-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathRegex.swift
118 lines (103 loc) · 3.56 KB
/
Regex.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
//
// Regex.swift
// Macro
//
// Created by Helge Hess.
// Copyright © 2021 ZeeZide GmbH. All rights reserved.
//
#if canImport(Foundation)
import class Foundation.NSRegularExpression
import class Foundation.NSString
import struct Foundation.NSRange
public extension String {
/**
* No Regex syntax in Swift. This is used to replicate those:
*
* arg.match(/^-/)
*
* TODO: `g` and `i` suffixes, don't know about `g`, but `i` is an option:
*
* arg.match("ain", .caseInsensitive)
*
* Note: Like JS this returns `nil` if no matches are found.
*/
@inlinable
func match(_ pattern: String,
options: NSRegularExpression.Options = [])
-> [ String ]?
{
guard let regex =
try? NSRegularExpression(pattern: pattern, options: options)
else {
assertionFailure("Could not parse regex: \(pattern)")
return nil
}
let range = NSRange(self.startIndex..<self.endIndex, in: self)
let matches = regex.matches(in: self, options: [], range: range)
if matches.isEmpty { return nil }
let nsSelf = self as NSString
var matchStrings = [ String ]()
matchStrings.reserveCapacity(matches.count)
for match in matches {
let matchString = nsSelf.substring(with: match.range)
matchStrings.append(matchString)
}
return matchStrings
}
/**
* No Regex syntax in Swift. This is used to replicate those:
*
* arg.replace(/^-+/, "")
* searchPath.replace(/\\/g, "\\\\"))
*
* TODO: What about `g`, what is that in NSRegEx? An option?
*/
@inlinable
func replace(_ pattern: String, _ replacement: String,
options: NSRegularExpression.Options = []) -> String
{
guard let regex =
try? NSRegularExpression(pattern: pattern, options: options)
else {
assertionFailure("Could not parse regex: \(pattern)")
return self
}
let range = NSRange(self.startIndex..<self.endIndex, in: self)
return regex.stringByReplacingMatches(in: self, options: [], range: range,
withTemplate: replacement)
}
/// Same like `String.split(separator:)`
@inlinable
func split(_ pattern: String, omitEmpty: Bool = false) -> [ String ] {
guard !self.isEmpty else { return [ "" ] } // That's what Node does
guard let regex = try? NSRegularExpression(pattern: pattern, options: [])
else {
assertionFailure("Could not parse regex: \(pattern)")
return [ self ]
}
let range = NSRange(self.startIndex..<self.endIndex, in: self)
let matches = regex.matches(in: self, options: [], range: range)
guard !matches.isEmpty else { return [ self ] }
let nsSelf = self as NSString
var idx = 0
var components = [ String ]()
components.reserveCapacity(matches.count + 1)
for match in matches {
let splitterRange = match.range
let componentRange = NSRange(location: idx,
length: splitterRange.location - idx)
assert(idx >= 0 && idx < nsSelf.length)
assert(componentRange.length >= 0)
assert(componentRange.location > 0)
let component = nsSelf.substring(with: componentRange)
if !omitEmpty || !component.isEmpty { components.append(component) }
idx = splitterRange.upperBound
}
if idx + 1 < nsSelf.length {
let component = nsSelf.substring(from: idx)
if !omitEmpty || !component.isEmpty { components.append(component) }
}
return components
}
}
#endif // canImport(Foundation)