UNPKG

@mseep/atlas-mcp-server

Version:

A Model Context Protocol (MCP) server for ATLAS, a Neo4j-powered task management system for LLM Agents - implementing a three-tier architecture (Projects, Tasks, Knowledge) to manage complex workflows.

178 lines (177 loc) 8.38 kB
import { randomUUID } from 'crypto'; import neo4j from 'neo4j-driver'; // Import the neo4j driver import { NodeLabels } from './types.js'; // Import NodeLabels import { Neo4jUtils } from './utils.js'; // Import Neo4jUtils /** * Helper functions for the Neo4j service */ /** * Generate a unique ID string * @returns A unique string ID (without hyphens) */ export function generateId() { return randomUUID().replace(/-/g, ''); } /** * Generate a timestamped ID with an optional prefix * @param prefix Optional prefix for the ID * @returns A unique ID with timestamp and random component */ export function generateTimestampedId(prefix) { const timestamp = Date.now().toString(36); const random = Math.random().toString(36).substring(2, 10); return prefix ? `${prefix}_${timestamp}${random}` : `${timestamp}${random}`; } // Removed unused toNeo4jParams function /** * Build a Neo4j update query dynamically based on provided fields * @param nodeLabel Neo4j node label * @param identifier Node identifier in the query (e.g., 'n') * @param updates Updates to apply * @returns Object with setClauses and params */ export function buildUpdateQuery(nodeLabel, // Keep nodeLabel for potential future use or context identifier, updates) { const params = {}; const setClauses = []; // Add update timestamp automatically const now = new Date().toISOString(); params.updatedAt = now; setClauses.push(`${identifier}.updatedAt = $updatedAt`); // Add update clauses for each provided field in the updates object for (const [key, value] of Object.entries(updates)) { // Ensure we don't try to overwrite the id or createdAt if (key !== 'id' && key !== 'createdAt' && value !== undefined) { params[key] = value; setClauses.push(`${identifier}.${key} = $${key}`); } } return { setClauses, params }; } /** * Builds dynamic Cypher queries for listing entities with filtering, sorting, and pagination. * * @param label The primary node label (e.g., NodeLabels.Task, NodeLabels.Knowledge) * @param returnProperties An array of properties or expressions to return for the data query (e.g., ['t.id as id', 'u.name as userName']) * @param filters Filter options based on ListQueryFilterOptions * @param pagination Pagination and sorting options based on ListQueryPaginationOptions * @param nodeAlias Alias for the primary node in the query (default: 'n') * @param additionalMatchClauses Optional string containing additional MATCH or OPTIONAL MATCH clauses (e.g., for relationships like assigned user or domain) * @returns ListQueryResult containing the count query, data query, and parameters */ export function buildListQuery(label, returnProperties, filters, pagination, nodeAlias = 'n', additionalMatchClauses = '') { const params = {}; let conditions = []; // --- Base MATCH Clause --- // projectId is handled directly in the MATCH for Task and Knowledge const projectIdFilter = filters.projectId ? `{projectId: $projectId}` : ''; if (filters.projectId) { params.projectId = filters.projectId; } let baseMatch = `MATCH (${nodeAlias}:${label} ${projectIdFilter})`; // --- Additional MATCH Clauses (Relationships) --- // Add user-provided MATCH/OPTIONAL MATCH clauses const fullMatchClause = `${baseMatch}\n${additionalMatchClauses}`; // --- WHERE Clause Conditions --- // Status filter if (filters.status) { if (Array.isArray(filters.status) && filters.status.length > 0) { params.statusList = filters.status; conditions.push(`${nodeAlias}.status IN $statusList`); } else if (typeof filters.status === 'string') { params.status = filters.status; conditions.push(`${nodeAlias}.status = $status`); } } // Priority filter (assuming it applies to the primary node) if (filters.priority) { if (Array.isArray(filters.priority) && filters.priority.length > 0) { params.priorityList = filters.priority; conditions.push(`${nodeAlias}.priority IN $priorityList`); } else if (typeof filters.priority === 'string') { params.priority = filters.priority; conditions.push(`${nodeAlias}.priority = $priority`); } } // TaskType filter (assuming it applies to the primary node) if (filters.taskType) { params.taskType = filters.taskType; conditions.push(`${nodeAlias}.taskType = $taskType`); } // Tags filter (using helper) if (filters.tags && filters.tags.length > 0) { // Ensure Neo4jUtils is accessible or import it if helpers.ts is separate // Assuming Neo4jUtils is available in scope or imported const tagQuery = Neo4jUtils.generateArrayInListQuery(nodeAlias, 'tags', 'tagsList', filters.tags); if (tagQuery.cypher) { conditions.push(tagQuery.cypher); Object.assign(params, tagQuery.params); } } // Text search filter (Knowledge specific, using regex for now) if (label === NodeLabels.Knowledge && filters.search) { // Use case-insensitive regex params.search = `(?i).*${filters.search.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')}.*`; conditions.push(`${nodeAlias}.text =~ $search`); // TODO: Consider switching to full-text index search for performance: // conditions.push(`apoc.index.search('${NodeLabels.Knowledge}_fulltext', $search) YIELD node as ${nodeAlias}`); // This would require changing the MATCH structure significantly. } // Domain filter is handled via additionalMatchClauses typically const whereClause = conditions.length > 0 ? `WHERE ${conditions.join(' AND ')}` : ''; // --- Sorting --- const sortField = pagination.sortBy || 'createdAt'; // Default sort field const sortDirection = pagination.sortDirection || 'desc'; // Default sort direction const orderByClause = `ORDER BY ${nodeAlias}.${sortField} ${sortDirection.toUpperCase()}`; // --- Pagination --- const page = Math.max(pagination.page || 1, 1); const limit = Math.min(Math.max(pagination.limit || 20, 1), 100); const skip = (page - 1) * limit; // Use neo4j.int() to ensure skip and limit are treated as integers params.skip = neo4j.int(skip); params.limit = neo4j.int(limit); const paginationClause = `SKIP $skip LIMIT $limit`; // --- Count Query --- const countQuery = ` ${fullMatchClause} ${whereClause} RETURN count(DISTINCT ${nodeAlias}) as total `; // --- Data Query --- // Use WITH clause to pass distinct nodes after filtering before collecting relationships // This is crucial if additionalMatchClauses involve OPTIONAL MATCH that could multiply rows const dataQuery = ` ${fullMatchClause} ${whereClause} WITH DISTINCT ${nodeAlias} ${additionalMatchClauses ? ', ' + additionalMatchClauses.split(' ')[1] : ''} // Pass distinct primary node and potentially relationship aliases ${orderByClause} // Order before skip/limit ${paginationClause} // Re-apply OPTIONAL MATCHes if needed after pagination to get related data for the paginated set ${additionalMatchClauses} // Re-apply OPTIONAL MATCH here if needed for RETURN RETURN ${returnProperties.join(',\n ')} `; // Refined Data Query structure (alternative): Apply OPTIONAL MATCH *after* pagination // This can be more efficient if relationship data is only needed for the final page results. const dataQueryAlternative = ` ${baseMatch} // Only match the primary node initially ${whereClause} // Apply filters on the primary node WITH ${nodeAlias} ${orderByClause} ${paginationClause} // Now apply OPTIONAL MATCHes for related data for the paginated nodes ${additionalMatchClauses} RETURN ${returnProperties.join(',\n ')} `; // Choosing dataQueryAlternative as it's generally more performant for pagination // Remove skip/limit from count params const countParams = { ...params }; delete countParams.skip; delete countParams.limit; return { countQuery: countQuery, dataQuery: dataQueryAlternative, // Use the alternative query params: params // Return params including skip/limit for the data query }; }