{"version":3,"sources":["../../../src/utils/config-validator.ts","../../../src/utils/config.ts","../../../src/utils/logger.ts","../../../src/api/init.ts"],"names":["path","load","winston","chalk","defaultConfig","dump"],"mappings":"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","file":"init.cjs","sourcesContent":["import { ValidationResult } from '../types.js';\nimport { LlmailConfig, ResolvedConfig } from './config.js';\n\n/**\n * Validates llmail configuration.\n * This is a focused utility that only handles config validation.\n * It ensures the config has the required structure and valid values.\n */\nexport class ConfigValidator {\n  /**\n   * Validate raw llmail configuration\n   */\n  validateConfig(config: LlmailConfig): ValidationResult {\n    const errors: string[] = [];\n\n    // Validate required arrays\n    if (!Array.isArray(config.types_list)) {\n      errors.push('types_list must be an array');\n    }\n\n    // Validate array contents if present\n    if (config.types_list?.some(type => typeof type !== 'string' || !type.match(/^[a-z0-9]+$/))) {\n      errors.push('types_list must contain only lowercase alphanumeric strings without hyphens');\n    }\n    if (config.active_states?.some(state => typeof state !== 'string' || !state.match(/^[a-z0-9]+$/))) {\n      errors.push('active_states must contain only lowercase alphanumeric strings without hyphens');\n    }\n    if (config.inactive_reasons?.some(reason => typeof reason !== 'string' || !reason.match(/^[a-z0-9]+$/))) {\n      errors.push('inactive_reasons must contain only lowercase alphanumeric strings without hyphens');\n    }\n\n    // Check for overlapping states if both arrays are present\n    if (config.active_states?.length > 0 && config.inactive_reasons?.length > 0) {\n      const activeStates = new Set(config.active_states);\n      for (const reason of config.inactive_reasons) {\n        if (activeStates.has(reason)) {\n          errors.push('States cannot be both active and inactive');\n          break;\n        }\n      }\n    }\n\n    // Validate required paths\n    if (!config.issues_dir) {\n      errors.push('issues_dir is required');\n    }\n    if (!config.inbox_dir) {\n      errors.push('inbox_dir is required');\n    }\n\n    // Validate plugins if present\n    if (config.plugins !== undefined) {\n      if (typeof config.plugins === 'string') {\n        // Single plugin string format\n        if (!config.plugins.match(/^[a-z0-9-]+$/)) {\n          errors.push('Plugin name must be lowercase alphanumeric with hyphens');\n        }\n      } else if (Array.isArray(config.plugins)) {\n        // Array format\n        for (const plugin of config.plugins) {\n          if (typeof plugin === 'string') {\n            // String array format\n            if (!plugin.match(/^[a-z0-9-]+$/)) {\n              errors.push('Plugin names must be lowercase alphanumeric with hyphens');\n            }\n          } else if (typeof plugin === 'object' && plugin !== null) {\n            // Object format with config\n            if (!plugin.name?.match(/^[a-z0-9-]+$/)) {\n              errors.push('Plugin names must be lowercase alphanumeric with hyphens');\n            }\n            if (plugin.config !== undefined && (typeof plugin.config !== 'object' || plugin.config === null)) {\n              errors.push('Plugin config must be an object');\n            }\n          } else {\n            errors.push('Invalid plugin configuration format');\n          }\n        }\n      } else {\n        errors.push('Invalid plugins configuration format');\n      }\n    }\n\n    return {\n      valid: errors.length === 0,\n      errors,\n      warnings: []\n    };\n  }\n\n  /**\n   * Validate resolved configuration\n   * This is used after the config has been normalized and paths resolved\n   */\n  validateResolvedConfig(config: ResolvedConfig): ValidationResult {\n    const errors: string[] = [];\n\n    // Validate required arrays are present\n    if (!Array.isArray(config.types_list)) {\n      errors.push('Resolved config must have types_list array');\n    }\n    if (!Array.isArray(config.active_states)) {\n      errors.push('Resolved config must have active_states array');\n    }\n    if (!Array.isArray(config.inactive_reasons)) {\n      errors.push('Resolved config must have inactive_reasons array');\n    }\n\n    // Validate required paths are absolute\n    if (!config.issues_dir?.startsWith('/')) {\n      errors.push('Resolved issues_dir must be absolute path');\n    }\n    if (!config.inbox_dir?.startsWith('/')) {\n      errors.push('Resolved inbox_dir must be absolute path');\n    }\n\n    return {\n      valid: errors.length === 0,\n      errors,\n      warnings: []\n    };\n  }\n} ","import fs from 'fs-extra';\nimport path from 'path';\nimport { dump, load } from 'js-yaml';\nimport { FileSystemAdapter } from './file-system-adapter.js';\nimport { ConfigValidator } from './config-validator.js';\n\n/**\n * Core LLMail configuration interface.\n * All paths are relative to the config file location.\n */\nexport interface LlmailConfig {\n  types_list: string[];\n  active_states: string[];\n  inactive_reasons: string[];\n  issues_dir: string;\n  inbox_dir: string;\n  type_dirs?: string[];\n  plugins?: string | string[] | Array<{\n    name: string;\n    config?: Record<string, any>;\n  }>;\n}\n\nexport type ResolvedConfig = LlmailConfig;\n\nconst defaultConfig: LlmailConfig = {\n  types_list: [],\n  active_states: [],\n  inactive_reasons: [],\n  issues_dir: 'issues',\n  inbox_dir: 'inbox'  // Changed to be relative to root\n};\n\n/**\n * Normalize paths in config.\n * All paths in the config should remain relative to the config file location.\n */\nfunction normalizeConfigPaths(config: LlmailConfig): LlmailConfig {\n  // Normalize paths but keep them relative\n  return {\n    ...config,\n    issues_dir: path.normalize(config.issues_dir),\n    inbox_dir: path.normalize(config.inbox_dir)\n  };\n}\n\n/**\n * Load and validate config from a file\n */\nexport async function loadConfigFromFile(fs: FileSystemAdapter, configPath: string, rootDir: string): Promise<LlmailConfig> {\n  const exists = await fs.exists(configPath);\n  \n  if (!exists) {\n    // Return default config with normalized paths\n    return normalizeConfigPaths(defaultConfig);\n  }\n\n  try {\n    const content = await fs.readFile(configPath);\n    let config: any;\n    \n    try {\n      config = load(content) as LlmailConfig;\n    } catch (err) {\n      throw new Error('Failed to load config');\n    }\n\n    // Merge with defaults, ensuring default types are preserved\n    const mergedConfig = {\n      ...defaultConfig,\n      ...config,\n      // Don't merge arrays - use config values if present, otherwise use defaults\n      types_list: config.types_list || defaultConfig.types_list,\n      active_states: config.active_states || defaultConfig.active_states,\n      inactive_reasons: config.inactive_reasons || defaultConfig.inactive_reasons\n    };\n\n    // Validate config\n    const validator = new ConfigValidator();\n    const result = validator.validateConfig(mergedConfig);\n    if (!result.valid) {\n      throw new Error(`Invalid config: ${result.errors[0]}`);\n    }\n\n    // Normalize paths but keep them relative\n    return normalizeConfigPaths(mergedConfig);\n  } catch (err) {\n    if (err instanceof Error) {\n      throw err;\n    }\n    throw new Error('Failed to load config');\n  }\n}\n\n/**\n * Load llmail config from the default location or specified path\n */\nexport async function loadLlmailConfig(fs: FileSystemAdapter, rootDir: string = process.cwd(), configPath?: string): Promise<LlmailConfig> {\n  const resolvedConfigPath = configPath || path.join(rootDir, 'llmail.yaml');\n  try {\n    return await loadConfigFromFile(fs, resolvedConfigPath, rootDir);\n  } catch (err) {\n    // If config file doesn't exist or can't be loaded, return defaults with normalized paths\n    return normalizeConfigPaths(defaultConfig);\n  }\n}\n\n/**\n * Save config to the default location\n */\nexport async function saveLlmailConfig(fs: FileSystemAdapter, config: LlmailConfig, configPath: string): Promise<void> {\n  const yamlContent = dump(normalizeConfigPaths(config));\n  \n  // Ensure the directory exists\n  const configDir = path.dirname(configPath);\n  await fs.ensureDir(configDir);\n  \n  await fs.writeFile(configPath, yamlContent);\n} ","import winston from 'winston';\nimport chalk from 'chalk';\nimport { IssueError, FileNotFoundError, InvalidFilenameError, InvalidTargetError, FileExistsError, handleError } from '../errors/errors.js';\n\n// Define our custom logger interface\nexport interface CustomLogger extends winston.Logger {\n  success(message: any, ...meta: any[]): winston.Logger;\n  debug(message: any, ...meta: any[]): winston.Logger;\n}\n\nconst { format, createLogger, transports } = winston;\nconst { combine, timestamp, printf } = format;\n\n// Custom format for different log levels\nconst customFormat = printf((info: winston.Logform.TransformableInfo) => {\n  const ts = new Date(info.timestamp?.toString() || Date.now()).toLocaleTimeString();\n  \n  switch (info.level) {\n    case 'error':\n      return `${chalk.gray(ts)} ${chalk.red('✖')} ${chalk.red(info.message)}`;\n    case 'warn':\n      return `${chalk.gray(ts)} ${chalk.yellow('⚠')} ${chalk.yellow(info.message)}`;\n    case 'info':\n      return `${chalk.gray(ts)} ${chalk.blue('ℹ')} ${info.message}`;\n    case 'debug':\n      return `${chalk.gray(ts)} ${chalk.gray('🔍')} ${chalk.gray(info.message)}`;\n    case 'success':\n      return `${chalk.gray(ts)} ${chalk.green('✔')} ${chalk.green(info.message)}`;\n    default:\n      return `${chalk.gray(ts)} ${info.message}`;\n  }\n});\n\n// Create custom success level\nconst customLevels = {\n  levels: {\n    error: 0,\n    warn: 1,\n    info: 2,\n    debug: 3,\n    success: 4\n  },\n  colors: {\n    error: 'red',\n    warn: 'yellow',\n    info: 'blue',\n    debug: 'gray',\n    success: 'green'\n  }\n};\n\n// Create a filter to completely silence debug messages unless explicitly enabled\nconst silenceDebug = format((info) => {\n  if (info.level === 'debug' && !process.env.DEBUG_ENABLED) {\n    return false;\n  }\n  return info;\n});\n\n// Create the logger with silent debug by default\nconst baseLogger = createLogger({\n  levels: customLevels.levels,\n  format: combine(\n    silenceDebug(),\n    timestamp(),\n    customFormat\n  ),\n  transports: [\n    new transports.Console({\n      level: 'info'\n    })\n  ]\n});\n\n// Add success method to match our interface\n(baseLogger as any).success = baseLogger.info;\n\nexport const logger = baseLogger as CustomLogger;\n\n// Add colors to winston\nwinston.addColors(customLevels.colors);\n\n// Function to configure logger based on debug flag\nexport function configureLogger(debug: boolean = false) {\n  process.env.DEBUG_ENABLED = debug ? 'true' : '';\n  const level = debug ? 'debug' : 'info';\n  logger.transports.forEach(transport => {\n    if (transport instanceof winston.transports.Console) {\n      transport.level = level;\n    }\n  });\n}\n\n// Re-export error types for convenience\nexport { IssueError, FileNotFoundError, InvalidFilenameError, InvalidTargetError, FileExistsError, handleError }; ","import { FileSystemAdapter } from '../../src/utils/file-system-adapter.js';\nimport { loadLlmailConfig } from '../../src/utils/config.js';\nimport { logger } from '../../src/utils/logger.js';\nimport { dump } from 'js-yaml';\nimport path from 'path';\n\nexport interface InitOptions {\n  configPath?: string;  // Path to config file or directory\n}\n\nexport interface InitResult {\n  success: boolean;\n  error?: string;\n  configPath?: string;\n}\n\nconst defaultConfig = {\n  version: 2,\n  types_list: ['bug', 'feature', 'task'],\n  active_states: ['active', 'blocked', 'inprogress'],\n  inactive_reasons: ['done', 'wontfix', 'duplicate'],\n  inactive_subfolders_for_types: true,\n  issues_dir: 'issues',\n  inbox_dir: 'issues/inbox',\n  require_inactive_reason: true\n};\n\n/**\n * Creates the directory structure for a new LLMail repository\n */\nasync function createDirectories(fs: FileSystemAdapter, config: any): Promise<void> {\n  logger.info('Creating directory structure...');\n\n  // Create root and inbox directories\n  await fs.ensureDir(config.issues_dir);\n  await fs.ensureDir(config.inbox_dir);\n  await fs.ensureDir(path.join(config.inbox_dir, '_templates'));\n\n  // Create type directories\n  for (const type of config.types_list) {\n    const typeDir = path.join(config.issues_dir, type);\n    await fs.ensureDir(typeDir);\n\n    // Create inactive reason subdirectories if enabled\n    if (config.inactive_subfolders_for_types) {\n      for (const reason of config.inactive_reasons) {\n        await fs.ensureDir(path.join(typeDir, reason));\n      }\n    }\n  }\n\n  // Create active state directories\n  for (const state of config.active_states) {\n    if (state !== 'inbox') { // inbox already created\n      await fs.ensureDir(path.join(config.issues_dir, `_${state}`));\n    }\n  }\n}\n\n/**\n * Initialize a new LLMail repository\n */\nexport async function init(fs: FileSystemAdapter, options: InitOptions = {}): Promise<InitResult> {\n  logger.info('Starting initialization...');\n  \n  try {\n    // Determine config path\n    const configDir = options.configPath || process.cwd();\n    const configPath = path.join(configDir, 'llmail.yaml');\n\n    // Write default config if it doesn't exist\n    if (!(await fs.exists(configPath))) {\n      logger.info('Creating default config file...');\n      await fs.ensureDir(configDir);\n      await fs.writeFile(configPath, dump(defaultConfig));\n    }\n\n    // Load config (either default or existing)\n    const config = await loadLlmailConfig(fs, configDir);\n\n    // Create directory structure\n    await createDirectories(fs, config);\n\n    logger.success('Initialization completed successfully');\n    return {\n      success: true,\n      configPath\n    };\n  } catch (err: any) {\n    logger.error(`Initialization failed: ${err.message}`);\n    return {\n      success: false,\n      error: err.message\n    };\n  }\n} "]}