@nalvietnam/avatar-cli 3.5.0-beta.6 → 3.5.0-beta.8
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/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -52,7 +52,7 @@ ${n}
|
|
|
52
52
|
`):g&&process.stderr.write(`${mo(g,30)}
|
|
53
53
|
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=go(e,".gitnexus","wiki","index.html");return uc(l)?(a.succeed(`Wiki ready: ${l}`),{ran:!0,skipped:!1,wikiPath:l}):(a.fail("Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y index.html"),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y ${l}`})}f();function wo(t){t.command("add").description("Th\xEAm t\xE0i nguy\xEAn v\xE0o workspace").command("repo").description("Clone 1 repo code v\xE0o src/ (repo c\xF3 s\u1EB5n / th\u01B0 m\u1EE5c / d\u1EF1 \xE1n m\u1EDBi)").option("--url <url>","URL git repo c\xF3 s\u1EB5n (b\u1ECF qua prompt ngu\u1ED3n)").option("--name <name>","T\xEAn th\u01B0 m\u1EE5c trong src/ (m\u1EB7c \u0111\u1ECBnh suy t\u1EEB URL)").option("--yes","Auto-confirm prompt").action(async o=>{try{await kc(o)}catch(i){s.error(i instanceof Error?i.message:String(i)),process.exit(1)}}),t.command("remove").description("G\u1EE1 t\xE0i nguy\xEAn kh\u1ECFi workspace").command("repo <name>").description("G\u1EE1 repo kh\u1ECFi src/ (x\xF3a kh\u1ECFi repos.json + t\xF9y ch\u1ECDn x\xF3a folder)").option("--keep-files","Ch\u1EC9 g\u1EE1 kh\u1ECFi manifest, GI\u1EEE folder src/<name>").action(async(o,i)=>{try{await yc(o,i)}catch(a){s.error(a instanceof Error?a.message:String(a)),process.exit(1)}}),t.command("list").description("Li\u1EC7t k\xEA t\xE0i nguy\xEAn trong workspace").command("repo").description("Li\u1EC7t k\xEA repo trong src/").action(async()=>{try{await bc()}catch(o){s.error(o instanceof Error?o.message:String(o)),process.exit(1)}})}function ko(t){let e=t.trim();return e.length===0?"T\xEAn b\u1EAFt bu\u1ED9c":/^[A-Za-z0-9._-]+$/.test(e)?!0:"T\xEAn repo ch\u1EC9 g\u1ED3m ch\u1EEF/s\u1ED1/d\u1EA5u . - _ (kh\xF4ng space, kh\xF4ng '/')."}function Sn(){let t=Tt(process.cwd());return t||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
54
54
|
C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function kc(t){let e=Sn();await oo();let n=!0,r=t.url,o=t.name;for(;n;){let i=await vc(r),a=null,c=null;if(i.mode==="new"){let l=o??await ho(i.url),u=await un({workspaceRoot:e,url:i.url,name:l});a=u.path,c=u.name,s.success(`\u2713 T\u1EA1o src/${u.name} (repo m\u1EDBi)`)}else{let l=o??await ho(i.url);s.info(`Clone ${i.url} \u2192 src/${l} ...`);let{cloned:u,skipped:d}=await Ur({workspaceRoot:e,url:i.url,name:l});d||!u?s.dim(`B\u1ECF qua repo ${l}.`):(a=u.path,c=u.name,s.success(`\u2713 \u0110\xE3 clone src/${u.name}`))}if(a&&c&&(await Ac(e,a,t.yes),s.success(`\u2713 Done repo: ${c}`)),r=void 0,o=void 0,t.yes)break;n=await Kt({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function yc(t,e){let n=Sn(),{validateRepoName:r}=await Promise.resolve().then(()=>(Ot(),dn)),o=r(t);o&&(s.error(`T\xEAn repo kh\xF4ng h\u1EE3p l\u1EC7: ${o}`),process.exit(1));let{readReposManifest:i,removeRepoFromManifest:a}=await Promise.resolve().then(()=>(W(),st)),{removeRecursive:c}=await Promise.resolve().then(()=>(w(),yt)),{join:l,relative:u,sep:d}=await import("path"),h=(await i(n)).find(T=>T.name===t),b=l(n,"src",t),S=l(n,"src"),I=u(S,b);(I===""||I===".."||I.startsWith(`..${d}`)||I.startsWith("/"))&&(s.error(`\u0110\u01B0\u1EDDng d\u1EABn repo escape kh\u1ECFi workspace/src \u2014 t\u1EEB ch\u1ED1i x\xF3a: ${b}`),process.exit(1));let{pathExists:V}=await Promise.resolve().then(()=>(w(),yt)),P=await V(b);if(!h&&!P){s.warn(`Repo "${t}" kh\xF4ng c\xF3 trong workspace (c\u1EA3 manifest l\u1EABn src/).`);return}await a(n,t),s.success(`\u2713 G\u1EE1 "${t}" kh\u1ECFi repos.json`),P&&!e.keepFiles?(s.warn(`\u26A0 Folder src/${t} ch\u1EE9a code. X\xF3a = m\u1EA5t data n\u1EBFu ch\u01B0a push remote.`),await Kt({message:`X\xF3a lu\xF4n folder src/${t}?`,default:!1})?(await c(b),s.success(`\u2713 \u0110\xE3 x\xF3a src/${t}`)):s.dim(`Gi\u1EEF folder src/${t}. (\u0110\xE3 g\u1EE1 kh\u1ECFi manifest \u2014 ch\u1EA1y 'avatar add repo' \u0111\u1EC3 add l\u1EA1i.)`)):P&&e.keepFiles&&s.dim(`Gi\u1EEF folder src/${t} (--keep-files).`)}async function bc(){let t=Sn(),{readReposManifest:e}=await Promise.resolve().then(()=>(W(),st)),{pathExists:n}=await Promise.resolve().then(()=>(w(),yt)),{join:r}=await import("path"),o=await e(t);if(o.length===0){s.dim("Ch\u01B0a c\xF3 repo n\xE0o. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}s.info(`Repo trong workspace (${o.length}):`);for(let i of o){let c=await n(r(t,"src",i.name))?"\u2713":"\u26A0 folder thi\u1EBFu";console.log(` ${c} ${i.name.padEnd(28)} ${i.url}`)}}async function vc(t){if(t)return{url:t,mode:"clone"};let e=await Cn({message:"Ngu\u1ED3n repo?",choices:[{name:"1. Repo c\xF3 s\u1EB5n (\u0111i\u1EC1n URL git)",value:"url"},{name:"2. Th\u01B0 m\u1EE5c \u0111\xE3 c\xF3 tr\xEAn m\xE1y (clone t\u1EEB remote c\u1EE7a n\xF3)",value:"folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n (t\u1EA1o repo GitHub)",value:"new"}]});if(e==="url")return{url:await Ht({message:"URL git repo:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),mode:"clone"};if(e==="folder"){let{statSync:i}=await import("fs"),a=fo(await Ht({message:"\u0110\u01B0\u1EDDng d\u1EABn folder:",validate:u=>{let d=u.trim();if(d.length===0)return"Path b\u1EAFt bu\u1ED9c";try{return i(fo(d)).isDirectory()?!0:`"${d}" kh\xF4ng ph\u1EA3i th\u01B0 m\u1EE5c (l\xE0 file?).`}catch{return`Th\u01B0 m\u1EE5c "${d}" kh\xF4ng t\u1ED3n t\u1EA1i. Ki\u1EC3m tra l\u1EA1i \u0111\u01B0\u1EDDng d\u1EABn.`}}})),c=await pn(a);if(c&&(s.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${c.slice(0,120)}`),s.warn(" Clone l\u1EA5y b\u1EA3n REMOTE \u2192 c\xE1c thay \u0111\u1ED5i local n\xE0y s\u1EBD KH\xD4NG c\xF3 trong src/<name>."),!await Kt({message:"Ti\u1EBFp t\u1EE5c clone t\u1EEB remote (b\u1ECF thay \u0111\u1ED5i local ch\u01B0a push)?",default:!1})))throw new Error("H\u1EE7y. Commit + push thay \u0111\u1ED5i trong folder g\u1ED1c r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar add repo'.");let l=await mn(a);return l?{url:l,mode:"clone"}:{url:await xc(a),mode:"clone"}}let n=await Ht({message:"T\xEAn repo m\u1EDBi:",validate:ko}),r=await Cn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]});return{url:await Hr(n.trim(),r),mode:"new"}}async function xc(t){let{basename:e}=await import("path");if(s.warn(`Folder ${t} ch\u01B0a c\xF3 remote origin.`),!await Kt({message:"T\u1EA1o GitHub repo cho folder n\xE0y + push code l\xEAn?",default:!0}))throw new Error("H\u1EE7y. C\u1EA7n remote \u0111\u1EC3 add repo. T\u1EF1 t\u1EA1o remote (gh repo create) r\u1ED3i ch\u1EA1y l\u1EA1i, ho\u1EB7c ch\u1ECDn ngu\u1ED3n kh\xE1c.");let{git:r}=await Promise.resolve().then(()=>(Xe(),wr)),{pathExists:o}=await Promise.resolve().then(()=>(w(),yt)),i=r(t);await o(`${t}/.git`)||(s.info("Folder ch\u01B0a git init \u2014 \u0111ang init..."),await i.init()),s.info("Commit code hi\u1EC7n t\u1EA1i trong folder (baseline cho push)..."),await i.add("."),await i.commit("chore: initial commit (avatar add repo)").catch(()=>{});let a=await Ht({message:"T\xEAn repo GitHub:",default:e(t),validate:ko}),c=await Cn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createGithubRemoteFromFolder:l}=await Promise.resolve().then(()=>(vn(),Xr)),u=await Kr(a.trim(),d=>l({folder:t,name:d,visibility:c}).sshUrl);return s.success(`\u2713 \u0110\xE3 t\u1EA1o remote + push: ${u}`),u}async function ho(t){let e=cn(t);return await Ht({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function Ac(t,e,n){if(!pt().installed){s.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(n||await Kt({message:"Index repo n\xE0y b\u1EB1ng GitNexus?",default:!0})){try{Pt(e),s.success(` \u2713 GitNexus indexed src/${e.split("/").pop()}`)}catch(i){s.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`);return}try{let i=await me(t,e);i.ran?s.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):i.reason==="subscription-mode"&&s.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(i){s.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`)}}}w();import{promises as Fl}from"fs";import{join as ri}from"path";import{confirm as Vl}from"@inquirer/prompts";import{confirm as Qc}from"@inquirer/prompts";w();import{join as vo}from"path";import{select as _c}from"@inquirer/prompts";f();import{input as Cc,password as Sc,select as Tc}from"@inquirer/prompts";var Pc="https://ai.nal.vn",yo=1e4;function J(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Ec(){return await Sc({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Rc(t=Pc){return(await Cc({message:"LLMLite base URL:",default:t,validate:n=>/^https?:\/\//.test(n)?!0:"Ph\u1EA3i l\xE0 URL h\u1EE3p l\u1EC7 (http/https)"})).replace(/\/+$/,"")}async function Tn(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),yo);try{let o=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(o.status===401||o.status===403)throw new Error(`API key invalid (HTTP ${o.status}).`);if(o.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!o.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${o.status}).`);let a=((await o.json()).data||[]).map(c=>typeof c.id=="string"?c.id:null).filter(c=>c!==null);if(a.length===0)throw new Error("LLMLite tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 admin NAL.");return a}catch(o){if(o.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: ai.nal.vn l\xE0 internal endpoint NAL \u2014 ki\u1EC3m tra VPN NAL \u0111\xE3 b\u1EADt ch\u01B0a, ho\u1EB7c tcp test b\u1EB1ng `curl -v https://ai.nal.vn`.":`
|
|
55
|
-
Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${yo/1e3}s.${c}`)}let i=o.message||String(o);if(i.toLowerCase().includes("fetch failed")||i.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${i}${c}`)}throw o}finally{clearTimeout(r)}}async function $c(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return s.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Tc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function bo(){let t=await Ec(),e=await Rc();s.info(`Verify key (${J(t)}) qua ${e}/v1/models...`);let n=await Tn(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await $c(n);return{apiKey:t,baseUrl:e,model:r,availableModels:n}}f();var Ic=".claude/state/ai-orchestrator-model-map.json",Nc=["opus","sonnet","haiku"];function xo(t){return vo(t,Ic)}async function Mc(t){let e=xo(t);if(!await m(e))return null;try{let r=(await y(e)).map;return r&&typeof r.hard=="string"&&typeof r.standard=="string"&&typeof r.fast=="string"?{hard:r.hard,standard:r.standard,fast:r.fast}:null}catch{return null}}async function Oc(t){let e=vo(t,".claude","settings.json");if(!await m(e))return{provider:"subscription",baseUrl:null,token:null};try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null,i=typeof n.avatarProvider=="string"?n.avatarProvider:null;if(!o)return{provider:"subscription",baseUrl:null,token:null};let a=d=>{let g=process.env[d];if(typeof g=="string"&&g.length>0)return g;let h=r[d];return typeof h=="string"&&h.length>0?h:null},c=!1;try{c=new URL(o).hostname==="api.anthropic.com"}catch{c=!1}let l=i==="anthropic"?"anthropic":i==="llmlite"?"llmlite":c?"anthropic":"llmlite",u=a(l==="anthropic"?"ANTHROPIC_API_KEY":"ANTHROPIC_AUTH_TOKEN");return{provider:l,baseUrl:o,token:u}}catch{return{provider:"subscription",baseUrl:null,token:null}}}function En(t){let e=t.filter(n=>n.toLowerCase().includes("claude"));return e.length>0?e:t}async function Lc(t){if(t.provider==="subscription"||!t.baseUrl||!t.token)return{models:[...Nc],source:"alias"};let e=await Tn(t.baseUrl,t.token);return{models:En(e),source:"/v1/models"}}async function Pn(t,e){return await _c({message:`Tier ${t} \u2192 ch\u1ECDn model:`,choices:e.map(n=>({name:n,value:n}))})}async function Gc(t,e,n,r){let o={map:e,provider:n,source:r,updatedAt:Math.floor(Date.now()/1e3)};await A(xo(t),o)}async function Ao(t,e={}){if(!e.forceRefresh&&await Mc(t))return s.dim("ai-orchestrator: state map \u0111\xE3 t\u1ED3n t\u1EA1i \u2014 gi\u1EEF map c\u0169 (kh\xF4ng h\u1ECFi l\u1EA1i)."),!0;if(!process.stdout.isTTY)return s.warn("ai-orchestrator c\u1EA7n interactive setup (ch\u1ECDn model cho 3 tier) \u2014 b\u1ECF qua trong m\xF4i tr\u01B0\u1EDDng non-TTY. Ch\u1EA1y 'avatar tools enable ai-orchestrator' trong terminal th\u1EADt \u0111\u1EC3 c\xE0i."),!1;let n=await Oc(t),r,o;try{let a=await Lc(n);r=a.models,o=a.source}catch(a){return s.error(`ai-orchestrator: kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c danh s\xE1ch model (${a.message}). Kh\xF4ng b\u1EADt tool. Ki\u1EC3m tra VPN/gateway r\u1ED3i th\u1EED l\u1EA1i.`),!1}if(r.length===1)return s.warn(`ai-orchestrator: key ch\u1EC9 c\xF3 1 model (${r[0]}) \u2014 route v\xF4 ngh\u0129a khi ch\u1EC9 1 model. KH\xD4NG b\u1EADt tool.`),!1;s.info(`ai-orchestrator: ${r.length} model kh\u1EA3 d\u1EE5ng (${n.provider}). Ch\u1ECDn model cho m\u1ED7i lo\u1EA1i vi\u1EC7c (Kh\xF3/V\u1EEBa/D\u1EC5). C\xF3 th\u1EC3 ch\u1ECDn tr\xF9ng.`);let i;try{let a=await Pn("Vi\u1EC7c kh\xF3 (model m\u1EA1nh)",r),c=await Pn("Vi\u1EC7c v\u1EEBa (model c\xE2n b\u1EB1ng)",r),l=await Pn("Vi\u1EC7c d\u1EC5 (model r\u1EBB)",r);i={hard:a,standard:c,fast:l}}catch{return s.dim("ai-orchestrator: h\u1EE7y ch\u1ECDn model \u2014 kh\xF4ng b\u1EADt tool."),!1}try{await Gc(t,i,n.provider,o)}catch(a){return s.error(`ai-orchestrator: ghi state map th\u1EA5t b\u1EA1i (${a.message}). Kh\xF4ng b\u1EADt tool.`),!1}return s.success(`ai-orchestrator: \u0111\xE3 map tier \u2192 model. Kh\xF3=${i.hard} \xB7 V\u1EEBa=${i.standard} \xB7 D\u1EC5=${i.fast}`),!0}f();import{spawnSync as Yc}from"child_process";w();import{promises as Fc}from"fs";import{join as Eo}from"path";w();f();import{promises as jc}from"fs";import Co,{join as Rn}from"path";async function Uc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return s.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let i=Rn(t,o),a=Co.resolve(t),c=Co.resolve(i);return!c.startsWith(`${a}/`)&&c!==a?(s.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await m(i)}function $n(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)+String(e.getMonth()+1).padStart(2,"0")+String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}${String(e.getSeconds()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Ft(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let i=typeof o=="string"?o:JSON.stringify(o);n.has(i)||(n.add(i),r.push(o))}return r}function Dc(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Hc(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let o of n)if(typeof o=="object"&&o!==null){let i=o.command;typeof i=="string"&&r.push(i)}return r}function So(t){let e=t.map((i,a)=>{let c=Hc(i),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:i,commands:c,hasVar:l}}),n=new Map;for(let i of e){let a=i.commands.map(Dc).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(i),n.set(a,c)}let r=new Set;for(let[,i]of n)if(!(i.length<2||!i.some(c=>c.hasVar)))for(let c of i)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((i,a)=>!r.has(a)),droppedCount:r.size}}function Kc(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=So(a);l>0&&(r[i]=c,o+=l,n.includes(i)||n.push(i))}for(let[i,a]of Object.entries(t)){let c=r[i]||[],l=Ft(c,a),{entries:u,droppedCount:d}=So(l);d>0&&(o+=d),u.length!==c.length&&(n.includes(i)||n.push(i)),r[i]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function de(t){let e=Rn(t,".claude","pack","templates","settings.json.tpl"),n=Rn(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await O(e);r=JSON.parse(u)}catch(u){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${u.message}. Path: ${e}`)}let o={},i=!1;if(await m(n)){i=!0;try{o=await y(n)}catch(u){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${u.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...o};if(typeof o.disableWorkflows!="boolean"&&(c.disableWorkflows=!0,a.push("disableWorkflows=true (Avatar t\u1EAFt Dynamic Workflows)")),r.statusLine&&!o.statusLine&&(await Uc(t,r.statusLine.command)?(c.statusLine=r.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${r.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof r.includeCoAuthoredBy=="boolean"&&typeof o.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!o.model&&(c.model=r.model,a.push("model added")),r.env||o.env){let u={...o.env||{}},d=!1,g=!1;for(let[h,b]of Object.entries(u))typeof b=="string"&&b.includes("llm.nal.vn")&&(u[h]=b.replace(/llm\.nal\.vn/g,"ai.nal.vn"),d=!0);if(d&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[h,b]of Object.entries(r.env))h in u||(u[h]=b,g=!0);g&&a.push("env vars added from pack"),(d||g)&&(c.env=u)}if(r.permissions){let u=o.permissions?.allow||[],d=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],b=Ft(u,g),S=Ft(d,h);(b.length!==u.length||S.length!==d.length)&&(c.permissions={allow:b,deny:S},a.push(`permissions union (+${b.length-u.length} allow, +${S.length-d.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=Kc(r.hooks,u);(g.length>0||h>0)&&(c.hooks=d,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),h>0&&a.push(`migrated ${h} stale relative-path hook entries to use \${CLAUDE_PROJECT_DIR} version (fixes hook failure when shell cwd changes)`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return i&&(l=$n(n),await jc.copyFile(n,l)),await A(n,c),{action:"merged",backupPath:l,changes:a}}function Vc(t,e){return Eo(t,".claude","pack","tools",e,"tool.json")}async function Vt(t,e){let n=Vc(t,e);if(!await m(n))return null;try{return await y(n)}catch{return null}}var Bc=[".claude","settings.json"];function Ro(t){return Eo(t,...Bc)}function ge(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function $o(t){let e=Ro(t);if(!await m(e))return{settings:{},existed:!1};try{return{settings:await y(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable tool.`)}}async function _o(t,e,n){let r=Ro(t),o;return n&&(o=$n(r),await Fc.copyFile(r,o)),await A(r,e),o}function To(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Po(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Wc(t,e){let n=new Set;for(let i of e)Po(i)&&n.add(To(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>Po(i)?!0:n.has(To(i))?(r++,!1):!0),droppedCount:r}}async function fe(t,e){let{settings:n,existed:r}=await $o(t),o={...n},i=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},d={...u},g=[],h=0;for(let[b,S]of Object.entries(a)){let I=u[b]??[],{migratedUser:V,droppedCount:P}=Wc(I,S);P>0&&(h+=P);let T=new Set(V.map(ge)),R=S.filter(rt=>!T.has(ge(rt)));(R.length>0||P>0)&&(d[b]=[...V,...R],g.push(b))}g.length>0&&(o.hooks=d,i.push(`hooks added: ${g.join(", ")}`),h>0&&i.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],d=Ft(u,c);d.length!==u.length&&(o.permissions={...n.permissions,deny:d},i.push(`deny +${d.length-u.length}`))}return i.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await _o(t,o,r),changes:i}}async function Io(t,e){let{settings:n,existed:r}=await $o(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:i,...a}=n,c={...a},l=[],u=e.settings.hooks??{},d=n.hooks??{},g={},h=[];for(let[P,T]of Object.entries(d)){let R=u[P];if(!R){g[P]=T;continue}let rt=new Set(R.map(ge)),kt=T.filter(ot=>!rt.has(ge(ot)));kt.length!==T.length&&h.push(P),kt.length>0&&(g[P]=kt)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let b=new Set(e.settings.permissions?.deny??[]),S=n.permissions?.deny??[],I=b.size>0?S.filter(P=>!b.has(P)):S;if(n.permissions){let{deny:P,...T}=n.permissions,R={...T,...I.length>0?{deny:I}:{}};Object.keys(R).length>0&&(c.permissions=R)}return I.length!==S.length&&l.push(`deny -${S.length-I.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await _o(t,c,r),changes:l}}f();w();import{join as qc}from"path";var zc=".claude/avatar-tools.json";function No(){return{tools:{}}}function Mo(t){return qc(t,zc)}async function Et(t){let e=Mo(t);if(!await m(e))return No();try{return{tools:(await y(e)).tools??{}}}catch{return No()}}async function Jc(t,e){await A(Mo(t),e)}async function he(t,e,n){let r=await Et(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Jc(t,r),r}async function Rt(t){let e=await Et(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Xc(t){if(!t)return;let e=Yc(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&s.warn(`Tool y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function Oo(t,e){switch(e.action){case"enabled":s.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"disabled":s.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"no-change":s.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function D(t,e,n={}){let r=await Vt(t,e);if(!r)return n.silent||s.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;if(Xc(r.requires?.runtime),e==="ai-orchestrator"&&!await Ao(t,{forceRefresh:n.forceRefresh}))return!1;let o=await fe(t,r);return Oo(e,o),await he(t,e,{enabled:!0,version:r.version}),!0}async function _n(t,e){let n=await Vt(t,e);if(!n){s.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let i=(await Et(t)).tools[e]?.version??"unknown";return await he(t,e,{enabled:!1,version:i}),!1}let r=await Io(t,n);return Oo(e,r),await he(t,e,{enabled:!1,version:n.version}),!0}var Zc="ai-orchestrator";async function we(t,e,n={}){if(!e||!e.ok||e.provider!=="llmlite"&&e.provider!=="anthropic")return;let r=e.availableModels??[],o=En(r);if(o.length<=1){s.dim(`AI Orchestrator: key ch\u1EC9 c\xF3 ${o.length} model ph\xE2n lo\u1EA1i \u0111\u01B0\u1EE3c \u2014 b\u1ECF qua (c\u1EA7n >1 model).`);return}if(n.autoYes){s.dim("AI Orchestrator: b\u1ECF qua trong ch\u1EBF \u0111\u1ED9 non-interactive \u2014 b\u1EADt sau b\u1EB1ng 'avatar tools enable ai-orchestrator'.");return}let i=n.forceRefresh?"AI Orchestrator: ch\u1ECDn l\u1EA1i model cho m\u1ED7i lo\u1EA1i vi\u1EC7c? (key c\xF3 th\u1EC3 \u0111\xE3 \u0111\u1ED5i)":"B\u1EADt AI Orchestrator? (t\u1EF1 ch\u1ECDn model theo \u0111\u1ED9 kh\xF3 vi\u1EC7c: Vi\u1EC7c kh\xF3 \u2192 model m\u1EA1nh, Vi\u1EC7c d\u1EC5 \u2192 model r\u1EBB)";if(!await Qc({message:i,default:!0})){s.dim(" B\u1ECF qua AI Orchestrator (user opt-out).");return}await D(t,Zc,{forceRefresh:n.forceRefresh})}import{promises as Ho}from"fs";import{homedir as al}from"os";import{join as zt}from"path";import{z as k}from"zod";var Lo=k.object({email:k.string().email(),name:k.string(),access_token:k.string().min(1),refresh_token:k.string().min(1),expires_at:k.string().datetime(),id_token:k.string().min(1)}),tl=k.object({installed_tools:k.record(k.string(),k.object({version:k.string().optional(),installed_at:k.string().datetime(),install_method:k.string()})).default({}),tool_inputs:k.record(k.string(),k.unknown()).default({})}),cf=k.object({$schema:k.string().optional(),includeCoAuthoredBy:k.boolean().optional(),env:k.record(k.string(),k.string()).default({}),permissions:k.object({allow:k.array(k.string()).default([]),deny:k.array(k.string()).default([])}).partial().optional(),hooks:k.record(k.string(),k.array(k.unknown())).optional(),statusLine:k.object({type:k.string(),command:k.string(),padding:k.number().optional()}).optional()}),lf=k.enum(["internal","client","library"]);w();var Jt=zt(al(),".avatar"),mt=zt(Jt,"config.json"),ff=zt(Jt,"state.json"),Un=zt(Jt,"audit.log"),hf=zt(Jt,"backups"),cl=384;async function Dn(){await x(Jt)}async function j(){if(!await m(mt))return null;let t=await y(mt),e=Lo.safeParse(t);return e.success?e.data:null}async function Hn(t){await Dn(),await A(mt,t,cl)}async function be(){if(await m(mt)){let{promises:t}=await import("fs");await t.unlink(mt)}}function dt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var qt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function ll(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Wt(),jn));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function ve(){let t=await j();if(!t)throw new qt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Wt(),jn));if(!await ll(t.id_token))return r(n(t.id_token)),t.id_token;let o;try{o=await e(t.refresh_token)}catch(a){throw new qt(`Token h\u1EBFt h\u1EA1n v\xE0 refresh th\u1EA5t b\u1EA1i (${a instanceof Error?a.message:a}). Ch\u1EA1y 'avatar login' l\u1EA1i.`)}if(!o.id_token)throw new qt("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let i={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await Hn(i),o.id_token}async function ul(){try{await Ho.chmod(Un,384)}catch{}}async function v(t,e){await Dn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
|
|
55
|
+
Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${yo/1e3}s.${c}`)}let i=o.message||String(o);if(i.toLowerCase().includes("fetch failed")||i.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${i}${c}`)}throw o}finally{clearTimeout(r)}}async function $c(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return s.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Tc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function bo(){let t=await Ec(),e=await Rc();s.info(`Verify key (${J(t)}) qua ${e}/v1/models...`);let n=await Tn(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await $c(n);return{apiKey:t,baseUrl:e,model:r,availableModels:n}}f();var Ic=".claude/state/ai-orchestrator-model-map.json",Nc=["opus","sonnet","haiku"];function xo(t){return vo(t,Ic)}async function Mc(t){let e=xo(t);if(!await m(e))return null;try{let r=(await y(e)).map;return r&&typeof r.hard=="string"&&typeof r.standard=="string"&&typeof r.fast=="string"?{hard:r.hard,standard:r.standard,fast:r.fast}:null}catch{return null}}async function Oc(t){let e=vo(t,".claude","settings.json");if(!await m(e))return{provider:"subscription",baseUrl:null,token:null};try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null,i=typeof n.avatarProvider=="string"?n.avatarProvider:null;if(!o)return{provider:"subscription",baseUrl:null,token:null};let a=d=>{let g=process.env[d];if(typeof g=="string"&&g.length>0)return g;let h=r[d];return typeof h=="string"&&h.length>0?h:null},c=!1;try{c=new URL(o).hostname==="api.anthropic.com"}catch{c=!1}let l=i==="anthropic"?"anthropic":i==="llmlite"?"llmlite":c?"anthropic":"llmlite",u=a(l==="anthropic"?"ANTHROPIC_API_KEY":"ANTHROPIC_AUTH_TOKEN");return{provider:l,baseUrl:o,token:u}}catch{return{provider:"subscription",baseUrl:null,token:null}}}function En(t){let e=t.filter(n=>n.toLowerCase().includes("claude"));return e.length>0?e:t}async function Lc(t){if(t.provider==="subscription"||!t.baseUrl||!t.token)return{models:[...Nc],source:"alias"};let e=await Tn(t.baseUrl,t.token);return{models:En(e),source:"/v1/models"}}async function Pn(t,e){return await _c({message:`Tier ${t} \u2192 ch\u1ECDn model:`,choices:e.map(n=>({name:n,value:n}))})}async function Gc(t,e,n,r){let o={map:e,provider:n,source:r,updatedAt:Math.floor(Date.now()/1e3)};await A(xo(t),o)}async function Ao(t,e={}){if(!e.forceRefresh&&await Mc(t))return s.dim("ai-orchestrator: state map \u0111\xE3 t\u1ED3n t\u1EA1i \u2014 gi\u1EEF map c\u0169 (kh\xF4ng h\u1ECFi l\u1EA1i)."),!0;if(!process.stdout.isTTY)return s.warn("ai-orchestrator c\u1EA7n interactive setup (ch\u1ECDn model cho 3 tier) \u2014 b\u1ECF qua trong m\xF4i tr\u01B0\u1EDDng non-TTY. Ch\u1EA1y 'avatar tools enable ai-orchestrator' trong terminal th\u1EADt \u0111\u1EC3 c\xE0i."),!1;let n=await Oc(t),r,o;try{let a=await Lc(n);r=a.models,o=a.source}catch(a){return s.error(`ai-orchestrator: kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c danh s\xE1ch model (${a.message}). Kh\xF4ng b\u1EADt tool. Ki\u1EC3m tra VPN/gateway r\u1ED3i th\u1EED l\u1EA1i.`),!1}if(r.length===1)return s.warn(`ai-orchestrator: key ch\u1EC9 c\xF3 1 model (${r[0]}) \u2014 route v\xF4 ngh\u0129a khi ch\u1EC9 1 model. KH\xD4NG b\u1EADt tool.`),!1;s.info(`ai-orchestrator: ${r.length} model kh\u1EA3 d\u1EE5ng (${n.provider}). Ch\u1ECDn model cho m\u1ED7i lo\u1EA1i vi\u1EC7c (Kh\xF3/V\u1EEBa/D\u1EC5). C\xF3 th\u1EC3 ch\u1ECDn tr\xF9ng.`);let i;try{let a=await Pn("Vi\u1EC7c kh\xF3 (model m\u1EA1nh)",r),c=await Pn("Vi\u1EC7c v\u1EEBa (model c\xE2n b\u1EB1ng)",r),l=await Pn("Vi\u1EC7c d\u1EC5 (model r\u1EBB)",r);i={hard:a,standard:c,fast:l}}catch{return s.dim("ai-orchestrator: h\u1EE7y ch\u1ECDn model \u2014 kh\xF4ng b\u1EADt tool."),!1}try{await Gc(t,i,n.provider,o)}catch(a){return s.error(`ai-orchestrator: ghi state map th\u1EA5t b\u1EA1i (${a.message}). Kh\xF4ng b\u1EADt tool.`),!1}return s.success(`ai-orchestrator: \u0111\xE3 map tier \u2192 model. Kh\xF3=${i.hard} \xB7 V\u1EEBa=${i.standard} \xB7 D\u1EC5=${i.fast}`),!0}f();import{spawnSync as Yc}from"child_process";w();import{promises as Fc}from"fs";import{join as Eo}from"path";w();f();import{promises as jc}from"fs";import Co,{join as Rn}from"path";async function Uc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return s.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let i=Rn(t,o),a=Co.resolve(t),c=Co.resolve(i);return!c.startsWith(`${a}/`)&&c!==a?(s.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await m(i)}function $n(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)+String(e.getMonth()+1).padStart(2,"0")+String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}${String(e.getSeconds()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Ft(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let i=typeof o=="string"?o:JSON.stringify(o);n.has(i)||(n.add(i),r.push(o))}return r}function Dc(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Hc(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let o of n)if(typeof o=="object"&&o!==null){let i=o.command;typeof i=="string"&&r.push(i)}return r}function So(t){let e=t.map((i,a)=>{let c=Hc(i),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:i,commands:c,hasVar:l}}),n=new Map;for(let i of e){let a=i.commands.map(Dc).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(i),n.set(a,c)}let r=new Set;for(let[,i]of n)if(!(i.length<2||!i.some(c=>c.hasVar)))for(let c of i)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((i,a)=>!r.has(a)),droppedCount:r.size}}function Kc(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=So(a);l>0&&(r[i]=c,o+=l,n.includes(i)||n.push(i))}for(let[i,a]of Object.entries(t)){let c=r[i]||[],l=Ft(c,a),{entries:u,droppedCount:d}=So(l);d>0&&(o+=d),u.length!==c.length&&(n.includes(i)||n.push(i)),r[i]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function de(t){let e=Rn(t,".claude","pack","templates","settings.json.tpl"),n=Rn(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await O(e);r=JSON.parse(u)}catch(u){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${u.message}. Path: ${e}`)}let o={},i=!1;if(await m(n)){i=!0;try{o=await y(n)}catch(u){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${u.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...o};if(typeof o.disableWorkflows!="boolean"&&(c.disableWorkflows=!0,a.push("disableWorkflows=true (Avatar t\u1EAFt Dynamic Workflows)")),r.statusLine&&!o.statusLine&&(await Uc(t,r.statusLine.command)?(c.statusLine=r.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${r.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof r.includeCoAuthoredBy=="boolean"&&typeof o.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!o.model&&(c.model=r.model,a.push("model added")),r.env||o.env){let u={...o.env||{}},d=!1,g=!1;for(let[h,b]of Object.entries(u))typeof b=="string"&&b.includes("llm.nal.vn")&&(u[h]=b.replace(/llm\.nal\.vn/g,"ai.nal.vn"),d=!0);if(d&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[h,b]of Object.entries(r.env))h in u||(u[h]=b,g=!0);g&&a.push("env vars added from pack"),(d||g)&&(c.env=u)}if(r.permissions){let u=o.permissions?.allow||[],d=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],b=Ft(u,g),S=Ft(d,h);(b.length!==u.length||S.length!==d.length)&&(c.permissions={allow:b,deny:S},a.push(`permissions union (+${b.length-u.length} allow, +${S.length-d.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=Kc(r.hooks,u);(g.length>0||h>0)&&(c.hooks=d,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),h>0&&a.push(`migrated ${h} stale relative-path hook entries to use \${CLAUDE_PROJECT_DIR} version (fixes hook failure when shell cwd changes)`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return i&&(l=$n(n),await jc.copyFile(n,l)),await A(n,c),{action:"merged",backupPath:l,changes:a}}function Vc(t,e){return Eo(t,".claude","pack","tools",e,"tool.json")}async function Vt(t,e){let n=Vc(t,e);if(!await m(n))return null;try{return await y(n)}catch{return null}}var Bc=[".claude","settings.json"];function Ro(t){return Eo(t,...Bc)}function ge(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function $o(t){let e=Ro(t);if(!await m(e))return{settings:{},existed:!1};try{return{settings:await y(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable tool.`)}}async function _o(t,e,n){let r=Ro(t),o;return n&&(o=$n(r),await Fc.copyFile(r,o)),await A(r,e),o}function To(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Po(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Wc(t,e){let n=new Set;for(let i of e)Po(i)&&n.add(To(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>Po(i)?!0:n.has(To(i))?(r++,!1):!0),droppedCount:r}}async function fe(t,e){let{settings:n,existed:r}=await $o(t),o={...n},i=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},d={...u},g=[],h=0;for(let[b,S]of Object.entries(a)){let I=u[b]??[],{migratedUser:V,droppedCount:P}=Wc(I,S);P>0&&(h+=P);let T=new Set(V.map(ge)),R=S.filter(rt=>!T.has(ge(rt)));(R.length>0||P>0)&&(d[b]=[...V,...R],g.push(b))}g.length>0&&(o.hooks=d,i.push(`hooks added: ${g.join(", ")}`),h>0&&i.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],d=Ft(u,c);d.length!==u.length&&(o.permissions={...n.permissions,deny:d},i.push(`deny +${d.length-u.length}`))}return i.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await _o(t,o,r),changes:i}}async function Io(t,e){let{settings:n,existed:r}=await $o(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:i,...a}=n,c={...a},l=[],u=e.settings.hooks??{},d=n.hooks??{},g={},h=[];for(let[P,T]of Object.entries(d)){let R=u[P];if(!R){g[P]=T;continue}let rt=new Set(R.map(ge)),kt=T.filter(ot=>!rt.has(ge(ot)));kt.length!==T.length&&h.push(P),kt.length>0&&(g[P]=kt)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let b=new Set(e.settings.permissions?.deny??[]),S=n.permissions?.deny??[],I=b.size>0?S.filter(P=>!b.has(P)):S;if(n.permissions){let{deny:P,...T}=n.permissions,R={...T,...I.length>0?{deny:I}:{}};Object.keys(R).length>0&&(c.permissions=R)}return I.length!==S.length&&l.push(`deny -${S.length-I.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await _o(t,c,r),changes:l}}f();w();import{join as qc}from"path";var zc=".claude/avatar-tools.json";function No(){return{tools:{}}}function Mo(t){return qc(t,zc)}async function Et(t){let e=Mo(t);if(!await m(e))return No();try{return{tools:(await y(e)).tools??{}}}catch{return No()}}async function Jc(t,e){await A(Mo(t),e)}async function he(t,e,n){let r=await Et(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Jc(t,r),r}async function Rt(t){let e=await Et(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Xc(t){if(!t)return;let e=Yc(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&s.warn(`Tool y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function Oo(t,e){switch(e.action){case"enabled":s.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"disabled":s.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"no-change":s.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function D(t,e,n={}){let r=await Vt(t,e);if(!r)return n.silent||s.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;if(Xc(r.requires?.runtime),e==="ai-orchestrator"&&!await Ao(t,{forceRefresh:n.forceRefresh}))return!1;let o=await fe(t,r);return Oo(e,o),await he(t,e,{enabled:!0,version:r.version}),!0}async function _n(t,e){let n=await Vt(t,e);if(!n){s.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let i=(await Et(t)).tools[e]?.version??"unknown";return await he(t,e,{enabled:!1,version:i}),!1}let r=await Io(t,n);return Oo(e,r),await he(t,e,{enabled:!1,version:n.version}),!0}var Zc="ai-orchestrator";async function we(t,e,n={}){if(!e||!e.ok)return;let r;if(e.provider==="subscription")r=3;else if(e.provider==="llmlite"||e.provider==="anthropic")r=En(e.availableModels??[]).length;else return;if(r<=1){s.dim(`AI Orchestrator: key ch\u1EC9 c\xF3 ${r} model ph\xE2n lo\u1EA1i \u0111\u01B0\u1EE3c \u2014 b\u1ECF qua (c\u1EA7n >1 model).`);return}if(n.autoYes){s.dim("AI Orchestrator: b\u1ECF qua trong ch\u1EBF \u0111\u1ED9 non-interactive \u2014 b\u1EADt sau b\u1EB1ng 'avatar tools enable ai-orchestrator'.");return}let o=n.forceRefresh?"AI Orchestrator: ch\u1ECDn l\u1EA1i model cho m\u1ED7i lo\u1EA1i vi\u1EC7c? (key c\xF3 th\u1EC3 \u0111\xE3 \u0111\u1ED5i)":"B\u1EADt AI Orchestrator? (t\u1EF1 ch\u1ECDn model theo \u0111\u1ED9 kh\xF3 vi\u1EC7c: Vi\u1EC7c kh\xF3 \u2192 model m\u1EA1nh, Vi\u1EC7c d\u1EC5 \u2192 model r\u1EBB)";if(!await Qc({message:o,default:!0})){s.dim(" B\u1ECF qua AI Orchestrator (user opt-out).");return}await D(t,Zc,{forceRefresh:n.forceRefresh})}import{promises as Ho}from"fs";import{homedir as al}from"os";import{join as zt}from"path";import{z as k}from"zod";var Lo=k.object({email:k.string().email(),name:k.string(),access_token:k.string().min(1),refresh_token:k.string().min(1),expires_at:k.string().datetime(),id_token:k.string().min(1)}),tl=k.object({installed_tools:k.record(k.string(),k.object({version:k.string().optional(),installed_at:k.string().datetime(),install_method:k.string()})).default({}),tool_inputs:k.record(k.string(),k.unknown()).default({})}),cf=k.object({$schema:k.string().optional(),includeCoAuthoredBy:k.boolean().optional(),env:k.record(k.string(),k.string()).default({}),permissions:k.object({allow:k.array(k.string()).default([]),deny:k.array(k.string()).default([])}).partial().optional(),hooks:k.record(k.string(),k.array(k.unknown())).optional(),statusLine:k.object({type:k.string(),command:k.string(),padding:k.number().optional()}).optional()}),lf=k.enum(["internal","client","library"]);w();var Jt=zt(al(),".avatar"),mt=zt(Jt,"config.json"),ff=zt(Jt,"state.json"),Un=zt(Jt,"audit.log"),hf=zt(Jt,"backups"),cl=384;async function Dn(){await x(Jt)}async function j(){if(!await m(mt))return null;let t=await y(mt),e=Lo.safeParse(t);return e.success?e.data:null}async function Hn(t){await Dn(),await A(mt,t,cl)}async function be(){if(await m(mt)){let{promises:t}=await import("fs");await t.unlink(mt)}}function dt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var qt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function ll(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Wt(),jn));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function ve(){let t=await j();if(!t)throw new qt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Wt(),jn));if(!await ll(t.id_token))return r(n(t.id_token)),t.id_token;let o;try{o=await e(t.refresh_token)}catch(a){throw new qt(`Token h\u1EBFt h\u1EA1n v\xE0 refresh th\u1EA5t b\u1EA1i (${a instanceof Error?a.message:a}). Ch\u1EA1y 'avatar login' l\u1EA1i.`)}if(!o.id_token)throw new qt("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let i={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await Hn(i),o.id_token}async function ul(){try{await Ho.chmod(Un,384)}catch{}}async function v(t,e){await Dn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
|
|
56
56
|
`;await Ho.appendFile(Un,r,{encoding:"utf8",mode:384}),await ul()}f();import{spawnSync as Kn}from"child_process";var Ko=6e4,pl="ok";function Fn(){let t=Kn("claude",["auth","status"],{encoding:"utf8"});if(t.error||t.status!==0)return{state:"not-authenticated"};let e=(t.stdout||"").trim();if(!e.startsWith("{"))return{state:"authenticated"};try{let n=JSON.parse(e);return n.loggedIn!==!0?{state:"not-authenticated"}:{state:"authenticated",email:n.email,subscriptionType:n.subscriptionType,apiProvider:n.apiProvider}}catch{return{state:"authenticated"}}}function Vn(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=Kn("claude",["auth","login"],{stdio:"inherit"});if(t.status!==0)throw new Error(`claude auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'claude auth login' tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function ml(t){let e=t.toLowerCase();return e.includes("credit_balance_too_low")||e.includes("credit balance too low")?"credit_balance_too_low":e.includes("insufficient_quota")||e.includes("insufficient quota")||e.includes("quota_exceeded")||e.includes("quota exceeded")||e.includes("usage limit")||e.includes("you've used all")?"insufficient_quota":e.includes("401")||e.includes("invalid authentication")||e.includes("authentication credentials")||e.includes("failed to authenticate")||e.includes("authentication failed")||e.includes("unauthorized")?"auth-expired":e.includes("invalid_api_key")||e.includes("invalid api key")?"invalid_api_key":e.includes("rate_limit")||e.includes("rate limit")||e.includes("429")?"rate_limit":"unknown"}function Fo(t){switch(t){case"auth-expired":return"Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n/b\u1ECB revoke. Ch\u1EA1y: `claude auth logout && claude auth login`.";case"credit_balance_too_low":case"insufficient_quota":return"H\u1EBFt quota subscription. Upgrade plan ho\u1EB7c d\xF9ng LLMLite (avatar ai setup \u2192 ch\u1ECDn LLMLite).";case"invalid_api_key":return"API key invalid. Re-login: `claude auth login`.";case"rate_limit":return"B\u1ECB rate limit t\u1EA1m th\u1EDDi. Ch\u1EDD v\xE0i ph\xFAt r\u1ED3i ch\u1EA1y `avatar ai setup`.";case"timeout":return"Timeout 60s: (1) m\u1EA1ng VN ch\u1EADm \u2014 th\u1EED VPN, (2) Anthropic API spike \u2014 retry v\xE0i ph\xFAt, (3) token v\u1EABn auth nh\u01B0ng quota h\u1EBFt \xE2m th\u1EA7m \u2014 check t\u1EA1i claude.ai/settings/usage.";default:return"L\u1ED7i ch\u01B0a bi\u1EBFt. Xem stderr \u1EDF tr\xEAn + ch\u1EA1y `claude --print ok` tay \u0111\u1EC3 debug."}}function Bn(){let t=Kn("claude",["--print",pl],{encoding:"utf8",timeout:Ko,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Ko/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",r=t.stdout||"";if(t.status===0)return{ok:!0};let o=r.trim(),i=n.toLowerCase();if(o.length>20&&!i.includes("error")&&!i.includes("limit")&&!i.includes("quota")&&!i.includes("401"))return s.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=ml(`${n}
|
|
57
57
|
${r}`);return a==="unknown"&&(s.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&s.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&s.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as Vo}from"child_process";var dl=5e3,gl=/(\d+\.\d+\.\d+)/;function fl(){let e=lt()==="win32"?"where":"which",n=Vo(e,["claude"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function hl(){let t=Vo("claude",["--version"],{encoding:"utf8",timeout:dl});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return gl.exec(e)?.[1]??null}var gt=null;function Yt(){if(gt!==null)return gt;let t=fl();return t?(gt={installed:!0,version:hl(),path:t},gt):(gt={installed:!1,version:null,path:null},gt)}function xe(){gt=null}import{spawnSync as wl}from"child_process";f();var Bo=300*1e3,Wo="@anthropic-ai/claude-code",ft=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function kl(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new ft("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Wo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new ft("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new ft("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function qo(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=wl("npm",["install","-g",Wo],{stdio:["inherit","inherit","pipe"],timeout:Bo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ft("timeout",`npm install timeout sau ${Bo/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),kl(t.status,t.stderr||"");xe();let e=Yt();if(!e.installed||!e.path)throw new ft("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `claude` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return s.success(`\u0110\xE3 c\xE0i Claude Code${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{readFileSync as yl}from"fs";import{homedir as bl}from"os";import{join as vl}from"path";import{select as zo}from"@inquirer/prompts";function xl(){return vl(bl(),".claude","settings.json")}function Wn(){let t=xl(),e;try{e=yl(t,"utf8")}catch{return{exists:!1,hasBaseUrl:!1,hasToken:!1}}let n;try{n=JSON.parse(e)}catch{return{exists:!0,hasBaseUrl:!1,hasToken:!1}}let r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,i=typeof r.ANTHROPIC_AUTH_TOKEN=="string"&&r.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!o,baseUrl:o,hasToken:i,model:a,rawSettings:n}}async function Jo(t=Wn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await zo({message:`Ph\xE1t hi\u1EC7n AI config global (base URL: ${t.baseUrl}). D\xF9ng cho project n\xE0y?`,choices:[{name:"a. Yes \u2014 copy config global v\xE0o .claude/settings.json (per-project)",value:"use-global"},{name:"b. No \u2014 setup ri\xEAng (ch\u1ECDn provider kh\xE1c)",value:"setup-fresh"}]})==="use-global"?"use-global":await zo({message:"Ch\u1ECDn provider cho AI tools:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLM key NAL (ai.nal.vn \u2014 gateway n\u1ED9i b\u1ED9, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}f();import{password as Al,select as Cl}from"@inquirer/prompts";var Ae="https://api.anthropic.com",Sl="2023-06-01",Yo=1e4;function Tl(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Pl(t){let e=t.trim();return e.length===0?"API key b\u1EAFt bu\u1ED9c":e.startsWith("sk-ant-")?!0:"Anthropic API key th\u01B0\u1EDDng b\u1EAFt \u0111\u1EA7u b\u1EB1ng 'sk-ant-' (l\u1EA5y t\u1EEB console.anthropic.com)."}async function El(){return await Al({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Pl})}async function Rl(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Yo);try{let r=await fetch(`${Ae}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Sl,Accept:"application/json"},signal:e.signal});if(r.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(r.status===403)throw new Error("API key b\u1ECB reject (HTTP 403). Key c\xF3 th\u1EC3 \u0111\xE3 b\u1ECB revoke ho\u1EB7c thi\u1EBFu permission.");if(r.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let i=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(i.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return i}catch(r){throw r.name==="AbortError"?new Error(`Connect ${Ae} timeout sau ${Yo/1e3}s.`):r}finally{clearTimeout(n)}}async function $l(t){if(t.length===1){let n=t[0];return s.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=i=>{let a=i.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await Cl({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function Xo(){let t=await El();s.info(`Verify key (${Tl(t)}) qua ${Ae}/v1/models...`);let e=await Rl(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await $l(e);return{apiKey:t,baseUrl:Ae,model:n,availableModels:e}}f();w();import{promises as _l}from"fs";import{join as Il}from"path";var qn=384;function Nl(t){return Il(t,".claude","settings.json")}async function Ml(t){if(!await m(t))return{};try{return await y(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function Ol(t,e){let{env:n,...r}=t,o={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:i,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function Qo(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function Ll(t,e,n,r,o){let a={...Qo(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function Gl(t,e,n,r,o){let a={...Qo(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function jl(t,e){let n=e.env||{},r=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...r?{model:r}:{}}}async function ht(t,e){let n=Nl(t),r=await Ml(n),o;switch(e.provider){case"subscription":o=Ol(r,e.model);break;case"llmlite":o=Ll(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=Gl(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=jl(r,e.sourceSettings);break}await A(n,o,qn);try{await _l.chmod(n,qn)}catch{}return{path:n,mode:qn}}var H="sonnet";async function Ce(t){try{s.info("Setup AI provider cho workspace...");let e=Yt();if(e.installed)s.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(s.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),qo(),xe(),e=Yt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=Wn();switch(await Jo(n)){case"subscription":{let o=Fn();if(o.state!=="authenticated"&&(Vn(),o=Fn()),o.state==="authenticated"&&o.subscriptionType)return await ht(t.workspacePath,{provider:"subscription",model:H}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${H}`),{ok:!0,provider:"subscription",model:H};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=Bn();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Vn(),i=Bn()),!i.ok&&(i.reason==="timeout"||i.reason==="unknown"))return s.warn(`Probe verify ${i.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),await ht(t.workspacePath,{provider:"subscription",model:H}),await v("ai_setup",`provider=subscription,result=ok,probe=${i.reason}-soft-pass`),s.success(`AI ready \xB7 Subscription (probe ${i.reason}, soft-pass) \xB7 model=${H}`),{ok:!0,provider:"subscription",model:H};if(!i.ok){let a=i.reason??"unknown";return await v("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),s.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),s.warn(` \u2192 ${Fo(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await ht(t.workspacePath,{provider:"subscription",model:H}),await v("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${H}`),{ok:!0,provider:"subscription",model:H}}case"llmlite":{let o=await bo();return await ht(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await v("ai_setup",`provider=llmlite,result=ok,model=${o.model},base=${o.baseUrl},storage=settings.json`),s.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model,availableModels:o.availableModels}}case"anthropic":{let o=await Xo();return await ht(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await v("ai_setup",`provider=anthropic,result=ok,model=${o.model},storage=settings.json`),s.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.model,availableModels:o.availableModels}}case"use-global":{if(!n.rawSettings)throw new Error("use-global ch\u1ECDn nh\u01B0ng kh\xF4ng \u0111\u1ECDc \u0111\u01B0\u1EE3c global settings.");return await ht(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await v("ai_setup","provider=use-global,result=ok"),s.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return s.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await v("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}f();import{spawnSync as Ul}from"child_process";f();var zn=1e4,Zo=3e4,ei=5,Jn="say ok",ti="2023-06-01";async function Dl(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${J(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),zn);try{let i=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(s.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),b=c.length>5?` ...+${c.length-5} more`:"";s.dim(` Models: ${h}${b}`)}s.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:Jn}],max_tokens:ei}),signal:r.signal});if(!l.ok){let h=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${h}`)}let u=await l.json(),d=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";s.success(`Response: "${String(d).trim().slice(0,100)}"`),s.dim(` Tokens used: ${g}`)}catch(i){throw i.name==="AbortError"?new Error(`Timeout ${zn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}function Hl(){s.info("Testing Subscription provider qua `claude --print`...");let t=Ul("claude",["--print",Jn],{encoding:"utf8",timeout:Zo});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Zo/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}s.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function Kl(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${J(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),zn);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":ti},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);s.success(`Connectivity OK \xB7 ${c.length} models available`),s.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":ti,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:ei,messages:[{role:"user",content:Jn}]}),signal:r.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let d=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);s.success(`Response: "${d}"`)}finally{clearTimeout(o)}}async function ni(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,r=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,i=typeof t.model=="string"?t.model:"default";return o&&n?(await Kl(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await Dl(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Hl(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Se(){let t=process.cwd(),e=Tt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
58
58
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|