forked from swiftlang/swift
- Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmacro_expand_body.swift
111 lines (90 loc) · 3.13 KB
/
macro_expand_body.swift
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
106
107
108
109
110
111
// REQUIRES: swift_swift_parser, executable_test, asserts, concurrency
// REQUIRES: swift_feature_PreambleMacros
// RUN: %empty-directory(%t)
// RUN: %host-build-swift -swift-version 5 -emit-library -o %t/%target-library-name(MacroDefinition) -module-name=MacroDefinition %S/Inputs/syntax_macro_definitions.swift -g -no-toolchain-stdlib-rpath -swift-version 5
// Diagnostics testing
// RUN: %target-typecheck-verify-swift -swift-version 5 -enable-experimental-feature PreambleMacros -load-plugin-library %t/%target-library-name(MacroDefinition) -module-name MacroUser -DTEST_DIAGNOSTICS
// Execution testing
// RUN: %target-build-swift -swift-version 5 -g -enable-experimental-feature PreambleMacros -load-plugin-library %t/%target-library-name(MacroDefinition) %s -o %t/main -module-name MacroUser
// RUN: %target-codesign %t/main
// RUN: %target-run %t/main | %FileCheck %s
@attached(body)
macro Remote()= #externalMacro(module:"MacroDefinition", type:"RemoteBodyMacro")
@attached(preamble)
macro Traced()= #externalMacro(module:"MacroDefinition", type:"TracedPreambleMacro")
@attached(preamble, names:named(logger))
macro Logged()= #externalMacro(module:"MacroDefinition", type:"LoggerMacro")
protocolConjureRemoteValue{
staticfunc conjureValue()->Self
}
extensionString:ConjureRemoteValue{
staticfunc conjureValue()->String{""}
}
structLogger{
func log(entering function:String){
print("Logger entering \(function)")
}
func log(_ message:String){
print("--- \(message)")
}
func log(exiting function:String){
print("Logger exiting \(function)")
}
}
func log(_ message:String){
print(message)
}
@available(SwiftStdlib 5.1,*)
func remoteCall<Result:ConjureRemoteValue>(function:String, arguments:[String:Any])asyncthrows->Result{
letprintedArgs= arguments.keys.sorted().map{ key in
"\(key): \(arguments[key]!)"
}.joined(separator:", ")
print("Remote call \(function)(\(printedArgs))")
returnResult.conjureValue()
}
@available(SwiftStdlib 5.1,*)
@Remote
func f(a:Int, b:String)asyncthrows-> String
@Traced
func doubleTheValue(value:Int)->Int{
return value *2
}
@Logged
func useLogger(){
letx=1
logger.log("use it")
print(x)
}
@available(SwiftStdlib 5.1,*)
@Remote
@Traced
@Logged
func g(a:Int, b:String)asyncthrows->String{
doesNotTypeCheck()
}
#if compiler(>=6.0) && TEST_DIAGNOSTICS
@available(SwiftStdlib 5.1,*)
@Remote
func h(a:Int, b:String)asyncthrows->String{
does not
// expected-error@-1{{consecutive statements on a line must be separated by ';'}}
parse
}
#endif
// CHECK: Entering doubleTheValue(value: 7)
// CHECK-NEXT: Exiting doubleTheValue(value:)
_ =doubleTheValue(value:7)
if #available(SwiftStdlib 5.1,*){
// CHECK: Remote call f(a: 5, b: Hello)
print(tryawaitf(a:5, b:"Hello"))
// CHECK: Entering g(a: 5, b: World)
// CHECK: Logger entering g(a: 5, b: World)
// CHECK: Remote call g(a: 5, b: World)
// CHECK: Logger exiting g(a:b:)
// CHECK: Exiting g(a:b:)
print(tryawaitg(a:5, b:"World"))
}
// CHECK: Logger entering useLogger()
// CHECK: --- use it
// CHECK: Logger exiting useLogger()
useLogger()