File: Plugin.swift

package info (click to toggle)
swiftlang 6.0.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,519,992 kB
  • sloc: cpp: 9,107,863; ansic: 2,040,022; asm: 1,135,751; python: 296,500; objc: 82,456; f90: 60,502; lisp: 34,951; pascal: 19,946; sh: 18,133; perl: 7,482; ml: 4,937; javascript: 4,117; makefile: 3,840; awk: 3,535; xml: 914; fortran: 619; cs: 573; ruby: 573
file content (91 lines) | stat: -rw-r--r-- 3,965 bytes parent folder | download
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
import PackagePlugin
import Foundation

@main
struct Plugin: CommandPlugin {
    func performCommand(context: PluginContext, arguments: [String]) async throws {
        var argumentExtractor = ArgumentExtractor(arguments)
        let targets = argumentExtractor.extractOption(named: "target")
        let sdk = argumentExtractor.extractOption(named: "sdk").last
        let parameters = PackageManager.BuildParameters()
        for target in targets {
            try extractFromTarget(target: target, sdk: sdk, parameters: parameters, context: context)
        }
    }

    func extractFromTarget(
        target: String,
        sdk: String?,
        parameters: PackageManager.BuildParameters,
        context: PluginContext
    ) throws {
        let buildResult = try packageManager.build(.target(target), parameters: parameters)
        guard buildResult.succeeded else {
            throw PluginError(description: "Failed to build \(target): \(buildResult.logText)")
        }
        // TODO: Add proper API to PackagePlugin to get data directory
        let dataPath = context.pluginWorkDirectory // output
            .removingLastComponent() // WITExtractorPlugin
            .removingLastComponent() // plugins
            .removingLastComponent() // .build (by default)

        let buildPath = dataPath.appending([parameters.configuration.rawValue])
        let llbuildManifest = dataPath.appending([parameters.configuration.rawValue + ".yaml"])
        guard let swiftcExecutable = ProcessInfo.processInfo.environment["WIT_EXTRACTOR_SWIFTC_PATH"]
                ?? inferSwiftcExecutablePath(llbuildManifest: llbuildManifest) else {
            throw PluginError(description: "Cloudn't infer `swiftc` command path from build directory. Please specify WIT_EXTRACTOR_SWIFTC_PATH")
        }
        let digesterExecutable = Path(swiftcExecutable).removingLastComponent().appending(["swift-api-digester"])

        let witOutputPath = context.pluginWorkDirectory.appending([target + ".wit"])
        let swiftOutputPath = context.pluginWorkDirectory.appending([target + "_WITOverlay.swift"])

        let tool = try context.tool(named: "WITTool")
        var arguments =  [
            "extract-wit",
            "--swift-api-digester", digesterExecutable.string,
            "--module-name", target,
            "--package-name", context.package.displayName,
            "--wit-output-path", witOutputPath.string,
            "--swift-output-path", swiftOutputPath.string,
            "-I", buildPath.string
        ]
        if let sdk {
            arguments += ["-sdk", sdk]
        }
        let process = try Process.run(URL(fileURLWithPath: tool.path.string), arguments: arguments)
        process.waitUntilExit()
        guard process.terminationStatus == 0 else {
            throw PluginError(
                description: "Failed to run \(([tool.path.string] + arguments).joined(separator: " "))"
            )
        }
        print("""
        {
            "witOutputPath": "\(witOutputPath)",
            "swiftOutputPath": "\(swiftOutputPath)"
        }
        """)
    }

    func inferSwiftcExecutablePath(llbuildManifest: Path) -> String? {
        // FIXME: This is completely not the right way but there is no right way for now...
        guard let contents = try? String(contentsOfFile: llbuildManifest.string, encoding: .utf8) else {
            return nil
        }
        for line in contents.split(separator: "\n") {
            let prefix = "    executable: \""
            if line.hasPrefix(prefix), line.hasSuffix("/swiftc\"") {
                let pathStart = line.index(line.startIndex, offsetBy: prefix.count)
                let pathEnd = line.index(before: line.endIndex)
                let executablePath = line[pathStart..<pathEnd]
                return String(executablePath)
            }
        }
        return nil
    }
}

struct PluginError: Error, CustomStringConvertible {
    let description: String
}