jja 2.3.17 → 2.3.18-beta.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.
Files changed (2) hide show
  1. package/bin.mjs +1 -1
  2. package/package.json +2 -2
package/bin.mjs CHANGED
@@ -1,2 +1,2 @@
1
1
  #!/usr/bin/env node
2
- import{Dog as e}from"@qqi/log";import{isFalse as t,isUndefined as n,isArray as o,isString as r,isEmptyObject as i,isNull as s,isBusinessEmptyString as a,isPlainObject as c,isEmptyArray as l}from"a-type-of-js";import{Command as p,enPrefixList as d}from"a-command";import{redPen as u,hexPen as g,randomPen as m,greenPen as f,yellowPen as w,pen as $,strInOneLineOnTerminal as y,hidePen as h,cyanPen as v,boldPen as b,italicPen as j}from"color-pen";import{_p as k,isWindows as D,runOtherCode as S,fileExist as V,detectPackageManager as A,cursorMoveUp as R,cursorAfterClear as E,getDirectoryBy as O,readFileToJsonSync as P,pathJoin as q,getNpmPkgInfo as T,waitingTips as C,colorLine as L,isTTY as x,__p as N,typewrite as M}from"a-node-tools";import F from"node:https";import{copyTextToClipboard as G}from"@qqi/copy-text";import{terminalRegExp as I}from"@color-pen/static";import{Table as X}from"colored-table";import{intersection as Y}from"a-js-tools";import B,{Resolver as H}from"node:dns";import J from"node:net";const U=new e({name:"jja",type:!1});t(U.type);const W=new p("jja");W.bind({"package <pkg> (包管理)":["--diff <-d> (分析当下包的差异)","--upDependencies <-u> (更新依赖,强制更新到 'latest',使用需谨慎)"],"remove <rm> (做一个简单的兼容的移除文件或文件夹的命令)":["--ignore <-i> (不打印日志(默认打印的))","--subdirectories <-s> (这是一个危险的命令,用于 windows 下递归删除执行命令下所有的给定文件名称)"],"update <up> (做一个简单的 npm 升级程序,对,简单的)":["--ignore <-i> (不建议你这么用,你会发现你像个傻子一样在那等结果)",`--dependencies <-d> (更新依赖,跟 ${m("npm update")} 一样)`],"clearScreen <cls> (清理终端显示屏幕,同 clearTerminal )":"","clearTerminal <clear> (清理终端显示屏幕,同 clearScreen )":"","git (一些关于 git 的操作)":[`commit (git 提交代码,是 ${u("commit")} 提交啊,不是 ${g("#666")("push")} 推送)`,"merge (合并两个分支)","tag (给提交打上 tag)"],"dns (使用 dns 获取 ip 地址)":["--domain <-d> (获取给定域名获取 ip 地址,默认 github.com )","--port <-p> (配置要校验的 port 端口号,默认为 443)","--not-covered <-n> (不覆盖解析过程,默认不展示)"]}).run(),U("参数已绑定,且已解析用户参数",W.args),U.warn("当前的执行状态",W.state),W.isEnd(!0);const _={log:!1,subdirectories:!1};async function z(e,t,n=5){return!await Reflect.apply(e,void 0,t)&&n--?(k(f("执行失败,现在重试中")),U.warn(t,"执行失败,第",5-n,"次执行"),await z(e,t,n)):await Reflect.apply(e,void 0,t)}function K(e,t){k(`清理 ${e} 文件${t?"完成":"失败"}`)}async function Q(e,t){_.log||k(g("#aaa")(` ${e} 文件/夹存在,准备删除`));let n={error:null};return D?t.isDirectory()?(n=await S(`rd /q /s ${e.replace(/\//,"\\")}`),K(e,n.success)):t.isFile()&&(n=await S(`del ${_.subdirectories?"/s":""} /q ${e.replace(/\//,"\\")}`),K(e,n.success)):(n=await S(`rm -rf ${e.replace(/\\/,"/")}`),K(e,n.success)),!!n.success||(console.log(n.error),!1)}async function Z(e){_.log||k(g("#336")(`当前清理文件为 ${e}`));const t=V(e);_.log||k(g("#666")(`正在检测 ${e} 文件/夹是否存在`)),U(`文件${e} 数据为 ${t}`),n(t)?_.log||k(w(`${e} 文件不存在`)):await z(Q,[e,t])}const ee=(()=>{const e=A();return"npm"===e?"npm install --save":"yarn"===e?"yarn add":"pnpm add"})();async function te(e=!0){const t=O("package.json","file");if(null==t)return k("当前目录下不存在 package.json 文件");const n=P(q(t,"package.json"))?.dependencies||{};if(e&&k(f("初始化完成,等待下一步命令")),Object.keys(n).length>0){let e="npm install --save";for(let t=Object.keys(n),o=t.length,r=0;r<o;r++){e+=` ${t[r]}@latest `}await S(e)}var o;e&&k(v("正在更新 dev 依赖")),await S("npm update --save"),e&&(o="开发依赖更新完毕",R(),E(),k(o))}const ne={log:!0};async function oe(e){Boolean(e["--dependencies"])&&await te(ne.log)}const re={local:null,online:null,dependenceList:{},updateDependence:[],latestDependence:[],binning:function(e,t=!1){if(n(e)||i(e)||s(this.local))return;const{overrides:o}=this.local;for(const n in e){if(r(o?.[n])){W.WARN(`${n} 被锁定在 ${o[n]}`);continue}const i=e[n];this.dependenceList[n]={type:t?"devDependencies":"dependencies",version:i,localVersion:"",latestVersion:"",onlineVersion:"",tag:"",time:""}}},preReleaseDependence:[],timeoutDependence:[],registry:void 0};const ie=b.green,se=g("0x993311"),ae=g("0x339999"),ce=g("0x666666");const le=C({show:!1,info:"请稍等",beforeDestroyed:e=>e&&W.end()});async function pe(){const e={"淘宝":"registry.npmmirror.com","官方":"registry.npmjs.org","腾讯":"mirrors.tencent.com/npm","中科大":"npmreg.proxy.ustclug.org",yarn:"registry.yarnpkg.com"},n=()=>Object.keys(e).map(t=>function(e){const t=Date.now();return new Promise(n=>{const o=`https://${e}/nry`,r=F.request(o,{method:"HEAD",timeout:3500},o=>{n((o?.statusCode??0)>=200&&(o?.statusCode??0)<300?{url:e,time:Date.now()-t,success:!0}:{url:e,time:Date.now()-t,success:!1})});r.on("error",()=>n({url:e,time:Date.now()-t,success:!1})),r.on("timeout",()=>{r.destroy(),n({url:e,time:Date.now()-t,success:!1})}),r.end()})}(e[t])),o={};for(let e=0;e<3;e++){le.run(`请稍等,正在进行第${["一","二","三","四","五"][e]}次安装源延迟测试`);(await Promise.all(n())).forEach(e=>{t(e.success)||(o[e.url]?o[e.url]=(o[e.url]+e.time)/2:o[e.url]=e.time)})}return le.destroyed(),Object.keys(e).reduce((t,n)=>o[e[n]]<(o[e[t]]??1/0)?n:t,"")}async function de(){const{dependenceList:e,timeoutDependence:t}=re,o=Object.keys(e);o.length>18&&(n(re.registry)||!["官方","淘宝","腾讯","中科大","yarn"].some(e=>e===re.registry))&&(re.registry=await pe(),k(`由于您未设置或者设置有误,现使用 ${v(re.registry)} npm 源`));const r=o.map(n=>new Promise(o=>{(async()=>{const r=e[n],i=e[n].version;le.run(`${d.current()}获取 ${n} 的本地安装信息`);const a=function(e){const t=P(q("node_modules/",e,"package.json"));let n="";return n=s(t)?"":t.version||"",re.dependenceList[e].localVersion=n,U(e,"本地安装的版本为:",n),n}(n);if(le.run(`${d.success()} ${n} 的本地安装版本:${r.localVersion}`),await async function(e){const t=await T(e,re.registry,9867);if("timeout"===t.status)return void U.warn("请求超时");if(s(t.data))return void U.warn("获取包信息失败");const n=Date.now();U("获取数据",e,"包信息");const o=re.dependenceList[e],{"dist-tags":r,time:i,version:a}=t.data;let c=a,l="latest";o.onlineVersion=c,o.tag=l;const p=Object.keys(r);U("共需比对",p.length);for(let e=0,t=p.length;e<t;e++){const t=p[e],n=r[t];i[n]>i[c]&&(c=n,l=t)}U("比对",e,"完成"),U("耗时",Date.now()-n),"latest"!==l&&(U(e,"最新版本为:",c,"tag为:",l),o.tag=l,o.latestVersion=c);const d=new Date(i[c]).toLocaleString();o.time=d,U(`${e} 的最后发布时间为:${d}`)}(n),""===r.onlineVersion&&""===r.latestVersion)return t.push(n),le.log(`${d.error()} ${n} 本地 ${r.localVersion} 请求错误`),o(!0);if(a===r.onlineVersion){const e=`${n} 的本地${i} 安装版本为 ${a} 最新是 ${r.onlineVersion} `;return""===r.latestVersion?(le.run(d.info()+e),o(!0)):(re.preReleaseDependence.push(n),le.log(`${d.info()} ${e};最新预发布版本为 ${f(r.latestVersion)}`),o(!0))}if(a===r.latestVersion)return le.log(`${d.info()} {key} 的本地版本${i} 安装版本为 ${a}`),o(!0);const c=`${n} 的本地 ${f(i)} 安装版本为 ${f(a)} 最新是 ${ae(r.onlineVersion)} `;re.latestDependence.push(n),""===r.latestVersion?(le.log(d.info()+c),o(!0)):(re.preReleaseDependence.push(n),le.log(`${d.info()} ${c};最新预发布版本为 ${f(r.latestVersion)}`),o(!0))})()}));return Promise.all(r).then(()=>{le.destroyed()})}async function ue(e){const{msg:t,list:n,type:i,copy:s}=e,a=$[i];k(),k(a.reversed(t),!1),k(s?ce.reversed`已复制到剪切板 📋`:""),k(),k(`${a(ee)} ${h`\\`}\n${function(e){if(!o(e)||e.some(e=>!r(e)))throw new TypeError("参数必须为字符串数组");return e.map((t,n)=>n%3==2&&n!==e.length-1?`${t} ${h`\\`}\n`:t.concat(" ")).join("")}(n)}`),k(),s&&G(`${ee} ${n.join(" ")}`.replace(I(),""))}function ge(e){return`${b(e)}@${ie`latest`}`}function me(e,t){return`${j(e)}@${j.random(t)}`}async function fe(){const{local:e,preReleaseDependence:t,latestDependence:n,dependenceList:o,timeoutDependence:r}=re;if(s(e))return;const{dependencies:i,devDependencies:a}=e;if(re.binning(i),re.binning(a,!0),await de(),0===t.length&&0===n.length&&0===r.length)return k(ce`看起来似乎没有依赖版本差异`);if(0===t.length&&0===n.length)return k(u`看起来网络不太好讷,所有的包线上版本的请求都出错了。或者是还没有执行 npm install 呐`);k($.brightGreen`\n版本差异的依赖为:\n`);const c=[...n,...t];var l;new X({header:[v("包名"),{content:"发布时间",color:"#f26"},f`最新版本`],body:[...c.map(e=>{const{type:t,latestVersion:n,time:r,onlineVersion:i}=o[e];return["dependencies"===t?b(e):j(e),m(r),n||j(i)]})]})(),k($.brightRed(`\n目前仅关注版本号是否为最新 ${$.brightMagenta("latest")}`)),k(`使用 ${ee} 命令安装更新\n`),r.length>0&&(l=u`有一些包没有返回结果,请注意:`,k(y(l)),new X([...r.map(e=>[e])])());const p=[...n.filter(e=>!1===t.includes(e)).map(e=>$.bold(e).concat("@latest")),...t.map(e=>me(e,o[e].tag))],d=n.map(e=>ge(e)),g=[...n.map(e=>ge(e)),...t.filter(e=>!1===n.includes(e)).map(e=>me(e,o[e].tag))],w=p.length<t.length+n.length,h=t.length>0,D=n.length>0;h&&(w&&await ue({msg:"‼️ 预发布版本优先:",list:p,type:"brightRed"}),await ue({msg:"⚠️ latest 版本优先:",list:g,type:"brightMagenta",copy:!D})),D&&await ue({msg:"🎉 最佳安装:",list:d,type:"brightGreen",copy:!0})}async function we(e){re.registry=e[0]?.toString()||void 0,await async function(){const e=O("package.json","file");if(n(e))return void U.warn("未找到当前包的 package.json 文件的位置");const t=P(q(e,"package.json"));if(s(t))return void U.error("未找到当前包的 package.json 文件,该事件发生的概率极低");const o=t.name||"",r=await T(o,re.registry,9800);re.local=t,re.online=r.data}();const{local:t,online:o}=re,r=t?.version||"",i=o?.version||"";if(s(t))return U("未找到包 package.json 文件退出"),k(se("未找到当前包 package.json"));const a=se(`当前包本地版本为: ${r}`);if(o){k(a);const e=" ".repeat(6);k(ae(`${e}线上版本为:${i}@latest`));const t=new Date(o&&o.time.modified||"").toLocaleString();W.INFO(`最后的发布时间为: ${t.toString()}`),L()}await fe()}async function $e(e){e["--upDependencies"]?await te(!0):e["--diff"]?await we(e["--diff"]):W.help("package")}async function ye(e){const t=O("package.json","file",process.cwd());if(null==t)return!0;const n=P(q(t,"package.json"));if(s(n)||!n.version)return!0;const o=n.version;await S({code:`git tag -a v${o} -m '${e}'`,cwd:t}),await S({code:"git push origin --tag",cwd:t})}async function he(e){e.commit?await async function(e="版本维护",t=!1){const n=O(".git","directory");if(null==n)return k(g("#ff0")("not a git repository(当前目录非 git 储存库)"));const o=await S({code:"git add .",cwd:n});if(!o.success)return null!=console.log(o.error);const r=await S({code:"git status",cwd:n});if(!/nothing to commit, working tree clean/.test(r.data||"")){const o=await S({code:"git add .",cwd:n});if(!o.success)return null!=console.log(o.error);""==e.trim()&&(e=await W.question({text:"请输入提交信息"})??"");const r=await S({code:`git commit -m "${e}"`,cwd:n});return r.success?(r.error&&console.log(r.error),await S({code:"git push",cwd:n}),t&&await ye(e),!0):null!=console.log(r.error)}return!1}(e.commit.join(" ")):e.merge?await async function(e){if(a(e)){const t=await S("git branch -a");console.log(t.data),e=await W.question({text:"请输入要合并分支的名称"})??""}if(a(e))return W.ERROR("没有获取到要合并的分支");const t=await W.selection({data:[{label:"正常快进合并",value:""},{label:"非快进合并 (--no-ff)",value:"--no-ff"},{label:"多提交记录合并为一条 (--squash)",value:"--squash",tip:"适用于将众多小的提交合并成一个提交"}]});n(t)?W.ERROR("您选择退出合并"):await S(`git merge ${e} ${t}`)}(e.merge.join("")):e.tag?await async function(){const e=await W.question({text:"请输入待标记的信息"});n(e)||await ye(e)}():W.help("git")}const ve=["A","AAAA","ANY","CAA","CNAME","MX","NAPTR","NS","PTR","SOA","SRV","TLSA","TXT"],be={domain:"github.com",port:443,notCovered:!1,dnsServers:["1.1.1.1","8.8.8.8","8.8.4.4","9.9.9.9","208.67.222.222","114.114.115.115","223.5.5.5","180.76.76.76"],ips:{},rrtype:["A"],reset(e){this.ips={},this.domain="github.com",this.port=443,this.notCovered=!1,this.rrtype=[],n(e)||e.forEach(e=>{const t=e["--domain"];!n(t)&&t.length>0&&(this.domain=t[0].toString());const o=e["--port"];!n(o)&&o.length>0&&(this.port=Number(o[0]));const r=e["--not-covered"];n(r)||!1===r[0]||(this.notCovered=!0);const i=e["--rrtype"];if(!n(i)&&i.length>0){const e=Y(ve,i);e.length>0&&(this.rrtype=e)}})}};async function je(e,t=3e3){const{ips:n,port:o}=be;if(void 0===n[e])return U(`校验 ${e} 的联通性`),new Promise(n=>{const r=new J.Socket;r.setTimeout(t),r.on("connect",()=>{r.destroy(),n(!0)}),r.on("timeout",()=>{r.destroy(),n(!1)}),r.on("error",()=>{r.destroy(),n(!1)}),U("测试的接口为:",o),r.connect(o,e)});U(`${e} 联通性已校验过 ${n[e]?"🔥":"⛓️‍💥"}`)}async function ke(e="1.1.1.1"){const t=new H;t.setServers([e]);const{domain:o,ips:r}=be;try{const i=await new Promise(n=>{t.resolve4(o,(t,r)=>{if(t)return U.error(`使用 ${e} 获取 ${o} 的 ip 出错`,t),void n(void 0);U(`使用 ${e} 获取 ${o} 的 ip 值为 ${r.join(" --- ")}`),n(r)})});if(n(i))return{dnsServer:e,error:"获取 ip 错误"};const s=await Promise.all(i.map(async e=>{const t=await je(e);return n(t)||(r[e]=t),{ip:e.padEnd(16," "),isAlive:n(t)?r[e]:t}}));return{dnsServer:e,results:s}}catch(t){return U.error("获取 ip 错误",t),{dnsServer:e,error:t}}}async function De(e){be.reset(e.options),await async function(){const{domain:e,ips:t}=be;try{const o=await new Promise(t=>{B.lookup(e,{all:!0,family:4},(n,o)=>{if(n)return U.error(`获取本地 ${e} 的 ip 出错`,n),void t(void 0);U(`获取本地 ${e} 的 ip 值为 ${o.join(" --- ")}`),t(o.map(e=>e.address))})});if(n(o))return{error:"获取 ip 错误"};const r=await Promise.all(o.map(async e=>{const o=await je(e);return n(o)||(t[e]=o),{ip:e.padEnd(16," "),isAlive:n(o)?t[e]:o}}));return r.length>0&&(k(se`本地配置 ${e} 的 ip 地址及联通性为:\n`),r.forEach(e=>{k(v`- ${e.ip}`,!1),k(e.isAlive?se` -> ✅`:ce` -> ❌`)}),k()),{results:r}}catch(e){return U.error(e),{error:e}}}(),await async function(){const{domain:e,dnsServers:t,notCovered:o}=be;let r=0;k(`${e} ip 列表:`);for(const e of t){const{dnsServer:t,results:i}=await ke(e);n(i)?U.warn(`${t} 获取 ip 失败`):(o||0===r||(R(r),E(!0)),k(),k($.reversed`DNS: ${t}`),k(),i.forEach(({ip:e,isAlive:t})=>{const n=t?`${f(e)} ${se`->`} ✅`:`${ce(e)} ${u`⛓️‍💥`} ❌`;k(y(n))}),r=3+i.length)}o||(R(r+1),E(!0))}();const{ips:t}=be;Object.keys(t).length>0?await async function(){const{domain:e,ips:t,port:n}=be;await M(`${$.brightMagenta`${e}`} 域名解析结果:`),k();for(const e in t)if(Object.prototype.hasOwnProperty.call(t,e)){const n=t[e];k(n?`${f`- ${e.padEnd(16)}`} ${se`->`} ✅`:`${ce`- ${e.padEnd(16)}`} ${u`⛓️‍💥`} ❌`)}k(),await M(u.italic.dim`${e} 联通性接口判断为 ${n.toString()}`),k()}():await async function(){const{domain:e,port:t}=be;await M(`\n找不到 ${u(e)} 的服务器 IP 地址 ${ce`测试使用 ${t.toString()} 端口`}\n`)}()}const Se=W.args.$arrMap;W.args.$isVoid?(U("没有匹配到子命令,打印帮助信息并退出"),W.help(),W.end()):l(W.args.$only)&&(console.log(`${W.args.$nomatch.map(e=>m(e)).join("、")} 不是有效的参数`),W.end());try{await async function e(){if(0===Se.length)return;const t=Se.shift();if(n(t))return await e();t.remove?(U("执行文件移除"),await async function(e){const t=e.value||[],n=e.options;for(n?.forEach(e=>{const n=e["--ignore"];n&&(_.log=!0,t.concat(n));const o=e["--subdirectories"];o&&(_.subdirectories=!0,t.concat(o))}),_.log||k($.brightCyan`当前系统为: ${D?"windows":"linux/mac"}`),0==t.length&&(U.warn("待删除列表",t),k($.random`没有待清理的文件/文件夹`));t.length;){let e=t.pop();e=void 0===e?"undefined":e.toString(),await Z(e)}}(t.remove)):t.clearScreen||t.clearTerminal?(U("执行清屏"),await async function(){x()?(U("执行第一遍清理"),await S({code:D?"cls":"clear",printLog:!1}),U("执行第二遍清理"),N("3J"),N("c",!1),R(1/0),E(!0)):(U.warn("当前系统不支持 TTY"),k("当前环境不支持 "))}()):t.git?(U("执行 git 相关命令"),await async function(e){if(c(e)&&(e={value:[]}),n(e.options)||0===e.options.length)return U.warn("没有配置项,直接返回的帮助文档"),W.help("git");const t=e.options;for(let e=0,n=t.length;e<n;e++)U("本次执行",t[e]),await he(t[e])}(t.git)):t.package?(U("执行 package 相关命令"),await async function(e){if(n(e.options)||0==e.options.length)return U.warn("没有参数输入"),W.help("package");const t=e.options;for(let e=0,n=t.length;e<n;e++)await $e(t[e])}(t.package)):t.update?(U("执行 update 相关命令"),await async function(e){if(n(e.options)||0==e.options.length)return U.warn("没有匹配到选项,自动抛出帮助文档"),W.help("update");const t=e.options;ne.log=!t.some(e=>e["--ignore"]);for(let e=0,n=t.length;e<n;e++)await oe(t[e])}(t.update)):t.dns?(U("执行 dns 相关的命令"),await De(t.dns)):t.runOtherCode&&U("执行运行其他命令");try{await e()}catch(e){U.error("执行 run 报错",e)}}(),L(" 终结分割线 ",!0)}catch(e){U.error("执行 run 报错",e)}
2
+ import{Dog as e}from"@qqi/log";import{isFalse as n,isUndefined as t,isArray as o,isString as i,isEmptyObject as r,isNull as s,isBusinessEmptyString as a,isPlainObject as c,isBoolean as l,isEmptyArray as p,isZero as u}from"a-type-of-js";import{Command as d,enPrefixList as g}from"a-command";import{redPen as m,hexPen as f,randomPen as $,greenPen as w,yellowPen as y,pen as h,strInOneLineOnTerminal as v,hidePen as b,cyanPen as j,boldPen as k,italicPen as D,reversedPen as S}from"color-pen";import{_p as V,isWindows as R,runOtherCode as O,fileExist as A,detectPackageManager as E,cursorMoveUp as P,cursorAfterClear as C,getDirectoryBy as q,readFileToJsonSync as T,pathJoin as x,getNpmPkgInfo as L,waitingTips as N,colorLine as M,isTTY as F,__p as G,typewrite as I}from"a-node-tools";import U from"node:https";import{copyTextToClipboard as X}from"@qqi/copy-text";import{terminalRegExp as Y}from"@color-pen/static";import{Table as _}from"colored-table";import{intersection as B}from"a-js-tools";import H,{Resolver as J}from"node:dns";import W from"node:net";import{spawn as z}from"node:child_process";const K=new e({name:"jja",type:!1}),Q=n(K.type),Z=new d("jja");Z.bind({"package <pkg> (包管理)":["--diff <-d> (分析当下包的差异)","--upDependencies <-u> (更新依赖,强制更新到 'latest',使用需谨慎)"],"remove <rm> (做一个简单的兼容的移除文件或文件夹的命令)":["--ignore <-i> (不打印日志(默认打印的))","--subdirectories <-s> (这是一个危险的命令,用于 windows 下递归删除执行命令下所有的给定文件名称)"],"update <up> (做一个简单的 npm 升级程序,对,简单的)":["--ignore <-i> (不建议你这么用,你会发现你像个傻子一样在那等结果)",`--dependencies <-d> (更新依赖,跟 ${$("npm update")} 一样)`],"clearScreen <cls> (清理终端显示屏幕,同 clearTerminal )":"","clearTerminal <clear> (清理终端显示屏幕,同 clearScreen )":"","git (一些关于 git 的操作)":[`commit (git 提交代码,是 ${m("commit")} 提交啊,不是 ${f("#666")("push")} 推送)`,"merge (合并两个分支)","tag (给提交打上 tag)"],"dns (使用 dns 获取 ip 地址)":["--domain <-d> (获取给定域名获取 ip 地址,默认 github.com )","--port <-p> (配置要校验的 port 端口号,默认为 443)"],"runOtherCode <run> (执行其他的代码,使用 )":[]}).run(),K("参数已绑定,且已解析用户参数",Z.args),K.warn("当前的执行状态",Z.state),Z.isEnd(!0);const ee={log:!1,subdirectories:!1};async function ne(e,n,t=5){return!await Reflect.apply(e,void 0,n)&&t--?(V(w("执行失败,现在重试中")),K.warn(n,"执行失败,第",5-t,"次执行"),await ne(e,n,t)):await Reflect.apply(e,void 0,n)}function te(e,n){V(`清理 ${e} 文件${n?"完成":"失败"}`)}async function oe(e,n){ee.log||V(f("#aaa")(` ${e} 文件/夹存在,准备删除`));let t={error:null};return R?n.isDirectory()?(t=await O(`rd /q /s ${e.replace(/\//,"\\")}`),te(e,t.success)):n.isFile()&&(t=await O(`del ${ee.subdirectories?"/s":""} /q ${e.replace(/\//,"\\")}`),te(e,t.success)):(t=await O(`rm -rf ${e.replace(/\\/,"/")}`),te(e,t.success)),!!t.success||(console.log(t.error),!1)}async function ie(e){ee.log||V(f("#336")(`当前清理文件为 ${e}`));const n=A(e);ee.log||V(f("#666")(`正在检测 ${e} 文件/夹是否存在`)),K(`文件${e} 数据为 ${n}`),t(n)?ee.log||V(y(`${e} 文件不存在`)):await ne(oe,[e,n])}const re=(()=>{const e=E();return"npm"===e?"npm install --save":"yarn"===e?"yarn add":"pnpm add"})();async function se(e=!0){const n=q("package.json","file");if(null==n)return V("当前目录下不存在 package.json 文件");const t=T(x(n,"package.json"))?.dependencies||{};if(e&&V(w("初始化完成,等待下一步命令")),Object.keys(t).length>0){let e="npm install --save";for(let n=Object.keys(t),o=n.length,i=0;i<o;i++){e+=` ${n[i]}@latest `}await O(e)}var o;e&&V(j("正在更新 dev 依赖")),await O("npm update --save"),e&&(o="开发依赖更新完毕",P(),C(),V(o))}const ae={log:!0};async function ce(e){Boolean(e["--dependencies"])&&await se(ae.log)}const le={local:null,online:null,dependenceList:{},updateDependence:[],latestDependence:[],binning:function(e,n=!1){if(t(e)||r(e)||s(this.local))return;const{overrides:o}=this.local;for(const t in e){if(i(o?.[t])){Z.WARN(`${t} 被锁定在 ${o[t]}`);continue}const r=e[t];this.dependenceList[t]={type:n?"devDependencies":"dependencies",version:r,localVersion:"",latestVersion:"",onlineVersion:"",tag:"",time:""}}},preReleaseDependence:[],timeoutDependence:[],registry:void 0};const pe=k.green,ue=f("0x993311"),de=f("0x339999"),ge=f("0x666666");const me=N({show:!1,info:"请稍等",beforeDestroyed:e=>e&&Z.end()});async function fe(){const e={"淘宝":"registry.npmmirror.com","官方":"registry.npmjs.org","腾讯":"mirrors.tencent.com/npm","中科大":"npmreg.proxy.ustclug.org",yarn:"registry.yarnpkg.com"},t=()=>Object.keys(e).map(n=>function(e){const n=Date.now();return new Promise(t=>{const o=`https://${e}/nry`,i=U.request(o,{method:"HEAD",timeout:3500},o=>{t((o?.statusCode??0)>=200&&(o?.statusCode??0)<300?{url:e,time:Date.now()-n,success:!0}:{url:e,time:Date.now()-n,success:!1})});i.on("error",()=>t({url:e,time:Date.now()-n,success:!1})),i.on("timeout",()=>{i.destroy(),t({url:e,time:Date.now()-n,success:!1})}),i.end()})}(e[n])),o={};for(let e=0;e<3;e++){me.run(`请稍等,正在进行第${["一","二","三","四","五"][e]}次安装源延迟测试`);(await Promise.all(t())).forEach(e=>{n(e.success)||(o[e.url]?o[e.url]=(o[e.url]+e.time)/2:o[e.url]=e.time)})}return me.destroyed(),Object.keys(e).reduce((n,t)=>o[e[t]]<(o[e[n]]??1/0)?t:n,"")}async function $e(){const{dependenceList:e,timeoutDependence:n}=le,o=Object.keys(e);o.length>18&&(t(le.registry)||!["官方","淘宝","腾讯","中科大","yarn"].some(e=>e===le.registry))&&(le.registry=await fe(),V(`由于您未设置或者设置有误,现使用 ${j(le.registry)} npm 源`));const i=o.map(t=>new Promise(o=>{(async()=>{const i=e[t],r=e[t].version;me.run(`${g.current()}获取 ${t} 的本地安装信息`);const a=function(e){const n=T(x("node_modules/",e,"package.json"));let t="";return t=s(n)?"":n.version||"",le.dependenceList[e].localVersion=t,K(e,"本地安装的版本为:",t),t}(t);if(me.run(`${g.success()} ${t} 的本地安装版本:${i.localVersion}`),await async function(e){const n=await L(e,le.registry,9867);if("timeout"===n.status)return void K.warn("请求超时");if(s(n.data))return void K.warn("获取包信息失败");const t=Date.now();K("获取数据",e,"包信息");const o=le.dependenceList[e],{"dist-tags":i,time:r,version:a}=n.data;let c=a,l="latest";o.onlineVersion=c,o.tag=l;const p=Object.keys(i);K("共需比对",p.length);for(let e=0,n=p.length;e<n;e++){const n=p[e],t=i[n];r[t]>r[c]&&(c=t,l=n)}K("比对",e,"完成"),K("耗时",Date.now()-t),"latest"!==l&&(K(e,"最新版本为:",c,"tag为:",l),o.tag=l,o.latestVersion=c);const u=new Date(r[c]).toLocaleString();o.time=u,K(`${e} 的最后发布时间为:${u}`)}(t),""===i.onlineVersion&&""===i.latestVersion)return n.push(t),me.log(`${g.error()} ${t} 本地 ${i.localVersion} 请求错误`),o(!0);if(a===i.onlineVersion){const e=`${t} 的本地${r} 安装版本为 ${a} 最新是 ${i.onlineVersion} `;return""===i.latestVersion?(me.run(g.info()+e),o(!0)):(le.preReleaseDependence.push(t),me.log(`${g.info()} ${e};最新预发布版本为 ${w(i.latestVersion)}`),o(!0))}if(a===i.latestVersion)return me.log(`${g.info()} {key} 的本地版本${r} 安装版本为 ${a}`),o(!0);const c=`${t} 的本地 ${w(r)} 安装版本为 ${w(a)} 最新是 ${de(i.onlineVersion)} `;le.latestDependence.push(t),""===i.latestVersion?(me.log(g.info()+c),o(!0)):(le.preReleaseDependence.push(t),me.log(`${g.info()} ${c};最新预发布版本为 ${w(i.latestVersion)}`),o(!0))})()}));return Promise.all(i).then(()=>{me.destroyed()})}async function we(e){const{msg:n,list:t,type:r,copy:s}=e,a=h[r];V(),V(a.reversed(n),!1),V(s?ge.reversed`已复制到剪切板 📋`:""),V(),V(`${a(re)} ${b`\\`}\n${function(e){if(!o(e)||e.some(e=>!i(e)))throw new TypeError("参数必须为字符串数组");return e.map((n,t)=>t%3==2&&t!==e.length-1?`${n} ${b`\\`}\n`:n.concat(" ")).join("")}(t)}`),V(),s&&X(`${re} ${t.join(" ")}`.replace(Y(),""))}function ye(e){return`${k(e)}@${pe`latest`}`}function he(e,n){return`${D(e)}@${D.random(n)}`}async function ve(){const{local:e,preReleaseDependence:n,latestDependence:t,dependenceList:o,timeoutDependence:i}=le;if(s(e))return;const{dependencies:r,devDependencies:a}=e;if(le.binning(r),le.binning(a,!0),await $e(),0===n.length&&0===t.length&&0===i.length)return V(ge`看起来似乎没有依赖版本差异`);if(0===n.length&&0===t.length)return V(m`看起来网络不太好讷,所有的包线上版本的请求都出错了。或者是还没有执行 npm install 呐`);V(h.brightGreen`\n版本差异的依赖为:\n`);const c=[...t,...n];var l;new _({header:[j("包名"),{content:"发布时间",color:"#f26"},w`最新版本`],body:[...c.map(e=>{const{type:n,latestVersion:t,time:i,onlineVersion:r}=o[e];return["dependencies"===n?k(e):D(e),$(i),t||D(r)]})]})(),V(h.brightRed(`\n目前仅关注版本号是否为最新 ${h.brightMagenta("latest")}`)),V(`使用 ${re} 命令安装更新\n`),i.length>0&&(l=m`有一些包没有返回结果,请注意:`,V(v(l)),new _([...i.map(e=>[e])])());const p=[...t.filter(e=>!1===n.includes(e)).map(e=>h.bold(e).concat("@latest")),...n.map(e=>he(e,o[e].tag))],u=t.map(e=>ye(e)),d=[...t.map(e=>ye(e)),...n.filter(e=>!1===t.includes(e)).map(e=>he(e,o[e].tag))],g=p.length<n.length+t.length,f=n.length>0,y=t.length>0;f&&(g&&await we({msg:"‼️ 预发布版本优先:",list:p,type:"brightRed"}),await we({msg:"⚠️ latest 版本优先:",list:d,type:"brightMagenta",copy:!y})),y&&await we({msg:"🎉 最佳安装:",list:u,type:"brightGreen",copy:!0})}async function be(e){le.registry=e[0]?.toString()||void 0,await async function(){const e=q("package.json","file");if(t(e))return void K.warn("未找到当前包的 package.json 文件的位置");const n=T(x(e,"package.json"));if(s(n))return void K.error("未找到当前包的 package.json 文件,该事件发生的概率极低");const o=n.name||"",i=await L(o,le.registry,9800);le.local=n,le.online=i.data}();const{local:n,online:o}=le,i=n?.version||"",r=o?.version||"";if(s(n))return K("未找到包 package.json 文件退出"),V(ue("未找到当前包 package.json"));const a=ue(`当前包本地版本为: ${i}`);if(o){V(a);const e=" ".repeat(6);V(de(`${e}线上版本为:${r}@latest`));const n=new Date(o&&o.time.modified||"").toLocaleString();Z.INFO(`最后的发布时间为: ${n.toString()}`),M()}await ve()}async function je(e){e["--upDependencies"]?await se(!0):e["--diff"]?await be(e["--diff"]):Z.help("package")}async function ke(e){const n=q("package.json","file",process.cwd());if(null==n)return!0;const t=T(x(n,"package.json"));if(s(t)||!t.version)return!0;const o=t.version;await O({code:`git tag -a v${o} -m '${e}'`,cwd:n}),await O({code:"git push origin --tag",cwd:n})}async function De(e){e.commit?await async function(e="版本维护",n=!1){const t=q(".git","directory");if(null==t)return V(f("#ff0")("not a git repository(当前目录非 git 储存库)"));const o=await O({code:"git add .",cwd:t});if(!o.success)return null!=console.log(o.error);const i=await O({code:"git status",cwd:t});if(!/nothing to commit, working tree clean/.test(i.data||"")){const o=await O({code:"git add .",cwd:t});if(!o.success)return null!=console.log(o.error);""==e.trim()&&(e=await Z.question({text:"请输入提交信息"})??"");const i=await O({code:`git commit -m "${e}"`,cwd:t});return i.success?(i.error&&console.log(i.error),await O({code:"git push",cwd:t}),n&&await ke(e),!0):null!=console.log(i.error)}return!1}(e.commit.join(" ")):e.merge?await async function(e){if(a(e)){const n=await O("git branch -a");console.log(n.data),e=await Z.question({text:"请输入要合并分支的名称"})??""}if(a(e))return Z.ERROR("没有获取到要合并的分支");const n=await Z.selection({data:[{label:"正常快进合并",value:""},{label:"非快进合并 (--no-ff)",value:"--no-ff"},{label:"多提交记录合并为一条 (--squash)",value:"--squash",tip:"适用于将众多小的提交合并成一个提交"}]});t(n)?Z.ERROR("您选择退出合并"):await O(`git merge ${e} ${n}`)}(e.merge.join("")):e.tag?await async function(){const e=await Z.question({text:"请输入待标记的信息"});t(e)||await ke(e)}():Z.help("git")}const Se=["A","AAAA","ANY","CAA","CNAME","MX","NAPTR","NS","PTR","SOA","SRV","TLSA","TXT"],Ve={domain:"github.com",port:443,dnsServers:["1.1.1.1","8.8.8.8","8.8.4.4","208.67.222.222","114.114.115.115","223.5.5.5","180.76.76.76"],ips:{},rrtype:["A"],reset(e){this.ips={},this.domain="github.com",this.port=443,this.rrtype=[],t(e)||e.forEach(e=>{const n=e["--domain"];!t(n)&&n.length>0&&(this.domain=n[0].toString());const o=e["--port"];!t(o)&&o.length>0&&(this.port=Number(o[0]));const i=e["--rrtype"];if(!t(i)&&i.length>0){const e=B(Se,i);e.length>0&&(this.rrtype=e)}})},noCopy:!0};async function Re(e){return Promise.all(e.map(e=>async function(e,n=3e3){const{ips:t,port:o}=Ve;if(l(t[e]))return Q||me.log(`${e} 联通性已校验过 ${t[e]?"🔥":"⛓️‍💥"}`),Promise.resolve(null);t[e]=!1,Q||me.log(`校验 ${e} 的联通性`);return new Promise(i=>{const r=new W.Socket,s=(n=!1)=>{t[e]=n,r.destroy(),i(null)};r.setTimeout(n),r.on("connect",()=>s(!0)),r.on("timeout",()=>s()),r.on("error",()=>s()),Q||me.log("测试的接口为:",o),r.connect(o,e)})}(e)))}async function Oe(e="1.1.1.1"){const{domain:n}=Ve;try{const t=await async function(e,n){return new Promise(t=>{const o=new J;me.run(`使用 ${n} 获取 ${e} 的 ip`),o.setServers([n]),o.resolve4(e,(o,i)=>{if(o)return Q||me.run(`使用 ${n} 获取 ${e} 的 ip 出错`),t(null);Q||me.log(`使用 ${n} 获取 ${e} 的 ip 值为 ${i.join(" --- ")}`),t(i)})})}(n,e);if(s(t))return Q||me.log(`${e} 获取 ${n} 的 ip 失败`),0;const o=(await Re(t)).filter(e=>!s(e));return p(o)?0:(me.destroyed(),V(),V(S`DNS: ${e}`),V(),o.forEach(({ip:e,isAlive:n})=>{const t=n?`${w(e)} ${ue`->`} ✅`:`${ge(e)} ${m`⛓️‍💥`} ❌`;V(v(t))}),3+o.length)}catch(e){return Q||me.log("'获取 ip 错误'",e),0}}async function Ae(){const{domain:e,ips:n}=Ve;me.run(`正在获取本地的 ${e} 的 ip 地址`);try{const t=await async function(e){return new Promise(n=>{H.lookup(e,{all:!0,family:4},(t,o)=>{if(t)return K.error(`获取本地 ${e} 的 ip 出错`,t),void n(null);Q||me.log(`获取本地 ${e} 的 ip 值为 ${o.join(" --- ")}`),n(o.map(e=>e.address))})})}(e);if(s(t))return;await Re(t),r(n)||function(e){const{ips:n}=Ve;V(ue`本地配置 ${e} 的 ip 地址及联通性为:\n`),Object.keys(n).forEach(e=>{const t=n[e];t&&(Ve.noCopy=!1),V(j`- ${e}`,!1),V(t?ue` -> ✅`:ge` -> ❌`)}),V()}(e)}catch(e){K.error(e)}}async function Ee(e){Ve.reset(e.options),await Ae(),await async function(){const{domain:e,dnsServers:n}=Ve;me.log(`${e} ip 列表:`);const t=n.map(e=>Oe(e)),o=await Promise.all(t);Q&&(P(o.reduce((e,n)=>e+n,0)+1),C(!0))}(),me.destroyed();const{ips:n}=Ve;Object.keys(n).length>0?await async function(){const{domain:e,ips:n,port:t}=Ve;await I(`${h.brightMagenta`${e}`} 域名解析结果:`),V();let o=Ve.noCopy;for(const t in n)if(Object.prototype.hasOwnProperty.call(n,t)){const i=n[t],r=i?`${w`- ${t.padEnd(16)}`} ${ue`->`} ✅${o?j`(已复制)`:""}`:`${ge`- ${t.padEnd(16)}`} ${m`⛓️‍💥`} ❌`;V(r),i&&o&&(X(`${t} ${e}`),o=!1)}V(),await I(m.italic.dim`${e} 联通性接口判断为 ${t.toString()}`),V()}():await async function(){const{domain:e,port:n}=Ve;await I(`\n找不到 ${m(e)} 的服务器 IP 地址 ${ge`测试使用 ${n.toString()} 端口`}\n`)}()}const Pe=Z.args.$arrMap;Z.args.$isVoid?(K("没有匹配到子命令,打印帮助信息并退出"),Z.help(),Z.end()):p(Z.args.$only)&&(console.log(`${Z.args.$nomatch.map(e=>$(e)).join("、")} 不是有效的参数`),Z.end());try{await async function e(){if(0===Pe.length)return;const n=Pe.shift();if(t(n))return await e();n.remove?(K("执行文件移除"),await async function(e){const n=e.value||[],t=e.options;for(t?.forEach(e=>{const t=e["--ignore"];t&&(ee.log=!0,n.concat(t));const o=e["--subdirectories"];o&&(ee.subdirectories=!0,n.concat(o))}),ee.log||V(h.brightCyan`当前系统为: ${R?"windows":"linux/mac"}`),0==n.length&&(K.warn("待删除列表",n),V(h.random`没有待清理的文件/文件夹`));n.length;){let e=n.pop();e=void 0===e?"undefined":e.toString(),await ie(e)}}(n.remove)):n.clearScreen||n.clearTerminal?(K("执行清屏"),await async function(){F()?(K("执行第一遍清理"),await O({code:R?"cls":"clear",printLog:!1}),K("执行第二遍清理"),G("3J"),G("c",!1),P(1/0),C(!0)):(K.warn("当前系统不支持 TTY"),V("当前环境不支持 "))}()):n.git?(K("执行 git 相关命令"),await async function(e){if(c(e)&&(e={value:[]}),t(e.options)||0===e.options.length)return K.warn("没有配置项,直接返回的帮助文档"),Z.help("git");const n=e.options;for(let e=0,t=n.length;e<t;e++)K("本次执行",n[e]),await De(n[e])}(n.git)):n.package?(K("执行 package 相关命令"),await async function(e){if(t(e.options)||0==e.options.length)return K.warn("没有参数输入"),Z.help("package");const n=e.options;for(let e=0,t=n.length;e<t;e++)await je(n[e])}(n.package)):n.update?(K("执行 update 相关命令"),await async function(e){if(t(e.options)||0==e.options.length)return K.warn("没有匹配到选项,自动抛出帮助文档"),Z.help("update");const n=e.options;ae.log=!n.some(e=>e["--ignore"]);for(let e=0,t=n.length;e<t;e++)await ce(n[e])}(n.update)):n.dns?(K("执行 dns 相关的命令"),await Ee(n.dns)):n.runOtherCode&&(K("执行运行其他命令"),await async function(e){const{value:n}=e;if(t(n)||u(n.length))return;for(const e in n){const t=n[e].toString();if(t.includes("=")){const[o,i]=t.split("=");o&&i?(process.env[o]=i,Z.SUCCESS(`设置环境变量 ${o}=${i} ✅`),n[e]=""):(Z.ERROR(`环境变量格式错误: ${t}`),Z.error())}break}const o=n.join(" ").trim();K("执行代码",o);const i=z(o,[],{stdio:"inherit",env:process.env,shell:!0});i.on("close",e=>e),i.on("error",e=>e),i.on("exit",e=>e)}(n.runOtherCode));try{await e()}catch(e){K.error("执行 run 报错",e)}}(),M(" 终结分割线 ",!0)}catch(e){K.error("执行 run 报错",e)}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "jja",
3
3
  "type": "module",
4
- "version": "2.3.17",
4
+ "version": "2.3.18-beta.0",
5
5
  "description": "一些在终端的执行动作 🥜",
6
6
  "license": "MIT",
7
7
  "dependencies": {
@@ -9,7 +9,7 @@
9
9
  "@qqi/copy-text": "^1.0.6",
10
10
  "@qqi/log": "^0.1.1",
11
11
  "a-command": "^2.3.15",
12
- "a-js-tools": "^1.0.10",
12
+ "a-js-tools": "^1.0.12",
13
13
  "a-node-tools": "^4.2.19",
14
14
  "a-type-of-js": "^1.0.7",
15
15
  "color-pen": "^2.0.13",