From 16612aa6636969da0b6c75b6723e53c096905bba Mon Sep 17 00:00:00 2001 From: breadhunter <6099829+EmiM@users.noreply.github.com> Date: Thu, 23 Nov 2023 16:40:25 +0100 Subject: [PATCH] Chore/master to develop (#2104) * Fix - js injection in message input (#1943) * use notarytool for macos notarization * Secure backend socket.io from other applications that can access localhost i.e. browser (#1940) * secure socket IO connection with token and origin, transform token from main.ts to backend and state manager * Add authorization headers to socketio android notifications client * Secure socketIO connection on iOS * Extend lastKnownPort to lastKnownSocketIOData on android * Handle socketIOSecret for iOS lifecycle event * feat: getRandomValues and concept for validating options on backend * fix: use secure crypto for ios socketio secret --------- Co-authored-by: Vin Kabuki Co-authored-by: siepra * feat: notifier component #1980 * feat: use mailto for support address #1980 * fix: building mobile package #1980 * Publish - @quiet/backend@1.9.5 - @quiet/common@1.8.2 - quiet@1.9.6 - e2e-tests@1.8.3 - integration-tests@1.9.2 - @quiet/mobile@1.10.9 - @quiet/state-manager@1.9.2 * fix: pass team id for notarization * chore: abort build on notarization failure (#2081) * chore: deactivate 'breaking changes warning' for mobile and desktop #2097 #2096 * fix: use default websocket port in case of none --------- Co-authored-by: Kacper Michalik Co-authored-by: Vin Kabuki Co-authored-by: Kacper-RF <111343884+Kacper-RF@users.noreply.github.com> Co-authored-by: siepra Co-authored-by: Wiktor Sieprawski Co-authored-by: github@zbay.llc --- .github/workflows/desktop-build.yml | 5 +- packages/backend/CHANGELOG.md | 1 + packages/backend/src/backendManager.ts | 20 +-- packages/backend/src/nest/app.module.ts | 36 +++- packages/backend/src/nest/types.ts | 1 + packages/backend/src/options.ts | 32 ++++ packages/common/CHANGELOG.md | 14 +- packages/common/package-lock.json | 154 +++++++++++++++++ packages/common/src/auth.test.ts | 19 ++ packages/common/src/auth.ts | 6 + packages/common/src/index.ts | 1 + packages/desktop/CHANGELOG.md | 21 ++- packages/desktop/scripts/notarize.js | 29 ++-- packages/desktop/src/main/main.ts | 7 +- packages/desktop/src/renderer/Root.tsx | 3 + .../update/UpdateModalComponent.stories.tsx | 53 ++++++ ...test.tsx => UpdateModalComponent.test.tsx} | 87 +++------- .../widgets/update/UpdateModalComponent.tsx | 79 +++++++++ .../BreakingChangesWarning.tsx | 63 +++++++ .../widgets/update/UpdateModal.test.tsx | 35 ++++ .../containers/widgets/update/UpdateModal.tsx | 33 +++- packages/desktop/src/renderer/index.tsx | 7 +- .../src/renderer/sagas/modals/modals.slice.ts | 3 +- .../src/renderer/sagas/modals/modals.types.ts | 1 + .../renderer/sagas/socket/socket.saga.test.ts | 41 +++++ .../src/renderer/sagas/socket/socket.saga.ts | 27 ++- .../src/renderer/testUtils/prepareStore.ts | 1 + packages/e2e-tests/CHANGELOG.md | 1 + .../src/tests/backwardsCompatibility.test.ts | 6 +- packages/integration-tests/CHANGELOG.md | 6 + packages/mobile/.storybook/index.js | 1 + packages/mobile/CHANGELOG.md | 3 + .../com/quietmobile/Backend/BackendWorker.kt | 24 ++- .../Scheme/WebsocketConnectionPayload.kt | 3 +- .../main/java/com/quietmobile/Utils/Utils.kt | 14 ++ .../mobile/assets/icons/update_graphics.png | Bin 0 -> 46275 bytes packages/mobile/ios/CommunicationModule.swift | 4 +- .../ios/Quiet.xcodeproj/project.pbxproj | 128 +++++++------- packages/mobile/ios/Quiet/AppDelegate.h | 2 + packages/mobile/ios/Quiet/AppDelegate.m | 13 +- packages/mobile/ios/Utils.swift | 23 +++ packages/mobile/src/App.tsx | 50 +++--- packages/mobile/src/assets.ts | 46 ++--- .../components/Button/Button.component.tsx | 7 +- .../src/components/Button/Button.stories.tsx | 3 +- .../src/components/Button/Button.types.ts | 1 + .../Notifier/Notifier.component.tsx | 44 +++++ .../components/Notifier/Notifier.stories.tsx | 19 ++ .../src/components/Notifier/Notifier.test.tsx | 47 +++++ .../src/components/Notifier/Notifier.types.ts | 10 ++ .../__snapshots__/Notifier.test.tsx.snap | 163 ++++++++++++++++++ .../components/Typography/Typography.types.ts | 2 +- packages/mobile/src/const/ScreenNames.enum.ts | 21 +-- packages/mobile/src/route.params.ts | 40 +++-- .../ConnectionProcess.screen.tsx | 4 +- .../src/screens/Notifier/Notifier.screen.tsx | 31 ++++ .../blindConnection/blindConnection.saga.ts | 8 +- .../store/init/deepLink/deepLink.saga.test.ts | 5 + .../mobile/src/store/init/init.selectors.ts | 4 +- packages/mobile/src/store/init/init.slice.ts | 9 +- .../mobile/src/store/init/init.transform.ts | 2 +- .../restoreConnection.saga.test.ts | 25 +-- .../restoreConnection.saga.ts | 12 +- .../startConnection/startConnection.saga.ts | 34 ++-- .../mobile/src/tests/joining.process.test.tsx | 6 +- .../mobile/src/tests/splash.screen.test.tsx | 6 +- .../mobile/src/tests/utils/prepareStore.ts | 2 +- packages/state-manager/CHANGELOG.md | 9 +- .../connection.selectors.test.ts | 13 ++ .../appConnection/connection.selectors.ts | 3 + .../sagas/appConnection/connection.slice.ts | 5 + .../appConnection/connection.transform.ts | 1 + 72 files changed, 1318 insertions(+), 321 deletions(-) create mode 100644 packages/backend/src/options.ts create mode 100644 packages/common/src/auth.test.ts create mode 100644 packages/common/src/auth.ts create mode 100644 packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.stories.tsx rename packages/desktop/src/renderer/components/widgets/update/{UpdateModal.test.tsx => UpdateModalComponent.test.tsx} (54%) create mode 100644 packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.tsx create mode 100644 packages/desktop/src/renderer/containers/widgets/breakingChangesWarning/BreakingChangesWarning.tsx create mode 100644 packages/desktop/src/renderer/containers/widgets/update/UpdateModal.test.tsx create mode 100644 packages/desktop/src/renderer/sagas/socket/socket.saga.test.ts create mode 100644 packages/mobile/assets/icons/update_graphics.png create mode 100644 packages/mobile/ios/Utils.swift create mode 100644 packages/mobile/src/components/Notifier/Notifier.component.tsx create mode 100644 packages/mobile/src/components/Notifier/Notifier.stories.tsx create mode 100644 packages/mobile/src/components/Notifier/Notifier.test.tsx create mode 100644 packages/mobile/src/components/Notifier/Notifier.types.ts create mode 100644 packages/mobile/src/components/Notifier/__snapshots__/Notifier.test.tsx.snap create mode 100644 packages/mobile/src/screens/Notifier/Notifier.screen.tsx diff --git a/.github/workflows/desktop-build.yml b/.github/workflows/desktop-build.yml index 84e6c79ab5..841c1c5776 100644 --- a/.github/workflows/desktop-build.yml +++ b/.github/workflows/desktop-build.yml @@ -138,8 +138,9 @@ jobs: GH_TOKEN: ${{ secrets.GH_TOKEN }} CSC_KEY_PASSWORD: ${{ secrets.MAC_CSC_KEY_PASSWORD }} CSC_LINK: ${{ secrets.MAC_CSC_LINK }} - APPLEID: ${{ secrets.APPLE_ID }} - APPLEIDPASS: ${{ secrets.APPLE_ID_PASS }} + APPLE_ID: ${{ secrets.APPLE_ID }} + APPLE_ID_PASS: ${{ secrets.APPLE_ID_PASS }} + APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }} AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }} USE_HARD_LINKS: false diff --git a/packages/backend/CHANGELOG.md b/packages/backend/CHANGELOG.md index 850cd66787..8318c9d408 100644 --- a/packages/backend/CHANGELOG.md +++ b/packages/backend/CHANGELOG.md @@ -193,6 +193,7 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline # [1.10.0-alpha.0](https://github.com/TryQuiet/backend/compare/@quiet/backend@1.9.0...@quiet/backend@1.10.0-alpha.0) (2023-08-29) +## [1.9.5](https://github.com/TryQuiet/backend/compare/@quiet/backend@1.9.4...@quiet/backend@1.9.5) (2023-11-09) **Note:** Version bump only for package @quiet/backend diff --git a/packages/backend/src/backendManager.ts b/packages/backend/src/backendManager.ts index f7e80b5e4e..b0abe6066e 100644 --- a/packages/backend/src/backendManager.ts +++ b/packages/backend/src/backendManager.ts @@ -8,8 +8,10 @@ import { ConnectionsManagerService } from './nest/connections-manager/connection import { TorControl } from './nest/tor/tor-control.service' import { torBinForPlatform, torDirForPlatform } from './nest/common/utils' import initRnBridge from './rn-bridge' - +import { INestApplicationContext } from '@nestjs/common' import logger from './nest/common/logger' +import { OpenServices, validateOptions } from './options' + const log = logger('backendManager') const program = new Command() @@ -25,21 +27,13 @@ program .option('-a, --appDataPath ', 'Path of application data directory') .option('-d, --socketIOPort ', 'Socket io data server port') .option('-r, --resourcesPath ', 'Application resources path') + .option('-scrt, --socketIOSecret ', 'socketIO secret') program.parse(process.argv) const options = program.opts() console.log('options', options) -interface OpenServices { - torControlPort?: any - socketIOPort?: any - httpTunnelPort?: any - authCookie?: any -} - -import { INestApplicationContext } from '@nestjs/common' - export const runBackendDesktop = async () => { const isDev = process.env.NODE_ENV === 'development' @@ -48,11 +42,14 @@ export const runBackendDesktop = async () => { // @ts-ignore global.crypto = webcrypto + validateOptions(options) + const resourcesPath = isDev ? null : options.resourcesPath.trim() const app = await NestFactory.createApplicationContext( AppModule.forOptions({ socketIOPort: options.socketIOPort, + socketIOSecret: options.socketIOSecret, torBinaryPath: torBinForPlatform(resourcesPath), torResourcesPath: torDirForPlatform(resourcesPath), torControlPort: await getPort(), @@ -87,7 +84,7 @@ export const runBackendDesktop = async () => { }) } -export const runBackendMobile = async (): Promise => { +export const runBackendMobile = async () => { // Enable triggering push notifications process.env['BACKEND'] = 'mobile' process.env['CONNECTION_TIME'] = (new Date().getTime() / 1000).toString() // Get time in seconds @@ -97,6 +94,7 @@ export const runBackendMobile = async (): Promise => { const app: INestApplicationContext = await NestFactory.createApplicationContext( AppModule.forOptions({ socketIOPort: options.dataPort, + socketIOSecret: options.socketIOSecret, httpTunnelPort: options.httpTunnelPort ? options.httpTunnelPort : null, torAuthCookie: options.authCookie ? options.authCookie : null, torControlPort: options.controlPort ? options.controlPort : await getPort(), diff --git a/packages/backend/src/nest/app.module.ts b/packages/backend/src/nest/app.module.ts index bbc422bbbb..cf2fdc655d 100644 --- a/packages/backend/src/nest/app.module.ts +++ b/packages/backend/src/nest/app.module.ts @@ -32,7 +32,7 @@ import { Server as SocketIO } from 'socket.io' import { StorageModule } from './storage/storage.module' import { IpfsModule } from './ipfs/ipfs.module' import { Level } from 'level' -import { getCors } from './common/utils' +import { verifyToken } from '@quiet/common' @Global() @Module({ @@ -94,10 +94,40 @@ export class AppModule { _app.use(cors()) const server = createServer(_app) const io = new SocketIO(server, { - cors: getCors(), + cors: { + origin: '127.0.0.1', + allowedHeaders: ['authorization'], + credentials: true, + }, pingInterval: 1000_000, pingTimeout: 1000_000, }) + io.engine.use((req, res, next) => { + const authHeader = req.headers['authorization'] + if (!authHeader) { + console.error('No authorization header') + res.writeHead(401, 'No authorization header') + res.end() + return + } + + const token = authHeader && authHeader.split(' ')[1] + if (!token) { + console.error('No auth token') + res.writeHead(401, 'No authorization token') + res.end() + return + } + + if (verifyToken(options.socketIOSecret, token)) { + next() + } else { + console.error('Wrong basic token') + res.writeHead(401, 'Unauthorized') + res.end() + } + }) + return { server, io } }, inject: [EXPRESS_PROVIDER], @@ -122,7 +152,7 @@ export class AppModule { }, { provide: LEVEL_DB, - useFactory: (dbPath: string) => new Level(dbPath, { valueEncoding: 'json' }), + useFactory: (dbPath: string) => new Level(dbPath, { valueEncoding: 'json' }), inject: [DB_PATH], }, ], diff --git a/packages/backend/src/nest/types.ts b/packages/backend/src/nest/types.ts index fa2c4e3943..ec6302de1a 100644 --- a/packages/backend/src/nest/types.ts +++ b/packages/backend/src/nest/types.ts @@ -5,6 +5,7 @@ import { Server as SocketIO } from 'socket.io' export class ConnectionsManagerTypes { options: Partial socketIOPort: number + socketIOSecret: string httpTunnelPort?: number torAuthCookie?: string torControlPort?: number diff --git a/packages/backend/src/options.ts b/packages/backend/src/options.ts new file mode 100644 index 0000000000..a19d0342c4 --- /dev/null +++ b/packages/backend/src/options.ts @@ -0,0 +1,32 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ +import commander from 'commander' + +export interface OpenServices { + torControlPort?: any + socketIOPort?: any + socketIOSecret?: any + httpTunnelPort?: any + authCookie?: any +} + +interface Options { + platform?: any + dataPath?: any + dataPort?: any + torBinary?: any + authCookie?: any + controlPort?: any + httpTunnelPort?: any + appDataPath?: string + socketIOPort?: number + resourcesPath?: string + socketIOSecret: string +} + +// concept +export const validateOptions = (_options: commander.OptionValues) => { + const options = _options as Options + if (!options.socketIOSecret) { + throw new Error('socketIOSecret is missing in options') + } +} diff --git a/packages/common/CHANGELOG.md b/packages/common/CHANGELOG.md index 58019f1a5d..8e887a1e7b 100644 --- a/packages/common/CHANGELOG.md +++ b/packages/common/CHANGELOG.md @@ -5,10 +5,6 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline ## [2.0.2-alpha.1](https://github.com/TryQuiet/quiet/compare/@quiet/common@2.0.2-alpha.0...@quiet/common@2.0.2-alpha.1) (2023-11-14) -**Note:** Version bump only for package @quiet/common - - - ## [2.0.2-alpha.0](https://github.com/TryQuiet/quiet/compare/@quiet/common@2.0.1-alpha.4...@quiet/common@2.0.2-alpha.0) (2023-10-26) @@ -107,9 +103,17 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline +# [1.9.0-alpha.0](/compare/@quiet/common@1.8.0...@quiet/common@1.9.0-alpha.0) (2023-08-29) + + +## [1.8.2](https://github.com/TryQuiet/quiet/compare/@quiet/common@1.8.1...@quiet/common@1.8.2) (2023-11-09) + +**Note:** Version bump only for package @quiet/common + + + -# [1.9.0-alpha.0](/compare/@quiet/common@1.8.0...@quiet/common@1.9.0-alpha.0) (2023-08-29) ## [1.8.1](https://github.com/TryQuiet/quiet/compare/@quiet/common@1.8.0...@quiet/common@1.8.1) (2023-09-15) **Note:** Version bump only for package @quiet/common diff --git a/packages/common/package-lock.json b/packages/common/package-lock.json index 8e6302db19..f59a8c3632 100644 --- a/packages/common/package-lock.json +++ b/packages/common/package-lock.json @@ -983,6 +983,88 @@ "@jridgewell/sourcemap-codec": "1.4.14" } }, + "node_modules/@peculiar/webcrypto": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/@peculiar/webcrypto/-/webcrypto-1.4.3.tgz", + "integrity": "sha512-VtaY4spKTdN5LjJ04im/d/joXuvLbQdgy5Z4DXF4MFZhQ+MTrejbNMkfZBp1Bs3O5+bFqnJgyGdPuZQflvIa5A==", + "dependencies": { + "@peculiar/asn1-schema": "^2.3.6", + "@peculiar/json-schema": "^1.1.12", + "pvtsutils": "^1.3.2", + "tslib": "^2.5.0", + "webcrypto-core": "^1.7.7" + }, + "engines": { + "node": ">=10.12.0" + } + }, + "node_modules/@peculiar/webcrypto/node_modules/@peculiar/asn1-schema": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/@peculiar/asn1-schema/-/asn1-schema-2.3.8.tgz", + "integrity": "sha512-ULB1XqHKx1WBU/tTFIA+uARuRoBVZ4pNdOA878RDrRbBfBGcSzi5HBkdScC6ZbHn8z7L8gmKCgPC1LHRrP46tA==", + "dependencies": { + "asn1js": "^3.0.5", + "pvtsutils": "^1.3.5", + "tslib": "^2.6.2" + } + }, + "node_modules/@peculiar/webcrypto/node_modules/@peculiar/json-schema": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/@peculiar/json-schema/-/json-schema-1.1.12.tgz", + "integrity": "sha512-coUfuoMeIB7B8/NMekxaDzLhaYmp0HZNPEjYRm9goRou8UZIC3z21s0sL9AWoCw4EG876QyO3kYrc61WNF9B/w==", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/@peculiar/webcrypto/node_modules/asn1js": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/asn1js/-/asn1js-3.0.5.tgz", + "integrity": "sha512-FVnvrKJwpt9LP2lAMl8qZswRNm3T4q9CON+bxldk2iwk3FFpuwhx2FfinyitizWHsVYyaY+y5JzDR0rCMV5yTQ==", + "dependencies": { + "pvtsutils": "^1.3.2", + "pvutils": "^1.1.3", + "tslib": "^2.4.0" + }, + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/@peculiar/webcrypto/node_modules/pvtsutils": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/pvtsutils/-/pvtsutils-1.3.5.tgz", + "integrity": "sha512-ARvb14YB9Nm2Xi6nBq1ZX6dAM0FsJnuk+31aUp4TrcZEdKUlSqOqsxJHUPJDNE3qiIp+iUPEIeR6Je/tgV7zsA==", + "dependencies": { + "tslib": "^2.6.1" + } + }, + "node_modules/@peculiar/webcrypto/node_modules/pvutils": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/pvutils/-/pvutils-1.1.3.tgz", + "integrity": "sha512-pMpnA0qRdFp32b1sJl1wOJNxZLQ2cbQx+k6tjNtZ8CpvVhNqEPRgivZ2WOUev2YMajecdH7ctUPDvEe87nariQ==", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@peculiar/webcrypto/node_modules/tslib": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" + }, + "node_modules/@peculiar/webcrypto/node_modules/webcrypto-core": { + "version": "1.7.7", + "resolved": "https://registry.npmjs.org/webcrypto-core/-/webcrypto-core-1.7.7.tgz", + "integrity": "sha512-7FjigXNsBfopEj+5DV2nhNpfic2vumtjjgPmeDKk45z+MJwXKKfhPB7118Pfzrmh4jqOMST6Ch37iPAHoImg5g==", + "dependencies": { + "@peculiar/asn1-schema": "^2.3.6", + "@peculiar/json-schema": "^1.1.12", + "asn1js": "^3.0.1", + "pvtsutils": "^1.3.2", + "tslib": "^2.4.0" + } + }, "node_modules/@sinonjs/commons": { "version": "1.8.6", "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.6.tgz", @@ -6982,6 +7064,78 @@ "@jridgewell/sourcemap-codec": "1.4.14" } }, + "@peculiar/webcrypto": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/@peculiar/webcrypto/-/webcrypto-1.4.3.tgz", + "integrity": "sha512-VtaY4spKTdN5LjJ04im/d/joXuvLbQdgy5Z4DXF4MFZhQ+MTrejbNMkfZBp1Bs3O5+bFqnJgyGdPuZQflvIa5A==", + "requires": { + "@peculiar/asn1-schema": "^2.3.6", + "@peculiar/json-schema": "^1.1.12", + "pvtsutils": "^1.3.2", + "tslib": "^2.5.0", + "webcrypto-core": "^1.7.7" + }, + "dependencies": { + "@peculiar/asn1-schema": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/@peculiar/asn1-schema/-/asn1-schema-2.3.8.tgz", + "integrity": "sha512-ULB1XqHKx1WBU/tTFIA+uARuRoBVZ4pNdOA878RDrRbBfBGcSzi5HBkdScC6ZbHn8z7L8gmKCgPC1LHRrP46tA==", + "requires": { + "asn1js": "^3.0.5", + "pvtsutils": "^1.3.5", + "tslib": "^2.6.2" + } + }, + "@peculiar/json-schema": { + "version": "1.1.12", + "resolved": "https://registry.npmjs.org/@peculiar/json-schema/-/json-schema-1.1.12.tgz", + "integrity": "sha512-coUfuoMeIB7B8/NMekxaDzLhaYmp0HZNPEjYRm9goRou8UZIC3z21s0sL9AWoCw4EG876QyO3kYrc61WNF9B/w==", + "requires": { + "tslib": "^2.0.0" + } + }, + "asn1js": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/asn1js/-/asn1js-3.0.5.tgz", + "integrity": "sha512-FVnvrKJwpt9LP2lAMl8qZswRNm3T4q9CON+bxldk2iwk3FFpuwhx2FfinyitizWHsVYyaY+y5JzDR0rCMV5yTQ==", + "requires": { + "pvtsutils": "^1.3.2", + "pvutils": "^1.1.3", + "tslib": "^2.4.0" + } + }, + "pvtsutils": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/pvtsutils/-/pvtsutils-1.3.5.tgz", + "integrity": "sha512-ARvb14YB9Nm2Xi6nBq1ZX6dAM0FsJnuk+31aUp4TrcZEdKUlSqOqsxJHUPJDNE3qiIp+iUPEIeR6Je/tgV7zsA==", + "requires": { + "tslib": "^2.6.1" + } + }, + "pvutils": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/pvutils/-/pvutils-1.1.3.tgz", + "integrity": "sha512-pMpnA0qRdFp32b1sJl1wOJNxZLQ2cbQx+k6tjNtZ8CpvVhNqEPRgivZ2WOUev2YMajecdH7ctUPDvEe87nariQ==" + }, + "tslib": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==" + }, + "webcrypto-core": { + "version": "1.7.7", + "resolved": "https://registry.npmjs.org/webcrypto-core/-/webcrypto-core-1.7.7.tgz", + "integrity": "sha512-7FjigXNsBfopEj+5DV2nhNpfic2vumtjjgPmeDKk45z+MJwXKKfhPB7118Pfzrmh4jqOMST6Ch37iPAHoImg5g==", + "requires": { + "@peculiar/asn1-schema": "^2.3.6", + "@peculiar/json-schema": "^1.1.12", + "asn1js": "^3.0.1", + "pvtsutils": "^1.3.2", + "tslib": "^2.4.0" + } + } + } + }, "@sinonjs/commons": { "version": "1.8.6", "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.6.tgz", diff --git a/packages/common/src/auth.test.ts b/packages/common/src/auth.test.ts new file mode 100644 index 0000000000..3d00770d7b --- /dev/null +++ b/packages/common/src/auth.test.ts @@ -0,0 +1,19 @@ +import { encodeSecret, verifyToken } from './auth' + +describe('Auth', () => { + it('correctly create secret, encode and decode', () => { + const secret = 'secret' + const token = encodeSecret(secret) + const decodedSecret = verifyToken(secret, token) + + expect(decodedSecret).toBeTruthy() + }) + + it('create token with wrong secret', () => { + const secret = 'secret' + const token = encodeSecret('test') + const decodedSecret = verifyToken(secret, token) + + expect(decodedSecret).toBeFalsy() + }) +}) diff --git a/packages/common/src/auth.ts b/packages/common/src/auth.ts new file mode 100644 index 0000000000..41bd90449e --- /dev/null +++ b/packages/common/src/auth.ts @@ -0,0 +1,6 @@ +export const encodeSecret = (secret: string) => Buffer.from(secret).toString('base64') + +export const verifyToken = (secret: string, token: string): boolean => { + const decoded = Buffer.from(token, 'base64').toString('ascii') + return decoded === secret +} diff --git a/packages/common/src/index.ts b/packages/common/src/index.ts index 19dcfce623..f15cd34520 100644 --- a/packages/common/src/index.ts +++ b/packages/common/src/index.ts @@ -9,4 +9,5 @@ export * from './naming' export * from './fileData' export * from './libp2p' export * from './tests' +export * from './auth' export * from './messages' diff --git a/packages/desktop/CHANGELOG.md b/packages/desktop/CHANGELOG.md index 96cee6fc28..89971d9147 100644 --- a/packages/desktop/CHANGELOG.md +++ b/packages/desktop/CHANGELOG.md @@ -63,14 +63,6 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline ## [2.0.3-alpha.2](https://github.com/TryQuiet/quiet/compare/@quiet/desktop@2.0.3-alpha.1...@quiet/desktop@2.0.3-alpha.2) (2023-11-09) -### Bug Fixes - -* trigger desktop ([2898bee](https://github.com/TryQuiet/quiet/commit/2898bee80bbf2f16cbda67281a29e47716faa77c)) - - - - - ## [2.0.3-alpha.1](https://github.com/TryQuiet/quiet/compare/@quiet/desktop@2.0.3-alpha.0...@quiet/desktop@2.0.3-alpha.1) (2023-11-08) **Note:** Version bump only for package @quiet/desktop @@ -159,6 +151,19 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline ## [2.0.1-alpha.1](https://github.com/TryQuiet/quiet/compare/quiet@2.0.1-alpha.0...quiet@2.0.1-alpha.1) (2023-09-25) + +## [1.9.7](https://github.com/TryQuiet/quiet/compare/quiet@1.9.6...quiet@1.9.7) (2023-11-17) + + +### Bug Fixes + +* trigger desktop ([2898bee](https://github.com/TryQuiet/quiet/commit/2898bee80bbf2f16cbda67281a29e47716faa77c)) +* pass team id for notarization ([ab86fc0](https://github.com/TryQuiet/quiet/commit/ab86fc0cefd5d8b3715712a4dd234bbe45f18cc2)) + + + +## [1.9.6](https://github.com/TryQuiet/quiet/compare/quiet@1.9.5...quiet@1.9.6) (2023-11-09) + **Note:** Version bump only for package quiet diff --git a/packages/desktop/scripts/notarize.js b/packages/desktop/scripts/notarize.js index f60c6b64e1..59b99c1ea0 100644 --- a/packages/desktop/scripts/notarize.js +++ b/packages/desktop/scripts/notarize.js @@ -2,24 +2,23 @@ const { notarize } = require('@electron/notarize') exports.default = async function notarizing(context) { - const { electronPlatformName, appOutDir } = context + const { electronPlatformName } = context + if (electronPlatformName !== 'darwin' || process.env.IS_E2E) { console.log('skipping notarization') return } - const appName = context.packager.appInfo.productFilename - console.log('notarization start') - try { - const response = await notarize({ - appBundleId: 'com.yourcompany.yourAppId', - appPath: `${appOutDir}/${appName}.app`, - appleId: process.env.APPLEID, - appleIdPassword: process.env.APPLEIDPASS - }) - console.log('notarization done') - return response - } catch (e) { - console.error(e) - } + console.log('notarization started') + + const response = await notarize({ + tool: 'notarytool', + appleId: process.env.APPLE_ID, + appleIdPassword: process.env.APPLE_ID_PASS, + teamId: process.env.APPLE_TEAM_ID + }) + + console.log('notarization done') + + return response } diff --git a/packages/desktop/src/main/main.ts b/packages/desktop/src/main/main.ts index c6c74db976..24174c5dd9 100644 --- a/packages/desktop/src/main/main.ts +++ b/packages/desktop/src/main/main.ts @@ -102,6 +102,8 @@ setEngine( }) ) +const SOCKET_IO_SECRET = webcrypto.getRandomValues(new Uint32Array(5)).join('') + export const isBrowserWindow = (window: BrowserWindow | null): window is BrowserWindow => { return window instanceof BrowserWindow } @@ -208,7 +210,7 @@ export const createWindow = async () => { mainWindow.loadURL( url.format({ pathname: path.join(__dirname, './index.html'), - search: `dataPort=${ports.dataServer}`, + search: `dataPort=${ports.dataServer}&socketIOSecret=${SOCKET_IO_SECRET}`, protocol: 'file:', slashes: true, hash: '/', @@ -333,6 +335,7 @@ app.on('ready', async () => { await createWindow() mainWindow?.webContents.on('did-finish-load', () => { + mainWindow?.webContents.send('socketIOSecret', SOCKET_IO_SECRET) if (splash && !splash.isDestroyed()) { const [width, height] = splash.getSize() mainWindow?.setSize(width, height) @@ -365,6 +368,8 @@ app.on('ready', async () => { `${process.resourcesPath}`, '-p', 'desktop', + '-scrt', + `${SOCKET_IO_SECRET}`, ] const backendBundlePath = path.normalize(require.resolve('backend-bundle')) diff --git a/packages/desktop/src/renderer/Root.tsx b/packages/desktop/src/renderer/Root.tsx index 608060d374..3cd16b9876 100644 --- a/packages/desktop/src/renderer/Root.tsx +++ b/packages/desktop/src/renderer/Root.tsx @@ -33,6 +33,8 @@ import UnregisteredModalContainer from './components/widgets/userLabel/unregiste import DuplicateModalContainer from './components/widgets/userLabel/duplicate/DuplicateModal.container' import UsernameTakenModalContainer from './components/widgets/usernameTakenModal/UsernameTakenModal.container' import PossibleImpersonationAttackModalContainer from './components/widgets/possibleImpersonationAttackModal/PossibleImpersonationAttackModal.container' +import BreakingChangesWarning from './containers/widgets/breakingChangesWarning/BreakingChangesWarning' +// Trigger lerna export const persistor = persistStore(store) export default () => { @@ -61,6 +63,7 @@ export default () => { + diff --git a/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.stories.tsx b/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.stories.tsx new file mode 100644 index 0000000000..d8c17ce5b9 --- /dev/null +++ b/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.stories.tsx @@ -0,0 +1,53 @@ +import React from 'react' +import { ComponentStory, ComponentMeta } from '@storybook/react' + +import UpdateModal, { UpdateModalProps } from './UpdateModalComponent' + +import { withTheme } from '../../../storybook/decorators' +import theme from '../../../theme' + +import Button from '@mui/material/Button' + +const Template: ComponentStory = args => { + return +} + +const args: UpdateModalProps = { + open: true, + handleClose: function (): void { + console.log('modal closed') + }, + buttons: [ + , + ], + title: 'Software update', + message: 'An update is available for Quiet.', +} + +export const Component = Template.bind({}) + +Component.args = args + +const component: ComponentMeta = { + title: 'Components/UpdateModalComponent', + decorators: [withTheme], + component: UpdateModal, +} + +export default component diff --git a/packages/desktop/src/renderer/components/widgets/update/UpdateModal.test.tsx b/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.test.tsx similarity index 54% rename from packages/desktop/src/renderer/components/widgets/update/UpdateModal.test.tsx rename to packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.test.tsx index d29212be1e..1a968eca97 100644 --- a/packages/desktop/src/renderer/components/widgets/update/UpdateModal.test.tsx +++ b/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.test.tsx @@ -1,11 +1,19 @@ import React from 'react' -import { UpdateModal } from './UpdateModal' import { renderComponent } from '../../../testUtils/renderComponent' +import UpdateModalComponent from './UpdateModalComponent' describe('UpdateModal', () => { it('renders component', () => { - const result = renderComponent() + const result = renderComponent( + + ) expect(result.baseElement).toMatchInlineSnapshot(` { style="width: 600px;" >
-
- -
+
-
-

- Software update -

-
+ Software update +
-
-

- A new update for Quiet is available and will be applied on your next restart. -

-
+ Update is available for Quiet. +

-
- -
-
-
- -
+ class="MuiGrid-root MuiGrid-container MuiGrid-spacing-xs-2 MuiGrid-direction-xs-column css-1bnhfwg-MuiGrid-root" + />
diff --git a/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.tsx b/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.tsx new file mode 100644 index 0000000000..0ffd86edbb --- /dev/null +++ b/packages/desktop/src/renderer/components/widgets/update/UpdateModalComponent.tsx @@ -0,0 +1,79 @@ +import React, { ReactElement } from 'react' + +import { styled } from '@mui/material/styles' + +import Typography from '@mui/material/Typography' +import Grid from '@mui/material/Grid' + +import Icon from '../../ui/Icon/Icon' +import updateIcon from '../../../static/images/updateIcon.svg' +import Modal from '../../ui/Modal/Modal' + +const PREFIX = 'UpdateModal' + +const classes = { + info: `${PREFIX}info`, + updateIcon: `${PREFIX}updateIcon`, + title: `${PREFIX}title`, + message: `${PREFIX}message`, +} + +const StyledModalContent = styled(Grid)(({ theme }) => ({ + backgroundColor: theme.palette.colors.white, + border: 'none', + + [`& .${classes.info}`]: { + marginTop: 38, + }, + + [`& .${classes.updateIcon}`]: { + width: 102, + height: 102, + }, + + [`& .${classes.title}`]: { + marginTop: 24, + marginBottom: 16, + textAlign: 'center', + }, + + [`& .${classes.message}`]: { + marginBottom: 32, + textAlign: 'center', + }, +})) + +export interface UpdateModalProps { + open: boolean + handleClose: () => void + buttons: ReactElement[] + title: string + message: string +} + +export const UpdateModalComponent: React.FC = ({ open, handleClose, buttons, title, message }) => { + return ( + + + + + + + {title} + + + {message} + + + {buttons.map((button, index) => ( + + {button} + + ))} + + + + ) +} + +export default UpdateModalComponent diff --git a/packages/desktop/src/renderer/containers/widgets/breakingChangesWarning/BreakingChangesWarning.tsx b/packages/desktop/src/renderer/containers/widgets/breakingChangesWarning/BreakingChangesWarning.tsx new file mode 100644 index 0000000000..9ef8817dca --- /dev/null +++ b/packages/desktop/src/renderer/containers/widgets/breakingChangesWarning/BreakingChangesWarning.tsx @@ -0,0 +1,63 @@ +import React, { useCallback, useEffect } from 'react' +import { ModalName } from '../../../sagas/modals/modals.types' +import { useModal } from '../../hooks' +import UpdateModalComponent from '../../../components/widgets/update/UpdateModalComponent' + +import Button from '@mui/material/Button' +import theme from '../../../theme' + +import { shell } from 'electron' +import { Site } from '@quiet/common' + +const BreakingChangesWarning = () => { + const modal = useModal(ModalName.breakingChangesWarning) + + const title = 'Update available' + const message = + 'Quiet’s next release makes joining communities faster and more reliable by letting people join when the owner is offline! 🎉 However, these changes are not backwards compatible, so you must re-install Quiet from tryquiet.org and re-create or re-join your community. 😥 This version of Quiet will no longer receive any updates or security fixes, so please re-install soon. We apologize for the inconvenience.' + + const updateAction = useCallback(() => { + shell.openExternal(`${Site.MAIN_PAGE}#Downloads`) + }, []) + + const updateButton = ( + + ) + + const dismissButton = ( + + ) + + return +} + +export default BreakingChangesWarning diff --git a/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.test.tsx b/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.test.tsx new file mode 100644 index 0000000000..25bbb22ed1 --- /dev/null +++ b/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.test.tsx @@ -0,0 +1,35 @@ +import React from 'react' +import { jest } from '@jest/globals' +import { fireEvent, screen } from '@testing-library/dom' +import { prepareStore, renderComponent } from '../../../testUtils' +import { StoreKeys } from '@quiet/state-manager' +import { ModalsInitialState } from '../../../sagas/modals/modals.slice' +import { ModalName } from '../../../sagas/modals/modals.types' +import * as UpdateModal from './UpdateModal' + +describe('Update Modal', () => { + test('triggers app update on button click', async () => { + const { store } = await prepareStore({ + [StoreKeys.Modals]: { + ...new ModalsInitialState(), + [ModalName.applicationUpdate]: { open: true, args: {} }, + }, + }) + + const update = jest.fn() + + const modal = + + // @ts-expect-error + jest.spyOn(UpdateModal, 'mapDispatchToProps').mockImplementation(() => ({ + handleUpdate: update, + })) + + renderComponent(modal, store) + + const button = screen.getByText('Update now') + fireEvent.click(button) + + expect(update).toHaveBeenCalled() + }) +}) diff --git a/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.tsx b/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.tsx index 30096a2c84..b2a54ad345 100644 --- a/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.tsx +++ b/packages/desktop/src/renderer/containers/widgets/update/UpdateModal.tsx @@ -1,11 +1,16 @@ import React from 'react' import { AnyAction, Dispatch, bindActionCreators } from 'redux' import { useDispatch } from 'react-redux' -import UpdateModal from '../../../components/widgets/update/UpdateModal' import updateHandlers from '../../../store/handlers/update' + import { useModal } from '../../hooks' import { ModalName } from '../../../sagas/modals/modals.types' +import UpdateModalComponent from '../../../components/widgets/update/UpdateModalComponent' + +import Button from '@mui/material/Button' +import theme from '../../../theme' + export const mapDispatchToProps = (dispatch: Dispatch) => bindActionCreators( { @@ -17,8 +22,32 @@ export const mapDispatchToProps = (dispatch: Dispatch) => const ApplicationUpdateModal: React.FC = () => { const dispatch = useDispatch() + const actions = mapDispatchToProps(dispatch) const modal = useModal(ModalName.applicationUpdate) - return + + const title = 'Software update' + const message = 'An update is availale for Quiet.' + + const button = ( + + ) + + return } + export default ApplicationUpdateModal diff --git a/packages/desktop/src/renderer/index.tsx b/packages/desktop/src/renderer/index.tsx index 49a82e3f13..3c478402b2 100644 --- a/packages/desktop/src/renderer/index.tsx +++ b/packages/desktop/src/renderer/index.tsx @@ -1,11 +1,10 @@ import React from 'react' import { createRoot } from 'react-dom/client' import { ipcRenderer } from 'electron' - import Root, { persistor } from './Root' import store from './store' import updateHandlers from './store/handlers/update' -import { communities } from '@quiet/state-manager' +import { communities, connection } from '@quiet/state-manager' import { InvitationData } from '@quiet/types' if (window && process.env.DEBUG) { @@ -27,6 +26,10 @@ ipcRenderer.on('invitation', (_event, invitation: { data: InvitationData }) => { store.dispatch(communities.actions.customProtocol(invitation.data)) }) +ipcRenderer.on('socketIOSecret', (_event, socketIOSecret) => { + store.dispatch(connection.actions.setSocketIOSecret(socketIOSecret)) +}) + const container = document.getElementById('root') if (!container) throw new Error('No root html element!') let root = createRoot(container) diff --git a/packages/desktop/src/renderer/sagas/modals/modals.slice.ts b/packages/desktop/src/renderer/sagas/modals/modals.slice.ts index fee515645f..31b217a520 100644 --- a/packages/desktop/src/renderer/sagas/modals/modals.slice.ts +++ b/packages/desktop/src/renderer/sagas/modals/modals.slice.ts @@ -37,7 +37,8 @@ export class ModalsInitialState { [ModalName.unregisteredUsernameModal] = { open: false, args: {} }; [ModalName.duplicatedUsernameModal] = { open: false, args: {} }; [ModalName.usernameTakenModal] = { open: false, args: {} }; - [ModalName.possibleImpersonationAttackModal] = { open: false, args: {} } + [ModalName.possibleImpersonationAttackModal] = { open: false, args: {} }; + [ModalName.breakingChangesWarning] = { open: false, args: {} }; } export const modalsSlice = createSlice({ diff --git a/packages/desktop/src/renderer/sagas/modals/modals.types.ts b/packages/desktop/src/renderer/sagas/modals/modals.types.ts index d79e2518e1..ddc4b9e8b3 100644 --- a/packages/desktop/src/renderer/sagas/modals/modals.types.ts +++ b/packages/desktop/src/renderer/sagas/modals/modals.types.ts @@ -1,5 +1,6 @@ export enum ModalName { applicationUpdate = 'applicationUpdate', + breakingChangesWarning = 'breakingChangesWarning', createChannel = 'createChannel', deleteChannel = 'deleteChannel', accountSettingsModal = 'accountSettingsModal', diff --git a/packages/desktop/src/renderer/sagas/socket/socket.saga.test.ts b/packages/desktop/src/renderer/sagas/socket/socket.saga.test.ts new file mode 100644 index 0000000000..d6c199291f --- /dev/null +++ b/packages/desktop/src/renderer/sagas/socket/socket.saga.test.ts @@ -0,0 +1,41 @@ +import { connection, getFactory, Store } from '@quiet/state-manager' +import { FactoryGirl } from 'factory-girl' +import { expectSaga } from 'redux-saga-test-plan' +import { socketActions, WebsocketConnectionPayload } from '../socket/socket.slice' +import { prepareStore } from '../../testUtils/prepareStore' +import { startConnectionSaga } from './socket.saga' + +describe('Start Connection Saga', () => { + const dataPort = 1234 + let store: Store + let factory: FactoryGirl + + beforeEach(async () => { + store = (await prepareStore()).store + factory = await getFactory(store) + }) + + it('socketIOSecret is null - take setSocketIOSecret', async () => { + const payload: WebsocketConnectionPayload = { + dataPort, + } + + await expectSaga(startConnectionSaga, socketActions.startConnection(payload)) + .withState(store.getState()) + .take(connection.actions.setSocketIOSecret) + .run() + }) + + it('socketIOSecret already exist', async () => { + const payload: WebsocketConnectionPayload = { + dataPort, + } + + store.dispatch(connection.actions.setSocketIOSecret('secret')) + + await expectSaga(startConnectionSaga, socketActions.startConnection(payload)) + .withState(store.getState()) + .not.take(connection.actions.setSocketIOSecret) + .run() + }) +}) diff --git a/packages/desktop/src/renderer/sagas/socket/socket.saga.ts b/packages/desktop/src/renderer/sagas/socket/socket.saga.ts index 03646371b5..99037aed24 100644 --- a/packages/desktop/src/renderer/sagas/socket/socket.saga.ts +++ b/packages/desktop/src/renderer/sagas/socket/socket.saga.ts @@ -1,22 +1,39 @@ import { io, Socket } from 'socket.io-client' -import { all, fork, takeEvery, call, put, cancel, FixedTask } from 'typed-redux-saga' +import { all, fork, takeEvery, call, put, cancel, FixedTask, select, take } from 'typed-redux-saga' import { PayloadAction } from '@reduxjs/toolkit' -import { socket as stateManager, messages } from '@quiet/state-manager' +import { socket as stateManager, messages, connection } from '@quiet/state-manager' import { socketActions } from './socket.slice' import { eventChannel } from 'redux-saga' import { displayMessageNotificationSaga } from '../notifications/notifications.saga' - import logger from '../../logger' +import { encodeSecret } from '@quiet/common' + const log = logger('socket') export function* startConnectionSaga( action: PayloadAction['payload']> ): Generator { - const dataPort = action.payload.dataPort + const { dataPort } = action.payload if (!dataPort) { log.error('About to start connection but no dataPort found') } - const socket = yield* call(io, `http://127.0.0.1:${dataPort}`) + + let socketIOSecret = yield* select(connection.selectors.socketIOSecret) + + if (!socketIOSecret) { + yield* take(connection.actions.setSocketIOSecret) + socketIOSecret = yield* select(connection.selectors.socketIOSecret) + } + + if (!socketIOSecret) return + + const token = encodeSecret(socketIOSecret) + const socket = yield* call(io, `http://127.0.0.1:${dataPort}`, { + withCredentials: true, + extraHeaders: { + authorization: `Basic ${token}`, + }, + }) yield* fork(handleSocketLifecycleActions, socket) // Handle opening/restoring connection diff --git a/packages/desktop/src/renderer/testUtils/prepareStore.ts b/packages/desktop/src/renderer/testUtils/prepareStore.ts index 6a8f15d65c..aff3b9f127 100644 --- a/packages/desktop/src/renderer/testUtils/prepareStore.ts +++ b/packages/desktop/src/renderer/testUtils/prepareStore.ts @@ -119,5 +119,6 @@ function* mockSocketConnectionSaga(socket: MockedSocket): Generator { socket.socketClient.emit('connect') }) }) + yield* put(connection.actions.setSocketIOSecret('socketIOSecret')) yield* put(socketActions.startConnection({ dataPort: 4677 })) } diff --git a/packages/e2e-tests/CHANGELOG.md b/packages/e2e-tests/CHANGELOG.md index 7866fc693b..a40de9a683 100644 --- a/packages/e2e-tests/CHANGELOG.md +++ b/packages/e2e-tests/CHANGELOG.md @@ -127,6 +127,7 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline # [1.9.0-alpha.0](/compare/e2e-tests@1.8.0...e2e-tests@1.9.0-alpha.0) (2023-08-29) +## [1.8.3](https://github.com/TryQuiet/quiet/compare/e2e-tests@1.8.2...e2e-tests@1.8.3) (2023-11-09) **Note:** Version bump only for package e2e-tests diff --git a/packages/e2e-tests/src/tests/backwardsCompatibility.test.ts b/packages/e2e-tests/src/tests/backwardsCompatibility.test.ts index eb463ef500..72cbd8eab9 100644 --- a/packages/e2e-tests/src/tests/backwardsCompatibility.test.ts +++ b/packages/e2e-tests/src/tests/backwardsCompatibility.test.ts @@ -19,8 +19,8 @@ describe.skip('Backwards Compatibility', () => { let generalChannel: Channel let secondChannel: Channel let messagesToCompare: WebElement[] - let sidebar: Sidebar + const dataDir = `e2e_${(Math.random() * 10 ** 18).toString(36)}` const communityName = 'testcommunity' const ownerUsername = 'bob' @@ -28,6 +28,8 @@ describe.skip('Backwards Compatibility', () => { const loopMessages = 'abc'.split('') const newChannelName = 'mid-night-club' + const isAlpha = process.env.FILE_NAME?.toString().includes('alpha') + beforeAll(async () => { ownerAppOldVersion = new App({ dataDir, fileName: 'Quiet-1.2.0-copy.AppImage' }) }) @@ -141,7 +143,7 @@ describe.skip('Backwards Compatibility', () => { await ownerAppNewVersion.open() }) - if (process.env.TEST_MODE) { + if (process.env.TEST_MODE && isAlpha) { it('Close debug modal', async () => { console.log('New version', 2) const debugModal = new DebugModeModal(ownerAppNewVersion.driver) diff --git a/packages/integration-tests/CHANGELOG.md b/packages/integration-tests/CHANGELOG.md index 756e9508a6..d874888668 100644 --- a/packages/integration-tests/CHANGELOG.md +++ b/packages/integration-tests/CHANGELOG.md @@ -140,6 +140,12 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline # [1.10.0-alpha.0](/compare/integration-tests@1.9.0...integration-tests@1.10.0-alpha.0) (2023-08-29) + + +## [1.9.2](https://github.com/TryQuiet/quiet/compare/integration-tests@1.9.1...integration-tests@1.9.2) (2023-11-09) + +**Note:** Version bump only for package integration-tests + ## [1.9.1](https://github.com/TryQuiet/quiet/compare/integration-tests@1.9.0...integration-tests@1.9.1) (2023-09-15) **Note:** Version bump only for package integration-tests diff --git a/packages/mobile/.storybook/index.js b/packages/mobile/.storybook/index.js index 703004a968..2042c91936 100644 --- a/packages/mobile/.storybook/index.js +++ b/packages/mobile/.storybook/index.js @@ -27,6 +27,7 @@ configure(() => { require('../src/components/DeleteChannel/DeleteChannel.stories') require('../src/components/QRCode/QRCode.stories') require('../src/components/Message/Message.stories') + require('../src/components/Notifier/Notifier.stories') require('../src/components/Chat/Chat.stories') require('../src/components/TextWithLink/TextWithLink.stories') require('../src/components/Typography/Typography.stories') diff --git a/packages/mobile/CHANGELOG.md b/packages/mobile/CHANGELOG.md index 425473677f..5940a80639 100644 --- a/packages/mobile/CHANGELOG.md +++ b/packages/mobile/CHANGELOG.md @@ -50,11 +50,13 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline ## [2.0.3-alpha.4](https://github.com/TryQuiet/quiet/compare/@quiet/mobile@2.0.3-alpha.3...@quiet/mobile@2.0.3-alpha.4) (2023-11-13) +## [1.10.10](https://github.com/TryQuiet/quiet/compare/@quiet/mobile@1.10.9...@quiet/mobile@1.10.10) (2023-11-09) ### Features * bump versionCode ([104c656](https://github.com/TryQuiet/quiet/commit/104c6569805efecffcc23a801a8ba91a352966fe)) +* bump versionCode ([08af810](https://github.com/TryQuiet/quiet/commit/08af81032b533beaea800cf1fd53035616c9d5d8)) @@ -346,6 +348,7 @@ See [Conventional Commits](https://conventionalcommits.org) for commit guideline # [1.11.0-alpha.0](/compare/@quiet/mobile@1.10.0...@quiet/mobile@1.11.0-alpha.0) (2023-08-29) +## [1.10.9](https://github.com/TryQuiet/quiet/compare/@quiet/mobile@1.10.8...@quiet/mobile@1.10.9) (2023-11-09) **Note:** Version bump only for package @quiet/mobile diff --git a/packages/mobile/android/app/src/main/java/com/quietmobile/Backend/BackendWorker.kt b/packages/mobile/android/app/src/main/java/com/quietmobile/Backend/BackendWorker.kt index 111310380a..e1a6d5792d 100644 --- a/packages/mobile/android/app/src/main/java/com/quietmobile/Backend/BackendWorker.kt +++ b/packages/mobile/android/app/src/main/java/com/quietmobile/Backend/BackendWorker.kt @@ -1,6 +1,7 @@ package com.quietmobile.Backend; import android.content.Context +import android.util.Base64 import android.util.Log import androidx.core.app.NotificationCompat import androidx.work.CoroutineWorker @@ -26,6 +27,7 @@ import org.json.JSONException import org.json.JSONObject import org.torproject.android.binary.TorResourceInstaller import java.util.concurrent.ThreadLocalRandom +import kotlin.collections.ArrayList class BackendWorker(private val context: Context, workerParams: WorkerParameters) : CoroutineWorker(context, workerParams) { @@ -88,8 +90,10 @@ class BackendWorker(private val context: Context, workerParams: WorkerParameters setForeground(createForegroundInfo()) withContext(Dispatchers.IO) { + // Get and store data port for usage in methods across the app val dataPort = Utils.getOpenPort(11000) + val socketIOSecret = Utils.generateRandomString(20) // Init nodejs project launch { @@ -98,7 +102,7 @@ class BackendWorker(private val context: Context, workerParams: WorkerParameters launch { notificationHandler = NotificationHandler(context) - subscribePushNotifications(dataPort) + subscribePushNotifications(dataPort, socketIOSecret) } launch { @@ -112,7 +116,7 @@ class BackendWorker(private val context: Context, workerParams: WorkerParameters * In any case, websocket won't connect until data server starts listening */ delay(WEBSOCKET_CONNECTION_DELAY) - startWebsocketConnection(dataPort) + startWebsocketConnection(dataPort, socketIOSecret) } val dataPath = Utils.createDirectory(context) @@ -122,7 +126,7 @@ class BackendWorker(private val context: Context, workerParams: WorkerParameters val platform = "mobile" - startNodeProjectWithArguments("bundle.cjs --torBinary $torBinary --dataPath $dataPath --dataPort $dataPort --platform $platform") + startNodeProjectWithArguments("bundle.cjs --torBinary $torBinary --dataPath $dataPath --dataPort $dataPort --platform $platform --socketIOSecret $socketIOSecret") } println("FINISHING BACKEND WORKER") @@ -167,8 +171,14 @@ class BackendWorker(private val context: Context, workerParams: WorkerParameters ) } - private fun subscribePushNotifications(port: Int) { - val webSocketClient = IO.socket("http://localhost:$port") + private fun subscribePushNotifications(port: Int, secret: String) { + val encodedSecret = Base64.encodeToString(secret.toByteArray(Charsets.UTF_8), Base64.NO_WRAP) + val options = IO.Options() + val headers = mutableMapOf>() + headers["Authorization"] = listOf("Basic $encodedSecret") + options.extraHeaders = headers + + val webSocketClient = IO.socket("http://127.0.0.1:$port", options) // Listen for events sent from nodejs webSocketClient.on("pushNotification", onPushNotification) // Client won't connect by itself (`connect()` method has to be called manually) @@ -190,10 +200,10 @@ class BackendWorker(private val context: Context, workerParams: WorkerParameters notificationHandler.notify(message, username) } - private fun startWebsocketConnection(port: Int) { + private fun startWebsocketConnection(port: Int, socketIOSecret: String) { Log.d("WEBSOCKET CONNECTION", "Starting on $port") // Proceed only if data port is defined - val websocketConnectionPayload = WebsocketConnectionPayload(port) + val websocketConnectionPayload = WebsocketConnectionPayload(port, socketIOSecret) CommunicationModule.handleIncomingEvents( CommunicationModule.WEBSOCKET_CONNECTION_CHANNEL, Gson().toJson(websocketConnectionPayload), diff --git a/packages/mobile/android/app/src/main/java/com/quietmobile/Scheme/WebsocketConnectionPayload.kt b/packages/mobile/android/app/src/main/java/com/quietmobile/Scheme/WebsocketConnectionPayload.kt index a5a490284a..ac8d89962c 100644 --- a/packages/mobile/android/app/src/main/java/com/quietmobile/Scheme/WebsocketConnectionPayload.kt +++ b/packages/mobile/android/app/src/main/java/com/quietmobile/Scheme/WebsocketConnectionPayload.kt @@ -1,5 +1,6 @@ package com.quietmobile.Scheme data class WebsocketConnectionPayload ( - val dataPort: Int + val dataPort: Int, + val socketIOSecret: String ) diff --git a/packages/mobile/android/app/src/main/java/com/quietmobile/Utils/Utils.kt b/packages/mobile/android/app/src/main/java/com/quietmobile/Utils/Utils.kt index 58da0628ba..123288a0d6 100644 --- a/packages/mobile/android/app/src/main/java/com/quietmobile/Utils/Utils.kt +++ b/packages/mobile/android/app/src/main/java/com/quietmobile/Utils/Utils.kt @@ -7,6 +7,7 @@ import java.io.* import java.net.ConnectException import java.net.InetSocketAddress import java.net.Socket +import java.security.SecureRandom import kotlin.coroutines.resume import kotlin.coroutines.suspendCoroutine @@ -29,6 +30,19 @@ object Utils { return dataDirectory.absolutePath } + fun generateRandomString(length: Int): String { + val CHARACTERS = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789" + val secureRandom = SecureRandom() + val randomString = StringBuilder(length) + + repeat(length) { + val randomIndex = secureRandom.nextInt(CHARACTERS.length) + randomString.append(CHARACTERS[randomIndex]) + } + + return randomString.toString() + } + suspend fun getOpenPort(starting: Int) = suspendCoroutine { continuation -> val port = checkPort(starting) continuation.resume(port) diff --git a/packages/mobile/assets/icons/update_graphics.png b/packages/mobile/assets/icons/update_graphics.png new file mode 100644 index 0000000000000000000000000000000000000000..7f3c8e281b725b38476f8e2dc6018b866dd3c201 GIT binary patch literal 46275 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 /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; - showEnvVarsInLog = 0; - }; 1827A9DE297828AB00245FD3 /* [CUSTOM NODEJS MOBILE] Mock .node files */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; @@ -5011,7 +4992,7 @@ shellPath = /bin/sh; shellScript = "find \"$CODESIGNING_FOLDER_PATH/nodejs-project/node_modules/\" -name \"python3\" | xargs rm\n"; }; - 1A881985853DC9243FFCAD06 /* [CP] Check Pods Manifest.lock */ = { + 3A0DBD8E832195F1C96149E4 /* [CP] Check Pods Manifest.lock */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; files = ( @@ -5026,48 +5007,48 @@ outputFileListPaths = ( ); outputPaths = ( - "$(DERIVED_FILE_DIR)/Pods-Quiet-checkManifestLockResult.txt", + "$(DERIVED_FILE_DIR)/Pods-Quiet-QuietTests-checkManifestLockResult.txt", ); runOnlyForDeploymentPostprocessing = 0; shellPath = /bin/sh; shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; showEnvVarsInLog = 0; }; - 33EE387FBE8BC810E57CE73A /* [CP] Embed Pods Frameworks */ = { + 75398FB9BE83762F7A4A343B /* [CP] Copy Pods Resources */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; files = ( ); inputFileListPaths = ( - "${PODS_ROOT}/Target Support Files/Pods-Quiet/Pods-Quiet-frameworks-${CONFIGURATION}-input-files.xcfilelist", + "${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-resources-${CONFIGURATION}-input-files.xcfilelist", ); - name = "[CP] Embed Pods Frameworks"; + name = "[CP] Copy Pods Resources"; outputFileListPaths = ( - "${PODS_ROOT}/Target Support Files/Pods-Quiet/Pods-Quiet-frameworks-${CONFIGURATION}-output-files.xcfilelist", + "${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-resources-${CONFIGURATION}-output-files.xcfilelist", ); runOnlyForDeploymentPostprocessing = 0; shellPath = /bin/sh; - shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Quiet/Pods-Quiet-frameworks.sh\"\n"; + shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-resources.sh\"\n"; showEnvVarsInLog = 0; }; - 96FE87B8CF60D56415A6D872 /* [CP] Copy Pods Resources */ = { + 897206F7B8FF9B5DD7F5561D /* [CP] Embed Pods Frameworks */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; files = ( ); inputFileListPaths = ( - "${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-resources-${CONFIGURATION}-input-files.xcfilelist", + "${PODS_ROOT}/Target Support Files/Pods-Quiet/Pods-Quiet-frameworks-${CONFIGURATION}-input-files.xcfilelist", ); - name = "[CP] Copy Pods Resources"; + name = "[CP] Embed Pods Frameworks"; outputFileListPaths = ( - "${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-resources-${CONFIGURATION}-output-files.xcfilelist", + "${PODS_ROOT}/Target Support Files/Pods-Quiet/Pods-Quiet-frameworks-${CONFIGURATION}-output-files.xcfilelist", ); runOnlyForDeploymentPostprocessing = 0; shellPath = /bin/sh; - shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-resources.sh\"\n"; + shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Quiet/Pods-Quiet-frameworks.sh\"\n"; showEnvVarsInLog = 0; }; - BCB5CD734D9D7837D92F6F6A /* [CP] Embed Pods Frameworks */ = { + 906843AE2BDF77EFA0025899 /* [CP] Embed Pods Frameworks */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; files = ( @@ -5084,7 +5065,29 @@ shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-Quiet-QuietTests/Pods-Quiet-QuietTests-frameworks.sh\"\n"; showEnvVarsInLog = 0; }; - C3B19D7F8A37CA1DEC5A8545 /* [CP] Copy Pods Resources */ = { + A66CC1900B2FEB9695F131B5 /* [CP] Check Pods Manifest.lock */ = { + isa = PBXShellScriptBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + ); + inputPaths = ( + "${PODS_PODFILE_DIR_PATH}/Podfile.lock", + "${PODS_ROOT}/Manifest.lock", + ); + name = "[CP] Check Pods Manifest.lock"; + outputFileListPaths = ( + ); + outputPaths = ( + "$(DERIVED_FILE_DIR)/Pods-Quiet-checkManifestLockResult.txt", + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n"; + showEnvVarsInLog = 0; + }; + F76D7A280A8CDA06072C296A /* [CP] Copy Pods Resources */ = { isa = PBXShellScriptBuildPhase; buildActionMask = 2147483647; files = ( @@ -5146,6 +5149,7 @@ 18FD2A3E296F009E00A2B8C0 /* AppDelegate.m in Sources */, 1868BCED292E9212001D6D5E /* NodeRunner.mm in Sources */, 1868C43C2930E255001D6D5E /* CommunicationModule.swift in Sources */, + 180E120B2AEFB7F900804659 /* Utils.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -5162,7 +5166,7 @@ /* Begin XCBuildConfiguration section */ 00E356F61AD99517003FC87E /* Debug */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 262E5B8A9D4174F60C5940B1 /* Pods-Quiet-QuietTests.debug.xcconfig */; + baseConfigurationReference = 20C39A27A55F5BFF2B63BC14 /* Pods-Quiet-QuietTests.debug.xcconfig */; buildSettings = { ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; BUNDLE_LOADER = "$(TEST_HOST)"; @@ -5195,7 +5199,7 @@ }; 00E356F71AD99517003FC87E /* Release */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 9CC7C58C143E70B860940210 /* Pods-Quiet-QuietTests.release.xcconfig */; + baseConfigurationReference = 599527D61D41D1899FC6F87B /* Pods-Quiet-QuietTests.release.xcconfig */; buildSettings = { ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; BUNDLE_LOADER = "$(TEST_HOST)"; @@ -5225,7 +5229,7 @@ }; 13B07F941A680F5B00A75B9A /* Debug */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 149DA46FB928274C5FB2FB01 /* Pods-Quiet.debug.xcconfig */; + baseConfigurationReference = F2C59D3CAFDB714D809D1115 /* Pods-Quiet.debug.xcconfig */; buildSettings = { ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; ARCHS = "$(ARCHS_STANDARD)"; @@ -5319,7 +5323,7 @@ }; 13B07F951A680F5B00A75B9A /* Release */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 56957883430CB6A0A09E6740 /* Pods-Quiet.release.xcconfig */; + baseConfigurationReference = CD39A872C9F80A6BA04B8A4A /* Pods-Quiet.release.xcconfig */; buildSettings = { ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES; ARCHS = "$(ARCHS_STANDARD)"; diff --git a/packages/mobile/ios/Quiet/AppDelegate.h b/packages/mobile/ios/Quiet/AppDelegate.h index 0f44aa8b5e..51fed2c14d 100644 --- a/packages/mobile/ios/Quiet/AppDelegate.h +++ b/packages/mobile/ios/Quiet/AppDelegate.h @@ -12,6 +12,8 @@ @property uint16_t dataPort; +@property NSString *socketIOSecret; + @property NSString *dataPath; @property RCTBridge *bridge; diff --git a/packages/mobile/ios/Quiet/AppDelegate.m b/packages/mobile/ios/Quiet/AppDelegate.m index 9f3984ca96..7706f61020 100644 --- a/packages/mobile/ios/Quiet/AppDelegate.m +++ b/packages/mobile/ios/Quiet/AppDelegate.m @@ -104,7 +104,7 @@ - (void) initWebsocketConnection { NSTimeInterval delayInSeconds = 5; dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC)); dispatch_after(popTime, dispatch_get_main_queue(), ^(void) { - [[self.bridge moduleForName:@"CommunicationModule"] sendDataPortWithPort:self.dataPort]; + [[self.bridge moduleForName:@"CommunicationModule"] sendDataPortWithPort:self.dataPort socketIOSecret:self.socketIOSecret]; }); }); } @@ -114,9 +114,13 @@ - (void) spinupBackend:(BOOL)init { // (1/6) Find ports to use in tor and backend configuration FindFreePort *findFreePort = [FindFreePort new]; + Utils *utils = [Utils new]; - self.dataPort = [findFreePort getFirstStartingFromPort:11000]; + if (self.socketIOSecret == nil) { + self.socketIOSecret = [utils generateSecretWithLength:(20)]; + } + self.dataPort = [findFreePort getFirstStartingFromPort:11000]; uint16_t socksPort = [findFreePort getFirstStartingFromPort:12000]; uint16_t controlPort = [findFreePort getFirstStartingFromPort:14000]; uint16_t httpTunnelPort = [findFreePort getFirstStartingFromPort:16000]; @@ -196,16 +200,17 @@ - (NSData *) getAuthCookieData { - (void) launchBackend:(uint16_t)controlPort:(uint16_t)httpTunnelPort:(NSString *)authCookie { self.nodeJsMobile = [RNNodeJsMobile new]; - [self.nodeJsMobile callStartNodeProject:[NSString stringWithFormat:@"bundle.cjs --dataPort %hu --dataPath %@ --controlPort %hu --httpTunnelPort %hu --authCookie %@ --platform %@", self.dataPort, self.dataPath, controlPort, httpTunnelPort, authCookie, platform]]; + [self.nodeJsMobile callStartNodeProject:[NSString stringWithFormat:@"bundle.cjs --dataPort %hu --dataPath %@ --controlPort %hu --httpTunnelPort %hu --authCookie %@ --platform %@ --socketIOSecret %@", self.dataPort, self.dataPath, controlPort, httpTunnelPort, authCookie, platform, self.socketIOSecret]]; } - (void) reviweServices:(uint16_t)controlPort:(uint16_t)httpTunnelPort:(NSString *)authCookie { NSString * dataPortPayload = [NSString stringWithFormat:@"%@:%hu", @"socketIOPort", self.dataPort]; + NSString * socketIOSecretPayload = [NSString stringWithFormat:@"%@:%@", @"socketIOSecret", self.socketIOSecret]; NSString * controlPortPayload = [NSString stringWithFormat:@"%@:%hu", @"torControlPort", controlPort]; NSString * httpTunnelPortPayload = [NSString stringWithFormat:@"%@:%hu", @"httpTunnelPort", httpTunnelPort]; NSString * authCookiePayload = [NSString stringWithFormat:@"%@:%@", @"authCookie", authCookie]; - NSString * payload = [NSString stringWithFormat:@"%@|%@|%@|%@", dataPortPayload, controlPortPayload, httpTunnelPortPayload, authCookiePayload]; + NSString * payload = [NSString stringWithFormat:@"%@|%@|%@|%@|%@", dataPortPayload, socketIOSecretPayload, controlPortPayload, httpTunnelPortPayload, authCookiePayload]; [self.nodeJsMobile sendMessageToNode:@"open":payload]; } diff --git a/packages/mobile/ios/Utils.swift b/packages/mobile/ios/Utils.swift new file mode 100644 index 0000000000..2c36113f99 --- /dev/null +++ b/packages/mobile/ios/Utils.swift @@ -0,0 +1,23 @@ +@objc(Utils) +class Utils: NSObject { + + @objc + func generateSecret(length: Int) -> String { + let characters = Array("ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789") + var randomString = "" + + for _ in 0.. - - - - - - + + - - @@ -112,6 +109,13 @@ function App(): JSX.Element { component={PossibleImpersonationAttackScreen} name={ScreenNames.PossibleImpersonationAttackScreen} /> + + + + + + + diff --git a/packages/mobile/src/assets.ts b/packages/mobile/src/assets.ts index b2d5be2dac..0970f3374f 100644 --- a/packages/mobile/src/assets.ts +++ b/packages/mobile/src/assets.ts @@ -1,38 +1,40 @@ -import quiet_icon from '../assets/icons/quiet_icon.png' -import quiet_icon_round from '../assets/icons/quiet_icon_round.png' -import icon_send from '../assets/icons/icon_send.png' -import icon_send_disabled from '../assets/icons/icon_send_disabled.png' -import icon_check_white from '../assets/icons/icon_check_white.png' -import check_circle_green from '../assets/icons/check_circle_green.png' -import check_circle_blank from '../assets/icons/check_circle_blank.png' -import username_registered from '../assets/icons/username_registered.png' import arrow_left from '../assets/icons/arrow_left.png' import arrow_right_short from '../assets/icons/arrow_right_short.png' -import icon_warning from '../assets/icons/icon_warning.png' -import icon_close from '../assets/icons/icon_close.png' -import file_document from '../assets/icons/file_document.png' +import check_circle_blank from '../assets/icons/check_circle_blank.png' +import check_circle_green from '../assets/icons/check_circle_green.png' import dots from '../assets/icons/dots.png' import paperclip_gray from '../assets/icons/paperclip_gray.png' +import file_document from '../assets/icons/file_document.png' +import icon_check_white from '../assets/icons/icon_check_white.png' +import icon_close from '../assets/icons/icon_close.png' +import icon_send from '../assets/icons/icon_send.png' +import icon_send_disabled from '../assets/icons/icon_send_disabled.png' +import icon_warning from '../assets/icons/icon_warning.png' +import quiet_icon from '../assets/icons/quiet_icon.png' +import quiet_icon_round from '../assets/icons/quiet_icon_round.png' +import update_graphics from '../assets/icons/update_graphics.png' +import username_registered from '../assets/icons/username_registered.png' /** * @description This assets are for the app. */ export const appImages = { - quiet_icon, - quiet_icon_round, - icon_send, - icon_send_disabled, - icon_check_white, - check_circle_green, - check_circle_blank, - username_registered, arrow_left, arrow_right_short, - icon_warning, - icon_close, - file_document, + check_circle_blank, + check_circle_green, dots, paperclip_gray, + file_document, + icon_check_white, + icon_close, + icon_send, + icon_send_disabled, + icon_warning, + quiet_icon, + quiet_icon_round, + update_graphics, + username_registered, } /** diff --git a/packages/mobile/src/components/Button/Button.component.tsx b/packages/mobile/src/components/Button/Button.component.tsx index 9c9fc97999..0393c7469d 100644 --- a/packages/mobile/src/components/Button/Button.component.tsx +++ b/packages/mobile/src/components/Button/Button.component.tsx @@ -2,10 +2,11 @@ import React, { FC } from 'react' import { TouchableWithoutFeedback, View } from 'react-native' import { ButtonProps } from './Button.types' import * as Progress from 'react-native-progress' + import { Typography } from '../Typography/Typography.component' import { defaultTheme } from '../../styles/themes/default.theme' -export const Button: FC = ({ onPress, title, width, loading, negative, disabled }) => { +export const Button: FC = ({ onPress, title, width, loading, negative, disabled, newDesign }) => { return ( { @@ -22,12 +23,12 @@ export const Button: FC = ({ onPress, title, width, loading, negati borderRadius: 8, justifyContent: 'center', alignItems: 'center', - minHeight: 45, + minHeight: newDesign ? 50 : 45, width, }} > {!loading ? ( - + {title} ) : ( diff --git a/packages/mobile/src/components/Button/Button.stories.tsx b/packages/mobile/src/components/Button/Button.stories.tsx index 2c4cb72cb3..d5ce0a479d 100644 --- a/packages/mobile/src/components/Button/Button.stories.tsx +++ b/packages/mobile/src/components/Button/Button.stories.tsx @@ -1,9 +1,10 @@ -import { storiesOf } from '@storybook/react-native' import React from 'react' +import { storiesOf } from '@storybook/react-native' import { storybookLog } from '../../utils/functions/storybookLog/storybookLog.function' import { Button } from './Button.component' storiesOf('Button', module) .add('Default', () =>