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
|
/*
This source file is part of the Swift.org open source project
Copyright (c) 2021 Apple Inc. and the Swift project authors
Licensed under Apache License v2.0 with Runtime Library Exception
See https://swift.org/LICENSE.txt for license information
See https://swift.org/CONTRIBUTORS.txt for Swift project authors
*/
import ArgumentParser
import Foundation
/// Resolves and validates a ``templateURL`` value that points to an HTML documentation template.
///
/// This value can be set via an environment variable.
public struct TemplateOption: ParsableArguments {
public init() {}
/// The environment variable key that can be used to set the ``templateURL`` property.
static let environmentVariableKey = "DOCC_HTML_DIR"
/// The path to an HTML template to be used during conversion as provided by
/// the environment variable `DOCC_HTML_DIR`.
public var templateURL: URL?
/// The location of the docc executable that's running.
/// This can be set to a known value for testing.
static var doccExecutableLocation: URL = {
// We rely on Bundle.main.executableURL here which is a robust API
// that should always return the current executable's URL, regardless of how it is invoked.
//
// If docc is invoked directly, then the executable's path is always the first
// command line argument provided. So, in the exceptional case where Bundle.main.executableURL
// is nil, we fall back to the value provided in CommandLine.arguments[0].
return Bundle.main.executableURL ?? URL(fileURLWithPath: CommandLine.arguments[0])
}()
/// The default template location.
var defaultTemplateURL: URL {
// This looks for the template relative to the docc executable
//
// executable: common-file-path/bin/docc
// template: common-file-path/share/docc/render/
let templatePath = TemplateOption.doccExecutableLocation
.deletingLastPathComponent() // docc
.deletingLastPathComponent() // bin
.appendingPathComponent("share", isDirectory: true)
.appendingPathComponent("docc", isDirectory: true)
.appendingPathComponent("render", isDirectory: true)
return templatePath
}
static func invalidHTMLTemplateError(
path templatePath: String,
expectedFile expectedFileName: String
) -> ValidationError {
return ValidationError(
"""
Invalid or missing HTML template directory configuration provided.
The directory at '\(templatePath)' does not contain a valid template. Missing '\(expectedFileName)' file.
Set the '\(TemplateOption.environmentVariableKey)' environment variable to use a custom HTML template.
"""
)
}
static func missingHTMLTemplateError(
path expectedTemplatePath: String
) -> ValidationError {
return ValidationError(
"""
Invalid or missing HTML template directory, relative to the docc executable, at: '\(expectedTemplatePath)'.
Set the '\(TemplateOption.environmentVariableKey)' environment variable to use a custom HTML template.
"""
)
}
public mutating func validate() throws {
templateURL = ProcessInfo.processInfo.environment[TemplateOption.environmentVariableKey]
.map { URL(fileURLWithPath: $0) }
// Validate that the provided template URL represents a directory
try URLArgumentValidator.validateHasDirectoryPath(templateURL,
forArgumentDescription: """
'\(TemplateOption.environmentVariableKey)' environment variable
""")
// Only perform further validation if a templateURL has been provided
guard let templateURL else {
if FileManager.default.fileExists(atPath: defaultTemplateURL.appendingPathComponent(HTMLTemplate.indexFileName.rawValue).path) {
self.templateURL = defaultTemplateURL
}
return
}
// Confirm that the provided directory contains an 'index.html' file which is a required part of
// an HTML template for docc.
guard FileManager.default.fileExists(atPath: templateURL.appendingPathComponent(HTMLTemplate.indexFileName.rawValue).path)
else {
throw Self.invalidHTMLTemplateError(path: templateURL.path, expectedFile: "index.html")
}
}
}
|