// tslint:disable-next-line:no-reference
///
import { bootstrap, JobQueueService, Logger } from '@vendure/core';
import { populate } from '@vendure/core/cli/populate';
import { clearAllTables, populateCustomers } from '@vendure/testing';
import stringify from 'csv-stringify';
import fs from 'fs';
import path from 'path';
import { initialData } from '../../core/mock-data/data-sources/initial-data';
import {
getLoadTestConfig,
getMysqlConnectionOptions,
getPostgresConnectionOptions,
getProductCount,
getProductCsvFilePath,
} from './load-test-config';
// tslint:disable:no-console
/**
* A script used to populate a database with test data for load testing.
*/
if (require.main === module) {
// Running from command line
const count = getProductCount();
const databaseName = `vendure-load-testing-${count}`;
isDatabasePopulated(databaseName)
.then(isPopulated => {
console.log(`isPopulated:`, isPopulated);
if (!isPopulated) {
const config = getLoadTestConfig('bearer', databaseName);
const csvFile = getProductCsvFilePath();
return clearAllTables(config, true)
.then(() => {
if (!fs.existsSync(csvFile)) {
return generateProductsCsv(count);
}
})
.then(() =>
populate(
() =>
bootstrap(config).then(async app => {
await app.get(JobQueueService).start();
return app;
}),
path.join(__dirname, '../../create/assets/initial-data.json'),
csvFile,
),
)
.then(async app => {
console.log('populating customers...');
await populateCustomers(app, 10, message => Logger.error(message));
return app.close();
});
} else {
console.log('Database is already populated!');
}
})
.then(
() => process.exit(0),
err => {
console.log(err);
process.exit(1);
},
);
}
/**
* Tests to see whether the load test database is already populated.
*/
async function isDatabasePopulated(databaseName: string): Promise {
const isPostgres = process.env.DB === 'postgres';
if (isPostgres) {
console.log(`Checking whether data is populated (postgres)`);
const pg = require('pg');
const postgresConnectionOptions = getPostgresConnectionOptions(databaseName);
const client = new pg.Client({
host: postgresConnectionOptions.host,
user: postgresConnectionOptions.username,
database: postgresConnectionOptions.database,
password: postgresConnectionOptions.password,
port: postgresConnectionOptions.port,
});
await client.connect();
try {
const res = await client.query('SELECT COUNT(id) as prodCount FROM product');
return true;
} catch (e) {
if (e.message === `relation "product" does not exist`) {
return false;
}
throw e;
}
} else {
const mysql = require('mysql');
const mysqlConnectionOptions = getMysqlConnectionOptions(databaseName);
const connection = mysql.createConnection({
host: mysqlConnectionOptions.host,
user: mysqlConnectionOptions.username,
password: mysqlConnectionOptions.password,
database: mysqlConnectionOptions.database,
});
return new Promise((resolve, reject) => {
connection.connect((error: any) => {
if (error) {
reject(error);
return;
}
connection.query('SELECT COUNT(id) as prodCount FROM product', (err: any, results: any) => {
if (err) {
if (err.code === 'ER_NO_SUCH_TABLE') {
resolve(false);
return;
}
reject(err);
return;
}
resolve(true);
});
});
});
}
}
/**
* Generates a CSV file of test product data which can then be imported into Vendure.
*/
function generateProductsCsv(productCount: number = 100): Promise {
const result: any[] = [];
const stringifier = stringify({
delimiter: ',',
});
const data: string[] = [];
console.log(`Generating ${productCount} rows of test product data...`);
stringifier.on('readable', () => {
let row;
// tslint:disable-next-line:no-conditional-assignment
while ((row = stringifier.read())) {
data.push(row);
}
});
return new Promise((resolve, reject) => {
const csvFile = getProductCsvFilePath();
stringifier.on('error', (err: any) => {
reject(err.message);
});
stringifier.on('finish', async () => {
fs.writeFileSync(csvFile, data.join(''));
console.log(`Done! Saved to ${csvFile}`);
resolve();
});
generateMockData(productCount, row => stringifier.write(row));
stringifier.end();
});
}
function generateMockData(productCount: number, writeFn: (row: string[]) => void) {
const headers: string[] = [
'name',
'slug',
'description',
'assets',
'facets',
'optionGroups',
'optionValues',
'sku',
'price',
'taxCategory',
'variantAssets',
'variantFacets',
'stockOnHand',
'trackInventory',
];
writeFn(headers);
const categories = getCategoryNames();
for (let i = 1; i <= productCount; i++) {
const outputRow = {
name: `Product ${i}`,
slug: `product-${i}`,
description: generateProductDescription(),
assets: 'product-image.jpg',
facets: `category:${categories[i % categories.length]}`,
optionGroups: '',
optionValues: '',
sku: `PRODID${i}`,
price: (Math.random() * 1000).toFixed(2),
taxCategory: 'standard',
variantAssets: '',
variantFacets: '',
stockOnHand: '1000',
trackInventory: 'false',
};
writeFn(Object.values(outputRow) as string[]);
}
}
function getCategoryNames() {
const allNames = new Set();
for (const collection of initialData.collections) {
for (const filter of collection.filters || []) {
filter.args.facetValueNames.forEach(name => allNames.add(name));
}
}
return Array.from(new Set(allNames));
}
const parts = [
`Now equipped with seventh-generation Intel Core processors`,
`Laptop is snappier than ever`,
`From daily tasks like launching apps and opening files to more advanced computing`,
`You can power through your day thanks to faster SSDs and Turbo Boost processing up to 3.6GHz`,
`Discover a truly immersive viewing experience with this monitor curved more deeply than any other`,
`Wrapping around your field of vision the 1,800 R screencreates a wider field of view`,
`This pc is optimised for gaming, and is also VR ready`,
`The Intel Core-i7 CPU and High Performance GPU give the computer the raw power it needs to function at a high level`,
`Boost your PC storage with this internal hard drive, designed just for desktop and all-in-one PCs`,
`Let all your colleagues know that you are typing on this exclusive, colorful klicky-klacky keyboard`,
`Solid conductors eliminate strand-interaction distortion and reduce jitter`,
`As the surface is made of high-purity silver`,
`the performance is very close to that of a solid silver cable`,
`but priced much closer to solid copper cable`,
`With its nostalgic design and simple point-and-shoot functionality`,
`the Instant Camera is the perfect pick to get started with instant photography`,
`This lens is a Di type lens using an optical system with improved multi-coating designed to function with digital SLR cameras as well as film cameras`,
`Capture vivid, professional-style photographs with help from this lightweight tripod`,
`The adjustable-height tripod makes it easy to achieve reliable stability`,
`Just the right angle when going after that award-winning shot`,
`Featuring a full carbon chassis - complete with cyclocross-specific carbon fork`,
`It's got the low weight, exceptional efficiency and brilliant handling`,
`You'll need to stay at the front of the pack`,
`When you're working out you need a quality rope that doesn't tangle at every couple of jumps`,
`Training gloves designed for optimum training`,
`Our gloves promote proper punching technique because they are conformed to the natural shape of your fist`,
`Dense, innovative two-layer foam provides better shock absorbency`,
`Full padding on the front, back and wrist to promote proper punching technique`,
`With tons of space inside (for max. 4 persons), full head height throughout`,
`This tent offers you everything you need`,
`Based on the 1970s iconic shape, but made to a larger 69cm size`,
`These skateboards are great for beginners to learn the foot spacing required`,
`Perfect for all-day cruising`,
`This football features high-contrast graphics for high-visibility during play`,
`Its machine-stitched tpu casing offers consistent performance`,
`With its ultra-light, uber-responsive magic foam`,
`The Running Shoe is ready to push you to victories both large and small`,
`A spiky yet elegant house cactus`,
`Perfect for the home or office`,
`Origin and habitat: Probably native only to the Andes of Peru`,
`Gloriously elegant`,
`It can go along with any interior as it is a neutral color and the most popular Phalaenopsis overall`,
`2 to 3 foot stems host large white flowers that can last for over 2 months`,
`Excellent semi-evergreen bonsai`,
`Indoors or out but needs some winter protection`,
`All trees sent will leave the nursery in excellent condition and will be of equal quality or better than the photograph shown`,
`Placing it at home or office can bring you fortune and prosperity`,
`Guards your house and ward off ill fortune`,
`Hand trowel for garden cultivating hammer finish epoxy-coated head`,
`For improved resistance to rust, scratches, humidity and alkalines in the soil`,
`A charming vintage white wooden chair`,
`Featuring an extremely spherical pink balloon`,
`The balloon may be detached and used for other purposes`,
`This premium, tan-brown bonded leather seat is part of the 'chill' sofa range`,
`The lever activated recline feature makes it easy to adjust to any position`,
`This smart, bustle back design with rounded tight padded arms has been designed with your comfort in mind`,
`This well-padded chair has foam pocket sprung seat cushions and fibre-filled back cushions`,
`Modern tapered white polycotton pendant shade with a metallic silver chrome interior`,
`For maximum light reflection`,
`Reversible gimble so it can be used as a ceiling shade or as a lamp shade`,
];
function generateProductDescription(): string {
const take = Math.ceil(Math.random() * 4);
return shuffle(parts).slice(0, take).join('. ');
}
/**
* Returns new copy of array in random order.
* https://stackoverflow.com/a/6274381/772859
*/
function shuffle(arr: T[]): T[] {
const a = arr.slice();
for (let i = a.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[a[i], a[j]] = [a[j], a[i]];
}
return a;
}