-
Notifications
You must be signed in to change notification settings - Fork 0
/
genstrings.swift
executable file
·142 lines (125 loc) · 5.77 KB
/
genstrings.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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
#!/usr/bin/swift
import Foundation
class GenStrings {
var str = "Hello, playground"
let fileManager = FileManager.default
let acceptedFileExtensions = ["swift"]
let excludedFolderNames = ["Carthage"]
let excludedFileNames = ["genstrings.swift"]
var regularExpresions = [String: NSRegularExpression]()
let localizedRegex = "(?<=\")([^\"]*)(?=\".(localized|localizedFormat))|(?<=(Localized|NSLocalizedString)\\(\")([^\"]*?)(?=\")"
let commentedRegex = "(?<=.commented\\(\")([^\"]*)(?=\")"
enum GenstringsError: Error {
case Error
}
// Performs the genstrings functionality
func perform(path: String? = nil) {
let directoryPath = path ?? fileManager.currentDirectoryPath
let rootPath = URL(fileURLWithPath: directoryPath)
let allFiles = fetchFilesInFolder(rootPath: rootPath)
// We use a set to avoid duplicates
var localizableStrings = Set<String>()
var comments = [String: String]()
for filePath in allFiles {
let (stringsInFile, commentsInFile) = localizableStringsInFile(filePath: filePath)
localizableStrings = localizableStrings.union(stringsInFile)
comments = comments.merging(commentsInFile, uniquingKeysWith: { (_, last) in last })
}
// We sort the strings
let sortedStrings = localizableStrings.sorted(by: { $0 < $1 })
var processedStrings = String()
for string in sortedStrings {
if let comment = comments[string] {
processedStrings.append("/* \(comment) */ \n")
}
processedStrings.append("\"\(string)\" = \"\(string)\"; \n")
}
print(processedStrings)
}
// Applies regex to a file at filePath.
func localizableStringsInFile(filePath: URL) -> (Set<String>, [String: String]) {
do {
let fileContentsData = try Data(contentsOf: filePath)
guard let fileContentsString = NSString(data: fileContentsData, encoding: String.Encoding.utf8.rawValue) else {
return (Set<String>(), [String: String]())
}
let localizedMatches = try regexMatches(pattern: localizedRegex, string: fileContentsString as String)
let commentedMatches = try regexMatches(pattern: commentedRegex, string: fileContentsString as String)
let localizedStringsArray = localizedMatches.map({fileContentsString.substring(with: $0.range)})
var comments = [String: String]()
for m in commentedMatches {
let comment = fileContentsString.substring(with: m.range)
let localizedRange = localizedMatches.filter { $0.range.location + $0.range.length <= m.range.location }.last?.range
if localizedRange != nil {
let localized = fileContentsString.substring(with: localizedRange!)
comments[localized] = comment
}
}
return (Set(localizedStringsArray), comments)
} catch {}
return (Set<String>(), [String: String]())
}
// MARK: Regex
func regexWithPattern(pattern: String) throws -> NSRegularExpression {
var safeRegex = regularExpresions
if let regex = safeRegex[pattern] {
return regex
} else {
do {
let currentPattern: NSRegularExpression
currentPattern = try NSRegularExpression(pattern: pattern, options: NSRegularExpression.Options.caseInsensitive)
safeRegex.updateValue(currentPattern, forKey: pattern)
regularExpresions = safeRegex
return currentPattern
} catch {
throw GenstringsError.Error
}
}
}
func regexMatches(pattern: String, string: String) throws -> [NSTextCheckingResult] {
do {
let internalString = string
let currentPattern = try regexWithPattern(pattern: pattern)
// NSRegularExpression accepts Swift strings but works with NSString under the hood. Safer to bridge to NSString for taking range.
let nsString = internalString as NSString
let stringRange = NSRange(location: 0, length: nsString.length)
let matches = currentPattern.matches(in: internalString, options: [], range: stringRange)
return matches
} catch {
throw GenstringsError.Error
}
}
// MARK: File manager
func fetchFilesInFolder(rootPath: URL) -> [URL] {
var files = [URL]()
do {
let directoryContents = try fileManager.contentsOfDirectory(at: rootPath as URL, includingPropertiesForKeys: [], options: .skipsHiddenFiles)
for urlPath in directoryContents {
let stringPath = urlPath.path
let lastPathComponent = urlPath.lastPathComponent
let pathExtension = urlPath.pathExtension
var isDir: ObjCBool = false
if fileManager.fileExists(atPath: stringPath, isDirectory: &isDir) {
if isDir.boolValue {
if !excludedFolderNames.contains(lastPathComponent) {
let dirFiles = fetchFilesInFolder(rootPath: urlPath)
files.append(contentsOf: dirFiles)
}
} else {
if acceptedFileExtensions.contains(pathExtension) && !excludedFileNames.contains(lastPathComponent) {
files.append(urlPath)
}
}
}
}
} catch {}
return files
}
}
let genStrings = GenStrings()
if CommandLine.arguments.count > 1 {
let path = CommandLine.arguments[1]
genStrings.perform(path: path)
} else {
genStrings.perform()
}