14
14
* limitations under the License.
15
15
*/
16
16
17
- import yargs from 'yargs' ;
17
+ const yargs = require ( 'yargs' ) ;
18
18
import { performance } from 'perf_hooks' ;
19
19
// eslint-disable-next-line node/no-unsupported-features/node-builtins
20
20
import { parentPort } from 'worker_threads' ;
21
21
import * as path from 'path' ;
22
22
import {
23
23
BLOCK_SIZE_IN_BYTES ,
24
24
cleanupFile ,
25
- DEFAULT_LARGE_FILE_SIZE_BYTES ,
26
- DEFAULT_PROJECT_ID ,
27
- DEFAULT_SMALL_FILE_SIZE_BYTES ,
28
25
generateRandomFile ,
29
26
generateRandomFileName ,
27
+ getLowHighFileSize ,
30
28
getValidationType ,
31
29
NODE_DEFAULT_HIGHWATER_MARK_BYTES ,
30
+ performanceTestCommand ,
32
31
performanceTestSetup ,
32
+ PERFORMANCE_TEST_TYPES ,
33
33
TestResult ,
34
34
} from './performanceUtils' ;
35
35
import { Bucket } from '../src' ;
36
+ import { rmSync } from 'fs' ;
36
37
37
38
const TEST_NAME_STRING = 'nodejs-perf-metrics' ;
38
39
const DEFAULT_NUMBER_OF_WRITES = 1 ;
39
40
const DEFAULT_NUMBER_OF_READS = 3 ;
40
- const DEFAULT_BUCKET_NAME = 'nodejs-perf-metrics' ;
41
+ const DEFAULT_RANGE_READS = 3 ;
41
42
42
43
let bucket : Bucket ;
43
44
const checkType = getValidationType ( ) ;
44
45
45
46
const argv = yargs ( process . argv . slice ( 2 ) )
46
- . options ( {
47
- bucket : { type : 'string' , default : DEFAULT_BUCKET_NAME } ,
48
- small : { type : 'number' , default : DEFAULT_SMALL_FILE_SIZE_BYTES } ,
49
- large : { type : 'number' , default : DEFAULT_LARGE_FILE_SIZE_BYTES } ,
50
- projectid : { type : 'string' , default : DEFAULT_PROJECT_ID } ,
51
- } )
47
+ . command ( performanceTestCommand )
52
48
. parseSync ( ) ;
53
49
54
50
/**
55
51
* Main entry point. This function performs a test iteration and posts the message back
56
52
* to the parent thread.
57
53
*/
58
54
async function main ( ) {
59
- const results = await performWriteReadTest ( ) ;
55
+ let results : TestResult [ ] = [ ] ;
56
+
57
+ ( { bucket} = await performanceTestSetup ( argv . project ! , argv . bucket ! ) ) ;
58
+
59
+ switch ( argv . test_type ) {
60
+ case PERFORMANCE_TEST_TYPES . WRITE_ONE_READ_THREE :
61
+ results = await performWriteReadTest ( ) ;
62
+ break ;
63
+ case PERFORMANCE_TEST_TYPES . RANGE_READ :
64
+ results = await performRangedReadTest ( ) ;
65
+ break ;
66
+ default :
67
+ break ;
68
+ }
69
+
60
70
parentPort ?. postMessage ( results ) ;
61
71
}
62
72
73
+ /**
74
+ * Performs an iteration of a ranged read test. Only the last result will be reported.
75
+ *
76
+ * @returns {Promise<TestResult[]> } Promise that resolves to an array of test results for the iteration.
77
+ */
78
+ async function performRangedReadTest ( ) : Promise < TestResult [ ] > {
79
+ const results : TestResult [ ] = [ ] ;
80
+ const fileSizeRange = getLowHighFileSize ( argv . object_size ) ;
81
+ const fileName = generateRandomFileName ( TEST_NAME_STRING ) ;
82
+ const sizeInBytes = generateRandomFile (
83
+ fileName ,
84
+ fileSizeRange . low ,
85
+ fileSizeRange . high ,
86
+ __dirname
87
+ ) ;
88
+ const file = bucket . file ( `${ fileName } ` ) ;
89
+ const destinationFileName = generateRandomFileName ( TEST_NAME_STRING ) ;
90
+ const destination = path . join ( __dirname , destinationFileName ) ;
91
+
92
+ const iterationResult : TestResult = {
93
+ op : 'READ' ,
94
+ objectSize : sizeInBytes ,
95
+ appBufferSize : BLOCK_SIZE_IN_BYTES ,
96
+ libBufferSize : NODE_DEFAULT_HIGHWATER_MARK_BYTES ,
97
+ crc32Enabled : false ,
98
+ md5Enabled : false ,
99
+ apiName : 'JSON' ,
100
+ elapsedTimeUs : 0 ,
101
+ cpuTimeUs : - 1 ,
102
+ status : '[OK]' ,
103
+ chunkSize : argv . range_read_size ,
104
+ workers : argv . workers ,
105
+ } ;
106
+
107
+ await bucket . upload ( `${ __dirname } /${ fileName } ` ) ;
108
+ cleanupFile ( fileName ) ;
109
+
110
+ for ( let i = 0 ; i < DEFAULT_RANGE_READS ; i ++ ) {
111
+ const start = performance . now ( ) ;
112
+ await file . download ( {
113
+ start : 0 ,
114
+ end : argv . range_read_size ,
115
+ destination,
116
+ } ) ;
117
+ const end = performance . now ( ) ;
118
+ cleanupFile ( destinationFileName ) ;
119
+ iterationResult . elapsedTimeUs = Math . round ( ( end - start ) * 1000 ) ;
120
+ }
121
+
122
+ rmSync ( TEST_NAME_STRING , { recursive : true , force : true } ) ;
123
+ await file . delete ( ) ;
124
+ results . push ( iterationResult ) ;
125
+ return results ;
126
+ }
127
+
63
128
/**
64
129
* Performs an iteration of the Write 1 / Read 3 performance measuring test.
65
130
*
66
131
* @returns {Promise<TestResult[]> } Promise that resolves to an array of test results for the iteration.
67
132
*/
68
133
async function performWriteReadTest ( ) : Promise < TestResult [ ] > {
69
134
const results : TestResult [ ] = [ ] ;
135
+ const fileSizeRange = getLowHighFileSize ( argv . object_size ) ;
70
136
const fileName = generateRandomFileName ( TEST_NAME_STRING ) ;
71
- const sizeInBytes = generateRandomFile ( fileName , argv . small , argv . large ) ;
72
-
73
- ( { bucket} = await performanceTestSetup ( argv . projectid , argv . bucket ) ) ;
137
+ const file = bucket . file ( `${ fileName } ` ) ;
138
+ const sizeInBytes = generateRandomFile (
139
+ fileName ,
140
+ fileSizeRange . low ,
141
+ fileSizeRange . high ,
142
+ __dirname
143
+ ) ;
74
144
75
145
for ( let j = 0 ; j < DEFAULT_NUMBER_OF_WRITES ; j ++ ) {
76
146
let start = 0 ;
@@ -88,6 +158,7 @@ async function performWriteReadTest(): Promise<TestResult[]> {
88
158
cpuTimeUs : - 1 ,
89
159
status : '[OK]' ,
90
160
chunkSize : sizeInBytes ,
161
+ workers : argv . workers ,
91
162
} ;
92
163
93
164
start = performance . now ( ) ;
@@ -98,23 +169,24 @@ async function performWriteReadTest(): Promise<TestResult[]> {
98
169
results . push ( iterationResult ) ;
99
170
}
100
171
172
+ const iterationResult : TestResult = {
173
+ op : 'READ' ,
174
+ objectSize : sizeInBytes ,
175
+ appBufferSize : BLOCK_SIZE_IN_BYTES ,
176
+ libBufferSize : NODE_DEFAULT_HIGHWATER_MARK_BYTES ,
177
+ crc32Enabled : checkType === 'crc32c' ,
178
+ md5Enabled : checkType === 'md5' ,
179
+ apiName : 'JSON' ,
180
+ elapsedTimeUs : 0 ,
181
+ cpuTimeUs : - 1 ,
182
+ status : '[OK]' ,
183
+ chunkSize : sizeInBytes ,
184
+ workers : argv . workers ,
185
+ } ;
186
+
101
187
for ( let j = 0 ; j < DEFAULT_NUMBER_OF_READS ; j ++ ) {
102
188
let start = 0 ;
103
189
let end = 0 ;
104
- const file = bucket . file ( `${ fileName } ` ) ;
105
- const iterationResult : TestResult = {
106
- op : `READ[${ j } ]` ,
107
- objectSize : sizeInBytes ,
108
- appBufferSize : BLOCK_SIZE_IN_BYTES ,
109
- libBufferSize : NODE_DEFAULT_HIGHWATER_MARK_BYTES ,
110
- crc32Enabled : checkType === 'crc32c' ,
111
- md5Enabled : checkType === 'md5' ,
112
- apiName : 'JSON' ,
113
- elapsedTimeUs : 0 ,
114
- cpuTimeUs : - 1 ,
115
- status : '[OK]' ,
116
- chunkSize : sizeInBytes ,
117
- } ;
118
190
119
191
const destinationFileName = generateRandomFileName ( TEST_NAME_STRING ) ;
120
192
const destination = path . join ( __dirname , destinationFileName ) ;
@@ -125,11 +197,11 @@ async function performWriteReadTest(): Promise<TestResult[]> {
125
197
126
198
cleanupFile ( destinationFileName ) ;
127
199
iterationResult . elapsedTimeUs = Math . round ( ( end - start ) * 1000 ) ;
128
- results . push ( iterationResult ) ;
129
200
}
130
201
131
- cleanupFile ( fileName ) ;
132
-
202
+ rmSync ( TEST_NAME_STRING , { recursive : true , force : true } ) ;
203
+ await file . delete ( ) ;
204
+ results . push ( iterationResult ) ;
133
205
return results ;
134
206
}
135
207
0 commit comments