Skip to main content
Glama
listViews.test.ts4.53 kB
import { CallToolResult } from '@modelcontextprotocol/sdk/types.js'; import { Server } from '../../server.js'; import invariant from '../../utils/invariant.js'; import { constrainViews, getListViewsTool } from './listViews.js'; import { mockView } from './mockView.js'; const mockViews = { pagination: { pageNumber: 1, pageSize: 10, totalAvailable: 1, }, views: [mockView], }; const mocks = vi.hoisted(() => ({ mockQueryViewsForSiteData: vi.fn(), })); vi.mock('../../restApiInstance.js', () => ({ useRestApi: vi.fn().mockImplementation(async ({ callback }) => callback({ viewsMethods: { queryViewsForSite: mocks.mockQueryViewsForSiteData, }, siteId: 'test-site-id', }), ), })); describe('listViewsTool', () => { beforeEach(() => { vi.clearAllMocks(); }); it('should create a tool instance with correct properties', () => { const listViewsTool = getListViewsTool(new Server()); expect(listViewsTool.name).toBe('list-views'); expect(listViewsTool.description).toContain( 'Retrieves a list of views on a Tableau site including their metadata such as name, owner, and the workbook they are found in.', ); expect(listViewsTool.paramsSchema).toMatchObject({ filter: expect.any(Object) }); }); it('should successfully get views', async () => { mocks.mockQueryViewsForSiteData.mockResolvedValue(mockViews); const result = await getToolResult({ filter: 'name:eq:Overview' }); expect(result.isError).toBe(false); expect(JSON.parse(`${result.content[0].text}`)).toMatchObject(mockViews.views); expect(mocks.mockQueryViewsForSiteData).toHaveBeenCalledWith({ siteId: 'test-site-id', filter: 'name:eq:Overview', includeUsageStatistics: true, pageNumber: undefined, pageSize: undefined, }); }); it('should handle API errors gracefully', async () => { const errorMessage = 'API Error'; mocks.mockQueryViewsForSiteData.mockRejectedValue(new Error(errorMessage)); const result = await getToolResult({ filter: 'name:eq:Overview' }); expect(result.isError).toBe(true); expect(result.content[0].text).toContain(errorMessage); }); describe('constrainViews', () => { it('should return empty result when no views are found', () => { const result = constrainViews({ views: [], boundedContext: { projectIds: null, datasourceIds: null, workbookIds: null }, }); invariant(result.type === 'empty'); expect(result.message).toBe( 'No views were found. Either none exist or you do not have permission to view them.', ); }); it('should return empty results when all views were filtered out by the bounded context', () => { const result = constrainViews({ views: mockViews.views, boundedContext: { projectIds: new Set(['123']), datasourceIds: null, workbookIds: null, }, }); invariant(result.type === 'empty'); expect(result.message).toBe( [ 'The set of allowed views that can be queried is limited by the server configuration.', 'While views were found, they were all filtered out by the server configuration.', ].join(' '), ); }); it('should return success result when no views were filtered out by the bounded context', () => { const result = constrainViews({ views: mockViews.views, boundedContext: { projectIds: null, datasourceIds: null, workbookIds: null, }, }); invariant(result.type === 'success'); expect(result.result).toBe(mockViews.views); }); it('should return success result when some views were filtered out by the bounded context', () => { const result = constrainViews({ views: mockViews.views, boundedContext: { projectIds: new Set([mockViews.views[0].project.id]), datasourceIds: null, workbookIds: new Set([mockViews.views[0].workbook.id]), }, }); invariant(result.type === 'success'); expect(result.result).toEqual([mockViews.views[0]]); }); }); }); async function getToolResult(params: { filter: string }): Promise<CallToolResult> { const listViewsTool = getListViewsTool(new Server()); return await listViewsTool.callback(params, { signal: new AbortController().signal, requestId: 'test-request-id', sendNotification: vi.fn(), sendRequest: vi.fn(), }); }

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/datalabs89/tableau-mcp'

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