|
|
|
const path = require('path');
|
|
|
|
const packageJson = require('./package.json');
|
|
|
|
const importOnce = require('node-sass-import-once');
|
|
|
|
const rimraf = require('rimraf');
|
|
|
|
const mkdirp = require('mkdirp');
|
|
|
|
const spectron = require('spectron');
|
|
|
|
const asar = require('asar');
|
|
|
|
const fs = require('fs');
|
|
|
|
const assert = require('assert');
|
|
|
|
const sass = require('node-sass');
|
|
|
|
|
|
|
|
/* eslint-disable more/no-then, no-console */
|
|
|
|
|
|
|
|
module.exports = grunt => {
|
|
|
|
const bower = grunt.file.readJSON('bower.json');
|
|
|
|
const components = [];
|
|
|
|
// eslint-disable-next-line guard-for-in, no-restricted-syntax
|
|
|
|
for (const i in bower.concat.app) {
|
|
|
|
components.push(bower.concat.app[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
const libtextsecurecomponents = [];
|
|
|
|
// eslint-disable-next-line guard-for-in, no-restricted-syntax
|
|
|
|
for (const i in bower.concat.libtextsecure) {
|
|
|
|
libtextsecurecomponents.push(bower.concat.libtextsecure[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
const liblokicomponents = [];
|
|
|
|
// eslint-disable-next-line guard-for-in, no-restricted-syntax
|
|
|
|
for (const i in bower.concat.libloki) {
|
|
|
|
liblokicomponents.push(bower.concat.libloki[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
grunt.loadNpmTasks('grunt-sass');
|
|
|
|
|
|
|
|
grunt.initConfig({
|
|
|
|
pkg: grunt.file.readJSON('package.json'),
|
|
|
|
concat: {
|
|
|
|
components: {
|
|
|
|
src: components,
|
|
|
|
dest: 'js/components.js',
|
|
|
|
},
|
|
|
|
util_worker: {
|
|
|
|
src: [
|
|
|
|
'components/bytebuffer/dist/ByteBufferAB.js',
|
|
|
|
'components/JSBI/dist/jsbi.mjs',
|
|
|
|
'libloki/proof-of-work.js',
|
|
|
|
'components/long/dist/Long.js',
|
|
|
|
'js/util_worker_tasks.js',
|
|
|
|
],
|
|
|
|
dest: 'js/util_worker.js',
|
|
|
|
},
|
|
|
|
libtextsecurecomponents: {
|
|
|
|
src: libtextsecurecomponents,
|
|
|
|
dest: 'libtextsecure/components.js',
|
|
|
|
},
|
|
|
|
liblokicomponents: {
|
|
|
|
src: liblokicomponents,
|
|
|
|
dest: 'libloki/test/components.js',
|
|
|
|
},
|
|
|
|
test: {
|
|
|
|
src: [
|
|
|
|
'node_modules/mocha/mocha.js',
|
|
|
|
'node_modules/chai/chai.js',
|
|
|
|
'test/_test.js',
|
|
|
|
],
|
|
|
|
dest: 'test/test.js',
|
|
|
|
},
|
|
|
|
// TODO: Move errors back down?
|
|
|
|
libtextsecure: {
|
|
|
|
options: {
|
|
|
|
banner: ';(function() {\n',
|
|
|
|
footer: '})();\n',
|
|
|
|
},
|
|
|
|
src: [
|
|
|
|
'libtextsecure/errors.js',
|
|
|
|
'libtextsecure/libsignal-protocol.js',
|
|
|
|
'libtextsecure/protocol_wrapper.js',
|
|
|
|
|
|
|
|
'libtextsecure/crypto.js',
|
|
|
|
'libtextsecure/storage.js',
|
|
|
|
'libtextsecure/storage/user.js',
|
|
|
|
'libtextsecure/storage/groups.js',
|
|
|
|
'libtextsecure/storage/unprocessed.js',
|
|
|
|
'libtextsecure/protobufs.js',
|
|
|
|
'libtextsecure/helpers.js',
|
|
|
|
'libtextsecure/stringview.js',
|
|
|
|
'libtextsecure/event_target.js',
|
|
|
|
'libtextsecure/account_manager.js',
|
|
|
|
'libtextsecure/websocket-resources.js',
|
|
|
|
'libtextsecure/http-resources.js',
|
|
|
|
'libtextsecure/message_receiver.js',
|
|
|
|
'libtextsecure/outgoing_message.js',
|
|
|
|
'libtextsecure/sendmessage.js',
|
|
|
|
'libtextsecure/sync_request.js',
|
|
|
|
'libtextsecure/contacts_parser.js',
|
|
|
|
'libtextsecure/ProvisioningCipher.js',
|
|
|
|
'libtextsecure/task_with_timeout.js',
|
|
|
|
],
|
|
|
|
dest: 'js/libtextsecure.js',
|
|
|
|
},
|
|
|
|
libloki: {
|
|
|
|
src: [
|
|
|
|
'libloki/api.js',
|
|
|
|
'libloki/friends.js',
|
|
|
|
'libloki/crypto.js',
|
|
|
|
'libloki/service_nodes.js',
|
|
|
|
'libloki/storage.js',
|
|
|
|
],
|
|
|
|
dest: 'js/libloki.js',
|
|
|
|
},
|
|
|
|
lokitest: {
|
|
|
|
src: [
|
|
|
|
'node_modules/mocha/mocha.js',
|
|
|
|
'node_modules/chai/chai.js',
|
|
|
|
'libloki/test/_test.js',
|
|
|
|
],
|
|
|
|
dest: 'libloki/test/test.js',
|
|
|
|
},
|
|
|
|
libtextsecuretest: {
|
|
|
|
src: [
|
|
|
|
'node_modules/jquery/dist/jquery.js',
|
|
|
|
'components/mock-socket/dist/mock-socket.js',
|
|
|
|
'node_modules/mocha/mocha.js',
|
|
|
|
'node_modules/chai/chai.js',
|
|
|
|
'libtextsecure/test/_test.js',
|
|
|
|
],
|
|
|
|
dest: 'libtextsecure/test/test.js',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
sass: {
|
|
|
|
options: {
|
|
|
|
implementation: sass,
|
|
|
|
sourceMap: true,
|
|
|
|
importer: importOnce,
|
|
|
|
},
|
|
|
|
dev: {
|
|
|
|
files: {
|
|
|
|
'stylesheets/manifest.css': 'stylesheets/manifest.scss',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
copy: {
|
|
|
|
deps: {
|
|
|
|
files: [
|
|
|
|
{
|
|
|
|
src: 'components/mp3lameencoder/lib/Mp3LameEncoder.js',
|
|
|
|
dest: 'js/Mp3LameEncoder.min.js',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
src: 'components/webaudiorecorder/lib/WebAudioRecorderMp3.js',
|
|
|
|
dest: 'js/WebAudioRecorderMp3.js',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
watch: {
|
|
|
|
libtextsecure: {
|
|
|
|
files: ['./libtextsecure/*.js', './libtextsecure/storage/*.js'],
|
|
|
|
tasks: ['concat:libtextsecure'],
|
|
|
|
},
|
|
|
|
utilworker: {
|
|
|
|
files: [
|
|
|
|
'components/bytebuffer/dist/ByteBufferAB.js',
|
|
|
|
'components/JSBI/dist/jsbi.mjs',
|
|
|
|
'libloki/proof-of-work.js',
|
|
|
|
'components/long/dist/Long.js',
|
|
|
|
'js/util_worker_tasks.js',
|
|
|
|
],
|
|
|
|
tasks: ['concat:util_worker'],
|
|
|
|
},
|
|
|
|
libloki: {
|
|
|
|
files: ['./libloki/*.js'],
|
|
|
|
tasks: ['concat:libloki'],
|
|
|
|
},
|
|
|
|
protobuf: {
|
|
|
|
files: ['./protos/SignalService.proto'],
|
|
|
|
tasks: ['exec:build-protobuf'],
|
|
|
|
},
|
|
|
|
sass: {
|
|
|
|
files: ['./stylesheets/*.scss'],
|
|
|
|
tasks: ['sass'],
|
|
|
|
},
|
|
|
|
transpile: {
|
Session v1.0 changes (#802)
* correct typo in readme
* include log
* decrypt file server response, remove debug, handle crypt before _sendToProxy, improve json parsing failure logging
* support file uploads on file proxy, fix _sendToProxy calling
* bump form-data to 3.0
* initial refactor of feaure flag detection statements in serverRequest()
* fix send-message line-height with multiple lines
* fix lint
* fix position of delete account modal
* Profile picture upload, fixes and copy
* Various changes suggested by redesign overview
* Scrolling button updated and animations to modals
* Display subscriber count for open chats
* Prevent illegal username and passwords
* Delete channel / group merge
* Solidification of minor changes w appview injections
* hide description field in group panel for now
* fix join publicgroups pulls
* increase min height respecting ratio
* allow space inside a display name but not at start or end
* fix height of leftpane overlay view
* add back typing indicator and read receipt setting under privacy
* Auto-focus new open chat input box
* Password lock screen and delete data screen
* touchups
* Resolving Bilb revisions
* Disable link previews as default per Kee on signup
* remove date, we have git
* add missing semicolon
* _sendToProxy pass headers/handle response refactor, lint
* fix my yarn conflict/resolve
* include IV in server response
* Sealed sender support
* Support sealed sender for friend requests
* fix lint
* Remove unused destinationRegistrationId; lint
* Update messages.json
* pull RSS through file proxy
* fix unit tests: remove not used count in scrolldown view and assert svg
present
* Disable auto-joining default loki open groups
* session-id-editable-textarea
* fix the textscramble for sessionID on registration
* speed up lint, add lint-full/format-full, make sure use lint-full
* add skipToken to establishConnection options, smuggle out secureRpcPubKey
* get latest version through snode proxy, remove clearfix from ExpiredAlertBanner
* expose semver and LokiAppDotNetServerAPI because we can't get ourKey from storage early enough
* update note
* fix upgrade link, wrap expiredWarning in span for styling, use br to clear the float, trim trailing whitespace
* designalify
* designalify
* designalify user agent
* continue designalification
* make expired banner legible
* remove ugly TLS hack
* disable unauthorization rejection when making https requests limited to lokiRpc
* Update main.js
Aspect ratio amendment
* Constants rework
* local commit
* event listeners
* address missing comma for lint
* fix header sessions message section
* fix profile image size conversation list with pending friend request
* textarea centering
* refresh files in group in group panel
* Looking into keyboard navigation
* Remove P2P
* cache eslint on `lint` but not `ready`
* Cleanup media view formatting
* force locale to be EN until our files are updated and translated
* Simplification of keyup
* Updated all icon references
* SASS fixup
* fix disabled state of message input on sent friendrequest
* trim pubkey when user can enter one to remove whitespaces
* remove lZ in path which fixes errors on svg and does not alter rendering
* fix text scramble animation on registration
* reload app on ctrl-r or f5 from anywhere
* add back file which should have not been deleted
* fix lint and clean code
* fix lint
* add .loki to have a self-signed cert
* Remove mixpanel
* use local shortcut instead of global shortcut
otherwise, ctrl+r is only caught bu the last loaded instance
* open the conversation when accepting a friend request
also, it does what is needed to show the new friend in the friend list
* make sure token comms are done over fileProxy, other notes, logging adjustment
* leftpane sections titles are Wasa bold
* minor refactor
* onboarding messageview
* linter
* fix padding buttons overlay
* do not render session-id-editable border when textarea disabled
* textarea sessionID SpaceMono font
* various touchups
* fix font of description to sfprodisplay
* reduce triple dots conversation header icon size
* reduce size of conversationHeader title font size
* fix font for session-search-input
* make conversationlistitem title font wasa
* fix green and white border under title in leftpane
* fix panel-text-divider font-size and family
* disable completely borders for profile images
* make profile image which where 48pixels big 36 noew, as no more border
* Complete conditional message onboarding
* cache file deletions
* Link preview warning on setting toggle
* Messages.json amendments
* Join channel generalisation
* Localise global vars
* remove eslintcache
* rm global launchcount
* Remove source field from envelope
* Session public chat icon
* CLosed groups ui initial listprops
* Desktop: enable useSnodeProxy feature flag
* file proxy needs to be able to talk to snode
- disable TLS check for fileProxy
- lokiHttpsAgent => snodeHttpsAgent (since we use for two different things now)
* enable useSealedSender too per Maxim
* lint
* lint
* window.extension.expiredPromise version
* better error checking
* use promise version to see if we're expired
* fix typo
* lint
* put back seemingly now required process.env.NODE_TLS_REJECT_UNAUTHORIZED
* fix querystring in file-proxy
* lint
* fix typo
* Remove more references to signal.org
* make sure TLS is forced on open groups, improve serverRequest error message
* Closed groups UI
* function params changes
* turn off snode proxy logging
* include useful info on error
* actually validate URL before starting up a bunch of timers
* Closed groups overlay integration
* move comments from connecting_to_server_dialog_view
* use attempt from window object to reduce code duplication
* refactor out validServer()
* lint
* lint caught typo
* Rename BACKGROUND_FRIEND_REQUEST to SESSION_REQUEST.
Don't trigger friend request logic if a message is aimed at a group.
* Linting
* Closed group joining completed w/o backend
* Fix friend request messages being sent to users you don't have a session in closed groups.
Disable typing messages and read receipts in groups.
Send out session request messages if you don't have a session with a member in the group.
* Remove unneeded boolean condition.
* Closed group update message stylgin
* constants renaming
* Message deletion fix
* gruntify
* fix grunt error
* expose isRss, don't close uncloseable Rss conversation on deleteMessages
* remove copyId and block user on RSS feeds
* remove options from RSS feed that don't make any sense and don't work
* fix grunt error
* squelch RSS duplicate messages
* extension.expiredStatus(), adjustable timers, improve guards
* allowing sending of messages if we're still waiting to hear back
* markRandomNodeUnreachable() refactor, notes/logging
* improve logging
* improve logging
* no need to validate empty token, support lokinet/getession file domains, mark broken snodes as bad, improve logging
* try to address travis-osx lint complaints
* not designed to have a period at the end of titleIsNow
* put period back at the end
* Catch a stray loki messenger
* fix stray loki messenger
* loki messenger isnt a thing
* lint
* Fix open group joining.
* guards incase there are no members yet, fixes dialog not showing up
* fixed file server holding up message sender init.
fix joining closed groups.
* Clean
* Don't wait for file server to return tokens when establishing home connection.
* Disable join public chat prompt
Co-authored-by: Audric Ackermann <audric.bilb@gmail.com>
Co-authored-by: Ryan Tharp <neuro@interx.net>
Co-authored-by: Vince <vincent@loki.network>
Co-authored-by: Maxim Shishmarev <msgmaxim@gmail.com>
5 years ago
|
|
|
files: ['./ts/**/*.ts', './ts/**/*.tsx', './ts/**/**/*.tsx'],
|
|
|
|
tasks: ['exec:transpile'],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
exec: {
|
|
|
|
'tx-pull-new': {
|
|
|
|
cmd: 'tx pull -a --minimum-perc=80',
|
|
|
|
},
|
|
|
|
'tx-pull': {
|
|
|
|
cmd: 'tx pull',
|
|
|
|
},
|
|
|
|
transpile: {
|
|
|
|
cmd: 'yarn transpile',
|
|
|
|
},
|
|
|
|
'build-protobuf': {
|
|
|
|
cmd: 'yarn build-protobuf',
|
|
|
|
},
|
|
|
|
},
|
|
|
|
'test-release': {
|
|
|
|
osx: {
|
|
|
|
archive: `mac/${
|
|
|
|
packageJson.productName
|
|
|
|
}.app/Contents/Resources/app.asar`,
|
|
|
|
appUpdateYML: `mac/${
|
|
|
|
packageJson.productName
|
|
|
|
}.app/Contents/Resources/app-update.yml`,
|
|
|
|
exe: `mac/${packageJson.productName}.app/Contents/MacOS/${
|
|
|
|
packageJson.productName
|
|
|
|
}`,
|
|
|
|
},
|
|
|
|
mas: {
|
|
|
|
archive: 'mas/Signal.app/Contents/Resources/app.asar',
|
|
|
|
appUpdateYML: 'mac/Signal.app/Contents/Resources/app-update.yml',
|
|
|
|
exe: `mas/${packageJson.productName}.app/Contents/MacOS/${
|
|
|
|
packageJson.productName
|
|
|
|
}`,
|
|
|
|
},
|
|
|
|
linux: {
|
|
|
|
archive: 'linux-unpacked/resources/app.asar',
|
|
|
|
exe: `linux-unpacked/${packageJson.name}`,
|
|
|
|
},
|
|
|
|
win: {
|
|
|
|
archive: 'win-unpacked/resources/app.asar',
|
|
|
|
appUpdateYML: 'win-unpacked/resources/app-update.yml',
|
|
|
|
exe: `win-unpacked/${packageJson.productName}.exe`,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
gitinfo: {}, // to be populated by grunt gitinfo
|
|
|
|
});
|
|
|
|
|
|
|
|
Object.keys(grunt.config.get('pkg').devDependencies).forEach(key => {
|
|
|
|
if (/^grunt(?!(-cli)?$)/.test(key)) {
|
|
|
|
// ignore grunt and grunt-cli
|
|
|
|
grunt.loadNpmTasks(key);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// Transifex does not understand placeholders, so this task patches all non-en
|
|
|
|
// locales with missing placeholders
|
|
|
|
grunt.registerTask('locale-patch', () => {
|
|
|
|
const en = grunt.file.readJSON('_locales/en/messages.json');
|
|
|
|
grunt.file.recurse('_locales', (abspath, rootdir, subdir, filename) => {
|
|
|
|
if (subdir === 'en' || filename !== 'messages.json') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const messages = grunt.file.readJSON(abspath);
|
|
|
|
|
|
|
|
// eslint-disable-next-line no-restricted-syntax
|
|
|
|
for (const key in messages) {
|
|
|
|
if (en[key] !== undefined && messages[key] !== undefined) {
|
|
|
|
if (
|
|
|
|
en[key].placeholders !== undefined &&
|
|
|
|
messages[key].placeholders === undefined
|
|
|
|
) {
|
|
|
|
messages[key].placeholders = en[key].placeholders;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
grunt.file.write(abspath, `${JSON.stringify(messages, null, 4)}\n`);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
function updateLocalConfig(update) {
|
|
|
|
const environment = process.env.SIGNAL_ENV || 'development';
|
|
|
|
const configPath = `config/local-${environment}.json`;
|
|
|
|
let localConfig;
|
|
|
|
try {
|
|
|
|
localConfig = grunt.file.readJSON(configPath);
|
|
|
|
} catch (e) {
|
|
|
|
//
|
|
|
|
}
|
|
|
|
localConfig = {
|
|
|
|
...localConfig,
|
|
|
|
...update,
|
|
|
|
};
|
|
|
|
grunt.file.write(configPath, `${JSON.stringify(localConfig)}\n`);
|
|
|
|
}
|
|
|
|
|
|
|
|
grunt.registerTask('getExpireTime', () => {
|
|
|
|
grunt.task.requires('gitinfo');
|
|
|
|
const gitinfo = grunt.config.get('gitinfo');
|
|
|
|
const committed = gitinfo.local.branch.current.lastCommitTime;
|
|
|
|
const time = Date.parse(committed) + 1000 * 60 * 60 * 24 * 90;
|
|
|
|
updateLocalConfig({ buildExpiration: time });
|
|
|
|
});
|
|
|
|
|
|
|
|
grunt.registerTask('getCommitHash', () => {
|
|
|
|
grunt.task.requires('gitinfo');
|
|
|
|
const gitinfo = grunt.config.get('gitinfo');
|
|
|
|
const hash = gitinfo.local.branch.current.SHA;
|
|
|
|
updateLocalConfig({ commitHash: hash });
|
|
|
|
});
|
|
|
|
|
|
|
|
grunt.registerTask('clean-release', () => {
|
|
|
|
rimraf.sync('release');
|
|
|
|
mkdirp.sync('release');
|
|
|
|
});
|
|
|
|
|
|
|
|
function runTests(environment, cb) {
|
|
|
|
let failure;
|
|
|
|
const { Application } = spectron;
|
|
|
|
const electronBinary =
|
|
|
|
process.platform === 'win32' ? 'electron.cmd' : 'electron';
|
|
|
|
const app = new Application({
|
|
|
|
path: path.join(__dirname, 'node_modules', '.bin', electronBinary),
|
|
|
|
args: [path.join(__dirname, 'main.js')],
|
|
|
|
env: {
|
|
|
|
NODE_ENV: environment,
|
|
|
|
},
|
|
|
|
requireName: 'unused',
|
|
|
|
});
|
|
|
|
|
|
|
|
function getMochaResults() {
|
|
|
|
// eslint-disable-next-line no-undef
|
|
|
|
return window.mochaResults;
|
|
|
|
}
|
|
|
|
|
|
|
|
app
|
|
|
|
.start()
|
|
|
|
.then(() =>
|
|
|
|
app.client.waitUntil(
|
|
|
|
() =>
|
|
|
|
app.client
|
|
|
|
.execute(getMochaResults)
|
|
|
|
.then(data => Boolean(data.value)),
|
|
|
|
25000,
|
|
|
|
'Expected to find window.mochaResults set!'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
.then(() => app.client.execute(getMochaResults))
|
|
|
|
.then(data => {
|
|
|
|
const results = data.value;
|
|
|
|
if (results.failures > 0) {
|
|
|
|
console.error(results.reports);
|
|
|
|
failure = () =>
|
|
|
|
grunt.fail.fatal(`Found ${results.failures} failing unit tests.`);
|
|
|
|
return app.client.log('browser');
|
|
|
|
}
|
|
|
|
grunt.log.ok(`${results.passes} tests passed.`);
|
|
|
|
return null;
|
|
|
|
})
|
|
|
|
.then(logs => {
|
|
|
|
if (logs) {
|
|
|
|
console.error();
|
|
|
|
console.error('Because tests failed, printing browser logs:');
|
|
|
|
console.error(logs);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
failure = () =>
|
|
|
|
grunt.fail.fatal(
|
|
|
|
`Something went wrong: ${error.message} ${error.stack}`
|
|
|
|
);
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
// We need to use the failure variable and this early stop to clean up before
|
|
|
|
// shutting down. Grunt's fail methods are the only way to set the return value,
|
|
|
|
// but they shut the process down immediately!
|
|
|
|
if (failure) {
|
|
|
|
console.log();
|
|
|
|
console.log('Main process logs:');
|
|
|
|
return app.client.getMainProcessLogs().then(logs => {
|
|
|
|
logs.forEach(log => {
|
|
|
|
console.log(log);
|
|
|
|
});
|
|
|
|
|
|
|
|
return app.stop();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return app.stop();
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
if (failure) {
|
|
|
|
failure();
|
|
|
|
}
|
|
|
|
cb();
|
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
console.error('Second-level error:', error.message, error.stack);
|
|
|
|
if (failure) {
|
|
|
|
failure();
|
|
|
|
}
|
|
|
|
cb();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
grunt.registerTask(
|
|
|
|
'unit-tests',
|
|
|
|
'Run unit tests w/Electron',
|
|
|
|
function thisNeeded() {
|
|
|
|
const environment = grunt.option('env') || 'test';
|
|
|
|
const done = this.async();
|
|
|
|
|
|
|
|
runTests(environment, done);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
grunt.registerTask(
|
|
|
|
'lib-unit-tests',
|
|
|
|
'Run libtextsecure unit tests w/Electron',
|
|
|
|
function thisNeeded() {
|
|
|
|
const environment = grunt.option('env') || 'test-lib';
|
|
|
|
const done = this.async();
|
|
|
|
|
|
|
|
runTests(environment, done);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
grunt.registerTask(
|
|
|
|
'loki-unit-tests',
|
|
|
|
'Run loki unit tests w/Electron',
|
|
|
|
function thisNeeded() {
|
|
|
|
const environment = grunt.option('env') || 'test-loki';
|
|
|
|
const done = this.async();
|
|
|
|
|
|
|
|
runTests(environment, done);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
grunt.registerMultiTask(
|
|
|
|
'test-release',
|
|
|
|
'Test packaged releases',
|
|
|
|
function thisNeeded() {
|
|
|
|
const dir = grunt.option('dir') || 'release';
|
|
|
|
const environment = grunt.option('env') || 'production';
|
|
|
|
const config = this.data;
|
|
|
|
const archive = [dir, config.archive].join('/');
|
|
|
|
const files = [
|
|
|
|
'config/default.json',
|
|
|
|
`config/${environment}.json`,
|
|
|
|
`config/local-${environment}.json`,
|
|
|
|
];
|
|
|
|
|
|
|
|
console.log(this.target, archive);
|
|
|
|
const releaseFiles = files.concat(config.files || []);
|
|
|
|
releaseFiles.forEach(fileName => {
|
|
|
|
console.log(fileName);
|
|
|
|
try {
|
|
|
|
asar.statFile(archive, fileName);
|
|
|
|
return true;
|
|
|
|
} catch (e) {
|
|
|
|
console.log(e);
|
|
|
|
throw new Error(`Missing file ${fileName}`);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if (config.appUpdateYML) {
|
|
|
|
const appUpdateYML = [dir, config.appUpdateYML].join('/');
|
|
|
|
if (fs.existsSync(appUpdateYML)) {
|
|
|
|
console.log('auto update ok');
|
|
|
|
} else {
|
|
|
|
throw new Error(`Missing auto update config ${appUpdateYML}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const done = this.async();
|
|
|
|
// A simple test to verify a visible window is opened with a title
|
|
|
|
const { Application } = spectron;
|
|
|
|
|
|
|
|
const app = new Application({
|
|
|
|
path: [dir, config.exe].join('/'),
|
|
|
|
requireName: 'unused',
|
|
|
|
});
|
|
|
|
|
|
|
|
app
|
|
|
|
.start()
|
|
|
|
.then(() => app.client.getWindowCount())
|
|
|
|
.then(count => {
|
|
|
|
assert.equal(count, 1);
|
|
|
|
console.log('window opened');
|
|
|
|
})
|
|
|
|
.then(() =>
|
|
|
|
// Get the window's title
|
|
|
|
app.client.getTitle()
|
|
|
|
)
|
|
|
|
.then(title => {
|
|
|
|
// TODO: restore once fixed on win
|
|
|
|
if (this.target !== 'win') {
|
|
|
|
// Verify the window's title
|
|
|
|
assert.equal(title, packageJson.productName);
|
|
|
|
console.log('title ok');
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.then(() => {
|
|
|
|
assert(
|
|
|
|
app.chromeDriver.logLines.indexOf(`NODE_ENV ${environment}`) > -1
|
|
|
|
);
|
|
|
|
console.log('environment ok');
|
|
|
|
})
|
|
|
|
.then(
|
|
|
|
() =>
|
|
|
|
// Successfully completed test
|
|
|
|
app.stop(),
|
|
|
|
error =>
|
|
|
|
// Test failed!
|
|
|
|
app.stop().then(() => {
|
|
|
|
grunt.fail.fatal(`Test failed: ${error.message} ${error.stack}`);
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.then(done);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
grunt.registerTask('tx', [
|
|
|
|
'exec:tx-pull-new',
|
|
|
|
'exec:tx-pull',
|
|
|
|
'locale-patch',
|
|
|
|
]);
|
|
|
|
grunt.registerTask('dev', ['default', 'watch']);
|
|
|
|
grunt.registerTask('test', [
|
|
|
|
'unit-tests',
|
|
|
|
'lib-unit-tests',
|
|
|
|
'loki-unit-tests',
|
|
|
|
]);
|
|
|
|
grunt.registerTask('date', ['gitinfo', 'getExpireTime']);
|
|
|
|
grunt.registerTask('default', [
|
|
|
|
'exec:build-protobuf',
|
|
|
|
'exec:transpile',
|
|
|
|
'concat',
|
|
|
|
'copy:deps',
|
|
|
|
'sass',
|
|
|
|
'date',
|
|
|
|
'getCommitHash',
|
|
|
|
]);
|
|
|
|
};
|