@frontfriend/tailwind 2.3.0 → 2.4.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../lib/core/constants.js", "../../../lib/core/errors.js", "../../../lib/core/file-utils.js", "../../../lib/core/path-utils.js", "../../../lib/core/local-token-reader.js", "../../../lib/core/api-client.js", "../../../lib/core/component-downloader.js"],
4
- "sourcesContent": ["/**\n * Configuration constants for FrontFriend Tailwind v2\n */\n\n// API Configuration\nconst DEFAULT_API_URL = 'https://app.frontfriend.dev';\nconst LEGACY_API_URL = 'https://tokens-studio-donux.up.railway.app/api';\n\n// Cache Configuration\nconst CACHE_DIR_NAME = '.cache/frontfriend';\nconst CACHE_TTL_DAYS = 7;\nconst CACHE_TTL_MS = CACHE_TTL_DAYS * 24 * 60 * 60 * 1000;\n\n// File names\nconst CACHE_FILES = {\n JS: [\n 'tokens.js',\n 'variables.js',\n 'semanticVariables.js',\n 'semanticDarkVariables.js',\n 'cls.js',\n 'custom.js'\n ],\n JSON: [\n 'fonts.json',\n 'icons.json',\n 'components-config.json',\n 'version.json',\n 'metadata.json'\n ]\n};\n\n// Environment variables\nconst ENV_VARS = {\n FF_ID: 'FF_ID',\n FF_API_URL: 'FF_API_URL',\n FF_USE_LOCAL: 'FF_USE_LOCAL'\n};\n\nmodule.exports = {\n DEFAULT_API_URL,\n LEGACY_API_URL,\n CACHE_DIR_NAME,\n CACHE_TTL_DAYS,\n CACHE_TTL_MS,\n CACHE_FILES,\n ENV_VARS\n};", "class APIError extends Error {\n constructor(message, statusCode, url) {\n super(message);\n this.name = 'APIError';\n this.statusCode = statusCode;\n this.url = url;\n this.code = `API_${statusCode}`;\n }\n}\n\nclass CacheError extends Error {\n constructor(message, operation) {\n super(message);\n this.name = 'CacheError';\n this.operation = operation;\n this.code = `CACHE_${operation.toUpperCase()}`;\n }\n}\n\nclass ConfigError extends Error {\n constructor(message, field) {\n super(message);\n this.name = 'ConfigError';\n this.field = field;\n this.code = `CONFIG_${field.toUpperCase()}`;\n }\n}\n\nclass ProcessingError extends Error {\n constructor(message, token) {\n super(message);\n this.name = 'ProcessingError';\n this.token = token;\n this.code = 'PROCESSING_ERROR';\n }\n}\n\nmodule.exports = {\n APIError,\n CacheError,\n ConfigError,\n ProcessingError\n};", "const fs = require('fs');\n\n/**\n * Read a JSON file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the JSON file\n * @returns {Object|null} Parsed JSON or null if file not found\n * @throws {Error} If JSON is invalid\n */\nfunction readJsonFileSafe(filepath) {\n try {\n const content = fs.readFileSync(filepath, 'utf8');\n return JSON.parse(content);\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found, return null like API would return 404\n }\n throw error;\n }\n}\n\n/**\n * Read a file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the file\n * @returns {string|null} File content or null if file not found\n * @throws {Error} For other file system errors\n */\nfunction readFileSafe(filepath) {\n try {\n return fs.readFileSync(filepath, 'utf8');\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found\n }\n throw error;\n }\n}\n\n/**\n * Write JSON data to a file with proper formatting\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to write\n * @param {number} indent - Indentation level (default: 2)\n */\nfunction writeJsonFile(filepath, data, indent = 2) {\n const content = JSON.stringify(data, null, indent);\n fs.writeFileSync(filepath, content, 'utf8');\n}\n\n/**\n * Write module.exports file with JSON data\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to export\n */\nfunction writeModuleExportsFile(filepath, data) {\n // Handle empty data cases\n if (data === null || data === undefined || \n (typeof data === 'object' && Object.keys(data).length === 0) ||\n (Array.isArray(data) && data.length === 0)) {\n const emptyContent = Array.isArray(data) ? '[]' : '{}';\n const content = `module.exports = ${emptyContent};`;\n fs.writeFileSync(filepath, content, 'utf8');\n } else {\n const content = `module.exports = ${JSON.stringify(data, null, 2)};`;\n fs.writeFileSync(filepath, content, 'utf8');\n }\n}\n\n/**\n * Check if a file exists\n * @param {string} filepath - Path to check\n * @returns {boolean} True if file exists\n */\nfunction fileExists(filepath) {\n return fs.existsSync(filepath);\n}\n\n/**\n * Create directory recursively\n * @param {string} dirpath - Directory path to create\n */\nfunction ensureDirectoryExists(dirpath) {\n fs.mkdirSync(dirpath, { recursive: true });\n}\n\n/**\n * Remove directory recursively\n * @param {string} dirpath - Directory path to remove\n */\nfunction removeDirectory(dirpath) {\n fs.rmSync(dirpath, { recursive: true, force: true });\n}\n\nmodule.exports = {\n readJsonFileSafe,\n readFileSafe,\n writeJsonFile,\n writeModuleExportsFile,\n fileExists,\n ensureDirectoryExists,\n removeDirectory\n};", "const path = require('path');\nconst { fileExists } = require('./file-utils');\n\n/**\n * Find a file by traversing up the directory tree\n * @param {string} filename - Name of the file to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the file or null if not found\n */\nfunction findFileUpward(filename, startDir = process.cwd()) {\n let currentDir = startDir;\n \n while (currentDir !== path.parse(currentDir).root) {\n const filePath = path.join(currentDir, filename);\n \n if (fileExists(filePath)) {\n return filePath;\n }\n \n currentDir = path.dirname(currentDir);\n }\n \n return null;\n}\n\n/**\n * Find a directory by traversing up and checking multiple possible paths\n * @param {string} dirname - Name of the directory to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @param {Function} validator - Optional function to validate the directory\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryUpward(dirname, startDir = process.cwd(), validator = null) {\n const possiblePaths = [];\n let currentDir = startDir;\n const maxLevels = 10; // Prevent infinite loop\n \n // Build list of possible paths\n for (let i = 0; i < maxLevels; i++) {\n // Direct path\n possiblePaths.push(path.join(currentDir, dirname));\n \n // Relative paths up to 3 levels\n if (i < 3) {\n possiblePaths.push(path.join(currentDir, '../'.repeat(i + 1) + dirname));\n }\n \n const parentDir = path.dirname(currentDir);\n if (parentDir === currentDir) break; // Reached root\n currentDir = parentDir;\n }\n \n // Also check from module directory (for when running from node_modules)\n possiblePaths.push(path.join(__dirname, '../../../', dirname));\n \n // Find first existing directory that passes validation\n for (const dirPath of possiblePaths) {\n if (fileExists(dirPath)) {\n if (!validator || validator(dirPath)) {\n return dirPath;\n }\n }\n }\n \n return null;\n}\n\n/**\n * Find a directory with multiple possible relative paths\n * @param {string[]} possiblePaths - Array of possible paths to check\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryWithPaths(possiblePaths, startDir = process.cwd()) {\n // Build absolute paths from the start directory\n const absolutePaths = possiblePaths.map(p => {\n if (path.isAbsolute(p)) {\n return p;\n }\n return path.join(startDir, p);\n });\n \n // Find first existing path\n for (const dirPath of absolutePaths) {\n if (fileExists(dirPath)) {\n return dirPath;\n }\n }\n \n return null;\n}\n\nmodule.exports = {\n findFileUpward,\n findDirectoryUpward,\n findDirectoryWithPaths\n};", "const path = require('path');\nconst { findDirectoryUpward } = require('./path-utils');\nconst { readJsonFileSafe, readFileSafe, fileExists } = require('./file-utils');\n\n/**\n * LocalTokenReader - Reads tokens directly from the local file system\n * Mimics the APIClient interface but reads from apps/tokens directory\n */\nclass LocalTokenReader {\n constructor(ffId, options = {}) {\n this.ffId = ffId;\n this.tokensBasePath = options.tokensBasePath || this.findTokensPath();\n this.folderMap = this.loadFolderMap();\n this.projectFolder = this.folderMap[ffId];\n \n if (!this.projectFolder) {\n throw new Error(`No folder mapping found for ff-id: ${ffId}`);\n }\n \n this.projectPath = path.join(this.tokensBasePath, this.projectFolder);\n }\n\n /**\n * Find the tokens directory by looking for apps/tokens in parent directories\n */\n findTokensPath() {\n // Try to find apps/tokens first\n let tokensPath = findDirectoryUpward('apps/tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n \n // If not found, try just 'tokens'\n if (!tokensPath) {\n tokensPath = findDirectoryUpward('tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n }\n \n if (!tokensPath) {\n throw new Error('Could not find tokens directory with folderMap.json');\n }\n \n return tokensPath;\n }\n\n /**\n * Load the folder mapping from folderMap.json\n */\n loadFolderMap() {\n const folderMapPath = path.join(this.tokensBasePath, 'folderMap.json');\n const folderMap = readJsonFileSafe(folderMapPath);\n \n if (!folderMap) {\n throw new Error(`Failed to load folderMap.json: File not found`);\n }\n \n return folderMap;\n }\n\n /**\n * Fetch all token files (global, colors, semantic)\n * Mimics the APIClient.fetchTokens() method\n */\n async fetchTokens() {\n const figmaPath = path.join(this.projectPath, 'figma');\n \n // Try different possible paths for tokens\n const tokenPaths = {\n global: [\n path.join(figmaPath, 'Global Tokens', 'Default.json'),\n path.join(figmaPath, 'global.json')\n ],\n colors: [\n path.join(figmaPath, 'Global Colors', 'Default.json'),\n path.join(figmaPath, 'Global Colors', 'Light.json')\n ],\n semantic: [\n path.join(figmaPath, 'Semantic', 'Light.json'),\n path.join(figmaPath, 'Semantic', 'Default.json')\n ],\n semanticDark: [\n path.join(figmaPath, 'Semantic', 'Dark.json')\n ]\n };\n\n const results = {};\n \n // Try each possible path for each token type\n for (const [key, paths] of Object.entries(tokenPaths)) {\n for (const tokenPath of paths) {\n const content = readJsonFileSafe(tokenPath);\n if (content !== null) {\n results[key] = content;\n break; // Found it, no need to try other paths\n }\n }\n // If not found in any path, set to null\n if (!results[key]) {\n results[key] = null;\n }\n }\n\n return results;\n }\n\n /**\n * Fetch pre-processed tokens (if available in cache folder)\n */\n async fetchProcessedTokens() {\n const cachePath = path.join(this.projectPath, 'cache');\n \n if (!fileExists(cachePath)) {\n return null;\n }\n\n // Read all the cache files that would be in processed tokens\n const cacheFiles = {\n 'tokens.js': 'hashedTokens.js',\n 'variables.js': 'hashedVariables.js', \n 'semanticVariables.js': 'hashedSemanticVariables.js',\n 'semanticDarkVariables.js': 'hashedSemanticDarkVariables.js',\n 'cls.js': 'cls.js',\n 'custom.js': 'custom.js',\n 'fonts.json': 'fonts.json',\n 'icons.json': 'icons.json',\n 'components-config.json': 'encoded-config.json',\n 'version.json': 'version.json'\n };\n\n const processedData = {};\n \n for (const [key, fileName] of Object.entries(cacheFiles)) {\n const filePath = path.join(cachePath, fileName);\n if (key.endsWith('.js')) {\n processedData[key] = readFileSafe(filePath);\n } else {\n processedData[key] = readJsonFileSafe(filePath);\n }\n }\n\n // Only return if we have at least some data\n const hasData = Object.values(processedData).some(val => val !== null);\n return hasData ? processedData : null;\n }\n\n /**\n * Fetch components configuration\n */\n async fetchComponentsConfig() {\n return readJsonFileSafe(path.join(this.projectPath, 'components-config.json'));\n }\n\n /**\n * Fetch fonts configuration\n */\n async fetchFonts() {\n return readJsonFileSafe(path.join(this.projectPath, 'font.json'));\n }\n\n /**\n * Fetch icons configuration\n */\n async fetchIcons() {\n return readJsonFileSafe(path.join(this.projectPath, 'icons.json'));\n }\n\n /**\n * Fetch version information\n */\n async fetchVersion() {\n return readJsonFileSafe(path.join(this.projectPath, 'version.json'));\n }\n\n /**\n * Fetch custom CSS\n */\n async fetchCustomCss() {\n return readFileSafe(path.join(this.projectPath, 'custom.css'));\n }\n\n /**\n * Fetch plan metadata\n */\n async fetchPlanMetadata() {\n const metadata = readJsonFileSafe(path.join(this.projectPath, 'metadata.json'));\n \n // Default to full plan if no metadata exists\n return metadata || {\n planType: 'full',\n allowedComponents: null\n };\n }\n\n // Compatibility methods to match APIClient interface\n fetchJson() {\n throw new Error('fetchJson is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n\n fetchRaw() {\n throw new Error('fetchRaw is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n}\n\nmodule.exports = { LocalTokenReader };", "const https = require('https');\nconst http = require('http');\nconst { URL } = require('url');\nconst { DEFAULT_API_URL, LEGACY_API_URL, ENV_VARS } = require('./constants');\nconst { APIError } = require('./errors');\nconst { LocalTokenReader } = require('./local-token-reader');\n\nclass APIClient {\n constructor(ffId, options = {}) {\n this.ffId = ffId;\n this.baseURL = options.baseURL || process.env[ENV_VARS.FF_API_URL] || DEFAULT_API_URL;\n \n // Check if we should use local token reader\n if (process.env[ENV_VARS.FF_USE_LOCAL] === 'true') {\n console.log(' \uD83C\uDFE0 Using local token reader');\n this.localReader = new LocalTokenReader(ffId, options);\n }\n }\n\n /**\n * Fetch JSON data from a URL using native https module\n * @param {string} url - The URL to fetch\n * @returns {Promise<any>} Parsed JSON data\n */\n fetchJson(url) {\n return new Promise((resolve, reject) => {\n // Determine if we should use http or https\n const protocol = url.startsWith('https') ? https : http;\n \n protocol.get(url, (response) => {\n let data = '';\n \n // Check for HTTP errors\n if (response.statusCode >= 400) {\n reject(new APIError(\n `HTTP ${response.statusCode}: ${response.statusMessage}`,\n response.statusCode,\n url\n ));\n return;\n }\n\n // Collect response chunks\n response.on('data', (chunk) => {\n data += chunk;\n });\n\n // Parse JSON when complete\n response.on('end', () => {\n try {\n const parsed = JSON.parse(data);\n resolve(parsed);\n } catch (error) {\n reject(new APIError(\n `Invalid JSON response: ${error.message}`,\n response.statusCode,\n url\n ));\n }\n });\n\n // Handle response errors\n response.on('error', (error) => {\n reject(new APIError(\n `Response error: ${error.message}`,\n response.statusCode,\n url\n ));\n });\n }).on('error', (error) => {\n // Handle network errors\n reject(new APIError(\n `Network error: ${error.message}`,\n 0,\n url\n ));\n });\n });\n }\n\n /**\n * Fetch raw content from a URL\n * @param {string} url - The URL to fetch\n * @returns {Promise<string>} Raw text content\n */\n fetchRaw(url) {\n return new Promise((resolve, reject) => {\n // Determine if we should use http or https\n const protocol = url.startsWith('https') ? https : http;\n \n protocol.get(url, (response) => {\n let data = '';\n \n // Check for HTTP errors\n if (response.statusCode >= 400) {\n reject(new APIError(\n `HTTP ${response.statusCode}: ${response.statusMessage}`,\n response.statusCode,\n url\n ));\n return;\n }\n\n // Collect response chunks\n response.on('data', (chunk) => {\n data += chunk;\n });\n\n // Return raw data when complete\n response.on('end', () => {\n resolve(data);\n });\n\n // Handle response errors\n response.on('error', (error) => {\n reject(new APIError(\n `Response error: ${error.message}`,\n response.statusCode,\n url\n ));\n });\n }).on('error', (error) => {\n // Handle network errors\n reject(new APIError(\n `Network error: ${error.message}`,\n 0,\n url\n ));\n });\n });\n }\n\n /**\n * Fetch all token files (global, colors, semantic)\n * @returns {Promise<Object>} Object with all token data\n */\n async fetchTokens() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchTokens();\n }\n \n // First try the new design system endpoint\n try {\n const designSystemUrl = `${this.baseURL}/api/design-systems/${this.ffId}/tokens`;\n const tokensData = await this.fetchJson(designSystemUrl);\n \n // If we have the new format, parse it\n if (tokensData && tokensData.tokens) {\n const tokens = tokensData.tokens;\n \n // Extract specific token sets from the new format\n return {\n global: tokens['Global Tokens/Default'] || tokens['global'] || null,\n colors: tokens['Global Colors/Default'] || tokens['colors'] || null,\n semantic: tokens['Semantic/Light'] || tokens['semantic/light'] || null,\n semanticDark: tokens['Semantic/Dark'] || tokens['semantic/dark'] || null\n };\n }\n } catch (err) {\n // If design system is not synced (400 error), throw with clear message\n if (err.statusCode === 400) {\n throw new APIError(\n 'Design system not synced with Figma. Please sync tokens from Figma before using this design system.',\n 400,\n `${this.baseURL}/api/design-systems/${this.ffId}/tokens`\n );\n }\n // Failed to fetch from design system endpoint, falling back to legacy URLs\n }\n \n // Fall back to legacy token URLs\n // Use legacy API URL for backward compatibility\n const legacyBaseURL = LEGACY_API_URL;\n const urls = {\n global: `${legacyBaseURL}/${this.ffId}/global-tokens/default.json`,\n colors: `${legacyBaseURL}/${this.ffId}/global-colors/default.json`,\n semantic: `${legacyBaseURL}/${this.ffId}/semantic/light.json`,\n semanticDark: `${legacyBaseURL}/${this.ffId}/semantic/dark.json`\n };\n\n const results = await Promise.all([\n this.fetchJson(urls.global).catch(err => err.statusCode === 404 ? null : Promise.reject(err)),\n this.fetchJson(urls.colors).catch(err => err.statusCode === 404 ? null : Promise.reject(err)),\n this.fetchJson(urls.semantic).catch(err => err.statusCode === 404 ? null : Promise.reject(err)),\n this.fetchJson(urls.semanticDark).catch(err => err.statusCode === 404 ? null : Promise.reject(err))\n ]);\n\n return {\n global: results[0],\n colors: results[1],\n semantic: results[2],\n semanticDark: results[3]\n };\n }\n\n /**\n * Fetch pre-processed tokens from the design system endpoint\n * @returns {Promise<Object|null>} Processed tokens ready for caching or null if not found\n */\n async fetchProcessedTokens() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchProcessedTokens();\n }\n \n const processedUrl = `${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;\n try {\n return await this.fetchJson(processedUrl);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n // If design system is not synced (400 error), throw with clear message\n if (error.statusCode === 400) {\n throw new APIError(\n 'Design system not synced with Figma. Please sync tokens from Figma before using this design system.',\n 400,\n processedUrl\n );\n }\n throw error;\n }\n }\n\n /**\n * Fetch components configuration\n * @returns {Promise<Object|null>} Components config or null if not found\n */\n async fetchComponentsConfig() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchComponentsConfig();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/components-config.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch fonts configuration\n * @returns {Promise<Object|null>} Fonts object with font1, font2, etc. URLs or null\n */\n async fetchFonts() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchFonts();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/font.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch icons configuration\n * @returns {Promise<Object|null>} Icons data or null if not found\n */\n async fetchIcons() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchIcons();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/icons.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch version information\n * @returns {Promise<Object|null>} Version data or null if not found\n */\n async fetchVersion() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchVersion();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/version.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch custom CSS\n * @returns {Promise<string|null>} Custom CSS string or null if not found\n */\n async fetchCustomCss() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchCustomCss();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/custom.css`;\n try {\n return await this.fetchRaw(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch plan metadata (plan type and allowed components)\n * @returns {Promise<Object|null>} Plan metadata or null if not found\n */\n async fetchPlanMetadata() {\n // Use local reader if available\n if (this.localReader) {\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 Using local reader for plan metadata');\n }\n return this.localReader.fetchPlanMetadata();\n }\n \n // Try to fetch from tokens server first\n const tokensUrl = `${LEGACY_API_URL}/${this.ffId}/metadata`;\n try {\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetching plan metadata from: ${tokensUrl}`);\n }\n return await this.fetchJson(tokensUrl);\n } catch (error) {\n // If not found in tokens server, return default full plan\n if (error.statusCode === 404) {\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 No metadata found, defaulting to full plan');\n }\n return {\n planType: 'full',\n allowedComponents: null\n };\n }\n throw error;\n }\n }\n}\n\nmodule.exports = { APIClient };", "const https = require('https');\nconst fs = require('fs');\nconst path = require('path');\nconst { URL } = require('url');\nconst os = require('os');\n\nconst REGISTRY_URL = process.env.FF_REGISTRY_URL || 'https://ff-registry.up.railway.app';\n\nclass ComponentDownloader {\n constructor(options = {}) {\n this.appRoot = options.appRoot || process.cwd();\n this.framework = options.framework || this.detectFramework();\n this.config = options.config || {};\n this.outputPath = options.outputPath || this.getDefaultOutputPath();\n this.overwrite = options.overwrite || false;\n this.downloadedComponents = new Set();\n this.ffId = this.config.ffId || this.config['ff-id'] || null;\n this.planMetadata = null; // Will be populated when needed\n }\n\n /**\n * Detect framework from package.json\n */\n detectFramework() {\n const packageJsonPath = path.join(this.appRoot, 'package.json');\n \n if (fs.existsSync(packageJsonPath)) {\n const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));\n const deps = { ...packageJson.dependencies, ...packageJson.devDependencies };\n \n if (deps.vue) {\n return 'vue';\n } else if (deps.react) {\n return 'react';\n }\n }\n \n // Default to React if can't detect\n return 'react';\n }\n\n /**\n * Get default output path based on framework and config\n */\n getDefaultOutputPath() {\n // Check if config has ui alias\n if (this.config.aliases && this.config.aliases.ui) {\n const uiAlias = this.config.aliases.ui;\n // Convert alias to actual path (e.g., @/components/ui -> components/ui)\n const cleanPath = uiAlias.replace(/^@\\//, '');\n return path.join(this.appRoot, cleanPath);\n }\n \n // Fallback to default paths\n if (this.framework === 'vue') {\n return path.join(this.appRoot, 'src/components/ui');\n }\n return path.join(this.appRoot, 'components/ui');\n }\n\n /**\n * Fetch JSON from URL or local file\n */\n fetchJson(url) {\n // Check if using local mode at runtime\n const USE_LOCAL = process.env.FF_USE_LOCAL === 'true';\n \n // If using local mode, read from file system\n if (USE_LOCAL) {\n // Try multiple paths to find the registry\n const possiblePaths = [\n '../../packages/registry',\n '../../../packages/registry',\n '../../../../packages/registry',\n path.join(__dirname, '../../../registry'),\n 'packages/registry'\n ];\n \n let registryPath;\n for (const p of possiblePaths) {\n const fullPath = path.isAbsolute(p) ? p : path.join(process.cwd(), p);\n if (fs.existsSync(fullPath)) {\n registryPath = fullPath;\n break;\n }\n }\n \n if (!registryPath) {\n return Promise.reject(new Error(`Local registry not found. Tried paths: ${possiblePaths.join(', ')}`));\n }\n \n // Extract the path from the URL and construct proper local path\n const urlPath = url.replace(REGISTRY_URL, '').replace(/^\\//, '');\n const localPath = path.join(registryPath, ...urlPath.split('/'));\n \n try {\n const content = fs.readFileSync(localPath, 'utf8');\n return Promise.resolve(JSON.parse(content));\n } catch (error) {\n if (error.code === 'ENOENT') {\n return Promise.reject(new Error('HTTP 404: Not Found'));\n }\n return Promise.reject(error);\n }\n }\n\n return new Promise((resolve, reject) => {\n https.get(url, (response) => {\n let data = '';\n \n if (response.statusCode >= 400) {\n reject(new Error(`HTTP ${response.statusCode}: ${response.statusMessage}`));\n return;\n }\n\n response.on('data', (chunk) => {\n data += chunk;\n });\n\n response.on('end', () => {\n try {\n resolve(JSON.parse(data));\n } catch (error) {\n reject(new Error(`Invalid JSON response: ${error.message}`));\n }\n });\n\n }).on('error', (error) => {\n reject(error);\n });\n });\n }\n\n /**\n * Get all available components from registry\n */\n async getAllComponents() {\n try {\n const url = `${REGISTRY_URL}/components/${this.framework}/index.json`;\n const index = await this.fetchJson(url);\n let components = index.components || [];\n \n // Filter components based on plan\n if (this.ffId) {\n // Ensure plan metadata is loaded\n await this.isComponentAllowed('dummy'); // This will load metadata\n \n if (this.planMetadata && this.planMetadata.planType === 'trial' && this.planMetadata.allowedComponents) {\n components = components.filter(comp => this.planMetadata.allowedComponents.includes(comp));\n \n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Trial plan: filtering to ${components.length} allowed components`);\n }\n }\n }\n \n return components;\n } catch (error) {\n console.error('Failed to fetch component index:', error.message);\n return [];\n }\n }\n\n /**\n * Get all custom components for this ff-id\n */\n async getCustomComponents() {\n if (!this.ffId) {\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 No ff-id found, skipping custom components');\n }\n return [];\n }\n \n try {\n const url = `${REGISTRY_URL}/custom/${this.ffId}/${this.framework}/index.json`;\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetching custom components from: ${url}`);\n }\n const index = await this.fetchJson(url);\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Found custom components: ${JSON.stringify(index.components)}`);\n }\n return index.components || [];\n } catch (error) {\n // Custom components might not exist, that's ok\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 No custom components found: ${error.message}`);\n }\n return [];\n }\n }\n\n /**\n * Download multiple components\n */\n async downloadComponents(components) {\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 ComponentDownloader initialized with ff-id: ${this.ffId}`);\n }\n \n const results = {\n successful: [],\n failed: [],\n dependencies: new Set()\n };\n\n for (const component of components) {\n try {\n const deps = await this.downloadComponent(component);\n results.successful.push(component);\n \n // Add dependencies\n deps.forEach(dep => results.dependencies.add(dep));\n } catch (error) {\n results.failed.push({\n component,\n error: error.message\n });\n }\n }\n\n return {\n successful: results.successful,\n failed: results.failed,\n dependencies: Array.from(results.dependencies)\n };\n }\n\n /**\n * Check if component is allowed based on plan\n */\n async isComponentAllowed(componentName) {\n // Fetch plan metadata if not already loaded\n if (!this.planMetadata && this.ffId) {\n const { APIClient } = require('./api-client');\n const apiClient = new APIClient(this.ffId, {\n baseURL: this.config['api-url'] || this.config.apiUrl\n });\n try {\n this.planMetadata = await apiClient.fetchPlanMetadata();\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetched plan metadata: ${this.planMetadata.planType}, allowed: ${this.planMetadata.allowedComponents ? this.planMetadata.allowedComponents.length + ' components' : 'all'}`);\n }\n } catch (error) {\n console.warn('Failed to fetch plan metadata:', error.message);\n // Default to full plan if we can't fetch metadata\n this.planMetadata = { planType: 'full', allowedComponents: null };\n }\n }\n\n // If full plan or no metadata, allow all components\n if (!this.planMetadata || this.planMetadata.planType === 'full') {\n return true;\n }\n\n // For trial plan, check if component is in allowed list\n if (this.planMetadata.planType === 'trial' && this.planMetadata.allowedComponents) {\n return this.planMetadata.allowedComponents.includes(componentName);\n }\n\n // Default to allowing if we can't determine\n return true;\n }\n\n /**\n * Download a single component and its registry dependencies\n */\n async downloadComponent(componentName) {\n // Avoid downloading the same component twice\n if (this.downloadedComponents.has(componentName)) {\n return [];\n }\n\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Checking if component \"${componentName}\" is allowed...`);\n console.log(` \u2192 FF-ID: ${this.ffId}`);\n }\n\n // Check if component is allowed based on plan\n const isAllowed = await this.isComponentAllowed(componentName);\n if (!isAllowed) {\n const errorMsg = `Component \"${componentName}\" is not available in your trial plan. Please upgrade to access all components.`;\n console.error(`\u274C ${errorMsg}`);\n throw new Error(errorMsg);\n }\n\n console.log(`\uD83D\uDCE5 Downloading ${componentName}...`);\n\n // Try to fetch from custom components first if ff-id is available\n let componentData;\n let isCustom = false;\n \n if (this.ffId) {\n const customUrl = `${REGISTRY_URL}/custom/${this.ffId}/${this.framework}/${componentName}.json`;\n try {\n componentData = await this.fetchJson(customUrl);\n isCustom = true;\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Found custom component for ${componentName}`);\n }\n } catch (error) {\n // Custom component not found, try standard registry\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 No custom component found for ${componentName}, trying standard`);\n }\n }\n }\n \n // If not found in custom, try standard registry\n if (!componentData) {\n const url = `${REGISTRY_URL}/components/${this.framework}/${componentName}.json`;\n \n try {\n componentData = await this.fetchJson(url);\n } catch (error) {\n // Check if it's a 404 (component not found)\n if (error.message.includes('404')) {\n throw new Error(`Component \"${componentName}\" not found`);\n }\n throw new Error(`Failed to fetch ${componentName}: ${error.message}`);\n }\n }\n\n // Validate component data\n if (!componentData.files || !Array.isArray(componentData.files)) {\n throw new Error(`Invalid component data for ${componentName}`);\n }\n\n // Write component files\n for (const file of componentData.files) {\n await this.writeComponentFile(file);\n }\n\n this.downloadedComponents.add(componentName);\n console.log(` \u2713 ${componentName} downloaded${isCustom && process.env.FF_DEBUG ? ' (custom)' : ''}`);\n\n // Download registry dependencies\n const allDependencies = [...(componentData.dependencies || [])];\n \n if (componentData.registryDependencies && componentData.registryDependencies.length > 0) {\n console.log(` \uD83D\uDCE6 Installing registry dependencies: ${componentData.registryDependencies.join(', ')}`);\n \n for (const dep of componentData.registryDependencies) {\n try {\n const depDeps = await this.downloadComponent(dep);\n depDeps.forEach(d => allDependencies.push(d));\n } catch (error) {\n console.warn(` \u26A0\uFE0F Failed to download dependency ${dep}: ${error.message}`);\n }\n }\n }\n\n return allDependencies;\n }\n\n /**\n * Write component file to disk\n */\n async writeComponentFile(file) {\n // Normalize the file path for the current platform\n // Convert forward slashes to the platform-specific separator\n const normalizedFileName = file.name.split('/').join(path.sep);\n const filePath = path.join(this.outputPath, normalizedFileName);\n const dir = path.dirname(filePath);\n\n // Create directory if it doesn't exist\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n // Check if file exists and overwrite setting\n if (fs.existsSync(filePath) && !this.overwrite) {\n console.log(` \u26A0\uFE0F Skipping ${file.name} (already exists)`);\n return;\n }\n\n // Normalize line endings to match the platform\n // This prevents git from showing every line as changed\n let normalizedContent = file.content;\n if (os.EOL !== '\\n') {\n // On Windows, replace LF with CRLF\n normalizedContent = file.content.replace(/\\n/g, os.EOL);\n }\n\n // Write file\n fs.writeFileSync(filePath, normalizedContent, 'utf8');\n }\n}\n\nmodule.exports = { ComponentDownloader };"],
5
- "mappings": "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",
6
- "names": ["require_constants", "__commonJSMin", "exports", "module", "DEFAULT_API_URL", "LEGACY_API_URL", "CACHE_DIR_NAME", "CACHE_FILES", "ENV_VARS", "require_errors", "__commonJSMin", "exports", "module", "APIError", "message", "statusCode", "url", "CacheError", "operation", "ConfigError", "field", "ProcessingError", "token", "require_file_utils", "__commonJSMin", "exports", "module", "fs", "readJsonFileSafe", "filepath", "content", "error", "readFileSafe", "writeJsonFile", "data", "indent", "writeModuleExportsFile", "fileExists", "ensureDirectoryExists", "dirpath", "removeDirectory", "require_path_utils", "__commonJSMin", "exports", "module", "path", "fileExists", "findFileUpward", "filename", "startDir", "currentDir", "filePath", "findDirectoryUpward", "dirname", "validator", "possiblePaths", "maxLevels", "i", "parentDir", "dirPath", "findDirectoryWithPaths", "absolutePaths", "p", "require_local_token_reader", "__commonJSMin", "exports", "module", "path", "findDirectoryUpward", "readJsonFileSafe", "readFileSafe", "fileExists", "LocalTokenReader", "ffId", "options", "tokensPath", "dir", "folderMapPath", "folderMap", "figmaPath", "tokenPaths", "results", "key", "paths", "tokenPath", "content", "cachePath", "cacheFiles", "processedData", "fileName", "filePath", "val", "require_api_client", "__commonJSMin", "exports", "module", "https", "http", "URL", "DEFAULT_API_URL", "LEGACY_API_URL", "ENV_VARS", "APIError", "LocalTokenReader", "APIClient", "ffId", "options", "url", "resolve", "reject", "response", "data", "chunk", "parsed", "error", "designSystemUrl", "tokensData", "tokens", "err", "legacyBaseURL", "urls", "results", "processedUrl", "tokensUrl", "https", "fs", "path", "URL", "os", "REGISTRY_URL", "ComponentDownloader", "options", "packageJsonPath", "packageJson", "deps", "cleanPath", "url", "possiblePaths", "registryPath", "p", "fullPath", "urlPath", "localPath", "content", "error", "resolve", "reject", "response", "data", "chunk", "components", "comp", "index", "results", "component", "dep", "componentName", "APIClient", "apiClient", "errorMsg", "componentData", "isCustom", "customUrl", "file", "allDependencies", "d", "normalizedFileName", "filePath", "dir", "normalizedContent"]
4
+ "sourcesContent": ["/**\n * Configuration constants for FrontFriend Tailwind v2\n */\n\n// API Configuration\nconst DEFAULT_API_URL = 'https://app.frontfriend.dev';\nconst LEGACY_API_URL = 'https://tokens-studio-donux.up.railway.app/api';\n\n// Cache Configuration\nconst CACHE_DIR_NAME = '.cache/frontfriend';\nconst CACHE_TTL_DAYS = 7;\nconst CACHE_TTL_MS = CACHE_TTL_DAYS * 24 * 60 * 60 * 1000;\n\n// File names\nconst CACHE_FILES = {\n JS: [\n 'tokens.js',\n 'variables.js',\n 'semanticVariables.js',\n 'semanticDarkVariables.js',\n 'safelist.js',\n 'custom.js'\n ],\n JSON: [\n 'fonts.json',\n 'icons.json',\n 'components-config.json',\n 'version.json',\n 'metadata.json'\n ]\n};\n\n// Environment variables\nconst ENV_VARS = {\n FF_ID: 'FF_ID',\n FF_API_URL: 'FF_API_URL',\n FF_USE_LOCAL: 'FF_USE_LOCAL'\n};\n\nmodule.exports = {\n DEFAULT_API_URL,\n LEGACY_API_URL,\n CACHE_DIR_NAME,\n CACHE_TTL_DAYS,\n CACHE_TTL_MS,\n CACHE_FILES,\n ENV_VARS\n};", "class APIError extends Error {\n constructor(message, statusCode, url) {\n super(message);\n this.name = 'APIError';\n this.statusCode = statusCode;\n this.url = url;\n this.code = `API_${statusCode}`;\n }\n}\n\nclass CacheError extends Error {\n constructor(message, operation) {\n super(message);\n this.name = 'CacheError';\n this.operation = operation;\n this.code = `CACHE_${operation.toUpperCase()}`;\n }\n}\n\nclass ConfigError extends Error {\n constructor(message, field) {\n super(message);\n this.name = 'ConfigError';\n this.field = field;\n this.code = `CONFIG_${field.toUpperCase()}`;\n }\n}\n\nclass ProcessingError extends Error {\n constructor(message, token) {\n super(message);\n this.name = 'ProcessingError';\n this.token = token;\n this.code = 'PROCESSING_ERROR';\n }\n}\n\nmodule.exports = {\n APIError,\n CacheError,\n ConfigError,\n ProcessingError\n};", "const fs = require('fs');\n\n/**\n * Read a JSON file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the JSON file\n * @returns {Object|null} Parsed JSON or null if file not found\n * @throws {Error} If JSON is invalid\n */\nfunction readJsonFileSafe(filepath) {\n try {\n const content = fs.readFileSync(filepath, 'utf8');\n return JSON.parse(content);\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found, return null like API would return 404\n }\n throw error;\n }\n}\n\n/**\n * Read a file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the file\n * @returns {string|null} File content or null if file not found\n * @throws {Error} For other file system errors\n */\nfunction readFileSafe(filepath) {\n try {\n return fs.readFileSync(filepath, 'utf8');\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found\n }\n throw error;\n }\n}\n\n/**\n * Write JSON data to a file with proper formatting\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to write\n * @param {number} indent - Indentation level (default: 2)\n */\nfunction writeJsonFile(filepath, data, indent = 2) {\n const content = JSON.stringify(data, null, indent);\n fs.writeFileSync(filepath, content, 'utf8');\n}\n\n/**\n * Write module.exports file with JSON data\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to export\n */\nfunction writeModuleExportsFile(filepath, data) {\n // Handle empty data cases\n if (data === null || data === undefined || \n (typeof data === 'object' && Object.keys(data).length === 0) ||\n (Array.isArray(data) && data.length === 0)) {\n const emptyContent = Array.isArray(data) ? '[]' : '{}';\n const content = `module.exports = ${emptyContent};`;\n fs.writeFileSync(filepath, content, 'utf8');\n } else {\n const content = `module.exports = ${JSON.stringify(data, null, 2)};`;\n fs.writeFileSync(filepath, content, 'utf8');\n }\n}\n\n/**\n * Check if a file exists\n * @param {string} filepath - Path to check\n * @returns {boolean} True if file exists\n */\nfunction fileExists(filepath) {\n return fs.existsSync(filepath);\n}\n\n/**\n * Create directory recursively\n * @param {string} dirpath - Directory path to create\n */\nfunction ensureDirectoryExists(dirpath) {\n fs.mkdirSync(dirpath, { recursive: true });\n}\n\n/**\n * Remove directory recursively\n * @param {string} dirpath - Directory path to remove\n */\nfunction removeDirectory(dirpath) {\n fs.rmSync(dirpath, { recursive: true, force: true });\n}\n\nmodule.exports = {\n readJsonFileSafe,\n readFileSafe,\n writeJsonFile,\n writeModuleExportsFile,\n fileExists,\n ensureDirectoryExists,\n removeDirectory\n};", "const path = require('path');\nconst { fileExists } = require('./file-utils');\n\n/**\n * Find a file by traversing up the directory tree\n * @param {string} filename - Name of the file to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the file or null if not found\n */\nfunction findFileUpward(filename, startDir = process.cwd()) {\n let currentDir = startDir;\n \n while (currentDir !== path.parse(currentDir).root) {\n const filePath = path.join(currentDir, filename);\n \n if (fileExists(filePath)) {\n return filePath;\n }\n \n currentDir = path.dirname(currentDir);\n }\n \n return null;\n}\n\n/**\n * Find a directory by traversing up and checking multiple possible paths\n * @param {string} dirname - Name of the directory to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @param {Function} validator - Optional function to validate the directory\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryUpward(dirname, startDir = process.cwd(), validator = null) {\n const possiblePaths = [];\n let currentDir = startDir;\n const maxLevels = 10; // Prevent infinite loop\n \n // Build list of possible paths\n for (let i = 0; i < maxLevels; i++) {\n // Direct path\n possiblePaths.push(path.join(currentDir, dirname));\n \n // Relative paths up to 3 levels\n if (i < 3) {\n possiblePaths.push(path.join(currentDir, '../'.repeat(i + 1) + dirname));\n }\n \n const parentDir = path.dirname(currentDir);\n if (parentDir === currentDir) break; // Reached root\n currentDir = parentDir;\n }\n \n // Also check from module directory (for when running from node_modules)\n possiblePaths.push(path.join(__dirname, '../../../', dirname));\n \n // Find first existing directory that passes validation\n for (const dirPath of possiblePaths) {\n if (fileExists(dirPath)) {\n if (!validator || validator(dirPath)) {\n return dirPath;\n }\n }\n }\n \n return null;\n}\n\n/**\n * Find a directory with multiple possible relative paths\n * @param {string[]} possiblePaths - Array of possible paths to check\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryWithPaths(possiblePaths, startDir = process.cwd()) {\n // Build absolute paths from the start directory\n const absolutePaths = possiblePaths.map(p => {\n if (path.isAbsolute(p)) {\n return p;\n }\n return path.join(startDir, p);\n });\n \n // Find first existing path\n for (const dirPath of absolutePaths) {\n if (fileExists(dirPath)) {\n return dirPath;\n }\n }\n \n return null;\n}\n\nmodule.exports = {\n findFileUpward,\n findDirectoryUpward,\n findDirectoryWithPaths\n};", "const path = require('path');\nconst { findDirectoryUpward } = require('./path-utils');\nconst { readJsonFileSafe, readFileSafe, fileExists } = require('./file-utils');\n\n/**\n * LocalTokenReader - Reads tokens directly from the local file system\n * Mimics the APIClient interface but reads from apps/tokens directory\n */\nclass LocalTokenReader {\n constructor(ffId, options = {}) {\n this.ffId = ffId;\n this.tokensBasePath = options.tokensBasePath || this.findTokensPath();\n this.folderMap = this.loadFolderMap();\n this.projectFolder = this.folderMap[ffId];\n \n if (!this.projectFolder) {\n throw new Error(`No folder mapping found for ff-id: ${ffId}`);\n }\n \n this.projectPath = path.join(this.tokensBasePath, this.projectFolder);\n }\n\n /**\n * Find the tokens directory by looking for apps/tokens in parent directories\n */\n findTokensPath() {\n // Try to find apps/tokens first\n let tokensPath = findDirectoryUpward('apps/tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n \n // If not found, try just 'tokens'\n if (!tokensPath) {\n tokensPath = findDirectoryUpward('tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n }\n \n if (!tokensPath) {\n throw new Error('Could not find tokens directory with folderMap.json');\n }\n \n return tokensPath;\n }\n\n /**\n * Load the folder mapping from folderMap.json\n */\n loadFolderMap() {\n const folderMapPath = path.join(this.tokensBasePath, 'folderMap.json');\n const folderMap = readJsonFileSafe(folderMapPath);\n \n if (!folderMap) {\n throw new Error(`Failed to load folderMap.json: File not found`);\n }\n \n return folderMap;\n }\n\n /**\n * Fetch all token files (global, colors, semantic)\n * Mimics the APIClient.fetchTokens() method\n */\n async fetchTokens() {\n const figmaPath = path.join(this.projectPath, 'figma');\n \n // Try different possible paths for tokens\n const tokenPaths = {\n global: [\n path.join(figmaPath, 'Global Tokens', 'Default.json'),\n path.join(figmaPath, 'global.json')\n ],\n colors: [\n path.join(figmaPath, 'Global Colors', 'Default.json'),\n path.join(figmaPath, 'Global Colors', 'Light.json')\n ],\n semantic: [\n path.join(figmaPath, 'Semantic', 'Light.json'),\n path.join(figmaPath, 'Semantic', 'Default.json')\n ],\n semanticDark: [\n path.join(figmaPath, 'Semantic', 'Dark.json')\n ]\n };\n\n const results = {};\n \n // Try each possible path for each token type\n for (const [key, paths] of Object.entries(tokenPaths)) {\n for (const tokenPath of paths) {\n const content = readJsonFileSafe(tokenPath);\n if (content !== null) {\n results[key] = content;\n break; // Found it, no need to try other paths\n }\n }\n // If not found in any path, set to null\n if (!results[key]) {\n results[key] = null;\n }\n }\n\n return results;\n }\n\n /**\n * Fetch pre-processed tokens (if available in cache folder)\n */\n async fetchProcessedTokens() {\n const cachePath = path.join(this.projectPath, 'cache');\n \n if (!fileExists(cachePath)) {\n return null;\n }\n\n // Read all the cache files that would be in processed tokens\n const cacheFiles = {\n 'tokens.js': 'hashedTokens.js',\n 'variables.js': 'hashedVariables.js', \n 'semanticVariables.js': 'hashedSemanticVariables.js',\n 'semanticDarkVariables.js': 'hashedSemanticDarkVariables.js',\n 'safelist.js': 'safelist.js',\n 'custom.js': 'custom.js',\n 'fonts.json': 'fonts.json',\n 'icons.json': 'icons.json',\n 'components-config.json': 'encoded-config.json',\n 'version.json': 'version.json'\n };\n\n const processedData = {};\n \n for (const [key, fileName] of Object.entries(cacheFiles)) {\n const filePath = path.join(cachePath, fileName);\n if (key.endsWith('.js')) {\n processedData[key] = readFileSafe(filePath);\n } else {\n processedData[key] = readJsonFileSafe(filePath);\n }\n }\n\n // Only return if we have at least some data\n const hasData = Object.values(processedData).some(val => val !== null);\n return hasData ? processedData : null;\n }\n\n /**\n * Fetch components configuration\n */\n async fetchComponentsConfig() {\n return readJsonFileSafe(path.join(this.projectPath, 'components-config.json'));\n }\n\n /**\n * Fetch fonts configuration\n */\n async fetchFonts() {\n return readJsonFileSafe(path.join(this.projectPath, 'font.json'));\n }\n\n /**\n * Fetch icons configuration\n */\n async fetchIcons() {\n return readJsonFileSafe(path.join(this.projectPath, 'icons.json'));\n }\n\n /**\n * Fetch version information\n */\n async fetchVersion() {\n return readJsonFileSafe(path.join(this.projectPath, 'version.json'));\n }\n\n /**\n * Fetch custom CSS\n */\n async fetchCustomCss() {\n return readFileSafe(path.join(this.projectPath, 'custom.css'));\n }\n\n /**\n * Fetch plan metadata\n */\n async fetchPlanMetadata() {\n const metadata = readJsonFileSafe(path.join(this.projectPath, 'metadata.json'));\n \n // Default to full plan if no metadata exists\n return metadata || {\n planType: 'full',\n allowedComponents: null\n };\n }\n\n // Compatibility methods to match APIClient interface\n fetchJson() {\n throw new Error('fetchJson is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n\n fetchRaw() {\n throw new Error('fetchRaw is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n}\n\nmodule.exports = { LocalTokenReader };", "const https = require('https');\nconst http = require('http');\nconst { URL } = require('url');\nconst { DEFAULT_API_URL, LEGACY_API_URL, ENV_VARS } = require('./constants');\nconst { APIError } = require('./errors');\nconst { LocalTokenReader } = require('./local-token-reader');\n\nclass APIClient {\n constructor(ffId, options = {}) {\n this.ffId = ffId;\n this.baseURL = options.baseURL || process.env[ENV_VARS.FF_API_URL] || DEFAULT_API_URL;\n \n // Check if we should use local token reader\n if (process.env[ENV_VARS.FF_USE_LOCAL] === 'true') {\n console.log(' \uD83C\uDFE0 Using local token reader');\n this.localReader = new LocalTokenReader(ffId, options);\n }\n }\n\n /**\n * Fetch JSON data from a URL using native https module\n * @param {string} url - The URL to fetch\n * @returns {Promise<any>} Parsed JSON data\n */\n fetchJson(url) {\n return new Promise((resolve, reject) => {\n // Determine if we should use http or https\n const protocol = url.startsWith('https') ? https : http;\n \n protocol.get(url, (response) => {\n let data = '';\n \n // Check for HTTP errors\n if (response.statusCode >= 400) {\n reject(new APIError(\n `HTTP ${response.statusCode}: ${response.statusMessage}`,\n response.statusCode,\n url\n ));\n return;\n }\n\n // Collect response chunks\n response.on('data', (chunk) => {\n data += chunk;\n });\n\n // Parse JSON when complete\n response.on('end', () => {\n try {\n const parsed = JSON.parse(data);\n resolve(parsed);\n } catch (error) {\n reject(new APIError(\n `Invalid JSON response: ${error.message}`,\n response.statusCode,\n url\n ));\n }\n });\n\n // Handle response errors\n response.on('error', (error) => {\n reject(new APIError(\n `Response error: ${error.message}`,\n response.statusCode,\n url\n ));\n });\n }).on('error', (error) => {\n // Handle network errors\n reject(new APIError(\n `Network error: ${error.message}`,\n 0,\n url\n ));\n });\n });\n }\n\n /**\n * Fetch raw content from a URL\n * @param {string} url - The URL to fetch\n * @returns {Promise<string>} Raw text content\n */\n fetchRaw(url) {\n return new Promise((resolve, reject) => {\n // Determine if we should use http or https\n const protocol = url.startsWith('https') ? https : http;\n \n protocol.get(url, (response) => {\n let data = '';\n \n // Check for HTTP errors\n if (response.statusCode >= 400) {\n reject(new APIError(\n `HTTP ${response.statusCode}: ${response.statusMessage}`,\n response.statusCode,\n url\n ));\n return;\n }\n\n // Collect response chunks\n response.on('data', (chunk) => {\n data += chunk;\n });\n\n // Return raw data when complete\n response.on('end', () => {\n resolve(data);\n });\n\n // Handle response errors\n response.on('error', (error) => {\n reject(new APIError(\n `Response error: ${error.message}`,\n response.statusCode,\n url\n ));\n });\n }).on('error', (error) => {\n // Handle network errors\n reject(new APIError(\n `Network error: ${error.message}`,\n 0,\n url\n ));\n });\n });\n }\n\n /**\n * Fetch all token files (global, colors, semantic)\n * @returns {Promise<Object>} Object with all token data\n */\n async fetchTokens() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchTokens();\n }\n \n // First try the new design system endpoint\n try {\n const designSystemUrl = `${this.baseURL}/api/design-systems/${this.ffId}/tokens`;\n const tokensData = await this.fetchJson(designSystemUrl);\n \n // If we have the new format, parse it\n if (tokensData && tokensData.tokens) {\n const tokens = tokensData.tokens;\n \n // Extract specific token sets from the new format\n return {\n global: tokens['Global Tokens/Default'] || tokens['global'] || null,\n colors: tokens['Global Colors/Default'] || tokens['colors'] || null,\n semantic: tokens['Semantic/Light'] || tokens['semantic/light'] || null,\n semanticDark: tokens['Semantic/Dark'] || tokens['semantic/dark'] || null\n };\n }\n } catch (err) {\n // If design system is not synced (400 error), throw with clear message\n if (err.statusCode === 400) {\n throw new APIError(\n 'Design system not synced with Figma. Please sync tokens from Figma before using this design system.',\n 400,\n `${this.baseURL}/api/design-systems/${this.ffId}/tokens`\n );\n }\n // Failed to fetch from design system endpoint, falling back to legacy URLs\n }\n \n // Fall back to legacy token URLs\n // Use legacy API URL for backward compatibility\n const legacyBaseURL = LEGACY_API_URL;\n const urls = {\n global: `${legacyBaseURL}/${this.ffId}/global-tokens/default.json`,\n colors: `${legacyBaseURL}/${this.ffId}/global-colors/default.json`,\n semantic: `${legacyBaseURL}/${this.ffId}/semantic/light.json`,\n semanticDark: `${legacyBaseURL}/${this.ffId}/semantic/dark.json`\n };\n\n const results = await Promise.all([\n this.fetchJson(urls.global).catch(err => err.statusCode === 404 ? null : Promise.reject(err)),\n this.fetchJson(urls.colors).catch(err => err.statusCode === 404 ? null : Promise.reject(err)),\n this.fetchJson(urls.semantic).catch(err => err.statusCode === 404 ? null : Promise.reject(err)),\n this.fetchJson(urls.semanticDark).catch(err => err.statusCode === 404 ? null : Promise.reject(err))\n ]);\n\n return {\n global: results[0],\n colors: results[1],\n semantic: results[2],\n semanticDark: results[3]\n };\n }\n\n /**\n * Fetch pre-processed tokens from the design system endpoint\n * @returns {Promise<Object|null>} Processed tokens ready for caching or null if not found\n */\n async fetchProcessedTokens() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchProcessedTokens();\n }\n \n const processedUrl = `${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;\n try {\n return await this.fetchJson(processedUrl);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n // If design system is not synced (400 error), throw with clear message\n if (error.statusCode === 400) {\n throw new APIError(\n 'Design system not synced with Figma. Please sync tokens from Figma before using this design system.',\n 400,\n processedUrl\n );\n }\n throw error;\n }\n }\n\n /**\n * Fetch components configuration\n * @returns {Promise<Object|null>} Components config or null if not found\n */\n async fetchComponentsConfig() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchComponentsConfig();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/components-config.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch fonts configuration\n * @returns {Promise<Object|null>} Fonts object with font1, font2, etc. URLs or null\n */\n async fetchFonts() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchFonts();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/font.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch icons configuration\n * @returns {Promise<Object|null>} Icons data or null if not found\n */\n async fetchIcons() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchIcons();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/icons.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch version information\n * @returns {Promise<Object|null>} Version data or null if not found\n */\n async fetchVersion() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchVersion();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/version.json`;\n try {\n return await this.fetchJson(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch custom CSS\n * @returns {Promise<string|null>} Custom CSS string or null if not found\n */\n async fetchCustomCss() {\n // Use local reader if available\n if (this.localReader) {\n return this.localReader.fetchCustomCss();\n }\n \n // Always use legacy URL for these endpoints\n const url = `${LEGACY_API_URL}/${this.ffId}/custom.css`;\n try {\n return await this.fetchRaw(url);\n } catch (error) {\n if (error.statusCode === 404) {\n return null;\n }\n throw error;\n }\n }\n\n /**\n * Fetch plan metadata (plan type and allowed components)\n * @returns {Promise<Object|null>} Plan metadata or null if not found\n */\n async fetchPlanMetadata() {\n // Use local reader if available\n if (this.localReader) {\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 Using local reader for plan metadata');\n }\n return this.localReader.fetchPlanMetadata();\n }\n \n // First try the saas app (processed-tokens endpoint) - this is the authoritative source\n const processedUrl = `${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;\n try {\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetching plan metadata from saas app: ${processedUrl}`);\n }\n \n const processedData = await this.fetchJson(processedUrl);\n \n // Extract metadata from processed tokens response\n if (processedData && processedData.metadata) {\n const metadata = {\n planType: processedData.metadata.planType || 'full',\n allowedComponents: processedData.metadata.allowedComponents || null\n };\n \n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Found plan metadata in saas app: ${metadata.planType}, allowed: ${metadata.allowedComponents ? metadata.allowedComponents.length + ' components' : 'all'}`);\n }\n \n return metadata;\n }\n } catch (saasError) {\n // If not found in saas app, try legacy server\n if (saasError.statusCode === 404) {\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 ff-id not found in saas app, trying legacy server');\n }\n \n // Try legacy tokens server (these are always full plan - no trial concept)\n const tokensUrl = `${LEGACY_API_URL}/${this.ffId}/metadata`;\n try {\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetching from legacy server: ${tokensUrl}`);\n }\n const legacyMetadata = await this.fetchJson(tokensUrl);\n \n // Use the plan metadata from legacy server if available\n const metadata = {\n planType: legacyMetadata.planType || 'full',\n allowedComponents: legacyMetadata.allowedComponents || null\n };\n \n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Found in legacy server: ${metadata.planType}, allowed: ${metadata.allowedComponents ? metadata.allowedComponents.length + ' components' : 'all'}`);\n }\n \n return metadata;\n } catch (legacyError) {\n if (legacyError.statusCode === 404) {\n // Not found in either place - DENY access\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 ff-id not found in any server - access denied');\n }\n \n // Return a denied plan type instead of defaulting to full\n return {\n planType: 'denied',\n allowedComponents: [],\n error: 'Design system not found. Please ensure your ff-id is valid.'\n };\n }\n throw legacyError;\n }\n }\n throw saasError;\n }\n }\n}\n\nmodule.exports = { APIClient };", "const https = require('https');\nconst fs = require('fs');\nconst path = require('path');\nconst { URL } = require('url');\nconst os = require('os');\n\nconst REGISTRY_URL = process.env.FF_REGISTRY_URL || 'https://ff-registry.up.railway.app';\n\nclass ComponentDownloader {\n constructor(options = {}) {\n this.appRoot = options.appRoot || process.cwd();\n this.framework = options.framework || this.detectFramework();\n this.config = options.config || {};\n this.outputPath = options.outputPath || this.getDefaultOutputPath();\n this.overwrite = options.overwrite || false;\n this.downloadedComponents = new Set();\n this.ffId = this.config.ffId || this.config['ff-id'] || null;\n this.planMetadata = null; // Will be populated when needed\n }\n\n /**\n * Detect framework from package.json\n */\n detectFramework() {\n const packageJsonPath = path.join(this.appRoot, 'package.json');\n \n if (fs.existsSync(packageJsonPath)) {\n const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));\n const deps = { ...packageJson.dependencies, ...packageJson.devDependencies };\n \n if (deps.vue) {\n return 'vue';\n } else if (deps.react) {\n return 'react';\n }\n }\n \n // Default to React if can't detect\n return 'react';\n }\n\n /**\n * Get default output path based on framework and config\n */\n getDefaultOutputPath() {\n // Check if config has ui alias\n if (this.config.aliases && this.config.aliases.ui) {\n const uiAlias = this.config.aliases.ui;\n // Convert alias to actual path (e.g., @/components/ui -> components/ui)\n const cleanPath = uiAlias.replace(/^@\\//, '');\n return path.join(this.appRoot, cleanPath);\n }\n \n // Fallback to default paths\n if (this.framework === 'vue') {\n return path.join(this.appRoot, 'src/components/ui');\n }\n return path.join(this.appRoot, 'components/ui');\n }\n\n /**\n * Fetch JSON from URL or local file\n */\n fetchJson(url) {\n // Check if using local mode at runtime\n const USE_LOCAL = process.env.FF_USE_LOCAL === 'true';\n \n // If using local mode, read from file system\n if (USE_LOCAL) {\n // Try multiple paths to find the registry\n const possiblePaths = [\n '../../packages/registry',\n '../../../packages/registry',\n '../../../../packages/registry',\n path.join(__dirname, '../../../registry'),\n 'packages/registry'\n ];\n \n let registryPath;\n for (const p of possiblePaths) {\n const fullPath = path.isAbsolute(p) ? p : path.join(process.cwd(), p);\n if (fs.existsSync(fullPath)) {\n registryPath = fullPath;\n break;\n }\n }\n \n if (!registryPath) {\n return Promise.reject(new Error(`Local registry not found. Tried paths: ${possiblePaths.join(', ')}`));\n }\n \n // Extract the path from the URL and construct proper local path\n const urlPath = url.replace(REGISTRY_URL, '').replace(/^\\//, '');\n const localPath = path.join(registryPath, ...urlPath.split('/'));\n \n try {\n const content = fs.readFileSync(localPath, 'utf8');\n return Promise.resolve(JSON.parse(content));\n } catch (error) {\n if (error.code === 'ENOENT') {\n return Promise.reject(new Error('HTTP 404: Not Found'));\n }\n return Promise.reject(error);\n }\n }\n\n return new Promise((resolve, reject) => {\n https.get(url, (response) => {\n let data = '';\n \n if (response.statusCode >= 400) {\n reject(new Error(`HTTP ${response.statusCode}: ${response.statusMessage}`));\n return;\n }\n\n response.on('data', (chunk) => {\n data += chunk;\n });\n\n response.on('end', () => {\n try {\n resolve(JSON.parse(data));\n } catch (error) {\n reject(new Error(`Invalid JSON response: ${error.message}`));\n }\n });\n\n }).on('error', (error) => {\n reject(error);\n });\n });\n }\n\n /**\n * Get all available components from registry\n */\n async getAllComponents() {\n try {\n const url = `${REGISTRY_URL}/components/${this.framework}/index.json`;\n const index = await this.fetchJson(url);\n let components = index.components || [];\n \n // Filter components based on plan\n if (this.ffId) {\n // Ensure plan metadata is loaded\n try {\n await this.isComponentAllowed('dummy'); // This will load metadata\n } catch (error) {\n // If access is denied, throw error\n if (this.planMetadata && this.planMetadata.planType === 'denied') {\n throw new Error(this.planMetadata.error || 'Access denied. Your ff-id was not found in any server.');\n }\n }\n \n if (this.planMetadata && this.planMetadata.planType === 'trial' && this.planMetadata.allowedComponents) {\n components = components.filter(comp => this.planMetadata.allowedComponents.includes(comp));\n \n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Trial plan: filtering to ${components.length} allowed components`);\n }\n }\n }\n \n return components;\n } catch (error) {\n console.error('Failed to fetch component index:', error.message);\n return [];\n }\n }\n\n /**\n * Get all custom components for this ff-id\n */\n async getCustomComponents() {\n if (!this.ffId) {\n if (process.env.FF_DEBUG) {\n console.log(' \u2192 No ff-id found, skipping custom components');\n }\n return [];\n }\n \n try {\n const url = `${REGISTRY_URL}/custom/${this.ffId}/${this.framework}/index.json`;\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetching custom components from: ${url}`);\n }\n const index = await this.fetchJson(url);\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Found custom components: ${JSON.stringify(index.components)}`);\n }\n return index.components || [];\n } catch (error) {\n // Custom components might not exist, that's ok\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 No custom components found: ${error.message}`);\n }\n return [];\n }\n }\n\n /**\n * Download multiple components\n */\n async downloadComponents(components) {\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 ComponentDownloader initialized with ff-id: ${this.ffId}`);\n }\n \n const results = {\n successful: [],\n failed: [],\n dependencies: new Set()\n };\n\n for (const component of components) {\n try {\n const deps = await this.downloadComponent(component);\n results.successful.push(component);\n \n // Add dependencies\n deps.forEach(dep => results.dependencies.add(dep));\n } catch (error) {\n results.failed.push({\n component,\n error: error.message\n });\n }\n }\n\n return {\n successful: results.successful,\n failed: results.failed,\n dependencies: Array.from(results.dependencies)\n };\n }\n\n /**\n * Check if component is allowed based on plan\n */\n async isComponentAllowed(componentName) {\n // Fetch plan metadata if not already loaded\n if (!this.planMetadata && this.ffId) {\n const { APIClient } = require('./api-client');\n const apiClient = new APIClient(this.ffId, {\n baseURL: this.config['api-url'] || this.config.apiUrl\n });\n try {\n this.planMetadata = await apiClient.fetchPlanMetadata();\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Fetched plan metadata: ${this.planMetadata.planType}, allowed: ${this.planMetadata.allowedComponents ? this.planMetadata.allowedComponents.length + ' components' : 'all'}`);\n }\n } catch (error) {\n console.warn('Failed to fetch plan metadata:', error.message);\n // Default to full plan if we can't fetch metadata\n this.planMetadata = { planType: 'full', allowedComponents: null };\n }\n }\n\n // Handle denied plan (ff-id not found anywhere)\n if (this.planMetadata && this.planMetadata.planType === 'denied') {\n throw new Error(this.planMetadata.error || 'Access denied. Your ff-id was not found in any server.');\n }\n \n // If full plan or no metadata, allow all components\n if (!this.planMetadata || this.planMetadata.planType === 'full') {\n return true;\n }\n\n // For trial plan, check if component is in allowed list\n if (this.planMetadata.planType === 'trial' && this.planMetadata.allowedComponents) {\n return this.planMetadata.allowedComponents.includes(componentName);\n }\n\n // Default to denying if we can't determine (safer)\n return false;\n }\n\n /**\n * Download a single component and its registry dependencies\n */\n async downloadComponent(componentName) {\n // Avoid downloading the same component twice\n if (this.downloadedComponents.has(componentName)) {\n return [];\n }\n\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Checking if component \"${componentName}\" is allowed...`);\n console.log(` \u2192 FF-ID: ${this.ffId}`);\n }\n\n // Check if component is allowed based on plan\n const isAllowed = await this.isComponentAllowed(componentName);\n if (!isAllowed) {\n const errorMsg = `Component \"${componentName}\" is not available in your trial plan. Please upgrade to access all components.`;\n console.error(`\u274C ${errorMsg}`);\n throw new Error(errorMsg);\n }\n\n console.log(`\uD83D\uDCE5 Downloading ${componentName}...`);\n\n // Try to fetch from custom components first if ff-id is available\n let componentData;\n let isCustom = false;\n \n if (this.ffId) {\n const customUrl = `${REGISTRY_URL}/custom/${this.ffId}/${this.framework}/${componentName}.json`;\n try {\n componentData = await this.fetchJson(customUrl);\n isCustom = true;\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 Found custom component for ${componentName}`);\n }\n } catch (error) {\n // Custom component not found, try standard registry\n if (process.env.FF_DEBUG) {\n console.log(` \u2192 No custom component found for ${componentName}, trying standard`);\n }\n }\n }\n \n // If not found in custom, try standard registry\n if (!componentData) {\n const url = `${REGISTRY_URL}/components/${this.framework}/${componentName}.json`;\n \n try {\n componentData = await this.fetchJson(url);\n } catch (error) {\n // Check if it's a 404 (component not found)\n if (error.message.includes('404')) {\n throw new Error(`Component \"${componentName}\" not found`);\n }\n throw new Error(`Failed to fetch ${componentName}: ${error.message}`);\n }\n }\n\n // Validate component data\n if (!componentData.files || !Array.isArray(componentData.files)) {\n throw new Error(`Invalid component data for ${componentName}`);\n }\n\n // Write component files\n for (const file of componentData.files) {\n await this.writeComponentFile(file);\n }\n\n this.downloadedComponents.add(componentName);\n console.log(` \u2713 ${componentName} downloaded${isCustom && process.env.FF_DEBUG ? ' (custom)' : ''}`);\n\n // Download registry dependencies\n const allDependencies = [...(componentData.dependencies || [])];\n \n if (componentData.registryDependencies && componentData.registryDependencies.length > 0) {\n console.log(` \uD83D\uDCE6 Installing registry dependencies: ${componentData.registryDependencies.join(', ')}`);\n \n for (const dep of componentData.registryDependencies) {\n try {\n const depDeps = await this.downloadComponent(dep);\n depDeps.forEach(d => allDependencies.push(d));\n } catch (error) {\n console.warn(` \u26A0\uFE0F Failed to download dependency ${dep}: ${error.message}`);\n }\n }\n }\n\n return allDependencies;\n }\n\n /**\n * Write component file to disk\n */\n async writeComponentFile(file) {\n // Normalize the file path for the current platform\n // Convert forward slashes to the platform-specific separator\n const normalizedFileName = file.name.split('/').join(path.sep);\n const filePath = path.join(this.outputPath, normalizedFileName);\n const dir = path.dirname(filePath);\n\n // Create directory if it doesn't exist\n if (!fs.existsSync(dir)) {\n fs.mkdirSync(dir, { recursive: true });\n }\n\n // Check if file exists and overwrite setting\n if (fs.existsSync(filePath) && !this.overwrite) {\n console.log(` \u26A0\uFE0F Skipping ${file.name} (already exists)`);\n return;\n }\n\n // Normalize line endings to match the platform\n // This prevents git from showing every line as changed\n let normalizedContent = file.content;\n if (os.EOL !== '\\n') {\n // On Windows, replace LF with CRLF\n normalizedContent = file.content.replace(/\\n/g, os.EOL);\n }\n\n // Write file\n fs.writeFileSync(filePath, normalizedContent, 'utf8');\n }\n}\n\nmodule.exports = { ComponentDownloader };"],
5
+ "mappings": "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",
6
+ "names": ["require_constants", "__commonJSMin", "exports", "module", "DEFAULT_API_URL", "LEGACY_API_URL", "CACHE_DIR_NAME", "CACHE_FILES", "ENV_VARS", "require_errors", "__commonJSMin", "exports", "module", "APIError", "message", "statusCode", "url", "CacheError", "operation", "ConfigError", "field", "ProcessingError", "token", "require_file_utils", "__commonJSMin", "exports", "module", "fs", "readJsonFileSafe", "filepath", "content", "error", "readFileSafe", "writeJsonFile", "data", "indent", "writeModuleExportsFile", "fileExists", "ensureDirectoryExists", "dirpath", "removeDirectory", "require_path_utils", "__commonJSMin", "exports", "module", "path", "fileExists", "findFileUpward", "filename", "startDir", "currentDir", "filePath", "findDirectoryUpward", "dirname", "validator", "possiblePaths", "maxLevels", "i", "parentDir", "dirPath", "findDirectoryWithPaths", "absolutePaths", "p", "require_local_token_reader", "__commonJSMin", "exports", "module", "path", "findDirectoryUpward", "readJsonFileSafe", "readFileSafe", "fileExists", "LocalTokenReader", "ffId", "options", "tokensPath", "dir", "folderMapPath", "folderMap", "figmaPath", "tokenPaths", "results", "key", "paths", "tokenPath", "content", "cachePath", "cacheFiles", "processedData", "fileName", "filePath", "val", "require_api_client", "__commonJSMin", "exports", "module", "https", "http", "URL", "DEFAULT_API_URL", "LEGACY_API_URL", "ENV_VARS", "APIError", "LocalTokenReader", "APIClient", "ffId", "options", "url", "resolve", "reject", "response", "data", "chunk", "parsed", "error", "designSystemUrl", "tokensData", "tokens", "err", "legacyBaseURL", "urls", "results", "processedUrl", "processedData", "metadata", "saasError", "tokensUrl", "legacyMetadata", "legacyError", "https", "fs", "path", "URL", "os", "REGISTRY_URL", "ComponentDownloader", "options", "packageJsonPath", "packageJson", "deps", "cleanPath", "url", "possiblePaths", "registryPath", "p", "fullPath", "urlPath", "localPath", "content", "error", "resolve", "reject", "response", "data", "chunk", "components", "comp", "index", "results", "component", "dep", "componentName", "APIClient", "apiClient", "errorMsg", "componentData", "isCustom", "customUrl", "file", "allDependencies", "d", "normalizedFileName", "filePath", "dir", "normalizedContent"]
7
7
  }
@@ -1,2 +1,2 @@
1
- var s="https://app.frontfriend.dev",_="https://tokens-studio-donux.up.railway.app/api",n=".cache/frontfriend";var o={JS:["tokens.js","variables.js","semanticVariables.js","semanticDarkVariables.js","cls.js","custom.js"],JSON:["fonts.json","icons.json","components-config.json","version.json","metadata.json"]},t={FF_ID:"FF_ID",FF_API_URL:"FF_API_URL",FF_USE_LOCAL:"FF_USE_LOCAL"};module.exports={DEFAULT_API_URL:s,LEGACY_API_URL:_,CACHE_DIR_NAME:n,CACHE_TTL_DAYS:7,CACHE_TTL_MS:6048e5,CACHE_FILES:o,ENV_VARS:t};
1
+ var s="https://app.frontfriend.dev",_="https://tokens-studio-donux.up.railway.app/api",n=".cache/frontfriend";var o={JS:["tokens.js","variables.js","semanticVariables.js","semanticDarkVariables.js","safelist.js","custom.js"],JSON:["fonts.json","icons.json","components-config.json","version.json","metadata.json"]},t={FF_ID:"FF_ID",FF_API_URL:"FF_API_URL",FF_USE_LOCAL:"FF_USE_LOCAL"};module.exports={DEFAULT_API_URL:s,LEGACY_API_URL:_,CACHE_DIR_NAME:n,CACHE_TTL_DAYS:7,CACHE_TTL_MS:6048e5,CACHE_FILES:o,ENV_VARS:t};
2
2
  //# sourceMappingURL=constants.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../lib/core/constants.js"],
4
- "sourcesContent": ["/**\n * Configuration constants for FrontFriend Tailwind v2\n */\n\n// API Configuration\nconst DEFAULT_API_URL = 'https://app.frontfriend.dev';\nconst LEGACY_API_URL = 'https://tokens-studio-donux.up.railway.app/api';\n\n// Cache Configuration\nconst CACHE_DIR_NAME = '.cache/frontfriend';\nconst CACHE_TTL_DAYS = 7;\nconst CACHE_TTL_MS = CACHE_TTL_DAYS * 24 * 60 * 60 * 1000;\n\n// File names\nconst CACHE_FILES = {\n JS: [\n 'tokens.js',\n 'variables.js',\n 'semanticVariables.js',\n 'semanticDarkVariables.js',\n 'cls.js',\n 'custom.js'\n ],\n JSON: [\n 'fonts.json',\n 'icons.json',\n 'components-config.json',\n 'version.json',\n 'metadata.json'\n ]\n};\n\n// Environment variables\nconst ENV_VARS = {\n FF_ID: 'FF_ID',\n FF_API_URL: 'FF_API_URL',\n FF_USE_LOCAL: 'FF_USE_LOCAL'\n};\n\nmodule.exports = {\n DEFAULT_API_URL,\n LEGACY_API_URL,\n CACHE_DIR_NAME,\n CACHE_TTL_DAYS,\n CACHE_TTL_MS,\n CACHE_FILES,\n ENV_VARS\n};"],
5
- "mappings": "AAKA,IAAMA,EAAkB,8BAClBC,EAAiB,iDAGjBC,EAAiB,qBAKvB,IAAMC,EAAc,CAClB,GAAI,CACF,YACA,eACA,uBACA,2BACA,SACA,WACF,EACA,KAAM,CACJ,aACA,aACA,yBACA,eACA,eACF,CACF,EAGMC,EAAW,CACf,MAAO,QACP,WAAY,aACZ,aAAc,cAChB,EAEA,OAAO,QAAU,CACf,gBAAAC,EACA,eAAAC,EACA,eAAAC,EACA,iBACA,oBACA,YAAAJ,EACA,SAAAC,CACF",
4
+ "sourcesContent": ["/**\n * Configuration constants for FrontFriend Tailwind v2\n */\n\n// API Configuration\nconst DEFAULT_API_URL = 'https://app.frontfriend.dev';\nconst LEGACY_API_URL = 'https://tokens-studio-donux.up.railway.app/api';\n\n// Cache Configuration\nconst CACHE_DIR_NAME = '.cache/frontfriend';\nconst CACHE_TTL_DAYS = 7;\nconst CACHE_TTL_MS = CACHE_TTL_DAYS * 24 * 60 * 60 * 1000;\n\n// File names\nconst CACHE_FILES = {\n JS: [\n 'tokens.js',\n 'variables.js',\n 'semanticVariables.js',\n 'semanticDarkVariables.js',\n 'safelist.js',\n 'custom.js'\n ],\n JSON: [\n 'fonts.json',\n 'icons.json',\n 'components-config.json',\n 'version.json',\n 'metadata.json'\n ]\n};\n\n// Environment variables\nconst ENV_VARS = {\n FF_ID: 'FF_ID',\n FF_API_URL: 'FF_API_URL',\n FF_USE_LOCAL: 'FF_USE_LOCAL'\n};\n\nmodule.exports = {\n DEFAULT_API_URL,\n LEGACY_API_URL,\n CACHE_DIR_NAME,\n CACHE_TTL_DAYS,\n CACHE_TTL_MS,\n CACHE_FILES,\n ENV_VARS\n};"],
5
+ "mappings": "AAKA,IAAMA,EAAkB,8BAClBC,EAAiB,iDAGjBC,EAAiB,qBAKvB,IAAMC,EAAc,CAClB,GAAI,CACF,YACA,eACA,uBACA,2BACA,cACA,WACF,EACA,KAAM,CACJ,aACA,aACA,yBACA,eACA,eACF,CACF,EAGMC,EAAW,CACf,MAAO,QACP,WAAY,aACZ,aAAc,cAChB,EAEA,OAAO,QAAU,CACf,gBAAAC,EACA,eAAAC,EACA,eAAAC,EACA,iBACA,oBACA,YAAAJ,EACA,SAAAC,CACF",
6
6
  "names": ["DEFAULT_API_URL", "LEGACY_API_URL", "CACHE_DIR_NAME", "CACHE_FILES", "ENV_VARS", "DEFAULT_API_URL", "LEGACY_API_URL", "CACHE_DIR_NAME"]
7
7
  }
@@ -1,2 +1,2 @@
1
- var m=(o,e)=>()=>(e||o((e={exports:{}}).exports,e),e.exports);var u=m((v,y)=>{var f=require("fs");function F(o){try{let e=f.readFileSync(o,"utf8");return JSON.parse(e)}catch(e){if(e.code==="ENOENT")return null;throw e}}function S(o){try{return f.readFileSync(o,"utf8")}catch(e){if(e.code==="ENOENT")return null;throw e}}function D(o,e,t=2){let n=JSON.stringify(e,null,t);f.writeFileSync(o,n,"utf8")}function g(o,e){if(e==null||typeof e=="object"&&Object.keys(e).length===0||Array.isArray(e)&&e.length===0){let n=`module.exports = ${Array.isArray(e)?"[]":"{}"};`;f.writeFileSync(o,n,"utf8")}else{let t=`module.exports = ${JSON.stringify(e,null,2)};`;f.writeFileSync(o,t,"utf8")}}function E(o){return f.existsSync(o)}function M(o){f.mkdirSync(o,{recursive:!0})}function x(o){f.rmSync(o,{recursive:!0,force:!0})}y.exports={readJsonFileSafe:F,readFileSafe:S,writeJsonFile:D,writeModuleExportsFile:g,fileExists:E,ensureDirectoryExists:M,removeDirectory:x}});var P=m((J,w)=>{var a=require("path"),{fileExists:j}=u();function C(o,e=process.cwd()){let t=e;for(;t!==a.parse(t).root;){let n=a.join(t,o);if(j(n))return n;t=a.dirname(t)}return null}function T(o,e=process.cwd(),t=null){let n=[],i=e,c=10;for(let r=0;r<c;r++){n.push(a.join(i,o)),r<3&&n.push(a.join(i,"../".repeat(r+1)+o));let l=a.dirname(i);if(l===i)break;i=l}n.push(a.join(__dirname,"../../../",o));for(let r of n)if(j(r)&&(!t||t(r)))return r;return null}function N(o,e=process.cwd()){let t=o.map(n=>a.isAbsolute(n)?n:a.join(e,n));for(let n of t)if(j(n))return n;return null}w.exports={findFileUpward:C,findDirectoryUpward:T,findDirectoryWithPaths:N}});var s=require("path"),{findDirectoryUpward:k}=P(),{readJsonFileSafe:h,readFileSafe:b,fileExists:d}=u(),p=class{constructor(e,t={}){if(this.ffId=e,this.tokensBasePath=t.tokensBasePath||this.findTokensPath(),this.folderMap=this.loadFolderMap(),this.projectFolder=this.folderMap[e],!this.projectFolder)throw new Error(`No folder mapping found for ff-id: ${e}`);this.projectPath=s.join(this.tokensBasePath,this.projectFolder)}findTokensPath(){let e=k("apps/tokens",process.cwd(),t=>d(s.join(t,"folderMap.json")));if(e||(e=k("tokens",process.cwd(),t=>d(s.join(t,"folderMap.json")))),!e)throw new Error("Could not find tokens directory with folderMap.json");return e}loadFolderMap(){let e=s.join(this.tokensBasePath,"folderMap.json"),t=h(e);if(!t)throw new Error("Failed to load folderMap.json: File not found");return t}async fetchTokens(){let e=s.join(this.projectPath,"figma"),t={global:[s.join(e,"Global Tokens","Default.json"),s.join(e,"global.json")],colors:[s.join(e,"Global Colors","Default.json"),s.join(e,"Global Colors","Light.json")],semantic:[s.join(e,"Semantic","Light.json"),s.join(e,"Semantic","Default.json")],semanticDark:[s.join(e,"Semantic","Dark.json")]},n={};for(let[i,c]of Object.entries(t)){for(let r of c){let l=h(r);if(l!==null){n[i]=l;break}}n[i]||(n[i]=null)}return n}async fetchProcessedTokens(){let e=s.join(this.projectPath,"cache");if(!d(e))return null;let t={"tokens.js":"hashedTokens.js","variables.js":"hashedVariables.js","semanticVariables.js":"hashedSemanticVariables.js","semanticDarkVariables.js":"hashedSemanticDarkVariables.js","cls.js":"cls.js","custom.js":"custom.js","fonts.json":"fonts.json","icons.json":"icons.json","components-config.json":"encoded-config.json","version.json":"version.json"},n={};for(let[c,r]of Object.entries(t)){let l=s.join(e,r);c.endsWith(".js")?n[c]=b(l):n[c]=h(l)}return Object.values(n).some(c=>c!==null)?n:null}async fetchComponentsConfig(){return h(s.join(this.projectPath,"components-config.json"))}async fetchFonts(){return h(s.join(this.projectPath,"font.json"))}async fetchIcons(){return h(s.join(this.projectPath,"icons.json"))}async fetchVersion(){return h(s.join(this.projectPath,"version.json"))}async fetchCustomCss(){return b(s.join(this.projectPath,"custom.css"))}async fetchPlanMetadata(){return h(s.join(this.projectPath,"metadata.json"))||{planType:"full",allowedComponents:null}}fetchJson(){throw new Error("fetchJson is not implemented in LocalTokenReader. Use specific fetch methods.")}fetchRaw(){throw new Error("fetchRaw is not implemented in LocalTokenReader. Use specific fetch methods.")}};module.exports={LocalTokenReader:p};
1
+ var m=(o,e)=>()=>(e||o((e={exports:{}}).exports,e),e.exports);var u=m((v,y)=>{var f=require("fs");function F(o){try{let e=f.readFileSync(o,"utf8");return JSON.parse(e)}catch(e){if(e.code==="ENOENT")return null;throw e}}function S(o){try{return f.readFileSync(o,"utf8")}catch(e){if(e.code==="ENOENT")return null;throw e}}function D(o,e,t=2){let n=JSON.stringify(e,null,t);f.writeFileSync(o,n,"utf8")}function g(o,e){if(e==null||typeof e=="object"&&Object.keys(e).length===0||Array.isArray(e)&&e.length===0){let n=`module.exports = ${Array.isArray(e)?"[]":"{}"};`;f.writeFileSync(o,n,"utf8")}else{let t=`module.exports = ${JSON.stringify(e,null,2)};`;f.writeFileSync(o,t,"utf8")}}function E(o){return f.existsSync(o)}function M(o){f.mkdirSync(o,{recursive:!0})}function x(o){f.rmSync(o,{recursive:!0,force:!0})}y.exports={readJsonFileSafe:F,readFileSafe:S,writeJsonFile:D,writeModuleExportsFile:g,fileExists:E,ensureDirectoryExists:M,removeDirectory:x}});var P=m((J,w)=>{var a=require("path"),{fileExists:j}=u();function C(o,e=process.cwd()){let t=e;for(;t!==a.parse(t).root;){let n=a.join(t,o);if(j(n))return n;t=a.dirname(t)}return null}function T(o,e=process.cwd(),t=null){let n=[],i=e,c=10;for(let r=0;r<c;r++){n.push(a.join(i,o)),r<3&&n.push(a.join(i,"../".repeat(r+1)+o));let l=a.dirname(i);if(l===i)break;i=l}n.push(a.join(__dirname,"../../../",o));for(let r of n)if(j(r)&&(!t||t(r)))return r;return null}function N(o,e=process.cwd()){let t=o.map(n=>a.isAbsolute(n)?n:a.join(e,n));for(let n of t)if(j(n))return n;return null}w.exports={findFileUpward:C,findDirectoryUpward:T,findDirectoryWithPaths:N}});var s=require("path"),{findDirectoryUpward:k}=P(),{readJsonFileSafe:h,readFileSafe:b,fileExists:d}=u(),p=class{constructor(e,t={}){if(this.ffId=e,this.tokensBasePath=t.tokensBasePath||this.findTokensPath(),this.folderMap=this.loadFolderMap(),this.projectFolder=this.folderMap[e],!this.projectFolder)throw new Error(`No folder mapping found for ff-id: ${e}`);this.projectPath=s.join(this.tokensBasePath,this.projectFolder)}findTokensPath(){let e=k("apps/tokens",process.cwd(),t=>d(s.join(t,"folderMap.json")));if(e||(e=k("tokens",process.cwd(),t=>d(s.join(t,"folderMap.json")))),!e)throw new Error("Could not find tokens directory with folderMap.json");return e}loadFolderMap(){let e=s.join(this.tokensBasePath,"folderMap.json"),t=h(e);if(!t)throw new Error("Failed to load folderMap.json: File not found");return t}async fetchTokens(){let e=s.join(this.projectPath,"figma"),t={global:[s.join(e,"Global Tokens","Default.json"),s.join(e,"global.json")],colors:[s.join(e,"Global Colors","Default.json"),s.join(e,"Global Colors","Light.json")],semantic:[s.join(e,"Semantic","Light.json"),s.join(e,"Semantic","Default.json")],semanticDark:[s.join(e,"Semantic","Dark.json")]},n={};for(let[i,c]of Object.entries(t)){for(let r of c){let l=h(r);if(l!==null){n[i]=l;break}}n[i]||(n[i]=null)}return n}async fetchProcessedTokens(){let e=s.join(this.projectPath,"cache");if(!d(e))return null;let t={"tokens.js":"hashedTokens.js","variables.js":"hashedVariables.js","semanticVariables.js":"hashedSemanticVariables.js","semanticDarkVariables.js":"hashedSemanticDarkVariables.js","safelist.js":"safelist.js","custom.js":"custom.js","fonts.json":"fonts.json","icons.json":"icons.json","components-config.json":"encoded-config.json","version.json":"version.json"},n={};for(let[c,r]of Object.entries(t)){let l=s.join(e,r);c.endsWith(".js")?n[c]=b(l):n[c]=h(l)}return Object.values(n).some(c=>c!==null)?n:null}async fetchComponentsConfig(){return h(s.join(this.projectPath,"components-config.json"))}async fetchFonts(){return h(s.join(this.projectPath,"font.json"))}async fetchIcons(){return h(s.join(this.projectPath,"icons.json"))}async fetchVersion(){return h(s.join(this.projectPath,"version.json"))}async fetchCustomCss(){return b(s.join(this.projectPath,"custom.css"))}async fetchPlanMetadata(){return h(s.join(this.projectPath,"metadata.json"))||{planType:"full",allowedComponents:null}}fetchJson(){throw new Error("fetchJson is not implemented in LocalTokenReader. Use specific fetch methods.")}fetchRaw(){throw new Error("fetchRaw is not implemented in LocalTokenReader. Use specific fetch methods.")}};module.exports={LocalTokenReader:p};
2
2
  //# sourceMappingURL=local-token-reader.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../lib/core/file-utils.js", "../../../lib/core/path-utils.js", "../../../lib/core/local-token-reader.js"],
4
- "sourcesContent": ["const fs = require('fs');\n\n/**\n * Read a JSON file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the JSON file\n * @returns {Object|null} Parsed JSON or null if file not found\n * @throws {Error} If JSON is invalid\n */\nfunction readJsonFileSafe(filepath) {\n try {\n const content = fs.readFileSync(filepath, 'utf8');\n return JSON.parse(content);\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found, return null like API would return 404\n }\n throw error;\n }\n}\n\n/**\n * Read a file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the file\n * @returns {string|null} File content or null if file not found\n * @throws {Error} For other file system errors\n */\nfunction readFileSafe(filepath) {\n try {\n return fs.readFileSync(filepath, 'utf8');\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found\n }\n throw error;\n }\n}\n\n/**\n * Write JSON data to a file with proper formatting\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to write\n * @param {number} indent - Indentation level (default: 2)\n */\nfunction writeJsonFile(filepath, data, indent = 2) {\n const content = JSON.stringify(data, null, indent);\n fs.writeFileSync(filepath, content, 'utf8');\n}\n\n/**\n * Write module.exports file with JSON data\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to export\n */\nfunction writeModuleExportsFile(filepath, data) {\n // Handle empty data cases\n if (data === null || data === undefined || \n (typeof data === 'object' && Object.keys(data).length === 0) ||\n (Array.isArray(data) && data.length === 0)) {\n const emptyContent = Array.isArray(data) ? '[]' : '{}';\n const content = `module.exports = ${emptyContent};`;\n fs.writeFileSync(filepath, content, 'utf8');\n } else {\n const content = `module.exports = ${JSON.stringify(data, null, 2)};`;\n fs.writeFileSync(filepath, content, 'utf8');\n }\n}\n\n/**\n * Check if a file exists\n * @param {string} filepath - Path to check\n * @returns {boolean} True if file exists\n */\nfunction fileExists(filepath) {\n return fs.existsSync(filepath);\n}\n\n/**\n * Create directory recursively\n * @param {string} dirpath - Directory path to create\n */\nfunction ensureDirectoryExists(dirpath) {\n fs.mkdirSync(dirpath, { recursive: true });\n}\n\n/**\n * Remove directory recursively\n * @param {string} dirpath - Directory path to remove\n */\nfunction removeDirectory(dirpath) {\n fs.rmSync(dirpath, { recursive: true, force: true });\n}\n\nmodule.exports = {\n readJsonFileSafe,\n readFileSafe,\n writeJsonFile,\n writeModuleExportsFile,\n fileExists,\n ensureDirectoryExists,\n removeDirectory\n};", "const path = require('path');\nconst { fileExists } = require('./file-utils');\n\n/**\n * Find a file by traversing up the directory tree\n * @param {string} filename - Name of the file to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the file or null if not found\n */\nfunction findFileUpward(filename, startDir = process.cwd()) {\n let currentDir = startDir;\n \n while (currentDir !== path.parse(currentDir).root) {\n const filePath = path.join(currentDir, filename);\n \n if (fileExists(filePath)) {\n return filePath;\n }\n \n currentDir = path.dirname(currentDir);\n }\n \n return null;\n}\n\n/**\n * Find a directory by traversing up and checking multiple possible paths\n * @param {string} dirname - Name of the directory to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @param {Function} validator - Optional function to validate the directory\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryUpward(dirname, startDir = process.cwd(), validator = null) {\n const possiblePaths = [];\n let currentDir = startDir;\n const maxLevels = 10; // Prevent infinite loop\n \n // Build list of possible paths\n for (let i = 0; i < maxLevels; i++) {\n // Direct path\n possiblePaths.push(path.join(currentDir, dirname));\n \n // Relative paths up to 3 levels\n if (i < 3) {\n possiblePaths.push(path.join(currentDir, '../'.repeat(i + 1) + dirname));\n }\n \n const parentDir = path.dirname(currentDir);\n if (parentDir === currentDir) break; // Reached root\n currentDir = parentDir;\n }\n \n // Also check from module directory (for when running from node_modules)\n possiblePaths.push(path.join(__dirname, '../../../', dirname));\n \n // Find first existing directory that passes validation\n for (const dirPath of possiblePaths) {\n if (fileExists(dirPath)) {\n if (!validator || validator(dirPath)) {\n return dirPath;\n }\n }\n }\n \n return null;\n}\n\n/**\n * Find a directory with multiple possible relative paths\n * @param {string[]} possiblePaths - Array of possible paths to check\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryWithPaths(possiblePaths, startDir = process.cwd()) {\n // Build absolute paths from the start directory\n const absolutePaths = possiblePaths.map(p => {\n if (path.isAbsolute(p)) {\n return p;\n }\n return path.join(startDir, p);\n });\n \n // Find first existing path\n for (const dirPath of absolutePaths) {\n if (fileExists(dirPath)) {\n return dirPath;\n }\n }\n \n return null;\n}\n\nmodule.exports = {\n findFileUpward,\n findDirectoryUpward,\n findDirectoryWithPaths\n};", "const path = require('path');\nconst { findDirectoryUpward } = require('./path-utils');\nconst { readJsonFileSafe, readFileSafe, fileExists } = require('./file-utils');\n\n/**\n * LocalTokenReader - Reads tokens directly from the local file system\n * Mimics the APIClient interface but reads from apps/tokens directory\n */\nclass LocalTokenReader {\n constructor(ffId, options = {}) {\n this.ffId = ffId;\n this.tokensBasePath = options.tokensBasePath || this.findTokensPath();\n this.folderMap = this.loadFolderMap();\n this.projectFolder = this.folderMap[ffId];\n \n if (!this.projectFolder) {\n throw new Error(`No folder mapping found for ff-id: ${ffId}`);\n }\n \n this.projectPath = path.join(this.tokensBasePath, this.projectFolder);\n }\n\n /**\n * Find the tokens directory by looking for apps/tokens in parent directories\n */\n findTokensPath() {\n // Try to find apps/tokens first\n let tokensPath = findDirectoryUpward('apps/tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n \n // If not found, try just 'tokens'\n if (!tokensPath) {\n tokensPath = findDirectoryUpward('tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n }\n \n if (!tokensPath) {\n throw new Error('Could not find tokens directory with folderMap.json');\n }\n \n return tokensPath;\n }\n\n /**\n * Load the folder mapping from folderMap.json\n */\n loadFolderMap() {\n const folderMapPath = path.join(this.tokensBasePath, 'folderMap.json');\n const folderMap = readJsonFileSafe(folderMapPath);\n \n if (!folderMap) {\n throw new Error(`Failed to load folderMap.json: File not found`);\n }\n \n return folderMap;\n }\n\n /**\n * Fetch all token files (global, colors, semantic)\n * Mimics the APIClient.fetchTokens() method\n */\n async fetchTokens() {\n const figmaPath = path.join(this.projectPath, 'figma');\n \n // Try different possible paths for tokens\n const tokenPaths = {\n global: [\n path.join(figmaPath, 'Global Tokens', 'Default.json'),\n path.join(figmaPath, 'global.json')\n ],\n colors: [\n path.join(figmaPath, 'Global Colors', 'Default.json'),\n path.join(figmaPath, 'Global Colors', 'Light.json')\n ],\n semantic: [\n path.join(figmaPath, 'Semantic', 'Light.json'),\n path.join(figmaPath, 'Semantic', 'Default.json')\n ],\n semanticDark: [\n path.join(figmaPath, 'Semantic', 'Dark.json')\n ]\n };\n\n const results = {};\n \n // Try each possible path for each token type\n for (const [key, paths] of Object.entries(tokenPaths)) {\n for (const tokenPath of paths) {\n const content = readJsonFileSafe(tokenPath);\n if (content !== null) {\n results[key] = content;\n break; // Found it, no need to try other paths\n }\n }\n // If not found in any path, set to null\n if (!results[key]) {\n results[key] = null;\n }\n }\n\n return results;\n }\n\n /**\n * Fetch pre-processed tokens (if available in cache folder)\n */\n async fetchProcessedTokens() {\n const cachePath = path.join(this.projectPath, 'cache');\n \n if (!fileExists(cachePath)) {\n return null;\n }\n\n // Read all the cache files that would be in processed tokens\n const cacheFiles = {\n 'tokens.js': 'hashedTokens.js',\n 'variables.js': 'hashedVariables.js', \n 'semanticVariables.js': 'hashedSemanticVariables.js',\n 'semanticDarkVariables.js': 'hashedSemanticDarkVariables.js',\n 'cls.js': 'cls.js',\n 'custom.js': 'custom.js',\n 'fonts.json': 'fonts.json',\n 'icons.json': 'icons.json',\n 'components-config.json': 'encoded-config.json',\n 'version.json': 'version.json'\n };\n\n const processedData = {};\n \n for (const [key, fileName] of Object.entries(cacheFiles)) {\n const filePath = path.join(cachePath, fileName);\n if (key.endsWith('.js')) {\n processedData[key] = readFileSafe(filePath);\n } else {\n processedData[key] = readJsonFileSafe(filePath);\n }\n }\n\n // Only return if we have at least some data\n const hasData = Object.values(processedData).some(val => val !== null);\n return hasData ? processedData : null;\n }\n\n /**\n * Fetch components configuration\n */\n async fetchComponentsConfig() {\n return readJsonFileSafe(path.join(this.projectPath, 'components-config.json'));\n }\n\n /**\n * Fetch fonts configuration\n */\n async fetchFonts() {\n return readJsonFileSafe(path.join(this.projectPath, 'font.json'));\n }\n\n /**\n * Fetch icons configuration\n */\n async fetchIcons() {\n return readJsonFileSafe(path.join(this.projectPath, 'icons.json'));\n }\n\n /**\n * Fetch version information\n */\n async fetchVersion() {\n return readJsonFileSafe(path.join(this.projectPath, 'version.json'));\n }\n\n /**\n * Fetch custom CSS\n */\n async fetchCustomCss() {\n return readFileSafe(path.join(this.projectPath, 'custom.css'));\n }\n\n /**\n * Fetch plan metadata\n */\n async fetchPlanMetadata() {\n const metadata = readJsonFileSafe(path.join(this.projectPath, 'metadata.json'));\n \n // Default to full plan if no metadata exists\n return metadata || {\n planType: 'full',\n allowedComponents: null\n };\n }\n\n // Compatibility methods to match APIClient interface\n fetchJson() {\n throw new Error('fetchJson is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n\n fetchRaw() {\n throw new Error('fetchRaw is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n}\n\nmodule.exports = { LocalTokenReader };"],
5
- "mappings": "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",
4
+ "sourcesContent": ["const fs = require('fs');\n\n/**\n * Read a JSON file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the JSON file\n * @returns {Object|null} Parsed JSON or null if file not found\n * @throws {Error} If JSON is invalid\n */\nfunction readJsonFileSafe(filepath) {\n try {\n const content = fs.readFileSync(filepath, 'utf8');\n return JSON.parse(content);\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found, return null like API would return 404\n }\n throw error;\n }\n}\n\n/**\n * Read a file safely, returning null if file doesn't exist\n * @param {string} filepath - Path to the file\n * @returns {string|null} File content or null if file not found\n * @throws {Error} For other file system errors\n */\nfunction readFileSafe(filepath) {\n try {\n return fs.readFileSync(filepath, 'utf8');\n } catch (error) {\n if (error.code === 'ENOENT') {\n return null; // File not found\n }\n throw error;\n }\n}\n\n/**\n * Write JSON data to a file with proper formatting\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to write\n * @param {number} indent - Indentation level (default: 2)\n */\nfunction writeJsonFile(filepath, data, indent = 2) {\n const content = JSON.stringify(data, null, indent);\n fs.writeFileSync(filepath, content, 'utf8');\n}\n\n/**\n * Write module.exports file with JSON data\n * @param {string} filepath - Path to write the file\n * @param {Object} data - Data to export\n */\nfunction writeModuleExportsFile(filepath, data) {\n // Handle empty data cases\n if (data === null || data === undefined || \n (typeof data === 'object' && Object.keys(data).length === 0) ||\n (Array.isArray(data) && data.length === 0)) {\n const emptyContent = Array.isArray(data) ? '[]' : '{}';\n const content = `module.exports = ${emptyContent};`;\n fs.writeFileSync(filepath, content, 'utf8');\n } else {\n const content = `module.exports = ${JSON.stringify(data, null, 2)};`;\n fs.writeFileSync(filepath, content, 'utf8');\n }\n}\n\n/**\n * Check if a file exists\n * @param {string} filepath - Path to check\n * @returns {boolean} True if file exists\n */\nfunction fileExists(filepath) {\n return fs.existsSync(filepath);\n}\n\n/**\n * Create directory recursively\n * @param {string} dirpath - Directory path to create\n */\nfunction ensureDirectoryExists(dirpath) {\n fs.mkdirSync(dirpath, { recursive: true });\n}\n\n/**\n * Remove directory recursively\n * @param {string} dirpath - Directory path to remove\n */\nfunction removeDirectory(dirpath) {\n fs.rmSync(dirpath, { recursive: true, force: true });\n}\n\nmodule.exports = {\n readJsonFileSafe,\n readFileSafe,\n writeJsonFile,\n writeModuleExportsFile,\n fileExists,\n ensureDirectoryExists,\n removeDirectory\n};", "const path = require('path');\nconst { fileExists } = require('./file-utils');\n\n/**\n * Find a file by traversing up the directory tree\n * @param {string} filename - Name of the file to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the file or null if not found\n */\nfunction findFileUpward(filename, startDir = process.cwd()) {\n let currentDir = startDir;\n \n while (currentDir !== path.parse(currentDir).root) {\n const filePath = path.join(currentDir, filename);\n \n if (fileExists(filePath)) {\n return filePath;\n }\n \n currentDir = path.dirname(currentDir);\n }\n \n return null;\n}\n\n/**\n * Find a directory by traversing up and checking multiple possible paths\n * @param {string} dirname - Name of the directory to find\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @param {Function} validator - Optional function to validate the directory\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryUpward(dirname, startDir = process.cwd(), validator = null) {\n const possiblePaths = [];\n let currentDir = startDir;\n const maxLevels = 10; // Prevent infinite loop\n \n // Build list of possible paths\n for (let i = 0; i < maxLevels; i++) {\n // Direct path\n possiblePaths.push(path.join(currentDir, dirname));\n \n // Relative paths up to 3 levels\n if (i < 3) {\n possiblePaths.push(path.join(currentDir, '../'.repeat(i + 1) + dirname));\n }\n \n const parentDir = path.dirname(currentDir);\n if (parentDir === currentDir) break; // Reached root\n currentDir = parentDir;\n }\n \n // Also check from module directory (for when running from node_modules)\n possiblePaths.push(path.join(__dirname, '../../../', dirname));\n \n // Find first existing directory that passes validation\n for (const dirPath of possiblePaths) {\n if (fileExists(dirPath)) {\n if (!validator || validator(dirPath)) {\n return dirPath;\n }\n }\n }\n \n return null;\n}\n\n/**\n * Find a directory with multiple possible relative paths\n * @param {string[]} possiblePaths - Array of possible paths to check\n * @param {string} startDir - Starting directory (defaults to cwd)\n * @returns {string|null} Full path to the directory or null if not found\n */\nfunction findDirectoryWithPaths(possiblePaths, startDir = process.cwd()) {\n // Build absolute paths from the start directory\n const absolutePaths = possiblePaths.map(p => {\n if (path.isAbsolute(p)) {\n return p;\n }\n return path.join(startDir, p);\n });\n \n // Find first existing path\n for (const dirPath of absolutePaths) {\n if (fileExists(dirPath)) {\n return dirPath;\n }\n }\n \n return null;\n}\n\nmodule.exports = {\n findFileUpward,\n findDirectoryUpward,\n findDirectoryWithPaths\n};", "const path = require('path');\nconst { findDirectoryUpward } = require('./path-utils');\nconst { readJsonFileSafe, readFileSafe, fileExists } = require('./file-utils');\n\n/**\n * LocalTokenReader - Reads tokens directly from the local file system\n * Mimics the APIClient interface but reads from apps/tokens directory\n */\nclass LocalTokenReader {\n constructor(ffId, options = {}) {\n this.ffId = ffId;\n this.tokensBasePath = options.tokensBasePath || this.findTokensPath();\n this.folderMap = this.loadFolderMap();\n this.projectFolder = this.folderMap[ffId];\n \n if (!this.projectFolder) {\n throw new Error(`No folder mapping found for ff-id: ${ffId}`);\n }\n \n this.projectPath = path.join(this.tokensBasePath, this.projectFolder);\n }\n\n /**\n * Find the tokens directory by looking for apps/tokens in parent directories\n */\n findTokensPath() {\n // Try to find apps/tokens first\n let tokensPath = findDirectoryUpward('apps/tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n \n // If not found, try just 'tokens'\n if (!tokensPath) {\n tokensPath = findDirectoryUpward('tokens', process.cwd(), (dir) => {\n return fileExists(path.join(dir, 'folderMap.json'));\n });\n }\n \n if (!tokensPath) {\n throw new Error('Could not find tokens directory with folderMap.json');\n }\n \n return tokensPath;\n }\n\n /**\n * Load the folder mapping from folderMap.json\n */\n loadFolderMap() {\n const folderMapPath = path.join(this.tokensBasePath, 'folderMap.json');\n const folderMap = readJsonFileSafe(folderMapPath);\n \n if (!folderMap) {\n throw new Error(`Failed to load folderMap.json: File not found`);\n }\n \n return folderMap;\n }\n\n /**\n * Fetch all token files (global, colors, semantic)\n * Mimics the APIClient.fetchTokens() method\n */\n async fetchTokens() {\n const figmaPath = path.join(this.projectPath, 'figma');\n \n // Try different possible paths for tokens\n const tokenPaths = {\n global: [\n path.join(figmaPath, 'Global Tokens', 'Default.json'),\n path.join(figmaPath, 'global.json')\n ],\n colors: [\n path.join(figmaPath, 'Global Colors', 'Default.json'),\n path.join(figmaPath, 'Global Colors', 'Light.json')\n ],\n semantic: [\n path.join(figmaPath, 'Semantic', 'Light.json'),\n path.join(figmaPath, 'Semantic', 'Default.json')\n ],\n semanticDark: [\n path.join(figmaPath, 'Semantic', 'Dark.json')\n ]\n };\n\n const results = {};\n \n // Try each possible path for each token type\n for (const [key, paths] of Object.entries(tokenPaths)) {\n for (const tokenPath of paths) {\n const content = readJsonFileSafe(tokenPath);\n if (content !== null) {\n results[key] = content;\n break; // Found it, no need to try other paths\n }\n }\n // If not found in any path, set to null\n if (!results[key]) {\n results[key] = null;\n }\n }\n\n return results;\n }\n\n /**\n * Fetch pre-processed tokens (if available in cache folder)\n */\n async fetchProcessedTokens() {\n const cachePath = path.join(this.projectPath, 'cache');\n \n if (!fileExists(cachePath)) {\n return null;\n }\n\n // Read all the cache files that would be in processed tokens\n const cacheFiles = {\n 'tokens.js': 'hashedTokens.js',\n 'variables.js': 'hashedVariables.js', \n 'semanticVariables.js': 'hashedSemanticVariables.js',\n 'semanticDarkVariables.js': 'hashedSemanticDarkVariables.js',\n 'safelist.js': 'safelist.js',\n 'custom.js': 'custom.js',\n 'fonts.json': 'fonts.json',\n 'icons.json': 'icons.json',\n 'components-config.json': 'encoded-config.json',\n 'version.json': 'version.json'\n };\n\n const processedData = {};\n \n for (const [key, fileName] of Object.entries(cacheFiles)) {\n const filePath = path.join(cachePath, fileName);\n if (key.endsWith('.js')) {\n processedData[key] = readFileSafe(filePath);\n } else {\n processedData[key] = readJsonFileSafe(filePath);\n }\n }\n\n // Only return if we have at least some data\n const hasData = Object.values(processedData).some(val => val !== null);\n return hasData ? processedData : null;\n }\n\n /**\n * Fetch components configuration\n */\n async fetchComponentsConfig() {\n return readJsonFileSafe(path.join(this.projectPath, 'components-config.json'));\n }\n\n /**\n * Fetch fonts configuration\n */\n async fetchFonts() {\n return readJsonFileSafe(path.join(this.projectPath, 'font.json'));\n }\n\n /**\n * Fetch icons configuration\n */\n async fetchIcons() {\n return readJsonFileSafe(path.join(this.projectPath, 'icons.json'));\n }\n\n /**\n * Fetch version information\n */\n async fetchVersion() {\n return readJsonFileSafe(path.join(this.projectPath, 'version.json'));\n }\n\n /**\n * Fetch custom CSS\n */\n async fetchCustomCss() {\n return readFileSafe(path.join(this.projectPath, 'custom.css'));\n }\n\n /**\n * Fetch plan metadata\n */\n async fetchPlanMetadata() {\n const metadata = readJsonFileSafe(path.join(this.projectPath, 'metadata.json'));\n \n // Default to full plan if no metadata exists\n return metadata || {\n planType: 'full',\n allowedComponents: null\n };\n }\n\n // Compatibility methods to match APIClient interface\n fetchJson() {\n throw new Error('fetchJson is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n\n fetchRaw() {\n throw new Error('fetchRaw is not implemented in LocalTokenReader. Use specific fetch methods.');\n }\n}\n\nmodule.exports = { LocalTokenReader };"],
5
+ "mappings": "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",
6
6
  "names": ["require_file_utils", "__commonJSMin", "exports", "module", "fs", "readJsonFileSafe", "filepath", "content", "error", "readFileSafe", "writeJsonFile", "data", "indent", "writeModuleExportsFile", "fileExists", "ensureDirectoryExists", "dirpath", "removeDirectory", "require_path_utils", "__commonJSMin", "exports", "module", "path", "fileExists", "findFileUpward", "filename", "startDir", "currentDir", "filePath", "findDirectoryUpward", "dirname", "validator", "possiblePaths", "maxLevels", "i", "parentDir", "dirPath", "findDirectoryWithPaths", "absolutePaths", "p", "path", "findDirectoryUpward", "readJsonFileSafe", "readFileSafe", "fileExists", "LocalTokenReader", "ffId", "options", "tokensPath", "dir", "folderMapPath", "folderMap", "figmaPath", "tokenPaths", "results", "key", "paths", "tokenPath", "content", "cachePath", "cacheFiles", "processedData", "fileName", "filePath", "val"]
7
7
  }
@@ -1,2 +1,2 @@
1
- var O=(v,e)=>()=>(e||v((e={exports:{}}).exports,e),e.exports);var x=O((W,V)=>{var b=class extends Error{constructor(e,s,o){super(e),this.name="APIError",this.statusCode=s,this.url=o,this.code=`API_${s}`}},C=class extends Error{constructor(e,s){super(e),this.name="CacheError",this.operation=s,this.code=`CACHE_${s.toUpperCase()}`}},k=class extends Error{constructor(e,s){super(e),this.name="ConfigError",this.field=s,this.code=`CONFIG_${s.toUpperCase()}`}},F=class extends Error{constructor(e,s){super(e),this.name="ProcessingError",this.token=s,this.code="PROCESSING_ERROR"}};V.exports={APIError:b,CacheError:C,ConfigError:k,ProcessingError:F}});var w=require("color"),N=require("https"),{ProcessingError:E}=x(),g=class{hexToHsl(e){try{let s=w(e),o=s.hsl(),t=Math.round(o.hue()),r=Math.round(o.saturationl()),a=Math.round(o.lightness()),i=s.alpha();if(i<1){let c=Math.round(i*100)/100;return`${t} ${r}% ${a}% / ${c}`}return`${t} ${r}% ${a}%`}catch(s){throw new E(`Failed to convert color ${e}: ${s.message}`,e)}}processColors(e){let s={},o={},t={backgroundColor:{},textColor:{},borderColor:{}};if(!e||typeof e!="object")return console.log(" \u26A0\uFE0F No valid color data to process"),{variables:s,colorMap:o,utilities:t};let r=(a,i="")=>{for(let[c,l]of Object.entries(a))if(l&&typeof l=="object")if(l.$value){let n=i?`${i}-${c}`:c,u=`--color-${n}`,f=this.hexToHsl(l.$value);s[u]=f,o[n]=u,t.backgroundColor[n]=`hsl(var(${u}))`,t.textColor[n]=`hsl(var(${u}))`,t.borderColor[n]=`hsl(var(${u}))`}else{let n=i?`${i}-${c}`:c;r(l,n)}};return e&&(e.colors?r(e.colors):r(e)),{variables:s,colorMap:o,utilities:t}}processSemanticTokens(e,s){let o={},t={backgroundColor:{},textColor:{},borderColor:{}},r=c=>c.replace(/ /g,"").replace(/\./g,"-").toLowerCase(),a=c=>{if(typeof c!="string")return null;let l=c.match(/^\{(.+)\}$/);if(!l)return null;let n=l[1].split(".");return n[0]==="colors"&&n.length>1?n[1]==="colors"&&n.length>2?n.slice(2).join("-"):n.slice(1).join("-"):n.join("-")},i=(c,l)=>{if(e[c])for(let[n,u]of Object.entries(e[c])){if(n==="disabled"){let f=a(u.$value);if(f){let m=s[f];if(m){let h=`--${c}-disabled`;o[h]=`var(${m})`,t[l].disabled=`var(${h})`}}continue}for(let[f,m]of Object.entries(u))for(let[h,p]of Object.entries(m)){let $=r(`${n}-${f}-${h}`);$.endsWith("-default")&&($=$.slice(0,-8));let d=`--${c}-${$}`,j=a(p.$value);if(j){let y=s[j];y&&(o[d]=`var(${y})`,t[l][$]=`var(${d})`)}else if(p.$value){let y=this.hexToHsl(p.$value);o[d]=y,t[l][$]=`hsl(var(${d}))`}}}};if(i("bg","backgroundColor"),i("text","textColor"),i("border","borderColor"),e.layer)for(let[c,l]of Object.entries(e.layer)){let n=`--layer-${c}`,u=a(l.$value);if(u){let f=u.replace(/\./g,"-").toLowerCase();if(f.endsWith("-default")&&(f=f.slice(0,-8)),f.startsWith("bg-")||f.startsWith("text-")||f.startsWith("border-")){let m=`--${f}`;o[n]=`var(${m})`,t.backgroundColor[`layer-${c}`]=`var(${n})`}else{let m=s[f];m&&(o[n]=`var(${m})`,t.backgroundColor[`layer-${c}`]=`var(${n})`)}}else if(l.$value){let f=this.hexToHsl(l.$value);o[n]=f,t.backgroundColor[`layer-${c}`]=`hsl(var(${n}))`}}if(e.overlay)for(let[c,l]of Object.entries(e.overlay)){let n=`--overlay-${c}`;l.$value&&(o[n]=l.$value,t.backgroundColor[`overlay-${c}`]=`var(${n})`)}if(e.icon){t.fill||(t.fill={});for(let[c,l]of Object.entries(e.icon)){if(c==="disabled"){let n=a(l.$value);if(n){let u=s[n];if(u){let f="--icon-disabled";o[f]=`var(${u})`,t.textColor["icon-disabled"]=`var(${f})`,t.fill["icon-disabled"]=`var(${f})`}}continue}for(let[n,u]of Object.entries(l))for(let[f,m]of Object.entries(u)){let h=r(`icon-${c}-${n}-${f}`);h.endsWith("-default")&&(h=h.slice(0,-8));let p=`--${h}`,$=a(m.$value);if($){let d=s[$];d&&(o[p]=`var(${d})`,t.textColor[h]=`var(${p})`,t.fill[h]=`var(${p})`)}}}}return{semanticVariables:o,tokens:t}}async fetchFontCss(e){return new Promise((s,o)=>{N.get(e,t=>{let r="";if(t.statusCode!==200){o(new Error(`Failed to fetch font CSS: ${t.statusCode}`));return}t.on("data",a=>{r+=a}),t.on("end",()=>{s(r)}),t.on("error",a=>{o(a)})}).on("error",t=>{o(t)})})}parseFontFaces(e){let s=/@font-face\s*{([^}]+)}/g,o=[],t;for(;(t=s.exec(e))!==null;){let r=t[1].trim().split(";").filter(Boolean),a={};r.forEach(i=>{let c=i.indexOf(":");if(c!==-1){let l=i.slice(0,c).trim(),n=i.slice(c+1).trim();l==="font-family"?a[l]=n.replace(/["']/g,""):a[l]=n}}),a.src&&(a.src=a.src.replace(/url\(["']?/,"url('").replace(/["']?\)/,"')")),o.push(a)}return o}async processFonts(e){let s=[];if(!e||typeof e!="object")return s;let o=Object.keys(e).filter(t=>t.startsWith("font"));for(let t of o){let r=e[t];if(typeof r=="string"&&r.startsWith("http"))try{let a=await this.fetchFontCss(r),i=this.parseFontFaces(a);i.length>0&&s.push(...i)}catch(a){console.warn(`Failed to process font ${t}:`,a.message)}}return s}processFontFamilies(e){let s={};if(!e||!e.font||!e.font.family)return s;let o=e.font.family;for(let[t,r]of Object.entries(o))if(r&&r.$value){let a=t.toLowerCase(),i=r.$value;typeof i=="string"&&i.startsWith("{")&&i.endsWith("}")&&(i=i.slice(1,-1));let c=l=>l.split(" ").map(n=>n.charAt(0).toUpperCase()+n.slice(1).toLowerCase()).join(" ");if(i.includes(","))s[a]=i;else{let l=c(i);l.includes(" ")&&!l.startsWith('"')&&!l.startsWith("'")?s[a]=`"${l}", sans-serif`:s[a]=`${l}, sans-serif`}}return s}processAnimations(e){let s={},o={};if(!e||typeof e!="object")return{keyframes:s,animations:o};for(let[t,r]of Object.entries(e))if(r&&typeof r=="object"){let a=`${t}`;if(r.keyframes){let c={};for(let[l,n]of Object.entries(r.keyframes))n&&typeof n=="object"&&(c[l]=n);s[a]=c}let i=[a,r.duration||"1s",r.easing||"ease",r.delay||"0s",r.iterations||"1",r.direction||"normal",r.fillMode||"none"].join(" ");o[t]=i}return{keyframes:s,animations:o}}generateSafelistClasses(e){let s=[];if(e.colorMap){let o=["hover:","focus:","active:","disabled:","dark:"],t=["bg-","text-","border-"];for(let r of Object.keys(e.colorMap))t.forEach(a=>{s.push(`${a}${r}`),o.forEach(i=>{s.push(`${i}${a}${r}`)})})}if(e.animations)for(let o of Object.keys(e.animations))s.push(`animate-${o}`);return s}extractClassesFromComponentsConfig(e){let s=new Set,o=r=>{typeof r=="string"&&r.split(/\s+/).filter(i=>i.length>0).forEach(i=>{s.add(i)})},t=r=>{if(!(!r||typeof r!="object"))for(let a in r){let i=r[a];typeof i=="string"?o(i):typeof i=="object"&&t(i)}};return e&&t(e),Array.from(s).sort()}processCustomCss(e){return!e||typeof e!="string"?{}:{raw:e}}async process(e){try{let s={variables:{},semanticVariables:{},semanticDarkVariables:{},utilities:{},fontFaces:[],keyframes:{},animations:{},safelist:[],metadata:{},custom:e.customCss?this.processCustomCss(e.customCss):{}},o=null;if(e.tokens&&typeof e.tokens=="object"){let t=e.tokens;o=t["Global Colors/Default"]||t["Global Colors"]||t.colors||t}else e.colors&&(o=e.colors);if(o){let t=this.processColors(o);s.variables={...s.variables,...t.variables},s.utilities={...s.utilities,...t.utilities},s.colorMap=t.colorMap||{}}else s.colorMap={};if(e.semantic&&s.colorMap){let t=this.processSemanticTokens(e.semantic,s.colorMap);s.semanticVariables=t.semanticVariables,t.tokens&&(s.tokens=t.tokens,s.utilities=t.tokens)}if(e.semanticDark&&s.colorMap){let t=this.processSemanticTokens(e.semanticDark,s.colorMap);s.semanticDarkVariables=t.semanticVariables}if(e.fonts&&(s.fontFaces=await this.processFonts(e.fonts)),e.globalTokens&&(s.fontFamilies=this.processFontFamilies(e.globalTokens),s.fontFamilies&&(s.utilities.fontFamily||(s.utilities.fontFamily={}),Object.assign(s.utilities.fontFamily,s.fontFamilies))),e.animations){let t=this.processAnimations(e.animations);s.keyframes=t.keyframes,s.animations=t.animations}return s.safelist=this.generateSafelistClasses(s),s.metadata={timestamp:new Date().toISOString(),version:e.version||"2.0.0",ffId:e.ffId},s}catch(s){throw console.error("Error processing tokens:",s),s}}};module.exports=g;
1
+ var V=(y,e)=>()=>(e||y((e={exports:{}}).exports,e),e.exports);var O=V((S,x)=>{var v=class extends Error{constructor(e,s,r){super(e),this.name="APIError",this.statusCode=s,this.url=r,this.code=`API_${s}`}},C=class extends Error{constructor(e,s){super(e),this.name="CacheError",this.operation=s,this.code=`CACHE_${s.toUpperCase()}`}},k=class extends Error{constructor(e,s){super(e),this.name="ConfigError",this.field=s,this.code=`CONFIG_${s.toUpperCase()}`}},F=class extends Error{constructor(e,s){super(e),this.name="ProcessingError",this.token=s,this.code="PROCESSING_ERROR"}};x.exports={APIError:v,CacheError:C,ConfigError:k,ProcessingError:F}});var N=require("color"),w=require("https"),{ProcessingError:E}=O(),g=class{hexToHsl(e){try{let s=N(e),r=s.hsl(),t=Math.round(r.hue()),o=Math.round(r.saturationl()),l=Math.round(r.lightness()),i=s.alpha();if(i<1){let a=Math.round(i*100)/100;return`${t} ${o}% ${l}% / ${a}`}return`${t} ${o}% ${l}%`}catch(s){throw new E(`Failed to convert color ${e}: ${s.message}`,e)}}processColors(e){let s={},r={},t={backgroundColor:{},textColor:{},borderColor:{}};if(!e||typeof e!="object")return console.log(" \u26A0\uFE0F No valid color data to process"),{variables:s,colorMap:r,utilities:t};let o=(l,i="")=>{for(let[a,c]of Object.entries(l))if(c&&typeof c=="object")if(c.$value){let n=i?`${i}-${a}`:a,u=`--color-${n}`,f=this.hexToHsl(c.$value);s[u]=f,r[n]=u,t.backgroundColor[n]=`hsl(var(${u}))`,t.textColor[n]=`hsl(var(${u}))`,t.borderColor[n]=`hsl(var(${u}))`}else{let n=i?`${i}-${a}`:a;o(c,n)}};return e&&(e.colors?o(e.colors):o(e)),{variables:s,colorMap:r,utilities:t}}processSemanticTokens(e,s){let r={},t={backgroundColor:{},textColor:{},borderColor:{}},o=a=>a.replace(/ /g,"").replace(/\./g,"-").toLowerCase(),l=a=>{if(typeof a!="string")return null;let c=a.match(/^\{(.+)\}$/);if(!c)return null;let n=c[1].split(".");return n[0]==="colors"&&n.length>1?n[1]==="colors"&&n.length>2?n.slice(2).join("-"):n.slice(1).join("-"):n.join("-")},i=(a,c)=>{if(e[a])for(let[n,u]of Object.entries(e[a])){if(n==="disabled"){let f=l(u.$value);if(f){let h=s[f];if(h){let m=`--${a}-disabled`;r[m]=`var(${h})`,t[c].disabled=`var(${m})`}}continue}for(let[f,h]of Object.entries(u))for(let[m,p]of Object.entries(h)){let $=o(`${n}-${f}-${m}`);$.endsWith("-default")&&($=$.slice(0,-8));let d=`--${a}-${$}`,j=l(p.$value);if(j){let b=s[j];b&&(r[d]=`var(${b})`,t[c][$]=`var(${d})`)}else if(p.$value){let b=this.hexToHsl(p.$value);r[d]=b,t[c][$]=`hsl(var(${d}))`}}}};if(i("bg","backgroundColor"),i("text","textColor"),i("border","borderColor"),e.layer)for(let[a,c]of Object.entries(e.layer)){let n=`--layer-${a}`,u=l(c.$value);if(u){let f=u.replace(/\./g,"-").toLowerCase();if(f.endsWith("-default")&&(f=f.slice(0,-8)),f.startsWith("bg-")||f.startsWith("text-")||f.startsWith("border-")){let h=`--${f}`;r[n]=`var(${h})`,t.backgroundColor[`layer-${a}`]=`var(${n})`}else{let h=s[f];h&&(r[n]=`var(${h})`,t.backgroundColor[`layer-${a}`]=`var(${n})`)}}else if(c.$value){let f=this.hexToHsl(c.$value);r[n]=f,t.backgroundColor[`layer-${a}`]=`hsl(var(${n}))`}}if(e.overlay)for(let[a,c]of Object.entries(e.overlay)){let n=`--overlay-${a}`;c.$value&&(r[n]=c.$value,t.backgroundColor[`overlay-${a}`]=`var(${n})`)}if(e.icon){t.fill||(t.fill={});for(let[a,c]of Object.entries(e.icon)){if(a==="disabled"){let n=l(c.$value);if(n){let u=s[n];if(u){let f="--icon-disabled";r[f]=`var(${u})`,t.textColor["icon-disabled"]=`var(${f})`,t.fill["icon-disabled"]=`var(${f})`}}continue}for(let[n,u]of Object.entries(c))for(let[f,h]of Object.entries(u)){let m=o(`icon-${a}-${n}-${f}`);m.endsWith("-default")&&(m=m.slice(0,-8));let p=`--${m}`,$=l(h.$value);if($){let d=s[$];d&&(r[p]=`var(${d})`,t.textColor[m]=`var(${p})`,t.fill[m]=`var(${p})`)}}}}return{semanticVariables:r,tokens:t}}async fetchFontCss(e){return new Promise((s,r)=>{w.get(e,t=>{let o="";if(t.statusCode!==200){r(new Error(`Failed to fetch font CSS: ${t.statusCode}`));return}t.on("data",l=>{o+=l}),t.on("end",()=>{s(o)}),t.on("error",l=>{r(l)})}).on("error",t=>{r(t)})})}parseFontFaces(e){let s=/@font-face\s*{([^}]+)}/g,r=[],t;for(;(t=s.exec(e))!==null;){let o=t[1].trim().split(";").filter(Boolean),l={};o.forEach(i=>{let a=i.indexOf(":");if(a!==-1){let c=i.slice(0,a).trim(),n=i.slice(a+1).trim();c==="font-family"?l[c]=n.replace(/["']/g,""):l[c]=n}}),l.src&&(l.src=l.src.replace(/url\(["']?/,"url('").replace(/["']?\)/,"')")),r.push(l)}return r}async processFonts(e){let s=[];if(!e||typeof e!="object")return s;let r=Object.keys(e).filter(t=>t.startsWith("font"));for(let t of r){let o=e[t];if(typeof o=="string"&&o.startsWith("http"))try{let l=await this.fetchFontCss(o),i=this.parseFontFaces(l);i.length>0&&s.push(...i)}catch(l){console.warn(`Failed to process font ${t}:`,l.message)}}return s}processFontFamilies(e){let s={};if(!e||!e.font||!e.font.family)return s;let r=e.font.family;for(let[t,o]of Object.entries(r))if(o&&o.$value){let l=t.toLowerCase(),i=o.$value;typeof i=="string"&&i.startsWith("{")&&i.endsWith("}")&&(i=i.slice(1,-1));let a=c=>c.split(" ").map(n=>n.charAt(0).toUpperCase()+n.slice(1).toLowerCase()).join(" ");if(i.includes(","))s[l]=i;else{let c=a(i);c.includes(" ")&&!c.startsWith('"')&&!c.startsWith("'")?s[l]=`"${c}", sans-serif`:s[l]=`${c}, sans-serif`}}return s}processAnimations(e){let s={},r={};if(!e||typeof e!="object")return{keyframes:s,animations:r};for(let[t,o]of Object.entries(e))if(o&&typeof o=="object"){let l=`${t}`;if(o.keyframes){let a={};for(let[c,n]of Object.entries(o.keyframes))n&&typeof n=="object"&&(a[c]=n);s[l]=a}let i=[l,o.duration||"1s",o.easing||"ease",o.delay||"0s",o.iterations||"1",o.direction||"normal",o.fillMode||"none"].join(" ");r[t]=i}return{keyframes:s,animations:r}}generateSafelistClasses(e){let s=[],r=["hover:","focus:","active:","disabled:","dark:"];if(e.backgroundColor)for(let t of Object.keys(e.backgroundColor))s.push(`bg-${t}`),r.forEach(o=>{s.push(`${o}bg-${t}`)});if(e.textColor)for(let t of Object.keys(e.textColor))s.push(`text-${t}`),r.forEach(o=>{s.push(`${o}text-${t}`)});if(e.borderColor)for(let t of Object.keys(e.borderColor))s.push(`border-${t}`),r.forEach(o=>{s.push(`${o}border-${t}`)});if(e.fill)for(let t of Object.keys(e.fill))s.push(`fill-${t}`),t.startsWith("icon-")&&s.push(`text-${t}`);if(e.fontFamily)for(let t of Object.keys(e.fontFamily))s.push(`font-${t}`);if(e.animations)for(let t of Object.keys(e.animations))s.push(`animate-${t}`);return s}extractClassesFromComponentsConfig(e){let s=new Set,r=o=>{typeof o=="string"&&o.split(/\s+/).filter(i=>i.length>0).forEach(i=>{s.add(i)})},t=o=>{if(!(!o||typeof o!="object"))for(let l in o){let i=o[l];typeof i=="string"?r(i):typeof i=="object"&&t(i)}};return e&&t(e),Array.from(s).sort()}processCustomCss(e){return!e||typeof e!="string"?{}:{raw:e}}async process(e){try{let s={variables:{},semanticVariables:{},semanticDarkVariables:{},utilities:{},fontFaces:[],keyframes:{},animations:{},safelist:[],metadata:{},custom:e.customCss?this.processCustomCss(e.customCss):{}},r=null;if(e.tokens&&typeof e.tokens=="object"){let t=e.tokens;r=t["Global Colors/Default"]||t["Global Colors"]||t.colors||t}else e.colors&&(r=e.colors);if(r){let t=this.processColors(r);s.variables={...s.variables,...t.variables},s.utilities={...s.utilities,...t.utilities},s.colorMap=t.colorMap||{}}else s.colorMap={};if(e.semantic&&s.colorMap){let t=this.processSemanticTokens(e.semantic,s.colorMap);s.semanticVariables=t.semanticVariables,t.tokens&&(s.tokens=t.tokens,s.utilities=t.tokens)}if(e.semanticDark&&s.colorMap){let t=this.processSemanticTokens(e.semanticDark,s.colorMap);s.semanticDarkVariables=t.semanticVariables}if(e.fonts&&(s.fontFaces=await this.processFonts(e.fonts)),e.globalTokens&&(s.fontFamilies=this.processFontFamilies(e.globalTokens),s.fontFamilies&&(s.utilities.fontFamily||(s.utilities.fontFamily={}),Object.assign(s.utilities.fontFamily,s.fontFamilies))),e.animations){let t=this.processAnimations(e.animations);s.keyframes=t.keyframes,s.animations=t.animations}return s.safelist=this.generateSafelistClasses(s.tokens||s.utilities),s.metadata={timestamp:new Date().toISOString(),version:e.version||"2.0.0",ffId:e.ffId},s}catch(s){throw console.error("Error processing tokens:",s),s}}};module.exports=g;
2
2
  //# sourceMappingURL=token-processor.js.map