From 94b24e07d62ca39fc619249c9b6bc21e74291289 Mon Sep 17 00:00:00 2001 From: Rafael Araujo Lehmkuhl Date: Tue, 10 Dec 2024 17:04:36 -0300 Subject: [PATCH 1/4] electron: Add support for filesystem storage through the IPC --- electron/main.ts | 2 ++ electron/preload.ts | 17 ++++++++++ electron/services/storage.ts | 61 ++++++++++++++++++++++++++++++++++++ src/libs/cosmos.ts | 3 +- src/stores/video.ts | 2 +- src/types/general.ts | 31 ++++++++++++++++++ src/types/video.ts | 4 --- 7 files changed, 114 insertions(+), 6 deletions(-) create mode 100644 electron/services/storage.ts diff --git a/electron/main.ts b/electron/main.ts index bf0685c89..3001cd00b 100644 --- a/electron/main.ts +++ b/electron/main.ts @@ -5,6 +5,7 @@ import { join } from 'path' import { setupAutoUpdater } from './services/auto-update' import store from './services/config-store' import { setupNetworkService } from './services/network' +import { setupFilesystemStorage } from './services/storage' // If the app is packaged, push logs to the system instead of the console if (app.isPackaged) { @@ -71,6 +72,7 @@ protocol.registerSchemesAsPrivileged([ }, ]) +setupFilesystemStorage() setupNetworkService() app.whenReady().then(async () => { diff --git a/electron/preload.ts b/electron/preload.ts index 08977a09b..a26e42dce 100644 --- a/electron/preload.ts +++ b/electron/preload.ts @@ -14,4 +14,21 @@ contextBridge.exposeInMainWorld('electronAPI', { downloadUpdate: () => ipcRenderer.send('download-update'), installUpdate: () => ipcRenderer.send('install-update'), cancelUpdate: () => ipcRenderer.send('cancel-update'), + setItem: async (key: string, value: Blob, subFolders?: string[]) => { + const arrayBuffer = await value.arrayBuffer() + await ipcRenderer.invoke('setItem', { key, value: new Uint8Array(arrayBuffer), subFolders }) + }, + getItem: async (key: string, subFolders?: string[]) => { + const arrayBuffer = await ipcRenderer.invoke('getItem', { key, subFolders }) + return arrayBuffer ? new Blob([arrayBuffer]) : null + }, + removeItem: async (key: string, subFolders?: string[]) => { + await ipcRenderer.invoke('removeItem', { key, subFolders }) + }, + clear: async (subFolders?: string[]) => { + await ipcRenderer.invoke('clear', { subFolders }) + }, + keys: async (subFolders?: string[]) => { + return await ipcRenderer.invoke('keys', { subFolders }) + }, }) diff --git a/electron/services/storage.ts b/electron/services/storage.ts new file mode 100644 index 000000000..f005494df --- /dev/null +++ b/electron/services/storage.ts @@ -0,0 +1,61 @@ +import { ipcMain } from 'electron' +import { app } from 'electron' +import * as fs from 'fs/promises' +import { dirname, join } from 'path' + +// Create a new storage interface for filesystem +const cockpitFolderPath = join(app.getPath('home'), 'Cockpit') +fs.mkdir(cockpitFolderPath, { recursive: true }) + +export const filesystemStorage = { + async setItem(key: string, value: ArrayBuffer, subFolders?: string[]): Promise { + const buffer = Buffer.from(value) + const filePath = join(cockpitFolderPath, ...(subFolders ?? []), key) + await fs.mkdir(dirname(filePath), { recursive: true }) + await fs.writeFile(filePath, buffer) + }, + async getItem(key: string, subFolders?: string[]): Promise { + const filePath = join(cockpitFolderPath, ...(subFolders ?? []), key) + try { + return await fs.readFile(filePath) + } catch (error) { + if (error.code === 'ENOENT') return null + throw error + } + }, + async removeItem(key: string, subFolders?: string[]): Promise { + const filePath = join(cockpitFolderPath, ...(subFolders ?? []), key) + await fs.unlink(filePath) + }, + async clear(subFolders?: string[]): Promise { + const dirPath = join(cockpitFolderPath, ...(subFolders ?? [])) + await fs.rm(dirPath, { recursive: true }) + }, + async keys(subFolders?: string[]): Promise { + const dirPath = join(cockpitFolderPath, ...(subFolders ?? [])) + try { + return await fs.readdir(dirPath) + } catch (error) { + if (error.code === 'ENOENT') return [] + throw error + } + }, +} + +export const setupFilesystemStorage = (): void => { + ipcMain.handle('setItem', async (_, data) => { + await filesystemStorage.setItem(data.key, data.value, data.subFolders) + }) + ipcMain.handle('getItem', async (_, data) => { + return await filesystemStorage.getItem(data.key, data.subFolders) + }) + ipcMain.handle('removeItem', async (_, data) => { + await filesystemStorage.removeItem(data.key, data.subFolders) + }) + ipcMain.handle('clear', async (_, data) => { + await filesystemStorage.clear(data.subFolders) + }) + ipcMain.handle('keys', async (_, data) => { + return await filesystemStorage.keys(data.subFolders) + }) +} diff --git a/src/libs/cosmos.ts b/src/libs/cosmos.ts index 810bef604..b74b2f9e8 100644 --- a/src/libs/cosmos.ts +++ b/src/libs/cosmos.ts @@ -1,5 +1,6 @@ import { isBrowser } from 'browser-or-node' +import { ElectronStorageDB } from '@/types/general' import { NetworkInfo } from '@/types/network' import { @@ -185,7 +186,7 @@ declare global { /** * Electron API exposed through preload script */ - electronAPI?: { + electronAPI?: ElectronStorageDB & { /** * Get network information from the main process * @returns Promise containing subnet information diff --git a/src/stores/video.ts b/src/stores/video.ts index 34d53ff4b..86308a73b 100644 --- a/src/stores/video.ts +++ b/src/stores/video.ts @@ -21,10 +21,10 @@ import { isEqual, sleep } from '@/libs/utils' import { useMainVehicleStore } from '@/stores/mainVehicle' import { useMissionStore } from '@/stores/mission' import { Alert, AlertLevel } from '@/types/alert' +import { StorageDB } from '@/types/general' import { type DownloadProgressCallback, type FileDescriptor, - type StorageDB, type StreamData, type UnprocessedVideoInfo, type VideoProcessingDetails, diff --git a/src/types/general.ts b/src/types/general.ts index 0ba903c59..f01f8ae57 100644 --- a/src/types/general.ts +++ b/src/types/general.ts @@ -33,3 +33,34 @@ export interface DialogActions { } export type ConfigComponent = DefineComponent, Record, unknown> | null + +export interface StorageDB { + getItem: (key: string) => Promise + setItem: (key: string, value: Blob) => Promise + removeItem: (key: string) => Promise + clear: () => Promise + keys: () => Promise +} + +export interface ElectronStorageDB { + /** + * Set an item in the filesystem storage + */ + setItem: (key: string, value: Blob, subFolders?: string[]) => Promise + /** + * Get an item from the filesystem storage + */ + getItem: (key: string, subFolders?: string[]) => Promise + /** + * Remove an item from the filesystem storage + */ + removeItem: (key: string, subFolders?: string[]) => Promise + /** + * Clear the filesystem storage + */ + clear: (subFolders?: string[]) => Promise + /** + * Get all keys from the filesystem storage + */ + keys: (subFolders?: string[]) => Promise +} diff --git a/src/types/video.ts b/src/types/video.ts index 5de9e0afc..5b58ba90d 100644 --- a/src/types/video.ts +++ b/src/types/video.ts @@ -108,10 +108,6 @@ export interface FileDescriptor { filename: string } -export interface StorageDB { - getItem: (key: string) => Promise -} - export type DownloadProgressCallback = (progress: number, total: number) => Promise export enum VideoExtensionContainer { From e78752624ef4a330f39abc021289f38cfd56b659 Mon Sep 17 00:00:00 2001 From: Rafael Araujo Lehmkuhl Date: Tue, 10 Dec 2024 17:12:06 -0300 Subject: [PATCH 2/4] video: Fix name of video files with unknown extensions When the extension was unknown, the video file was being generated with two dots in the end (e.g.: 'video..webm'). This also created a side-effect of the system not being able to find the correspondent telemetry logs and leaving them behind when clearing the videos. --- src/stores/video.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/stores/video.ts b/src/stores/video.ts index 86308a73b..cba04ecf5 100644 --- a/src/stores/video.ts +++ b/src/stores/video.ts @@ -639,7 +639,7 @@ export const useVideoStore = defineStore('video', () => { updateLastProcessingUpdate(hash) debouncedUpdateFileProgress(info.fileName, 75, `Saving video file.`) - await videoStoringDB.setItem(`${info.fileName}.${extensionContainer || '.webm'}`, durFixedBlob ?? mergedBlob) + await videoStoringDB.setItem(`${info.fileName}.${extensionContainer || 'webm'}`, durFixedBlob ?? mergedBlob) updateLastProcessingUpdate(hash) From 188dbb095030015824a6b8b7493053cc2088393b Mon Sep 17 00:00:00 2001 From: Rafael Araujo Lehmkuhl Date: Tue, 10 Dec 2024 17:12:52 -0300 Subject: [PATCH 3/4] video: Use the filesystem to store videos on the electron-based app For that an abstraction layer was created. It coordinates the usage of the IndexedDB for the browser version and the filesystem for the Electron version. --- src/components/VideoLibraryModal.vue | 9 +- .../mini-widgets/MiniVideoRecorder.vue | 3 +- src/libs/videoStorage.ts | 127 ++++++++++++++++++ src/stores/video.ts | 85 ++++++------ 4 files changed, 173 insertions(+), 51 deletions(-) create mode 100644 src/libs/videoStorage.ts diff --git a/src/components/VideoLibraryModal.vue b/src/components/VideoLibraryModal.vue index facff754d..2ea3d5379 100644 --- a/src/components/VideoLibraryModal.vue +++ b/src/components/VideoLibraryModal.vue @@ -889,11 +889,12 @@ const fetchVideosAndLogData = async (): Promise => { const logFileOperations: Promise[] = [] // Fetch processed videos and logs - await videoStore.videoStoringDB.iterate((value, key) => { + const keys = await videoStore.videoStorage.keys() + for (const key of keys) { if (videoStore.isVideoFilename(key)) { videoFilesOperations.push( (async () => { - const videoBlob = await videoStore.videoStoringDB.getItem(key) + const videoBlob = await videoStore.videoStorage.getItem(key) let url = '' let isProcessed = true if (videoBlob instanceof Blob) { @@ -910,7 +911,7 @@ const fetchVideosAndLogData = async (): Promise => { if (key.endsWith('.ass')) { logFileOperations.push( (async () => { - const videoBlob = await videoStore.videoStoringDB.getItem(key) + const videoBlob = await videoStore.videoStorage.getItem(key) let url = '' if (videoBlob instanceof Blob) { url = URL.createObjectURL(videoBlob) @@ -923,7 +924,7 @@ const fetchVideosAndLogData = async (): Promise => { })() ) } - }) + } // Fetch unprocessed videos const unprocessedVideos = await videoStore.unprocessedVideos diff --git a/src/components/mini-widgets/MiniVideoRecorder.vue b/src/components/mini-widgets/MiniVideoRecorder.vue index 4991521ab..c99ef9940 100644 --- a/src/components/mini-widgets/MiniVideoRecorder.vue +++ b/src/components/mini-widgets/MiniVideoRecorder.vue @@ -188,7 +188,8 @@ watch(nameSelectedStream, (newName) => { // Fetch number of temporary videos on storage const fetchNumberOfTempVideos = async (): Promise => { - const nProcessedVideos = (await videoStore.videoStoringDB.keys()).filter((k) => videoStore.isVideoFilename(k)).length + const keys = await videoStore.videoStorage.keys() + const nProcessedVideos = keys.filter((k) => videoStore.isVideoFilename(k)).length const nFailedUnprocessedVideos = Object.keys(videoStore.keysFailedUnprocessedVideos).length numberOfVideosOnDB.value = nProcessedVideos + nFailedUnprocessedVideos } diff --git a/src/libs/videoStorage.ts b/src/libs/videoStorage.ts new file mode 100644 index 000000000..2ffab2f90 --- /dev/null +++ b/src/libs/videoStorage.ts @@ -0,0 +1,127 @@ +import localforage from 'localforage' + +import type { ElectronStorageDB, StorageDB } from '@/types/general' + +import { isElectron } from './utils' + +const throwIfNotElectron = (): void => { + if (!isElectron()) { + console.warn('Filesystem storage is only available in Electron.') + return + } + if (!window.electronAPI) { + console.error('electronAPI is not available on window object') + console.debug('Available window properties:', Object.keys(window)) + throw new Error('Electron filesystem API is not properly initialized. This is likely a setup issue.') + } +} + +/** + * Electron storage implementation. + * Uses the exposed IPC renderer API to store and retrieve data in the filesystem. + */ +class ElectronStorage implements ElectronStorageDB { + subFolders: string[] + electronAPI: ElectronStorageDB + + /** + * Creates a new instance of the ElectronStorage class. + * @param {string[]} subFolders - The subfolders to store the data in. + */ + constructor(subFolders: string[]) { + throwIfNotElectron() + + this.subFolders = subFolders + this.electronAPI = window.electronAPI as StorageDB + } + + setItem = async (key: string, value: Blob): Promise => { + throwIfNotElectron() + await this.electronAPI.setItem(key, value, this.subFolders) + } + + getItem = async (key: string): Promise => { + throwIfNotElectron() + return await this.electronAPI.getItem(key, this.subFolders) + } + + removeItem = async (key: string): Promise => { + throwIfNotElectron() + await this.electronAPI.removeItem(key, this.subFolders) + } + + clear = async (): Promise => { + throwIfNotElectron() + await this.electronAPI.clear(this.subFolders) + } + + keys = async (): Promise => { + throwIfNotElectron() + return await this.electronAPI.keys(this.subFolders) + } +} + +/** + * LocalForage storage implementation. + * Uses the localforage library to store and retrieve data in the IndexedDB. + */ +class LocalForageStorage implements StorageDB { + localForage: LocalForage + + /** + * Creates a new instance of the LocalForageStorage class. + * @param {string} name - The name of the localforage instance. + * @param {string} storeName - The name of the store to store the data in. + * @param {number} version - The version of the localforage instance. + * @param {string} description - The description of the localforage instance. + */ + constructor(name: string, storeName: string, version: number, description: string) { + this.localForage = localforage.createInstance({ + driver: localforage.INDEXEDDB, + name: name, + storeName: storeName, + version: version, + description: description, + }) + } + + setItem = async (key: string, value: Blob): Promise => { + await this.localForage.setItem(key, value) + } + + getItem = async (key: string): Promise => { + return await this.localForage.getItem(key) + } + + removeItem = async (key: string): Promise => { + await this.localForage.removeItem(key) + } + + clear = async (): Promise => { + await this.localForage.clear() + } + + keys = async (): Promise => { + return await this.localForage.keys() + } +} + +const tempVideoChunksIndexdedDB: StorageDB = new LocalForageStorage( + 'Cockpit - Temporary Video', + 'cockpit-temp-video-db', + 1.0, + 'Database for storing the chunks of an ongoing recording, to be merged afterwards.' +) + +const videoStoringIndexedDB: StorageDB = new LocalForageStorage( + 'Cockpit - Video Recovery', + 'cockpit-video-recovery-db', + 1.0, + 'Cockpit video recordings and their corresponding telemetry subtitles.' +) + +const electronVideoStorage = new ElectronStorage(['videos']) +const temporaryElectronVideoStorage = new ElectronStorage(['videos', 'temporary-video-chunks']) + +export const videoStorage = isElectron() ? electronVideoStorage : videoStoringIndexedDB +export const tempVideoStorage = isElectron() ? temporaryElectronVideoStorage : tempVideoChunksIndexdedDB diff --git a/src/stores/video.ts b/src/stores/video.ts index cba04ecf5..0d8cb4946 100644 --- a/src/stores/video.ts +++ b/src/stores/video.ts @@ -2,7 +2,6 @@ import { useDebounceFn, useStorage, useThrottleFn, useTimestamp } from '@vueuse/ import { BlobReader, BlobWriter, ZipWriter } from '@zip.js/zip.js' import { differenceInSeconds, format } from 'date-fns' import { saveAs } from 'file-saver' -import localforage from 'localforage' import { defineStore } from 'pinia' import { v4 as uuid } from 'uuid' import { computed, ref, watch } from 'vue' @@ -18,6 +17,7 @@ import eventTracker from '@/libs/external-telemetry/event-tracking' import { availableCockpitActions, registerActionCallback } from '@/libs/joystick/protocols/cockpit-actions' import { datalogger } from '@/libs/sensors-logging' import { isEqual, sleep } from '@/libs/utils' +import { tempVideoStorage, videoStorage } from '@/libs/videoStorage' import { useMainVehicleStore } from '@/stores/mainVehicle' import { useMissionStore } from '@/stores/mission' import { Alert, AlertLevel } from '@/types/alert' @@ -281,7 +281,7 @@ export const useVideoStore = defineStore('video', () => { let recordingHash = '' let refreshHash = true - const namesCurrentChunksOnDB = await tempVideoChunksDB.keys() + const namesCurrentChunksOnDB = await tempVideoStorage.keys() while (refreshHash) { recordingHash = uuid().slice(0, 8) refreshHash = namesCurrentChunksOnDB.some((chunkName) => chunkName.includes(recordingHash)) @@ -370,7 +370,7 @@ export const useVideoStore = defineStore('video', () => { const chunkName = `${recordingHash}_${chunksCount}` try { - await tempVideoChunksDB.setItem(chunkName, e.data) + await tempVideoStorage.setItem(chunkName, e.data) sequentialLostChunks = 0 } catch { sequentialLostChunks++ @@ -387,7 +387,7 @@ export const useVideoStore = defineStore('video', () => { // Gets the thumbnail from the first chunk if (chunksCount === 0) { try { - const videoChunk = await tempVideoChunksDB.getItem(chunkName) + const videoChunk = await tempVideoStorage.getItem(chunkName) if (videoChunk) { const firstChunkBlob = new Blob([videoChunk as Blob]) const thumbnail = await extractThumbnailFromVideo(firstChunkBlob) @@ -436,13 +436,13 @@ export const useVideoStore = defineStore('video', () => { const discardProcessedFilesFromVideoDB = async (fileNames: string[]): Promise => { console.debug(`Discarding files from the video recovery database: ${fileNames.join(', ')}`) for (const filename of fileNames) { - await videoStoringDB.removeItem(filename) + await videoStorage.removeItem(filename) } } const discardUnprocessedFilesFromVideoDB = async (hashes: string[]): Promise => { for (const hash of hashes) { - await tempVideoChunksDB.removeItem(hash) + await tempVideoStorage.removeItem(hash) delete unprocessedVideos.value[hash] } } @@ -462,7 +462,7 @@ export const useVideoStore = defineStore('video', () => { } const downloadFiles = async ( - db: StorageDB, + db: StorageDB | LocalForage, keys: string[], shouldZip = false, zipFilenamePrefix = 'Cockpit-Video-Files', @@ -496,9 +496,9 @@ export const useVideoStore = defineStore('video', () => { console.debug(`Downloading files from the video recovery database: ${fileNames.join(', ')}`) if (zipMultipleFiles.value) { const ZipFilename = fileNames.length > 1 ? 'Cockpit-Video-Recordings' : 'Cockpit-Video-Recording' - await downloadFiles(videoStoringDB, fileNames, true, ZipFilename, progressCallback) + await downloadFiles(videoStorage, fileNames, true, ZipFilename, progressCallback) } else { - await downloadFiles(videoStoringDB, fileNames) + await downloadFiles(videoStorage, fileNames) } } @@ -506,48 +506,34 @@ export const useVideoStore = defineStore('video', () => { console.debug(`Downloading ${hashes.length} video chunks from the temporary database.`) for (const hash of hashes) { - const fileNames = (await tempVideoChunksDB.keys()).filter((filename) => filename.includes(hash)) + const fileNames = (await tempVideoStorage.keys()).filter((filename) => filename.includes(hash)) const zipFilenamePrefix = `Cockpit-Unprocessed-Video-Chunks-${hash}` - await downloadFiles(tempVideoChunksDB, fileNames, true, zipFilenamePrefix, progressCallback) + await downloadFiles(tempVideoStorage, fileNames, true, zipFilenamePrefix, progressCallback) } } // Used to clear the temporary video database const clearTemporaryVideoDB = async (): Promise => { - await tempVideoChunksDB.clear() + await tempVideoStorage.clear() } const temporaryVideoDBSize = async (): Promise => { let totalSizeBytes = 0 - await tempVideoChunksDB.iterate((chunk) => { - totalSizeBytes += (chunk as Blob).size - }) + const keys = await tempVideoStorage.keys() + for (const key of keys) { + const blob = await tempVideoStorage.getItem(key) + if (blob) { + totalSizeBytes += blob.size + } + } return totalSizeBytes } const videoStorageFileSize = async (filename: string): Promise => { - const file = await videoStoringDB.getItem(filename) + const file = await videoStorage.getItem(filename) return file ? (file as Blob).size : undefined } - // Used to store chunks of an ongoing recording, that will be merged into a video file when the recording is stopped - const tempVideoChunksDB = localforage.createInstance({ - driver: localforage.INDEXEDDB, - name: 'Cockpit - Temporary Video', - storeName: 'cockpit-temp-video-db', - version: 1.0, - description: 'Database for storing the chunks of an ongoing recording, to be merged afterwards.', - }) - - // Offer download of backuped videos - const videoStoringDB = localforage.createInstance({ - driver: localforage.INDEXEDDB, - name: 'Cockpit - Video Recovery', - storeName: 'cockpit-video-recovery-db', - version: 1.0, - description: 'Local backups of Cockpit video recordings to be retrieved in case of failure.', - }) - const updateLastProcessingUpdate = (recordingHash: string): void => { const info = unprocessedVideos.value[recordingHash] info.dateLastProcessingUpdate = new Date() @@ -599,11 +585,14 @@ export const useVideoStore = defineStore('video', () => { const dateFinish = new Date(info.dateFinish!) debouncedUpdateFileProgress(info.fileName, 30, 'Grouping video chunks.') - await tempVideoChunksDB.iterate((videoChunk, chunkName) => { - if (chunkName.includes(hash)) { - chunks.push({ blob: videoChunk as Blob, name: chunkName }) + const keys = await tempVideoStorage.keys() + const filteredKeys = keys.filter((key) => key.includes(hash)) + for (const key of filteredKeys) { + const blob = await tempVideoStorage.getItem(key) + if (blob && blob.size > 0) { + chunks.push({ blob, name: key }) } - }) + } // As we advance through the processing, we update the last processing update date, so consumers know this is ongoing updateLastProcessingUpdate(hash) @@ -639,7 +628,7 @@ export const useVideoStore = defineStore('video', () => { updateLastProcessingUpdate(hash) debouncedUpdateFileProgress(info.fileName, 75, `Saving video file.`) - await videoStoringDB.setItem(`${info.fileName}.${extensionContainer || 'webm'}`, durFixedBlob ?? mergedBlob) + await videoStorage.setItem(`${info.fileName}.${extensionContainer || 'webm'}`, durFixedBlob ?? mergedBlob) updateLastProcessingUpdate(hash) @@ -653,7 +642,7 @@ export const useVideoStore = defineStore('video', () => { const videoTelemetryLog = datalogger.getSlice(telemetryLog, dateStart, dateFinish) const assLog = datalogger.toAssOverlay(videoTelemetryLog, info.vWidth!, info.vHeight!, dateStart.getTime()) const logBlob = new Blob([assLog], { type: 'text/plain' }) - videoStoringDB.setItem(`${info.fileName}.ass`, logBlob) + videoStorage.setItem(`${info.fileName}.ass`, logBlob) updateLastProcessingUpdate(hash) @@ -666,7 +655,11 @@ export const useVideoStore = defineStore('video', () => { // Remove temp chunks and video metadata from the database const cleanupProcessedData = async (recordingHash: string): Promise => { - await tempVideoChunksDB.removeItem(recordingHash) + const keys = await tempVideoStorage.keys() + const filteredKeys = keys.filter((key) => key.includes(recordingHash) && key.includes('_')) + for (const key of filteredKeys) { + await tempVideoStorage.removeItem(key) + } delete unprocessedVideos.value[recordingHash] } @@ -705,7 +698,7 @@ export const useVideoStore = defineStore('video', () => { if (keysFailedUnprocessedVideos.value.isEmpty()) return console.log(`Processing unprocessed videos: ${keysFailedUnprocessedVideos.value.join(', ')}`) - const chunks = await tempVideoChunksDB.keys() + const chunks = await tempVideoStorage.keys() if (chunks.length === 0) { discardUnprocessedVideos() throw new Error('No video recording data found. Discarding leftover info.') @@ -732,14 +725,14 @@ export const useVideoStore = defineStore('video', () => { console.log('Discarding unprocessed videos.') const keysUnprocessedVideos = includeNotFailed ? keysAllUnprocessedVideos.value : keysFailedUnprocessedVideos.value - const currentChunks = await tempVideoChunksDB.keys() + const currentChunks = await tempVideoStorage.keys() const chunksUnprocessedVideos = currentChunks.filter((chunkName) => { return keysUnprocessedVideos.some((key) => chunkName.includes(key)) }) unprocessedVideos.value = {} for (const chunk of chunksUnprocessedVideos) { - tempVideoChunksDB.removeItem(chunk) + tempVideoStorage.removeItem(chunk) } } @@ -922,8 +915,8 @@ export const useVideoStore = defineStore('video', () => { jitterBufferTarget, zipMultipleFiles, namesAvailableStreams, - videoStoringDB, - tempVideoChunksDB, + videoStorage, + tempVideoStorage, streamsCorrespondency, namessAvailableAbstractedStreams, externalStreamId, From b7bbd6eeedf8f7cb2dd257a765ece6ebe282c5ad Mon Sep 17 00:00:00 2001 From: Rafael Araujo Lehmkuhl Date: Tue, 17 Dec 2024 15:46:26 -0300 Subject: [PATCH 4/4] electron: Add buttons to open the video folder as well as the cockpit folder --- electron/preload.ts | 2 ++ electron/services/storage.ts | 13 +++++++++-- src/components/VideoLibraryModal.vue | 24 +++++++++++++++++++ src/libs/cosmos.ts | 8 +++++++ src/views/ConfigurationGeneralView.vue | 32 +++++++++++++++++++++++--- 5 files changed, 74 insertions(+), 5 deletions(-) diff --git a/electron/preload.ts b/electron/preload.ts index a26e42dce..b3929e0a1 100644 --- a/electron/preload.ts +++ b/electron/preload.ts @@ -31,4 +31,6 @@ contextBridge.exposeInMainWorld('electronAPI', { keys: async (subFolders?: string[]) => { return await ipcRenderer.invoke('keys', { subFolders }) }, + openCockpitFolder: () => ipcRenderer.invoke('open-cockpit-folder'), + openVideoFolder: () => ipcRenderer.invoke('open-video-folder'), }) diff --git a/electron/services/storage.ts b/electron/services/storage.ts index f005494df..f6349f72e 100644 --- a/electron/services/storage.ts +++ b/electron/services/storage.ts @@ -1,10 +1,10 @@ -import { ipcMain } from 'electron' +import { ipcMain, shell } from 'electron' import { app } from 'electron' import * as fs from 'fs/promises' import { dirname, join } from 'path' // Create a new storage interface for filesystem -const cockpitFolderPath = join(app.getPath('home'), 'Cockpit') +export const cockpitFolderPath = join(app.getPath('home'), 'Cockpit') fs.mkdir(cockpitFolderPath, { recursive: true }) export const filesystemStorage = { @@ -58,4 +58,13 @@ export const setupFilesystemStorage = (): void => { ipcMain.handle('keys', async (_, data) => { return await filesystemStorage.keys(data.subFolders) }) + ipcMain.handle('open-cockpit-folder', async () => { + await fs.mkdir(cockpitFolderPath, { recursive: true }) + await shell.openPath(cockpitFolderPath) + }) + ipcMain.handle('open-video-folder', async () => { + const videoFolderPath = join(cockpitFolderPath, 'videos') + await fs.mkdir(videoFolderPath, { recursive: true }) + await shell.openPath(videoFolderPath) + }) } diff --git a/src/components/VideoLibraryModal.vue b/src/components/VideoLibraryModal.vue index 2ea3d5379..a44575a30 100644 --- a/src/components/VideoLibraryModal.vue +++ b/src/components/VideoLibraryModal.vue @@ -484,6 +484,7 @@ import { ref, watch } from 'vue' import { useInteractionDialog } from '@/composables/interactionDialog' import { useSnackbar } from '@/composables/snackbar' +import { isElectron } from '@/libs/utils' import { useAppInterfaceStore } from '@/stores/appInterface' import { useVideoStore } from '@/stores/video' import { DialogActions } from '@/types/general' @@ -580,8 +581,31 @@ const fileActionButtons = computed(() => [ disabled: showOnScreenProgress.value === true || isPreparingDownload.value === true, action: () => downloadVideoAndTelemetryFiles(), }, + { + name: 'Open Folder', + icon: 'mdi-folder-outline', + size: 28, + tooltip: 'Open videos folder', + confirmAction: false, + show: isElectron(), + disabled: false, + action: () => openVideoFolder(), + }, ]) +const openVideoFolder = (): void => { + if (isElectron() && window.electronAPI) { + window.electronAPI?.openVideoFolder() + } else { + showSnackbar({ + message: 'This feature is only available in the desktop version of Cockpit.', + duration: 3000, + variant: 'error', + closeButton: true, + }) + } +} + const closeModal = (): void => { isVisible.value = false emits('update:openModal', false) diff --git a/src/libs/cosmos.ts b/src/libs/cosmos.ts index b74b2f9e8..ecd0c0a97 100644 --- a/src/libs/cosmos.ts +++ b/src/libs/cosmos.ts @@ -224,6 +224,14 @@ declare global { * Register callback for download progress event */ onDownloadProgress: (callback: (info: any) => void) => void + /** + * Open cockpit folder + */ + openCockpitFolder: () => void + /** + * Open video folder + */ + openVideoFolder: () => void } } } diff --git a/src/views/ConfigurationGeneralView.vue b/src/views/ConfigurationGeneralView.vue index 029378016..1548bf100 100644 --- a/src/views/ConfigurationGeneralView.vue +++ b/src/views/ConfigurationGeneralView.vue @@ -32,9 +32,20 @@ @click="missionStore.changeUsername" /> - - Show tutorial - +
+ + Show tutorial + + + Open Cockpit folder + +
@@ -292,6 +303,7 @@ import { onMounted, ref, watch } from 'vue' import { defaultGlobalAddress } from '@/assets/defaults' import ExpansiblePanel from '@/components/ExpansiblePanel.vue' import VehicleDiscoveryDialog from '@/components/VehicleDiscoveryDialog.vue' +import { useSnackbar } from '@/composables/snackbar' import * as Connection from '@/libs/connection/connection' import { ConnectionManager } from '@/libs/connection/connection-manager' import { isValidNetworkAddress, reloadCockpit } from '@/libs/utils' @@ -306,6 +318,7 @@ import BaseConfigurationView from './BaseConfigurationView.vue' const mainVehicleStore = useMainVehicleStore() const interfaceStore = useAppInterfaceStore() const missionStore = useMissionStore() +const { showSnackbar } = useSnackbar() const globalAddressForm = ref() const globalAddressFormValid = ref(false) @@ -517,6 +530,19 @@ onMounted(() => { }) const showDiscoveryDialog = ref(false) + +const openCockpitFolder = (): void => { + if (isElectron() && window.electronAPI) { + window.electronAPI?.openCockpitFolder() + } else { + showSnackbar({ + message: 'This feature is only available in the desktop version of Cockpit.', + duration: 3000, + variant: 'error', + closeButton: true, + }) + } +}