.xcodeproj
나 .xcworkspace
대신 폴더구조+package manifest 로만 구성된다.CodableMacroMacros
라는 폴더가 추가됐다. 보기 싫어서 기본으로 설정된 package 파일을 아래와 같이 변경해줬다. // swift-tools-version: 5.9
// swift 5.9를 지원하기 위해 변경
import PackageDescription
import CompilerPluginSupport
let package = Package(
name: "CodableMacro",
platforms: [.macOS(.v13), .iOS(.v15), .watchOS(.v6), .macCatalyst(.v13)], // *불필요한 platform 제외
products: [
.library(
name: "CodableMacro",
targets: ["CodableMacro"]
),
// *executable은 제외해줬다. 이렇게 하면 package import 시 CodableMacroClient에 있는 main 파일은 실행할 수 없다.
],
dependencies: [
.package(url: "https://github.com/swiftlang/swift-syntax.git", from: "600.0.0-latest"),
],
targets: [
.macro(
name: "CodableMacroCore",
dependencies: [
.product(name: "SwiftSyntaxMacros", package: "swift-syntax"),
.product(name: "SwiftCompilerPlugin", package: "swift-syntax")
]
),
.target(name: "CodableMacro", dependencies: ["CodableMacroCore"]),
.executableTarget(name: "CodableMacroClient", dependencies: ["CodableMacro"]),
.testTarget(
name: "CodableMacroTests",
dependencies: [
"CodableMacroCore",
.product(name: "SwiftSyntaxMacrosTestSupport", package: "swift-syntax"),
]
),
]
)
// AS-IS
@freestanding(expression)
public macro stringify<T>(_ value: T) -> (T, String) = #externalMacro(module: "CodableMacroMacros", type: "StringifyMacro")
// TO-BE
@freestanding(expression)
public macro stringify<T>(_ value: T) -> (T, String) = #externalMacro(module: "CodableMacroCore", type: "StringifyMacro")
The value 42 was produced by the code "a + b"
가 console에 출력된다!패키지를 생성하면 기본으로 #stringify
매크로가 작성되어 있을 거다. 이걸 이용해서 대강 패키지의 구조와 매크로가 작성 순서를 알아볼 수 있다.
// CodableMacro.swift
// 실제로 외부에 제공되는 인터페이스들 정의
@freestanding(expression)
public macro stringify<T>(_ value: T) -> (T, String) = #externalMacro(module: "CodableMacroCore", type: "StringifyMacro")
#externalMacro(module: "CodableMacroCore", type: "StringifyMacro")
는 매크로 구현부는 CodableMacroCore 폴더(모듈) 안의 StringfyMacro 타입을 참고하라고 말해주는거다. // CodableMacroCore.swift
// 실제 #stringify 매크로의 구현부
import SwiftCompilerPlugin // Swift 컴파일러와 연결되어 매크로 기능을 등록하고 실행할 수 있게 해준다.
import SwiftSyntax // 소스 코드를 Sytax Tree 구조로 표현해준다.
import SwiftSyntaxBuilder // Syntax Tree 구성을 위한 편리 API를 제공한다.
import SwiftSyntaxMacros // 매크로 작성에 필요한 프로토콜과 타입을 제공한다.
public struct StringifyMacro: ExpressionMacro {
public static func expansion(
of node: some FreestandingMacroExpansionSyntax,
in context: some MacroExpansionContext
) -> ExprSyntax {
guard let argument = node.arguments.first?.expression else {
fatalError("compiler bug: the macro does not have any arguments")
}
return "(\(argument), \(literal: argument.description))"
}
}
/// 매크로 기능을 Swift 컴파일러에 등록하는 진입점
/// @main은 이 구조체가 프로그램의 시작점임을 나타낸다.
///
/// CompilerPlugin을 채택하여:
/// 1. 컴파일러가 이 매크로 패키지를 플러그인으로 인식하게 하고
/// 2. providingMacros 배열을 통해 사용 가능한 매크로 목록을 컴파일러에 알린다.
///
/// 새로운 매크로를 추가하려면 providingMacros 배열에 해당 매크로 타입을 추가해야한다.
@main
struct CodableMacroPlugin: CompilerPlugin {
let providingMacros: [Macro.Type] = [
StringifyMacro.self,
]
}
// CodableMacroClient로 scheme을 변경하면 테스트 가능
import CodableMacro
let a = 17
let b = 25
let (result, code) = #stringify(a + b)
print("The value \(result) was produced by the code \"\(code)\"")