1
1
use proc_macro2:: { Span , TokenStream } ;
2
+ use std:: fs;
2
3
use std:: path:: { Path , PathBuf } ;
3
4
use syn:: parse:: { Error , Parse , ParseStream , Result } ;
4
5
use syn:: punctuated:: Punctuated ;
5
6
use syn:: { token, Token } ;
6
- use wit_bindgen_core:: wit_parser:: { PackageId , Resolve , UnresolvedPackage , WorldId } ;
7
+ use wit_bindgen_core:: wit_parser:: { self , PackageId , Resolve , UnresolvedPackage , WorldId } ;
7
8
use wit_bindgen_rust:: Opts ;
8
9
9
10
#[ proc_macro]
@@ -32,6 +33,7 @@ impl Parse for Config {
32
33
let mut opts = Opts :: default ( ) ;
33
34
let mut world = None ;
34
35
let mut source = None ;
36
+ let mut substitutions = None ;
35
37
36
38
if input. peek ( token:: Brace ) {
37
39
let content;
@@ -57,6 +59,24 @@ impl Parse for Config {
57
59
}
58
60
source = Some ( Source :: Inline ( s. value ( ) ) ) ;
59
61
}
62
+ Opt :: SubstitutionsPath ( s) => {
63
+ if substitutions. is_some ( ) {
64
+ return Err ( Error :: new (
65
+ s. span ( ) ,
66
+ "cannot specify second substitutions" ,
67
+ ) ) ;
68
+ }
69
+ substitutions = Some ( Source :: Path ( s. value ( ) ) ) ;
70
+ }
71
+ Opt :: SubstitutionsInline ( s) => {
72
+ if substitutions. is_some ( ) {
73
+ return Err ( Error :: new (
74
+ s. span ( ) ,
75
+ "cannot specify second substitutions" ,
76
+ ) ) ;
77
+ }
78
+ substitutions = Some ( Source :: Inline ( s. value ( ) ) ) ;
79
+ }
60
80
Opt :: UseStdFeature => opts. std_feature = true ,
61
81
Opt :: RawStrings => opts. raw_strings = true ,
62
82
Opt :: MacroExport => opts. macro_export = true ,
@@ -71,8 +91,8 @@ impl Parse for Config {
71
91
source = Some ( Source :: Path ( input. parse :: < syn:: LitStr > ( ) ?. value ( ) ) ) ;
72
92
}
73
93
}
74
- let ( resolve, pkg, files) =
75
- parse_source ( & source ) . map_err ( |err| Error :: new ( call_site, format ! ( "{err:?}" ) ) ) ?;
94
+ let ( resolve, pkg, files) = parse_source ( & source , & substitutions )
95
+ . map_err ( |err| Error :: new ( call_site, format ! ( "{err:?}" ) ) ) ?;
76
96
let world = resolve
77
97
. select_world ( pkg, world. as_deref ( ) )
78
98
. map_err ( |e| Error :: new ( call_site, format ! ( "{e:?}" ) ) ) ?;
@@ -85,7 +105,10 @@ impl Parse for Config {
85
105
}
86
106
}
87
107
88
- fn parse_source ( source : & Option < Source > ) -> anyhow:: Result < ( Resolve , PackageId , Vec < PathBuf > ) > {
108
+ fn parse_source (
109
+ source : & Option < Source > ,
110
+ substitutions : & Option < Source > ,
111
+ ) -> anyhow:: Result < ( Resolve , PackageId , Vec < PathBuf > ) > {
89
112
let mut resolve = Resolve :: default ( ) ;
90
113
let mut files = Vec :: new ( ) ;
91
114
let root = PathBuf :: from ( std:: env:: var ( "CARGO_MANIFEST_DIR" ) . unwrap ( ) ) ;
@@ -108,7 +131,14 @@ fn parse_source(source: &Option<Source>) -> anyhow::Result<(Resolve, PackageId,
108
131
Some ( Source :: Path ( s) ) => parse ( & root. join ( & s) ) ?,
109
132
None => parse ( & root. join ( "wit" ) ) ?,
110
133
} ;
111
-
134
+ match substitutions {
135
+ Some ( Source :: Inline ( s) ) => wit_parser:: expand ( & mut resolve, toml:: from_str ( s) ?) ?,
136
+ Some ( Source :: Path ( s) ) => wit_parser:: expand (
137
+ & mut resolve,
138
+ toml:: from_str ( & fs:: read_to_string ( & root. join ( & s) ) ?) ?,
139
+ ) ?,
140
+ None => ( ) ,
141
+ }
112
142
Ok ( ( resolve, pkg, files) )
113
143
}
114
144
@@ -146,12 +176,16 @@ mod kw {
146
176
syn:: custom_keyword!( world) ;
147
177
syn:: custom_keyword!( path) ;
148
178
syn:: custom_keyword!( inline) ;
179
+ syn:: custom_keyword!( substitutions_path) ;
180
+ syn:: custom_keyword!( substitutions_inline) ;
149
181
}
150
182
151
183
enum Opt {
152
184
World ( syn:: LitStr ) ,
153
185
Path ( syn:: LitStr ) ,
154
186
Inline ( syn:: LitStr ) ,
187
+ SubstitutionsPath ( syn:: LitStr ) ,
188
+ SubstitutionsInline ( syn:: LitStr ) ,
155
189
UseStdFeature ,
156
190
RawStrings ,
157
191
MacroExport ,
@@ -171,6 +205,14 @@ impl Parse for Opt {
171
205
input. parse :: < kw:: inline > ( ) ?;
172
206
input. parse :: < Token ! [ : ] > ( ) ?;
173
207
Ok ( Opt :: Inline ( input. parse ( ) ?) )
208
+ } else if l. peek ( kw:: substitutions_path) {
209
+ input. parse :: < kw:: substitutions_path > ( ) ?;
210
+ input. parse :: < Token ! [ : ] > ( ) ?;
211
+ Ok ( Opt :: SubstitutionsPath ( input. parse ( ) ?) )
212
+ } else if l. peek ( kw:: substitutions_inline) {
213
+ input. parse :: < kw:: substitutions_inline > ( ) ?;
214
+ input. parse :: < Token ! [ : ] > ( ) ?;
215
+ Ok ( Opt :: SubstitutionsInline ( input. parse ( ) ?) )
174
216
} else if l. peek ( kw:: world) {
175
217
input. parse :: < kw:: world > ( ) ?;
176
218
input. parse :: < Token ! [ : ] > ( ) ?;
0 commit comments