3
3
//! `./x.py test` (aka [`Kind::Test`]) is currently allowed to reach build steps in other modules.
4
4
//! However, this contains ~all test parts we expect people to be able to build and run locally.
5
5
6
+ use std:: collections:: HashSet ;
6
7
use std:: ffi:: { OsStr , OsString } ;
7
8
use std:: path:: { Path , PathBuf } ;
8
9
use std:: { env, fs, iter} ;
9
10
10
11
use clap_complete:: shells;
11
12
13
+ use crate :: core:: build_steps:: compile:: run_cargo;
12
14
use crate :: core:: build_steps:: doc:: DocumentationFormat ;
13
15
use crate :: core:: build_steps:: synthetic_targets:: MirOptPanicAbortSyntheticTarget ;
14
16
use crate :: core:: build_steps:: tool:: { self , SourceType , Tool } ;
@@ -2168,9 +2170,11 @@ NOTE: if you're sure you want to do this, please open an issue as to why. In the
2168
2170
#[ derive( Debug , Clone , PartialEq , Eq , Hash ) ]
2169
2171
struct BookTest {
2170
2172
compiler : Compiler ,
2173
+ target : TargetSelection ,
2171
2174
path : PathBuf ,
2172
2175
name : & ' static str ,
2173
2176
is_ext_doc : bool ,
2177
+ dependencies : Vec < & ' static str > ,
2174
2178
}
2175
2179
2176
2180
impl Step for BookTest {
@@ -2223,6 +2227,63 @@ impl BookTest {
2223
2227
// Books often have feature-gated example text.
2224
2228
rustbook_cmd. env ( "RUSTC_BOOTSTRAP" , "1" ) ;
2225
2229
rustbook_cmd. env ( "PATH" , new_path) . arg ( "test" ) . arg ( path) ;
2230
+
2231
+ // Books may also need to build dependencies. For example, `TheBook` has
2232
+ // code samples which use the `trpl` crate. For the `rustdoc` invocation
2233
+ // to find them them successfully, they need to be built first and their
2234
+ // paths used to generate the
2235
+ let libs = if !self . dependencies . is_empty ( ) {
2236
+ let mut lib_paths = vec ! [ ] ;
2237
+ for dep in self . dependencies {
2238
+ let mode = Mode :: ToolRustc ;
2239
+ let target = builder. config . build ;
2240
+ // CHECKME: is this correct, or should it be using `builder::Cargo::new`?
2241
+ let cargo = tool:: prepare_tool_cargo (
2242
+ builder,
2243
+ self . compiler ,
2244
+ mode,
2245
+ builder. config . build ,
2246
+ Kind :: Build ,
2247
+ dep,
2248
+ SourceType :: Submodule ,
2249
+ & [ ] ,
2250
+ ) ;
2251
+ // CHECKME: this is used for the "stamp" for this `run_cargo`;
2252
+ // is there a better way to do this?!?
2253
+ let dep_path = PathBuf :: from ( dep) ;
2254
+ let file_name = dep_path. file_name ( ) . unwrap ( ) ;
2255
+ let file_name = PathBuf :: from ( file_name) ;
2256
+
2257
+ let stamp = builder
2258
+ . cargo_out ( self . compiler , mode, target)
2259
+ . join ( file_name)
2260
+ . with_extension ( "stamp" ) ;
2261
+
2262
+ let output_paths = run_cargo ( builder, cargo, vec ! [ ] , & stamp, vec ! [ ] , false , false ) ;
2263
+ let directories = output_paths
2264
+ . into_iter ( )
2265
+ . filter_map ( |p| p. parent ( ) . map ( ToOwned :: to_owned) )
2266
+ . fold ( HashSet :: new ( ) , |mut set, dir| {
2267
+ set. insert ( dir) ;
2268
+ set
2269
+ } ) ;
2270
+ lib_paths. extend ( directories) ;
2271
+ }
2272
+ lib_paths
2273
+ } else {
2274
+ vec ! [ ]
2275
+ } ;
2276
+
2277
+ if !libs. is_empty ( ) {
2278
+ let comma_separated_paths = libs
2279
+ . into_iter ( )
2280
+ . map ( |path| format ! ( "{}" , path. display( ) ) )
2281
+ . collect :: < Vec < String > > ( )
2282
+ . join ( "," ) ;
2283
+
2284
+ rustbook_cmd. args ( vec ! [ String :: from( "--library-path" ) , comma_separated_paths] ) ;
2285
+ }
2286
+
2226
2287
builder. add_rust_test_threads ( & mut rustbook_cmd) ;
2227
2288
let _guard = builder. msg (
2228
2289
Kind :: Test ,
@@ -2281,12 +2342,14 @@ macro_rules! test_book {
2281
2342
$name: ident, $path: expr, $book_name: expr,
2282
2343
default =$default: expr
2283
2344
$( , submodules = $submodules: expr) ?
2345
+ $( , dependencies=$dependencies: expr) ?
2284
2346
;
2285
2347
) +) => {
2286
2348
$(
2287
2349
#[ derive( Debug , Clone , PartialEq , Eq , Hash ) ]
2288
2350
pub struct $name {
2289
2351
compiler: Compiler ,
2352
+ target: TargetSelection ,
2290
2353
}
2291
2354
2292
2355
impl Step for $name {
@@ -2301,6 +2364,7 @@ macro_rules! test_book {
2301
2364
fn make_run( run: RunConfig <' _>) {
2302
2365
run. builder. ensure( $name {
2303
2366
compiler: run. builder. compiler( run. builder. top_stage, run. target) ,
2367
+ target: run. target,
2304
2368
} ) ;
2305
2369
}
2306
2370
@@ -2310,11 +2374,22 @@ macro_rules! test_book {
2310
2374
builder. require_submodule( submodule, None ) ;
2311
2375
}
2312
2376
) *
2377
+
2378
+ let dependencies = vec![ ] ;
2379
+ $(
2380
+ let mut dependencies = dependencies;
2381
+ for dep in $dependencies {
2382
+ dependencies. push( dep) ;
2383
+ }
2384
+ ) ?
2385
+
2313
2386
builder. ensure( BookTest {
2314
2387
compiler: self . compiler,
2388
+ target: self . target,
2315
2389
path: PathBuf :: from( $path) ,
2316
2390
name: $book_name,
2317
2391
is_ext_doc: !$default,
2392
+ dependencies,
2318
2393
} ) ;
2319
2394
}
2320
2395
}
@@ -2329,7 +2404,7 @@ test_book!(
2329
2404
RustcBook , "src/doc/rustc" , "rustc" , default =true ;
2330
2405
RustByExample , "src/doc/rust-by-example" , "rust-by-example" , default =false , submodules=[ "src/doc/rust-by-example" ] ;
2331
2406
EmbeddedBook , "src/doc/embedded-book" , "embedded-book" , default =false , submodules=[ "src/doc/embedded-book" ] ;
2332
- TheBook , "src/doc/book" , "book" , default =false , submodules=[ "src/doc/book" ] ;
2407
+ TheBook , "src/doc/book" , "book" , default =false , submodules=[ "src/doc/book" ] , dependencies= [ "src/doc/book/packages/trpl" ] ;
2333
2408
UnstableBook , "src/doc/unstable-book" , "unstable-book" , default =true ;
2334
2409
EditionGuide , "src/doc/edition-guide" , "edition-guide" , default =false , submodules=[ "src/doc/edition-guide" ] ;
2335
2410
) ;
0 commit comments