Skip to content

Commit 98d3b06

Browse files
authored
improve test structure (#533)
* Refactors test list declarations to be tuples of an explicit test ID, a comment string, and the args. * Moves most test list declarations to the single place they are used. * Simplifies running tests. * Deletes some unused regression test helper binaries.
1 parent 5e47403 commit 98d3b06

File tree

8 files changed

+1209
-1716
lines changed

8 files changed

+1209
-1716
lines changed

enclone_exec/tests/enclone_test2.rs

+289-419
Large diffs are not rendered by default.

enclone_exec/tests/enclone_test3.rs

+39-52
Original file line numberDiff line numberDiff line change
@@ -170,18 +170,23 @@ fn test_site_examples() {
170170
#[test]
171171
fn test_enclone_examples() {
172172
PrettyTrace::new().on();
173-
for t in 0..EXAMPLES.len() {
174-
let testn = format!("{}", EXAMPLES[t]);
175-
let out_file = format!("../enclone_help/src/example{}", t + 1);
173+
let examples = [
174+
(1, r###"BCR=123089 CDR3=CARRYFGVVADAFDIW"###),
175+
(
176+
2,
177+
r###"BCR=123085 GEX=123217 LVARSP=gex,IGHV2-5_g_μ CDR3=CALMGTYCSGDNCYSWFDPW"###,
178+
),
179+
];
180+
for (num, args) in examples {
181+
let out_file = format!("../enclone_help/src/example{}", num);
176182
let old = read_to_string(&out_file).unwrap();
177-
let args = testn.split(' ').collect::<Vec<&str>>();
178183
let mut new = Command::new(env!("CARGO_BIN_EXE_enclone"));
179184
let mut new = new.arg(format!(
180185
"PRE=../enclone-data/big_inputs/version{}",
181186
TEST_FILES_VERSION
182187
));
183-
for i in 0..args.len() {
184-
new = new.arg(&args[i]);
188+
for arg in args.split(' ') {
189+
new = new.arg(arg);
185190
}
186191
let new = new
187192
.arg("FORCE_EXTERNAL")
@@ -192,7 +197,7 @@ fn test_enclone_examples() {
192197
if new.status.code() != Some(0) {
193198
eprint!(
194199
"\nenclone_test_examples: example{} failed to execute, stderr =\n{}",
195-
t + 1,
200+
num,
196201
strme(&new.stderr),
197202
);
198203
eprintln!("If it's not clear what is happening, make sure you've run ./build.\n");
@@ -201,7 +206,7 @@ fn test_enclone_examples() {
201206
if old != new2 {
202207
eprintln!(
203208
"\nenclone_test_examples: the file enclone_help/src/example{} is not up to date\n",
204-
t + 1
209+
num
205210
);
206211
eprintln!("old output =\n{}", old);
207212
eprintln!("new output =\n{}\n", new2);
@@ -371,19 +376,17 @@ fn test_enclone_prebuild() {
371376
remove_file(&mb).unwrap();
372377
}
373378

374-
let test_id = 48;
375-
let it = test_id - 1;
376-
let testn = TESTS[it];
377-
let out_file = format!("testx/inputs/outputs/enclone_test{}_output", test_id);
379+
let (test_num, comment, args) = TESTS[47];
380+
assert_eq!(48, test_num);
381+
let out_file = format!("testx/inputs/outputs/enclone_test{}_output", test_num);
378382
let old = read_to_string(&out_file).unwrap();
379-
let args = testn.split(' ').collect::<Vec<&str>>();
380383
let mut new = Command::new(env!("CARGO_BIN_EXE_enclone"));
381384
let mut new = new.arg(format!(
382385
"PRE=../enclone-data/big_inputs/version{}",
383386
TEST_FILES_VERSION
384387
));
385-
for i in 0..args.len() {
386-
new = new.arg(&args[i]);
388+
for arg in args.split(' ') {
389+
new = new.arg(arg);
387390
}
388391
// dubious use of expect:
389392
let new = new
@@ -398,7 +401,7 @@ fn test_enclone_prebuild() {
398401
"\nenclone_test_prebuild: first pass output has changed.\n\
399402
If you are OK with the new output, it should work to type:\n\
400403
enclone {} > enclone_exec/{}\n",
401-
testn, out_file
404+
args, out_file
402405
);
403406
eprintln!("old output =\n{}\n", old);
404407
eprintln!("new output =\n{}\n", new2);
@@ -407,16 +410,13 @@ fn test_enclone_prebuild() {
407410
}
408411

409412
// Second pass: run without PREBUILD
410-
411-
let testn = TESTS[it];
412-
let args = testn.split(' ').collect::<Vec<&str>>();
413413
let mut new = Command::new(env!("CARGO_BIN_EXE_enclone"));
414414
let mut new = new.arg(format!(
415415
"PRE=../enclone-data/big_inputs/version{}",
416416
TEST_FILES_VERSION
417417
));
418-
for i in 0..args.len() {
419-
new = new.arg(&args[i]);
418+
for arg in args.split(' ') {
419+
new = new.arg(arg);
420420
}
421421
// dubious use of expect:
422422
let new = new
@@ -562,42 +562,29 @@ fn test_proto_write() -> Result<(), Error> {
562562

563563
#[cfg(not(feature = "cpu"))]
564564
#[test]
565-
fn test_annotated_example() {
565+
fn test_annotated_example() -> Result<(), String> {
566566
PrettyTrace::new().on();
567-
let it = 0;
568-
let test = "BCR=123085 CDR3=CTRDRDLRGATDAFDIW";
569-
let mut out = String::new();
570-
let mut log = String::new();
571-
let mut ok = false;
572567
run_test(
573568
env!("CARGO_BIN_EXE_enclone"),
574-
it,
575-
"",
576-
&test,
577569
"annotated_example_test",
578-
&mut ok,
579-
&mut log,
580-
&mut out,
581570
0,
582-
);
583-
print!("{}", log);
584-
if !ok {
585-
let mut log = Vec::<u8>::new();
586-
emit_red_escape(&mut log);
587-
fwriteln!(
588-
log,
589-
"Oh no: the results for the annotated example on the landing page have \
590-
changed. Assuming that\nthe change is intentional, to fix this you \
591-
need to do two things:\n\
592-
1. Update the test output.\n\
593-
2. Manually update the annotated example output.\n\
594-
Because the second item is such a big pain, we stopped doing it, but you should\n\
595-
check to make sure that it has not changed too much from what is shown."
596-
);
597-
emit_end_escape(&mut log);
598-
eprintln!("{}", strme(&log));
599-
panic!("failed");
600-
}
571+
1,
572+
"",
573+
"BCR=123085 CDR3=CTRDRDLRGATDAFDIW",
574+
)
575+
.map_err(|res| {
576+
format!(
577+
"{}\n\nOh no: the results for the annotated example on the landing page have \
578+
changed. Assuming that the change is intentional, to fix this you \
579+
need to do two things:\n\
580+
1. Update the test output.\n\
581+
2. Manually update the annotated example output.\n\
582+
Because the second item is such a big pain, we stopped doing it, but you should \
583+
check to make sure that it has not changed too much from what is shown.",
584+
res.log
585+
)
586+
})?;
587+
Ok(())
601588
}
602589

603590
// ▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓▓

enclone_testlist/src/lib.rs

-212
Original file line numberDiff line numberDiff line change
@@ -3,220 +3,8 @@
33
// Information about enclone tests.
44
pub mod main_testlist;
55

6-
pub fn enclone_testdata() -> String {
7-
include_str!["enclone.testdata"].to_string()
8-
}
9-
pub fn enclone_testdata_public_bcr_human() -> String {
10-
include_str!["testdata.public.bcr.human"].to_string()
11-
}
12-
pub fn enclone_testdata_public_tcr_human() -> String {
13-
include_str!["testdata.public.tcr.human"].to_string()
14-
}
15-
pub fn enclone_testdata_public_tcr_mouse() -> String {
16-
include_str!["testdata.public.tcr.mouse"].to_string()
17-
}
18-
pub fn enclone_testdata_public_gex_human() -> String {
19-
include_str!["testdata.public.gex.human"].to_string()
20-
}
21-
226
pub const TEST_FILES_VERSION: u8 = 15;
237

24-
// Unaccounted time test. This is separated out so that we can avoid running it in parallel with
25-
// other tests, to reduce the sporadic failure rate. Also this is off in GitHub Actions.
26-
27-
pub const UNAC_TESTS: [&str; 1] = [
28-
// 1. enforce no unaccounted time
29-
r###"BCR=123085 COMPE UNACCOUNTED NOPRINT EXPECT_OK"###,
30-
];
31-
32-
// Tests that are affected by the D region alignment algorithm. All such tests should go here.
33-
34-
pub const DTESTS: [&str; 15] = [
35-
// 1. test ALIGN_2ND<n>
36-
r###"BCR=123085 CDR3=CKVMLYDSRGSDYYYVMDVW ALIGN_2ND1 CVARS=d1_name"###,
37-
// 2. test JALIGN_2ND<n>
38-
r###"BCR=123085 CDR3=CKVMLYDSRGSDYYYVMDVW JALIGN_2ND1 CVARS=d2_name"###,
39-
// 3. test ALIGN_JALIGN_CONSISTENCY
40-
r###"BCR=123085 CELLS=1 CHAINS=2 ALIGN1 JALIGN1 ALIGN_JALIGN_CONSISTENCY AMINO=cdr3
41-
PLAIN NOPAGER EXPECT_OK"###,
42-
// 4. test D_INCONSISTENT, and lock number of inconsistencies
43-
r###"BCR=123085 D_INCONSISTENT CVARS=d1_name COMPLETE NGROUP"###,
44-
// 5. the JALIGN1 in this example had a boundary location that was off by one
45-
r###"BCR=165807 JALIGN1 AMINO=cdr3 CVARS=d1_score,d2_score CDR3=CAKEYYDFWSGYSDVRGVIPNIDYW"###,
46-
// 6. the JALIGN1 in this example had a boundary location that was off by one
47-
r###"BCR=123085 CELLS=2 JALIGN1 AMINO=cdr3 CVARS=d1_name CDR3=CAKAGPTESGYYVWYFDLW"###,
48-
// 7. test d_inconsistent_{%,n}
49-
r###"BCR=123085 GVARS=d_inconsistent_%,d_inconsistent_n NOPRINT SUMMARY SUMMARY_CLEAN"###,
50-
// 8. test ALIGN<n>
51-
r###"BCR=123085 CDR3=CKVMLYDSRGSDYYYVMDVW ALIGN1 CVARS=d1_name"###,
52-
// 9. test ALIGN<n> and JALIGN<n>, case where there's a D segment
53-
r###"BCR=85333 ALIGN1 JALIGN1 CDR3=CARGYDFWSGYLVGNWAGDYYYYMDVW"###,
54-
// 10. test ALIGN<n> and JALIGN<n>, case where there is no D segment
55-
r###"BCR=85333 ALIGN1 JALIGN1 CDR3=CAKGKGFRNYYYYMDVW"###,
56-
// 11. test d1 etc.
57-
r###"BCR=123085 CVARS=d1_name,d2_name,d_Δ,d_delta AMINO=cdr3 CDR3=CARVRDILTGDYGMDVW"###,
58-
// 12. test GROUP_VDJ_REFNAME_HEAVY (deprecated but supported)
59-
r###"BCR=86237 GROUP_VDJ_REFNAME_HEAVY CDR3="CAKAVAGKAVAGGWDYW|CAKVSTGIAVAGPGDYW" COMPLETE"###,
60-
// 13. test GROUP_VJ_REFNAME_HEAVY (deprecated but supported)
61-
r###"BCR=86237 GROUP_VJ_REFNAME_HEAVY CDR3="CARGVLWFGELGAFDIW|CARAGLGVVLAARGAFDIW""###,
62-
// 14. test placement of indel, needed shifting right
63-
r###"BCR=123085 CELLS=1 CHAINS=2 AMINO=cdr3 JALIGN2 CDR3=CAKDKSRPPTHYYGSGSYYSRILDNW"###,
64-
// 15. test placement of indel, needed shifting left
65-
r###"BCR=123085 CELLS=1 CHAINS=2 AMINO=cdr3 JALIGN2 CDR3=CARMAQFYSGSGTYYIGPYYFEYW"###,
66-
];
67-
68-
// Tests that are affected by the grouping algorithm. All such tests should go here, if not
69-
// already in DTESTS.
70-
71-
pub const GTESTS: [&str; 21] = [
72-
// 1. test 5/8 for newline correctness (this grouping option deprecated but supported)
73-
r###"BCR=85333 GROUP_VJ_REFNAME MIN_GROUP=2 AMINO= PLAIN SET_IN_STONE"###,
74-
// 2. test 6/8 for newline correctness (this grouping option deprecated but supported)
75-
r###"BCR=85333 GROUP_VJ_REFNAME MIN_GROUP=2 AMINO= PLAIN NGROUP SET_IN_STONE"###,
76-
// 3. test 7/8 for newline correctness (this grouping option deprecated but supported)
77-
r###"BCR=85333 GROUP_VJ_REFNAME MIN_GROUP=2 AMINO= PLAIN HTML SET_IN_STONE"###,
78-
// 4. test 8/8 for newline correctness (this grouping option deprecated but supported)
79-
r###"BCR=85333 GROUP_VJ_REFNAME MIN_GROUP=2 AMINO= PLAIN HTML NGROUP SET_IN_STONE"###,
80-
// 5. test of GROUP
81-
r###"BCR=123085 GROUP=vj_refname,cdr3_aa_heavy≥80%,cdr3_aa_light≥80% CVARS=cdr3_len
82-
AMINO=cdr3 CDR3="CARHLQWELP.*W""###,
83-
// 6. test of GROUP
84-
r###"BCR=123085 GROUP=vj_refname,len,cdr3_len MIN_GROUP=2 MIN_CHAINS=2 CDR3="CQQSY.*TLATF"
85-
CVARS=cdr3_len"###,
86-
// 7. test of GROUP
87-
r###"BCR=123085 GROUP=cdr3_aa_heavy≥100% MIN_GROUP=2 MIN_CHAINS=2 CVARS=cdr3_len
88-
CDR3=CARPKSDYIIDAFDIW"###,
89-
// 8. test of GROUP
90-
r###"BCR=123085 GROUP=cdr3_aa_light≥100% MIN_GROUP=2 MIN_CHAINS=2 CVARS=cdr3_len
91-
CDR3=CQTWGTGPWVF"###,
92-
// 9. test of GROUP
93-
r###"BCR=123085 GROUP=vj_refname,aa_heavy≥100% MIN_GROUP=2 MIN_CHAINS=2 CVARS=cdr3_len
94-
CDR3=CARVPYYYDRSYYYYGMDVW"###,
95-
// 10. test of AGROUP
96-
r###"BCR=123085 AGROUP AG_CENTER=from_filters CDR3=CARHSYSSGWYDEWDYW
97-
AG_DIST_FORMULA=cdr3_edit_distance AG_DIST_BOUND=top=2"###,
98-
// 11. test of AGROUP
99-
r###"BCR=123085 AGROUP AG_CENTER=from_filters CDR3=CAKDGGEHYYDSSGYYASYYFDYW
100-
AG_DIST_FORMULA=cdr3_edit_distance AG_DIST_BOUND=max=14"###,
101-
// 12. test of AGROUP
102-
r###"BCR=123085 AGROUP AG_CENTER=from_filters CDR3=CAKDGGEHYYDSSGYYASYYFDYW
103-
AG_DIST_FORMULA=cdr3_edit_distance AG_DIST_BOUND=max=13"###,
104-
// 13. test of AGROUP
105-
r###"BCR=123085 AGROUP AG_CENTER=copy_filters MIN_CELLS=2 MAX_CELLS=2
106-
AG_DIST_FORMULA=cdr3_edit_distance AG_DIST_BOUND=max=3 MIN_GROUP=2"###,
107-
// 14. test symmetric grouping stats
108-
r###"BCR=123085 GROUP=vj_refname,cdr3_aa_heavy≥80%,cdr3_aa_light≥80% NOPRINT
109-
SUMMARY SUMMARY_CLEAN"###,
110-
// 15. test of GROUP
111-
r###"BCR=123085 GROUP=cdr3_heavy≥100% MIN_GROUP=2 MIN_CHAINS=2 CVARS=cdr3_len
112-
CDR3=CARPKSDYIIDAFDIW"###,
113-
// 16. test of GROUP
114-
r###"BCR=123085 GROUP="cdr3_light>=100%" MIN_GROUP=2 MIN_CHAINS=2 CVARS=cdr3_len
115-
CDR3=CQTWGTGPWVF"###,
116-
// 17. test of GROUP
117-
r###"BCR=123085 GROUP=vj_refname,heavy≥96.6% MIN_GROUP=2 MIN_CHAINS=2
118-
CDR3="CARVIVGPKKLEGRLYSSSLHFDCW|CARVIVGPEKQEGRLYSSSLHFDYW" POUT=stdout PCOLS=vj_seq1"###,
119-
// 18. test of GROUP
120-
r###"BCR=123085 GROUP=vj_heavy_refname,cdr3_heavy_len,cdr3_heavy≥80% LVARS=n,donors,dref
121-
CVARS=const,cdr3_len AMINO=cdr3 CHAINS=2 MIN_GROUP=2
122-
CDR3="CARDLHGYDPYGMDVW|CARELRHYDTYGMDVW""###,
123-
// 19. test of GROUP
124-
r###"BCR=123085 GROUP=vj_refname,cdr3_light_len LVARS=n,donors,dref CVARS=const,cdr3_len
125-
AMINO=cdr3 CHAINS=2 MIN_GROUP=2 CDR3="CARESAVAGDMDVW|CARDYGDYRWWVDGMDVW""###,
126-
// 20. test of group
127-
r###"BCR=123085 GROUP=vj_refname GROUP_CDR3=CACFGRIGVVVRAAHYW"###,
128-
// 21. test of group, asserted at one time
129-
r###"BCR=123085 GROUP=vj_refname,cdr3_len MIN_GROUP=3 HONEY=out=stdout,color=var,u1
130-
EXPECT_OK"###,
131-
];
132-
133-
// Crash tests. These are tests to make sure that certain options do not result in a crash, even
134-
// when run on a large and complex dataset. The options are in groups because not all are
135-
// compatible with each other. The datasets are defined by a single fixed list, to be enlarged
136-
// over time based on discovery of pathologies in particular PUBLIC datasets.
137-
// All run with certain shared options.
138-
139-
pub const CRASH_DATA: &str = "BCR=\"45977;123085;testx/inputs/flaky\"";
140-
pub const CRASH_OPTS: &str = "NOPRINT BUILT_IN EXPECT_OK NO_PRE NFORCE";
141-
pub const CRASH_SETS: [&str; 6] = [
142-
/* 1 */ "CONP SEQC SUM MEAN BARCODES DIFF_STYLE=C1 GROUP_VJ_REFNAME",
143-
//
144-
/* 2 */ "CONX FULL_SEQC DIFF_STYLE=C2 POUT=stdout PCOLS=count_CAR",
145-
//
146-
/* 3 */
147-
"AMINO=fwr1,cdr1,fwr2,cdr2,fwr3,cdr3,fwr4 CVARS=d1_name,d2_name,d_delta,d_Δ,cigar",
148-
//
149-
/* 4 */
150-
"PLOT_BY_ISOTYPE=stdout MIN_CELLS=3 GROUP_VJ_REFNAME_HEAVY ALIGN1 JALIGN1",
151-
//
152-
/* 5 */
153-
"GROUP_VDJ_REFNAME_HEAVY GVARS=d_inconsistent_%,d_inconsistent_n",
154-
//
155-
/* 6 */
156-
"GROUP=vj_refname,cdr3_aa_heavy≥90%,cdr3_aa_light≥90%",
157-
];
158-
159-
// Test using datasets that are either in the extended public dataset collection,
160-
// or which require samtools.
161-
162-
pub const EXTENDED_TESTS: [&str; 12] = [
163-
// 1. Make sure that POUT works on full dataset.
164-
// If we experience failures on other PUBLIC ids, we can add them to this list.
165-
r###"BCR=86237 RE POUT=/dev/null NOPRINT EXPECT_OK NO_PRE NFORCE"###,
166-
// 2. tests nd2
167-
r###"BCR=47199,47200,47212 AMINO=cdr3 NCROSS LVARS=nd2 CDR3=CVKGKSGSFWYYFENW
168-
NO_PRE NFORCE"###,
169-
// 3. test sec and mem [requires samtools]
170-
r###"BCR=123085 GEX=123217 LVARSP=sec,mem CDR3=CVKDRVTGTITELDYW"###,
171-
// 4. crashed at one point
172-
r###"BCR=128037,128040 GEX=127798,127801 LVARSP=pe1 NOPRINT EXPECT_OK NO_PRE NFORCE"###,
173-
//
174-
// 5. this added because it got better when a bug in bads detection was fixed
175-
r###"TCR=163914 CDR3=CASRLGGEETQYF NO_PRE NFORCE"###,
176-
// 6. Test PCHAINS=max. For this we need a clonotype having at least five chains, and the
177-
// question is whether the header line represents cvars for all the chains. The output of
178-
// this is expected to change whenever variables are added.
179-
r###"BCR=123085,123089,124547 NWEAK_CHAINS NDOUBLET MIN_CHAINS=5 POUT=stdout PCHAINS=max
180-
NOPRINT RE NO_PRE NFORCE"###,
181-
// 7. test MIN_GROUP_DONORS
182-
r###"BCR="40953;43899" MIX_DONORS MIN_GROUP=2 NO_PRE NFORCE
183-
GROUP="cdr3_len,cdr3_aa_heavy>=85%,cdr3_aa_light>=85%,vj_refname" MIN_GROUP_DONORS=2"###,
184-
// 8. this asserted at one point
185-
r###"BUILT_IN GROUP=vj_refname,cdr3_aa_heavy≥90% MIN_CHAINS_EXACT=2 MIN_GROUP=2
186-
KEEP_CLONO_IF_CELL_MEAN="cdr3_len1>=18" BCR=1018096-1018098 JALIGN1 NO_PRE NFORCE
187-
EXPECT_OK"###,
188-
// 9. this clonotype included a junk chain before we made a change, and test "/outs"
189-
r###"TCR=163911/outs CDR3=CAPSAGDKIIF AMINO=donor NO_PRE NFORCE"###,
190-
// 10. test case where digit rows are just barely present
191-
r###"TCR=163911 CDR3=CASSLVQPSTDTQYF AMINO=donor NO_PRE NFORCE"###,
192-
// 11. this added because it got better when a noise filter was added, also tests u_max
193-
r###"TCR=163914 CDR3=CASSLVQPSTDTQYF CVARSP=u_max NO_PRE NFORCE"###,
194-
// 12. this added because it got better when a noise filter was added; also test FASTA
195-
r###"TCR=163914 CDR3=CAFRGGSYIPTF FASTA=stdout NO_PRE NFORCE"###,
196-
];
197-
198-
// Tests of internal features.
199-
200-
pub const INTERNAL_TESTS: [&str; 3] = [
201-
// 1. gave wrong result
202-
r###"123085 CDR3=CARDRIAGRFGYGMDVW NFORCE"###,
203-
// 2. test human + IMGT; note that specifying by number forces BCR+TCR reference checks
204-
r###"123085 REQUIRE_UNBROKEN_OK IMGT ACCEPT_BROKEN EXPECT_NULL"###,
205-
// 3. this crashed; it is not exactly an internal feature test but uses an internal feature
206-
// (IMGT) to exhibit the phenomenon
207-
r###"BCR=123085 IMGT RE ACCEPT_BROKEN POUT=stdout PCELL BARCODE=AGCAGCCCATTAGGCT-1
208-
EXPECT_OK"###,
209-
];
210-
211-
// List of examples in documentation.
212-
213-
pub const EXAMPLES: [&str; 2] = [
214-
// 1.
215-
r###"BCR=123089 CDR3=CARRYFGVVADAFDIW"###,
216-
// 2.
217-
r###"BCR=123085 GEX=123217 LVARSP=gex,IGHV2-5_g_μ CDR3=CALMGTYCSGDNCYSWFDPW"###,
218-
];
219-
2208
// List of examples on site.
2219

22210
pub const SITE_EXAMPLES: [(&str, &str); 28] = [

0 commit comments

Comments
 (0)