import Foundation public enum TextParsing { /// Removes ANSI escape sequences so regex parsing works on colored terminal output. public static func stripANSICodes(_ text: String) -> String { // CSI sequences: ESC [ ... ending in 0x40–0x7E let pattern = #"\u001B\[[0-?]*[ -/]*[@-~]"# guard let regex = try? NSRegularExpression(pattern: pattern, options: []) else { return text } let range = NSRange(text.startIndex.. Double? { guard let regex = try? NSRegularExpression(pattern: pattern, options: [.caseInsensitive]) else { return nil } let range = NSRange(text.startIndex..= 2, let r = Range(match.range(at: 1), in: text) else { return nil } let raw = text[r].replacingOccurrences(of: ",", with: "") return Double(raw) } public static func firstInt(pattern: String, text: String) -> Int? { guard let v = firstNumber(pattern: pattern, text: text) else { return nil } return Int(v) } public static func firstLine(matching pattern: String, text: String) -> String? { guard let regex = try? NSRegularExpression(pattern: pattern, options: [.caseInsensitive]) else { return nil } let range = NSRange(text.startIndex.. Int? { guard let pct = firstInt(pattern: #"([0-9]{1,3})%\s+left"#, text: line) else { return nil } return pct } public static func resetString(fromLine line: String) -> String? { guard let regex = try? NSRegularExpression(pattern: #"resets?\s+(.+)"#, options: [.caseInsensitive]) else { return nil } let range = NSRange(line.startIndex..= 2, let r = Range(match.range(at: 1), in: line) else { return nil } // Return the tail text only (drop the "resets" prefix). return String(line[r]).trimmingCharacters(in: .whitespacesAndNewlines) } }