From a09e370e1d8c0cdd29e4639e93c3587402b6a6bf Mon Sep 17 00:00:00 2001 From: Mikunj Date: Thu, 27 Feb 2020 12:11:08 +1100 Subject: [PATCH 1/2] Added testing on pull request Disable proxy in tests. --- .github/workflows/pull-request.yml | 67 +++ js/models/messages.js | 2 +- js/modules/loki_message_api.js | 7 +- js/modules/loki_rpc.js | 7 +- js/modules/loki_snode_api.js | 21 +- libloki/test/_test.js | 5 +- libtextsecure/message_receiver.js | 5 +- libtextsecure/test/_test.js | 5 +- package.json | 3 +- preload.js | 42 +- test/_test.js | 5 +- test/backup_test.js | 16 +- test/index.html | 517 +++++++++--------- test/spellcheck_test.js | 10 +- .../conversation/CreateGroupDialog.tsx | 6 +- ts/util/lint/linter.ts | 1 + yarn.lock | 10 +- 17 files changed, 426 insertions(+), 303 deletions(-) create mode 100644 .github/workflows/pull-request.yml 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" From b7a1c5fd5a485a7dbd6124fdf8e3b66cfb89544d Mon Sep 17 00:00:00 2001 From: Mikunj Date: Fri, 28 Feb 2020 14:54:12 +1100 Subject: [PATCH 2/2] Remove old CI files --- .gitlab-ci.yml | 61 -------------------------------------------------- .travis.yml | 33 --------------------------- appveyor.yml | 24 -------------------- aptly.sh | 52 ------------------------------------------ preload.js | 2 -- travis.sh | 13 ----------- 6 files changed, 185 deletions(-) delete mode 100644 .gitlab-ci.yml delete mode 100644 .travis.yml delete mode 100644 appveyor.yml delete mode 100755 aptly.sh delete mode 100755 travis.sh diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml deleted file mode 100644 index 76d3a0c03..000000000 --- a/.gitlab-ci.yml +++ /dev/null @@ -1,61 +0,0 @@ -# TODO: Figure out a way to use nvm in the linux build -linux: - image: node:10.13.0 - tags: - - docker - script: - - whoami - - node -v - - yarn -v - - yarn install --frozen-lockfile - - export SIGNAL_ENV=production - - yarn generate - - $(yarn bin)/electron-builder --config.extraMetadata.environment=$SIGNAL_ENV --config.mac.bundleVersion='$CI_COMMIT_REF_SLUG' --publish=never --config.directories.output=release - cache: - paths: - - node_modules/ - artifacts: - paths: - - release/ - -osx: - tags: - - osx - script: - - nvm install - - npm install --global yarn - - yarn install --frozen-lockfile - - export SIGNAL_ENV=production - - yarn generate - - $(yarn bin)/electron-builder --config.extraMetadata.environment=$SIGNAL_ENV --config.mac.bundleVersion='$CI_COMMIT_REF_SLUG' --publish=never --config.directories.output=release - cache: - paths: - - node_modules/ - artifacts: - paths: - - release/ - -windows: - tags: - - windows-cmd - script: - # install - - set PATH=%PATH%;C:\Users\Administrator\AppData\Local\nvs\ - - set SIGNAL_ENV=production - - set /p NVMRC_VER=<.nvmrc - - call nvs add %NVMRC_VER% - - call nvs use %NVMRC_VER% - - call "C:\\PROGRA~2\\MICROS~1\\2017\\BuildTools\\Common7\\Tools\\VsDevCmd.bat" - - call yarn install --frozen-lockfile - # build - - call yarn generate - - call node build\grunt.js - - call yarn prepare-beta-build - - call node_modules\.bin\electron-builder --config.extraMetadata.environment=%SIGNAL_ENV% --publish=never --config.directories.output=release - - call node build\grunt.js test-release:win - cache: - paths: - - node_modules/ - artifacts: - paths: - - release/ diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index 5f2a7a5a6..000000000 --- a/.travis.yml +++ /dev/null @@ -1,33 +0,0 @@ -language: node_js -cache: - yarn: true - directories: - - node_modules -node_js: - - '10.13.0' -install: - - travis_wait 30 yarn install --frozen-lockfile --network-timeout 1000000 -script: - - yarn generate - - yarn lint-windows - - yarn test -env: - global: - - SIGNAL_ENV: production -sudo: false -notifications: - email: false - -matrix: - include: - - name: 'Linux' - os: linux - dist: trusty - before_install: - - sudo apt-get install -y libgtk2.0-0 libgtk-3-0 libgconf-2-4 libasound2 libxtst6 libxss1 libnss3 xvfb hunspell-en-us - before_script: - - Xvfb -ac -screen scrn 1280x2000x24 :9.0 & - - export DISPLAY=:9.0 - - export LC_ALL=en_US - - name: 'OSX' - os: osx diff --git a/appveyor.yml b/appveyor.yml deleted file mode 100644 index 6b3167e51..000000000 --- a/appveyor.yml +++ /dev/null @@ -1,24 +0,0 @@ -platform: - - x64 - -cache: - - '%LOCALAPPDATA%\electron\Cache' - - node_modules -> package.json - -install: - - systeminfo | findstr /C:"OS" - - set PATH=C:\Ruby23-x64\bin;%PATH% - - ps: Install-Product node 10.13.0 x64 - - yarn install --frozen-lockfile - -build_script: - - node build\grunt.js - - yarn generate - - yarn lint-windows - - yarn test-node - -test_script: - - node build\grunt.js test - -environment: - SIGNAL_ENV: production diff --git a/aptly.sh b/aptly.sh deleted file mode 100755 index a761b806f..000000000 --- a/aptly.sh +++ /dev/null @@ -1,52 +0,0 @@ -#!/bin/bash -# Setup - creates the local repo which will be mirrored up to S3, then back-fill it. Your -# future deploys will eliminate all old versions without these backfill steps: -# aptly repo create signal-desktop -# aptly mirror create -ignore-signatures backfill-mirror https://updates.signal.org/desktop/apt xenial -# aptly mirror update -ignore-signatures backfill-mirror -# aptly repo import backfill-mirror signal-desktop signal-desktop signal-desktop-beta -# aptly repo show -with-packages signal-desktop -# -# First run on a machine - uncomment the first set of 'aptly publish snapshot' commands, -# comment the other two. Sets up the two publish channels, one local, one to S3. -# -# Testing - comment out the lines with s3:$ENDPOINT to publish only locally. To eliminate -# effects of testing, remove package from repo, then move back to old snapshot: -# aptly repo remove signal-desktop signal-desktop_1.0.35_amd64 -# aptly publish switch -gpg-key=57F6FB06 xenial signal-desktop_v1.0.34 -# -# Pruning package set - we generally want 2-3 versions of each stream available, -# production and beta. You can remove old packages like this: -# aptly repo show -with-packages signal-desktop -# aptly repo remove signal-desktop signal-desktop_1.0.34_amd64 -# -# Release: -# NAME=signal-desktop(-beta) VERSION=X.X.X ./aptly.sh - -echo "Releasing $NAME build version $VERSION" - -REPO=signal-desktop -CURRENT=xenial -# PREVIOUS=xenial -ENDPOINT=signal-desktop-apt # Matches endpoint name in .aptly.conf -SNAPSHOT=signal-desktop_v$VERSION -GPG_KEYID=57F6FB06 - -aptly repo add $REPO release/$NAME\_$VERSION\_*.deb -aptly snapshot create $SNAPSHOT from repo $REPO - -# run these only on first release to a given repo from a given machine. the first set is -# for local testing, the second set is to set up the production server. -# https://www.aptly.info/doc/aptly/publish/snapshot/ -# aptly publish snapshot -gpg-key=$GPG_KEYID -distribution=$CURRENT $SNAPSHOT -# aptly publish snapshot -gpg-key=$GPG_KEYID -distribution=$PREVIOUS $SNAPSHOT -# aptly publish snapshot -gpg-key=$GPG_KEYID -distribution=$CURRENT -config=.aptly.conf $SNAPSHOT s3:$ENDPOINT: -# aptly publish snapshot -gpg-key=$GPG_KEYID -distribution=$PREVIOUS -config=.aptly.conf $SNAPSHOT s3:$ENDPOINT: - -# these update already-published repos, run every time after that -# https://www.aptly.info/doc/aptly/publish/switch/ -aptly publish switch -gpg-key=$GPG_KEYID $CURRENT $SNAPSHOT -# aptly publish switch -gpg-key=$GPG_KEYID $PREVIOUS $SNAPSHOT -aptly publish switch -gpg-key=$GPG_KEYID -config=.aptly.conf $CURRENT s3:$ENDPOINT: $SNAPSHOT -# aptly publish switch -gpg-key=$GPG_KEYID -config=.aptly.conf $PREVIOUS s3:$ENDPOINT: $SNAPSHOT - diff --git a/preload.js b/preload.js index c7b161af3..901dd64d2 100644 --- a/preload.js +++ b/preload.js @@ -494,7 +494,6 @@ Promise.prototype.ignore = function() { }; 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 = { @@ -504,7 +503,6 @@ if (config.environment.includes('test')) { path: require('path'), basePath: __dirname, attachmentsPath: window.Signal.Migrations.attachmentsPath, - isTravis, isWindows, }; /* eslint-enable global-require, import/no-extraneous-dependencies */ diff --git a/travis.sh b/travis.sh deleted file mode 100755 index 8515d5851..000000000 --- a/travis.sh +++ /dev/null @@ -1,13 +0,0 @@ -#!/usr/bin/env bash - -set -e - -if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then - export DISPLAY=:99.0 - sh -e /etc/init.d/xvfb start - sleep 3 -fi - -yarn test-electron - -NODE_ENV=production yarn grunt test-release:$TRAVIS_OS_NAME