/** * Input Validator Tests * * Tests verify: * - Zod schema validation * - Input sanitization * - Authentication schemas * - Command and path schemas */ import { describe, it, expect } from 'vitest'; import { InputValidator, sanitizeString, sanitizeHtml, sanitizePath, SafeStringSchema, IdentifierSchema, EmailSchema, PasswordSchema, UUIDSchema, HttpsUrlSchema, PortSchema, UserRoleSchema, PermissionSchema, LoginRequestSchema, CreateUserSchema, TaskInputSchema, CommandArgumentSchema, PathSchema, PATTERNS, LIMITS, } from '../../security/input-validator.js'; describe('InputValidator', () => { describe('SafeStringSchema', () => { it('should accept safe strings', () => { expect(() => SafeStringSchema.parse('hello world')).not.toThrow(); }); it('should reject empty strings', () => { expect(() => SafeStringSchema.parse('')).toThrow(); }); it('should reject strings with shell metacharacters', () => { const dangerous = [';', '&&', '||', '|', '`', '$()', '${}', '>', '<']; for (const char of dangerous) { expect(() => SafeStringSchema.parse(`hello${char}world`)).toThrow(); } }); }); describe('IdentifierSchema', () => { it('should accept valid identifiers', () => { expect(() => IdentifierSchema.parse('validId')).not.toThrow(); expect(() => IdentifierSchema.parse('valid-id')).not.toThrow(); expect(() => IdentifierSchema.parse('valid_id')).not.toThrow(); expect(() => IdentifierSchema.parse('validId123')).not.toThrow(); }); it('should reject identifiers starting with number', () => { expect(() => IdentifierSchema.parse('123invalid')).toThrow(); }); it('should reject identifiers with special characters', () => { expect(() => IdentifierSchema.parse('invalid@id')).toThrow(); expect(() => IdentifierSchema.parse('invalid id')).toThrow(); }); it('should reject empty identifiers', () => { expect(() => IdentifierSchema.parse('')).toThrow(); }); }); describe('EmailSchema', () => { it('should accept valid emails', () => { expect(() => EmailSchema.parse('user@example.com')).not.toThrow(); expect(() => EmailSchema.parse('user.name@example.co.uk')).not.toThrow(); }); it('should reject invalid emails', () => { expect(() => EmailSchema.parse('notanemail')).toThrow(); expect(() => EmailSchema.parse('@nodomain.com')).toThrow(); expect(() => EmailSchema.parse('no@')).toThrow(); }); it('should lowercase emails', () => { const result = EmailSchema.parse('USER@EXAMPLE.COM'); expect(result).toBe('user@example.com'); }); it('should reject too long emails', () => { const longEmail = 'a'.repeat(300) + '@example.com'; expect(() => EmailSchema.parse(longEmail)).toThrow(); }); }); describe('PasswordSchema', () => { it('should accept valid passwords', () => { expect(() => PasswordSchema.parse('SecurePass123')).not.toThrow(); }); it('should reject short passwords', () => { expect(() => PasswordSchema.parse('Short1')).toThrow(); }); it('should reject passwords without uppercase', () => { expect(() => PasswordSchema.parse('lowercase123')).toThrow(); }); it('should reject passwords without lowercase', () => { expect(() => PasswordSchema.parse('UPPERCASE123')).toThrow(); }); it('should reject passwords without digits', () => { expect(() => PasswordSchema.parse('NoDigitsHere')).toThrow(); }); }); describe('UUIDSchema', () => { it('should accept valid UUIDs', () => { expect(() => UUIDSchema.parse('550e8400-e29b-41d4-a716-446655440000')).not.toThrow(); }); it('should reject invalid UUIDs', () => { expect(() => UUIDSchema.parse('not-a-uuid')).toThrow(); expect(() => UUIDSchema.parse('550e8400-e29b-41d4-a716')).toThrow(); }); }); describe('HttpsUrlSchema', () => { it('should accept HTTPS URLs', () => { expect(() => HttpsUrlSchema.parse('https://example.com')).not.toThrow(); expect(() => HttpsUrlSchema.parse('https://example.com/path')).not.toThrow(); }); it('should reject HTTP URLs', () => { expect(() => HttpsUrlSchema.parse('http://example.com')).toThrow(); }); it('should reject invalid URLs', () => { expect(() => HttpsUrlSchema.parse('not-a-url')).toThrow(); }); }); describe('PortSchema', () => { it('should accept valid ports', () => { expect(() => PortSchema.parse(80)).not.toThrow(); expect(() => PortSchema.parse(443)).not.toThrow(); expect(() => PortSchema.parse(3000)).not.toThrow(); expect(() => PortSchema.parse(65535)).not.toThrow(); }); it('should reject invalid ports', () => { expect(() => PortSchema.parse(0)).toThrow(); expect(() => PortSchema.parse(-1)).toThrow(); expect(() => PortSchema.parse(65536)).toThrow(); expect(() => PortSchema.parse(3.14)).toThrow(); }); }); describe('UserRoleSchema', () => { it('should accept valid roles', () => { expect(() => UserRoleSchema.parse('admin')).not.toThrow(); expect(() => UserRoleSchema.parse('operator')).not.toThrow(); expect(() => UserRoleSchema.parse('developer')).not.toThrow(); expect(() => UserRoleSchema.parse('viewer')).not.toThrow(); expect(() => UserRoleSchema.parse('service')).not.toThrow(); }); it('should reject invalid roles', () => { expect(() => UserRoleSchema.parse('superuser')).toThrow(); expect(() => UserRoleSchema.parse('root')).toThrow(); }); }); describe('PermissionSchema', () => { it('should accept valid permissions', () => { expect(() => PermissionSchema.parse('swarm.create')).not.toThrow(); expect(() => PermissionSchema.parse('agent.spawn')).not.toThrow(); expect(() => PermissionSchema.parse('system.admin')).not.toThrow(); }); it('should reject invalid permissions', () => { expect(() => PermissionSchema.parse('invalid.permission')).toThrow(); }); }); describe('LoginRequestSchema', () => { it('should accept valid login request', () => { expect(() => LoginRequestSchema.parse({ email: 'user@example.com', password: 'password123', })).not.toThrow(); }); it('should accept login with MFA code', () => { expect(() => LoginRequestSchema.parse({ email: 'user@example.com', password: 'password123', mfaCode: '123456', })).not.toThrow(); }); it('should reject invalid MFA code length', () => { expect(() => LoginRequestSchema.parse({ email: 'user@example.com', password: 'password123', mfaCode: '12345', // 5 digits instead of 6 })).toThrow(); }); }); describe('CreateUserSchema', () => { it('should accept valid user creation', () => { expect(() => CreateUserSchema.parse({ email: 'user@example.com', password: 'SecurePass123', role: 'developer', })).not.toThrow(); }); it('should require strong password', () => { expect(() => CreateUserSchema.parse({ email: 'user@example.com', password: 'weak', role: 'developer', })).toThrow(); }); }); describe('TaskInputSchema', () => { it('should accept valid task input', () => { expect(() => TaskInputSchema.parse({ taskId: '550e8400-e29b-41d4-a716-446655440000', content: 'Implement new feature', agentType: 'coder', })).not.toThrow(); }); it('should reject task with shell characters in content', () => { expect(() => TaskInputSchema.parse({ taskId: '550e8400-e29b-41d4-a716-446655440000', content: 'Implement feature; rm -rf /', agentType: 'coder', })).toThrow(); }); }); describe('CommandArgumentSchema', () => { it('should accept safe arguments', () => { expect(() => CommandArgumentSchema.parse('--flag')).not.toThrow(); expect(() => CommandArgumentSchema.parse('value')).not.toThrow(); expect(() => CommandArgumentSchema.parse('path/to/file')).not.toThrow(); }); it('should reject arguments with null bytes', () => { expect(() => CommandArgumentSchema.parse('arg\x00injected')).toThrow(); }); it('should reject arguments with shell metacharacters', () => { expect(() => CommandArgumentSchema.parse('arg;injected')).toThrow(); expect(() => CommandArgumentSchema.parse('arg&&injected')).toThrow(); expect(() => CommandArgumentSchema.parse('arg|injected')).toThrow(); }); }); describe('PathSchema', () => { it('should accept valid paths', () => { expect(() => PathSchema.parse('/path/to/file.ts')).not.toThrow(); expect(() => PathSchema.parse('./relative/path')).not.toThrow(); }); it('should reject paths with traversal', () => { expect(() => PathSchema.parse('/path/../etc/passwd')).toThrow(); }); it('should reject paths with null bytes', () => { expect(() => PathSchema.parse('/path/file\x00.jpg')).toThrow(); }); }); describe('Sanitization Functions', () => { describe('sanitizeString', () => { it('should remove null bytes', () => { expect(sanitizeString('hello\x00world')).toBe('helloworld'); }); it('should remove HTML brackets', () => { expect(sanitizeString('')).toBe('scriptalert(1)/script'); }); it('should remove javascript: protocol', () => { expect(sanitizeString('javascript:alert(1)')).toBe('alert(1)'); }); it('should trim whitespace', () => { expect(sanitizeString(' hello ')).toBe('hello'); }); }); describe('sanitizeHtml', () => { it('should escape HTML entities', () => { expect(sanitizeHtml('