@nalvietnam/avatar-cli 3.14.0 → 3.15.1
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 +73 -71
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.js
CHANGED
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
// @nalvietnam/avatar-cli — built with tsup
|
|
2
|
-
var
|
|
3
|
-
`,n)}async function
|
|
4
|
-
`).filter(c=>c.trim().length>0)):n(new Error(`tar list exit ${a}: ${
|
|
5
|
-
`),r=
|
|
6
|
-
`),outcome:"replaced-block"}}return{content:e,outcome:"replaced-whole"}}var
|
|
7
|
-
`;await
|
|
2
|
+
var aa=Object.defineProperty;var M=(t,e)=>()=>(t&&(e=t(t=0)),e);var z=(t,e)=>{for(var n in e)aa(t,n,{get:e[n],enumerable:!0})};var yt={};z(yt,{copyDirRecursive:()=>ln,downloadFile:()=>un,ensureDir:()=>S,extractTarballToDir:()=>pn,pathExists:()=>d,readJson:()=>v,readText:()=>D,relativeFromCwd:()=>ma,removeRecursive:()=>J,writeJsonAtomic:()=>T,writeTextAtomic:()=>B});import{spawn as Mr,spawnSync as ca}from"child_process";import{createHash as Zm}from"crypto";import{constants as la,createReadStream as ed,promises as L}from"fs";import{dirname as Or,join as cn,relative as Nr,sep as ua}from"path";import{Readable as pa}from"stream";async function d(t){try{return await L.access(t,la.F_OK),!0}catch{return!1}}async function S(t){await L.mkdir(t,{recursive:!0})}async function D(t){return await L.readFile(t,"utf8")}async function v(t){return JSON.parse(await D(t))}async function B(t,e,n){await S(Or(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`,o=!1;try{await L.writeFile(r,e,"utf8"),n!==void 0&&await L.chmod(r,n),await L.rename(r,t),o=!0}finally{o||await L.rm(r,{force:!0}).catch(()=>{})}}async function T(t,e,n){await B(t,`${JSON.stringify(e,null,2)}
|
|
3
|
+
`,n)}async function ln(t,e,n=[]){await S(e);let r=await L.readdir(t,{withFileTypes:!0});for(let o of r){if(n.includes(o.name))continue;let s=cn(t,o.name),a=cn(e,o.name);if(o.isDirectory())await ln(s,a,n);else if(o.isSymbolicLink()){let c=await L.readlink(s);await L.symlink(c,a)}else await L.copyFile(s,a)}}async function J(t){await L.rm(t,{recursive:!0,force:!0})}function ma(t){return Nr(process.cwd(),t)}async function un(t,e,n=da){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await S(Or(e));let{createWriteStream:s}=await import("fs");try{let a=await fetch(t,{signal:r.signal});if(!a.ok)throw new Error(`Download fail (${a.status}) t\u1EEB ${t.slice(0,80)}...`);if(!a.body)throw new Error("Response kh\xF4ng c\xF3 body \u0111\u1EC3 t\u1EA3i.");let c=s(e);await new Promise((l,u)=>{let m=pa.fromWeb(a.body);m.on("error",u),c.on("error",u),c.on("finish",()=>l()),m.pipe(c)})}catch(a){throw await L.rm(e,{force:!0}).catch(()=>{}),a instanceof Error&&a.name==="AbortError"?new Error(`T\u1EA3i file qu\xE1 ${n/1e3}s (signed URL c\xF3 th\u1EC3 \u0111\xE3 h\u1EBFt h\u1EA1n).`):a}finally{clearTimeout(o)}}function ga(){return ue!==null||(ue=ca("tar",["--version"],{stdio:"ignore"}).status===0),ue}async function fa(t){return await new Promise((e,n)=>{let r=Mr("tar",["-tzf",t],{stdio:["ignore","pipe","pipe"]}),o="",s="";r.stdout.on("data",a=>{o+=a.toString()}),r.stderr.on("data",a=>{s+=a.toString()}),r.on("error",n),r.on("close",a=>{a===0?e(o.split(`
|
|
4
|
+
`).filter(c=>c.trim().length>0)):n(new Error(`tar list exit ${a}: ${s.trim()}`))})})}function ha(t,e){if(t.startsWith("/")||t.startsWith("~"))return!1;let n=cn(e,t),r=Nr(e,n);return r!==".."&&!r.startsWith(`..${ua}`)}async function pn(t,e){if(!ga())throw new Error("Kh\xF4ng t\xECm th\u1EA5y l\u1EC7nh `tar` tr\xEAn h\u1EC7 th\u1ED1ng. C\u1EA7n \u0111\u1EC3 gi\u1EA3i n\xE9n pack.\n Windows: c\xE0i qua WSL2, Git Bash, ho\u1EB7c MSYS2.\n macOS/Linux: tar th\u01B0\u1EDDng c\xF3 s\u1EB5n \u2014 ki\u1EC3m tra PATH.");await S(e);let r=(await fa(t)).find(o=>!ha(o,e));if(r)throw new Error(`Tarball ch\u1EE9a path kh\xF4ng an to\xE0n (path-traversal): "${r}". T\u1EEB ch\u1ED1i gi\u1EA3i n\xE9n \u0111\u1EC3 tr\xE1nh ghi \u0111\xE8 file ngo\xE0i th\u01B0 m\u1EE5c \u0111\xEDch.`);await new Promise((o,s)=>{let a=Mr("tar",["-xzf",t,"-C",e],{stdio:["ignore","ignore","pipe"]}),c="";a.stderr.on("data",l=>{c+=l.toString()}),a.on("error",s),a.on("close",l=>{l===0?o():s(new Error(`tar extract exit ${l}: ${c.trim()}`))})})}var da,ue,k=M(()=>{"use strict";da=3e5;ue=null});var Lr={};z(Lr,{addSubmodule:()=>ba,checkoutBranchHeadInSubmodule:()=>xa,checkoutTagInSubmodule:()=>va,currentBranch:()=>ya,currentCommitSha:()=>Sa,git:()=>I,isGitRepo:()=>ka,listTags:()=>Aa,tagAtHead:()=>Ca,workingTreeIsDirty:()=>Ta});import{join as mn}from"path";import{simpleGit as wa}from"simple-git";function I(t=process.cwd()){return wa({baseDir:t,binary:"git"})}async function ka(t=process.cwd()){return await d(mn(t,".git"))}async function ya(t=process.cwd()){return(await I(t).revparse(["--abbrev-ref","HEAD"])).trim()}async function ba(t,e,n=process.cwd()){await I(n).subModule(["add",t,e])}async function va(t,e,n=process.cwd()){let r=mn(n,t);await I(r).fetch(["--tags"]),await I(r).checkout(e)}async function xa(t,e,n=process.cwd()){let r=mn(n,t);await I(r).fetch(["origin"]),await I(r).checkout(["-B",e,`origin/${e}`])}async function Aa(t=process.cwd()){return(await I(t).tags()).all}async function Ca(t=process.cwd()){try{return(await I(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Sa(t=process.cwd()){return(await I(t).revparse(["HEAD"])).trim()}async function Ta(t=process.cwd()){return!(await I(t).status()).isClean()}var dn=M(()=>{"use strict";k()});function Gr(t,e){return t.replace($a,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var $a,Ur=M(()=>{"use strict";$a=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as Pa}from"fs";import{dirname as Dr,join as Gt}from"path";import{fileURLToPath as Ea}from"url";function Ma(t){let e=t;for(;;){if(Pa(Gt(e,"package.json")))return e;let n=Dr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function Oa(t){return await D(Gt(_a,`${t}.tpl`))}async function pe(t,e){let n=await Oa(t);return Gr(n,e)}async function Hr(t){return await D(Gt(Ia,`${t}.sh.tpl`))}var Ra,jr,_a,Ia,Fr=M(()=>{"use strict";k();Ur();Ra=Dr(Ea(import.meta.url)),jr=Ma(Ra),_a=Gt(jr,"src","templates"),Ia=Gt(jr,"src","hooks")});var Wr={};z(Wr,{MANAGED_END_PREFIX:()=>Br,MANAGED_START_PREFIX:()=>Vr,mergeClaudeMdManagedBlock:()=>Na});function Kr(t,e){return t.findIndex(n=>n.trimStart().startsWith(e))}function Na(t,e){if(t===void 0||t.trim()==="")return{content:e,outcome:"created"};let n=t.split(`
|
|
5
|
+
`),r=Kr(n,Vr),o=Kr(n,Br);if(r!==-1&&o!==-1&&r<o){let s=n.slice(0,r),a=n.slice(o+1);return{content:[...s,e.trimEnd(),...a].join(`
|
|
6
|
+
`),outcome:"replaced-block"}}return{content:e,outcome:"replaced-whole"}}var Vr,Br,qr=M(()=>{"use strict";Vr="<!-- AVATAR:MANAGED:START",Br="<!-- AVATAR:MANAGED:END"});var Zr={};z(Zr,{AVATAR_MANAGED_PATHS:()=>Jr,AVATAR_WORKSPACE_MARKER_RELATIVE:()=>Qr,backupIfExists:()=>Yr,createClaudeDirTree:()=>gn,installGitHook:()=>me,mergeRootClaudeMdManaged:()=>kn,writeAvatarWorkspaceMarker:()=>fn,writeClaudeGitignore:()=>bn,writeProjectKnowledgeFiles:()=>hn,writeProjectSettings:()=>yn,writeRootClaudeMd:()=>wn});import{promises as zr}from"fs";import{join as H}from"path";async function Yr(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,r=n,o=1;for(;await d(r);)if(r=`${n}-${o}`,o++,o>5)throw new Error(`Could not find free backup name for ${t}`);return await zr.rename(t,r),r}async function Xr(t,e,n){let r=await Yr(t);return await B(t,e,n),r}async function gn(t){let e=H(t,".claude");await S(e);for(let n of La){let r=H(e,n);await S(r),await B(H(r,".gitkeep"),"")}}async function fn(t,e){let n=H(t,Qr),r=`${JSON.stringify({avatarWorkspace:!0,avatarVersion:e.avatarVersion,workspaceName:e.workspaceName,createdAt:new Date().toISOString()},null,2)}
|
|
7
|
+
`;await B(n,r)}async function hn(t,e){return[]}async function wn(t,e){let n=await pe("CLAUDE.md",e);return await Xr(H(t,"CLAUDE.md"),n)}async function kn(t,e){let n=H(t,"CLAUDE.md");if(!await d(n))return{outcome:"skipped-no-file"};let r=await zr.readFile(n,"utf8"),o=await pe("CLAUDE.md",e),{mergeClaudeMdManagedBlock:s}=await Promise.resolve().then(()=>(qr(),Wr)),a=s(r,o);return await B(n,a.content),{outcome:a.outcome}}async function yn(t,e){let n=await pe("settings.json",e);return await Xr(H(t,".claude","settings.json"),n)}async function bn(t){let e=H(t,".claude",".gitignore");await B(e,Ga)}async function me(t,e){let n=await Hr(e),r=H(t,"hooks");await S(r);let o=H(r,e);await B(o,n,493)}var Jr,La,Qr,Ga,bt=M(()=>{"use strict";k();Fr();Jr=[".claude","CLAUDE.md"];La=["state","_pending","_backup"];Qr=".claude/avatar.json";Ga=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
|
|
8
8
|
# Workspace root KH\xD4NG git; file n\xE0y ph\xF2ng tr\u01B0\u1EDDng h\u1EE3p .claude/ b\u1ECB track nh\u1EA7m.
|
|
9
9
|
settings.json
|
|
10
10
|
settings.json.backup-*
|
|
@@ -13,52 +13,52 @@ settings.json.backup-*
|
|
|
13
13
|
_pending/
|
|
14
14
|
_backup/
|
|
15
15
|
state/session-*.json
|
|
16
|
-
`});var at={};
|
|
17
|
-
`),await o.add("."),await o.commit("chore: initial commit (avatar add repo \u2014 d\u1EF1 \xE1n m\u1EDBi)").catch(()=>{}),await o.addRemote("origin",t.url).catch(()=>{}),await o.push(["-u","origin","HEAD"]).catch(a=>{}),await
|
|
16
|
+
`});var at={};z(at,{REPOS_MANIFEST_RELATIVE:()=>to,addRepoToManifest:()=>ge,readReposManifest:()=>de,removeRepoFromManifest:()=>Da,repoNameExists:()=>fe});import{join as Ua}from"path";function vn(t){return Ua(t,to)}async function de(t){let e=vn(t);if(!await d(e))return[];try{let n=await v(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function ge(t,e){let r=(await de(t)).filter(o=>o.name!==e.name);r.push(e),await T(vn(t),{repos:r})}async function Da(t,e){let n=await de(t);await T(vn(t),{repos:n.filter(r=>r.name!==e)})}async function fe(t,e){return(await de(t)).some(r=>r.name===e)}var to,Y=M(()=>{"use strict";k();to=".claude/repos.json"});var Pn={};z(Pn,{RepoAlreadyInWorkspaceError:()=>lt,cloneCodeRepoIntoSrc:()=>Cn,folderDirtyStatus:()=>Tn,inferRepoNameFromUrl:()=>An,readFolderRemoteUrl:()=>$n,scaffoldNewRepoInSrc:()=>Sn,validateRepoName:()=>xn});import{join as ct}from"path";function xn(t){return!t||t.trim().length===0?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng.":/[/\\]/.test(t)||t.includes("..")?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..'.":t==="pack"?"T\xEAn 'pack' b\u1ECB gi\u1EEF cho team-ai-pack.":null}function An(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function Cn(t){let e=xn(t.name);if(e)throw new lt(e);let n=ct(t.workspaceRoot,"src");await S(n);let r=ct(n,t.name),o=await d(r),s=await fe(t.workspaceRoot,t.name);if(o)throw new lt(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc (avatar remove repo).`);if(s){let{removeRepoFromManifest:c}=await Promise.resolve().then(()=>(Y(),at));await c(t.workspaceRoot,t.name)}await I(n).clone(t.url,t.name);let a=ct(r,".git");return await me(a,"pre-push").catch(()=>{}),await ge(t.workspaceRoot,{name:t.name,path:`src/${t.name}`,url:t.url,addedAt:new Date().toISOString()}),{name:t.name,path:r,url:t.url}}async function Sn(t){let e=xn(t.name);if(e)throw new Error(e);if(await fe(t.workspaceRoot,t.name))throw new Error(`Repo "${t.name}" \u0111\xE3 c\xF3 trong workspace. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc.`);let n=ct(t.workspaceRoot,"src");await S(n);let r=ct(n,t.name);if(await d(r))throw new Error(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c.`);await S(r);let o=I(r);await o.init();let{writeTextAtomic:s}=await Promise.resolve().then(()=>(k(),yt));return await s(ct(r,"README.md"),`# ${t.name}
|
|
17
|
+
`),await o.add("."),await o.commit("chore: initial commit (avatar add repo \u2014 d\u1EF1 \xE1n m\u1EDBi)").catch(()=>{}),await o.addRemote("origin",t.url).catch(()=>{}),await o.push(["-u","origin","HEAD"]).catch(a=>{}),await me(ct(r,".git"),"pre-push").catch(()=>{}),await ge(t.workspaceRoot,{name:t.name,path:`src/${t.name}`,url:t.url,addedAt:new Date().toISOString()}),{name:t.name,path:r,url:t.url}}async function Tn(t){try{let e=await I(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function $n(t){try{let n=(await I(t).getRemotes(!0)).find(r=>r.name==="origin");return n?.refs.fetch??n?.refs.push??null}catch{return null}}var lt,Ut=M(()=>{"use strict";k();dn();bt();Y();lt=class extends Error{constructor(e){super(e),this.name="RepoAlreadyInWorkspaceError"}}});import p from"chalk";import Ha from"ora";function X(t){return Ha({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Dt(t){let e=Date.now(),n=X(`${t} (0:00)`),r=()=>{let s=Math.floor((Date.now()-e)/1e3),a=Math.floor(s/60),c=s%60;return`${a}:${String(c).padStart(2,"0")}`},o=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:s=>{clearInterval(o),n.succeed(`${s} (${r()})`)},fail:s=>{clearInterval(o),n.fail(`${s} (${r()})`)},stop:()=>{clearInterval(o),n.stop()}}}var i,h=M(()=>{"use strict";i={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
|
|
18
18
|
`),success:t=>process.stdout.write(`${p.green("\u2713")} ${t}
|
|
19
19
|
`),warn:t=>process.stdout.write(`${p.yellow("\u26A0")} ${t}
|
|
20
20
|
`),error:t=>process.stderr.write(`${p.red("\u2717")} ${t}
|
|
21
21
|
`),dim:t=>process.stdout.write(`${p.dim(t)}
|
|
22
22
|
`),plain:t=>process.stdout.write(`${t}
|
|
23
|
-
`)}});import{spawnSync as
|
|
24
|
-
`),new
|
|
25
|
-
`);let
|
|
26
|
-
`),r=
|
|
27
|
-
Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)`);break;case"network":
|
|
23
|
+
`)}});import{spawnSync as Ya}from"child_process";function ke(){let t=Ya("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"],timeout:1e4});if(t.signal==="SIGTERM"||t.error?.code==="ETIMEDOUT")throw new Error("Qu\xE1 th\u1EDDi gian l\u1EA5y GitHub username (10s). Ki\u1EC3m tra m\u1EA1ng / k\u1EBFt n\u1ED1i GitHub r\u1ED3i th\u1EED l\u1EA1i.");if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var Rn=M(()=>{"use strict"});import{spawnSync as Xa}from"child_process";function _n(t){let e=t.toLowerCase();return e.includes("name already exists")||e.includes("already exists")?"name-exists":e.includes("403")||e.includes("permission")||e.includes("not have access")||e.includes("resource not accessible")?"no-permission":e.includes("could not resolve")||e.includes("network")||e.includes("timeout")?"network":"unknown"}async function ao(t,e,n){let o=`${n??ke()}/${t}`;i.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let s=Xa("gh",["repo","create",o,`--${e}`],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(s.status!==0){let a=(s.stderr||"").trim();throw a&&process.stderr.write(`${a}
|
|
24
|
+
`),new Ht(_n(a),`T\u1EA1o repo ${o} th\u1EA5t b\u1EA1i.`)}return i.success(`\u0110\xE3 t\u1EA1o: git@github.com:${o}.git`),`git@github.com:${o}.git`}var Ht,In=M(()=>{"use strict";Rn();h();Ht=class extends Error{reason;constructor(e,n){super(n),this.name="RepoCreateError",this.reason=e}}});import{spawnSync as pc}from"child_process";import{existsSync as mc,statSync as dc}from"fs";function go(t){let e=`${t.org}/${t.name}`;if(!mc(t.folder)||!dc(t.folder).isDirectory())throw new Ct("missing-folder",`Folder ngu\u1ED3n kh\xF4ng t\u1ED3n t\u1EA1i: ${t.folder}. Ki\u1EC3m tra l\u1EA1i \u0111\u01B0\u1EDDng d\u1EABn.`);let n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=pc("gh",n,{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(r.status!==0){let o=(r.stderr||"").trim();o&&process.stderr.write(`${o}
|
|
25
|
+
`);let s=_n(o);throw s==="name-exists"?new Mn(e):s==="no-permission"?new Ct("no-permission",`Kh\xF4ng c\xF3 quy\u1EC1n t\u1EA1o repo ${e}. Ki\u1EC3m tra \u0111\u0103ng nh\u1EADp \u0111\xFAng account/org ch\u01B0a (gh auth status).`):s==="network"?new Ct("network",`L\u1ED7i m\u1EA1ng khi t\u1EA1o repo ${e}.`):new Ct("unknown",`gh repo create th\u1EA5t b\u1EA1i (exit ${r.status}).`)}return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}var Mn,Ct,fo=M(()=>{"use strict";In();Mn=class extends Error{constructor(e){super(`Repo "${e}" \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. \u0110\u1ED5i t\xEAn ho\u1EB7c x\xF3a repo c\u0169.`),this.name="RepoAlreadyExistsError"}},Ct=class extends Error{reason;constructor(e,n){super(n),this.name="GhRepoCreateError",this.reason=e}}});function ho(t){if(!gc.test(t))throw new On(t)}function wo(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}var gc,On,ko=M(()=>{"use strict";gc=/^[a-zA-Z0-9._-]{1,100}$/,On=class extends Error{constructor(e){super(`T\xEAn repo "${e}" kh\xF4ng h\u1EE3p l\u1EC7. Ch\u1EC9 d\xF9ng ch\u1EEF/s\u1ED1/d\u1EA5u ch\u1EA5m/g\u1EA1ch/underscore, d\xE0i 1-100 k\xFD t\u1EF1.`),this.name="InvalidRepoNameError"}}});var yo={};z(yo,{createGithubRemoteFromFolder:()=>Nn});function Nn(t){ho(t.name),wo(t.visibility);let e=t.org??ke();i.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=go({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return i.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}var Ln=M(()=>{"use strict";fo();Rn();h();ko()});var tr={};z(tr,{ALLOWED_HOSTED_DOMAINS:()=>Re,HOSTED_DOMAIN:()=>Ll,SCOPES:()=>si,buildUserConfig:()=>Qn,buildVerificationUrl:()=>Zn,decodeIdToken:()=>Yn,pollForToken:()=>Jn,refreshAccessToken:()=>Hl,requestDeviceCode:()=>zn,revokeToken:()=>zt,verifyHostedDomain:()=>Xn,verifyIdTokenClaims:()=>ci});async function zn(){let t=new URLSearchParams({client_id:_e,scope:si.join(" ")}),e=await fetch(Gl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:t});if(!e.ok){let n=await e.text();throw new Error(`Device code request failed (${e.status}): ${n}`)}return await e.json()}async function Jn(t){let e=new URLSearchParams({client_id:_e,client_secret:ii,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(ai,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(n.ok)return await n.json();let r="";try{r=(await n.json()).error??""}catch{r=""}if(r==="authorization_pending"||r==="slow_down")return null;throw r==="access_denied"?new Error("User t\u1EEB ch\u1ED1i quy\u1EC1n truy c\u1EADp"):r==="expired_token"?new Error("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."):new Error(`OAuth token endpoint tr\u1EA3 l\u1ED7i: ${r||n.status}`)}function Yn(t){let e=t.split(".");if(e.length!==3)throw new Error("id_token format kh\xF4ng h\u1EE3p l\u1EC7");let n=e[1];if(!n)throw new Error("id_token thi\u1EBFu payload");let r=n.replace(/-/g,"+").replace(/_/g,"/"),o=Buffer.from(r,"base64").toString("utf8");return JSON.parse(o)}function ci(t){if(!Dl.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==_e)throw new Error("id_token audience kh\xF4ng kh\u1EDBp client ID Avatar. Token c\xF3 th\u1EC3 \u0111\u01B0\u1EE3c sign cho app kh\xE1c.");let e=Math.floor(Date.now()/1e3);if(t.exp+jl<e){let n=e-t.exp;throw new Error(`id_token \u0111\xE3 h\u1EBFt h\u1EA1n ${n}s tr\u01B0\u1EDBc. Login l\u1EA1i \u0111\u1EC3 l\u1EA5y token m\u1EDBi.`)}if(!t.hd||!Re.includes(t.hd))throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u ${Re.map(n=>`@${n}`).join(" ho\u1EB7c ")}). Nh\u1EADn: ${t.email}${t.hd?` (hd=${t.hd})`:" (kh\xF4ng ph\u1EA3i t\xE0i kho\u1EA3n Workspace)"}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function Qn(t,e){let n=new Date(Date.now()+t.expires_in*1e3).toISOString();return{email:e.email,name:e.name??e.email,access_token:t.access_token,refresh_token:t.refresh_token,expires_at:n,id_token:t.id_token}}async function Hl(t){let e=new URLSearchParams({client_id:_e,client_secret:ii,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(ai,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(!n.ok){let r=await n.text();throw new Error(`Refresh token failed (${n.status}): ${r}`)}return await n.json()}async function zt(t){let e=new URLSearchParams({token:t}),n=new AbortController,r=setTimeout(()=>n.abort(),1e4);await fetch(Ul,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e,signal:n.signal}).catch(()=>{}).finally(()=>clearTimeout(r))}function Zn(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.toString()}var _e,ii,Re,Ll,si,Gl,ai,Ul,Dl,jl,Xn,Jt=M(()=>{"use strict";_e="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",ii="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",Re=["nal.vn","nal-software.com"],Ll=Re[0],si=["openid","email","profile"],Gl="https://oauth2.googleapis.com/device/code",ai="https://oauth2.googleapis.com/token",Ul="https://oauth2.googleapis.com/revoke";Dl=new Set(["https://accounts.google.com","accounts.google.com"]),jl=60;Xn=ci});import{Command as zm}from"commander";Ut();import{resolve as No}from"path";import{confirm as Ae,input as Bt,select as Dn}from"@inquirer/prompts";Ut();k();import{join as Ba}from"path";import{confirm as Wa,input as we,select as io}from"@inquirer/prompts";function vt(t){let e=t.trim();if(!e)return null;let n=e.match(/^https?:\/\/([^/]+)/);if(n)return n[1].toLowerCase();let r=e.match(/^git@([^:]+):/);return r?r[1].toLowerCase():null}function he(t){return t?t.toLowerCase()==="github.com":!1}import{input as no}from"@inquirer/prompts";import{spawnSync as ja}from"child_process";function eo(t){let n=[`protocol=${t.protocol??"https"}`,`host=${t.host}`,`username=${t.username}`,`password=${t.token}`,""].join(`
|
|
26
|
+
`),r=ja("git",["credential","approve"],{input:n,encoding:"utf8",stdio:["pipe","pipe","pipe"]});return r.status===0?{ok:!0}:{ok:!1,error:(r.stderr||"").trim()||(r.error?.message??"git credential approve th\u1EA5t b\u1EA1i")}}h();async function jt(t){let e=`https://${t}/-/user_settings/personal_access_tokens`;i.info(`C\u1EA7n Personal Access Token (PAT) \u0111\u1EC3 truy c\u1EADp ${t}.`),i.dim(` T\u1EA1o PAT t\u1EA1i: ${e}`),i.dim(" Scope t\u1ED1i thi\u1EC3u: read_repository (th\xEAm write_repository n\u1EBFu c\u1EA7n push)."),i.dim(' L\u01B0u \xFD: "password" git y\xEAu c\u1EA7u l\xE0 PAT \u2014 KH\xD4NG ph\u1EA3i m\u1EADt kh\u1EA9u web login.');let n=await no({message:`Username tr\xEAn ${t} (th\u01B0\u1EDDng l\xE0 email ho\u1EB7c account handle):`,validate:s=>s.trim().length>0?!0:"Username b\u1EAFt bu\u1ED9c"}),r=await no({message:`Personal Access Token (PAT) cho ${t}:`,validate:s=>s.trim().length>0?!0:"Token b\u1EAFt bu\u1ED9c"}),o=eo({host:t,username:n.trim(),token:r.trim()});return o.ok?(i.success(`Credential \u0111\xE3 l\u01B0u v\xE0o keychain cho ${t}. \u0110ang verify l\u1EA1i...`),!0):(i.warn(`L\u01B0u credential th\u1EA5t b\u1EA1i: ${o.error??"l\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh"}`),i.dim(" Th\u1EED ch\u1EA1y tay: git credential approve, ho\u1EB7c check git credential helper config."),!1)}h();h();import{spawnSync as Fa}from"child_process";function xt(){i.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=Fa("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);i.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}import{spawnSync as Ka}from"child_process";var En=5e3;function Va(t){let e=t.toLowerCase();return e.includes("authentication")||e.includes("could not read username")||e.includes("terminal prompts disabled")||e.includes("permission denied")||e.includes("403")||e.includes("access denied")||e.includes("repository not found")?"no-access":e.includes("404")||e.includes("does not exist")?"not-found":e.includes("could not resolve host")||e.includes("network")||e.includes("connection refused")||e.includes("connection timed out")?"network":"unknown"}function At(t){let e=Ka("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:En,stdio:["ignore","pipe","pipe"],env:{...process.env,GIT_TERMINAL_PROMPT:"0"}});if(e.error){let o=e.error;return o.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:o.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${En/1e3}s`}:{ok:!1,reason:"unknown",detail:o.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${En/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:Va(n),detail:n.slice(0,300)}}function qa(t,e,n){switch(i.warn(`\u26D4 Kh\xF4ng clone \u0111\u01B0\u1EE3c ${t}`),e){case"not-found":i.dim(" \u2192 Repo kh\xF4ng t\u1ED3n t\u1EA1i (sai URL, ho\u1EB7c repo ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o tr\xEAn GitHub).");break;case"no-access":i.dim(` \u2192 Kh\xF4ng c\xF3 quy\u1EC1n truy c\u1EADp (repo private + sai account), HO\u1EB6C repo kh\xF4ng t\u1ED3n t\u1EA1i.
|
|
27
|
+
Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)`);break;case"network":i.dim(" \u2192 L\u1ED7i m\u1EA1ng (m\u1EA5t k\u1EBFt n\u1ED1i / DNS / timeout).");break;default:i.dim(` \u2192 L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh${n?`: ${n}`:""}.`)}}function ro(){i.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");try{xt()}catch(t){i.warn(`${t instanceof Error?t.message:t}. C\xF3 th\u1EC3 ch\u1EA1y 'gh auth login' tay r\u1ED3i retry.`)}}async function oo(t,e){let n=vt(e),r=he(n),o=[{name:"Th\u1EED l\u1EA1i (retry)",value:"retry"},{name:"Nh\u1EADp URL repo kh\xE1c",value:"new-url"}];return new Set(["no-access","unknown","timeout"]).has(t)&&(r?o.push({name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch-account"}):n&&o.push({name:`Nh\u1EADp Personal Access Token cho ${n} (l\u01B0u v\xE0o keychain)`,value:"enter-pat"})),o.push({name:"B\u1ECF qua repo n\xE0y",value:"skip"}),await io({message:"C\xE1ch x\u1EED l\xFD?",choices:o})}async function so(t){let e=t.url,n=t.name;for(;;){let r=At(e);if(!r.ok){qa(e,r.reason??"unknown",r.detail);let o=await oo(r.reason??"unknown",e);if(o==="skip")return i.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){ro();continue}if(o==="enter-pat"){let s=vt(e);s&&await jt(s);continue}if(o==="new-url"){if(e=await we({message:"URL repo kh\xE1c:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),await Wa({message:`\u0110\u1ED5i t\xEAn th\u01B0 m\u1EE5c trong src/ theo URL m\u1EDBi? (hi\u1EC7n: ${n})`,default:!1})){let{inferRepoNameFromUrl:a}=await Promise.resolve().then(()=>(Ut(),Pn));n=await we({message:"T\xEAn th\u01B0 m\u1EE5c:",default:a(e)})}continue}continue}try{return{cloned:await Cn({workspaceRoot:t.workspaceRoot,url:e,name:n}),skipped:!1}}catch(o){let s=o instanceof Error?o.message:String(o);if(o instanceof lt){let c=await za(t.workspaceRoot,n);if(c&&Ja(c,e))return i.warn(`\u26A0 Repo n\xE0y \u0111\xE3 c\xF3 trong workspace (src/${n}). D\xF9ng tr\u1EF1c ti\u1EBFp, kh\xF4ng add l\u1EA1i.`),i.dim(` Mu\u1ED1n thay b\u1EB1ng b\u1EA3n m\u1EDBi? G\u1EE1 tr\u01B0\u1EDBc: avatar remove repo ${n}`),{cloned:null,skipped:!0};if(i.warn(`\u26A0 ${s}`),await io({message:"C\xE1ch x\u1EED l\xFD?",choices:[{name:"\u0110\u1ED5i t\xEAn th\u01B0 m\u1EE5c kh\xE1c",value:"rename"},{name:"B\u1ECF qua repo n\xE0y",value:"skip"}]})==="skip")return{cloned:null,skipped:!0};n=await we({message:"T\xEAn th\u01B0 m\u1EE5c m\u1EDBi trong src/:",validate:u=>u.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"});continue}i.warn(`Clone fail: ${s}`),await J(Ba(t.workspaceRoot,"src",n)).catch(()=>{});let a=await oo("unknown",e);if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){ro();continue}if(a==="enter-pat"){let c=vt(e);c&&await jt(c);continue}a==="new-url"&&(e=await we({message:"URL repo kh\xE1c:",validate:c=>c.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}))}}}async function za(t,e){let{readReposManifest:n}=await Promise.resolve().then(()=>(Y(),at));return(await n(t)).find(o=>o.name===e)?.url??null}function Ja(t,e){let n=r=>r.trim().toLowerCase().replace(/^git@/,"").replace(/^https?:\/\//,"").replace(/:/,"/").replace(/\.git$/,"").replace(/\/+$/,"");return n(t)===n(e)}In();import{input as Za,select as tc}from"@inquirer/prompts";h();import{input as rg,select as Qa}from"@inquirer/prompts";var $=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function F(t){i.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&i.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await Qa({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}h();function ec(t){switch(t){case"name-exists":i.warn("\u26D4 Repo t\xEAn n\xE0y \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub."),i.dim(" \u2192 \u0110\u1ED5i t\xEAn kh\xE1c, ho\u1EB7c d\xF9ng account/org kh\xE1c.");break;case"no-permission":i.warn("\u26D4 Kh\xF4ng c\xF3 quy\u1EC1n t\u1EA1o repo (sai account / org)."),i.dim(" \u2192 Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)");break;case"network":i.warn("\u26D4 L\u1ED7i m\u1EA1ng khi t\u1EA1o repo.");break;default:i.warn("\u26D4 T\u1EA1o repo th\u1EA5t b\u1EA1i (l\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh)."),i.dim(" \u2192 Ki\u1EC3m tra gh auth status + th\u1EED l\u1EA1i.")}}async function nc(t){let e=[];return t==="name-exists"&&e.push({name:"\u0110\u1ED5i t\xEAn repo kh\xE1c",value:"rename"}),(t==="no-permission"||t==="unknown")&&e.push({name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch-account"}),e.push({name:"Th\u1EED l\u1EA1i (retry)",value:"retry"}),e.push({name:"H\u1EE7y",value:"abort"}),await tc({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function co(t,e){let n=t;for(;;)try{return await ao(n,e)}catch(r){let o=r instanceof Ht?r.reason:"unknown",s=await uo(o,n);if(s.action==="retry")continue;n=s.newName}}async function lo(t,e){let n=t;for(;;)try{return e(n)}catch(r){let o=r instanceof Error&&r.name==="RepoAlreadyExistsError"?"name-exists":"unknown",s=await uo(o,n);if(s.action==="retry")continue;n=s.newName}}async function uo(t,e){ec(t);let n=await nc(t);if(n==="abort")throw new $("User h\u1EE7y t\u1EA1i b\u01B0\u1EDBc t\u1EA1o repo GitHub.");return n==="switch-account"?(xt(),{action:"retry",newName:e}):n==="rename"?{action:"rename",newName:(await Za({message:"T\xEAn repo m\u1EDBi:",default:e,validate:o=>o.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"})).trim()}:{action:"retry",newName:e}}import{spawnSync as po}from"child_process";import{platform as rc}from"os";function j(){let t=rc();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var oc=5e3,ic=/(\d+\.\d+\.\d+)/;function sc(){let e=j()==="win32"?"where":"which",n=po(e,["gitnexus"],{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 ac(){let t=po("gitnexus",["--version"],{encoding:"utf8",timeout:oc});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return ic.exec(e)?.[1]??null}var ut=null;function pt(){if(ut!==null)return ut;let t=sc();return t?(ut={installed:!0,version:ac(),path:t},ut):(ut={installed:!1,version:null,path:null},ut)}function ye(){ut=null}import{spawnSync as cc}from"child_process";function be(){let t=cc("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as lc}from"child_process";function uc(t){let e=j();return lc(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function mo(){let t=j(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(uc(n))return n;return null}import{spawnSync as xo}from"child_process";import{input as xc,select as Ac}from"@inquirer/prompts";Ln();k();h();import{spawnSync as Gn}from"child_process";import{promises as fc}from"fs";import{basename as hc,join as bo}from"path";import{confirm as wc,select as kc}from"@inquirer/prompts";function yc(){let t=new Date;return`${t.getFullYear().toString().slice(-2)}${String(t.getMonth()+1).padStart(2,"0")}${String(t.getDate()).padStart(2,"0")}-${String(t.getHours()).padStart(2,"0")}${String(t.getMinutes()).padStart(2,"0")}`}async function bc(t){let e=bo(t,".git");if(!await d(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${yc()}`,r=bo(t,n);return await fc.rename(e,r),i.success(`Backup .git \u2192 ${n}`),r}function vc(t){let e=Gn("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);i.success("Git init m\u1EDBi (branch main)");let n=Gn("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0&&i.warn(`git add -A th\u1EA5t b\u1EA1i: ${(n.stderr||"").slice(0,200)}`);let r=Gn("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});r.status!==0?i.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(r.stderr||"").slice(0,200)}`):i.success("Initial commit")}async function vo(t){let e=hc(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await wc({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
|
|
28
28
|
1. Backup .git \u2192 .git.backup-{timestamp}
|
|
29
29
|
2. Git init m\u1EDBi (branch main, initial commit)
|
|
30
30
|
3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
|
|
31
|
-
Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await
|
|
31
|
+
Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await kc({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await bc(t.folderPath);return vc(t.folderPath),{newRemoteUrl:Nn({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}h();var Ft=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Cc(){let t=xo("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Sc(){i.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=xo("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&i.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Tc(t,e,n){switch(t){case"no-access":return n?`Repo c\xF3 th\u1EC3 private v\xE0 account '${n}' kh\xF4ng c\xF3 quy\u1EC1n access. Switch sang account c\xF3 quy\u1EC1n, ho\u1EB7c xin invite t\u1EEB owner repo.`:"Repo c\xF3 th\u1EC3 private v\xE0 gh CLI ch\u01B0a login. Login tr\u01B0\u1EDBc r\u1ED3i retry.";case"not-found":return`URL c\xF3 th\u1EC3 sai ch\xEDnh t\u1EA3, ho\u1EB7c repo \u0111\xE3 b\u1ECB x\xF3a/rename. Nh\u1EADp l\u1EA1i URL \u0111\xFAng. Check: ${e}`;case"network":return"Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c GitHub. Check m\u1EA1ng / VPN / firewall.";case"timeout":return"Network ch\u1EADm > 5s. Check m\u1EA1ng r\u1ED3i retry.";default:return"L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh. URL c\xF3 th\u1EC3 sai \u2014 nh\u1EADp l\u1EA1i, ho\u1EB7c check gh auth status."}}function $c(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function Ao(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=Cc();i.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),i.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),i.info(Tc(n,e,o)),o&&i.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);let s=vt(e),a=he(s),c=new Set(["no-access","unknown","timeout"]),l=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"}];c.has(n)&&(a?l.push({name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"}):s&&l.push({name:`Nh\u1EADp Personal Access Token cho ${s} (l\u01B0u v\xE0o keychain)`,value:"enter-pat"})),l.push({name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}),t.folderPath&&l.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),l.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let u=await Ac({message:"C\xE1ch x\u1EED l\xFD?",choices:l});if(u==="abort")throw new Ft(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(u==="reset-folder"){if(!t.folderPath){i.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let g=await vo({folderPath:t.folderPath,visibility:t.defaultVisibility});return i.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${g.backupPath}`),i.success(`Remote m\u1EDBi: ${g.newRemoteUrl}`),{resolvedUrl:g.newRemoteUrl}}catch(g){i.warn(`Reset folder th\u1EA5t b\u1EA1i: ${g.message}`);continue}}u==="re-input-url"&&(e=(await xc({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:f=>$c(f)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),u==="switch"&&Sc(),u==="enter-pat"&&s&&await jt(s),i.info(`Verify remote l\u1EA1i: ${e}...`);let m=At(e);if(m.ok)return i.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=m.reason??"unknown",r=m.detail}}h();import{spawnSync as Pc}from"child_process";var Ec={brew:{cmd:"brew",args:["install","gh"]},apt:{cmd:"sudo",args:["apt-get","install","-y","gh"]},dnf:{cmd:"sudo",args:["dnf","install","-y","gh"]},pacman:{cmd:"sudo",args:["pacman","-S","--noconfirm","github-cli"]},winget:{cmd:"winget",args:["install","--id","GitHub.cli","-e","--silent"]}};function Co(t){let e=Ec[t];i.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=Pc(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);i.success("\u0110\xE3 c\xE0i gh CLI")}h();import{spawnSync as Rc}from"child_process";function So(){if(Rc("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){i.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}i.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}h();async function To(t){for(;be()==="not-installed";){i.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=mo();if(!e){if(await F({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{Co(e)}catch(n){if(await F({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;be()==="not-authenticated";){i.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{xt()}catch(e){if(await F({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(be()!=="authenticated"&&await F({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(i.success("gh CLI s\u1EB5n s\xE0ng"),So(),t){let e=At(t);return e.ok?(i.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await Ao({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as St}from"fs";import{dirname as _c,join as Tt}from"path";var Ic=5;function Mc(t){if(St(Tt(t,".claude","avatar.json")))return!0;let e=St(Tt(t,".claude")),n=St(Tt(t,"CLAUDE.md")),r=St(Tt(t,"src"));if(!(e&&n&&r))return!1;let o=St(Tt(t,".claude","repos.json")),s=St(Tt(t,".claude","pack"));return o||s}function $t(t){let e=t;for(let n=0;n<Ic;n++){if(Mc(e))return e;let r=_c(e);if(r===e)return null;e=r}return null}import{appendFileSync as Nc,existsSync as $o,readFileSync as Lc,writeFileSync as Gc}from"fs";import{join as Un}from"path";import{spawn as Oc}from"child_process";function E(t,e,n={}){let{cwd:r,timeoutMs:o,forwardSigint:s=!0}=n;return new Promise((a,c)=>{let u=Oc(t,e,{cwd:r,stdio:["ignore","pipe","pipe"]}),m="",g="",f=!1,w=!1,C=o?setTimeout(()=>{f=!0,u.kill("SIGTERM")},o):null,R=()=>u.kill("SIGINT");s&&process.on("SIGINT",R);let U=()=>{C&&clearTimeout(C),s&&process.removeListener("SIGINT",R)};u.stdout?.on("data",A=>{m+=A.toString()}),u.stderr?.on("data",A=>{g+=A.toString()}),u.on("error",A=>{w||(w=!0,U(),c(A))}),u.on("close",(A,N)=>{w||(w=!0,U(),a({status:A,signal:N,stdout:m,stderr:g,timedOut:f}))})})}h();var Uc=120*1e3,Dc=300*1e3,Q=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,o=null,s){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=o,this.stderr=s}};function Po(t,e,n,r){if(n==="SIGTERM")return new Q(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,r);let o=r.toLowerCase();return o.includes("eacces")||o.includes("permission denied")?new Q(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new Q(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function ve(t,e){return t.split(`
|
|
32
32
|
`).slice(-e).join(`
|
|
33
|
-
`)}async function
|
|
34
|
-
`):r&&process.stderr.write(`${
|
|
35
|
-
`),
|
|
36
|
-
`):
|
|
37
|
-
`),
|
|
33
|
+
`)}async function Eo(){let t=Dt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=await E("gitnexus",["setup"],{timeoutMs:Uc});if(e.status!==0||e.timedOut){t.fail("GitNexus setup failed");let n=e.stderr.trim(),r=e.stdout.trim();throw n?process.stderr.write(`${ve(n,30)}
|
|
34
|
+
`):r&&process.stderr.write(`${ve(r,30)}
|
|
35
|
+
`),Po("setup",e.status,e.timedOut?"SIGTERM":null,n)}t.succeed("GitNexus setup OK (global skills installed)")}async function Pt(t){let e=Dt(`Analyze ${t} (1-3 ph\xFAt)`),n=await E("gitnexus",["analyze",".","--index-only"],{cwd:t,timeoutMs:Dc});if(n.status!==0||n.timedOut){e.fail("Analyze failed");let o=n.stderr.trim(),s=n.stdout.trim();throw o?process.stderr.write(`${ve(o,30)}
|
|
36
|
+
`):s&&process.stderr.write(`${ve(s,30)}
|
|
37
|
+
`),Po("analyze",n.status,n.timedOut?"SIGTERM":null,o)}let r=Un(t,".gitnexus","meta.json");if(!$o(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new Q("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);jc(t),e.succeed(`Analyze OK (index t\u1EA1i ${Un(t,".gitnexus")}, \u0111\xE3 gitignore)`)}function jc(t){let e=Un(t,".gitignore"),n=".gitnexus/";try{let r="";if($o(e)&&(r=Lc(e,"utf8"),/^\.gitnexus\/?\s*$/m.test(r)))return;let s=`${r.length>0&&!r.endsWith(`
|
|
38
38
|
`)?`
|
|
39
39
|
`:""}
|
|
40
40
|
# GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
|
|
41
41
|
${n}
|
|
42
|
-
`;r.length>0?
|
|
42
|
+
`;r.length>0?Nc(e,s):Gc(e,`# GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
|
|
43
43
|
${n}
|
|
44
|
-
`)}catch{}}import{existsSync as
|
|
44
|
+
`)}catch{}}import{existsSync as qc}from"fs";import{join as Oo}from"path";import{confirm as zc}from"@inquirer/prompts";var Hc=[/^claude-(opus|sonnet)-4/i,/^claude-(opus|sonnet|haiku)-[5-9]/i,/^o1(-|$)/i,/^o3(-|$)/i,/^o4(-|$)/i,/nal-claude-(opus|sonnet)-?4/i];function Ro(t){return t?Hc.some(e=>e.test(t)):!1}k();h();import{promises as Kt}from"fs";import{homedir as Fc}from"os";import{dirname as Kc,join as _o}from"path";var Vc=_o(Fc(),".gitnexus"),Vt=_o(Vc,"config.json");async function Bc(){try{let t=await Kt.readFile(Vt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Wc(t){await Kt.mkdir(Kc(Vt),{recursive:!0});let e=`${Vt}.tmp-${process.pid}`;await Kt.writeFile(e,t,{mode:384}),await Kt.rename(e,Vt)}async function Io(t){let n={...await Bc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Wc(JSON.stringify(n,null,2)),await Kt.chmod(Vt,384).catch(()=>{})}var Jc=900*1e3,Yc="nal-claude",Xc="claude-sonnet-4-5";function Qc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Zc(t){let e=Oo(t,".claude","settings.json");if(!await d(e))return null;try{let n=await v(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!o)return null;let s=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=s.length>0?s:a,l=f=>{let w=process.env[f];if(typeof w=="string"&&w.length>0)return w;let C=r[f];return typeof C=="string"&&C.length>0?C:null},u=typeof n.avatarProvider=="string"?n.avatarProvider:null,m=!1;try{m=new URL(o).hostname==="api.anthropic.com"}catch{m=!1}if((u==="anthropic"?"anthropic":u==="llmlite"?"llmlite":m?"anthropic":"llmlite")==="anthropic"){let f=l("ANTHROPIC_API_KEY");if(f)return{provider:"anthropic",apiKey:f,baseUrl:Qc(o),model:c.length>0?c:Xc}}else{let f=l("ANTHROPIC_AUTH_TOKEN");if(f)return{provider:"llmlite",apiKey:f,baseUrl:o,model:c.length>0?c:Yc}}return null}catch{return null}}async function tl(t,e){return await zc({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function Mo(t,e){return t.split(`
|
|
45
45
|
`).slice(-e).join(`
|
|
46
|
-
`)}async function
|
|
46
|
+
`)}async function xe(t,e=t){let n=await Zc(t);if(!n)return i.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
|
|
47
47
|
\u0110\xE3 ki\u1EC3m tra: process.env \u2192 .envrc Avatar block \u2192 settings.json.env
|
|
48
48
|
Nguy\xEAn nh\xE2n c\xF3 th\u1EC3:
|
|
49
49
|
\u2022 Subscription mode (OAuth, kh\xF4ng c\xF3 key)
|
|
50
50
|
\u2022 Key ch\u01B0a setup \u2014 ch\u1EA1y 'avatar ai setup' ho\u1EB7c 'avatar secrets set ANTHROPIC_API_KEY'
|
|
51
|
-
\u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),
|
|
52
|
-
gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await
|
|
53
|
-
`):g&&process.stderr.write(`${
|
|
54
|
-
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=
|
|
55
|
-
C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function
|
|
56
|
-
Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${Mo/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 rl(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 Zc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Lo(){let t=await el(),e=await nl();s.info(`Verify key (${X(t)}) qua ${e}/v1/models...`);let n=await Gn(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await rl(n);return{apiKey:t,baseUrl:e,model:r,availableModels:n}}h();var il=".claude/state/ai-orchestrator-model-map.json",sl=["opus","sonnet","haiku"];function Do(t){return Dn(t,il)}async function al(t){let e=Do(t);if(!await d(e))return null;try{let r=(await b(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 Go(t){let e=Dn(t,".claude","settings.json");if(!await d(e))return{provider:"subscription",baseUrl:null,token:null};try{let n=await b(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=m=>{let g=process.env[m];if(typeof g=="string"&&g.length>0)return g;let f=r[m];return typeof f=="string"&&f.length>0?f: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 jn(t){let e=t.filter(n=>n.toLowerCase().includes("claude"));return e.length>0?e:t}async function cl(t){if(t.provider==="subscription"||!t.baseUrl||!t.token)return{models:[...sl],source:"alias"};let e=await Gn(t.baseUrl,t.token);return{models:jn(e),source:"/v1/models"}}async function Un(t,e){return await ol({message:`Tier ${t} \u2192 ch\u1ECDn model:`,choices:e.map(n=>({name:n,value:n}))})}async function ll(t,e,n,r){let o={map:e,provider:n,source:r,updatedAt:Math.floor(Date.now()/1e3)};await T(Do(t),o)}async function Uo(t,e,n){if(n==="subscription")return;let r=No(e),o=Dn(t,".claude","settings.json");try{if(!await d(o))return;let i=await b(o);i.env={...i.env||{},...r},await T(o,i),s.dim("ai-orchestrator: set env slot tier (OPUS=hard, SONNET=standard, HAIKU=fast) \u2014 alias spawn expand \u0111\xFAng model \u0111\xE3 map.")}catch(i){s.warn(`ai-orchestrator: kh\xF4ng ghi \u0111\u01B0\u1EE3c tier-slot env (${i.message}) \u2014 route c\xF3 th\u1EC3 ch\u1EA1y sai model. B\u1ECF qua, tool v\u1EABn b\u1EADt.`)}}async function jo(t,e={}){if(!e.forceRefresh){let a=await al(t);if(a){s.dim("ai-orchestrator: state map \u0111\xE3 t\u1ED3n t\u1EA1i \u2014 gi\u1EEF map c\u0169 (kh\xF4ng h\u1ECFi l\u1EA1i).");let c=await Go(t);return await Uo(t,a,c.provider),!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 Go(t),r,o;try{let a=await cl(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 Un("Vi\u1EC7c kh\xF3 (model m\u1EA1nh)",r),c=await Un("Vi\u1EC7c v\u1EEBa (model c\xE2n b\u1EB1ng)",r),l=await Un("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 ll(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 await Uo(t,i,n.provider),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}h();import{spawnSync as xl}from"child_process";k();import{promises as fl}from"fs";import{join as Bo}from"path";k();h();import{promises as ul}from"fs";import Ho,{join as ve}from"path";function Hn(t,e){let n=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!n?.[2])return{safe:!0};let r=n[2];if(r.includes("${CLAUDE_PROJECT_DIR}"))return r.includes("..")?{safe:!1,reason:`path traversal trong "${r}"`}:{safe:!0};if(r.startsWith("/"))return{safe:!1,reason:`absolute path "${r}" (ch\u1EC9 accept relative t\u1EDBi workspace)`};let o=Ho.resolve(t),i=Ho.resolve(ve(t,r));return!i.startsWith(`${o}/`)&&i!==o?{safe:!1,reason:`path "${r}" resolve ra ngo\xE0i workspace (traversal)`}:{safe:!0}}async function pl(t,e){let n=Hn(t,e);if(!n.safe)return s.warn(`Pack statusLine reject: ${n.reason}.`),!1;let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);return!r?.[2]||r[2].includes("${CLAUDE_PROJECT_DIR}")?!0:await d(ve(t,r[2]))}function Fn(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 Bt(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 ml(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function dl(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 Fo(t){let e=t.map((i,a)=>{let c=dl(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(ml).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 gl(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=Fo(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=Bt(c,a),{entries:u,droppedCount:m}=Fo(l);m>0&&(o+=m),u.length!==c.length&&(n.includes(i)||n.push(i)),r[i]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function xe(t){let e=ve(t,".claude","pack","templates","settings.json.tpl"),n=ve(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await G(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 d(n)){i=!0;try{o=await b(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 pl(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||{}},m=!1,g=!1;for(let[f,w]of Object.entries(u))typeof w=="string"&&w.includes("llm.nal.vn")&&(u[f]=w.replace(/llm\.nal\.vn/g,"ai.nal.vn"),m=!0);if(m&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[f,w]of Object.entries(r.env))f in u||(u[f]=w,g=!0);g&&a.push("env vars added from pack"),(m||g)&&(c.env=u)}if(r.permissions){let u=o.permissions?.allow||[],m=o.permissions?.deny||[],g=r.permissions.allow||[],f=r.permissions.deny||[],w=Bt(u,g),S=Bt(m,f);(w.length!==u.length||S.length!==m.length)&&(c.permissions={allow:w,deny:S},a.push(`permissions union (+${w.length-u.length} allow, +${S.length-m.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:m,touchedEvents:g,migratedCount:f}=gl(r.hooks,u);(g.length>0||f>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),f>0&&a.push(`migrated ${f} 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=Fn(n),await ul.copyFile(n,l)),await T(n,c),{action:"merged",backupPath:l,changes:a}}function hl(t,e){return Bo(t,".claude","pack","tools",e,"tool.json")}async function Wt(t,e){let n=hl(t,e);if(!await d(n))return null;try{return await b(n)}catch{return null}}var wl=[".claude","settings.json"];function Wo(t){return Bo(t,...wl)}function Ae(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function qo(t){let e=Wo(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await b(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 zo(t,e,n){let r=Wo(t),o;return n&&(o=Fn(r),await fl.copyFile(r,o)),await T(r,e),o}function Ko(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Vo(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function kl(t,e){let n=new Set;for(let i of e)Vo(i)&&n.add(Ko(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>Vo(i)?!0:n.has(Ko(i))?(r++,!1):!0),droppedCount:r}}async function Se(t,e){let{settings:n,existed:r}=await qo(t),o={...n},i=[];for(let[u,m]of Object.entries(e.settings.hooks??{}))for(let g of m)for(let f of g.hooks??[]){if(!f.command)continue;let w=Hn(t,f.command);if(!w.safe)throw new Error(`Tool "${e.name}" hook (${u}) b\u1ECB t\u1EEB ch\u1ED1i: ${w.reason}. Hook pack PH\u1EA2I d\xF9ng path t\u01B0\u01A1ng \u0111\u1ED1i t\u1EDBi workspace (vd \${CLAUDE_PROJECT_DIR}/.claude/...).`)}let a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],f=0;for(let[w,S]of Object.entries(a)){let E=u[w]??[],{migratedUser:L,droppedCount:v}=kl(E,S);v>0&&(f+=v);let x=new Set(L.map(Ae)),R=S.filter(it=>!x.has(Ae(it)));(R.length>0||v>0)&&(m[w]=[...L,...R],g.push(w))}g.length>0&&(o.hooks=m,i.push(`hooks added: ${g.join(", ")}`),f>0&&i.push(`migrated ${f} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],m=Bt(u,c);m.length!==u.length&&(o.permissions={...n.permissions,deny:m},i.push(`deny +${m.length-u.length}`))}return i.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await zo(t,o,r),changes:i}}async function Jo(t,e){let{settings:n,existed:r}=await qo(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:i,...a}=n,c={...a},l=[],u=e.settings.hooks??{},m=n.hooks??{},g={},f=[];for(let[v,x]of Object.entries(m)){let R=u[v];if(!R){g[v]=x;continue}let it=new Set(R.map(Ae)),yt=x.filter(st=>!it.has(Ae(st)));yt.length!==x.length&&f.push(v),yt.length>0&&(g[v]=yt)}Object.keys(g).length>0&&(c.hooks=g),f.length>0&&l.push(`hooks removed: ${f.join(", ")}`);let w=new Set(e.settings.permissions?.deny??[]),S=n.permissions?.deny??[],E=w.size>0?S.filter(v=>!w.has(v)):S;if(n.permissions){let{deny:v,...x}=n.permissions,R={...x,...E.length>0?{deny:E}:{}};Object.keys(R).length>0&&(c.permissions=R)}return E.length!==S.length&&l.push(`deny -${S.length-E.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await zo(t,c,r),changes:l}}h();k();import{join as yl}from"path";var bl=".claude/avatar-tools.json";function Yo(){return{tools:{}}}function Xo(t){return yl(t,bl)}async function Rt(t){let e=Xo(t);if(!await d(e))return Yo();try{return{tools:(await b(e)).tools??{}}}catch{return Yo()}}async function vl(t,e){await T(Xo(t),e)}async function Ce(t,e,n){let r=await Rt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await vl(t,r),r}async function _t(t){let e=await Rt(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Al(t){if(!t)return;let e=xl(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 Qo(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 V(t,e,n={}){let r=await Wt(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(Al(r.requires?.runtime),e==="ai-orchestrator"&&!await jo(t,{forceRefresh:n.forceRefresh}))return!1;let o=await Se(t,r);return Qo(e,o),await Ce(t,e,{enabled:!0,version:r.version}),!0}async function Kn(t,e){let n=await Wt(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 Rt(t)).tools[e]?.version??"unknown";return await Ce(t,e,{enabled:!1,version:i}),!1}let r=await Jo(t,n);return Qo(e,r),await Ce(t,e,{enabled:!1,version:n.version}),!0}var Sl="ai-orchestrator";async function Te(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=jn(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: c\u1EA7n ch\u1ECDn model interactive \u2014 b\u1ECF qua \u1EDF ch\u1EBF \u0111\u1ED9 non-interactive. B\u1EADt sau b\u1EB1ng 'avatar tools enable ai-orchestrator'.");return}s.info("AI Orchestrator: key c\xF3 >1 model \u2192 b\u1EADt (ch\u1ECDn model cho m\u1ED7i lo\u1EA1i vi\u1EC7c). G\u1EE1 sau b\u1EB1ng 'avatar tools disable ai-orchestrator'."),await V(t,Sl,{forceRefresh:n.forceRefresh})}import{promises as oi}from"fs";import{homedir as Il}from"os";import{join as Yt}from"path";import{z as y}from"zod";var Zo=y.object({email:y.email(),name:y.string(),access_token:y.string().min(1),refresh_token:y.string().min(1),expires_at:y.iso.datetime(),id_token:y.string().min(1)}),Cl=y.object({installed_tools:y.record(y.string(),y.object({version:y.string().optional(),installed_at:y.iso.datetime(),install_method:y.string()})).default({}),tool_inputs:y.record(y.string(),y.unknown()).default({})}),ch=y.object({$schema:y.string().optional(),includeCoAuthoredBy:y.boolean().optional(),env:y.record(y.string(),y.string()).default({}),permissions:y.object({allow:y.array(y.string()).default([]),deny:y.array(y.string()).default([])}).partial().optional(),hooks:y.record(y.string(),y.array(y.unknown())).optional(),statusLine:y.object({type:y.string(),command:y.string(),padding:y.number().optional()}).optional()}),lh=y.enum(["internal","client","library"]);k();var Xt=Yt(Il(),".avatar"),mt=Yt(Xt,"config.json"),fh=Yt(Xt,"state.json"),Xn=Yt(Xt,"audit.log"),hh=Yt(Xt,"backups"),Ol=384;async function Qn(){await C(Xt)}async function H(){if(!await d(mt))return null;let t=await b(mt),e=Zo.safeParse(t);return e.success?e.data:null}async function Zn(t){await Qn(),await T(mt,t,Ol)}async function Ee(){if(await d(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 Jt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function Nl(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(zt(),Yn));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function Re(){let t=await H();if(!t)throw new Jt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(zt(),Yn));if(!await Nl(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 Jt(`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 Jt("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 Zn(i),o.id_token}async function Ml(){try{await oi.chmod(Xn,384)}catch{}}async function A(t,e){await Qn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
|
|
57
|
-
`;await
|
|
58
|
-
${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 ci}from"child_process";var Ul=5e3,Dl=/(\d+\.\d+\.\d+)/;function jl(){let e=U()==="win32"?"where":"which",n=ci(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=ci("claude",["--version"],{encoding:"utf8",timeout:Ul});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Dl.exec(e)?.[1]??null}var gt=null;function Qt(){if(gt!==null)return gt;let t=jl();return t?(gt={installed:!0,version:Hl(),path:t},gt):(gt={installed:!1,version:null,path:null},gt)}function _e(){gt=null}import{spawnSync as Fl}from"child_process";h();var li=300*1e3,ui="@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 ${ui} 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 pi(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Fl("npm",["install","-g",ui],{stdio:["inherit","inherit","pipe"],timeout:li,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ft("timeout",`npm install timeout sau ${li/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||"");_e();let e=Qt();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 Vl}from"fs";import{homedir as Bl}from"os";import{join as Wl}from"path";import{select as mi}from"@inquirer/prompts";function ql(){return Wl(Bl(),".claude","settings.json")}function rr(){let t=ql(),e;try{e=Vl(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 di(t=rr()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await mi({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 mi({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"}]})}h();import{password as zl,select as Jl}from"@inquirer/prompts";var Ie="https://api.anthropic.com",Yl="2023-06-01",gi=1e4;function Xl(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Ql(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 Zl(){return await zl({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Ql})}async function tu(t){let e=new AbortController,n=setTimeout(()=>e.abort(),gi);try{let r=await fetch(`${Ie}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Yl,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 ${Ie} timeout sau ${gi/1e3}s.`):r}finally{clearTimeout(n)}}async function eu(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 Jl({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function fi(){let t=await Zl();s.info(`Verify key (${Xl(t)}) qua ${Ie}/v1/models...`);let e=await tu(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await eu(e);return{apiKey:t,baseUrl:Ie,model:n,availableModels:e}}h();k();import{promises as nu}from"fs";import{join as ru}from"path";var or=384;function ou(t){return ru(t,".claude","settings.json")}async function iu(t){if(!await d(t))return{};try{return await b(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 su(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 hi(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function au(t,e,n,r,o){let a={...hi(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function cu(t,e,n,r,o){let a={...hi(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function lu(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=ou(t),r=await iu(n),o;switch(e.provider){case"subscription":o=su(r,e.model);break;case"llmlite":o=au(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=cu(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=lu(r,e.sourceSettings);break}await T(n,o,or);try{await nu.chmod(n,or)}catch{}return{path:n,mode:or}}var B="sonnet";async function Oe(t){try{s.info("Setup AI provider cho workspace...");let e=Qt();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."),pi(),_e(),e=Qt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=rr();switch(await di(n)){case"subscription":{let o=tr();if(o.state!=="authenticated"&&(er(),o=tr()),o.state==="authenticated"&&o.subscriptionType)return await ht(t.workspacePath,{provider:"subscription",model:B}),await A("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${B}`),{ok:!0,provider:"subscription",model:B};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=await nr();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),er(),i=await nr()),!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:B}),await A("ai_setup",`provider=subscription,result=ok,probe=${i.reason}-soft-pass`),s.success(`AI ready \xB7 Subscription (probe ${i.reason}, soft-pass) \xB7 model=${B}`),{ok:!0,provider:"subscription",model:B};if(!i.ok){let a=i.reason??"unknown";return await A("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 ${ai(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await ht(t.workspacePath,{provider:"subscription",model:B}),await A("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${B}`),{ok:!0,provider:"subscription",model:B}}case"llmlite":{let o=await Lo();return await ht(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await A("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 fi();return await ht(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await A("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 A("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 A("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}h();h();var ir=1e4,wi=3e4,yi=5,sr="say ok",ki="2023-06-01";async function uu(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${X(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),ir);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(f=>typeof f.id=="string"?f.id:null).filter(f=>f!==null);if(s.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let f=c.slice(0,5).join(", "),w=c.length>5?` ...+${c.length-5} more`:"";s.dim(` Models: ${f}${w}`)}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:sr}],max_tokens:yi}),signal:r.signal});if(!l.ok){let f=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${f}`)}let u=await l.json(),m=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";s.success(`Response: "${String(m).trim().slice(0,100)}"`),s.dim(` Tokens used: ${g}`)}catch(i){throw i.name==="AbortError"?new Error(`Timeout ${ir/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}async function pu(){s.info("Testing Subscription provider qua `claude --print`...");let t=await M("claude",["--print",sr],{timeoutMs:wi});if(t.timedOut)throw new Error(`Timeout ${wi/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 mu(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${X(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),ir);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":ki},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":ki,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:yi,messages:[{role:"user",content:sr}]}),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 m=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);s.success(`Response: "${m}"`)}finally{clearTimeout(o)}}async function bi(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 mu(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await uu(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(await pu(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Ne(){let t=process.cwd(),e=Pt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
51
|
+
\u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),i.dim(`\u0110\u1EC3 gen wiki sau khi \u0111\xE3 c\xF3 key, ch\u1EA1y manual:
|
|
52
|
+
gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await tl(n.baseUrl,n.model))return i.dim("User decline wiki gen \u2014 workspace OK kh\xF4ng c\xF3 wiki. Ch\u1EA1y `gitnexus wiki` manual sau khi c\u1EA7n."),{ran:!1,skipped:!0,reason:"user-declined"};let o=Ro(n.model);await Io({apiKey:n.apiKey,baseUrl:n.baseUrl,model:n.model,isReasoningModel:o});let s=["wiki",".","--base-url",n.baseUrl,"--model",n.model];o&&s.push("--reasoning-model");let a=Dt(`Generating wiki via ${n.baseUrl} (${n.provider}) model=${n.model}${o?" [reasoning]":""}`),c=await E("gitnexus",s,{cwd:e,timeoutMs:Jc});if(c.status!==0||c.timedOut){let u=c.timedOut?"timeout":"non-zero-exit";a.fail(`Wiki gen ${u} (exit ${c.status??"null"})`);let m=(c.stderr||"").trim(),g=(c.stdout||"").trim();return m?process.stderr.write(`${Mo(m,30)}
|
|
53
|
+
`):g&&process.stderr.write(`${Mo(g,30)}
|
|
54
|
+
`),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=Oo(e,".gitnexus","wiki","index.html");return qc(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}`})}h();function Go(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 el(o)}catch(s){i.error(s instanceof Error?s.message:String(s)),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,s)=>{try{await nl(o,s)}catch(a){i.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 rl()}catch(o){i.error(o instanceof Error?o.message:String(o)),process.exit(1)}})}function Uo(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 jn(){let t=$t(process.cwd());return t||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
55
|
+
C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function el(t){let e=jn();await To();let n=!0,r=t.url,o=t.name;for(;n;){let s=await ol(r),a=null,c=null;if(s.mode==="new"){let l=o??await Lo(s.url),u=await Sn({workspaceRoot:e,url:s.url,name:l});a=u.path,c=u.name,i.success(`\u2713 T\u1EA1o src/${u.name} (repo m\u1EDBi)`)}else{let l=o??await Lo(s.url);i.info(`Clone ${s.url} \u2192 src/${l} ...`);let{cloned:u,skipped:m}=await so({workspaceRoot:e,url:s.url,name:l});m||!u?i.dim(`B\u1ECF qua repo ${l}.`):(a=u.path,c=u.name,i.success(`\u2713 \u0110\xE3 clone src/${u.name}`))}if(a&&c&&(await sl(e,a),i.success(`\u2713 Done repo: ${c}`)),r=void 0,o=void 0,t.yes)break;n=await Ae({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function nl(t,e){let n=jn(),{validateRepoName:r}=await Promise.resolve().then(()=>(Ut(),Pn)),o=r(t);o&&(i.error(`T\xEAn repo kh\xF4ng h\u1EE3p l\u1EC7: ${o}`),process.exit(1));let{readReposManifest:s,removeRepoFromManifest:a}=await Promise.resolve().then(()=>(Y(),at)),{removeRecursive:c}=await Promise.resolve().then(()=>(k(),yt)),{join:l,relative:u,sep:m}=await import("path"),f=(await s(n)).find(N=>N.name===t),w=l(n,"src",t),C=l(n,"src"),R=u(C,w);(R===""||R===".."||R.startsWith(`..${m}`)||R.startsWith("/"))&&(i.error(`\u0110\u01B0\u1EDDng d\u1EABn repo escape kh\u1ECFi workspace/src \u2014 t\u1EEB ch\u1ED1i x\xF3a: ${w}`),process.exit(1));let{pathExists:U}=await Promise.resolve().then(()=>(k(),yt)),A=await U(w);if(!f&&!A){i.warn(`Repo "${t}" kh\xF4ng c\xF3 trong workspace (c\u1EA3 manifest l\u1EABn src/).`);return}await a(n,t),i.success(`\u2713 G\u1EE1 "${t}" kh\u1ECFi repos.json`),A&&!e.keepFiles?(i.warn(`\u26A0 Folder src/${t} ch\u1EE9a code. X\xF3a = m\u1EA5t data n\u1EBFu ch\u01B0a push remote.`),await Ae({message:`X\xF3a lu\xF4n folder src/${t}?`,default:!1})?(await c(w),i.success(`\u2713 \u0110\xE3 x\xF3a src/${t}`)):i.dim(`Gi\u1EEF folder src/${t}. (\u0110\xE3 g\u1EE1 kh\u1ECFi manifest \u2014 ch\u1EA1y 'avatar add repo' \u0111\u1EC3 add l\u1EA1i.)`)):A&&e.keepFiles&&i.dim(`Gi\u1EEF folder src/${t} (--keep-files).`)}async function rl(){let t=jn(),{readReposManifest:e}=await Promise.resolve().then(()=>(Y(),at)),{pathExists:n}=await Promise.resolve().then(()=>(k(),yt)),{join:r}=await import("path"),o=await e(t);if(o.length===0){i.dim("Ch\u01B0a c\xF3 repo n\xE0o. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}i.info(`Repo trong workspace (${o.length}):`);for(let s of o){let c=await n(r(t,"src",s.name))?"\u2713":"\u26A0 folder thi\u1EBFu";i.plain(` ${c} ${s.name.padEnd(28)} ${s.url}`)}}async function ol(t){if(t)return{url:t,mode:"clone"};let e=await Dn({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 Bt({message:"URL git repo:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),mode:"clone"};if(e==="folder"){let{statSync:s}=await import("fs"),a=No(await Bt({message:"\u0110\u01B0\u1EDDng d\u1EABn folder:",validate:u=>{let m=u.trim();if(m.length===0)return"Path b\u1EAFt bu\u1ED9c";try{return s(No(m)).isDirectory()?!0:`"${m}" kh\xF4ng ph\u1EA3i th\u01B0 m\u1EE5c (l\xE0 file?).`}catch{return`Th\u01B0 m\u1EE5c "${m}" kh\xF4ng t\u1ED3n t\u1EA1i. Ki\u1EC3m tra l\u1EA1i \u0111\u01B0\u1EDDng d\u1EABn.`}}})),c=await Tn(a);if(c&&(i.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${c.slice(0,120)}`),i.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 Ae({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 $n(a);return l?{url:l,mode:"clone"}:{url:await il(a),mode:"clone"}}let n=await Bt({message:"T\xEAn repo m\u1EDBi:",validate:Uo}),r=await Dn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]});return{url:await co(n.trim(),r),mode:"new"}}async function il(t){let{basename:e}=await import("path");if(i.warn(`Folder ${t} ch\u01B0a c\xF3 remote origin.`),!await Ae({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(()=>(dn(),Lr)),{pathExists:o}=await Promise.resolve().then(()=>(k(),yt)),s=r(t);await o(`${t}/.git`)||(i.info("Folder ch\u01B0a git init \u2014 \u0111ang init..."),await s.init()),i.info("Commit code hi\u1EC7n t\u1EA1i trong folder (baseline cho push)..."),await s.add("."),await s.commit("chore: initial commit (avatar add repo)").catch(()=>{});let a=await Bt({message:"T\xEAn repo GitHub:",default:e(t),validate:Uo}),c=await Dn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createGithubRemoteFromFolder:l}=await Promise.resolve().then(()=>(Ln(),yo)),u=await lo(a.trim(),m=>l({folder:t,name:m,visibility:c}).sshUrl);return i.success(`\u2713 \u0110\xE3 t\u1EA1o remote + push: ${u}`),u}async function Lo(t){let e=An(t);return await Bt({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function sl(t,e){if(!pt().installed){i.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}try{await Pt(e),i.success(` \u2713 GitNexus indexed src/${e.split("/").pop()}`)}catch(r){i.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${r instanceof Error?r.message:r}`);return}try{let r=await xe(t,e);r.ran?i.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):r.reason==="subscription-mode"&&i.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(r){i.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${r instanceof Error?r.message:r}`)}}k();import{promises as Au}from"fs";import{join as Ti}from"path";import{confirm as Cu}from"@inquirer/prompts";k();import{join as Kn}from"path";import{select as gl}from"@inquirer/prompts";function Do(t){return{ANTHROPIC_DEFAULT_OPUS_MODEL:t.hard,ANTHROPIC_DEFAULT_SONNET_MODEL:t.standard,ANTHROPIC_DEFAULT_HAIKU_MODEL:t.fast}}h();import{input as al,password as cl,select as ll}from"@inquirer/prompts";var ul="https://ai.nal.vn",jo=1e4;function Z(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function pl(){return await cl({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function ml(t=ul){return(await al({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 Hn(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),jo);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`.":`
|
|
56
|
+
Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${jo/1e3}s.${c}`)}let s=o.message||String(o);if(s.toLowerCase().includes("fetch failed")||s.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}: ${s}${c}`)}throw o}finally{clearTimeout(r)}}async function dl(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return i.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await ll({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Ho(){let t=await pl(),e=await ml();i.info(`Verify key (${Z(t)}) qua ${e}/v1/models...`);let n=await Hn(e,t);i.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await dl(n);return{apiKey:t,baseUrl:e,model:r,availableModels:n}}h();var fl=".claude/state/ai-orchestrator-model-map.json",hl=["opus","sonnet","haiku"];function Vo(t){return Kn(t,fl)}async function wl(t){let e=Vo(t);if(!await d(e))return null;try{let r=(await v(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 Fo(t){let e=Kn(t,".claude","settings.json");if(!await d(e))return{provider:"subscription",baseUrl:null,token:null};try{let n=await v(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null,s=typeof n.avatarProvider=="string"?n.avatarProvider:null;if(!o)return{provider:"subscription",baseUrl:null,token:null};let a=m=>{let g=process.env[m];if(typeof g=="string"&&g.length>0)return g;let f=r[m];return typeof f=="string"&&f.length>0?f:null},c=!1;try{c=new URL(o).hostname==="api.anthropic.com"}catch{c=!1}let l=s==="anthropic"?"anthropic":s==="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 Vn(t){let e=t.filter(n=>n.toLowerCase().includes("claude"));return e.length>0?e:t}async function kl(t){if(t.provider==="subscription"||!t.baseUrl||!t.token)return{models:[...hl],source:"alias"};let e=await Hn(t.baseUrl,t.token);return{models:Vn(e),source:"/v1/models"}}async function Fn(t,e){return await gl({message:`Tier ${t} \u2192 ch\u1ECDn model:`,choices:e.map(n=>({name:n,value:n}))})}async function yl(t,e,n,r){let o={map:e,provider:n,source:r,updatedAt:Math.floor(Date.now()/1e3)};await T(Vo(t),o)}async function Ko(t,e,n){if(n==="subscription")return;let r=Do(e),o=Kn(t,".claude","settings.json");try{if(!await d(o))return;let s=await v(o);s.env={...s.env||{},...r},await T(o,s),i.dim("ai-orchestrator: set env slot tier (OPUS=hard, SONNET=standard, HAIKU=fast) \u2014 alias spawn expand \u0111\xFAng model \u0111\xE3 map.")}catch(s){i.warn(`ai-orchestrator: kh\xF4ng ghi \u0111\u01B0\u1EE3c tier-slot env (${s.message}) \u2014 route c\xF3 th\u1EC3 ch\u1EA1y sai model. B\u1ECF qua, tool v\u1EABn b\u1EADt.`)}}async function Bo(t,e={}){if(!e.forceRefresh){let a=await wl(t);if(a){i.dim("ai-orchestrator: state map \u0111\xE3 t\u1ED3n t\u1EA1i \u2014 gi\u1EEF map c\u0169 (kh\xF4ng h\u1ECFi l\u1EA1i).");let c=await Fo(t);return await Ko(t,a,c.provider),!0}}if(!process.stdout.isTTY)return i.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 Fo(t),r,o;try{let a=await kl(n);r=a.models,o=a.source}catch(a){return i.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 i.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;i.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 s;try{let a=await Fn("Vi\u1EC7c kh\xF3 (model m\u1EA1nh)",r),c=await Fn("Vi\u1EC7c v\u1EEBa (model c\xE2n b\u1EB1ng)",r),l=await Fn("Vi\u1EC7c d\u1EC5 (model r\u1EBB)",r);s={hard:a,standard:c,fast:l}}catch{return i.dim("ai-orchestrator: h\u1EE7y ch\u1ECDn model \u2014 kh\xF4ng b\u1EADt tool."),!1}try{await yl(t,s,n.provider,o)}catch(a){return i.error(`ai-orchestrator: ghi state map th\u1EA5t b\u1EA1i (${a.message}). Kh\xF4ng b\u1EADt tool.`),!1}return await Ko(t,s,n.provider),i.success(`ai-orchestrator: \u0111\xE3 map tier \u2192 model. Kh\xF3=${s.hard} \xB7 V\u1EEBa=${s.standard} \xB7 D\u1EC5=${s.fast}`),!0}h();import{spawnSync as Il}from"child_process";k();import{promises as Sl}from"fs";import{join as Yo}from"path";k();h();import{promises as bl}from"fs";import Wo,{join as Ce}from"path";function Bn(t,e){let n=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!n?.[2])return{safe:!0};let r=n[2];if(r.includes("${CLAUDE_PROJECT_DIR}"))return r.includes("..")?{safe:!1,reason:`path traversal trong "${r}"`}:{safe:!0};if(r.startsWith("/"))return{safe:!1,reason:`absolute path "${r}" (ch\u1EC9 accept relative t\u1EDBi workspace)`};let o=Wo.resolve(t),s=Wo.resolve(Ce(t,r));return!s.startsWith(`${o}/`)&&s!==o?{safe:!1,reason:`path "${r}" resolve ra ngo\xE0i workspace (traversal)`}:{safe:!0}}async function vl(t,e){let n=Bn(t,e);if(!n.safe)return i.warn(`Pack statusLine reject: ${n.reason}.`),!1;let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);return!r?.[2]||r[2].includes("${CLAUDE_PROJECT_DIR}")?!0:await d(Ce(t,r[2]))}function Wn(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 Wt(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let s=typeof o=="string"?o:JSON.stringify(o);n.has(s)||(n.add(s),r.push(o))}return r}function xl(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Al(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 s=o.command;typeof s=="string"&&r.push(s)}return r}function qo(t){let e=t.map((s,a)=>{let c=Al(s),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:s,commands:c,hasVar:l}}),n=new Map;for(let s of e){let a=s.commands.map(xl).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(s),n.set(a,c)}let r=new Set;for(let[,s]of n)if(!(s.length<2||!s.some(c=>c.hasVar)))for(let c of s)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((s,a)=>!r.has(a)),droppedCount:r.size}}function Cl(t,e){let n=[],r={...e},o=0;for(let s of Object.keys(r)){let a=r[s]||[],{entries:c,droppedCount:l}=qo(a);l>0&&(r[s]=c,o+=l,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=r[s]||[],l=Wt(c,a),{entries:u,droppedCount:m}=qo(l);m>0&&(o+=m),u.length!==c.length&&(n.includes(s)||n.push(s)),r[s]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function Se(t){let e=Ce(t,".claude","pack","templates","settings.json.tpl"),n=Ce(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await D(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={},s=!1;if(await d(n)){s=!0;try{o=await v(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 vl(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||{}},m=!1,g=!1;for(let[f,w]of Object.entries(u))typeof w=="string"&&w.includes("llm.nal.vn")&&(u[f]=w.replace(/llm\.nal\.vn/g,"ai.nal.vn"),m=!0);if(m&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[f,w]of Object.entries(r.env))f in u||(u[f]=w,g=!0);g&&a.push("env vars added from pack"),(m||g)&&(c.env=u)}if(r.permissions){let u=o.permissions?.allow||[],m=o.permissions?.deny||[],g=r.permissions.allow||[],f=r.permissions.deny||[],w=Wt(u,g),C=Wt(m,f);(w.length!==u.length||C.length!==m.length)&&(c.permissions={allow:w,deny:C},a.push(`permissions union (+${w.length-u.length} allow, +${C.length-m.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:m,touchedEvents:g,migratedCount:f}=Cl(r.hooks,u);(g.length>0||f>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),f>0&&a.push(`migrated ${f} 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 s&&(l=Wn(n),await bl.copyFile(n,l)),await T(n,c),{action:"merged",backupPath:l,changes:a}}function Tl(t,e){return Yo(t,".claude","pack","tools",e,"tool.json")}async function qt(t,e){let n=Tl(t,e);if(!await d(n))return null;try{return await v(n)}catch{return null}}var $l=[".claude","settings.json"];function Xo(t){return Yo(t,...$l)}function Te(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Qo(t){let e=Xo(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await v(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 Zo(t,e,n){let r=Xo(t),o;return n&&(o=Wn(r),await Sl.copyFile(r,o)),await T(r,e),o}function zo(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Jo(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Pl(t,e){let n=new Set;for(let s of e)Jo(s)&&n.add(zo(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(s=>Jo(s)?!0:n.has(zo(s))?(r++,!1):!0),droppedCount:r}}async function $e(t,e){let{settings:n,existed:r}=await Qo(t),o={...n},s=[];for(let[u,m]of Object.entries(e.settings.hooks??{}))for(let g of m)for(let f of g.hooks??[]){if(!f.command)continue;let w=Bn(t,f.command);if(!w.safe)throw new Error(`Tool "${e.name}" hook (${u}) b\u1ECB t\u1EEB ch\u1ED1i: ${w.reason}. Hook pack PH\u1EA2I d\xF9ng path t\u01B0\u01A1ng \u0111\u1ED1i t\u1EDBi workspace (vd \${CLAUDE_PROJECT_DIR}/.claude/...).`)}let a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],f=0;for(let[w,C]of Object.entries(a)){let R=u[w]??[],{migratedUser:U,droppedCount:A}=Pl(R,C);A>0&&(f+=A);let N=new Set(U.map(Te)),b=C.filter(_=>!N.has(Te(_)));(b.length>0||A>0)&&(m[w]=[...U,...b],g.push(w))}g.length>0&&(o.hooks=m,s.push(`hooks added: ${g.join(", ")}`),f>0&&s.push(`migrated ${f} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],m=Wt(u,c);m.length!==u.length&&(o.permissions={...n.permissions,deny:m},s.push(`deny +${m.length-u.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await Zo(t,o,r),changes:s}}async function ti(t,e){let{settings:n,existed:r}=await Qo(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:s,...a}=n,c={...a},l=[],u=e.settings.hooks??{},m=n.hooks??{},g={},f=[];for(let[A,N]of Object.entries(m)){let b=u[A];if(!b){g[A]=N;continue}let _=new Set(b.map(Te)),G=N.filter(le=>!_.has(Te(le)));G.length!==N.length&&f.push(A),G.length>0&&(g[A]=G)}Object.keys(g).length>0&&(c.hooks=g),f.length>0&&l.push(`hooks removed: ${f.join(", ")}`);let w=new Set(e.settings.permissions?.deny??[]),C=n.permissions?.deny??[],R=w.size>0?C.filter(A=>!w.has(A)):C;if(n.permissions){let{deny:A,...N}=n.permissions,b={...N,...R.length>0?{deny:R}:{}};Object.keys(b).length>0&&(c.permissions=b)}return R.length!==C.length&&l.push(`deny -${C.length-R.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Zo(t,c,r),changes:l}}h();k();import{join as El}from"path";var Rl=".claude/avatar-tools.json";function ei(){return{tools:{}}}function ni(t){return El(t,Rl)}async function Et(t){let e=ni(t);if(!await d(e))return ei();try{return{tools:(await v(e)).tools??{}}}catch{return ei()}}async function _l(t,e){await T(ni(t),e)}async function Pe(t,e,n){let r=await Et(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await _l(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 Ml(t){if(!t)return;let e=Il(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&i.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 ri(t,e){switch(e.action){case"enabled":i.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&i.dim(` Backup: ${e.backupPath}`);break;case"disabled":i.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&i.dim(` Backup: ${e.backupPath}`);break;case"no-change":i.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 W(t,e,n={}){let r=await qt(t,e);if(!r)return n.silent||i.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(Ml(r.requires?.runtime),e==="ai-orchestrator"&&!await Bo(t,{forceRefresh:n.forceRefresh}))return!1;let o=await $e(t,r);return ri(e,o),await Pe(t,e,{enabled:!0,version:r.version}),!0}async function qn(t,e){let n=await qt(t,e);if(!n){i.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 s=(await Et(t)).tools[e]?.version??"unknown";return await Pe(t,e,{enabled:!1,version:s}),!1}let r=await ti(t,n);return ri(e,r),await Pe(t,e,{enabled:!1,version:n.version}),!0}var Ol="ai-orchestrator";async function Ee(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=Vn(e.availableModels??[]).length;else return;if(r<=1){i.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){i.dim("AI Orchestrator: c\u1EA7n ch\u1ECDn model interactive \u2014 b\u1ECF qua \u1EDF ch\u1EBF \u0111\u1ED9 non-interactive. B\u1EADt sau b\u1EB1ng 'avatar tools enable ai-orchestrator'.");return}i.info("AI Orchestrator: key c\xF3 >1 model \u2192 b\u1EADt (ch\u1ECDn model cho m\u1ED7i lo\u1EA1i vi\u1EC7c). G\u1EE1 sau b\u1EB1ng 'avatar tools disable ai-orchestrator'."),await W(t,Ol,{forceRefresh:n.forceRefresh})}import{promises as li}from"fs";import{homedir as Fl}from"os";import{join as Xt}from"path";import{z as y}from"zod";var oi=y.object({email:y.email(),name:y.string(),access_token:y.string().min(1),refresh_token:y.string().min(1),expires_at:y.iso.datetime(),id_token:y.string().min(1)}),Nl=y.object({installed_tools:y.record(y.string(),y.object({version:y.string().optional(),installed_at:y.iso.datetime(),install_method:y.string()})).default({}),tool_inputs:y.record(y.string(),y.unknown()).default({})}),$h=y.object({$schema:y.string().optional(),includeCoAuthoredBy:y.boolean().optional(),env:y.record(y.string(),y.string()).default({}),permissions:y.object({allow:y.array(y.string()).default([]),deny:y.array(y.string()).default([])}).partial().optional(),hooks:y.record(y.string(),y.array(y.unknown())).optional(),statusLine:y.object({type:y.string(),command:y.string(),padding:y.number().optional()}).optional()}),Ph=y.enum(["internal","client","library"]);k();var Qt=Xt(Fl(),".avatar"),mt=Xt(Qt,"config.json"),Oh=Xt(Qt,"state.json"),er=Xt(Qt,"audit.log"),Nh=Xt(Qt,"backups"),Kl=384;async function nr(){await S(Qt)}async function K(){if(!await d(mt))return null;let t=await v(mt),e=oi.safeParse(t);return e.success?e.data:null}async function rr(t){await nr(),await T(mt,t,Kl)}async function Ie(){if(await d(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 Yt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function Vl(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Jt(),tr));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function Me(){let t=await K();if(!t)throw new Yt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Jt(),tr));if(!await Vl(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 Yt(`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 Yt("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 s={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await rr(s),o.id_token}async function Bl(){try{await li.chmod(er,384)}catch{}}async function x(t,e){await nr();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
|
|
57
|
+
`;await li.appendFile(er,r,{encoding:"utf8",mode:384}),await Bl()}import{spawnSync as pi}from"child_process";h();var ui=6e4,Wl="ok";function or(){let t=pi("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 ir(){i.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=pi("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.`);i.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function ql(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 mi(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."}}async function sr(){let t=await E("claude",["--print",Wl],{timeoutMs:ui});if(t.timedOut||t.status===143)return{ok:!1,reason:"timeout",detail:`claude --print > ${ui/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(),s=n.toLowerCase();if(o.length>20&&!s.includes("error")&&!s.includes("limit")&&!s.includes("quota")&&!s.includes("401"))return i.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=ql(`${n}
|
|
58
|
+
${r}`);return a==="unknown"&&(i.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&i.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&i.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as di}from"child_process";var zl=5e3,Jl=/(\d+\.\d+\.\d+)/;function Yl(){let e=j()==="win32"?"where":"which",n=di(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 Xl(){let t=di("claude",["--version"],{encoding:"utf8",timeout:zl});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Jl.exec(e)?.[1]??null}var gt=null;function Zt(){if(gt!==null)return gt;let t=Yl();if(!t)return gt={installed:!1,version:null,path:null},gt;let e=Xl();return gt={installed:!!(t&&e),version:e,path:t},gt}function Oe(){gt=null}import{spawnSync as Ql}from"child_process";h();var gi=300*1e3,fi="@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 Zl(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 ${fi} 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 hi(){i.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Ql("npm",["install","-g",fi],{stdio:["inherit","inherit","pipe"],timeout:gi,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ft("timeout",`npm install timeout sau ${gi/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Zl(t.status,t.stderr||"");Oe();let e=Zt();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 i.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 tu}from"fs";import{homedir as eu}from"os";import{join as nu}from"path";import{select as wi}from"@inquirer/prompts";function ru(){return nu(eu(),".claude","settings.json")}function ar(){let t=ru(),e;try{e=tu(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,s=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:s,model:a,rawSettings:n}}async function ki(t=ar()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await wi({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 wi({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"}]})}h();import{password as ou,select as iu}from"@inquirer/prompts";var Ne="https://api.anthropic.com",su="2023-06-01",yi=1e4;function au(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function cu(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 lu(){return await ou({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:cu})}async function uu(t){let e=new AbortController,n=setTimeout(()=>e.abort(),yi);try{let r=await fetch(`${Ne}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":su,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 s=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(s.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return s}catch(r){throw r.name==="AbortError"?new Error(`Connect ${Ne} timeout sau ${yi/1e3}s.`):r}finally{clearTimeout(n)}}async function pu(t){if(t.length===1){let n=t[0];return i.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=s=>{let a=s.toLowerCase();return a.includes("opus")?0:a.includes("sonnet")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await iu({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function bi(){let t=await lu();i.info(`Verify key (${au(t)}) qua ${Ne}/v1/models...`);let e=await uu(t);i.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await pu(e);return{apiKey:t,baseUrl:Ne,model:n,availableModels:e}}h();k();import{promises as mu}from"fs";import{join as du}from"path";var cr=384;function gu(t){return du(t,".claude","settings.json")}async function fu(t){if(!await d(t))return{};try{return await v(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 hu(t,e){let{env:n,...r}=t,o={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:s,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function vi(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function wu(t,e,n,r,o){let a={...vi(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function ku(t,e,n,r,o){let a={...vi(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function yu(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=gu(t),r=await fu(n),o;switch(e.provider){case"subscription":o=hu(r,e.model);break;case"llmlite":o=wu(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=ku(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=yu(r,e.sourceSettings);break}await T(n,o,cr);try{await mu.chmod(n,cr)}catch{}return{path:n,mode:cr}}var q="opus";async function Le(t){try{i.info("Setup AI provider cho workspace...");let e=Zt();if(e.installed)i.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(i.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),hi(),Oe(),e=Zt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=ar();switch(await ki(n)){case"subscription":{let o=or();if(o.state!=="authenticated"&&(ir(),o=or()),o.state==="authenticated"&&o.subscriptionType)return await ht(t.workspacePath,{provider:"subscription",model:q}),await x("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),i.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${q}`),{ok:!0,provider:"subscription",model:q};i.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=await sr();if(!s.ok&&s.reason==="auth-expired"&&(i.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),ir(),s=await sr()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return i.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&i.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await ht(t.workspacePath,{provider:"subscription",model:q}),await x("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),i.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${q}`),{ok:!0,provider:"subscription",model:q};if(!s.ok){let a=s.reason??"unknown";return await x("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),i.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&i.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),i.warn(` \u2192 ${mi(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await ht(t.workspacePath,{provider:"subscription",model:q}),await x("ai_setup","provider=subscription,result=ok"),i.success(`AI ready \xB7 Subscription \xB7 model=${q}`),{ok:!0,provider:"subscription",model:q}}case"llmlite":{let o=await Ho();return await ht(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await x("ai_setup",`provider=llmlite,result=ok,model=${o.model},base=${o.baseUrl},storage=settings.json`),i.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 bi();return await ht(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await x("ai_setup",`provider=anthropic,result=ok,model=${o.model},storage=settings.json`),i.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 x("ai_setup","provider=use-global,result=ok"),i.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 i.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),i.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await x("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}h();h();var lr=1e4,xi=3e4,Ci=5,ur="say ok",Ai="2023-06-01";async function bu(t,e,n){i.info(`Testing LLMLite provider: ${t} (key: ${Z(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),lr);try{let s=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(f=>typeof f.id=="string"?f.id:null).filter(f=>f!==null);if(i.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let f=c.slice(0,5).join(", "),w=c.length>5?` ...+${c.length-5} more`:"";i.dim(` Models: ${f}${w}`)}i.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:ur}],max_tokens:Ci}),signal:r.signal});if(!l.ok){let f=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${f}`)}let u=await l.json(),m=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";i.success(`Response: "${String(m).trim().slice(0,100)}"`),i.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${lr/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(o)}}async function vu(){i.info("Testing Subscription provider qua `claude --print`...");let t=await E("claude",["--print",ur],{timeoutMs:xi});if(t.timedOut)throw new Error(`Timeout ${xi/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)}`)}i.success(`Response: "${t.stdout.trim().slice(0,100)}"`)}async function xu(t,e,n){i.info(`Testing Anthropic Direct provider: ${t} (key: ${Z(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),lr);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Ai},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);i.success(`Connectivity OK \xB7 ${c.length} models available`),i.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":Ai,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Ci,messages:[{role:"user",content:ur}]}),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 m=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);i.success(`Response: "${m}"`)}finally{clearTimeout(o)}}async function Si(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,s=typeof t.model=="string"?t.model:"default";return o&&n?(await xu(n,o,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await bu(n,r,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(await vu(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Ge(){let t=process.cwd(),e=$t(t);return e||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
59
59
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|
|
60
60
|
B\u1EA1n \u0111ang \u1EDF: ${t}
|
|
61
|
-
Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&
|
|
61
|
+
Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&i.dim(`Detected workspace root: ${e}`),e}async function pr(t){let e=Ti(t,".claude","settings.json");if(!await d(e))return{};try{return await v(e)}catch{return{}}}async function Su(){let t=await Ge(),e=await Le({workspacePath:t});try{await Ee(t,e,{forceRefresh:!0})}catch(n){i.warn(`Setup AI Orchestrator fail: ${n instanceof Error?n.message:n}`)}}async function Tu(){let t=await Ge(),e=await pr(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,s=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;s?(c=r?.includes("api.anthropic.com")||s.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=Z(s)):r&&o?(c="LLMLite",l=Z(o)):o?(c="Custom",l=Z(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),i.info(`Project: ${t}`),i.info(`Provider: ${c}${r?` (${r})`:""}`),i.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),i.info(`Token: ${l}`)}async function $u(){let t=await Ge(),e=await pr(t);try{let n=await Si(e);i.success(`\u2713 ${n.message}`)}catch(n){i.error(`Test fail: ${n.message}`),process.exit(1)}}async function Pu(t){let e=await Ge(),n=Ti(e,".claude","settings.json"),r=await pr(e);if(!t.yes&&!await Cu({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){i.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...s}=r,a={...s};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...m}=o;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await Au.unlink(n).catch(()=>{}),i.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await T(n,a,384),i.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function $i(t){let e=t.command("ai").description("Qu\u1EA3n l\xFD AI provider config (M12)");e.command("setup").description("Wizard setup/re-config AI provider cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await Su()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Tu()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await $u()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Pu(n)})}h();function Pi(t){t.command("debate [file]").description("2-agent ph\u1EA3n bi\u1EC7n plan/doc \u2014 ch\u1EA1y trong Claude Code: /avatar:debate").action(e=>{let n=e??"<file>";process.stdout.write(`${p.cyan("\u{1F4AC}")} ${p.bold("avatar debate")} l\xE0 ${p.bold("slash command")} ch\u1EA1y TRONG Claude Code \u2014 kh\xF4ng ph\u1EA3i l\u1EC7nh CLI.
|
|
62
62
|
(debate c\u1EA7n Task subagent + hook + LLM, Node CLI kh\xF4ng t\u1EF1 ch\u1EA1y \u0111\u01B0\u1EE3c.)
|
|
63
63
|
|
|
64
64
|
C\xE1ch d\xF9ng:
|
|
@@ -69,35 +69,37 @@ ${r}`);return a==="unknown"&&(s.warn(`[debug] claude --print exit=${t.status} si
|
|
|
69
69
|
|
|
70
70
|
2 agent (Skeptic \u21C4 Refiner) ph\u1EA3n bi\u1EC7n \u0111\xF3ng khung trong artifact \u0111\u1EBFn khi h\u1EBFt blocking;
|
|
71
71
|
vi\u1EC7c c\u1EA7n b\u1EA1n quy\u1EBFt \u2192 ${p.bold("NEEDS_DECISION")} (tool kh\xF4ng t\u1EF1 ch\u1ED1t thay b\u1EA1n).
|
|
72
|
-
`),process.exit(0)})}import{spawnSync as
|
|
72
|
+
`),process.exit(0)})}import{spawnSync as fr}from"child_process";import{promises as Hi}from"fs";import{join as ne}from"path";import Ju from"boxen";import{join as Eu}from"path";k();function Ru(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[o,s]of Object.entries(n)){let a=r[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of s)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function Ei(t){let e=await Rt(t);if(e.length===0)return[];let n=Eu(t,".claude","settings.json"),r={};if(await d(n))try{r=await v(n)}catch{r={}}let o=[];for(let s of e){let a=await qt(t,s);if(!a){o.push({name:`Tool: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}Ru(r,a)?o.push({name:`Tool: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${s}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await $e(t,a)}})}return o}import{spawnSync as dr}from"child_process";h();var te=["gitleaks","trivy"],_u=300*1e3,Ri=5e3;function mr(t){let e=j()==="win32",n=dr(e?"where":"which",[t],{encoding:"utf8",timeout:Ri});return!n.error&&n.status===0&&(n.stdout||"").trim().length>0}function Iu(){let t=dr("brew",["--version"],{encoding:"utf8",timeout:Ri});return!t.error&&t.status===0}function Mu(t){i.info(`C\xE0i security scanner qua brew: ${t.join(", ")} (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...`);let e=dr("brew",["install",...t],{stdio:["inherit","inherit","pipe"],timeout:_u,encoding:"utf8"});return e.status!==0?(e.stderr&&process.stderr.write(e.stderr),!1):!0}function Ue(){let t=te.filter(e=>!mr(e));if(t.length===0)return te.map(e=>({name:e,installed:!0}));if(!Iu())return i.warn(`Thi\u1EBFu security scanner (${t.join(", ")}) v\xE0 kh\xF4ng t\xECm th\u1EA5y brew. Security gate s\u1EBD t\u1EF1 skip t\u1EDBi khi c\xE0i. C\xE0i tay: xem trivy.dev / github.com/gitleaks/gitleaks.`),te.map(e=>({name:e,installed:!t.includes(e)}));try{Mu(t)?i.success(`\u0110\xE3 c\xE0i security scanner: ${t.join(", ")}.`):i.warn(`C\xE0i scanner qua brew th\u1EA5t b\u1EA1i (${t.join(", ")}). Security gate skip t\u1EDBi khi c\xE0i. Th\u1EED tay: brew install `+t.join(" "))}catch(e){i.warn(`L\u1ED7i c\xE0i scanner (${e.message}). B\u1ECF qua \u2014 workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c (gate skip).`)}return te.map(e=>({name:e,installed:mr(e)}))}function _i(){return te.map(t=>({name:t,installed:mr(t)}))}k();h();var gr=1e4,Ii=2e4,Ou=18e4;function je(t){if(!t)return null;let e=t.match(/(\d+\.\d+\.\d+)/);return e?e[1]:null}function Nu(t){if(!t)return null;let e=t.split(`
|
|
73
|
+
`).map(n=>n.trim()).filter(Boolean).at(-1);return je(e)}function Lu(t){if(!t)return null;try{let r=JSON.parse(t).formulae?.[0]?.versions?.stable;if(r)return je(r)}catch{}let e=t.match(/stable\s+(\d+\.\d+\.\d+)/i);return e?e[1]:je(t)}function Oi(t){return t.manager==="npm"?{cmd:"npm",args:["install","-g",`${t.pkg}@latest`]}:{cmd:"brew",args:["upgrade",t.pkg]}}async function Gu(t){try{let e=process.platform==="win32"?"where":"which";return(await E(e,[t],{timeoutMs:gr})).status===0}catch{return!1}}async function Uu(t){try{return(await E(t,["--version"],{timeoutMs:gr})).status===0}catch{return!1}}async function De(t){try{let e=await E(t.bin,t.versionArgs,{timeoutMs:gr});return e.status!==0?null:je(`${e.stdout}
|
|
74
|
+
${e.stderr}`)}catch{return null}}async function Du(t){try{if(t.manager==="npm"){let n=await E("npm",["view",`${t.pkg}@latest`,"version"],{timeoutMs:Ii});return n.status===0?Nu(n.stdout):null}let e=await E("brew",["info","--json=v2",t.pkg],{timeoutMs:Ii});return e.status===0?Lu(e.stdout):null}catch{return null}}async function Mi(t){try{let{cmd:e,args:n}=Oi(t);return(await E(e,n,{timeoutMs:Ou})).status===0}catch{return!1}}async function Ni(t,e){let n={name:t.name,installed:!1,current:null,latest:null};if(!await Gu(t.bin))return{...n,action:"not-installed"};if(!await Uu(t.manager)){let a=await De(t);return{...n,installed:!0,current:a,action:"skipped-no-manager",detail:`${t.manager} kh\xF4ng c\xF3 \u2014 b\u1ECF qua update`}}let r=await De(t);if(r===null){if(e.autoUpdate&&await Mi(t)){let u=await De(t);if(u!==null)return{name:t.name,installed:!0,current:u,latest:null,action:"updated",detail:"reinstall do binary h\u1ECFng"}}let{cmd:a,args:c}=Oi(t);return{name:t.name,installed:!0,current:null,latest:null,action:"binary-broken",detail:`${t.name} c\xE0i d\u1EDF, \`${t.bin} ${t.versionArgs.join(" ")}\` l\u1ED7i; reinstall kh\xF4ng kh\xF4i ph\u1EE5c \u2014 ch\u1EA1y th\u1EE7 c\xF4ng: ${a} ${c.join(" ")}`}}let o=await Du(t);if(!o||r===o)return{name:t.name,installed:!0,current:r,latest:o,action:"up-to-date"};if(!e.autoUpdate)return{name:t.name,installed:!0,current:r,latest:o,action:"update-available"};if(await Mi(t)){let a=await De(t);return{name:t.name,installed:!0,current:a,latest:o,action:"updated"}}return{name:t.name,installed:!0,current:r,latest:o,action:"update-failed",detail:`Kh\xF4ng c\u1EADp nh\u1EADt \u0111\u01B0\u1EE3c (d\xF9ng b\u1EA3n c\u0169 v${r})`}}import{spawnSync as ju}from"child_process";h();var Li=300*1e3,Gi="gitnexus",wt=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function Hu(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new wt("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Gi} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new wt("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new wt("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Ui(){i.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=ju("npm",["install","-g",Gi],{stdio:["inherit","inherit","pipe"],timeout:Li,encoding:"utf8"});if(t.signal==="SIGTERM")throw new wt("timeout",`npm install timeout sau ${Li/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Hu(t.status,t.stderr||"");ye();let e=pt();if(!e.installed||!e.path)throw new wt("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return i.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}h();var Di="gitnexus",Fu=15e3,Ku=300*1e3;async function Vu(){try{let t=await E("npm",["view",`${Di}@latest`,"version"],{timeoutMs:Fu});if(t.status!==0||!t.stdout)return null;let e=t.stdout.split(`
|
|
75
|
+
`).map(n=>n.trim()).filter(Boolean).at(-1);return e&&/^\d+\.\d+\.\d+/.test(e)?e:null}catch{return null}}async function Bu(){try{return(await E("npm",["install","-g",`${Di}@latest`],{timeoutMs:Ku})).status===0}catch{return!1}}async function _t(){let t=pt();if(!t.installed){i.info("GitNexus ch\u01B0a c\xE0i \u2014 \u0111ang c\xE0i (h\u1EA1 t\u1EA7ng c\u1ED1t l\xF5i c\u1EE7a Avatar, c\xE0i m\u1EB7c \u0111\u1ECBnh)...");try{return{installed:!0,version:Ui().version,updated:!0}}catch(n){let r=n instanceof Error?n.message:String(n);return i.warn(`Kh\xF4ng c\xE0i \u0111\u01B0\u1EE3c GitNexus (workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${r}`),i.dim("C\xE0i l\u1EA1i sau: avatar gitnexus install (ho\u1EB7c npm i -g gitnexus)."),{installed:!1,version:null,updated:!1,reason:"install-failed"}}}let e=await Vu();return e&&t.version&&e!==t.version?(i.info(`GitNexus c\xF3 b\u1EA3n m\u1EDBi (${t.version} \u2192 ${e}) \u2014 \u0111ang c\u1EADp nh\u1EADt...`),await Bu()?(ye(),t=pt(),i.success(`GitNexus \u0111\xE3 c\u1EADp nh\u1EADt \u2192 v${t.version??e}`),{installed:!0,version:t.version,updated:!0}):(i.warn(`Kh\xF4ng c\u1EADp nh\u1EADt \u0111\u01B0\u1EE3c GitNexus (d\xF9ng b\u1EA3n c\u0169 v${t.version}).`),{installed:!0,version:t.version,updated:!1,reason:"update-failed"})):{installed:!0,version:t.version,updated:!1}}h();var Wu=[{name:"Claude Code",bin:"claude",manager:"npm",pkg:"@anthropic-ai/claude-code",versionArgs:["--version"]},{name:"Supabase",bin:"supabase",manager:"brew",pkg:"supabase",versionArgs:["--version"]},{name:"git",bin:"git",manager:"brew",pkg:"git",versionArgs:["--version"]},{name:"gitleaks",bin:"gitleaks",manager:"brew",pkg:"gitleaks",versionArgs:["version"]},{name:"trivy",bin:"trivy",manager:"brew",pkg:"trivy",versionArgs:["--version"]}];async function qu(){try{let t=await _t();return t.installed?{name:"GitNexus",installed:!0,current:t.version,latest:t.version,action:t.updated?"updated":t.reason==="update-failed"?"update-failed":"up-to-date",detail:t.reason}:{name:"GitNexus",installed:!1,current:null,latest:null,action:"not-installed",detail:t.reason}}catch(t){return{name:"GitNexus",installed:!1,current:null,latest:null,action:"not-installed",detail:t instanceof Error?t.message:String(t)}}}async function zu(t){t.action==="updated"?await x("cli_update",`name=${t.name},to=${t.current??"?"},result=ok`):t.action==="update-failed"&&await x("cli_update",`name=${t.name},result=failed`)}async function ee(t){let n=[...await Promise.all(Wu.map(r=>Ni(r,{autoUpdate:t.autoUpdate}).catch(o=>({name:r.name,installed:!1,current:null,latest:null,action:"not-installed",detail:o instanceof Error?o.message:String(o)})))),await qu()];for(let r of n)await zu(r);return n}function ji(t){for(let e of t)e.action==="updated"?i.success(` \u2713 ${e.name}: c\u1EADp nh\u1EADt \u2192 v${e.current??"?"}`):e.action==="up-to-date"?i.dim(` \xB7 ${e.name}: v${e.current??"?"} (m\u1EDBi nh\u1EA5t)`):e.action==="update-available"?i.info(` \u2191 ${e.name}: v${e.current} \u2192 v${e.latest} c\xF3 b\u1EA3n m\u1EDBi`):e.action==="update-failed"?i.warn(` \u2717 ${e.name}: kh\xF4ng c\u1EADp nh\u1EADt \u0111\u01B0\u1EE3c (gi\u1EEF v${e.current})`):e.action==="skipped-no-manager"?i.dim(` \xB7 ${e.name}: ${e.detail??"b\u1ECF qua"}`):e.action==="not-installed"?i.dim(` \xB7 ${e.name}: ch\u01B0a c\xE0i`):e.action==="binary-broken"&&i.warn(` \u2717 ${e.name}: c\xE0i d\u1EDF (binary l\u1ED7i) \u2014 ${e.detail??"reinstall: npm install -g @anthropic-ai/claude-code"}`)}function Fi(t){t.command("doctor").description("Ch\u1EA9n \u0111o\xE1n c\xE0i \u0111\u1EB7t Avatar: hooks, MCP, login, submodule, ...").option("--fix","T\u1EF1 \u0111\u1ED9ng fix c\xE1c issue c\xF3 th\u1EC3 fix t\u1EF1 \u0111\u1ED9ng").action(async e=>{try{let n=await Yu(process.cwd());Xu(n),e.fix&&await Qu(n)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Yu(t){let e=[],n=process.versions.node,[r,o]=n.split(".").map(b=>Number.parseInt(b,10)),s=(r??0)>18||(r??0)===18&&(o??0)>=17;e.push({name:"Node.js version",status:s?"ok":"fail",detail:`v${n}${s?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await K();a?dt(a)?e.push({name:"Login status",status:"warn",detail:`Token h\u1EBFt h\u1EA1n (${a.email}) \u2014 ch\u1EA1y 'avatar login'`,fixable:!1}):e.push({name:"Login status",status:"ok",detail:`Logged in: ${a.email}`,fixable:!1}):e.push({name:"Login status",status:"fail",detail:"Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y 'avatar login'",fixable:!1});let c=ne(t,".claude","pack"),l=ne(t,"CLAUDE.md"),[u,m]=await Promise.all([d(c),d(l)]);if(e.push({name:"team-ai-pack installation",status:u?"ok":"warn",detail:u?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:m?"ok":"warn",detail:m?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),u){let b=ne(t,".claude",".gitignore"),_=!1;await d(b)&&(_=(await Hi.readFile(b,"utf8")).includes("settings.json")),e.push({name:"\u{1F512} settings.json gitignored (.claude/.gitignore)",status:_?"ok":"fail",detail:_?"an to\xE0n \u2014 settings.json kh\xF4ng commit nh\u1EA7m":"CRITICAL: settings.json ch\u1EE9a API key \u2014 ch\u1EA1y 'avatar doctor --fix' \u0111\u1EC3 b\u1EA3o v\u1EC7",fixable:!_,fix:_?void 0:async()=>{let{writeClaudeGitignore:G}=await Promise.resolve().then(()=>(bt(),Zr));await G(t)}})}let g=fr("which",["python"]),f=fr("which",["python3"]),w=g.status===0,C=f.status===0;C&&!w?e.push({name:"Python binary alias",status:"warn",detail:`Ch\u1EC9 c\xF3 python3 (modern macOS). Pack scripts th\u01B0\u1EDDng ref 'python' \u2192 suggest: ln -s ${f.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):w?e.push({name:"Python binary",status:"ok",detail:`python: ${g.stdout.toString().trim()}`,fixable:!1}):C&&e.push({name:"Python binary",status:"ok",detail:`python3: ${f.stdout.toString().trim()}`,fixable:!1});let R=ne(t,".claude","settings.json");if(await d(R))try{let b=await Hi.readFile(R,"utf8"),_=JSON.parse(b);if(_.statusLine?.command){let le=_.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(le?.[2]){let Lt=le[2],sa=Lt.startsWith("/")?Lt:ne(t,Lt),Ir=await d(sa);e.push({name:"statusLine command",status:Ir?"ok":"fail",detail:Ir?`ref OK: ${Lt}`:`BROKEN: settings.json ref '${Lt}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let U=fr("which",["claude"]),A=U.status===0;if(e.push({name:"Claude Code CLI",status:A?"ok":"warn",detail:A?U.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),u){let b=await Ei(t);e.push(...b)}for(let b of _i())e.push({name:`Scanner: ${b.name}`,status:b.installed?"ok":"warn",detail:b.installed?"\u0111\xE3 c\xE0i":`ch\u01B0a c\xE0i \u2014 security gate skip ph\u1EA7n n\xE0y (c\xE0i: brew install ${b.name})`,fixable:!1});let N=await ee({autoUpdate:!1});for(let b of N){let _=b.action==="update-available",G=b.action==="binary-broken";e.push({name:`CLI: ${b.name}`,status:_||G||b.action==="not-installed"?"warn":"ok",detail:_?`v${b.current} \u2192 v${b.latest} c\xF3 b\u1EA3n m\u1EDBi`:G?b.detail??"c\xE0i d\u1EDF \u2014 binary l\u1ED7i":b.action==="not-installed"?"ch\u01B0a c\xE0i":b.action==="skipped-no-manager"?b.detail??"b\u1ECF qua":`v${b.current??"?"} (m\u1EDBi nh\u1EA5t)`,fixable:_||G,..._||G?{fix:async()=>{await ee({autoUpdate:!0})}}:{}})}return e}function Xu(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,o=0;for(let s of t){let a=s.status==="ok"?p.green("\u2713"):s.status==="warn"?p.yellow("\u26A0"):p.red("\u2717");e.push(`${a} ${s.name.padEnd(28)} ${p.dim(s.detail)}`),s.status==="ok"?n+=1:(r+=1,s.fixable&&(o+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${o>0?` (${o} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${Ju(e.join(`
|
|
73
76
|
`),{padding:1,borderStyle:"round"})}
|
|
74
|
-
`)}async function
|
|
75
|
-
`).map(n=>n.trim()).filter(Boolean).at(-1);return e&&/^\d+\.\d+\.\d+/.test(e)?e:null}catch{return null}}async function Ou(){try{return(await M("npm",["install","-g",`${Ii}@latest`],{timeoutMs:_u})).status===0}catch{return!1}}async function Le(){let t=pt();if(!t.installed){s.info("GitNexus ch\u01B0a c\xE0i \u2014 \u0111ang c\xE0i (h\u1EA1 t\u1EA7ng c\u1ED1t l\xF5i c\u1EE7a Avatar, c\xE0i m\u1EB7c \u0111\u1ECBnh)...");try{return{installed:!0,version:_i().version,updated:!0}}catch(n){let r=n instanceof Error?n.message:String(n);return s.warn(`Kh\xF4ng c\xE0i \u0111\u01B0\u1EE3c GitNexus (workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${r}`),s.dim("C\xE0i l\u1EA1i sau: avatar gitnexus install (ho\u1EB7c npm i -g gitnexus)."),{installed:!1,version:null,updated:!1,reason:"install-failed"}}}let e=await Iu();return e&&t.version&&e!==t.version?(s.info(`GitNexus c\xF3 b\u1EA3n m\u1EDBi (${t.version} \u2192 ${e}) \u2014 \u0111ang c\u1EADp nh\u1EADt...`),await Ou()?(he(),t=pt(),s.success(`GitNexus \u0111\xE3 c\u1EADp nh\u1EADt \u2192 v${t.version??e}`),{installed:!0,version:t.version,updated:!0}):(s.warn(`Kh\xF4ng c\u1EADp nh\u1EADt \u0111\u01B0\u1EE3c GitNexus (d\xF9ng b\u1EA3n c\u0169 v${t.version}).`),{installed:!0,version:t.version,updated:!1,reason:"update-failed"})):{installed:!0,version:t.version,updated:!1}}k();h();import{promises as Mi}from"fs";import{homedir as Nu}from"os";import{join as Mu}from"path";var Oi=384,Ni={command:"gitnexus",args:["mcp"]};function Lu(){return Mu(Nu(),".claude","mcp_servers.json")}function Gu(t,e){return t===e?!0:typeof t!="object"||typeof e!="object"||t===null||e===null?!1:JSON.stringify(t)===JSON.stringify(e)}async function Uu(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await Mi.copyFile(t,n),n}async function Li(){let t=Lu(),e={},n=!1;if(await d(t)){n=!0;try{e=await b(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&Gu(r,Ni))return s.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await Uu(t),s.dim(`Backup ${t} \u2192 ${o}`));let i={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:Ni}};await T(t,i,Oi);try{await Mi.chmod(t,Oi)}catch{}return s.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}h();async function Du(t){for(;;)try{return await Et(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof Y&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",o=await j({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function Ge(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{s.info("=== Phase 10: GitNexus Setup ===");let n=await Le();if(!n.installed)return await A("gitnexus_setup",`result=skipped,reason=${n.reason}`),e.reason=n.reason,e;e.installed=!0,s.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{await So()}catch(r){s.warn(`gitnexus setup fail: ${r.message}`),s.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)s.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await Du(t.workspacePath))return await A("gitnexus_setup","result=skipped,reason=analyze-skipped"),s.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await ye(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&s.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await Li();e.mcpRegistered=!0,r.wasUpdated||s.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){s.warn(`MCP server register fail: ${r.message}`),s.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await A("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),s.success("GitNexus ready"),e}catch(n){if(n instanceof $)throw n;let r=n instanceof Error?n.message:String(n);return s.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await A("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}h();function pr(){let t=process.cwd(),e=Pt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
77
|
+
`)}async function Qu(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),i.success(`Fixed: ${n.name}`),e+=1}catch(r){i.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&i.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}k();import{promises as qi}from"fs";import{join as re}from"path";k();h();import{promises as Bi}from"fs";import{homedir as Zu}from"os";import{join as tp}from"path";var Ki=384,Vi={command:"gitnexus",args:["mcp"]};function ep(){return tp(Zu(),".claude","mcp_servers.json")}function np(t,e){return t===e?!0:typeof t!="object"||typeof e!="object"||t===null||e===null?!1:JSON.stringify(t)===JSON.stringify(e)}async function rp(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await Bi.copyFile(t,n),n}async function Wi(){let t=ep(),e={},n=!1;if(await d(t)){n=!0;try{e=await v(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&np(r,Vi))return i.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await rp(t),i.dim(`Backup ${t} \u2192 ${o}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:Vi}};await T(t,s,Ki);try{await Bi.chmod(t,Ki)}catch{}return i.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}h();async function op(t){for(;;)try{return await Pt(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof Q&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",o=await F({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function He(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{i.info("=== Phase 10: GitNexus Setup ===");let n=await _t();if(!n.installed)return await x("gitnexus_setup",`result=skipped,reason=${n.reason}`),e.reason=n.reason,e;e.installed=!0,i.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{await Eo()}catch(r){i.warn(`gitnexus setup fail: ${r.message}`),i.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)i.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await op(t.workspacePath))return await x("gitnexus_setup","result=skipped,reason=analyze-skipped"),i.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await xe(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&i.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await Wi();e.mcpRegistered=!0,r.wasUpdated||i.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){i.warn(`MCP server register fail: ${r.message}`),i.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await x("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),i.success("GitNexus ready"),e}catch(n){if(n instanceof $)throw n;let r=n instanceof Error?n.message:String(n);return i.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),i.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await x("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}h();function hr(){let t=process.cwd(),e=$t(t);return e||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
|
|
76
78
|
Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
|
|
77
79
|
B\u1EA1n \u0111ang \u1EDF: ${t}
|
|
78
|
-
Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&
|
|
79
|
-
Stale \u2192 'avatar gitnexus analyze <name>' \u0111\u1EC3 c\u1EADp nh\u1EADt.`)}async function
|
|
80
|
-
`);let r=
|
|
80
|
+
Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&i.dim(`Detected workspace root: ${e}`),e}async function ip(){let t=hr(),e=await He({workspacePath:t,skipAnalyze:!0});e.ok||e.installed?(i.success("GitNexus c\xE0i + MCP \u0111\u0103ng k\xFD xong."),i.dim("Index repo: avatar gitnexus analyze <name> | --all | (menu)")):i.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function sp(t){try{let e=re(t,".git","HEAD"),n=(await qi.readFile(e,"utf8")).trim();if(n.startsWith("ref:")){let r=n.slice(4).trim();try{return(await qi.readFile(re(t,".git",r),"utf8")).trim()}catch{return null}}return n}catch{return null}}async function ap(){let t=hr(),{readReposManifest:e}=await Promise.resolve().then(()=>(Y(),at)),n=await e(t);if(n.length===0){i.dim("Ch\u01B0a c\xF3 repo n\xE0o trong workspace. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}i.info(`GitNexus index status (${n.length} repo):`);for(let r of n){let o=re(t,"src",r.name),s=re(o,".gitnexus","meta.json");if(!await d(s)){i.plain(` \u2717 ${r.name.padEnd(28)} ch\u01B0a index \u2014 'avatar gitnexus analyze ${r.name}'`);continue}try{let a=await v(s),c=!1;if(a.lastCommit){let u=await sp(o);u&&(c=u!==a.lastCommit)}let l=c?"\u26A0 stale":"\u2713 fresh";i.plain(` ${l} ${r.name.padEnd(28)} indexed ${a.indexedAt?.slice(0,10)??"?"}`)}catch{i.plain(` ? ${r.name.padEnd(28)} (\u0111\u1ECDc meta.json fail)`)}}i.dim(`
|
|
81
|
+
Stale \u2192 'avatar gitnexus analyze <name>' \u0111\u1EC3 c\u1EADp nh\u1EADt.`)}async function cp(t,e){let n=hr(),{readReposManifest:r}=await Promise.resolve().then(()=>(Y(),at)),o=await r(n);if(o.length===0){i.warn("Ch\u01B0a c\xF3 repo n\xE0o trong workspace. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}let s;if(e.all)s=o.map(c=>c.name);else if(t)o.some(c=>c.name===t)||(i.error(`Repo "${t}" kh\xF4ng c\xF3 trong workspace. Xem: avatar list repo`),process.exit(1)),s=[t];else{let{select:c}=await import("@inquirer/prompts"),l=await c({message:"Index repo n\xE0o?",choices:[...o.map(u=>({name:u.name,value:u.name})),{name:"\u2500\u2500 T\u1EA5t c\u1EA3 \u2500\u2500",value:"__all__"}]});s=l==="__all__"?o.map(u=>u.name):[l]}let a=0;for(let c of s){let l=re(n,"src",c);try{await Pt(l),a+=1}catch(u){i.warn(` ! ${c} fail: ${u.message}`)}}i.success(`Index xong ${a}/${s.length} repo. Xem: avatar gitnexus status`)}function zi(t){let e=t.command("gitnexus").description("Qu\u1EA3n l\xFD GitNexus code intelligence (M10)");e.command("install").description("C\xE0i + setup GitNexus cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await ip()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await ap()}),e.command("analyze [name]").description("Index 1 repo trong src/ (t\xEAn), --all cho t\u1EA5t c\u1EA3, kh\xF4ng t\xEAn \u2192 menu").option("--all","Index m\u1ECDi repo trong src/").action(async(n,r)=>{await cp(n,r)})}import{resolve as Xp}from"path";import{input as Qp}from"@inquirer/prompts";import wr from"chalk";var Fe=[" \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2557 ","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2554\u255D","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u255A\u2588\u2588\u2557 \u2588\u2588\u2554\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2551 \u2588\u2588\u2551 \u255A\u2588\u2588\u2588\u2588\u2554\u255D \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551","\u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u2550\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D"],Ke=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function kr(t,e,n){return Math.round(t+(e-t)*n)}function lp(t){let n=Math.max(0,Math.min(1,t))*(Ke.length-1),r=Math.floor(n),o=Math.min(Ke.length-1,r+1),s=n-r,a=Ke[r],c=Ke[o];return[kr(a[0],c[0],s),kr(a[1],c[1],s),kr(a[2],c[2],s)]}function yr(t){if(!((process.stdout.isTTY??!1)&&wr.level>0))return[...Fe,...t?.tagline?["",t.tagline]:[]].join(`
|
|
82
|
+
`);let r=Fe.map((o,s)=>{let a=Fe.length===1?0:s/(Fe.length-1),[c,l,u]=lp(a);return wr.rgb(c,l,u).bold(o)});return t?.tagline&&(r.push(""),r.push(wr.dim(t.tagline))),r.join(`
|
|
81
83
|
`)}function It(t){process.stdout.write(`
|
|
82
|
-
${
|
|
84
|
+
${yr(t)}
|
|
83
85
|
|
|
84
|
-
`)}import{readdirSync as
|
|
85
|
-
`).find(r=>r.trimStart().startsWith("description:"));return n?n.replace(/^\s*description:\s*/,"").replace(/^['"]|['"]$/g,"").trim():""}async function
|
|
86
|
-
`);return
|
|
87
|
-
`)}function
|
|
86
|
+
`)}import{readdirSync as Ay}from"fs";import{select as Sy}from"@inquirer/prompts";import{simpleGit as $y}from"simple-git";import{existsSync as Vk,statSync as Bk}from"fs";import{join as qk}from"path";import{simpleGit as Yk}from"simple-git";import{existsSync as Zk}from"fs";import{join as ey}from"path";import{readFileSync as oy}from"fs";import{dirname as up,join as Ve}from"path";import{fileURLToPath as pp}from"url";var Be=up(pp(import.meta.url)),ay=[Ve(Be,"templates","gitignore"),Ve(Be,"..","templates","gitignore"),Ve(Be,"..","..","src","templates","gitignore"),Ve(Be,"..","src","templates","gitignore")],br="# === avatar ===",We="# === /avatar ===";h();import{existsSync as uy,readFileSync as py,writeFileSync as my}from"fs";import{join as gy}from"path";var qe=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as es}from"path";k();import{join as Ji}from"path";var mp=".avatar-pack-manifest.json",dp=".pack-version";function Yi(t){return Ji(t,mp)}async function ze(t,e){await T(Yi(t),e)}async function gp(t){let e=Yi(t);if(!await d(e))return null;try{return await v(e)}catch{return null}}async function Je(t){let e=await gp(t);if(e?.version)return e.version;let n=Ji(t,dp);if(await d(n)){let r=(await D(n)).trim();if(r)return r}return null}k();import{promises as oe}from"fs";import{join as vr}from"path";var fp="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack",Xi=2e4,Qi=[".supabase.co",".nal.vn"];function hp(){let t=process.env.AVATAR_GET_PACK_ENDPOINT??fp,e;try{e=new URL(t)}catch{throw new V(`AVATAR_GET_PACK_ENDPOINT kh\xF4ng ph\u1EA3i URL h\u1EE3p l\u1EC7: ${t}`)}if(e.protocol!=="https:")throw new V(`AVATAR_GET_PACK_ENDPOINT ph\u1EA3i d\xF9ng HTTPS (nh\u1EADn: ${e.protocol}). T\u1EEB ch\u1ED1i g\u1EEDi token qua k\u1EBFt n\u1ED1i kh\xF4ng m\xE3 h\xF3a.`);let n=e.hostname.toLowerCase();if(!Qi.some(o=>n===o.slice(1)||n.endsWith(o)))throw new V(`AVATAR_GET_PACK_ENDPOINT host kh\xF4ng \u0111\u01B0\u1EE3c ph\xE9p: ${n}. Ch\u1EC9 ch\u1EA5p nh\u1EADn: ${Qi.join(", ")}. T\u1EEB ch\u1ED1i g\u1EEDi id_token t\u1EDBi host l\u1EA1.`);return t}var kt=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},tt=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},V=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function Ye(t,e){let n,r=new AbortController,o=setTimeout(()=>r.abort(),Xi);try{n=await fetch(hp(),{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify({version:e}),signal:r.signal})}catch(a){throw a instanceof Error&&a.name==="AbortError"?new tt(`get-pack qu\xE1 ${Xi/1e3}s kh\xF4ng ph\u1EA3n h\u1ED3i. Th\u1EED l\u1EA1i.`):new tt(`Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Supabase get-pack: ${a instanceof Error?a.message:a}`)}finally{clearTimeout(o)}if(n.status===401||n.status===403){let a=await Zi(n);throw new kt(`Supabase t\u1EEB ch\u1ED1i (${n.status}): ${a}. Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn.`)}if(n.status>=500)throw new tt(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let a=await Zi(n);throw new V(`get-pack tr\u1EA3 ${n.status}: ${a}`)}let s;try{s=await n.json()}catch{throw new V("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(s.error)throw new V(`get-pack b\xE1o l\u1ED7i: ${s.error}`);if(!s.url||!s.object)throw new V("get-pack thi\u1EBFu url/object trong response.");return{url:s.url,object:s.object,expiresIn:s.expires_in??0,requestedBy:s.requested_by??""}}var wp=new Set(["plans","docs",".pytest_cache",".ruff_cache","__pycache__",".serena"]);async function ts(t){let e;try{e=await oe.readdir(t,{withFileTypes:!0})}catch{return}for(let n of e){let r=vr(t,n.name);wp.has(n.name)?await J(r).catch(()=>{}):n.isDirectory()&&await ts(r)}}async function Xe(t,e){let n=vr(e,"..");await S(n);let r=vr(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await un(t,r)}catch(c){throw new tt(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await J(o);try{await pn(r,o)}catch(c){throw new V(`Gi\u1EA3i n\xE9n tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await ts(o);let s=`${e}.old-${process.pid}`,a=await d(e);a&&await oe.rename(e,s);try{await oe.rename(o,e)}catch(c){throw a&&await oe.rename(s,e).catch(()=>{}),c}a&&await J(s).catch(()=>{})}finally{await oe.rm(r,{force:!0}).catch(()=>{}),await J(o).catch(()=>{})}}async function Zi(t){try{let e=await t.json();return e.error??JSON.stringify(e).slice(0,200)}catch{return(await t.text().catch(()=>"")).slice(0,200)}}var et=".claude/pack",Mt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function ns(t,e){let n=await Me(),r=await Ye(n,e),o=es(t,et);await Xe(r.url,o);let s=e??kp(r.object);return await ze(o,{version:s,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:s}}function kp(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Qe(t){let e=es(t,et);return await Je(e)??"(unknown)"}h();k();import{join as en,relative as Pp,resolve as us}from"path";import{input as ms,select as Ep}from"@inquirer/prompts";import Rp from"boxen";h();import{readdir as rs,readFile as yp}from"fs/promises";import{join as Ze}from"path";import bp from"boxen";function vp(t){let e=t.match(/^---\n([\s\S]*?)\n---/);if(!e?.[1])return"";let n=e[1].split(`
|
|
87
|
+
`).find(r=>r.trimStart().startsWith("description:"));return n?n.replace(/^\s*description:\s*/,"").replace(/^['"]|['"]$/g,"").trim():""}async function xp(t){let e;try{e=await rs(t,{withFileTypes:!0})}catch{return[]}let n=[],r=new Set;for(let o of e){if(!o.isFile()||!o.name.endsWith(".md"))continue;let s=o.name.replace(/\.md$/,"");r.add(s);try{let a=await yp(Ze(t,o.name),"utf8");n.push({cmd:`/avatar:${s}`,desc:vp(a)})}catch{}}for(let o of e){if(!o.isDirectory()||r.has(o.name))continue;let s=Ze(t,o.name),a=await xr(s);a!==0&&n.push({cmd:`/avatar:${o.name}:*`,desc:`nh\xF3m ${a} l\u1EC7nh (g\xF5 /avatar:${o.name}: \u0111\u1EC3 xem)`})}return n.sort((o,s)=>o.cmd.localeCompare(s.cmd)),n}async function xr(t){let e=0,n;try{n=await rs(t,{withFileTypes:!0})}catch{return 0}for(let r of n)r.isDirectory()?e+=await xr(Ze(t,r.name)):r.name.endsWith(".md")&&(e+=1);return e}async function os(t){let e=Ze(t,"commands","avatar"),n=await xp(e);if(n.length===0)return null;let r=await xr(e),o=Math.max(...n.map(m=>m.cmd.length)),s=p.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),a=p.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),c=n.map(m=>{let g=p.cyan(m.cmd.padEnd(o));return m.desc?` ${g} ${p.dim(m.desc)}`:` ${g}`}),l=p.dim(`${n.length} l\u1EC7nh g\u1ED1c (\u0111\u1EA7y \u0111\u1EE7 ${r} g\u1ED3m sub-command). Sub: th\xEAm :fast / :hard / :auto / :parallel sau l\u1EC7nh g\u1ED1c.`),u=[s,a,"",...c,"",l].join(`
|
|
88
|
+
`);return bp(u,{padding:1,borderStyle:"round",borderColor:"cyan"})}h();import Ap from"boxen";var is=[{name:"SKILLS",role:"k\u1EF9 n\u0103ng t\xE1i d\xF9ng \u2014 Claude g\u1ECDi khi c\u1EA7n"},{name:"AGENTS",role:"subagent chuy\xEAn tr\xE1ch \u2014 skills \u0111i\u1EC1u ph\u1ED1i"},{name:"WORKFLOW",role:"orchestrate nhi\u1EC1u agent song song"},{name:"TOOLS",role:"hook g\xE1c c\u1ED5ng m\u1ECDi thao t\xE1c"},{name:"SECURITY",role:"ch\u1EB7n push/secret r\u1EE7i ro tr\u01B0\u1EDBc khi x\u1EA3y ra"},{name:"LOG TRACKING",role:"ghi nh\u1EADn m\u1ECDi h\xE0nh \u0111\u1ED9ng \u0111\u1EC3 truy v\u1EBFt"},{name:"AUTO SUPPORT",role:"t\u1EF1 x\u1EED l\xFD / nh\u1EAFc \u0111\xFAng l\xFAc c\u1EA7n"},{name:"KNOWLEDGE",role:"s\u1ED5 kinh nghi\u1EC7m nu\xF4i c\u1EA3 h\u1EC7 th\u1ED1ng"}],Ar=[125,88,217],Cp=[205,95,130],ss=[150,120,200];function as(){return(process.stdout.isTTY??!1)&&p.level>0}function Ot(t,e,n=!1){if(!as())return e;let r=p.rgb(t[0],t[1],t[2]);return n?r.bold(e):r(e)}function Cr(t){return as()?p.dim(t):t}function Sp(){let t=[];return t.push(`${Ot(Ar,"\u25C6",!0)} ${Ot(Ar,"AVATAR",!0)} ${Cr("\u2014 l\xF5i \u0111i\u1EC1u ph\u1ED1i")}`),t.push(Ot(ss,"\u2502")),is.forEach((e,n)=>{let r=n===is.length-1,o=Ot(ss,r?"\u2514\u2500\u2500":"\u251C\u2500\u2500");t.push(`${o} ${Ot(Cp,e.name,!0)} ${Cr(e.role)}`)}),t.join(`
|
|
89
|
+
`)}function cs(){let t=Ot(Ar,"Ki\u1EBFn tr\xFAc Avatar \u2014 8 kh\u1ED1i quanh m\u1ED9t l\xF5i",!0),e=Cr("M\u1ECDi kh\u1ED1i chia s\u1EBB c\xF9ng m\u1ED9t l\xF5i, h\u1EE3p th\xE0nh 1 h\u1EC7 th\u1ED1ng cho c\u1EA3 \u0111\u1ED9i."),n=`${t}
|
|
88
90
|
|
|
89
|
-
${
|
|
91
|
+
${Sp()}
|
|
90
92
|
|
|
91
93
|
${e}`;process.stdout.write(`
|
|
92
|
-
${
|
|
93
|
-
`)}h();k();
|
|
94
|
-
Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=
|
|
94
|
+
${Ap(n,{padding:1,borderStyle:"round",borderColor:"magenta",textAlignment:"left"})}
|
|
95
|
+
`)}h();k();bt();import{readdir as Tp}from"fs/promises";import{join as $p}from"path";async function tn(t){if(!await d(t))return!0;try{return(await Tp(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function ls(t,e,n=10){for(let r=2;r<n;r++){let o=$p(t,`${e}-${r}`);if(await tn(o))return o}return null}function ps(t,e){if(/[/\\]/.test(e)||e==="."||e===".."||e.includes(".."))throw new Error(`T\xEAn workspace kh\xF4ng an to\xE0n (ch\u1EE9a path separator ho\u1EB7c '..'): "${e}".
|
|
96
|
+
Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=us(en(t,e)),r=us(t);if(!n.startsWith(`${r}/`)&&n!==r)throw new Error(`T\xEAn workspace "${e}" resolve ra ngo\xE0i parent dir (path traversal). Block.`)}async function ds(t,e,n){ps(t,e);let r=en(t,e);if(await tn(r))return r;for(i.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await ls(t,e);if(n&&o)return i.info(`--force: d\xF9ng ${o}`),o;let s=[];o&&s.push({name:`D\xF9ng "${o}" (suggest)`,value:"use-alt"}),s.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),s.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await Ep({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await ms({message:"T\xEAn workspace m\u1EDBi:",validate:u=>{let m=u.trim();return m.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[/\\]/.test(m)||m==="."||m===".."||m.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});ps(t,c.trim());let l=en(t,c.trim());if(await tn(l))return l;i.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function gs(t){return await ms({message:"Team owner email:",default:t})}async function fs(t){try{await S(t)}catch(e){let n=e;throw n.code==="EACCES"||n.code==="EPERM"?new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": kh\xF4ng c\xF3 quy\u1EC1n ghi.`):n.code==="ENAMETOOLONG"?new Error(`\u0110\u01B0\u1EDDng d\u1EABn workspace qu\xE1 d\xE0i: "${t}".`):n.code==="ENOSPC"?new Error("H\u1EBFt dung l\u01B0\u1EE3ng \u0111\u0129a, kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace."):new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": ${n.message}`)}}function _p(t){if(t===null)return` ${p.yellow("AI:")} skipped \xB7 ${p.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${p.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${p.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${p.cyan("avatar ai setup")}`}function Ip(t){if(t===null)return` ${p.yellow("GitNexus:")} skipped \xB7 ${p.cyan("avatar gitnexus install")} \u0111\u1EC3 setup sau`;if(t.ok){let e=["ready"];return t.analyzed&&e.push("indexed"),t.wikiGenerated&&e.push("wiki"),t.mcpRegistered&&e.push("mcp"),` ${p.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${p.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${p.cyan("avatar gitnexus install")}`}async function hs(t,e=null,n=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Pp(process.cwd(),t)||t}`,_p(e),Ip(n),"",` ${p.cyan(`cd ${t}`)}`,` ${p.cyan("avatar add repo")} Th\xEAm repo code v\xE0o src/`,` ${p.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${p.cyan("avatar sync")} C\u1EADp nh\u1EADt team-ai-pack m\u1EDBi`,` ${p.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${Rp(r.join(`
|
|
95
97
|
`),{padding:1,borderStyle:"round"})}
|
|
96
|
-
`);let o=
|
|
97
|
-
${
|
|
98
|
-
`)}
|
|
99
|
-
`);process.stdout.write(`${
|
|
100
|
-
`),
|
|
98
|
+
`);let o=en(t,et);if(await d(o)){let s=await os(o);s&&process.stdout.write(`
|
|
99
|
+
${s}
|
|
100
|
+
`)}cs()}import Mp from"boxen";import Op from"open";Jt();h();function ws(t){t.command("login").description("\u0110\u0103ng nh\u1EADp Google SSO (workspace @nal.vn)").option("--reset","X\xF3a credential c\u0169 v\xE0 \u0111\u0103ng nh\u1EADp l\u1EA1i").action(async e=>{try{await Sr(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Sr(t){if(It({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await Ie(),await x("login_reset");else{let g=await K();if(g&&!dt(g)){i.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=X("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await zn(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=Zn(n),o=[`1. Truy c\u1EADp: ${p.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${p.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${p.green("Allow")}...`].join(`
|
|
101
|
+
`);process.stdout.write(`${Mp(o,{padding:1,borderStyle:"round"})}
|
|
102
|
+
`),Op(r).catch(()=>{i.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=X("\u0110ang ch\u1EDD x\xE1c nh\u1EADn trong browser..."),a=n.interval*1e3,c=Date.now()+n.expires_in*1e3,l=null;for(;Date.now()<c;){await Np(a);try{if(l=await Jn(n.device_code),l)break}catch(g){throw s.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(s.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),s.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=Yn(l.id_token);try{Xn(u)}catch(g){throw await zt(l.access_token),g}let m=Qn(l,u);await rr(m),await x("login",m.email),i.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),i.success(`Verify hosted domain: ${u.hd} \u2713`),i.success(`L\u01B0u credential v\xE0o ${mt} (chmod 600)`)}function Np(t){return new Promise(e=>setTimeout(e,t))}import{join as on}from"path";h();var ks=3;async function ys(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await ns(t,e)).pinnedTag,skipped:!1}}catch(s){if(s instanceof Mt)throw s;if(s instanceof kt){i.error(s.message);let l=await F({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:s.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return i.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(s instanceof tt&&o<ks){o+=1,i.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${ks}): ${s.message}`),await Lp(1e3*o);continue}let a=s instanceof Error?s.message:String(s),c=await F({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new $("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return i.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function Lp(t){return new Promise(e=>setTimeout(e,t))}k();bt();import{readFileSync as Gp}from"fs";import{dirname as Up,resolve as Dp}from"path";import{fileURLToPath as jp}from"url";var Nt=null;function O(){if(Nt!==null)return Nt;let t=Up(jp(import.meta.url));for(let e=0;e<5;e++){let n=Dp(t,...Array(e).fill(".."),"package.json");try{let r=Gp(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return Nt=o.version,Nt}catch{}}return Nt="unknown",Nt}k();import{promises as Hp}from"fs";import{join as bs}from"path";var Fp=[".claude","pack","tools"];function Kp(t){return bs(t,...Fp)}async function ie(t){let e=Kp(t);if(!await d(e))return[];let n=await Hp.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await d(bs(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function se(t){let[e,n]=await Promise.all([ie(t),Rt(t)]);return{available:e,enabled:n}}h();var Vp=new Set(["ai-orchestrator"]);async function vs(t,e={}){let r=(await ie(t)).filter(o=>!Vp.has(o));if(r.length!==0){i.info(`B\u1EADt ${r.length} tool m\u1EB7c \u0111\u1ECBnh (g\u1EE1 sau b\u1EB1ng 'avatar tools disable <t\xEAn>')...`);for(let o of r)await W(t,o,{silent:!0}),i.dim(` \u2713 ${o}`)}}import{promises as nn}from"fs";import{dirname as Tr,join as As,relative as $r}from"path";import{promises as Bp}from"fs";function Wp(){let t=new Date,e=n=>n.toString().padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}-${e(t.getHours())}${e(t.getMinutes())}`}async function xs(t){let e=`${t}.backup-${Wp()}`;return await Bp.rename(t,e),e}k();var Pr=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function qp(t){try{return(await nn.lstat(t)).isSymbolicLink()}catch{return!1}}async function zp(t,e,n){let r=$r(Tr(e),e)||e;if(!await d(t))return{dir:r,action:"source-missing"};if(await d(e))if(await qp(e))await nn.unlink(e);else if(n){let s=await xs(e),a=$r(Tr(e),t);return await nn.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:s}}else return{dir:r,action:"skipped-conflict"};let o=$r(Tr(e),t);return await nn.symlink(o,e),{dir:r,action:"created"}}async function rn(t,e,n){let r=[];for(let o of Pr){let s=As(t,o),a=As(e,o);r.push(await zp(s,a,n))}return r}h();function Jp(t){return t?`
|
|
101
103
|
### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
|
|
102
104
|
|
|
103
105
|
Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
|
|
@@ -125,32 +127,32 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
|
|
|
125
127
|
\`\`\`bash
|
|
126
128
|
gitnexus wiki . --api-key <key> --base-url <url>
|
|
127
129
|
\`\`\`
|
|
128
|
-
`:""}function
|
|
130
|
+
`:""}function Cs(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:O(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:Jp(t.gitnexusReady??!1)}}async function Ss(t){await fs(t.workspacePath),await S(on(t.workspacePath,"src")),await S(on(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)i.dim("Skip team-ai-pack (--skip-team-pack).");else{X("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await ys(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||i.success(`team-ai-pack: ${e}`)}let n=Cs({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await gn(t.workspacePath),await fn(t.workspacePath,{avatarVersion:O(),workspaceName:t.workspaceName}),await hn(t.workspacePath,n),await wn(t.workspacePath,n),await yn(t.workspacePath,n),await bn(t.workspacePath),await Yp(t.workspacePath,t.autoYes),await x("init",`workspace=${t.workspaceName}`);let r=null;t.aiSkip?i.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau: avatar ai setup"):r=await Le({workspacePath:t.workspacePath});try{await Ee(t.workspacePath,r,{autoYes:t.autoYes})}catch(a){i.warn(` \u2717 Setup AI Orchestrator fail: ${a instanceof Error?a.message:a}`)}let o=null;t.aiSkip||t.gitnexusSkip?i.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await He({workspacePath:t.workspacePath,skipAnalyze:!0});try{i.info("Ki\u1EC3m tra c\u1EADp nh\u1EADt CLI ph\u1EE5 thu\u1ED9c...");let a=await ee({autoUpdate:!0});ji(a)}catch(a){i.warn(`C\u1EADp nh\u1EADt CLI ph\u1EE5 thu\u1ED9c b\u1ECF qua: ${a instanceof Error?a.message:a}`)}await hs(t.workspacePath,r,o)}async function Yp(t,e){let n=on(t,et);if(!await d(n)){i.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=on(t,".claude");i.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");let o=!1;try{let s=await rn(n,r,!1),a=s.filter(l=>l.action==="created"||l.action==="updated").length,c=s.filter(l=>l.action==="source-missing").length;i.success(` \u2713 Symlinks: ${a} created${c>0?`, ${c} source-missing`:""}`)}catch(s){o=!0,i.warn(` \u2717 Symlink farm fail: ${s instanceof Error?s.message:s}`)}try{let s=await Se(t);s.action==="merged"&&i.success(` \u2713 settings.json merged (${s.changes.join("; ")})`)}catch(s){o=!0,i.warn(` \u2717 Merge settings fail: ${s instanceof Error?s.message:s}`)}try{await vs(t,{autoYes:e})}catch(s){o=!0,i.warn(` \u2717 Setup default tools fail: ${s instanceof Error?s.message:s}`)}try{Ue()}catch(s){i.warn(` ! Security scanner check fail: ${s instanceof Error?s.message:s}`)}o&&(i.warn("\u26A0 Auto-sync pack CH\u01AFA ho\xE0n t\u1EA5t \u2014 workspace c\xF3 th\u1EC3 thi\u1EBFu symlinks/settings."),i.warn(" Ch\u1EA1y `avatar sync` \u0111\u1EC3 ho\xE0n t\u1EA5t setup."))}function Ts(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar workspace (m\xF4i tr\u01B0\u1EDDng) \u2014 repo c\xE0i sau b\u1EB1ng 'avatar add repo'").argument("[workspace-name]","T\xEAn workspace (b\u1EAFt bu\u1ED9c, h\u1ECFi n\u1EBFu thi\u1EBFu)").option("--workspace-parent <path>","Th\u01B0 m\u1EE5c cha t\u1EA1o workspace (m\u1EB7c \u0111\u1ECBnh . \u2014 CWD)").option("--pack-version <tag>","Pin team-ai-pack v\xE0o tag c\u1EE5 th\u1EC3 (vd v0.2.0-beta.0)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a workspace").option("--skip-team-pack","B\u1ECF qua t\u1EA3i team-ai-pack (test mode)").option("--force","B\u1ECF qua prompt khi workspace path \u0111\xE3 t\u1ED3n t\u1EA1i").option("--yes","Auto-confirm t\u1EA5t c\u1EA3 prompt").option("--ai-skip","B\u1ECF qua phase AI setup (ch\u1EA1y 'avatar ai setup' sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup").action(async(e,n)=>{try{await Zp(e,n)}catch(r){(r instanceof qe||r instanceof Mt||r instanceof Ft||r instanceof $)&&(i.dim(r.message),process.exit(0)),i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function Zp(t,e){e.yes||It({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await K();for(;!n||dt(n);){i.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp (ho\u1EB7c token h\u1EBFt h\u1EA1n) \u2014 ch\u1EA1y login tr\u01B0\u1EDBc...");try{await Sr({})}catch(c){throw c instanceof $||i.error(`Login th\u1EA5t b\u1EA1i: ${c.message}`),c}if(n=await K(),n&&!dt(n))break;throw new $("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await Qp({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await gs(n.email),s=Xp(e.workspaceParent??"."),a=await ds(s,r.trim(),e.force);await Ss({workspacePath:a,workspaceName:r.trim(),teamOwner:o,description:e.description??`Avatar workspace: ${r.trim()}`,packVersion:e.packVersion,autoYes:e.yes,skipTeamPack:e.skipTeamPack,aiSkip:e.aiSkip,gitnexusSkip:e.gitnexusSkip})}Jt();h();function $s(t){t.command("logout").description("\u0110\u0103ng xu\u1EA5t \u2014 revoke token Google + x\xF3a credential local").option("--keep-remote","Ch\u1EC9 x\xF3a credential local, KH\xD4NG revoke v\u1EDBi Google").action(async e=>{try{await tm(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function tm(t={}){let e=await K();if(!e){i.dim("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 \u0111\u0103ng xu\u1EA5t.");return}let n=e.email;if(!t.keepRemote){let r=X("Revoke token v\u1EDBi Google...");await zt(e.refresh_token),r.succeed("\u0110\xE3 revoke token v\u1EDBi Google")}await Ie(),await x("logout",n),i.success(`\u0110\xE3 \u0111\u0103ng xu\u1EA5t: ${n}`),i.dim("\u0110\u0103ng nh\u1EADp l\u1EA1i: avatar login")}h();function nt(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
|
|
129
131
|
`),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${p.cyan(e)}
|
|
130
132
|
`),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
|
|
131
|
-
`),process.exit(0)}}function
|
|
132
|
-
`):
|
|
133
|
+
`),process.exit(0)}}function Ps(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(nt("mcp-run","Milestone 09"))}k();import{join as em}from"path";import Es from"boxen";h();var nm=".claude/pack";function Rs(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB version pack \u0111ang c\xE0i").option("--json","Output JSON cho script").action(async n=>{try{let r=await rm(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
|
|
134
|
+
`):om(r)}catch(r){i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function rm(t){let e=em(t,nm);return await d(e)?{installed:!0,currentVersion:await Qe(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function om(t){if(!t.installed){let r=[`${p.bold("team-ai-pack")} \xB7 ${p.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),p.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${Es(r.join(`
|
|
133
135
|
`),{padding:1,borderStyle:"round"})}
|
|
134
|
-
`);return}let e=t.currentVersion??p.yellow("(unknown)"),n=[`${p.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${p.dim("Version hi\u1EC7n t\u1EA1i:")} ${e}`,"",p.dim("C\u1EADp nh\u1EADt m\u1EDBi nh\u1EA5t: avatar sync")];process.stdout.write(`${
|
|
136
|
+
`);return}let e=t.currentVersion??p.yellow("(unknown)"),n=[`${p.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${p.dim("Version hi\u1EC7n t\u1EA1i:")} ${e}`,"",p.dim("C\u1EADp nh\u1EADt m\u1EDBi nh\u1EA5t: avatar sync")];process.stdout.write(`${Es(n.join(`
|
|
135
137
|
`),{padding:1,borderStyle:"round"})}
|
|
136
|
-
`)}function
|
|
137
|
-
`):
|
|
138
|
-
`).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function
|
|
138
|
+
`)}function _s(t){t.command("restore").description("Kh\xF4i ph\u1EE5c .claude/pack/ t\u1EEB backup (M08)").option("--backup <name>","T\xEAn backup folder trong .claude/_backup/").option("--list","Li\u1EC7t k\xEA c\xE1c backup hi\u1EC7n c\xF3").action(nt("restore","Milestone 08"))}function Is(t){t.command("review").description("Review pending proposals t\u1EEB avatar scan (M08)").option("--accept-all","Approve m\u1ECDi pending kh\xF4ng h\u1ECFi (CI mode)").option("--reject-all","X\xF3a m\u1ECDi pending kh\xF4ng h\u1ECFi").action(nt("review","Milestone 08"))}function Ms(t){t.command("scan").description("Ch\u1EA1y project scanner v\xE0 \u0111\u1EC1 xu\u1EA5t knowledge update (M06)").option("--incremental","Ch\u1EC9 scan c\xE1c file thay \u0111\u1ED5i t\u1EEB commit cu\u1ED1i").option("--full","Scan to\xE0n b\u1ED9 d\u1EF1 \xE1n (default)").option("--scanners <list>","tech-stack,conventions,architecture,domain,git-pattern").option("--quiet","Ch\u1EA1y ng\u1EA7m, \xEDt output (d\xF9ng cho git hook)").action(nt("scan","Milestone 06"))}import{promises as cm}from"fs";import{join as sn}from"path";import lm from"boxen";k();k();import{promises as im}from"fs";import{join as sm}from"path";var am="_backup";async function Os(t){let e=sm(t,".claude",am);return await d(e)?(await im.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}h();function Ns(t){t.command("status").description("Snapshot t\u1EE9c th\xEC: project, pack version, pending, backup").option("--json","Output JSON cho script").action(async e=>{try{let n=await um(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
|
|
139
|
+
`):dm(n)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function um(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=sn(t,".claude");if(!await d(n))return{projectName:e,cliVersion:O(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=sn(n,"_pending"),[s,a,c,l,u]=await Promise.all([(async()=>await d(sn(n,"pack"))?Qe(t).catch(()=>null):null)(),(async()=>await d(o)?(await cm.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),Os(t).then(m=>m.length).catch(()=>0),pm(n).catch(()=>"(read error)"),se(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:O(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:u.enabled,toolsAvailableCount:u.available.length}}async function pm(t){let e=sn(t,"project","tech-stack.md");return await d(e)?(await D(e)).split(`
|
|
140
|
+
`).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function mm(t){return t.toolsEnabled.length>0?`${t.toolsEnabled.join(", ")} (${t.toolsEnabled.length}/${t.toolsAvailableCount} available)`:t.toolsAvailableCount>0?`none enabled (${t.toolsAvailableCount} available \u2014 avatar tool list)`:"none"}function dm(t){let e=[`${p.bold("Avatar Status")} \xB7 ${p.cyan(t.projectName)}`,"\u2500".repeat(48),`${p.dim("CLI version:")} ${t.cliVersion}`,`${p.dim("Pack version:")} ${t.packVersion??p.yellow("not installed")}`,`${p.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?p.dim(" (avatar review)"):""}`,`${p.dim("Backups:")} ${t.backupCount}`,`${p.dim("Tech stack:")} ${t.techStackSummary}`,`${p.dim("Tools:")} ${mm(t)}`];process.stdout.write(`${lm(e.join(`
|
|
139
141
|
`),{padding:1,borderStyle:"round"})}
|
|
140
|
-
`)}import{readFile as
|
|
141
|
-
Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await
|
|
142
|
-
Mount dir statuses:`);for(let l of c.mountDirStatuses)
|
|
143
|
-
Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await
|
|
144
|
-
${
|
|
145
|
-
`,"utf8")}async function
|
|
146
|
-
`),o=[],
|
|
147
|
-
`).trim();a.length===0?await
|
|
148
|
-
`,"utf8")}h();h();import{spawnSync as
|
|
142
|
+
`)}import{readFile as fm}from"fs/promises";import{basename as hm,join as Er}from"path";k();k();import{join as Ls}from"path";async function gm(t,e,n){let r=Ls(t,n),o=Ls(e,n);if(!await d(r))return"source-missing";if(!await d(o))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(o)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function Gs(t,e,n){let r=await Je(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",s=[];for(let a of Pr)s.push({dir:a,status:await gm(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:s}}bt();h();async function Us(t){let e=await Rt(t);if(e.length!==0){i.info(`Re-apply ${e.length} tool(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await W(t,n)}}h();function wm(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function km(t,e){let n=hm(t)||"workspace",r="";try{r=(await fm(e,"utf8")).match(/^-\s*Team owner:\s*(.+)$/m)?.[1]?.trim()??""}catch{}return{projectName:n,projectDescription:"",teamOwner:r}}async function ym(t){let e=process.cwd(),n=Er(e,".claude"),r=Er(e,et);if(await d(n)||(i.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
|
|
143
|
+
Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await Gs(r,n,t.packVersion);i.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),i.info(`Target version: ${c.targetVersion}`),i.info(`
|
|
144
|
+
Mount dir statuses:`);for(let l of c.mountDirStatuses)i.plain(` ${l.dir.padEnd(12)} ${l.status}`);i.info(`
|
|
145
|
+
Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await Me()}catch(c){i.error(c instanceof Error?c.message:String(c)),process.exit(1)}i.info(t.packVersion?`T\u1EA3i team-ai-pack ${t.packVersion} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let s;try{let c=await Ye(o,t.packVersion);await Xe(c.url,r),s=t.packVersion??wm(c.object),await ze(r,{version:s,downloadedObject:c.object,extractedAt:new Date().toISOString()}),i.success(`\u0110\xE3 c\xE0i pack ${s}`)}catch(c){c instanceof kt?i.error(`${c.message}`):i.error(`T\u1EA3i pack th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`),process.exit(1)}i.info("T\u1EA1o symlink farm...");let a=await rn(r,n,t.force===!0);bm(a,t.force===!0),i.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await Se(e);switch(c.action){case"merged":i.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":i.info(" - settings.json \u0111\xE3 sync.");break;case"no-pack-template":i.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip.");break}}catch(c){i.warn(` ! Merge settings.json fail: ${c instanceof Error?c.message:c}.`)}try{await Us(e)}catch(c){i.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}i.info("C\u1EADp nh\u1EADt managed block trong CLAUDE.md...");try{let c=await km(e,Er(e,"CLAUDE.md")),l={projectName:c.projectName,projectDescription:c.projectDescription,teamOwner:c.teamOwner,avatarVersion:O(),packVersion:s,lastScan:new Date().toISOString(),gitnexusSection:""};switch((await kn(e,l)).outcome){case"replaced-block":i.success(" \u2713 CLAUDE.md managed block \u0111\xE3 c\u1EADp nh\u1EADt (gi\u1EEF context user ngo\xE0i block).");break;case"replaced-whole":i.success(" \u2713 CLAUDE.md (ch\u01B0a c\xF3 marker) \u2192 thay ho\xE0n to\xE0n b\u1EB1ng b\u1EA3n Avatar m\u1EDBi.");break;case"skipped-no-file":i.dim(" - Ch\u01B0a c\xF3 CLAUDE.md (ch\u1EA1y 'avatar init' \u0111\u1EC3 t\u1EA1o), skip.");break}}catch(c){i.warn(` ! C\u1EADp nh\u1EADt CLAUDE.md fail: ${c instanceof Error?c.message:c}.`)}try{await _t()}catch(c){i.warn(` ! GitNexus check fail: ${c instanceof Error?c.message:c}.`)}try{Ue()}catch(c){i.warn(` ! Scanner check fail: ${c instanceof Error?c.message:c}.`)}i.success(`Synced team-ai-pack \u2192 ${s}.`)}function bm(t,e){for(let r of t)switch(r.action){case"created":i.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":i.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":i.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":i.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":i.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override.`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&i.warn(`${n} mount dir(s) skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force.`)}function Ds(t){t.command("sync").description("T\u1EA3i team-ai-pack m\u1EDBi nh\u1EA5t (tarball Supabase) + t\u1EA1o symlink farm").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--pack-version <tag>","Pin version c\u1EE5 th\u1EC3 (vd: v0.2.0 ho\u1EB7c v0.2.0-beta.0)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply").action(ym)}import{resolve as vm}from"path";import{checkbox as xm,confirm as Am}from"@inquirer/prompts";h();function js(t,e){let n=new Set(e);return t.map(r=>({name:n.has(r)?`${r} (installed)`:r,value:r,checked:n.has(r)}))}function Hs(t){return t.map(e=>({name:e,value:e,checked:!1}))}function Fs(t){return[...t]}function ae(t){return vm(t.target??process.cwd())}async function Ks(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return Fs(t);let s=r==="add"?js(t,e):Hs(t);return await xm({message:r==="add"?"Ch\u1ECDn tool \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn tool \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:s})}async function Cm(t){let e=ae(t),{available:n,enabled:r}=await se(e);if(n.length===0){i.dim("Kh\xF4ng c\xF3 tool available (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync).");return}let o=await Ks(n,r,t,"add");if(o.length===0){i.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let s of o)await W(e,s)}async function Sm(t){let e=ae(t),{enabled:n}=await se(e);if(n.length===0){i.dim("Kh\xF4ng c\xF3 tool n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await Ks(n,[],t,"remove");if(r.length===0){i.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await Am({message:`G\u1EE1 ${r.length} tool: ${r.join(", ")}?`,default:!0})){i.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 tool n\xE0o.");return}for(let s of r)await qn(e,s)}async function Tm(t){let e=ae(t),n=await ie(e),r=await Et(e);if(n.length===0&&Object.keys(r.tools).length===0){i.dim("Kh\xF4ng c\xF3 tool n\xE0o (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let o=[...new Set([...n,...Object.keys(r.tools)])].sort();i.info("Tools:"),i.plain(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let s of o){let a=n.includes(s)?"yes":"no",c=r.tools[s],l=c?.enabled?"yes":"no",u=c?.version??"-";i.plain(` ${s.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${u}`)}}function Vs(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD tool toggle (prompt-scoring, ...) b\u1EADt/t\u1EAFt nguy\xEAn kh\u1ED1i");e.command("enable <name>").description("B\u1EADt tool: merge manifest v\xE0o settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let o=ae(r);await W(o,n)||process.exit(1)}),e.command("disable <name>").description("T\u1EAFt tool: r\xFAt \u0111\xFAng entry manifest ra kh\u1ECFi settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let o=ae(r);await qn(o,n)}),e.command("list").description("Li\u1EC7t k\xEA tool: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await Tm(n)}),e.command("add").description("Ch\u1ECDn (multi-select) tool available \u0111\u1EC3 c\xE0i").option("--target <path>","Workspace root (default: cwd)").option("--all","C\xE0i h\u1EBFt tool available (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Cm(n)}),e.command("remove").description("Ch\u1ECDn (multi-select) tool \u0111ang b\u1EADt \u0111\u1EC3 g\u1EE1").option("--target <path>","Workspace root (default: cwd)").option("--all","G\u1EE1 h\u1EBFt tool \u0111ang b\u1EADt (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Sm(n)})}import{relative as Dm}from"path";import{confirm as jm}from"@inquirer/prompts";import Hm from"boxen";import{cp as an,mkdir as Bs,writeFile as $m}from"fs/promises";import{homedir as Pm}from"os";import{basename as Em,join as rt}from"path";var Rm=rt(Pm(),".avatar","uninstall-backups");async function Ws(t,e,n){let r=Em(t),o=new Date().toISOString().replace(/[:.]/g,"-"),s=rt(Rm,`${r}-${o}`);if(await Bs(s,{recursive:!0,mode:448}),e.claudeDir&&await an(e.claudeDir,rt(s,".claude"),{recursive:!0}),e.claudeMd&&await an(e.claudeMd,rt(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=rt(s,"hooks");await Bs(c,{recursive:!0}),e.postMergeHook&&await an(e.postMergeHook,rt(c,"post-merge")),e.prePushHook&&await an(e.prePushHook,rt(c,"pre-push"))}let a={projectName:r,projectPath:t,timestamp:o,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await $m(rt(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as _m}from"fs";import{join as ot}from"path";function it(t){return _m(t)?t:null}function qs(t){let e=it(ot(t,".claude")),n=it(ot(t,"CLAUDE.md")),r=it(ot(t,".git","hooks","post-merge")),o=it(ot(t,".git","modules","src","hooks","pre-push")),s=it(ot(t,".gitignore")),a=it(ot(t,".gitmodules")),c=it(ot(t,"notes")),l=it(ot(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||o),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:o,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as zs,rm as st,writeFile as Js}from"fs/promises";async function Ys(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let s of o)s!=="pack"&&await st(r(t.claudeDir,s),{recursive:!0,force:!0})}else await st(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await st(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await st(t.postMergeHook,{force:!0}),t.prePushHook&&await st(t.prePushHook,{force:!0})),t.gitignorePath&&await Im(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Mm(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await st(n,{recursive:!0,force:!0})}}async function Im(t){let e=await zs(t,"utf8"),n=e.indexOf(br),r=e.indexOf(We);if(n===-1||r===-1)return;let o=e.slice(0,n),s=e.slice(r+We.length),a=`${o.trimEnd()}
|
|
146
|
+
${s.trimStart()}`.trim();a.length===0?await st(t,{force:!0}):await Js(t,`${a}
|
|
147
|
+
`,"utf8")}async function Mm(t,e){let r=(await zs(t,"utf8")).split(`
|
|
148
|
+
`),o=[],s=!1;for(let c of r){if(c.trim().startsWith("[submodule")&&c.includes(e)){s=!0;continue}s&&c.trim().startsWith("[")&&(s=!1),s||o.push(c)}let a=o.join(`
|
|
149
|
+
`).trim();a.length===0?await st(t,{force:!0}):await Js(t,`${a}
|
|
150
|
+
`,"utf8")}h();h();import{spawnSync as Rr}from"child_process";var Xs=5e3,Om=120*1e3,Nm=[{name:"gitnexus",manager:"npm"},{name:"gitleaks",manager:"brew"},{name:"trivy",manager:"brew"}];function Lm(t){let e=Rr("which",[t],{encoding:"utf8",timeout:Xs});return!e.error&&e.status===0&&(e.stdout||"").trim().length>0}function Gm(t){let e=Rr(t,["--version"],{encoding:"utf8",timeout:Xs});return!e.error&&e.status===0}function Um(t){if(!Lm(t.name))return"skipped";if(!Gm(t.manager))return i.warn(` ! ${t.name}: kh\xF4ng c\xF3 ${t.manager} \u0111\u1EC3 g\u1EE1 \u2014 b\u1ECF qua (g\u1EE1 tay n\u1EBFu c\u1EA7n).`),"skipped";let e=t.manager==="brew"?["uninstall",t.name]:["uninstall","-g",t.name],n=Rr(t.manager,e,{stdio:["ignore","pipe","pipe"],timeout:Om,encoding:"utf8"});return n.status!==0?(i.warn(` ! G\u1EE1 ${t.name} th\u1EA5t b\u1EA1i: ${(n.stderr||"").slice(0,120)} \u2014 b\u1ECF qua.`),"failed"):"removed"}function Qs(){i.info("G\u1EE1 global tools avatar \u0111\xE3 c\xE0i (gitnexus, gitleaks, trivy)...");let t=[],e=[];for(let n of Nm){let r=Um(n);r==="removed"?t.push(n.name):r==="skipped"&&e.push(n.name)}t.length&&i.success(` \u2713 \u0110\xE3 g\u1EE1: ${t.join(", ")}`),e.length&&i.dim(` - B\u1ECF qua (ch\u01B0a c\xE0i): ${e.join(", ")}`)}function Zs(t){t.command("uninstall").description("G\u1EE1 Avatar kh\u1ECFi project \u2014 backup t\u1EF1 \u0111\u1ED9ng (M11)").option("--yes","Skip confirm prompt").option("--no-backup","Kh\xF4ng t\u1EA1o backup tr\u01B0\u1EDBc khi x\xF3a (nguy hi\u1EC3m)").option("--keep-submodule","Gi\u1EEF submodule .claude/pack/").option("--keep-hooks","Gi\u1EEF git hooks post-merge, pre-push").option("--dry-run","Hi\u1EC3n th\u1ECB danh s\xE1ch s\u1EBD x\xF3a, kh\xF4ng th\u1EF1c thi").action(async e=>{try{await Fm(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Fm(t){let e=process.cwd(),n=qs(e);if(!n.hasAnyArtifact){i.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Km(e,n,t),t.dryRun){i.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await jm({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){i.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await Ws(e,n,O()),i.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await Ys(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),Qs(),await x("uninstall",`project=${e},backup=${r??"skipped"}`),Vm(r)}function Km(t,e,n){i.info(`Project: ${t}`),i.plain(""),i.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&i.plain(` ${p.red("\u2717")} ${Dm(t,e.claudeDir)||".claude/"}`),e.claudeMd&&i.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&i.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&i.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&i.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&i.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),i.plain(` ${p.red("\u2717")} Global tools (G\u1EE0 KH\u1ECEI M\xC1Y): gitnexus, gitleaks, trivy`),i.plain(` ${p.dim("(d\xF9ng chung m\u1ECDi workspace \u2014 workspace kh\xE1c s\u1EBD t\u1EF1 c\xE0i l\u1EA1i khi sync)")}`),i.plain(""),i.plain("Kh\xF4ng \u0111\u1EE5ng:"),i.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),i.plain(` ${p.green("\u2713")} Git history`),i.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),i.plain(` ${p.green("\u2713")} Secrets trong keychain`),i.plain("")}function Vm(t){let e=[`${p.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${p.dim("Backup:")} ${t}`),e.push(` ${p.dim("Restore:")} ${p.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${Hm(e.join(`
|
|
149
151
|
`),{padding:1,borderStyle:"round"})}
|
|
150
|
-
`)}import{spawnSync as
|
|
151
|
-
`).map(r=>r.trim()).filter(Boolean).at(-1)??"";return/^\d+\.\d+\.\d+(-[\w.]+)?$/.test(n)?n:null}function
|
|
152
|
-
${
|
|
152
|
+
`)}import{spawnSync as ta}from"child_process";var ce="@nalvietnam/avatar-cli";function ea(){return j()==="win32"}function Bm(t){return["view",`${ce}@${t}`,"version"]}function Wm(t){return["install","-g",`${ce}@${t}`]}function qm(t){let n=t.split(`
|
|
153
|
+
`).map(r=>r.trim()).filter(Boolean).at(-1)??"";return/^\d+\.\d+\.\d+(-[\w.]+)?$/.test(n)?n:null}function na(t){let e=ta("npm",Bm(t),{encoding:"utf-8",shell:ea()});return e.status!==0||!e.stdout?null:qm(e.stdout)}function ra(t){return ta("npm",Wm(t),{stdio:"inherit",shell:ea()}).status??1}h();function oa(t){t.command("update").description("Update avatar CLI l\xEAn b\u1EA3n m\u1EDBi nh\u1EA5t (npm install -g)").option("--tag <dist-tag>","npm dist-tag mu\u1ED1n theo (latest | next)","latest").action(async e=>{let n=O(),r=na(e.tag);if(r||(i.error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c version cho tag '${e.tag}' t\u1EEB npm registry. Ki\u1EC3m tra m\u1EA1ng/npm r\u1ED3i th\u1EED l\u1EA1i, ho\u1EB7c ch\u1EA1y tay: npm i -g ${ce}@${e.tag}`),process.exit(1)),r===n){i.success(`avatar CLI \u0111\xE3 \u1EDF b\u1EA3n m\u1EDBi nh\u1EA5t c\u1EE7a tag '${e.tag}' (v${n}).`);return}i.info(`Update avatar CLI: v${n} \u2192 v${r} (tag '${e.tag}')...`);let o=ra(e.tag);o!==0&&(i.error(`npm install th\u1EA5t b\u1EA1i (exit ${o}). Th\u1EED ch\u1EA1y tay: npm i -g ${ce}@${e.tag}`),process.exit(o)),i.success(`\u0110\xE3 update avatar CLI \u2192 v${r}. M\u1EDF terminal m\u1EDBi n\u1EBFu version ch\u01B0a \u0111\u1ED5i.`)})}var _r=O(),P=new zm;P.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(_r,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
|
|
154
|
+
${yr({tagline:`v${_r} \xB7 AI harness CLI for NAL Vietnam`})}
|
|
153
155
|
|
|
154
|
-
`);var
|
|
156
|
+
`);var ia=process.argv[2],Jm=ia==="-v"||ia==="--version";Jm&&(It({tagline:`v${_r} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));ws(P);$s(P);Ts(P);Ds(P);Ms(P);Is(P);Ns(P);Fi(P);_s(P);Vs(P);Pi(P);Ps(P);$i(P);zi(P);Rs(P);Go(P);Zs(P);oa(P);P.parseAsync(process.argv).catch(t=>{let e=t instanceof Error?t.message:String(t);process.stderr.write(`\u2717 L\u1ED7i kh\xF4ng x\u1EED l\xFD \u0111\u01B0\u1EE3c: ${e}
|
|
155
157
|
`),process.exit(1)});
|
|
156
158
|
//# sourceMappingURL=index.js.map
|