From 89611fb4180f63c8376e8151d96af9d78dfae359 Mon Sep 17 00:00:00 2001 From: Philippe Martin Date: Mon, 22 Jan 2024 13:30:52 +0100 Subject: [PATCH 1/6] Do not download model again --- .../src/managers/applicationManager.ts | 32 +++++++++++++------ 1 file changed, 22 insertions(+), 10 deletions(-) diff --git a/packages/backend/src/managers/applicationManager.ts b/packages/backend/src/managers/applicationManager.ts index 85bf17003..8f707a1ce 100644 --- a/packages/backend/src/managers/applicationManager.ts +++ b/packages/backend/src/managers/applicationManager.ts @@ -141,17 +141,29 @@ export class ApplicationManager { container => container.arch === undefined || container.arch === arch(), ); - // Download model - taskUtil.setTask({ - id: model.id, - state: 'loading', - name: `Downloading model ${model.name}`, - labels: { - 'model-pulling': model.id, - }, - }); + const localModels = this.getLocalModels(); + if (!localModels.map(m => m.id).includes(model.id)) { + // Download model + taskUtil.setTask({ + id: model.id, + state: 'loading', + name: `Downloading model ${model.name}`, + labels: { + 'model-pulling': model.id, + }, + }); - await this.downloadModelMain(model.id, model.url, taskUtil); + await this.downloadModelMain(model.id, model.url, taskUtil); + } else { + taskUtil.setTask({ + id: model.id, + state: 'success', + name: `Model ${model.name} already present on disk`, + labels: { + 'model-pulling': model.id, + }, + }); + } filteredContainers.forEach(container => { taskUtil.setTask({ From 6b4e88d087d24de4a41303e5ea9b08a790d6b10e Mon Sep 17 00:00:00 2001 From: Philippe Martin Date: Mon, 22 Jan 2024 16:27:59 +0100 Subject: [PATCH 2/6] add some unit tests for applicationManager --- .../src/managers/applicationManager.spec.ts | 56 +++++++++++++++++++ 1 file changed, 56 insertions(+) create mode 100644 packages/backend/src/managers/applicationManager.spec.ts diff --git a/packages/backend/src/managers/applicationManager.spec.ts b/packages/backend/src/managers/applicationManager.spec.ts new file mode 100644 index 000000000..692cf1068 --- /dev/null +++ b/packages/backend/src/managers/applicationManager.spec.ts @@ -0,0 +1,56 @@ +import { expect, test, vi } from 'vitest'; +import { ApplicationManager } from './applicationManager'; +import type { RecipeStatusRegistry } from '../registries/RecipeStatusRegistry'; +import type { ExtensionContext } from '@podman-desktop/api'; +import type { GitManager } from './gitManager'; +import os from 'os'; +import fs, { type Dirent } from 'fs'; +import path from 'path'; + +test('appUserDirectory should be under home directory', () => { + vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); + const manager = new ApplicationManager({} as GitManager, {} as RecipeStatusRegistry, {} as ExtensionContext); + expect(manager.appUserDirectory).toMatch(/^(\/|\\)home(\/|\\)user/); +}); + +test('getLocalModels should return models in local directory', () => { + vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); + // eslint-disable-next-line @typescript-eslint/no-explicit-any + vi.spyOn(fs, 'readdirSync').mockImplementation((dir: string): any => { + // TODO(feloy): fix any + if (dir.endsWith('model-id-1') || dir.endsWith('model-id-2')) { + const base = path.basename(dir); + return [base + '-model']; + } else { + return [ + { + isDirectory: () => true, + path: '/home/user/appstudio-dir', + name: 'model-id-1', + }, + { + isDirectory: () => true, + path: '/home/user/appstudio-dir', + name: 'model-id-2', + }, + { + isDirectory: () => false, + path: '/home/user/appstudio-dir', + name: 'other-file-should-be-ignored.txt', + }, + ] as Dirent[]; + } + }); + const manager = new ApplicationManager({} as GitManager, {} as RecipeStatusRegistry, {} as ExtensionContext); + const models = manager.getLocalModels(); + expect(models).toEqual([ + { + id: 'model-id-1', + file: 'model-id-1-model', + }, + { + id: 'model-id-2', + file: 'model-id-2-model', + }, + ]); +}); From fe94bbe37066c912efb1b9fc55e3623657431c2e Mon Sep 17 00:00:00 2001 From: Philippe Martin Date: Mon, 22 Jan 2024 17:19:35 +0100 Subject: [PATCH 3/6] test applicationManager.pullApplication --- .../src/managers/applicationManager.spec.ts | 84 ++++++++++++++++++- 1 file changed, 83 insertions(+), 1 deletion(-) diff --git a/packages/backend/src/managers/applicationManager.spec.ts b/packages/backend/src/managers/applicationManager.spec.ts index 692cf1068..7e6d913a8 100644 --- a/packages/backend/src/managers/applicationManager.spec.ts +++ b/packages/backend/src/managers/applicationManager.spec.ts @@ -4,8 +4,20 @@ import type { RecipeStatusRegistry } from '../registries/RecipeStatusRegistry'; import type { ExtensionContext } from '@podman-desktop/api'; import type { GitManager } from './gitManager'; import os from 'os'; -import fs, { type Dirent } from 'fs'; +import fs, { Stats, type Dirent } from 'fs'; import path from 'path'; +import type { Recipe } from '@shared/src/models/IRecipe'; +import type { ModelInfo } from '@shared/src/models/IModelInfo'; + +const mocks = vi.hoisted(() => { + return { + parseYamlMock: vi.fn(), + }; +}); + +vi.mock('../models/AIConfig', () => ({ + parseYaml: mocks.parseYamlMock, +})); test('appUserDirectory should be under home directory', () => { vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); @@ -54,3 +66,73 @@ test('getLocalModels should return models in local directory', () => { }, ]); }); + +test('pullApplication should clone repository and call downloadModelMain', async () => { + const setStatusMock = vi.fn(); + const cloneRepositoryMock = vi.fn(); + vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); + vi.spyOn(fs, 'mkdirSync').mockReturnValue(undefined); + vi.spyOn(fs, 'existsSync').mockImplementation((path: string) => { + if (path.endsWith('recipe1')) { + return false; + } else if (path.endsWith('ai-studio.yaml')) { + return true; + } + return false; + }); + vi.spyOn(fs, 'statSync').mockImplementation((path: string) => { + if (path.endsWith('recipe1')) { + const stat = new Stats(); + stat.isDirectory = () => true; + return stat; + } else if (path.endsWith('ai-studio.yaml')) { + const stat = new Stats(); + stat.isDirectory = () => false; + return stat; + } + }); + vi.spyOn(fs, 'readFileSync').mockImplementation((_path: string) => { + return ''; + }); + mocks.parseYamlMock.mockReturnValue({ + application: { + containers: [], + }, + }); + const manager = new ApplicationManager( + { + cloneRepository: cloneRepositoryMock, + } as unknown as GitManager, + { + setStatus: setStatusMock, + } as unknown as RecipeStatusRegistry, + {} as ExtensionContext, + ); + + const getLocalModelsSpy = vi.spyOn(manager, 'getLocalModels'); + getLocalModelsSpy.mockReturnValue([]); + const downloadModelMainSpy = vi.spyOn(manager, 'downloadModelMain'); + downloadModelMainSpy.mockResolvedValue(''); + + const recipe: Recipe = { + id: 'recipe1', + name: 'Recipe 1', + categories: [], + description: '', + readme: '', + repository: 'repo', + }; + const model: ModelInfo = { + id: 'model1', + description: '', + hw: '', + license: '', + name: 'Model 1', + popularity: 1, + registry: '', + url: '', + }; + await manager.pullApplication(recipe, model); + expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '/home/user/podman-desktop/ai-studio/recipe1'); + expect(downloadModelMainSpy).toHaveBeenCalledOnce(); +}); From b1316e315db0881d13543847cc082d8a8d91098d Mon Sep 17 00:00:00 2001 From: Philippe Martin Date: Mon, 22 Jan 2024 17:32:13 +0100 Subject: [PATCH 4/6] tests: fix windows paths --- .../backend/src/managers/applicationManager.spec.ts | 12 ++++++++++-- 1 file changed, 10 insertions(+), 2 deletions(-) diff --git a/packages/backend/src/managers/applicationManager.spec.ts b/packages/backend/src/managers/applicationManager.spec.ts index 7e6d913a8..04321edd4 100644 --- a/packages/backend/src/managers/applicationManager.spec.ts +++ b/packages/backend/src/managers/applicationManager.spec.ts @@ -22,7 +22,11 @@ vi.mock('../models/AIConfig', () => ({ test('appUserDirectory should be under home directory', () => { vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); const manager = new ApplicationManager({} as GitManager, {} as RecipeStatusRegistry, {} as ExtensionContext); - expect(manager.appUserDirectory).toMatch(/^(\/|\\)home(\/|\\)user/); + if (process.platform === 'win32') { + expect(manager.appUserDirectory).toMatch(/^\\home\\user/); + } else { + expect(manager.appUserDirectory).toMatch(/^\/home\/user/); + } }); test('getLocalModels should return models in local directory', () => { @@ -133,6 +137,10 @@ test('pullApplication should clone repository and call downloadModelMain', async url: '', }; await manager.pullApplication(recipe, model); - expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '/home/user/podman-desktop/ai-studio/recipe1'); + if (process.platform === 'win32') { + expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '\\home\\user\\podman-desktop\\ai-studio\\recipe1'); + } else { + expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '/home/user/podman-desktop/ai-studio/recipe1'); + } expect(downloadModelMainSpy).toHaveBeenCalledOnce(); }); From 0224f4fc75767045c22a4fee1825930969781120 Mon Sep 17 00:00:00 2001 From: Philippe Martin Date: Mon, 22 Jan 2024 19:45:34 +0100 Subject: [PATCH 5/6] tests: more applicationManager.pullApplication tests --- .../src/managers/applicationManager.spec.ts | 244 +++++++++++++----- packages/backend/vitest.config.js | 10 +- 2 files changed, 185 insertions(+), 69 deletions(-) diff --git a/packages/backend/src/managers/applicationManager.spec.ts b/packages/backend/src/managers/applicationManager.spec.ts index 04321edd4..fe835be8a 100644 --- a/packages/backend/src/managers/applicationManager.spec.ts +++ b/packages/backend/src/managers/applicationManager.spec.ts @@ -1,4 +1,4 @@ -import { expect, test, vi } from 'vitest'; +import { type MockInstance, describe, expect, test, vi, beforeEach } from 'vitest'; import { ApplicationManager } from './applicationManager'; import type { RecipeStatusRegistry } from '../registries/RecipeStatusRegistry'; import type { ExtensionContext } from '@podman-desktop/api'; @@ -8,10 +8,13 @@ import fs, { Stats, type Dirent } from 'fs'; import path from 'path'; import type { Recipe } from '@shared/src/models/IRecipe'; import type { ModelInfo } from '@shared/src/models/IModelInfo'; +import type { LocalModelInfo } from '@shared/src/models/ILocalModelInfo'; +import type { RecipeStatusUtils } from '../utils/recipeStatusUtils'; const mocks = vi.hoisted(() => { return { parseYamlMock: vi.fn(), + builImageMock: vi.fn(), }; }); @@ -19,6 +22,16 @@ vi.mock('../models/AIConfig', () => ({ parseYaml: mocks.parseYamlMock, })); +vi.mock('@podman-desktop/api', () => ({ + containerEngine: { + buildImage: mocks.builImageMock, + }, +})); + +beforeEach(() => { + vi.resetAllMocks(); +}); + test('appUserDirectory should be under home directory', () => { vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); const manager = new ApplicationManager({} as GitManager, {} as RecipeStatusRegistry, {} as ExtensionContext); @@ -71,76 +84,175 @@ test('getLocalModels should return models in local directory', () => { ]); }); -test('pullApplication should clone repository and call downloadModelMain', async () => { +describe('pullApplication', () => { + interface mockForPullApplicationOptions { + recipeFolderExists: boolean; + } + const setStatusMock = vi.fn(); const cloneRepositoryMock = vi.fn(); - vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); - vi.spyOn(fs, 'mkdirSync').mockReturnValue(undefined); - vi.spyOn(fs, 'existsSync').mockImplementation((path: string) => { - if (path.endsWith('recipe1')) { + let manager: ApplicationManager; + let getLocalModelsSpy: MockInstance<[], LocalModelInfo[]>; + let downloadModelMainSpy: MockInstance< + [modelId: string, url: string, taskUtil: RecipeStatusUtils, destFileName?: string], + Promise + >; + + function mockForPullApplication(options: mockForPullApplicationOptions) { + vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); + vi.spyOn(fs, 'mkdirSync').mockReturnValue(undefined); + vi.spyOn(fs, 'existsSync').mockImplementation((path: string) => { + if (path.endsWith('recipe1')) { + return options.recipeFolderExists; + } else if (path.endsWith('ai-studio.yaml')) { + return true; + } else if (path.endsWith('contextdir1')) { + return true; + } return false; - } else if (path.endsWith('ai-studio.yaml')) { - return true; - } - return false; - }); - vi.spyOn(fs, 'statSync').mockImplementation((path: string) => { - if (path.endsWith('recipe1')) { - const stat = new Stats(); - stat.isDirectory = () => true; - return stat; - } else if (path.endsWith('ai-studio.yaml')) { - const stat = new Stats(); - stat.isDirectory = () => false; - return stat; + }); + vi.spyOn(fs, 'statSync').mockImplementation((path: string) => { + if (path.endsWith('recipe1')) { + const stat = new Stats(); + stat.isDirectory = () => true; + return stat; + } else if (path.endsWith('ai-studio.yaml')) { + const stat = new Stats(); + stat.isDirectory = () => false; + return stat; + } + }); + vi.spyOn(fs, 'readFileSync').mockImplementation((_path: string) => { + return ''; + }); + mocks.parseYamlMock.mockReturnValue({ + application: { + containers: [ + { + name: 'container1', + contextdir: 'contextdir1', + containerfile: 'Containerfile', + }, + ], + }, + }); + mocks.builImageMock.mockResolvedValue(undefined); + + manager = new ApplicationManager( + { + cloneRepository: cloneRepositoryMock, + } as unknown as GitManager, + { + setStatus: setStatusMock, + } as unknown as RecipeStatusRegistry, + {} as ExtensionContext, + ); + + getLocalModelsSpy = vi.spyOn(manager, 'getLocalModels'); + downloadModelMainSpy = vi.spyOn(manager, 'downloadModelMain'); + downloadModelMainSpy.mockResolvedValue(''); + } + + test('pullApplication should clone repository and call downloadModelMain and buildImage', async () => { + mockForPullApplication({ + recipeFolderExists: false, + }); + getLocalModelsSpy.mockReturnValue([]); + + const recipe: Recipe = { + id: 'recipe1', + name: 'Recipe 1', + categories: [], + description: '', + readme: '', + repository: 'repo', + }; + const model: ModelInfo = { + id: 'model1', + description: '', + hw: '', + license: '', + name: 'Model 1', + popularity: 1, + registry: '', + url: '', + }; + + await manager.pullApplication(recipe, model); + if (process.platform === 'win32') { + expect(cloneRepositoryMock).toHaveBeenNthCalledWith( + 1, + 'repo', + '\\home\\user\\podman-desktop\\ai-studio\\recipe1', + ); + } else { + expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '/home/user/podman-desktop/ai-studio/recipe1'); } + expect(downloadModelMainSpy).toHaveBeenCalledOnce(); + expect(mocks.builImageMock).toHaveBeenCalledOnce(); }); - vi.spyOn(fs, 'readFileSync').mockImplementation((_path: string) => { - return ''; + + test('pullApplication should not clone repository if folder already exists locally', async () => { + mockForPullApplication({ + recipeFolderExists: true, + }); + getLocalModelsSpy.mockReturnValue([]); + + const recipe: Recipe = { + id: 'recipe1', + name: 'Recipe 1', + categories: [], + description: '', + readme: '', + repository: 'repo', + }; + const model: ModelInfo = { + id: 'model1', + description: '', + hw: '', + license: '', + name: 'Model 1', + popularity: 1, + registry: '', + url: '', + }; + + await manager.pullApplication(recipe, model); + expect(cloneRepositoryMock).not.toHaveBeenCalled(); }); - mocks.parseYamlMock.mockReturnValue({ - application: { - containers: [], - }, + + test('pullApplication should not download model if already on disk', async () => { + mockForPullApplication({ + recipeFolderExists: true, + }); + getLocalModelsSpy.mockReturnValue([ + { + id: 'model1', + file: 'model1.file', + }, + ]); + + const recipe: Recipe = { + id: 'recipe1', + name: 'Recipe 1', + categories: [], + description: '', + readme: '', + repository: 'repo', + }; + const model: ModelInfo = { + id: 'model1', + description: '', + hw: '', + license: '', + name: 'Model 1', + popularity: 1, + registry: '', + url: '', + }; + + await manager.pullApplication(recipe, model); + expect(cloneRepositoryMock).not.toHaveBeenCalled(); + expect(downloadModelMainSpy).not.toHaveBeenCalled(); }); - const manager = new ApplicationManager( - { - cloneRepository: cloneRepositoryMock, - } as unknown as GitManager, - { - setStatus: setStatusMock, - } as unknown as RecipeStatusRegistry, - {} as ExtensionContext, - ); - - const getLocalModelsSpy = vi.spyOn(manager, 'getLocalModels'); - getLocalModelsSpy.mockReturnValue([]); - const downloadModelMainSpy = vi.spyOn(manager, 'downloadModelMain'); - downloadModelMainSpy.mockResolvedValue(''); - - const recipe: Recipe = { - id: 'recipe1', - name: 'Recipe 1', - categories: [], - description: '', - readme: '', - repository: 'repo', - }; - const model: ModelInfo = { - id: 'model1', - description: '', - hw: '', - license: '', - name: 'Model 1', - popularity: 1, - registry: '', - url: '', - }; - await manager.pullApplication(recipe, model); - if (process.platform === 'win32') { - expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '\\home\\user\\podman-desktop\\ai-studio\\recipe1'); - } else { - expect(cloneRepositoryMock).toHaveBeenNthCalledWith(1, 'repo', '/home/user/podman-desktop/ai-studio/recipe1'); - } - expect(downloadModelMainSpy).toHaveBeenCalledOnce(); }); diff --git a/packages/backend/vitest.config.js b/packages/backend/vitest.config.js index 0c675e04b..d86d61da3 100644 --- a/packages/backend/vitest.config.js +++ b/packages/backend/vitest.config.js @@ -23,9 +23,13 @@ const PACKAGE_ROOT = __dirname; const config = { test: { - include: ['**/*.{test,spec}.?(c|m)[jt]s?(x)', '../shared/**/*.{test,spec}.?(c|m)[jt]s?(x)'] - }, - resolve: { + include: ['**/*.{test,spec}.?(c|m)[jt]s?(x)', '../shared/**/*.{test,spec}.?(c|m)[jt]s?(x)'], + coverage: { + provider: 'v8', + reporter: ['lcov', 'text'], + }, +}, +resolve: { alias: { '@podman-desktop/api': path.resolve(__dirname, '__mocks__/@podman-desktop/api.js'), '/@/': join(PACKAGE_ROOT, 'src') + '/', From 1ff9821c3a8bcdf31801042d74d9ecef574ab507 Mon Sep 17 00:00:00 2001 From: Philippe Martin Date: Tue, 23 Jan 2024 09:06:38 +0100 Subject: [PATCH 6/6] fix any in tests --- packages/backend/src/managers/applicationManager.spec.ts | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/packages/backend/src/managers/applicationManager.spec.ts b/packages/backend/src/managers/applicationManager.spec.ts index fe835be8a..f6dd4927d 100644 --- a/packages/backend/src/managers/applicationManager.spec.ts +++ b/packages/backend/src/managers/applicationManager.spec.ts @@ -45,8 +45,11 @@ test('appUserDirectory should be under home directory', () => { test('getLocalModels should return models in local directory', () => { vi.spyOn(os, 'homedir').mockReturnValue('/home/user'); // eslint-disable-next-line @typescript-eslint/no-explicit-any - vi.spyOn(fs, 'readdirSync').mockImplementation((dir: string): any => { - // TODO(feloy): fix any + const readdirSyncMock = vi.spyOn(fs, 'readdirSync') as unknown as MockInstance< + [path: string], + string[] | fs.Dirent[] + >; + readdirSyncMock.mockImplementation((dir: string) => { if (dir.endsWith('model-id-1') || dir.endsWith('model-id-2')) { const base = path.basename(dir); return [base + '-model'];