Files
SwiftFormat/Sources/CommandLine.swift
T
2019-11-24 10:27:26 +00:00

851 lines
34 KiB
Swift

//
// CommandLine.swift
// SwiftFormat
//
// Created by Nick Lockwood on 10/01/2017.
// Copyright 2017 Nick Lockwood
//
// Distributed under the permissive MIT license
// Get the latest version from here:
//
// https://github.com/nicklockwood/SwiftFormat
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
//
import Foundation
/// Public interface for the SwiftFormat command-line functions
public struct CLI {
/// Output type for printed content
public enum OutputType {
case info
case success
case error
case warning
case content
case raw
}
/// Output handler - override this to intercept output from the CLI
public static var print: (String, OutputType) -> Void = { _, _ in
fatalError("No print hook set.")
}
/// Input handler - override this to inject input into the CLI
/// Injected lines should include the terminating newline character
public static var readLine: () -> String? = {
Swift.readLine(strippingNewline: false)
}
/// Run the CLI with the specified input arguments
public static func run(in directory: String, with args: [String] = CommandLine.arguments) -> ExitCode {
return processArguments(args, in: directory)
}
/// Run the CLI with the specified input string (this will be parsed into multiple arguments)
public static func run(in directory: String, with argumentString: String) -> ExitCode {
return run(in: directory, with: parseArguments(argumentString))
}
}
private var quietMode = false
private func print(_ message: String, as type: CLI.OutputType = .info) {
if !quietMode || [.raw, .content, .error].contains(type) {
CLI.print(message, type)
}
}
private func printWarnings(_ errors: [Error]) {
for error in errors {
print("warning: \(error).", as: .warning)
}
}
// Represents the exit codes to the command line. See `man sysexits` for more information.
public enum ExitCode: Int32 {
case ok = 0 // EX_OK
case lintFailure = 1
case error = 70 // EX_SOFTWARE
}
func printOptions(as type: CLI.OutputType) {
print("")
print(FormatOptions.Descriptor.formatting.compactMap {
guard !$0.isDeprecated else { return nil }
var result = "--\($0.argumentName)"
for _ in 0 ..< 19 - result.count {
result += " "
}
return result + stripMarkdown($0.help)
}.sorted().joined(separator: "\n"), as: type)
print("")
}
func printRuleInfo(for name: String, as type: CLI.OutputType) throws {
guard let rule = FormatRules.byName[name] else {
if name.isEmpty {
throw FormatError.options("--ruleinfo command expects a rule name")
}
throw FormatError.options("'\(name)' rule does not exist")
}
print("")
print(name, as: type)
print("", as: type)
print(stripMarkdown(rule.help), as: type)
if !rule.options.isEmpty {
print("\nOptions:\n", as: type)
print(rule.options.compactMap {
guard let descriptor = FormatOptions.Descriptor.byName[$0], !descriptor.isDeprecated else {
return nil
}
var result = "--\(descriptor.argumentName)"
for _ in 0 ..< 19 - result.count {
result += " "
}
return result + stripMarkdown(descriptor.help)
}.sorted().joined(separator: "\n"), as: type)
}
if var examples = rule.examples {
examples = examples
.replacingOccurrences(of: "```diff\n", with: "")
.replacingOccurrences(of: "```\n", with: "")
if examples.hasSuffix("```") {
examples = String(examples.dropLast(3))
}
print("\nExamples:\n", as: type)
print(examples, as: type)
}
print("")
}
func printHelp(as type: CLI.OutputType) {
print("")
print("""
SwiftFormat, version \(version)
Copyright (c) 2016 Nick Lockwood
--help Print this help page
--version Print the currently installed swiftformat version
SwiftFormat can operate on files & directories, or directly on input from stdin.
Usage: swiftformat [<file> <file> ...] [--inferoptions] [--output path] [...]
<file> <file> ... One or more swift files or directory paths to be processed
--config Path to a configuration file containing rules and options
--inferoptions Instead of formatting input, use it to infer format options
--output Output path for formatted file(s) (defaults to input path)
--exclude Comma-delimited list of ignored paths (supports glob syntax)
--unexclude Paths to not exclude, even if excluded elsewhere in config
--symlinks How symlinks are handled: "follow" or "ignore" (default)
--fragment \(stripMarkdown(FormatOptions.Descriptor.fragment.help))
--conflictmarkers \(stripMarkdown(FormatOptions.Descriptor.ignoreConflictMarkers.help))
--swiftversion \(stripMarkdown(FormatOptions.Descriptor.swiftVersion.help))
--cache Path to cache file, or "clear" or "ignore" the default cache
--verbose Display detailed formatting output and warnings/errors
--quiet Disables non-critical output messages and warnings
--dryrun Run in "dry" mode (without actually changing any files)
--lint Like --dryrun, but returns an error if formatting is needed
SwiftFormat has a number of rules that can be enabled or disabled. By default
most rules are enabled. Use --rules to display all enabled/disabled rules.
--rules The list of rules to apply. Pass nothing to print all rules
--disable A list of format rules to be disabled (comma-delimited)
--enable A list of disabled rules to be re-enabled (comma-delimited)
SwiftFormat's rules can be configured using options. A given option may affect
multiple rules. Options have no effect if the related rules have been disabled.
--ruleinfo Display options for a given rule or rules (comma-delimited)
--options Prints a list of all formatting options and their usage
""", as: type)
print("")
}
func timeEvent(block: () throws -> Void) rethrows -> TimeInterval {
#if os(macOS)
let start = CFAbsoluteTimeGetCurrent()
try block()
return CFAbsoluteTimeGetCurrent() - start
#else
let start = Date.timeIntervalSinceReferenceDate
try block()
return Date.timeIntervalSinceReferenceDate - start
#endif
}
private func formatTime(_ time: TimeInterval) -> String {
let time = round(time * 100) / 100 // round to nearest 10ms
return String(format: "%gs", time)
}
private func serializeOptions(_ options: Options, to outputURL: URL?) throws {
if let outputURL = outputURL {
let file = serialize(options: options) + "\n"
do {
try file.write(to: outputURL, atomically: true, encoding: .utf8)
} catch {
throw FormatError.writing("Failed to write options to \(outputURL.path)")
}
} else {
print(serialize(options: options, excludingDefaults: true, separator: " "), as: .content)
print("")
}
}
typealias OutputFlags = (
filesWritten: Int,
filesChecked: Int,
filesSkipped: Int,
filesFailed: Int
)
func processArguments(_ args: [String], in directory: String) -> ExitCode {
var errors = [Error]()
var verbose = false
quietMode = false
defer {
// Reset quiet mode on exit to prevent side-effects between unit tests
quietMode = false
}
do {
// Get arguments
var args = try preprocessArguments(args, commandLineArguments)
// Quiet mode
quietMode = (args["quiet"] != nil)
// Verbose
verbose = (args["verbose"] != nil)
// Lint
let lint = (args["lint"] != nil)
// Dry run
let dryrun = lint || (args["dryrun"] != nil)
// Input path(s)
var inputURLs = [URL]()
while let inputPath = args[String(inputURLs.count + 1)] {
inputURLs += try parsePaths(inputPath, for: "input", in: directory)
}
// Warnings
for warning in warningsForArguments(args) {
print(warning, as: .warning)
}
// Show help
if args["help"] != nil {
printHelp(as: .content)
return .ok
}
// Show version
if args["version"] != nil {
print(version, as: .content)
return .ok
}
// Show options
if args["options"] != nil {
printOptions(as: .content)
return .ok
}
// Show rule info
if let names = try args["ruleinfo"].map(parseRules) {
let names = names.isEmpty ? allRules.sorted() : names.sorted()
for name in names {
try printRuleInfo(for: name, as: .content)
}
return .ok
}
// Display rules (must be checked before merging config)
let showRules: Bool = args["rules"].map {
if $0 == "" {
args["rules"] = nil
return true
}
return false
} ?? false
// Config file
if let configURL = try args["config"].map({ try parsePath($0, for: "--config", in: directory) }) {
if args["config"] == "" {
throw FormatError.options("--config argument expects a value")
}
if !FileManager.default.fileExists(atPath: configURL.path) {
throw FormatError.reading("Specified config file does not exist: \(configURL.path)")
}
let data: Data
do {
data = try Data(contentsOf: configURL)
} catch {
throw FormatError.reading("Failed to read config file at \(configURL.path), \(error)")
}
var config = try parseConfigFile(data)
// Ensure exclude paths in config file are treated as relative to the file itself
// TODO: find a better way/place to do this
let directory = configURL.deletingLastPathComponent().path
if let exclude = config["exclude"] {
let excluded = expandGlobs(exclude, in: directory)
if excluded.isEmpty {
print("warning: --exclude value '\(exclude)' did not match any files in \(directory).", as: .warning)
config["exclude"] = nil
} else {
config["exclude"] = excluded.map { $0.description }.sorted().joined(separator: ",")
}
}
if let unexclude = config["unexclude"] {
let unexcluded = expandGlobs(unexclude, in: directory)
if unexcluded.isEmpty {
print("warning: --unexclude value '\(unexclude)' did not match any files in \(directory).", as: .warning)
config["unexclude"] = nil
} else {
config["unexclude"] = unexcluded.map { $0.description }.sorted().joined(separator: ",")
}
}
args = try mergeArguments(args, into: config)
}
// Options
let options = try Options(args, in: directory)
// Show rules
if showRules {
print("")
let rules = options.rules ?? allRules.subtracting(FormatRules.disabledByDefault)
for name in Array(allRules).sorted() {
print(" \(name)\(rules.contains(name) ? "" : " (disabled)")", as: .content)
}
print("")
return .ok
}
// FormatOption overrides
var overrides = [String: String]()
for key in formattingArguments + rulesArguments {
overrides[key] = args[key]
}
// Treat values for arguments that do not take a value as input paths
func addInputPaths(for argName: String) throws {
guard let arg = args[argName], !arg.isEmpty else {
return
}
guard inputURLs.isEmpty || "/.,".contains(where: { arg.contains($0) }) else {
throw FormatError.options("--\(argName) argument does not expect a value")
}
inputURLs += try parsePaths(arg, for: "input", in: directory)
}
try addInputPaths(for: "quiet")
try addInputPaths(for: "verbose")
try addInputPaths(for: "dryrun")
try addInputPaths(for: "lint")
try addInputPaths(for: "inferoptions")
// Output path
let outputURL = try args["output"].map { try parsePath($0, for: "--output", in: directory) }
if outputURL != nil {
if args["output"] == "" {
throw FormatError.options("--output argument expects a value")
} else if inputURLs.count > 1 {
throw FormatError.options("--output argument is only valid for a single input file or directory")
}
}
// Infer options
if args["inferoptions"] != nil {
guard args["config"] == nil else {
throw FormatError.options("--inferoptions option can't be used along with a config file")
}
if !inputURLs.isEmpty {
print("Inferring swiftformat options from source file(s)...", as: .info)
var filesParsed = 0, formatOptions = FormatOptions.default, errors = [Error]()
let fileOptions = options.fileOptions ?? .default
let time = formatTime(timeEvent {
(filesParsed, formatOptions, errors) = inferOptions(from: inputURLs, options: fileOptions)
})
printWarnings(errors)
if filesParsed == 0 {
throw FormatError.parsing("Failed to to infer options")
}
var filesChecked = filesParsed
for case let error as FormatError in errors {
switch error {
case .parsing, .reading:
filesChecked += 1
case .writing:
assertionFailure()
case .options:
break
}
}
print("Options inferred from \(filesParsed)/\(filesChecked) files in \(time).", as: .info)
print("")
var options = options
options.formatOptions = formatOptions
try serializeOptions(options, to: outputURL)
return .ok
}
}
// Cache path
var cacheURL: URL?
let defaultCacheFileName = "swiftformat.cache"
let manager = FileManager.default
func setDefaultCacheURL() {
var cacheDirectory: URL!
#if os(macOS)
if let cachePath = NSSearchPathForDirectoriesInDomains(.cachesDirectory, .userDomainMask, true).first {
cacheDirectory = URL(fileURLWithPath: cachePath)
}
#endif
cacheDirectory = (cacheDirectory ?? URL(fileURLWithPath: "/var/tmp/"))
.appendingPathComponent("com.charcoaldesign.swiftformat")
do {
try manager.createDirectory(at: cacheDirectory, withIntermediateDirectories: true, attributes: nil)
cacheURL = cacheDirectory.appendingPathComponent(defaultCacheFileName)
} catch {
errors.append(FormatError.writing("Failed to create cache directory at \(cacheDirectory.path)"))
}
}
if let cache = args["cache"] {
switch cache {
case "":
throw FormatError.options("--cache option expects a value")
case "ignore":
break
case "clear":
setDefaultCacheURL()
if let cacheURL = cacheURL, manager.fileExists(atPath: cacheURL.path) {
do {
try manager.removeItem(at: cacheURL)
} catch {
errors.append(FormatError.writing("Failed to delete cache file at \(cacheURL.path)"))
}
}
default:
cacheURL = try parsePath(cache, for: "--cache", in: directory)
guard cacheURL != nil else {
throw FormatError.options("Invalid --cache value '\(cache)'")
}
var isDirectory: ObjCBool = false
if manager.fileExists(atPath: cacheURL!.path, isDirectory: &isDirectory), isDirectory.boolValue {
cacheURL = cacheURL!.appendingPathComponent(defaultCacheFileName)
}
}
} else {
setDefaultCacheURL()
}
func printRunningMessage() {
print("Running SwiftFormat...", as: .info)
if lint {
print("(lint mode - no files will be changed.)", as: .info)
} else if dryrun {
print("(dryrun mode - no files will be changed.)", as: .info)
}
}
// If no input file, try stdin
if inputURLs.isEmpty {
var input: String?
var finished = false
var fatalError: Error?
DispatchQueue.global(qos: .userInitiated).async {
while let line = CLI.readLine() {
input = (input ?? "") + line
}
if let input = input {
do {
if args["inferoptions"] != nil {
let tokens = tokenize(input)
var options = options
options.formatOptions = inferFormatOptions(from: tokens)
try serializeOptions(options, to: outputURL)
} else if let outputURL = outputURL {
printRunningMessage()
let output = try format(input, options: options, verbose: verbose)
if (try? String(contentsOf: outputURL)) != output {
if dryrun {
if verbose {
print("Would have updated \(outputURL.path).", as: .info)
}
} else {
do {
try output.write(to: outputURL, atomically: true, encoding: .utf8)
} catch {
throw FormatError.writing("Failed to write file \(outputURL.path)")
}
}
}
print("Swiftformat completed successfully.", as: .success)
} else {
// Write to stdout
let output = try format(input, options: options, verbose: false)
print(output, as: .raw)
}
} catch {
fatalError = error
}
}
finished = true
}
// Wait for input
let start = NSDate()
while start.timeIntervalSinceNow > -0.01 {}
// If no input received by now, assume none is coming
if input != nil {
while !finished, start.timeIntervalSinceNow > -30 {}
if let fatalError = fatalError {
throw fatalError
}
} else if args["inferoptions"] != nil {
throw FormatError.options("--inferoptions requires one or more input files")
} else {
printHelp(as: .info)
}
return .ok
}
printRunningMessage()
// Format the code
var outputFlags: OutputFlags = (0, 0, 0, 0)
let time = formatTime(timeEvent {
var _errors: [Error]
(outputFlags, _errors) = processInput(inputURLs,
andWriteToOutput: outputURL,
options: options,
overrides: overrides,
verbose: verbose,
dryrun: dryrun,
cacheURL: cacheURL)
errors += _errors
})
if outputFlags.filesWritten == 0 {
if outputFlags.filesChecked == 0 {
if let error = errors.first {
errors.removeAll()
throw error
}
if outputFlags.filesSkipped == 0 {
let inputPaths = inputURLs.map { $0.path }.joined(separator: ", ")
throw FormatError.options("No eligible files found at \(inputPaths)")
}
} else if !dryrun, !errors.isEmpty {
throw FormatError.options("Failed to format any files")
}
}
if verbose {
print("")
}
printWarnings(errors)
print("SwiftFormat completed in \(time).", as: .success)
return printResult(dryrun, lint, outputFlags)
} catch {
if !verbose {
// Warnings would be redundant at this point
printWarnings(errors)
}
// Fatal error
print("error: \(error).", as: .error)
return .error
}
}
func printResult(_ dryrun: Bool, _ lint: Bool, _ flags: OutputFlags) -> ExitCode {
let (written, checked, skipped, failed) = flags
let ignored = (skipped == 0) ? "" : ", \(skipped) file\(skipped == 1 ? "" : "s") skipped"
if checked == 0 {
print("0 files formatted\(ignored).", as: .info)
} else if lint {
if failed > 0 {
print("Source input did not pass lint check.", as: .error)
}
print("\(failed)/\(checked) files require formatting\(ignored).", as: .info)
} else if dryrun {
print("\(failed)/\(checked) files would have been formatted\(ignored).", as: .info)
} else {
print("\(written)/\(checked) files formatted\(ignored).", as: .info)
}
return lint && failed > 0 ? .lintFailure : .ok
}
func inferOptions(from inputURLs: [URL], options: FileOptions) -> (Int, FormatOptions, [Error]) {
var tokens = [Token]()
var errors = [Error]()
var filesParsed = 0
let baseOptions = Options(fileOptions: options)
for inputURL in inputURLs {
errors += enumerateFiles(withInputURL: inputURL, options: baseOptions) { inputURL, _, _ in
guard let input = try? String(contentsOf: inputURL) else {
throw FormatError.reading("Failed to read file \(inputURL.path)")
}
let _tokens = tokenize(input)
return {
filesParsed += 1
tokens += _tokens
}
}
}
return (filesParsed, inferFormatOptions(from: tokens), errors)
}
func computeHash(_ source: String) -> String {
var count = 0
var hash: UInt64 = 5381
for byte in source.utf8 {
count += 1
hash = 127 &* (hash & 0x00FF_FFFF_FFFF_FFFF) &+ UInt64(byte)
}
return "\(count)\(hash)"
}
func format(_ source: String, options: Options, verbose: Bool) throws -> String {
// Parse source
let originalTokens = tokenize(source)
var tokens = originalTokens
// Get rules
let rulesByName = FormatRules.byName
let ruleNames = Array(options.rules ?? allRules.subtracting(FormatRules.disabledByDefault)).sorted()
let rules = ruleNames.compactMap { rulesByName[$0] }
var rulesApplied = Set<String>()
let callback: ((Int, [Token]) -> Void)? = verbose ? { i, updatedTokens in
if updatedTokens != tokens {
rulesApplied.insert(ruleNames[i])
tokens = updatedTokens
}
} : nil
// Apply rules
let formatOptions = options.formatOptions ?? .default
tokens = try applyRules(rules, to: tokens, with: formatOptions, callback: callback)
// Display info
if verbose {
if rulesApplied.isEmpty || tokens == originalTokens {
print(" -- no changes", as: .success)
} else {
let sortedNames = Array(rulesApplied).sorted().joined(separator: ", ")
print(" -- rules applied: \(sortedNames)", as: .success)
}
}
// Output
return sourceCode(for: tokens)
}
func processInput(_ inputURLs: [URL],
andWriteToOutput outputURL: URL?,
options: Options,
overrides: [String: String],
verbose: Bool,
dryrun: Bool,
cacheURL: URL?) -> (OutputFlags, [Error]) {
// Load cache
let cacheDirectory = cacheURL?.deletingLastPathComponent().absoluteURL
var cache: [String: String]?
if let cacheURL = cacheURL {
cache = NSDictionary(contentsOf: cacheURL) as? [String: String] ?? [:]
}
// Logging skipped files
var outputFlags: OutputFlags = (0, 0, 0, 0)
let skippedHandler: FileEnumerationHandler? = verbose ? { inputURL, _, _ in
print("Skipping \(inputURL.path)", as: .info)
print("-- ignored", as: .warning)
return {}
} : { _, _, _ in
{ outputFlags.filesSkipped += 1 }
}
// Swift version
var showedConfigurationWarnings = false
func showConfigurationWarnings(_ options: Options) {
let formatOptions = options.formatOptions ?? .default
guard !showedConfigurationWarnings else {
return
}
if formatOptions.swiftVersion == .undefined {
print("warning: No swift version was specified, so some formatting features were disabled. Specify the version of swift you are using with the --swiftversion command line option, or by adding a \(swiftVersionFile) file to your project.", as: .warning)
}
if formatOptions.useTabs, formatOptions.tabWidth <= 0 {
print("warning: The --indent option is set to tabs, but no --tabwidth was specified.", as: .warning)
}
showedConfigurationWarnings = true
}
// Format files
var errors = [Error]()
for inputURL in inputURLs {
errors += enumerateFiles(withInputURL: inputURL,
outputURL: outputURL,
options: options,
concurrent: !verbose,
skipped: skippedHandler) { inputURL, outputURL, options in
guard let input = try? String(contentsOf: inputURL) else {
throw FormatError.reading("Failed to read file \(inputURL.path)")
}
// Override options
var options = options
try options.addArguments(overrides, in: "") // No need for directory as overrides are formatOptions only
// Validate options
let formatOptions = options.formatOptions ?? .default
if formatOptions.useTabs, formatOptions.tabWidth <= 0 {
throw FormatError.options("The --maxwidth option requires --tabwidth to also be set")
}
// Check cache
let rules = options.rules ?? allRules.subtracting(FormatRules.disabledByDefault)
let cachePrefix = "\(version);\(formatOptions)\(rules.sorted().joined(separator: ","));"
let cacheKey: String = {
var path = inputURL.absoluteURL.path
if let cacheDirectory = cacheDirectory {
let commonPrefix = path.commonPrefix(with: cacheDirectory.path)
path = String(path[commonPrefix.endIndex ..< path.endIndex])
}
return path
}()
do {
if verbose {
print("Formatting \(inputURL.path)", as: .info)
}
var cacheHash: String?
var sourceHash: String?
if let cacheEntry = cache?[cacheKey], cacheEntry.hasPrefix(cachePrefix) {
cacheHash = String(cacheEntry[cachePrefix.endIndex...])
sourceHash = computeHash(input)
}
let output: String
if let cacheHash = cacheHash, cacheHash == sourceHash {
output = input
if verbose {
print("-- no changes (cached)", as: .success)
}
} else {
output = try format(input, options: options, verbose: verbose)
if output != input {
sourceHash = nil
}
}
let cacheValue = cache.map { _ in
// Only bother computing this if cache is enabled
cachePrefix + (sourceHash ?? computeHash(output))
}
if outputURL != inputURL, (try? String(contentsOf: outputURL)) != output {
if !dryrun {
do {
try FileManager.default.createDirectory(at: outputURL.deletingLastPathComponent(),
withIntermediateDirectories: true,
attributes: nil)
} catch {
throw FormatError.writing("Failed to create directory at \(outputURL.path), \(error)")
}
}
} else if output == input {
// No changes needed
return {
outputFlags.filesChecked += 1
cache?[cacheKey] = cacheValue
showConfigurationWarnings(options)
}
}
if dryrun {
if verbose {
print("Would have updated \(outputURL.path)", as: .info)
}
return {
outputFlags.filesChecked += 1
outputFlags.filesFailed += 1
showConfigurationWarnings(options)
}
} else {
do {
if verbose {
print("Writing \(outputURL.path)", as: .info)
}
try output.write(to: outputURL, atomically: true, encoding: .utf8)
return {
outputFlags.filesChecked += 1
outputFlags.filesFailed += 1
outputFlags.filesWritten += 1
cache?[cacheKey] = cacheValue
showConfigurationWarnings(options)
}
} catch {
throw FormatError.writing("Failed to write file \(outputURL.path), \(error)")
}
}
} catch {
if verbose {
print("-- error: \(error)", as: .error)
}
return {
outputFlags.filesChecked += 1
showConfigurationWarnings(options)
if case let FormatError.parsing(string) = error {
throw FormatError.parsing("\(string) in \(inputURL.path)")
}
throw error
}
}
}
}
if verbose {
var errorCount = errors.count
errors = errors.filter { error in
guard let error = error as? FormatError else {
return true
}
switch error {
case .options, .reading:
return true
case .parsing, .writing:
return false
}
}
errorCount -= errors.count
if errorCount > 0 {
// Replace individual warnings with a generic message, to avoid repetition
errors.append(FormatError.writing("\(errorCount) file\(errorCount == 1 ? "" : "s") could not be formatted"))
}
}
if outputFlags.filesChecked > 0 {
// Save cache
if let cache = cache, let cacheURL = cacheURL, let cacheDirectory = cacheDirectory {
if !(cache as NSDictionary).write(to: cacheURL, atomically: true) {
if FileManager.default.fileExists(atPath: cacheDirectory.path) {
errors.append(FormatError.writing("Failed to write cache file at \(cacheURL.path)"))
} else {
errors.append(FormatError.reading("Specified cache file directory does not exist: \(cacheDirectory.path)"))
}
}
}
}
return (outputFlags, errors)
}