creevey
Version:
creevey is a tool for automated visual testing, that tightly integrated with storybook
247 lines (205 loc) • 22.5 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _path = _interopRequireDefault(require("path"));
var _fs = require("fs");
var _util = require("util");
var _events = require("events");
var _types = require("../../types");
var _pool = _interopRequireDefault(require("./pool"));
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
const copyFileAsync = (0, _util.promisify)(_fs.copyFile);
const mkdirAsync = (0, _util.promisify)(_fs.mkdir);
class Runner extends _events.EventEmitter {
get isRunning() {
return Object.values(this.pools).some(pool => pool.isRunning);
}
constructor(config, tests) {
super();
this.tests = tests;
_defineProperty(this, "screenDir", void 0);
_defineProperty(this, "reportDir", void 0);
_defineProperty(this, "browsers", void 0);
_defineProperty(this, "pools", {});
_defineProperty(this, "handlePoolMessage", message => {
const {
id,
status,
result
} = message;
const test = this.tests[id];
if (!test) return;
test.status = status;
if (!result) {
this.sendUpdate({
tests: {
[id]: {
path: test.path,
status
}
}
});
return;
}
if (!test.results) {
test.results = [];
}
test.results.push(result);
this.sendUpdate({
tests: {
[id]: {
path: test.path,
status,
results: [result]
}
}
});
});
_defineProperty(this, "handlePoolStop", () => {
if (!this.isRunning) {
this.sendUpdate({
isRunning: false
});
this.emit('stop');
}
});
this.screenDir = config.screenDir;
this.reportDir = config.reportDir;
this.browsers = Object.keys(config.browsers);
this.browsers.map(browser => this.pools[browser] = new _pool.default(config, browser)).map(pool => pool.on('test', this.handlePoolMessage));
}
async init() {
await Promise.all(Object.values(this.pools).map(pool => pool.init()));
}
updateTests(testsDiff) {
const tests = {};
const removedTests = [];
Object.entries(testsDiff).forEach(([id, newTest]) => {
const oldTest = this.tests[id];
if (newTest) {
if (oldTest) {
this.tests[id] = { ...newTest,
status: 'unknown',
retries: oldTest.retries,
results: oldTest.results,
approved: oldTest.approved
};
} else this.tests[id] = newTest; // eslint-disable-next-line @typescript-eslint/no-unused-vars
const {
story,
fn,
...restTest
} = newTest;
tests[id] = { ...restTest,
status: 'unknown'
};
} else {
if (oldTest) removedTests.push(oldTest.path);
delete this.tests[id];
}
});
this.sendUpdate({
tests,
removedTests
});
}
start(ids) {
if (this.isRunning) return;
const testsToStart = ids.map(id => this.tests[id]).filter(_types.isDefined).filter(test => !test.skip);
if (testsToStart.length == 0) return;
this.sendUpdate({
isRunning: true,
tests: testsToStart.reduce((update, {
id
}) => {
var _this$tests$id;
return { ...update,
[id]: {
path: (_this$tests$id = this.tests[id]) === null || _this$tests$id === void 0 ? void 0 : _this$tests$id.path,
status: 'pending'
}
};
}, {})
});
const testsByBrowser = testsToStart.reduce((tests, test) => {
const {
id,
path
} = test;
const [browser, ...restPath] = path;
test.status = 'pending';
return { ...tests,
[browser]: [...(tests[browser] || []), {
id,
path: restPath
}]
};
}, {});
this.browsers.forEach(browser => {
const pool = this.pools[browser];
const tests = testsByBrowser[browser];
if (tests && tests.length > 0 && pool.start(tests)) {
pool.once('stop', this.handlePoolStop);
}
});
}
stop() {
if (!this.isRunning) return;
this.browsers.forEach(browser => this.pools[browser].stop());
}
get status() {
const tests = {};
Object.values(this.tests).filter(_types.isDefined) // eslint-disable-next-line @typescript-eslint/no-unused-vars
.forEach(({
story,
fn,
...test
}) => tests[test.id] = test);
return {
isRunning: this.isRunning,
tests
};
}
async approve({
id,
retry,
image
}) {
const test = this.tests[id];
if (!test || !test.results) return;
const result = test.results[retry];
if (!result || !result.images) return;
const images = result.images[image];
if (!images) return;
if (!test.approved) {
test.approved = {};
}
const [browser, ...restPath] = test.path;
const testPath = _path.default.join(...restPath.reverse(), image == browser ? '' : browser);
const srcImagePath = _path.default.join(this.reportDir, testPath, images.actual);
const dstImagePath = _path.default.join(this.screenDir, testPath, `${image}.png`);
await mkdirAsync(_path.default.join(this.screenDir, testPath), {
recursive: true
});
await copyFileAsync(srcImagePath, dstImagePath);
test.approved[image] = retry;
this.sendUpdate({
tests: {
[id]: {
path: test.path,
approved: {
[image]: retry
}
}
}
});
}
sendUpdate(data) {
this.emit('update', data);
}
}
exports.default = Runner;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/server/master/runner.ts"],"names":["copyFileAsync","copyFile","mkdirAsync","mkdir","Runner","EventEmitter","isRunning","Object","values","pools","some","pool","constructor","config","tests","message","id","status","result","test","sendUpdate","path","results","push","emit","screenDir","reportDir","browsers","keys","map","browser","Pool","on","handlePoolMessage","init","Promise","all","updateTests","testsDiff","removedTests","entries","forEach","newTest","oldTest","retries","approved","story","fn","restTest","start","ids","testsToStart","filter","isDefined","skip","length","reduce","update","testsByBrowser","restPath","once","handlePoolStop","stop","approve","retry","image","images","testPath","join","reverse","srcImagePath","actual","dstImagePath","recursive","data"],"mappings":"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","sourcesContent":["import path from 'path';\nimport { copyFile, mkdir } from 'fs';\nimport { promisify } from 'util';\nimport { EventEmitter } from 'events';\nimport {\n  Config,\n  CreeveyStatus,\n  TestResult,\n  ApprovePayload,\n  isDefined,\n  CreeveyUpdate,\n  TestStatus,\n  ServerTest,\n} from '../../types';\nimport Pool from './pool';\n\nconst copyFileAsync = promisify(copyFile);\nconst mkdirAsync = promisify(mkdir);\n\nexport default class Runner extends EventEmitter {\n  private screenDir: string;\n  private reportDir: string;\n  private browsers: string[];\n  private pools: { [browser: string]: Pool } = {};\n  public get isRunning(): boolean {\n    return Object.values(this.pools).some((pool) => pool.isRunning);\n  }\n  constructor(config: Config, private tests: Partial<{ [id: string]: ServerTest }>) {\n    super();\n\n    this.screenDir = config.screenDir;\n    this.reportDir = config.reportDir;\n    this.browsers = Object.keys(config.browsers);\n    this.browsers\n      .map((browser) => (this.pools[browser] = new Pool(config, browser)))\n      .map((pool) => pool.on('test', this.handlePoolMessage));\n  }\n\n  private handlePoolMessage = (message: { id: string; status: TestStatus; result?: TestResult }): void => {\n    const { id, status, result } = message;\n    const test = this.tests[id];\n    if (!test) return;\n    test.status = status;\n    if (!result) {\n      this.sendUpdate({ tests: { [id]: { path: test.path, status } } });\n      return;\n    }\n    if (!test.results) {\n      test.results = [];\n    }\n    test.results.push(result);\n    this.sendUpdate({ tests: { [id]: { path: test.path, status, results: [result] } } });\n  };\n\n  private handlePoolStop = (): void => {\n    if (!this.isRunning) {\n      this.sendUpdate({ isRunning: false });\n      this.emit('stop');\n    }\n  };\n\n  public async init(): Promise<void> {\n    await Promise.all(Object.values(this.pools).map((pool) => pool.init()));\n  }\n\n  public updateTests(testsDiff: Partial<{ [id: string]: ServerTest }>): void {\n    const tests: CreeveyStatus['tests'] = {};\n    const removedTests: string[][] = [];\n    Object.entries(testsDiff).forEach(([id, newTest]) => {\n      const oldTest = this.tests[id];\n      if (newTest) {\n        if (oldTest) {\n          this.tests[id] = {\n            ...newTest,\n            status: 'unknown',\n            retries: oldTest.retries,\n            results: oldTest.results,\n            approved: oldTest.approved,\n          };\n        } else this.tests[id] = newTest;\n\n        // eslint-disable-next-line @typescript-eslint/no-unused-vars\n        const { story, fn, ...restTest } = newTest;\n        tests[id] = { ...restTest, status: 'unknown' };\n      } else {\n        if (oldTest) removedTests.push(oldTest.path);\n        delete this.tests[id];\n      }\n    });\n    this.sendUpdate({ tests, removedTests });\n  }\n\n  public start(ids: string[]): void {\n    interface TestsByBrowser {\n      [browser: string]: { id: string; path: string[] }[];\n    }\n    if (this.isRunning) return;\n\n    const testsToStart = ids\n      .map((id) => this.tests[id])\n      .filter(isDefined)\n      .filter((test) => !test.skip);\n\n    if (testsToStart.length == 0) return;\n\n    this.sendUpdate({\n      isRunning: true,\n      tests: testsToStart.reduce(\n        (update, { id }) => ({ ...update, [id]: { path: this.tests[id]?.path, status: 'pending' } }),\n        {},\n      ),\n    });\n\n    const testsByBrowser: Partial<TestsByBrowser> = testsToStart.reduce((tests: TestsByBrowser, test) => {\n      const { id, path } = test;\n      const [browser, ...restPath] = path;\n      test.status = 'pending';\n      return {\n        ...tests,\n        [browser]: [...(tests[browser] || []), { id, path: restPath }],\n      };\n    }, {});\n\n    this.browsers.forEach((browser) => {\n      const pool = this.pools[browser];\n      const tests = testsByBrowser[browser];\n\n      if (tests && tests.length > 0 && pool.start(tests)) {\n        pool.once('stop', this.handlePoolStop);\n      }\n    });\n  }\n\n  public stop(): void {\n    if (!this.isRunning) return;\n    this.browsers.forEach((browser) => this.pools[browser].stop());\n  }\n\n  public get status(): CreeveyStatus {\n    const tests: CreeveyStatus['tests'] = {};\n    Object.values(this.tests)\n      .filter(isDefined)\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      .forEach(({ story, fn, ...test }) => (tests[test.id] = test));\n    return {\n      isRunning: this.isRunning,\n      tests,\n    };\n  }\n\n  public async approve({ id, retry, image }: ApprovePayload): Promise<void> {\n    const test = this.tests[id];\n    if (!test || !test.results) return;\n    const result = test.results[retry];\n    if (!result || !result.images) return;\n    const images = result.images[image];\n    if (!images) return;\n    if (!test.approved) {\n      test.approved = {};\n    }\n    const [browser, ...restPath] = test.path;\n    const testPath = path.join(...restPath.reverse(), image == browser ? '' : browser);\n    const srcImagePath = path.join(this.reportDir, testPath, images.actual);\n    const dstImagePath = path.join(this.screenDir, testPath, `${image}.png`);\n    await mkdirAsync(path.join(this.screenDir, testPath), { recursive: true });\n    await copyFileAsync(srcImagePath, dstImagePath);\n    test.approved[image] = retry;\n    this.sendUpdate({ tests: { [id]: { path: test.path, approved: { [image]: retry } } } });\n  }\n\n  private sendUpdate(data: CreeveyUpdate): void {\n    this.emit('update', data);\n  }\n}\n"]}