From cc3d76d72ee7a2cb361d7b9f4ab634934958294e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Kurczewski?= Date: Thu, 14 Mar 2024 14:36:41 +0100 Subject: [PATCH] [CP-2490] Backup (#1742) --- .stylelintrc.js | 3 +- apps/mudita-center/src/main.ts | 5 + demo-data/kompakt-img.ts | 2 +- libs/core/__deprecated__/main/main.ts | 593 ------------------ .../components/rest/header/header.test.tsx | 4 + .../components/rest/menu/menu.test.tsx | 4 + .../renderer/locales/default/en-US.json | 29 +- .../__deprecated__/renderer/store/reducers.ts | 2 + .../app-update-step-modal.component.test.tsx | 4 + .../contact-support-flow.container.test.tsx | 4 + .../contacts/contacts.component.test.tsx | 4 + libs/core/core/application.module.ts | 5 +- libs/core/core/builder/result.builder.ts | 8 +- .../apps/base-app/base-app-routes.tsx | 166 ++--- .../apps/base-app/base-app.component.tsx | 13 +- libs/core/core/errors/app-error.ts | 10 +- libs/core/core/errors/index.ts | 1 + .../api-device-initialization-modal-flow.tsx | 9 +- .../actions/set-active-device.action.ts | 4 +- .../actions/delete-files.action.test.ts | 23 +- .../actions/get-files.action.test.ts | 18 +- libs/core/help/components/help.test.tsx | 4 + .../harmony-overview.component.test.tsx | 4 + .../pure-overview.component.test.tsx | 4 + .../components/overview/overview.test.tsx | 4 + .../system-update-text.test.tsx | 4 + .../update-os-flow.component.test.tsx | 4 + .../settings/components/about/about.test.tsx | 4 + .../store/schemas/generate-application-id.ts | 28 + .../settings/store/schemas/settings.schema.ts | 24 +- libs/device/feature/src/index.ts | 3 + libs/device/feature/src/lib/api-module.ts | 21 +- .../feature/src/lib/backup/backup.service.ts | 2 +- .../lib/backup/start-pre-backup.request.ts | 2 +- .../lib/file-manager/file-manager.service.ts | 177 ++++++ .../feature/src/lib/file-manager/index.ts | 11 + .../open-backup-directory.request.ts | 17 + .../read-backup-directory.request.ts | 17 + .../file-manager/read-directory.request.ts | 16 + .../file-manager/save-backup-file.request.ts | 21 + .../src/lib/file-manager/save-file.request.ts | 18 + .../file-transfer/file-transfer.service.ts | 339 ++++++++++ .../src/lib/file-transfer/get-file.request.ts | 21 + .../feature/src/lib/file-transfer/index.ts | 11 + .../lib/file-transfer/send-clear.request.ts | 13 + .../lib/file-transfer/send-file.request.ts | 21 + .../start-pre-get-file.request.ts | 19 + .../start-pre-send-file.request.ts | 26 + .../src/lib/restore/restore.service.ts | 2 +- .../feature/src/lib/service-bridge/index.ts | 6 + .../src/lib/service-bridge/service-bridge.ts | 55 ++ libs/device/feature/tsconfig.json | 3 +- libs/device/models/src/index.ts | 1 + .../models/src/lib/api-request.model.ts | 4 + .../file-transfer/file-transfer-statuses.ts | 9 + .../src/lib/file-transfer/file-transfer.ts | 33 + .../models/src/lib/file-transfer/index.ts | 7 + libs/device/models/src/lib/general-error.ts | 10 + .../api-backup-service-events.ts | 1 + .../api-file-transfer-service-events.ts | 12 + .../file-manager-service-events.ts | 13 + .../src/lib/renderer-to-main-events/index.ts | 2 + libs/generic-view/feature/src/index.ts | 1 + .../feature/src/lib/api-device-modals.tsx | 40 ++ libs/generic-view/store/src/index.ts | 6 + .../store/src/lib/action-names.ts | 15 +- .../store/src/lib/backup/actions.ts | 20 +- .../src/lib/backup/create-backup.action.ts | 166 +++++ .../store/src/lib/backup/reducer.ts | 98 ++- .../lib/backup/refresh-backup-list.action.ts | 60 ++ .../store/src/lib/file-transfer/actions.ts | 29 + .../src/lib/file-transfer/get-file.action.ts | 129 ++++ .../store/src/lib/file-transfer/reducer.ts | 129 ++++ .../src/lib/file-transfer/send-file.action.ts | 91 +++ .../generic-view/store/src/lib/hooks/index.ts | 2 + .../hooks/use-api-serial-port-listeners.ts | 14 +- .../src/lib/hooks/use-app-events-listeners.ts | 27 + .../store/src/lib/hooks/use-backup-list.ts | 24 + .../store/src/lib/hooks/use-outbox.ts | 5 + .../selectors/active-device-configuration.ts | 21 + .../lib/selectors/active-device-features.ts | 15 +- .../src/lib/selectors/backup-location.ts | 18 + .../lib/selectors/backup-process-status.ts | 12 + .../src/lib/selectors/backup-progress.ts | 40 ++ .../store/src/lib/selectors/device-backups.ts | 4 +- .../store/src/lib/selectors/index.ts | 3 + .../store/src/lib/views/reducer.ts | 9 + libs/generic-view/theme/src/lib/color.ts | 1 + libs/generic-view/theme/src/lib/font-size.ts | 2 + .../theme/src/lib/generic-theme-provider.tsx | 18 + .../generic-view/theme/src/lib/line-height.ts | 2 + libs/generic-view/ui/src/index.ts | 3 +- .../lib/buttons/button-base/button-base.tsx | 3 +- .../buttons/button-base/use-button-action.ts | 71 ++- .../ui/src/lib/buttons/button-primary.tsx | 5 +- .../ui/src/lib/buttons/button-secondary.tsx | 10 +- .../ui/src/lib/buttons/button-text.tsx | 34 +- .../ui/src/lib/icon/get-icon.helper.tsx | 16 + libs/generic-view/ui/src/lib/icon/icon.tsx | 15 +- .../ui/src/lib/icon/svg/backup.svg | 8 + .../ui/src/lib/icon/svg/confirm.svg | 4 + .../ui/src/lib/icon/svg/failed.svg | 4 + .../ui/src/lib/icon/svg/folder.svg | 5 + .../ui/src/lib/icon/svg/password-hide.svg | 8 + .../ui/src/lib/icon/svg/password-show.svg | 8 + .../ui/src/lib/icon/svg/settings.svg | 8 + .../ui/src/lib/interactive/form/form.tsx | 24 + .../src/lib/interactive/input/text-input.tsx | 180 ++++++ .../ui/src/lib/interactive/interactive.ts | 6 + .../ui/src/lib/interactive/modal/index.ts | 9 + .../src/lib/interactive/modal/modal-base.tsx | 70 +-- .../lib/interactive/modal/modal-helpers.tsx | 160 +++++ .../ui/src/lib/interactive/modal/modal.tsx | 22 +- .../src/lib/interactive/modal/text-modal.tsx | 33 +- .../interactive/progress-bar/progress-bar.tsx | 85 +++ .../predefined/backup-restore-available.tsx | 2 + .../lib/predefined/backup/backup-create.tsx | 183 ++++++ .../lib/predefined/backup/backup-error.tsx | 49 ++ .../lib/predefined/backup/backup-features.tsx | 88 +++ .../lib/predefined/backup/backup-password.tsx | 141 +++++ .../lib/predefined/backup/backup-progress.tsx | 72 +++ .../lib/predefined/backup/backup-success.tsx | 90 +++ .../src/lib/predefined/overview-predefined.ts | 4 +- .../generic-view/ui/src/lib/shared/button.tsx | 1 + .../utils/src/lib/models/button.types.ts | 11 +- .../utils/src/lib/models/icons.types.ts | 7 + .../views/src/lib/mc-overview/mc-overview.ts | 27 - .../section-backup/backup-create-modal.ts | 35 ++ .../section-backup/section-backup.ts | 12 +- .../section-update/section-update.ts | 1 + .../utils/src/lib/call-renderer.helper.ts | 3 +- .../utils/src/lib/main-event.constant.ts | 4 + libs/system-utils/feature/.babelrc | 20 + libs/system-utils/feature/.eslintrc.json | 18 + libs/system-utils/feature/README.md | 7 + libs/system-utils/feature/jest.config.ts | 11 + libs/system-utils/feature/project.json | 20 + libs/system-utils/feature/src/index.ts | 6 + .../src/lib/directory/directory.service.ts | 26 + .../feature/src/lib/system-utils.module.ts | 16 + libs/system-utils/feature/tsconfig.json | 20 + libs/system-utils/feature/tsconfig.lib.json | 24 + libs/system-utils/feature/tsconfig.spec.json | 20 + libs/system-utils/models/.babelrc | 20 + libs/system-utils/models/.eslintrc.json | 18 + libs/system-utils/models/README.md | 7 + libs/system-utils/models/jest.config.ts | 11 + libs/system-utils/models/project.json | 20 + libs/system-utils/models/src/index.ts | 6 + .../src/lib/directory-service-events.ts | 8 + libs/system-utils/models/tsconfig.json | 20 + libs/system-utils/models/tsconfig.lib.json | 24 + libs/system-utils/models/tsconfig.spec.json | 20 + package-lock.json | 36 +- package.json | 8 +- tsconfig.base.json | 4 +- 156 files changed, 3858 insertions(+), 958 deletions(-) delete mode 100644 libs/core/__deprecated__/main/main.ts create mode 100644 libs/core/settings/store/schemas/generate-application-id.ts create mode 100644 libs/device/feature/src/lib/file-manager/file-manager.service.ts create mode 100644 libs/device/feature/src/lib/file-manager/index.ts create mode 100644 libs/device/feature/src/lib/file-manager/open-backup-directory.request.ts create mode 100644 libs/device/feature/src/lib/file-manager/read-backup-directory.request.ts create mode 100644 libs/device/feature/src/lib/file-manager/read-directory.request.ts create mode 100644 libs/device/feature/src/lib/file-manager/save-backup-file.request.ts create mode 100644 libs/device/feature/src/lib/file-manager/save-file.request.ts create mode 100644 libs/device/feature/src/lib/file-transfer/file-transfer.service.ts create mode 100644 libs/device/feature/src/lib/file-transfer/get-file.request.ts create mode 100644 libs/device/feature/src/lib/file-transfer/index.ts create mode 100644 libs/device/feature/src/lib/file-transfer/send-clear.request.ts create mode 100644 libs/device/feature/src/lib/file-transfer/send-file.request.ts create mode 100644 libs/device/feature/src/lib/file-transfer/start-pre-get-file.request.ts create mode 100644 libs/device/feature/src/lib/file-transfer/start-pre-send-file.request.ts create mode 100644 libs/device/feature/src/lib/service-bridge/index.ts create mode 100644 libs/device/feature/src/lib/service-bridge/service-bridge.ts create mode 100644 libs/device/models/src/lib/file-transfer/file-transfer-statuses.ts create mode 100644 libs/device/models/src/lib/file-transfer/file-transfer.ts create mode 100644 libs/device/models/src/lib/file-transfer/index.ts create mode 100644 libs/device/models/src/lib/renderer-to-main-events/api-file-transfer-service-events.ts create mode 100644 libs/device/models/src/lib/renderer-to-main-events/file-manager-service-events.ts create mode 100644 libs/generic-view/feature/src/lib/api-device-modals.tsx create mode 100644 libs/generic-view/store/src/lib/backup/create-backup.action.ts create mode 100644 libs/generic-view/store/src/lib/backup/refresh-backup-list.action.ts create mode 100644 libs/generic-view/store/src/lib/file-transfer/actions.ts create mode 100644 libs/generic-view/store/src/lib/file-transfer/get-file.action.ts create mode 100644 libs/generic-view/store/src/lib/file-transfer/reducer.ts create mode 100644 libs/generic-view/store/src/lib/file-transfer/send-file.action.ts create mode 100644 libs/generic-view/store/src/lib/hooks/use-app-events-listeners.ts create mode 100644 libs/generic-view/store/src/lib/hooks/use-backup-list.ts create mode 100644 libs/generic-view/store/src/lib/selectors/active-device-configuration.ts create mode 100644 libs/generic-view/store/src/lib/selectors/backup-location.ts create mode 100644 libs/generic-view/store/src/lib/selectors/backup-process-status.ts create mode 100644 libs/generic-view/store/src/lib/selectors/backup-progress.ts create mode 100644 libs/generic-view/ui/src/lib/icon/svg/backup.svg create mode 100644 libs/generic-view/ui/src/lib/icon/svg/confirm.svg create mode 100644 libs/generic-view/ui/src/lib/icon/svg/failed.svg create mode 100644 libs/generic-view/ui/src/lib/icon/svg/folder.svg create mode 100644 libs/generic-view/ui/src/lib/icon/svg/password-hide.svg create mode 100644 libs/generic-view/ui/src/lib/icon/svg/password-show.svg create mode 100644 libs/generic-view/ui/src/lib/icon/svg/settings.svg create mode 100644 libs/generic-view/ui/src/lib/interactive/form/form.tsx create mode 100644 libs/generic-view/ui/src/lib/interactive/input/text-input.tsx create mode 100644 libs/generic-view/ui/src/lib/interactive/modal/index.ts create mode 100644 libs/generic-view/ui/src/lib/interactive/modal/modal-helpers.tsx create mode 100644 libs/generic-view/ui/src/lib/interactive/progress-bar/progress-bar.tsx create mode 100644 libs/generic-view/ui/src/lib/predefined/backup/backup-create.tsx create mode 100644 libs/generic-view/ui/src/lib/predefined/backup/backup-error.tsx create mode 100644 libs/generic-view/ui/src/lib/predefined/backup/backup-features.tsx create mode 100644 libs/generic-view/ui/src/lib/predefined/backup/backup-password.tsx create mode 100644 libs/generic-view/ui/src/lib/predefined/backup/backup-progress.tsx create mode 100644 libs/generic-view/ui/src/lib/predefined/backup/backup-success.tsx create mode 100644 libs/generic-view/views/src/lib/mc-overview/section-backup/backup-create-modal.ts create mode 100644 libs/system-utils/feature/.babelrc create mode 100644 libs/system-utils/feature/.eslintrc.json create mode 100644 libs/system-utils/feature/README.md create mode 100644 libs/system-utils/feature/jest.config.ts create mode 100644 libs/system-utils/feature/project.json create mode 100644 libs/system-utils/feature/src/index.ts create mode 100644 libs/system-utils/feature/src/lib/directory/directory.service.ts create mode 100644 libs/system-utils/feature/src/lib/system-utils.module.ts create mode 100644 libs/system-utils/feature/tsconfig.json create mode 100644 libs/system-utils/feature/tsconfig.lib.json create mode 100644 libs/system-utils/feature/tsconfig.spec.json create mode 100644 libs/system-utils/models/.babelrc create mode 100644 libs/system-utils/models/.eslintrc.json create mode 100644 libs/system-utils/models/README.md create mode 100644 libs/system-utils/models/jest.config.ts create mode 100644 libs/system-utils/models/project.json create mode 100644 libs/system-utils/models/src/index.ts create mode 100644 libs/system-utils/models/src/lib/directory-service-events.ts create mode 100644 libs/system-utils/models/tsconfig.json create mode 100644 libs/system-utils/models/tsconfig.lib.json create mode 100644 libs/system-utils/models/tsconfig.spec.json diff --git a/.stylelintrc.js b/.stylelintrc.js index d2bc3333a..c7f62c7ef 100644 --- a/.stylelintrc.js +++ b/.stylelintrc.js @@ -2,5 +2,6 @@ module.exports = { extends: "@mudita/stylelint-config", rules: { "no-descending-specificity": null, - } + "selector-type-no-unknown": [true, { ignoreTypes: ["$dummyValue"] }], + }, } diff --git a/apps/mudita-center/src/main.ts b/apps/mudita-center/src/main.ts index 92a1be41c..523f01e75 100644 --- a/apps/mudita-center/src/main.ts +++ b/apps/mudita-center/src/main.ts @@ -81,6 +81,7 @@ import installExtension, { REDUX_DEVTOOLS, REACT_DEVELOPER_TOOLS, } from "electron-devtools-installer" +import { AppEvents, callRenderer } from "shared/utils" // AUTO DISABLED - fix me if you like :) // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call @@ -190,6 +191,10 @@ const createWindow = async () => { app.exit() }) + win.on("focus", () => { + callRenderer(AppEvents.WindowFocused) + }) + new MetadataInitializer(metadataStore).init() const registerDownloadListener = createDownloadListenerRegistrar(win) diff --git a/demo-data/kompakt-img.ts b/demo-data/kompakt-img.ts index 2f681f1af..8f0d3660c 100644 --- a/demo-data/kompakt-img.ts +++ b/demo-data/kompakt-img.ts @@ -1,2 +1,2 @@ export const kompaktImg = - "" + "" diff --git a/libs/core/__deprecated__/main/main.ts b/libs/core/__deprecated__/main/main.ts deleted file mode 100644 index ecafc194c..000000000 --- a/libs/core/__deprecated__/main/main.ts +++ /dev/null @@ -1,593 +0,0 @@ -/** - * Copyright (c) Mudita sp. z o.o. All rights reserved. - * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md - */ - -import "reflect-metadata" -import { check as checkPort } from "tcp-port-used" -import { - app, - BrowserWindow, - BrowserWindowConstructorOptions, - shell, -} from "electron" -import { ipcMain } from "electron-better-ipc" -import * as path from "path" -import * as url from "url" -import packageInfo from "Root/apps/mudita-center/package.json" -import registerPureOsDownloadListener from "Core/__deprecated__/main/functions/register-pure-os-download-listener" -import registerNewsListener from "Core/__deprecated__/main/functions/register-news-listener/register-news-listener" -import registerAppLogsListeners from "Core/__deprecated__/main/functions/register-app-logs-listener" -import registerContactsExportListener from "Core/contacts/backend/export-contacts" -import registerWriteFileListener from "Core/__deprecated__/main/functions/register-write-file-listener" -import registerCopyFileListener from "Core/__deprecated__/main/functions/register-copy-file-listener" -import registerWriteGzipListener from "Core/__deprecated__/main/functions/register-write-gzip-listener" -import registerRmdirListener from "Core/__deprecated__/main/functions/register-rmdir-listener" -import registerArchiveFilesListener from "Core/__deprecated__/main/functions/register-archive-files-listener" -import createDownloadListenerRegistrar from "Core/__deprecated__/main/functions/create-download-listener-registrar" -import { - registerDownloadHelpHandler, - removeDownloadHelpHandler, -} from "Core/__deprecated__/main/functions/download-help-handler" -import { - registerSetHelpStoreHandler, - removeSetHelpStoreHandler, -} from "Core/__deprecated__/main/functions/set-help-store-handler" -import { - registerGetHelpStoreHandler, - removeGetHelpStoreHandler, -} from "Core/__deprecated__/main/functions/get-help-store-handler" -import { GoogleAuthActions } from "Core/__deprecated__/common/enums/google-auth-actions.enum" -import { - authServerPort, - createAuthServer, - killAuthServer, -} from "Core/__deprecated__/main/auth-server" -import logger from "Core/__deprecated__/main/utils/logger" -import { Scope } from "Core/__deprecated__/renderer/models/external-providers/google/google.interface" -import { OutlookAuthActions } from "Core/__deprecated__/common/enums/outlook-auth-actions.enum" -import { - clientId, - redirectUrl, -} from "Core/__deprecated__/renderer/models/external-providers/outlook/outlook.constants" -import { TokenRequester } from "Core/__deprecated__/renderer/models/external-providers/outlook/token-requester" -import { - GOOGLE_AUTH_WINDOW_SIZE, - WINDOW_SIZE, - DEFAULT_WINDOWS_SIZE, -} from "./config" -import autoupdate, { mockAutoupdate } from "./autoupdate" -import { - URL_MAIN, - URL_OVERVIEW, -} from "Core/__deprecated__/renderer/constants/urls" -import { Mode } from "Core/__deprecated__/common/enums/mode.enum" -import { HelpActions } from "Core/__deprecated__/common/enums/help-actions.enum" -import { AboutActions } from "Core/__deprecated__/common/enums/about-actions.enum" -import { PureSystemActions } from "Core/__deprecated__/common/enums/pure-system-actions.enum" -import { BrowserActions } from "Core/__deprecated__/common/enums/browser-actions.enum" -import { - createMetadataStore, - MetadataStore, - MetadataInitializer, - registerMetadataAllGetValueListener, - registerMetadataGetValueListener, - registerMetadataSetValueListener, -} from "Core/metadata" -import { registerOsUpdateAlreadyDownloadedCheck } from "Core/update/requests" -import { createSettingsService } from "Core/settings/containers/settings.container" -import { ApplicationModule } from "Core/core/application.module" -import registerExternalUsageDevice from "Core/device/listeners/register-external-usage-device.listner" -import installExtension, { - REDUX_DEVTOOLS, - REACT_DEVELOPER_TOOLS, -} from "electron-devtools-installer" -import isPrereleaseSet from "Core/utils/is-prerelease-set" - -// AUTO DISABLED - fix me if you like :) -// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call -require("dotenv").config() - -// FIXME: electron v12 added changes to the remote module. This module has many subtle pitfalls. -// There is almost always a better way to accomplish your task than using this module. -// You can read more in https://github.com/electron/remote#migrating-from-remote -require("@electron/remote/main").initialize() - -logger.info("Starting the app") - -let win: BrowserWindow | null -let helpWindow: BrowserWindow | null = null -let googleAuthWindow: BrowserWindow | null = null -let outlookAuthWindow: BrowserWindow | null = null -const licenseWindow: BrowserWindow | null = null -const termsWindow: BrowserWindow | null = null -const policyWindow: BrowserWindow | null = null -const metadataStore: MetadataStore = createMetadataStore() - -// Disabling browser security features -// to address CORS issue between local and remote servers. -// To be handled as part of ticket https://appnroll.atlassian.net/browse/CP-2242 -app.commandLine.appendSwitch( - "disable-features", - "BlockInsecurePrivateNetworkRequests,PrivateNetworkAccessSendPreflights" -) - -const gotTheLock = app.requestSingleInstanceLock() - -// Fetch and log all errors -process.on("uncaughtException", (error) => { - logger.error(error) - // TODO: Add contact support modal -}) - -const productionEnvironment = process.env.NODE_ENV === "production" -const commonWindowOptions: BrowserWindowConstructorOptions = { - resizable: true, - fullscreen: false, - fullscreenable: true, - useContentSize: true, - webPreferences: { - nodeIntegration: true, - webSecurity: false, - // FIXME: electron v12 throw error: 'Require' is not defined. `contextIsolation` default value is changed to `true`. - // You can read more in https://www.electronjs.org/blog/electron-12-0#breaking-changes - contextIsolation: false, - }, -} -const getWindowOptions = ( - extendedWindowOptions?: BrowserWindowConstructorOptions -) => ({ - ...commonWindowOptions, - ...extendedWindowOptions, -}) - -const installElectronDevToolExtensions = async () => { - try { - await installExtension([REDUX_DEVTOOLS, REACT_DEVELOPER_TOOLS], { - loadExtensionOptions: { - allowFileAccess: true, - }, - }) - console.info(`[INFO] Successfully added devtools extensions`) - } catch (err) { - console.warn( - "[WARN] An error occurred while trying to add devtools extensions:\n", - err - ) - } -} - -const createWindow = async () => { - const version = packageInfo.version - const prereleaseSet = isPrereleaseSet(version) - const title = prereleaseSet ? `Mudita Center - ${version}` : "Mudita Center" - - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-call - ;(global as any).__static = require("path") - .join(__dirname, "/static") - .replace(/\\/g, "\\\\") - - win = new BrowserWindow( - getWindowOptions({ - minWidth: WINDOW_SIZE.minWidth, - width: WINDOW_SIZE.width, - minHeight: WINDOW_SIZE.minHeight, - height: WINDOW_SIZE.height, - title, - }) - ) - // FIXME: electron v12 added changes to the remote module. This module has many subtle pitfalls. - // There is almost always a better way to accomplish your task than using this module. - // You can read more in https://github.com/electron/remote#migrating-from-remote - require("@electron/remote/main").enable(win.webContents) - win.removeMenu() - - win.webContents.on("before-input-event", (event, input) => { - if ((input.control || input.meta) && input.key.toLowerCase() === "r") { - event.preventDefault() - } - }) - - win.on("closed", () => { - win = null - app.exit() - }) - - new MetadataInitializer(metadataStore).init() - - const registerDownloadListener = createDownloadListenerRegistrar(win) - - const settingsService = createSettingsService() - settingsService.init() - - const appModules = new ApplicationModule(ipcMain, win) - - registerPureOsDownloadListener(registerDownloadListener) - registerOsUpdateAlreadyDownloadedCheck() - registerNewsListener() - registerAppLogsListeners() - registerContactsExportListener() - registerWriteFileListener() - registerCopyFileListener() - registerRmdirListener() - registerWriteGzipListener() - registerArchiveFilesListener() - registerMetadataAllGetValueListener() - registerMetadataGetValueListener() - registerMetadataSetValueListener() - registerExternalUsageDevice() - - if (productionEnvironment) { - win.setMenuBarVisibility(false) - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - win.loadURL( - url.format({ - pathname: path.join(__dirname, "index.html"), - protocol: "file:", - slashes: true, - }) - ) - autoupdate(win) - } else { - await installElectronDevToolExtensions() - process.env.ELECTRON_DISABLE_SECURITY_WARNINGS = "1" - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - win.loadURL(`http://localhost:2003`) - mockAutoupdate(win) - } - - win.webContents.setWindowOpenHandler(({ url }) => { - shell.openExternal(url) - return { - action: "deny", - overrideBrowserWindowOptions: {}, - } - }) - - if (productionEnvironment) { - win.webContents.once("dom-ready", () => { - appModules.lateInitialization() - }) - } else { - // Open DevTools, see https://github.com/electron/electron/issues/12438 for why we wait for dom-ready - win.webContents.once("dom-ready", () => { - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-non-null-assertion - win!.webContents.openDevTools() - appModules.lateInitialization() - }) - - win.webContents.once("dom-ready", () => { - win!.webContents.once("devtools-opened", () => { - win!.focus() - }) - win!.webContents.openDevTools() - }) - } - - logger.updateMetadata() -} - -if (!gotTheLock) { - app.quit() -} else { - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-misused-promises - app.on("ready", createWindow) - - app.on("window-all-closed", () => { - app.quit() - }) - - app.on("activate", () => { - if (win === null) { - void createWindow() - } - }) -} - -ipcMain.answerRenderer(HelpActions.OpenWindow, () => { - const title = "Mudita Center - Help" - if (helpWindow === null) { - helpWindow = new BrowserWindow( - getWindowOptions({ - width: DEFAULT_WINDOWS_SIZE.width, - height: DEFAULT_WINDOWS_SIZE.height, - title, - }) - ) - // FIXME: electron v12 added changes to the remote module. This module has many subtle pitfalls. - // There is almost always a better way to accomplish your task than using this module. - // You can read more in https://github.com/electron/remote#migrating-from-remote - require("@electron/remote/main").enable(helpWindow.webContents) - helpWindow.removeMenu() - - helpWindow.on("closed", () => { - removeDownloadHelpHandler() - removeSetHelpStoreHandler() - removeGetHelpStoreHandler() - helpWindow = null - }) - - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - helpWindow.loadURL( - !productionEnvironment - ? `http://localhost:2003/?mode=${Mode.Help}#${URL_MAIN.help}` - : url.format({ - pathname: path.join(__dirname, "index.html"), - protocol: "file:", - slashes: true, - hash: URL_MAIN.help, - search: `?mode=${Mode.Help}`, - }) - ) - registerDownloadHelpHandler() - registerSetHelpStoreHandler() - registerGetHelpStoreHandler() - } else { - helpWindow.show() - } -}) - -const createOpenWindowListener = ( - channel: string, - mode: string, - urlMain: string, - title: string, - newWindow: BrowserWindow | null = null -) => { - ipcMain.answerRenderer(channel, async () => { - if (newWindow === null) { - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/await-thenable - newWindow = new BrowserWindow( - getWindowOptions({ - width: DEFAULT_WINDOWS_SIZE.width, - height: DEFAULT_WINDOWS_SIZE.height, - title, - }) - ) - // FIXME: electron v12 added changes to the remote module. This module has many subtle pitfalls. - // There is almost always a better way to accomplish your task than using this module. - // You can read more in https://github.com/electron/remote#migrating-from-remote - require("@electron/remote/main").enable(newWindow.webContents) - newWindow.removeMenu() - - newWindow.on("closed", () => { - newWindow = null - }) - - await newWindow.loadURL( - !productionEnvironment - ? `http://localhost:2003/?mode=${mode}#${urlMain}` - : url.format({ - pathname: path.join(__dirname, "index.html"), - protocol: "file:", - slashes: true, - hash: urlMain, - search: `?mode=${mode}`, - }) - ) - - newWindow.webContents.setWindowOpenHandler(({ url }) => { - shell.openExternal(url) - return { - action: "allow", - overrideBrowserWindowOptions: {}, - } - }) - } else { - newWindow.show() - } - }) -} - -ipcMain.answerRenderer(BrowserActions.PolicyOpenBrowser, () => - shell.openExternal( - `${process.env.MUDITA_CENTER_SERVER_URL ?? ""}/privacy-policy-url` - ) -) -ipcMain.answerRenderer(BrowserActions.UpdateOpenBrowser, () => - shell.openExternal("https://mudita.com") -) - -ipcMain.answerRenderer(BrowserActions.AppleOpenBrowser, () => - shell.openExternal( - "https://support.apple.com/en-al/guide/contacts/adrbdcfd32e6/mac#:~:text=In%20the%20Contacts%20app%20on,vcf)%20only." - ) -) - -createOpenWindowListener( - AboutActions.LicenseOpenWindow, - Mode.License, - URL_MAIN.license, - "Mudita Center - License", - licenseWindow -) - -createOpenWindowListener( - AboutActions.TermsOpenWindow, - Mode.TermsOfService, - URL_MAIN.termsOfService, - "Mudita Center - Terms of service", - termsWindow -) - -createOpenWindowListener( - AboutActions.PolicyOpenWindow, - Mode.PrivacyPolicy, - URL_MAIN.privacyPolicy, - "Mudita Center - Privacy policy", - policyWindow -) - -createOpenWindowListener( - PureSystemActions.SarOpenWindow, - Mode.Sar, - URL_OVERVIEW.sar, - "Mudita Center - SAR information", - policyWindow -) - -const createErrorWindow = async (googleAuthWindow: BrowserWindow) => { - return await googleAuthWindow.loadURL( - !productionEnvironment - ? `http://localhost:2003/?mode=${Mode.ServerError}#${URL_MAIN.error}` - : url.format({ - pathname: path.join(__dirname, "index.html"), - protocol: "file:", - slashes: true, - hash: URL_MAIN.error, - search: `?mode=${Mode.ServerError}`, - }) - ) -} - -ipcMain.answerRenderer(GoogleAuthActions.OpenWindow, async (scope: Scope) => { - const title = "Mudita Center - Google Auth" - if (process.env.MUDITA_CENTER_SERVER_URL) { - const cb = (data: string) => { - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - ipcMain.callRenderer( - win as BrowserWindow, - GoogleAuthActions.GotCredentials, - data - ) - } - - if (googleAuthWindow === null) { - googleAuthWindow = new BrowserWindow( - getWindowOptions({ - width: GOOGLE_AUTH_WINDOW_SIZE.width, - height: GOOGLE_AUTH_WINDOW_SIZE.height, - title, - webPreferences: { - nodeIntegration: true, - }, - }) - ) - googleAuthWindow.removeMenu() - - googleAuthWindow.on("close", () => { - void ipcMain.callRenderer( - win as BrowserWindow, - GoogleAuthActions.CloseWindow - ) - googleAuthWindow = null - killAuthServer() - }) - - if (await checkPort(authServerPort)) { - await createErrorWindow(googleAuthWindow) - return - } - - createAuthServer(cb) - - let scopeUrl: string - - switch (scope) { - case Scope.Calendar: - scopeUrl = "https://www.googleapis.com/auth/calendar" - break - case Scope.Contacts: - scopeUrl = "https://www.googleapis.com/auth/contacts" - break - } - const url = `${process.env.MUDITA_CENTER_SERVER_URL}/google-auth-init` - void (await googleAuthWindow.loadURL(`${url}?scope=${scopeUrl}`)) - } else { - googleAuthWindow.show() - } - } else { - console.log("No Google Auth URL defined!") - } -}) - -ipcMain.answerRenderer(GoogleAuthActions.CloseWindow, () => { - killAuthServer() - googleAuthWindow?.close() -}) - -ipcMain.answerRenderer( - OutlookAuthActions.OpenWindow, - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/require-await - async (data: { authorizationUrl: string; scope: string }) => { - const title = "Mudita Center - Outlook Auth" - const { authorizationUrl, scope } = data - if (clientId) { - if (outlookAuthWindow === null) { - outlookAuthWindow = new BrowserWindow( - getWindowOptions({ - width: 600, - height: 600, - title, - }) - ) - outlookAuthWindow.removeMenu() - - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - outlookAuthWindow.loadURL(authorizationUrl) - - outlookAuthWindow.on("close", () => { - void ipcMain.callRenderer( - win as BrowserWindow, - OutlookAuthActions.CloseWindow - ) - outlookAuthWindow = null - }) - - const { - session: { webRequest }, - } = outlookAuthWindow.webContents - webRequest.onBeforeRequest( - { - urls: [`${redirectUrl}*`], - }, // * character is used to "catch all" url params - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-misused-promises - async ({ url }) => { - const code = new URL(url).searchParams.get("code") || "" - try { - const tokenRequester = new TokenRequester() - const tokens = await tokenRequester.requestTokens(code, scope) - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - ipcMain.callRenderer( - win as BrowserWindow, - OutlookAuthActions.GotCredentials, - tokens - ) - } catch (error) { - // AUTO DISABLED - fix me if you like :) - // eslint-disable-next-line @typescript-eslint/no-floating-promises - ipcMain.callRenderer( - win as BrowserWindow, - OutlookAuthActions.GotCredentials, - { error } - ) - } - - outlookAuthWindow?.close() - outlookAuthWindow = null - } - ) - } else { - outlookAuthWindow.show() - } - } else { - logger.info("No Outlook Auth URL defined!") - } - } -) - -ipcMain.answerRenderer(OutlookAuthActions.CloseWindow, () => { - outlookAuthWindow?.close() -}) diff --git a/libs/core/__deprecated__/renderer/components/rest/header/header.test.tsx b/libs/core/__deprecated__/renderer/components/rest/header/header.test.tsx index 27c04aac7..c73fba719 100644 --- a/libs/core/__deprecated__/renderer/components/rest/header/header.test.tsx +++ b/libs/core/__deprecated__/renderer/components/rest/header/header.test.tsx @@ -16,6 +16,10 @@ import store from "Core/__deprecated__/renderer/store" jest.mock("Core/feature-flags") +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + test("matches snapshot without tabs", () => { const currentLocation = "/overview" const { container } = renderWithThemeAndIntl( diff --git a/libs/core/__deprecated__/renderer/components/rest/menu/menu.test.tsx b/libs/core/__deprecated__/renderer/components/rest/menu/menu.test.tsx index 72541e13c..93acab9dd 100644 --- a/libs/core/__deprecated__/renderer/components/rest/menu/menu.test.tsx +++ b/libs/core/__deprecated__/renderer/components/rest/menu/menu.test.tsx @@ -27,6 +27,10 @@ import { } from "Core/device-initialization/reducers/device-initialization.interface" import { DeviceManagerState } from "Core/device-manager/reducers/device-manager.interface" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("Core/feature-flags") type Props = MenuProps diff --git a/libs/core/__deprecated__/renderer/locales/default/en-US.json b/libs/core/__deprecated__/renderer/locales/default/en-US.json index 0a39cadf3..eb857aba5 100644 --- a/libs/core/__deprecated__/renderer/locales/default/en-US.json +++ b/libs/core/__deprecated__/renderer/locales/default/en-US.json @@ -862,5 +862,32 @@ "module.genericBackup.createButtonLabel": "Create backup", "module.genericBackup.restoreButtonLabel": "Restore from backup", "module.genericViews.update.tag": "Update is available ({version}).", - "module.genericViews.update.actionLabel": "You can update it on your device." + "module.genericViews.update.actionLabel": "You can update it on your device.", + "module.genericViews.backup.features.title": "Create backup", + "module.genericViews.backup.features.description": "All backup data stays on your computer.", + "module.genericViews.backup.features.cancelButtonLabel": "Cancel", + "module.genericViews.backup.features.createButtonLabel": "Create backup", + "module.genericViews.backup.password.title": "Create password for backup", + "module.genericViews.backup.password.subtitle": "(optional)", + "module.genericViews.backup.password.description": "You can protect backup with a new password.", + "module.genericViews.backup.password.description2": "* You can't change/recover the password later.", + "module.genericViews.backup.password.passwordPlaceholder": "Password", + "module.genericViews.backup.password.passwordRepeatPlaceholder": "Repeat password", + "module.genericViews.backup.password.passwordRepeatNotMatchingError": "Passwords do not match", + "module.genericViews.backup.password.confirmButtonLabel": "Confirm password", + "module.genericViews.backup.password.skipButtonLabel": "Skip password", + "module.genericViews.backup.progress.title": "Creating backup", + "module.genericViews.backup.progress.description": "Please wait and do not unplug your device from computer.", + "module.genericViews.backup.progress.progressDetails": "This might take a few minutes", + "module.genericViews.backup.progress.progressDetailsForFeature": "Backing up {featureLabel}", + "module.genericViews.backup.success.title": "Backup complete", + "module.genericViews.backup.success.description": "Your data was successfully secured.\nOpen the backup folder to see your backup data or close this window.", + "module.genericViews.backup.success.openBackupButtonLabel": "Open backup folder", + "module.genericViews.backup.success.closeButtonLabel": "Close", + "module.genericViews.backup.failure.title": "Backup failed", + "module.genericViews.backup.failure.defaultErrorMessage": "The backup process was interrupted.", + "module.genericViews.backup.failure.closeButtonLabel": "Close", + "module.genericViews.backup.directoryOpenFailure.title": "Directory open failed", + "module.genericViews.backup.directoryOpenFailure.defaultErrorMessage": "The directory could not be opened.\n\nYou can still try to find the backup in the default location:", + "module.genericViews.backup.directoryOpenFailure.closeButtonLabel": "Close" } diff --git a/libs/core/__deprecated__/renderer/store/reducers.ts b/libs/core/__deprecated__/renderer/store/reducers.ts index 9d3d8435a..bf20841ea 100644 --- a/libs/core/__deprecated__/renderer/store/reducers.ts +++ b/libs/core/__deprecated__/renderer/store/reducers.ts @@ -24,6 +24,7 @@ import { appInitializationReducer } from "Core/app-initialization/reducers/app-i import { deviceManagerReducer } from "Core/device-manager/reducers/device-manager.reducer" import { genericBackupsReducer, + genericFileTransferReducer, genericModalsReducer, genericViewsReducer, } from "generic-view/store" @@ -51,6 +52,7 @@ export const reducers = { genericViews: genericViewsReducer, genericModals: genericModalsReducer, genericBackups: genericBackupsReducer, + genericFileTransfer: genericFileTransferReducer, appState: appStateReducer, } diff --git a/libs/core/__deprecated__/renderer/wrappers/app-update-step-modal/app-update-step-modal.component.test.tsx b/libs/core/__deprecated__/renderer/wrappers/app-update-step-modal/app-update-step-modal.component.test.tsx index aac20a9ef..ad344d4fb 100644 --- a/libs/core/__deprecated__/renderer/wrappers/app-update-step-modal/app-update-step-modal.component.test.tsx +++ b/libs/core/__deprecated__/renderer/wrappers/app-update-step-modal/app-update-step-modal.component.test.tsx @@ -17,6 +17,10 @@ import { AppUpdateStepModalTestIds } from "Core/__deprecated__/renderer/wrappers import { Provider } from "react-redux" import store from "../../store" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + const onCloseMock = jest.fn() const openExternalMock = jest.fn() diff --git a/libs/core/contact-support/containers/contact-support-flow.container.test.tsx b/libs/core/contact-support/containers/contact-support-flow.container.test.tsx index 987be5d2e..3d622c96c 100644 --- a/libs/core/contact-support/containers/contact-support-flow.container.test.tsx +++ b/libs/core/contact-support/containers/contact-support-flow.container.test.tsx @@ -11,6 +11,10 @@ import ModalsManager from "Core/modals-manager/components/modals-manager.contain import ContactSupportFlow from "Core/contact-support/containers/contact-support-flow.container" import { ContactSupportFlowTestIds } from "Core/contact-support/components/contact-support-flow-test-ids.component" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + type Props = ComponentProps const defaultProps: Props = {} diff --git a/libs/core/contacts/components/contacts/contacts.component.test.tsx b/libs/core/contacts/components/contacts/contacts.component.test.tsx index 5a0717a9d..2c8c09856 100644 --- a/libs/core/contacts/components/contacts/contacts.component.test.tsx +++ b/libs/core/contacts/components/contacts/contacts.component.test.tsx @@ -38,6 +38,10 @@ window.IntersectionObserver = jest type Props = ComponentProps +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("@electron/remote", () => ({ Menu: () => ({ popup: jest.fn(), diff --git a/libs/core/core/application.module.ts b/libs/core/core/application.module.ts index 400dbc3a6..e220132f3 100644 --- a/libs/core/core/application.module.ts +++ b/libs/core/core/application.module.ts @@ -46,6 +46,7 @@ import { } from "Core/device-manager/services" import { APIModule } from "device/feature" import { FileSystemDialogModule } from "shared/app-state" +import { SystemUtilsModule } from "system-utils/feature" export class ApplicationModule { public modules: Module[] = [ @@ -89,6 +90,7 @@ export class ApplicationModule { new DeviceResolverService(), this.eventEmitter ) + private systemUtilsModule = new SystemUtilsModule() constructor( private ipc: MainProcessIpc, @@ -105,11 +107,12 @@ export class ApplicationModule { this.initializeInitializer = new InitializeInitializer() this.modules.forEach(this.initModule) - this.apiModule = new APIModule(this.deviceManager) + this.apiModule = new APIModule(this.deviceManager, this.systemUtilsModule) this.controllerInitializer.initialize(this.apiModule.getAPIServices()) this.controllerInitializer.initialize( FileSystemDialogModule.getControllers() ) + this.controllerInitializer.initialize(this.systemUtilsModule.getServices()) } lateInitialization(): void { diff --git a/libs/core/core/builder/result.builder.ts b/libs/core/core/builder/result.builder.ts index 2b585bb36..32ac0cdaf 100644 --- a/libs/core/core/builder/result.builder.ts +++ b/libs/core/core/builder/result.builder.ts @@ -3,11 +3,11 @@ * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md */ -import { AppError } from "Core/core/errors" +import { AppError, AppErrorType } from "Core/core/errors" export type ResultObject< Data, - ErrorType extends string = string, + ErrorType extends AppErrorType = AppErrorType, ErrorData = unknown > = SuccessResult | FailedResult @@ -18,7 +18,7 @@ export class SuccessResult { constructor(public data: Data) {} } -export class FailedResult { +export class FailedResult { public ok: false = false constructor( @@ -32,7 +32,7 @@ export class Result { return new SuccessResult(data) } - static failed( + static failed( error: AppError, data?: Data ): FailedResult { diff --git a/libs/core/core/components/apps/base-app/base-app-routes.tsx b/libs/core/core/components/apps/base-app/base-app-routes.tsx index 94162dd1b..9d8d615c2 100644 --- a/libs/core/core/components/apps/base-app/base-app-routes.tsx +++ b/libs/core/core/components/apps/base-app/base-app-routes.tsx @@ -35,99 +35,101 @@ import ConfiguredDevicesDiscovery from "Core/discovery-device/components/configu import DevicesInitialization from "Core/device-initialization/components/devices-initialization.component" import AvailableDeviceListContainer from "Core/discovery-device/components/available-device-list.container" import DeviceConnecting from "Core/discovery-device/components/device-connecting.component" -import { GenericView } from "generic-view/feature" +import { ApiDeviceModals, GenericView } from "generic-view/feature" import { APIConnectionDemo } from "generic-view/ui" // AUTO DISABLED - fix me if you like :) // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types export default () => ( - - - + <> + + + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + - - - - - - - - - - - - - + + - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + ) diff --git a/libs/core/core/components/apps/base-app/base-app.component.tsx b/libs/core/core/components/apps/base-app/base-app.component.tsx index 6efa9595b..887230e9a 100644 --- a/libs/core/core/components/apps/base-app/base-app.component.tsx +++ b/libs/core/core/components/apps/base-app/base-app.component.tsx @@ -18,8 +18,12 @@ import { useDeviceDetachedEffect } from "Core/core/hooks/use-device-detached-eff import { useDeviceConnectFailedEffect } from "Core/core/hooks/use-device-connect-failed-effect" import { useDiscoveryRedirectEffect } from "Core/core/hooks/use-discovery-redirect-effect" import { useRouterListener } from "Core/core/hooks" -import { useAPISerialPortListeners, useOutbox } from "generic-view/store" - +import { + OutboxWrapper, + useAPISerialPortListeners, + useBackupList, + useAppEventsListeners, +} from "generic-view/store" const BaseApp: FunctionComponent = () => { useRouterListener() @@ -33,10 +37,11 @@ const BaseApp: FunctionComponent = () => { useDiscoveryRedirectEffect() // API useAPISerialPortListeners() - useOutbox() - + useAppEventsListeners() + useBackupList() return ( <> + diff --git a/libs/core/core/errors/app-error.ts b/libs/core/core/errors/app-error.ts index f9757e16f..6dd05f26e 100644 --- a/libs/core/core/errors/app-error.ts +++ b/libs/core/core/errors/app-error.ts @@ -3,13 +3,19 @@ * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md */ -export class AppError extends Error { +export type AppErrorType = string | number + +export class AppError< + Type extends AppErrorType = AppErrorType, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + Payload = any +> extends Error { constructor( public readonly type: Type, public readonly message: string = "", // AUTO DISABLED - fix me if you like :) // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types, @typescript-eslint/no-explicit-any - public readonly payload?: any + public readonly payload?: Payload ) { super() diff --git a/libs/core/core/errors/index.ts b/libs/core/core/errors/index.ts index bbd1e4cf1..e66ade5fb 100644 --- a/libs/core/core/errors/index.ts +++ b/libs/core/core/errors/index.ts @@ -4,3 +4,4 @@ */ export { AppError } from "./app-error" +export type { AppErrorType } from "./app-error" diff --git a/libs/core/device-initialization/components/devices-initialization-modal-flows/api-device-initialization-modal-flow.tsx b/libs/core/device-initialization/components/devices-initialization-modal-flows/api-device-initialization-modal-flow.tsx index fbf72cb21..de82c2273 100644 --- a/libs/core/device-initialization/components/devices-initialization-modal-flows/api-device-initialization-modal-flow.tsx +++ b/libs/core/device-initialization/components/devices-initialization-modal-flows/api-device-initialization-modal-flow.tsx @@ -19,6 +19,7 @@ import { ApiError } from "device/models" import { intl } from "Core/__deprecated__/renderer/utils/intl" import { defineMessages } from "react-intl" import { + closeButtonStyles, IconButton, ModalBase, ModalCenteredContent, @@ -122,9 +123,9 @@ export const APIDeviceInitializationModalFlow: FunctionComponent = () => { variant={"small"} overlayHidden closeButton={ - + - + } > @@ -151,3 +152,7 @@ const ConnectingText = styled.p` font-weight: ${({ theme }) => theme.fontWeight.bold}; margin: 2.4rem 0 0; ` + +const CloseButton = styled(IconButton)` + ${closeButtonStyles}; +` diff --git a/libs/core/device-manager/actions/set-active-device.action.ts b/libs/core/device-manager/actions/set-active-device.action.ts index c8cc07b10..1b0f2fbcf 100644 --- a/libs/core/device-manager/actions/set-active-device.action.ts +++ b/libs/core/device-manager/actions/set-active-device.action.ts @@ -8,12 +8,14 @@ import { ReduxRootState } from "Core/__deprecated__/renderer/store" import { DeviceManagerEvent } from "Core/device-manager/constants" import { setActiveDeviceRequest } from "Core/device-manager/requests" import { DeviceId } from "Core/device/constants/device-id" +import { cleanBackupProcess } from "generic-view/store" export const setActiveDevice = createAsyncThunk< DeviceId | undefined, DeviceId | undefined, { state: ReduxRootState } ->(DeviceManagerEvent.SetActiveDevice, async (payload) => { +>(DeviceManagerEvent.SetActiveDevice, async (payload, { dispatch }) => { await setActiveDeviceRequest(payload) + dispatch(cleanBackupProcess()) return payload }) diff --git a/libs/core/files-manager/actions/delete-files.action.test.ts b/libs/core/files-manager/actions/delete-files.action.test.ts index 5299b0b54..faefeee1b 100644 --- a/libs/core/files-manager/actions/delete-files.action.test.ts +++ b/libs/core/files-manager/actions/delete-files.action.test.ts @@ -12,11 +12,18 @@ import { Result, ResultObject } from "Core/core/builder" import { deleteFiles } from "Core/files-manager/actions/delete-files.action" import { AppError } from "Core/core/errors" import { testError } from "Core/__deprecated__/renderer/store/constants" -import { fulfilledAction, pendingAction } from "Core/__deprecated__/renderer/store" +import { + fulfilledAction, + pendingAction, +} from "Core/__deprecated__/renderer/store" import * as loadStorageInfoActionModule from "Core/device/actions/load-storage-info.action" import * as loadDeviceDataActionModule from "Core/device/actions/load-device-data.action" import { DeviceEvent } from "Core/device" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("Core/files-manager/requests/delete-files.request") jest .spyOn(loadStorageInfoActionModule, "loadStorageInfoAction") @@ -26,14 +33,12 @@ jest type: pendingAction(DeviceEvent.LoadStorageInfo), } as unknown as jest.Mock) ) -jest - .spyOn(loadDeviceDataActionModule, "loadDeviceData") - .mockImplementation( - () => - ({ - type: fulfilledAction(DeviceEvent.LoadDeviceData), - } as unknown as jest.Mock) - ) +jest.spyOn(loadDeviceDataActionModule, "loadDeviceData").mockImplementation( + () => + ({ + type: fulfilledAction(DeviceEvent.LoadDeviceData), + } as unknown as jest.Mock) +) const filePaths = [ "user/music/example_file_name.mp3", "user/music/second_example_file_name.wav", diff --git a/libs/core/files-manager/actions/get-files.action.test.ts b/libs/core/files-manager/actions/get-files.action.test.ts index a5d0662d6..383e6cda9 100644 --- a/libs/core/files-manager/actions/get-files.action.test.ts +++ b/libs/core/files-manager/actions/get-files.action.test.ts @@ -18,16 +18,18 @@ import * as loadDeviceDataActionModule from "Core/device/actions/load-device-dat import { fulfilledAction } from "Core/__deprecated__/renderer/store" import { DeviceEvent } from "Core/device" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("Core/files-manager/requests/get-files.request") -jest - .spyOn(loadDeviceDataActionModule, "loadDeviceData") - .mockImplementation( - () => - ({ - type: fulfilledAction(DeviceEvent.LoadDeviceData), - } as unknown as jest.Mock) - ) +jest.spyOn(loadDeviceDataActionModule, "loadDeviceData").mockImplementation( + () => + ({ + type: fulfilledAction(DeviceEvent.LoadDeviceData), + } as unknown as jest.Mock) +) const successObjectResult: ResultObject = Result.success([ { diff --git a/libs/core/help/components/help.test.tsx b/libs/core/help/components/help.test.tsx index e7f34d615..49ff37bfd 100644 --- a/libs/core/help/components/help.test.tsx +++ b/libs/core/help/components/help.test.tsx @@ -12,6 +12,10 @@ import { HelpComponentTestIds } from "Core/help/components/help.enum" import Help from "Core/help/components/help.component" import { data } from "Core/__deprecated__/seeds/help" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("@electron/remote", () => ({ dialog: { showOpenDialog: jest.fn(), diff --git a/libs/core/overview/components/overview-screens/harmony-overview/harmony-overview.component.test.tsx b/libs/core/overview/components/overview-screens/harmony-overview/harmony-overview.component.test.tsx index 670612a17..f95182e85 100644 --- a/libs/core/overview/components/overview-screens/harmony-overview/harmony-overview.component.test.tsx +++ b/libs/core/overview/components/overview-screens/harmony-overview/harmony-overview.component.test.tsx @@ -16,6 +16,10 @@ import { Provider } from "react-redux" import { CheckForUpdateState } from "Core/update/constants/check-for-update-state.constant" import { CaseColour } from "Core/device" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("Core/feature-flags") jest.mock("@electron/remote", () => ({ diff --git a/libs/core/overview/components/overview-screens/pure-overview/pure-overview.component.test.tsx b/libs/core/overview/components/overview-screens/pure-overview/pure-overview.component.test.tsx index 09d3a079f..cbbc6a53f 100644 --- a/libs/core/overview/components/overview-screens/pure-overview/pure-overview.component.test.tsx +++ b/libs/core/overview/components/overview-screens/pure-overview/pure-overview.component.test.tsx @@ -20,6 +20,10 @@ import { CheckForUpdateState } from "Core/update/constants/check-for-update-stat // TODO [mw] add integration tests for update process - scope of the next PR (after all the changes from CP-1681 are done) +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("Core/feature-flags") jest.mock("@electron/remote", () => ({ diff --git a/libs/core/overview/components/overview/overview.test.tsx b/libs/core/overview/components/overview/overview.test.tsx index be5145546..fa903a57e 100644 --- a/libs/core/overview/components/overview/overview.test.tsx +++ b/libs/core/overview/components/overview/overview.test.tsx @@ -23,6 +23,10 @@ import { CheckForUpdateState } from "Core/update/constants/check-for-update-stat type Props = ComponentProps +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("@electron/remote", () => ({ Menu: () => ({ popup: jest.fn(), diff --git a/libs/core/overview/components/system-update-text/system-update-text.test.tsx b/libs/core/overview/components/system-update-text/system-update-text.test.tsx index ddbbc5608..9ebd89871 100644 --- a/libs/core/overview/components/system-update-text/system-update-text.test.tsx +++ b/libs/core/overview/components/system-update-text/system-update-text.test.tsx @@ -15,6 +15,10 @@ import { initialState as update } from "Core/update/reducers" import { OsRelease } from "Core/update/dto" import { OsReleaseType, Product } from "Core/update/constants" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + const defaultProps: SystemUpdateTextProps = { checkForUpdateFailed: false, checkForUpdateInProgress: false, diff --git a/libs/core/overview/components/update-os-flow/update-os-flow.component.test.tsx b/libs/core/overview/components/update-os-flow/update-os-flow.component.test.tsx index 01e4dd6cc..fc509587c 100644 --- a/libs/core/overview/components/update-os-flow/update-os-flow.component.test.tsx +++ b/libs/core/overview/components/update-os-flow/update-os-flow.component.test.tsx @@ -22,6 +22,10 @@ import { renderWithThemeAndIntl } from "Core/__deprecated__/renderer/utils/rende import React from "react" import { CheckForUpdateState } from "Core/update/constants/check-for-update-state.constant" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("@electron/remote", () => ({ Menu: () => ({ popup: jest.fn(), diff --git a/libs/core/settings/components/about/about.test.tsx b/libs/core/settings/components/about/about.test.tsx index 5fdfe510d..4428473d2 100644 --- a/libs/core/settings/components/about/about.test.tsx +++ b/libs/core/settings/components/about/about.test.tsx @@ -15,6 +15,10 @@ import { flags } from "Core/feature-flags" import store from "Core/__deprecated__/renderer/store" import { Provider } from "react-redux" +jest.mock("Core/settings/store/schemas/generate-application-id", () => ({ + generateApplicationId: () => "123", +})) + jest.mock("Core/feature-flags") jest.mock("electron-better-ipc", () => { return { diff --git a/libs/core/settings/store/schemas/generate-application-id.ts b/libs/core/settings/store/schemas/generate-application-id.ts new file mode 100644 index 000000000..582179706 --- /dev/null +++ b/libs/core/settings/store/schemas/generate-application-id.ts @@ -0,0 +1,28 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import getMAC from "getmac" + +export const generateApplicationId = (): string | null => { + const maxApplicationIdLength = 16 + + const mac = getMACOrNull() + if (mac !== null) { + const uniqueValue = mac.replace(/:/g, "").slice(-maxApplicationIdLength) + const padLength = maxApplicationIdLength - uniqueValue.length + const pad = Math.random().toString(16).slice(-padLength) + return `${pad}${uniqueValue}` + } else { + return null + } +} + +const getMACOrNull = (): string | null => { + try { + return getMAC() + } catch (ex) { + return null + } +} diff --git a/libs/core/settings/store/schemas/settings.schema.ts b/libs/core/settings/store/schemas/settings.schema.ts index b7c0af229..c149d772c 100644 --- a/libs/core/settings/store/schemas/settings.schema.ts +++ b/libs/core/settings/store/schemas/settings.schema.ts @@ -5,33 +5,11 @@ import path from "path" import { Schema } from "electron-store" -import getMAC from "getmac" import getAppPath from "Core/__deprecated__/main/utils/get-app-path" import { Settings } from "Core/settings/dto" import { ConversionFormat, Convert } from "Core/settings/constants" import translationConfig from "App/translations.config.json" - -const generateApplicationId = (): string | null => { - const maxApplicationIdLength = 16 - - const mac = getMACOrNull() - if (mac !== null) { - const uniqueValue = mac.replace(/:/g, "").slice(-maxApplicationIdLength) - const padLength = maxApplicationIdLength - uniqueValue.length - const pad = Math.random().toString(16).slice(-padLength) - return `${pad}${uniqueValue}` - } else { - return null - } -} - -const getMACOrNull = (): string | null => { - try { - return getMAC() - } catch (ex) { - return null - } -} +import { generateApplicationId } from "Core/settings/store/schemas/generate-application-id" export const settingsSchema: Schema = { applicationId: { diff --git a/libs/device/feature/src/index.ts b/libs/device/feature/src/index.ts index 76e03c6f5..80c9b2826 100644 --- a/libs/device/feature/src/index.ts +++ b/libs/device/feature/src/index.ts @@ -10,3 +10,6 @@ export * from "./lib/api-features" export * from "./lib/outbox" export * from "./lib/menu" export * from "./lib/server" +export * from "./lib/file-transfer" +export * from "./lib/file-manager" +export * from "./lib/backup" diff --git a/libs/device/feature/src/lib/api-module.ts b/libs/device/feature/src/lib/api-module.ts index 7b36415bc..e2c065031 100644 --- a/libs/device/feature/src/lib/api-module.ts +++ b/libs/device/feature/src/lib/api-module.ts @@ -7,9 +7,14 @@ import { DeviceManager } from "Core/device-manager/services" import { APIConfigService } from "./api-config/api-config.service" import { APIFeaturesService } from "./api-features/api-features.service" import { APIBackupService } from "./backup" +import { FileManager } from "./file-manager" import { APIMenuService } from "./menu" import { APIOutboxService } from "./outbox/outbox.service" import { ServerService } from "./server/server.service" +import { APIFileTransferService } from "./file-transfer" +import { ServiceBridge } from "./service-bridge" +import { SystemUtilsModule } from "system-utils/feature" +import { createSettingsService } from "Core/settings/containers/settings.container" export class APIModule { private apiConfigService: APIConfigService @@ -18,14 +23,26 @@ export class APIModule { private apiMenuService: APIMenuService private serverService: ServerService private backupService: APIBackupService + private fileTransferService: APIFileTransferService + private fileManager: FileManager + private serviceBridge: ServiceBridge - constructor(deviceManager: DeviceManager) { + constructor( + deviceManager: DeviceManager, + systemUtilsModule: SystemUtilsModule + ) { + this.serviceBridge = new ServiceBridge() this.apiConfigService = new APIConfigService(deviceManager) this.apiFeaturesService = new APIFeaturesService(deviceManager) this.apiOutboxService = new APIOutboxService(deviceManager) this.apiMenuService = new APIMenuService(deviceManager) this.serverService = new ServerService() this.backupService = new APIBackupService(deviceManager) + this.fileTransferService = new APIFileTransferService(deviceManager) + this.fileManager = new FileManager(deviceManager, this.serviceBridge) + this.serviceBridge.systemUtilsModule = systemUtilsModule + this.serviceBridge.fileTransfer = this.fileTransferService + this.serviceBridge.settingsService = createSettingsService() } public getAPIServices() { @@ -36,6 +53,8 @@ export class APIModule { this.apiMenuService, this.serverService, this.backupService, + this.fileTransferService, + this.fileManager, ] } } diff --git a/libs/device/feature/src/lib/backup/backup.service.ts b/libs/device/feature/src/lib/backup/backup.service.ts index 8e3301293..010190cf6 100644 --- a/libs/device/feature/src/lib/backup/backup.service.ts +++ b/libs/device/feature/src/lib/backup/backup.service.ts @@ -112,7 +112,7 @@ export class APIBackupService { } private parsePreBackupResponse( - response: ResultObject, string, unknown>, + response: ResultObject, string | number, unknown>, features: string[] ) { if (!response.ok) { diff --git a/libs/device/feature/src/lib/backup/start-pre-backup.request.ts b/libs/device/feature/src/lib/backup/start-pre-backup.request.ts index 7d7de2292..b7ba91c77 100644 --- a/libs/device/feature/src/lib/backup/start-pre-backup.request.ts +++ b/libs/device/feature/src/lib/backup/start-pre-backup.request.ts @@ -8,7 +8,7 @@ import { ipcRenderer } from "electron-better-ipc" import { APIBackupServiceEvents, PreBackup } from "device/models" import { DeviceId } from "Core/device/constants/device-id" -export const getStartPreBackupRequest = ( +export const startPreBackupRequest = ( features: string[], deviceId?: DeviceId ): Promise> => { diff --git a/libs/device/feature/src/lib/file-manager/file-manager.service.ts b/libs/device/feature/src/lib/file-manager/file-manager.service.ts new file mode 100644 index 000000000..24ca45d0e --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/file-manager.service.ts @@ -0,0 +1,177 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { Result, ResultObject } from "Core/core/builder" +import { IpcEvent } from "Core/core/decorators" +import { ServiceBridge } from "../service-bridge" +import { FileManagerServiceEvents, GeneralError } from "device/models" +import { AppError } from "Core/core/errors" +import { DeviceManager } from "Core/device-manager/services" +import { DeviceId } from "Core/device/constants/device-id" +import packageInfo from "../../../../../../apps/mudita-center/package.json" +import { writeFileSync, writeJSONSync, mkdirSync, readdirSync } from "fs-extra" +import AES from "crypto-js/aes" +import path from "path" + +export class FileManager { + constructor( + private deviceManager: DeviceManager, + private serviceBridge: ServiceBridge + ) {} + + @IpcEvent(FileManagerServiceEvents.SaveFile) + public saveFileByTransferId({ + filePath, + transferId, + }: { + filePath: string + transferId: number + }): ResultObject { + try { + const file = + this.serviceBridge.fileTransfer.getFileByTransferId(transferId) + writeFileSync(filePath, file.chunks.join(), "base64") + + return Result.success(undefined) + } catch (e) { + console.log(e) + return Result.failed(new AppError(GeneralError.IncorrectResponse)) + } + } + + @IpcEvent(FileManagerServiceEvents.GetBackupPath) + public getBackupPath({ deviceId }: { deviceId?: DeviceId }) { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + if (!device) { + return Result.failed(new AppError(GeneralError.NoDevice, "")) + } + + const vendorId = device.portInfo.vendorId ?? "unknown" + const productId = device.portInfo.productId ?? "unknown" + + const { osBackupLocation } = + this.serviceBridge.settingsService.getSettings() + const backupLocation = path.join( + osBackupLocation, + `${vendorId}-${productId}` + ) + return Result.success(backupLocation) + } + + @IpcEvent(FileManagerServiceEvents.OpenBackupDirectory) + public openBackupDirectory({ deviceId }: { deviceId?: DeviceId }) { + const backupDirectory = this.getBackupPath({ deviceId }) + if (!backupDirectory.ok) { + return Result.failed(new AppError(GeneralError.InternalError, "")) + } + return this.serviceBridge.systemUtilsModule.directory.open({ + path: backupDirectory.data, + }) + } + + @IpcEvent(FileManagerServiceEvents.SaveBackupFile) + public saveBackupFile({ + deviceId, + featureToTransferId, + password, + }: { + deviceId?: DeviceId + featureToTransferId: Record + password?: string + }) { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + if (!device) { + return Result.failed(new AppError(GeneralError.NoDevice, "")) + } + try { + const vendorId = device.portInfo.vendorId ?? "" + const productId = device.portInfo.productId ?? "" + const serialNumber = device.portInfo.serialNumber ?? "" + const timestamp = new Date().getTime() + const appVersion = packageInfo.version + const backupDirectory = this.getBackupPath({ deviceId }) + + if (!backupDirectory.ok) { + return Result.failed(new AppError(GeneralError.InternalError, "")) + } + + const backupFilePath = path.join( + backupDirectory.data, + `${timestamp}_${serialNumber}.mcbackup` + ) + + const data = Object.entries(featureToTransferId).reduce( + (acc, [feature, transferId]) => { + const transfer = + this.serviceBridge.fileTransfer.getFileByTransferId(transferId) + + const featureData = password + ? AES.encrypt(transfer.chunks.join(), password).toString() + : transfer.chunks.join() + + return { ...acc, [feature]: featureData } + }, + {} + ) + + const fileToSave = { + header: { + vendorId, + productId, + serialNumber, + appVersion, + ...(password && { crypto: "AES" }), + }, + data, + } + + mkdirSync(backupDirectory.data, { recursive: true }) + + writeJSONSync(backupFilePath, fileToSave) + + return Result.success(undefined) + } catch (e) { + console.log(e) + return Result.failed(new AppError(GeneralError.InternalError)) + } + } + + @IpcEvent(FileManagerServiceEvents.ReadDirectory) + public readDirectory({ path }: { path: string }): ResultObject { + try { + const result = readdirSync(path) + + return Result.success(result) + } catch (e) { + console.log(e) + return Result.failed(new AppError(GeneralError.InternalError)) + } + } + + @IpcEvent(FileManagerServiceEvents.ReadBackupDirectory) + public readBackupDirectory({ + deviceId, + }: { + deviceId?: DeviceId + }): ResultObject { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + const pathResult = this.getBackupPath({ deviceId }) + + if (!pathResult.ok) { + return Result.failed(new AppError(GeneralError.InternalError)) + } + + return this.readDirectory({ path: pathResult.data }) + } +} diff --git a/libs/device/feature/src/lib/file-manager/index.ts b/libs/device/feature/src/lib/file-manager/index.ts new file mode 100644 index 000000000..5b597ed81 --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/index.ts @@ -0,0 +1,11 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./file-manager.service" +export * from "./save-file.request" +export * from "./save-backup-file.request" +export * from "./open-backup-directory.request" +export * from "./read-directory.request" +export * from "./read-backup-directory.request" diff --git a/libs/device/feature/src/lib/file-manager/open-backup-directory.request.ts b/libs/device/feature/src/lib/file-manager/open-backup-directory.request.ts new file mode 100644 index 000000000..6e4579614 --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/open-backup-directory.request.ts @@ -0,0 +1,17 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ResultObject } from "Core/core/builder" +import { DeviceId } from "Core/device/constants/device-id" +import { FileManagerServiceEvents } from "device/models" +import { ipcRenderer } from "electron-better-ipc" + +export const openBackupDirectoryRequest = ( + deviceId?: DeviceId +): Promise> => { + return ipcRenderer.callMain(FileManagerServiceEvents.OpenBackupDirectory, { + deviceId, + }) +} diff --git a/libs/device/feature/src/lib/file-manager/read-backup-directory.request.ts b/libs/device/feature/src/lib/file-manager/read-backup-directory.request.ts new file mode 100644 index 000000000..85637dd09 --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/read-backup-directory.request.ts @@ -0,0 +1,17 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ResultObject } from "Core/core/builder" +import { DeviceId } from "Core/device/constants/device-id" +import { FileManagerServiceEvents } from "device/models" +import { ipcRenderer } from "electron-better-ipc" + +export const readBackupDirectoryRequest = ( + deviceId: DeviceId +): Promise> => { + return ipcRenderer.callMain(FileManagerServiceEvents.ReadBackupDirectory, { + deviceId, + }) +} diff --git a/libs/device/feature/src/lib/file-manager/read-directory.request.ts b/libs/device/feature/src/lib/file-manager/read-directory.request.ts new file mode 100644 index 000000000..ca58ed62f --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/read-directory.request.ts @@ -0,0 +1,16 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ResultObject } from "Core/core/builder" +import { FileManagerServiceEvents } from "device/models" +import { ipcRenderer } from "electron-better-ipc" + +export const readDirectoryRequest = ( + path: string +): Promise> => { + return ipcRenderer.callMain(FileManagerServiceEvents.ReadDirectory, { + path, + }) +} diff --git a/libs/device/feature/src/lib/file-manager/save-backup-file.request.ts b/libs/device/feature/src/lib/file-manager/save-backup-file.request.ts new file mode 100644 index 000000000..a7e2dc51c --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/save-backup-file.request.ts @@ -0,0 +1,21 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ResultObject } from "Core/core/builder" +import { DeviceId } from "Core/device/constants/device-id" +import { FileManagerServiceEvents } from "device/models" +import { ipcRenderer } from "electron-better-ipc" + +export const saveBackupFileRequest = ( + featureToTransferId: Record, + deviceId?: DeviceId, + password?: string +): Promise> => { + return ipcRenderer.callMain(FileManagerServiceEvents.SaveBackupFile, { + featureToTransferId, + deviceId, + password, + }) +} diff --git a/libs/device/feature/src/lib/file-manager/save-file.request.ts b/libs/device/feature/src/lib/file-manager/save-file.request.ts new file mode 100644 index 000000000..e24b70cee --- /dev/null +++ b/libs/device/feature/src/lib/file-manager/save-file.request.ts @@ -0,0 +1,18 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ResultObject } from "Core/core/builder" +import { FileManagerServiceEvents } from "device/models" +import { ipcRenderer } from "electron-better-ipc" + +export const saveFileRequest = ( + filePath: string, + transferId: number +): Promise> => { + return ipcRenderer.callMain(FileManagerServiceEvents.SaveFile, { + filePath, + transferId, + }) +} diff --git a/libs/device/feature/src/lib/file-transfer/file-transfer.service.ts b/libs/device/feature/src/lib/file-transfer/file-transfer.service.ts new file mode 100644 index 000000000..467e31a04 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/file-transfer.service.ts @@ -0,0 +1,339 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { Result, ResultObject } from "Core/core/builder" +import { IpcEvent } from "Core/core/decorators" +import { AppError, AppErrorType } from "Core/core/errors" +import { DeviceManager } from "Core/device-manager/services" +import { DeviceId } from "Core/device/constants/device-id" +import { + ApiFileTransferError, + ApiFileTransferServiceEvents, + FileTransferStatuses, + GeneralError, + PreTransferGet, + PreTransferGetValidator, + PreTransferSendValidator, + TransferGetValidator, + TransferSend, + TransferSendValidator, +} from "device/models" +import { readFileSync } from "fs-extra" +import crc from "js-crc" + +interface Transfer { + crc32: string + fileSize: number + filePath: string + chunks: string[] +} + +const DEFAULT_MAX_REPEATS = 2 + +export class APIFileTransferService { + constructor( + private deviceManager: DeviceManager, + private transfers: Record = {} + ) {} + + private prepareFile(path: string) { + const file = readFileSync(path, { + encoding: "base64", + }) + return { + file, + crc32: crc.crc32(file), + } + } + + public getFileByTransferId(transferId: number) { + return this.transfers[transferId] + } + + // Sending files to device + @IpcEvent(ApiFileTransferServiceEvents.PreSend) + public async preTransferSend({ + filePath, + targetPath, + deviceId, + }: { + filePath: string + targetPath: string + deviceId?: DeviceId + }): Promise< + ResultObject<{ + transferId: number + chunksCount: number + }> + > { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + if (!device) { + return Result.failed(new AppError(GeneralError.NoDevice, "")) + } + const { crc32, file } = this.prepareFile(filePath) + + const response = await device.request({ + endpoint: "PRE_FILE_TRANSFER", + method: "POST", + body: { + filePath: targetPath, + fileSize: file.length, + crc32, + }, + }) + + if (response.ok) { + const preTransferResponse = PreTransferSendValidator.safeParse( + response.data.body + ) + + const success = preTransferResponse.success + + if (!success) { + return handleError(response.data.status) + } + + this.transfers[preTransferResponse.data.transferId] = { + crc32, + fileSize: file.length, + filePath, + chunks: + file.match( + new RegExp(`.{1,${preTransferResponse.data.chunkSize}}`, "g") + ) || [], + } + + return Result.success({ + transferId: preTransferResponse.data.transferId, + chunksCount: + this.transfers[preTransferResponse.data.transferId].chunks.length, + }) + } + + return handleError(response.error.type) + } + + @IpcEvent(ApiFileTransferServiceEvents.Send) + public async transferSend({ + transferId, + chunkNumber, + deviceId, + repeats = 0, + maxRepeats = DEFAULT_MAX_REPEATS, + }: { + transferId: number + chunkNumber: number + deviceId?: DeviceId + repeats: number + maxRepeats: number + }): Promise> { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + if (!device) { + return Result.failed(new AppError(GeneralError.NoDevice, "")) + } + + const data = this.transfers[transferId].chunks[chunkNumber - 1] + + const response = await device.request({ + endpoint: "FILE_TRANSFER", + method: "POST", + body: { + transferId, + chunkNumber, + data, + }, + }) + + if (!response.ok) { + if (repeats < maxRepeats) { + return this.transferSend({ + deviceId, + transferId, + chunkNumber, + repeats: repeats + 1, + maxRepeats, + }) + } else { + return handleError(response.error.type) + } + } + + const transferResponse = TransferSendValidator.safeParse(response.data.body) + + const success = + transferResponse.success && + [ + FileTransferStatuses.WholeFileTransferred, + FileTransferStatuses.FileChunkTransferred, + ].includes(response.data.status as number) + + if (!success) { + return handleError(response.data.status) + } + + return Result.success(transferResponse.data) + } + + private validateChecksum(transferId: number) { + const transfer = this.transfers[transferId] + const data = transfer.chunks.join("") + const crc32 = crc.crc32(data) + return crc32.toLowerCase() === transfer.crc32.toLowerCase() + } + + @IpcEvent(ApiFileTransferServiceEvents.PreGet) + public async preTransferGet({ + filePath, + deviceId, + }: { + filePath: string + deviceId?: DeviceId + }): Promise> { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + if (!device) { + return Result.failed(new AppError(GeneralError.NoDevice, "")) + } + + const response = await device.request({ + endpoint: "PRE_FILE_TRANSFER", + method: "GET", + body: { + filePath, + }, + }) + + if (response.ok) { + const preTransferResponse = PreTransferGetValidator.safeParse( + response.data.body + ) + + const success = preTransferResponse.success + + if (!success) { + return handleError(response.data.status) + } + + this.transfers[preTransferResponse.data.transferId] = { + crc32: preTransferResponse.data.crc32, + fileSize: preTransferResponse.data.fileSize, + filePath, + chunks: [], + } + + return Result.success(preTransferResponse.data) + } + + return handleError(response.error.type) + } + + @IpcEvent(ApiFileTransferServiceEvents.Get) + public async transferGet({ + deviceId, + transferId, + chunkNumber, + repeats = 0, + maxRepeats = DEFAULT_MAX_REPEATS, + }: { + deviceId?: DeviceId + transferId: number + chunkNumber: number + repeats: number + maxRepeats: number + }): Promise> { + const device = deviceId + ? this.deviceManager.getAPIDeviceById(deviceId) + : this.deviceManager.apiDevice + + if (!device) { + return Result.failed(new AppError(GeneralError.NoDevice, "")) + } + + const response = await device.request({ + endpoint: "FILE_TRANSFER", + method: "GET", + body: { + transferId, + chunkNumber, + }, + }) + + if (!response.ok) { + if (repeats < maxRepeats) { + return this.transferGet({ + deviceId, + transferId, + chunkNumber, + repeats: repeats + 1, + maxRepeats, + }) + } else { + return handleError(response.error.type) + } + } + + const transferResponse = TransferGetValidator.safeParse(response.data.body) + + const success = + transferResponse.success && + [ + FileTransferStatuses.WholeFileTransferred, + FileTransferStatuses.FileChunkTransferred, + ].includes(response.data.status as number) + + if (!success) { + return handleError(response.data.status) + } + + this.transfers[transferId].chunks[chunkNumber - 1] = + transferResponse.data.data + + if ( + (response.data.status as number) === + FileTransferStatuses.WholeFileTransferred + ) { + if (this.validateChecksum(transferId)) { + return Result.success(undefined) + } else { + return Result.failed( + new AppError( + ApiFileTransferError.CRCMismatch, + ApiFileTransferError[ApiFileTransferError.CRCMismatch] + ) + ) + } + } + return Result.success(undefined) + } + + @IpcEvent(ApiFileTransferServiceEvents.Clear) + public transferClear({ transferId }: { transferId: number }) { + delete this.transfers[transferId] + } +} + +const handleError = (responseStatus: AppErrorType) => { + if (ApiFileTransferError[responseStatus as ApiFileTransferError]) { + return Result.failed< + { transferId?: number; filePath: string }, + AppErrorType + >( + new AppError( + responseStatus, + ApiFileTransferError[responseStatus as ApiFileTransferError] + ) + ) + } else { + return Result.failed(new AppError(GeneralError.IncorrectResponse, "")) + } +} diff --git a/libs/device/feature/src/lib/file-transfer/get-file.request.ts b/libs/device/feature/src/lib/file-transfer/get-file.request.ts new file mode 100644 index 000000000..980aef7b1 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/get-file.request.ts @@ -0,0 +1,21 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ipcRenderer } from "electron-better-ipc" +import { ApiFileTransferServiceEvents } from "device/models" +import { DeviceId } from "Core/device/constants/device-id" +import { ResultObject } from "Core/core/builder" + +export const getFileRequest = ( + transferId: number, + chunkNumber: number, + deviceId?: DeviceId +): Promise> => { + return ipcRenderer.callMain(ApiFileTransferServiceEvents.Get, { + transferId, + chunkNumber, + deviceId, + }) +} diff --git a/libs/device/feature/src/lib/file-transfer/index.ts b/libs/device/feature/src/lib/file-transfer/index.ts new file mode 100644 index 000000000..69516bff5 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/index.ts @@ -0,0 +1,11 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./file-transfer.service" +export * from "./start-pre-send-file.request" +export * from "./send-file.request" +export * from "./send-clear.request" +export * from "./start-pre-get-file.request" +export * from "./get-file.request" diff --git a/libs/device/feature/src/lib/file-transfer/send-clear.request.ts b/libs/device/feature/src/lib/file-transfer/send-clear.request.ts new file mode 100644 index 000000000..1377055d3 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/send-clear.request.ts @@ -0,0 +1,13 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ipcRenderer } from "electron-better-ipc" +import { ApiFileTransferServiceEvents } from "device/models" + +export const sendClearRequest = (transferId: number): Promise => { + return ipcRenderer.callMain(ApiFileTransferServiceEvents.Clear, { + transferId, + }) +} diff --git a/libs/device/feature/src/lib/file-transfer/send-file.request.ts b/libs/device/feature/src/lib/file-transfer/send-file.request.ts new file mode 100644 index 000000000..9441b6d80 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/send-file.request.ts @@ -0,0 +1,21 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ipcRenderer } from "electron-better-ipc" +import { ApiFileTransferServiceEvents, TransferSend } from "device/models" +import { DeviceId } from "Core/device/constants/device-id" +import { ResultObject } from "Core/core/builder" + +export const sendFileRequest = ( + transferId: number, + chunkNumber: number, + deviceId?: DeviceId +): Promise> => { + return ipcRenderer.callMain(ApiFileTransferServiceEvents.Send, { + transferId, + chunkNumber, + deviceId, + }) +} diff --git a/libs/device/feature/src/lib/file-transfer/start-pre-get-file.request.ts b/libs/device/feature/src/lib/file-transfer/start-pre-get-file.request.ts new file mode 100644 index 000000000..1d79f7d17 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/start-pre-get-file.request.ts @@ -0,0 +1,19 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ipcRenderer } from "electron-better-ipc" +import { ApiFileTransferServiceEvents, PreTransferGet } from "device/models" +import { DeviceId } from "Core/device/constants/device-id" +import { ResultObject } from "Core/core/builder" + +export const startPreGetFileRequest = ( + filePath: string, + deviceId?: DeviceId +): Promise> => { + return ipcRenderer.callMain(ApiFileTransferServiceEvents.PreGet, { + filePath, + deviceId, + }) +} diff --git a/libs/device/feature/src/lib/file-transfer/start-pre-send-file.request.ts b/libs/device/feature/src/lib/file-transfer/start-pre-send-file.request.ts new file mode 100644 index 000000000..51ef56856 --- /dev/null +++ b/libs/device/feature/src/lib/file-transfer/start-pre-send-file.request.ts @@ -0,0 +1,26 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { ipcRenderer } from "electron-better-ipc" +import { ApiFileTransferServiceEvents } from "device/models" +import { DeviceId } from "Core/device/constants/device-id" +import { ResultObject } from "Core/core/builder" + +export const startPreSendFileRequest = ( + filePath: string, + targetPath: string, + deviceId?: DeviceId +): Promise< + ResultObject<{ + transferId: number + chunksCount: number + }> +> => { + return ipcRenderer.callMain(ApiFileTransferServiceEvents.PreSend, { + filePath, + targetPath, + deviceId, + }) +} diff --git a/libs/device/feature/src/lib/restore/restore.service.ts b/libs/device/feature/src/lib/restore/restore.service.ts index de16311f8..3ec4f1f33 100644 --- a/libs/device/feature/src/lib/restore/restore.service.ts +++ b/libs/device/feature/src/lib/restore/restore.service.ts @@ -118,7 +118,7 @@ export class APIRestoreService { } private parseRestoreResponse( - response: ResultObject, string, unknown> + response: ResultObject, string | number, unknown> ) { if (!response.ok) { return Result.failed(response.error) diff --git a/libs/device/feature/src/lib/service-bridge/index.ts b/libs/device/feature/src/lib/service-bridge/index.ts new file mode 100644 index 000000000..9bdd0cc18 --- /dev/null +++ b/libs/device/feature/src/lib/service-bridge/index.ts @@ -0,0 +1,6 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./service-bridge" diff --git a/libs/device/feature/src/lib/service-bridge/service-bridge.ts b/libs/device/feature/src/lib/service-bridge/service-bridge.ts new file mode 100644 index 000000000..34b930e57 --- /dev/null +++ b/libs/device/feature/src/lib/service-bridge/service-bridge.ts @@ -0,0 +1,55 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { SettingsService } from "Core/settings/services" +import { APIFileTransferService } from "../file-transfer" +import { SystemUtilsModule } from "system-utils/feature" + +export class ServiceBridge { + private _fileTransfer?: APIFileTransferService + + get fileTransfer(): APIFileTransferService { + if (!this._fileTransfer) { + throw new Error( + "APIFileTransferService reference has not been set in ServiceBridge" + ) + } + return this._fileTransfer + } + set fileTransfer(value: APIFileTransferService) { + this._fileTransfer = value + } + + private _settingsService?: SettingsService + + get settingsService(): SettingsService { + if (!this._settingsService) { + throw new Error( + "SettingsService reference has not been set in ServiceBridge" + ) + } + return this._settingsService + } + set settingsService(value: SettingsService) { + this._settingsService = value + } + + private _systemUtilsModule?: SystemUtilsModule + + get systemUtilsModule(): SystemUtilsModule { + if (!this._systemUtilsModule) { + throw new Error( + "SystemUtilsService reference has not been set in ServiceBridge" + ) + } + return this._systemUtilsModule + } + + set systemUtilsModule(value: SystemUtilsModule) { + this._systemUtilsModule = value + } + + constructor() {} +} diff --git a/libs/device/feature/tsconfig.json b/libs/device/feature/tsconfig.json index 4daaf45cd..ac56d3d8c 100644 --- a/libs/device/feature/tsconfig.json +++ b/libs/device/feature/tsconfig.json @@ -4,7 +4,8 @@ "allowJs": false, "esModuleInterop": false, "allowSyntheticDefaultImports": true, - "strict": true + "strict": true, + "resolveJsonModule": true, }, "files": [], "include": [], diff --git a/libs/device/models/src/index.ts b/libs/device/models/src/index.ts index ad6e17ff5..f14d789b6 100644 --- a/libs/device/models/src/index.ts +++ b/libs/device/models/src/index.ts @@ -13,4 +13,5 @@ export * from "./lib/api-config" export * from "./lib/server" export * from "./lib/general-error" export * from "./lib/backup" +export * from "./lib/file-transfer" export * from "./lib/restore" diff --git a/libs/device/models/src/lib/api-request.model.ts b/libs/device/models/src/lib/api-request.model.ts index 50a66d0f0..fb483c896 100644 --- a/libs/device/models/src/lib/api-request.model.ts +++ b/libs/device/models/src/lib/api-request.model.ts @@ -12,6 +12,8 @@ const APIEndpoints = { Outbox: "OUTBOX", PreBackup: "PRE_BACKUP", PostBackup: "POST_BACKUP", + PreFileTransfer: "PRE_FILE_TRANSFER", + FileTransfer: "FILE_TRANSFER", PreRestore: "PRE_RESTORE", Restore: "RESTORE", } as const @@ -36,6 +38,8 @@ const APIRequests = { OUTBOX: [APIMethods.GET], PRE_BACKUP: [APIMethods.POST, APIMethods.GET], POST_BACKUP: [APIMethods.POST], + PRE_FILE_TRANSFER: [APIMethods.POST, APIMethods.GET], + FILE_TRANSFER: [APIMethods.POST, APIMethods.GET], PRE_RESTORE: [APIMethods.POST], RESTORE: [APIMethods.POST, APIMethods.GET], } as const diff --git a/libs/device/models/src/lib/file-transfer/file-transfer-statuses.ts b/libs/device/models/src/lib/file-transfer/file-transfer-statuses.ts new file mode 100644 index 000000000..59428258d --- /dev/null +++ b/libs/device/models/src/lib/file-transfer/file-transfer-statuses.ts @@ -0,0 +1,9 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export enum FileTransferStatuses { + WholeFileTransferred = 200, + FileChunkTransferred = 206, +} diff --git a/libs/device/models/src/lib/file-transfer/file-transfer.ts b/libs/device/models/src/lib/file-transfer/file-transfer.ts new file mode 100644 index 000000000..adb9e4bd4 --- /dev/null +++ b/libs/device/models/src/lib/file-transfer/file-transfer.ts @@ -0,0 +1,33 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { z } from "zod" + +export const PreTransferSendValidator = z.object({ + transferId: z.number(), + chunkSize: z.number().positive(), +}) + +export const TransferSendValidator = z.object({ + transferId: z.number(), + chunkNumber: z.number().positive(), +}) + +export type TransferSend = z.infer + +export const PreTransferGetValidator = z.object({ + transferId: z.number(), + chunkSize: z.number().positive(), + fileSize: z.number().positive(), + crc32: z.string(), +}) + +export type PreTransferGet = z.infer + +export const TransferGetValidator = z.object({ + transferId: z.number(), + chunkNumber: z.number().positive(), + data: z.string().min(1), +}) diff --git a/libs/device/models/src/lib/file-transfer/index.ts b/libs/device/models/src/lib/file-transfer/index.ts new file mode 100644 index 000000000..465d6dc58 --- /dev/null +++ b/libs/device/models/src/lib/file-transfer/index.ts @@ -0,0 +1,7 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./file-transfer" +export * from "./file-transfer-statuses" diff --git a/libs/device/models/src/lib/general-error.ts b/libs/device/models/src/lib/general-error.ts index 3f81b3c71..2cf094253 100644 --- a/libs/device/models/src/lib/general-error.ts +++ b/libs/device/models/src/lib/general-error.ts @@ -6,8 +6,18 @@ export enum GeneralError { NoDevice = "no-device", IncorrectResponse = "incorrect-response", + InternalError = "internal-error", } export enum ApiError { DeviceLocked = 423, } + +export enum ApiFileTransferError { + AccessRestricted = 403, + IncorrectPath = 404, + FileAlreadyExists = 409, + CRCMismatch = 422, + Unknown = 500, + NotEnoughSpace = 507, +} diff --git a/libs/device/models/src/lib/renderer-to-main-events/api-backup-service-events.ts b/libs/device/models/src/lib/renderer-to-main-events/api-backup-service-events.ts index 03828399e..4d07a21ab 100644 --- a/libs/device/models/src/lib/renderer-to-main-events/api-backup-service-events.ts +++ b/libs/device/models/src/lib/renderer-to-main-events/api-backup-service-events.ts @@ -7,4 +7,5 @@ export enum APIBackupServiceEvents { StartPreBackup = "apiservice_backup-start-pre-backup", CheckPreBackup = "apiservice_backup-check-pre-backup", PostBackup = "apiservice_backup-post-backup", + GetBackupList = "apiservice_backup-get-backup-list", } diff --git a/libs/device/models/src/lib/renderer-to-main-events/api-file-transfer-service-events.ts b/libs/device/models/src/lib/renderer-to-main-events/api-file-transfer-service-events.ts new file mode 100644 index 000000000..692a18774 --- /dev/null +++ b/libs/device/models/src/lib/renderer-to-main-events/api-file-transfer-service-events.ts @@ -0,0 +1,12 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export enum ApiFileTransferServiceEvents { + PreGet = "apiservice_file_transfer-pre-transfer-get", + Get = "apiservice_file_transfer-transfer-get", + PreSend = "apiservice_file_transfer-pre-transfer-send", + Send = "apiservice_file_transfer-transfer-send", + Clear = "apiservice_file_transfer-transfer-clear", +} diff --git a/libs/device/models/src/lib/renderer-to-main-events/file-manager-service-events.ts b/libs/device/models/src/lib/renderer-to-main-events/file-manager-service-events.ts new file mode 100644 index 000000000..89aabbbd1 --- /dev/null +++ b/libs/device/models/src/lib/renderer-to-main-events/file-manager-service-events.ts @@ -0,0 +1,13 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export enum FileManagerServiceEvents { + SaveFile = "apiservice_file_manager-save-file", + SaveBackupFile = "apiservice_file_manager-save-backup-file", + GetBackupPath = "apiservice_file_manager-get-backup-path", + OpenBackupDirectory = "apiservice_file_manager-open-directory", + ReadDirectory = "apiservice_file_manager-read-directory", + ReadBackupDirectory = "apiservice_file_manager-read-backup-directory", +} diff --git a/libs/device/models/src/lib/renderer-to-main-events/index.ts b/libs/device/models/src/lib/renderer-to-main-events/index.ts index 050709159..29c32b38a 100644 --- a/libs/device/models/src/lib/renderer-to-main-events/index.ts +++ b/libs/device/models/src/lib/renderer-to-main-events/index.ts @@ -10,3 +10,5 @@ export * from "./api-config-service-events" export * from "./api-backup-service-events" export * from "./api-restore-service-events" export * from "./server-service-events" +export * from "./api-file-transfer-service-events" +export * from "./file-manager-service-events" diff --git a/libs/generic-view/feature/src/index.ts b/libs/generic-view/feature/src/index.ts index 38d9faca2..604cd4e7c 100644 --- a/libs/generic-view/feature/src/index.ts +++ b/libs/generic-view/feature/src/index.ts @@ -5,3 +5,4 @@ export * from "./lib/generic-view" export * from "./lib/recursive-layout" +export * from "./lib/api-device-modals" diff --git a/libs/generic-view/feature/src/lib/api-device-modals.tsx b/libs/generic-view/feature/src/lib/api-device-modals.tsx new file mode 100644 index 000000000..551958e70 --- /dev/null +++ b/libs/generic-view/feature/src/lib/api-device-modals.tsx @@ -0,0 +1,40 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import { GenericThemeProvider } from "generic-view/theme" +import { BackupError, ModalBase, ModalCenteredContent } from "generic-view/ui" +import { useDispatch, useSelector } from "react-redux" +import { + cleanBackupProcess, + selectActiveDevice, + selectBackupProcessStatus, +} from "generic-view/store" + +export const ApiDeviceModals: FunctionComponent = () => { + return ( + + + + ) +} + +const BackupErrorModal: FunctionComponent = () => { + const dispatch = useDispatch() + const activeDevice = useSelector(selectActiveDevice) + const backupStatus = useSelector(selectBackupProcessStatus) + const opened = backupStatus === "FAILED" && !activeDevice + + const onClose = () => { + dispatch(cleanBackupProcess()) + } + return ( + + + + + + ) +} diff --git a/libs/generic-view/store/src/index.ts b/libs/generic-view/store/src/index.ts index 744653a29..9e5c3adfd 100644 --- a/libs/generic-view/store/src/index.ts +++ b/libs/generic-view/store/src/index.ts @@ -13,3 +13,9 @@ export * from "./lib/modals/reducer" export * from "./lib/modals/actions" export * from "./lib/backup/reducer" export * from "./lib/backup/actions" +export * from "./lib/backup/create-backup.action" +export * from "./lib/backup/refresh-backup-list.action" +export * from "./lib/file-transfer/reducer" +export * from "./lib/file-transfer/actions" +export * from "./lib/file-transfer/send-file.action" +export * from "./lib/file-transfer/get-file.action" diff --git a/libs/generic-view/store/src/lib/action-names.ts b/libs/generic-view/store/src/lib/action-names.ts index 0a5241497..bb742dd14 100644 --- a/libs/generic-view/store/src/lib/action-names.ts +++ b/libs/generic-view/store/src/lib/action-names.ts @@ -18,5 +18,18 @@ export enum ActionName { CloseAllModals = "generic-modals/close-all-modals", ReplaceModal = "generic-modals/replace-modal", CloseDomainModals = "generic-modals/close-domain-modals", - AddBackupFiles = "generic-backups/add-backup-files", + SetBackupProcess = "generic-backups/set-backup-process", + CleanBackupProcess = "generic-backups/clean-backup-process", + BackupProcessStatus = "generic-backups/backup-process-status", + SetBackupProcessFileStatus = "generic-backups/set-backup-process-file-status", + CreateBackup = "generic-backups/create-backup", + RefreshBackupList = "generic-backups/refresh-backup-list", + FileTransferSend = "generic-file-transfer/send", + PreFileTransferSend = "generic-file-transfer/pre-send", + ChunkFileTransferSend = "generic-file-transfer/chunk-sent", + ClearFileTransferSendError = "generic-file-transfer/clear-send-errors", + PreFileTransferGet = "generic-file-transfer/pre-get", + FileTransferGet = "generic-file-transfer/get", + ChunkFileTransferGet = "generic-file-transfer/chunk-get", + ClearFileTransferGetError = "generic-file-transfer/clear-get-errors", } diff --git a/libs/generic-view/store/src/lib/backup/actions.ts b/libs/generic-view/store/src/lib/backup/actions.ts index 1dea01132..329b84967 100644 --- a/libs/generic-view/store/src/lib/backup/actions.ts +++ b/libs/generic-view/store/src/lib/backup/actions.ts @@ -4,8 +4,24 @@ */ import { createAction } from "@reduxjs/toolkit" -import { Backup } from "./reducer" +import { + BackupProcess, + BackupProcessFileStatus, + BackupProcessStatus, +} from "./reducer" import { ActionName } from "../action-names" -export const setBackupFiles = createAction(ActionName.AddBackupFiles) +export const setBackupProcess = createAction( + ActionName.SetBackupProcess +) +export const cleanBackupProcess = createAction(ActionName.CleanBackupProcess) + +export const setBackupProcessFileStatus = createAction<{ + feature: string + status: BackupProcessFileStatus +}>(ActionName.SetBackupProcessFileStatus) + +export const setBackupProcessStatus = createAction( + ActionName.BackupProcessStatus +) diff --git a/libs/generic-view/store/src/lib/backup/create-backup.action.ts b/libs/generic-view/store/src/lib/backup/create-backup.action.ts new file mode 100644 index 000000000..414a1a91c --- /dev/null +++ b/libs/generic-view/store/src/lib/backup/create-backup.action.ts @@ -0,0 +1,166 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createAsyncThunk } from "@reduxjs/toolkit" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" +import { + checkPreBackupRequest, + postBackupRequest, + saveBackupFileRequest, + sendClearRequest, + startPreBackupRequest, +} from "device/feature" + +import { ActionName } from "../action-names" +import { getFile } from "../file-transfer/get-file.action" +import { + setBackupProcess, + setBackupProcessFileStatus, + setBackupProcessStatus, +} from "./actions" +import { refreshBackupList } from "./refresh-backup-list.action" + +export const createBackup = createAsyncThunk< + undefined, + { + features: string[] + password?: string + }, + { state: ReduxRootState } +>( + ActionName.CreateBackup, + async ( + { features, password }, + { getState, dispatch, rejectWithValue, signal } + ) => { + let aborted = false + + const abortListener = async () => { + signal.removeEventListener("abort", abortListener) + aborted = true + abortFileRequest?.() + await clearTransfers?.() + if (backupId && deviceId) { + await postBackupRequest(backupId, deviceId) + } + } + signal.addEventListener("abort", abortListener) + + if (aborted) { + return rejectWithValue(undefined) + } + + dispatch( + setBackupProcess({ + status: "PRE_BACKUP", + featureFilesTransfer: features.reduce((acc, item) => { + return { ...acc, [item]: { done: false } } + }, {}), + }) + ) + + const deviceId = getState().genericViews.activeDevice + + if (deviceId === undefined || aborted) { + return rejectWithValue(undefined) + } + + const startPreBackupResponse = await startPreBackupRequest( + features, + deviceId + ) + + if (!startPreBackupResponse.ok || aborted) { + console.log(startPreBackupResponse.error) + return rejectWithValue(undefined) + } + + const backupId = startPreBackupResponse.data.backupId + let backupFeaturesFiles = startPreBackupResponse.data.features + let abortFileRequest: VoidFunction + const featureToTransferId: Record = {} + + const clearTransfers = () => { + return Promise.all( + Object.values(featureToTransferId).map(async (transferId) => { + await sendClearRequest(transferId) + }) + ) + } + + while (backupFeaturesFiles === undefined) { + if (aborted) { + return rejectWithValue(undefined) + } + const checkPreBackupResponse = await checkPreBackupRequest( + backupId, + features, + deviceId + ) + + if (!checkPreBackupResponse.ok) { + console.log(checkPreBackupResponse.error) + return rejectWithValue(undefined) + } + + backupFeaturesFiles = checkPreBackupResponse.data.features + } + dispatch(setBackupProcessStatus("FILES_TRANSFER")) + + for (let i = 0; i < features.length; ++i) { + if (aborted) { + return rejectWithValue(undefined) + } + const feature = features[i] + + dispatch(setBackupProcessFileStatus({ feature, status: "IN_PROGRESS" })) + const filePromise = dispatch( + getFile({ + deviceId, + filePath: backupFeaturesFiles[feature], + targetPath: "", + }) + ) + abortFileRequest = filePromise.abort + const file = await filePromise + if ( + file.meta.requestStatus === "fulfilled" && + file.payload && + "transferId" in file.payload + ) { + featureToTransferId[feature] = file.payload.transferId + dispatch(setBackupProcessFileStatus({ feature, status: "DONE" })) + } else if (!aborted) { + console.log("Error while downloading file") + await clearTransfers() + await postBackupRequest(backupId, deviceId) + return rejectWithValue(undefined) + } + } + if (aborted) { + return rejectWithValue(undefined) + } + dispatch(setBackupProcessStatus("SAVE_FILE")) + const saveBackupFileResponse = await saveBackupFileRequest( + featureToTransferId, + deviceId, + password + ) + + if (!saveBackupFileResponse.ok) { + console.log("Error while saving file") + await clearTransfers() + await postBackupRequest(backupId, deviceId) + return rejectWithValue(undefined) + } + + dispatch(refreshBackupList()) + + if (!aborted) { + await postBackupRequest(backupId, deviceId) + } + return undefined + } +) diff --git a/libs/generic-view/store/src/lib/backup/reducer.ts b/libs/generic-view/store/src/lib/backup/reducer.ts index b180a2878..8763ca3a2 100644 --- a/libs/generic-view/store/src/lib/backup/reducer.ts +++ b/libs/generic-view/store/src/lib/backup/reducer.ts @@ -4,43 +4,91 @@ */ import { createReducer } from "@reduxjs/toolkit" -import { setBackupFiles } from "./actions" +import { + cleanBackupProcess, + setBackupProcess, + setBackupProcessFileStatus, + setBackupProcessStatus, +} from "./actions" +import { createBackup } from "./create-backup.action" +import { refreshBackupList } from "./refresh-backup-list.action" export interface Backup { fileName: string date: Date - features: string[] - device: { - serialNumber: string - vendorId: string - productId: string - osVersion: string - } + serialNumber: string +} + +export type BackupProcessStatus = + | "PRE_BACKUP" + | "FILES_TRANSFER" + | "SAVE_FILE" + | "DONE" + | "FAILED" + +export type BackupProcessFileStatus = "PENDING" | "IN_PROGRESS" | "DONE" + +export interface BackupProcess { + status: "PRE_BACKUP" | "FILES_TRANSFER" | "SAVE_FILE" | "DONE" | "FAILED" + featureFilesTransfer: Record< + string, + { transferId?: number; status: BackupProcessFileStatus } + > } interface BackupState { - files: Backup[] + lastBackupRefresh: number + backups: Backup[] + backupProcess?: BackupProcess } const initialState: BackupState = { - // Demo data - files: [ - { - fileName: "backup1.json", - date: new Date(), - features: ["calls", "messages"], - device: { - serialNumber: "0123456789ABCDEF", - vendorId: "0e8d", - productId: "2006", - osVersion: "12", - }, - }, - ], + lastBackupRefresh: 0, + backups: [], } export const genericBackupsReducer = createReducer(initialState, (builder) => { - builder.addCase(setBackupFiles, (state, action) => { - state.files = action.payload + builder.addCase(cleanBackupProcess, (state, action) => { + delete state.backupProcess + }) + builder.addCase(setBackupProcess, (state, action) => { + state.backupProcess = action.payload + }) + builder.addCase(setBackupProcessFileStatus, (state, action) => { + if (state.backupProcess) { + state.backupProcess.featureFilesTransfer[action.payload.feature].status = + action.payload.status + } + }) + builder.addCase(setBackupProcessStatus, (state, action) => { + if (state.backupProcess) { + state.backupProcess.status = action.payload + } + }) + builder.addCase(createBackup.rejected, (state, action) => { + if (state.backupProcess) { + state.backupProcess.status = "FAILED" + } else { + state.backupProcess = { + status: "FAILED", + featureFilesTransfer: {}, + } + } + }) + builder.addCase(createBackup.fulfilled, (state, action) => { + if (state.backupProcess) { + state.backupProcess.status = "DONE" + } else { + state.backupProcess = { + status: "DONE", + featureFilesTransfer: {}, + } + } + }) + builder.addCase(refreshBackupList.fulfilled, (state, action) => { + if (state.lastBackupRefresh < action.payload.refreshTimestamp) { + state.lastBackupRefresh = action.payload.refreshTimestamp + state.backups = action.payload.backups + } }) }) diff --git a/libs/generic-view/store/src/lib/backup/refresh-backup-list.action.ts b/libs/generic-view/store/src/lib/backup/refresh-backup-list.action.ts new file mode 100644 index 000000000..3fe54fec8 --- /dev/null +++ b/libs/generic-view/store/src/lib/backup/refresh-backup-list.action.ts @@ -0,0 +1,60 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createAsyncThunk } from "@reduxjs/toolkit" +import { DeviceId } from "Core/device/constants/device-id" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" +import { readBackupDirectoryRequest } from "device/feature" + +import { ActionName } from "../action-names" +import { Backup } from "./reducer" + +export const refreshBackupList = createAsyncThunk< + { refreshTimestamp: number; backups: Backup[]; deviceId: DeviceId }, + undefined, + { state: ReduxRootState } +>( + ActionName.RefreshBackupList, + async (_, { getState, dispatch, rejectWithValue }) => { + const refreshTimestamp = new Date().getTime() + + const deviceId = getState().genericViews.activeDevice + if (!deviceId) { + return rejectWithValue(undefined) + } + + const backupsList = await readBackupDirectoryRequest(deviceId) + + if (!backupsList.ok) { + return rejectWithValue(undefined) + } + + const backups = + (backupsList.data + .map((item) => { + const isFormatValid = /^\d+[_][a-zA-Z0-9]+[.]mcbackup$/i.test(item) + + if (!isFormatValid) { + return null + } + + const [fileName] = item.split(".") + const [timestamp, serialNumber] = fileName.split("_") + const result: Backup = { + date: new Date(Number(timestamp)), + fileName: item, + serialNumber, + } + return result + }) + .filter(Boolean) as Backup[]) ?? [] + + return { + refreshTimestamp, + backups, + deviceId, + } + } +) diff --git a/libs/generic-view/store/src/lib/file-transfer/actions.ts b/libs/generic-view/store/src/lib/file-transfer/actions.ts new file mode 100644 index 000000000..5f45b9914 --- /dev/null +++ b/libs/generic-view/store/src/lib/file-transfer/actions.ts @@ -0,0 +1,29 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createAction } from "@reduxjs/toolkit" +import { ActionName } from "../action-names" +import { FileProgress } from "./reducer" +import { SendFileErrorPayload } from "./send-file.action" +import { GetFileErrorPayload } from "Libs/generic-view/store/src" + +export const fileTransferSendPrepared = createAction< + Pick +>(ActionName.PreFileTransferSend) +export const fileTransferChunkSent = createAction< + Pick +>(ActionName.ChunkFileTransferSend) +export const fileTransferGetPrepared = createAction< + Pick +>(ActionName.PreFileTransferGet) +export const fileTransferChunkGet = createAction< + Pick +>(ActionName.ChunkFileTransferGet) +export const clearSendErrors = createAction( + ActionName.ClearFileTransferSendError +) +export const clearGetErrors = createAction( + ActionName.ClearFileTransferGetError +) diff --git a/libs/generic-view/store/src/lib/file-transfer/get-file.action.ts b/libs/generic-view/store/src/lib/file-transfer/get-file.action.ts new file mode 100644 index 000000000..c5dd38ea7 --- /dev/null +++ b/libs/generic-view/store/src/lib/file-transfer/get-file.action.ts @@ -0,0 +1,129 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createAsyncThunk } from "@reduxjs/toolkit" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" +import { + getFileRequest, + saveFileRequest, + sendClearRequest, + startPreGetFileRequest, +} from "device/feature" +import { DeviceId } from "Core/device/constants/device-id" +import { ActionName } from "../action-names" +import { fileTransferChunkGet, fileTransferGetPrepared } from "./actions" +import { AppError, AppErrorType } from "Core/core/errors" +import { GeneralError, PreTransferGet } from "device/models" +import { Result } from "Core/core/builder" + +export type GetFileErrorPayload = { + transferId?: number + filePath: string +} + +interface GetFileError { + deviceId: DeviceId + error: AppError +} + +export const getFile = createAsyncThunk< + { transferId: number }, + { + deviceId: DeviceId + filePath: string + targetPath: string + preTransfer?: PreTransferGet + }, + { state: ReduxRootState; rejectValue: GetFileError | undefined } +>( + ActionName.FileTransferGet, + async ( + { deviceId, filePath, targetPath, preTransfer }, + { rejectWithValue, dispatch, signal } + ) => { + let aborted = false + + const abortListener = async () => { + signal.removeEventListener("abort", abortListener) + aborted = true + const transferId = preTransferResponse?.ok + ? preTransferResponse.data.transferId + : preTransferResponse?.error.payload + if (transferId) { + await sendClearRequest(transferId) + } + } + signal.addEventListener("abort", abortListener) + + if (aborted) { + return rejectWithValue(undefined) + } + + const preTransferResponse = preTransfer + ? Result.success(preTransfer) + : await startPreGetFileRequest(filePath, deviceId) + + if (preTransferResponse.ok && !aborted) { + const { transferId, chunkSize, fileSize } = preTransferResponse.data + const chunksCount = Math.ceil(fileSize / chunkSize) + dispatch( + fileTransferGetPrepared({ + transferId, + chunksCount, + filePath, + }) + ) + + for (let chunkNumber = 1; chunkNumber <= chunksCount; chunkNumber++) { + if (aborted) { + return rejectWithValue(undefined) + } + const { ok, error } = await getFileRequest(transferId, chunkNumber) + if (!ok) { + await sendClearRequest(transferId) + return rejectWithValue({ + deviceId, + error: { + ...error, + payload: { + transferId, + filePath, + }, + }, + }) + } + dispatch( + fileTransferChunkGet({ + transferId, + chunksTransferred: chunkNumber, + }) + ) + } + + if (aborted) { + return rejectWithValue(undefined) + } + + if (targetPath) { + await saveFileRequest(targetPath, transferId) + await sendClearRequest(transferId) + } + return { transferId } + } else { + return rejectWithValue({ + deviceId, + error: { + name: GeneralError.IncorrectResponse, + type: GeneralError.IncorrectResponse, + message: "Incorrect response", + payload: { + transferId: preTransferResponse.error?.payload, + filePath, + }, + }, + }) + } + } +) diff --git a/libs/generic-view/store/src/lib/file-transfer/reducer.ts b/libs/generic-view/store/src/lib/file-transfer/reducer.ts new file mode 100644 index 000000000..443597039 --- /dev/null +++ b/libs/generic-view/store/src/lib/file-transfer/reducer.ts @@ -0,0 +1,129 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createReducer } from "@reduxjs/toolkit" +import { AppErrorType } from "Core/core/errors" +import { + clearGetErrors, + clearSendErrors, + fileTransferChunkGet, + fileTransferChunkSent, + fileTransferGetPrepared, + fileTransferSendPrepared, +} from "./actions" +import { sendFile } from "./send-file.action" +import { getFile } from "./get-file.action" + +export interface FileTransferError { + code?: AppErrorType + message?: string + filePath?: string + transferId?: number +} + +export interface FileProgress { + transferId: number + chunksCount: number + chunksTransferred: number + filePath?: string +} + +interface FileTransferState { + sendingFilesProgress: { + [transferId: number]: FileProgress + } + sendingErrors?: FileTransferError[] + receivingFilesProgress: { + [transferId: number]: FileProgress + } + receivingErrors?: FileTransferError[] +} + +const initialState: FileTransferState = { + sendingFilesProgress: {}, + sendingErrors: [], + receivingFilesProgress: {}, + receivingErrors: [], +} + +export const genericFileTransferReducer = createReducer( + initialState, + (builder) => { + builder.addCase(fileTransferSendPrepared, (state, action) => { + state.sendingFilesProgress[action.payload.transferId] = { + transferId: action.payload.transferId, + chunksCount: action.payload.chunksCount, + chunksTransferred: 0, + } + }) + builder.addCase(fileTransferChunkSent, (state, action) => { + state.sendingFilesProgress[action.payload.transferId].chunksTransferred = + action.payload.chunksTransferred + }) + builder.addCase(sendFile.fulfilled, (state, action) => { + delete state.sendingFilesProgress[action.payload.transferId] + }) + builder.addCase(sendFile.rejected, (state, action) => { + const { transferId, filePath } = action.payload?.error.payload || {} + if (transferId) { + delete state.sendingFilesProgress[transferId] + } + state.sendingErrors?.push({ + code: action.payload?.error.type, + message: action.payload?.error.message, + transferId, + filePath, + }) + }) + builder.addCase(clearSendErrors, (state, action) => { + state.sendingErrors = state.sendingErrors?.filter( + (error) => error.transferId !== action.payload.transferId + ) + }) + builder.addCase(fileTransferGetPrepared, (state, action) => { + state.receivingFilesProgress[action.payload.transferId] = { + transferId: action.payload.transferId, + chunksCount: action.payload.chunksCount, + chunksTransferred: 0, + filePath: action.payload.filePath, + } + }) + builder.addCase(fileTransferChunkGet, (state, action) => { + state.receivingFilesProgress[ + action.payload.transferId + ].chunksTransferred = action.payload.chunksTransferred + }) + builder.addCase(getFile.fulfilled, (state, action) => { + delete state.receivingFilesProgress[action.payload.transferId] + }) + builder.addCase(getFile.rejected, (state, action) => { + if (action.meta.aborted) { + const transfer = Object.entries(state.receivingFilesProgress).find( + ([, item]) => item.filePath === action.meta.arg.filePath + ) + if (transfer) { + const transferId = Number(transfer[0]) + delete state.receivingFilesProgress[transferId] + } + } else { + const { transferId, filePath } = action.payload?.error.payload || {} + if (transferId) { + delete state.receivingFilesProgress[transferId] + } + state.receivingErrors?.push({ + code: action.payload?.error.type, + message: action.payload?.error.message, + transferId, + filePath: action.payload?.error.payload?.filePath, + }) + } + }) + builder.addCase(clearGetErrors, (state, action) => { + state.receivingErrors = state.receivingErrors?.filter( + (error) => error.transferId !== action.payload.transferId + ) + }) + } +) diff --git a/libs/generic-view/store/src/lib/file-transfer/send-file.action.ts b/libs/generic-view/store/src/lib/file-transfer/send-file.action.ts new file mode 100644 index 000000000..78782564b --- /dev/null +++ b/libs/generic-view/store/src/lib/file-transfer/send-file.action.ts @@ -0,0 +1,91 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createAsyncThunk } from "@reduxjs/toolkit" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" +import { + sendClearRequest, + sendFileRequest, + startPreSendFileRequest, +} from "device/feature" +import { DeviceId } from "Core/device/constants/device-id" +import { ActionName } from "../action-names" +import { fileTransferChunkSent, fileTransferSendPrepared } from "./actions" +import { AppError, AppErrorType } from "Core/core/errors" +import { GeneralError } from "device/models" + +export type SendFileErrorPayload = { + transferId?: number + filePath: string +} + +interface SendFileError { + deviceId: DeviceId + error: AppError +} + +export const sendFile = createAsyncThunk< + { transferId: number }, + { deviceId: DeviceId; filePath: string; targetPath: string }, + { state: ReduxRootState; rejectValue: SendFileError } +>( + ActionName.FileTransferSend, + async ({ deviceId, filePath, targetPath }, { rejectWithValue, dispatch }) => { + const preTransferResponse = await startPreSendFileRequest( + filePath, + targetPath, + deviceId + ) + + if (preTransferResponse.ok) { + const { transferId, chunksCount } = preTransferResponse.data + dispatch( + fileTransferSendPrepared({ + transferId, + chunksCount, + }) + ) + + for (let chunkNumber = 1; chunkNumber <= chunksCount; chunkNumber++) { + // TODO: consider using signal to abort + const { ok, error } = await sendFileRequest(transferId, chunkNumber) + if (!ok) { + await sendClearRequest(transferId) + return rejectWithValue({ + deviceId, + error: { + ...error, + payload: { + transferId, + filePath, + }, + }, + }) + } + dispatch( + fileTransferChunkSent({ + transferId, + chunksTransferred: chunkNumber, + }) + ) + } + await sendClearRequest(transferId) + return { transferId } + } else { + return rejectWithValue({ + deviceId, + error: { + name: GeneralError.IncorrectResponse, + type: GeneralError.IncorrectResponse, + message: "Incorrect response", + payload: { + transferId: preTransferResponse.error.payload, + filePath, + }, + }, + }) + } + } +) diff --git a/libs/generic-view/store/src/lib/hooks/index.ts b/libs/generic-view/store/src/lib/hooks/index.ts index 1ad759f15..29cc2b7aa 100644 --- a/libs/generic-view/store/src/lib/hooks/index.ts +++ b/libs/generic-view/store/src/lib/hooks/index.ts @@ -7,3 +7,5 @@ export * from "./use-api-serial-port-listeners" export * from "./use-outbox" export * from "./use-screen-title" export * from "./use-locked-device-handler" +export * from "./use-backup-list" +export * from "./use-app-events-listeners" diff --git a/libs/generic-view/store/src/lib/hooks/use-api-serial-port-listeners.ts b/libs/generic-view/store/src/lib/hooks/use-api-serial-port-listeners.ts index e32ea0e89..3e7585bbd 100644 --- a/libs/generic-view/store/src/lib/hooks/use-api-serial-port-listeners.ts +++ b/libs/generic-view/store/src/lib/hooks/use-api-serial-port-listeners.ts @@ -5,15 +5,19 @@ import { Device } from "Core/device-manager/reducers/device-manager.interface" import { useEffect } from "react" -import { useDispatch } from "react-redux" +import { useDispatch, useSelector } from "react-redux" import { answerMain, DeviceManagerMainEvent } from "shared/utils" import { detachDevice } from "../views/actions" import { getAPIConfig } from "../get-api-config" import { Dispatch } from "Core/__deprecated__/renderer/store" import { DeviceType } from "Core/device" +import { setBackupProcessStatus } from "../backup/actions" +import { closeAllModals } from "../modals/actions" +import { selectBackupProcessStatus } from "../selectors" export const useAPISerialPortListeners = () => { const dispatch = useDispatch() + const backupProcess = useSelector(selectBackupProcessStatus) useEffect(() => { const unregisterFailListener = answerMain( @@ -39,12 +43,16 @@ export const useAPISerialPortListeners = () => { ) const unregisterDetachedListener = answerMain( DeviceManagerMainEvent.DeviceDetached, - (properties) => { + async (properties) => { const { id, deviceType } = properties if (deviceType !== DeviceType.APIDevice) { return } dispatch(detachDevice({ deviceId: id })) + dispatch(closeAllModals()) + if (backupProcess) { + dispatch(setBackupProcessStatus("FAILED")) + } } ) return () => { @@ -52,5 +60,5 @@ export const useAPISerialPortListeners = () => { unregisterConnectListener() unregisterFailListener() } - }, [dispatch]) + }, [backupProcess, dispatch]) } diff --git a/libs/generic-view/store/src/lib/hooks/use-app-events-listeners.ts b/libs/generic-view/store/src/lib/hooks/use-app-events-listeners.ts new file mode 100644 index 000000000..251e1942d --- /dev/null +++ b/libs/generic-view/store/src/lib/hooks/use-app-events-listeners.ts @@ -0,0 +1,27 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { useEffect } from "react" +import { useDispatch } from "react-redux" +import { answerMain, AppEvents } from "shared/utils" +import { Dispatch } from "Core/__deprecated__/renderer/store" +import { refreshBackupList } from "../backup/refresh-backup-list.action" + +export const useAppEventsListeners = () => { + const dispatch = useDispatch() + + useEffect(() => { + const unregisterWindowFocusedListener = answerMain( + AppEvents.WindowFocused, + () => { + dispatch(refreshBackupList()) + } + ) + + return () => { + unregisterWindowFocusedListener() + } + }, [dispatch]) +} diff --git a/libs/generic-view/store/src/lib/hooks/use-backup-list.ts b/libs/generic-view/store/src/lib/hooks/use-backup-list.ts new file mode 100644 index 000000000..e1292c44c --- /dev/null +++ b/libs/generic-view/store/src/lib/hooks/use-backup-list.ts @@ -0,0 +1,24 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { Dispatch } from "Core/__deprecated__/renderer/store" +import { useEffect } from "react" +import { useDispatch } from "react-redux" +import { refreshBackupList } from "../backup/refresh-backup-list.action" + +export const useBackupList = () => { + const dispatch = useDispatch() + + useEffect(() => { + const interval = setInterval(() => { + dispatch(refreshBackupList()) + }, 5000) + return () => { + clearInterval(interval) + } + }, [dispatch]) + + return undefined +} diff --git a/libs/generic-view/store/src/lib/hooks/use-outbox.ts b/libs/generic-view/store/src/lib/hooks/use-outbox.ts index ec8e5dc8c..e3024cbcc 100644 --- a/libs/generic-view/store/src/lib/hooks/use-outbox.ts +++ b/libs/generic-view/store/src/lib/hooks/use-outbox.ts @@ -31,3 +31,8 @@ export const useOutbox = () => { // eslint-disable-next-line react-hooks/exhaustive-deps }, [activeDevice, lastRefreshTimestamp]) } + +export const OutboxWrapper = () => { + useOutbox() + return null +} diff --git a/libs/generic-view/store/src/lib/selectors/active-device-configuration.ts b/libs/generic-view/store/src/lib/selectors/active-device-configuration.ts new file mode 100644 index 000000000..9cc922d87 --- /dev/null +++ b/libs/generic-view/store/src/lib/selectors/active-device-configuration.ts @@ -0,0 +1,21 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createSelector } from "@reduxjs/toolkit" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" +import { selectActiveDevice } from "./active-device" + +export const selectActiveDeviceConfiguration = createSelector( + [ + selectActiveDevice, + (state: ReduxRootState) => state.genericViews.devicesConfiguration, + ], + (activeDevice, devicesConfiguration) => { + if (activeDevice) { + return devicesConfiguration[activeDevice] + } + return undefined + } +) diff --git a/libs/generic-view/store/src/lib/selectors/active-device-features.ts b/libs/generic-view/store/src/lib/selectors/active-device-features.ts index 339532184..606ffed49 100644 --- a/libs/generic-view/store/src/lib/selectors/active-device-features.ts +++ b/libs/generic-view/store/src/lib/selectors/active-device-features.ts @@ -4,18 +4,11 @@ */ import { createSelector } from "@reduxjs/toolkit" -import { ReduxRootState } from "Core/__deprecated__/renderer/store" -import { selectActiveDevice } from "./active-device" +import { selectActiveDeviceConfiguration } from "./active-device-configuration" export const selectActiveDeviceFeatures = createSelector( - [ - selectActiveDevice, - (state: ReduxRootState) => state.genericViews.devicesConfiguration, - ], - (activeDevice, devicesConfiguration) => { - if (activeDevice) { - return devicesConfiguration[activeDevice].features - } - return undefined + selectActiveDeviceConfiguration, + (activeDeviceConfiguration) => { + return activeDeviceConfiguration?.features } ) diff --git a/libs/generic-view/store/src/lib/selectors/backup-location.ts b/libs/generic-view/store/src/lib/selectors/backup-location.ts new file mode 100644 index 000000000..5f1cfb38e --- /dev/null +++ b/libs/generic-view/store/src/lib/selectors/backup-location.ts @@ -0,0 +1,18 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createSelector } from "@reduxjs/toolkit" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" +import { selectActiveDeviceConfiguration } from "./active-device-configuration" + +export const selectBackupLocation = createSelector( + selectActiveDeviceConfiguration, + (state: ReduxRootState) => state.settings.osBackupLocation, + (deviceConfiguration, location) => { + const { vendorId, productId } = deviceConfiguration?.apiConfig || {} + const deviceDirectory = `${vendorId}_${productId}` + return vendorId && productId ? `${location}/${deviceDirectory}` : location + } +) diff --git a/libs/generic-view/store/src/lib/selectors/backup-process-status.ts b/libs/generic-view/store/src/lib/selectors/backup-process-status.ts new file mode 100644 index 000000000..4825f5b87 --- /dev/null +++ b/libs/generic-view/store/src/lib/selectors/backup-process-status.ts @@ -0,0 +1,12 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createSelector } from "@reduxjs/toolkit" +import { ReduxRootState } from "Core/__deprecated__/renderer/store" + +export const selectBackupProcessStatus = createSelector( + (state: ReduxRootState) => state.genericBackups.backupProcess?.status, + (status) => status +) diff --git a/libs/generic-view/store/src/lib/selectors/backup-progress.ts b/libs/generic-view/store/src/lib/selectors/backup-progress.ts new file mode 100644 index 000000000..e0e65d9e6 --- /dev/null +++ b/libs/generic-view/store/src/lib/selectors/backup-progress.ts @@ -0,0 +1,40 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { createSelector } from "@reduxjs/toolkit" + +import { ReduxRootState } from "Core/__deprecated__/renderer/store" + +export const backupProgress = createSelector( + [(state: ReduxRootState) => state.genericBackups.backupProcess], + (backupProcess) => { + if (!backupProcess) { + return { progress: 0 } + } + + if (backupProcess.status === "DONE") { + return { progress: 100 } + } + + const features = Object.values(backupProcess.featureFilesTransfer) + const downloadedFilesCount = features.filter( + (item) => item.status === "DONE" + ).length + + if (features.length <= downloadedFilesCount) { + return { progress: 90 } + } + + const [featureInProgress] = + Object.entries(backupProcess.featureFilesTransfer).find( + ([, item]) => item.status === "IN_PROGRESS" + ) ?? [] + + return { + progress: Math.floor(10 + (downloadedFilesCount / features.length) * 80), + featureInProgress, + } + } +) diff --git a/libs/generic-view/store/src/lib/selectors/device-backups.ts b/libs/generic-view/store/src/lib/selectors/device-backups.ts index baa959d05..c0e976ef8 100644 --- a/libs/generic-view/store/src/lib/selectors/device-backups.ts +++ b/libs/generic-view/store/src/lib/selectors/device-backups.ts @@ -9,10 +9,10 @@ import { ReduxRootState } from "Core/__deprecated__/renderer/store" export const selectDeviceBackups = createSelector( selectActiveDevice, - (state: ReduxRootState) => state.genericBackups.files, + (state: ReduxRootState) => state.genericBackups.backups, (activeDevice, backups) => { return backups - .filter((backup) => backup.device.serialNumber === activeDevice) + .filter((backup) => backup.serialNumber === activeDevice) .sort((a, b) => b.date.getTime() - a.date.getTime()) } ) diff --git a/libs/generic-view/store/src/lib/selectors/index.ts b/libs/generic-view/store/src/lib/selectors/index.ts index b93425e6c..c18301dd2 100644 --- a/libs/generic-view/store/src/lib/selectors/index.ts +++ b/libs/generic-view/store/src/lib/selectors/index.ts @@ -12,3 +12,6 @@ export * from "./component" export * from "./view" export * from "./api-error" export * from "./device-backups" +export * from "./backup-progress" +export * from "./backup-process-status" +export * from "./backup-location" diff --git a/libs/generic-view/store/src/lib/views/reducer.ts b/libs/generic-view/store/src/lib/views/reducer.ts index ae7852056..4a24af18f 100644 --- a/libs/generic-view/store/src/lib/views/reducer.ts +++ b/libs/generic-view/store/src/lib/views/reducer.ts @@ -50,6 +50,7 @@ interface GenericState { lastResponse: unknown lastRefresh?: number activeDevice?: DeviceId + pendingDevice?: DeviceId devicesConfiguration: Record apiErrors: Record } @@ -152,12 +153,17 @@ export const genericViewsReducer = createReducer(initialState, (builder) => { : {}), } } + if (state.activeDevice === undefined && state.pendingDevice === deviceId) { + state.activeDevice = deviceId + state.pendingDevice = undefined + } }) builder.addCase(activateDevice, (state, action) => { const { deviceId } = action.payload state.activeDevice = state.devicesConfiguration?.[deviceId]?.apiConfig ? deviceId : undefined + state.pendingDevice = deviceId }) builder.addCase(detachDevice, (state, action) => { const { deviceId } = action.payload @@ -167,6 +173,9 @@ export const genericViewsReducer = createReducer(initialState, (builder) => { if (state.activeDevice === deviceId) { state.activeDevice = undefined } + if (state.pendingDevice === deviceId) { + state.pendingDevice = undefined + } }) builder.addCase(getOutboxData.fulfilled, (state, action) => { const { deviceId, timestamp } = action.payload diff --git a/libs/generic-view/theme/src/lib/color.ts b/libs/generic-view/theme/src/lib/color.ts index bc48ef3ed..289ec6d75 100644 --- a/libs/generic-view/theme/src/lib/color.ts +++ b/libs/generic-view/theme/src/lib/color.ts @@ -13,4 +13,5 @@ export const color = { grey4: "#D2D6DB", grey5: "#F4F5F6", grey6: "#FBFBFB", + red1: "#E96A6A", } as const diff --git a/libs/generic-view/theme/src/lib/font-size.ts b/libs/generic-view/theme/src/lib/font-size.ts index 447b66624..1eb3aeea6 100644 --- a/libs/generic-view/theme/src/lib/font-size.ts +++ b/libs/generic-view/theme/src/lib/font-size.ts @@ -17,4 +17,6 @@ export const fontSize = { buttonLink: "1.4rem", buttonText: "1.2rem", detailText: "1.2rem", + labelText: "1.2rem", + modalTitle: "2rem", } as const diff --git a/libs/generic-view/theme/src/lib/generic-theme-provider.tsx b/libs/generic-view/theme/src/lib/generic-theme-provider.tsx index 43b5b58ea..29b13ff39 100644 --- a/libs/generic-view/theme/src/lib/generic-theme-provider.tsx +++ b/libs/generic-view/theme/src/lib/generic-theme-provider.tsx @@ -58,5 +58,23 @@ const GlobalStyle = createGlobalStyle<{ theme: Theme }>` box-shadow: 0 2rem 10rem 0 ${({ theme }) => theme.color.black + "26"}; display: flex; flex-direction: column; + position: relative; + + .modal-close-button:nth-child(2) { + display: none; + } + } + + *::-webkit-scrollbar { + width: 0.5rem; + } + + *::-webkit-scrollbar-track { + background-color: transparent; + } + + *::-webkit-scrollbar-thumb { + background-color: ${({ theme }) => theme.color.grey2}; + border-radius: 0.25rem; } ` diff --git a/libs/generic-view/theme/src/lib/line-height.ts b/libs/generic-view/theme/src/lib/line-height.ts index 7739e2c77..4b4357c11 100644 --- a/libs/generic-view/theme/src/lib/line-height.ts +++ b/libs/generic-view/theme/src/lib/line-height.ts @@ -17,4 +17,6 @@ export const lineHeight = { buttonLink: "2.2rem", buttonText: "2rem", detailText: "2rem", + labelText: "2rem", + modalTitle: "3.2rem", } as const diff --git a/libs/generic-view/ui/src/index.ts b/libs/generic-view/ui/src/index.ts index 9529bdef9..16d34ebeb 100644 --- a/libs/generic-view/ui/src/index.ts +++ b/libs/generic-view/ui/src/index.ts @@ -13,7 +13,9 @@ import { buttons } from "./lib/buttons/buttons" export { default as Icon } from "./lib/icon/icon" export * from "./lib/api-connection-demo" export * from "./lib/interactive/modal/modal-base" +export * from "./lib/interactive/modal/modal-helpers" export * from "./lib/shared/shared" +export * from "./lib/predefined/backup/backup-error" const apiComponents = { ...predefinedComponents, @@ -27,4 +29,3 @@ const apiComponents = { export default apiComponents export type APIComponents = typeof apiComponents - diff --git a/libs/generic-view/ui/src/lib/buttons/button-base/button-base.tsx b/libs/generic-view/ui/src/lib/buttons/button-base/button-base.tsx index 5924e1eb3..24b12061a 100644 --- a/libs/generic-view/ui/src/lib/buttons/button-base/button-base.tsx +++ b/libs/generic-view/ui/src/lib/buttons/button-base/button-base.tsx @@ -11,11 +11,12 @@ import { DefaultButton } from "../../shared/button" interface Props extends HTMLAttributes { action: ButtonAction viewKey?: string + disabled?: boolean } export const ButtonBase: FunctionComponent = ({ action, ...props }) => { const callButtonAction = useButtonAction(props.viewKey as string) const callAction = () => callButtonAction(action) - return + return } diff --git a/libs/generic-view/ui/src/lib/buttons/button-base/use-button-action.ts b/libs/generic-view/ui/src/lib/buttons/button-base/use-button-action.ts index 26dc7a269..ffd4aca8e 100644 --- a/libs/generic-view/ui/src/lib/buttons/button-base/use-button-action.ts +++ b/libs/generic-view/ui/src/lib/buttons/button-base/use-button-action.ts @@ -7,20 +7,31 @@ import { closeAllModals, closeDomainModals, closeModal, + createBackup, openModal, replaceModal, + selectActiveDevice, + sendFile, useScreenTitle, } from "generic-view/store" import { ButtonAction } from "generic-view/utils" -import { useDispatch } from "react-redux" +import { useDispatch, useSelector } from "react-redux" import { useHistory } from "react-router" +import { Dispatch, ReduxRootState } from "Core/__deprecated__/renderer/store" +import { getPaths } from "shared/app-state" +import { PayloadAction } from "@reduxjs/toolkit" +import { ResultObject } from "Core/core/builder" +import { platform } from "Core/__deprecated__/renderer/utils/platform" export const useButtonAction = (viewKey: string) => { - const dispatch = useDispatch() + const dispatch = useDispatch() const navigate = useHistory() const currentViewName = useScreenTitle(viewKey) + const restore = useButtonRestoreAction() + const backup = useButtonBackupAction() - return (action: ButtonAction) => { + return (action?: ButtonAction) => { + if (!action) return switch (action.type) { case "open-modal": dispatch( @@ -57,6 +68,60 @@ export const useButtonAction = (viewKey: string) => { }, }) break + case "restore-data": + void restore() + break + case "backup-data": + void backup() + break + case "custom": + action.callback() + break + default: + break + } + } +} + +const useButtonRestoreAction = () => { + const dispatch = useDispatch() + const osBackupLocation = useSelector( + (state: ReduxRootState) => state.settings.osBackupLocation + ) + const deviceId = useSelector(selectActiveDevice) + + return async () => { + const { payload: getPathsPayload } = (await dispatch( + getPaths({ + properties: ["openFile"], + defaultPath: osBackupLocation, + }) + )) as PayloadAction> + const location = getPathsPayload.ok && (getPathsPayload.data as string[])[0] + if (location && deviceId) { + const [fileName] = location + .split(platform.windows() ? "\\" : "/") + .reverse() + dispatch( + sendFile({ + deviceId: deviceId, + filePath: location, + targetPath: `/storage/emulated/0/Documents/${fileName}`, + }) + ) + } + } +} + +const useButtonBackupAction = () => { + const dispatch = useDispatch() + const deviceId = useSelector(selectActiveDevice) + + return async () => { + if (deviceId) { + await dispatch( + createBackup({ features: ["CONTACTS_LIST", "MESSAGES", "CALL_LOG"] }) + ) } } } diff --git a/libs/generic-view/ui/src/lib/buttons/button-primary.tsx b/libs/generic-view/ui/src/lib/buttons/button-primary.tsx index ce58776e9..b40032157 100644 --- a/libs/generic-view/ui/src/lib/buttons/button-primary.tsx +++ b/libs/generic-view/ui/src/lib/buttons/button-primary.tsx @@ -10,10 +10,11 @@ import { ButtonBase } from "./button-base/button-base" import Icon from "../icon/icon" import { withConfig } from "../utils/with-config" -interface Config { +export interface Config { text: string icon?: IconType action: ButtonAction + disabled?: boolean } export const ButtonPrimary: APIFC = ({ @@ -22,7 +23,7 @@ export const ButtonPrimary: APIFC = ({ ...props }) => { return ( - diff --git a/libs/generic-view/ui/src/lib/buttons/button-secondary.tsx b/libs/generic-view/ui/src/lib/buttons/button-secondary.tsx index fd0afc94d..6cee4f449 100644 --- a/libs/generic-view/ui/src/lib/buttons/button-secondary.tsx +++ b/libs/generic-view/ui/src/lib/buttons/button-secondary.tsx @@ -5,15 +5,9 @@ import React from "react" import styled from "styled-components" -import { APIFC, ButtonAction, IconType } from "generic-view/utils" +import { APIFC } from "generic-view/utils" import { withConfig } from "../utils/with-config" -import { ButtonPrimary } from "./button-primary" - -interface Config { - text: string - icon?: IconType - action: ButtonAction -} +import { ButtonPrimary, Config } from "./button-primary" export const ButtonSecondary: APIFC = (props) => { return ) @@ -39,30 +35,39 @@ export const ButtonText: APIFC = ({ export default withConfig(ButtonText) const Button = styled(ButtonBase)<{ $modifiers?: ButtonModifiers[] }>` + color: ${({ theme }) => theme.color.grey1}; + + &:hover { + color: ${({ theme }) => theme.color.black}; + } + span { font-size: ${({ theme }) => theme.fontSize.buttonLink}; line-height: ${({ theme }) => theme.lineHeight.buttonLink}; - color: ${({ theme }) => theme.color.grey1}; letter-spacing: 0.05em; text-transform: unset; transition: color 0.15s ease-in-out; } - &:hover span { - color: ${({ theme }) => theme.color.black}; - } ${({ $modifiers }) => $modifiers?.includes("link") && buttonLinkModifier}; ${({ $modifiers }) => $modifiers?.includes("uppercase") && buttonUpperCaseModifier}; ${({ $modifiers }) => $modifiers?.includes("hover-underline") && buttonHoverUnderlineModifier}; + + .icon { + width: 2.2rem; + height: 2.2rem; + + svg * { + fill: currentColor; + } + } ` const buttonLinkModifier = css` - span { - color: ${({ theme }) => theme.color.blue2}; - } - &:hover span { + color: ${({ theme }) => theme.color.blue2}; + &:hover { color: ${({ theme }) => theme.color.blue2}; } ` @@ -73,6 +78,7 @@ const buttonUpperCaseModifier = css` font-size: ${({ theme }) => theme.fontSize.buttonText}; line-height: ${({ theme }) => theme.lineHeight.buttonText}; letter-spacing: 0.1em; + margin-top: 0.1rem; } ` diff --git a/libs/generic-view/ui/src/lib/icon/get-icon.helper.tsx b/libs/generic-view/ui/src/lib/icon/get-icon.helper.tsx index 366e3130d..c1088c1c2 100644 --- a/libs/generic-view/ui/src/lib/icon/get-icon.helper.tsx +++ b/libs/generic-view/ui/src/lib/icon/get-icon.helper.tsx @@ -29,6 +29,15 @@ import CloseIcon from "Core/__deprecated__/renderer/svg/close.svg" import Device from "Core/__deprecated__/renderer/svg/device.svg" import Mudita from "Core/__deprecated__/renderer/svg/mudita.svg" import Spinner from "Core/__deprecated__/renderer/svg/spinner.svg" + +import Backup from "./svg/backup.svg" +import Settings from "./svg/settings.svg" +import PasswordShow from "./svg/password-show.svg" +import PasswordHide from "./svg/password-hide.svg" +import Success from "./svg/confirm.svg" +import Failure from "./svg/failed.svg" +import Folder from "./svg/folder.svg" + import { IconType } from "generic-view/utils" const typeToIcon: Record = { @@ -56,6 +65,13 @@ const typeToIcon: Record = { [IconType.Device]: Device, [IconType.Mudita]: Mudita, [IconType.Spinner]: Spinner, + [IconType.Backup]: Backup, + [IconType.Settings]: Settings, + [IconType.PasswordShow]: PasswordShow, + [IconType.PasswordHide]: PasswordHide, + [IconType.Success]: Success, + [IconType.Failure]: Failure, + [IconType.Folder]: Folder } export const getIcon = ( diff --git a/libs/generic-view/ui/src/lib/icon/icon.tsx b/libs/generic-view/ui/src/lib/icon/icon.tsx index 67126e73a..3edf171fa 100644 --- a/libs/generic-view/ui/src/lib/icon/icon.tsx +++ b/libs/generic-view/ui/src/lib/icon/icon.tsx @@ -3,18 +3,17 @@ * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md */ -import { IconType } from "generic-view/utils" -import { APIFC } from "generic-view/utils" +import { APIFC, IconType } from "generic-view/utils" import React from "react" import styled from "styled-components" import { getIcon } from "./get-icon.helper" -interface IconProps { +interface Data { type: IconType } const StyledIcon = styled.div` - color: pink; + color: inherit; width: 3.2rem; height: 3.2rem; & > * { @@ -23,7 +22,7 @@ const StyledIcon = styled.div` } ` -const Icon: APIFC = ({ className, data, ...rest }) => { +const Icon: APIFC = ({ data, ...rest }) => { if (!data) { return null } @@ -31,11 +30,7 @@ const Icon: APIFC = ({ className, data, ...rest }) => { const SVGToDisplay = getIcon(data.type) return ( - + ) diff --git a/libs/generic-view/ui/src/lib/icon/svg/backup.svg b/libs/generic-view/ui/src/lib/icon/svg/backup.svg new file mode 100644 index 000000000..725259c33 --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/backup.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/libs/generic-view/ui/src/lib/icon/svg/confirm.svg b/libs/generic-view/ui/src/lib/icon/svg/confirm.svg new file mode 100644 index 000000000..cf840aa80 --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/confirm.svg @@ -0,0 +1,4 @@ + + + + diff --git a/libs/generic-view/ui/src/lib/icon/svg/failed.svg b/libs/generic-view/ui/src/lib/icon/svg/failed.svg new file mode 100644 index 000000000..3a889e881 --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/failed.svg @@ -0,0 +1,4 @@ + + + + diff --git a/libs/generic-view/ui/src/lib/icon/svg/folder.svg b/libs/generic-view/ui/src/lib/icon/svg/folder.svg new file mode 100644 index 000000000..a5fe7ef7e --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/folder.svg @@ -0,0 +1,5 @@ + + + diff --git a/libs/generic-view/ui/src/lib/icon/svg/password-hide.svg b/libs/generic-view/ui/src/lib/icon/svg/password-hide.svg new file mode 100644 index 000000000..dba916fb8 --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/password-hide.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/libs/generic-view/ui/src/lib/icon/svg/password-show.svg b/libs/generic-view/ui/src/lib/icon/svg/password-show.svg new file mode 100644 index 000000000..936f800fb --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/password-show.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/libs/generic-view/ui/src/lib/icon/svg/settings.svg b/libs/generic-view/ui/src/lib/icon/svg/settings.svg new file mode 100644 index 000000000..c66b4e815 --- /dev/null +++ b/libs/generic-view/ui/src/lib/icon/svg/settings.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/libs/generic-view/ui/src/lib/interactive/form/form.tsx b/libs/generic-view/ui/src/lib/interactive/form/form.tsx new file mode 100644 index 000000000..17a536f96 --- /dev/null +++ b/libs/generic-view/ui/src/lib/interactive/form/form.tsx @@ -0,0 +1,24 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React from "react" +import { APIFC } from "generic-view/utils" +import { UseFormProps } from "react-hook-form/dist/types/form" +import { FormProvider, useForm } from "react-hook-form" +import { withConfig } from "../../utils/with-config" + +interface Config { + formOptions?: Pick +} + +export const Form: APIFC = ({ config, children }) => { + const methods = useForm({ + mode: "onTouched", + ...config?.formOptions, + }) + return {children} +} + +export default withConfig(Form) diff --git a/libs/generic-view/ui/src/lib/interactive/input/text-input.tsx b/libs/generic-view/ui/src/lib/interactive/input/text-input.tsx new file mode 100644 index 000000000..7c6187419 --- /dev/null +++ b/libs/generic-view/ui/src/lib/interactive/input/text-input.tsx @@ -0,0 +1,180 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { useEffect, useId, useState } from "react" +import { APIFC, IconType } from "generic-view/utils" +import { withConfig } from "../../utils/with-config" +import { withData } from "../../utils/with-data" +import styled, { css } from "styled-components" +import { IconButton } from "../../shared/button" +import Icon from "../../icon/icon" +import { useFormContext } from "react-hook-form" +import { RegisterOptions } from "react-hook-form/dist/types/validator" + +interface Data { + value: string +} + +interface Config { + name: string + label: string + type: "text" | "password" | "email" | "tel" | "url" + validation?: Pick< + RegisterOptions, + "required" | "pattern" | "maxLength" | "minLength" | "deps" | "validate" + > +} + +export const TextInput: APIFC = ({ data, config }) => { + const id = useId() + const { + register, + watch, + setValue, + formState: { errors }, + } = useFormContext() + const value = (watch(config!.name) as string) || "" + const [passwordVisible, setPasswordVisible] = useState(false) + + const error = errors[config!.name] + const inputType = + config?.type === "password" && !passwordVisible ? "password" : "text" + + const togglePasswordVisibility = () => { + setPasswordVisible((prevState) => !prevState) + } + + useEffect(() => { + if (config?.name) { + setValue(config.name, data?.value) + } + }, [config?.name, data?.value, setValue]) + + return ( + + + + + {config?.type === "password" && value.length > 0 && ( + + + + )} + + {error && {error?.message?.toString()}} + + ) +} + +export default withData(withConfig(TextInput)) + +const Wrapper = styled.div` + display: flex; + flex-direction: column; + width: 100%; +` + +const Label = styled.label<{ $inactive: boolean; $withError?: boolean }>` + color: ${({ theme }) => theme.color.grey2}; + letter-spacing: 0.04em; + font-size: ${({ theme }) => theme.fontSize.labelText}; + line-height: ${({ theme }) => theme.lineHeight.labelText}; + transition: all 0.2s ease-in-out; + position: relative; + z-index: 1; + + ${({ $inactive, theme }) => + $inactive && + css` + pointer-events: none; + font-size: ${theme.fontSize.paragraph3}; + letter-spacing: 0.05em; + transform: translateY(2.6rem); + `} + + ${({ $withError, $inactive, theme }) => + $withError && + !$inactive && + css` + color: ${theme.color.red1}; + `} +` + +const inputFocusStyles = css` + border-bottom-color: ${({ theme }) => theme.color.black}; +` + +const Input = styled.input<{ $withError?: boolean }>` + color: ${({ theme }) => theme.color.black}; + font-size: ${({ theme }) => theme.fontSize.paragraph3}; + line-height: ${({ theme }) => theme.lineHeight.paragraph3}; + letter-spacing: 0.05em; + padding: 0 3.2rem 0 0; + min-height: 3.2rem; + border: none; + border-bottom: 0.1rem solid ${({ theme }) => theme.color.grey4}; + box-sizing: content-box; + flex: 1; + outline: none; + transition: border-bottom-color 0.2s ease-in-out; + + &[type="password"] { + font-family: Arial, sans-serif; + letter-spacing: 0.15em; + font-weight: bold; + } + + &:focus { + ${inputFocusStyles}; + } + + ${({ $withError, theme }) => + $withError && + css` + border-bottom-color: ${theme.color.red1} !important; + `} +` + +const InputWrapper = styled.div` + display: flex; + flex-direction: row; + width: 100%; + + &:hover { + ${Input} { + ${inputFocusStyles}; + } + } + + button { + margin-left: -3.2rem; + } +` + +const Error = styled.p` + color: ${({ theme }) => theme.color.red1}; + font-size: ${({ theme }) => theme.fontSize.labelText}; + line-height: ${({ theme }) => theme.lineHeight.labelText}; + min-height: ${({ theme }) => theme.lineHeight.labelText}; + letter-spacing: 0.04em; + margin: 0.4rem 0 0; +` diff --git a/libs/generic-view/ui/src/lib/interactive/interactive.ts b/libs/generic-view/ui/src/lib/interactive/interactive.ts index 50cd7f367..409338318 100644 --- a/libs/generic-view/ui/src/lib/interactive/interactive.ts +++ b/libs/generic-view/ui/src/lib/interactive/interactive.ts @@ -5,8 +5,14 @@ import Modal from "./modal/modal" import { TextModal } from "./modal/text-modal" +import TextInput from "./input/text-input" +import ProgressBar from "./progress-bar/progress-bar" +import Form from "./form/form" export const interactive = { modal: Modal, "text-modal": TextModal, + "text-input": TextInput, + "progress-bar": ProgressBar, + form: Form, } diff --git a/libs/generic-view/ui/src/lib/interactive/modal/index.ts b/libs/generic-view/ui/src/lib/interactive/modal/index.ts new file mode 100644 index 000000000..7603ff0a1 --- /dev/null +++ b/libs/generic-view/ui/src/lib/interactive/modal/index.ts @@ -0,0 +1,9 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./modal-base" +export * from "./modal" +export * from "./text-modal" +export * from "./modal-helpers" diff --git a/libs/generic-view/ui/src/lib/interactive/modal/modal-base.tsx b/libs/generic-view/ui/src/lib/interactive/modal/modal-base.tsx index 114b68cc0..675c80974 100644 --- a/libs/generic-view/ui/src/lib/interactive/modal/modal-base.tsx +++ b/libs/generic-view/ui/src/lib/interactive/modal/modal-base.tsx @@ -10,9 +10,6 @@ import React, { } from "react" import ReactModal from "react-modal" import { ModalLayers } from "Core/modals-manager/constants/modal-layers.enum" -import styled from "styled-components" -import Icon from "../../icon/icon" -import { IconType } from "generic-view/utils" interface Props extends PropsWithChildren { opened: boolean @@ -21,7 +18,6 @@ interface Props extends PropsWithChildren { } variant?: "default" | "small" closeButton?: ReactElement - headerDisabled?: boolean overlayHidden?: boolean } @@ -31,7 +27,6 @@ export const ModalBase: FunctionComponent = ({ children, variant = "default", closeButton, - headerDisabled, overlayHidden, }) => { return ( @@ -46,74 +41,17 @@ export const ModalBase: FunctionComponent = ({ zIndex: ModalLayers.Default, }, content: { - width: config?.width || (variant === "small" ? 408 : 614), + width: config?.width || (variant === "small" ? 384 : 614), + maxHeight: 574, zIndex: ModalLayers.Default, // @ts-ignore - "--modal-padding": variant === "small" ? "3.6rem" : "4.8rem", + "--modal-padding": variant === "small" ? "2.4rem" : "4.8rem", }, }} closeTimeoutMS={400} > - {!headerDisabled && {closeButton}} + {closeButton} {children} ) } - -export const ModalCloseIcon = styled(Icon).attrs({ - data: { type: IconType.Close }, -})` - cursor: pointer; - width: 1.6rem; - height: 1.6rem; -` - -export const ModalTitleIcon = styled(Icon)` - width: 8rem; - height: 8rem; - padding: ${({ theme }) => theme.space.lg}; - border-radius: 50%; - background-color: ${({ theme }) => theme.color.grey5}; -` - -export const ModalCenteredContent = styled.div` - display: flex; - flex-direction: column; - align-items: center; - padding: var(--modal-padding); - gap: 4rem; - - ${ModalTitleIcon} { - margin-bottom: -2.6rem; - } - - h1 { - margin: 0; - font-size: ${({ theme }) => theme.fontSize.headline3}; - font-weight: ${({ theme }) => theme.fontWeight.bold}; - line-height: ${({ theme }) => theme.lineHeight.headline3}; - text-align: center; - } - - p { - margin: 0; - font-size: ${({ theme }) => theme.fontSize.paragraph1}; - line-height: ${({ theme }) => theme.lineHeight.paragraph1}; - text-align: center; - color: ${({ theme }) => theme.color.grey1}; - letter-spacing: 0.02em; - } -` - -export const ModalHeader = styled.header` - display: flex; - flex-direction: row; - justify-content: flex-end; - align-items: flex-end; - padding: ${({ theme }) => theme.space.xl} ${({ theme }) => theme.space.xl} 0 0; - min-height: 5.6rem; - - & + ${ModalCenteredContent} { - padding-top: 0; - } -` diff --git a/libs/generic-view/ui/src/lib/interactive/modal/modal-helpers.tsx b/libs/generic-view/ui/src/lib/interactive/modal/modal-helpers.tsx new file mode 100644 index 000000000..d0cd978ef --- /dev/null +++ b/libs/generic-view/ui/src/lib/interactive/modal/modal-helpers.tsx @@ -0,0 +1,160 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import styled, { css } from "styled-components" +import Icon from "../../icon/icon" +import { ButtonAction, IconType } from "generic-view/utils" +import { iconButtonStyles } from "../../shared/button" +import { ButtonBase } from "../../buttons/button-base/button-base" + +export const ModalTitleIcon = styled(Icon)` + width: 6.8rem; + height: 6.8rem; + padding: ${({ theme }) => theme.space.md}; + border-radius: 50%; + background-color: ${({ theme }) => theme.color.grey5}; +` + +export const ModalScrollableContent = styled.div` + overflow-y: auto; + + &::-webkit-scrollbar { + width: 0.2rem; + } + + &::-webkit-scrollbar-track { + background-color: transparent; + } + + &::-webkit-scrollbar-thumb { + background-color: ${({ theme }) => theme.color.grey2}; + } +` + +const listBulletStyle = css` + content: url('data:image/svg+xml, '); +` + +export const ModalCenteredContent = styled.div` + display: flex; + flex-direction: column; + align-items: center; + padding: var(--modal-padding); + height: 100%; + overflow: hidden; + gap: ${({ theme }) => theme.space.xl}; + + ${ModalTitleIcon} { + margin-bottom: -1rem; + + + h1 { + margin: 0; + font-size: ${({ theme }) => theme.fontSize.modalTitle}; + line-height: ${({ theme }) => theme.lineHeight.modalTitle}; + font-weight: ${({ theme }) => theme.fontWeight.bold}; + text-align: center; + } + } + + & > p, + article p { + font-size: ${({ theme }) => theme.fontSize.paragraph1}; + line-height: ${({ theme }) => theme.lineHeight.paragraph1}; + text-align: center; + color: ${({ theme }) => theme.color.grey1}; + letter-spacing: 0.02em; + margin: 0; + white-space: pre-line; + } + + & > ul, + article ul { + margin: 0; + padding-left: 2.9rem; + display: flex; + flex-direction: column; + gap: 0.4rem; + + li { + padding: 0.4rem 1.2rem 0.4rem 2.1rem; + font-size: ${({ theme }) => theme.fontSize.paragraph1}; + line-height: ${({ theme }) => theme.lineHeight.paragraph1}; + letter-spacing: 0.02em; + color: ${({ theme }) => theme.color.grey1}; + text-align: left; + + &::marker { + ${listBulletStyle}; + } + } + } + + *:has(${ModalScrollableContent}) { + overflow: hidden; + height: fit-content; + display: flex; + flex-direction: column; + } +` + +export const ModalCloseIcon = styled(Icon).attrs({ + data: { type: IconType.Close }, +})` + cursor: pointer; + width: 1.6rem; + height: 1.6rem; +` + +export const closeButtonStyles = css` + position: absolute; + right: ${({ theme }) => theme.space.xl}; + top: ${({ theme }) => theme.space.xl}; + z-index: 2; +` + +const ModalClose = styled(ButtonBase)` + ${iconButtonStyles}; + ${closeButtonStyles}; +` + +export const ModalCloseButton: FunctionComponent<{ + action?: ButtonAction +}> = ({ action }) => { + if (!action) return null + return ( + + + + ) +} + +export const ModalButtons = styled.div<{ $vertical?: boolean }>` + display: grid; + + ${({ $vertical }) => + $vertical + ? css` + grid-template-columns: 15.6rem; + grid-auto-flow: row; + grid-auto-rows: auto; + row-gap: 1.4rem; + ` + : css` + width: 100%; + grid-template-columns: 1fr 1fr; + grid-template-rows: auto; + column-gap: 2.4rem; + `} + + button { + flex: 1; + min-height: 3.2rem; + } +` diff --git a/libs/generic-view/ui/src/lib/interactive/modal/modal.tsx b/libs/generic-view/ui/src/lib/interactive/modal/modal.tsx index f4c5a3955..2e437b73c 100644 --- a/libs/generic-view/ui/src/lib/interactive/modal/modal.tsx +++ b/libs/generic-view/ui/src/lib/interactive/modal/modal.tsx @@ -5,13 +5,11 @@ import React from "react" import { BaseGenericComponent, ModalAction } from "generic-view/utils" -import styled from "styled-components" -import { ButtonBase } from "../../buttons/button-base/button-base" import { useModalsQueue } from "./use-modals-queue" import { withData } from "../../utils/with-data" import { withConfig } from "../../utils/with-config" -import { ModalCloseIcon, ModalBase, ModalHeader } from "./modal-base" -import { iconButtonStyles } from "../../shared/button" +import { ModalBase } from "./modal-base" +import { ModalCloseButton } from "./modal-helpers" interface Config { closeButtonAction?: ModalAction @@ -26,24 +24,14 @@ export const Modal: BaseGenericComponent< > = ({ children, componentKey, config }) => { const { opened } = useModalsQueue(componentKey) - const closeAction: ModalAction = config?.closeButtonAction - ? config.closeButtonAction - : { type: "close-modal", modalKey: componentKey } - return ( - - - - - + {config?.closeButtonAction && ( + + )} {children} ) } export default withConfig(withData(Modal)) - -const ModalClose = styled(ButtonBase)` - ${iconButtonStyles}; -` diff --git a/libs/generic-view/ui/src/lib/interactive/modal/text-modal.tsx b/libs/generic-view/ui/src/lib/interactive/modal/text-modal.tsx index dcdffb9da..6657a240a 100644 --- a/libs/generic-view/ui/src/lib/interactive/modal/text-modal.tsx +++ b/libs/generic-view/ui/src/lib/interactive/modal/text-modal.tsx @@ -6,12 +6,11 @@ import React, { UIEventHandler, useState } from "react" import { BaseGenericComponent, ModalAction } from "generic-view/utils" import styled, { css } from "styled-components" -import { ButtonBase } from "../../buttons/button-base/button-base" import { useModalsQueue } from "./use-modals-queue" import { withData } from "../../utils/with-data" import { withConfig } from "../../utils/with-config" -import { ModalCloseIcon, ModalBase, ModalHeader } from "./modal-base" -import { iconButtonStyles } from "../../shared/button" +import { ModalBase } from "./modal-base" +import { ModalCloseButton } from "./modal-helpers" interface Config { closeButtonAction?: ModalAction @@ -41,13 +40,10 @@ export const TextModal: BaseGenericComponent< config={{ width: config?.width || 678, }} - headerDisabled > - - - - - +
+ +
{children} ) @@ -55,19 +51,28 @@ export const TextModal: BaseGenericComponent< export default withConfig(withData(TextModal)) -const ModalClose = styled(ButtonBase)` - ${iconButtonStyles}; -` - const headerWhileScrollingStyles = css` box-shadow: 0 1rem 5rem 0 rgba(0, 0, 0, 0.08); ` -const TextModalHeader = styled(ModalHeader)<{ $active: boolean }>` +const Header = styled.header<{ $active: boolean }>` + position: relative; + width: 100%; + display: flex; + flex-direction: row; + justify-content: flex-end; + align-items: flex-end; + min-height: 8rem; padding: ${({ theme }) => theme.space.xl}; transition: box-shadow 0.3s ease-in-out; background-color: ${({ theme }) => theme.color.white}; ${({ $active }) => $active && headerWhileScrollingStyles}; + + button { + position: relative; + top: 0; + right: 0; + } ` const ScrollContainer = styled.div` diff --git a/libs/generic-view/ui/src/lib/interactive/progress-bar/progress-bar.tsx b/libs/generic-view/ui/src/lib/interactive/progress-bar/progress-bar.tsx new file mode 100644 index 000000000..666f26f9f --- /dev/null +++ b/libs/generic-view/ui/src/lib/interactive/progress-bar/progress-bar.tsx @@ -0,0 +1,85 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { useId } from "react" +import { APIFC } from "generic-view/utils" +import styled from "styled-components" +import { withConfig } from "../../utils/with-config" +import { withData } from "../../utils/with-data" + +interface Data { + value: number + message?: string +} + +interface Config { + maxValue: number + valueUnit?: string +} + +export const ProgressBar: APIFC = ({ + data, + config, + ...props +}) => { + const id = useId() + return ( + + {data?.message && {data.message}} + + + + ) +} + +export default withData(withConfig(ProgressBar)) + +const Wrapper = styled.div` + width: 100%; + display: flex; + flex-direction: column; + align-items: center; + gap: ${({ theme }) => theme.space.sm}; +` + +const Message = styled.p` + font-size: ${({ theme }) => theme.fontSize.paragraph4}; + line-height: ${({ theme }) => theme.lineHeight.paragraph4}; + color: ${({ theme }) => theme.color.grey2}; + font-weight: ${({ theme }) => theme.fontWeight.light}; + letter-spacing: 0.05em; + margin: 0 0 0.6rem 0; +` + +const Progress = styled.progress` + width: 100%; + height: 0.4rem; + border-radius: 0.2rem; + overflow: hidden; + + &::-webkit-progress-bar { + background-color: ${({ theme }) => theme.color.grey5}; + } + + &::-webkit-progress-value { + background-color: ${({ theme }) => theme.color.grey1}; + border-radius: 0.2rem; + transition: width 0.15s linear; + } +` + +const Label = styled.label` + font-size: ${({ theme }) => theme.fontSize.paragraph3}; + line-height: ${({ theme }) => theme.lineHeight.paragraph3}; + color: ${({ theme }) => theme.color.black}; + letter-spacing: 0.05em; +` diff --git a/libs/generic-view/ui/src/lib/predefined/backup-restore-available.tsx b/libs/generic-view/ui/src/lib/predefined/backup-restore-available.tsx index b88c997a6..e89f1b25a 100644 --- a/libs/generic-view/ui/src/lib/predefined/backup-restore-available.tsx +++ b/libs/generic-view/ui/src/lib/predefined/backup-restore-available.tsx @@ -23,3 +23,5 @@ export const BackupRestoreAvailable: APIFC = ({ return children } + +export default BackupRestoreAvailable diff --git a/libs/generic-view/ui/src/lib/predefined/backup/backup-create.tsx b/libs/generic-view/ui/src/lib/predefined/backup/backup-create.tsx new file mode 100644 index 000000000..ed116cda1 --- /dev/null +++ b/libs/generic-view/ui/src/lib/predefined/backup/backup-create.tsx @@ -0,0 +1,183 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent, useEffect, useRef, useState } from "react" +import { APIFC, ButtonAction } from "generic-view/utils" +import { withConfig } from "../../utils/with-config" +import { BackupFeatures, Feature } from "./backup-features" +import { BackupPassword } from "./backup-password" +import { useFormContext } from "react-hook-form" +import { BackupProgress } from "./backup-progress" +import { ModalCenteredContent, ModalCloseButton } from "../../interactive/modal" +import { BackupSuccess } from "./backup-success" +import { BackupError } from "./backup-error" +import { Form } from "../../interactive/form/form" +import { useDispatch, useSelector } from "react-redux" +import { Dispatch } from "Core/__deprecated__/renderer/store" +import { + cleanBackupProcess, + closeModal as closeModalAction, + createBackup, + selectBackupProcessStatus, +} from "generic-view/store" + +enum Step { + Features, + Password, + Progress, + Success, + Error, +} + +interface Config { + features?: Feature[] + modalKey?: string +} + +const BackupCreateForm: FunctionComponent = ({ + features = [], + modalKey, +}) => { + const dispatch = useDispatch() + const { handleSubmit } = useFormContext() + const backupProcessStatus = useSelector(selectBackupProcessStatus) + const backupAbortReference = useRef() + const [step, setStep] = useState(Step.Features) + + const featuresKeys = features?.map((item) => item.key) ?? [] + const closeButtonVisible = [ + Step.Features, + Step.Password, + Step.Success, + ].includes(step) + const abortButtonVisible = step === Step.Progress + + const closeModal = () => { + dispatch(closeModalAction({ key: modalKey! })) + dispatch(cleanBackupProcess()) + } + + const startBackup = (password?: string) => { + const promise = dispatch( + createBackup({ + features: featuresKeys, + password, + }) + ) + backupAbortReference.current = ( + promise as unknown as { + abort: VoidFunction + } + ).abort + } + + const backupCloseButtonAction: ButtonAction = { + type: "custom", + callback: closeModal, + } + + const backupAbortButtonAction: ButtonAction = { + type: "custom", + callback: () => { + backupAbortReference.current?.() + }, + } + + const backupCreateButtonAction: ButtonAction = { + type: "custom", + callback: () => { + dispatch(cleanBackupProcess()) + setStep(Step.Password) + }, + } + + const passwordSkipButtonAction: ButtonAction = { + type: "custom", + callback: () => { + handleSubmit(() => { + startBackup() + })() + }, + } + + const passwordConfirmButtonAction: ButtonAction = { + type: "custom", + callback: () => { + handleSubmit((data) => { + startBackup(data.password) + })() + }, + } + + useEffect(() => { + switch (backupProcessStatus) { + case "DONE": + setStep(Step.Success) + break + case "FAILED": + setStep(Step.Error) + break + case "PRE_BACKUP": + case "FILES_TRANSFER": + case "SAVE_FILE": + setStep(Step.Progress) + break + } + }, [backupProcessStatus]) + + useEffect(() => { + return () => { + backupAbortReference.current?.() + } + }, []) + + return ( + <> + {closeButtonVisible && ( + + )} + {abortButtonVisible && ( + + )} + + {step === Step.Features && ( + + )} + {step === Step.Password && ( + + )} + {step === Step.Progress && } + {step === Step.Success && ( + + )} + {step === Step.Error && ( + + )} + + + ) +} + +export const BackupCreate: APIFC = ({ + data, + config, + children, + ...props +}) => { + return ( +
+ + + ) +} + +export default withConfig(BackupCreate) diff --git a/libs/generic-view/ui/src/lib/predefined/backup/backup-error.tsx b/libs/generic-view/ui/src/lib/predefined/backup/backup-error.tsx new file mode 100644 index 000000000..f5dc11ecf --- /dev/null +++ b/libs/generic-view/ui/src/lib/predefined/backup/backup-error.tsx @@ -0,0 +1,49 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import { ButtonAction, IconType } from "generic-view/utils" +import { ModalButtons, ModalTitleIcon } from "../../interactive/modal" +import { ButtonSecondary } from "../../buttons/button-secondary" +import { defineMessages } from "react-intl" +import { intl } from "Core/__deprecated__/renderer/utils/intl" + +const messages = defineMessages({ + title: { + id: "module.genericViews.backup.failure.title", + }, + defaultErrorMessage: { + id: "module.genericViews.backup.failure.defaultErrorMessage", + }, + closeButtonLabel: { + id: "module.genericViews.backup.failure.closeButtonLabel", + }, +}) + +interface Props { + closeAction: ButtonAction +} + +export const BackupError: FunctionComponent = ({ closeAction }) => { + return ( + <> + +

{intl.formatMessage(messages.title)}

+

{intl.formatMessage(messages.defaultErrorMessage)}

+ + + + + ) +} diff --git a/libs/generic-view/ui/src/lib/predefined/backup/backup-features.tsx b/libs/generic-view/ui/src/lib/predefined/backup/backup-features.tsx new file mode 100644 index 000000000..bfc36172e --- /dev/null +++ b/libs/generic-view/ui/src/lib/predefined/backup/backup-features.tsx @@ -0,0 +1,88 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import styled from "styled-components" +import { ButtonAction, IconType } from "generic-view/utils" +import { ButtonSecondary } from "../../buttons/button-secondary" +import { ButtonPrimary } from "../../buttons/button-primary" +import { + ModalButtons, + ModalScrollableContent, + ModalTitleIcon, +} from "../../interactive/modal" +import { defineMessages } from "react-intl" +import { intl } from "Core/__deprecated__/renderer/utils/intl" + +const messages = defineMessages({ + title: { + id: "module.genericViews.backup.features.title", + }, + description: { + id: "module.genericViews.backup.features.description", + }, + cancelButtonLabel: { + id: "module.genericViews.backup.features.cancelButtonLabel", + }, + createButtonLabel: { + id: "module.genericViews.backup.features.createButtonLabel", + }, +}) + +export interface Feature { + label: string + key: string +} + +interface Props { + features: Feature[] + closeAction: ButtonAction + nextAction: ButtonAction +} + +export const BackupFeatures: FunctionComponent = ({ + features, + closeAction, + nextAction, +}) => { + return ( + <> + +

{intl.formatMessage(messages.title)}

+
+

{intl.formatMessage(messages.description)}

+ +
    + {features.map((feature, index) => ( +
  • {feature.label}
  • + ))} +
+
+
+ + + + + + ) +} + +const Article = styled.article` + width: 100%; + + & > p { + padding-bottom: 1.4rem; + } +` diff --git a/libs/generic-view/ui/src/lib/predefined/backup/backup-password.tsx b/libs/generic-view/ui/src/lib/predefined/backup/backup-password.tsx new file mode 100644 index 000000000..1a2a1b362 --- /dev/null +++ b/libs/generic-view/ui/src/lib/predefined/backup/backup-password.tsx @@ -0,0 +1,141 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import styled from "styled-components" +import { ButtonAction, IconType } from "generic-view/utils" +import { withConfig } from "../../utils/with-config" +import { TextInput } from "../../interactive/input/text-input" +import { ButtonPrimary } from "../../buttons/button-primary" +import { ButtonText } from "../../buttons/button-text" +import { useFormContext } from "react-hook-form" +import { ModalButtons, ModalTitleIcon } from "../../interactive/modal" +import { defineMessages } from "react-intl" +import { intl } from "Core/__deprecated__/renderer/utils/intl" + +const messages = defineMessages({ + title: { + id: "module.genericViews.backup.password.title", + }, + subtitle: { + id: "module.genericViews.backup.password.subtitle", + }, + description: { + id: "module.genericViews.backup.password.description", + }, + description2: { + id: "module.genericViews.backup.password.description2", + }, + passwordPlaceholder: { + id: "module.genericViews.backup.password.passwordPlaceholder", + }, + passwordRepeatPlaceholder: { + id: "module.genericViews.backup.password.passwordRepeatPlaceholder", + }, + confirmButtonLabel: { + id: "module.genericViews.backup.password.confirmButtonLabel", + }, + skipButtonLabel: { + id: "module.genericViews.backup.password.skipButtonLabel", + }, + passwordRepeatNotMatchingError: { + id: "module.genericViews.backup.password.passwordRepeatNotMatchingError", + }, +}) + +interface Props { + skipAction: ButtonAction + nextAction: ButtonAction +} + +export const BackupPassword: FunctionComponent = ({ + skipAction, + nextAction, +}) => { + const { watch, formState } = useFormContext() + const password = watch("password") + const passwordRepeat = watch("passwordRepeat") + + const passwordsMatching = password === passwordRepeat + + return ( + <> + +

+ {intl.formatMessage(messages.title)} + + {intl.formatMessage(messages.subtitle)} + +

+ + {intl.formatMessage(messages.description)} + {intl.formatMessage(messages.description2)} + + + { + return ( + value === formValues.password || + intl.formatMessage(messages.passwordRepeatNotMatchingError) + ) + }, + }, + }} + /> + + + + + + ) +} + +export default withConfig(BackupPassword) + +const HeadlineOptional = styled.span` + margin: -0.2rem 0 0; + display: block; + text-align: center; + font-size: ${({ theme }) => theme.fontSize.paragraph1}; + line-height: ${({ theme }) => theme.lineHeight.paragraph1}; + font-weight: ${({ theme }) => theme.fontWeight.regular}; + letter-spacing: 0.02em; +` + +const Text = styled.p` + span { + display: block; + margin: 0; + color: ${({ theme }) => theme.color.grey2}; + font-weight: ${({ theme }) => theme.fontWeight.light}; + } +` diff --git a/libs/generic-view/ui/src/lib/predefined/backup/backup-progress.tsx b/libs/generic-view/ui/src/lib/predefined/backup/backup-progress.tsx new file mode 100644 index 000000000..73cc71808 --- /dev/null +++ b/libs/generic-view/ui/src/lib/predefined/backup/backup-progress.tsx @@ -0,0 +1,72 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import styled from "styled-components" +import { IconType } from "generic-view/utils" +import { ProgressBar } from "../../interactive/progress-bar/progress-bar" +import { ModalTitleIcon } from "../../interactive/modal" +import { defineMessages } from "react-intl" +import { intl } from "Core/__deprecated__/renderer/utils/intl" +import { useSelector } from "react-redux" +import { backupProgress } from "generic-view/store" + +const messages = defineMessages({ + title: { + id: "module.genericViews.backup.progress.title", + }, + description: { + id: "module.genericViews.backup.progress.description", + }, + progressDetails: { + id: "module.genericViews.backup.progress.progressDetails", + }, + progressDetailsForFeature: { + id: "module.genericViews.backup.progress.progressDetailsForFeature", + }, +}) + +export interface Feature { + label: string + key: string +} + +interface Props { + features: Feature[] +} + +export const BackupProgress: FunctionComponent = ({ features }) => { + const progressStatus = useSelector(backupProgress) + + const featureLabel = features.find( + (item) => item.key === progressStatus.featureInProgress + )?.label + const detailMessage = featureLabel + ? intl.formatMessage(messages.progressDetailsForFeature, { featureLabel }) + : intl.formatMessage(messages.progressDetails) + + return ( + <> + +

{intl.formatMessage(messages.title)}

+

{intl.formatMessage(messages.description)}

+ + + ) +} + +const Progress = styled(ProgressBar)` + progress { + max-width: 23.3rem; + } +` diff --git a/libs/generic-view/ui/src/lib/predefined/backup/backup-success.tsx b/libs/generic-view/ui/src/lib/predefined/backup/backup-success.tsx new file mode 100644 index 000000000..e473718cc --- /dev/null +++ b/libs/generic-view/ui/src/lib/predefined/backup/backup-success.tsx @@ -0,0 +1,90 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import React, { FunctionComponent } from "react" +import { IconType } from "generic-view/utils" +import { ModalButtons, ModalTitleIcon } from "../../interactive/modal" +import { ButtonSecondary } from "../../buttons/button-secondary" +import { defineMessages } from "react-intl" +import { intl } from "Core/__deprecated__/renderer/utils/intl" +import { openBackupDirectoryRequest } from "device/feature" +import styled from "styled-components" +import { ButtonText } from "../../buttons/button-text" + +const messages = defineMessages({ + title: { + id: "module.genericViews.backup.success.title", + }, + description: { + id: "module.genericViews.backup.success.description", + }, + openBackupButtonLabel: { + id: "module.genericViews.backup.success.openBackupButtonLabel", + }, + closeButtonLabel: { + id: "module.genericViews.backup.success.closeButtonLabel", + }, +}) + +export interface Feature { + label: string + key: string +} + +interface Props { + onClose: VoidFunction +} + +export const BackupSuccess: FunctionComponent = ({ onClose }) => { + const openBackupCallback = async () => { + const openDirectoryResponse = await openBackupDirectoryRequest() + if (openDirectoryResponse.ok) { + onClose() + } + } + return ( + <> + +

{intl.formatMessage(messages.title)}

+
+

{intl.formatMessage(messages.description)}

+ +
+ + + + + ) +} + +const Article = styled.article` + display: flex; + flex-direction: column; + align-items: center; + gap: 1.4rem; + + button { + height: 3.2rem; + } +` diff --git a/libs/generic-view/ui/src/lib/predefined/overview-predefined.ts b/libs/generic-view/ui/src/lib/predefined/overview-predefined.ts index cdbff31e8..185a3c87f 100644 --- a/libs/generic-view/ui/src/lib/predefined/overview-predefined.ts +++ b/libs/generic-view/ui/src/lib/predefined/overview-predefined.ts @@ -6,11 +6,13 @@ import OverviewOsVersion from "./overview-os-version" import AboutDataBox from "./about-data-box" import LastBackupDate from "./last-backup-date" -import { BackupRestoreAvailable } from "./backup-restore-available" +import BackupRestoreAvailable from "./backup-restore-available" +import BackupCreate from "./backup/backup-create" export const predefinedComponents = { "overview-os-version": OverviewOsVersion, "about-data-box": AboutDataBox, "last-backup-date": LastBackupDate, "backup-restore-available": BackupRestoreAvailable, + "backup-create": BackupCreate, } diff --git a/libs/generic-view/ui/src/lib/shared/button.tsx b/libs/generic-view/ui/src/lib/shared/button.tsx index d71afb1a1..9ea3e21a1 100644 --- a/libs/generic-view/ui/src/lib/shared/button.tsx +++ b/libs/generic-view/ui/src/lib/shared/button.tsx @@ -16,6 +16,7 @@ export const DefaultButton = styled.button` display: flex; flex-direction: row; align-items: center; + justify-content: center; gap: 0.4rem; ` diff --git a/libs/generic-view/utils/src/lib/models/button.types.ts b/libs/generic-view/utils/src/lib/models/button.types.ts index 36464cd26..bd23f2ecb 100644 --- a/libs/generic-view/utils/src/lib/models/button.types.ts +++ b/libs/generic-view/utils/src/lib/models/button.types.ts @@ -48,4 +48,13 @@ export interface NavigateAction { viewKey: string } -export type ButtonAction = ModalAction | NavigateAction | BackupAction +export interface CustomAction { + type: "custom" + callback: VoidFunction +} + +export type ButtonAction = + | ModalAction + | NavigateAction + | BackupAction + | CustomAction diff --git a/libs/generic-view/utils/src/lib/models/icons.types.ts b/libs/generic-view/utils/src/lib/models/icons.types.ts index 78806e724..c853c8f0c 100644 --- a/libs/generic-view/utils/src/lib/models/icons.types.ts +++ b/libs/generic-view/utils/src/lib/models/icons.types.ts @@ -28,4 +28,11 @@ export enum IconType { Device = "device", Mudita = "mudita", Spinner = "spinner", + Backup = "backup", + Settings = "settings", + PasswordShow = "password-show", + PasswordHide = "password-hide", + Success = "success", + Failure = "failure", + Folder = "folder", } diff --git a/libs/generic-view/views/src/lib/mc-overview/mc-overview.ts b/libs/generic-view/views/src/lib/mc-overview/mc-overview.ts index 53450b0e2..3a42fb3dd 100644 --- a/libs/generic-view/views/src/lib/mc-overview/mc-overview.ts +++ b/libs/generic-view/views/src/lib/mc-overview/mc-overview.ts @@ -30,33 +30,6 @@ export const generateMcOverviewLayout: ViewGenerator = ( ) => { const summary = generateMcOverviewSummaryLayout(config.summary) - // Push a demo data for backup section - config.sections?.push({ - type: "mc-overview-backup", - dataKey: "backup", - title: "Backup", - backupFeatures: [ - { - label: "Contacts list", - key: "contacts-list", - }, - { - label: "Call log", - key: "call-log", - }, - ], - restoreFeatures: [ - { - label: "Contacts list", - keys: ["contacts-list"], - }, - { - label: "Call log", - keys: ["call-log"], - }, - ], - }) - const sections = config.sections?.map((section) => { switch (section?.type) { diff --git a/libs/generic-view/views/src/lib/mc-overview/section-backup/backup-create-modal.ts b/libs/generic-view/views/src/lib/mc-overview/section-backup/backup-create-modal.ts new file mode 100644 index 000000000..3b16702dd --- /dev/null +++ b/libs/generic-view/views/src/lib/mc-overview/section-backup/backup-create-modal.ts @@ -0,0 +1,35 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { Subview, ViewGenerator } from "generic-view/utils" +import { BackupTileConfig } from "device/models" + +export enum BackupModalsKeys { + Domain = "backup", + Create = "backup-create", + CreateContent = "backup-create-content", +} + +export const generateBackupCreateModalLayout: ViewGenerator< + BackupTileConfig, + Subview +> = (config) => { + return { + [config.dataKey + BackupModalsKeys.Create]: { + component: "modal", + config: { + variant: "small", + }, + childrenKeys: [config.dataKey + BackupModalsKeys.CreateContent], + }, + [config.dataKey + BackupModalsKeys.CreateContent]: { + component: "backup-create", + config: { + features: config.backupFeatures, + modalKey: config.dataKey + BackupModalsKeys.Create, + }, + }, + } +} diff --git a/libs/generic-view/views/src/lib/mc-overview/section-backup/section-backup.ts b/libs/generic-view/views/src/lib/mc-overview/section-backup/section-backup.ts index cb9696194..2f257ecc3 100644 --- a/libs/generic-view/views/src/lib/mc-overview/section-backup/section-backup.ts +++ b/libs/generic-view/views/src/lib/mc-overview/section-backup/section-backup.ts @@ -6,6 +6,10 @@ import { intl } from "Core/__deprecated__/renderer/utils/intl" import { BackupTileConfig } from "device/models" import { Subview, ViewGenerator } from "generic-view/utils" +import { + BackupModalsKeys, + generateBackupCreateModalLayout, +} from "./backup-create-modal" enum BackupKeys { BackupInfo = "backup-info", @@ -90,7 +94,7 @@ export const generateMcOverviewBackupLayout: ViewGenerator< id: "module.genericBackup.restoreButtonLabel", }), action: { - type: "restore-data", + type: "backup-data", features: config.restoreFeatures, }, }, @@ -106,11 +110,13 @@ export const generateMcOverviewBackupLayout: ViewGenerator< id: "module.genericBackup.createButtonLabel", }), action: { - type: "backup-data", - features: config.backupFeatures, + type: "open-modal", + modalKey: config.dataKey + BackupModalsKeys.Create, + domain: BackupModalsKeys.Domain, }, }, }, + ...generateBackupCreateModalLayout(config), } : {}), } diff --git a/libs/generic-view/views/src/lib/mc-overview/section-update/section-update.ts b/libs/generic-view/views/src/lib/mc-overview/section-update/section-update.ts index a0b7eaeae..2000d9e58 100644 --- a/libs/generic-view/views/src/lib/mc-overview/section-update/section-update.ts +++ b/libs/generic-view/views/src/lib/mc-overview/section-update/section-update.ts @@ -73,6 +73,7 @@ export const generateMcOverviewUpdateData = ( semver.coerce(baseUpdateData.version as string)?.raw ?? "" ) } catch { + console.log("error") updateAvailable = false } if (updateAvailable) { diff --git a/libs/shared/utils/src/lib/call-renderer.helper.ts b/libs/shared/utils/src/lib/call-renderer.helper.ts index a931e90d7..bd1368a02 100644 --- a/libs/shared/utils/src/lib/call-renderer.helper.ts +++ b/libs/shared/utils/src/lib/call-renderer.helper.ts @@ -8,13 +8,14 @@ import { LoggerFactory } from "Core/core/factories" import { ApiSerialPortToRendererEvents } from "device/models" import { PureStrategyMainEvent } from "Core/device/strategies" import { getMainAppWindow } from "./get-main-app-window" -import { DeviceManagerMainEvent } from "./main-event.constant" +import { AppEvents, DeviceManagerMainEvent } from "./main-event.constant" const logger = LoggerFactory.getInstance() export type CallRendererEvent = | ApiSerialPortToRendererEvents | DeviceManagerMainEvent + | AppEvents | PureStrategyMainEvent export const callRenderer = (event: CallRendererEvent, payload?: unknown) => { diff --git a/libs/shared/utils/src/lib/main-event.constant.ts b/libs/shared/utils/src/lib/main-event.constant.ts index 3685d8918..611da78c0 100644 --- a/libs/shared/utils/src/lib/main-event.constant.ts +++ b/libs/shared/utils/src/lib/main-event.constant.ts @@ -8,3 +8,7 @@ export enum DeviceManagerMainEvent { DeviceConnected = "device-manager-device-connected", DeviceConnectFailed = "device-manager-device-connect-failed", } + +export enum AppEvents { + WindowFocused = "app-events-window-focused", +} diff --git a/libs/system-utils/feature/.babelrc b/libs/system-utils/feature/.babelrc new file mode 100644 index 000000000..ef4889c1a --- /dev/null +++ b/libs/system-utils/feature/.babelrc @@ -0,0 +1,20 @@ +{ + "presets": [ + [ + "@nx/react/babel", + { + "runtime": "automatic", + "useBuiltIns": "usage" + } + ] + ], + "plugins": [ + [ + "styled-components", + { + "pure": true, + "ssr": true + } + ] + ] +} diff --git a/libs/system-utils/feature/.eslintrc.json b/libs/system-utils/feature/.eslintrc.json new file mode 100644 index 000000000..cacbe2621 --- /dev/null +++ b/libs/system-utils/feature/.eslintrc.json @@ -0,0 +1,18 @@ +{ + "extends": ["../../../.eslintrc.js"], + "ignorePatterns": ["!**/*"], + "overrides": [ + { + "files": ["*.ts", "*.tsx", "*.js", "*.jsx"], + "rules": {} + }, + { + "files": ["*.ts", "*.tsx"], + "rules": {} + }, + { + "files": ["*.js", "*.jsx"], + "rules": {} + } + ] +} diff --git a/libs/system-utils/feature/README.md b/libs/system-utils/feature/README.md new file mode 100644 index 000000000..09870cdc1 --- /dev/null +++ b/libs/system-utils/feature/README.md @@ -0,0 +1,7 @@ +# system-utils-feature + +This library was generated with [Nx](https://nx.dev). + +## Running unit tests + +Run `nx test system-utils-feature` to execute the unit tests via [Jest](https://jestjs.io). diff --git a/libs/system-utils/feature/jest.config.ts b/libs/system-utils/feature/jest.config.ts new file mode 100644 index 000000000..103272573 --- /dev/null +++ b/libs/system-utils/feature/jest.config.ts @@ -0,0 +1,11 @@ +/* eslint-disable */ +export default { + displayName: "system-utils-feature", + preset: "../../../jest.preset.js", + transform: { + "^(?!.*\\.(js|jsx|ts|tsx|css|json)$)": "@nx/react/plugins/jest", + "^.+\\.[tj]sx?$": ["babel-jest", { presets: ["@nx/react/babel"] }], + }, + moduleFileExtensions: ["ts", "tsx", "js", "jsx"], + coverageDirectory: "../../../coverage/libs/system-utils/feature", +} diff --git a/libs/system-utils/feature/project.json b/libs/system-utils/feature/project.json new file mode 100644 index 000000000..175ca13e7 --- /dev/null +++ b/libs/system-utils/feature/project.json @@ -0,0 +1,20 @@ +{ + "name": "system-utils-feature", + "$schema": "../../../node_modules/nx/schemas/project-schema.json", + "sourceRoot": "libs/system-utils/feature/src", + "projectType": "library", + "tags": [], + "targets": { + "lint": { + "executor": "@nx/eslint:lint", + "outputs": ["{options.outputFile}"] + }, + "test": { + "executor": "@nx/jest:jest", + "outputs": ["{workspaceRoot}/coverage/{projectRoot}"], + "options": { + "jestConfig": "libs/system-utils/feature/jest.config.ts" + } + } + } +} diff --git a/libs/system-utils/feature/src/index.ts b/libs/system-utils/feature/src/index.ts new file mode 100644 index 000000000..01a4f31c0 --- /dev/null +++ b/libs/system-utils/feature/src/index.ts @@ -0,0 +1,6 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./lib/system-utils.module" diff --git a/libs/system-utils/feature/src/lib/directory/directory.service.ts b/libs/system-utils/feature/src/lib/directory/directory.service.ts new file mode 100644 index 000000000..36d227a6d --- /dev/null +++ b/libs/system-utils/feature/src/lib/directory/directory.service.ts @@ -0,0 +1,26 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { shell } from "electron" +import { IpcEvent } from "Core/core/decorators" +import { DirectoryServiceEvents } from "system-utils/models" +import { Result } from "Core/core/builder" +import { AppError } from "Core/core/errors" +import { GeneralError } from "device/models" + +export class Directory { + constructor() {} + + @IpcEvent(DirectoryServiceEvents.OpenDirectory) + public async open({ path }: { path: string }) { + const errorMessage = await shell.openPath(path) + if (errorMessage) { + return Result.failed( + new AppError(GeneralError.InternalError, errorMessage) + ) + } + return Result.success(undefined) + } +} diff --git a/libs/system-utils/feature/src/lib/system-utils.module.ts b/libs/system-utils/feature/src/lib/system-utils.module.ts new file mode 100644 index 000000000..90f5b229a --- /dev/null +++ b/libs/system-utils/feature/src/lib/system-utils.module.ts @@ -0,0 +1,16 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +import { Directory } from "./directory/directory.service" + +export class SystemUtilsModule { + public directory = new Directory() + + constructor() {} + + public getServices() { + return [this.directory] + } +} diff --git a/libs/system-utils/feature/tsconfig.json b/libs/system-utils/feature/tsconfig.json new file mode 100644 index 000000000..4daaf45cd --- /dev/null +++ b/libs/system-utils/feature/tsconfig.json @@ -0,0 +1,20 @@ +{ + "compilerOptions": { + "jsx": "react-jsx", + "allowJs": false, + "esModuleInterop": false, + "allowSyntheticDefaultImports": true, + "strict": true + }, + "files": [], + "include": [], + "references": [ + { + "path": "./tsconfig.lib.json" + }, + { + "path": "./tsconfig.spec.json" + } + ], + "extends": "../../../tsconfig.base.json" +} diff --git a/libs/system-utils/feature/tsconfig.lib.json b/libs/system-utils/feature/tsconfig.lib.json new file mode 100644 index 000000000..21799b3e6 --- /dev/null +++ b/libs/system-utils/feature/tsconfig.lib.json @@ -0,0 +1,24 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../../dist/out-tsc", + "types": [ + "node", + + "@nx/react/typings/cssmodule.d.ts", + "@nx/react/typings/image.d.ts" + ] + }, + "exclude": [ + "jest.config.ts", + "src/**/*.spec.ts", + "src/**/*.test.ts", + "src/**/*.spec.tsx", + "src/**/*.test.tsx", + "src/**/*.spec.js", + "src/**/*.test.js", + "src/**/*.spec.jsx", + "src/**/*.test.jsx" + ], + "include": ["src/**/*.js", "src/**/*.jsx", "src/**/*.ts", "src/**/*.tsx"] +} diff --git a/libs/system-utils/feature/tsconfig.spec.json b/libs/system-utils/feature/tsconfig.spec.json new file mode 100644 index 000000000..25b7af8f6 --- /dev/null +++ b/libs/system-utils/feature/tsconfig.spec.json @@ -0,0 +1,20 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../../dist/out-tsc", + "module": "commonjs", + "types": ["jest", "node"] + }, + "include": [ + "jest.config.ts", + "src/**/*.test.ts", + "src/**/*.spec.ts", + "src/**/*.test.tsx", + "src/**/*.spec.tsx", + "src/**/*.test.js", + "src/**/*.spec.js", + "src/**/*.test.jsx", + "src/**/*.spec.jsx", + "src/**/*.d.ts" + ] +} diff --git a/libs/system-utils/models/.babelrc b/libs/system-utils/models/.babelrc new file mode 100644 index 000000000..ef4889c1a --- /dev/null +++ b/libs/system-utils/models/.babelrc @@ -0,0 +1,20 @@ +{ + "presets": [ + [ + "@nx/react/babel", + { + "runtime": "automatic", + "useBuiltIns": "usage" + } + ] + ], + "plugins": [ + [ + "styled-components", + { + "pure": true, + "ssr": true + } + ] + ] +} diff --git a/libs/system-utils/models/.eslintrc.json b/libs/system-utils/models/.eslintrc.json new file mode 100644 index 000000000..cacbe2621 --- /dev/null +++ b/libs/system-utils/models/.eslintrc.json @@ -0,0 +1,18 @@ +{ + "extends": ["../../../.eslintrc.js"], + "ignorePatterns": ["!**/*"], + "overrides": [ + { + "files": ["*.ts", "*.tsx", "*.js", "*.jsx"], + "rules": {} + }, + { + "files": ["*.ts", "*.tsx"], + "rules": {} + }, + { + "files": ["*.js", "*.jsx"], + "rules": {} + } + ] +} diff --git a/libs/system-utils/models/README.md b/libs/system-utils/models/README.md new file mode 100644 index 000000000..588e511d7 --- /dev/null +++ b/libs/system-utils/models/README.md @@ -0,0 +1,7 @@ +# system-utils-models + +This library was generated with [Nx](https://nx.dev). + +## Running unit tests + +Run `nx test system-utils-models` to execute the unit tests via [Jest](https://jestjs.io). diff --git a/libs/system-utils/models/jest.config.ts b/libs/system-utils/models/jest.config.ts new file mode 100644 index 000000000..56827b88e --- /dev/null +++ b/libs/system-utils/models/jest.config.ts @@ -0,0 +1,11 @@ +/* eslint-disable */ +export default { + displayName: "system-utils-models", + preset: "../../../jest.preset.js", + transform: { + "^(?!.*\\.(js|jsx|ts|tsx|css|json)$)": "@nx/react/plugins/jest", + "^.+\\.[tj]sx?$": ["babel-jest", { presets: ["@nx/react/babel"] }], + }, + moduleFileExtensions: ["ts", "tsx", "js", "jsx"], + coverageDirectory: "../../../coverage/libs/system-utils/models", +} diff --git a/libs/system-utils/models/project.json b/libs/system-utils/models/project.json new file mode 100644 index 000000000..9fad57830 --- /dev/null +++ b/libs/system-utils/models/project.json @@ -0,0 +1,20 @@ +{ + "name": "system-utils-models", + "$schema": "../../../node_modules/nx/schemas/project-schema.json", + "sourceRoot": "libs/system-utils/models/src", + "projectType": "library", + "tags": [], + "targets": { + "lint": { + "executor": "@nx/eslint:lint", + "outputs": ["{options.outputFile}"] + }, + "test": { + "executor": "@nx/jest:jest", + "outputs": ["{workspaceRoot}/coverage/{projectRoot}"], + "options": { + "jestConfig": "libs/system-utils/models/jest.config.ts" + } + } + } +} diff --git a/libs/system-utils/models/src/index.ts b/libs/system-utils/models/src/index.ts new file mode 100644 index 000000000..fa2aed964 --- /dev/null +++ b/libs/system-utils/models/src/index.ts @@ -0,0 +1,6 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export * from "./lib/directory-service-events" diff --git a/libs/system-utils/models/src/lib/directory-service-events.ts b/libs/system-utils/models/src/lib/directory-service-events.ts new file mode 100644 index 000000000..bbbe99e1f --- /dev/null +++ b/libs/system-utils/models/src/lib/directory-service-events.ts @@ -0,0 +1,8 @@ +/** + * Copyright (c) Mudita sp. z o.o. All rights reserved. + * For licensing, see https://github.com/mudita/mudita-center/blob/master/LICENSE.md + */ + +export enum DirectoryServiceEvents { + OpenDirectory = "directoryservice-open-directory", +} diff --git a/libs/system-utils/models/tsconfig.json b/libs/system-utils/models/tsconfig.json new file mode 100644 index 000000000..4daaf45cd --- /dev/null +++ b/libs/system-utils/models/tsconfig.json @@ -0,0 +1,20 @@ +{ + "compilerOptions": { + "jsx": "react-jsx", + "allowJs": false, + "esModuleInterop": false, + "allowSyntheticDefaultImports": true, + "strict": true + }, + "files": [], + "include": [], + "references": [ + { + "path": "./tsconfig.lib.json" + }, + { + "path": "./tsconfig.spec.json" + } + ], + "extends": "../../../tsconfig.base.json" +} diff --git a/libs/system-utils/models/tsconfig.lib.json b/libs/system-utils/models/tsconfig.lib.json new file mode 100644 index 000000000..21799b3e6 --- /dev/null +++ b/libs/system-utils/models/tsconfig.lib.json @@ -0,0 +1,24 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../../dist/out-tsc", + "types": [ + "node", + + "@nx/react/typings/cssmodule.d.ts", + "@nx/react/typings/image.d.ts" + ] + }, + "exclude": [ + "jest.config.ts", + "src/**/*.spec.ts", + "src/**/*.test.ts", + "src/**/*.spec.tsx", + "src/**/*.test.tsx", + "src/**/*.spec.js", + "src/**/*.test.js", + "src/**/*.spec.jsx", + "src/**/*.test.jsx" + ], + "include": ["src/**/*.js", "src/**/*.jsx", "src/**/*.ts", "src/**/*.tsx"] +} diff --git a/libs/system-utils/models/tsconfig.spec.json b/libs/system-utils/models/tsconfig.spec.json new file mode 100644 index 000000000..25b7af8f6 --- /dev/null +++ b/libs/system-utils/models/tsconfig.spec.json @@ -0,0 +1,20 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../../dist/out-tsc", + "module": "commonjs", + "types": ["jest", "node"] + }, + "include": [ + "jest.config.ts", + "src/**/*.test.ts", + "src/**/*.spec.ts", + "src/**/*.test.tsx", + "src/**/*.spec.tsx", + "src/**/*.test.js", + "src/**/*.spec.js", + "src/**/*.test.jsx", + "src/**/*.spec.jsx", + "src/**/*.d.ts" + ] +} diff --git a/package-lock.json b/package-lock.json index aed8be02d..06c6c1036 100644 --- a/package-lock.json +++ b/package-lock.json @@ -14,6 +14,8 @@ "libs" ], "dependencies": { + "crypto-js": "^4.2.0", + "js-crc": "^0.2.0", "react-is": "18.2.0", "react-markdown": "^9.0.1", "serialport": "10.1.0", @@ -64,11 +66,13 @@ "@testing-library/user-event": "^14.5.1", "@types/archiver": "^5.3.1", "@types/chai": "^4.3.3", + "@types/crypto-js": "^4.2.2", "@types/elasticlunr": "^0.9.5", "@types/electron-devtools-installer": "^2.2.2", "@types/electron-localshortcut": "^3.1.0", "@types/history": "^4.7.9", "@types/jest": "^29.5.6", + "@types/js-crc": "^0.2.3", "@types/lodash": "^4.14.182", "@types/mime-types": "2.1.2", "@types/mock-fs": "^4.13.1", @@ -149,7 +153,7 @@ "file-loader": "^6.2.0", "fork-ts-checker-webpack-plugin": "^7.2.13", "fs-extra": "^10.1.0", - "getmac": "^5.20.0", + "getmac": "^5.21.0", "googleapis": "^133.0.0", "history": "^4.10.1", "html-webpack-plugin": "^5.5.0", @@ -179,7 +183,7 @@ "react-beautiful-dnd": "^13.1.1", "react-dom": "^18.2.0", "react-google-button": "^0.7.2", - "react-hook-form": "~7.47.0", + "react-hook-form": "~7.50.1", "react-intersection-observer": "^9.4.0", "react-intl": "6.4.7", "react-modal": "^3.15.1", @@ -9467,6 +9471,12 @@ "@types/node": "*" } }, + "node_modules/@types/crypto-js": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@types/crypto-js/-/crypto-js-4.2.2.tgz", + "integrity": "sha512-sDOLlVbHhXpAUAL0YHDUUwDZf3iN4Bwi4W6a0W0b+QcAezUbRtH4FVb+9J4h+XFPW7l/gQ9F8qC7P+Ec4k8QVQ==", + "dev": true + }, "node_modules/@types/debug": { "version": "4.1.12", "resolved": "https://registry.npmjs.org/@types/debug/-/debug-4.1.12.tgz", @@ -9756,6 +9766,12 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/@types/js-crc": { + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/@types/js-crc/-/js-crc-0.2.3.tgz", + "integrity": "sha512-XQNB5dDWZ1S0AS/NzY0F/Bl8c8uOTC1MbTZrIj08HzkwT4POxgeA7+wWesrP4MvJEdvy+tsHya3OHzcLnQhfIg==", + "dev": true + }, "node_modules/@types/jsdom": { "version": "20.0.1", "resolved": "https://registry.npmjs.org/@types/jsdom/-/jsdom-20.0.1.tgz", @@ -17235,6 +17251,11 @@ "node": "*" } }, + "node_modules/crypto-js": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/crypto-js/-/crypto-js-4.2.0.tgz", + "integrity": "sha512-KALDyEYgpY+Rlob/iriUtjV6d5Eq+Y191A5g4UqLAi8CyGP9N1+FdVbkc1SxKc2r4YAYqG8JzO2KGL+AizD70Q==" + }, "node_modules/crypto-random-string": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/crypto-random-string/-/crypto-random-string-2.0.0.tgz", @@ -28016,6 +28037,11 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/js-crc": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/js-crc/-/js-crc-0.2.0.tgz", + "integrity": "sha512-8DdCSAOACpF8WDAjyDFBC2rj8OS4HUP9mNZBDfl8jCiPCnJG+2bkuycalxwZh6heFy6PrMvoWTp47lp6gzT65A==" + }, "node_modules/js-tokens": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", @@ -34945,9 +34971,9 @@ } }, "node_modules/react-hook-form": { - "version": "7.47.0", - "resolved": "https://registry.npmjs.org/react-hook-form/-/react-hook-form-7.47.0.tgz", - "integrity": "sha512-F/TroLjTICipmHeFlMrLtNLceO2xr1jU3CyiNla5zdwsGUGu2UOxxR4UyJgLlhMwLW/Wzp4cpJ7CPfgJIeKdSg==", + "version": "7.50.1", + "resolved": "https://registry.npmjs.org/react-hook-form/-/react-hook-form-7.50.1.tgz", + "integrity": "sha512-3PCY82oE0WgeOgUtIr3nYNNtNvqtJ7BZjsbxh6TnYNbXButaD5WpjOmTjdxZfheuHKR68qfeFnEDVYoSSFPMTQ==", "dev": true, "engines": { "node": ">=12.22.0" diff --git a/package.json b/package.json index 914e5cb42..cec0df44f 100644 --- a/package.json +++ b/package.json @@ -88,11 +88,13 @@ "@testing-library/user-event": "^14.5.1", "@types/archiver": "^5.3.1", "@types/chai": "^4.3.3", + "@types/crypto-js": "^4.2.2", "@types/elasticlunr": "^0.9.5", "@types/electron-devtools-installer": "^2.2.2", "@types/electron-localshortcut": "^3.1.0", "@types/history": "^4.7.9", "@types/jest": "^29.5.6", + "@types/js-crc": "^0.2.3", "@types/lodash": "^4.14.182", "@types/mime-types": "2.1.2", "@types/mock-fs": "^4.13.1", @@ -173,7 +175,7 @@ "file-loader": "^6.2.0", "fork-ts-checker-webpack-plugin": "^7.2.13", "fs-extra": "^10.1.0", - "getmac": "^5.20.0", + "getmac": "^5.21.0", "googleapis": "^133.0.0", "history": "^4.10.1", "html-webpack-plugin": "^5.5.0", @@ -203,7 +205,7 @@ "react-beautiful-dnd": "^13.1.1", "react-dom": "^18.2.0", "react-google-button": "^0.7.2", - "react-hook-form": "~7.47.0", + "react-hook-form": "~7.50.1", "react-intersection-observer": "^9.4.0", "react-intl": "6.4.7", "react-modal": "^3.15.1", @@ -265,6 +267,8 @@ "npm": "9.5.1" }, "dependencies": { + "crypto-js": "^4.2.0", + "js-crc": "^0.2.0", "react-is": "18.2.0", "react-markdown": "^9.0.1", "serialport": "10.1.0", diff --git a/tsconfig.base.json b/tsconfig.base.json index b12a082f4..acafde405 100644 --- a/tsconfig.base.json +++ b/tsconfig.base.json @@ -41,7 +41,9 @@ "generic-view/utils": ["libs/generic-view/utils/src/index.ts"], "generic-view/views": ["libs/generic-view/views/src/index.ts"], "shared/app-state": ["libs/shared/app-state/src/index.ts"], - "shared/utils": ["libs/shared/utils/src/index.ts"] + "shared/utils": ["libs/shared/utils/src/index.ts"], + "system-utils/feature": ["libs/system-utils/feature/src/index.ts"], + "system-utils/models": ["libs/system-utils/models/src/index.ts"] } }, "exclude": [