ai-projects 1.9.0 → 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +3 -0
- package/dist/index.js +46 -46
- package/package.json +1 -1
package/dist/index.d.ts
CHANGED
|
@@ -334,6 +334,7 @@ declare const commands: {
|
|
|
334
334
|
assignee: zod.ZodOptional<zod.ZodString>;
|
|
335
335
|
project: zod.ZodOptional<zod.ZodString>;
|
|
336
336
|
body: zod.ZodOptional<zod.ZodString>;
|
|
337
|
+
log: zod.ZodOptional<zod.ZodString>;
|
|
337
338
|
}, "strip", zod.ZodTypeAny, {
|
|
338
339
|
project?: string | undefined;
|
|
339
340
|
status?: string | undefined;
|
|
@@ -341,6 +342,7 @@ declare const commands: {
|
|
|
341
342
|
name?: string | undefined;
|
|
342
343
|
assignee?: string | undefined;
|
|
343
344
|
body?: string | undefined;
|
|
345
|
+
log?: string | undefined;
|
|
344
346
|
}, {
|
|
345
347
|
project?: string | undefined;
|
|
346
348
|
status?: string | undefined;
|
|
@@ -348,6 +350,7 @@ declare const commands: {
|
|
|
348
350
|
name?: string | undefined;
|
|
349
351
|
assignee?: string | undefined;
|
|
350
352
|
body?: string | undefined;
|
|
353
|
+
log?: string | undefined;
|
|
351
354
|
}>, ZodObject<{
|
|
352
355
|
task: zod.ZodOptional<zod.ZodString>;
|
|
353
356
|
}, "strip", zod.ZodTypeAny, {
|
package/dist/index.js
CHANGED
|
@@ -1,26 +1,26 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import {z as z$1}from'zod';import {parser}from'zod-opts';import'dotenv';import it from'path';import {spawn,exec}from'child_process';import G from'fs/promises';import
|
|
2
|
+
import {z as z$1}from'zod';import {parser}from'zod-opts';import'dotenv';import it from'path';import {spawn,exec}from'child_process';import G from'fs/promises';import b from'lodash';import V from'moment';import {fileURLToPath}from'url';import {inspect}from'util';import Et from'yaml';import ut from'fs';var ze=Object.defineProperty;var f=(t,e)=>()=>(t&&(e=t(t=0)),e);var Be=(t,e)=>{for(var o in e)ze(t,o,{get:e[o],enumerable:true});};var x,J=f(()=>{x={name:"ai-projects",version:"2.0.0",type:"module",description:"Useful CLI for AI agents to create and manage projects, tasks, skills and agents",main:"dist/index.js",types:"dist/index.d.ts",bin:{aip:"dist/index.js"},files:["dist/**/*","README.md","LICENSE"],scripts:{build:"npm run build:map && npm run build:code","build:map":"tsx bin/map-commands.ts","build:code":"tsup --no-dts","build:full":"npm run build:map && tsup","build:check":"tsc --noEmit","build:watch":"npm run build:code -- --watch","build:clean":"rimraf dist/*",start:"dist/index.js",dev:'npm run build:watch -- --onSuccess "dist/index.js"',cli:"npm run build && npm start --",ts:"tsx",vitest:"VITE_CJS_IGNORE_WARNING=true vitest",test:"npm run test:types && npm run test:unit","test:unit":"npm run vitest -- run","test:types:test":"tsc -p tsconfig.test.json --incremental --tsBuildInfoFile test.tsbuildinfo","test:types:bin":"tsc -p bin/tsconfig.json --incremental --tsBuildInfoFile bin.tsbuildinfo","test:types":"npm run test:types:test && npm run test:types:bin","test:full":"npm run lint:full && npm run test",eslint:"eslint --cache",lint:"npm run eslint -- '{src,bin}/**/*.ts'","lint:fix":"npm run lint -- --fix","lint:full":"npm run build:check && npm run test:types:bin && npm run lint:fix","lint:full:silent":"npm run -s lint:full && echo LINT OK",prepare:'[ "$CI" != "true" ] && [ -d node_modules/husky ] && husky || true',prepack:"npm run test:full && npm run build:full","publish:dry":"npm pack --dry-run","version:patch":"npm version patch","version:minor":"npm version minor","version:major":"npm version major","agent:stop":"npm run -s lint:full:silent"},keywords:["ai","aip","project","task","kanban","cli","agent","automation"],author:"Ariel Flesler <aflesler@gmail.com>",license:"MIT",repository:{type:"git",url:"git+https://github.com/flesler/ai-projects.git"},homepage:"https://github.com/flesler/ai-projects#readme",bugs:{url:"https://github.com/flesler/ai-projects/issues"},engines:{node:">=20.0.0"},dependencies:{dotenv:"^17.3.1",lodash:"^4.17.21",moment:"^2.30.1",tslib:"^2.8.1",yaml:"^2.8.0",zod:"^3.25.76","zod-opts":"^1.0.0"},devDependencies:{"@eslint/js":"^10.0.1","@stylistic/eslint-plugin":"^5.7.0","@types/dotenv":"^8.2.3","@types/eslint":"^9.6.1","@types/lodash":"^4.17.23","@types/moment":"^2.13.0","@types/node":"^25.0.9","@types/source-map-support":"^0.5.10","@typescript-eslint/eslint-plugin":"^8.53.0","@typescript-eslint/parser":"^8.53.0",eslint:"^10.0.3","eslint-plugin-import-x":"^4.16.2","eslint-plugin-unused-imports":"^4.4.1","fast-glob":"^3.3.3",husky:"^9.1.7",rimraf:"^6.0.1","source-map-support":"^0.5.21",tsup:"^8.5.1",tsx:"^4.20.3","types-package-json":"^2.0.39",typescript:"^5.8.3",vitest:"^4.0.17"}};});function d(t){let{description:e,options:o=z$1.object({}),args:r,handler:s}=t,i=o.shape,a={};for(let[m,l]of Object.entries(i))a[m]={type:l};let p=parser().options(a);if(r){let m=r.shape,l=Object.entries(m).map(([g,y])=>({name:g,type:y}));l.length>0&&(p=p.args(l));}return e&&(p=p.description(e)),{description:e,options:o,args:r,parser:p,handler:s,cli:m=>s(p.parse(m))}}var k=f(()=>{});var c,P=f(()=>{c={dirs:{PROJECTS:"projects",SKILLS:"skills",AGENTS:"agents",TASKS:"tasks",INPUTS:"inputs",OUTPUTS:"outputs",HOOKS:"hooks",SCRIPTS:"scripts"},files:{MAIN:"main.md",LOG:"log.tsv",SKILL:"SKILL.md"},hookTypes:["pre-create","post-create","pre-complete","post-complete","pre-start","post-start","pre-update","post-update"],languages:["ts","js","sh","py"],targets:["project","task"]};});function Ze(t){return t.startsWith("~")?it.join(process.env.HOME??"",t.slice(1)):t}var W,qe,S,L=f(()=>{P();W=process.env.AIP_HOME?Ze(process.env.AIP_HOME):void 0;if(!W){W=process.cwd();let t=W.split(it.sep);for(let e of [c.dirs.PROJECTS,c.dirs.SKILLS,c.dirs.AGENTS]){let o=t.indexOf(e);if(o!==-1){W=t.slice(0,o).join(it.sep);break}}}qe={AIP_HOME:W,ERROR_LOG:process.env.AIP_ERROR_LOG},S=qe;});var T,n,j=f(()=>{L();T={REPO:fileURLToPath(import.meta.url.replace(/\/(dist|src)\/.*/,"/")),join:(...t)=>it.join(...t),onRepo:(...t)=>T.join(T.REPO,...t),joinHome:(...t)=>T.join(S.AIP_HOME,...t),async ensureDir(t){await G.mkdir(t,{recursive:true});},async write(t,e){await T.ensureDir(it.dirname(t)),await G.writeFile(t,e,"utf8");},async append(t,e){await T.ensureDir(it.dirname(t)),await G.appendFile(t,e,"utf8");},async read(t){return G.readFile(t,"utf8")},async readRepo(...t){return T.read(T.onRepo(...t))},async readMany(...t){return (await Promise.all(t.map(async o=>{let r=await T.fileExists(o)?await T.read(o):"ERROR: File not found";return `# ${o}
|
|
3
3
|
|
|
4
|
-
${r}`}))).filter(Boolean)},async logFiles(...t){let e=await
|
|
4
|
+
${r}`}))).filter(Boolean)},async logFiles(...t){let e=await T.readMany(...t);console.log(e.join(`
|
|
5
5
|
|
|
6
6
|
---
|
|
7
7
|
|
|
8
|
-
`));},dumpCommandMapLines(t,e){let o=e?
|
|
8
|
+
`));},dumpCommandMapLines(t,e){let o=e?b.pick(t,e):t,r=[];for(let[s,i]of T.entriesOf(o))r.push(`${s} {${Object.keys(i).join("|")}}`);return r},async fileExists(t){try{return await G.access(t),!0}catch{return false}},async listDir(t){try{return await G.readdir(t)}catch{return []}},spawn:spawn,cd(t){process.chdir(t);},async exec(t,e){return new Promise(o=>{exec(t,{encoding:"utf8",...e},(r,s,i)=>{let a=r?.code;o({stdout:s??"",stderr:i??"",code:r?typeof a=="number"?a:1:0});});})},noop:()=>{},omitByDeep:(t,e)=>b.isArray(t)?t.map(o=>T.omitByDeep(o,e)):b.isPlainObject(t)?b.mapValues(b.omitBy(t,e),o=>T.omitByDeep(o,e)):t,omitNilsDeep:t=>T.omitByDeep(t,b.isNil),omitUndefinedsDeep:t=>T.omitByDeep(t,b.isUndefined),cloneDeep:t=>b.cloneDeep(t),int:t=>Number.parseInt(t,10),keysOf:t=>Object.keys(t),entriesOf:t=>Object.entries(t),isKeyOf:(t,e)=>!b.isNil(t)&&t in e,typeOf:t=>typeof t,never:{},undef:t=>t??void 0,toNull:t=>t??null,same:t=>t,notNil:t=>{if(b.isNil(t))throw new Error(`Received unexpected ${t} value`);return t},randomInt:(t,e)=>Math.floor(Math.random()*(e-t+1))+t,promise:()=>{let t,e=new Promise((o,r)=>{t={resolve:o,reject:r};});return b.extend(e,t)},isPromise:t=>b.isObject(t)&&"catch"in t,delay:t=>{let e=b.isObject(t)?T.toMS(t):t;return new Promise(o=>setTimeout(o,e))},promiseMap:(t,e)=>Promise.all(t.map(e)),promiseEach:(t,e)=>Promise.all(t.map(e)).then(()=>{}),memoize:t=>b.memoize(t,T.memoizeKey),toMS:t=>V.duration(t).asMilliseconds(),toSecs:t=>V.duration(t).asSeconds(),inspect:(t,e)=>inspect(t,{colors:false,showHidden:false,depth:null,compact:true,breakLength:1/0,...e}),stringify(t,e,o){let r=new WeakSet;return JSON.stringify(t,(s,i)=>{if(b.isObject(i)){if(r.has(i))return "[Circular]";r.add(i);}return b.isFunction(e)?e(s,i):i},o)},dump:t=>T.oneLine(T.stringify(t)),compactWhitespace:t=>t.replace(/\\n/g,`
|
|
9
9
|
`).replace(/\r/g,"").replace(/[ \t]*\n[ \t]*/g,`
|
|
10
10
|
`).replace(/\n{3,}/g,`
|
|
11
11
|
|
|
12
|
-
`).replace(/ {2,}/g," ").trim(),oneLine:t=>t.replace(/\n/g,"\\n").replace(/\t/g,"\\t").replace(/ +/g," ").trim(),fuzzySearch:(t,e)=>
|
|
12
|
+
`).replace(/ {2,}/g," ").trim(),oneLine:t=>t.replace(/\n/g,"\\n").replace(/\t/g,"\\t").replace(/ +/g," ").trim(),fuzzySearch:(t,e)=>b.includes(b.toLower(t||""),b.toLower(e||"")),elapsed:(t,e=Date.now())=>V.duration(V(e).diff(t||0)),iso:t=>V.utc(t).toISOString(),isoDate:t=>T.iso(t).slice(0,10),memoizeKey:(...t)=>t.join(""),exit:(t=0)=>{process.exit(t);},set:(t,e,o)=>(t[e]=o,t),run:async t=>{try{await t(),process.exit(0);}catch(e){console.error(e),process.exit(1);}},isMain:()=>{let t=process.argv[1]||"";return t.includes("/index.")||t.endsWith("/bin/aip")},errorMessage:t=>t instanceof Error?t.message:String(t),slugify:t=>t.toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/^-+|-+$/g,""),matchesSearch:(t,e)=>{let o=e?.toLowerCase().split(/\s+/).filter(Boolean);if(!o?.length)return true;let r=t.toLowerCase();return o.every(s=>r.includes(s))}},n=T;});var eo,rt,oo,Ot=f(()=>{j();eo=" ",rt=class{filename;context;constructor(t){this.filename=t;}log=(t,e,...o)=>{this.filename&&!this.context&&(this.context=`<${oo(this.filename)}>`),this.context&&o.unshift(this.context);let[r,s]=new Date().toISOString().split(/[T.]/);o.unshift(e,r,s);let i=o.map(this.stringify).map(this.sanitize).join(eo);return t?console.error(i):console.log(i),i};stringify=t=>{if(!t)return "-";if(b.isNumber(t))return `${t}`;if(b.isError(t))return n.oneLine(t.message||"");if(b.isFunction(t))return n.oneLine(t.toString());if(b.isObject(t)||b.isArray(t))try{return n.dump(t)}catch(e){return `Failed to stringify: ${e.message}`}return b.toString(t)};sanitize=t=>t.replace(/ {2,}/g," ");debug=(...t)=>{this.log(false,"[DEBUG]",...t);};info=(...t)=>{this.log(false,"[INFO]",...t);};warn=(...t)=>{this.log(true,"[WARN]",...t);};error=(...t)=>{this.log(true,"[ERROR]",...t);}},oo=t=>{let e=t.split(it.sep),[o,r,...s]=e.reverse();if(!s.length)return t;let i=it.basename(o,".js").replace(".ts","");if(i==="index")return r;let a=e.indexOf("commands");if(a!==-1&&a<e.length-1){let p=e.slice(a+1);if(p.length>=2){let[m,l]=p;return `<${m}/${l.replace(".js","").replace(".ts","")}>`}}return i};});var ro,st,M,kt,Z,Dt=f(()=>{j();Ot();ro=new rt,st=t=>{let e=t.trim();if(!e.startsWith("---"))return null;let o=e.match(/^---\n([\s\S]*?)\n---\n?([\s\S]*)$/);return o?{frontmatter:Et.parse(o[1]),content:o[2].trim()}:null},M=async(t,e,o="")=>{let s=`---
|
|
13
13
|
${Et.stringify(e).trim()}
|
|
14
14
|
---
|
|
15
15
|
|
|
16
16
|
${o}`.trim();await n.write(t,s+`
|
|
17
|
-
`);},kt=async(t,e)=>{let o=await n.read(t),r=st(o);if(!r)throw new Error(`No frontmatter found in ${t}`);let s={...r.frontmatter,...e};return await M(t,s,r.content),s},
|
|
18
|
-
Expected at: ${o}`);console.log(e);}});});var
|
|
19
|
-
Expected at: ${o}`);let i=[r,o];await n.logFiles(...i);}});});var
|
|
20
|
-
`,r.length>0||s.length>0){let
|
|
17
|
+
`);},kt=async(t,e)=>{let o=await n.read(t),r=st(o);if(!r)throw new Error(`No frontmatter found in ${t}`);let s={...r.frontmatter,...e};return await M(t,s,r.content),s},Z=async t=>{if(!await n.fileExists(t))return null;let o=await n.read(t),r=st(o);return r?r.frontmatter:(ro.warn(`No frontmatter found in ${t}`),null)};});var z,nt,Rt,Ct,so,no,u,v=f(()=>{P();L();Dt();j();z=(a=>(a.BACKLOG="backlog",a.IN_PROGRESS="in-progress",a.ONGOING="ongoing",a.DONE="done",a.BLOCKED="blocked",a.TO_DO="to-do",a))(z||{}),nt=Object.values(z),Rt=["backlog","in-progress","ongoing"],Ct=[c.dirs.HOOKS,c.dirs.INPUTS,c.dirs.OUTPUTS,c.dirs.SCRIPTS],so=[c.dirs.TASKS,...Ct],no={getProjectDir(t){return n.joinHome(c.dirs.PROJECTS,t)},getTaskDir(t,e){return n.join(this.getProjectDir(t),c.dirs.TASKS,e)},async findTask(t,e){if(e)return {project:e,task:t};let o=await this.listProjects(),r=[];if(await n.promiseEach(o,async s=>{(await this.listTasks(s)).includes(t)&&r.push(s);}),r.length===0)throw new Error(`Task '${t}' not found. Use --project to specify the project.`);if(r.length>1)throw new Error(`Task '${t}' found in multiple projects: ${r.join(", ")}. Use --project to specify which one.`);return {project:r[0],task:t}},getAgentDir(t){return n.join(S.AIP_HOME,c.dirs.AGENTS,t)},async listProjects(){let t=n.join(S.AIP_HOME,c.dirs.PROJECTS),e=await n.listDir(t);return (await n.promiseMap(e,async r=>{let s=this.getProjectDir(r);return await n.fileExists(s)&&r!==c.dirs.AGENTS?r:null})).filter(r=>r!==null).sort()},async listTasks(t){let e=n.join(this.getProjectDir(t),c.dirs.TASKS);return await n.fileExists(e)?await n.listDir(e):[]},async listAgents(){let t=n.join(S.AIP_HOME,c.dirs.AGENTS);return await n.fileExists(t)?await n.listDir(t):[]},async createProject(t,e,o){let r=this.getProjectDir(t);return await n.ensureDir(r),await Promise.all(so.map(s=>n.ensureDir(n.join(r,s)))),await M(n.join(r,c.files.MAIN),e,o),await n.write(n.join(r,c.files.LOG),""),r},async createTask(t,e,o,r){let s=this.getTaskDir(t,e);return await n.ensureDir(s),await Promise.all(Ct.map(i=>n.ensureDir(n.join(s,i)))),await M(n.join(s,c.files.MAIN),o,r),await n.write(n.join(s,c.files.LOG),""),s},async createAgent(t,e){let o=this.getAgentDir(t);return await n.ensureDir(o),await M(n.join(o,c.files.MAIN),e),o},async getProject(t){let e=n.join(this.getProjectDir(t),c.files.MAIN);return await Z(e)},async getTask(t,e){let o=n.join(this.getTaskDir(t,e),c.files.MAIN);return await Z(o)},async getAgent(t){let e=n.join(this.getAgentDir(t),c.files.MAIN);return await Z(e)},async updateProject(t,e){let o=n.join(this.getProjectDir(t),c.files.MAIN);return await kt(o,e)},async updateBody(t,e){let o=await n.read(t),r=st(o);if(!r)throw new Error(`No frontmatter found in ${t}`);await M(t,r.frontmatter,e);},async updateTask(t,e,o){let r=n.join(this.getTaskDir(t,e),c.files.MAIN);if((await Z(r))?.status==="ongoing")throw new Error("Cannot update ongoing task. Should be done by user or update manually");return await kt(r,o)},async ingestTask(t,e){let o=this.getTaskDir(t,e),r=this.getProjectDir(t),s=[n.join(r,c.files.MAIN),n.join(o,c.files.MAIN),n.join(o,c.files.LOG)];await n.logFiles(...s);},async ingestProject(t){let e=this.getProjectDir(t),o=await this.listTasks(t),r=[n.join(e,c.files.MAIN),n.join(e,c.files.LOG),...o.map(s=>n.join(this.getTaskDir(t,s),c.files.MAIN))];await n.logFiles(...r);},async ingestFiles(t){await n.logFiles(...t);}},u=no;});var Nt,It=f(()=>{k();j();v();Nt=d({options:z$1.object({description:z$1.string().describe("Agent description"),status:z$1.string().default("active").describe("Initial status")}),args:z$1.object({name:z$1.string().describe("Agent name")}),handler:async({name:t,description:e,status:o})=>{let r=n.slugify(t);await u.createAgent(r,{name:t,description:e,status:o,created:new Date().toISOString()}),console.log(`Agent created: ${r}`),console.log(` Path: ${u.getAgentDir(r)}`);}});});var _t,Lt=f(()=>{k();_t=d({options:z$1.object({}),handler:async()=>{let t=process.env.CURRENT_AGENT;t||(console.error("CURRENT_AGENT not set"),process.exit(1)),console.log(t);}});});var Ft,Ut=f(()=>{k();v();Ft=d({options:z$1.object({}),handler:async()=>{let t=await u.listAgents(),e=[];for(let o of t){let r=await u.getAgent(o);r&&e.push({slug:o,description:r.description});}if(e.length===0)return console.log("No agents found");console.log("Agents:");for(let o of e)console.log(`- ${o.slug}: ${o.description||""}`);}});});var Ht,Gt=f(()=>{P();k();L();j();Ht=d({description:"Output agent directory path (for cd)",options:z$1.object({}),args:z$1.object({name:z$1.string().describe("Agent name (directory name)")}),handler:async({name:t})=>{let e=n.join(S.AIP_HOME,c.dirs.AGENTS,t),o=n.join(e,c.files.MAIN);if(!await n.fileExists(o))throw new Error(`Agent not found: ${t}
|
|
18
|
+
Expected at: ${o}`);console.log(e);}});});var Kt,Mt=f(()=>{P();k();L();j();Kt=d({description:"Start an agent: read SOUL.md and AGENTS.md content",options:z$1.object({}),args:z$1.object({name:z$1.string().describe("Agent name (directory name)")}),handler:async({name:t})=>{let e=n.join(S.AIP_HOME,c.dirs.AGENTS,t),o=n.join(e,c.files.MAIN),r=n.join(e,"SOUL.md");if(!await n.fileExists(o))throw new Error(`Agent not found: ${t}
|
|
19
|
+
Expected at: ${o}`);let i=[r,o];await n.logFiles(...i);}});});var zt,Bt,po,Jt,Wt=f(()=>{k();zt=t=>{let e=t.shape,o=[];for(let[r,s]of Object.entries(e)){let i=s._def,a="unknown",p,m;i.typeName==="ZodString"?a="string":i.typeName==="ZodNumber"?a="number":i.typeName==="ZodBoolean"?a="boolean":i.typeName==="ZodEnum"?a=`enum: ${i.values.join(" | ")}`:i.typeName==="ZodOptional"?a=Bt(i.innerType):i.typeName==="ZodDefault"&&(a=Bt(i.innerType),m=i.defaultValue()),i.description&&(p=i.description);let l=i.typeName!=="ZodOptional"&&i.typeName!=="ZodDefault";o.push({name:r,type:a,required:l,description:p,defaultValue:m});}return o},Bt=t=>{let e=t._def;return e.typeName==="ZodString"?"string":e.typeName==="ZodNumber"?"number":e.typeName==="ZodBoolean"?"boolean":e.typeName==="ZodEnum"?`enum: ${e.values.join(" | ")}`:"unknown"},po=(t,e,o)=>{let r=zt(o.options),s=o.args?zt(o.args):[],i=o.description,a=`\`${t} ${e}\``;if(i&&(a+=`: ${i}`),a+=`
|
|
20
|
+
`,r.length>0||s.length>0){let m=[];for(let l of s){let g=l.required?`<${l.name}>`:`[${l.name}]`,y=l.description||"";l.defaultValue!==void 0&&(y+=y?` (default: ${l.defaultValue})`:`(default: ${l.defaultValue})`),m.push(`\xB7 ${g}${y?": "+y:""}`);}for(let l of r){let g=l.description||"";l.defaultValue!==void 0&&(g+=g?` (default: ${l.defaultValue})`:`(default: ${l.defaultValue})`),m.push(`\xB7 --${l.name}${g?": "+g:""}`);}a+=m.join(`
|
|
21
21
|
`)+`
|
|
22
22
|
`;}return a+=`
|
|
23
|
-
`,a},
|
|
23
|
+
`,a},Jt=d({description:"Print generated CLI reference from command schemas",options:z$1.object({}),handler:async()=>{let{default:t}=await Promise.resolve().then(()=>(Y(),jt)),e="Prepend `aip` to each command\n\n",o=Object.keys(t).sort();for(let r of o){let s=t[r],i=Object.keys(s).sort();for(let a of i){let p=s[a];e+=po(r,a,p);}}console.log(e);}});});var mo,Vt,Zt=f(()=>{k();mo=`# AIP Hooks
|
|
24
24
|
|
|
25
25
|
Automated scripts that run before/after project/task actions.
|
|
26
26
|
|
|
@@ -200,30 +200,30 @@ DEBUG=1 aip task update --status done
|
|
|
200
200
|
---
|
|
201
201
|
|
|
202
202
|
*Hooks automate workflow enforcement. Pre-hooks guard quality, post-hooks handle cleanup.*
|
|
203
|
-
`,
|
|
203
|
+
`,Vt=d({description:"Hook types, env vars, and patterns",options:z$1.object({}),handler:async()=>{console.log(mo);}});});var A,qt,fo,Yt,Qt=f(()=>{J();P();k();A=Object.keys(x.bin)[0],{LOG:qt}=c.files,fo=`# Quick Start
|
|
204
204
|
|
|
205
205
|
## New Task/Project
|
|
206
206
|
|
|
207
207
|
\`\`\`bash
|
|
208
208
|
# 1. Create project (if needed)
|
|
209
|
-
${
|
|
209
|
+
${A} project create "my-project" --description "What I'm building" --body $'# Goals
|
|
210
210
|
- Build X
|
|
211
211
|
- Solve Y'
|
|
212
212
|
|
|
213
213
|
# 2. Create task (heredoc works too)
|
|
214
|
-
${
|
|
214
|
+
${A} task create my-project "first-task" --description "Start here" --body "$(cat <<< '# Task Details
|
|
215
215
|
- Step 1
|
|
216
216
|
- Step 2')"
|
|
217
217
|
|
|
218
218
|
# 3. Navigate to task
|
|
219
|
-
cd $(${
|
|
219
|
+
cd $(${A} task path first-task)
|
|
220
220
|
|
|
221
221
|
# 4. Start working (implicit from PWD)
|
|
222
|
-
${
|
|
222
|
+
${A} task start
|
|
223
223
|
# \u2192 sets status to in-progress, reads all context to stdout
|
|
224
224
|
|
|
225
225
|
# 5. Work session
|
|
226
|
-
# Log progress: ${
|
|
226
|
+
# Log progress: ${A} log append "API integration complete"
|
|
227
227
|
# Save deliverables to outputs/ (e.g. reports, leads, etc.)
|
|
228
228
|
# Save intermediate/raw (lower value) data to inputs/ (e.g. raw API responses, downloads)
|
|
229
229
|
\`\`\`
|
|
@@ -232,27 +232,27 @@ ${$} task start
|
|
|
232
232
|
|
|
233
233
|
\`\`\`bash
|
|
234
234
|
# Find your tasks (searches all projects), can pass --project or --search to filter
|
|
235
|
-
${
|
|
235
|
+
${A} task list
|
|
236
236
|
|
|
237
237
|
# Navigate to task
|
|
238
|
-
cd $(${
|
|
238
|
+
cd $(${A} task path task-slug)
|
|
239
239
|
|
|
240
240
|
# All commands work implicitly from PWD:
|
|
241
|
-
${
|
|
241
|
+
${A} task start # Start working (sets status to in-progress, reads all context to stdout)
|
|
242
242
|
\`\`\`
|
|
243
243
|
|
|
244
244
|
## Completing
|
|
245
245
|
|
|
246
246
|
\`\`\`bash
|
|
247
247
|
# Mark done (implicit from PWD)
|
|
248
|
-
${
|
|
249
|
-
${
|
|
248
|
+
${A} task update --status done
|
|
249
|
+
${A} log append "if you want to log something extra"
|
|
250
250
|
\`\`\`
|
|
251
251
|
|
|
252
252
|
## Check current task
|
|
253
253
|
|
|
254
254
|
\`\`\`bash
|
|
255
|
-
${
|
|
255
|
+
${A} task current
|
|
256
256
|
\`\`\`
|
|
257
257
|
|
|
258
258
|
**Note:** All task commands (except create) accept an optional task slug, but use PWD if not provided. Navigate with \`cd\` first for smoother workflow.
|
|
@@ -261,10 +261,10 @@ ${$} task current
|
|
|
261
261
|
|
|
262
262
|
$AIP_HOME/projects/slug/
|
|
263
263
|
\u251C\u2500\u2500 main.md # Definition + body
|
|
264
|
-
\u251C\u2500\u2500 ${
|
|
264
|
+
\u251C\u2500\u2500 ${qt} # Log (auto-appended, has key task updates too)
|
|
265
265
|
\u2514\u2500\u2500 tasks/task/
|
|
266
266
|
\u251C\u2500\u2500 main.md # Task definition + body
|
|
267
|
-
\u2514\u2500\u2500 ${
|
|
267
|
+
\u2514\u2500\u2500 ${qt} # Updates
|
|
268
268
|
|
|
269
269
|
**Log format:** TSV with columns: date, time, entityType, slug, action, text
|
|
270
270
|
**Example:** \`2026-03-30<TAB>14:30:00<TAB>task<TAB>api-integration<TAB>log<TAB>API integration complete\`
|
|
@@ -273,15 +273,15 @@ $AIP_HOME/projects/slug/
|
|
|
273
273
|
|
|
274
274
|
\`\`\`bash
|
|
275
275
|
# Get all nouns and verbs
|
|
276
|
-
${
|
|
276
|
+
${A} --help
|
|
277
277
|
|
|
278
278
|
# All verbs for a noun
|
|
279
|
-
${
|
|
279
|
+
${A} <noun> --help
|
|
280
280
|
|
|
281
281
|
# All parameters and --options for a command
|
|
282
|
-
${
|
|
282
|
+
${A} <noun> <verb> --help
|
|
283
283
|
\`\`\`
|
|
284
|
-
`,
|
|
284
|
+
`,Yt=d({description:"Quick start guide for new users",options:z$1.object({}),handler:async()=>{console.log(fo);}});});var go,ko,ho,yo,jo,To,bo,te,Tt,Q,bt,ee,oe=f(()=>{J();P();k();v();go=Object.values(z).join("|"),{PROJECTS:ko,TASKS:ho,HOOKS:yo,INPUTS:jo,OUTPUTS:To,SCRIPTS:bo}=c.dirs,{MAIN:te,LOG:Tt}=c.files,Q=Object.keys(x.bin)[0],bt=`# AIP - Project Management Skill
|
|
285
285
|
|
|
286
286
|
Automated project and task creation with proper structure, front-matter, and file organization.
|
|
287
287
|
|
|
@@ -299,16 +299,16 @@ Streamline creation of new projects and tasks with the correct directory structu
|
|
|
299
299
|
## File Structure Overview
|
|
300
300
|
|
|
301
301
|
\`\`\`
|
|
302
|
-
$AIP_HOME/${
|
|
303
|
-
\u251C\u2500\u2500 ${
|
|
302
|
+
$AIP_HOME/${ko}/{project-slug}/
|
|
303
|
+
\u251C\u2500\u2500 ${te} # Goals + body (YAML frontmatter + markdown content)
|
|
304
304
|
\u251C\u2500\u2500 ${Tt} # Chronological log (TSV: date, time, entityType, slug, action, text)
|
|
305
|
-
\u251C\u2500\u2500 ${
|
|
306
|
-
\u251C\u2500\u2500 ${
|
|
307
|
-
\u251C\u2500\u2500 ${
|
|
308
|
-
\u251C\u2500\u2500 ${
|
|
309
|
-
\u2514\u2500\u2500 ${
|
|
305
|
+
\u251C\u2500\u2500 ${yo}/ # pre|post-{create,start,update,complete}.*
|
|
306
|
+
\u251C\u2500\u2500 ${jo}/ # Intermediate/raw data (e.g. API responses, downloads)
|
|
307
|
+
\u251C\u2500\u2500 ${To}/ # Deliverables (e.g. reports, leads, etc.)
|
|
308
|
+
\u251C\u2500\u2500 ${bo}/ # Automation
|
|
309
|
+
\u2514\u2500\u2500 ${ho}/
|
|
310
310
|
\u2514\u2500\u2500 {task-slug}/
|
|
311
|
-
\u251C\u2500\u2500 ${
|
|
311
|
+
\u251C\u2500\u2500 ${te} # Task definition + body
|
|
312
312
|
\u251C\u2500\u2500 ${Tt} # Activity log
|
|
313
313
|
\u2514\u2500\u2500 ... (same structure)
|
|
314
314
|
\`\`\`
|
|
@@ -320,7 +320,7 @@ $AIP_HOME/${go}/{project-slug}/
|
|
|
320
320
|
name: Clear Title
|
|
321
321
|
description: One-line summary
|
|
322
322
|
assignee: agent-name
|
|
323
|
-
status: ${
|
|
323
|
+
status: ${go}
|
|
324
324
|
created: YYYY-MM-DD
|
|
325
325
|
---
|
|
326
326
|
\`\`\`
|
|
@@ -337,10 +337,10 @@ Markdown content after frontmatter. See \`aip help quickstart\` for examples of
|
|
|
337
337
|
## For Agents
|
|
338
338
|
|
|
339
339
|
Typical workflow: read \`main.md\` for context, check \`${Tt}\` for history, work, and log progress with \`aip log append\`.
|
|
340
|
-
`,
|
|
340
|
+
`,ee=d({description:"Project management skill for creating and managing projects/tasks",options:z$1.object({mode:z$1.enum(["md","claude","hermes"]).default("md").describe("Output format: md (no header), claude (name+description), hermes (full frontmatter)")}),handler:async t=>{let e=Object.keys(x.bin)[0];if(t.mode==="md"){console.log(bt);return}if(t.mode==="claude"){console.log(`---
|
|
341
341
|
name: ${e}
|
|
342
342
|
description: ${x.description}
|
|
343
|
-
---`),console.log(),console.log(
|
|
343
|
+
---`),console.log(),console.log(bt);return}if(t.mode==="hermes"){console.log(`---
|
|
344
344
|
name: ${e}
|
|
345
345
|
description: ${x.description}
|
|
346
346
|
version: ${x.version}
|
|
@@ -349,9 +349,9 @@ license: ${x.license}
|
|
|
349
349
|
metadata:
|
|
350
350
|
hermes:
|
|
351
351
|
tags: [${x.keywords.join(", ")}]
|
|
352
|
-
---`),console.log(),console.log(
|
|
353
|
-
`)),console.log(),console.log("Run `aip help quickstart` for a quick start guide"),console.log("Run `aip help skill` for a broader overview");else {let s=r[0]??`${t} {?}`;console.log(o,`Usage: aip ${s} [options]`);}}});});var Pt,
|
|
354
|
-
`)},getTargetDir(t,e=process.cwd()){let o=this.getCurrentContext(e);if(t==="project"){let r=this.getProjectFromPwd(e);if(!r)throw new Error("Not in a project directory. Use --project flag or cd into a project.");let s=n.joinHome(c.dirs.PROJECTS,r);return {targetDir:s,projectDir:s,entityType:"project"}}if(t==="task"){if(!o.project||!o.task)throw new Error("Not in a task directory. Use --project and --task flags or cd into a task.");let r=n.joinHome(c.dirs.PROJECTS,o.project);return {targetDir:n.join(r,c.dirs.TASKS,o.task),projectDir:r,entityType:"task"}}if(o.task&&o.project){let r=n.joinHome(c.dirs.PROJECTS,o.project);return {targetDir:n.join(r,c.dirs.TASKS,o.task),projectDir:r,entityType:"task"}}if(o.project){let r=n.joinHome(c.dirs.PROJECTS,o.project);return {targetDir:r,projectDir:r,entityType:"project"}}throw new Error("Not in a project or task directory. Specify --target flag.")}},h=
|
|
352
|
+
---`),console.log(),console.log(bt);return}}});});var re,se=f(()=>{J();k();j();re=d({description:"Print compact CLI usage (all nouns, or one noun if name is given)",options:z$1.object({}),args:z$1.object({name:z$1.string().optional().describe("Noun (e.g. task) to list verbs for")}),handler:async({name:t})=>{let{default:e}=await Promise.resolve().then(()=>(Y(),jt)),o=`${x.name} ${x.version} -`,r=n.dumpCommandMapLines(e,t);if(!t)console.log(o,"Usage: aip <noun> <verb> [options]"),console.log(r.map(s=>`-> ${s}`).join(`
|
|
353
|
+
`)),console.log(),console.log("Run `aip help quickstart` for a quick start guide"),console.log("Run `aip help skill` for a broader overview");else {let s=r[0]??`${t} {?}`;console.log(o,`Usage: aip ${s} [options]`);}}});});var Pt,Po,h,D=f(()=>{P();j();Pt=ut.realpathSync(n.joinHome(c.dirs.PROJECTS)),Po={getProjectFromPwd(t=process.cwd()){let e=it.relative(Pt,t);if(!e||e.startsWith(".."))return null;let o=e.split(it.sep).filter(Boolean);return o.length===0?null:o[0]},getTaskFromPwd(t=process.cwd()){let e=it.relative(Pt,t);if(!e||e.startsWith(".."))return null;let o=e.split(it.sep).filter(Boolean);return o.length<3||o[1]!==c.dirs.TASKS?null:o[2]},getCurrentContext(t=process.cwd()){return {project:this.getProjectFromPwd(t),task:this.getTaskFromPwd(t)}},requireProject(t=process.cwd()){let e=this.getProjectFromPwd(t);if(!e)throw new Error(`Not in a project directory. PWD: ${t}, projects: ${Pt}`);return e},requireTask(t=process.cwd()){let e=this.getCurrentContext(t);if(!e.project||!e.task)throw new Error(`Not in a task directory. PWD: ${t}`);return {project:e.project,task:e.task}},exportContext(t=process.cwd()){let e=this.getCurrentContext(t),o=[];return e.project&&o.push(`export CURRENT_PROJECT="${e.project}"`),e.task&&o.push(`export CURRENT_TASK="${e.task}"`),process.env.CURRENT_AGENT&&o.push(`export CURRENT_AGENT="${process.env.CURRENT_AGENT}"`),o.join(`
|
|
354
|
+
`)},getTargetDir(t,e=process.cwd()){let o=this.getCurrentContext(e);if(t==="project"){let r=this.getProjectFromPwd(e);if(!r)throw new Error("Not in a project directory. Use --project flag or cd into a project.");let s=n.joinHome(c.dirs.PROJECTS,r);return {targetDir:s,projectDir:s,entityType:"project"}}if(t==="task"){if(!o.project||!o.task)throw new Error("Not in a task directory. Use --project and --task flags or cd into a task.");let r=n.joinHome(c.dirs.PROJECTS,o.project);return {targetDir:n.join(r,c.dirs.TASKS,o.task),projectDir:r,entityType:"task"}}if(o.task&&o.project){let r=n.joinHome(c.dirs.PROJECTS,o.project);return {targetDir:n.join(r,c.dirs.TASKS,o.task),projectDir:r,entityType:"task"}}if(o.project){let r=n.joinHome(c.dirs.PROJECTS,o.project);return {targetDir:r,projectDir:r,entityType:"project"}}throw new Error("Not in a project or task directory. Specify --target flag.")}},h=Po;});var ne,ie=f(()=>{k();j();D();P();ne=d({options:z$1.object({lang:z$1.enum(c.languages).default("ts").describe("Language"),target:z$1.enum(c.targets).default("project").describe("Target level")}),args:z$1.object({type:z$1.enum(c.hookTypes).describe("Hook type (e.g., pre-create, post-complete)")}),handler:async({type:t,lang:e,target:o})=>{let r=e,s=`.${r}`,{targetDir:i,entityType:a}=h.getTargetDir(o),p=n.join(i,c.dirs.HOOKS);await n.ensureDir(p);let m=n.join(p,`${t}${s}`);if(await n.fileExists(m))throw new Error(`Hook already exists: ${m}`);let g;if(r==="ts")g=`#!/usr/bin/env tsx
|
|
355
355
|
/** ${t} hook for ${a} */
|
|
356
356
|
|
|
357
357
|
import { env } from 'node:process'
|
|
@@ -398,8 +398,8 @@ if os.environ.get('TASK_SLUG'):
|
|
|
398
398
|
|
|
399
399
|
# Exit with non-zero to prevent action (for pre-hooks)
|
|
400
400
|
# sys.exit(1)
|
|
401
|
-
`;else throw new Error(`Unsupported language: ${r}`);await n.write(
|
|
402
|
-
`},async append(t,e,o,r,s){let i=n.join(t,c.files.LOG),a=this.formatEntry(e,o,r,s);await n.append(i,a);},async read(t){let e=n.join(t,c.files.LOG);return await n.fileExists(e)?await n.read(e):""},async task(t,e,o){let r=h.getCurrentContext();if(!r.project||!r.task)throw new Error("Not in a task directory");let s=n.joinHome(c.dirs.PROJECTS,r.project,c.dirs.TASKS,r.task);await this.append(s,"task",t,"log",o?`${o}: ${e}`:e);},async project(t,e,o){let r=h.getProjectFromPwd();if(!r)throw new Error("Not in a project directory");let s=n.joinHome(c.dirs.PROJECTS,r);await this.append(s,"project",t,"log",o?`${o}: ${e}`:e);}},
|
|
403
|
-
Expected at: ${o}`);await n.logFiles(o);}});});var
|
|
404
|
-
`)+
|
|
405
|
-
`);}function
|
|
401
|
+
`;else throw new Error(`Unsupported language: ${r}`);await n.write(m,g);let{chmod:y}=await import('fs/promises');await y(m,493),console.log(`Hook created: ${m}`);}});});var vo,$,B=f(()=>{P();j();vo={async findHooks(t,e){let o=n.join(t,c.dirs.HOOKS);if(!await n.fileExists(o))return [];let s=await n.listDir(o),i=`${e}.`;return s.filter(a=>a.startsWith(i)).map(a=>n.join(o,a))},async executeHook(t,e,o){if(!await n.fileExists(t))return true;let s=e.project?n.joinHome(c.dirs.PROJECTS,e.project):o,i=e.task?n.join(s,c.dirs.TASKS,e.task):"",a={...process.env,HOOK_TYPE:e.action,ENTITY_TYPE:e.entityType,TARGET_DIR:o,PROJECT_DIR:s,...e.project?{PROJECT_SLUG:e.project}:{},...e.task?{TASK_SLUG:e.task,TASK_DIR:i}:{}};return new Promise(p=>{let m=spawn(t,[],{stdio:"inherit",env:a,cwd:o,shell:false});m.on("close",(l,g)=>{l===0?p(true):(console.error(`Hook ${it.basename(t)} failed (code=${l??g})`),p(false));}),m.on("error",l=>{console.error(`Failed to execute hook ${t}: ${l.message}`),p(false);});})},async runHooks(t,e,o){let r=await this.findHooks(t,e),s=e.startsWith("pre-");for(let i of r)if(!await this.executeHook(i,o,t)&&s)return false;return true},async runHooksForContext(t,e,o,r){return !(!await this.runHooks(t,o,r)||e&&!await this.runHooks(e,o,r))}},$=vo;});var ae,ce=f(()=>{k();B();D();P();ae=d({options:z$1.object({target:z$1.enum(c.targets).optional().describe("Target level")}),args:z$1.object({type:z$1.enum(c.hookTypes).describe("Hook type to run")}),handler:async({type:t,target:e})=>{let{targetDir:o,entityType:r}=h.getTargetDir(e),s={action:t,entityType:r,project:h.getProjectFromPwd()||void 0,task:h.getTaskFromPwd()||void 0};await $.runHooks(o,t,s)||(console.error(`Hook ${t} failed`),process.exit(1)),console.log(`Hook ${t} completed successfully`);}});});var Ao,w,F=f(()=>{P();D();j();Ao={formatEntry(t,e,o,r){let s=new Date,i=s.getFullYear(),a=String(s.getMonth()+1).padStart(2,"0"),p=String(s.getDate()).padStart(2,"0"),m=s.toLocaleTimeString("en-US",{hour12:false,hour:"2-digit",minute:"2-digit",second:"2-digit"});return `${`${i}-${a}-${p}`} ${m} ${t} ${e} ${o} ${r}
|
|
402
|
+
`},async append(t,e,o,r,s){let i=n.join(t,c.files.LOG),a=this.formatEntry(e,o,r,s);await n.append(i,a);},async read(t){let e=n.join(t,c.files.LOG);return await n.fileExists(e)?await n.read(e):""},async task(t,e,o){let r=h.getCurrentContext();if(!r.project||!r.task)throw new Error("Not in a task directory");let s=n.joinHome(c.dirs.PROJECTS,r.project,c.dirs.TASKS,r.task);await this.append(s,"task",t,"log",o?`${o}: ${e}`:e);},async project(t,e,o){let r=h.getProjectFromPwd();if(!r)throw new Error("Not in a project directory");let s=n.joinHome(c.dirs.PROJECTS,r);await this.append(s,"project",t,"log",o?`${o}: ${e}`:e);}},w=Ao;});var pe,le=f(()=>{P();D();k();j();F();pe=d({description:"Append a custom entry to log.tsv",options:z$1.object({project:z$1.string().optional().describe("Project slug (uses current project from $PWD if not provided)"),task:z$1.string().optional().describe("Task slug (uses current task from $PWD if not provided)")}),args:z$1.object({text:z$1.string().describe("Text to append")}),handler:async({project:t,task:e,text:o})=>{let r=h.getCurrentContext(),s=t||r.project;if(!s)throw new Error("No project specified. Use --project or run from a project/task directory");let i=n.joinHome(c.dirs.PROJECTS,s);if(e||r.task){let a=e||r.task;if(!a)throw new Error("No task specified");let p=n.join(i,c.dirs.TASKS,a);await w.append(p,"task",a,"log",o),console.log(`Appended to task ${a} log.tsv`);}else await w.append(i,"project",s,"log",o),console.log(`Appended to project ${s} log.tsv`);}});});var de,me=f(()=>{P();D();k();j();F();de=d({description:"Read log.tsv history",options:z$1.object({project:z$1.string().optional().describe("Project slug (uses current project from $PWD if not provided)"),task:z$1.string().optional().describe("Task slug (uses current task from $PWD if not provided)")}),args:z$1.object({}),handler:async({project:t,task:e})=>{let o=h.getCurrentContext(),r=t||o.project;if(!r)throw new Error("No project specified. Use --project or run from a project/task directory");let s=n.joinHome(c.dirs.PROJECTS,r);if(e||o.task){let i=e||o.task;if(!i)throw new Error("No task specified");let a=n.join(s,c.dirs.TASKS,i),p=await w.read(a);console.log(p||`No log.tsv found for task ${i}`);}else {let i=await w.read(s);console.log(i||`No log.tsv found for project ${r}`);}}});});var ue,fe=f(()=>{k();j();F();v();ue=d({description:"Create a new project with name, description, optional status, assignee and body",options:z$1.object({description:z$1.string().describe("Project description"),status:z$1.string().default("active").describe("Initial status"),assignee:z$1.string().optional().describe("Assignee agent slug"),body:z$1.string().optional().describe("Initial body/content (markdown)")}),args:z$1.object({name:z$1.string().describe("Project name")}),handler:async({name:t,description:e,status:o,assignee:r,body:s})=>{let i=n.slugify(t),a=u.getProjectDir(i);await u.createProject(i,{name:t,description:e,status:o,assignee:r,created:new Date().toISOString()},s),await w.append(a,"project",i,"created",`${t} > status is ${o}`),console.log(`Project created: ${i}. Path: ${a}`);}});});var ge,ke=f(()=>{k();D();ge=d({options:z$1.object({}),handler:async()=>{let t=h.getProjectFromPwd();t||(console.error("Not in a project directory"),process.exit(1)),console.log(t);}});});var ye,je=f(()=>{k();v();ye=d({description:"List all projects, optionally filtered by status",options:z$1.object({status:z$1.string().optional().describe("Filter by status")}),handler:async({status:t})=>{let e=await u.listProjects(),o=[];for(let r of e){let s=await u.getProject(r);s&&(t&&s.status!==t||o.push({slug:r,status:s.status,assignee:s.assignee}));}if(o.length===0)return console.log("No projects found");console.log("Projects:");for(let r of o)console.log(`- ${r.slug}: ${r.status||""} ${r.assignee||""}`);}});});var Te,be=f(()=>{k();v();Te=d({description:"Output project directory path (for cd)",options:z$1.object({}),args:z$1.object({name:z$1.string().describe("Project name (slug)")}),handler:async({name:t})=>{let e=u.getProjectDir(t);console.log(e);}});});var we,Pe=f(()=>{P();D();k();B();j();v();F();we=d({description:"Update project properties: name, description, status, assignee, or replace body",options:z$1.object({name:z$1.string().optional().describe("New name"),description:z$1.string().optional().describe("New description"),status:z$1.string().optional().describe("New status"),assignee:z$1.string().optional().describe("New assignee"),body:z$1.string().optional().describe("Replace entire body/content (markdown)")}),args:z$1.object({project:z$1.string().optional().describe("Project slug (defaults to current project from $PWD)")}),handler:async({project:t,name:e,description:o,status:r,assignee:s,body:i})=>{let a=t||h.getProjectFromPwd();if(!a)throw new Error("No project specified and not in a project directory");let p=u.getProjectDir(a),m=n.join(p,c.files.MAIN);if(!await $.runHooks(p,"pre-update",{action:"pre-update",entityType:"project"}))throw new Error("Pre-update hook failed, aborting update");let g={};if(e&&(g.name=e),o&&(g.description=o),r&&(g.status=r),s&&(g.assignee=s),Object.keys(g).length===0&&!i){console.log("No updates provided");return}if(Object.keys(g).length>0){await u.updateProject(a,g);let y=Object.entries(g).map(([O,E])=>`${O}=${E}`).join(", ");await w.append(p,"project",a,"updated",y);}i!==void 0&&(await u.updateBody(m,i),await w.append(p,"project",a,"updated",`body replaced (${i.length} chars)`)),await $.runHooks(p,"post-update",{action:"post-update",entityType:"project"}),console.log(`Project ${a} updated`);}});});var Se,xe=f(()=>{P();k();L();j();Se=d({description:"Read a skill's SKILL.md file into the console for agent context",options:z$1.object({}),args:z$1.object({name:z$1.string().describe("Skill name (directory name)")}),handler:async({name:t})=>{let e=n.join(S.AIP_HOME,c.dirs.SKILLS,t),o=n.join(e,c.files.SKILL);if(!await n.fileExists(o))throw new Error(`Skill not found: ${t}
|
|
403
|
+
Expected at: ${o}`);await n.logFiles(o);}});});var ve,Ae=f(()=>{k();B();j();v();F();ve=d({description:"Create a new task with assignee, initial status and optional body",options:z$1.object({description:z$1.string().optional().describe("Task description"),assignee:z$1.string().optional().describe("Assignee agent slug"),status:z$1.string().default("backlog").describe("Initial status"),body:z$1.string().optional().describe("Initial body/content (markdown)")}),args:z$1.object({project:z$1.string().describe("Project slug"),name:z$1.string().describe("Task name")}),handler:async({project:t,name:e,description:o,assignee:r,status:s,body:i})=>{let a=n.slugify(e),p=u.getProjectDir(t),m=u.getTaskDir(t,a);if(!await $.runHooksForContext(p,m,"pre-create",{action:"pre-create",entityType:"task",project:t}))throw new Error("Pre-create hook failed, aborting task creation");await u.createTask(t,a,{name:e,description:o,assignee:r,status:s,created:new Date().toISOString()},i),await w.append(m,"task",a,"created",e),await w.append(p,"task",a,"created",`${a} > status is ${s}`),await $.runHooksForContext(p,m,"post-create",{action:"post-create",entityType:"task",project:t}),console.log(`Task created: ${a}`),console.log(` Project: ${t}`),console.log(` Path: ${m}`);}});});var $e,Oe=f(()=>{k();D();$e=d({description:"Get the current task slug from PWD",options:z$1.object({}),handler:async()=>{let t=h.getTaskFromPwd();t||(console.error("Not in a task directory"),process.exit(1)),console.log(t);}});});var Ee,De=f(()=>{D();k();v();Ee=d({description:"Output full task context (main.md, log.tsv) for ingestion by agents",options:z$1.object({project:z$1.string().optional().describe("Project slug (searches all projects if not provided)")}),args:z$1.object({task:z$1.string().optional().describe("Task slug (default: from $PWD)")}),handler:async({project:t,task:e})=>{let o=h.getCurrentContext(),r=e??o.task;if(!r)throw new Error("No task specified or inferred. cd into task dir before: `cd $(aip task path my-task) && aip task ingest`");let{project:s}=await u.findTask(r,t);await u.ingestTask(s,r);}});});var Re,Ce=f(()=>{k();j();v();Re=d({description:"List tasks, optionally filtered by project, status, assignee, or search query",options:z$1.object({project:z$1.string().optional().describe("Project slug (searches all projects if not provided)"),statuses:z$1.array(z$1.string()).default([]).describe("Filter by statuses (multiple allowed)"),assignee:z$1.string().optional().describe("Filter by assignee"),all:z$1.boolean().default(false).describe("Include all tasks (including done/blocked)"),search:z$1.string().optional().describe("Search query (matches task slug, case-insensitive, multi-part AND, any order)")}),handler:async({project:t,statuses:e,assignee:o,all:r,search:s})=>{if(e&&e.length>0){for(let l of e)if(!nt.includes(l))throw new Error(`Invalid status: ${l}. Valid values: ${nt.join(", ")}`)}let i;e&&e.length>0?i=e:r?i=nt:i=Rt;let a=t?[t]:await u.listProjects(),p={};await n.promiseEach(a,async l=>{let g=await u.listTasks(l),y=[];await n.promiseEach(g,async O=>{let E=await u.getTask(l,O);E&&(i.length>0&&!i.includes(E.status||"")||o&&E.assignee!==o||n.matchesSearch(O,s)&&y.push({slug:O,description:E.description}));}),y.length>0&&(p[l]=y.sort((O,E)=>O.slug.localeCompare(E.slug)));});let m=Object.keys(p).sort();if(m.length===0)return console.log("No tasks found");for(let l of m){console.log(`${l}:`);for(let g of p[l])console.log(`- ${g.slug}: ${g.description||""}`);}}});});var Ne,Ie=f(()=>{k();v();Ne=d({description:"Output task directory path (for cd)",options:z$1.object({project:z$1.string().optional().describe("Project slug (searches all projects if not provided)")}),args:z$1.object({task:z$1.string().describe("Task slug")}),handler:async({project:t,task:e})=>{let{project:o,task:r}=await u.findTask(e,t),s=u.getTaskDir(o,r);console.log(s);}});});var dt,vt=f(()=>{P();D();k();B();j();F();v();dt=d({description:"Update task properties: name, description, status, assignee, or replace body",options:z$1.object({name:z$1.string().optional().describe("New name"),description:z$1.string().optional().describe("New description"),status:z$1.string().optional().describe("New status"),assignee:z$1.string().optional().describe("New assignee"),project:z$1.string().optional().describe("Project slug (searches all projects if not provided)"),body:z$1.string().optional().describe("Replace entire body/content (markdown)"),log:z$1.string().optional().describe("Log message to include in log.tsv")}),args:z$1.object({task:z$1.string().optional().describe("Task slug (default: from $PWD)")}),handler:async({project:t,task:e,name:o,description:r,status:s,assignee:i,body:a,log:p})=>{let m=h.getCurrentContext(),l=e??m.task;if(!l)throw new Error("No task specified or inferred. Provide it or cd into task dir before: `cd $(aip task path my-task)`");let{project:g}=await u.findTask(l,t),y=u.getTaskDir(g,l),O=u.getProjectDir(g),E=n.join(y,c.files.MAIN);if(!await $.runHooksForContext(O,y,"pre-update",{action:"pre-update",entityType:"task",project:g,task:l}))throw new Error("Pre-update hook failed, aborting update");let R={};if(o&&(R.name=o),r&&(R.description=r),s&&(R.status=s),i&&(R.assignee=i),Object.keys(R).length===0&&!a&&!p){console.log("No updates provided");return}if(Object.keys(R).length>0){await u.updateTask(g,l,R);let At=Object.entries(R).map(([Ke,Me])=>`${Ke}=${Me}`).join(", ");p&&(At+=` | ${p}`),await w.append(y,"task",l,"updated",At);}else p&&await w.append(y,"task",l,"log",p);a!==void 0&&(await u.updateBody(E,a),await w.append(y,"task",l,"updated",`body replaced (${a.length} chars)`)),R.status&&await w.append(O,"task",l,"updated",`status to ${R.status}`),await $.runHooksForContext(u.getProjectDir(g),y,"post-update",{action:"post-update",entityType:"task",project:g,task:l}),console.log(`Task ${l} updated`);}});});var _e,Le=f(()=>{D();k();B();v();vt();_e=d({description:"Start a task: set status to in-progress, optionally print context",options:z$1.object({project:z$1.string().optional().describe("Project slug (searches all projects if not provided)"),ingest:z$1.boolean().default(true).describe("Also output context for this task")}),args:z$1.object({task:z$1.string().optional().describe("Task slug (default: from $PWD)")}),handler:async({project:t,task:e,ingest:o})=>{let r=h.getCurrentContext(),s=e??r.task;if(!s)throw new Error("No task specified or inferred. cd into task dir before: `cd $(aip task path my-task) && aip task ingest`");let{project:i}=await u.findTask(s,t),a=u.getProjectDir(i),p=u.getTaskDir(i,s);if(!await $.runHooksForContext(a,p,"pre-start",{action:"pre-start",entityType:"task",project:i,task:s}))throw new Error("Pre-start hook failed, aborting task start");let l=await u.getTask(i,s);l?.status!=="in-progress"&&l?.status!=="ongoing"&&await dt.handler({project:i,task:s,status:"in-progress"}),o&&await u.ingestTask(i,s),await $.runHooksForContext(a,p,"post-start",{action:"post-start",entityType:"task",project:i,task:s});}});});var Fe,Ue=f(()=>{k();j();Fe=d({description:"Read one or more files into the console",options:z$1.object({}),args:z$1.object({paths:z$1.array(z$1.string()).describe("File paths to read")}),handler:async({paths:t})=>{await n.logFiles(...t);}});});var jt={};Be(jt,{default:()=>N});var Eo,N,Y=f(()=>{It();Lt();Ut();Gt();Mt();Wt();Zt();Qt();oe();se();ie();ce();le();me();fe();ke();je();be();Pe();xe();Ae();Oe();De();Ce();Ie();Le();vt();Ue();Eo={agent:{create:Nt,current:_t,list:Ft,path:Ht,start:Kt},help:{api:Jt,hooks:Vt,quickstart:Yt,skill:ee,usage:re},hook:{create:ne,run:ae},log:{append:pe,read:de},project:{create:ue,current:ge,list:ye,path:Te,update:we},skill:{read:Se},task:{create:ve,current:$e,ingest:Ee,list:Re,path:Ne,start:_e,update:dt},util:{read:Fe}},N=Eo;});J();Y();j();var Do={task:{complete:{noun:"task",verb:"update",prependArgs:["--status","done"]},done:{noun:"task",verb:"update",prependArgs:["--status","done"]},stop:{noun:"task",verb:"update",prependArgs:["--status","done"]},log:{noun:"log",verb:"append"},show:{noun:"task",verb:"ingest"},status:{noun:"task",verb:"current"}},project:{stop:{noun:"project",verb:"update",prependArgs:["--status","done"]},done:{noun:"project",verb:"update",prependArgs:["--status","done"]}}},Ro={task:{update:{summary:"log",message:"log"}}};function Co(t,e,o){let r=Do[t]?.[e];return r?{noun:r.noun,verb:r.verb,args:[...r.prependArgs||[],...o]}:{noun:t,verb:e,args:o}}function No(t,e,o){let r=Ro[t]?.[e];return r?o.map(s=>{if(s.startsWith("--")){let i=s.slice(2);if(i in r)return `--${r[i]}`}return s}):o}function Io(t){let e=[];for(let o of t){let r=o.indexOf("=");r>0&&!o.startsWith("-")?e.push(`--${o.slice(0,r)}`,o.slice(r+1)):e.push(o);}return e}function He(t){let[e="",o="",...r]=t,s=Co(e,o,r),i=Io(s.args),a=No(s.noun,s.verb,i);return [s.noun,s.verb,...a]}Y();L();j();var Lo="date time noun verb extra_args error";function ot(t,e){if(!S.ERROR_LOG)return;let r=new Date().toISOString(),[s,i]=r.split("T"),a=i.replace(/\.\d+Z$/,""),p="",m="";n.isKeyOf(t[0],N)&&(p=t[0],n.isKeyOf(t[1],N[p])&&(m=t[1]));let l=(p?1:0)+(m?1:0),g=t.slice(l).join(" "),y=[s,a,p,m,g,n.errorMessage(e)].map(Fo).join(" "),O=it.dirname(S.ERROR_LOG);ut.existsSync(O)||ut.mkdirSync(O,{recursive:true});let E=ut.existsSync(S.ERROR_LOG);ut.appendFileSync(S.ERROR_LOG,(E?"":Lo+`
|
|
404
|
+
`)+y+`
|
|
405
|
+
`);}function Fo(t){return (t||"-").replace(/\\/g,"\\\\").replace(/\t/g,"\\t").replace(/\n/g,"\\n").replace(/\r/g,"")}process.on("unhandledRejection",t=>{console.error("Unhandled Rejection:",t),process.exit(1);});process.on("uncaughtException",t=>{console.error("Unhandled Exception:",t),process.exit(1);});async function Ge(t){let[e,o,...r]=He(t);(e==="--version"||e==="-v")&&(console.log(x.version),process.exit(0));try{let s=!e||e==="--help"||e==="-h";(s||!n.isKeyOf(e,N))&&(await N.help.usage.handler({}),s||ot(t,`Unknown command: ${e}`),process.exit(1));let i=N[e];s=!o||o==="--help"||o==="-h",(s||!n.isKeyOf(o,i))&&(s||ot(t,`Unknown command: ${e} ${o}`),await N.help.usage.handler({name:e}),process.exit(1));let a=i[o];a.parser._internalHandler(m=>{m.type==="error"&&ot(t,m.error||"parse error");});let p=Uo(r,a.args);await a.handler(a.parser.name(`${e} ${o}`).parse(p));}catch(s){console.error("Error:",n.errorMessage(s)),ot(t,s),process.exit(1);}}var la=Ge,da=N;n.isMain()&&Ge(process.argv.slice(2));function Uo(t,e){if(!e)return t;let o=new Set(Object.keys(e.shape)),r=[],s=0;for(;s<t.length;){if(t[s].startsWith("--")){let i=t[s].slice(2);if(o.has(i)&&s+1<t.length&&!t[s+1].startsWith("--")){r.push(t[s+1]),s+=2;continue}}r.push(t[s]),s++;}return r}export{da as commands,la as default};
|