serve-command.test.tsβ’8.95 kB
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import { execSync } from 'child_process';
import fs from 'fs';
import path from 'path';
import os from 'os';
describe('n8n-mcp serve Command', () => {
let tempDir: string;
let mockEntrypointPath: string;
// Clean environment for tests - only include essential variables
const cleanEnv = {
PATH: process.env.PATH,
HOME: process.env.HOME,
NODE_ENV: process.env.NODE_ENV
};
beforeEach(() => {
tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'serve-command-test-'));
mockEntrypointPath = path.join(tempDir, 'mock-entrypoint.sh');
});
afterEach(() => {
if (fs.existsSync(tempDir)) {
fs.rmSync(tempDir, { recursive: true });
}
});
/**
* Create a mock entrypoint script that simulates the behavior
* of the real docker-entrypoint.sh for testing purposes
*/
function createMockEntrypoint(content: string): void {
fs.writeFileSync(mockEntrypointPath, content, { mode: 0o755 });
}
describe('Command transformation', () => {
it('should detect "n8n-mcp serve" and set MCP_MODE=http', () => {
const mockScript = `#!/bin/sh
# Simplified version of the entrypoint logic
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
echo "MCP_MODE=\$MCP_MODE"
echo "Remaining args: \$@"
else
echo "Normal execution"
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(`"${mockEntrypointPath}" n8n-mcp serve`, { encoding: 'utf8', env: cleanEnv });
expect(output).toContain('MCP_MODE=http');
expect(output).toContain('Remaining args:');
});
it('should preserve additional arguments after serve command', () => {
const mockScript = `#!/bin/sh
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
echo "MCP_MODE=\$MCP_MODE"
echo "Args: \$@"
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(
`"${mockEntrypointPath}" n8n-mcp serve --port 8080 --verbose --debug`,
{ encoding: 'utf8', env: cleanEnv }
);
expect(output).toContain('MCP_MODE=http');
expect(output).toContain('Args: --port 8080 --verbose --debug');
});
it('should not affect other commands', () => {
const mockScript = `#!/bin/sh
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
echo "Serve mode activated"
else
echo "Command: \$@"
echo "MCP_MODE=\${MCP_MODE:-not-set}"
fi
`;
createMockEntrypoint(mockScript);
// Test with different command
const output1 = execSync(`"${mockEntrypointPath}" node index.js`, { encoding: 'utf8', env: cleanEnv });
expect(output1).toContain('Command: node index.js');
expect(output1).toContain('MCP_MODE=not-set');
// Test with n8n-mcp but not serve
const output2 = execSync(`"${mockEntrypointPath}" n8n-mcp validate`, { encoding: 'utf8', env: cleanEnv });
expect(output2).toContain('Command: n8n-mcp validate');
expect(output2).not.toContain('Serve mode activated');
});
});
describe('Integration with config loading', () => {
it('should load config before processing serve command', () => {
const configPath = path.join(tempDir, 'config.json');
const config = {
custom_var: 'from-config',
port: 9000
};
fs.writeFileSync(configPath, JSON.stringify(config));
const mockScript = `#!/bin/sh
# Simulate config loading
if [ -f "${configPath}" ]; then
export CUSTOM_VAR='from-config'
export PORT='9000'
fi
# Process serve command
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
echo "MCP_MODE=\$MCP_MODE"
echo "CUSTOM_VAR=\$CUSTOM_VAR"
echo "PORT=\$PORT"
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(`"${mockEntrypointPath}" n8n-mcp serve`, { encoding: 'utf8', env: cleanEnv });
expect(output).toContain('MCP_MODE=http');
expect(output).toContain('CUSTOM_VAR=from-config');
expect(output).toContain('PORT=9000');
});
});
describe('Command line variations', () => {
it('should handle serve command with equals sign notation', () => {
const mockScript = `#!/bin/sh
# Handle both space and equals notation
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
echo "Standard notation worked"
echo "Args: \$@"
elif echo "\$@" | grep -q "n8n-mcp.*serve"; then
echo "Alternative notation detected"
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(`"${mockEntrypointPath}" n8n-mcp serve --port=8080`, { encoding: 'utf8', env: cleanEnv });
expect(output).toContain('Standard notation worked');
expect(output).toContain('Args: --port=8080');
});
it('should handle quoted arguments correctly', () => {
const mockScript = `#!/bin/sh
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
shift 2
echo "Args received:"
for arg in "\$@"; do
echo " - '\$arg'"
done
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(
`"${mockEntrypointPath}" n8n-mcp serve --message "Hello World" --path "/path with spaces"`,
{ encoding: 'utf8', env: cleanEnv }
);
expect(output).toContain("- '--message'");
expect(output).toContain("- 'Hello World'");
expect(output).toContain("- '--path'");
expect(output).toContain("- '/path with spaces'");
});
});
describe('Error handling', () => {
it('should handle serve command with missing AUTH_TOKEN in HTTP mode', () => {
const mockScript = `#!/bin/sh
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
# Check for AUTH_TOKEN (simulate entrypoint validation)
if [ -z "\$AUTH_TOKEN" ] && [ -z "\$AUTH_TOKEN_FILE" ]; then
echo "ERROR: AUTH_TOKEN or AUTH_TOKEN_FILE is required for HTTP mode" >&2
exit 1
fi
fi
`;
createMockEntrypoint(mockScript);
try {
execSync(`"${mockEntrypointPath}" n8n-mcp serve`, { encoding: 'utf8', env: cleanEnv });
expect.fail('Should have thrown an error');
} catch (error: any) {
expect(error.status).toBe(1);
expect(error.stderr.toString()).toContain('AUTH_TOKEN or AUTH_TOKEN_FILE is required');
}
});
it('should succeed with AUTH_TOKEN provided', () => {
const mockScript = `#!/bin/sh
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
# Check for AUTH_TOKEN
if [ -z "\$AUTH_TOKEN" ] && [ -z "\$AUTH_TOKEN_FILE" ]; then
echo "ERROR: AUTH_TOKEN or AUTH_TOKEN_FILE is required for HTTP mode" >&2
exit 1
fi
echo "Server starting with AUTH_TOKEN"
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(
`"${mockEntrypointPath}" n8n-mcp serve`,
{ encoding: 'utf8', env: { ...cleanEnv, AUTH_TOKEN: 'test-token' } }
);
expect(output).toContain('Server starting with AUTH_TOKEN');
});
});
describe('Backwards compatibility', () => {
it('should maintain compatibility with direct HTTP mode setting', () => {
const mockScript = `#!/bin/sh
# Direct MCP_MODE setting should still work
echo "Initial MCP_MODE=\${MCP_MODE:-not-set}"
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
echo "Serve command: MCP_MODE=\$MCP_MODE"
else
echo "Direct mode: MCP_MODE=\${MCP_MODE:-stdio}"
fi
`;
createMockEntrypoint(mockScript);
// Test with explicit MCP_MODE
const output1 = execSync(
`"${mockEntrypointPath}" node index.js`,
{ encoding: 'utf8', env: { ...cleanEnv, MCP_MODE: 'http' } }
);
expect(output1).toContain('Initial MCP_MODE=http');
expect(output1).toContain('Direct mode: MCP_MODE=http');
// Test with serve command
const output2 = execSync(`"${mockEntrypointPath}" n8n-mcp serve`, { encoding: 'utf8', env: cleanEnv });
expect(output2).toContain('Serve command: MCP_MODE=http');
});
});
describe('Command construction', () => {
it('should properly construct the node command after transformation', () => {
const mockScript = `#!/bin/sh
if [ "\$1" = "n8n-mcp" ] && [ "\$2" = "serve" ]; then
export MCP_MODE="http"
shift 2
# Simulate the actual command that would be executed
echo "Would execute: node /app/dist/mcp/index.js \$@"
fi
`;
createMockEntrypoint(mockScript);
const output = execSync(
`"${mockEntrypointPath}" n8n-mcp serve --port 8080 --host 0.0.0.0`,
{ encoding: 'utf8', env: cleanEnv }
);
expect(output).toContain('Would execute: node /app/dist/mcp/index.js --port 8080 --host 0.0.0.0');
});
});
});