Skip to content

Commit

Permalink
Merge pull request swiftlang#2268 from rintaro/macro-resolveerror-rda…
Browse files Browse the repository at this point in the history
…r115571427

[Macros] Granular diagnostics when macro type is not found in a plugin
  • Loading branch information
rintaro authored Oct 17, 2023
2 parents 8decbe4 + e4e6a95 commit 975ac26
Show file tree
Hide file tree
Showing 4 changed files with 25 additions and 25 deletions.
21 changes: 13 additions & 8 deletions Sources/SwiftCompilerPlugin/CompilerPlugin.swift
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,8 @@ public protocol CompilerPlugin {
}

extension CompilerPlugin {
func resolveMacro(moduleName: String, typeName: String) -> Macro.Type? {
@_spi(Testing)
public func resolveMacro(moduleName: String, typeName: String) throws -> Macro.Type {
let qualifedName = "\(moduleName).\(typeName)"

for type in providingMacros {
Expand All @@ -74,12 +75,9 @@ extension CompilerPlugin {
return type
}
}
return nil
}

// @testable
public func _resolveMacro(moduleName: String, typeName: String) -> Macro.Type? {
resolveMacro(moduleName: moduleName, typeName: typeName)
let pluginPath = CommandLine.arguments.first ?? Bundle.main.executablePath ?? ProcessInfo.processInfo.processName
throw CompilerPluginError(message: "macro implementation type '\(moduleName).\(typeName)' could not be found in executable plugin '\(pluginPath)'")
}
}

Expand All @@ -88,8 +86,8 @@ struct MacroProviderAdapter<Plugin: CompilerPlugin>: PluginProvider {
init(plugin: Plugin) {
self.plugin = plugin
}
func resolveMacro(moduleName: String, typeName: String) -> Macro.Type? {
plugin.resolveMacro(moduleName: moduleName, typeName: typeName)
func resolveMacro(moduleName: String, typeName: String) throws -> Macro.Type {
try plugin.resolveMacro(moduleName: moduleName, typeName: typeName)
}
}

Expand Down Expand Up @@ -242,3 +240,10 @@ private extension FileHandle {
}
}
}

struct CompilerPluginError: Error, CustomStringConvertible {
var description: String
init(message: String) {
self.description = message
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ public enum PluginFeature: String {
/// A type that provides the actual plugin functions.
public protocol PluginProvider {
/// Resolve macro type by the module name and the type name.
func resolveMacro(moduleName: String, typeName: String) -> Macro.Type?
func resolveMacro(moduleName: String, typeName: String) throws -> Macro.Type

/// Load dynamic link library at `libraryPath`. Implementations can use
/// `moduleName` to associate the loaded library with it.
Expand Down
13 changes: 4 additions & 9 deletions Sources/SwiftCompilerPluginMessageHandling/Macros.swift
Original file line number Diff line number Diff line change
Expand Up @@ -19,8 +19,8 @@ import SwiftSyntaxMacros

extension CompilerPluginMessageHandler {
/// Get concrete macro type from a pair of module name and type name.
private func resolveMacro(_ ref: PluginMessage.MacroReference) -> Macro.Type? {
provider.resolveMacro(moduleName: ref.moduleName, typeName: ref.typeName)
private func resolveMacro(_ ref: PluginMessage.MacroReference) throws -> Macro.Type {
try provider.resolveMacro(moduleName: ref.moduleName, typeName: ref.typeName)
}

/// Expand `@freestainding(XXX)` macros.
Expand All @@ -43,10 +43,7 @@ extension CompilerPluginMessageHandler {
guard let macroSyntax = syntax.asProtocol(FreestandingMacroExpansionSyntax.self) else {
throw MacroExpansionError.freestandingMacroSyntaxIsNotMacro
}
guard let macroDefinition = resolveMacro(macro) else {
throw MacroExpansionError.macroTypeNotFound(macro)
}

let macroDefinition = try resolveMacro(macro)
let macroRole: MacroRole
if let pluginMacroRole {
macroRole = MacroRole(messageMacroRole: pluginMacroRole)
Expand Down Expand Up @@ -113,9 +110,7 @@ extension CompilerPluginMessageHandler {
// TODO: Make this a 'String?' and remove non-'hasExpandMacroResult' branches.
let expandedSources: [String]?
do {
guard let macroDefinition = resolveMacro(macro) else {
throw MacroExpansionError.macroTypeNotFound(macro)
}
let macroDefinition = try resolveMacro(macro)
let role = MacroRole(messageMacroRole: macroRole)

let expansions = SwiftSyntaxMacroExpansion.expandAttachedMacroWithoutCollapsing(
Expand Down
14 changes: 7 additions & 7 deletions Tests/SwiftCompilerPluginTest/CompilerPluginTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
//
//===----------------------------------------------------------------------===//

import SwiftCompilerPlugin
@_spi(Testing) import SwiftCompilerPlugin
import SwiftSyntax
import SwiftSyntaxMacros
import XCTest
Expand Down Expand Up @@ -51,20 +51,20 @@ public class CompilerPluginTests: XCTestCase {
func testResolveMacro() {
let plugin = MyPlugin()

let registeredMacro = plugin._resolveMacro(
let registeredMacro = try? plugin.resolveMacro(
moduleName: "SwiftCompilerPluginTest",
typeName: "RegisteredMacro"
)
XCTAssertNotNil(registeredMacro)
XCTAssertTrue(registeredMacro == RegisteredMacro.self)

/// Test the plugin doesn't provide unregistered macros.
let dummyMacro = plugin._resolveMacro(
moduleName: "SwiftCompilerPluginTest",
typeName: "DummyMacro"
XCTAssertThrowsError(
try plugin.resolveMacro(
moduleName: "SwiftCompilerPluginTest",
typeName: "DummyMacro"
)
)
XCTAssertNil(dummyMacro)
XCTAssertFalse(dummyMacro == DummyMacro.self)

}
}

0 comments on commit 975ac26

Please sign in to comment.