diff --git a/.github/workflows/build-binaries.yml b/.github/workflows/build-binaries.yml new file mode 100644 index 000000000..e3dfd3a8e --- /dev/null +++ b/.github/workflows/build-binaries.yml @@ -0,0 +1,85 @@ +# This script will build session production binaries anytime a branch is updated +name: Session Build Binaries + +on: + push: + branches: + - master + - 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: + - 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 lint-full + + - name: Build windows production binaries + if: runner.os == 'Windows' + run: node_modules\.bin\electron-builder --config.extraMetadata.environment=%SIGNAL_ENV% --publish=never --config.directories.output=release + + - name: Build mac production binaries + if: runner.os == 'macOS' + run: $(yarn bin)/electron-builder --config.extraMetadata.environment=$SIGNAL_ENV --config.mac.bundleVersion=${{ github.ref }} --publish=never --config.directories.output=release + env: + CSC_LINK: ${{ secrets.MAC_CERTIFICATE }} + CSC_KEY_PASSWORD: ${{ secrets.MAC_CERTIFICATE_PASSWORD }} + SIGNING_APPLE_ID: ${{ secrets.SIGNING_APPLE_ID }} + SIGNING_APP_PASSWORD: ${{ secrets.SIGNING_APP_PASSWORD }} + SIGNING_TEAM_ID: ${{ secrets.SIGNING_TEAM_ID }} + + - name: Build linux production binaries + if: runner.os == 'Linux' + run: $(yarn bin)/electron-builder --config.extraMetadata.environment=$SIGNAL_ENV --publish=never --config.directories.output=release + + - name: Remove unpacked files + run: | + ls -d -- */ | xargs -I{} echo "Removing {}" + ls -d -- */ | xargs -I{} rm -rf {} + shell: bash + working-directory: ./release/ + + - name: Remaining files + run: ls . + shell: bash + working-directory: ./release/ + + - name: Upload Production Artifacts + uses: actions/upload-artifact@v1 + with: + name: ${{ runner.OS }}-production + path: release diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml new file mode 100644 index 000000000..2ea6585df --- /dev/null +++ b/.github/workflows/release.yml @@ -0,0 +1,64 @@ +# This script will build binaries and publish a draft on github release page with the the tag v[package-version] +name: Session Draft Release + +on: + push: + branches: + - master + +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: + - 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 lint-full + + - name: Build windows production binaries + if: runner.os == 'Windows' + run: node_modules\.bin\electron-builder --config.extraMetadata.environment=%SIGNAL_ENV% --publish=always + + - name: Build mac production binaries + if: runner.os == 'macOS' + run: $(yarn bin)/electron-builder --config.extraMetadata.environment=$SIGNAL_ENV --config.mac.bundleVersion=${{ github.ref }} --publish=always + env: + CSC_LINK: ${{ secrets.MAC_CERTIFICATE }} + CSC_KEY_PASSWORD: ${{ secrets.MAC_CERTIFICATE_PASSWORD }} + SIGNING_APPLE_ID: ${{ secrets.SIGNING_APPLE_ID }} + SIGNING_APP_PASSWORD: ${{ secrets.SIGNING_APP_PASSWORD }} + SIGNING_TEAM_ID: ${{ secrets.SIGNING_TEAM_ID }} + + - name: Build linux production binaries + if: runner.os == 'Linux' + run: $(yarn bin)/electron-builder --config.extraMetadata.environment=$SIGNAL_ENV --publish=always diff --git a/BUILDING.md b/BUILDING.md new file mode 100644 index 000000000..ae7ed1c51 --- /dev/null +++ b/BUILDING.md @@ -0,0 +1,42 @@ +# Building + +Building session binaries is done using github actions. Windows and linux binaries will build right out of the box but there are some extra steps needed for Mac OS + +## Mac OS + +The build script for Mac OS requires you to have a valid `Developer ID Application` certificate. Without this the build script cannot sign and notarize the mac binary which is needed for Catalina 10.15 and above. +If you would like to disable this then comment out `"afterSign": "build/notarize.js",` in package.json. + +You will also need an [App-specific password](https://support.apple.com/en-al/HT204397) for the apple account you wish to notarize with + +### Setup + +Once you have your `Developer ID Application` you need to export it into a `.p12` file. Keep a note of the password used to encrypt this file as it will be needed later. + +We need to Base64 encode this file, so run the following command: + +``` +base64 -i certificate.p12 -o encoded.txt +``` + +#### On GitHub: + +1. Navigate to the main page of the repository. +2. Under your repository name, click **Settings**. +3. In the left sidebar, click **Secrets**. +4. Add the following secrets: + 1. Certificate + * Name: `MAC_CERTIFICATE` + * Value: The encoded Base64 certificate + 2. Certificate password + * Name: `MAC_CERTIFICATE_PASSWORD` + * Value: The password that was set when the certificate was exported. + 3. Apple ID + * Name: `SIGNING_APPLE_ID` + * Value: The apple id (email) to use for signing + 4. Apple Password + * Name: `SIGNING_APP_PASSWORD` + * Value: The app-specific password that was generated for the apple id + 5. Team ID (Optional) + * Name: `SIGNING_TEAM_ID` + * Value: The apple team id if you're sigining the application for a team diff --git a/README.md b/README.md index 766d7db52..ce39ccced 100644 --- a/README.md +++ b/README.md @@ -2,15 +2,11 @@ [![Build Status](https://travis-ci.org/loki-project/loki-messenger.svg?branch=development)](https://travis-ci.org/loki-project/loki-messenger) -Session allows for truly decentralized, end to end, and private encrypted chats. Session is built to handle both online and fully Asynchronous offline messages. Session implements the Signal protocol for message encryption. Our Client interface is a fork of [Signal Messenger](https://signal.org/). All communication that passes through Session is routed through [Lokinet](https://github.com/loki-project/loki-network). +Session allows for truly decentralized, end to end, and private encrypted chats. Session is built to handle both online and fully Asynchronous offline messages. Session implements the Signal protocol for message encryption. Our Client interface is a fork of [Signal Messenger](https://signal.org/). ## Summary -Session integrates directly with Loki [Service Nodes](https://lokidocs.com/ServiceNodes/SNOverview/), which are a set of distributed, decentralized and Sybil resistant nodes. Service Nodes act as both federated servers which store messages offline, and a set of nodes which allow for onion routing functionality obfuscating users IP Addresses. For a full understanding of how Session works, read the [Loki whitepaper](https://loki.network/whitepaper). - -**Online Messages** - -If Alice and Bob are both online they can simply resolve each others public keys, to introduction sets, this functionality is handled by interfacing with [Lokinet](https://github.com/loki-project/loki-network). With the appropriate introduction sets Alice and Bob can create a path and using onion routing pass messages through the Loki network without giving away personally identifiable information like their IP address. +Session integrates directly with Loki [Service Nodes](https://lokidocs.com/ServiceNodes/SNOverview/), which are a set of distributed, decentralized and Sybil resistant nodes. Service Nodes act as servers which store messages offline, and a set of nodes which allow for onion routing functionality obfuscating users IP Addresses. For a full understanding of how Session works, read the [Loki whitepaper](https://loki.network/whitepaper). **Offline messages** diff --git a/_locales/en/messages.json b/_locales/en/messages.json index 896892a4e..34a519aa4 100644 --- a/_locales/en/messages.json +++ b/_locales/en/messages.json @@ -2162,6 +2162,11 @@ "message": "Leave Closed Group", "description": "Button action that the user can click to leave the group" }, + "leaveClosedGroupConfirmation": { + "message": "Leave this Closed Group?", + "description": + "Confirmation dialog text that tells the user what will happen if they leave the closed group." + }, "leaveGroupDialogTitle": { "message": "Are you sure you want to leave this group?", "description": diff --git a/app/sql.js b/app/sql.js index e9cb2390c..29f19027d 100644 --- a/app/sql.js +++ b/app/sql.js @@ -901,7 +901,7 @@ async function updateToLokiSchemaVersion1(currentVersion, instance) { id: 'rss://loki.network/feed/', rssFeed: 'https://loki.network/feed/', closable: true, - name: 'Loki.network News', + name: 'Loki News', profileAvatar: 'images/session/session_chat_icon.png', }; @@ -910,7 +910,7 @@ async function updateToLokiSchemaVersion1(currentVersion, instance) { id: 'rss://loki.network/category/messenger-updates/feed/', rssFeed: 'https://loki.network/category/messenger-updates/feed/', closable: false, - name: 'Messenger updates', + name: 'Session Updates', profileAvatar: 'images/session/session_chat_icon.png', }; diff --git a/build/notarize.js b/build/notarize.js index ac5a2c138..1c797b4e7 100644 --- a/build/notarize.js +++ b/build/notarize.js @@ -10,19 +10,36 @@ const { notarize } = require('electron-notarize'); Notarizing: https://kilianvalkhof.com/2019/electron/notarizing-your-electron-application/ */ +const log = msg => console.log(`\n${msg}`); +const isEmpty = v => !v || v.length === 0; + exports.default = async function notarizing(context) { const { electronPlatformName, appOutDir } = context; if (electronPlatformName !== 'darwin') { return; } + log('Notarizing mac application'); const appName = context.packager.appInfo.productFilename; + const { + SIGNING_APPLE_ID, + SIGNING_APP_PASSWORD, + SIGNING_TEAM_ID, + } = process.env; + + if (isEmpty(SIGNING_APPLE_ID) || isEmpty(SIGNING_APP_PASSWORD)) { + log( + 'SIGNING_APPLE_ID or SIGNING_APP_PASSWORD not set.\nTerminating noratization.' + ); + return; + } - return notarize({ - appBundleId: 'com.loki-project.messenger-desktop', + const options = { + appBundleId: 'org.getsession.desktop', appPath: `${appOutDir}/${appName}.app`, - appleId: process.env.SIGNING_APPLE_ID, - appleIdPassword: process.env.SIGNING_APP_PASSWORD, - ascProvider: process.env.SIGNING_TEAM_ID, - }); + appleId: SIGNING_APPLE_ID, + appleIdPassword: SIGNING_APP_PASSWORD, + }; + if (!isEmpty(SIGNING_TEAM_ID)) options.ascProvider = SIGNING_TEAM_ID; + return notarize(options); }; diff --git a/js/background.js b/js/background.js index 85f040d99..123dca978 100644 --- a/js/background.js +++ b/js/background.js @@ -763,9 +763,10 @@ const ev = new Event('group'); - const ourKey = textsecure.storage.user.getNumber(); - - const allMembers = [ourKey, ...members]; + const primaryDeviceKey = + window.storage.get('primaryDevicePubKey') || + textsecure.storage.user.getNumber(); + const allMembers = [primaryDeviceKey, ...members]; ev.groupDetails = { id: groupId, @@ -794,7 +795,7 @@ window.friends.friendRequestStatusEnum.friends ); - convo.updateGroupAdmins([ourKey]); + convo.updateGroupAdmins([primaryDeviceKey]); appView.openConversation(groupId, {}); }; @@ -1059,6 +1060,7 @@ window.setMediaPermissions(!mediaPermissions); }; + // attempts a connection to an open group server window.attemptConnection = async (serverURL, channelId) => { let rawserverURL = serverURL .replace(/^https?:\/\//i, '') @@ -2187,9 +2189,7 @@ }, }); } else { - window.log.verbose( - `Already seen session restore for pubkey: ${pubkey}` - ); + window.log.debug(`Already seen session restore for pubkey: ${pubkey}`); if (ev.confirm) { ev.confirm(); } diff --git a/js/conversation_controller.js b/js/conversation_controller.js index 32b5e8e10..5e102420d 100644 --- a/js/conversation_controller.js +++ b/js/conversation_controller.js @@ -161,26 +161,32 @@ if (!conversation) { return; } - if (conversation.isPublic()) { + + // Close group leaving + if (conversation.isClosedGroup()) { + await conversation.leaveGroup(); + } else if (conversation.isPublic()) { const channelAPI = await conversation.getPublicSendData(); if (channelAPI === null) { log.warn(`Could not get API for public conversation ${id}`); } else { channelAPI.serverAPI.partChannel(channelAPI.channelId); } + } else if (conversation.isPrivate()) { + const deviceIds = await textsecure.storage.protocol.getDeviceIds(id); + await Promise.all( + deviceIds.map(deviceId => { + const address = new libsignal.SignalProtocolAddress(id, deviceId); + const sessionCipher = new libsignal.SessionCipher( + textsecure.storage.protocol, + address + ); + return sessionCipher.deleteAllSessionsForDevice(); + }) + ); } + await conversation.destroyMessages(); - const deviceIds = await textsecure.storage.protocol.getDeviceIds(id); - await Promise.all( - deviceIds.map(deviceId => { - const address = new libsignal.SignalProtocolAddress(id, deviceId); - const sessionCipher = new libsignal.SessionCipher( - textsecure.storage.protocol, - address - ); - return sessionCipher.deleteAllSessionsForDevice(); - }) - ); await window.Signal.Data.removeConversation(id, { Conversation: Whisper.Conversation, }); diff --git a/js/expire.js b/js/expire.js index 32f9555fc..c068921e5 100644 --- a/js/expire.js +++ b/js/expire.js @@ -10,6 +10,8 @@ '', // no pubkey needed window.getDefaultFileServer() ); + // use the anonymous access token + window.tokenlessFileServerAdnAPI.token = 'loki'; window.tokenlessFileServerAdnAPI.pubKey = window.Signal.Crypto.base64ToArrayBuffer( LokiFileServerAPI.secureRpcPubKey ); diff --git a/js/models/conversations.js b/js/models/conversations.js index 29c9ab317..7471c5798 100644 --- a/js/models/conversations.js +++ b/js/models/conversations.js @@ -198,13 +198,33 @@ isOnline() { return this.isMe() || this.get('isOnline'); }, - isMe() { + return this.isOurLocalDevice() || this.isOurPrimaryDevice(); + }, + isOurPrimaryDevice() { return this.id === window.storage.get('primaryDevicePubKey'); }, + async isOurDevice() { + if (this.isMe()) { + return true; + } + + const ourDevices = await window.libloki.storage.getPairedDevicesFor( + this.ourNumber + ); + return ourDevices.includes(this.id); + }, + isOurLocalDevice() { + return this.id === this.ourNumber; + }, isPublic() { return !!(this.id && this.id.match(/^publicChat:/)); }, + isClosedGroup() { + return ( + this.get('type') === Message.GROUP && !this.isPublic() && !this.isRss() + ); + }, isClosable() { return !this.isRss() || this.get('closable'); }, @@ -2712,13 +2732,16 @@ }, deleteContact() { - const title = this.isPublic() - ? i18n('deletePublicChannel') - : i18n('deleteContact'); + let title = i18n('deleteContact'); + let message = i18n('deleteContactConfirmation'); - const message = this.isPublic() - ? i18n('deletePublicChannelConfirmation') - : i18n('deleteContactConfirmation'); + if (this.isPublic()) { + title = i18n('deletePublicChannel'); + message = i18n('deletePublicChannelConfirmation'); + } else if (this.isClosedGroup()) { + title = i18n('leaveClosedGroup'); + message = i18n('leaveClosedGroupConfirmation'); + } window.confirmationDialog({ title, diff --git a/js/models/messages.js b/js/models/messages.js index 0fb0a6e61..c79ca9c11 100644 --- a/js/models/messages.js +++ b/js/models/messages.js @@ -205,7 +205,7 @@ }, getLokiNameForNumber(number) { const conversation = ConversationController.get(number); - if (!conversation) { + if (!conversation || !conversation.getLokiProfile()) { return number; } return conversation.getLokiProfile().displayName; @@ -1898,6 +1898,8 @@ const authorisation = await libloki.storage.getGrantAuthorisationForSecondaryPubKey( source ); + const primarySource = + (authorisation && authorisation.primaryDevicePubKey) || source; const isGroupMessage = !!initialMessage.group; if (isGroupMessage) { conversationId = initialMessage.group.id; @@ -1916,10 +1918,12 @@ const knownMembers = conversation.get('members'); if (!newGroup && knownMembers) { - const fromMember = knownMembers.includes(source); + const fromMember = knownMembers.includes(primarySource); if (!fromMember) { - window.log.warn(`Ignoring group message from non-member: ${source}`); + window.log.warn( + `Ignoring group message from non-member: ${primarySource}` + ); confirm(); return null; } @@ -1938,7 +1942,9 @@ ); } - const fromAdmin = conversation.get('groupAdmins').includes(source); + const fromAdmin = conversation + .get('groupAdmins') + .includes(primarySource); if (!fromAdmin) { // Make sure the message is not removing members / renaming the group @@ -2016,11 +2022,11 @@ .getConversations() .models.filter(c => c.get('members')) .reduce((acc, x) => window.Lodash.concat(acc, x.get('members')), []) - .includes(source); + .includes(primarySource); if (groupMember) { window.log.info( - `Auto accepting a 'group' friend request for a known group member: ${groupMember}` + `Auto accepting a 'group' friend request for a known group member: ${primarySource}` ); window.libloki.api.sendBackgroundMessage(message.get('source')); @@ -2355,6 +2361,12 @@ await sendingDeviceConversation.onFriendRequestAccepted(); } } + + // We need to map the original message source to the primary device + if (source !== ourNumber) { + message.set({ source: primarySource }); + } + const id = await window.Signal.Data.saveMessage(message.attributes, { Message: Whisper.Message, }); diff --git a/js/modules/loki_app_dot_net_api.js b/js/modules/loki_app_dot_net_api.js index 38b229c73..293d6a522 100644 --- a/js/modules/loki_app_dot_net_api.js +++ b/js/modules/loki_app_dot_net_api.js @@ -34,8 +34,17 @@ class LokiAppDotNetServerAPI { log.info(`LokiAppDotNetAPI registered server ${url}`); } + async open() { + // check token, we're not sure how long we were asleep, token may have expired + await this.getOrRefreshServerToken(); + // now that we have a working token, start up pollers + this.channels.forEach(channel => channel.open()); + } + async close() { this.channels.forEach(channel => channel.stop()); + // match sure our pending requests are finished + // in case it's still starting up if (this.tokenPromise) { await this.tokenPromise; } @@ -70,6 +79,7 @@ class LokiAppDotNetServerAPI { } async partChannel(channelId) { + log.info('partChannel', channelId, 'from', this.baseServerUrl); await this.serverRequest(`channels/${channelId}/subscribe`, { method: 'DELETE', }); @@ -78,6 +88,7 @@ class LokiAppDotNetServerAPI { // deallocate resources channel uses unregisterChannel(channelId) { + log.info('unregisterChannel', channelId, 'from', this.baseServerUrl); let thisChannel; let i = 0; for (; i < this.channels.length; i += 1) { @@ -110,16 +121,23 @@ class LokiAppDotNetServerAPI { ); */ + // You cannot use null to clear the profile name + // the name key has to be set to know what value we want changed + const pName = profileName || ''; + const res = await this.serverRequest('users/me', { method: 'PATCH', objBody: { - name: profileName, + name: pName, }, }); // no big deal if it fails... if (res.err || !res.response || !res.response.data) { if (res.err) { - log.error(`setProfileName Error ${res.err}`); + log.error( + `setProfileName Error ${res.err} ${res.statusCode}`, + this.baseServerUrl + ); } return []; } @@ -187,8 +205,9 @@ class LokiAppDotNetServerAPI { // if no token to verify, just bail now if (!token) { - // + // if we haven't forced it if (!forceRefresh) { + // try one more time with requesting a fresh token token = await this.getOrRefreshServerToken(true); } return token; @@ -204,10 +223,14 @@ class LokiAppDotNetServerAPI { tokenRes.response.data.user ) { // get our profile name - // FIXME: should this be window.storage.get('primaryDevicePubKey')? - const ourNumber = textsecure.storage.user.getNumber(); + // this should be primaryDevicePubKey + // because the rest of the profile system uses that... + const ourNumber = + window.storage.get('primaryDevicePubKey') || + textsecure.storage.user.getNumber(); const profileConvo = ConversationController.get(ourNumber); - const profileName = profileConvo.getProfileName(); + const profile = profileConvo.getLokiProfile(); + const profileName = profile && profile.displayName; // if doesn't match, write it to the network if (tokenRes.response.data.user.name !== profileName) { // update our profile name if it got out of sync @@ -317,7 +340,7 @@ class LokiAppDotNetServerAPI { // activate token async submitToken(token) { - const options = { + const fetchOptions = { method: 'POST', headers: { 'Content-Type': 'application/json', @@ -331,7 +354,8 @@ class LokiAppDotNetServerAPI { try { const res = await this.proxyFetch( `${this.baseServerUrl}/loki/v1/submit_challenge`, - options + fetchOptions, + { textResponse: true } ); return res.ok; } catch (e) { @@ -340,7 +364,7 @@ class LokiAppDotNetServerAPI { } } - async proxyFetch(urlObj, fetchOptions = { method: 'GET' }) { + async proxyFetch(urlObj, fetchOptions = { method: 'GET' }, options = {}) { if ( window.lokiFeatureFlags.useSnodeProxy && (this.baseServerUrl === 'https://file-dev.lokinet.org' || @@ -356,7 +380,8 @@ class LokiAppDotNetServerAPI { const endpoint = urlStr.replace(`${this.baseServerUrl}/`, ''); const { response, result } = await this._sendToProxy( endpoint, - finalOptions + finalOptions, + options ); // emulate nodeFetch response... return { @@ -364,15 +389,20 @@ class LokiAppDotNetServerAPI { json: () => response, }; } - return nodeFetch(urlObj, fetchOptions); + return nodeFetch(urlObj, fetchOptions, options); } - async _sendToProxy(endpoint, fetchOptions) { + async _sendToProxy(endpoint, pFetchOptions, options = {}) { const randSnode = await lokiSnodeAPI.getRandomSnodeAddress(); const url = `https://${randSnode.ip}:${randSnode.port}/file_proxy`; + const fetchOptions = pFetchOptions; // make lint happy + // safety issue with file server, just safer to have this + if (fetchOptions.headers === undefined) { + fetchOptions.headers = {}; + } + const payloadObj = { - // I think this is a stream, we may need to collect it all? body: fetchOptions.body, // might need to b64 if binary... endpoint, method: fetchOptions.method, @@ -444,9 +474,13 @@ class LokiAppDotNetServerAPI { const result = await nodeFetch(url, firstHopOptions); const txtResponse = await result.text(); - if (txtResponse === 'Service node is not ready: not in any swarm; \n') { + if (txtResponse.match(/^Service node is not ready: not in any swarm/i)) { // mark snode bad - log.warn('Marking random snode bad', randSnode); + log.warn( + `Marking random snode bad, internet address ${randSnode.ip}:${ + randSnode.port + }` + ); lokiSnodeAPI.markRandomNodeUnreachable(randSnode); // retry (hopefully with new snode) // FIXME: max number of retries... @@ -457,7 +491,10 @@ class LokiAppDotNetServerAPI { try { response = JSON.parse(txtResponse); } catch (e) { - log.warn(`_sendToProxy Could not parse outer JSON [${txtResponse}]`); + log.warn( + `_sendToProxy Could not parse outer JSON [${txtResponse}]`, + endpoint + ); } if (response.meta && response.meta.code === 200) { @@ -471,18 +508,22 @@ class LokiAppDotNetServerAPI { ivAndCiphertextResponse ); const textDecoder = new TextDecoder(); - const json = textDecoder.decode(decrypted); + const respStr = textDecoder.decode(decrypted); // replace response try { - response = JSON.parse(json); + response = options.textResponse ? respStr : JSON.parse(respStr); } catch (e) { - log.warn(`_sendToProxy Could not parse inner JSON [${json}]`); + log.warn( + `_sendToProxy Could not parse inner JSON [${respStr}]`, + endpoint + ); } } else { log.warn( 'file server secure_rpc gave an non-200 response: ', response, - ` txtResponse[${txtResponse}]` + ` txtResponse[${txtResponse}]`, + endpoint ); } return { result, txtResponse, response }; @@ -552,7 +593,8 @@ class LokiAppDotNetServerAPI { .replace(`${this.baseServerUrl}/`, ''); ({ response, txtResponse, result } = await this._sendToProxy( endpointWithQS, - fetchOptions + fetchOptions, + options )); } else { // disable check for .loki @@ -563,7 +605,8 @@ class LokiAppDotNetServerAPI { // always make sure this check is enabled process.env.NODE_TLS_REJECT_UNAUTHORIZED = 1; txtResponse = await result.text(); - response = JSON.parse(txtResponse); + // hrm cloudflare timeouts (504s) will be html... + response = options.textResponse ? txtResponse : JSON.parse(txtResponse); } } catch (e) { if (txtResponse) { @@ -571,10 +614,16 @@ class LokiAppDotNetServerAPI { `serverRequest ${mode} error`, e.code, e.message, - `json: ${txtResponse}` + `json: ${txtResponse}`, 'attempting connection to', url ); } else { - log.info(`serverRequest ${mode} error`, e.code, e.message); + log.info( + `serverRequest ${mode} error`, + e.code, + e.message, + 'attempting connection to', + url + ); } return { err: e, @@ -879,7 +928,6 @@ class LokiPublicChannelAPI { this.modStatus = false; this.deleteLastId = 1; this.timers = {}; - this.running = true; this.myPrivateKey = false; // can escalated to SQL if it start uses too much memory this.logMop = {}; @@ -895,12 +943,7 @@ class LokiPublicChannelAPI { }` ); // start polling - this.pollForMessages(); - this.pollForDeletions(); - this.pollForChannel(); - this.pollForModerators(); - - // TODO: poll for group members here? + this.open(); } async getPrivateKey() { @@ -929,19 +972,64 @@ class LokiPublicChannelAPI { return true; } + open() { + log.info( + `LokiPublicChannel open ${this.channelId} on ${ + this.serverAPI.baseServerUrl + }` + ); + if (this.running) { + log.warn( + `LokiPublicChannel already open ${this.channelId} on ${ + this.serverAPI.baseServerUrl + }` + ); + } + this.running = true; + if (!this.timers.channel) { + this.pollForChannel(); + } + if (!this.timers.moderator) { + this.pollForModerators(); + } + if (!this.timers.delete) { + this.pollForDeletions(); + } + if (!this.timers.message) { + this.pollForMessages(); + } + // TODO: poll for group members here? + } + stop() { + log.info( + `LokiPublicChannel close ${this.channelId} on ${ + this.serverAPI.baseServerUrl + }` + ); + if (!this.running) { + log.warn( + `LokiPublicChannel already open ${this.channelId} on ${ + this.serverAPI.baseServerUrl + }` + ); + } this.running = false; if (this.timers.channel) { clearTimeout(this.timers.channel); + this.timers.channel = false; } if (this.timers.moderator) { clearTimeout(this.timers.moderator); + this.timers.moderator = false; } if (this.timers.delete) { clearTimeout(this.timers.delete); + this.timers.delete = false; } if (this.timers.message) { clearTimeout(this.timers.message); + this.timers.message = false; } } @@ -981,15 +1069,17 @@ class LokiPublicChannelAPI { const res = await this.serverRequest( `loki/v1/channels/${this.channelId}/moderators` ); - // FIXME: should this be window.storage.get('primaryDevicePubKey')? - const ourNumber = textsecure.storage.user.getNumber(); + const ourNumberDevice = textsecure.storage.user.getNumber(); + const ourNumberProfile = window.storage.get('primaryDevicePubKey'); // Get the list of moderators if no errors occurred const moderators = !res.err && res.response && res.response.moderators; // if we encountered problems then we'll keep the old mod status if (moderators) { - this.modStatus = moderators.includes(ourNumber); + this.modStatus = + (ourNumberProfile && moderators.includes(ourNumberProfile)) || + moderators.includes(ourNumberDevice); } await this.conversation.setModerators(moderators || []); @@ -1392,8 +1482,10 @@ class LokiPublicChannelAPI { let pendingMessages = []; // get our profile name - // FIXME: should this be window.storage.get('primaryDevicePubKey')? - const ourNumber = textsecure.storage.user.getNumber(); + const ourNumberDevice = textsecure.storage.user.getNumber(); + // if no primaryDevicePubKey fall back to ourNumberDevice + const ourNumberProfile = + window.storage.get('primaryDevicePubKey') || ourNumberDevice; let lastProfileName = false; // the signature forces this to be async @@ -1466,7 +1558,7 @@ class LokiPublicChannelAPI { const from = adnMessage.user.name || 'Anonymous'; // profileName // if us - if (pubKey === ourNumber) { + if (pubKey === ourNumberProfile || pubKey === ourNumberDevice) { // update the last name we saw from ourself lastProfileName = from; } @@ -1616,7 +1708,7 @@ class LokiPublicChannelAPI { const slaveKey = messageData.source; // prevent our own device sent messages from coming back in - if (slaveKey === ourNumber) { + if (slaveKey === ourNumberDevice) { // we originally sent these return; } @@ -1647,7 +1739,7 @@ class LokiPublicChannelAPI { // if we received one of our own messages if (lastProfileName !== false) { // get current profileName - const profileConvo = ConversationController.get(ourNumber); + const profileConvo = ConversationController.get(ourNumberProfile); const profileName = profileConvo.getProfileName(); // check to see if it out of sync if (profileName !== lastProfileName) { diff --git a/js/modules/loki_file_server_api.js b/js/modules/loki_file_server_api.js index c38609467..1bd5fbc51 100644 --- a/js/modules/loki_file_server_api.js +++ b/js/modules/loki_file_server_api.js @@ -245,6 +245,9 @@ class LokiHomeServerInstance extends LokiFileServerInstance { isPrimary: isPrimary ? '1' : '0', authorisations, }; + if (!this._server.token) { + log.warn('_setOurDeviceMapping no token yet'); + } return this._server.setSelfAnnotation( DEVICE_MAPPING_USER_ANNOTATION_TYPE, content @@ -269,11 +272,14 @@ class LokiHomeServerInstance extends LokiFileServerInstance { // you only upload to your own home server // you can download from any server... uploadAvatar(data) { + if (!this._server.token) { + log.warn('uploadAvatar no token yet'); + } return this._server.uploadAvatar(data); } - uploadPrivateAttachment(data) { - return this._server.uploadData(data); + static uploadPrivateAttachment(data) { + return window.tokenlessFileServerAdnAPI.uploadData(data); } clearOurDeviceMappingAnnotations() { @@ -298,6 +304,7 @@ class LokiFileServerFactoryAPI { if (!thisServer) { thisServer = new LokiHomeServerInstance(this.ourKey); log.info(`Registering HomeServer ${serverUrl}`); + // not await, so a failure or slow connection doesn't hinder loading of the app thisServer.establishConnection(serverUrl); this.servers.push(thisServer); } diff --git a/js/modules/loki_public_chat_api.js b/js/modules/loki_public_chat_api.js index 8e4cd0f3d..0dff04c9b 100644 --- a/js/modules/loki_public_chat_api.js +++ b/js/modules/loki_public_chat_api.js @@ -13,6 +13,13 @@ class LokiPublicChatFactoryAPI extends EventEmitter { this.primaryUserProfileName = {}; } + // MessageReceiver.connect calls this + // start polling in all existing registered channels + async open() { + await Promise.all(this.servers.map(server => server.open())); + } + + // MessageReceiver.close async close() { await Promise.all(this.servers.map(server => server.close())); } diff --git a/js/modules/loki_rpc.js b/js/modules/loki_rpc.js index 1cd99b9c7..2579d27f0 100644 --- a/js/modules/loki_rpc.js +++ b/js/modules/loki_rpc.js @@ -85,7 +85,20 @@ const sendToProxy = async (options = {}, targetNode) => { try { const jsonRes = JSON.parse(plaintext); // emulate nodeFetch response... - jsonRes.json = () => JSON.parse(jsonRes.body); + jsonRes.json = () => { + try { + return JSON.parse(jsonRes.body); + } catch (e) { + log.error( + 'lokiRpc sendToProxy error', + e.code, + e.message, + 'json', + jsonRes.body + ); + } + return false; + }; return jsonRes; } catch (e) { log.error( diff --git a/js/modules/loki_snode_api.js b/js/modules/loki_snode_api.js index eb6bdc2f5..3418c827f 100644 --- a/js/modules/loki_snode_api.js +++ b/js/modules/loki_snode_api.js @@ -156,6 +156,15 @@ class LokiSnodeAPI { '/storage_rpc/v1', snode ); + if (!result) { + log.warn( + `getSwarmNodes lokiRpc on ${snode.ip}:${ + snode.port + } returned falsish value`, + result + ); + return []; + } const snodes = result.snodes.filter(tSnode => tSnode.ip !== '0.0.0.0'); return snodes; } catch (e) { diff --git a/js/modules/web_api.js b/js/modules/web_api.js index 7d446bace..c92681b05 100644 --- a/js/modules/web_api.js +++ b/js/modules/web_api.js @@ -865,7 +865,7 @@ function initialize({ filename: 'attachment', }); - return lokiFileServerAPI.uploadPrivateAttachment(formData); + return lokiFileServerAPI.constructor.uploadPrivateAttachment(formData); } function putAvatar(bin) { diff --git a/js/views/app_view.js b/js/views/app_view.js index 9b62d9a06..7fbf40b8a 100644 --- a/js/views/app_view.js +++ b/js/views/app_view.js @@ -238,13 +238,16 @@ this.el.append(dialog.el); }, showLeaveGroupDialog(groupConvo) { - const title = groupConvo.isPublic() - ? i18n('deletePublicChannel') - : i18n('deleteContact'); + let title = i18n('deleteContact'); + let message = i18n('deleteContactConfirmation'); - const message = groupConvo.isPublic() - ? i18n('deletePublicChannelConfirmation') - : i18n('deleteContactConfirmation'); + if (groupConvo.isPublic()) { + title = i18n('deletePublicChannel'); + message = i18n('deletePublicChannelConfirmation'); + } else if (groupConvo.isClosedGroup()) { + title = i18n('leaveClosedGroup'); + message = i18n('leaveClosedGroupConfirmation'); + } window.confirmationDialog({ title, diff --git a/libloki/crypto.js b/libloki/crypto.js index f9888d70e..dfaaf78e4 100644 --- a/libloki/crypto.js +++ b/libloki/crypto.js @@ -48,6 +48,9 @@ // Should we use ephemeral key pairs here rather than long term keys on each side? async encrypt(plaintext) { const myKeyPair = await textsecure.storage.protocol.getIdentityKeyPair(); + if (!myKeyPair) { + throw new Error('Failed to get keypair for encryption'); + } const myPrivateKey = myKeyPair.privKey; const symmetricKey = libsignal.Curve.calculateAgreement( this.pubKey, @@ -63,6 +66,9 @@ async decrypt(ivAndCiphertext) { const myKeyPair = await textsecure.storage.protocol.getIdentityKeyPair(); + if (!myKeyPair) { + throw new Error('Failed to get keypair for decryption'); + } const myPrivateKey = myKeyPair.privKey; const symmetricKey = libsignal.Curve.calculateAgreement( this.pubKey, @@ -169,6 +175,9 @@ data[len] = type; const myKeyPair = await textsecure.storage.protocol.getIdentityKeyPair(); + if (!myKeyPair) { + throw new Error('Failed to get keypair for pairing signature generation'); + } const signature = await libsignal.Curve.async.calculateSignature( myKeyPair.privKey, data.buffer @@ -291,7 +300,11 @@ const serverPubKey = new Uint8Array( dcodeIO.ByteBuffer.fromBase64(serverPubKey64).toArrayBuffer() ); - const { privKey } = await textsecure.storage.protocol.getIdentityKeyPair(); + const keyPair = await textsecure.storage.protocol.getIdentityKeyPair(); + if (!keyPair) { + throw new Error('Failed to get keypair for token decryption'); + } + const { privKey } = keyPair; const symmetricKey = libsignal.Curve.calculateAgreement( serverPubKey, privKey diff --git a/libloki/storage.js b/libloki/storage.js index dd3889fba..3a2d084e2 100644 --- a/libloki/storage.js +++ b/libloki/storage.js @@ -240,6 +240,10 @@ return secondaryPubKeys.concat(primaryDevicePubKey); } + function getPairedDevicesFor(pubkey) { + return window.Signal.Data.getPairedDevicesFor(pubkey); + } + window.libloki.storage = { getPreKeyBundleForContact, saveContactPreKeyBundle, @@ -250,6 +254,7 @@ removePairingAuthorisationForSecondaryPubKey, getGrantAuthorisationForSecondaryPubKey, getAuthorisationForSecondaryPubKey, + getPairedDevicesFor, getAllDevicePubKeysForPrimaryPubKey, getSecondaryDevicesFor, getPrimaryDeviceMapping, diff --git a/libtextsecure/message_receiver.js b/libtextsecure/message_receiver.js index 6146f86dc..0af1a190b 100644 --- a/libtextsecure/message_receiver.js +++ b/libtextsecure/message_receiver.js @@ -48,6 +48,17 @@ function MessageReceiver(username, password, signalingKey, options = {}) { if (options.retryCached) { this.pending = this.queueAllCached(); } + + // only do this once to prevent duplicates + if (lokiPublicChatAPI) { + // bind events + lokiPublicChatAPI.on( + 'publicMessage', + this.handleUnencryptedMessage.bind(this) + ); + } else { + window.log.error('Can not handle open group data, API is not available'); + } } MessageReceiver.stringToArrayBuffer = string => @@ -79,11 +90,11 @@ MessageReceiver.prototype.extend({ handleRequest: this.handleRequest.bind(this), }); this.httpPollingResource.pollServer(); + + // start polling all open group rooms you have registered + // if not registered yet, they'll get started when they're created if (lokiPublicChatAPI) { - lokiPublicChatAPI.on( - 'publicMessage', - this.handleUnencryptedMessage.bind(this) - ); + lokiPublicChatAPI.open(); } // set up pollers for any RSS feeds feeds.forEach(feed => { @@ -166,6 +177,7 @@ MessageReceiver.prototype.extend({ this.wsr.close(3000, 'called close'); } + // stop polling all open group rooms if (lokiPublicChatAPI) { await lokiPublicChatAPI.close(); } @@ -1314,8 +1326,9 @@ MessageReceiver.prototype.extend({ primaryPubKey ); + // If we don't have a mapping on the primary then we have been unlinked if (!primaryMapping) { - return false; + return true; } // We expect the primary device to have updated its mapping @@ -1366,7 +1379,11 @@ MessageReceiver.prototype.extend({ } } - if (friendRequest) { + // If we got a friend request message or + // if we're not friends with the current user that sent this private message + // Check to see if we need to auto accept their friend request + const isGroupMessage = !!groupId; + if (friendRequest || (!isGroupMessage && !conversation.isFriend())) { if (isMe) { window.log.info('refusing to add a friend request to ourselves'); throw new Error('Cannot add a friend request for ourselves!'); diff --git a/libtextsecure/outgoing_message.js b/libtextsecure/outgoing_message.js index 5be3b7368..00e4be293 100644 --- a/libtextsecure/outgoing_message.js +++ b/libtextsecure/outgoing_message.js @@ -350,23 +350,33 @@ OutgoingMessage.prototype = { } catch (e) { // do nothing } - if ( - conversation && - !conversation.isFriend() && - !conversation.hasReceivedFriendRequest() && - !this.isGroup - ) { - // We want to send an automated friend request if: - // - We aren't already friends - // - We haven't received a friend request from this device - // - We haven't sent a friend request recently - if (conversation.friendRequestTimerIsExpired()) { - isMultiDeviceRequest = true; - thisDeviceMessageType = 'friend-request'; - } else { - // Throttle automated friend requests - this.successfulNumbers.push(devicePubKey); - return null; + if (conversation && !this.isGroup) { + const isOurDevice = await conversation.isOurDevice(); + const isFriends = + conversation.isFriend() || + conversation.hasReceivedFriendRequest(); + // We should only send a friend request to our device if we don't have keys + const shouldSendAutomatedFR = isOurDevice ? !keysFound : !isFriends; + if (shouldSendAutomatedFR) { + // We want to send an automated friend request if: + // - We aren't already friends + // - We haven't received a friend request from this device + // - We haven't sent a friend request recently + if (conversation.friendRequestTimerIsExpired()) { + isMultiDeviceRequest = true; + thisDeviceMessageType = 'friend-request'; + } else { + // Throttle automated friend requests + this.successfulNumbers.push(devicePubKey); + return null; + } + } + + // If we're not friends with our own device then we should become friends + if (isOurDevice && keysFound && !isFriends) { + conversation.setFriendRequestStatus( + window.friends.friendRequestStatusEnum.friends + ); } } } diff --git a/libtextsecure/sendmessage.js b/libtextsecure/sendmessage.js index cb1fea2ad..2f0dbc001 100644 --- a/libtextsecure/sendmessage.js +++ b/libtextsecure/sendmessage.js @@ -411,7 +411,11 @@ MessageSender.prototype = { const ourNumber = textsecure.storage.user.getNumber(); - numbers.forEach(number => { + // Note: Since we're just doing independant tasks, + // using `async` in the `forEach` loop should be fine. + // If however we want to use the results from forEach then + // we would need to convert this to a Promise.all(numbers.map(...)) + numbers.forEach(async number => { // Note: if we are sending a private group message, we do our best to // ensure we have signal protocol sessions with every member, but if we // fail, let's at least send messages to those members with which we do: @@ -420,9 +424,17 @@ MessageSender.prototype = { s => s.number === number ); + let keysFound = false; + // If we don't have a session but we already have prekeys to + // start communication then we should use them + if (!haveSession && !options.isPublic) { + keysFound = await outgoing.getKeysForNumber(number, []); + } + if ( number === ourNumber || haveSession || + keysFound || options.isPublic || options.messageType === 'friend-request' ) { @@ -640,6 +652,10 @@ MessageSender.prototype = { }, async sendContactSyncMessage(contactConversation) { + if (!contactConversation.isPrivate()) { + return Promise.resolve(); + } + const primaryDeviceKey = window.storage.get('primaryDevicePubKey'); const allOurDevices = (await libloki.storage.getAllDevicePubKeysForPrimaryPubKey( primaryDeviceKey @@ -869,8 +885,13 @@ MessageSender.prototype = { }, sendGroupProto(providedNumbers, proto, timestamp = Date.now(), options = {}) { - const me = textsecure.storage.user.getNumber(); - const numbers = providedNumbers.filter(number => number !== me); + // We always assume that only primary device is a member in the group + const primaryDeviceKey = + window.storage.get('primaryDevicePubKey') || + textsecure.storage.user.getNumber(); + const numbers = providedNumbers.filter( + number => number !== primaryDeviceKey + ); if (numbers.length === 0) { return Promise.resolve({ successfulNumbers: [], @@ -881,7 +902,7 @@ MessageSender.prototype = { }); } - return new Promise((resolve, reject) => { + const sendPromise = new Promise((resolve, reject) => { const silent = true; const callback = res => { res.dataMessage = proto.toArrayBuffer(); @@ -901,6 +922,13 @@ MessageSender.prototype = { options ); }); + + return sendPromise.then(result => { + // Sync the group message to our other devices + const encoded = textsecure.protobuf.DataMessage.encode(proto); + this.sendSyncMessage(encoded, timestamp, null, null, [], [], options); + return result; + }); }, async getMessageProto( @@ -1085,8 +1113,11 @@ MessageSender.prototype = { profileKey, options ) { - const me = textsecure.storage.user.getNumber(); - let numbers = groupNumbers.filter(number => number !== me); + // We always assume that only primary device is a member in the group + const primaryDeviceKey = + window.storage.get('primaryDevicePubKey') || + textsecure.storage.user.getNumber(); + let numbers = groupNumbers.filter(number => number !== primaryDeviceKey); if (options.isPublic) { numbers = [groupId]; } @@ -1130,8 +1161,10 @@ MessageSender.prototype = { proto.group.name = name; proto.group.members = members; - const ourPK = textsecure.storage.user.getNumber(); - proto.group.admins = [ourPK]; + const primaryDeviceKey = + window.storage.get('primaryDevicePubKey') || + textsecure.storage.user.getNumber(); + proto.group.admins = [primaryDeviceKey]; return this.makeAttachmentPointer(avatar).then(attachment => { proto.group.avatar = attachment; diff --git a/package.json b/package.json index 34d44bb26..6f1aceb85 100644 --- a/package.json +++ b/package.json @@ -3,7 +3,7 @@ "productName": "Session", "description": "Private messaging from your desktop", "repository": "https://github.com/loki-project/loki-messenger.git", - "version": "1.0.0", + "version": "1.0.1", "license": "GPL-3.0", "author": { "name": "Loki Project", @@ -166,7 +166,7 @@ "chai": "4.1.2", "dashdash": "1.14.1", "electron": "4.1.2", - "electron-builder": "21.2.0", + "electron-builder": "22.3.2", "electron-icon-maker": "0.0.3", "electron-notarize": "^0.2.0", "eslint": "4.14.0", @@ -207,8 +207,8 @@ "build": { "appId": "com.loki-project.messenger-desktop", "afterSign": "build/notarize.js", + "artifactName": "${name}-${os}-${version}.${ext}", "mac": { - "artifactName": "${name}-mac-${version}.${ext}", "category": "public.app-category.social-networking", "icon": "build/icons/mac/icon.icns", "target": [ @@ -225,21 +225,16 @@ }, "win": { "asarUnpack": "node_modules/spellchecker/vendor/hunspell_dictionaries", - "artifactName": "${name}-win-${version}.${ext}", "publisherName": "Loki Project", "icon": "build/icons/win/icon.ico", - "publish": [ - { - "provider": "generic", - "url": "https://getsession.org/" - } - ], "target": [ "nsis" ] }, "nsis": { - "deleteAppDataOnUninstall": true + "deleteAppDataOnUninstall": true, + "oneClick": false, + "allowToChangeInstallationDirectory": true }, "linux": { "category": "Network", @@ -248,7 +243,8 @@ }, "asarUnpack": "node_modules/spellchecker/vendor/hunspell_dictionaries", "target": [ - "deb" + "deb", + "AppImage" ], "icon": "build/icons/png" }, @@ -318,7 +314,8 @@ "node_modules/socks/build/common/*.js", "node_modules/socks/build/client/*.js", "node_modules/smart-buffer/build/*.js", - "!node_modules/@journeyapps/sqlcipher/deps/*" + "!node_modules/@journeyapps/sqlcipher/deps/*", + "!build/*.js" ] } } diff --git a/stylesheets/_modules.scss b/stylesheets/_modules.scss index 097920c68..99234384c 100644 --- a/stylesheets/_modules.scss +++ b/stylesheets/_modules.scss @@ -1241,6 +1241,7 @@ margin: 10px auto; padding: 5px 20px; border-radius: 4px; + word-break: break-word; } .module-group-notification__contact { diff --git a/ts/components/EditProfileDialog.tsx b/ts/components/EditProfileDialog.tsx index 555986cdc..1a560dfc7 100644 --- a/ts/components/EditProfileDialog.tsx +++ b/ts/components/EditProfileDialog.tsx @@ -71,7 +71,12 @@ export class EditProfileDialog extends React.Component { const viewDefault = this.state.mode === 'default'; const viewEdit = this.state.mode === 'edit'; const viewQR = this.state.mode === 'qr'; - const sessionID = window.textsecure.storage.user.getNumber(); + + /* tslint:disable:no-backbone-get-set-outside-model */ + const sessionID = + window.textsecure.storage.get('primaryDevicePubKey') || + window.textsecure.storage.user.getNumber(); + /* tslint:enable:no-backbone-get-set-outside-model */ const backButton = viewEdit || viewQR diff --git a/ts/components/conversation/GroupNotification.tsx b/ts/components/conversation/GroupNotification.tsx index 9cec3d1bf..b1484bbb4 100644 --- a/ts/components/conversation/GroupNotification.tsx +++ b/ts/components/conversation/GroupNotification.tsx @@ -38,7 +38,7 @@ export class GroupNotification extends React.Component { key={`external-${contact.phoneNumber}`} className="module-group-notification__contact" > - {contact.profileName} + {contact.profileName || contact.phoneNumber} ); diff --git a/ts/components/session/LeftPaneChannelSection.tsx b/ts/components/session/LeftPaneChannelSection.tsx index c5131f448..e47d8c215 100644 --- a/ts/components/session/LeftPaneChannelSection.tsx +++ b/ts/components/session/LeftPaneChannelSection.tsx @@ -364,7 +364,8 @@ export class LeftPaneChannelSection extends React.Component { return false; } - const regexURL = /(http:\/\/www\.|https:\/\/www\.|http:\/\/|https:\/\/)?[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,5}(:[0-9]{1,5})?(\/.*)?/; + // longest TLD is now (20/02/06) 24 characters per https://jasontucker.blog/8945/what-is-the-longest-tld-you-can-get-for-a-domain-name + const regexURL = /(http:\/\/|https:\/\/)?[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,24}(:[0-9]{1,5})?(\/.*)?/; if (groupUrl.length <= 0) { window.pushToast({ @@ -387,7 +388,7 @@ export class LeftPaneChannelSection extends React.Component { } joinChannelStateManager(this, groupUrl, () => { - this.handleToggleOverlay(SessionGroupType.Open); + this.handleToggleOverlay(undefined); }); return true; diff --git a/ts/components/session/LeftPaneMessageSection.tsx b/ts/components/session/LeftPaneMessageSection.tsx index 418439a77..9841257ea 100644 --- a/ts/components/session/LeftPaneMessageSection.tsx +++ b/ts/components/session/LeftPaneMessageSection.tsx @@ -69,7 +69,8 @@ export class LeftPaneMessageSection extends React.Component { this.state = { showComposeView: false, pubKeyPasted: '', - shouldRenderMessageOnboarding: length === 0 && renderOnboardingSetting && false, + shouldRenderMessageOnboarding: + length === 0 && renderOnboardingSetting && false, connectSuccess: false, loading: false, }; diff --git a/ts/components/session/RegistrationTabs.tsx b/ts/components/session/RegistrationTabs.tsx index 75c46b50b..f70dd9f79 100644 --- a/ts/components/session/RegistrationTabs.tsx +++ b/ts/components/session/RegistrationTabs.tsx @@ -827,7 +827,10 @@ export class RegistrationTabs extends React.Component<{}, State> { const onError = async (error: any) => { window.log.error(error); - + // clear the ... to make sure the user realize we're not doing anything + this.setState({ + loading: false, + }); await this.resetRegistration(); }; @@ -839,6 +842,11 @@ export class RegistrationTabs extends React.Component<{}, State> { const validationError = c.validateNumber(); if (validationError) { onError('Invalid public key').ignore(); + window.pushToast({ + title: window.i18n('invalidNumberError'), + type: 'error', + id: 'invalidNumberError', + }); return; } diff --git a/ts/components/session/SessionClosableOverlay.tsx b/ts/components/session/SessionClosableOverlay.tsx index 53d7ef327..88024618c 100644 --- a/ts/components/session/SessionClosableOverlay.tsx +++ b/ts/components/session/SessionClosableOverlay.tsx @@ -52,14 +52,16 @@ export class SessionClosableOverlay extends React.Component { } public getContacts() { - const conversations = window.getConversations(); - - let conversationList = conversations; - if (conversationList !== undefined) { - conversationList = conversationList.filter((conv: any) => { - return !conv.isRss() && !conv.isPublic() && conv.attributes.lastMessage; - }); - } + const conversations = window.getConversations() || []; + + const conversationList = conversations.filter((conversation: any) => { + return ( + !conversation.isMe() && + conversation.isPrivate() && + !conversation.isSecondaryDevice() && + conversation.isFriend() + ); + }); return conversationList.map((d: any) => { const lokiProfile = d.getLokiProfile(); diff --git a/yarn.lock b/yarn.lock index b86fad54d..081674812 100644 --- a/yarn.lock +++ b/yarn.lock @@ -179,7 +179,7 @@ resolved "https://registry.yarnpkg.com/@types/config/-/config-0.0.34.tgz#123f91bdb5afdd702294b9de9ca04d9ea11137b0" integrity sha512-jWi9DXx77hnzN4kHCNEvP/kab+nchRLTg9yjXYxjTcMBkuc5iBb3QuwJ4sPrb+nzy1GQjrfyfMqZOdR4i7opRQ== -"@types/debug@^4.1.4": +"@types/debug@^4.1.5": version "4.1.5" resolved "https://registry.yarnpkg.com/@types/debug/-/debug-4.1.5.tgz#b14efa8852b7768d898906613c23f688713e02cd" integrity sha512-Q1y515GcOdTHgagaVFhHnIFQ38ygs/kmxdNpvpou+raI9UO3YZcHDngBSYKQklcKlvA7iuQlmIKbzvmxcOE9CQ== @@ -208,6 +208,13 @@ dependencies: "@types/node" "*" +"@types/fs-extra@^8.0.1": + version "8.0.1" + resolved "https://registry.yarnpkg.com/@types/fs-extra/-/fs-extra-8.0.1.tgz#a2378d6e7e8afea1564e44aafa2e207dadf77686" + integrity sha512-J00cVDALmi/hJOYsunyT52Hva5TnJeKP5yd1r+mH/ZU0mbYZflR0Z5kw5kITtKTRYMhm1JMClOFYdHnQszEvqw== + dependencies: + "@types/node" "*" + "@types/glob@*": version "7.1.1" resolved "https://registry.yarnpkg.com/@types/glob/-/glob-7.1.1.tgz#aa59a1c6e3fbc421e07ccd31a944c30eba521575" @@ -586,7 +593,7 @@ ansi-styles@^3.2.0, ansi-styles@^3.2.1: dependencies: color-convert "^1.9.0" -ansi-styles@^4.0.0: +ansi-styles@^4.0.0, ansi-styles@^4.1.0: version "4.2.1" resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.2.1.tgz#90ae75c424d008d2624c5bf29ead3177ebfcf359" integrity sha512-9VGjrMsG1vePxcSweQsN20KY/c4zN0h9fLjqAbwbPfahM3t+NL+M9HC8xeXG2I8pX5NoamTGNuomEUFI7fcUjA== @@ -607,38 +614,66 @@ anymatch@^2.0.0: micromatch "^3.1.4" normalize-path "^2.1.1" -app-builder-bin@3.4.3: - version "3.4.3" - resolved "https://registry.yarnpkg.com/app-builder-bin/-/app-builder-bin-3.4.3.tgz#58a74193eb882f029be6b7f0cd3f0c6805927a6b" - integrity sha512-qMhayIwi3juerQEVJMQ76trObEbfQT0nhUdxZz9a26/3NLT3pE6awmQ8S1cEnrGugaaM5gYqR8OElcDezfmEsg== +app-builder-bin@3.5.2: + version "3.5.2" + resolved "https://registry.yarnpkg.com/app-builder-bin/-/app-builder-bin-3.5.2.tgz#fba56e6e9ef76fcd37816738c5f9a0b3992d7183" + integrity sha512-tYeNHp8js5c5MT+mzGEMsZhDBXRVAkCgrvz80rzNLlCojG6YrNLRSzfCWZiKJCv5W2faHjxTkKQoRAMY3RWaow== -app-builder-lib@21.2.0, app-builder-lib@~21.2.0: - version "21.2.0" - resolved "https://registry.yarnpkg.com/app-builder-lib/-/app-builder-lib-21.2.0.tgz#fa1d1604601431e2c3476857e9b9b61d33ad26cc" - integrity sha512-aOX/nv77/Bti6NymJDg7p9T067xD8m1ipIEJR7B4Mm1GsJWpMm9PZdXtCRiMNRjHtQS5KIljT0g17781y6qn5A== +app-builder-lib@22.3.2: + version "22.3.2" + resolved "https://registry.yarnpkg.com/app-builder-lib/-/app-builder-lib-22.3.2.tgz#d43e0bdff91d484c0bd07d7248043dbb2665b8ac" + integrity sha512-QHaDelJUP3R+HUmnQJzHvjlCCJGG6t0bcFSZTOtUx+44B42VxzKpxtoG55xRFbyrU5l2++n7dQqEZAGk8iL6Qg== dependencies: "7zip-bin" "~5.0.3" "@develar/schema-utils" "~2.1.0" async-exit-hook "^2.0.1" bluebird-lst "^1.0.9" - builder-util "21.2.0" - builder-util-runtime "8.3.0" + builder-util "22.3.2" + builder-util-runtime "8.6.0" chromium-pickle-js "^0.2.0" debug "^4.1.1" - ejs "^2.6.2" - electron-publish "21.2.0" + ejs "^3.0.1" + electron-publish "22.3.2" fs-extra "^8.1.0" - hosted-git-info "^2.7.1" + hosted-git-info "^3.0.2" is-ci "^2.0.0" - isbinaryfile "^4.0.2" + isbinaryfile "^4.0.4" js-yaml "^3.13.1" lazy-val "^1.0.4" minimatch "^3.0.4" normalize-package-data "^2.5.0" - read-config-file "5.0.0" - sanitize-filename "^1.6.2" - semver "^6.3.0" - temp-file "^3.3.4" + read-config-file "5.0.1" + sanitize-filename "^1.6.3" + semver "^7.1.1" + temp-file "^3.3.6" + +app-builder-lib@~22.3.2: + version "22.3.3" + resolved "https://registry.yarnpkg.com/app-builder-lib/-/app-builder-lib-22.3.3.tgz#9a95a3c14f69fb6131834dd840fba561191c9998" + integrity sha512-zZJyuF3djIA5K6tbx8t3w40M0iVoBR6K2k4KMHOu96+ffmfvdlu+UrsvDqvP1N1cgwFoSSyvW/Hg9/SP12pnEQ== + dependencies: + "7zip-bin" "~5.0.3" + "@develar/schema-utils" "~2.1.0" + async-exit-hook "^2.0.1" + bluebird-lst "^1.0.9" + builder-util "22.3.3" + builder-util-runtime "8.6.0" + chromium-pickle-js "^0.2.0" + debug "^4.1.1" + ejs "^3.0.1" + electron-publish "22.3.3" + fs-extra "^8.1.0" + hosted-git-info "^3.0.2" + is-ci "^2.0.0" + isbinaryfile "^4.0.4" + js-yaml "^3.13.1" + lazy-val "^1.0.4" + minimatch "^3.0.4" + normalize-package-data "^2.5.0" + read-config-file "5.0.1" + sanitize-filename "^1.6.3" + semver "^7.1.1" + temp-file "^3.3.6" append-transform@^0.4.0: version "0.4.0" @@ -1221,19 +1256,19 @@ bower@1.8.2: resolved "https://registry.yarnpkg.com/bower/-/bower-1.8.2.tgz#adf53529c8d4af02ef24fb8d5341c1419d33e2f7" integrity sha1-rfU1KcjUrwLvJPuNU0HBQZ0z4vc= -boxen@^3.0.0: - version "3.2.0" - resolved "https://registry.yarnpkg.com/boxen/-/boxen-3.2.0.tgz#fbdff0de93636ab4450886b6ff45b92d098f45eb" - integrity sha512-cU4J/+NodM3IHdSL2yN8bqYqnmlBTidDR4RC7nJs61ZmtGz8VZzM3HLQX0zY5mrSmPtR3xWwsq2jOUQqFZN8+A== +boxen@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/boxen/-/boxen-4.2.0.tgz#e411b62357d6d6d36587c8ac3d5d974daa070e64" + integrity sha512-eB4uT9RGzg2odpER62bBwSLvUeGC+WbRjjyyFhGsKnc8wp/m0+hQsMUvUe3H2V0D5vw0nBdO1hCJoZo5mKeuIQ== dependencies: ansi-align "^3.0.0" camelcase "^5.3.1" - chalk "^2.4.2" + chalk "^3.0.0" cli-boxes "^2.2.0" - string-width "^3.0.0" - term-size "^1.2.0" - type-fest "^0.3.0" - widest-line "^2.0.0" + string-width "^4.1.0" + term-size "^2.1.0" + type-fest "^0.8.1" + widest-line "^3.1.0" brace-expansion@^1.0.0, brace-expansion@^1.1.7: version "1.1.11" @@ -1424,32 +1459,52 @@ buffers@~0.1.1: resolved "https://registry.yarnpkg.com/buffers/-/buffers-0.1.1.tgz#b24579c3bed4d6d396aeee6d9a8ae7f5482ab7bb" integrity sha1-skV5w77U1tOWru5tmorn9Ugqt7s= -builder-util-runtime@8.3.0: - version "8.3.0" - resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-8.3.0.tgz#f5fac9139af6facf42a21fbe4d3aebed88fda33e" - integrity sha512-CSOdsYqf4RXIHh1HANPbrZHlZ9JQJXSuDDloblZPcWQVN62inyYoTQuSmY3KrgefME2Sv3Kn2MxHvbGQHRf8Iw== +builder-util-runtime@8.6.0: + version "8.6.0" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-8.6.0.tgz#b7007c30126da9a90e99932128d2922c8c178649" + integrity sha512-WTDhTUVrm7zkFyd6Qn7AXgmWifjpZ/fYnEdV3XCOIDMNNb/KPddBTbQ8bUlxxVeuOYlhGpcLUypG+4USdGL1ww== dependencies: debug "^4.1.1" sax "^1.2.4" -builder-util@21.2.0, builder-util@~21.2.0: - version "21.2.0" - resolved "https://registry.yarnpkg.com/builder-util/-/builder-util-21.2.0.tgz#aba721190e4e841009d9fb4b88f1130ed616522f" - integrity sha512-Nd6CUb6YgDY8EXAXEIegx+1kzKqyFQ5ZM5BoYkeunAlwz/zDJoH1UCyULjoS5wQe5czNClFQy07zz2bzYD0Z4A== +builder-util@22.3.2: + version "22.3.2" + resolved "https://registry.yarnpkg.com/builder-util/-/builder-util-22.3.2.tgz#23c61aaf0f0006f994087b33a26e47cdaec7aa8d" + integrity sha512-jNeyA/AgyujE6NQLoEkERY/qrilFdCFAC6cBM1ylSnRuzlD1tQQCI3QJhuMJG/gLUCgyNsfFQhN0Mw6IqsDAqw== dependencies: "7zip-bin" "~5.0.3" - "@types/debug" "^4.1.4" - app-builder-bin "3.4.3" + "@types/debug" "^4.1.5" + app-builder-bin "3.5.2" bluebird-lst "^1.0.9" - builder-util-runtime "8.3.0" - chalk "^2.4.2" + builder-util-runtime "8.6.0" + chalk "^3.0.0" debug "^4.1.1" fs-extra "^8.1.0" is-ci "^2.0.0" js-yaml "^3.13.1" - source-map-support "^0.5.13" - stat-mode "^0.3.0" - temp-file "^3.3.4" + source-map-support "^0.5.16" + stat-mode "^1.0.0" + temp-file "^3.3.6" + +builder-util@22.3.3, builder-util@~22.3.2, builder-util@~22.3.3: + version "22.3.3" + resolved "https://registry.yarnpkg.com/builder-util/-/builder-util-22.3.3.tgz#62f0527ceaa1a2e4a60596a9b38ad1ffe3e20ae6" + integrity sha512-VzQALenLDdeaz7hXaQgS9N0Xz3zlgkK64Dp2Vn61XTbhI0MgVneTeEKKDFwdBC/l7v0cHsOPeao/xeWmyznC2g== + dependencies: + "7zip-bin" "~5.0.3" + "@types/debug" "^4.1.5" + "@types/fs-extra" "^8.0.1" + app-builder-bin "3.5.2" + bluebird-lst "^1.0.9" + builder-util-runtime "8.6.0" + chalk "^3.0.0" + debug "^4.1.1" + fs-extra "^8.1.0" + is-ci "^2.0.0" + js-yaml "^3.13.1" + source-map-support "^0.5.16" + stat-mode "^1.0.0" + temp-file "^3.3.6" builtin-modules@^1.1.1: version "1.1.1" @@ -1664,6 +1719,14 @@ chalk@^2.0.0, chalk@^2.0.1, chalk@^2.1.0, chalk@^2.3.0, chalk@^2.3.1, chalk@^2.3 escape-string-regexp "^1.0.5" supports-color "^5.3.0" +chalk@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-3.0.0.tgz#3f73c2bf526591f574cc492c51e2456349f844e4" + integrity sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg== + dependencies: + ansi-styles "^4.1.0" + supports-color "^7.1.0" + character-entities-html4@^1.0.0: version "1.1.3" resolved "https://registry.yarnpkg.com/character-entities-html4/-/character-entities-html4-1.1.3.tgz#5ce6e01618e47048ac22f34f7f39db5c6fd679ef" @@ -1833,14 +1896,14 @@ cliui@^4.0.0: strip-ansi "^4.0.0" wrap-ansi "^2.0.0" -cliui@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/cliui/-/cliui-5.0.0.tgz#deefcfdb2e800784aa34f46fa08e06851c7bbbc5" - integrity sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA== +cliui@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-6.0.0.tgz#511d702c0c4e41ca156d7d0e96021f23e13225b1" + integrity sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ== dependencies: - string-width "^3.1.0" - strip-ansi "^5.2.0" - wrap-ansi "^5.1.0" + string-width "^4.2.0" + strip-ansi "^6.0.0" + wrap-ansi "^6.2.0" clone-response@1.0.2, clone-response@^1.0.2: version "1.0.2" @@ -2104,17 +2167,17 @@ config@1.28.1: json5 "0.4.0" os-homedir "1.0.2" -configstore@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/configstore/-/configstore-4.0.0.tgz#5933311e95d3687efb592c528b922d9262d227e7" - integrity sha512-CmquAXFBocrzaSM8mtGPMM/HiWmyIpr4CcJl/rgY2uCObZ/S7cKU0silxslqJejl+t/T9HS8E0PUNQD81JGUEQ== +configstore@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/configstore/-/configstore-5.0.0.tgz#37de662c7a49b5fe8dbcf8f6f5818d2d81ed852b" + integrity sha512-eE/hvMs7qw7DlcB5JPRnthmrITuHMmACUJAp89v6PT6iOqzoLS7HRWhBtuHMlhNHo2AhUSA/3Dh1bKNJHcublQ== dependencies: - dot-prop "^4.1.0" + dot-prop "^5.1.0" graceful-fs "^4.1.2" - make-dir "^1.0.0" - unique-string "^1.0.0" - write-file-atomic "^2.0.0" - xdg-basedir "^3.0.0" + make-dir "^3.0.0" + unique-string "^2.0.0" + write-file-atomic "^3.0.0" + xdg-basedir "^4.0.0" connect-history-api-fallback@^1.3.0: version "1.6.0" @@ -2315,10 +2378,10 @@ crypto-browserify@^3.11.0: randombytes "^2.0.0" randomfill "^1.0.3" -crypto-random-string@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-1.0.0.tgz#a230f64f568310e1498009940790ec99545bca7e" - integrity sha1-ojD2T1aDEOFJgAmUB5DsmVRbyn4= +crypto-random-string@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-2.0.0.tgz#ef2a7a966ec11083388369baa02ebead229b30d5" + integrity sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA== css-animation@^1.3.2: version "1.6.1" @@ -2749,18 +2812,18 @@ dir-glob@^2.0.0: dependencies: path-type "^3.0.0" -dmg-builder@21.2.0: - version "21.2.0" - resolved "https://registry.yarnpkg.com/dmg-builder/-/dmg-builder-21.2.0.tgz#a9c883557cacb9abdb66c7133b30fe921c1a3ba7" - integrity sha512-9cJEclnGy7EyKFCoHDYDf54pub/t92CQapyiUxU0w9Bj2vUvfoDagP1PMiX4XD5rPp96141h9A+QN0OB4VgvQg== +dmg-builder@22.3.2: + version "22.3.2" + resolved "https://registry.yarnpkg.com/dmg-builder/-/dmg-builder-22.3.2.tgz#4c052f75d601e3358da1ff9d7d57738e1c01b157" + integrity sha512-szx+nmCNjpfp2yy3zVCMQLj2nRHL3LfZyzvQECDGHr0ZHK7//48+MoJckkbOCB22ofBvx5Y2M1YqCjK8b2slYQ== dependencies: - app-builder-lib "~21.2.0" + app-builder-lib "~22.3.2" bluebird-lst "^1.0.9" - builder-util "~21.2.0" + builder-util "~22.3.2" fs-extra "^8.1.0" - iconv-lite "^0.5.0" + iconv-lite "^0.5.1" js-yaml "^3.13.1" - sanitize-filename "^1.6.2" + sanitize-filename "^1.6.3" dns-equal@^1.0.0: version "1.0.0" @@ -2824,19 +2887,19 @@ dompurify@^2.0.7: resolved "https://registry.yarnpkg.com/dompurify/-/dompurify-2.0.7.tgz#f8266ad38fe1602fb5b3222f31eedbf5c16c4fd5" integrity sha512-S3O0lk6rFJtO01ZTzMollCOGg+WAtCwS3U5E2WSDY/x/sy7q70RjEC4Dmrih5/UqzLLB9XoKJ8KqwBxaNvBu4A== -dot-prop@^4.1.0: - version "4.2.0" - resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-4.2.0.tgz#1f19e0c2e1aa0e32797c49799f2837ac6af69c57" - integrity sha512-tUMXrxlExSW6U2EXiiKGSBVdYgtV8qlHL+C10TsW4PURY/ic+eaysnSkwB4kA/mBlCyy/IKDJ+Lc3wbWeaXtuQ== +dot-prop@^5.1.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-5.2.0.tgz#c34ecc29556dc45f1f4c22697b6f4904e0cc4fcb" + integrity sha512-uEUyaDKoSQ1M4Oq8l45hSE26SnTxL6snNnqvK/VWx5wJhmff5z0FUVJDKDanor/6w3kzE3i7XZOk+7wC0EXr1A== dependencies: - is-obj "^1.0.0" + is-obj "^2.0.0" dotenv-expand@^5.1.0: version "5.1.0" resolved "https://registry.yarnpkg.com/dotenv-expand/-/dotenv-expand-5.1.0.tgz#3fbaf020bfd794884072ea26b1e9791d45a629f0" integrity sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA== -dotenv@^8.0.0: +dotenv@^8.2.0: version "8.2.0" resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-8.2.0.tgz#97e619259ada750eea3e4ea3e26bceea5424b16a" integrity sha512-8sJ78ElpbDJBHNeBzUbUVLsqKdccaa/BXF1uPTw3GrvQTBgrQrtObr2mUrE38vzYd8cEv+m/JBfDLioYcfXoaw== @@ -2881,34 +2944,34 @@ ee-first@1.1.1: resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0= -ejs@^2.6.2: - version "2.7.4" - resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.7.4.tgz#48661287573dcc53e366c7a1ae52c3a120eec9ba" - integrity sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA== +ejs@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/ejs/-/ejs-3.0.1.tgz#30c8f6ee9948502cc32e85c37a3f8b39b5a614a5" + integrity sha512-cuIMtJwxvzumSAkqaaoGY/L6Fc/t6YvoP9/VIaK0V/CyqKLEQ8sqODmYfy/cjXEdZ9+OOL8TecbJu+1RsofGDw== ejs@~2.5.6: version "2.5.9" resolved "https://registry.yarnpkg.com/ejs/-/ejs-2.5.9.tgz#7ba254582a560d267437109a68354112475b0ce5" integrity sha512-GJCAeDBKfREgkBtgrYSf9hQy9kTb3helv0zGdzqhM7iAkW8FA/ZF97VQDbwFiwIT8MQLLOe5VlPZOEvZAqtUAQ== -electron-builder@21.2.0: - version "21.2.0" - resolved "https://registry.yarnpkg.com/electron-builder/-/electron-builder-21.2.0.tgz#b68ec4def713fc0b8602654ce842f972432f50c5" - integrity sha512-x8EXrqFbAb2L3N22YlGar3dGh8vwptbB3ovo3OF6K7NTpcsmM2zEoJv7GhFyX73rNzSG2HaWpXwGAtOp2JWiEw== +electron-builder@22.3.2: + version "22.3.2" + resolved "https://registry.yarnpkg.com/electron-builder/-/electron-builder-22.3.2.tgz#902d150fc0670cb90213262e5e0aa3c4f299ffa4" + integrity sha512-bDjHfKtA4DapI6qqy4FC18fzLsOJtlSVGBqjSjhrgv+gbcppp3tjR6ASsUX5K64/8L9MGjhRGdfQ7iP78OLx8g== dependencies: - app-builder-lib "21.2.0" + app-builder-lib "22.3.2" bluebird-lst "^1.0.9" - builder-util "21.2.0" - builder-util-runtime "8.3.0" - chalk "^2.4.2" - dmg-builder "21.2.0" + builder-util "22.3.2" + builder-util-runtime "8.6.0" + chalk "^3.0.0" + dmg-builder "22.3.2" fs-extra "^8.1.0" is-ci "^2.0.0" lazy-val "^1.0.4" - read-config-file "5.0.0" - sanitize-filename "^1.6.2" - update-notifier "^3.0.1" - yargs "^13.3.0" + read-config-file "5.0.1" + sanitize-filename "^1.6.3" + update-notifier "^4.0.0" + yargs "^15.1.0" electron-chromedriver@~3.0.0: version "3.0.0" @@ -3004,15 +3067,29 @@ electron-notarize@^0.2.0: debug "^4.1.1" fs-extra "^8.1.0" -electron-publish@21.2.0: - version "21.2.0" - resolved "https://registry.yarnpkg.com/electron-publish/-/electron-publish-21.2.0.tgz#cc225cb46aa62e74b899f2f7299b396c9802387d" - integrity sha512-mWavuoWJe87iaeKd0I24dNWIaR+0yRzshjNVqGyK019H766fsPWl3caQJnVKFaEyrZRP397v4JZVG0e7s16AxA== +electron-publish@22.3.2: + version "22.3.2" + resolved "https://registry.yarnpkg.com/electron-publish/-/electron-publish-22.3.2.tgz#d2e60caf7a9643fe57e501c20acaf32c737b1c50" + integrity sha512-nUGepzRtoGT8Tv83RHnnrR5szbw4+HpLLBdZU+aO5Kt3uoReTsp7NSm+agJfbI+H5rVpYCG8h6nPIejxGZ7LTg== dependencies: bluebird-lst "^1.0.9" - builder-util "~21.2.0" - builder-util-runtime "8.3.0" - chalk "^2.4.2" + builder-util "~22.3.2" + builder-util-runtime "8.6.0" + chalk "^3.0.0" + fs-extra "^8.1.0" + lazy-val "^1.0.4" + mime "^2.4.4" + +electron-publish@22.3.3: + version "22.3.3" + resolved "https://registry.yarnpkg.com/electron-publish/-/electron-publish-22.3.3.tgz#7d1e757a20ce0558fdc42900b6e3d773fdae9d9e" + integrity sha512-QfdS6gyqdjX+JBm3DhRT8nwO2TKQF9Z2dsZBXxCfE+FXYe2XmxMXWeXY2vPBHxSOpBYeAYVIkBiNL+gWcSfA+w== + dependencies: + "@types/fs-extra" "^8.0.1" + bluebird-lst "^1.0.9" + builder-util "~22.3.3" + builder-util-runtime "8.6.0" + chalk "^3.0.0" fs-extra "^8.1.0" lazy-val "^1.0.4" mime "^2.4.4" @@ -3749,12 +3826,13 @@ find-up@^2.0.0, find-up@^2.1.0: dependencies: locate-path "^2.0.0" -find-up@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/find-up/-/find-up-3.0.0.tgz#49169f1d7993430646da61ecc5ae355c21c97b73" - integrity sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg== +find-up@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19" + integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw== dependencies: - locate-path "^3.0.0" + locate-path "^5.0.0" + path-exists "^4.0.0" findup-sync@~0.3.0: version "0.3.0" @@ -4195,12 +4273,12 @@ glob@~7.0.0: once "^1.3.0" path-is-absolute "^1.0.0" -global-dirs@^0.1.0: - version "0.1.1" - resolved "https://registry.yarnpkg.com/global-dirs/-/global-dirs-0.1.1.tgz#b319c0dd4607f353f3be9cca4c72fc148c49f445" - integrity sha1-sxnA3UYH81PzvpzKTHL8FIxJ9EU= +global-dirs@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/global-dirs/-/global-dirs-2.0.1.tgz#acdf3bb6685bcd55cb35e8a052266569e9469201" + integrity sha512-5HqUqdhkEovj2Of/ms3IeS/EekcO54ytHRLV4PEY2rhRwrHXLQjeVEES0Lhka0xwNDtGYn58wyC4s5+MHsOO6A== dependencies: - ini "^1.3.4" + ini "^1.3.5" global-modules@1.0.0, global-modules@^1.0.0: version "1.0.0" @@ -4518,6 +4596,11 @@ has-flag@^3.0.0: resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0= +has-flag@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" + integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== + has-symbol-support-x@^1.4.1: version "1.4.2" resolved "https://registry.yarnpkg.com/has-symbol-support-x/-/has-symbol-support-x-1.4.2.tgz#1409f98bc00247da45da67cee0a36f282ff26455" @@ -4652,11 +4735,18 @@ hooker@~0.2.3: resolved "https://registry.yarnpkg.com/hooker/-/hooker-0.2.3.tgz#b834f723cc4a242aa65963459df6d984c5d3d959" integrity sha1-uDT3I8xKJCqmWWNFnfbZhMXT2Vk= -hosted-git-info@^2.1.4, hosted-git-info@^2.7.1: +hosted-git-info@^2.1.4: version "2.8.5" resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.5.tgz#759cfcf2c4d156ade59b0b2dfabddc42a6b9c70c" integrity sha512-kssjab8CvdXfcXMXVcvsXum4Hwdq9XGtRD3TteMEvEbq0LXyiNQr6AprqKqfeaDXze7SxWvRxdpwE6ku7ikLkg== +hosted-git-info@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-3.0.2.tgz#8b7e3bd114b59b51786f8bade0f39ddc80275a97" + integrity sha512-ezZMWtHXm7Eb7Rq4Mwnx2vs79WUx2QmRg3+ZqeGroKzfDO+EprOcgRPYghsOP9JuYBfK18VojmRTGCg8Ma+ktw== + dependencies: + lru-cache "^5.1.1" + hpack.js@^2.1.6: version "2.1.6" resolved "https://registry.yarnpkg.com/hpack.js/-/hpack.js-2.1.6.tgz#87774c0949e513f42e84575b3c45681fade2a0b2" @@ -4822,10 +4912,10 @@ iconv-lite@0.4.24, iconv-lite@^0.4.17, iconv-lite@^0.4.4, iconv-lite@~0.4.13: dependencies: safer-buffer ">= 2.1.2 < 3" -iconv-lite@^0.5.0: - version "0.5.0" - resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.5.0.tgz#59cdde0a2a297cc2aeb0c6445a195ee89f127550" - integrity sha512-NnEhI9hIEKHOzJ4f697DMz9IQEXr/MMJ5w64vN2/4Ai+wRnvV7SBrL0KLoRlwaKVghOc7LQ5YkPLuX146b6Ydw== +iconv-lite@^0.5.1: + version "0.5.1" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.5.1.tgz#b2425d3c7b18f7219f2ca663d103bddb91718d64" + integrity sha512-ONHr16SQvKZNSqjQT9gy5z24Jw+uqfO02/ngBSBoqChZ+W8qXX7GPRa1RoUnzGADw8K63R1BXUMzarCVQBpY8Q== dependencies: safer-buffer ">= 2.1.2 < 3" @@ -4926,7 +5016,7 @@ inherits@2.0.3: resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.3.tgz#633c2c83e3da42a502f52466022480f4208261de" integrity sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4= -ini@^1.3.4, ini@~1.3.0: +ini@^1.3.4, ini@^1.3.5, ini@~1.3.0: version "1.3.5" resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.5.tgz#eee25f56db1c9ec6085e0c22778083f596abf927" integrity sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw== @@ -5215,18 +5305,18 @@ is-in-browser@^1.1.3: resolved "https://registry.yarnpkg.com/is-in-browser/-/is-in-browser-1.1.3.tgz#56ff4db683a078c6082eb95dad7dc62e1d04f835" integrity sha1-Vv9NtoOgeMYILrldrX3GLh0E+DU= -is-installed-globally@^0.1.0: - version "0.1.0" - resolved "https://registry.yarnpkg.com/is-installed-globally/-/is-installed-globally-0.1.0.tgz#0dfd98f5a9111716dd535dda6492f67bf3d25a80" - integrity sha1-Df2Y9akRFxbdU13aZJL2e/PSWoA= +is-installed-globally@^0.3.1: + version "0.3.1" + resolved "https://registry.yarnpkg.com/is-installed-globally/-/is-installed-globally-0.3.1.tgz#679afef819347a72584617fd19497f010b8ed35f" + integrity sha512-oiEcGoQbGc+3/iijAijrK2qFpkNoNjsHOm/5V5iaeydyrS/hnwaRCEgH5cpW0P3T1lSjV5piB7S5b5lEugNLhg== dependencies: - global-dirs "^0.1.0" - is-path-inside "^1.0.0" + global-dirs "^2.0.1" + is-path-inside "^3.0.1" -is-npm@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/is-npm/-/is-npm-3.0.0.tgz#ec9147bfb629c43f494cf67936a961edec7e8053" - integrity sha512-wsigDr1Kkschp2opC4G3yA6r9EgVA6NjRpWzIi9axXqeIaAATPRJc4uLujXe3Nd9uO8KoDyA4MD6aZSeXTADhA== +is-npm@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/is-npm/-/is-npm-4.0.0.tgz#c90dd8380696df87a7a6d823c20d0b12bbe3c84d" + integrity sha512-96ECIfh9xtDDlPylNPXhzjsykHsMJZ18ASpaWzQyBr4YRTcVjUvzaHayDAES2oU/3KpljhHUjtSRNiDwi0F0ig== is-number@^2.1.0: version "2.1.0" @@ -5247,11 +5337,16 @@ is-number@^4.0.0: resolved "https://registry.yarnpkg.com/is-number/-/is-number-4.0.0.tgz#0026e37f5454d73e356dfe6564699867c6a7f0ff" integrity sha512-rSklcAIlf1OmFdyAqbnWTLVelsQ58uvZ66S/ZyawjWqIviTWCjg2PzVGw8WUA+nNuPTqb4wgA+NszrJ+08LlgQ== -is-obj@^1.0.0, is-obj@^1.0.1: +is-obj@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-1.0.1.tgz#3e4729ac1f5fde025cd7d83a896dab9f4f67db0f" integrity sha1-PkcprB9f3gJc19g6iW2rn09n2w8= +is-obj@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz#473fb05d973705e3fd9620545018ca8e22ef4982" + integrity sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w== + is-object@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/is-object/-/is-object-1.0.1.tgz#8952688c5ec2ffd6b03ecc85e769e02903083470" @@ -5276,6 +5371,11 @@ is-path-inside@^1.0.0: dependencies: path-is-inside "^1.0.1" +is-path-inside@^3.0.1: + version "3.0.2" + resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.2.tgz#f5220fc82a3e233757291dddc9c5877f2a1f3017" + integrity sha512-/2UGPSgmtqwo1ktx8NDHjuPwZWmHhO+gj0f93EkhLB5RgW9RZevWYYlIkS6zePc6U2WpOdQYIwHe9YC4DWEBVg== + is-plain-obj@^1.0.0, is-plain-obj@^1.1.0: version "1.1.0" resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-1.1.0.tgz#71a50c8429dfca773c92a390a4a03b39fcd51d3e" @@ -5399,7 +5499,7 @@ isarray@1.0.0, isarray@^1.0.0, isarray@~1.0.0: resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" integrity sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE= -isbinaryfile@^4.0.2: +isbinaryfile@^4.0.4: version "4.0.4" resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.4.tgz#6803f81a8944201c642b6e17da041e24deb78712" integrity sha512-pEutbN134CzcjlLS1myKX/uxNjwU5eBVSprvkpv3+3dqhBHUZLIWJQowC40w5c0Zf19vBY8mrZl88y5J4RAPbQ== @@ -5643,7 +5743,7 @@ json5@^1.0.1: dependencies: minimist "^1.2.0" -json5@^2.1.0: +json5@^2.1.1: version "2.1.1" resolved "https://registry.yarnpkg.com/json5/-/json5-2.1.1.tgz#81b6cb04e9ba496f1c7005d07b4368a2638f90b6" integrity sha512-l+3HXD0GEI3huGq1njuqtzYK8OYJyXMkOLtQ53pjWh89tvWS2h6l+1zMkYWqlb57+SiQodKZyvMEFb2X+KrFhQ== @@ -5928,13 +6028,17 @@ locate-path@^2.0.0: p-locate "^2.0.0" path-exists "^3.0.0" -locate-path@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-3.0.0.tgz#dbec3b3ab759758071b58fe59fc41871af21400e" - integrity sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A== +locate-path@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-5.0.0.tgz#1afba396afd676a6d42504d0a67a3a7eb9f62aa0" + integrity sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g== dependencies: - p-locate "^3.0.0" - path-exists "^3.0.0" + p-locate "^4.1.0" + +lodash-es@^4.2.1: + version "4.17.15" + resolved "https://registry.yarnpkg.com/lodash-es/-/lodash-es-4.17.15.tgz#21bd96839354412f23d7a10340e5eac6ee455d78" + integrity sha512-rlrc3yU3+JNOpZ9zj5pQtxnx2THmvRykwL4Xlxoa8I9lHBlVbbyPhgyPMioxVZ4NqyxaVVtaJnzsyOidQIhyyQ== lodash-es@^4.2.1: version "4.17.15" @@ -6098,6 +6202,13 @@ lru-cache@^4.0.1, lru-cache@^4.1.1, lru-cache@^4.1.2: pseudomap "^1.0.2" yallist "^2.1.2" +lru-cache@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-5.1.1.tgz#1da27e6710271947695daf6848e847f01d84b920" + integrity sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w== + dependencies: + yallist "^3.0.2" + magic-string@^0.25.3: version "0.25.6" resolved "https://registry.yarnpkg.com/magic-string/-/magic-string-0.25.6.tgz#5586387d1242f919c6d223579cc938bf1420795e" @@ -6112,6 +6223,13 @@ make-dir@^1.0.0: dependencies: pify "^3.0.0" +make-dir@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.0.0.tgz#1b5f39f6b9270ed33f9f054c5c0f84304989f801" + integrity sha512-grNJDhb8b1Jm1qeqW5R/O63wUo4UXo2v2HMic6YT9i/HBlF93S8jkMgH7yugvY9ABDShH4VZMn8I+U8+fCNegw== + dependencies: + semver "^6.0.0" + map-cache@^0.2.2: version "0.2.2" resolved "https://registry.yarnpkg.com/map-cache/-/map-cache-0.2.2.tgz#c32abd0bd6525d9b051645bb4f26ac5dc98a0dbf" @@ -7143,7 +7261,7 @@ p-limit@^1.0.0, p-limit@^1.1.0: dependencies: p-try "^1.0.0" -p-limit@^2.0.0: +p-limit@^2.2.0: version "2.2.2" resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.2.2.tgz#61279b67721f5287aa1c13a9a7fbbc48c9291b1e" integrity sha512-WGR+xHecKTr7EbUEhyLSh5Dube9JtdiG78ufaeLxTgpudf/20KqyMioIUZJAezlTIi6evxuoUs9YXc11cU+yzQ== @@ -7157,12 +7275,12 @@ p-locate@^2.0.0: dependencies: p-limit "^1.1.0" -p-locate@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-3.0.0.tgz#322d69a05c0264b25997d9f40cd8a891ab0064a4" - integrity sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ== +p-locate@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-4.1.0.tgz#a3428bb7088b3a60292f66919278b7c297ad4f07" + integrity sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A== dependencies: - p-limit "^2.0.0" + p-limit "^2.2.0" p-map@^1.1.1: version "1.2.0" @@ -7336,6 +7454,11 @@ path-exists@^3.0.0: resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" integrity sha1-zg6+ql94yxiSXqfYENe1mwEP1RU= +path-exists@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" + integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== + path-is-absolute@^1.0.0, path-is-absolute@~1.0.0: version "1.0.1" resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" @@ -8460,16 +8583,16 @@ read-chunk@^1.0.1: resolved "https://registry.yarnpkg.com/read-chunk/-/read-chunk-1.0.1.tgz#5f68cab307e663f19993527d9b589cace4661194" integrity sha1-X2jKswfmY/GZk1J9m1icrORmEZQ= -read-config-file@5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/read-config-file/-/read-config-file-5.0.0.tgz#1487c983fae9c1b672d3acda5cac899a2d451f02" - integrity sha512-jIKUu+C84bfnKxyJ5j30CxCqgXWYjZLXuVE/NYlMEpeni+dhESgAeZOZd0JZbg1xTkMmnCdxksDoarkOyfEsOg== +read-config-file@5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/read-config-file/-/read-config-file-5.0.1.tgz#ead3df0d9822cc96006ca16322eaa79dac8591c2" + integrity sha512-75zp4PDbvtBlECoZK1KEkNlesr9OWdMWL8oi4xq+HXAM+kKHKU+Cx2ksFt+ie2BkrmkLBOKSfONDuz+WIKWoXA== dependencies: - dotenv "^8.0.0" + dotenv "^8.2.0" dotenv-expand "^5.1.0" fs-extra "^8.1.0" js-yaml "^3.13.1" - json5 "^2.1.0" + json5 "^2.1.1" lazy-val "^1.0.4" read-last-lines@1.3.0: @@ -8587,8 +8710,8 @@ redent@^1.0.0: resolved "https://registry.yarnpkg.com/redent/-/redent-1.0.0.tgz#cf916ab1fd5f1f16dfb20822dd6ec7f730c2afde" integrity sha1-z5Fqsf1fHxbfsggi3W7H9zDCr94= dependencies: - indent-string "^2.1.0" - strip-indent "^1.0.1" + loose-envify "^1.4.0" + symbol-observable "^1.2.0" reduce-css-calc@^1.2.6: version "1.3.0" @@ -9089,7 +9212,7 @@ samsam@1.3.0: resolved "https://registry.yarnpkg.com/samsam/-/samsam-1.3.0.tgz#8d1d9350e25622da30de3e44ba692b5221ab7c50" integrity sha512-1HwIYD/8UlOtFS3QO3w7ey+SdSDFE4HRNLZoZRYVQefrOY3l17epswImeB1ijgJFQJodIaHcwkp3r/myBjFVbg== -sanitize-filename@^1.6.2: +sanitize-filename@^1.6.3: version "1.6.3" resolved "https://registry.yarnpkg.com/sanitize-filename/-/sanitize-filename-1.6.3.tgz#755ebd752045931977e30b2025d340d7c9090378" integrity sha512-y/52Mcy7aw3gRm7IrcGDFx/bCk4AhRh2eI9luHOQM86nZsqwiRkkq2GekHXBBD+SmPidc8i2PqtYZl+pWJ8Oeg== @@ -9147,14 +9270,14 @@ selfsigned@^1.9.1: dependencies: node-forge "0.9.0" -semver-diff@^2.0.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/semver-diff/-/semver-diff-2.1.0.tgz#4bbb8437c8d37e4b0cf1a68fd726ec6d645d6d36" - integrity sha1-S7uEN8jTfksM8aaP1ybsbWRdbTY= +semver-diff@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/semver-diff/-/semver-diff-3.1.1.tgz#05f77ce59f325e00e2706afd67bb506ddb1ca32b" + integrity sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg== dependencies: - semver "^5.0.3" + semver "^6.3.0" -"semver@2 || 3 || 4 || 5", semver@^5.0.1, semver@^5.0.3, semver@^5.3.0, semver@^5.4.1: +"semver@2 || 3 || 4 || 5", semver@^5.0.1, semver@^5.3.0, semver@^5.4.1: version "5.7.1" resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== @@ -9164,11 +9287,16 @@ semver@5.4.1: resolved "https://registry.yarnpkg.com/semver/-/semver-5.4.1.tgz#e059c09d8571f0540823733433505d3a2f00b18e" integrity sha512-WfG/X9+oATh81XtllIo/I8gOiY9EXRdv1cQdyykeXK17YcUW3EXUAi2To4pcH6nZtJPr7ZOpM5OMyWJZm+8Rsg== -semver@^6.2.0, semver@^6.3.0: +semver@^6.0.0, semver@^6.2.0, semver@^6.3.0: version "6.3.0" resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== +semver@^7.1.1: + version "7.1.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.1.2.tgz#847bae5bce68c5d08889824f02667199b70e3d87" + integrity sha512-BJs9T/H8sEVHbeigqzIEo57Iu/3DG6c4QoqTfbQB3BPA4zgzAomh/Fk9E7QtjWQ8mx2dgA9YCfSF4y9k9bHNpQ== + semver@~5.3.0: version "5.3.0" resolved "https://registry.yarnpkg.com/semver/-/semver-5.3.0.tgz#9b2ce5d3de02d17c6012ad326aa6b4d0cf54f94f" @@ -9464,7 +9592,7 @@ source-map-resolve@^0.5.0, source-map-resolve@^0.5.2: source-map-url "^0.4.0" urix "^0.1.0" -source-map-support@^0.5.13: +source-map-support@^0.5.16: version "0.5.16" resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.16.tgz#0ae069e7fe3ba7538c64c98515e35339eac5a042" integrity sha512-efyLRJDr68D9hBBNIPWFjhpFzURh+KJykQwvMyW5UiZzYwoF6l4YMMDIJJEyFWxWCqfyxLzz6tSfUFR+kXXsVQ== @@ -9630,10 +9758,10 @@ ssri@^5.2.4: dependencies: safe-buffer "^5.1.1" -stat-mode@^0.3.0: - version "0.3.0" - resolved "https://registry.yarnpkg.com/stat-mode/-/stat-mode-0.3.0.tgz#69283b081f851582b328d2a4ace5f591ce52f54b" - integrity sha512-QjMLR0A3WwFY2aZdV0okfFEJB5TRjkggXZjxP3A1RsWsNHNu3YPv8btmtc6iCFZ0Rul3FE93OYogvhOUClU+ng== +stat-mode@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/stat-mode/-/stat-mode-1.0.0.tgz#68b55cb61ea639ff57136f36b216a291800d1465" + integrity sha512-jH9EhtKIjuXZ2cWxmXS8ZP80XyC3iasQxMDV8jzhNJpfDb7VbQLVW4Wvsxz9QZvzV+G4YoSfBUVKDOyxLzi/sg== state-toggle@^1.0.0: version "1.0.2" @@ -9733,7 +9861,7 @@ string-width@^1.0.1, string-width@^1.0.2: is-fullwidth-code-point "^2.0.0" strip-ansi "^4.0.0" -string-width@^3.0.0, string-width@^3.1.0: +string-width@^3.0.0: version "3.1.0" resolved "https://registry.yarnpkg.com/string-width/-/string-width-3.1.0.tgz#22767be21b62af1081574306f69ac51b62203961" integrity sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w== @@ -9742,7 +9870,7 @@ string-width@^3.0.0, string-width@^3.1.0: is-fullwidth-code-point "^2.0.0" strip-ansi "^5.1.0" -string-width@^4.2.0: +string-width@^4.0.0, string-width@^4.1.0, string-width@^4.2.0: version "4.2.0" resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.0.tgz#952182c46cc7b2c313d1596e623992bd163b72b5" integrity sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg== @@ -9819,7 +9947,7 @@ strip-ansi@^4.0.0: dependencies: ansi-regex "^3.0.0" -strip-ansi@^5.0.0, strip-ansi@^5.1.0, strip-ansi@^5.2.0: +strip-ansi@^5.1.0: version "5.2.0" resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-5.2.0.tgz#8c9a536feb6afc962bdfa5b104a5091c1ad9c0ae" integrity sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA== @@ -9903,6 +10031,13 @@ supports-color@^5.1.0, supports-color@^5.3.0, supports-color@^5.4.0: dependencies: has-flag "^3.0.0" +supports-color@^7.1.0: + version "7.1.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.1.0.tgz#68e32591df73e25ad1c4b49108a2ec507962bfd1" + integrity sha512-oRSIpR8pxT1Wr2FquTNnGet79b3BWljqOuoW/h4oBhxJ/HUbX5nX6JSruTkvXDCFMwDPvsaTTbvMLKZWSy0R5g== + dependencies: + has-flag "^4.0.0" + supports-color@~5.0.0: version "5.0.1" resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.0.1.tgz#1c5331f22250c84202805b2f17adf16699f3a39a" @@ -10003,7 +10138,7 @@ tar@^4: safe-buffer "^5.1.2" yallist "^3.0.3" -temp-file@^3.3.4: +temp-file@^3.3.6: version "3.3.6" resolved "https://registry.yarnpkg.com/temp-file/-/temp-file-3.3.6.tgz#bd7a1951338bf93b59380b498ec1804d5b76c449" integrity sha512-7TPldi8QJqRlPIF/Y33mVvo8+xDfi6+aVTCK4CrCaLqCoaOnVtf3SA4hCU0T5nhYDdOC7erw7o2uWfvijlk4Ug== @@ -10011,12 +10146,10 @@ temp-file@^3.3.4: async-exit-hook "^2.0.1" fs-extra "^8.1.0" -term-size@^1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/term-size/-/term-size-1.2.0.tgz#458b83887f288fc56d6fffbfad262e26638efa69" - integrity sha1-RYuDiH8oj8Vtb/+/rSYuJmOO+mk= - dependencies: - execa "^0.7.0" +term-size@^2.1.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/term-size/-/term-size-2.2.0.tgz#1f16adedfe9bdc18800e1776821734086fcc6753" + integrity sha512-a6sumDlzyHVJWb8+YofY4TW112G6p2FCPEAFk+59gIYHv3XHRhm9ltVQ9kli4hNWeQBwSpe8cRN25x0ROunMOw== test-exclude@^4.1.1: version "4.2.3" @@ -10358,10 +10491,10 @@ type-detect@4.0.8, type-detect@^4.0.0, type-detect@^4.0.5, type-detect@^4.0.8: resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.0.8.tgz#7646fb5f18871cfbb7749e69bd39a6388eb7450c" integrity sha512-0fr/mIH1dlO+x7TlcMy+bIDqKPsw/70tVyeHW787goQjhmqaZe10uwLujubK9q9Lg6Fiho1KUKDYz0Z7k7g5/g== -type-fest@^0.3.0: - version "0.3.1" - resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.3.1.tgz#63d00d204e059474fe5e1b7c011112bbd1dc29e1" - integrity sha512-cUGJnCdr4STbePCgqNFbpVNCepa+kAVohJs1sLhxzdH+gnEoOd8VhbYa7pD3zZYGiURWM2xzEII3fQcRizDkYQ== +type-fest@^0.8.1: + version "0.8.1" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.8.1.tgz#09e249ebde851d3b1e48d27c105444667f17b83d" + integrity sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA== type-is@~1.6.10, type-is@~1.6.17, type-is@~1.6.18: version "1.6.18" @@ -10529,12 +10662,12 @@ unique-slug@^2.0.0: dependencies: imurmurhash "^0.1.4" -unique-string@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/unique-string/-/unique-string-1.0.0.tgz#9e1057cca851abb93398f8b33ae187b99caec11a" - integrity sha1-nhBXzKhRq7kzmPizOuGHuZyuwRo= +unique-string@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/unique-string/-/unique-string-2.0.0.tgz#39c6451f81afb2749de2b233e3f7c5e8843bd89d" + integrity sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg== dependencies: - crypto-random-string "^1.0.0" + crypto-random-string "^2.0.0" unist-util-is@^3.0.0: version "3.0.0" @@ -10603,23 +10736,23 @@ upath@^1.1.1: resolved "https://registry.yarnpkg.com/upath/-/upath-1.2.0.tgz#8f66dbcd55a883acdae4408af8b035a5044c1894" integrity sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg== -update-notifier@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/update-notifier/-/update-notifier-3.0.1.tgz#78ecb68b915e2fd1be9f767f6e298ce87b736250" - integrity sha512-grrmrB6Zb8DUiyDIaeRTBCkgISYUgETNe7NglEbVsrLWXeESnlCSP50WfRSj/GmzMPl6Uchj24S/p80nP/ZQrQ== +update-notifier@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/update-notifier/-/update-notifier-4.0.0.tgz#f344a6f8b03e00e31b323d632a0e632e9f0e0654" + integrity sha512-p9zf71hWt5GVXM4iEBujpUgx8mK9AWiCCapEJm/O1z5ntCim83Z1ATqzZFBHFYqx03laMqv8LiDgs/7ikXjf/g== dependencies: - boxen "^3.0.0" - chalk "^2.0.1" - configstore "^4.0.0" + boxen "^4.2.0" + chalk "^3.0.0" + configstore "^5.0.0" has-yarn "^2.1.0" import-lazy "^2.1.0" is-ci "^2.0.0" - is-installed-globally "^0.1.0" - is-npm "^3.0.0" + is-installed-globally "^0.3.1" + is-npm "^4.0.0" is-yarn-global "^0.3.0" latest-version "^5.0.0" - semver-diff "^2.0.0" - xdg-basedir "^3.0.0" + semver-diff "^3.1.1" + xdg-basedir "^4.0.0" uri-js@^4.2.2: version "4.2.2" @@ -10992,12 +11125,12 @@ wide-align@^1.1.0: dependencies: string-width "^1.0.2 || 2" -widest-line@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/widest-line/-/widest-line-2.0.1.tgz#7438764730ec7ef4381ce4df82fb98a53142a3fc" - integrity sha512-Ba5m9/Fa4Xt9eb2ELXt77JxVDV8w7qQrH0zS/TWSJdLyAwQjWoOzpzj5lwVftDz6n/EOu3tNACS84v509qwnJA== +widest-line@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/widest-line/-/widest-line-3.1.0.tgz#8292333bbf66cb45ff0de1603b136b7ae1496eca" + integrity sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg== dependencies: - string-width "^2.1.1" + string-width "^4.0.0" word-wrap@~1.2.3: version "1.2.3" @@ -11024,14 +11157,14 @@ wrap-ansi@^2.0.0: string-width "^1.0.1" strip-ansi "^3.0.1" -wrap-ansi@^5.1.0: - version "5.1.0" - resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-5.1.0.tgz#1fd1f67235d5b6d0fee781056001bfb694c03b09" - integrity sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q== +wrap-ansi@^6.2.0: + version "6.2.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-6.2.0.tgz#e9393ba07102e6c91a3b221478f0257cd2856e53" + integrity sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA== dependencies: - ansi-styles "^3.2.0" - string-width "^3.0.0" - strip-ansi "^5.0.0" + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" wrappy@1: version "1.0.2" @@ -11045,16 +11178,17 @@ write-file-atomic@^1.1.4: dependencies: graceful-fs "^4.1.11" imurmurhash "^0.1.4" - slide "^1.1.5" + signal-exit "^3.0.2" -write-file-atomic@^2.0.0: - version "2.4.3" - resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-2.4.3.tgz#1fd2e9ae1df3e75b8d8c367443c692d4ca81f481" - integrity sha512-GaETH5wwsX+GcnzhPgKcKjJ6M2Cq3/iZp1WyY/X1CSqrW+jVNM9Y7D8EC2sM4ZG/V8wZlSniJnCKWPmBYAucRQ== +write-file-atomic@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-3.0.1.tgz#558328352e673b5bb192cf86500d60b230667d4b" + integrity sha512-JPStrIyyVJ6oCSz/691fAjFtefZ6q+fP6tm+OS4Qw6o+TGQxNp1ziY2PgS+X/m0V8OWhZiO/m4xSj+Pr4RrZvw== dependencies: - graceful-fs "^4.1.11" imurmurhash "^0.1.4" + is-typedarray "^1.0.0" signal-exit "^3.0.2" + typedarray-to-buffer "^3.1.5" write@^0.2.1: version "0.2.1" @@ -11068,10 +11202,10 @@ x-is-string@^0.1.0: resolved "https://registry.yarnpkg.com/x-is-string/-/x-is-string-0.1.0.tgz#474b50865af3a49a9c4657f05acd145458f77d82" integrity sha1-R0tQhlrzpJqcRlfwWs0UVFj3fYI= -xdg-basedir@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/xdg-basedir/-/xdg-basedir-3.0.0.tgz#496b2cc109eca8dbacfe2dc72b603c17c5870ad4" - integrity sha1-SWsswQnsqNus/i3HK2A8F8WHCtQ= +xdg-basedir@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/xdg-basedir/-/xdg-basedir-4.0.0.tgz#4bc8d9984403696225ef83a1573cbbcb4e79db13" + integrity sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q== xhr@^2.0.1: version "2.5.0" @@ -11143,10 +11277,10 @@ yallist@^3.0.0, yallist@^3.0.2, yallist@^3.0.3: resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd" integrity sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g== -yargs-parser@^13.1.1: - version "13.1.1" - resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-13.1.1.tgz#d26058532aa06d365fe091f6a1fc06b2f7e5eca0" - integrity sha512-oVAVsHz6uFrg3XQheFII8ESO2ssAf9luWuAd6Wexsu4F3OtIW0o8IribPXYrD4WC24LWtPrJlGy87y5udK+dxQ== +yargs-parser@^16.1.0: + version "16.1.0" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-16.1.0.tgz#73747d53ae187e7b8dbe333f95714c76ea00ecf1" + integrity sha512-H/V41UNZQPkUMIT5h5hiwg4QKIY1RPvoBV4XcjUbRM8Bk2oKqqyZ0DIEbTFZB0XjbtSPG8SAa/0DxCQmiRgzKg== dependencies: camelcase "^5.0.0" decamelize "^1.2.0" @@ -11209,21 +11343,22 @@ yargs@^10.0.3: y18n "^3.2.1" yargs-parser "^8.1.0" -yargs@^13.3.0: - version "13.3.0" - resolved "https://registry.yarnpkg.com/yargs/-/yargs-13.3.0.tgz#4c657a55e07e5f2cf947f8a366567c04a0dedc83" - integrity sha512-2eehun/8ALW8TLoIl7MVaRUrg+yCnenu8B4kBlRxj3GJGDKU1Og7sMXPNm1BYyM1DOJmTZ4YeN/Nwxv+8XJsUA== +yargs@^15.1.0: + version "15.1.0" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-15.1.0.tgz#e111381f5830e863a89550bd4b136bb6a5f37219" + integrity sha512-T39FNN1b6hCW4SOIk1XyTOWxtXdcen0t+XYrysQmChzSipvhBO8Bj0nK1ozAasdk24dNWuMZvr4k24nz+8HHLg== dependencies: - cliui "^5.0.0" - find-up "^3.0.0" + cliui "^6.0.0" + decamelize "^1.2.0" + find-up "^4.1.0" get-caller-file "^2.0.1" require-directory "^2.1.1" require-main-filename "^2.0.0" set-blocking "^2.0.0" - string-width "^3.0.0" + string-width "^4.2.0" which-module "^2.0.0" y18n "^4.0.0" - yargs-parser "^13.1.1" + yargs-parser "^16.1.0" yargs@^7.0.0: version "7.1.0"