UNPKG

@mickdarling/dollhousemcp

Version:

DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.

101 lines 13.5 kB
/** * Input validation and sanitization functions */ import { SECURITY_LIMITS, VALIDATION_PATTERNS } from './constants.js'; import { VALID_CATEGORIES } from '../config/constants.js'; /** * Validate and sanitize a filename */ export function validateFilename(filename) { if (!filename || typeof filename !== 'string') { throw new Error('Filename must be a non-empty string'); } if (filename.length > SECURITY_LIMITS.MAX_FILENAME_LENGTH) { throw new Error(`Filename too long (max ${SECURITY_LIMITS.MAX_FILENAME_LENGTH} characters)`); } // Remove any path separators and dangerous characters const sanitized = filename.replace(/[\/\\:*?"<>|]/g, '').replace(/^\.+/, ''); if (!VALIDATION_PATTERNS.SAFE_FILENAME.test(sanitized)) { throw new Error('Invalid filename format. Use alphanumeric characters, hyphens, underscores, and dots only.'); } return sanitized; } /** * Validate and sanitize a path */ export function validatePath(inputPath) { if (!inputPath || typeof inputPath !== 'string') { throw new Error('Path must be a non-empty string'); } // Remove leading/trailing slashes and normalize // Length limits added to prevent ReDoS attacks const normalized = inputPath.replace(/^\/{1,100}|\/{1,100}$/g, '').replace(/\/{1,100}/g, '/'); if (!VALIDATION_PATTERNS.SAFE_PATH.test(normalized)) { throw new Error('Invalid path format. Use alphanumeric characters, hyphens, underscores, dots, and forward slashes only.'); } // Check for path traversal attempts if (normalized.includes('..') || normalized.includes('./') || normalized.includes('/.')) { throw new Error('Path traversal not allowed'); } // Validate path depth const depth = normalized.split('/').length; if (depth > SECURITY_LIMITS.MAX_PATH_DEPTH) { throw new Error(`Path too deep (max ${SECURITY_LIMITS.MAX_PATH_DEPTH} levels)`); } return normalized; } /** * Validate and sanitize a username */ export function validateUsername(username) { if (!username || typeof username !== 'string') { throw new Error('Username must be a non-empty string'); } if (!VALIDATION_PATTERNS.SAFE_USERNAME.test(username)) { throw new Error('Invalid username format. Use alphanumeric characters, hyphens, underscores, and dots only.'); } return username.toLowerCase(); } /** * Validate a category */ export function validateCategory(category) { if (!category || typeof category !== 'string') { throw new Error('Category must be a non-empty string'); } if (!VALIDATION_PATTERNS.SAFE_CATEGORY.test(category)) { throw new Error('Invalid category format. Use alphabetic characters, hyphens, and underscores only.'); } const normalized = category.toLowerCase(); if (!VALID_CATEGORIES.includes(normalized)) { throw new Error(`Invalid category. Must be one of: ${VALID_CATEGORIES.join(', ')}`); } return normalized; } /** * Validate content size */ export function validateContentSize(content, maxSize = SECURITY_LIMITS.MAX_CONTENT_LENGTH) { if (!content || typeof content !== 'string') { throw new Error('Content must be a non-empty string'); } const sizeBytes = Buffer.byteLength(content, 'utf8'); if (sizeBytes > maxSize) { throw new Error(`Content too large (${sizeBytes} bytes, max ${maxSize} bytes)`); } } /** * General input sanitization */ export function sanitizeInput(input, maxLength = 1000) { if (!input || typeof input !== 'string') { return ''; } // Remove potentially dangerous characters and limit length return input .replace(/[\x00-\x1F\x7F]/g, '') // Remove control characters .replace(/[<>'"&]/g, '') // Remove HTML-dangerous characters .substring(0, maxLength) .trim(); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"InputValidator.js","sourceRoot":"","sources":["../../src/security/InputValidator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Input validation and sanitization functions\n */\n\nimport { SECURITY_LIMITS, VALIDATION_PATTERNS } from './constants.js';\nimport { VALID_CATEGORIES } from '../config/constants.js';\n\n/**\n * Validate and sanitize a filename\n */\nexport function validateFilename(filename: string): string {\n  if (!filename || typeof filename !== 'string') {\n    throw new Error('Filename must be a non-empty string');\n  }\n  \n  if (filename.length > SECURITY_LIMITS.MAX_FILENAME_LENGTH) {\n    throw new Error(`Filename too long (max ${SECURITY_LIMITS.MAX_FILENAME_LENGTH} characters)`);\n  }\n  \n  // Remove any path separators and dangerous characters\n  const sanitized = filename.replace(/[\\/\\\\:*?\"<>|]/g, '').replace(/^\\.+/, '');\n  \n  if (!VALIDATION_PATTERNS.SAFE_FILENAME.test(sanitized)) {\n    throw new Error('Invalid filename format. Use alphanumeric characters, hyphens, underscores, and dots only.');\n  }\n  \n  return sanitized;\n}\n\n/**\n * Validate and sanitize a path\n */\nexport function validatePath(inputPath: string): string {\n  if (!inputPath || typeof inputPath !== 'string') {\n    throw new Error('Path must be a non-empty string');\n  }\n  \n  // Remove leading/trailing slashes and normalize\n  // Length limits added to prevent ReDoS attacks\n  const normalized = inputPath.replace(/^\\/{1,100}|\\/{1,100}$/g, '').replace(/\\/{1,100}/g, '/');\n  \n  if (!VALIDATION_PATTERNS.SAFE_PATH.test(normalized)) {\n    throw new Error('Invalid path format. Use alphanumeric characters, hyphens, underscores, dots, and forward slashes only.');\n  }\n  \n  // Check for path traversal attempts\n  if (normalized.includes('..') || normalized.includes('./') || normalized.includes('/.')) {\n    throw new Error('Path traversal not allowed');\n  }\n  \n  // Validate path depth\n  const depth = normalized.split('/').length;\n  if (depth > SECURITY_LIMITS.MAX_PATH_DEPTH) {\n    throw new Error(`Path too deep (max ${SECURITY_LIMITS.MAX_PATH_DEPTH} levels)`);\n  }\n  \n  return normalized;\n}\n\n/**\n * Validate and sanitize a username\n */\nexport function validateUsername(username: string): string {\n  if (!username || typeof username !== 'string') {\n    throw new Error('Username must be a non-empty string');\n  }\n  \n  if (!VALIDATION_PATTERNS.SAFE_USERNAME.test(username)) {\n    throw new Error('Invalid username format. Use alphanumeric characters, hyphens, underscores, and dots only.');\n  }\n  \n  return username.toLowerCase();\n}\n\n/**\n * Validate a category\n */\nexport function validateCategory(category: string): string {\n  if (!category || typeof category !== 'string') {\n    throw new Error('Category must be a non-empty string');\n  }\n  \n  if (!VALIDATION_PATTERNS.SAFE_CATEGORY.test(category)) {\n    throw new Error('Invalid category format. Use alphabetic characters, hyphens, and underscores only.');\n  }\n  \n  const normalized = category.toLowerCase();\n  \n  if (!VALID_CATEGORIES.includes(normalized)) {\n    throw new Error(`Invalid category. Must be one of: ${VALID_CATEGORIES.join(', ')}`);\n  }\n  \n  return normalized;\n}\n\n/**\n * Validate content size\n */\nexport function validateContentSize(content: string, maxSize: number = SECURITY_LIMITS.MAX_CONTENT_LENGTH): void {\n  if (!content || typeof content !== 'string') {\n    throw new Error('Content must be a non-empty string');\n  }\n  \n  const sizeBytes = Buffer.byteLength(content, 'utf8');\n  if (sizeBytes > maxSize) {\n    throw new Error(`Content too large (${sizeBytes} bytes, max ${maxSize} bytes)`);\n  }\n}\n\n/**\n * General input sanitization\n */\nexport function sanitizeInput(input: string, maxLength: number = 1000): string {\n  if (!input || typeof input !== 'string') {\n    return '';\n  }\n  \n  // Remove potentially dangerous characters and limit length\n  return input\n    .replace(/[\\x00-\\x1F\\x7F]/g, '') // Remove control characters\n    .replace(/[<>'\"&]/g, '') // Remove HTML-dangerous characters\n    .substring(0, maxLength)\n    .trim();\n}"]}