From d0564808669c0211a11e1b0d3c4520b883a65e38 Mon Sep 17 00:00:00 2001 From: Robin Huang Date: Tue, 5 Nov 2024 15:45:37 -0800 Subject: [PATCH] Refactor tests directory to be outside src. --- tests/e2e/startup.test.ts | 26 +++ .../startup-darwin.png | Bin 0 -> 30263 bytes tests/unit/comfyConfigManager.test.ts | 165 ++++++++++++++++++ tests/unit/main.test.ts | 105 +++++++++++ 4 files changed, 296 insertions(+) create mode 100644 tests/e2e/startup.test.ts create mode 100644 tests/e2e/startup.test.ts-snapshots/startup-darwin.png create mode 100644 tests/unit/comfyConfigManager.test.ts create mode 100644 tests/unit/main.test.ts diff --git a/tests/e2e/startup.test.ts b/tests/e2e/startup.test.ts new file mode 100644 index 00000000..e1d87ca0 --- /dev/null +++ b/tests/e2e/startup.test.ts @@ -0,0 +1,26 @@ +import { test, _electron as electron, expect } from '@playwright/test'; + +test('launch app', async () => { + const electronApp = await electron.launch({ args: ['.'] }); + electronApp.process().stdout?.on?.('data', (data) => { + console.log(`Electron stdout: ${data}`); + }); + electronApp.process().stderr?.on?.('data', (data) => { + console.error(`Electron stderr: ${data}`); + }); + + const isPackaged = await electronApp.evaluate(async ({ app }) => { + // This runs in Electron's main process, parameter here is always + // the result of the require('electron') in the main app script. + return app.isPackaged; + }); + + expect(isPackaged).toBe(false); + + // Wait for the first BrowserWindow to open + // and return its Page object + const window = await electronApp.firstWindow(); + await expect(window).toHaveScreenshot('startup.png'); + + await electronApp.close(); +}); diff --git a/tests/e2e/startup.test.ts-snapshots/startup-darwin.png b/tests/e2e/startup.test.ts-snapshots/startup-darwin.png new file mode 100644 index 0000000000000000000000000000000000000000..2c5a12b065fc27891cc1da3085fec09573605663 GIT binary patch literal 30263 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 literal 0 HcmV?d00001 diff --git a/tests/unit/comfyConfigManager.test.ts b/tests/unit/comfyConfigManager.test.ts new file mode 100644 index 00000000..2532abc4 --- /dev/null +++ b/tests/unit/comfyConfigManager.test.ts @@ -0,0 +1,165 @@ +import fs from 'fs'; +import { ComfyConfigManager, DirectoryStructure } from '../../src/config/comfyConfigManager'; + +// Mock the fs module +jest.mock('fs'); +jest.mock('electron-log/main', () => ({ + info: jest.fn(), + error: jest.fn(), + warn: jest.fn(), +})); + +describe('ComfyConfigManager', () => { + // Reset all mocks before each test + beforeEach(() => { + jest.clearAllMocks(); + (fs.existsSync as jest.Mock).mockReset(); + (fs.mkdirSync as jest.Mock).mockReset(); + (fs.writeFileSync as jest.Mock).mockReset(); + (fs.renameSync as jest.Mock).mockReset(); + }); + + describe('setUpComfyUI', () => { + it('should use existing directory when it contains ComfyUI structure', () => { + // Mock isComfyUIDirectory to return true for the input path + (fs.existsSync as jest.Mock).mockImplementation((path: string) => { + const requiredDirs = [ + '/existing/ComfyUI/models', + '/existing/ComfyUI/input', + '/existing/ComfyUI/user', + '/existing/ComfyUI/output', + '/existing/ComfyUI/custom_nodes', + ]; + return requiredDirs.includes(path); + }); + + const result = ComfyConfigManager.setUpComfyUI('/existing/ComfyUI'); + + expect(result).toBe('/existing/ComfyUI'); + }); + + it('should create ComfyUI subdirectory when it is missing', () => { + (fs.existsSync as jest.Mock).mockImplementationOnce((path: string) => { + if (path === '/some/base/path/ComfyUI') { + return false; + } + return true; + }); + + const result = ComfyConfigManager.setUpComfyUI('/some/base/path'); + + expect(result).toBe('/some/base/path/ComfyUI'); + }); + }); + + describe('isComfyUIDirectory', () => { + it('should return true when all required directories exist', () => { + (fs.existsSync as jest.Mock).mockImplementation((path: string) => { + const requiredDirs = [ + '/fake/path/models', + '/fake/path/input', + '/fake/path/user', + '/fake/path/output', + '/fake/path/custom_nodes', + ]; + return requiredDirs.includes(path); + }); + + const result = ComfyConfigManager.isComfyUIDirectory('/fake/path'); + + expect(result).toBe(true); + expect(fs.existsSync).toHaveBeenCalledTimes(5); + }); + + it('should return false when some required directories are missing', () => { + (fs.existsSync as jest.Mock) + .mockReturnValueOnce(true) // models exists + .mockReturnValueOnce(true) // input exists + .mockReturnValueOnce(false) // user missing + .mockReturnValueOnce(true) // output exists + .mockReturnValueOnce(true); // custom_nodes exists + + const result = ComfyConfigManager.isComfyUIDirectory('/fake/path'); + + expect(result).toBe(false); + }); + }); + + describe('createComfyDirectories', () => { + it('should create all necessary directories when none exist', () => { + (fs.existsSync as jest.Mock).mockReturnValue(false); + + ComfyConfigManager.createComfyDirectories('/fake/path/ComfyUI'); + + // Verify each required directory was created + expect(fs.mkdirSync).toHaveBeenCalledWith('/fake/path/ComfyUI/models', { recursive: true }); + expect(fs.mkdirSync).toHaveBeenCalledWith('/fake/path/ComfyUI/input', { recursive: true }); + expect(fs.mkdirSync).toHaveBeenCalledWith('/fake/path/ComfyUI/user', { recursive: true }); + expect(fs.mkdirSync).toHaveBeenCalledWith('/fake/path/ComfyUI/output', { recursive: true }); + expect(fs.mkdirSync).toHaveBeenCalledWith('/fake/path/ComfyUI/custom_nodes', { recursive: true }); + }); + }); + + describe('createComfyConfigFile', () => { + it('should create new config file when none exists', () => { + (fs.existsSync as jest.Mock).mockReturnValue(false); + + ComfyConfigManager.createComfyConfigFile('/fake/path', false); + + expect(fs.writeFileSync).toHaveBeenCalledTimes(1); + expect(fs.renameSync).not.toHaveBeenCalled(); + }); + + it('should backup existing config file when overwrite is true', () => { + (fs.existsSync as jest.Mock).mockImplementation((path: string) => { + return path === '/user/default/comfy.settings.json'; + }); + + ComfyConfigManager.createComfyConfigFile('/user/default', true); + + expect(fs.renameSync).toHaveBeenCalledTimes(1); + expect(fs.writeFileSync).toHaveBeenCalledTimes(1); + }); + + it('should handle backup failure gracefully', () => { + (fs.existsSync as jest.Mock).mockReturnValue(true); + (fs.renameSync as jest.Mock).mockImplementation(() => { + throw new Error('Backup failed'); + }); + + ComfyConfigManager.createComfyConfigFile('/fake/path', true); + + expect(fs.writeFileSync).not.toHaveBeenCalled(); + }); + }); + + describe('createNestedDirectories', () => { + it('should create nested directory structure correctly', () => { + (fs.existsSync as jest.Mock).mockReturnValue(false); + + const structure = ['dir1', ['dir2', ['subdir1', 'subdir2']], ['dir3', [['subdir3', ['subsubdir1']]]]]; + + ComfyConfigManager['createNestedDirectories']('/fake/path', structure); + + // Verify the correct paths were created + expect(fs.mkdirSync).toHaveBeenCalledWith(expect.stringContaining('dir1'), expect.any(Object)); + expect(fs.mkdirSync).toHaveBeenCalledWith(expect.stringContaining('dir2'), expect.any(Object)); + expect(fs.mkdirSync).toHaveBeenCalledWith(expect.stringContaining('subdir1'), expect.any(Object)); + expect(fs.mkdirSync).toHaveBeenCalledWith(expect.stringContaining('subsubdir1'), expect.any(Object)); + }); + + it('should handle invalid directory structure items', () => { + const invalidStructure = [ + 'dir1', + ['dir2'], // Invalid: array with only one item + [123, ['subdir1']], // Invalid: non-string directory name + ]; + + ComfyConfigManager['createNestedDirectories']('/fake/path', invalidStructure as DirectoryStructure); + + // Verify only valid directories were created + expect(fs.mkdirSync).toHaveBeenCalledWith(expect.stringContaining('dir1'), expect.any(Object)); + expect(fs.mkdirSync).not.toHaveBeenCalledWith(expect.stringContaining('subdir1'), expect.any(Object)); + }); + }); +}); diff --git a/tests/unit/main.test.ts b/tests/unit/main.test.ts new file mode 100644 index 00000000..50f3ae0f --- /dev/null +++ b/tests/unit/main.test.ts @@ -0,0 +1,105 @@ +import { expect, jest, describe, it } from '@jest/globals'; +import { createWindow } from '../../src/main'; +import { BrowserWindow } from 'electron'; + +global.MAIN_WINDOW_VITE_DEV_SERVER_URL = 'http://localhost:5173'; +global.MAIN_WINDOW_VITE_NAME = 'index.html'; + +jest.mock('node:path', () => ({ + join: jest.fn((...args) => { + return 'preload.js'; + }), +})); + +jest.mock('@sentry/electron/main', () => ({ + init: jest.fn(), + captureException: jest.fn(), +})); + +jest.mock('tar', () => ({ + extract: jest.fn(), +})); +jest.mock('axios'); +jest.mock('fs'); +jest.mock('node:fs/promises'); + +const mockMenuInstance = { + append: jest.fn(), + popup: jest.fn(), + closePopup: jest.fn(), +}; + +const MockMenu = jest.fn(() => mockMenuInstance) as jest.Mock & { + buildFromTemplate: jest.Mock; +}; +MockMenu.buildFromTemplate = jest.fn().mockReturnValue({ + items: [], +}); + +jest.mock('electron', () => ({ + app: { + isPackaged: false, + isReady: true, + on: jest.fn(), + getPath: jest.fn(), + requestSingleInstanceLock: jest.fn().mockReturnValue(true), + }, + BrowserWindow: jest.fn().mockImplementation((options) => { + return { + loadURL: jest.fn(), + on: jest.fn(), + webContents: { + openDevTools: jest.fn(), + }, + }; + }), + ipcMain: { + on: jest.fn(), + handle: jest.fn(), + }, + screen: { + getPrimaryDisplay: jest.fn().mockReturnValue({ + workAreaSize: { width: 1920, height: 1080 }, + }), + }, + // Add this line to mock Tray + Tray: jest.fn().mockImplementation(() => ({ + setToolTip: jest.fn(), + setContextMenu: jest.fn(), + on: jest.fn(), + setPressedImage: jest.fn(), + })), + // Add this line to mock Menu + Menu: MockMenu, + // Mock other Electron modules if necessary +})); + +jest.mock('electron-log/main', () => ({ + initialize: jest.fn(), + info: jest.fn(), + error: jest.fn(), + // Add other methods you might use from electron-log +})); + +describe('createWindow', () => { + // it('should create a new BrowserWindow with correct options', async () => { + // const window = await createWindow('/'); + + // expect(BrowserWindow).toHaveBeenCalledWith( + // expect.objectContaining({ + // title: 'ComfyUI', + // webPreferences: expect.objectContaining({ + // preload: expect.stringContaining('preload.js'), + // nodeIntegration: true, + // contextIsolation: true, + // }), + // autoHideMenuBar: true, + // }) + // ); + // expect(window.loadURL).toHaveBeenCalled(); + // }); + + it('just passes', () => { + expect(true).toBe(true); + }); +});