diff --git a/packages/main/src/mainWindow.ts b/packages/main/src/mainWindow.ts index 6e851322e44f3..41d6f8171071b 100644 --- a/packages/main/src/mainWindow.ts +++ b/packages/main/src/mainWindow.ts @@ -131,6 +131,22 @@ async function createWindow(): Promise { }); }); + ipcMain.on('dialog:saveFile', (_, param: { dialogId: string; message: string; defaultPath: string }) => { + dialog + .showSaveDialog(browserWindow, { + title: param.message, + defaultPath: param.defaultPath, + }) + .then(response => { + if (!response.canceled && response.filePath) { + browserWindow.webContents.send('dialog:open-file-or-folder-response', param.dialogId, response); + } + }) + .catch((err: unknown) => { + console.error('Error saving file', err); + }); + }); + let configurationRegistry: ConfigurationRegistry; ipcMain.on('configuration-registry', (_, data) => { configurationRegistry = data; diff --git a/packages/main/src/plugin/index.ts b/packages/main/src/plugin/index.ts index ff84b9e6af521..f6dea54b3e152 100644 --- a/packages/main/src/plugin/index.ts +++ b/packages/main/src/plugin/index.ts @@ -147,6 +147,7 @@ import { ImageCheckerImpl } from './image-checker.js'; import type { ImageCheckerInfo } from './api/image-checker-info.js'; import { AppearanceInit } from './appearance-init.js'; import type { KubeContext } from './kubernetes-context.js'; +import { Troubleshooting } from './troubleshooting.js'; import { KubernetesInformerManager } from './kubernetes-informer-registry.js'; import type { KubernetesInformerResourcesType } from './api/kubernetes-informer-info.js'; import { OpenDevToolsInit } from './open-devtools-init.js'; @@ -733,6 +734,8 @@ export class PluginSystem { const imageChecker = new ImageCheckerImpl(apiSender); + const troubleshooting = new Troubleshooting(apiSender); + const contributionManager = new ContributionManager(apiSender, directories, containerProviderRegistry, exec); const webviewRegistry = new WebviewRegistry(apiSender); @@ -1278,6 +1281,24 @@ export class PluginSystem { return cliToolRegistry.getCliToolInfos(); }); + this.ipcHandle( + 'troubleshooting:saveLogs', + async ( + _listener, + consoleLogs: { logType: LogType; message: string }[], + destinaton: string, + ): Promise => { + return troubleshooting.saveLogs(consoleLogs, destinaton); + }, + ); + + this.ipcHandle( + 'troubleshooting:generateLogFileName', + async (_listener, filename: string, prefix?: string): Promise => { + return troubleshooting.generateLogFileName(filename, prefix); + }, + ); + this.ipcHandle( 'cli-tool-registry:updateCliTool', async (_listener, id: string, loggerId: string): Promise => { diff --git a/packages/main/src/plugin/troubleshooting.spec.ts b/packages/main/src/plugin/troubleshooting.spec.ts new file mode 100644 index 0000000000000..d11da3a316e9c --- /dev/null +++ b/packages/main/src/plugin/troubleshooting.spec.ts @@ -0,0 +1,212 @@ +/********************************************************************** + * Copyright (C) 2024 Red Hat, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ***********************************************************************/ + +import { beforeEach, expect, test, vi } from 'vitest'; +import { Troubleshooting } from './troubleshooting.js'; +import type { TroubleshootingFileMap, LogType } from './troubleshooting.js'; +import * as fs from 'node:fs'; +import type { ApiSenderType } from './api.js'; + +const writeZipMock = vi.fn(); +const addFileMock = vi.fn(); + +const apiSender: ApiSenderType = { + send: vi.fn(), + receive: vi.fn(), +}; + +vi.mock('electron', () => { + return { + ipcMain: { + emit: vi.fn(), + on: vi.fn(), + }, + }; +}); + +vi.mock('adm-zip', () => { + return { + default: class { + addFile = addFileMock; + writeZip = writeZipMock; + }, + }; +}); + +beforeEach(() => { + vi.clearAllMocks(); +}); + +// Test the saveLogsToZip function +test('Should save a zip file with the correct content', async () => { + const zipFile = new Troubleshooting(apiSender); + const fileMaps = [ + { + filename: 'file1', + content: 'content1', + }, + { + filename: 'file2', + content: 'content2', + }, + ]; + + const zipSpy = vi.spyOn(zipFile, 'saveLogsToZip'); + + await zipFile.saveLogsToZip(fileMaps, 'test.zip'); + expect(zipSpy).toHaveBeenCalledWith(fileMaps, 'test.zip'); + + expect(writeZipMock).toHaveBeenCalledWith('test.zip'); +}); + +// Do not expect writeZipMock to have been called if fileMaps is empty +test('Should not save a zip file if fileMaps is empty', async () => { + const zipFile = new Troubleshooting(apiSender); + const fileMaps: TroubleshootingFileMap[] = []; + + const zipSpy = vi.spyOn(zipFile, 'saveLogsToZip'); + + await zipFile.saveLogsToZip(fileMaps, 'test.zip'); + expect(zipSpy).toHaveBeenCalledWith(fileMaps, 'test.zip'); + + expect(writeZipMock).not.toHaveBeenCalled(); +}); + +// Expect the file name to have a .txt extension +test('Should have a .txt extension in the file name', async () => { + const zipFile = new Troubleshooting(apiSender); + const fileMaps = [ + { + filename: 'file1', + content: '', + }, + { + filename: 'file2', + content: '', + }, + ]; + + const zipSpy = vi.spyOn(zipFile, 'saveLogsToZip'); + + await zipFile.saveLogsToZip(fileMaps, 'test.zip'); + expect(zipSpy).toHaveBeenCalledWith(fileMaps, 'test.zip'); + + expect(addFileMock).toHaveBeenCalledWith('file1', expect.any(Object)); + expect(addFileMock).toHaveBeenCalledWith('file2', expect.any(Object)); +}); + +// Expect getConsoleLogs to correctly format the console logs passed in +test('Should correctly format console logs', async () => { + const zipFile = new Troubleshooting(apiSender); + const consoleLogs = [ + { + logType: 'log' as LogType, + date: new Date(), + message: 'message1', + }, + { + logType: 'log' as LogType, + date: new Date(), + message: 'message2', + }, + ]; + + const zipSpy = vi.spyOn(zipFile, 'getConsoleLogs'); + + const fileMaps = zipFile.getConsoleLogs(consoleLogs); + expect(zipSpy).toHaveBeenCalledWith(consoleLogs); + + expect(fileMaps[0].filename).toContain('console'); + expect(fileMaps[0].content).toContain('log : message1'); + expect(fileMaps[0].content).toContain('log : message2'); +}); + +// Expect getSystemLogs to return getMacSystemLogs if the platform is darwin +// mock the private getMacSystemLogs function so we can spy on it +test('Should return getMacSystemLogs if the platform is darwin', async () => { + // Mock platform to be darwin + vi.spyOn(process, 'platform', 'get').mockReturnValue('darwin'); + + const readFileMock = vi.spyOn(fs.promises, 'readFile'); + readFileMock.mockResolvedValue('content'); + + // Mock exists to be true + vi.mock('node:fs'); + vi.spyOn(fs, 'existsSync').mockImplementation(() => { + return true; + }); + + const zipFile = new Troubleshooting(apiSender); + const getSystemLogsSpy = vi.spyOn(zipFile, 'getSystemLogs'); + + await zipFile.getSystemLogs(); + expect(getSystemLogsSpy).toHaveBeenCalled(); + + // Expect it to have been called twice as it checked stdout and stderr + expect(readFileMock).toHaveBeenCalledTimes(2); + + // Expect readFileMock to have been called with /Library/Logs/Podman Desktop/launchd-stdout.log but CONTAINED in the path + expect(readFileMock).toHaveBeenCalledWith( + expect.stringContaining('/Library/Logs/Podman Desktop/launchd-stdout'), + 'utf-8', + ); + expect(readFileMock).toHaveBeenCalledWith( + expect.stringContaining('/Library/Logs/Podman Desktop/launchd-stderr'), + 'utf-8', + ); +}); + +// Should return getWindowsSystemLogs if the platform is win32 +// ~/AppData/Roaming/Podman Desktop/logs/podman-desktop.log +test('Should return getWindowsSystemLogs if the platform is win32', async () => { + // Mock exists to be true + vi.mock('node:fs'); + vi.spyOn(fs, 'existsSync').mockImplementation(() => { + return true; + }); + + // Mock platform to be win32 + vi.spyOn(process, 'platform', 'get').mockReturnValue('win32'); + + const readFileMock = vi.spyOn(fs.promises, 'readFile'); + readFileMock.mockResolvedValue('content'); + + const zipFile = new Troubleshooting(apiSender); + const getSystemLogsSpy = vi.spyOn(zipFile, 'getSystemLogs'); + + await zipFile.getSystemLogs(); + expect(getSystemLogsSpy).toHaveBeenCalled(); + + // Expect it to have been called once as it checked podman-desktop.log + expect(readFileMock).toHaveBeenCalledTimes(1); + + // Expect readFileMock to have been called with ~/AppData/Roaming/Podman Desktop/logs/podman-desktop.log but CONTAINED in the path + expect(readFileMock).toHaveBeenCalledWith( + expect.stringContaining('/AppData/Roaming/Podman Desktop/logs/podman-desktop'), + 'utf-8', + ); +}); + +test('test generateLogFileName', async () => { + const ts = new Troubleshooting(apiSender); + const filename = ts.generateLogFileName('test'); + + // Simple regex to check that the file name is in the correct format (YYYMMDDHHmmss) + expect(filename).toMatch(/[0-9]{14}/); + expect(filename).toContain('test'); +}); diff --git a/packages/main/src/plugin/troubleshooting.ts b/packages/main/src/plugin/troubleshooting.ts new file mode 100644 index 0000000000000..9fa86aeae3361 --- /dev/null +++ b/packages/main/src/plugin/troubleshooting.ts @@ -0,0 +1,112 @@ +/********************************************************************** + * Copyright (C) 2024 Red Hat, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ***********************************************************************/ + +import AdmZip from 'adm-zip'; +import moment from 'moment'; +import * as os from 'node:os'; +import * as fs from 'node:fs'; +import type { ApiSenderType } from './api.js'; + +const SYSTEM_FILENAME = 'system'; + +export interface TroubleshootingFileMap { + filename: string; + content: string; +} + +export type LogType = 'log' | 'warn' | 'trace' | 'debug' | 'error'; + +export class Troubleshooting { + constructor(private apiSender: ApiSenderType) {} + + // The "main" function that is exposes that is used to gather + // all the logs and save them to a zip file. + // this also takes in the console logs and adds them to the zip file (see preload/src/index.ts) regarding memoryLogs + async saveLogs(console: { logType: LogType; message: string }[], destination: string): Promise { + const systemLogs = await this.getSystemLogs(); + const consoleLogs = this.getConsoleLogs(console); + const fileMaps = [...systemLogs, ...consoleLogs]; + await this.saveLogsToZip(fileMaps, destination); + return fileMaps.map(fileMap => fileMap.filename); + } + + async saveLogsToZip(fileMaps: TroubleshootingFileMap[], destination: string): Promise { + if (fileMaps.length === 0) { + return; + } + + const zip = new AdmZip(); + fileMaps.forEach(fileMap => { + zip.addFile(fileMap.filename, Buffer.from(fileMap.content, 'utf8')); + }); + zip.writeZip(destination); + } + + getConsoleLogs(consoleLogs: { logType: LogType; message: string }[]): TroubleshootingFileMap[] { + const content = consoleLogs.map(log => `${log.logType} : ${log.message}`).join('\n'); + return [{ filename: this.generateLogFileName('console'), content }]; + } + + async getSystemLogs(): Promise { + switch (os.platform()) { + case 'darwin': + return this.getLogsFromFiles( + ['launchd-stdout.log', 'launchd-stderr.log'], + `${os.homedir()}/Library/Logs/Podman Desktop`, + ); + case 'win32': + return this.getLogsFromFiles(['podman-desktop'], `${os.homedir()}/AppData/Roaming/Podman Desktop/logs`); + default: + // Unsupported platform, so do not return anything + return []; + } + } + + private async getFileSystemContent(filePath: string, logName: string): Promise { + const content = await fs.promises.readFile(filePath, 'utf-8'); + return { filename: this.generateLogFileName(SYSTEM_FILENAME + '-' + logName), content }; + } + + private async getLogsFromFiles(logFiles: string[], logDir: string): Promise { + const logs: TroubleshootingFileMap[] = []; + for (const file of logFiles) { + try { + const filePath = `${logDir}/${file}`; + + // Check if the file exists, if not, skip it. + if (!fs.existsSync(filePath)) { + continue; + } + + const fileMap = await this.getFileSystemContent(filePath, file); + logs.push(fileMap); + } catch (error) { + console.error(`Error reading ${file}: `, error); + } + } + return logs; + } + generateLogFileName(filename: string, extension?: string): string { + // If the filename has an extension like .log, move it to the end ofthe generated name + // otherwise just use .txt + const filenameParts = filename.split('.'); + // Use the file extension if it's provided, otherwise use the one from the file name, or default to txt + const fileExtension = extension ? extension : filenameParts.length > 1 ? filenameParts[1] : 'txt'; + return `${filenameParts[0]}-${moment().format('YYYYMMDDHHmmss')}.${fileExtension}`; + } +} diff --git a/packages/preload/src/index.ts b/packages/preload/src/index.ts index 0a5c66043a35d..a9b5ad0774833 100644 --- a/packages/preload/src/index.ts +++ b/packages/preload/src/index.ts @@ -1076,6 +1076,17 @@ function initExposure(): void { }, ); + contextBridge.exposeInMainWorld('troubleshootingSaveLogs', async (destinaton: string): Promise => { + return ipcInvoke('troubleshooting:saveLogs', memoryLogs, destinaton); + }); + + contextBridge.exposeInMainWorld( + 'troubleshootingGenerateLogFileName', + async (filename: string, extension?: string): Promise => { + return ipcInvoke('troubleshooting:generateLogFileName', filename, extension); + }, + ); + contextBridge.exposeInMainWorld('getContributedMenus', async (context: string): Promise => { return ipcInvoke('menu-registry:getContributedMenus', context); }); @@ -1289,6 +1300,30 @@ function initExposure(): void { const dialogResponses = new Map(); + contextBridge.exposeInMainWorld('saveFileDialog', async (message: string, defaultPath: string) => { + // generate id + const dialogId = idDialog; + idDialog++; + + // create defer object + const defer = new Deferred(); + + // store the dialogID + dialogResponses.set(`${dialogId}`, (result: Electron.SaveDialogReturnValue) => { + defer.resolve(result); + }); + + // ask to open file dialog + ipcRenderer.send('dialog:saveFile', { + dialogId: `${dialogId}`, + message, + defaultPath, + }); + + // wait for response + return defer.promise; + }); + contextBridge.exposeInMainWorld( 'openFileDialog', async (message: string, filter?: { extensions: string[]; name: string }) => { diff --git a/packages/renderer/src/lib/troubleshooting/TroubleshootingGatherLogs.svelte b/packages/renderer/src/lib/troubleshooting/TroubleshootingGatherLogs.svelte new file mode 100644 index 0000000000000..6e92ee0de82e9 --- /dev/null +++ b/packages/renderer/src/lib/troubleshooting/TroubleshootingGatherLogs.svelte @@ -0,0 +1,55 @@ + + +
+
+ +
Gather Log Files
+
+
+
Bundle all available logs into a .zip
+
+ +
+ {#if logs.length > 0} +
+
    + {#each logs as log} +
  • +
    +
    + {log} generated +
    +
    +
  • + {/each} +
  • +
    +
    + {logs.length} logs collected and bundled as a .zip +
    +
    +
  • +
+
+ {/if} +
diff --git a/packages/renderer/src/lib/troubleshooting/TroubleshootingPage.svelte b/packages/renderer/src/lib/troubleshooting/TroubleshootingPage.svelte index 64079d2079459..52e9e0d42132f 100644 --- a/packages/renderer/src/lib/troubleshooting/TroubleshootingPage.svelte +++ b/packages/renderer/src/lib/troubleshooting/TroubleshootingPage.svelte @@ -2,6 +2,7 @@ import FormPage from '../ui/FormPage.svelte'; import Tab from '../ui/Tab.svelte'; import TroubleshootingDevToolsConsoleLogs from './TroubleshootingDevToolsConsoleLogs.svelte'; +import TroubleshootingGatherLogs from './TroubleshootingGatherLogs.svelte'; import TroubleshootingPageProviders from './TroubleshootingPageProviders.svelte'; import TroubleshootingPageStores from './TroubleshootingPageStores.svelte'; import Route from '/@/Route.svelte'; @@ -13,6 +14,7 @@ import Route from '/@/Route.svelte'; + @@ -24,6 +26,10 @@ import Route from '/@/Route.svelte'; + + + +