Skip to main content
Glama
invoice-list.test.ts12.3 kB
/** * Tests for invoice_list tool */ import { describe, it, expect, beforeEach, vi } from 'vitest'; import { invoiceListTool } from '../../../src/tools/invoice/invoice-list.js'; import { createMockClientWrapper } from '../../mocks/client.js'; import { mockInvoiceListResponse, mockInvoiceEmptyListResponse, } from '../../mocks/responses/invoice.js'; import { mockUnauthorizedError, mockRateLimitError, mockServerError, mockNetworkTimeoutError, } from '../../mocks/errors/freshbooks-errors.js'; // Mock the FreshBooks SDK query builders vi.mock('@freshbooks/api/dist/models/builders/index.js', () => ({ SearchQueryBuilder: class { private filters: any[] = []; equals(field: string, value: any) { this.filters.push({ type: 'equals', field, value }); return this; } like(field: string, value: string) { this.filters.push({ type: 'like', field, value }); return this; } between(field: string, range: { min: string; max: string }) { this.filters.push({ type: 'between', field, range }); return this; } build() { return this.filters; } }, PaginationQueryBuilder: class { private _page: number = 1; private _perPage: number = 30; page(value: number) { this._page = value; return this; } perPage(value: number) { this._perPage = value; return this; } build() { return { page: this._page, perPage: this._perPage }; } }, })); describe('invoice_list tool', () => { let mockClient: ReturnType<typeof createMockClientWrapper>; beforeEach(() => { mockClient = createMockClientWrapper(); vi.clearAllMocks(); }); describe('successful operations', () => { it('should return invoices with default pagination', async () => { const mockResponse = mockInvoiceListResponse(10); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123' }, mockClient as any ); expect(result.invoices).toHaveLength(10); expect(result.pagination.page).toBe(1); expect(result.pagination.perPage).toBe(30); expect(result.pagination.total).toBe(10); }); it('should return invoices with custom pagination', async () => { const mockResponse = mockInvoiceListResponse(5, 2, 5); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', page: 2, perPage: 5 }, mockClient as any ); expect(result.invoices).toHaveLength(5); expect(result.pagination.page).toBe(2); expect(result.pagination.perPage).toBe(5); }); it('should return empty array when no invoices exist', async () => { const mockResponse = mockInvoiceEmptyListResponse(); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123' }, mockClient as any ); expect(result.invoices).toHaveLength(0); expect(result.pagination.total).toBe(0); }); it('should apply customerId filter correctly', async () => { const mockResponse = mockInvoiceListResponse(3); let capturedQueryBuilders: any[] = []; mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn((accountId, queryBuilders) => { capturedQueryBuilders = queryBuilders; return Promise.resolve(mockResponse); }), }, }; return apiCall(client); }); await invoiceListTool.execute( { accountId: 'ABC123', customerId: 12345 }, mockClient as any ); expect(capturedQueryBuilders).toBeDefined(); expect(capturedQueryBuilders.length).toBeGreaterThan(0); }); it('should apply status filter correctly', async () => { const mockResponse = mockInvoiceListResponse(5); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', status: 'sent' }, mockClient as any ); expect(result.invoices).toHaveLength(5); }); it('should apply paymentStatus filter correctly', async () => { const mockResponse = mockInvoiceListResponse(3); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', paymentStatus: 'unpaid' }, mockClient as any ); expect(result.invoices).toHaveLength(3); }); it('should apply date range filters correctly', async () => { const mockResponse = mockInvoiceListResponse(2); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', dateMin: '2024-01-01', dateMax: '2024-12-31' }, mockClient as any ); expect(result.invoices).toHaveLength(2); }); it('should apply updatedSince filter correctly', async () => { const mockResponse = mockInvoiceListResponse(4); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', updatedSince: '2024-01-01T00:00:00Z' }, mockClient as any ); expect(result.invoices).toHaveLength(4); }); it('should apply multiple filters together', async () => { const mockResponse = mockInvoiceListResponse(2); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', customerId: 12345, status: 'sent', paymentStatus: 'unpaid', page: 1, perPage: 10, }, mockClient as any ); expect(result.invoices).toHaveLength(2); }); }); describe('error handling', () => { it('should handle unauthorized error', async () => { mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockUnauthorizedError()), }, }; return apiCall(client); }); await expect( invoiceListTool.execute({ accountId: 'ABC123' }, mockClient as any) ).rejects.toThrow(); }); it('should handle rate limit error', async () => { mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockRateLimitError(60)), }, }; return apiCall(client); }); await expect( invoiceListTool.execute({ accountId: 'ABC123' }, mockClient as any) ).rejects.toThrow(); }); it('should handle server error', async () => { mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockServerError()), }, }; return apiCall(client); }); await expect( invoiceListTool.execute({ accountId: 'ABC123' }, mockClient as any) ).rejects.toThrow(); }); it('should handle network timeout', async () => { mockClient.executeWithRetry.mockRejectedValueOnce(mockNetworkTimeoutError()); await expect( invoiceListTool.execute({ accountId: 'ABC123' }, mockClient as any) ).rejects.toThrow(); }); }); describe('input validation', () => { it('should require accountId', async () => { await expect( invoiceListTool.execute({} as any, mockClient as any) ).rejects.toThrow(); }); it('should reject invalid page number (zero)', async () => { await expect( invoiceListTool.execute( { accountId: 'ABC123', page: 0 }, mockClient as any ) ).rejects.toThrow(); }); it('should reject invalid page number (negative)', async () => { await expect( invoiceListTool.execute( { accountId: 'ABC123', page: -1 }, mockClient as any ) ).rejects.toThrow(); }); it('should reject perPage exceeding maximum', async () => { await expect( invoiceListTool.execute( { accountId: 'ABC123', perPage: 101 }, mockClient as any ) ).rejects.toThrow(); }); it('should reject perPage less than 1', async () => { await expect( invoiceListTool.execute( { accountId: 'ABC123', perPage: 0 }, mockClient as any ) ).rejects.toThrow(); }); }); describe('edge cases', () => { it('should handle maximum pagination values', async () => { const mockResponse = mockInvoiceListResponse(100, 1, 100); mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', perPage: 100 }, mockClient as any ); expect(result.invoices).toHaveLength(100); }); it('should handle invoices with null optional fields', async () => { const mockResponse = mockInvoiceListResponse(1); mockResponse.data.invoices[0].dueDate = null; mockResponse.data.invoices[0].notes = null; mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123' }, mockClient as any ); expect(result.invoices[0].dueDate).toBeNull(); }); it('should handle request beyond last page', async () => { const mockResponse = mockInvoiceListResponse(0, 999, 30); mockResponse.data.invoices = []; mockClient.executeWithRetry.mockImplementation(async (operation, apiCall) => { const client = { invoices: { list: vi.fn().mockResolvedValue(mockResponse), }, }; return apiCall(client); }); const result = await invoiceListTool.execute( { accountId: 'ABC123', page: 999 }, mockClient as any ); expect(result.invoices).toHaveLength(0); }); }); });

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/Good-Samaritan-Software-LLC/freshbooks-mcp'

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