test.rs 11.4 KB
Newer Older
1
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
2 3 4 5 6 7 8 9 10
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

11
use std::cell::RefCell;
12
use std::char;
13
use std::io;
14
use std::io::{Command, TempDir};
15 16
use std::os;
use std::str;
17
use std::string::String;
F
Felix S. Klock II 已提交
18
use std::unstable::dynamic_lib::DynamicLibrary;
19

20
use collections::{HashSet, HashMap};
L
Liigo Zhuang 已提交
21
use testing;
A
Alex Crichton 已提交
22
use rustc::back::link;
N
Nick Cameron 已提交
23
use rustc::driver::config;
24 25
use rustc::driver::driver;
use rustc::driver::session;
26
use rustc::metadata::creader::Loader;
27 28
use syntax::ast;
use syntax::codemap::{CodeMap, dummy_spanned};
29
use syntax::diagnostic;
30
use syntax::parse::token;
31 32 33 34 35 36 37 38 39

use core;
use clean;
use clean::Clean;
use fold::DocFolder;
use html::markdown;
use passes;
use visit_ast::RustdocVisitor;

40
pub fn run(input: &str,
41
           cfgs: Vec<String>,
42
           libs: HashSet<Path>,
43
           mut test_args: Vec<String>)
44
           -> int {
45 46
    let input_path = Path::new(input);
    let input = driver::FileInput(input_path.clone());
47

N
Nick Cameron 已提交
48
    let sessopts = config::Options {
E
Eduard Burtescu 已提交
49 50
        maybe_sysroot: Some(os::self_exe_path().unwrap().dir_path()),
        addl_lib_search_paths: RefCell::new(libs.clone()),
N
Nick Cameron 已提交
51 52
        crate_types: vec!(config::CrateTypeDylib),
        ..config::basic_options().clone()
53 54 55
    };


E
Eduard Burtescu 已提交
56
    let codemap = CodeMap::new();
57
    let diagnostic_handler = diagnostic::default_handler(diagnostic::Auto);
58
    let span_diagnostic_handler =
E
Eduard Burtescu 已提交
59
    diagnostic::mk_span_handler(diagnostic_handler, codemap);
60

N
Nick Cameron 已提交
61
    let sess = session::build_session_(sessopts,
62
                                      Some(input_path.clone()),
63 64
                                      span_diagnostic_handler);

N
Nick Cameron 已提交
65
    let mut cfg = config::build_configuration(&sess);
66
    cfg.extend(cfgs.move_iter().map(|cfg_| {
67
        let cfg_ = token::intern_and_get_ident(cfg_.as_slice());
68 69
        @dummy_spanned(ast::MetaWord(cfg_))
    }));
E
Eduard Burtescu 已提交
70
    let krate = driver::phase_1_parse_input(&sess, cfg, &input);
E
Eduard Burtescu 已提交
71
    let (krate, _) = driver::phase_2_configure_and_expand(&sess, &mut Loader::new(&sess), krate,
E
Eduard Burtescu 已提交
72
                                                          &from_str("rustdoc-test").unwrap());
73 74

    let ctx = @core::DocContext {
75
        krate: krate,
E
Eduard Burtescu 已提交
76
        maybe_typed: core::NotTyped(sess),
77
        src: input_path,
78
        external_paths: RefCell::new(Some(HashMap::new())),
79 80
        external_traits: RefCell::new(None),
        external_typarams: RefCell::new(None),
81
        inlined: RefCell::new(None),
82
        populated_crate_impls: RefCell::new(HashSet::new()),
83
    };
84
    super::ctxtkey.replace(Some(ctx));
85

A
Alex Crichton 已提交
86
    let mut v = RustdocVisitor::new(ctx, None);
87 88 89 90
    v.visit(&ctx.krate);
    let krate = v.clean();
    let (krate, _) = passes::unindent_comments(krate);
    let (krate, _) = passes::collapse_docs(krate);
91

92
    let mut collector = Collector::new(krate.name.to_string(),
93 94 95
                                       libs,
                                       false,
                                       false);
96
    collector.fold_crate(krate);
97

98
    test_args.unshift("rustdoctest".to_string());
99

100
    testing::test_main(test_args.as_slice(),
101
                       collector.tests.move_iter().collect());
102 103 104
    0
}

105
fn runtest(test: &str, cratename: &str, libs: HashSet<Path>, should_fail: bool,
106 107
           no_run: bool, loose_feature_gating: bool) {
    let test = maketest(test, cratename, loose_feature_gating);
108
    let input = driver::StrInput(test.to_string());
109

N
Nick Cameron 已提交
110
    let sessopts = config::Options {
E
Eduard Burtescu 已提交
111 112
        maybe_sysroot: Some(os::self_exe_path().unwrap().dir_path()),
        addl_lib_search_paths: RefCell::new(libs),
N
Nick Cameron 已提交
113
        crate_types: vec!(config::CrateTypeExecutable),
114
        output_types: vec!(link::OutputTypeExe),
115
        no_trans: no_run,
N
Nick Cameron 已提交
116
        cg: config::CodegenOptions {
117
            prefer_dynamic: true,
N
Nick Cameron 已提交
118
            .. config::basic_codegen_options()
119
        },
N
Nick Cameron 已提交
120
        ..config::basic_options().clone()
121 122
    };

123 124 125 126 127 128 129 130 131 132 133
    // Shuffle around a few input and output handles here. We're going to pass
    // an explicit handle into rustc to collect output messages, but we also
    // want to catch the error message that rustc prints when it fails.
    //
    // We take our task-local stderr (likely set by the test runner), and move
    // it into another task. This helper task then acts as a sink for both the
    // stderr of this task and stderr of rustc itself, copying all the info onto
    // the stderr channel we originally started with.
    //
    // The basic idea is to not use a default_handler() for rustc, and then also
    // not print things by default to the actual stderr.
134 135
    let (tx, rx) = channel();
    let w1 = io::ChanWriter::new(tx);
136
    let w2 = w1.clone();
137
    let old = io::stdio::set_stderr(box w1);
138
    spawn(proc() {
139
        let mut p = io::ChanReader::new(rx);
140
        let mut err = old.unwrap_or(box io::stderr() as Box<Writer:Send>);
141 142
        io::util::copy(&mut p, &mut err).unwrap();
    });
143
    let emitter = diagnostic::EmitterWriter::new(box w2);
144 145

    // Compile the code
E
Eduard Burtescu 已提交
146
    let codemap = CodeMap::new();
147
    let diagnostic_handler = diagnostic::mk_handler(box emitter);
148
    let span_diagnostic_handler =
E
Eduard Burtescu 已提交
149
        diagnostic::mk_span_handler(diagnostic_handler, codemap);
150

N
Nick Cameron 已提交
151
    let sess = session::build_session_(sessopts,
152
                                      None,
153 154 155 156
                                      span_diagnostic_handler);

    let outdir = TempDir::new("rustdoctest").expect("rustdoc needs a tempdir");
    let out = Some(outdir.path().clone());
N
Nick Cameron 已提交
157
    let cfg = config::build_configuration(&sess);
F
Felix S. Klock II 已提交
158
    let libdir = sess.target_filesearch().get_lib_path();
159 160
    driver::compile_input(sess, cfg, &input, &out, &None);

161 162
    if no_run { return }

163
    // Run the code!
F
Felix S. Klock II 已提交
164 165 166 167 168 169 170 171 172 173 174 175
    //
    // We're careful to prepend the *target* dylib search path to the child's
    // environment to ensure that the target loads the right libraries at
    // runtime. It would be a sad day if the *host* libraries were loaded as a
    // mistake.
    let exe = outdir.path().join("rust_out");
    let env = {
        let mut path = DynamicLibrary::search_path();
        path.insert(0, libdir.clone());

        // Remove the previous dylib search path var
        let var = DynamicLibrary::envvar();
176
        let mut env: Vec<(String,String)> = os::env().move_iter().collect();
F
Felix S. Klock II 已提交
177 178 179 180 181 182 183
        match env.iter().position(|&(ref k, _)| k.as_slice() == var) {
            Some(i) => { env.remove(i); }
            None => {}
        };

        // Add the new dylib search path var
        let newpath = DynamicLibrary::create_path(path.as_slice());
R
Richo Healey 已提交
184 185
        env.push((var.to_string(),
                  str::from_utf8(newpath.as_slice()).unwrap().to_string()));
F
Felix S. Klock II 已提交
186 187 188
        env
    };
    match Command::new(exe).env(env.as_slice()).output() {
189 190 191 192
        Err(e) => fail!("couldn't run the test: {}{}", e,
                        if e.kind == io::PermissionDenied {
                            " - maybe your tempdir is mounted with noexec?"
                        } else { "" }),
A
Alex Crichton 已提交
193
        Ok(out) => {
194 195 196
            if should_fail && out.status.success() {
                fail!("test executable succeeded when it should have failed");
            } else if !should_fail && !out.status.success() {
S
Steven Fackler 已提交
197 198
                fail!("test executable failed:\n{}",
                      str::from_utf8(out.error.as_slice()));
199 200 201 202 203
            }
        }
    }
}

204 205
fn maketest(s: &str, cratename: &str, loose_feature_gating: bool) -> String {
    let mut prog = String::from_str(r"
206 207
#![deny(warnings)]
#![allow(unused_variable, dead_assignment, unused_mut, attribute_usage, dead_code)]
208
");
209 210 211 212

    if loose_feature_gating {
        // FIXME #12773: avoid inserting these when the tutorial & manual
        // etc. have been updated to not use them so prolifically.
213
        prog.push_str("#![feature(macro_rules, globs, struct_variant, managed_boxes) ]\n");
214 215
    }

216 217
    if !s.contains("extern crate") {
        if s.contains(cratename) {
218 219
            prog.push_str(format!("extern crate {};\n",
                                  cratename).as_slice());
220
        }
221 222 223 224 225 226 227 228 229
    }
    if s.contains("fn main") {
        prog.push_str(s);
    } else {
        prog.push_str("fn main() {\n");
        prog.push_str(s);
        prog.push_str("\n}");
    }

230
    return prog
231 232 233
}

pub struct Collector {
234
    pub tests: Vec<testing::TestDescAndFn>,
235
    names: Vec<String>,
236 237 238
    libs: HashSet<Path>,
    cnt: uint,
    use_headers: bool,
239 240
    current_header: Option<String>,
    cratename: String,
241 242

    loose_feature_gating: bool
243 244 245
}

impl Collector {
246
    pub fn new(cratename: String, libs: HashSet<Path>,
247
               use_headers: bool, loose_feature_gating: bool) -> Collector {
248
        Collector {
249 250
            tests: Vec::new(),
            names: Vec::new(),
251 252 253 254
            libs: libs,
            cnt: 0,
            use_headers: use_headers,
            current_header: None,
255 256 257
            cratename: cratename,

            loose_feature_gating: loose_feature_gating
258 259 260
        }
    }

261
    pub fn add_test(&mut self, test: String, should_fail: bool, no_run: bool, should_ignore: bool) {
262 263
        let name = if self.use_headers {
            let s = self.current_header.as_ref().map(|s| s.as_slice()).unwrap_or("");
A
Alex Crichton 已提交
264
            format!("{}_{}", s, self.cnt)
265
        } else {
A
Alex Crichton 已提交
266
            format!("{}_{}", self.names.connect("::"), self.cnt)
267
        };
268
        self.cnt += 1;
E
Eduard Burtescu 已提交
269
        let libs = self.libs.clone();
R
Richo Healey 已提交
270
        let cratename = self.cratename.to_string();
271
        let loose_feature_gating = self.loose_feature_gating;
272
        debug!("Creating test {}: {}", name, test);
L
Liigo Zhuang 已提交
273 274 275
        self.tests.push(testing::TestDescAndFn {
            desc: testing::TestDesc {
                name: testing::DynTestName(name),
276
                ignore: should_ignore,
277
                should_fail: false, // compiler failures are test failures
278
            },
L
Liigo Zhuang 已提交
279
            testfn: testing::DynTestFn(proc() {
280
                runtest(test.as_slice(),
281
                        cratename.as_slice(),
282 283 284 285
                        libs,
                        should_fail,
                        no_run,
                        loose_feature_gating);
286 287 288
            }),
        });
    }
289 290 291 292 293 294 295 296 297 298 299 300

    pub fn register_header(&mut self, name: &str, level: u32) {
        if self.use_headers && level == 1 {
            // we use these headings as test names, so it's good if
            // they're valid identifiers.
            let name = name.chars().enumerate().map(|(i, c)| {
                    if (i == 0 && char::is_XID_start(c)) ||
                        (i != 0 && char::is_XID_continue(c)) {
                        c
                    } else {
                        '_'
                    }
301
                }).collect::<String>();
302 303 304 305 306 307

            // new header => reset count.
            self.cnt = 0;
            self.current_header = Some(name);
        }
    }
308 309 310 311 312 313
}

impl DocFolder for Collector {
    fn fold_item(&mut self, item: clean::Item) -> Option<clean::Item> {
        let pushed = match item.name {
            Some(ref name) if name.len() == 0 => false,
314
            Some(ref name) => { self.names.push(name.to_string()); true }
315 316 317 318 319
            None => false
        };
        match item.doc_value() {
            Some(doc) => {
                self.cnt = 0;
320
                markdown::find_testable_code(doc, &mut *self);
321 322 323 324 325 326 327 328 329 330
            }
            None => {}
        }
        let ret = self.fold_item_recur(item);
        if pushed {
            self.names.pop();
        }
        return ret;
    }
}