gitpick 4.15.0-canary.3 → 4.15.0-canary.4

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/README.md CHANGED
@@ -165,7 +165,6 @@ Contributions welcome — any help is appreciated!
165
165
  - Fork the repo and create a branch (use descriptive names, e.g. feat/<name> or fix/<name>).
166
166
  - Make your changes, add tests if applicable, and run the checks:
167
167
  - bun install
168
- - bun run sync:external
169
168
  - bun test
170
169
  - Follow the existing code style and commit message conventions (use conventional commits: feat, fix, docs, chore).
171
170
  - Open a PR describing the change, motivation, and any migration notes; link related issues.
package/dist/index.mjs CHANGED
@@ -1,41 +1,38 @@
1
1
  #!/usr/bin/env node
2
- import e from"node:fs";import t from"node:path";import{parseArgs as n,stripVTControlCharacters as r}from"node:util";import i from"node:os";import a from"node:process";import*as o from"node:readline/promises";import{fileURLToPath as s}from"node:url";import{pipeline as c}from"node:stream/promises";import{on as l,once as u}from"node:events";import{spawn as d}from"node:child_process";import f from"node:fs/promises";import p from"node:tty";import ee from"process";var m=Object.defineProperty,te=(e,t)=>{let n={};for(var r in e)m(n,r,{get:e[r],enumerable:!0});return t||m(n,Symbol.toStringTag,{value:`Module`}),n};const h=async(n,r)=>{let i=await e.promises.readdir(n,{withFileTypes:!0});await e.promises.mkdir(r,{recursive:!0});for(let a of i){if(a.name===`.git`)continue;let i=t.join(n,a.name),o=t.join(r,a.name);if(a.isDirectory())await h(i,o);else if(a.isSymbolicLink()){let t=await e.promises.readlink(i);await e.promises.symlink(t,o)}else await e.promises.copyFile(i,o)}},ne=e=>({start:a.hrtime.bigint(),command:e.map(e=>re(r(e))).join(` `),state:{stdout:``,stderr:``,output:``,isIterating:{},nonIterable:[!1,!1]}}),re=e=>/[^\w./-]/.test(e)?`'${e.replaceAll(`'`,`'\\''`)}'`:e,g=async function*(e,{state:t},n,r){if(t.isIterating[n]===!1)throw Error(`The subprocess must be iterated right away, for example:
3
- for await (const line of spawn(...)) { ... }`);t.isIterating[n]=!0;try{let{[n]:i}=await e.nodeChildProcess;if(!i){t.nonIterable[r]=!0;let e=t.nonIterable.every(Boolean)?"either the option `stdout` or `stderr`":`the option \`${n}\``;throw TypeError(`The subprocess cannot be iterated unless ${e} is 'pipe'.`)}ie(e),yield*o.createInterface({input:i})}finally{await e}},ie=async e=>{try{await e}catch{}},ae=async function*({state:e},...t){try{let n=[];for(;t.length>0;){n=t.map((t,r)=>n[r]??oe(t,r,e));let[{value:r,done:i},a]=await Promise.race(n.map((e,t)=>Promise.all([e,t]))),[o]=t.splice(a,1);n.splice(a,1),i||(t.push(o),yield r)}}finally{await Promise.all(t.map(e=>e.return()))}},oe=async(e,t,{nonIterable:n})=>{try{return await e.next()}catch(r){return se(n,t)?e.return():e.throw(r)}},se=(e,t)=>e.every(Boolean)?t!==e.length-1:e[t],ce=({stdin:e,stdout:n,stderr:r,stdio:i=[e,n,r],env:o,preferLocal:c,cwd:l=`.`,...u})=>{let d=l instanceof URL?s(l):t.resolve(l),f=o?{...a.env,...o}:void 0,p=i[0]?.string;return{...u,input:p,stdio:p===void 0?i:[`pipe`,...i.slice(1)],env:c?le(f??a.env,d):f,cwd:d}},le=({Path:e=``,PATH:n=e,...r},i)=>{let a=n.split(t.delimiter),o=_([],t.resolve(i)).map(e=>t.join(e,`node_modules/.bin`)).filter(e=>!a.includes(e));return{...r,PATH:[...o,n].filter(Boolean).join(t.delimiter)}},_=(e,n)=>e.at(-1)===n?e:_([...e,n],t.resolve(n,`..`)),ue=async e=>{let[[t,n]]=await Promise.all([Promise.allSettled(e),de(e)]);if(n.reason)throw n.reason.pipedFrom=t.reason??t.value,n.reason;if(t.reason)throw t.reason;return{...n.value,pipedFrom:t.value}},de=async e=>{try{let[{stdout:t},{stdin:n}]=await Promise.all(e.map(({nodeChildProcess:e})=>e));if(n===null)throw Error(`The "stdin" option must be set on the first "spawn()" call in the pipeline.`);if(t===null)throw Error(`The "stdout" option must be set on the last "spawn()" call in the pipeline.`);c(t,n).catch(()=>{})}catch(t){throw await Promise.allSettled(e.map(({nodeChildProcess:e})=>fe(e))),t}},fe=async e=>{let{stdin:t}=await e;t.end()},pe=async(e,{input:t},n)=>{let r=await e;t!==void 0&&r.stdin.end(t);let i=u(r,`close`);try{return await Promise.race([i,...r.stdio.filter(Boolean).map(e=>me(e))]),he(n,b(r)),_e(n)}catch(e){throw await Promise.allSettled([i]),v(e,r,n)}},me=async e=>{for await(let[t]of l(e,`error`))if(![`ERR_STREAM_PREMATURE_CLOSE`,`EPIPE`].includes(t?.code))throw t},he=({command:e},{exitCode:t,signalName:n})=>{if(n!==void 0)throw new y(`Command was terminated with ${n}: ${e}`);if(t!==void 0)throw new y(`Command failed with exit code ${t}: ${e}`)},v=(e,t,n)=>Object.assign(ge(e,n),b(t),_e(n)),ge=(e,{command:t})=>e instanceof y?e:new y(`Command failed: ${t}`,{cause:e});var y=class extends Error{name=`SubprocessError`};const b=({exitCode:e,signalCode:t})=>({...e<1?{}:{exitCode:e},...t===null?{}:{signalName:t}}),_e=({state:{stdout:e,stderr:t,output:n},command:r,start:i})=>({stdout:x(e),stderr:x(t),output:x(n),command:r,durationMs:Number(a.hrtime.bigint()-i)/1e6}),x=e=>e.at(-1)===`
4
- `?e.slice(0,e.at(-2)===`\r`?-2:-1):e,ve=async(e,t,n)=>await ye(e,n)?[w(e),t.map(e=>we(e)),{...n,shell:!0}]:[e,t,n],ye=async(e,{shell:t,cwd:n,env:r=a.env})=>a.platform===`win32`&&!t&&!await be(e,n,r),be=(e,t,{Path:n=``,PATH:r=n})=>C.some(t=>e.toLowerCase().endsWith(t))||Ce(e,t,r),xe={},S=e=>(...t)=>xe[t.join(`\0`)]??=e(...t),Se=S(f.access),Ce=S(async(e,n,r)=>{let i=r.split(t.delimiter).filter(Boolean).map(e=>e.replace(/^"(.*)"$/,`$1`));try{await Promise.any([n,...i].flatMap(n=>C.map(r=>Se(`${t.resolve(n,e)}${r}`))))}catch{return!1}return!0}),C=[`.exe`,`.com`],we=e=>w(w(`"${e.replaceAll(/(\\*)"/g,`$1$1\\"`).replace(/(\\*)$/,`$1$1`)}"`)),w=e=>e.replaceAll(/([()\][%!^"`<>&|;, *?])/g,`^$1`),Te=async(e,t,n,r)=>{try{[`node`,`node.exe`].includes(e.toLowerCase())&&(e=a.execPath,t=[...a.execArgv.filter(e=>!e.startsWith(`--inspect`)),...t]),[e,t,n]=await ve(e,t,n),[e,t,n]=Ee(e,t,n);let i=d(e,t,n);return De(i.stdout,r,`stdout`),De(i.stderr,r,`stderr`),i.once(`error`,()=>{}),await u(i,`spawn`),i}catch(e){throw v(e,{},r)}},Ee=(e,t,n)=>n.shell&&t.length>0?[[e,...t].join(` `),[],n]:[e,t,n],De=(e,{state:t},n)=>{e&&(e.setEncoding(`utf8`),t.isIterating[n]||(t.isIterating[n]=!1,e.on(`data`,e=>{t[n]+=e,t.output+=e})))};function T(e,t,n,r){let[i=[],a={}]=Array.isArray(t)?[t,n]:[[],t],o=ne([e,...i]),s=ce(a),c=Te(e,i,s,o),l=pe(c,s,o);Object.assign(l,{nodeChildProcess:c}),l=r?ue([r,l]):l;let u=g(l,o,`stdout`,0),d=g(l,o,`stderr`,1);return Object.assign(l,{nodeChildProcess:c,stdout:u,stderr:d,[Symbol.asyncIterator]:()=>ae(o,u,d),pipe:(e,t,n)=>T(e,t,n,l)})}var Oe=te({bgBlack:()=>He,bgBlue:()=>Ke,bgBlueBright:()=>st,bgCyan:()=>Je,bgCyanBright:()=>lt,bgGray:()=>Xe,bgGreen:()=>We,bgGreenBright:()=>at,bgMagenta:()=>qe,bgMagentaBright:()=>ct,bgRed:()=>Ue,bgRedBright:()=>it,bgWhite:()=>Ye,bgWhiteBright:()=>ut,bgYellow:()=>Ge,bgYellowBright:()=>ot,black:()=>Re,blue:()=>j,blueBright:()=>et,bold:()=>D,cyan:()=>M,cyanBright:()=>nt,dim:()=>je,gray:()=>Ve,green:()=>k,greenBright:()=>Qe,hidden:()=>Ie,inverse:()=>Fe,italic:()=>Me,magenta:()=>ze,magentaBright:()=>tt,overline:()=>Pe,red:()=>O,redBright:()=>Ze,reset:()=>Ae,strikethrough:()=>Le,underline:()=>Ne,white:()=>Be,whiteBright:()=>rt,yellow:()=>A,yellowBright:()=>$e});const ke=p?.WriteStream?.prototype?.hasColors?.()??!1,E=(e,t)=>{if(!ke)return e=>e;let n=`\u001B[${e}m`,r=`\u001B[${t}m`;return e=>{let i=e+``,a=i.indexOf(r);if(a===-1)return n+i+r;let o=n,s=0,c=(t===22?r:``)+n;for(;a!==-1;)o+=i.slice(s,a)+c,s=a+r.length,a=i.indexOf(r,s);return o+=i.slice(s)+r,o}},Ae=E(0,0),D=E(1,22),je=E(2,22),Me=E(3,23),Ne=E(4,24),Pe=E(53,55),Fe=E(7,27),Ie=E(8,28),Le=E(9,29),Re=E(30,39),O=E(31,39),k=E(32,39),A=E(33,39),j=E(34,39),ze=E(35,39),M=E(36,39),Be=E(37,39),Ve=E(90,39),He=E(40,49),Ue=E(41,49),We=E(42,49),Ge=E(43,49),Ke=E(44,49),qe=E(45,49),Je=E(46,49),Ye=E(47,49),Xe=E(100,49),Ze=E(91,39),Qe=E(92,39),$e=E(93,39),et=E(94,39),tt=E(95,39),nt=E(96,39),rt=E(97,39),it=E(101,49),at=E(102,49),ot=E(103,49),st=E(104,49),ct=E(105,49),lt=E(106,49),ut=E(107,49),N=a.platform!==`win32`||!!a.env.WT_SESSION||a.env.TERM_PROGRAM===`vscode`,P=e=>!!(e.isTTY&&a.env.TERM!==`dumb`&&!(`CI`in a.env)),dt=j(N?`ℹ`:`i`),ft=k(N?`✔`:`√`),pt=A(N?`⚠`:`‼`),mt=O(N?`✖`:`×`),F={frames:N?[`⠋`,`⠙`,`⠹`,`⠸`,`⠼`,`⠴`,`⠦`,`⠧`,`⠇`,`⠏`]:[`-`,`\\`,`|`,`/`],interval:80},I=new Set;var ht=class{#e;#t;#n=-1;#r;#i;#a;#o;#s=0;#c;#l;#u=0;#d=!1;#f=new Map;#p=!1;#m=!1;constructor(e={}){let t=e.spinner??F;if(!Array.isArray(t.frames)||t.frames.length===0||t.frames.some(e=>typeof e!=`string`))throw Error("The `spinner.frames` option must be a non-empty array of strings");if(t.interval!==void 0&&!(Number.isInteger(t.interval)&&t.interval>0))throw Error("The `spinner.interval` option must be a positive integer");this.#e=t.frames,this.#t=t.interval??F.interval,this.#i=e.text??``,this.#a=e.stream??a.stderr,this.#o=e.color??`cyan`,this.#l=P(this.#a),this.#c=this.#A.bind(this)}#h(e){this.#p=!0;try{return e()}finally{this.#p=!1}}#g(e,t){if(e==null)return``;if(typeof e==`string`)return e;if(Buffer.isBuffer(e)||ArrayBuffer.isView(e)){let n=typeof t==`string`&&t!==``&&t!==`buffer`?t:`utf8`;return Buffer.from(e).toString(n)}return String(e)}#_(e){if(!this.#l)return e();try{return this.#w(`\x1B[?2026h`),e()}finally{this.#w(`\x1B[?2026l`)}}#v(e){if(!e||this.#f.has(e)||typeof e.write!=`function`||I.has(e))return;let t=e.write,n=(...n)=>this.#x(e,t,n);this.#f.set(e,{originalWrite:t,hookedWrite:n}),I.add(e),e.write=n}#y(){if(!this.#l||this.#f.size>0)return;let e=new Set([this.#a]);(this.#a===a.stdout||this.#a===a.stderr)&&(P(a.stdout)&&e.add(a.stdout),P(a.stderr)&&e.add(a.stderr));for(let t of e)this.#v(t)}#b(){for(let[e,t]of this.#f)e.write===t.hookedWrite&&(e.write=t.originalWrite),I.delete(e);this.#f.clear()}#x(e,t,n){let[r,i,a]=n,o=i,s=a;if(typeof o==`function`&&(s=o,o=void 0),this.#p||!this.isSpinning)return t.call(e,r,o,s);this.#s>0&&this.clear();let c=this.#g(r,o),l=c.at(-1)===`
5
- `,u=t.call(e,r,o,s);return l?this.#m=!1:c!==``&&(this.#m=!0),this.isSpinning&&!this.#m&&this.#C(),u}start(e){return e&&(this.#i=e),this.isSpinning?this:(this.#d=!0,this.#E(),this.#y(),this.#C(),this.#O(),this.#l&&(this.#r=setInterval(()=>{this.#C()},this.#t)),this)}stop(e){if(!this.isSpinning)return this;let t=this.#m;if(this.#d=!1,this.#r&&=(clearInterval(this.#r),void 0),this.#m=!1,this.#b(),this.#D(),this.clear(),this.#k(),e){let n=t?`
6
- `:``;this.#a.write(`${n}${e}\n`)}return this}#S(e,t){return this.stop(`${e} ${t??this.#i}`)}success(e){return this.#S(ft,e)}error(e){return this.#S(mt,e)}warning(e){return this.#S(pt,e)}info(e){return this.#S(dt,e)}get isSpinning(){return this.#d}get text(){return this.#i}set text(e){this.#i=e??``,this.#C()}get color(){return this.#o}set color(e){this.#o=e,this.#C()}clear(){return!this.#l||this.#s===0?this:(this.#h(()=>{this.#a.cursorTo(0);for(let e=0;e<this.#s;e++)e>0&&this.#a.moveCursor(0,-1),this.#a.clearLine(1)}),this.#s=0,this)}#C(){if(this.#m)return;let e=this.#l,t=Date.now();(this.#n===-1||t-this.#u>=this.#t)&&(this.#n=++this.#n%this.#e.length,this.#u=t);let n=Oe[this.#o]??M,r=this.#e[this.#n],i=`${n(r)} ${this.#i}`;this.#l||(i+=`
7
- `),e?this.#_(()=>{this.clear(),this.#w(i)}):this.#w(i),this.#l&&(this.#s=this.#T(i))}#w(e){this.#h(()=>{this.#a.write(e)})}#T(e){let t=this.#a.columns??80,n=r(e).split(`
8
- `),i=0;for(let e of n)i+=Math.max(1,Math.ceil(e.length/t));return i}#E(){this.#l&&this.#w(`\x1B[?25l`)}#D(){this.#l&&this.#w(`\x1B[?25h`)}#O(){a.once(`SIGINT`,this.#c),a.once(`SIGTERM`,this.#c)}#k(){a.off(`SIGINT`,this.#c),a.off(`SIGTERM`,this.#c)}#A(e){this.isSpinning&&this.stop();let t=e===`SIGINT`?130:e===`SIGTERM`?143:1;a.exit(t)}};function gt(e){return new ht(e)}const L=async(n,r,a)=>{process.platform===`win32`&&await T(`git`,[`config`,`--global`,`core.longpaths`,`true`]);let o=`https://${n.token?n.token+`@`:n.token}github.com/${n.owner}/${n.repository}.git`,s=t.resolve(i.tmpdir(),`${n.repository}-${Date.now()}${Math.random().toString(16).slice(2,6)}`),c=gt(),l=performance.now();r.watch||c.start(`Picking ${n.type}${n.type===`repository`?` without .git`:` from repository`}...`);try{await T(`git`,[`clone`,o,s,`--branch`,n.branch,`--depth`,`1`,`--single-branch`,...r.recursive?[`--recursive`]:[]])}catch{await T(`git`,[`clone`,o,s,...r.recursive?[`--recursive`]:[]]),await T(`git`,[`checkout`,n.branch],{cwd:s})}let u=t.resolve(s,n.path);(await e.promises.stat(u)).isDirectory()?(await e.promises.mkdir(a,{recursive:!0}),await h(u,a)):(await e.promises.mkdir(a.split(`/`).slice(0,-1).join(`/`),{recursive:!0}),await e.promises.copyFile(u,a)),r.watch?console.log(`- Synced at `+new Date().toLocaleTimeString()):c.success(`Picked ${n.type}${n.type===`repository`?` without .git`:` from repository`} in ${((performance.now()-l)/1e3).toFixed(2)} seconds.`),await e.promises.rm(s,{recursive:!0,force:!0})};function R(e){if(typeof e==`number`||/^\d+$/.test(e))return typeof e==`number`?e:parseInt(e,10);let t=/(\d+)([hms])/g,n=0,r;for(;(r=t.exec(e))!==null;){let e=parseInt(r[1],10);switch(r[2]){case`h`:n+=e*36e5;break;case`m`:n+=e*6e4;break;case`s`:n+=e*1e3;break}}return n}const _t=async e=>{let t=(await T(`git`,[`ls-remote`,e])).stdout,n=t.match(/(.+)\s+HEAD/)?.[1],r=t.match(RegExp(`${n}\\s+refs/heads/(.+)`))?.[1];if(!r)throw Error(`Could not determine default branch!`);return r};async function vt(e,{branch:t,target:n}){let r=/^https:\/\/([^@]+)@github\.com/,i=e.match(r),a=``;i&&(a=i[1],e=e.replace(r,`https://github.com`));for(let t of[`git@github.com:`,`https://github.com/`,`https://raw.githubusercontent.com/`])if(e.startsWith(t)){e=e.replace(t,``);break}let o=e.split(`/`),s=o[0],c=o[1].endsWith(`.git`)?o[1].slice(0,-4):o[1],l=o[2]===`blob`?`blob`:o[2]===`tree`?`tree`:o[2]+o[3]===`refsheads`?`raw`:`repository`,u=t||(l===`repository`?await _t(`https://${a&&a+`@`}github.com/${s}/${c}`):l===`raw`?o[4]:o[3]),d=l?l===`raw`?o.slice(5).join(`/`):o.slice(4).join(`/`):o.slice(2).join(`/`)||`/`,f=n||(l===`blob`?`.`:d.split(`/`).pop()||c);return{token:a,owner:s,repository:c,type:l,branch:u,path:d,target:f}}const z=Symbol(`singleComment`),B=Symbol(`multiComment`),yt=()=>``,bt=(e,t,n)=>e.slice(t,n).replace(/[^ \t\r\n]/g,` `),xt=(e,t)=>{let n=t-1,r=0;for(;e[n]===`\\`;)--n,r+=1;return!!(r%2)};function St(e,{whitespace:t=!0,trailingCommas:n=!1}={}){if(typeof e!=`string`)throw TypeError(`Expected argument \`jsonString\` to be a \`string\`, got \`${typeof e}\``);let r=t?bt:yt,i=!1,a=!1,o=0,s=``,c=``,l=-1;for(let t=0;t<e.length;t++){let u=e[t],d=e[t+1];if(!a&&u===`"`&&(xt(e,t)||(i=!i)),!i)if(!a&&u+d===`//`)s+=e.slice(o,t),o=t,a=z,t++;else if(a===z&&u+d===`\r
9
- `){t++,a=!1,s+=r(e,o,t),o=t;continue}else if(a===z&&u===`
10
- `)a=!1,s+=r(e,o,t),o=t;else if(!a&&u+d===`/*`){s+=e.slice(o,t),o=t,a=B,t++;continue}else if(a===B&&u+d===`*/`){t++,a=!1,s+=r(e,o,t+1),o=t+1;continue}else n&&!a&&(l===-1?u===`,`&&(c+=s+e.slice(o,t),s=``,o=t,l=t):u===`}`||u===`]`?(s+=e.slice(o,t),c+=r(s,0,1)+s.slice(1),s=``,o=t,l=-1):u!==` `&&u!==` `&&u!==`\r`&&u!==`
11
- `&&(s+=e.slice(o,t),o=t,l=-1))}let u=a===z?r(e,o):e.slice(o);return c+s+u}const Ct=[`.gitpick.json`,`.gitpick.jsonc`],wt=async()=>{let n;for(let r of Ct){let i=t.resolve(r);if(e.existsSync(i)){n=i;break}}if(!n)return!1;let r=await e.promises.readFile(n,`utf-8`),i=JSON.parse(St(r).replace(/,\s*([}\]])/g,`$1`));if(!Array.isArray(i)||!i.every(e=>typeof e==`string`))throw Error(`${t.basename(n)} must be an array of strings`);for(let e of i)await T(process.argv[0],[...process.argv.slice(1),...e.split(/\s+/),`-o`],{stdio:`inherit`});return!0},V=globalThis.window?.document!==void 0;globalThis.process?.versions?.node,globalThis.process?.versions?.bun,globalThis.Deno?.version?.deno,globalThis.process?.versions?.electron,globalThis.navigator?.userAgent?.includes(`jsdom`),typeof WorkerGlobalScope<`u`&&globalThis instanceof WorkerGlobalScope,typeof DedicatedWorkerGlobalScope<`u`&&globalThis instanceof DedicatedWorkerGlobalScope,typeof SharedWorkerGlobalScope<`u`&&globalThis instanceof SharedWorkerGlobalScope,typeof ServiceWorkerGlobalScope<`u`&&globalThis instanceof ServiceWorkerGlobalScope;const H=globalThis.navigator?.userAgentData?.platform;H===`macOS`||globalThis.navigator?.platform===`MacIntel`||globalThis.navigator?.userAgent?.includes(` Mac `)===!0||globalThis.process?.platform,H===`Windows`||globalThis.navigator?.platform===`Win32`||globalThis.process?.platform,H===`Linux`||globalThis.navigator?.platform?.startsWith(`Linux`)===!0||globalThis.navigator?.userAgent?.includes(` Linux `)===!0||globalThis.process?.platform,H===`iOS`||globalThis.navigator?.platform===`MacIntel`&&globalThis.navigator?.maxTouchPoints>1||/iPad|iPhone|iPod/.test(globalThis.navigator?.platform),H===`Android`||globalThis.navigator?.platform===`Android`||globalThis.navigator?.userAgent?.includes(` Android `)===!0||globalThis.process?.platform,!V&&a.env.TERM_PROGRAM;const Tt=!V&&a.platform===`win32`,Et=!V&&(a.env.TERM?.startsWith(`screen`)||a.env.TERM?.startsWith(`tmux`)||a.env.TMUX!==void 0);V||a.cwd;const U=e=>Et?`\x1BPtmux;`+e.replaceAll(`\x1B`,`\x1B\x1B`)+`\x1B\\`:e,W=`\x1B[2J`;`${W}`,(()=>{if(V||!Tt)return!1;let e=i.release().split(`.`),t=Number(e[0]),n=Number(e[2]??0);return t<10||t===10&&n<10586})(),`${W}`;const Dt=(e,t)=>{let n=U(`]8;;${t}`),r=U(`\x1B]8;;\x07`);return n+e+r};function G(e,t=ee.argv){let n=e.startsWith(`-`)?``:e.length===1?`-`:`--`,r=t.indexOf(n+e),i=t.indexOf(`--`);return r!==-1&&(i===-1||r<i)}function K(e,t=globalThis.Deno?globalThis.Deno.args:a.argv){let n=e.startsWith(`-`)?``:e.length===1?`-`:`--`,r=t.indexOf(n+e),i=t.indexOf(`--`);return r!==-1&&(i===-1||r<i)}const{env:q}=a;let J;K(`no-color`)||K(`no-colors`)||K(`color=false`)||K(`color=never`)?J=0:(K(`color`)||K(`colors`)||K(`color=true`)||K(`color=always`))&&(J=1);function Ot(){if(!(`FORCE_COLOR`in q))return;if(q.FORCE_COLOR===`true`)return 1;if(q.FORCE_COLOR===`false`)return 0;if(q.FORCE_COLOR.length===0)return 1;let e=Math.min(Number.parseInt(q.FORCE_COLOR,10),3);if([0,1,2,3].includes(e))return e}function kt(e){return e===0?!1:{level:e,hasBasic:!0,has256:e>=2,has16m:e>=3}}function At(e,{streamIsTTY:t,sniffFlags:n=!0}={}){let r=Ot();r!==void 0&&(J=r);let o=n?J:r;if(o===0)return 0;if(n){if(K(`color=16m`)||K(`color=full`)||K(`color=truecolor`))return 3;if(K(`color=256`))return 2}if(`TF_BUILD`in q&&`AGENT_NAME`in q)return 1;if(e&&!t&&o===void 0)return 0;let s=o||0;if(q.TERM===`dumb`)return s;if(a.platform===`win32`){let e=i.release().split(`.`);return Number(e[0])>=10&&Number(e[2])>=10586?Number(e[2])>=14931?3:2:1}if(`CI`in q)return[`GITHUB_ACTIONS`,`GITEA_ACTIONS`,`CIRCLECI`].some(e=>e in q)?3:[`TRAVIS`,`APPVEYOR`,`GITLAB_CI`,`BUILDKITE`,`DRONE`].some(e=>e in q)||q.CI_NAME===`codeship`?1:s;if(`TEAMCITY_VERSION`in q)return/^(9\.(0*[1-9]\d*)\.|\d{2,}\.)/.test(q.TEAMCITY_VERSION)?1:0;if(q.COLORTERM===`truecolor`||q.TERM===`xterm-kitty`||q.TERM===`xterm-ghostty`||q.TERM===`wezterm`)return 3;if(`TERM_PROGRAM`in q){let e=Number.parseInt((q.TERM_PROGRAM_VERSION||``).split(`.`)[0],10);switch(q.TERM_PROGRAM){case`iTerm.app`:return e>=3?3:2;case`Apple_Terminal`:return 2}}return/-256(color)?$/i.test(q.TERM)?2:/^screen|^xterm|^vt100|^vt220|^rxvt|color|ansi|cygwin|linux/i.test(q.TERM)||`COLORTERM`in q?1:s}function Y(e,t={}){return kt(At(e,{streamIsTTY:e&&e.isTTY,...t}))}Y({isTTY:p.isatty(1)}),Y({isTTY:p.isatty(2)});function X(e=``){if(/^\d{3,4}$/.test(e)){let t=/(\d{1,2})(\d{2})/.exec(e)??[];return{major:0,minor:Number.parseInt(t[1],10),patch:Number.parseInt(t[2],10)}}let t=(e??``).split(`.`).map(e=>Number.parseInt(e,10));return{major:t[0],minor:t[1],patch:t[2]}}function Z(e){let{CI:t,CURSOR_TRACE_ID:n,FORCE_HYPERLINK:r,NETLIFY:i,TEAMCITY_VERSION:o,TERM_PROGRAM:s,TERM_PROGRAM_VERSION:c,VTE_VERSION:l,TERM:u}=a.env;if(r)return!(r.length>0&&Number.parseInt(r,10)===0);if(G(`no-hyperlink`)||G(`no-hyperlinks`)||G(`hyperlink=false`)||G(`hyperlink=never`))return!1;if(G(`hyperlink=true`)||G(`hyperlink=always`)||i)return!0;if(!Y(e)||e&&!e.isTTY)return!1;if(`WT_SESSION`in a.env)return!0;if(a.platform===`win32`||t||o)return!1;if(s){let e=X(c);switch(s){case`iTerm.app`:return e.major===3?e.minor>=1:e.major>3;case`WezTerm`:return/^0-unstable-\d{4}-\d{2}-\d{2}$/.test(c)?c.slice(11)>=`2020-06-20`:e.major>=20200620;case`vscode`:return n?!0:e.major>1||e.major===1&&e.minor>=72;case`ghostty`:return!0;case`zed`:return!0}}if(l){if(l===`0.50.0`)return!1;let e=X(l);return e.major>0||e.minor>=50}switch(u){case`alacritty`:return!0;case`xterm-kitty`:return!0}return!1}const Q={stdout:Z(a.stdout),stderr:Z(a.stderr)};function $(e,t,{target:n=`stdout`,...r}={}){return Q[n]?Dt(e,t):r.fallback===!1?e:typeof r.fallback==`function`?r.fallback(e,t):`${e} ${t}`}$.isSupported=Q.stdout,$.stderr=(e,t,n={})=>$(e,t,{target:`stderr`,...n}),$.stderr.isSupported=Q.stderr;var jt=`gitpick`,Mt=`4.15.0-canary.3`;const Nt=`
12
- With ${D(`${$(`GitPick`,`https://github.com/nrjdalal/gitpick`)}`)} clone specific directories or files from GitHub!
2
+ import e from"node:fs";import t from"node:path";import{parseArgs as n,stripVTControlCharacters as r}from"node:util";import i from"node:os";import{spawn as a}from"node:child_process";import{once as o}from"node:events";import s from"node:fs/promises";import c from"node:process";import{fileURLToPath as l}from"node:url";import u from"node:tty";const d=async(n,r)=>{let i=await e.promises.readdir(n,{withFileTypes:!0});await e.promises.mkdir(r,{recursive:!0});for(let a of i){if(a.name===`.git`)continue;let i=t.join(n,a.name),o=t.join(r,a.name);if(a.isDirectory())await d(i,o);else if(a.isSymbolicLink()){let t=await e.promises.readlink(i);await e.promises.symlink(t,o)}else await e.promises.copyFile(i,o)}};var f=class extends Error{constructor(...e){super(...e),this.name=`SubprocessError`,this.stdout=``,this.stderr=``}};const p=[`.exe`,`.com`],m={},h=e=>(...t)=>m[t.join(`\0`)]??=e(...t),g=h(async(...e)=>{try{return await s.access(e[0]),!0}catch{return!1}}),_=h(async(e,n,r)=>{let i=r.split(t.delimiter).filter(Boolean).map(e=>e.replace(/^"(.*)"$/,`$1`));try{await Promise.any([n,...i].flatMap(n=>p.map(r=>g(`${t.resolve(n,e)}${r}`))))}catch{return!1}return!0}),v=async(e,t)=>c.platform===`win32`&&!t.shell&&!p.some(t=>e.toLowerCase().endsWith(t))&&!await _(e,t.cwd??`.`,(c.env.PATH||c.env.Path)??``),y=e=>e.replaceAll(/([()\][%!^"`<>&|;, *?])/g,`^$1`),b=e=>y(y(`"${e.replaceAll(/(\\*)"/g,`$1$1\\"`).replace(/(\\*)$/,`$1$1`)}"`)),x=e=>/[^\w./-]/.test(e)?`'${e.replaceAll(`'`,`'\\''`)}'`:e;async function S(e,n=[],i={}){let{stdin:s,stdout:u,stderr:d,stdio:p,cwd:m=`.`,env:h,...g}=i,_=m instanceof URL?l(m):t.resolve(m),S=h?{...c.env,...h}:void 0,C=p??[s,u,d],w=[e,...n].map(e=>x(r(e))).join(` `);[`node`,`node.exe`].includes(e.toLowerCase())&&(e=c.execPath,n=[...c.execArgv.filter(e=>!e.startsWith(`--inspect`)),...n]);let T={...g,stdio:C,env:S,cwd:_};await v(e,T)&&(n=n.map(e=>b(e)),e=y(e),T={...T,shell:!0}),T.shell&&n.length>0&&(e=[e,...n].join(` `),n=[]);let E=a(e,n,T),D=``,O=``;E.stdout&&(E.stdout.setEncoding(`utf8`),E.stdout.on(`data`,e=>D+=e)),E.stderr&&(E.stderr.setEncoding(`utf8`),E.stderr.on(`data`,e=>O+=e)),E.once(`error`,()=>{});try{await o(E,`spawn`)}catch(e){throw Object.assign(new f(`Command failed: ${w}`,{cause:e}),{stdout:D,stderr:O})}await o(E,`close`);let k=e=>e.at(-1)===`
3
+ `?e.slice(0,e.at(-2)===`\r`?-2:-1):e;if(E.exitCode&&E.exitCode>0)throw Object.assign(new f(`Command failed with exit code ${E.exitCode}: ${w}`),{stdout:k(D),stderr:k(O),exitCode:E.exitCode});if(E.signalCode)throw Object.assign(new f(`Command was terminated with ${E.signalCode}: ${w}`),{stdout:k(D),stderr:k(O)});return{stdout:k(D),stderr:k(O)}}const C=u?.WriteStream?.prototype?.hasColors?.()??!1,w=(e,t)=>{if(!C)return e=>e;let n=`\u001B[${e}m`,r=`\u001B[${t}m`;return e=>{let i=e+``,a=i.indexOf(r);if(a===-1)return n+i+r;let o=n,s=0,c=(t===22?r:``)+n;for(;a!==-1;)o+=i.slice(s,a)+c,s=a+r.length,a=i.indexOf(r,s);return o+=i.slice(s)+r,o}},T=w(1,22),E=w(31,39),D=w(32,39),O=w(33,39),k=w(36,39),A=c.platform!==`win32`||!!c.env.WT_SESSION||c.env.TERM_PROGRAM===`vscode`,j=e=>!!(e.isTTY&&c.env.TERM!==`dumb`&&!(`CI`in c.env)),M=D(A?`✔`:`√`),N=A?[`⠋`,`⠙`,`⠹`,`⠸`,`⠼`,`⠴`,`⠦`,`⠧`,`⠇`,`⠏`]:[`-`,`\\`,`|`,`/`],P=(e={})=>{let t=-1,n,i=e.text??``,a=e.stream??c.stderr,o=j(a),s=0,l=0,u=!1,d=e=>a.write(e),f=()=>{if(!(!o||s===0)){a.cursorTo(0);for(let e=0;e<s;e++)e>0&&a.moveCursor(0,-1),a.clearLine(1);s=0}},p=e=>{let t=a.columns??80,n=r(e).split(`
4
+ `),i=0;for(let e of n)i+=Math.max(1,Math.ceil(e.length/t));return i},m=()=>{let e=Date.now();(t===-1||e-l>=80)&&(t=++t%N.length,l=e);let n=N[t],r=`${k(n)} ${i}`;o?(f(),d(r),s=p(r)):d(r+`
5
+ `)};return{start(e){return i=e,u=!0,o&&d(`\x1B[?25l`),m(),o&&(n=setInterval(m,80)),this},success(e){return u?(u=!1,n&&=(clearInterval(n),void 0),f(),o&&d(`\x1B[?25h`),d(`${M} ${e??i}\n`),this):this}}},F=async(n,r,a)=>{process.platform===`win32`&&await S(`git`,[`config`,`--global`,`core.longpaths`,`true`]);let o=`https://${n.token?n.token+`@`:n.token}github.com/${n.owner}/${n.repository}.git`,s=t.resolve(i.tmpdir(),`${n.repository}-${Date.now()}${Math.random().toString(16).slice(2,6)}`),c=P(),l=performance.now();r.watch||c.start(`Picking ${n.type}${n.type===`repository`?` without .git`:` from repository`}...`);try{await S(`git`,[`clone`,o,s,`--branch`,n.branch,`--depth`,`1`,`--single-branch`,...r.recursive?[`--recursive`]:[]])}catch{await S(`git`,[`clone`,o,s,...r.recursive?[`--recursive`]:[]]),await S(`git`,[`checkout`,n.branch],{cwd:s})}let u=t.resolve(s,n.path);(await e.promises.stat(u)).isDirectory()?(await e.promises.mkdir(a,{recursive:!0}),await d(u,a)):(await e.promises.mkdir(a.split(`/`).slice(0,-1).join(`/`),{recursive:!0}),await e.promises.copyFile(u,a)),r.watch?console.log(`- Synced at `+new Date().toLocaleTimeString()):c.success(`Picked ${n.type}${n.type===`repository`?` without .git`:` from repository`} in ${((performance.now()-l)/1e3).toFixed(2)} seconds.`),await e.promises.rm(s,{recursive:!0,force:!0})};function I(e){if(typeof e==`number`||/^\d+$/.test(e))return typeof e==`number`?e:parseInt(e,10);let t=/(\d+)([hms])/g,n=0,r;for(;(r=t.exec(e))!==null;){let e=parseInt(r[1],10);switch(r[2]){case`h`:n+=e*36e5;break;case`m`:n+=e*6e4;break;case`s`:n+=e*1e3;break}}return n}const L=async e=>{let t=(await S(`git`,[`ls-remote`,e])).stdout,n=t.match(/(.+)\s+HEAD/)?.[1],r=t.match(RegExp(`${n}\\s+refs/heads/(.+)`))?.[1];if(!r)throw Error(`Could not determine default branch!`);return r};async function R(e,{branch:t,target:n}){let r=/^https:\/\/([^@]+)@github\.com/,i=e.match(r),a=``;i&&(a=i[1],e=e.replace(r,`https://github.com`));for(let t of[`git@github.com:`,`https://github.com/`,`https://raw.githubusercontent.com/`])if(e.startsWith(t)){e=e.replace(t,``);break}let o=e.split(`/`),s=o[0],c=o[1].endsWith(`.git`)?o[1].slice(0,-4):o[1],l=o[2]===`blob`?`blob`:o[2]===`tree`?`tree`:o[2]+o[3]===`refsheads`?`raw`:`repository`,u=t||(l===`repository`?await L(`https://${a&&a+`@`}github.com/${s}/${c}`):l===`raw`?o[4]:o[3]),d=l?l===`raw`?o.slice(5).join(`/`):o.slice(4).join(`/`):o.slice(2).join(`/`)||`/`,f=n||(l===`blob`?`.`:d.split(`/`).pop()||c);return{token:a,owner:s,repository:c,type:l,branch:u,path:d,target:f}}const z=Symbol(`singleComment`),B=Symbol(`multiComment`),V=(e,t,n)=>e.slice(t,n).replace(/[^ \t\r\n]/g,` `),H=(e,t)=>{let n=t-1,r=0;for(;e[n]===`\\`;)--n,r+=1;return!!(r%2)};function U(e){if(typeof e!=`string`)throw TypeError(`Expected argument \`jsonString\` to be a \`string\`, got \`${typeof e}\``);let t=!1,n=!1,r=0,i=``,a=``,o=-1;for(let s=0;s<e.length;s++){let c=e[s],l=e[s+1];if(!n&&c===`"`&&(H(e,s)||(t=!t)),!t)if(!n&&c+l===`//`)i+=e.slice(r,s),r=s,n=z,s++;else if(n===z&&c+l===`\r
6
+ `){s++,n=!1,i+=V(e,r,s),r=s;continue}else if(n===z&&c===`
7
+ `)n=!1,i+=V(e,r,s),r=s;else if(!n&&c+l===`/*`){i+=e.slice(r,s),r=s,n=B,s++;continue}else if(n===B&&c+l===`*/`){s++,n=!1,i+=V(e,r,s+1),r=s+1;continue}else n||(o===-1?c===`,`&&(a+=i+e.slice(r,s),i=``,r=s,o=s):c===`}`||c===`]`?(i+=e.slice(r,s),a+=V(i,0,1)+i.slice(1),i=``,r=s,o=-1):c!==` `&&c!==` `&&c!==`\r`&&c!==`
8
+ `&&(i+=e.slice(r,s),r=s,o=-1))}let s=n===z?V(e,r):e.slice(r);return a+i+s}const W=[`.gitpick.json`,`.gitpick.jsonc`],G=async()=>{let n;for(let r of W){let i=t.resolve(r);if(e.existsSync(i)){n=i;break}}if(!n)return!1;let r=await e.promises.readFile(n,`utf-8`),i=JSON.parse(U(r));if(!Array.isArray(i)||!i.every(e=>typeof e==`string`))throw Error(`${t.basename(n)} must be an array of strings`);for(let e of i)await S(process.argv[0],[...process.argv.slice(1),...e.split(/\s+/),`-o`],{stdio:`inherit`});return!0};var K=`gitpick`,q=`4.15.0-canary.4`;const J=(e,t)=>`\x1b]8;;${t}\x07${e}\x1b]8;;\x07`,Y=`
9
+ With ${T(`${J(`GitPick`,`https://github.com/nrjdalal/gitpick`)}`)} clone specific directories or files from GitHub!
13
10
 
14
- $ gitpick ${A(`<url>`)} ${k(`[target]`)} ${M(`[options]`)}
11
+ $ gitpick ${O(`<url>`)} ${D(`[target]`)} ${k(`[options]`)}
15
12
 
16
- ${D(`Hint:`)}
13
+ ${T(`Hint:`)}
17
14
  [target] and [options] are optional and if not specified,
18
15
  GitPick fallbacks to the default behavior of \`git clone\`
19
16
 
20
- ${D(`Arguments:`)}
21
- ${A(`url`)} GitHub URL with path to file/folder/repository
22
- ${k(`target`)} Directory to clone into (optional)
17
+ ${T(`Arguments:`)}
18
+ ${O(`url`)} GitHub URL with path to file/folder/repository
19
+ ${D(`target`)} Directory to clone into (optional)
23
20
 
24
- ${D(`Options:`)}
25
- ${M(`-b, --branch `)} Branch/SHA to clone
26
- ${M(`-o, --overwrite`)} Skip overwrite prompt
27
- ${M(`-r, --recursive`)} Clone submodules
28
- ${M(`-w, --watch [time]`)} Watch the repository and sync every [time]
21
+ ${T(`Options:`)}
22
+ ${k(`-b, --branch `)} Branch/SHA to clone
23
+ ${k(`-o, --overwrite`)} Skip overwrite prompt
24
+ ${k(`-r, --recursive`)} Clone submodules
25
+ ${k(`-w, --watch [time]`)} Watch the repository and sync every [time]
29
26
  (e.g. 1h, 30m, 15s)
30
- ${M(`-h, --help`)} display help for command
31
- ${M(`-v, --version`)} display the version number
27
+ ${k(`-h, --help`)} display help for command
28
+ ${k(`-v, --version`)} display the version number
32
29
 
33
- ${D(`Examples:`)}
30
+ ${T(`Examples:`)}
34
31
  $ gitpick <url>
35
32
  $ gitpick <url> [target]
36
33
  $ gitpick <url> [target] -b [branch/SHA]
37
34
  $ gitpick <url> [target] -w [time]
38
35
  $ gitpick <url> [target] -b [branch/SHA] -w [time]
39
36
 
40
- 🚀 More awesome tools at ${M(`https://github.com/nrjdalal`)}`,Pt=e=>{try{return n(e)}catch(e){throw Error(`Error parsing arguments: ${e.message}`)}};(async()=>{try{let{positionals:n,values:r}=Pt({allowPositionals:!0,options:{branch:{type:`string`,short:`b`},force:{type:`boolean`,short:`f`},help:{type:`boolean`,short:`h`},overwrite:{type:`boolean`,short:`o`},recursive:{type:`boolean`,short:`r`},version:{type:`boolean`,short:`v`},watch:{type:`string`,short:`w`}}});n.length||(r.version&&(console.log(`\n${jt}@${Mt}`),process.exit(0)),await wt()&&process.exit(0),console.log(Nt),process.exit(0)),n[0]===`clone`&&n.shift();let[i,a]=n,o={branch:r.branch,force:r.force,overwrite:r.overwrite,recursive:r.recursive,watch:r.watch};console.log(`\nWith ${D(`${$(`GitPick`,`https://github.com/nrjdalal/gitpick`)}`)} clone specific files, folders, branches, commits and more from GitHub!`);let s=await vt(i,{branch:o.branch,target:a});if(s.type===`blob`){let e=s.target.split(`/`).filter(e=>e!==``),t=e[e.length-1];t!==`.`&&t!==`..`&&t.includes(`.`)?e.pop():t=s.path.split(`/`).pop()||t,s.target=[...e,t].join(`/`)}console.info(`\n${k(`✔`)} ${s.owner}/${s.repository} ${M(s.type+`:`+s.branch)} ${s.type===`repository`?`> ${k(s.target)}`:`${s.path.length?A(s.path)+` >`:`>`} ${k(s.target)}`}`);let c=t.resolve(s.target);if(o.overwrite=o.overwrite||o.force,o.watch&&(o.overwrite=!0),e.existsSync(c)&&!o.overwrite&&(s.type===`blob`&&(console.log(`${A(`\nWarning: The target file exists at ${k(s.target)}. Use ${M(`-f`)} or ${M(`-o`)} to overwrite.`)}`),process.exit(1)),(await e.promises.readdir(c)).length&&(console.log(`${A(`\nWarning: The target directory exists at ${k(s.target)} and is not empty. Use ${M(`-f`)} or ${M(`-o`)} to overwrite.`)}`),process.exit(1))),o.watch){console.log(`\n👀 Watching every ${R(o.watch)/1e3+`s`}\n`),await L(s,o,c);let e=R(o.watch);setInterval(async()=>await L(s,o,c),e)}else await L(s,o,c),process.exit(0)}catch(e){e instanceof Error?console.log(D(`\n${O(`Error: `)}`)+e.message):console.log(D(`${O(`
37
+ 🚀 More awesome tools at ${k(`https://github.com/nrjdalal`)}`,X=e=>{try{return n(e)}catch(e){throw Error(`Error parsing arguments: ${e.message}`)}};(async()=>{try{let{positionals:n,values:r}=X({allowPositionals:!0,options:{branch:{type:`string`,short:`b`},force:{type:`boolean`,short:`f`},help:{type:`boolean`,short:`h`},overwrite:{type:`boolean`,short:`o`},recursive:{type:`boolean`,short:`r`},version:{type:`boolean`,short:`v`},watch:{type:`string`,short:`w`}}});n.length||(r.version&&(console.log(`\n${K}@${q}`),process.exit(0)),await G()&&process.exit(0),console.log(Y),process.exit(0)),n[0]===`clone`&&n.shift();let[i,a]=n,o={branch:r.branch,force:r.force,overwrite:r.overwrite,recursive:r.recursive,watch:r.watch};console.log(`\nWith ${T(`${J(`GitPick`,`https://github.com/nrjdalal/gitpick`)}`)} clone specific files, folders, branches, commits and more from GitHub!`);let s=await R(i,{branch:o.branch,target:a});if(s.type===`blob`){let e=s.target.split(`/`).filter(e=>e!==``),t=e[e.length-1];t!==`.`&&t!==`..`&&t.includes(`.`)?e.pop():t=s.path.split(`/`).pop()||t,s.target=[...e,t].join(`/`)}console.info(`\n${D(`✔`)} ${s.owner}/${s.repository} ${k(s.type+`:`+s.branch)} ${s.type===`repository`?`> ${D(s.target)}`:`${s.path.length?O(s.path)+` >`:`>`} ${D(s.target)}`}`);let c=t.resolve(s.target);if(o.overwrite=o.overwrite||o.force,o.watch&&(o.overwrite=!0),e.existsSync(c)&&!o.overwrite&&(s.type===`blob`&&(console.log(`${O(`\nWarning: The target file exists at ${D(s.target)}. Use ${k(`-f`)} or ${k(`-o`)} to overwrite.`)}`),process.exit(1)),(await e.promises.readdir(c)).length&&(console.log(`${O(`\nWarning: The target directory exists at ${D(s.target)} and is not empty. Use ${k(`-f`)} or ${k(`-o`)} to overwrite.`)}`),process.exit(1))),o.watch){console.log(`\n👀 Watching every ${I(o.watch)/1e3+`s`}\n`),await F(s,o,c);let e=I(o.watch);setInterval(async()=>await F(s,o,c),e)}else await F(s,o,c),process.exit(0)}catch(e){e instanceof Error?console.log(T(`\n${E(`Error: `)}`)+e.message):console.log(T(`${E(`
41
38
  Unexpected Error: `)}`)+JSON.stringify(e,null,2)),process.exit(1)}})();export{};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "gitpick",
3
- "version": "4.15.0-canary.3",
3
+ "version": "4.15.0-canary.4",
4
4
  "description": "Clone exactly what you need aka straightforward project scaffolding!",
5
5
  "keywords": [
6
6
  "clone",