import { describe, it, expect, beforeEach, vi } from 'vitest';
import {
  workflowNodeSchema,
  workflowConnectionSchema,
  workflowSettingsSchema,
  defaultWorkflowSettings,
  validateWorkflowNode,
  validateWorkflowConnections,
  validateWorkflowSettings,
  cleanWorkflowForCreate,
  cleanWorkflowForUpdate,
  validateWorkflowStructure,
  hasWebhookTrigger,
  getWebhookUrl,
  getWorkflowStructureExample,
  getWorkflowFixSuggestions,
} from '../../../src/services/n8n-validation';
import { WorkflowBuilder } from '../../utils/builders/workflow.builder';
import { z } from 'zod';
import { WorkflowNode, WorkflowConnection, Workflow } from '../../../src/types/n8n-api';

function webhookNode(id: string, name: string, type: string, typeVersion = 2): WorkflowNode {
  return { id, name, type, typeVersion, position: [250, 300] as [number, number], parameters: {} };
}

function workflowWithNodes(nodes: WorkflowNode[]): Partial<Workflow> {
  return { name: 'Test', nodes, connections: {} };
}

describe('n8n-validation', () => {
  describe('Zod Schemas', () => {
    describe('workflowNodeSchema', () => {
      it('should validate a complete valid node', () => {
        const validNode = {
          id: 'node-1',
          name: 'Test Node',
          type: 'n8n-nodes-base.set',
          typeVersion: 3,
          position: [100, 200],
          parameters: { key: 'value' },
          credentials: { api: 'cred-id' },
          disabled: false,
          notes: 'Test notes',
          notesInFlow: true,
          continueOnFail: true,
          retryOnFail: true,
          maxTries: 3,
          waitBetweenTries: 1000,
          alwaysOutputData: true,
          executeOnce: false,
        };

        const result = workflowNodeSchema.parse(validNode);
        expect(result).toEqual(validNode);
      });

      it('should validate a minimal valid node', () => {
        const minimalNode = {
          id: 'node-1',
          name: 'Test Node',
          type: 'n8n-nodes-base.set',
          typeVersion: 3,
          position: [100, 200],
          parameters: {},
        };

        const result = workflowNodeSchema.parse(minimalNode);
        expect(result).toEqual(minimalNode);
      });

      it('should reject node with missing required fields', () => {
        const invalidNode = {
          name: 'Test Node',
          type: 'n8n-nodes-base.set',
        };

        expect(() => workflowNodeSchema.parse(invalidNode)).toThrow();
      });

      it('should reject node with invalid position format', () => {
        const invalidNode = {
          id: 'node-1',
          name: 'Test Node',
          type: 'n8n-nodes-base.set',
          typeVersion: 3,
          position: [100], // Should be tuple of 2 numbers
          parameters: {},
        };

        expect(() => workflowNodeSchema.parse(invalidNode)).toThrow();
      });

      it('should reject node with invalid type values', () => {
        const invalidNode = {
          id: 'node-1',
          name: 'Test Node',
          type: 'n8n-nodes-base.set',
          typeVersion: '3', // Should be number
          position: [100, 200],
          parameters: {},
        };

        expect(() => workflowNodeSchema.parse(invalidNode)).toThrow();
      });
    });

    describe('workflowConnectionSchema', () => {
      it('should validate valid connections', () => {
        const validConnections = {
          'node-1': {
            main: [[{ node: 'node-2', type: 'main', index: 0 }]],
          },
          'node-2': {
            main: [
              [
                { node: 'node-3', type: 'main', index: 0 },
                { node: 'node-4', type: 'main', index: 0 },
              ],
            ],
          },
        };

        const result = workflowConnectionSchema.parse(validConnections);
        expect(result).toEqual(validConnections);
      });

      it('should validate empty connections', () => {
        const emptyConnections = {};
        const result = workflowConnectionSchema.parse(emptyConnections);
        expect(result).toEqual(emptyConnections);
      });

      it('should reject invalid connection structure', () => {
        const invalidConnections = {
          'node-1': {
            main: [{ node: 'node-2', type: 'main', index: 0 }], // Should be array of arrays
          },
        };

        expect(() => workflowConnectionSchema.parse(invalidConnections)).toThrow();
      });

      it('should reject connections missing required fields', () => {
        const invalidConnections = {
          'node-1': {
            main: [[{ node: 'node-2' }]], // Missing type and index
          },
        };

        expect(() => workflowConnectionSchema.parse(invalidConnections)).toThrow();
      });
    });

    describe('workflowSettingsSchema', () => {
      it('should validate complete settings', () => {
        const completeSettings = {
          executionOrder: 'v1' as const,
          timezone: 'America/New_York',
          saveDataErrorExecution: 'all' as const,
          saveDataSuccessExecution: 'all' as const,
          saveManualExecutions: true,
          saveExecutionProgress: true,
          executionTimeout: 300,
          errorWorkflow: 'error-handler-workflow',
        };

        const result = workflowSettingsSchema.parse(completeSettings);
        expect(result).toEqual(completeSettings);
      });

      it('should apply defaults for missing fields', () => {
        const minimalSettings = {};
        const result = workflowSettingsSchema.parse(minimalSettings);
        
        expect(result).toEqual({
          executionOrder: 'v1',
          saveDataErrorExecution: 'all',
          saveDataSuccessExecution: 'all',
          saveManualExecutions: true,
          saveExecutionProgress: true,
        });
      });

      it('should reject invalid enum values', () => {
        const invalidSettings = {
          executionOrder: 'v2', // Invalid enum value
        };

        expect(() => workflowSettingsSchema.parse(invalidSettings)).toThrow();
      });
    });
  });

  describe('Validation Functions', () => {
    describe('validateWorkflowNode', () => {
      it('should validate and return a valid node', () => {
        const node = {
          id: 'test-1',
          name: 'Test',
          type: 'n8n-nodes-base.webhook',
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: {},
        };

        const result = validateWorkflowNode(node);
        expect(result).toEqual(node);
      });

      it('should throw for invalid node', () => {
        const invalidNode = { name: 'Test' };
        expect(() => validateWorkflowNode(invalidNode)).toThrow();
      });
    });

    describe('validateWorkflowConnections', () => {
      it('should validate and return valid connections', () => {
        const connections = {
          'Node1': {
            main: [[{ node: 'Node2', type: 'main', index: 0 }]],
          },
        };

        const result = validateWorkflowConnections(connections);
        expect(result).toEqual(connections);
      });

      it('should throw for invalid connections', () => {
        const invalidConnections = {
          'Node1': {
            main: 'invalid', // Should be array
          },
        };

        expect(() => validateWorkflowConnections(invalidConnections)).toThrow();
      });
    });

    describe('validateWorkflowSettings', () => {
      it('should validate and return valid settings', () => {
        const settings = {
          executionOrder: 'v1' as const,
          timezone: 'UTC',
        };

        const result = validateWorkflowSettings(settings);
        expect(result).toMatchObject(settings);
      });

      it('should apply defaults and validate', () => {
        const result = validateWorkflowSettings({});
        expect(result).toMatchObject(defaultWorkflowSettings);
      });
    });
  });

  describe('Workflow Cleaning Functions', () => {
    describe('cleanWorkflowForCreate', () => {
      it('should remove read-only fields', () => {
        const workflow = {
          id: 'should-be-removed',
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          createdAt: '2023-01-01',
          updatedAt: '2023-01-01',
          versionId: 'v123',
          meta: { test: 'data' },
          active: true,
          tags: ['tag1'],
        };

        const cleaned = cleanWorkflowForCreate(workflow as any);
        
        expect(cleaned).not.toHaveProperty('id');
        expect(cleaned).not.toHaveProperty('createdAt');
        expect(cleaned).not.toHaveProperty('updatedAt');
        expect(cleaned).not.toHaveProperty('versionId');
        expect(cleaned).not.toHaveProperty('meta');
        expect(cleaned).not.toHaveProperty('active');
        expect(cleaned).not.toHaveProperty('tags');
        expect(cleaned.name).toBe('Test Workflow');
      });

      it('should add default settings if not present', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
        };

        const cleaned = cleanWorkflowForCreate(workflow as Workflow);
        expect(cleaned.settings).toEqual(defaultWorkflowSettings);
      });

      it('should preserve existing settings', () => {
        const customSettings = {
          executionOrder: 'v0' as const,
          timezone: 'America/New_York',
        };

        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          settings: customSettings,
        };

        const cleaned = cleanWorkflowForCreate(workflow as Workflow);
        expect(cleaned.settings).toEqual(customSettings);
      });

      it('should inject webhookId on webhook nodes missing it', () => {
        const workflow = workflowWithNodes([
          webhookNode('1', 'Webhook', 'n8n-nodes-base.webhook'),
        ]);

        const cleaned = cleanWorkflowForCreate(workflow as Workflow);
        expect(cleaned.nodes![0].webhookId).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-/);
      });

      it('should preserve existing webhookId on webhook nodes', () => {
        const workflow = workflowWithNodes([
          { ...webhookNode('1', 'Webhook', 'n8n-nodes-base.webhook'), webhookId: 'existing-id' },
        ]);

        const cleaned = cleanWorkflowForCreate(workflow as Workflow);
        expect(cleaned.nodes![0].webhookId).toBe('existing-id');
      });

      it('should inject webhookId on formTrigger and chatTrigger nodes', () => {
        const workflow = workflowWithNodes([
          webhookNode('1', 'Form', 'n8n-nodes-base.formTrigger'),
          webhookNode('2', 'Chat', '@n8n/n8n-nodes-langchain.chatTrigger'),
        ]);

        const cleaned = cleanWorkflowForCreate(workflow as Workflow);
        expect(cleaned.nodes![0].webhookId).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-/);
        expect(cleaned.nodes![1].webhookId).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-/);
      });

      it('should not inject webhookId on non-webhook nodes', () => {
        const workflow = workflowWithNodes([
          webhookNode('1', 'Set', 'n8n-nodes-base.set', 3.4),
        ]);

        const cleaned = cleanWorkflowForCreate(workflow as Workflow);
        expect(cleaned.nodes![0].webhookId).toBeUndefined();
      });
    });

    describe('cleanWorkflowForUpdate', () => {
      it('should remove all read-only and computed fields', () => {
        const workflow = {
          id: 'keep-id',
          name: 'Updated Workflow',
          nodes: [],
          connections: {},
          createdAt: '2023-01-01',
          updatedAt: '2023-01-01',
          versionId: 'v123',
          versionCounter: 5, // n8n 1.118.1+ field
          meta: { test: 'data' },
          staticData: { some: 'data' },
          pinData: { pin: 'data' },
          tags: ['tag1'],
          isArchived: false,
          usedCredentials: ['cred1'],
          sharedWithProjects: ['proj1'],
          triggerCount: 5,
          shared: true,
          active: true,
          settings: { executionOrder: 'v1' },
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        
        // Should remove all these fields
        expect(cleaned).not.toHaveProperty('id');
        expect(cleaned).not.toHaveProperty('createdAt');
        expect(cleaned).not.toHaveProperty('updatedAt');
        expect(cleaned).not.toHaveProperty('versionId');
        expect(cleaned).not.toHaveProperty('versionCounter'); // n8n 1.118.1+ compatibility
        expect(cleaned).not.toHaveProperty('meta');
        expect(cleaned).not.toHaveProperty('staticData');
        expect(cleaned).not.toHaveProperty('pinData');
        expect(cleaned).not.toHaveProperty('tags');
        expect(cleaned).not.toHaveProperty('isArchived');
        expect(cleaned).not.toHaveProperty('usedCredentials');
        expect(cleaned).not.toHaveProperty('sharedWithProjects');
        expect(cleaned).not.toHaveProperty('triggerCount');
        expect(cleaned).not.toHaveProperty('shared');
        expect(cleaned).not.toHaveProperty('active');
        
        // Should keep name and filter settings to safe properties
        expect(cleaned.name).toBe('Updated Workflow');
        expect(cleaned.settings).toEqual({ executionOrder: 'v1' });
      });

      it('should exclude versionCounter for n8n 1.118.1+ compatibility', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          versionId: 'v123',
          versionCounter: 5, // n8n 1.118.1 returns this but rejects it in PUT
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);

        expect(cleaned).not.toHaveProperty('versionCounter');
        expect(cleaned).not.toHaveProperty('versionId');
        expect(cleaned.name).toBe('Test Workflow');
      });

      it('should exclude description field for n8n API compatibility (Issue #431)', () => {
        const workflow = {
          name: 'Test Workflow',
          description: 'This is a test workflow description',
          nodes: [],
          connections: {},
          versionId: 'v123',
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);

        expect(cleaned).not.toHaveProperty('description');
        expect(cleaned).not.toHaveProperty('versionId');
        expect(cleaned.name).toBe('Test Workflow');
      });

      it('should provide empty settings when no settings provided (Issue #431)', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        // Empty settings get minimal defaults to avoid API rejection (Issue #431)
        expect(cleaned.settings).toEqual({ executionOrder: 'v1' });
      });

      it('should filter settings to safe properties to prevent API errors (Issue #248 - final fix)', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          settings: {
            executionOrder: 'v1' as const,
            saveDataSuccessExecution: 'none' as const,
            callerPolicy: 'workflowsFromSameOwner' as const, // Whitelisted (n8n 1.119+)
            timeSavedPerExecution: 5, // Whitelisted (n8n 1.119+, PR #21297)
            unknownProperty: 'should be filtered', // Unknown properties ARE filtered
          },
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);

        // All 4 properties from n8n 1.119+ are whitelisted, unknown properties filtered
        expect(cleaned.settings).toEqual({
          executionOrder: 'v1',
          saveDataSuccessExecution: 'none',
          callerPolicy: 'workflowsFromSameOwner',
          timeSavedPerExecution: 5,
        });
        expect(cleaned.settings).not.toHaveProperty('unknownProperty');
      });

      it('should preserve callerPolicy and availableInMCP (n8n 1.121+ settings)', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          settings: {
            executionOrder: 'v1' as const,
            callerPolicy: 'workflowsFromSameOwner' as const, // Now whitelisted
            availableInMCP: true, // New in n8n 1.121
            errorWorkflow: 'N2O2nZy3aUiBRGFN',
          },
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);

        // callerPolicy and availableInMCP now whitelisted (n8n 1.121+)
        expect(cleaned.settings).toEqual({
          executionOrder: 'v1',
          callerPolicy: 'workflowsFromSameOwner',
          availableInMCP: true,
          errorWorkflow: 'N2O2nZy3aUiBRGFN'
        });
      });

      it('should preserve all whitelisted settings properties including callerPolicy (Issue #248 - updated for n8n 1.121)', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          settings: {
            executionOrder: 'v0' as const,
            timezone: 'UTC',
            saveDataErrorExecution: 'all' as const,
            saveDataSuccessExecution: 'none' as const,
            saveManualExecutions: false,
            saveExecutionProgress: false,
            executionTimeout: 300,
            errorWorkflow: 'error-workflow-id',
            callerPolicy: 'workflowsFromAList' as const, // Now whitelisted (n8n 1.121+)
            availableInMCP: false, // New in n8n 1.121
          },
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);

        // All whitelisted properties kept including callerPolicy and availableInMCP
        expect(cleaned.settings).toEqual({
          executionOrder: 'v0',
          timezone: 'UTC',
          saveDataErrorExecution: 'all',
          saveDataSuccessExecution: 'none',
          saveManualExecutions: false,
          saveExecutionProgress: false,
          executionTimeout: 300,
          errorWorkflow: 'error-workflow-id',
          callerPolicy: 'workflowsFromAList',
          availableInMCP: false
        });
      });

      it('should handle workflows without settings gracefully', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        // Empty settings get minimal defaults to avoid API rejection (Issue #431)
        expect(cleaned.settings).toEqual({ executionOrder: 'v1' });
      });

      it('should return minimal defaults when only non-whitelisted properties exist (Issue #431)', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          settings: {
            timeSavedPerExecution: 5, // Whitelisted (n8n 1.119+)
            someOtherProperty: 'value', // Filtered out (unknown)
          },
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        // timeSavedPerExecution is now whitelisted, someOtherProperty is filtered out
        // n8n API now accepts empty or partial settings {} - server preserves existing values
        expect(cleaned.settings).toEqual({ timeSavedPerExecution: 5 });
        expect(cleaned.settings).not.toHaveProperty('someOtherProperty');
      });

      it('should preserve whitelisted settings when mixed with non-whitelisted (Issue #431)', () => {
        const workflow = {
          name: 'Test Workflow',
          nodes: [],
          connections: {},
          settings: {
            executionOrder: 'v1' as const, // Whitelisted
            callerPolicy: 'workflowsFromSameOwner' as const, // Now whitelisted (n8n 1.121+)
            timezone: 'America/New_York', // Whitelisted
            someOtherProperty: 'value', // Filtered out
          },
        } as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        // Should keep only whitelisted properties (callerPolicy now whitelisted)
        expect(cleaned.settings).toEqual({
          executionOrder: 'v1',
          callerPolicy: 'workflowsFromSameOwner',
          timezone: 'America/New_York'
        });
        expect(cleaned.settings).not.toHaveProperty('someOtherProperty');
      });

      it('should inject webhookId on webhook nodes missing it', () => {
        const workflow = workflowWithNodes([
          webhookNode('1', 'Webhook', 'n8n-nodes-base.webhook'),
        ]) as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        expect(cleaned.nodes![0].webhookId).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-/);
      });

      it('should preserve existing webhookId on webhook nodes', () => {
        const workflow = workflowWithNodes([
          { ...webhookNode('1', 'Webhook', 'n8n-nodes-base.webhook'), webhookId: 'existing-id' },
        ]) as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        expect(cleaned.nodes![0].webhookId).toBe('existing-id');
      });

      it('should inject webhookId on formTrigger and chatTrigger nodes', () => {
        const workflow = workflowWithNodes([
          webhookNode('1', 'Form', 'n8n-nodes-base.formTrigger'),
          webhookNode('2', 'Chat', '@n8n/n8n-nodes-langchain.chatTrigger'),
        ]) as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        expect(cleaned.nodes![0].webhookId).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-/);
        expect(cleaned.nodes![1].webhookId).toMatch(/^[0-9a-f]{8}-[0-9a-f]{4}-/);
      });

      it('should not inject webhookId on non-webhook nodes', () => {
        const workflow = workflowWithNodes([
          webhookNode('1', 'Set', 'n8n-nodes-base.set', 3.4),
        ]) as any;

        const cleaned = cleanWorkflowForUpdate(workflow);
        expect(cleaned.nodes![0].webhookId).toBeUndefined();
      });
    });
  });

  describe('validateWorkflowStructure', () => {
    it('should return no errors for valid workflow', () => {
      const workflow = new WorkflowBuilder('Valid Workflow')
        .addWebhookNode({ id: 'webhook-1', name: 'Webhook' })
        .addSlackNode({ id: 'slack-1', name: 'Send Slack' })
        .connect('Webhook', 'Send Slack')
        .build();

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toEqual([]);
    });

    it('should detect missing workflow name', () => {
      const workflow = {
        nodes: [],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toContain('Workflow name is required');
    });

    it('should detect missing nodes', () => {
      const workflow = {
        name: 'Test',
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toContain('Workflow must have at least one node');
    });

    it('should detect empty nodes array', () => {
      const workflow = {
        name: 'Test',
        nodes: [],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toContain('Workflow must have at least one node');
    });

    it('should detect missing connections', () => {
      const workflow = {
        name: 'Test',
        nodes: [{ id: 'node-1', name: 'Node 1', type: 'n8n-nodes-base.set', typeVersion: 1, position: [0, 0] as [number, number], parameters: {} }],
      };

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toContain('Workflow connections are required');
    });

    it('should allow single webhook node workflow', () => {
      const workflow = {
        name: 'Webhook Only',
        nodes: [{
          id: 'webhook-1',
          name: 'Webhook',
          type: 'n8n-nodes-base.webhook',
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toEqual([]);
    });

    it('should reject single non-webhook node workflow', () => {
      const workflow = {
        name: 'Invalid Single Node',
        nodes: [{
          id: 'set-1',
          name: 'Set',
          type: 'n8n-nodes-base.set',
          typeVersion: 3,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors.some(e => e.includes('Single non-webhook node workflow is invalid'))).toBe(true);
    });

    it('should detect empty connections in multi-node workflow', () => {
      const workflow = {
        name: 'Disconnected Nodes',
        nodes: [
          {
            id: 'node-1',
            name: 'Node 1',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'node-2',
            name: 'Node 2',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [550, 300] as [number, number],
            parameters: {},
          },
        ],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors.some(e => e.includes('Multi-node workflow has no connections between nodes'))).toBe(true);
    });

    it('should validate node type format - missing package prefix', () => {
      const workflow = {
        name: 'Invalid Node Type',
        nodes: [{
          id: 'node-1',
          name: 'Node 1',
          type: 'webhook', // Missing package prefix
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain('Invalid node type "webhook" at index 0. Node types must include package prefix (e.g., "n8n-nodes-base.webhook").');
    });

    it('should validate node type format - wrong prefix format', () => {
      const workflow = {
        name: 'Invalid Node Type',
        nodes: [{
          id: 'node-1',
          name: 'Node 1',
          type: 'nodes-base.webhook', // Wrong prefix
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain('Invalid node type "nodes-base.webhook" at index 0. Use "n8n-nodes-base.webhook" instead.');
    });

    it('should detect invalid node structure', () => {
      const workflow = {
        name: 'Invalid Node',
        nodes: [{
          name: 'Missing Required Fields',
          // Missing id, type, typeVersion, position, parameters
        } as any],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow);
      // The validation will fail because the node is missing required fields
      expect(errors.some(e => e.includes('Invalid node at index 0'))).toBe(true);
    });

    it('should detect non-existent connection source by name', () => {
      const workflow = {
        name: 'Bad Connection',
        nodes: [{
          id: 'node-1',
          name: 'Node 1',
          type: 'n8n-nodes-base.set',
          typeVersion: 3,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {
          'Non-existent Node': {
            main: [[{ node: 'Node 1', type: 'main', index: 0 }]],
          },
        },
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain('Connection references non-existent node: Non-existent Node');
    });

    it('should detect non-existent connection target by name', () => {
      const workflow = {
        name: 'Bad Connection Target',
        nodes: [{
          id: 'node-1',
          name: 'Node 1',
          type: 'n8n-nodes-base.set',
          typeVersion: 3,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {
          'Node 1': {
            main: [[{ node: 'Non-existent Node', type: 'main', index: 0 }]],
          },
        },
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain('Connection references non-existent target node: Non-existent Node (from Node 1[0][0])');
    });

    it('should detect when node ID is used instead of name in connection source', () => {
      const workflow = {
        name: 'ID Instead of Name',
        nodes: [
          {
            id: 'node-1',
            name: 'First Node',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'node-2',
            name: 'Second Node',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [550, 300] as [number, number],
            parameters: {},
          },
        ],
        connections: {
          'node-1': { // Using ID instead of name
            main: [[{ node: 'Second Node', type: 'main', index: 0 }]],
          },
        },
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain("Connection uses node ID 'node-1' but must use node name 'First Node'. Change connections.node-1 to connections['First Node']");
    });

    it('should detect when node ID is used instead of name in connection target', () => {
      const workflow = {
        name: 'ID Instead of Name in Target',
        nodes: [
          {
            id: 'node-1',
            name: 'First Node',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'node-2',
            name: 'Second Node',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [550, 300] as [number, number],
            parameters: {},
          },
        ],
        connections: {
          'First Node': {
            main: [[{ node: 'node-2', type: 'main', index: 0 }]], // Using ID instead of name
          },
        },
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain("Connection target uses node ID 'node-2' but must use node name 'Second Node' (from First Node[0][0])");
    });

    it('should handle complex multi-output connections', () => {
      const workflow = {
        name: 'Complex Connections',
        nodes: [
          {
            id: 'if-1',
            name: 'IF Node',
            type: 'n8n-nodes-base.if',
            typeVersion: 2,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'true-1',
            name: 'True Branch',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [450, 200] as [number, number],
            parameters: {},
          },
          {
            id: 'false-1',
            name: 'False Branch',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [450, 400] as [number, number],
            parameters: {},
          },
        ],
        connections: {
          'IF Node': {
            main: [
              [{ node: 'True Branch', type: 'main', index: 0 }],
              [{ node: 'False Branch', type: 'main', index: 0 }],
            ],
          },
        },
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toEqual([]);
    });

    it('should validate invalid connections structure', () => {
      const workflow = {
        name: 'Invalid Connections',
        nodes: [
          {
            id: 'node-1',
            name: 'Node 1',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'node-2',
            name: 'Node 2',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [550, 300] as [number, number],
            parameters: {},
          }
        ],
        connections: {
          'Node 1': 'invalid', // Should be an object
        } as any,
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors.some(e => e.includes('Invalid connections'))).toBe(true);
    });

    // Issue #503: mcpTrigger nodes should not be flagged as disconnected
    describe('AI connection types (Issue #503)', () => {
      it('should NOT flag mcpTrigger as disconnected when it has ai_tool inbound connections', () => {
        const workflow = {
          name: 'MCP Server Workflow',
          nodes: [
            {
              id: 'mcp-server',
              name: 'MCP Server',
              type: '@n8n/n8n-nodes-langchain.mcpTrigger',
              typeVersion: 1,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'tool-1',
              name: 'Get Weather Tool',
              type: '@n8n/n8n-nodes-langchain.toolWorkflow',
              typeVersion: 1.3,
              position: [300, 200] as [number, number],
              parameters: {},
            },
            {
              id: 'tool-2',
              name: 'Search Tool',
              type: '@n8n/n8n-nodes-langchain.toolWorkflow',
              typeVersion: 1.3,
              position: [300, 400] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'Get Weather Tool': {
              ai_tool: [[{ node: 'MCP Server', type: 'ai_tool', index: 0 }]],
            },
            'Search Tool': {
              ai_tool: [[{ node: 'MCP Server', type: 'ai_tool', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via ai_languageModel', () => {
        const workflow = {
          name: 'AI Agent Workflow',
          nodes: [
            {
              id: 'agent-1',
              name: 'AI Agent',
              type: '@n8n/n8n-nodes-langchain.agent',
              typeVersion: 1.6,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'llm-1',
              name: 'OpenAI Model',
              type: '@n8n/n8n-nodes-langchain.lmChatOpenAi',
              typeVersion: 1,
              position: [300, 300] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'OpenAI Model': {
              ai_languageModel: [[{ node: 'AI Agent', type: 'ai_languageModel', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via ai_memory', () => {
        const workflow = {
          name: 'AI Memory Workflow',
          nodes: [
            {
              id: 'agent-1',
              name: 'AI Agent',
              type: '@n8n/n8n-nodes-langchain.agent',
              typeVersion: 1.6,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'memory-1',
              name: 'Buffer Memory',
              type: '@n8n/n8n-nodes-langchain.memoryBufferWindow',
              typeVersion: 1,
              position: [300, 400] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'Buffer Memory': {
              ai_memory: [[{ node: 'AI Agent', type: 'ai_memory', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via ai_embedding', () => {
        const workflow = {
          name: 'Vector Store Workflow',
          nodes: [
            {
              id: 'vs-1',
              name: 'Vector Store',
              type: '@n8n/n8n-nodes-langchain.vectorStorePinecone',
              typeVersion: 1,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'embed-1',
              name: 'OpenAI Embeddings',
              type: '@n8n/n8n-nodes-langchain.embeddingsOpenAi',
              typeVersion: 1,
              position: [300, 300] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'OpenAI Embeddings': {
              ai_embedding: [[{ node: 'Vector Store', type: 'ai_embedding', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via ai_vectorStore', () => {
        const workflow = {
          name: 'Retriever Workflow',
          nodes: [
            {
              id: 'retriever-1',
              name: 'Vector Store Retriever',
              type: '@n8n/n8n-nodes-langchain.retrieverVectorStore',
              typeVersion: 1,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'vs-1',
              name: 'Pinecone Store',
              type: '@n8n/n8n-nodes-langchain.vectorStorePinecone',
              typeVersion: 1,
              position: [300, 300] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'Pinecone Store': {
              ai_vectorStore: [[{ node: 'Vector Store Retriever', type: 'ai_vectorStore', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via error output', () => {
        const workflow = {
          name: 'Error Handling Workflow',
          nodes: [
            {
              id: 'http-1',
              name: 'HTTP Request',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 4.2,
              position: [300, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'set-1',
              name: 'Handle Error',
              type: 'n8n-nodes-base.set',
              typeVersion: 3.4,
              position: [500, 400] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'HTTP Request': {
              error: [[{ node: 'Handle Error', type: 'error', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via ai_outputParser', () => {
        const workflow = {
          name: 'AI Output Parser Workflow',
          nodes: [
            {
              id: 'agent-1',
              name: 'AI Agent',
              type: '@n8n/n8n-nodes-langchain.agent',
              typeVersion: 1.6,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'parser-1',
              name: 'Structured Output Parser',
              type: '@n8n/n8n-nodes-langchain.outputParserStructured',
              typeVersion: 1,
              position: [300, 400] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'Structured Output Parser': {
              ai_outputParser: [[{ node: 'AI Agent', type: 'ai_outputParser', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should NOT flag nodes as disconnected when connected via ai_document or ai_textSplitter', () => {
        const workflow = {
          name: 'Document Processing Workflow',
          nodes: [
            {
              id: 'vs-1',
              name: 'Pinecone Vector Store',
              type: '@n8n/n8n-nodes-langchain.vectorStorePinecone',
              typeVersion: 1,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'doc-1',
              name: 'Default Data Loader',
              type: '@n8n/n8n-nodes-langchain.documentDefaultDataLoader',
              typeVersion: 1,
              position: [300, 400] as [number, number],
              parameters: {},
            },
            {
              id: 'splitter-1',
              name: 'Text Splitter',
              type: '@n8n/n8n-nodes-langchain.textSplitterRecursiveCharacterTextSplitter',
              typeVersion: 1,
              position: [100, 400] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'Default Data Loader': {
              ai_document: [[{ node: 'Pinecone Vector Store', type: 'ai_document', index: 0 }]],
            },
            'Text Splitter': {
              ai_textSplitter: [[{ node: 'Default Data Loader', type: 'ai_textSplitter', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors).toHaveLength(0);
      });

      it('should still flag truly disconnected nodes in AI workflows', () => {
        const workflow = {
          name: 'AI Workflow with Disconnected Node',
          nodes: [
            {
              id: 'agent-1',
              name: 'AI Agent',
              type: '@n8n/n8n-nodes-langchain.agent',
              typeVersion: 1.6,
              position: [500, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'llm-1',
              name: 'OpenAI Model',
              type: '@n8n/n8n-nodes-langchain.lmChatOpenAi',
              typeVersion: 1,
              position: [300, 300] as [number, number],
              parameters: {},
            },
            {
              id: 'disconnected-1',
              name: 'Disconnected Set',
              type: 'n8n-nodes-base.set',
              typeVersion: 3.4,
              position: [700, 300] as [number, number],
              parameters: {},
            },
          ],
          connections: {
            'OpenAI Model': {
              ai_languageModel: [[{ node: 'AI Agent', type: 'ai_languageModel', index: 0 }]],
            },
          },
        };

        const errors = validateWorkflowStructure(workflow);
        const disconnectedErrors = errors.filter(e => e.includes('Disconnected'));
        expect(disconnectedErrors.length).toBeGreaterThan(0);
        expect(disconnectedErrors[0]).toContain('Disconnected Set');
      });
    });
  });

  describe('hasWebhookTrigger', () => {
    it('should return true for workflow with webhook node', () => {
      const workflow = new WorkflowBuilder()
        .addWebhookNode()
        .build() as Workflow;

      expect(hasWebhookTrigger(workflow)).toBe(true);
    });

    it('should return true for workflow with webhookTrigger node', () => {
      const workflow = {
        name: 'Test',
        nodes: [{
          id: 'webhook-1',
          name: 'Webhook Trigger',
          type: 'n8n-nodes-base.webhookTrigger',
          typeVersion: 1,
          position: [250, 300] as [number, number],
          parameters: {},
        }],
        connections: {},
      } as Workflow;

      expect(hasWebhookTrigger(workflow)).toBe(true);
    });

    it('should return false for workflow without webhook nodes', () => {
      const workflow = new WorkflowBuilder()
        .addSlackNode()
        .addHttpRequestNode()
        .build() as Workflow;

      expect(hasWebhookTrigger(workflow)).toBe(false);
    });

    it('should return true even if webhook is not the first node', () => {
      const workflow = new WorkflowBuilder()
        .addSlackNode()
        .addWebhookNode()
        .addHttpRequestNode()
        .build() as Workflow;

      expect(hasWebhookTrigger(workflow)).toBe(true);
    });
  });

  describe('getWebhookUrl', () => {
    it('should return webhook path from webhook node', () => {
      const workflow = {
        name: 'Test',
        nodes: [{
          id: 'webhook-1',
          name: 'Webhook',
          type: 'n8n-nodes-base.webhook',
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: {
            path: 'my-custom-webhook',
          },
        }],
        connections: {},
      } as Workflow;

      expect(getWebhookUrl(workflow)).toBe('my-custom-webhook');
    });

    it('should return webhook path from webhookTrigger node', () => {
      const workflow = {
        name: 'Test',
        nodes: [{
          id: 'webhook-1',
          name: 'Webhook Trigger',
          type: 'n8n-nodes-base.webhookTrigger',
          typeVersion: 1,
          position: [250, 300] as [number, number],
          parameters: {
            path: 'trigger-webhook-path',
          },
        }],
        connections: {},
      } as Workflow;

      expect(getWebhookUrl(workflow)).toBe('trigger-webhook-path');
    });

    it('should return null if no webhook node exists', () => {
      const workflow = new WorkflowBuilder()
        .addSlackNode()
        .build() as Workflow;

      expect(getWebhookUrl(workflow)).toBe(null);
    });

    it('should return null if webhook node has no parameters', () => {
      const workflow = {
        name: 'Test',
        nodes: [{
          id: 'webhook-1',
          name: 'Webhook',
          type: 'n8n-nodes-base.webhook',
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: undefined as any,
        }],
        connections: {},
      } as Workflow;

      expect(getWebhookUrl(workflow)).toBe(null);
    });

    it('should return null if webhook node has no path parameter', () => {
      const workflow = {
        name: 'Test',
        nodes: [{
          id: 'webhook-1',
          name: 'Webhook',
          type: 'n8n-nodes-base.webhook',
          typeVersion: 2,
          position: [250, 300] as [number, number],
          parameters: {
            method: 'POST',
            // No path parameter
          },
        }],
        connections: {},
      } as Workflow;

      expect(getWebhookUrl(workflow)).toBe(null);
    });

    it('should return first webhook path when multiple webhooks exist', () => {
      const workflow = {
        name: 'Test',
        nodes: [
          {
            id: 'webhook-1',
            name: 'Webhook 1',
            type: 'n8n-nodes-base.webhook',
            typeVersion: 2,
            position: [250, 300] as [number, number],
            parameters: {
              path: 'first-webhook',
            },
          },
          {
            id: 'webhook-2',
            name: 'Webhook 2',
            type: 'n8n-nodes-base.webhook',
            typeVersion: 2,
            position: [550, 300] as [number, number],
            parameters: {
              path: 'second-webhook',
            },
          },
        ],
        connections: {},
      } as Workflow;

      expect(getWebhookUrl(workflow)).toBe('first-webhook');
    });
  });

  describe('getWorkflowStructureExample', () => {
    it('should return a string containing example workflow structure', () => {
      const example = getWorkflowStructureExample();
      
      expect(example).toContain('Minimal Workflow Example');
      expect(example).toContain('Manual Trigger');
      expect(example).toContain('Set Data');
      expect(example).toContain('connections');
      expect(example).toContain('IMPORTANT: In connections, use the node NAME');
    });

    it('should contain valid JSON structure in example', () => {
      const example = getWorkflowStructureExample();
      // Extract the JSON part between the first { and last }
      const match = example.match(/\{[\s\S]*\}/);
      expect(match).toBeTruthy();
      
      if (match) {
        // Should not throw when parsing
        expect(() => JSON.parse(match[0])).not.toThrow();
      }
    });
  });

  describe('getWorkflowFixSuggestions', () => {
    it('should suggest fixes for empty connections', () => {
      const errors = ['Multi-node workflow has empty connections'];
      const suggestions = getWorkflowFixSuggestions(errors);
      
      expect(suggestions).toContain('Add connections between your nodes. Each node (except endpoints) should connect to another node.');
      expect(suggestions).toContain('Connection format: connections: { "Source Node Name": { "main": [[{ "node": "Target Node Name", "type": "main", "index": 0 }]] } }');
    });

    it('should suggest fixes for single-node workflows', () => {
      const errors = ['Single-node workflows are only valid for webhooks'];
      const suggestions = getWorkflowFixSuggestions(errors);
      
      expect(suggestions).toContain('Add at least one more node to process data. Common patterns: Trigger → Process → Output');
      expect(suggestions).toContain('Examples: Manual Trigger → Set, Webhook → HTTP Request, Schedule Trigger → Database Query');
    });

    it('should suggest fixes for node ID usage instead of names', () => {
      const errors = ["Connection uses node ID 'set-1' but must use node name 'Set Data' instead of node name"];
      const suggestions = getWorkflowFixSuggestions(errors);
      
      expect(suggestions.some(s => s.includes('Replace node IDs with node names'))).toBe(true);
      expect(suggestions.some(s => s.includes('connections: { "set-1": {...} }'))).toBe(true);
    });

    it('should return empty array for no errors', () => {
      const suggestions = getWorkflowFixSuggestions([]);
      expect(suggestions).toEqual([]);
    });

    it('should handle multiple error types', () => {
      const errors = [
        'Multi-node workflow has empty connections',
        'Single-node workflows are only valid for webhooks',
        "Connection uses node ID instead of node name",
      ];
      const suggestions = getWorkflowFixSuggestions(errors);
      
      expect(suggestions.length).toBeGreaterThan(3);
      expect(suggestions).toContain('Add connections between your nodes. Each node (except endpoints) should connect to another node.');
      expect(suggestions).toContain('Add at least one more node to process data. Common patterns: Trigger → Process → Output');
      expect(suggestions).toContain('Replace node IDs with node names in connections. The name is what appears in the node header.');
    });

    it('should not duplicate suggestions for similar errors', () => {
      const errors = [
        "Connection uses node ID 'id1' instead of node name",
        "Connection uses node ID 'id2' instead of node name",
      ];
      const suggestions = getWorkflowFixSuggestions(errors);
      
      // Should only have 2 suggestions for this error type
      const idSuggestions = suggestions.filter(s => s.includes('Replace node IDs'));
      expect(idSuggestions.length).toBe(1);
    });
  });

  describe('Edge Cases and Error Conditions', () => {
    it('should handle workflow with null values gracefully', () => {
      const workflow = {
        name: 'Test',
        nodes: null as any,
        connections: null as any,
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain('Workflow must have at least one node');
      expect(errors).toContain('Workflow connections are required');
    });

    it('should handle undefined parameters in cleaning functions', () => {
      const workflow = {
        name: undefined as any,
        nodes: undefined as any,
        connections: undefined as any,
      };

      expect(() => cleanWorkflowForCreate(workflow)).not.toThrow();
      expect(() => cleanWorkflowForUpdate(workflow as any)).not.toThrow();
    });

    it('should handle circular references in workflow structure', () => {
      const node1: any = {
        id: 'node-1',
        name: 'Node 1',
        type: 'n8n-nodes-base.set',
        typeVersion: 3,
        position: [250, 300],
        parameters: {},
      };
      
      // Create circular reference
      node1.parameters.circular = node1;

      const workflow = {
        name: 'Circular Ref',
        nodes: [node1],
        connections: {},
      };

      // Should handle circular references without crashing
      expect(() => validateWorkflowStructure(workflow)).not.toThrow();
    });

    it('should validate very large position values', () => {
      const node = {
        id: 'node-1',
        name: 'Test Node',
        type: 'n8n-nodes-base.set',
        typeVersion: 3,
        position: [Number.MAX_SAFE_INTEGER, Number.MAX_SAFE_INTEGER] as [number, number],
        parameters: {},
      };

      expect(() => validateWorkflowNode(node)).not.toThrow();
    });

    it('should handle special characters in node names', () => {
      const workflow = {
        name: 'Special Chars',
        nodes: [
          {
            id: 'node-1',
            name: 'Node with "quotes" & special <chars>',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'node-2',
            name: 'Normal Node',
            type: 'n8n-nodes-base.set',
            typeVersion: 3,
            position: [550, 300] as [number, number],
            parameters: {},
          },
        ],
        connections: {
          'Node with "quotes" & special <chars>': {
            main: [[{ node: 'Normal Node', type: 'main', index: 0 }]],
          },
        },
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toEqual([]);
    });

    it('should handle empty string values', () => {
      const workflow = {
        name: '',
        nodes: [{
          id: '',
          name: '',
          type: '',
          typeVersion: 1,
          position: [0, 0] as [number, number],
          parameters: {},
        }],
        connections: {},
      };

      const errors = validateWorkflowStructure(workflow);
      expect(errors).toContain('Workflow name is required');
      // Empty string for type will be caught as invalid
      expect(errors.some(e => e.includes('Invalid node at index 0') || e.includes('Node types must include package prefix'))).toBe(true);
    });

    it('should handle negative position values', () => {
      const node = {
        id: 'node-1',
        name: 'Test Node',
        type: 'n8n-nodes-base.set',
        typeVersion: 3,
        position: [-100, -200] as [number, number],
        parameters: {},
      };

      // Negative positions are valid
      expect(() => validateWorkflowNode(node)).not.toThrow();
    });

    it('should validate settings with additional unknown properties', () => {
      const settings = {
        executionOrder: 'v1' as const,
        timezone: 'UTC',
        unknownProperty: 'should be allowed',
        anotherUnknown: { nested: 'object' },
      };

      // Zod by default strips unknown properties
      const result = validateWorkflowSettings(settings);
      expect(result).toHaveProperty('executionOrder', 'v1');
      expect(result).toHaveProperty('timezone', 'UTC');
      expect(result).not.toHaveProperty('unknownProperty');
      expect(result).not.toHaveProperty('anotherUnknown');
    });
  });

  describe('Integration Tests', () => {
    it('should validate a complete real-world workflow', () => {
      const workflow = new WorkflowBuilder('Production Workflow')
        .addWebhookNode({ 
          id: 'webhook-1', 
          name: 'Order Webhook',
          parameters: {
            path: 'new-order',
            method: 'POST',
          },
        })
        .addIfNode({
          id: 'if-1',
          name: 'Check Order Value',
          parameters: {
            conditions: {
              options: { caseSensitive: true, leftValue: '', typeValidation: 'strict' },
              conditions: [{
                id: '1',
                leftValue: '={{ $json.orderValue }}',
                rightValue: '100',
                operator: { type: 'number', operation: 'gte' },
              }],
              combinator: 'and',
            },
          },
        })
        .addSlackNode({
          id: 'slack-1',
          name: 'Notify High Value',
          parameters: {
            channel: '#high-value-orders',
            text: 'High value order received: ${{ $json.orderId }}',
          },
        })
        .addHttpRequestNode({
          id: 'http-1',
          name: 'Update Inventory',
          parameters: {
            method: 'POST',
            url: 'https://api.inventory.com/update',
            sendBody: true,
            bodyParametersJson: '={{ $json }}',
          },
        })
        .connect('Order Webhook', 'Check Order Value')
        .connect('Check Order Value', 'Notify High Value', 0) // True output
        .connect('Check Order Value', 'Update Inventory', 1) // False output
        .setSettings({
          executionOrder: 'v1',
          timezone: 'America/New_York',
          saveDataErrorExecution: 'all',
          saveDataSuccessExecution: 'none',
          executionTimeout: 300,
        })
        .build();

      const errors = validateWorkflowStructure(workflow as any);
      expect(errors).toEqual([]);

      // Validate individual components
      workflow.nodes.forEach(node => {
        expect(() => validateWorkflowNode(node)).not.toThrow();
      });
      expect(() => validateWorkflowConnections(workflow.connections)).not.toThrow();
      expect(() => validateWorkflowSettings(workflow.settings!)).not.toThrow();
    });

    it('should clean and validate workflow for API operations', () => {
      const originalWorkflow = {
        id: 'wf-123',
        name: 'API Test Workflow',
        nodes: [
          {
            id: 'manual-1',
            name: 'Manual Trigger',
            type: 'n8n-nodes-base.manualTrigger',
            typeVersion: 1,
            position: [250, 300] as [number, number],
            parameters: {},
          },
          {
            id: 'set-1',
            name: 'Set Data',
            type: 'n8n-nodes-base.set',
            typeVersion: 3.4,
            position: [450, 300] as [number, number],
            parameters: {
              mode: 'manual',
              assignments: {
                assignments: [{
                  id: '1',
                  name: 'testKey',
                  value: 'testValue',
                  type: 'string',
                }],
              },
            },
          }
        ],
        connections: {
          'Manual Trigger': {
            main: [[{
              node: 'Set Data',
              type: 'main',
              index: 0,
            }]],
          },
        },
        createdAt: '2023-01-01T00:00:00Z',
        updatedAt: '2023-01-02T00:00:00Z',
        versionId: 'v123',
        active: true,
        tags: ['test', 'api'],
        meta: { instanceId: 'instance-123' },
      };

      // Test create cleaning
      const forCreate = cleanWorkflowForCreate(originalWorkflow);
      expect(forCreate).not.toHaveProperty('id');
      expect(forCreate).not.toHaveProperty('createdAt');
      expect(forCreate).not.toHaveProperty('updatedAt');
      expect(forCreate).not.toHaveProperty('versionId');
      expect(forCreate).not.toHaveProperty('active');
      expect(forCreate).not.toHaveProperty('tags');
      expect(forCreate).not.toHaveProperty('meta');
      expect(forCreate).toHaveProperty('settings');
      expect(validateWorkflowStructure(forCreate)).toEqual([]);

      // Test update cleaning
      const forUpdate = cleanWorkflowForUpdate(originalWorkflow as any);
      expect(forUpdate).not.toHaveProperty('id');
      expect(forUpdate).not.toHaveProperty('createdAt');
      expect(forUpdate).not.toHaveProperty('updatedAt');
      expect(forUpdate).not.toHaveProperty('versionId');
      expect(forUpdate).not.toHaveProperty('active');
      expect(forUpdate).not.toHaveProperty('tags');
      expect(forUpdate).not.toHaveProperty('meta');
      // Empty settings get minimal defaults to avoid API rejection (Issue #431)
      expect(forUpdate.settings).toEqual({ executionOrder: 'v1' });
      expect(validateWorkflowStructure(forUpdate)).toEqual([]);
    });
  });

  describe('Sticky Notes Bug Fix', () => {
    describe('sticky notes should be excluded from disconnected nodes validation', () => {
      it('should allow workflow with sticky notes and connected functional nodes', () => {
        const workflow: Partial<Workflow> = {
          name: 'Test Workflow',
          nodes: [
            {
              id: '1',
              name: 'Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/test' }
            },
            {
              id: '2',
              name: 'HTTP Request',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 3,
              position: [450, 300],
              parameters: {}
            },
            {
              id: 'sticky1',
              name: 'Documentation Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 100],
              parameters: { content: 'This is a documentation note' }
            }
          ],
          connections: {
            'Webhook': {
              main: [[{ node: 'HTTP Request', type: 'main', index: 0 }]]
            }
          }
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors).toEqual([]);
      });

      it('should handle multiple sticky notes without errors', () => {
        const workflow: Partial<Workflow> = {
          name: 'Documented Workflow',
          nodes: [
            {
              id: '1',
              name: 'Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/test' }
            },
            {
              id: '2',
              name: 'Process',
              type: 'n8n-nodes-base.set',
              typeVersion: 3,
              position: [450, 300],
              parameters: {}
            },
            ...Array.from({ length: 10 }, (_, i) => ({
              id: `sticky${i}`,
              name: `Note ${i}`,
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [100 + i * 50, 100] as [number, number],
              parameters: { content: `Documentation note ${i}` }
            }))
          ],
          connections: {
            'Webhook': {
              main: [[{ node: 'Process', type: 'main', index: 0 }]]
            }
          }
        };

        const errors = validateWorkflowStructure(workflow);
        expect(errors).toEqual([]);
      });

      it('should handle all sticky note type variations', () => {
        const stickyTypes = [
          'n8n-nodes-base.stickyNote',
          'nodes-base.stickyNote',
          '@n8n/n8n-nodes-base.stickyNote'
        ];

        stickyTypes.forEach((stickyType, index) => {
          const workflow: Partial<Workflow> = {
            name: 'Test Workflow',
            nodes: [
              {
                id: '1',
                name: 'Webhook',
                type: 'n8n-nodes-base.webhook',
                typeVersion: 1,
                position: [250, 300],
                parameters: { path: '/test' }
              },
              {
                id: `sticky${index}`,
                name: `Note ${index}`,
                type: stickyType,
                typeVersion: 1,
                position: [250, 100],
                parameters: { content: `Note ${index}` }
              }
            ],
            connections: {}
          };

          const errors = validateWorkflowStructure(workflow);

          expect(errors.every(e => !e.includes(`Note ${index}`))).toBe(true);
        });
      });

      it('should handle complex workflow with multiple sticky notes (real-world scenario)', () => {
        const workflow: Partial<Workflow> = {
          name: 'POST /auth/login',
          nodes: [
            {
              id: 'webhook1',
              name: 'Webhook Trigger',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/auth/login', httpMethod: 'POST' }
            },
            {
              id: 'http1',
              name: 'Authenticate',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 3,
              position: [450, 300],
              parameters: {}
            },
            {
              id: 'respond1',
              name: 'Return Success',
              type: 'n8n-nodes-base.respondToWebhook',
              typeVersion: 1,
              position: [650, 250],
              parameters: {}
            },
            {
              id: 'respond2',
              name: 'Return Error',
              type: 'n8n-nodes-base.respondToWebhook',
              typeVersion: 1,
              position: [650, 350],
              parameters: {}
            },
            {
              id: 'sticky1',
              name: 'Webhook Trigger Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 150],
              parameters: { content: 'Receives login request' }
            },
            {
              id: 'sticky2',
              name: 'Authenticate with Supabase Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [450, 150],
              parameters: { content: 'Validates credentials' }
            },
            {
              id: 'sticky3',
              name: 'Return Tokens Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [650, 150],
              parameters: { content: 'Returns access and refresh tokens' }
            },
            {
              id: 'sticky4',
              name: 'Return Error Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [650, 450],
              parameters: { content: 'Returns error message' }
            }
          ],
          connections: {
            'Webhook Trigger': {
              main: [[{ node: 'Authenticate', type: 'main', index: 0 }]]
            },
            'Authenticate': {
              main: [
                [{ node: 'Return Success', type: 'main', index: 0 }],
                [{ node: 'Return Error', type: 'main', index: 0 }]
              ]
            }
          }
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors).toEqual([]);
      });
    });

    describe('validation should still detect truly disconnected functional nodes', () => {
      it('should detect disconnected HTTP node but ignore sticky note', () => {
        const workflow: Partial<Workflow> = {
          name: 'Test Workflow',
          nodes: [
            {
              id: '1',
              name: 'Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/test' }
            },
            {
              id: '2',
              name: 'Disconnected HTTP',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 3,
              position: [450, 300],
              parameters: {}
            },
            {
              id: 'sticky1',
              name: 'Sticky Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 100],
              parameters: { content: 'Note' }
            }
          ],
          connections: {}
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors.length).toBeGreaterThan(0);
        const disconnectedError = errors.find(e => e.includes('Disconnected'));
        expect(disconnectedError).toBeDefined();
        expect(disconnectedError).toContain('Disconnected HTTP');
        expect(disconnectedError).not.toContain('Sticky Note');
      });

      it('should detect multiple disconnected functional nodes but ignore sticky notes', () => {
        const workflow: Partial<Workflow> = {
          name: 'Test Workflow',
          nodes: [
            {
              id: '1',
              name: 'Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/test' }
            },
            {
              id: '2',
              name: 'Disconnected HTTP',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 3,
              position: [450, 300],
              parameters: {}
            },
            {
              id: '3',
              name: 'Disconnected Set',
              type: 'n8n-nodes-base.set',
              typeVersion: 3,
              position: [650, 300],
              parameters: {}
            },
            {
              id: 'sticky1',
              name: 'Note 1',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 100],
              parameters: { content: 'Note 1' }
            },
            {
              id: 'sticky2',
              name: 'Note 2',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [450, 100],
              parameters: { content: 'Note 2' }
            }
          ],
          connections: {}
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors.length).toBeGreaterThan(0);
        const connectionError = errors.find(e => e.includes('no connections') || e.includes('Disconnected'));
        expect(connectionError).toBeDefined();
        expect(connectionError).not.toContain('Note 1');
        expect(connectionError).not.toContain('Note 2');
      });

      it('should allow sticky notes but still validate functional node connections', () => {
        const workflow: Partial<Workflow> = {
          name: 'Test Workflow',
          nodes: [
            {
              id: '1',
              name: 'Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/test' }
            },
            {
              id: '2',
              name: 'Connected HTTP',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 3,
              position: [450, 300],
              parameters: {}
            },
            {
              id: '3',
              name: 'Disconnected Set',
              type: 'n8n-nodes-base.set',
              typeVersion: 3,
              position: [650, 300],
              parameters: {}
            },
            {
              id: 'sticky1',
              name: 'Sticky Note',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 100],
              parameters: { content: 'Note' }
            }
          ],
          connections: {
            'Webhook': {
              main: [[{ node: 'Connected HTTP', type: 'main', index: 0 }]]
            }
          }
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors.length).toBeGreaterThan(0);
        const disconnectedError = errors.find(e => e.includes('Disconnected'));
        expect(disconnectedError).toBeDefined();
        expect(disconnectedError).toContain('Disconnected Set');
        expect(disconnectedError).not.toContain('Connected HTTP');
        expect(disconnectedError).not.toContain('Sticky Note');
      });
    });

    describe('regression tests - ensure sticky notes work like in n8n UI', () => {
      it('single webhook with sticky notes should be valid (matches n8n UI behavior)', () => {
        const workflow: Partial<Workflow> = {
          name: 'Webhook Only with Notes',
          nodes: [
            {
              id: '1',
              name: 'Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/test' }
            },
            {
              id: 'sticky1',
              name: 'Usage Instructions',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 100],
              parameters: { content: 'Call this webhook to trigger the workflow' }
            }
          ],
          connections: {}
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors).toEqual([]);
      });

      it('workflow with only sticky notes should be invalid (no executable nodes)', () => {
        const workflow: Partial<Workflow> = {
          name: 'Only Notes',
          nodes: [
            {
              id: 'sticky1',
              name: 'Note 1',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250, 100],
              parameters: { content: 'Note 1' }
            },
            {
              id: 'sticky2',
              name: 'Note 2',
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [450, 100],
              parameters: { content: 'Note 2' }
            }
          ],
          connections: {}
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors.length).toBeGreaterThan(0);
        expect(errors.some(e => e.includes('at least one executable node'))).toBe(true);
      });

      it('complex production workflow structure should validate correctly', () => {
        const workflow: Partial<Workflow> = {
          name: 'Production API Endpoint',
          nodes: [
            {
              id: 'webhook1',
              name: 'API Webhook',
              type: 'n8n-nodes-base.webhook',
              typeVersion: 1,
              position: [250, 300],
              parameters: { path: '/api/endpoint' }
            },
            {
              id: 'validate1',
              name: 'Validate Input',
              type: 'n8n-nodes-base.code',
              typeVersion: 2,
              position: [450, 300],
              parameters: {}
            },
            {
              id: 'branch1',
              name: 'Check Valid',
              type: 'n8n-nodes-base.if',
              typeVersion: 2,
              position: [650, 300],
              parameters: {}
            },
            {
              id: 'process1',
              name: 'Process Request',
              type: 'n8n-nodes-base.httpRequest',
              typeVersion: 3,
              position: [850, 250],
              parameters: {}
            },
            {
              id: 'success1',
              name: 'Return Success',
              type: 'n8n-nodes-base.respondToWebhook',
              typeVersion: 1,
              position: [1050, 250],
              parameters: {}
            },
            {
              id: 'error1',
              name: 'Return Error',
              type: 'n8n-nodes-base.respondToWebhook',
              typeVersion: 1,
              position: [850, 350],
              parameters: {}
            },
            ...Array.from({ length: 11 }, (_, i) => ({
              id: `sticky${i}`,
              name: `Documentation ${i}`,
              type: 'n8n-nodes-base.stickyNote',
              typeVersion: 1,
              position: [250 + i * 100, 100] as [number, number],
              parameters: { content: `Documentation section ${i}` }
            }))
          ],
          connections: {
            'API Webhook': {
              main: [[{ node: 'Validate Input', type: 'main', index: 0 }]]
            },
            'Validate Input': {
              main: [[{ node: 'Check Valid', type: 'main', index: 0 }]]
            },
            'Check Valid': {
              main: [
                [{ node: 'Process Request', type: 'main', index: 0 }],
                [{ node: 'Return Error', type: 'main', index: 0 }]
              ]
            },
            'Process Request': {
              main: [[{ node: 'Return Success', type: 'main', index: 0 }]]
            }
          }
        };

        const errors = validateWorkflowStructure(workflow);

        expect(errors).toEqual([]);
      });
    });
  });
});