diff --git a/.github/workflows/pull-request.yml b/.github/workflows/pull-request.yml new file mode 100644 index 000000000..c500c9ca0 --- /dev/null +++ b/.github/workflows/pull-request.yml @@ -0,0 +1,67 @@ +# This script will run tests anytime a pull request is added +name: Session Test + +on: + pull_request: + branches: + - development + - clearnet + - github-actions + +jobs: + build: + runs-on: ${{ matrix.os }} + strategy: + fail-fast: false + matrix: + os: [windows-2016, macos-latest, ubuntu-latest] + env: + SIGNAL_ENV: production + GH_TOKEN: ${{ secrets.GITHUB_TOKEN }} + steps: + - run: git config --global core.autocrlf false + + - name: Checkout git repo + uses: actions/checkout@v1 + + - name: Install node + uses: actions/setup-node@v1 + with: + node-version: 10.13.0 + + - name: Setup node for windows + if: runner.os == 'Windows' + run: | + npm install --global --production windows-build-tools@4.0.0 + npm install --global node-gyp@latest + npm config set python python2.7 + npm config set msvs_version 2015 + + - name: Install yarn + run: npm install yarn --no-save + + - name: Install Dependencies + run: yarn install --frozen-lockfile + + - name: Generate and concat files + run: yarn generate + + - name: Lint Files + run: | + yarn format-full --list-different + yarn eslint + yarn tslint + + - name: Make linux use en_US locale + if: runner.os == 'Linux' + run: | + sudo apt-get install -y hunspell-en-us + sudo locale-gen en_US.UTF-8 + sudo dpkg-reconfigure locales + echo ::set-env name=DISPLAY:::9.0 + echo ::set-env name=LANG::en_US.UTF-8 + + - name: Test + uses: GabrielBB/xvfb-action@v1.0 + with: + run: yarn test diff --git a/js/models/messages.js b/js/models/messages.js index 7b024886b..e6dc4f074 100644 --- a/js/models/messages.js +++ b/js/models/messages.js @@ -1422,7 +1422,7 @@ if (this.get('type') !== 'friend-request') { const c = this.getConversation(); // Don't bother sending sync messages to public chats - if (!c.isPublic()) { + if (c && !c.isPublic()) { this.sendSyncMessage(); } } diff --git a/js/modules/loki_message_api.js b/js/modules/loki_message_api.js index 98411fd85..88b02b7af 100644 --- a/js/modules/loki_message_api.js +++ b/js/modules/loki_message_api.js @@ -217,7 +217,12 @@ class LokiMessageAPI { } return true; } catch (e) { - log.warn('Loki send message error:', e.code, e.message, `from ${address}`); + log.warn( + 'Loki send message error:', + e.code, + e.message, + `from ${address}` + ); if (e instanceof textsecure.WrongSwarmError) { const { newSwarm } = e; await lokiSnodeAPI.updateSwarmNodes(params.pubKey, newSwarm); diff --git a/js/modules/loki_rpc.js b/js/modules/loki_rpc.js index 61fb83926..5f29f6c38 100644 --- a/js/modules/loki_rpc.js +++ b/js/modules/loki_rpc.js @@ -70,7 +70,10 @@ const sendToProxy = async (options = {}, targetNode) => { // detect SNode is not ready (not in swarm; not done syncing) if (response.status === 503) { const ciphertext = await response.text(); - log.error(`lokiRpc sendToProxy snode ${randSnode.ip}:${randSnode.port} error`, ciphertext); + log.error( + `lokiRpc sendToProxy snode ${randSnode.ip}:${randSnode.port} error`, + ciphertext + ); // mark as bad for this round (should give it some time and improve success rates) lokiSnodeAPI.markRandomNodeUnreachable(randSnode); // retry for a new working snode @@ -104,7 +107,7 @@ const sendToProxy = async (options = {}, targetNode) => { const textDecoder = new TextDecoder(); plaintext = textDecoder.decode(plaintextBuffer); - } catch(e) { + } catch (e) { log.error( 'lokiRpc sendToProxy decode error', e.code, diff --git a/js/modules/loki_snode_api.js b/js/modules/loki_snode_api.js index 7bc2944ed..7d163a215 100644 --- a/js/modules/loki_snode_api.js +++ b/js/modules/loki_snode_api.js @@ -31,7 +31,10 @@ class LokiSnodeAPI { ]; } - async initialiseRandomPool(seedNodes = [...window.seedNodeList], consecutiveErrors = 0) { + async initialiseRandomPool( + seedNodes = [...window.seedNodeList], + consecutiveErrors = 0 + ) { const params = { limit: 20, active_only: true, @@ -71,7 +74,10 @@ class LokiSnodeAPI { if (consecutiveErrors < 3) { // retry after a possible delay setTimeout(() => { - log.info('Retrying initialising random snode pool, try #', consecutiveErrors); + log.info( + 'Retrying initialising random snode pool, try #', + consecutiveErrors + ); this.initialiseRandomPool(seedNodes, consecutiveErrors + 1); }, consecutiveErrors * consecutiveErrors * 5000); } else { @@ -181,7 +187,12 @@ class LokiSnodeAPI { const snodes = result.snodes.filter(tSnode => tSnode.ip !== '0.0.0.0'); return snodes; } catch (e) { - log.error('getSnodesForPubkey error', e.code, e.message, `for ${snode.ip}:${snode.port}`); + log.error( + 'getSnodesForPubkey error', + e.code, + e.message, + `for ${snode.ip}:${snode.port}` + ); this.markRandomNodeUnreachable(snode); return []; } @@ -197,7 +208,9 @@ class LokiSnodeAPI { const resList = await this.getSnodesForPubkey(rSnode, pubKey); // should we only activate entries that are in all results? resList.map(item => { - const hasItem = snodes.some(hItem => item.ip === hItem.ip && item.port === hItem.port); + const hasItem = snodes.some( + hItem => item.ip === hItem.ip && item.port === hItem.port + ); if (!hasItem) { snodes.push(item); } diff --git a/libloki/test/_test.js b/libloki/test/_test.js index e26c8ab40..ab439a5e8 100644 --- a/libloki/test/_test.js +++ b/libloki/test/_test.js @@ -1,6 +1,9 @@ /* global window, mocha, chai, assert, Whisper */ -mocha.setup('bdd'); +mocha + .setup('bdd') + .fullTrace() + .timeout(10000); window.assert = chai.assert; window.PROTO_ROOT = '../../protos'; diff --git a/libtextsecure/message_receiver.js b/libtextsecure/message_receiver.js index 0e6204016..b6b24f355 100644 --- a/libtextsecure/message_receiver.js +++ b/libtextsecure/message_receiver.js @@ -59,7 +59,10 @@ function MessageReceiver(username, password, signalingKey, options = {}) { lokiPublicChatAPI.removeAllListeners('publicMessage'); // we only need one MR in the system handling these // bind events - lokiPublicChatAPI.on('publicMessage', this.handleUnencryptedMessage.bind(this)); + lokiPublicChatAPI.on( + 'publicMessage', + this.handleUnencryptedMessage.bind(this) + ); openGroupBound = true; } } else { diff --git a/libtextsecure/test/_test.js b/libtextsecure/test/_test.js index a77e132e9..7edd32586 100644 --- a/libtextsecure/test/_test.js +++ b/libtextsecure/test/_test.js @@ -1,6 +1,9 @@ /* global mocha, chai, assert */ -mocha.setup('bdd'); +mocha + .setup('bdd') + .fullTrace() + .timeout(10000); window.assert = chai.assert; window.PROTO_ROOT = '../../protos'; diff --git a/package.json b/package.json index c5a4d1a00..206065e8d 100644 --- a/package.json +++ b/package.json @@ -42,6 +42,7 @@ "test-node-coverage": "nyc --reporter=lcov --reporter=text mocha --recursive test/app test/modules ts/test libloki/test/node", "test-node-coverage-html": "nyc --reporter=lcov --reporter=html mocha --recursive test/a/* */pp test/modules ts/test libloki/test/node", "eslint": "eslint --cache .", + "eslint-fix": "eslint --fix .", "eslint-full": "eslint .", "lint": "yarn format --list-different && yarn lint-windows", "lint-full": "yarn format-full --list-different; yarn lint-windows-full", @@ -124,7 +125,7 @@ "reselect": "4.0.0", "rimraf": "2.6.2", "semver": "5.4.1", - "spellchecker": "3.5.1", + "spellchecker": "3.7.0", "tar": "4.4.8", "testcheck": "1.0.0-rc.2", "tmp": "0.0.33", diff --git a/preload.js b/preload.js index c9383e136..c7b161af3 100644 --- a/preload.js +++ b/preload.js @@ -475,23 +475,6 @@ contextMenu({ // /tmp mounted as noexec on Linux. require('./js/spell_check'); -if (config.environment === 'test') { - const isTravis = 'TRAVIS' in process.env && 'CI' in process.env; - const isWindows = process.platform === 'win32'; - /* eslint-disable global-require, import/no-extraneous-dependencies */ - window.test = { - glob: require('glob'), - fse: require('fs-extra'), - tmp: require('tmp'), - path: require('path'), - basePath: __dirname, - attachmentsPath: window.Signal.Migrations.attachmentsPath, - isTravis, - isWindows, - }; - /* eslint-enable global-require, import/no-extraneous-dependencies */ -} - window.shortenPubkey = pubkey => `(...${pubkey.substring(pubkey.length - 6)})`; window.pubkeyPattern = /@[a-fA-F0-9]{64,66}\b/g; @@ -509,3 +492,28 @@ Promise.prototype.ignore = function() { // eslint-disable-next-line more/no-then this.then(() => {}); }; + +if (config.environment.includes('test')) { + const isTravis = 'TRAVIS' in process.env && 'CI' in process.env; + const isWindows = process.platform === 'win32'; + /* eslint-disable global-require, import/no-extraneous-dependencies */ + window.test = { + glob: require('glob'), + fse: require('fs-extra'), + tmp: require('tmp'), + path: require('path'), + basePath: __dirname, + attachmentsPath: window.Signal.Migrations.attachmentsPath, + isTravis, + isWindows, + }; + /* eslint-enable global-require, import/no-extraneous-dependencies */ + window.lokiFeatureFlags = {}; + window.lokiSnodeAPI = { + refreshSwarmNodesForPubKey: () => [], + getFreshSwarmNodes: () => [], + updateSwarmNodes: () => {}, + updateLastHash: () => {}, + getSwarmNodesForPubKey: () => [], + }; +} diff --git a/test/_test.js b/test/_test.js index 6e1e03ae2..2f9a3a30b 100644 --- a/test/_test.js +++ b/test/_test.js @@ -1,6 +1,9 @@ /* global chai, Whisper, _, Backbone */ -mocha.setup('bdd'); +mocha + .setup('bdd') + .fullTrace() + .timeout(10000); window.assert = chai.assert; window.PROTO_ROOT = '../protos'; diff --git a/test/backup_test.js b/test/backup_test.js index f5146cc09..7d7ef8eea 100644 --- a/test/backup_test.js +++ b/test/backup_test.js @@ -239,20 +239,12 @@ describe('Backup', () => { it('exports then imports to produce the same data we started with', async function thisNeeded() { this.timeout(6000); - const { - attachmentsPath, - fse, - glob, - path, - tmp, - isTravis, - isWindows, - } = window.test; - - // Skip this test on travis windows + const { attachmentsPath, fse, glob, path, tmp, isWindows } = window.test; + + // Skip this test on windows // because it always fails due to lstat permission error. // Don't know how to fix it so this is a temp work around. - if (isTravis && isWindows) { + if (isWindows) { console.log( 'Skipping exports then imports to produce the same data we started' ); diff --git a/test/index.html b/test/index.html index a5bc6a046..54d1dd46f 100644 --- a/test/index.html +++ b/test/index.html @@ -1,7 +1,7 @@ - + TextSecure test runner @@ -11,90 +11,90 @@
-
+
-
+
- - - - - - - - - - - - - - - - - - - - - - - - - - + @@ -505,50 +505,62 @@ - + - - - + + + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -559,11 +571,10 @@ - + - diff --git a/test/spellcheck_test.js b/test/spellcheck_test.js index 57f7b529f..6937d9d9e 100644 --- a/test/spellcheck_test.js +++ b/test/spellcheck_test.js @@ -1,6 +1,12 @@ describe('spellChecker', () => { it('should work', () => { - assert(window.spellChecker.spellCheck('correct')); - assert(!window.spellChecker.spellCheck('fhqwgads')); + assert( + window.spellChecker.spellCheck('correct'), + 'Spellchecker returned false on a correct word.' + ); + assert( + !window.spellChecker.spellCheck('fhqwgads'), + 'Spellchecker returned true on a incorrect word.' + ); }); }); diff --git a/ts/components/conversation/CreateGroupDialog.tsx b/ts/components/conversation/CreateGroupDialog.tsx index ce853d4ce..482a35579 100644 --- a/ts/components/conversation/CreateGroupDialog.tsx +++ b/ts/components/conversation/CreateGroupDialog.tsx @@ -95,7 +95,11 @@ export class CreateGroupDialog extends React.Component { ); return ( - null}> + null} + >

{this.state.errorMessage}

diff --git a/ts/util/lint/linter.ts b/ts/util/lint/linter.ts index a481a7db1..1ada85d60 100644 --- a/ts/util/lint/linter.ts +++ b/ts/util/lint/linter.ts @@ -73,6 +73,7 @@ const excludedFiles = [ '^js/Mp3LameEncoder.min.js', // Test files + '^libloki/test/*', '^libtextsecure/test/*', '^test/*', diff --git a/yarn.lock b/yarn.lock index c97426256..dc8d73cae 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9704,13 +9704,13 @@ speedometer@~0.1.2: resolved "https://registry.yarnpkg.com/speedometer/-/speedometer-0.1.4.tgz#9876dbd2a169d3115402d48e6ea6329c8816a50d" integrity sha1-mHbb0qFp0xFUAtSObqYynIgWpQ0= -spellchecker@3.5.1: - version "3.5.1" - resolved "https://registry.yarnpkg.com/spellchecker/-/spellchecker-3.5.1.tgz#72cc2bcbb0c610536bc2a36df1ced9b2665816cc" - integrity sha512-R1qUBsDZzio+7MFZN6/AtPUe5NGvnc0wywckuXAlp9akASaYSFqKuI5O8p3rSiA+yKP31qC7Iijjoygmzkh6xw== +spellchecker@3.7.0: + version "3.7.0" + resolved "https://registry.yarnpkg.com/spellchecker/-/spellchecker-3.7.0.tgz#d63e6fd612352b0108e7bbf942f271665ff63c8b" + integrity sha512-saQT4BR9nivbK70s0YjyIlSbZzO6bfWRULcGL2JU7fi7wotOnWl70P0QoUwwLywNQJQ47osgCo6GmOlqzRTxbQ== dependencies: any-promise "^1.3.0" - nan "^2.10.0" + nan "^2.14.0" split-string@^3.0.1, split-string@^3.0.2: version "3.1.0"