Skip to main content
Glama

n8n-MCP

by 88-888
node-similarity-service.test.ts6.63 kB
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest'; import { NodeSimilarityService } from '@/services/node-similarity-service'; import { NodeRepository } from '@/database/node-repository'; import type { ParsedNode } from '@/parsers/node-parser'; vi.mock('@/database/node-repository'); describe('NodeSimilarityService', () => { let service: NodeSimilarityService; let mockRepository: NodeRepository; const createMockNode = (type: string, displayName: string, description = ''): any => ({ nodeType: type, displayName, description, version: 1, defaults: {}, inputs: ['main'], outputs: ['main'], properties: [], package: 'n8n-nodes-base', typeVersion: 1 }); beforeEach(() => { vi.clearAllMocks(); mockRepository = new NodeRepository({} as any); service = new NodeSimilarityService(mockRepository); }); afterEach(() => { vi.restoreAllMocks(); }); describe('Cache Management', () => { it('should invalidate cache when requested', () => { service.invalidateCache(); expect(service['nodeCache']).toBeNull(); expect(service['cacheVersion']).toBeGreaterThan(0); }); it('should refresh cache with new data', async () => { const nodes = [ createMockNode('nodes-base.httpRequest', 'HTTP Request'), createMockNode('nodes-base.webhook', 'Webhook') ]; vi.spyOn(mockRepository, 'getAllNodes').mockReturnValue(nodes); await service.refreshCache(); expect(service['nodeCache']).toEqual(nodes); expect(mockRepository.getAllNodes).toHaveBeenCalled(); }); it('should use stale cache on refresh error', async () => { const staleNodes = [createMockNode('nodes-base.slack', 'Slack')]; service['nodeCache'] = staleNodes; service['cacheExpiry'] = Date.now() + 1000; // Set cache as not expired vi.spyOn(mockRepository, 'getAllNodes').mockImplementation(() => { throw new Error('Database error'); }); const nodes = await service['getCachedNodes'](); expect(nodes).toEqual(staleNodes); }); it('should refresh cache when expired', async () => { service['cacheExpiry'] = Date.now() - 1000; // Cache expired const nodes = [createMockNode('nodes-base.httpRequest', 'HTTP Request')]; vi.spyOn(mockRepository, 'getAllNodes').mockReturnValue(nodes); const result = await service['getCachedNodes'](); expect(result).toEqual(nodes); expect(mockRepository.getAllNodes).toHaveBeenCalled(); }); }); describe('Edit Distance Optimization', () => { it('should return 0 for identical strings', () => { const distance = service['getEditDistance']('test', 'test'); expect(distance).toBe(0); }); it('should early terminate for length difference exceeding max', () => { const distance = service['getEditDistance']('a', 'abcdefghijk', 3); expect(distance).toBe(4); // maxDistance + 1 }); it('should calculate correct edit distance within threshold', () => { const distance = service['getEditDistance']('kitten', 'sitting', 10); expect(distance).toBe(3); }); it('should use early termination when min distance exceeds max', () => { const distance = service['getEditDistance']('abc', 'xyz', 2); expect(distance).toBe(3); // Should terminate early and return maxDistance + 1 }); }); describe('Node Suggestions', () => { beforeEach(() => { const nodes = [ createMockNode('nodes-base.httpRequest', 'HTTP Request', 'Make HTTP requests'), createMockNode('nodes-base.webhook', 'Webhook', 'Receive webhooks'), createMockNode('nodes-base.slack', 'Slack', 'Send messages to Slack'), createMockNode('nodes-langchain.openAi', 'OpenAI', 'Use OpenAI models') ]; vi.spyOn(mockRepository, 'getAllNodes').mockReturnValue(nodes); }); it('should find similar nodes for exact match', async () => { const suggestions = await service.findSimilarNodes('httpRequest', 3); expect(suggestions).toHaveLength(1); expect(suggestions[0].nodeType).toBe('nodes-base.httpRequest'); expect(suggestions[0].confidence).toBeGreaterThan(0.5); // Adjusted based on actual implementation }); it('should find nodes for typo queries', async () => { const suggestions = await service.findSimilarNodes('htpRequest', 3); expect(suggestions.length).toBeGreaterThan(0); expect(suggestions[0].nodeType).toBe('nodes-base.httpRequest'); expect(suggestions[0].confidence).toBeGreaterThan(0.4); // Adjusted based on actual implementation }); it('should find nodes for partial matches', async () => { const suggestions = await service.findSimilarNodes('slack', 3); expect(suggestions.length).toBeGreaterThan(0); expect(suggestions[0].nodeType).toBe('nodes-base.slack'); }); it('should return empty array for no matches', async () => { const suggestions = await service.findSimilarNodes('nonexistent', 3); expect(suggestions).toEqual([]); }); it('should respect the limit parameter', async () => { const suggestions = await service.findSimilarNodes('request', 2); expect(suggestions.length).toBeLessThanOrEqual(2); }); it('should provide appropriate confidence levels', async () => { const suggestions = await service.findSimilarNodes('HttpRequest', 3); if (suggestions.length > 0) { expect(suggestions[0].confidence).toBeGreaterThan(0.5); expect(suggestions[0].reason).toBeDefined(); } }); it('should handle package prefix normalization', async () => { // Add a node with the exact type we're searching for const nodes = [ createMockNode('nodes-base.httpRequest', 'HTTP Request', 'Make HTTP requests') ]; vi.spyOn(mockRepository, 'getAllNodes').mockReturnValue(nodes); const suggestions = await service.findSimilarNodes('nodes-base.httpRequest', 3); expect(suggestions.length).toBeGreaterThan(0); expect(suggestions[0].nodeType).toBe('nodes-base.httpRequest'); }); }); describe('Constants Usage', () => { it('should use proper constants for scoring', () => { expect(NodeSimilarityService['SCORING_THRESHOLD']).toBe(50); expect(NodeSimilarityService['TYPO_EDIT_DISTANCE']).toBe(2); expect(NodeSimilarityService['SHORT_SEARCH_LENGTH']).toBe(5); expect(NodeSimilarityService['CACHE_DURATION_MS']).toBe(5 * 60 * 1000); expect(NodeSimilarityService['AUTO_FIX_CONFIDENCE']).toBe(0.9); }); }); });

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/88-888/n8n-mcp'

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