@frontfriend/tailwind 2.3.0 → 2.5.0
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.
- package/dist/cli.js +40 -28
- package/dist/index.js +3 -3
- package/dist/index.js.map +3 -3
- package/dist/lib/core/api-client.js +1 -1
- package/dist/lib/core/api-client.js.map +3 -3
- package/dist/lib/core/cache-manager.js +13 -1
- package/dist/lib/core/component-downloader.js +2 -2
- package/dist/lib/core/component-downloader.js.map +3 -3
- package/dist/lib/core/constants.js +1 -1
- package/dist/lib/core/constants.js.map +2 -2
- package/dist/lib/core/local-token-reader.js +1 -1
- package/dist/lib/core/local-token-reader.js.map +2 -2
- package/dist/lib/core/token-processor.js +1 -1
- package/dist/lib/core/token-processor.js.map +3 -3
- package/dist/next.js +1 -1
- package/dist/next.js.map +3 -3
- package/dist/vite.js +6 -6
- package/dist/vite.js.map +3 -3
- package/package.json +2 -2
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
var m=(n,t)=>()=>(t||n((t={exports:{}}).exports,t),t.exports);var _=m((st,R)=>{var M="https://app.frontfriend.dev",N="https://tokens-studio-donux.up.railway.app/api",v=".cache/frontfriend",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"]},V={FF_ID:"FF_ID",FF_API_URL:"FF_API_URL",FF_USE_LOCAL:"FF_USE_LOCAL"};R.exports={DEFAULT_API_URL:M,LEGACY_API_URL:N,CACHE_DIR_NAME:v,CACHE_TTL_DAYS:7,CACHE_TTL_MS:6048e5,CACHE_FILES:O,ENV_VARS:V}});var $=m((ot,E)=>{var p=class extends Error{constructor(t,e,s){super(t),this.name="APIError",this.statusCode=e,this.url=s,this.code=`API_${e}`}},j=class extends Error{constructor(t,e){super(t),this.name="CacheError",this.operation=e,this.code=`CACHE_${e.toUpperCase()}`}},y=class extends Error{constructor(t,e){super(t),this.name="ConfigError",this.field=e,this.code=`CONFIG_${e.toUpperCase()}`}},w=class extends Error{constructor(t,e){super(t),this.name="ProcessingError",this.token=e,this.code="PROCESSING_ERROR"}};E.exports={APIError:p,CacheError:j,ConfigError:y,ProcessingError:w}});var C=m((nt,b)=>{var f=require("fs");function q(n){try{let t=f.readFileSync(n,"utf8");return JSON.parse(t)}catch(t){if(t.code==="ENOENT")return null;throw t}}function G(n){try{return f.readFileSync(n,"utf8")}catch(t){if(t.code==="ENOENT")return null;throw t}}function H(n,t,e=2){let s=JSON.stringify(t,null,e);f.writeFileSync(n,s,"utf8")}function B(n,t){if(t==null||typeof t=="object"&&Object.keys(t).length===0||Array.isArray(t)&&t.length===0){let s=`module.exports = ${Array.isArray(t)?"[]":"{}"};`;f.writeFileSync(n,s,"utf8")}else{let e=`module.exports = ${JSON.stringify(t,null,2)};`;f.writeFileSync(n,e,"utf8")}}function Y(n){return f.existsSync(n)}function W(n){f.mkdirSync(n,{recursive:!0})}function z(n){f.rmSync(n,{recursive:!0,force:!0})}b.exports={readJsonFileSafe:q,readFileSafe:G,writeJsonFile:H,writeModuleExportsFile:B,fileExists:Y,ensureDirectoryExists:W,removeDirectory:z}});var S=m((rt,D)=>{var l=require("path"),{fileExists:g}=C();function K(n,t=process.cwd()){let e=t;for(;e!==l.parse(e).root;){let s=l.join(e,n);if(g(s))return s;e=l.dirname(e)}return null}function Q(n,t=process.cwd(),e=null){let s=[],r=t,o=10;for(let a=0;a<o;a++){s.push(l.join(r,n)),a<3&&s.push(l.join(r,"../".repeat(a+1)+n));let i=l.dirname(r);if(i===r)break;r=i}s.push(l.join(__dirname,"../../../",n));for(let a of s)if(g(a)&&(!e||e(a)))return a;return null}function X(n,t=process.cwd()){let e=n.map(s=>l.isAbsolute(s)?s:l.join(t,s));for(let s of e)if(g(s))return s;return null}D.exports={findFileUpward:K,findDirectoryUpward:Q,findDirectoryWithPaths:X}});var T=m((at,U)=>{var c=require("path"),{findDirectoryUpward:L}=S(),{readJsonFileSafe:u,readFileSafe:A,fileExists:k}=C(),F=class{constructor(t,e={}){if(this.ffId=t,this.tokensBasePath=e.tokensBasePath||this.findTokensPath(),this.folderMap=this.loadFolderMap(),this.projectFolder=this.folderMap[t],!this.projectFolder)throw new Error(`No folder mapping found for ff-id: ${t}`);this.projectPath=c.join(this.tokensBasePath,this.projectFolder)}findTokensPath(){let t=L("apps/tokens",process.cwd(),e=>k(c.join(e,"folderMap.json")));if(t||(t=L("tokens",process.cwd(),e=>k(c.join(e,"folderMap.json")))),!t)throw new Error("Could not find tokens directory with folderMap.json");return t}loadFolderMap(){let t=c.join(this.tokensBasePath,"folderMap.json"),e=u(t);if(!e)throw new Error("Failed to load folderMap.json: File not found");return e}async fetchTokens(){let t=c.join(this.projectPath,"figma"),e={global:[c.join(t,"Global Tokens","Default.json"),c.join(t,"global.json")],colors:[c.join(t,"Global Colors","Default.json"),c.join(t,"Global Colors","Light.json")],semantic:[c.join(t,"Semantic","Light.json"),c.join(t,"Semantic","Default.json")],semanticDark:[c.join(t,"Semantic","Dark.json")]},s={};for(let[r,o]of Object.entries(e)){for(let a of o){let i=u(a);if(i!==null){s[r]=i;break}}s[r]||(s[r]=null)}return s}async fetchProcessedTokens(){let t=c.join(this.projectPath,"cache");if(!k(t))return null;let e={"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"},s={};for(let[o,a]of Object.entries(e)){let i=c.join(t,a);o.endsWith(".js")?s[o]=A(i):s[o]=u(i)}return Object.values(s).some(o=>o!==null)?s:null}async fetchComponentsConfig(){return u(c.join(this.projectPath,"components-config.json"))}async fetchFonts(){return u(c.join(this.projectPath,"font.json"))}async fetchIcons(){return u(c.join(this.projectPath,"icons.json"))}async fetchVersion(){return u(c.join(this.projectPath,"version.json"))}async fetchCustomCss(){return A(c.join(this.projectPath,"custom.css"))}async fetchPlanMetadata(){return u(c.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.")}};U.exports={LocalTokenReader:F}});var I=require("https"),J=require("http"),{URL:it}=require("url"),{DEFAULT_API_URL:Z,LEGACY_API_URL:d,ENV_VARS:x}=_(),{APIError:h}=$(),{LocalTokenReader:tt}=T(),P=class{constructor(t,e={}){this.ffId=t,this.baseURL=e.baseURL||process.env[x.FF_API_URL]||Z,process.env[x.FF_USE_LOCAL]==="true"&&(console.log(" \u{1F3E0} Using local token reader"),this.localReader=new tt(t,e))}fetchJson(t){return new Promise((e,s)=>{(t.startsWith("https")?I:J).get(t,o=>{let a="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,t));return}o.on("data",i=>{a+=i}),o.on("end",()=>{try{let i=JSON.parse(a);e(i)}catch(i){s(new h(`Invalid JSON response: ${i.message}`,o.statusCode,t))}}),o.on("error",i=>{s(new h(`Response error: ${i.message}`,o.statusCode,t))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,t))})})}fetchRaw(t){return new Promise((e,s)=>{(t.startsWith("https")?I:J).get(t,o=>{let a="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,t));return}o.on("data",i=>{a+=i}),o.on("end",()=>{e(a)}),o.on("error",i=>{s(new h(`Response error: ${i.message}`,o.statusCode,t))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,t))})})}async fetchTokens(){if(this.localReader)return this.localReader.fetchTokens();try{let r=`${this.baseURL}/api/design-systems/${this.ffId}/tokens`,o=await this.fetchJson(r);if(o&&o.tokens){let a=o.tokens;return{global:a["Global Tokens/Default"]||a.global||null,colors:a["Global Colors/Default"]||a.colors||null,semantic:a["Semantic/Light"]||a["semantic/light"]||null,semanticDark:a["Semantic/Dark"]||a["semantic/dark"]||null}}}catch(r){if(r.statusCode===400)throw new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,`${this.baseURL}/api/design-systems/${this.ffId}/tokens`)}let t=d,e={global:`${t}/${this.ffId}/global-tokens/default.json`,colors:`${t}/${this.ffId}/global-colors/default.json`,semantic:`${t}/${this.ffId}/semantic/light.json`,semanticDark:`${t}/${this.ffId}/semantic/dark.json`},s=await Promise.all([this.fetchJson(e.global).catch(r=>r.statusCode===404?null:Promise.reject(r)),this.fetchJson(e.colors).catch(r=>r.statusCode===404?null:Promise.reject(r)),this.fetchJson(e.semantic).catch(r=>r.statusCode===404?null:Promise.reject(r)),this.fetchJson(e.semanticDark).catch(r=>r.statusCode===404?null:Promise.reject(r))]);return{global:s[0],colors:s[1],semantic:s[2],semanticDark:s[3]}}async fetchProcessedTokens(){if(this.localReader)return this.localReader.fetchProcessedTokens();let t=`${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e.statusCode===400?new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,t):e}}async fetchComponentsConfig(){if(this.localReader)return this.localReader.fetchComponentsConfig();let t=`${d}/${this.ffId}/components-config.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchFonts(){if(this.localReader)return this.localReader.fetchFonts();let t=`${d}/${this.ffId}/font.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchIcons(){if(this.localReader)return this.localReader.fetchIcons();let t=`${d}/${this.ffId}/icons.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchVersion(){if(this.localReader)return this.localReader.fetchVersion();let t=`${d}/${this.ffId}/version.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchCustomCss(){if(this.localReader)return this.localReader.fetchCustomCss();let t=`${d}/${this.ffId}/custom.css`;try{return await this.fetchRaw(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchPlanMetadata(){if(this.localReader)return process.env.FF_DEBUG&&console.log(" \u2192 Using local reader for plan metadata"),this.localReader.fetchPlanMetadata();let t=`${d}/${this.ffId}/metadata`;try{return process.env.FF_DEBUG&&console.log(` \u2192 Fetching plan metadata from: ${t}`),await this.fetchJson(t)}catch(e){if(e.statusCode===404)return process.env.FF_DEBUG&&console.log(" \u2192 No metadata found, defaulting to full plan"),{planType:"full",allowedComponents:null};throw e}}};module.exports={APIClient:P};
|
|
1
|
+
var p=(r,t)=>()=>(t||r((t={exports:{}}).exports,t),t.exports);var E=p((st,_)=>{var M="https://app.frontfriend.dev",x="https://tokens-studio-donux.up.railway.app/api",N=".cache/frontfriend",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"]},G={FF_ID:"FF_ID",FF_API_URL:"FF_API_URL",FF_USE_LOCAL:"FF_USE_LOCAL"};_.exports={DEFAULT_API_URL:M,LEGACY_API_URL:x,CACHE_DIR_NAME:N,CACHE_TTL_DAYS:7,CACHE_TTL_MS:6048e5,CACHE_FILES:O,ENV_VARS:G}});var R=p((ot,$)=>{var m=class extends Error{constructor(t,e,s){super(t),this.name="APIError",this.statusCode=e,this.url=s,this.code=`API_${e}`}},j=class extends Error{constructor(t,e){super(t),this.name="CacheError",this.operation=e,this.code=`CACHE_${e.toUpperCase()}`}},y=class extends Error{constructor(t,e){super(t),this.name="ConfigError",this.field=e,this.code=`CONFIG_${e.toUpperCase()}`}},w=class extends Error{constructor(t,e){super(t),this.name="ProcessingError",this.token=e,this.code="PROCESSING_ERROR"}};$.exports={APIError:m,CacheError:j,ConfigError:y,ProcessingError:w}});var g=p((nt,D)=>{var f=require("fs");function V(r){try{let t=f.readFileSync(r,"utf8");return JSON.parse(t)}catch(t){if(t.code==="ENOENT")return null;throw t}}function q(r){try{return f.readFileSync(r,"utf8")}catch(t){if(t.code==="ENOENT")return null;throw t}}function B(r,t,e=2){let s=JSON.stringify(t,null,e);f.writeFileSync(r,s,"utf8")}function H(r,t){if(t==null||typeof t=="object"&&Object.keys(t).length===0||Array.isArray(t)&&t.length===0){let s=`module.exports = ${Array.isArray(t)?"[]":"{}"};`;f.writeFileSync(r,s,"utf8")}else{let e=`module.exports = ${JSON.stringify(t,null,2)};`;f.writeFileSync(r,e,"utf8")}}function Y(r){return f.existsSync(r)}function W(r){f.mkdirSync(r,{recursive:!0})}function z(r){f.rmSync(r,{recursive:!0,force:!0})}D.exports={readJsonFileSafe:V,readFileSafe:q,writeJsonFile:B,writeModuleExportsFile:H,fileExists:Y,ensureDirectoryExists:W,removeDirectory:z}});var S=p((rt,b)=>{var l=require("path"),{fileExists:C}=g();function K(r,t=process.cwd()){let e=t;for(;e!==l.parse(e).root;){let s=l.join(e,r);if(C(s))return s;e=l.dirname(e)}return null}function Q(r,t=process.cwd(),e=null){let s=[],n=t,o=10;for(let a=0;a<o;a++){s.push(l.join(n,r)),a<3&&s.push(l.join(n,"../".repeat(a+1)+r));let i=l.dirname(n);if(i===n)break;n=i}s.push(l.join(__dirname,"../../../",r));for(let a of s)if(C(a)&&(!e||e(a)))return a;return null}function X(r,t=process.cwd()){let e=r.map(s=>l.isAbsolute(s)?s:l.join(t,s));for(let s of e)if(C(s))return s;return null}b.exports={findFileUpward:K,findDirectoryUpward:Q,findDirectoryWithPaths:X}});var A=p((at,L)=>{var c=require("path"),{findDirectoryUpward:U}=S(),{readJsonFileSafe:d,readFileSafe:T,fileExists:F}=g(),k=class{constructor(t,e={}){if(this.ffId=t,this.tokensBasePath=e.tokensBasePath||this.findTokensPath(),this.folderMap=this.loadFolderMap(),this.projectFolder=this.folderMap[t],!this.projectFolder)throw new Error(`No folder mapping found for ff-id: ${t}`);this.projectPath=c.join(this.tokensBasePath,this.projectFolder)}findTokensPath(){let t=U("apps/tokens",process.cwd(),e=>F(c.join(e,"folderMap.json")));if(t||(t=U("tokens",process.cwd(),e=>F(c.join(e,"folderMap.json")))),!t)throw new Error("Could not find tokens directory with folderMap.json");return t}loadFolderMap(){let t=c.join(this.tokensBasePath,"folderMap.json"),e=d(t);if(!e)throw new Error("Failed to load folderMap.json: File not found");return e}async fetchTokens(){let t=c.join(this.projectPath,"figma"),e={global:[c.join(t,"Global Tokens","Default.json"),c.join(t,"global.json")],colors:[c.join(t,"Global Colors","Default.json"),c.join(t,"Global Colors","Light.json")],semantic:[c.join(t,"Semantic","Light.json"),c.join(t,"Semantic","Default.json")],semanticDark:[c.join(t,"Semantic","Dark.json")]},s={};for(let[n,o]of Object.entries(e)){for(let a of o){let i=d(a);if(i!==null){s[n]=i;break}}s[n]||(s[n]=null)}return s}async fetchProcessedTokens(){let t=c.join(this.projectPath,"cache");if(!F(t))return null;let e={"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"},s={};for(let[o,a]of Object.entries(e)){let i=c.join(t,a);o.endsWith(".js")?s[o]=T(i):s[o]=d(i)}return Object.values(s).some(o=>o!==null)?s:null}async fetchComponentsConfig(){return d(c.join(this.projectPath,"components-config.json"))}async fetchFonts(){return d(c.join(this.projectPath,"font.json"))}async fetchIcons(){return d(c.join(this.projectPath,"icons.json"))}async fetchVersion(){return d(c.join(this.projectPath,"version.json"))}async fetchCustomCss(){return T(c.join(this.projectPath,"custom.css"))}async fetchPlanMetadata(){return d(c.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.")}};L.exports={LocalTokenReader:k}});var I=require("https"),v=require("http"),{URL:it}=require("url"),{DEFAULT_API_URL:Z,LEGACY_API_URL:u,ENV_VARS:J}=E(),{APIError:h}=R(),{LocalTokenReader:tt}=A(),P=class{constructor(t,e={}){this.ffId=t,this.baseURL=e.baseURL||process.env[J.FF_API_URL]||Z,process.env[J.FF_USE_LOCAL]==="true"&&(console.log(" \u{1F3E0} Using local token reader"),this.localReader=new tt(t,e))}fetchJson(t){return new Promise((e,s)=>{(t.startsWith("https")?I:v).get(t,o=>{let a="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,t));return}o.on("data",i=>{a+=i}),o.on("end",()=>{try{let i=JSON.parse(a);e(i)}catch(i){s(new h(`Invalid JSON response: ${i.message}`,o.statusCode,t))}}),o.on("error",i=>{s(new h(`Response error: ${i.message}`,o.statusCode,t))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,t))})})}fetchRaw(t){return new Promise((e,s)=>{(t.startsWith("https")?I:v).get(t,o=>{let a="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,t));return}o.on("data",i=>{a+=i}),o.on("end",()=>{e(a)}),o.on("error",i=>{s(new h(`Response error: ${i.message}`,o.statusCode,t))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,t))})})}async fetchTokens(){if(this.localReader)return this.localReader.fetchTokens();try{let n=`${this.baseURL}/api/design-systems/${this.ffId}/tokens`,o=await this.fetchJson(n);if(o&&o.tokens){let a=o.tokens;return{global:a["Global Tokens/Default"]||a.global||null,colors:a["Global Colors/Default"]||a.colors||null,semantic:a["Semantic/Light"]||a["semantic/light"]||null,semanticDark:a["Semantic/Dark"]||a["semantic/dark"]||null}}}catch(n){if(n.statusCode===400)throw new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,`${this.baseURL}/api/design-systems/${this.ffId}/tokens`)}let t=u,e={global:`${t}/${this.ffId}/global-tokens/default.json`,colors:`${t}/${this.ffId}/global-colors/default.json`,semantic:`${t}/${this.ffId}/semantic/light.json`,semanticDark:`${t}/${this.ffId}/semantic/dark.json`},s=await Promise.all([this.fetchJson(e.global).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(e.colors).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(e.semantic).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(e.semanticDark).catch(n=>n.statusCode===404?null:Promise.reject(n))]);return{global:s[0],colors:s[1],semantic:s[2],semanticDark:s[3]}}async fetchProcessedTokens(){if(this.localReader)return this.localReader.fetchProcessedTokens();let t=`${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e.statusCode===400?new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,t):e}}async fetchComponentsConfig(){if(this.localReader)return this.localReader.fetchComponentsConfig();let t=`${u}/${this.ffId}/components-config.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchFonts(){if(this.localReader)return this.localReader.fetchFonts();let t=`${u}/${this.ffId}/font.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchIcons(){if(this.localReader)return this.localReader.fetchIcons();let t=`${u}/${this.ffId}/icons.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchVersion(){if(this.localReader)return this.localReader.fetchVersion();let t=`${u}/${this.ffId}/version.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchCustomCss(){if(this.localReader)return this.localReader.fetchCustomCss();let t=`${u}/${this.ffId}/custom.css`;try{return await this.fetchRaw(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchPlanMetadata(){if(this.localReader)return process.env.FF_DEBUG&&console.log(" \u2192 Using local reader for plan metadata"),this.localReader.fetchPlanMetadata();let t=`${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;try{process.env.FF_DEBUG&&console.log(` \u2192 Fetching plan metadata from saas app: ${t}`);let e=await this.fetchJson(t);if(e&&e.metadata){let s={planType:e.metadata.planType||"full",allowedComponents:e.metadata.allowedComponents||null};return process.env.FF_DEBUG&&console.log(` \u2192 Found plan metadata in saas app: ${s.planType}, allowed: ${s.allowedComponents?s.allowedComponents.length+" components":"all"}`),s}}catch(e){if(e.statusCode===404){process.env.FF_DEBUG&&console.log(" \u2192 ff-id not found in saas app, trying legacy server");let s=`${u}/${this.ffId}/metadata`;try{process.env.FF_DEBUG&&console.log(` \u2192 Fetching from legacy server: ${s}`);let n=await this.fetchJson(s),o={planType:n.planType||"full",allowedComponents:n.allowedComponents||null};return process.env.FF_DEBUG&&console.log(` \u2192 Found in legacy server: ${o.planType}, allowed: ${o.allowedComponents?o.allowedComponents.length+" components":"all"}`),o}catch(n){if(n.statusCode===404)return process.env.FF_DEBUG&&console.log(" \u2192 ff-id not found in any server - access denied"),{planType:"denied",allowedComponents:[],error:"Design system not found. Please ensure your ff-id is valid."};throw n}}throw e}}};module.exports={APIClient:P};
|
|
2
2
|
//# sourceMappingURL=api-client.js.map
|
|
@@ -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"],
|
|
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 };"],
|
|
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", "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"]
|
|
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 };"],
|
|
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", "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"]
|
|
7
7
|
}
|
|
@@ -105,6 +105,12 @@ class CacheManager {
|
|
|
105
105
|
delete data['components-config'];
|
|
106
106
|
}
|
|
107
107
|
|
|
108
|
+
// Backward compatibility: rename cls to safelist if present
|
|
109
|
+
if (data.cls && !data.safelist) {
|
|
110
|
+
data.safelist = data.cls;
|
|
111
|
+
delete data.cls;
|
|
112
|
+
}
|
|
113
|
+
|
|
108
114
|
return data;
|
|
109
115
|
} catch (error) {
|
|
110
116
|
throw new CacheError(`Failed to load cache: ${error.message}`, 'read');
|
|
@@ -135,7 +141,7 @@ class CacheManager {
|
|
|
135
141
|
'variables',
|
|
136
142
|
'semanticVariables',
|
|
137
143
|
'semanticDarkVariables',
|
|
138
|
-
'
|
|
144
|
+
'safelist',
|
|
139
145
|
'custom'
|
|
140
146
|
];
|
|
141
147
|
|
|
@@ -145,6 +151,12 @@ class CacheManager {
|
|
|
145
151
|
writeModuleExportsFile(filePath, data[key]);
|
|
146
152
|
}
|
|
147
153
|
}
|
|
154
|
+
|
|
155
|
+
// Also check for safelist specifically since it might be an array
|
|
156
|
+
if (data.safelist && Array.isArray(data.safelist)) {
|
|
157
|
+
const safelistPath = path.join(this.cacheDir, 'safelist.js');
|
|
158
|
+
writeModuleExportsFile(safelistPath, data.safelist);
|
|
159
|
+
}
|
|
148
160
|
|
|
149
161
|
// Save .json files
|
|
150
162
|
const jsonData = {
|
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
var w=(i,t)=>()=>(t||i((t={exports:{}}).exports,t),t.exports);var U=w((ht,R)=>{var H="https://app.frontfriend.dev",Y="https://tokens-studio-donux.up.railway.app/api",W=".cache/frontfriend",z={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"]},K={FF_ID:"FF_ID",FF_API_URL:"FF_API_URL",FF_USE_LOCAL:"FF_USE_LOCAL"};R.exports={DEFAULT_API_URL:H,LEGACY_API_URL:Y,CACHE_DIR_NAME:W,CACHE_TTL_DAYS:7,CACHE_TTL_MS:6048e5,CACHE_FILES:z,ENV_VARS:K}});var L=w((dt,A)=>{var j=class extends Error{constructor(t,e,s){super(t),this.name="APIError",this.statusCode=e,this.url=s,this.code=`API_${e}`}},C=class extends Error{constructor(t,e){super(t),this.name="CacheError",this.operation=e,this.code=`CACHE_${e.toUpperCase()}`}},F=class extends Error{constructor(t,e){super(t),this.name="ConfigError",this.field=e,this.code=`CONFIG_${e.toUpperCase()}`}},$=class extends Error{constructor(t,e){super(t),this.name="ProcessingError",this.token=e,this.code="PROCESSING_ERROR"}};A.exports={APIError:j,CacheError:C,ConfigError:F,ProcessingError:$}});var k=w((ut,I)=>{var d=require("fs");function Q(i){try{let t=d.readFileSync(i,"utf8");return JSON.parse(t)}catch(t){if(t.code==="ENOENT")return null;throw t}}function X(i){try{return d.readFileSync(i,"utf8")}catch(t){if(t.code==="ENOENT")return null;throw t}}function Z(i,t,e=2){let s=JSON.stringify(t,null,e);d.writeFileSync(i,s,"utf8")}function tt(i,t){if(t==null||typeof t=="object"&&Object.keys(t).length===0||Array.isArray(t)&&t.length===0){let s=`module.exports = ${Array.isArray(t)?"[]":"{}"};`;d.writeFileSync(i,s,"utf8")}else{let e=`module.exports = ${JSON.stringify(t,null,2)};`;d.writeFileSync(i,e,"utf8")}}function et(i){return d.existsSync(i)}function st(i){d.mkdirSync(i,{recursive:!0})}function ot(i){d.rmSync(i,{recursive:!0,force:!0})}I.exports={readJsonFileSafe:Q,readFileSafe:X,writeJsonFile:Z,writeModuleExportsFile:tt,fileExists:et,ensureDirectoryExists:st,removeDirectory:ot}});var b=w((pt,T)=>{var f=require("path"),{fileExists:P}=k();function nt(i,t=process.cwd()){let e=t;for(;e!==f.parse(e).root;){let s=f.join(e,i);if(P(s))return s;e=f.dirname(e)}return null}function rt(i,t=process.cwd(),e=null){let s=[],n=t,o=10;for(let r=0;r<o;r++){s.push(f.join(n,i)),r<3&&s.push(f.join(n,"../".repeat(r+1)+i));let a=f.dirname(n);if(a===n)break;n=a}s.push(f.join(__dirname,"../../../",i));for(let r of s)if(P(r)&&(!e||e(r)))return r;return null}function at(i,t=process.cwd()){let e=i.map(s=>f.isAbsolute(s)?s:f.join(t,s));for(let s of e)if(P(s))return s;return null}T.exports={findFileUpward:nt,findDirectoryUpward:rt,findDirectoryWithPaths:at}});var x=w((mt,J)=>{var c=require("path"),{findDirectoryUpward:v}=b(),{readJsonFileSafe:u,readFileSafe:M,fileExists:E}=k(),_=class{constructor(t,e={}){if(this.ffId=t,this.tokensBasePath=e.tokensBasePath||this.findTokensPath(),this.folderMap=this.loadFolderMap(),this.projectFolder=this.folderMap[t],!this.projectFolder)throw new Error(`No folder mapping found for ff-id: ${t}`);this.projectPath=c.join(this.tokensBasePath,this.projectFolder)}findTokensPath(){let t=v("apps/tokens",process.cwd(),e=>E(c.join(e,"folderMap.json")));if(t||(t=v("tokens",process.cwd(),e=>E(c.join(e,"folderMap.json")))),!t)throw new Error("Could not find tokens directory with folderMap.json");return t}loadFolderMap(){let t=c.join(this.tokensBasePath,"folderMap.json"),e=u(t);if(!e)throw new Error("Failed to load folderMap.json: File not found");return e}async fetchTokens(){let t=c.join(this.projectPath,"figma"),e={global:[c.join(t,"Global Tokens","Default.json"),c.join(t,"global.json")],colors:[c.join(t,"Global Colors","Default.json"),c.join(t,"Global Colors","Light.json")],semantic:[c.join(t,"Semantic","Light.json"),c.join(t,"Semantic","Default.json")],semanticDark:[c.join(t,"Semantic","Dark.json")]},s={};for(let[n,o]of Object.entries(e)){for(let r of o){let a=u(r);if(a!==null){s[n]=a;break}}s[n]||(s[n]=null)}return s}async fetchProcessedTokens(){let t=c.join(this.projectPath,"cache");if(!E(t))return null;let e={"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"},s={};for(let[o,r]of Object.entries(e)){let a=c.join(t,r);o.endsWith(".js")?s[o]=M(a):s[o]=u(a)}return Object.values(s).some(o=>o!==null)?s:null}async fetchComponentsConfig(){return u(c.join(this.projectPath,"components-config.json"))}async fetchFonts(){return u(c.join(this.projectPath,"font.json"))}async fetchIcons(){return u(c.join(this.projectPath,"icons.json"))}async fetchVersion(){return u(c.join(this.projectPath,"version.json"))}async fetchCustomCss(){return M(c.join(this.projectPath,"custom.css"))}async fetchPlanMetadata(){return u(c.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.")}};J.exports={LocalTokenReader:_}});var q=w((gt,N)=>{var O=require("https"),G=require("http"),{URL:wt}=require("url"),{DEFAULT_API_URL:it,LEGACY_API_URL:m,ENV_VARS:B}=U(),{APIError:h}=L(),{LocalTokenReader:ct}=x(),D=class{constructor(t,e={}){this.ffId=t,this.baseURL=e.baseURL||process.env[B.FF_API_URL]||it,process.env[B.FF_USE_LOCAL]==="true"&&(console.log(" \u{1F3E0} Using local token reader"),this.localReader=new ct(t,e))}fetchJson(t){return new Promise((e,s)=>{(t.startsWith("https")?O:G).get(t,o=>{let r="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,t));return}o.on("data",a=>{r+=a}),o.on("end",()=>{try{let a=JSON.parse(r);e(a)}catch(a){s(new h(`Invalid JSON response: ${a.message}`,o.statusCode,t))}}),o.on("error",a=>{s(new h(`Response error: ${a.message}`,o.statusCode,t))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,t))})})}fetchRaw(t){return new Promise((e,s)=>{(t.startsWith("https")?O:G).get(t,o=>{let r="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,t));return}o.on("data",a=>{r+=a}),o.on("end",()=>{e(r)}),o.on("error",a=>{s(new h(`Response error: ${a.message}`,o.statusCode,t))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,t))})})}async fetchTokens(){if(this.localReader)return this.localReader.fetchTokens();try{let n=`${this.baseURL}/api/design-systems/${this.ffId}/tokens`,o=await this.fetchJson(n);if(o&&o.tokens){let r=o.tokens;return{global:r["Global Tokens/Default"]||r.global||null,colors:r["Global Colors/Default"]||r.colors||null,semantic:r["Semantic/Light"]||r["semantic/light"]||null,semanticDark:r["Semantic/Dark"]||r["semantic/dark"]||null}}}catch(n){if(n.statusCode===400)throw new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,`${this.baseURL}/api/design-systems/${this.ffId}/tokens`)}let t=m,e={global:`${t}/${this.ffId}/global-tokens/default.json`,colors:`${t}/${this.ffId}/global-colors/default.json`,semantic:`${t}/${this.ffId}/semantic/light.json`,semanticDark:`${t}/${this.ffId}/semantic/dark.json`},s=await Promise.all([this.fetchJson(e.global).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(e.colors).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(e.semantic).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(e.semanticDark).catch(n=>n.statusCode===404?null:Promise.reject(n))]);return{global:s[0],colors:s[1],semantic:s[2],semanticDark:s[3]}}async fetchProcessedTokens(){if(this.localReader)return this.localReader.fetchProcessedTokens();let t=`${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e.statusCode===400?new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,t):e}}async fetchComponentsConfig(){if(this.localReader)return this.localReader.fetchComponentsConfig();let t=`${m}/${this.ffId}/components-config.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchFonts(){if(this.localReader)return this.localReader.fetchFonts();let t=`${m}/${this.ffId}/font.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchIcons(){if(this.localReader)return this.localReader.fetchIcons();let t=`${m}/${this.ffId}/icons.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchVersion(){if(this.localReader)return this.localReader.fetchVersion();let t=`${m}/${this.ffId}/version.json`;try{return await this.fetchJson(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchCustomCss(){if(this.localReader)return this.localReader.fetchCustomCss();let t=`${m}/${this.ffId}/custom.css`;try{return await this.fetchRaw(t)}catch(e){if(e.statusCode===404)return null;throw e}}async fetchPlanMetadata(){if(this.localReader)return process.env.FF_DEBUG&&console.log(" \u2192 Using local reader for plan metadata"),this.localReader.fetchPlanMetadata();let t=`${m}/${this.ffId}/metadata`;try{return process.env.FF_DEBUG&&console.log(` \u2192 Fetching plan metadata from: ${t}`),await this.fetchJson(t)}catch(e){if(e.statusCode===404)return process.env.FF_DEBUG&&console.log(" \u2192 No metadata found, defaulting to full plan"),{planType:"full",allowedComponents:null};throw e}}};N.exports={APIClient:D}});var lt=require("https"),p=require("fs"),l=require("path"),{URL:yt}=require("url"),V=require("os"),g=process.env.FF_REGISTRY_URL||"https://ff-registry.up.railway.app",S=class{constructor(t={}){this.appRoot=t.appRoot||process.cwd(),this.framework=t.framework||this.detectFramework(),this.config=t.config||{},this.outputPath=t.outputPath||this.getDefaultOutputPath(),this.overwrite=t.overwrite||!1,this.downloadedComponents=new Set,this.ffId=this.config.ffId||this.config["ff-id"]||null,this.planMetadata=null}detectFramework(){let t=l.join(this.appRoot,"package.json");if(p.existsSync(t)){let e=JSON.parse(p.readFileSync(t,"utf8")),s={...e.dependencies,...e.devDependencies};if(s.vue)return"vue";if(s.react)return"react"}return"react"}getDefaultOutputPath(){if(this.config.aliases&&this.config.aliases.ui){let e=this.config.aliases.ui.replace(/^@\//,"");return l.join(this.appRoot,e)}return this.framework==="vue"?l.join(this.appRoot,"src/components/ui"):l.join(this.appRoot,"components/ui")}fetchJson(t){if(process.env.FF_USE_LOCAL==="true"){let s=["../../packages/registry","../../../packages/registry","../../../../packages/registry",l.join(__dirname,"../../../registry"),"packages/registry"],n;for(let a of s){let y=l.isAbsolute(a)?a:l.join(process.cwd(),a);if(p.existsSync(y)){n=y;break}}if(!n)return Promise.reject(new Error(`Local registry not found. Tried paths: ${s.join(", ")}`));let o=t.replace(g,"").replace(/^\//,""),r=l.join(n,...o.split("/"));try{let a=p.readFileSync(r,"utf8");return Promise.resolve(JSON.parse(a))}catch(a){return a.code==="ENOENT"?Promise.reject(new Error("HTTP 404: Not Found")):Promise.reject(a)}}return new Promise((s,n)=>{lt.get(t,o=>{let r="";if(o.statusCode>=400){n(new Error(`HTTP ${o.statusCode}: ${o.statusMessage}`));return}o.on("data",a=>{r+=a}),o.on("end",()=>{try{s(JSON.parse(r))}catch(a){n(new Error(`Invalid JSON response: ${a.message}`))}})}).on("error",o=>{n(o)})})}async getAllComponents(){try{let t=`${g}/components/${this.framework}/index.json`,s=(await this.fetchJson(t)).components||[];return this.ffId&&(await this.isComponentAllowed("dummy"),this.planMetadata&&this.planMetadata.planType==="trial"&&this.planMetadata.allowedComponents&&(s=s.filter(n=>this.planMetadata.allowedComponents.includes(n)),process.env.FF_DEBUG&&console.log(` \u2192 Trial plan: filtering to ${s.length} allowed components`))),s}catch(t){return console.error("Failed to fetch component index:",t.message),[]}}async getCustomComponents(){if(!this.ffId)return process.env.FF_DEBUG&&console.log(" \u2192 No ff-id found, skipping custom components"),[];try{let t=`${g}/custom/${this.ffId}/${this.framework}/index.json`;process.env.FF_DEBUG&&console.log(` \u2192 Fetching custom components from: ${t}`);let e=await this.fetchJson(t);return process.env.FF_DEBUG&&console.log(` \u2192 Found custom components: ${JSON.stringify(e.components)}`),e.components||[]}catch(t){return process.env.FF_DEBUG&&console.log(` \u2192 No custom components found: ${t.message}`),[]}}async downloadComponents(t){process.env.FF_DEBUG&&console.log(` \u2192 ComponentDownloader initialized with ff-id: ${this.ffId}`);let e={successful:[],failed:[],dependencies:new Set};for(let s of t)try{let n=await this.downloadComponent(s);e.successful.push(s),n.forEach(o=>e.dependencies.add(o))}catch(n){e.failed.push({component:s,error:n.message})}return{successful:e.successful,failed:e.failed,dependencies:Array.from(e.dependencies)}}async isComponentAllowed(t){if(!this.planMetadata&&this.ffId){let{APIClient:e}=q(),s=new e(this.ffId,{baseURL:this.config["api-url"]||this.config.apiUrl});try{this.planMetadata=await s.fetchPlanMetadata(),process.env.FF_DEBUG&&console.log(` \u2192 Fetched plan metadata: ${this.planMetadata.planType}, allowed: ${this.planMetadata.allowedComponents?this.planMetadata.allowedComponents.length+" components":"all"}`)}catch(n){console.warn("Failed to fetch plan metadata:",n.message),this.planMetadata={planType:"full",allowedComponents:null}}}return!this.planMetadata||this.planMetadata.planType==="full"?!0:this.planMetadata.planType==="trial"&&this.planMetadata.allowedComponents?this.planMetadata.allowedComponents.includes(t):!0}async downloadComponent(t){if(this.downloadedComponents.has(t))return[];if(process.env.FF_DEBUG&&(console.log(` \u2192 Checking if component "${t}" is allowed...`),console.log(` \u2192 FF-ID: ${this.ffId}`)),!await this.isComponentAllowed(t)){let r=`Component "${t}" is not available in your trial plan. Please upgrade to access all components.`;throw console.error(`\u274C ${r}`),new Error(r)}console.log(`\u{1F4E5} Downloading ${t}...`);let s,n=!1;if(this.ffId){let r=`${g}/custom/${this.ffId}/${this.framework}/${t}.json`;try{s=await this.fetchJson(r),n=!0,process.env.FF_DEBUG&&console.log(` \u2192 Found custom component for ${t}`)}catch{process.env.FF_DEBUG&&console.log(` \u2192 No custom component found for ${t}, trying standard`)}}if(!s){let r=`${g}/components/${this.framework}/${t}.json`;try{s=await this.fetchJson(r)}catch(a){throw a.message.includes("404")?new Error(`Component "${t}" not found`):new Error(`Failed to fetch ${t}: ${a.message}`)}}if(!s.files||!Array.isArray(s.files))throw new Error(`Invalid component data for ${t}`);for(let r of s.files)await this.writeComponentFile(r);this.downloadedComponents.add(t),console.log(` \u2713 ${t} downloaded${n&&process.env.FF_DEBUG?" (custom)":""}`);let o=[...s.dependencies||[]];if(s.registryDependencies&&s.registryDependencies.length>0){console.log(` \u{1F4E6} Installing registry dependencies: ${s.registryDependencies.join(", ")}`);for(let r of s.registryDependencies)try{(await this.downloadComponent(r)).forEach(y=>o.push(y))}catch(a){console.warn(` \u26A0\uFE0F Failed to download dependency ${r}: ${a.message}`)}}return o}async writeComponentFile(t){let e=t.name.split("/").join(l.sep),s=l.join(this.outputPath,e),n=l.dirname(s);if(p.existsSync(n)||p.mkdirSync(n,{recursive:!0}),p.existsSync(s)&&!this.overwrite){console.log(` \u26A0\uFE0F Skipping ${t.name} (already exists)`);return}let o=t.content;V.EOL!==`
|
|
2
|
-
`&&(o=
|
|
1
|
+
var y=(i,e)=>()=>(e||i((e={exports:{}}).exports,e),e.exports);var v=y((ue,T)=>{var W="https://app.frontfriend.dev",z="https://tokens-studio-donux.up.railway.app/api",K=".cache/frontfriend",Q={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"]},X={FF_ID:"FF_ID",FF_API_URL:"FF_API_URL",FF_USE_LOCAL:"FF_USE_LOCAL"};T.exports={DEFAULT_API_URL:W,LEGACY_API_URL:z,CACHE_DIR_NAME:K,CACHE_TTL_DAYS:7,CACHE_TTL_MS:6048e5,CACHE_FILES:Q,ENV_VARS:X}});var L=y((pe,A)=>{var F=class extends Error{constructor(e,t,s){super(e),this.name="APIError",this.statusCode=t,this.url=s,this.code=`API_${t}`}},C=class extends Error{constructor(e,t){super(e),this.name="CacheError",this.operation=t,this.code=`CACHE_${t.toUpperCase()}`}},$=class extends Error{constructor(e,t){super(e),this.name="ConfigError",this.field=t,this.code=`CONFIG_${t.toUpperCase()}`}},E=class extends Error{constructor(e,t){super(e),this.name="ProcessingError",this.token=t,this.code="PROCESSING_ERROR"}};A.exports={APIError:F,CacheError:C,ConfigError:$,ProcessingError:E}});var k=y((me,I)=>{var u=require("fs");function Z(i){try{let e=u.readFileSync(i,"utf8");return JSON.parse(e)}catch(e){if(e.code==="ENOENT")return null;throw e}}function ee(i){try{return u.readFileSync(i,"utf8")}catch(e){if(e.code==="ENOENT")return null;throw e}}function te(i,e,t=2){let s=JSON.stringify(e,null,t);u.writeFileSync(i,s,"utf8")}function se(i,e){if(e==null||typeof e=="object"&&Object.keys(e).length===0||Array.isArray(e)&&e.length===0){let s=`module.exports = ${Array.isArray(e)?"[]":"{}"};`;u.writeFileSync(i,s,"utf8")}else{let t=`module.exports = ${JSON.stringify(e,null,2)};`;u.writeFileSync(i,t,"utf8")}}function oe(i){return u.existsSync(i)}function ne(i){u.mkdirSync(i,{recursive:!0})}function re(i){u.rmSync(i,{recursive:!0,force:!0})}I.exports={readJsonFileSafe:Z,readFileSafe:ee,writeJsonFile:te,writeModuleExportsFile:se,fileExists:oe,ensureDirectoryExists:ne,removeDirectory:re}});var M=y((we,b)=>{var d=require("path"),{fileExists:P}=k();function ae(i,e=process.cwd()){let t=e;for(;t!==d.parse(t).root;){let s=d.join(t,i);if(P(s))return s;t=d.dirname(t)}return null}function ie(i,e=process.cwd(),t=null){let s=[],n=e,o=10;for(let r=0;r<o;r++){s.push(d.join(n,i)),r<3&&s.push(d.join(n,"../".repeat(r+1)+i));let a=d.dirname(n);if(a===n)break;n=a}s.push(d.join(__dirname,"../../../",i));for(let r of s)if(P(r)&&(!t||t(r)))return r;return null}function ce(i,e=process.cwd()){let t=i.map(s=>d.isAbsolute(s)?s:d.join(e,s));for(let s of t)if(P(s))return s;return null}b.exports={findFileUpward:ae,findDirectoryUpward:ie,findDirectoryWithPaths:ce}});var O=y((ge,G)=>{var c=require("path"),{findDirectoryUpward:J}=M(),{readJsonFileSafe:p,readFileSafe:x,fileExists:_}=k(),D=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=c.join(this.tokensBasePath,this.projectFolder)}findTokensPath(){let e=J("apps/tokens",process.cwd(),t=>_(c.join(t,"folderMap.json")));if(e||(e=J("tokens",process.cwd(),t=>_(c.join(t,"folderMap.json")))),!e)throw new Error("Could not find tokens directory with folderMap.json");return e}loadFolderMap(){let e=c.join(this.tokensBasePath,"folderMap.json"),t=p(e);if(!t)throw new Error("Failed to load folderMap.json: File not found");return t}async fetchTokens(){let e=c.join(this.projectPath,"figma"),t={global:[c.join(e,"Global Tokens","Default.json"),c.join(e,"global.json")],colors:[c.join(e,"Global Colors","Default.json"),c.join(e,"Global Colors","Light.json")],semantic:[c.join(e,"Semantic","Light.json"),c.join(e,"Semantic","Default.json")],semanticDark:[c.join(e,"Semantic","Dark.json")]},s={};for(let[n,o]of Object.entries(t)){for(let r of o){let a=p(r);if(a!==null){s[n]=a;break}}s[n]||(s[n]=null)}return s}async fetchProcessedTokens(){let e=c.join(this.projectPath,"cache");if(!_(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"},s={};for(let[o,r]of Object.entries(t)){let a=c.join(e,r);o.endsWith(".js")?s[o]=x(a):s[o]=p(a)}return Object.values(s).some(o=>o!==null)?s:null}async fetchComponentsConfig(){return p(c.join(this.projectPath,"components-config.json"))}async fetchFonts(){return p(c.join(this.projectPath,"font.json"))}async fetchIcons(){return p(c.join(this.projectPath,"icons.json"))}async fetchVersion(){return p(c.join(this.projectPath,"version.json"))}async fetchCustomCss(){return x(c.join(this.projectPath,"custom.css"))}async fetchPlanMetadata(){return p(c.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.")}};G.exports={LocalTokenReader:D}});var H=y((je,V)=>{var B=require("https"),q=require("http"),{URL:ye}=require("url"),{DEFAULT_API_URL:le,LEGACY_API_URL:w,ENV_VARS:N}=v(),{APIError:h}=L(),{LocalTokenReader:fe}=O(),R=class{constructor(e,t={}){this.ffId=e,this.baseURL=t.baseURL||process.env[N.FF_API_URL]||le,process.env[N.FF_USE_LOCAL]==="true"&&(console.log(" \u{1F3E0} Using local token reader"),this.localReader=new fe(e,t))}fetchJson(e){return new Promise((t,s)=>{(e.startsWith("https")?B:q).get(e,o=>{let r="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,e));return}o.on("data",a=>{r+=a}),o.on("end",()=>{try{let a=JSON.parse(r);t(a)}catch(a){s(new h(`Invalid JSON response: ${a.message}`,o.statusCode,e))}}),o.on("error",a=>{s(new h(`Response error: ${a.message}`,o.statusCode,e))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,e))})})}fetchRaw(e){return new Promise((t,s)=>{(e.startsWith("https")?B:q).get(e,o=>{let r="";if(o.statusCode>=400){s(new h(`HTTP ${o.statusCode}: ${o.statusMessage}`,o.statusCode,e));return}o.on("data",a=>{r+=a}),o.on("end",()=>{t(r)}),o.on("error",a=>{s(new h(`Response error: ${a.message}`,o.statusCode,e))})}).on("error",o=>{s(new h(`Network error: ${o.message}`,0,e))})})}async fetchTokens(){if(this.localReader)return this.localReader.fetchTokens();try{let n=`${this.baseURL}/api/design-systems/${this.ffId}/tokens`,o=await this.fetchJson(n);if(o&&o.tokens){let r=o.tokens;return{global:r["Global Tokens/Default"]||r.global||null,colors:r["Global Colors/Default"]||r.colors||null,semantic:r["Semantic/Light"]||r["semantic/light"]||null,semanticDark:r["Semantic/Dark"]||r["semantic/dark"]||null}}}catch(n){if(n.statusCode===400)throw new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,`${this.baseURL}/api/design-systems/${this.ffId}/tokens`)}let e=w,t={global:`${e}/${this.ffId}/global-tokens/default.json`,colors:`${e}/${this.ffId}/global-colors/default.json`,semantic:`${e}/${this.ffId}/semantic/light.json`,semanticDark:`${e}/${this.ffId}/semantic/dark.json`},s=await Promise.all([this.fetchJson(t.global).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(t.colors).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(t.semantic).catch(n=>n.statusCode===404?null:Promise.reject(n)),this.fetchJson(t.semanticDark).catch(n=>n.statusCode===404?null:Promise.reject(n))]);return{global:s[0],colors:s[1],semantic:s[2],semanticDark:s[3]}}async fetchProcessedTokens(){if(this.localReader)return this.localReader.fetchProcessedTokens();let e=`${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;try{return await this.fetchJson(e)}catch(t){if(t.statusCode===404)return null;throw t.statusCode===400?new h("Design system not synced with Figma. Please sync tokens from Figma before using this design system.",400,e):t}}async fetchComponentsConfig(){if(this.localReader)return this.localReader.fetchComponentsConfig();let e=`${w}/${this.ffId}/components-config.json`;try{return await this.fetchJson(e)}catch(t){if(t.statusCode===404)return null;throw t}}async fetchFonts(){if(this.localReader)return this.localReader.fetchFonts();let e=`${w}/${this.ffId}/font.json`;try{return await this.fetchJson(e)}catch(t){if(t.statusCode===404)return null;throw t}}async fetchIcons(){if(this.localReader)return this.localReader.fetchIcons();let e=`${w}/${this.ffId}/icons.json`;try{return await this.fetchJson(e)}catch(t){if(t.statusCode===404)return null;throw t}}async fetchVersion(){if(this.localReader)return this.localReader.fetchVersion();let e=`${w}/${this.ffId}/version.json`;try{return await this.fetchJson(e)}catch(t){if(t.statusCode===404)return null;throw t}}async fetchCustomCss(){if(this.localReader)return this.localReader.fetchCustomCss();let e=`${w}/${this.ffId}/custom.css`;try{return await this.fetchRaw(e)}catch(t){if(t.statusCode===404)return null;throw t}}async fetchPlanMetadata(){if(this.localReader)return process.env.FF_DEBUG&&console.log(" \u2192 Using local reader for plan metadata"),this.localReader.fetchPlanMetadata();let e=`${this.baseURL}/api/design-systems/${this.ffId}/processed-tokens`;try{process.env.FF_DEBUG&&console.log(` \u2192 Fetching plan metadata from saas app: ${e}`);let t=await this.fetchJson(e);if(t&&t.metadata){let s={planType:t.metadata.planType||"full",allowedComponents:t.metadata.allowedComponents||null};return process.env.FF_DEBUG&&console.log(` \u2192 Found plan metadata in saas app: ${s.planType}, allowed: ${s.allowedComponents?s.allowedComponents.length+" components":"all"}`),s}}catch(t){if(t.statusCode===404){process.env.FF_DEBUG&&console.log(" \u2192 ff-id not found in saas app, trying legacy server");let s=`${w}/${this.ffId}/metadata`;try{process.env.FF_DEBUG&&console.log(` \u2192 Fetching from legacy server: ${s}`);let n=await this.fetchJson(s),o={planType:n.planType||"full",allowedComponents:n.allowedComponents||null};return process.env.FF_DEBUG&&console.log(` \u2192 Found in legacy server: ${o.planType}, allowed: ${o.allowedComponents?o.allowedComponents.length+" components":"all"}`),o}catch(n){if(n.statusCode===404)return process.env.FF_DEBUG&&console.log(" \u2192 ff-id not found in any server - access denied"),{planType:"denied",allowedComponents:[],error:"Design system not found. Please ensure your ff-id is valid."};throw n}}throw t}}};V.exports={APIClient:R}});var de=require("https"),m=require("fs"),f=require("path"),{URL:Fe}=require("url"),Y=require("os"),j=process.env.FF_REGISTRY_URL||"https://ff-registry.up.railway.app",S=class{constructor(e={}){this.appRoot=e.appRoot||process.cwd(),this.framework=e.framework||this.detectFramework(),this.config=e.config||{},this.outputPath=e.outputPath||this.getDefaultOutputPath(),this.overwrite=e.overwrite||!1,this.downloadedComponents=new Set,this.ffId=this.config.ffId||this.config["ff-id"]||null,this.planMetadata=null}detectFramework(){let e=f.join(this.appRoot,"package.json");if(m.existsSync(e)){let t=JSON.parse(m.readFileSync(e,"utf8")),s={...t.dependencies,...t.devDependencies};if(s.vue)return"vue";if(s.react)return"react"}return"react"}getDefaultOutputPath(){if(this.config.aliases&&this.config.aliases.ui){let t=this.config.aliases.ui.replace(/^@\//,"");return f.join(this.appRoot,t)}return this.framework==="vue"?f.join(this.appRoot,"src/components/ui"):f.join(this.appRoot,"components/ui")}fetchJson(e){let t=process.env.FF_USE_LOCAL==="true",s=process.env.FF_USE_LOCAL_REGISTRY==="true",n=e.includes(j);if(t||s&&n){process.env.FF_DEBUG&&s&&!t&&console.log(" \u2192 Using local registry for components");let o=["../../packages/registry","../../../packages/registry","../../../../packages/registry",f.join(__dirname,"../../../registry"),"packages/registry"],r;for(let l of o){let U=f.isAbsolute(l)?l:f.join(process.cwd(),l);if(m.existsSync(U)){r=U;break}}if(!r)return Promise.reject(new Error(`Local registry not found. Tried paths: ${o.join(", ")}`));let a=e.replace(j,"").replace(/^\//,""),g=f.join(r,...a.split("/"));try{let l=m.readFileSync(g,"utf8");return Promise.resolve(JSON.parse(l))}catch(l){return l.code==="ENOENT"?Promise.reject(new Error("HTTP 404: Not Found")):Promise.reject(l)}}return new Promise((o,r)=>{de.get(e,a=>{let g="";if(a.statusCode>=400){r(new Error(`HTTP ${a.statusCode}: ${a.statusMessage}`));return}a.on("data",l=>{g+=l}),a.on("end",()=>{try{o(JSON.parse(g))}catch(l){r(new Error(`Invalid JSON response: ${l.message}`))}})}).on("error",a=>{r(a)})})}async getAllComponents(){try{let e=`${j}/components/${this.framework}/index.json`,s=(await this.fetchJson(e)).components||[];if(this.ffId){try{await this.isComponentAllowed("dummy")}catch{if(this.planMetadata&&this.planMetadata.planType==="denied")throw new Error(this.planMetadata.error||"Access denied. Your ff-id was not found in any server.")}this.planMetadata&&this.planMetadata.planType==="trial"&&this.planMetadata.allowedComponents&&(s=s.filter(n=>this.planMetadata.allowedComponents.includes(n)),process.env.FF_DEBUG&&console.log(` \u2192 Trial plan: filtering to ${s.length} allowed components`))}return s}catch(e){return console.error("Failed to fetch component index:",e.message),[]}}async getCustomComponents(){if(!this.ffId)return process.env.FF_DEBUG&&console.log(" \u2192 No ff-id found, skipping custom components"),[];try{let e=`${j}/custom/${this.ffId}/${this.framework}/index.json`;process.env.FF_DEBUG&&console.log(` \u2192 Fetching custom components from: ${e}`);let t=await this.fetchJson(e);return process.env.FF_DEBUG&&console.log(` \u2192 Found custom components: ${JSON.stringify(t.components)}`),t.components||[]}catch(e){return process.env.FF_DEBUG&&console.log(` \u2192 No custom components found: ${e.message}`),[]}}async downloadComponents(e){process.env.FF_DEBUG&&console.log(` \u2192 ComponentDownloader initialized with ff-id: ${this.ffId}`);let t={successful:[],failed:[],dependencies:new Set};for(let s of e)try{let n=await this.downloadComponent(s);t.successful.push(s),n.forEach(o=>t.dependencies.add(o))}catch(n){t.failed.push({component:s,error:n.message})}return{successful:t.successful,failed:t.failed,dependencies:Array.from(t.dependencies)}}async isComponentAllowed(e){if(!this.planMetadata&&this.ffId){let{APIClient:t}=H(),s=new t(this.ffId,{baseURL:this.config["api-url"]||this.config.apiUrl});try{this.planMetadata=await s.fetchPlanMetadata(),process.env.FF_DEBUG&&console.log(` \u2192 Fetched plan metadata: ${this.planMetadata.planType}, allowed: ${this.planMetadata.allowedComponents?this.planMetadata.allowedComponents.length+" components":"all"}`)}catch(n){console.warn("Failed to fetch plan metadata:",n.message),this.planMetadata={planType:"full",allowedComponents:null}}}if(this.planMetadata&&this.planMetadata.planType==="denied")throw new Error(this.planMetadata.error||"Access denied. Your ff-id was not found in any server.");return!this.planMetadata||this.planMetadata.planType==="full"?!0:this.planMetadata.planType==="trial"&&this.planMetadata.allowedComponents?this.planMetadata.allowedComponents.includes(e):!1}async downloadComponent(e){if(this.downloadedComponents.has(e))return[];if(process.env.FF_DEBUG&&(console.log(` \u2192 Checking if component "${e}" is allowed...`),console.log(` \u2192 FF-ID: ${this.ffId}`)),!await this.isComponentAllowed(e)){let r=`Component "${e}" is not available in your trial plan. Please upgrade to access all components.`;throw console.error(`\u274C ${r}`),new Error(r)}console.log(`\u{1F4E5} Downloading ${e}...`);let s,n=!1;if(this.ffId){let r=`${j}/custom/${this.ffId}/${this.framework}/${e}.json`;try{s=await this.fetchJson(r),n=!0,process.env.FF_DEBUG&&console.log(` \u2192 Found custom component for ${e}`)}catch{process.env.FF_DEBUG&&console.log(` \u2192 No custom component found for ${e}, trying standard`)}}if(!s){let r=`${j}/components/${this.framework}/${e}.json`;try{s=await this.fetchJson(r)}catch(a){throw a.message.includes("404")?new Error(`Component "${e}" not found`):new Error(`Failed to fetch ${e}: ${a.message}`)}}if(!s.files||!Array.isArray(s.files))throw new Error(`Invalid component data for ${e}`);for(let r of s.files)await this.writeComponentFile(r);this.downloadedComponents.add(e),console.log(` \u2713 ${e} downloaded${n&&process.env.FF_DEBUG?" (custom)":""}`);let o=[...s.dependencies||[]];if(s.registryDependencies&&s.registryDependencies.length>0){console.log(` \u{1F4E6} Installing registry dependencies: ${s.registryDependencies.join(", ")}`);for(let r of s.registryDependencies)try{(await this.downloadComponent(r)).forEach(g=>o.push(g))}catch(a){console.warn(` \u26A0\uFE0F Failed to download dependency ${r}: ${a.message}`)}}return o}async writeComponentFile(e){let t=e.name.split("/").join(f.sep),s=f.join(this.outputPath,t),n=f.dirname(s);if(m.existsSync(n)||m.mkdirSync(n,{recursive:!0}),m.existsSync(s)&&!this.overwrite){console.log(` \u26A0\uFE0F Skipping ${e.name} (already exists)`);return}let o=e.content;Y.EOL!==`
|
|
2
|
+
`&&(o=e.content.replace(/\n/g,Y.EOL)),m.writeFileSync(s,o,"utf8")}};module.exports={ComponentDownloader:S};
|
|
3
3
|
//# sourceMappingURL=component-downloader.js.map
|