embark/packages/plugins/scripts-runner/test/script-runner.spec.js
Michael Bradley, Jr 3693ebd90d fix: ensure that packages properly specify their dependencies
Many packages in the monorepo did not specify all of their dependencies; they
were effectively relying on resolution in the monorepo's root
`node_modules`. In a production release of `embark` and `embark[js]-*` packages
this can lead to broken packages.

To fix the problem currently and to help prevent it from happening again, make
use of the `eslint-plugin-import` package's `import/no-extraneous-dependencies`
and `import/no-unresolved` rules. In the root `tslint.json` set
`"no-implicit-dependencies": true`, wich is the tslint equivalent of
`import/no-extraneous-dependencies`; there is no tslint equivalent for
`import/no-unresolved`, but we will eventually replace tslint with an eslint
configuration that checks both `.js` and `.ts` files.

For `import/no-unresolved` to work in our monorepo setup, in most packages add
an `index.js` that has:

```js
module.exports = require('./dist'); // or './dist/lib' in some cases
```

And point `"main"` in `package.json` to `"./index.js"`. Despite what's
indicated in npm's documentation for `package.json`, it's also necessary to add
`"index.js"` to the `"files"` array.

Make sure that all `.js` files that can and should be linted are in fact
linted. For example, files in `packages/embark/src/cmd/` weren't being linted
and many test suites weren't being linted.

Bump all relevant packages to `eslint@6.8.0`.

Fix all linter errors that arose after these changes.

Implement a `check-yarn-lock` script that's run as part of `"ci:full"` and
`"qa:full"`, and can manually be invoked via `yarn cylock` in the root of the
monorepo. The script exits with error if any specifiers are found in
`yarn.lock` for `embark[js][-*]` and/or `@embarklabs/*` (with a few exceptions,
cf. `scripts/check-yarn-lock.js`).
2020-02-25 14:52:10 -06:00

174 lines
5.2 KiB
JavaScript

import sinon from 'sinon';
import assert from 'assert';
import path from 'path';
import { fakeEmbark } from 'embark-testing';
import ScriptsRunnerPlugin, { ScriptsRunnerCommand } from '../src/';
import { file as tmpFile, dir as tmpDir } from 'tmp-promise';
import { promises } from 'fs';
// Due to our `DAPP_PATH` dependency in `embark-utils` `dappPath()`, we need to
// ensure that this environment variable is defined.
process.env.DAPP_PATH = 'something';
async function prepareScriptFile(content, dir) {
const file = await tmpFile({ postfix: '.js', dir});
await promises.writeFile(file.path, content);
return file;
}
const web3Mock = {
eth: {
getBlock: sinon.spy((_number, _flag) => 'testhash')
}
};
describe('plugins/scripts-runner', () => {
// eslint-disable-next-line no-unused-vars
let scriptRunner,
testTracker,
runCodeCommandHandler,
blockchainClientProviderCommandHandler,
contractsListCommandHandler,
embark;
beforeEach(async () => {
const testBed = fakeEmbark({
contractsConfig: {
},
embarkConfig: {
migrations: 'migrations'
}
});
testTracker = {
web3: web3Mock,
ensureTrackingFile: sinon.spy((_hash, _env) => {}),
track: sinon.spy(_config => {}),
isTracked: sinon.spy(() => false),
setWeb3: sinon.spy(_web3 => {})
};
embark = testBed.embark;
scriptRunner = new ScriptsRunnerPlugin(embark, { tracker: testTracker });
runCodeCommandHandler = sinon.spy((code, cb) => {
// `ScriptsRunnerPlugin` requests code evaluation two times.
// It expects a boolean for the first one and an object for
// the second one.
if (code.indexOf('!==') > 0) {
cb(null, true);
}
cb(null, {});
});
blockchainClientProviderCommandHandler = sinon.spy((name, cb) => {
cb(null, 'http://localhost:8545');
});
contractsListCommandHandler = sinon.spy(cb => {
cb(null, [
{ className: 'SimpleStorage' },
{ className: 'AnotherOne' },
{ className: 'Foo' }
]);
});
embark.events.setCommandHandler('contracts:list', contractsListCommandHandler);
embark.events.setCommandHandler('runcode:eval', runCodeCommandHandler);
embark.events.setCommandHandler('blockchain:client:provider', blockchainClientProviderCommandHandler);
await embark.events.request2(ScriptsRunnerCommand.Initialize);
});
afterEach(() => {
embark.teardown();
sinon.restore();
});
it('should execute script', async (done) => {
const scriptFile = await prepareScriptFile(`module.exports = () => { return 'done'; }`);
embark.events.request(ScriptsRunnerCommand.Execute, scriptFile.path, false, (err, result) => {
assert.equal(result, 'done');
scriptFile.cleanup();
done();
});
});
it('should execute all scripts in a directory', async (done) => {
const scriptsDir = await tmpDir();
// eslint-disable-next-line no-unused-vars
const scriptFile1 = await prepareScriptFile(
`module.exports = () => { return 'done' }`,
scriptsDir.path
);
// eslint-disable-next-line no-unused-vars
const scriptFile2 = await prepareScriptFile(
`module.exports = () => { return 'done2' }`,
scriptsDir.path
);
embark.events.request(ScriptsRunnerCommand.Execute, scriptsDir.path, false, (err, result) => {
assert.ok(result.includes('done'));
assert.ok(result.includes('done2'));
scriptsDir.cleanup();
done();
});
});
it('should force track scripts if --track option is applied', async (done) => {
const scriptFile = await prepareScriptFile(`module.exports = () => { return 'done'; }`);
const expectedResult = {
scriptName: path.basename(scriptFile.path),
scriptDirectory: path.basename(path.dirname(scriptFile.path)),
forceTracking: true
};
embark.events.request(ScriptsRunnerCommand.Execute, scriptFile.path, true, (err, result) => {
assert.equal(result, 'done');
assert(testTracker.track.calledOnce);
assert(testTracker.track.calledWith(expectedResult));
scriptFile.cleanup();
done();
});
});
it('should track automatically if script directory equals migrations directory', async (done) => {
const scriptsDir = await tmpDir();
const migrationsDir = path.join(scriptsDir.path, embark.config.embarkConfig.migrations);
await promises.mkdir(migrationsDir);
const scriptFile = await prepareScriptFile(
`module.exports = () => { return 'done' }`,
migrationsDir
);
const expectedResult = {
scriptName: path.basename(scriptFile.path),
scriptDirectory: path.basename(migrationsDir),
forceTracking: false
};
embark.events.request(ScriptsRunnerCommand.Execute, scriptFile.path, false, (err, result) => {
assert.equal(result, 'done');
assert(testTracker.track.calledOnce);
assert(testTracker.track.calledWith(expectedResult));
done();
});
});
it('should not execute script if it was tracked', async(done) => {
const scriptFile = await prepareScriptFile(`module.exports = () => { return 'done'; }`);
embark.events.request(ScriptsRunnerCommand.Execute, scriptFile.path, true, (err, result) => {
assert.equal(result, undefined);
done();
});
});
});