parent
4be69a0fbd
commit
ea2afe9816
@ -0,0 +1,10 @@ |
||||
module.exports = { |
||||
name: 'remix-tests', |
||||
preset: '../../jest.config.js', |
||||
transform: { |
||||
'^.+\\.[tj]sx?$': 'ts-jest', |
||||
}, |
||||
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'html', 'json', 'sol'], |
||||
coverageDirectory: '../../coverage/libs/remix-tests', |
||||
}; |
||||
|
@ -0,0 +1,282 @@ |
||||
import * as async from 'async' |
||||
import Web3 from 'web3'; |
||||
import * as assert from 'assert' |
||||
import { Provider } from '@remix-project/remix-simulator' |
||||
|
||||
import { compileFileOrFiles } from '../src/compiler' |
||||
import { deployAll } from '../src/deployer' |
||||
import { runTest, compilationInterface } from '../src/index' |
||||
import { ResultsInterface, TestCbInterface, ResultCbInterface } from '../src/index' |
||||
|
||||
// deepEqualExcluding allows us to exclude specific keys whose values vary.
|
||||
// In this specific test, we'll use this helper to exclude `time` keys.
|
||||
// Assertions for the existance of these will be made at the correct places.
|
||||
function deepEqualExcluding(a: any, b: any, excludedKeys: string[]) { |
||||
function removeKeysFromObject(obj: any, excludedKeys: string[]) { |
||||
if (obj !== Object(obj)) { |
||||
return obj |
||||
} |
||||
|
||||
if(Object.prototype.toString.call(obj) !== '[object Array]') { |
||||
obj = Object.assign({}, obj) |
||||
for (const key of excludedKeys) { |
||||
delete obj[key] |
||||
} |
||||
|
||||
return obj |
||||
} |
||||
|
||||
let newObj = [] |
||||
for (const idx in obj) { |
||||
newObj[idx] = removeKeysFromObject(obj[idx], excludedKeys); |
||||
} |
||||
|
||||
return newObj |
||||
} |
||||
|
||||
let aStripped: any = removeKeysFromObject(a, excludedKeys); |
||||
let bStripped: any = removeKeysFromObject(b, excludedKeys); |
||||
assert.deepEqual(aStripped, bStripped) |
||||
} |
||||
|
||||
let accounts: string[] |
||||
let provider = new Provider() |
||||
|
||||
async function compileAndDeploy(filename: string, callback: Function) { |
||||
let web3: Web3 = new Web3() |
||||
let sourceASTs: any = {} |
||||
await provider.init() |
||||
web3.setProvider(provider) |
||||
let compilationData: object |
||||
async.waterfall([ |
||||
function getAccountList(next: Function): void { |
||||
web3.eth.getAccounts((_err: Error | null | undefined, _accounts: string[]) => { |
||||
accounts = _accounts |
||||
web3.eth.defaultAccount = accounts[0] |
||||
next(_err) |
||||
}) |
||||
}, |
||||
function compile(next: Function): void { |
||||
compileFileOrFiles(filename, false, { accounts }, next) |
||||
}, |
||||
function deployAllContracts(compilationResult: compilationInterface, asts, next: Function): void { |
||||
for(const filename in asts) { |
||||
if(filename.endsWith('_test.sol')) |
||||
sourceASTs[filename] = asts[filename].ast |
||||
} |
||||
try { |
||||
compilationData = compilationResult |
||||
deployAll(compilationResult, web3, false, next) |
||||
} catch (e) { |
||||
throw e |
||||
} |
||||
} |
||||
], function (_err: Error | null | undefined, contracts: any): void { |
||||
callback(null, compilationData, contracts, sourceASTs, accounts) |
||||
}) |
||||
} |
||||
|
||||
|
||||
describe('testRunner', () => { |
||||
let tests: any[] = [], results: ResultsInterface; |
||||
|
||||
const testCallback: TestCbInterface = (err, test) => { |
||||
if (err) { throw err } |
||||
|
||||
if (test.type === 'testPass' || test.type === 'testFailure') { |
||||
assert.ok(test.time, 'test time not reported') |
||||
assert.ok(!Number.isInteger(test.time || 0), 'test time should not be an integer') |
||||
} |
||||
|
||||
tests.push(test) |
||||
} |
||||
|
||||
const resultsCallback: Function = (done) => { |
||||
return (err, _results) => { |
||||
if (err) { throw err } |
||||
results = _results |
||||
done() |
||||
} |
||||
} |
||||
|
||||
describe('#runTest', () => { |
||||
describe('test with beforeAll', () => { |
||||
const filename: string = '../tests/examples_1/simple_storage_test.sol' |
||||
|
||||
beforeAll((done) => { |
||||
compileAndDeploy(filename, (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) => { |
||||
runTest('MyTest', contracts.MyTest, compilationData[filename]['MyTest'], asts[filename], { accounts }, testCallback, resultsCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
afterAll(() => { tests = [] }) |
||||
|
||||
it('should have 3 passing test', () => { |
||||
assert.equal(results.passingNum, 3) |
||||
}) |
||||
|
||||
it('should have 1 failing test', () => { |
||||
assert.equal(results.failureNum, 1) |
||||
}) |
||||
|
||||
it('should return 6 messages', () => { |
||||
deepEqualExcluding(tests, [ |
||||
{ type: 'accountList', value: accounts }, |
||||
{ type: 'contract', value: 'MyTest', filename: 'tests/examples_1/simple_storage_test.sol' }, |
||||
{ type: 'testPass', value: 'Initial value should be100', context: 'MyTest' }, |
||||
{ type: 'testPass', value: 'Initial value should not be200', context: 'MyTest' }, |
||||
{ type: 'testFailure', value: 'Should trigger one fail', errMsg: 'uint test 1 fails', context: 'MyTest' }, |
||||
{ type: 'testPass', value: 'Should trigger one pass', context: 'MyTest' } |
||||
], ['time']) |
||||
}) |
||||
}) |
||||
|
||||
describe('test with beforeEach', () => { |
||||
const filename: string = 'tests/examples_2/simple_storage_test.sol' |
||||
|
||||
beforeAll(done => { |
||||
compileAndDeploy(filename, function (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) { |
||||
runTest('MyTest', contracts.MyTest, compilationData[filename]['MyTest'], asts[filename], { accounts }, testCallback, resultsCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
afterAll(() => { tests = [] }) |
||||
|
||||
it('should have 2 passing tests', () => { |
||||
assert.equal(results.passingNum, 2) |
||||
}) |
||||
|
||||
it('should 0 failing tests', () => { |
||||
assert.equal(results.failureNum, 0) |
||||
}) |
||||
|
||||
it('should return 4 messages', () => { |
||||
deepEqualExcluding(tests, [ |
||||
{ type: 'accountList', value: accounts }, |
||||
{ type: 'contract', value: 'MyTest', filename: 'tests/examples_2/simple_storage_test.sol' }, |
||||
{ type: 'testPass', value: 'Initial value should be100', context: 'MyTest' }, |
||||
{ type: 'testPass', value: 'Value is set200', context: 'MyTest' } |
||||
], ['time']) |
||||
}) |
||||
}) |
||||
|
||||
// Test string equality
|
||||
describe('test string equality', () => { |
||||
const filename: string = 'tests/examples_3/simple_string_test.sol' |
||||
|
||||
beforeAll(done => { |
||||
compileAndDeploy(filename, function (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) { |
||||
runTest('StringTest', contracts.StringTest, compilationData[filename]['StringTest'], asts[filename], { accounts }, testCallback, resultsCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
afterAll(() => { tests = [] }) |
||||
|
||||
it('should 2 passing tests', () => { |
||||
assert.equal(results.passingNum, 2) |
||||
}) |
||||
|
||||
it('should return 4 messages', () => { |
||||
deepEqualExcluding(tests, [ |
||||
{ type: 'accountList', value: accounts }, |
||||
{ type: 'contract', value: 'StringTest', filename: 'tests/examples_3/simple_string_test.sol' }, |
||||
{ type: 'testPass', value: 'Initial value should be hello world', context: 'StringTest' }, |
||||
{ type: 'testPass', value: 'Value should not be hello wordl', context: 'StringTest' } |
||||
], ['time']) |
||||
}) |
||||
}) |
||||
|
||||
// Test multiple directory import in test contract
|
||||
describe('test multiple directory import in test contract', () => { |
||||
const filename: string = 'tests/examples_5/test/simple_storage_test.sol' |
||||
|
||||
beforeAll(done => { |
||||
compileAndDeploy(filename, function (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) { |
||||
runTest('StorageResolveTest', contracts.StorageResolveTest, compilationData[filename]['StorageResolveTest'], asts[filename], { accounts }, testCallback, resultsCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
afterAll(() => { tests = [] }) |
||||
|
||||
it('should 3 passing tests', () => { |
||||
assert.equal(results.passingNum, 3) |
||||
}) |
||||
|
||||
it('should return 4 messages', () => { |
||||
deepEqualExcluding(tests, [ |
||||
{ type: 'accountList', value: accounts }, |
||||
{ type: 'contract', value: 'StorageResolveTest', filename: 'tests/examples_5/test/simple_storage_test.sol' }, |
||||
{ type: 'testPass', value: 'Initial value should be100', context: 'StorageResolveTest' }, |
||||
{ type: 'testPass', value: 'Check if even', context: 'StorageResolveTest' }, |
||||
{ type: 'testPass', value: 'Check if odd', context: 'StorageResolveTest' } |
||||
], ['time']) |
||||
}) |
||||
}) |
||||
|
||||
//Test signed/unsigned integer weight
|
||||
describe('test number weight', () => { |
||||
const filename: string = 'tests/number/number_test.sol' |
||||
|
||||
beforeAll(done => { |
||||
compileAndDeploy(filename, function (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) { |
||||
runTest('IntegerTest', contracts.IntegerTest, compilationData[filename]['IntegerTest'], asts[filename], { accounts }, testCallback, resultsCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
afterAll(() => { tests = [] }) |
||||
|
||||
it('should have 6 passing tests', () => { |
||||
assert.equal(results.passingNum, 6) |
||||
}) |
||||
it('should have 2 failing tests', () => { |
||||
assert.equal(results.failureNum, 2) |
||||
}) |
||||
}) |
||||
|
||||
// Test Transaction with custom sender & value
|
||||
describe('various sender', () => { |
||||
const filename: string = 'tests/various_sender/sender_and_value_test.sol' |
||||
|
||||
beforeAll(done => { |
||||
compileAndDeploy(filename, function (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) { |
||||
runTest('SenderAndValueTest', contracts.SenderAndValueTest, compilationData[filename]['SenderAndValueTest'], asts[filename], { accounts }, testCallback, resultsCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
afterAll(() => { tests = [] }) |
||||
|
||||
it('should have 17 passing tests', () => { |
||||
assert.equal(results.passingNum, 17) |
||||
}) |
||||
it('should have 0 failing tests', () => { |
||||
assert.equal(results.failureNum, 0) |
||||
}) |
||||
}) |
||||
|
||||
// Test `runTest` method without sending contract object (should throw error)
|
||||
describe('runTest method without contract json interface', () => { |
||||
const filename: string = 'tests/various_sender/sender_and_value_test.sol' |
||||
const errorCallback: Function = (done) => { |
||||
return (err, _results) => { |
||||
if (err && err.message.includes('Contract interface not available')) {
|
||||
results = _results |
||||
done()
|
||||
} |
||||
else throw err |
||||
} |
||||
} |
||||
beforeAll(done => { |
||||
compileAndDeploy(filename, function (_err: Error | null | undefined, compilationData: object, contracts: any, asts: any, accounts: string[]) { |
||||
runTest('SenderAndValueTest', undefined, compilationData[filename]['SenderAndValueTest'], asts[filename], { accounts }, testCallback, errorCallback(done)) |
||||
}) |
||||
}) |
||||
|
||||
it('should have 0 passing tests', () => { |
||||
assert.equal(results.passingNum, 0) |
||||
}) |
||||
it('should have 0 failing tests', () => { |
||||
assert.equal(results.failureNum, 0) |
||||
}) |
||||
}) |
||||
}) |
||||
}) |
@ -0,0 +1,17 @@ |
||||
{ |
||||
"extends": "./tsconfig.json", |
||||
"compilerOptions": { |
||||
"outDir": "../../dist/out-tsc", |
||||
"module": "commonjs", |
||||
"types": ["jest", "node"] |
||||
}, |
||||
"include": [ |
||||
"**/*.sol", |
||||
"**/*.spec.ts", |
||||
"**/*.spec.tsx", |
||||
"**/*.spec.js", |
||||
"**/*.spec.jsx", |
||||
"**/*.d.ts" |
||||
] |
||||
} |
||||
|
File diff suppressed because it is too large
Load Diff
Loading…
Reference in new issue