42
42
#![ feature( staged_api) ]
43
43
#![ feature( question_mark) ]
44
44
#![ feature( panic_unwind) ]
45
+ #![ feature( mpsc_recv_timeout) ]
45
46
46
47
extern crate getopts;
47
48
extern crate term;
@@ -73,6 +74,8 @@ use std::sync::{Arc, Mutex};
73
74
use std:: thread;
74
75
use std:: time:: { Instant , Duration } ;
75
76
77
+ const TEST_WARN_TIMEOUT_S : u64 = 60 ;
78
+
76
79
// to be used by rustc to compile tests in libtest
77
80
pub mod test {
78
81
pub use { Bencher , TestName , TestResult , TestDesc , TestDescAndFn , TestOpts , TrFailed ,
@@ -592,6 +595,12 @@ impl<T: Write> ConsoleTestState<T> {
592
595
}
593
596
}
594
597
598
+ pub fn write_timeout ( & mut self , desc : & TestDesc ) -> io:: Result < ( ) > {
599
+ self . write_plain ( & format ! ( "test {} has been running for over {} seconds\n " ,
600
+ desc. name,
601
+ TEST_WARN_TIMEOUT_S ) )
602
+ }
603
+
595
604
pub fn write_log ( & mut self , test : & TestDesc , result : & TestResult ) -> io:: Result < ( ) > {
596
605
match self . log_out {
597
606
None => Ok ( ( ) ) ,
@@ -709,6 +718,7 @@ pub fn run_tests_console(opts: &TestOpts, tests: Vec<TestDescAndFn>) -> io::Resu
709
718
match ( * event) . clone ( ) {
710
719
TeFiltered ( ref filtered_tests) => st. write_run_start ( filtered_tests. len ( ) ) ,
711
720
TeWait ( ref test, padding) => st. write_test_start ( test, padding) ,
721
+ TeTimeout ( ref test) => st. write_timeout ( test) ,
712
722
TeResult ( test, result, stdout) => {
713
723
st. write_log ( & test, & result) ?;
714
724
st. write_result ( & result) ?;
@@ -830,6 +840,7 @@ enum TestEvent {
830
840
TeFiltered ( Vec < TestDesc > ) ,
831
841
TeWait ( TestDesc , NamePadding ) ,
832
842
TeResult ( TestDesc , TestResult , Vec < u8 > ) ,
843
+ TeTimeout ( TestDesc ) ,
833
844
}
834
845
835
846
pub type MonitorMsg = ( TestDesc , TestResult , Vec < u8 > ) ;
@@ -838,6 +849,9 @@ pub type MonitorMsg = (TestDesc, TestResult, Vec<u8>);
838
849
fn run_tests < F > ( opts : & TestOpts , tests : Vec < TestDescAndFn > , mut callback : F ) -> io:: Result < ( ) >
839
850
where F : FnMut ( TestEvent ) -> io:: Result < ( ) >
840
851
{
852
+ use std:: collections:: HashMap ;
853
+ use std:: sync:: mpsc:: RecvTimeoutError ;
854
+
841
855
let mut filtered_tests = filter_tests ( opts, tests) ;
842
856
if !opts. bench_benchmarks {
843
857
filtered_tests = convert_benchmarks_to_tests ( filtered_tests) ;
@@ -867,6 +881,29 @@ fn run_tests<F>(opts: &TestOpts, tests: Vec<TestDescAndFn>, mut callback: F) ->
867
881
868
882
let ( tx, rx) = channel :: < MonitorMsg > ( ) ;
869
883
884
+ let mut running_tests: HashMap < TestDesc , Instant > = HashMap :: new ( ) ;
885
+
886
+ fn get_timed_out_tests ( running_tests : & mut HashMap < TestDesc , Instant > ) -> Vec < TestDesc > {
887
+ let now = Instant :: now ( ) ;
888
+ let timed_out = running_tests. iter ( )
889
+ . filter_map ( |( desc, timeout) | if & now >= timeout { Some ( desc. clone ( ) ) } else { None } )
890
+ . collect ( ) ;
891
+ for test in & timed_out {
892
+ running_tests. remove ( test) ;
893
+ }
894
+ timed_out
895
+ } ;
896
+
897
+ fn calc_timeout ( running_tests : & HashMap < TestDesc , Instant > ) -> Option < Duration > {
898
+ running_tests. values ( ) . min ( ) . map ( |next_timeout| {
899
+ let now = Instant :: now ( ) ;
900
+ if * next_timeout >= now {
901
+ * next_timeout - now
902
+ } else {
903
+ Duration :: new ( 0 , 0 )
904
+ } } )
905
+ } ;
906
+
870
907
while pending > 0 || !remaining. is_empty ( ) {
871
908
while pending < concurrency && !remaining. is_empty ( ) {
872
909
let test = remaining. pop ( ) . unwrap ( ) ;
@@ -876,11 +913,31 @@ fn run_tests<F>(opts: &TestOpts, tests: Vec<TestDescAndFn>, mut callback: F) ->
876
913
// that hang forever.
877
914
callback ( TeWait ( test. desc . clone ( ) , test. testfn . padding ( ) ) ) ?;
878
915
}
916
+ let timeout = Instant :: now ( ) + Duration :: from_secs ( TEST_WARN_TIMEOUT_S ) ;
917
+ running_tests. insert ( test. desc . clone ( ) , timeout) ;
879
918
run_test ( opts, !opts. run_tests , test, tx. clone ( ) ) ;
880
919
pending += 1 ;
881
920
}
882
921
883
- let ( desc, result, stdout) = rx. recv ( ) . unwrap ( ) ;
922
+ let mut res;
923
+ loop {
924
+ if let Some ( timeout) = calc_timeout ( & running_tests) {
925
+ res = rx. recv_timeout ( timeout) ;
926
+ for test in get_timed_out_tests ( & mut running_tests) {
927
+ callback ( TeTimeout ( test) ) ?;
928
+ }
929
+ if res != Err ( RecvTimeoutError :: Timeout ) {
930
+ break ;
931
+ }
932
+ } else {
933
+ res = rx. recv ( ) . map_err ( |_| RecvTimeoutError :: Disconnected ) ;
934
+ break ;
935
+ }
936
+ }
937
+
938
+ let ( desc, result, stdout) = res. unwrap ( ) ;
939
+ running_tests. remove ( & desc) ;
940
+
884
941
if concurrency != 1 {
885
942
callback ( TeWait ( desc. clone ( ) , PadNone ) ) ?;
886
943
}
0 commit comments