Skip to content

Commit 0501127

Browse files
authored
test: add ranged read test, performance test code cleanups (#2161)
* test: add ranged read test to performance tests * test: add ranged read test, performance test code cleanups * use require for yargs since we do not have esModuleInterop set * fix output type variable name * clean up naming * fix w1r3 directory * use as const instead of enum * only report last read for w1r3
1 parent 6b8a848 commit 0501127

5 files changed

+326
-186
lines changed

internal-tooling/performApplicationPerformanceTest.ts

+20-24
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
* limitations under the License.
1515
*/
1616

17-
import yargs from 'yargs';
17+
const yargs = require('yargs');
1818
import {promises as fsp, rmSync} from 'fs';
1919
import {
2020
Bucket,
@@ -28,33 +28,24 @@ import {performance} from 'perf_hooks';
2828
import {parentPort} from 'worker_threads';
2929
import {
3030
BLOCK_SIZE_IN_BYTES,
31-
DEFAULT_PROJECT_ID,
32-
DEFAULT_NUMBER_OF_OBJECTS,
33-
DEFAULT_SMALL_FILE_SIZE_BYTES,
34-
DEFAULT_LARGE_FILE_SIZE_BYTES,
3531
NODE_DEFAULT_HIGHWATER_MARK_BYTES,
3632
generateRandomDirectoryStructure,
3733
getValidationType,
3834
performanceTestSetup,
3935
TestResult,
36+
performanceTestCommand,
37+
getLowHighFileSize,
38+
PERFORMANCE_TEST_TYPES,
4039
} from './performanceUtils';
41-
import {TRANSFER_MANAGER_TEST_TYPES} from './performanceTest';
4240

4341
const TEST_NAME_STRING = 'nodejs-perf-metrics-application';
44-
const DEFAULT_BUCKET_NAME = 'nodejs-perf-metrics-shaffeeullah';
4542

4643
let bucket: Bucket;
4744

4845
const checkType = getValidationType();
4946

5047
const argv = yargs(process.argv.slice(2))
51-
.options({
52-
bucket: {type: 'string', default: DEFAULT_BUCKET_NAME},
53-
small: {type: 'number', default: DEFAULT_SMALL_FILE_SIZE_BYTES},
54-
large: {type: 'number', default: DEFAULT_LARGE_FILE_SIZE_BYTES},
55-
projectid: {type: 'string', default: DEFAULT_PROJECT_ID},
56-
numobjects: {type: 'number', default: DEFAULT_NUMBER_OF_OBJECTS},
57-
})
48+
.command(performanceTestCommand)
5849
.parseSync();
5950

6051
/**
@@ -74,15 +65,16 @@ async function main() {
7465
cpuTimeUs: 0,
7566
status: '[OK]',
7667
chunkSize: 0,
68+
workers: argv.workers,
7769
};
7870

79-
({bucket} = await performanceTestSetup(argv.projectid, argv.bucket));
71+
({bucket} = await performanceTestSetup(argv.project!, argv.bucket!));
8072

81-
switch (argv.testtype) {
82-
case TRANSFER_MANAGER_TEST_TYPES.APPLICATION_UPLOAD_MULTIPLE_OBJECTS:
73+
switch (argv.test_type) {
74+
case PERFORMANCE_TEST_TYPES.APPLICATION_UPLOAD_MULTIPLE_OBJECTS:
8375
result = await performWriteTest();
8476
break;
85-
case TRANSFER_MANAGER_TEST_TYPES.APPLICATION_DOWNLOAD_MULTIPLE_OBJECTS:
77+
case PERFORMANCE_TEST_TYPES.APPLICATION_DOWNLOAD_MULTIPLE_OBJECTS:
8678
result = await performReadTest();
8779
break;
8880
// case TRANSFER_MANAGER_TEST_TYPES.APPLICATION_LARGE_FILE_DOWNLOAD:
@@ -129,11 +121,12 @@ async function downloadInParallel(bucket: Bucket, options: DownloadOptions) {
129121
async function performWriteTest(): Promise<TestResult> {
130122
await bucket.deleteFiles(); //start clean
131123

124+
const fileSizeRange = getLowHighFileSize(argv.object_size);
132125
const creationInfo = generateRandomDirectoryStructure(
133-
argv.numobjects,
126+
argv.num_objects,
134127
TEST_NAME_STRING,
135-
argv.small,
136-
argv.large
128+
fileSizeRange.low,
129+
fileSizeRange.high
137130
);
138131

139132
const start = performance.now();
@@ -155,6 +148,7 @@ async function performWriteTest(): Promise<TestResult> {
155148
cpuTimeUs: -1,
156149
status: '[OK]',
157150
chunkSize: creationInfo.totalSizeInBytes,
151+
workers: argv.workers,
158152
};
159153
return result;
160154
}
@@ -165,12 +159,13 @@ async function performWriteTest(): Promise<TestResult> {
165159
* @returns {Promise<TestResult>} Promise that resolves to an array of test results for the iteration.
166160
*/
167161
async function performReadTest(): Promise<TestResult> {
162+
const fileSizeRange = getLowHighFileSize(argv.object_size);
168163
await bucket.deleteFiles(); // start clean
169164
const creationInfo = generateRandomDirectoryStructure(
170-
argv.numobjects,
165+
argv.num_objects,
171166
TEST_NAME_STRING,
172-
argv.small,
173-
argv.large
167+
fileSizeRange.low,
168+
fileSizeRange.high
174169
);
175170
await uploadInParallel(bucket, creationInfo.paths, {validation: checkType});
176171

@@ -190,6 +185,7 @@ async function performReadTest(): Promise<TestResult> {
190185
cpuTimeUs: -1,
191186
status: '[OK]',
192187
chunkSize: creationInfo.totalSizeInBytes,
188+
workers: argv.workers,
193189
};
194190

195191
rmSync(TEST_NAME_STRING, {recursive: true, force: true});

internal-tooling/performPerformanceTest.ts

+104-32
Original file line numberDiff line numberDiff line change
@@ -14,63 +14,133 @@
1414
* limitations under the License.
1515
*/
1616

17-
import yargs from 'yargs';
17+
const yargs = require('yargs');
1818
import {performance} from 'perf_hooks';
1919
// eslint-disable-next-line node/no-unsupported-features/node-builtins
2020
import {parentPort} from 'worker_threads';
2121
import * as path from 'path';
2222
import {
2323
BLOCK_SIZE_IN_BYTES,
2424
cleanupFile,
25-
DEFAULT_LARGE_FILE_SIZE_BYTES,
26-
DEFAULT_PROJECT_ID,
27-
DEFAULT_SMALL_FILE_SIZE_BYTES,
2825
generateRandomFile,
2926
generateRandomFileName,
27+
getLowHighFileSize,
3028
getValidationType,
3129
NODE_DEFAULT_HIGHWATER_MARK_BYTES,
30+
performanceTestCommand,
3231
performanceTestSetup,
32+
PERFORMANCE_TEST_TYPES,
3333
TestResult,
3434
} from './performanceUtils';
3535
import {Bucket} from '../src';
36+
import {rmSync} from 'fs';
3637

3738
const TEST_NAME_STRING = 'nodejs-perf-metrics';
3839
const DEFAULT_NUMBER_OF_WRITES = 1;
3940
const DEFAULT_NUMBER_OF_READS = 3;
40-
const DEFAULT_BUCKET_NAME = 'nodejs-perf-metrics';
41+
const DEFAULT_RANGE_READS = 3;
4142

4243
let bucket: Bucket;
4344
const checkType = getValidationType();
4445

4546
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)
5248
.parseSync();
5349

5450
/**
5551
* Main entry point. This function performs a test iteration and posts the message back
5652
* to the parent thread.
5753
*/
5854
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+
6070
parentPort?.postMessage(results);
6171
}
6272

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+
63128
/**
64129
* Performs an iteration of the Write 1 / Read 3 performance measuring test.
65130
*
66131
* @returns {Promise<TestResult[]>} Promise that resolves to an array of test results for the iteration.
67132
*/
68133
async function performWriteReadTest(): Promise<TestResult[]> {
69134
const results: TestResult[] = [];
135+
const fileSizeRange = getLowHighFileSize(argv.object_size);
70136
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+
);
74144

75145
for (let j = 0; j < DEFAULT_NUMBER_OF_WRITES; j++) {
76146
let start = 0;
@@ -88,6 +158,7 @@ async function performWriteReadTest(): Promise<TestResult[]> {
88158
cpuTimeUs: -1,
89159
status: '[OK]',
90160
chunkSize: sizeInBytes,
161+
workers: argv.workers,
91162
};
92163

93164
start = performance.now();
@@ -98,23 +169,24 @@ async function performWriteReadTest(): Promise<TestResult[]> {
98169
results.push(iterationResult);
99170
}
100171

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+
101187
for (let j = 0; j < DEFAULT_NUMBER_OF_READS; j++) {
102188
let start = 0;
103189
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-
};
118190

119191
const destinationFileName = generateRandomFileName(TEST_NAME_STRING);
120192
const destination = path.join(__dirname, destinationFileName);
@@ -125,11 +197,11 @@ async function performWriteReadTest(): Promise<TestResult[]> {
125197

126198
cleanupFile(destinationFileName);
127199
iterationResult.elapsedTimeUs = Math.round((end - start) * 1000);
128-
results.push(iterationResult);
129200
}
130201

131-
cleanupFile(fileName);
132-
202+
rmSync(TEST_NAME_STRING, {recursive: true, force: true});
203+
await file.delete();
204+
results.push(iterationResult);
133205
return results;
134206
}
135207

0 commit comments

Comments
 (0)