Skip to main content
Glama
run.test.ts16.6 kB
/** * @vitest-environment node */ import { vi } from 'vitest'; import { asProjectKey, asRunId, asCommitOid, asBranchName, asGraphQLNodeId, } from '../../types/branded'; import type { IAnalysisRunRepository } from '../../domain/aggregates/analysis-run/analysis-run.repository'; import type { AnalysisRun } from '../../domain/aggregates/analysis-run/analysis-run.aggregate'; import type { Logger } from '../../utils/logging/logger'; import { IssueCount } from '../../domain/value-objects/issue-count'; // Create mock logger const mockLogger = { debug: vi.fn(), info: vi.fn(), warn: vi.fn(), error: vi.fn(), }; // Mock modules before importing the implementation vi.mock('../../utils/logging/logger', () => ({ createLogger: vi.fn(() => mockLogger), })); // Mock the repository and factory const mockFindByRunId = vi.fn(); const mockFindByCommit = vi.fn(); const mockAnalysisRunRepository = { findByRunId: mockFindByRunId, findByCommit: mockFindByCommit, } as unknown as IAnalysisRunRepository; const mockCreateAnalysisRunRepository = vi.fn(() => mockAnalysisRunRepository); const mockRepositoryFactory = vi.fn(() => ({ createAnalysisRunRepository: mockCreateAnalysisRunRepository, })); vi.mock('../../infrastructure/factories/repository.factory', () => ({ RepositoryFactory: mockRepositoryFactory, })); // Import the modules under test AFTER mocking const { handleDeepsourceRun, createRunHandlerWithRepo } = await import('../../handlers/run'); describe('Run Handler', () => { // Environment backup let originalEnv: Record<string, string | undefined>; beforeEach(() => { // Backup environment originalEnv = { ...process.env }; process.env.DEEPSOURCE_API_KEY = 'test-api-key'; // Reset mocks vi.clearAllMocks(); }); afterEach(() => { // Restore environment process.env = originalEnv; }); describe('createRunHandlerWithRepo', () => { it('should create a handler that finds run by runId', async () => { // Mock domain analysis run const projectKey = asProjectKey('test-project'); const runId = asRunId('run-123'); const mockRun = { runId, projectKey, repositoryId: asGraphQLNodeId('repo-1'), commitInfo: { oid: asCommitOid('commit-abc'), branch: asBranchName('main'), baseOid: asCommitOid('base-def'), }, status: 'SUCCESS' as const, timestamps: { createdAt: new Date('2023-01-01T00:00:00Z'), startedAt: new Date('2023-01-01T00:01:00Z'), finishedAt: new Date('2023-01-01T00:05:00Z'), }, summary: { totalIntroduced: IssueCount.create(3), totalResolved: IssueCount.create(1), totalSuppressed: IssueCount.create(0), byAnalyzer: [ { analyzerShortcode: 'eslint', introduced: IssueCount.create(2), resolved: IssueCount.create(1), suppressed: IssueCount.create(0), }, ], byCategory: [ { category: 'BUG_RISK', introduced: IssueCount.create(3), resolved: IssueCount.create(1), suppressed: IssueCount.create(0), }, ], }, } as AnalysisRun; // Set up the mock to return the run mockFindByRunId.mockResolvedValue(mockRun); const handler = createRunHandlerWithRepo({ analysisRunRepository: mockAnalysisRunRepository, logger: mockLogger as unknown as Logger, }); // Call the handler with runId const result = await handler({ projectKey: 'test-project', runIdentifier: 'run-123', isCommitOid: false, }); // Verify repository was used correctly expect(mockFindByRunId).toHaveBeenCalledWith(runId); expect(mockFindByCommit).not.toHaveBeenCalled(); // Verify the response const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.run.id).toBe('run-123'); expect(parsedContent.run.runUid).toBe('run-123'); expect(parsedContent.run.status).toBe('SUCCESS'); expect(parsedContent.run.commitOid).toBe('commit-abc'); expect(parsedContent.run.branchName).toBe('main'); expect(parsedContent.run.summary.occurrencesIntroduced).toBe(3); expect(parsedContent.run.summary.occurrencesResolved).toBe(1); expect(parsedContent.run.summary.occurrencesSuppressed).toBe(0); expect(parsedContent.analysis.analyzers_used).toEqual(['eslint']); expect(parsedContent.analysis.issue_categories).toEqual(['BUG_RISK']); expect(parsedContent.related_tools).toBeDefined(); }); it('should create a handler that finds run by commitOid', async () => { // Mock domain analysis run const projectKey = asProjectKey('test-project'); const commitOid = asCommitOid('commit-xyz'); const mockRun = { runId: asRunId('run-456'), projectKey, repositoryId: asGraphQLNodeId('repo-1'), commitInfo: { oid: commitOid, branch: asBranchName('feature'), baseOid: asCommitOid('base-ghi'), }, status: 'FAILURE' as const, timestamps: { createdAt: new Date('2023-01-02T00:00:00Z'), finishedAt: new Date('2023-01-02T00:03:00Z'), }, summary: { totalIntroduced: IssueCount.create(5), totalResolved: IssueCount.create(0), totalSuppressed: IssueCount.create(1), byAnalyzer: [], byCategory: [], }, } as AnalysisRun; // Set up the mock to return the run mockFindByCommit.mockResolvedValue(mockRun); const handler = createRunHandlerWithRepo({ analysisRunRepository: mockAnalysisRunRepository, logger: mockLogger as unknown as Logger, }); // Call the handler with commitOid const result = await handler({ projectKey: 'test-project', runIdentifier: 'commit-xyz', isCommitOid: true, }); // Verify repository was used correctly expect(mockFindByCommit).toHaveBeenCalledWith(projectKey, commitOid); expect(mockFindByRunId).not.toHaveBeenCalled(); // Verify the response const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.run.id).toBe('run-456'); expect(parsedContent.run.status).toBe('FAILURE'); expect(parsedContent.run.commitOid).toBe('commit-xyz'); expect(parsedContent.run.branchName).toBe('feature'); expect(parsedContent.analysis.status_info).toContain('failed'); }); it('should handle run not found', async () => { // Set up the mock to return null mockFindByRunId.mockResolvedValue(null); const handler = createRunHandlerWithRepo({ analysisRunRepository: mockAnalysisRunRepository, logger: mockLogger as unknown as Logger, }); // Call the handler const result = await handler({ projectKey: 'test-project', runIdentifier: 'nonexistent-run', isCommitOid: false, }); // Verify error response expect(result.isError).toBe(true); const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.error).toContain('Run with runUid "nonexistent-run" not found'); expect(parsedContent.details).toBe('Failed to retrieve run'); }); it('should handle run not found by commitOid', async () => { // Set up the mock to return null mockFindByCommit.mockResolvedValue(null); const handler = createRunHandlerWithRepo({ analysisRunRepository: mockAnalysisRunRepository, logger: mockLogger as unknown as Logger, }); // Call the handler const result = await handler({ projectKey: 'test-project', runIdentifier: 'nonexistent-commit', isCommitOid: true, }); // Verify error response expect(result.isError).toBe(true); const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.error).toContain('Run with commitOid "nonexistent-commit" not found'); }); it('should handle errors using injected logger', async () => { // Set up the mock to throw an error const testError = new Error('Repository connection failed'); mockFindByRunId.mockRejectedValue(testError); const handler = createRunHandlerWithRepo({ analysisRunRepository: mockAnalysisRunRepository, logger: mockLogger as unknown as Logger, }); // Call the handler const result = await handler({ projectKey: 'test-project', runIdentifier: 'run-123', isCommitOid: false, }); // Verify error was logged using injected logger expect(mockLogger.error).toHaveBeenCalledWith('Error in handleRun', expect.any(Object)); // Verify error response expect(result.isError).toBe(true); const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent).toEqual({ error: 'Repository connection failed', details: 'Failed to retrieve run', }); }); it('should handle run with missing optional data', async () => { // Mock domain analysis run with minimal data const mockRun = { runId: asRunId('run-minimal'), projectKey: asProjectKey('test-project'), repositoryId: asGraphQLNodeId('repo-1'), commitInfo: { oid: asCommitOid('commit-minimal'), branch: asBranchName('main'), baseOid: asCommitOid('base-minimal'), }, status: 'PENDING' as const, timestamps: { createdAt: new Date('2023-01-01T00:00:00Z'), }, summary: { totalIntroduced: IssueCount.create(0), totalResolved: IssueCount.create(0), totalSuppressed: IssueCount.create(0), byAnalyzer: [], byCategory: [], }, } as AnalysisRun; mockFindByRunId.mockResolvedValue(mockRun); const handler = createRunHandlerWithRepo({ analysisRunRepository: mockAnalysisRunRepository, logger: mockLogger as unknown as Logger, }); // Call the handler const result = await handler({ projectKey: 'test-project', runIdentifier: 'run-minimal', isCommitOid: false, }); // Verify the response handles missing optional data gracefully const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.run.status).toBe('PENDING'); expect(parsedContent.run.finishedAt).toBeUndefined(); expect(parsedContent.analysis.analyzers_used).toEqual([]); expect(parsedContent.analysis.issue_categories).toEqual([]); expect(parsedContent.analysis.status_info).toContain('queued'); }); }); describe('handleDeepsourceRun', () => { it('should throw an error if DEEPSOURCE_API_KEY is not set', async () => { // Unset API key delete process.env.DEEPSOURCE_API_KEY; // Call the handler and expect it to throw await expect( handleDeepsourceRun({ projectKey: 'test-project', runIdentifier: 'run-123', isCommitOid: false, }) ).rejects.toThrow('Configuration error: DeepSource API key is required but not configured'); }); it('should return run data successfully by runId', async () => { // Mock domain analysis run const runId = asRunId('run-success'); const mockRun = { runId, projectKey: asProjectKey('test-project'), repositoryId: asGraphQLNodeId('repo-1'), commitInfo: { oid: asCommitOid('commit-success'), branch: asBranchName('main'), baseOid: asCommitOid('base-success'), }, status: 'SUCCESS' as const, timestamps: { createdAt: new Date('2023-01-01T00:00:00Z'), startedAt: new Date('2023-01-01T00:01:00Z'), finishedAt: new Date('2023-01-01T00:05:00Z'), }, summary: { totalIntroduced: IssueCount.create(2), totalResolved: IssueCount.create(1), totalSuppressed: IssueCount.create(0), byAnalyzer: [], byCategory: [], }, } as AnalysisRun; // Set up the mock to return the run mockFindByRunId.mockResolvedValue(mockRun); // Call the handler const result = await handleDeepsourceRun({ projectKey: 'test-project', runIdentifier: 'run-success', isCommitOid: false, }); // Verify factory was created with the API key expect(mockRepositoryFactory).toHaveBeenCalledWith({ apiKey: 'test-api-key' }); // Verify createAnalysisRunRepository was called expect(mockCreateAnalysisRunRepository).toHaveBeenCalled(); // Verify findByRunId was called expect(mockFindByRunId).toHaveBeenCalledWith(runId); // Verify logging behavior expect(mockLogger.info).toHaveBeenCalledWith( 'Fetching run from repository', expect.any(Object) ); // Verify the response structure expect(result).toHaveProperty('content'); expect(result.content).toHaveLength(1); expect(result.content[0]).toHaveProperty('type', 'text'); // Parse and verify the content const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.run.id).toBe('run-success'); expect(parsedContent.run.status).toBe('SUCCESS'); }); it('should return run data successfully by commitOid', async () => { // Mock domain analysis run const projectKey = asProjectKey('test-project'); const commitOid = asCommitOid('commit-by-oid'); const mockRun = { runId: asRunId('run-by-commit'), projectKey, repositoryId: asGraphQLNodeId('repo-1'), commitInfo: { oid: commitOid, branch: asBranchName('develop'), baseOid: asCommitOid('base-by-oid'), }, status: 'SUCCESS' as const, timestamps: { createdAt: new Date('2023-01-01T00:00:00Z'), finishedAt: new Date('2023-01-01T00:03:00Z'), }, summary: { totalIntroduced: IssueCount.create(1), totalResolved: IssueCount.create(2), totalSuppressed: IssueCount.create(0), byAnalyzer: [], byCategory: [], }, } as AnalysisRun; // Set up the mock to return the run mockFindByCommit.mockResolvedValue(mockRun); // Call the handler const result = await handleDeepsourceRun({ projectKey: 'test-project', runIdentifier: 'commit-by-oid', isCommitOid: true, }); // Verify findByCommit was called with correct parameters expect(mockFindByCommit).toHaveBeenCalledWith(projectKey, commitOid); // Parse and verify the content const parsedContent = JSON.parse(result.content[0].text); expect(parsedContent.run.commitOid).toBe('commit-by-oid'); expect(parsedContent.run.branchName).toBe('develop'); }); it('should throw error when repository fails', async () => { // Set up the mock to throw an error const testError = new Error('Repository connection failed'); mockFindByRunId.mockRejectedValue(testError); // Call the handler and expect it to throw await expect( handleDeepsourceRun({ projectKey: 'test-project', runIdentifier: 'run-123', isCommitOid: false, }) ).rejects.toThrow('Repository connection failed'); }); it('should throw error when run not found', async () => { // Set up the mock to return null mockFindByRunId.mockResolvedValue(null); // Call the handler and expect it to throw await expect( handleDeepsourceRun({ projectKey: 'test-project', runIdentifier: 'nonexistent-run', isCommitOid: false, }) ).rejects.toThrow('Run with runUid "nonexistent-run" not found'); }); it('should handle non-Error type exceptions', async () => { // Set up the mock to throw a non-Error value mockFindByRunId.mockRejectedValue('Just a string error'); // Call the handler and expect it to throw await expect( handleDeepsourceRun({ projectKey: 'test-project', runIdentifier: 'run-123', isCommitOid: false, }) ).rejects.toThrow('Unknown error'); }); }); });

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/sapientpants/deepsource-mcp-server'

If you have feedback or need assistance with the MCP directory API, please join our Discord server