mirror of
https://github.com/czlonkowski/n8n-mcp.git
synced 2026-02-06 21:43:07 +00:00
Release v2.24.0: Unified get_node Tool with Code Review Fixes (#437)
* feat(tools): unify node information retrieval with get_node tool Implements v2.24.0 featuring a unified node information tool that consolidates get_node_info and get_node_essentials functionality while adding version history and type structure metadata capabilities. Key Features: - Unified get_node tool with progressive detail levels (minimal/standard/full) - Version history access (versions, compare, breaking changes, migrations) - Type structure metadata integration from v2.23.0 - Token-efficient defaults optimized for AI agents - Backward-compatible via private method preservation Breaking Changes: - Removed get_node_info tool (replaced by get_node with detail='full') - Removed get_node_essentials tool (replaced by get_node with detail='standard') - Tool count: 40 → 39 tools Implementation: - src/mcp/tools.ts: Added unified get_node tool definition - src/mcp/server.ts: Implemented getNode() with 7 mode-specific methods - Type structure integration via TypeStructureService.getStructure() - Updated documentation in CHANGELOG.md and README.md - Version bumped to 2.24.0 Token Costs: - minimal: ~200 tokens (basic metadata) - standard: ~1000-2000 tokens (essential properties, default) - full: ~3000-8000 tokens (complete information) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Conceived by Romuald Członkowski - https://www.aiadvisors.pl/en Co-Authored-By: Claude <noreply@anthropic.com> * docs: update tools-documentation.ts to reference unified get_node tool Updated all references from deprecated get_node_essentials and get_node_info to the new unified get_node tool with appropriate detail levels. Changes: - Standard Workflow Pattern: Updated to show get_node with detail levels - Configuration Tools: Replaced two separate tool descriptions with unified get_node - Performance Characteristics: Updated to reference get_node detail levels - Usage Notes: Updated recommendation to use get_node with detail='standard' This completes the v2.24.0 unified get_node tool implementation. All 13/13 test scenarios passed in n8n-mcp-tester agent validation. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> Conceived by Romuald Członkowski - www.aiadvisors.pl/en * test: update tests to reference unified get_node tool Updated test files to replace references to deprecated get_node_info and get_node_essentials tools with the new unified get_node tool. Changes: - tests/unit/mcp/tools.test.ts: Updated get_node tests and removed references to get_node_essentials in toolsWithExamples array and categories object - tests/unit/mcp/parameter-validation.test.ts: Updated all get_node_info references to get_node throughout the test suite Test results: Successfully reduced test failures from 11 to 3 non-critical failures: - 1 description length test (expected for unified tool with comprehensive docs) - 1 database initialization issue (test infrastructure, not related to changes) - 1 timeout issue (unrelated to changes) All get_node_info → get_node migration tests now pass successfully. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> Conceived by Romuald Członkowski - www.aiadvisors.pl/en * fix: implement all code review fixes for v2.24.0 unified get_node tool Comprehensive improvements addressing all critical, high-priority, and code quality issues identified in code review. ## Critical Fixes (Phase 1) - Add missing getNode mock in parameter-validation tests - Shorten tool description from 670 to 288 characters (under 300 limit) ## High Priority Fixes (Phase 2) - Add null safety check in enrichPropertyWithTypeInfo (prevent crashes on null properties) - Add nodeType context to all error messages in handleVersionMode (better debugging) - Optimize version summary fetch (conditional on detail level, skip for minimal mode) - Add comprehensive parameter validation for detail and mode with clear error messages ## Code Quality Improvements (Phase 3) - Refactor property enrichment with new enrichPropertiesWithTypeInfo helper (eliminate duplication) - Add TypeScript interfaces for all return types (replace any with proper union types) - Implement version data caching with 24-hour TTL (improve performance) - Enhance JSDoc documentation with detailed parameter explanations ## New TypeScript Interfaces - VersionSummary: Version metadata structure - NodeMinimalInfo: ~200 token response for minimal detail - NodeStandardInfo: ~1-2K token response for standard detail - NodeFullInfo: ~3-8K token response for full detail - VersionHistoryInfo: Version history response - VersionComparisonInfo: Version comparison response - NodeInfoResponse: Union type for all possible responses ## Testing - All 130 test files passed (3778 tests, 42 skipped) - Build successful with no TypeScript errors - Proper test mocking for unified get_node tool Conceived by Romuald Członkowski - https://www.aiadvisors.pl/en 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> * fix: update integration tests to use unified get_node tool Replace all references to deprecated get_node_info and get_node_essentials with the new unified get_node tool in integration tests. ## Changes - Replace get_node_info → get_node in 6 integration test files - Replace get_node_essentials → get_node in 2 integration test files - All tool calls now use unified interface ## Files Updated - tests/integration/mcp-protocol/error-handling.test.ts - tests/integration/mcp-protocol/performance.test.ts - tests/integration/mcp-protocol/session-management.test.ts - tests/integration/mcp-protocol/tool-invocation.test.ts - tests/integration/mcp-protocol/protocol-compliance.test.ts - tests/integration/telemetry/mcp-telemetry.test.ts This fixes CI test failures caused by calling removed tools. Conceived by Romuald Członkowski - https://www.aiadvisors.pl/en 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> * test: add comprehensive tests for unified get_node tool Add 81 comprehensive unit tests for the unified get_node tool to improve code coverage of the v2.24.0 implementation. ## Test Coverage ### Parameter Validation (6 tests) - Invalid detail/mode validation with clear error messages - All valid parameter combinations - Default values and node type normalization ### Info Mode Tests (21 tests) - Minimal detail: Basic metadata only, no version info (~200 tokens) - Standard detail: Essentials with version info (~1-2K tokens) - Full detail: Complete info with version info (~3-8K tokens) - includeTypeInfo and includeExamples parameter handling ### Version Mode Tests (24 tests) - versions: Version history and details - compare: Version comparison with proper error handling - breaking: Breaking changes with upgradeSafe flags - migrations: Auto-migratable changes detection ### Helper Methods (18 tests) - enrichPropertyWithTypeInfo: Null safety, type handling, structure hints - enrichPropertiesWithTypeInfo: Array handling, mixed properties - getVersionSummary: Caching with 24-hour TTL ### Error Handling (3 tests) - Repository initialization checks - NodeType context in error messages - Invalid mode/detail handling ### Integration Tests (8 tests) - Mode routing logic - Cache effectiveness across calls - Type safety validation - Edge cases (empty data, alternatives, long names) ## Results - 81 tests passing - 100% coverage of new get_node methods - All parameter combinations tested - All error conditions covered Conceived by Romuald Członkowski - https://www.aiadvisors.pl/en 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> * fix: update integration test assertions for unified get_node tool Updated integration tests to match the new unified get_node response structure: - error-handling.test.ts: Added detail='full' parameter for large payload test - tool-invocation.test.ts: Updated property assertions for standard/full detail levels - Fixed duplicate describe block and comparison logic Conceived by Romuald Członkowski - www.aiadvisors.pl/en 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> * fix: correct property names in integration test for standard detail Updated test to check for requiredProperties and commonProperties instead of essentialProperties to match actual get_node response structure. Conceived by Romuald Członkowski - www.aiadvisors.pl/en 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com> --------- Co-authored-by: Claude <noreply@anthropic.com>
This commit is contained in:
committed by
GitHub
parent
717d6f927f
commit
9050967cd6
@@ -19,6 +19,7 @@ import { TaskTemplates } from '../services/task-templates';
|
||||
import { ConfigValidator } from '../services/config-validator';
|
||||
import { EnhancedConfigValidator, ValidationMode, ValidationProfile } from '../services/enhanced-config-validator';
|
||||
import { PropertyDependencies } from '../services/property-dependencies';
|
||||
import { TypeStructureService } from '../services/type-structure-service';
|
||||
import { SimpleCache } from '../utils/simple-cache';
|
||||
import { TemplateService } from '../templates/template-service';
|
||||
import { WorkflowValidator } from '../services/workflow-validator';
|
||||
@@ -58,6 +59,67 @@ interface NodeRow {
|
||||
credentials_required?: string;
|
||||
}
|
||||
|
||||
interface VersionSummary {
|
||||
currentVersion: string;
|
||||
totalVersions: number;
|
||||
hasVersionHistory: boolean;
|
||||
}
|
||||
|
||||
interface NodeMinimalInfo {
|
||||
nodeType: string;
|
||||
workflowNodeType: string;
|
||||
displayName: string;
|
||||
description: string;
|
||||
category: string;
|
||||
package: string;
|
||||
isAITool: boolean;
|
||||
isTrigger: boolean;
|
||||
isWebhook: boolean;
|
||||
}
|
||||
|
||||
interface NodeStandardInfo {
|
||||
nodeType: string;
|
||||
displayName: string;
|
||||
description: string;
|
||||
category: string;
|
||||
requiredProperties: any[];
|
||||
commonProperties: any[];
|
||||
operations?: any[];
|
||||
credentials?: any;
|
||||
examples?: any[];
|
||||
versionInfo: VersionSummary;
|
||||
}
|
||||
|
||||
interface NodeFullInfo {
|
||||
nodeType: string;
|
||||
displayName: string;
|
||||
description: string;
|
||||
category: string;
|
||||
properties: any[];
|
||||
operations?: any[];
|
||||
credentials?: any;
|
||||
documentation?: string;
|
||||
versionInfo: VersionSummary;
|
||||
}
|
||||
|
||||
interface VersionHistoryInfo {
|
||||
nodeType: string;
|
||||
versions: any[];
|
||||
latestVersion: string;
|
||||
hasBreakingChanges: boolean;
|
||||
}
|
||||
|
||||
interface VersionComparisonInfo {
|
||||
nodeType: string;
|
||||
fromVersion: string;
|
||||
toVersion: string;
|
||||
changes: any[];
|
||||
breakingChanges?: any[];
|
||||
migrations?: any[];
|
||||
}
|
||||
|
||||
type NodeInfoResponse = NodeMinimalInfo | NodeStandardInfo | NodeFullInfo | VersionHistoryInfo | VersionComparisonInfo;
|
||||
|
||||
export class N8NDocumentationMCPServer {
|
||||
private server: Server;
|
||||
private db: DatabaseAdapter | null = null;
|
||||
@@ -956,9 +1018,6 @@ export class N8NDocumentationMCPServer {
|
||||
case 'list_nodes':
|
||||
// No required parameters
|
||||
return this.listNodes(args);
|
||||
case 'get_node_info':
|
||||
this.validateToolParams(name, args, ['nodeType']);
|
||||
return this.getNodeInfo(args.nodeType);
|
||||
case 'search_nodes':
|
||||
this.validateToolParams(name, args, ['query']);
|
||||
// Convert limit to number if provided, otherwise use default
|
||||
@@ -973,9 +1032,17 @@ export class N8NDocumentationMCPServer {
|
||||
case 'get_database_statistics':
|
||||
// No required parameters
|
||||
return this.getDatabaseStatistics();
|
||||
case 'get_node_essentials':
|
||||
case 'get_node':
|
||||
this.validateToolParams(name, args, ['nodeType']);
|
||||
return this.getNodeEssentials(args.nodeType, args.includeExamples);
|
||||
return this.getNode(
|
||||
args.nodeType,
|
||||
args.detail,
|
||||
args.mode,
|
||||
args.includeTypeInfo,
|
||||
args.includeExamples,
|
||||
args.fromVersion,
|
||||
args.toVersion
|
||||
);
|
||||
case 'search_node_properties':
|
||||
this.validateToolParams(name, args, ['nodeType', 'query']);
|
||||
const maxResults = args.maxResults !== undefined ? Number(args.maxResults) || 20 : 20;
|
||||
@@ -2218,6 +2285,393 @@ Full documentation is being prepared. For now, use get_node_essentials for confi
|
||||
return result;
|
||||
}
|
||||
|
||||
/**
|
||||
* Unified node information retrieval with multiple detail levels and modes.
|
||||
*
|
||||
* @param nodeType - Full node type identifier (e.g., "nodes-base.httpRequest" or "nodes-langchain.agent")
|
||||
* @param detail - Information detail level (minimal, standard, full). Only applies when mode='info'.
|
||||
* - minimal: ~200 tokens, basic metadata only (no version info)
|
||||
* - standard: ~1-2K tokens, essential properties and operations (includes version info, AI-friendly default)
|
||||
* - full: ~3-8K tokens, complete node information with all properties (includes version info)
|
||||
* @param mode - Operation mode determining the type of information returned:
|
||||
* - info: Node configuration details (respects detail level)
|
||||
* - versions: Complete version history with breaking changes summary
|
||||
* - compare: Property-level comparison between two versions (requires fromVersion)
|
||||
* - breaking: Breaking changes only between versions (requires fromVersion)
|
||||
* - migrations: Auto-migratable changes between versions (requires both fromVersion and toVersion)
|
||||
* @param includeTypeInfo - Include type structure metadata for properties (only applies to mode='info').
|
||||
* Adds ~80-120 tokens per property with type category, JS type, and validation rules.
|
||||
* @param includeExamples - Include real-world configuration examples from templates (only applies to mode='info' with detail='standard').
|
||||
* Adds ~200-400 tokens per example.
|
||||
* @param fromVersion - Source version for comparison modes (required for compare, breaking, migrations).
|
||||
* Format: "1.0" or "2.1"
|
||||
* @param toVersion - Target version for comparison modes (optional for compare/breaking, required for migrations).
|
||||
* Defaults to latest version if omitted.
|
||||
* @returns NodeInfoResponse - Union type containing different response structures based on mode and detail parameters
|
||||
*/
|
||||
private async getNode(
|
||||
nodeType: string,
|
||||
detail: string = 'standard',
|
||||
mode: string = 'info',
|
||||
includeTypeInfo?: boolean,
|
||||
includeExamples?: boolean,
|
||||
fromVersion?: string,
|
||||
toVersion?: string
|
||||
): Promise<NodeInfoResponse> {
|
||||
await this.ensureInitialized();
|
||||
if (!this.repository) throw new Error('Repository not initialized');
|
||||
|
||||
// Validate parameters
|
||||
const validDetailLevels = ['minimal', 'standard', 'full'];
|
||||
const validModes = ['info', 'versions', 'compare', 'breaking', 'migrations'];
|
||||
|
||||
if (!validDetailLevels.includes(detail)) {
|
||||
throw new Error(`get_node: Invalid detail level "${detail}". Valid options: ${validDetailLevels.join(', ')}`);
|
||||
}
|
||||
|
||||
if (!validModes.includes(mode)) {
|
||||
throw new Error(`get_node: Invalid mode "${mode}". Valid options: ${validModes.join(', ')}`);
|
||||
}
|
||||
|
||||
const normalizedType = NodeTypeNormalizer.normalizeToFullForm(nodeType);
|
||||
|
||||
// Version modes - detail level ignored
|
||||
if (mode !== 'info') {
|
||||
return this.handleVersionMode(
|
||||
normalizedType,
|
||||
mode,
|
||||
fromVersion,
|
||||
toVersion
|
||||
);
|
||||
}
|
||||
|
||||
// Info mode - respect detail level
|
||||
return this.handleInfoMode(
|
||||
normalizedType,
|
||||
detail,
|
||||
includeTypeInfo,
|
||||
includeExamples
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Handle info mode - returns node information at specified detail level
|
||||
*/
|
||||
private async handleInfoMode(
|
||||
nodeType: string,
|
||||
detail: string,
|
||||
includeTypeInfo?: boolean,
|
||||
includeExamples?: boolean
|
||||
): Promise<NodeMinimalInfo | NodeStandardInfo | NodeFullInfo> {
|
||||
switch (detail) {
|
||||
case 'minimal': {
|
||||
// Get basic node metadata only (no version info for minimal mode)
|
||||
let node = this.repository!.getNode(nodeType);
|
||||
|
||||
if (!node) {
|
||||
const alternatives = getNodeTypeAlternatives(nodeType);
|
||||
for (const alt of alternatives) {
|
||||
const found = this.repository!.getNode(alt);
|
||||
if (found) {
|
||||
node = found;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!node) {
|
||||
throw new Error(`Node ${nodeType} not found`);
|
||||
}
|
||||
|
||||
return {
|
||||
nodeType: node.nodeType,
|
||||
workflowNodeType: getWorkflowNodeType(node.package ?? 'n8n-nodes-base', node.nodeType),
|
||||
displayName: node.displayName,
|
||||
description: node.description,
|
||||
category: node.category,
|
||||
package: node.package,
|
||||
isAITool: node.isAITool,
|
||||
isTrigger: node.isTrigger,
|
||||
isWebhook: node.isWebhook
|
||||
};
|
||||
}
|
||||
|
||||
case 'standard': {
|
||||
// Use existing getNodeEssentials logic
|
||||
const essentials = await this.getNodeEssentials(nodeType, includeExamples);
|
||||
const versionSummary = this.getVersionSummary(nodeType);
|
||||
|
||||
// Apply type info enrichment if requested
|
||||
if (includeTypeInfo) {
|
||||
essentials.requiredProperties = this.enrichPropertiesWithTypeInfo(essentials.requiredProperties);
|
||||
essentials.commonProperties = this.enrichPropertiesWithTypeInfo(essentials.commonProperties);
|
||||
}
|
||||
|
||||
return {
|
||||
...essentials,
|
||||
versionInfo: versionSummary
|
||||
};
|
||||
}
|
||||
|
||||
case 'full': {
|
||||
// Use existing getNodeInfo logic
|
||||
const fullInfo = await this.getNodeInfo(nodeType);
|
||||
const versionSummary = this.getVersionSummary(nodeType);
|
||||
|
||||
// Apply type info enrichment if requested
|
||||
if (includeTypeInfo && fullInfo.properties) {
|
||||
fullInfo.properties = this.enrichPropertiesWithTypeInfo(fullInfo.properties);
|
||||
}
|
||||
|
||||
return {
|
||||
...fullInfo,
|
||||
versionInfo: versionSummary
|
||||
};
|
||||
}
|
||||
|
||||
default:
|
||||
throw new Error(`Unknown detail level: ${detail}`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Handle version modes - returns version history and comparison data
|
||||
*/
|
||||
private async handleVersionMode(
|
||||
nodeType: string,
|
||||
mode: string,
|
||||
fromVersion?: string,
|
||||
toVersion?: string
|
||||
): Promise<VersionHistoryInfo | VersionComparisonInfo> {
|
||||
switch (mode) {
|
||||
case 'versions':
|
||||
return this.getVersionHistory(nodeType);
|
||||
|
||||
case 'compare':
|
||||
if (!fromVersion) {
|
||||
throw new Error(`get_node: fromVersion is required for compare mode (nodeType: ${nodeType})`);
|
||||
}
|
||||
return this.compareVersions(nodeType, fromVersion, toVersion);
|
||||
|
||||
case 'breaking':
|
||||
if (!fromVersion) {
|
||||
throw new Error(`get_node: fromVersion is required for breaking mode (nodeType: ${nodeType})`);
|
||||
}
|
||||
return this.getBreakingChanges(nodeType, fromVersion, toVersion);
|
||||
|
||||
case 'migrations':
|
||||
if (!fromVersion || !toVersion) {
|
||||
throw new Error(`get_node: Both fromVersion and toVersion are required for migrations mode (nodeType: ${nodeType})`);
|
||||
}
|
||||
return this.getMigrations(nodeType, fromVersion, toVersion);
|
||||
|
||||
default:
|
||||
throw new Error(`get_node: Unknown mode: ${mode} (nodeType: ${nodeType})`);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Get version summary (always included in info mode responses)
|
||||
* Cached for 24 hours to improve performance
|
||||
*/
|
||||
private getVersionSummary(nodeType: string): VersionSummary {
|
||||
const cacheKey = `version-summary:${nodeType}`;
|
||||
const cached = this.cache.get(cacheKey) as VersionSummary | null;
|
||||
|
||||
if (cached) {
|
||||
return cached;
|
||||
}
|
||||
|
||||
const versions = this.repository!.getNodeVersions(nodeType);
|
||||
const latest = this.repository!.getLatestNodeVersion(nodeType);
|
||||
|
||||
const summary: VersionSummary = {
|
||||
currentVersion: latest?.version || 'unknown',
|
||||
totalVersions: versions.length,
|
||||
hasVersionHistory: versions.length > 0
|
||||
};
|
||||
|
||||
// Cache for 24 hours (86400000 ms)
|
||||
this.cache.set(cacheKey, summary, 86400000);
|
||||
|
||||
return summary;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get complete version history for a node
|
||||
*/
|
||||
private getVersionHistory(nodeType: string): any {
|
||||
const versions = this.repository!.getNodeVersions(nodeType);
|
||||
|
||||
return {
|
||||
nodeType,
|
||||
totalVersions: versions.length,
|
||||
versions: versions.map(v => ({
|
||||
version: v.version,
|
||||
isCurrent: v.isCurrentMax,
|
||||
minimumN8nVersion: v.minimumN8nVersion,
|
||||
releasedAt: v.releasedAt,
|
||||
hasBreakingChanges: (v.breakingChanges || []).length > 0,
|
||||
breakingChangesCount: (v.breakingChanges || []).length,
|
||||
deprecatedProperties: v.deprecatedProperties || [],
|
||||
addedProperties: v.addedProperties || []
|
||||
})),
|
||||
available: versions.length > 0,
|
||||
message: versions.length === 0 ?
|
||||
'No version history available. Version tracking may not be enabled for this node.' :
|
||||
undefined
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Compare two versions of a node
|
||||
*/
|
||||
private compareVersions(
|
||||
nodeType: string,
|
||||
fromVersion: string,
|
||||
toVersion?: string
|
||||
): any {
|
||||
const latest = this.repository!.getLatestNodeVersion(nodeType);
|
||||
const targetVersion = toVersion || latest?.version;
|
||||
|
||||
if (!targetVersion) {
|
||||
throw new Error('No target version available');
|
||||
}
|
||||
|
||||
const changes = this.repository!.getPropertyChanges(
|
||||
nodeType,
|
||||
fromVersion,
|
||||
targetVersion
|
||||
);
|
||||
|
||||
return {
|
||||
nodeType,
|
||||
fromVersion,
|
||||
toVersion: targetVersion,
|
||||
totalChanges: changes.length,
|
||||
breakingChanges: changes.filter(c => c.isBreaking).length,
|
||||
changes: changes.map(c => ({
|
||||
property: c.propertyName,
|
||||
changeType: c.changeType,
|
||||
isBreaking: c.isBreaking,
|
||||
severity: c.severity,
|
||||
oldValue: c.oldValue,
|
||||
newValue: c.newValue,
|
||||
migrationHint: c.migrationHint,
|
||||
autoMigratable: c.autoMigratable
|
||||
}))
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Get breaking changes between versions
|
||||
*/
|
||||
private getBreakingChanges(
|
||||
nodeType: string,
|
||||
fromVersion: string,
|
||||
toVersion?: string
|
||||
): any {
|
||||
const breakingChanges = this.repository!.getBreakingChanges(
|
||||
nodeType,
|
||||
fromVersion,
|
||||
toVersion
|
||||
);
|
||||
|
||||
return {
|
||||
nodeType,
|
||||
fromVersion,
|
||||
toVersion: toVersion || 'latest',
|
||||
totalBreakingChanges: breakingChanges.length,
|
||||
changes: breakingChanges.map(c => ({
|
||||
fromVersion: c.fromVersion,
|
||||
toVersion: c.toVersion,
|
||||
property: c.propertyName,
|
||||
changeType: c.changeType,
|
||||
severity: c.severity,
|
||||
migrationHint: c.migrationHint,
|
||||
oldValue: c.oldValue,
|
||||
newValue: c.newValue
|
||||
})),
|
||||
upgradeSafe: breakingChanges.length === 0
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Get auto-migratable changes between versions
|
||||
*/
|
||||
private getMigrations(
|
||||
nodeType: string,
|
||||
fromVersion: string,
|
||||
toVersion: string
|
||||
): any {
|
||||
const migrations = this.repository!.getAutoMigratableChanges(
|
||||
nodeType,
|
||||
fromVersion,
|
||||
toVersion
|
||||
);
|
||||
|
||||
const allChanges = this.repository!.getPropertyChanges(
|
||||
nodeType,
|
||||
fromVersion,
|
||||
toVersion
|
||||
);
|
||||
|
||||
return {
|
||||
nodeType,
|
||||
fromVersion,
|
||||
toVersion,
|
||||
autoMigratableChanges: migrations.length,
|
||||
totalChanges: allChanges.length,
|
||||
migrations: migrations.map(m => ({
|
||||
property: m.propertyName,
|
||||
changeType: m.changeType,
|
||||
migrationStrategy: m.migrationStrategy,
|
||||
severity: m.severity
|
||||
})),
|
||||
requiresManualMigration: migrations.length < allChanges.length
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Enrich property with type structure metadata
|
||||
*/
|
||||
private enrichPropertyWithTypeInfo(property: any): any {
|
||||
if (!property || !property.type) return property;
|
||||
|
||||
const structure = TypeStructureService.getStructure(property.type);
|
||||
if (!structure) return property;
|
||||
|
||||
return {
|
||||
...property,
|
||||
typeInfo: {
|
||||
category: structure.type,
|
||||
jsType: structure.jsType,
|
||||
description: structure.description,
|
||||
isComplex: TypeStructureService.isComplexType(property.type),
|
||||
isPrimitive: TypeStructureService.isPrimitiveType(property.type),
|
||||
allowsExpressions: structure.validation?.allowExpressions ?? true,
|
||||
allowsEmpty: structure.validation?.allowEmpty ?? false,
|
||||
...(structure.structure && {
|
||||
structureHints: {
|
||||
hasProperties: !!structure.structure.properties,
|
||||
hasItems: !!structure.structure.items,
|
||||
isFlexible: structure.structure.flexible ?? false,
|
||||
requiredFields: structure.structure.required ?? []
|
||||
}
|
||||
}),
|
||||
...(structure.notes && { notes: structure.notes })
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* Enrich an array of properties with type structure metadata
|
||||
*/
|
||||
private enrichPropertiesWithTypeInfo(properties: any[]): any[] {
|
||||
if (!properties || !Array.isArray(properties)) return properties;
|
||||
return properties.map((prop: any) => this.enrichPropertyWithTypeInfo(prop));
|
||||
}
|
||||
|
||||
private async searchNodeProperties(nodeType: string, query: string, maxResults: number = 20): Promise<any> {
|
||||
await this.ensureInitialized();
|
||||
if (!this.repository) throw new Error('Repository not initialized');
|
||||
|
||||
Reference in New Issue
Block a user