@@ -3,25 +3,35 @@ import {
3
3
AllureRuntime ,
4
4
AllureStep ,
5
5
AllureTest ,
6
+ Attachment ,
6
7
ExecutableItem ,
7
8
ExecutableItemWrapper ,
8
9
FixtureResult ,
10
+ Status ,
9
11
StatusDetails ,
10
12
TestResult ,
11
13
} from 'allure-js-commons' ;
12
14
import getUuid from 'uuid-by-string' ;
13
15
import getUuidByString from 'uuid-by-string' ;
14
16
import { parseAllure } from 'allure-js-parser' ;
15
- import { copyFile , copyFileSync , existsSync , mkdirSync , readFile , readFileSync , writeFile , writeFileSync } from 'fs' ;
17
+ import { copyFileSync , existsSync , mkdirSync , readFile , readFileSync , writeFileSync } from 'fs' ;
16
18
import path , { basename } from 'path' ;
17
19
import glob from 'fast-glob' ;
18
20
import { ReporterOptions } from './allure' ;
19
21
import Debug from 'debug' ;
20
22
import { GlobalHooks } from './allure-global-hook' ;
21
- import { AllureTaskArgs , LabelName , Stage , Status , StatusType , UNKNOWN } from './allure-types' ;
22
- import { delay , extname , packageLog } from '../common' ;
23
+ import { AllureTaskArgs , LabelName , Stage , StatusType , UNKNOWN } from './allure-types' ;
24
+ import { extname , packageLog } from '../common' ;
23
25
import type { ContentType } from '../common/types' ;
24
26
import { randomUUID } from 'crypto' ;
27
+ import {
28
+ copyAttachments ,
29
+ copyFileCp ,
30
+ copyTest ,
31
+ mkdirSyncWithTry ,
32
+ waitWhileCondition ,
33
+ writeResultFile ,
34
+ } from './fs-tools' ;
25
35
26
36
const beforeEachHookName = '"before each" hook' ;
27
37
const beforeAllHookName = '"before all" hook' ;
@@ -37,24 +47,93 @@ const log = (...args: unknown[]) => {
37
47
debug ( args ) ;
38
48
} ;
39
49
40
- const writeTestFile = ( testFile : string , content : string , callBack : ( ) => void ) => {
41
- writeFile ( testFile , content , errWrite => {
42
- if ( errWrite ) {
43
- log ( `error test file ${ errWrite . message } ` ) ;
50
+ const createNewContentForContainer = ( nameAttAhc : string , existingContents : Buffer , ext : string , specname : string ) => {
51
+ const containerJSON = JSON . parse ( existingContents . toString ( ) ) ;
52
+
53
+ const after : ExecutableItem = {
54
+ name : 'video' ,
55
+ attachments : [
56
+ {
57
+ name : `${ specname } ${ ext } ` ,
58
+ type : 'video/mp4' ,
59
+ source : nameAttAhc ,
60
+ } ,
61
+ ] ,
62
+ parameters : [ ] ,
63
+ start : Date . now ( ) ,
64
+ stop : Date . now ( ) ,
65
+ status : Status . PASSED ,
66
+ statusDetails : { } ,
67
+ stage : Stage . FINISHED ,
68
+ steps : [ ] ,
69
+ } ;
70
+
71
+ if ( ! containerJSON . afters ) {
72
+ containerJSON . afters = [ ] ;
73
+ }
74
+
75
+ containerJSON . afters . push ( after ) ;
76
+
77
+ return containerJSON ;
78
+ } ;
44
79
45
- return ;
80
+ /**
81
+ * Will copy test results and all attachments to watch folder
82
+ * for results to appear in TestOps
83
+ * @param input
84
+ * @param allureResultsWatch
85
+ */
86
+ const copyFileToWatch = async (
87
+ input : { test : string ; attachments : { name : string ; type : string ; source : string } [ ] } ,
88
+ allureResultsWatch : string ,
89
+ ) => {
90
+ const { test : allureResultFile , attachments } = input ;
91
+ const allureResults = path . dirname ( allureResultFile ) ;
92
+
93
+ if ( allureResults === allureResultsWatch ) {
94
+ log ( `afterSpec allureResultsWatch the same as allureResults ${ allureResults } , will not copy` ) ;
95
+
96
+ return ;
97
+ }
98
+ mkdirSyncWithTry ( allureResultsWatch ) ;
99
+
100
+ log ( `allureResults: ${ allureResults } ` ) ;
101
+ log ( `allureResultsWatch: ${ allureResultsWatch } ` ) ;
102
+ log ( `attachments: ${ JSON . stringify ( attachments ) } ` ) ;
103
+
104
+ await copyAttachments ( attachments , allureResultsWatch , allureResultFile ) ;
105
+ await copyTest ( allureResultFile , allureResultsWatch ) ;
106
+ } ;
107
+
108
+ /**
109
+ * Get all attachments for test to move them to watch folder
110
+ * @param item test item
111
+ */
112
+ const getAllAttachments = ( item : ExecutableItem ) => {
113
+ const attachmentsResult : Attachment [ ] = [ ] ;
114
+
115
+ const inner = ( steps : ExecutableItem [ ] , accumulatedRes : Attachment [ ] ) : Attachment [ ] => {
116
+ if ( steps . length === 0 ) {
117
+ return accumulatedRes ;
46
118
}
47
- log ( `write test file done ${ testFile } ` ) ;
48
- callBack ( ) ;
49
- } ) ;
119
+
120
+ const [ first , ...rest ] = steps ;
121
+ const newRes = [ ...accumulatedRes , ...first . attachments ] ;
122
+
123
+ return inner ( rest , newRes ) ;
124
+ } ;
125
+
126
+ return inner ( item . steps , attachmentsResult ) ;
50
127
} ;
128
+
51
129
// all tests for session
52
130
const allTests : { specRelative : string | undefined ; fullTitle : string ; uuid : string ; mochaId : string } [ ] = [ ] ;
53
131
54
132
export class AllureReporter {
55
133
// todo config
56
134
private showDuplicateWarn : boolean ;
57
135
private allureResults : string ;
136
+ private allureResultsWatch : string ;
58
137
private allureAddVideoOnPass : boolean ;
59
138
private allureSkipSteps : RegExp [ ] ;
60
139
private videos : string ;
@@ -78,6 +157,7 @@ export class AllureReporter {
78
157
constructor ( opts : ReporterOptions ) {
79
158
this . showDuplicateWarn = opts . showDuplicateWarn ;
80
159
this . allureResults = opts . allureResults ;
160
+ this . allureResultsWatch = opts . techAllureResults ;
81
161
this . allureAddVideoOnPass = opts . allureAddVideoOnPass ;
82
162
this . videos = opts . videos ;
83
163
this . screenshots = opts . screenshots ;
@@ -408,14 +488,14 @@ export class AllureReporter {
408
488
* @param arg {path: string}
409
489
*/
410
490
async attachVideoToContainers ( arg : { path : string } ) {
411
- // this happens after test has already finished
491
+ // this happens after test and suites have already finished
412
492
const { path : videoPath } = arg ;
413
493
log ( `attachVideoToTests: ${ videoPath } ` ) ;
414
494
const ext = '.mp4' ;
415
495
const specname = basename ( videoPath , ext ) ;
416
496
log ( specname ) ;
417
497
418
- // when video uploads everything is uploaded already(TestOps) except containers
498
+ // when video uploads everything is uploaded already (TestOps) except containers
419
499
const res = parseAllure ( this . allureResults ) ;
420
500
421
501
const tests = res
@@ -431,15 +511,20 @@ export class AllureReporter {
431
511
432
512
let doneFiles = 0 ;
433
513
434
- readFile ( videoPath , ( errVideo , _contentVideo ) => {
514
+ readFile ( videoPath , errVideo => {
435
515
if ( errVideo ) {
436
516
console . error ( `Could not read video: ${ errVideo } ` ) ;
437
517
438
518
return ;
439
519
}
440
520
441
521
testsAttach . forEach ( test => {
442
- const containerFile = `${ this . allureResults } /${ test . parent ?. uuid } -container.json` ;
522
+ if ( ! test . parent ) {
523
+ console . error ( `not writing videos since test has no parent suite: ${ test . fullName } ` ) ;
524
+
525
+ return ;
526
+ }
527
+ const containerFile = `${ this . allureResults } /${ test . parent . uuid } -container.json` ;
443
528
log ( `ATTACHING to container: ${ containerFile } ` ) ;
444
529
445
530
readFile ( containerFile , ( err , contents ) => {
@@ -448,71 +533,34 @@ export class AllureReporter {
448
533
449
534
return ;
450
535
}
451
- const testCon = JSON . parse ( contents . toString ( ) ) ;
452
536
const uuid = randomUUID ( ) ;
453
-
454
537
const nameAttAhc = `${ uuid } -attachment${ ext } ` ;
455
538
const newPath = path . join ( this . allureResults , nameAttAhc ) ;
539
+ const newContentJson = createNewContentForContainer ( nameAttAhc , contents , ext , specname ) ;
540
+ const newContent = JSON . stringify ( newContentJson ) ;
456
541
457
- const after = {
458
- name : 'video' ,
459
- attachments : [
460
- {
461
- name : `${ specname } ${ ext } ` ,
462
- type : 'video/mp4' ,
463
- source : nameAttAhc ,
464
- } ,
465
- ] ,
466
- parameters : [ ] ,
467
- start : Date . now ( ) ,
468
- stop : Date . now ( ) ,
469
- status : 'passed' ,
470
- statusDetails : { } ,
471
- stage : 'finished' ,
472
- steps : [ ] ,
542
+ const writeContainer = ( ) => {
543
+ log ( `write result file ${ containerFile } ` ) ;
544
+ writeResultFile ( containerFile , newContent , ( ) => {
545
+ doneFiles = doneFiles + 1 ;
546
+ } ) ;
473
547
} ;
474
548
475
- if ( ! testCon . afters ) {
476
- testCon . afters = [ after ] ;
477
- } else {
478
- testCon . afters . push ( after ) ;
479
- }
480
-
481
549
if ( existsSync ( newPath ) ) {
482
- log ( `not writing! video file ${ newPath } ` ) ;
483
-
484
- writeTestFile ( containerFile , JSON . stringify ( testCon ) , ( ) => {
485
- doneFiles = doneFiles + 1 ;
486
- } ) ;
550
+ log ( `not writing video, file already exist in path ${ newPath } ` ) ;
551
+ writeContainer ( ) ;
487
552
488
553
return ;
489
554
}
490
555
491
- log ( `write video file ${ newPath } ` ) ;
492
- copyFile ( videoPath , newPath , errCopy => {
493
- if ( errCopy ) {
494
- log ( `error copy file ${ errCopy . message } ` ) ;
495
-
496
- return ;
497
- }
498
- log ( `write test file ${ containerFile } ` ) ;
499
- writeTestFile ( containerFile , JSON . stringify ( testCon ) , ( ) => {
500
- doneFiles = doneFiles + 1 ;
501
- } ) ;
556
+ copyFileCp ( videoPath , newPath , false , ( ) => {
557
+ writeContainer ( ) ;
502
558
} ) ;
503
559
} ) ;
504
560
} ) ;
505
561
} ) ;
506
- const started = Date . now ( ) ;
507
- const timeout = 10000 ;
508
562
509
- while ( doneFiles < testsAttach . length ) {
510
- if ( Date . now ( ) - started >= timeout ) {
511
- console . error ( `Could not write all video attachments in ${ timeout } ms` ) ;
512
- break ;
513
- }
514
- await delay ( 100 ) ;
515
- }
563
+ await waitWhileCondition ( ( ) => doneFiles < testsAttach . length ) ;
516
564
}
517
565
518
566
endGroup ( ) {
@@ -752,7 +800,7 @@ export class AllureReporter {
752
800
}
753
801
}
754
802
755
- endTest ( arg : AllureTaskArgs < 'testEnded' > ) : void {
803
+ async endTest ( arg : AllureTaskArgs < 'testEnded' > ) : Promise < void > {
756
804
const { result, details } = arg ;
757
805
const storedStatus = this . testStatusStored ;
758
806
const storedDetails = this . testDetailsStored ;
@@ -788,6 +836,9 @@ export class AllureReporter {
788
836
789
837
this . applyGroupLabels ( ) ;
790
838
const uid = this . currentTest . uuid ;
839
+
840
+ //const resAtt: Attachment[] = [...this.currentTest.wrappedItem.attachments];
841
+ const attachments = getAllAttachments ( this . currentTest . wrappedItem ) ;
791
842
this . currentTest . endTest ( ) ;
792
843
this . tests . pop ( ) ;
793
844
this . descriptionHtml = [ ] ;
@@ -807,6 +858,11 @@ export class AllureReporter {
807
858
}
808
859
} ;
809
860
waitResultWritten ( this . allureResults , `${ this . allureResults } /${ uid } -result.json` ) ;
861
+
862
+ // move to watch
863
+
864
+ log ( 'testEnded: will move result to watch folder' ) ;
865
+ await copyFileToWatch ( { test : `${ this . allureResults } /${ uid } -result.json` , attachments } , this . allureResultsWatch ) ;
810
866
}
811
867
812
868
startStep ( arg : AllureTaskArgs < 'stepStarted' > ) {
0 commit comments