* fix: Fix critical session cleanup stack overflow bug (v2.19.2)
This commit fixes a critical P0 bug that caused stack overflow during
container restart, making the service unusable for all users with
session persistence enabled.
Root Causes:
1. Missing await in cleanupExpiredSessions() line 206 caused
overlapping async cleanup attempts
2. Transport event handlers (onclose, onerror) triggered recursive
cleanup during shutdown
3. No recursion guard to prevent concurrent cleanup of same session
Fixes Applied:
- Added cleanupInProgress Set recursion guard
- Added isShuttingDown flag to prevent recursive event handlers
- Implemented safeCloseTransport() with timeout protection (3s)
- Updated removeSession() with recursion guard and safe close
- Fixed cleanupExpiredSessions() to properly await with error isolation
- Updated all transport event handlers to check shutdown flag
- Enhanced shutdown() method for proper sequential cleanup
Impact:
- Service now survives container restarts without stack overflow
- No more hanging requests after restart
- Individual session cleanup failures don't cascade
- All 77 session lifecycle tests passing
Version: 2.19.2
Severity: CRITICAL
Priority: P0
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
* chore: Bump package.runtime.json to v2.19.2
* test: Fix transport cleanup test to work with safeCloseTransport
The test was manually triggering mockTransport.onclose() to simulate
cleanup, but our stack overflow fix sets transport.onclose = undefined
in safeCloseTransport() before closing.
Updated the test to call removeSession() directly instead of manually
triggering the onclose handler. This properly tests the cleanup behavior
with the new recursion-safe approach.
Changes:
- Call removeSession() directly to test cleanup
- Verify transport.close() is called
- Verify onclose and onerror handlers are cleared
- Verify all session data structures are cleaned up
Test Results: All 115 session tests passing ✅🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
---------
Co-authored-by: Claude <noreply@anthropic.com>
Updates session-management-api.test.ts to align with the relaxed
session ID validation policy introduced for MCP proxy compatibility.
Changes:
- Remove short session IDs from invalid test cases (they're now valid)
- Add new test "should accept short session IDs (relaxed for MCP proxy compatibility)"
- Keep testing truly invalid IDs: empty strings, too long (101+), invalid chars
- Add more comprehensive invalid character tests (spaces, special chars)
Valid short session IDs now accepted:
- 'short' (5 chars)
- 'a' (1 char)
- 'only-nineteen-chars' (19 chars)
- '12345' (5 digits)
Invalid session IDs still rejected:
- Empty strings
- Over 100 characters
- Contains invalid characters (spaces, special chars, quotes, slashes)
This maintains security (character whitelist, max length) while
improving MCP proxy compatibility.
Resolves the last failing CI test in PR #312🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Fixes 5 failing CI tests by relaxing session ID validation to accept
any non-empty string with safe characters (alphanumeric, hyphens, underscores).
Changes:
- Remove 20-character minimum length requirement
- Keep maximum 100-character length for DoS protection
- Maintain character whitelist for injection protection
- Update tests to reflect relaxed validation policy
- Fix mock setup for N8NDocumentationMCPServer in tests
Security protections maintained:
- Character whitelist prevents SQL/NoSQL injection and path traversal
- Maximum length limit prevents DoS attacks
- Empty string validation ensures non-empty session IDs
Tests fixed:
✅ DELETE /mcp endpoint now returns 404 (not 400) for non-existent sessions
✅ Session ID validation accepts short IDs like '12345', 'short-id'
✅ Idempotent session creation tests pass with proper mock setup
Related to PR #312 (Complete Session Persistence Implementation)
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Phase 1 - Lazy Session Restoration (REQ-1, REQ-2, REQ-8):
- Add onSessionNotFound hook for restoring sessions from external storage
- Implement idempotent session creation to prevent race conditions
- Add session ID validation for security (prevent injection attacks)
- Comprehensive error handling (400/408/500 status codes)
- 13 integration tests covering all scenarios
Phase 2 - Session Management API (REQ-5):
- getActiveSessions(): Get all active session IDs
- getSessionState(sessionId): Get session state for persistence
- getAllSessionStates(): Bulk session state retrieval
- restoreSession(sessionId, context): Manual session restoration
- deleteSession(sessionId): Manual session termination
- 21 unit tests covering all API methods
Benefits:
- Sessions survive container restarts
- Horizontal scaling support (no session stickiness needed)
- Zero-downtime deployments
- 100% backwards compatible
Implementation Details:
- Backend methods in http-server-single-session.ts
- Public API methods in mcp-engine.ts
- SessionState type exported from index.ts
- Synchronous session creation and deletion for reliable testing
- Version updated from 2.18.10 to 2.19.0
Tests: 34 passing (13 integration + 21 unit)
Coverage: Full API coverage with edge cases
Security: Session ID validation prevents SQL/NoSQL injection and path traversal
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
- Updated "should return 400 for empty session ID" test to expect "Mcp-Session-Id header is required"
instead of "Invalid session ID format" (empty strings are treated as missing headers)
- Updated "should return 404 for non-existent session" test to verify any non-empty string format is accepted
- Updated "should accept any non-empty string as session ID" test to comprehensively test all session ID formats
- All 38 session management tests now pass
This aligns with the relaxed session ID validation introduced in PR #309 for multi-tenant support.
The server now accepts any non-empty string as a session ID to support various MCP clients
(UUIDv4, instance-prefixed, mcp-remote, custom formats).
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Updated test expectation to match the new validation that accepts
EITHER range OR columns for Google Sheets append operation. This
fixes the CI test failure.
Test was expecting old message: 'Range is required for append operation'
Now expects: 'Range or columns mapping is required for append operation'
Related to #304 - Google Sheets v4+ resourceMapper validation
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Root cause analysis revealed validator was looking at wrong path for
modes data. n8n stores modes at top level of properties, not nested
in typeOptions.
Changes:
- config-validator.ts: Changed from prop.typeOptions?.resourceLocator?.modes
to prop.modes (lines 273-310)
- property-extractor.ts: Added modes field to normalizeProperties to
capture mode definitions from n8n nodes
- Updated all test cases to match real n8n schema structure with modes
at property top level
- Rebuilt database with modes field
Results:
- 100% coverage: All 70 resourceLocator nodes now have modes defined
- Schema-based validation now ACTIVE (was being skipped before)
- False positive eliminated: Google Sheets "name" mode now validates
- Helpful error messages showing actual allowed modes from schema
Testing:
- All 33 unit tests pass
- Verified with n8n-mcp-tester: valid "name" mode passes, invalid modes
fail with clear error listing allowed options [list, url, id, name]
Fixes#304 (Google Sheets false positive)
Related to #306 (validator improvements)
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Update test expectation to include troubleshooting array in error
response details. This field was added as part of environment-aware
debugging improvements in PR #303.
The handleHealthCheck error response now includes troubleshooting
steps to help users diagnose API connectivity issues.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Added isDocker and cloudPlatform fields to session_start telemetry events to enable measurement of the v2.17.1 user ID stability fix.
Changes:
- Added detectCloudPlatform() method to event-tracker.ts
- Updated trackSessionStart() to include isDocker and cloudPlatform
- Added 16 comprehensive unit tests for environment detection
- Tests for all 8 cloud platforms (Railway, Render, Fly, Heroku, AWS, K8s, GCP, Azure)
- Tests for Docker detection, local env, and combined scenarios
- Version bumped to 2.18.1
- Comprehensive CHANGELOG entry
Impact:
- Enables validation of v2.17.1 boot_id-based user ID stability
- Allows segmentation of metrics by environment
- 100% backward compatible - only adds new fields
- All tests passing, TypeScript compilation successful
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Updated test to reflect critical typeVersion validation fix from v2.17.4.
## Issue
CI test failing: "should skip node repository lookup for langchain nodes"
Expected getNode() NOT to be called for langchain nodes.
## Root Cause
Test was written before v2.17.4 when langchain nodes completely bypassed
validation. In v2.17.4, we fixed critical bug where langchain nodes with
invalid typeVersion (e.g., 99999) passed validation but failed at runtime.
## Fix
Updated test to reflect new correct behavior:
- Langchain nodes SHOULD call getNode() for typeVersion validation
- Prevents invalid typeVersion from bypassing validation
- Parameter validation still skipped (handled by AI validators)
## Changes
1. Renamed test to clarify what it tests
2. Changed expectation: getNode() SHOULD be called
3. Check for no typeVersion errors (AI errors may exist)
4. Added new test for invalid typeVersion detection
## Impact
- Zero breaking changes (only test update)
- Validates v2.17.4 critical bug fix works correctly
- Ensures langchain nodes don't bypass typeVersion validation
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Added comprehensive TypeScript type definitions for n8n node parsing while
maintaining zero compilation errors. Uses pragmatic "70% benefit with 0%
breakage" approach with strategic `any` assertions.
## Type Definitions (src/types/node-types.ts)
- NodeClass union type replaces `any` in method signatures
- Type guards: isVersionedNodeInstance(), isVersionedNodeClass()
- Utility functions for safe node handling
## Parser Updates
- node-parser.ts: All methods use NodeClass (15+ methods)
- simple-parser.ts: Strongly typed method signatures
- property-extractor.ts: Typed extraction methods
- 30+ method signatures improved
## Strategic Pattern
- Strong types in public method signatures (caller type safety)
- Strategic `as any` assertions for internal union type access
- Pattern: const desc = description as any; // Access union properties
## Benefits
- Better IDE support and auto-complete
- Compile-time safety at call sites
- Type-based documentation
- Zero compilation errors
- Bug prevention (would have caught v2.17.4 baseDescription issue)
## Test Updates
- All test files updated with `as any` for mock objects
- Zero compilation errors maintained
## Known Limitations
- ~70% type coverage (signatures typed, internal logic uses assertions)
- Union types (INodeTypeBaseDescription vs INodeTypeDescription) not fully resolved
- Future work: Conditional types or overloads for 100% type safety
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Address code review feedback from PR #285:
1. Fix Failing Test (CRITICAL)
- Updated test from baseDescription.defaultVersion to description.defaultVersion
- Added test to verify baseDescription is correctly ignored (legacy bug)
2. Add Missing Test Coverage (HIGH PRIORITY)
- Test currentVersion priority over description.defaultVersion
- Test currentVersion = 0 edge case (version 0 should be valid)
- All 34 tests now passing
3. Enhanced Documentation
- Added comprehensive JSDoc for extractVersion() explaining priority chain
- Enhanced validation comments explaining why typeVersion must run before langchain skip
- Clarified that parameter validation (not typeVersion) is skipped for langchain nodes
Test Results:
- ✅ 34/34 tests passing
- ✅ Version extraction priority chain validated
- ✅ Edge cases covered (version 0, missing properties)
- ✅ Legacy bug prevention tested
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
This fixes a critical validation gap where AI agents could create invalid
configurations for nodes using resourceLocator properties (primarily AI model
nodes like OpenAI Chat Model v1.2+, Anthropic, Cohere, etc.).
Before this fix, AI agents could incorrectly pass a string value like:
model: "gpt-4o-mini"
Instead of the required object format:
model: { mode: "list", value: "gpt-4o-mini" }
These invalid configs would pass validation but fail at runtime in n8n.
Changes:
- Added resourceLocator type validation in config-validator.ts (lines 237-274)
- Validates value is an object with required 'mode' and 'value' properties
- Provides helpful error messages with exact fix suggestions
- Added 10 comprehensive test cases (100% passing)
- Updated version to 2.17.3
- Added CHANGELOG entry
Affected nodes: OpenAI Chat Model (v1.2+), Anthropic, Cohere, DeepSeek,
Groq, Mistral, OpenRouter, xAI Grok Chat Models, and embeddings nodes.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Fixed failing tests by adding the new getMostRecentTemplateDate method
to the mock repository in template service tests.
Fixes test failures in:
- should handle update mode with existing templates
- should handle update mode with no new templates
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Fixes critical issue where Docker and cloud deployments generated new
anonymous user IDs on every container recreation, causing 100-200x
inflation in unique user counts.
Changes:
- Use host's boot_id for stable identification across container updates
- Auto-detect Docker (IS_DOCKER=true) and 8 cloud platforms
- Defensive fallback chain: boot_id → combined signals → generic ID
- Zero configuration required
Impact:
- Resolves ~1000x/month inflation in stdio mode
- Resolves ~180x/month inflation in HTTP mode (6 releases/day)
- Improves telemetry accuracy: 3,996 apparent users → ~2,400-2,800 actual
Testing:
- 18 new unit tests for boot_id functionality
- 16 new integration tests for Docker/cloud detection
- All 60 telemetry tests passing (100%)
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Updated test "should skip node repository lookup for langchain nodes" to verify that getNode is NOT called for langchain nodes, matching the new behavior where langchain nodes bypass all node repository validation and are handled exclusively by AI-specific validators.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Calculator and Think tools have built-in descriptions in n8n, so toolDescription parameter is optional. Updated unit tests to match actual n8n behavior and integration test expectations.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Fixed multiple TypeScript errors preventing clean build:
- Fixed import paths for ValidationResponse type (5 test files)
- Fixed validateBasicLLMChain function signature (removed extra workflow parameter)
- Enhanced ValidationResponse interface to include missing properties:
- Added code, nodeName fields to errors/warnings
- Added info array for informational messages
- Added suggestions array
- Fixed type assertion in mergeConnections helper
- Fixed implicit any type in chat-trigger-validation test
All tests now compile cleanly with no TypeScript errors.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Root Cause:
- SSRF protection added DNS resolution via dns/promises.lookup()
- n8n-api-client.test.ts did not mock DNS module
- Tests failed with "DNS resolution failed" error in CI
Fix:
- Added vi.mock('dns/promises') before imports
- Imported dns module for type safety
- Implemented DNS mock in beforeEach to simulate real behavior:
- localhost → 127.0.0.1
- IP addresses → returned as-is
- Real hostnames → 8.8.8.8 (public IP)
Test Results:
- All 50 n8n-api-client tests now pass ✅
- Type checking passes ✅
- Matches pattern from ssrf-protection.test.ts
🤖 Generated with Claude Code
Co-Authored-By: Claude <noreply@anthropic.com>
This commit implements HIGH-02 (Rate Limiting) and HIGH-03 (SSRF Protection)
from the security audit, protecting against brute force attacks and
Server-Side Request Forgery.
Security Enhancements:
- Rate limiting: 20 attempts per 15 minutes per IP (configurable)
- SSRF protection: Three security modes (strict/moderate/permissive)
- DNS rebinding prevention
- Cloud metadata blocking in all modes
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
The test expected empty strings to pass validation, but our Issue #275
fix intentionally rejects empty strings to prevent TypeErrors.
Change:
- Updated test from "should pass" to "should reject"
- Now expects error: "String parameters cannot be empty"
- Aligns with Issue #275 fix that eliminated 57.4% of production errors
The old behavior (allowing empty strings) caused TypeErrors in
getNodeTypeAlternatives(). The new behavior (rejecting empty strings)
provides clear error messages and prevents crashes.
Related: Issue #275 - TypeError prevention
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Remove UpdateConnectionOperation completely as planned for v2.16.0.
This is a breaking change - users should use removeConnection + addConnection
or the new rewireConnection operation instead.
Changes:
- Remove UpdateConnectionOperation type definition
- Remove validateUpdateConnection and applyUpdateConnection methods
- Remove updateConnection cases from validation/apply switches
- Remove updateConnection tests (4 tests)
- Remove UpdateConnectionOperation import from tests
All 137 tests passing.
Related: #272 Phase 1
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Found by n8n-mcp-tester agent: IF nodes in n8n store connections as:
IF.main[0] (true branch)
IF.main[1] (false branch)
NOT as IF.true and IF.false
Previous implementation (WRONG):
- branch='true' → sourceOutput='true'
Correct implementation (FIXED):
- branch='true' → sourceIndex=0, sourceOutput='main'
- branch='false' → sourceIndex=1, sourceOutput='main'
Changes:
- resolveSmartParameters(): branch now sets sourceIndex, not sourceOutput
- Type definition comments updated to reflect correct mapping
- All unit tests fixed to expect connections under 'main' with correct indices
- All 141 tests passing with correct behavior
This was caught by integration testing against real n8n API, not by unit tests.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Add intuitive semantic parameters for working with IF and Switch nodes:
- branch='true'|'false' for IF nodes (maps to sourceOutput)
- case=N for Switch nodes (maps to sourceIndex)
- Smart parameters resolve to technical parameters automatically
- Explicit parameters always override smart parameters
Implementation:
- Added branch and case parameters to AddConnectionOperation and RewireConnectionOperation interfaces
- Created resolveSmartParameters() helper method to map semantic to technical parameters
- Updated applyAddConnection() to use smart parameter resolution
- Updated applyRewireConnection() to use smart parameter resolution
- Updated validateRewireConnection() to validate with resolved smart parameters
Tests:
- Added 8 comprehensive tests for smart parameters feature
- All 141 workflow diff engine tests passing
- Coverage: 91.7% overall
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Fixes#270
## Problem
Connection operations (addConnection, removeConnection, etc.) failed when node
names contained special characters like apostrophes, quotes, or backslashes.
Default n8n Manual Trigger node: "When clicking 'Execute workflow'" caused:
- Error: "Source node not found: \"When clicking 'Execute workflow'\""
- Node shown in available nodes list but string matching failed
- Users had to use node IDs as workaround
## Root Cause
The `findNode()` method in WorkflowDiffEngine performed exact string matching
without normalization. When node names contained special characters, escaping
differences between input strings and stored node names caused match failures.
## Solution
### 1. String Normalization (Primary Fix)
Added `normalizeNodeName()` helper method:
- Unescapes single quotes: \' → '
- Unescapes double quotes: \" → "
- Unescapes backslashes: \\ → \
- Normalizes whitespace
Updated `findNode()` to normalize both search string and node names before
comparison, while preserving exact UUID matching for node IDs.
### 2. Improved Error Messages
Enhanced validation error messages to show:
- Node IDs (first 8 characters) for quick reference
- Available nodes with both names and ID prefixes
- Helpful tip about using node IDs for special characters
### 3. Comprehensive Tests
Added 6 new test cases covering:
- Apostrophes (default Manual Trigger scenario)
- Double quotes
- Backslashes
- Mixed special characters
- removeConnection with special chars
- updateNode with special chars
All tests passing: 116/116 in workflow-diff-engine.test.ts
### 4. Documentation
Updated tool documentation to note:
- Special character support since v2.15.6
- Node IDs preferred for best compatibility
## Affected Operations
All 8 operations using findNode() now support special characters:
- addConnection, removeConnection, updateConnection
- removeNode, updateNode, moveNode
- enableNode, disableNode
## Testing
Validated with n8n-mcp-tester agent:
✅ addConnection with apostrophes works
✅ Default Manual Trigger name works
✅ Improved error messages show IDs
✅ Double quotes handled correctly
✅ Node IDs work as alternative
## Impact
- Fixes common user pain point with default n8n node names
- Backward compatible (only makes matching MORE permissive)
- Minimal performance impact (normalization only during validation)
- Centralized fix (one method fixes all 8 operations)
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Fixed type errors caused by changing WorkflowListParams.tags from string[] to string:
1. cleanup-helpers.ts: Changed tags: [tag] to tags: tag (line 221)
2. n8n-api-client.test.ts: Changed tags: ['test'] to tags: 'test,production' (line 384)
3. Added unit tests for handleDeleteWorkflow and handleListWorkflows (100% coverage)
All tests pass, lint clean.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Updated tests to match new settings filtering behavior:
- Settings are now filtered to OpenAPI spec whitelisted properties
- Unsafe properties like callerPolicy are removed
- Safe properties are preserved
- Empty object still used when no settings provided
All 72 tests passing.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Add 11 new test cases to achieve 100% coverage of the SHORT form detection
logic added in the P0 bug fix.
## New Test Cases
1. Detect nodes-base.* SHORT form with proper error
2. Detect nodes-langchain.* SHORT form with proper error
3. Detect multiple SHORT form nodes (3 nodes)
4. Allow FULL form n8n-nodes-base.* without error
5. Allow FULL form @n8n/n8n-nodes-langchain.* without error
6. Detect SHORT form in mixed FULL/SHORT workflow
7. Handle null node type gracefully
8. Handle undefined node type gracefully
9. Handle empty nodes array gracefully
10. Handle undefined nodes array (Zod validation)
11. Verify correct node index in error messages
## Coverage Improvements
Before: 32 tests
After: 43 tests (+11 tests, 34% increase)
## Test Quality
- All tests follow existing mocking patterns
- Clear, descriptive test names
- Comprehensive edge case coverage
- Tests both success and failure paths
- Verifies exact error message content
- Tests telemetry tracking
Addresses Codecov patch coverage requirement.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
## Bug Description
handleCreateWorkflow and handleUpdateFullWorkflow were incorrectly
normalizing node types from FULL form (n8n-nodes-base.webhook) to
SHORT form (nodes-base.webhook) before validation and API calls.
This caused 100% failure rate for workflow creation because:
- n8n API requires FULL form (n8n-nodes-base.*)
- Database stores SHORT form (nodes-base.*)
- NodeTypeNormalizer converts TO SHORT form (for database)
- But was being used BEFORE API calls (incorrect)
## Root Cause
NodeTypeNormalizer was designed for database lookups but was
incorrectly applied to API operations. The method name
`normalizeToFullForm()` is misleading - it actually normalizes
TO SHORT form.
## Changes
1. handlers-n8n-manager.ts:
- Removed NodeTypeNormalizer.normalizeWorkflowNodeTypes() from
handleCreateWorkflow (line 288)
- Removed normalization from handleUpdateFullWorkflow (line 544-557)
- Added proactive SHORT form detection with helpful errors
- Added comments explaining n8n API expects FULL form
2. node-type-normalizer.ts:
- Added prominent WARNING about not using before API calls
- Added examples showing CORRECT vs INCORRECT usage
- Clarified this is FOR DATABASE OPERATIONS ONLY
3. handlers-n8n-manager.test.ts:
- Fixed test to expect FULL form (not SHORT) sent to API
- Removed incorrect expectedNormalizedInput assertion
4. NEW: workflow-creation-node-type-format.test.ts:
- 7 integration tests with real validation (unmocked)
- Tests FULL form acceptance, SHORT form rejection
- Tests real-world workflows (webhook, schedule trigger)
- Regression test to prevent bug reintroduction
## Verification
Before fix:
❌ Manual Trigger → Set: FAILED
❌ Webhook → HTTP Request: FAILED
Failure rate: 100%
After fix:
✅ Manual Trigger → Set: SUCCESS (ID: kTAaDZwdpzj8gqzM)
✅ Webhook → HTTP Request: SUCCESS (ID: aPtQUb54uuHIqX52)
✅ All 39 tests passing (32 unit + 7 integration)
Success rate: 100%
## Impact
- Fixes: Complete blocking bug preventing all workflow creation
- Risk: Zero (removing buggy behavior)
- Breaking: None (external API unchanged)
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Remove references to get_node_for_task tool that was removed in v2.15.0
as part of P0-R3 implementation.
Changes:
- parameter-validation.test.ts: Remove getNodeForTask mock spy
- parameter-validation.test.ts: Remove get_node_for_task from validation test array
- tools.test.ts: Remove get_node_for_task from templates category
Test results:
✅ parameter-validation.test.ts: 52/52 passed
✅ tools.test.ts: 57/57 passed
This completes the removal of get_node_for_task tool across the entire codebase.
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>
Root cause: Tests used in-memory database without populating node data,
causing "Node not found" errors when getNodeEssentials() tried lookups.
Changes:
- Add beforeEach() setup to populate test nodes in both test files
- Insert test nodes with SHORT form node types (nodes-base.xxx)
- Fix error handling test expectations (empty array vs undefined)
- Fix searchNodesLIKE test expectations (object with results array)
- Add comments explaining SHORT form requirement
Database stores node types in SHORT form (nodes-base.webhook), not full
form (n8n-nodes-base.webhook). NodeTypeNormalizer.normalizeToFullForm()
actually normalizes TO short form despite the misleading name.
Test results:
✅ get-node-essentials-examples.test.ts: 16/16 passed
✅ search-nodes-examples.test.ts: 14/14 passed
Files modified:
- tests/unit/mcp/get-node-essentials-examples.test.ts
- tests/unit/mcp/search-nodes-examples.test.ts
🤖 Generated with [Claude Code](https://claude.com/claude-code)
Co-Authored-By: Claude <noreply@anthropic.com>