Compare commits
77 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 57e4d1c043 | |||
| f495f85bdb | |||
| d53e8fec6d | |||
| 00fef1ae06 | |||
| 4c1608cf94 | |||
| e0c4cf2983 | |||
| e3eb0af434 | |||
| 8d25d28b54 | |||
| 4237e49f14 | |||
| 019f7e2d88 | |||
| abaeb7be6c | |||
| ab714377ba | |||
| 450b3ba379 | |||
| 0daae0bd07 | |||
| 826f3a6c63 | |||
| c1a4671ce1 | |||
| 73a6d5bc31 | |||
| e15a569391 | |||
| 74809bd8d4 | |||
| 5b852d9602 | |||
| f68254eea7 | |||
| 5935c67f2e | |||
| 98f5353744 | |||
| 1372a2bb7e | |||
| fa16aaa2c1 | |||
| c3892096b6 | |||
| 4e6646b133 | |||
| da6739be6a | |||
| f72217250d | |||
| 4c36174b0d | |||
| 5f34bc8042 | |||
| 6d354a10d0 | |||
| 8208ba2970 | |||
| 3c2b51d47e | |||
| 12fab84daa | |||
| 4b5b683fff | |||
| ec1958d882 | |||
| 1f59cd4ba1 | |||
| 74c0d537cc | |||
| c4706e96cb | |||
| 628b86cf3d | |||
| d0c6ebb0df | |||
| dd8c30e7cf | |||
| 8f861d86c9 | |||
| fe2581b533 | |||
| db906cea1a | |||
| 9c1dca9ace | |||
| c620549476 | |||
| 90697584d7 | |||
| 1252fa8f97 | |||
| cdbc9823f1 | |||
| 9d7023e739 | |||
| 3bb3860f44 | |||
| fc22ea9e4d | |||
| 1afa78307c | |||
| ab9ca8f15f | |||
| e51e692fef | |||
| 52bb5bd302 | |||
| 139349a38c | |||
| f01e275044 | |||
| 82b6654b94 | |||
| b10059bb1b | |||
| 87c1a9a29d | |||
| 551287d34d | |||
| 64fc8abe07 | |||
| af7f47c340 | |||
| 8e6b9b669e | |||
| f3057ede95 | |||
| 2e20c5b3cc | |||
| 939016347d | |||
|
|
8a2cc29969 | ||
| a716455ef4 | |||
| a67a946fa6 | |||
| 929a7c7fbb | |||
| bf9071b093 | |||
| 917250772f | |||
| e16de55993 |
28
.gitignore
vendored
28
.gitignore
vendored
@@ -1,7 +1,23 @@
|
||||
.idea/
|
||||
node_modules/
|
||||
.nogit/
|
||||
|
||||
# artifacts
|
||||
coverage/
|
||||
docs/
|
||||
ts/typings/
|
||||
ts/**/*.js
|
||||
ts/**/*.js.map
|
||||
public/
|
||||
|
||||
# installs
|
||||
node_modules/
|
||||
|
||||
# caches
|
||||
.yarn/
|
||||
.cache/
|
||||
.rpt2_cache
|
||||
|
||||
# builds
|
||||
dist/
|
||||
dist_*/
|
||||
|
||||
# AI
|
||||
.claude/
|
||||
.serena/
|
||||
|
||||
#------# custom
|
||||
12
.travis.yml
12
.travis.yml
@@ -1,12 +0,0 @@
|
||||
language: node_js
|
||||
node_js:
|
||||
- '0.11'
|
||||
- '0.10'
|
||||
deploy:
|
||||
provider: npm
|
||||
email: npm@lossless.digital
|
||||
api_key:
|
||||
secure: tHwuWIQl9/AL42/TP84DnZ/QutUPlja9p4lxLtP7Nm0LXRJtAnXkZ4qKqXagLMKy2KdQiJykrHTo1bwr/e/NGtmtBLhfLpvkCKOAKfX+o9ZB7vw998aSSCKKr7IFVR/78JMkXOcmgeKaPXdkqHwX5DTVMkl5ImaX7dOuRB/VL5t+nLRchVP/vAXiyuIyKYPEPEBR7mXwRGaGvKj3Ko87qtkIx4jUaMGyA9BMYT5u9iVGY3q8IgcCIoVk+hbBx+UkISCs0UOegILldrTRS36/OMnOitDbCV43o7j9/R7VhT7pDikITmdJQ7ePC+rdCn8JsbRLntqFnvrrjWuJThF6EmiH2C8aKwcZ1YzeJf/kDgo23j5fsBFDDNQfBVuCitY0fvE7aFG5Za2JySfXn58HyROF/LM1fX8dFM9LOLf+UHNH/iMxyy+WRPrAApbjHbq/rfwbQj09oAkWB6Wvbehn9h86VxsABjE3OBFAP3y0zA/t0PB6IcGtJ9edazB6VuHyiMnucKrr7XPOBRf/t9kgWo7itGPbVfSwDUUbUpUTIJxJtQ8VMltb4RbrwVQRF5J15hEpt1Mit1aVIbcp4hyewkNvnttnO58S//A5fU4m9sHaGxi3yNq5hDpxH5Uu4cCNvC2/y2JQDUywPuwme44iKPGxGkMmeJ5Ea5hU+MCgrvM=
|
||||
on:
|
||||
tags: true
|
||||
repo: GitZoneTools/node-g
|
||||
29
.vscode/launch.json
vendored
Normal file
29
.vscode/launch.json
vendored
Normal file
@@ -0,0 +1,29 @@
|
||||
{
|
||||
"version": "0.2.0",
|
||||
"configurations": [
|
||||
{
|
||||
"name": "current file",
|
||||
"type": "node",
|
||||
"request": "launch",
|
||||
"args": [
|
||||
"${relativeFile}"
|
||||
],
|
||||
"runtimeArgs": ["-r", "@git.zone/tsrun"],
|
||||
"cwd": "${workspaceRoot}",
|
||||
"protocol": "inspector",
|
||||
"internalConsoleOptions": "openOnSessionStart"
|
||||
},
|
||||
{
|
||||
"name": "test.ts",
|
||||
"type": "node",
|
||||
"request": "launch",
|
||||
"args": [
|
||||
"test/test.ts"
|
||||
],
|
||||
"runtimeArgs": ["-r", "@git.zone/tsrun"],
|
||||
"cwd": "${workspaceRoot}",
|
||||
"protocol": "inspector",
|
||||
"internalConsoleOptions": "openOnSessionStart"
|
||||
}
|
||||
]
|
||||
}
|
||||
26
.vscode/settings.json
vendored
Normal file
26
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,26 @@
|
||||
{
|
||||
"json.schemas": [
|
||||
{
|
||||
"fileMatch": ["/npmextra.json"],
|
||||
"schema": {
|
||||
"type": "object",
|
||||
"properties": {
|
||||
"npmci": {
|
||||
"type": "object",
|
||||
"description": "settings for npmci"
|
||||
},
|
||||
"gitzone": {
|
||||
"type": "object",
|
||||
"description": "settings for gitzone",
|
||||
"properties": {
|
||||
"projectType": {
|
||||
"type": "string",
|
||||
"enum": ["website", "element", "service", "npm", "wcc"]
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
2
LICENSE
2
LICENSE
@@ -1,6 +1,6 @@
|
||||
The MIT License (MIT)
|
||||
|
||||
Copyright (c) 2016 Git.Zone
|
||||
Copyright (c) 2016 Task Venture Capital GmbH
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
|
||||
131
changelog.md
Normal file
131
changelog.md
Normal file
@@ -0,0 +1,131 @@
|
||||
# Changelog
|
||||
|
||||
## 2026-02-05 - 3.2.0 - feat(update)
|
||||
enhance package manager detection, version reporting, and add verbose option
|
||||
|
||||
- Add IPackageManagerInfo interface and detectPackageManager() to robustly detect npm/yarn/pnpm via 'which' and '--version' fallbacks
|
||||
- Make isAvailable() delegate to detectPackageManager() and return structured detection info
|
||||
- Add getPackageManagerVersion() to obtain current and latest versions (parses local --version and queries npm registry)
|
||||
- Update run() to support a verbose flag, show a package-manager status table, and collect detectedPMs with version/update status
|
||||
- Update CLI help and command handling to accept --verbose/-v and pass it through to mod_update.run()
|
||||
|
||||
## 2026-02-03 - 3.1.3 - fix(mod_update)
|
||||
try private registry (verdaccio.lossless.digital) first when fetching package versions; fall back to public npm; handle unknown latest versions gracefully in output
|
||||
|
||||
- getLatestVersion now attempts a direct API request to https://verdaccio.lossless.digital/<encoded-package> and parses dist-tags.latest
|
||||
- Falls back to npm view when the private registry request fails
|
||||
- Scoped package names are URL-encoded (replaces '/' with '%2f') before querying the private registry
|
||||
- Packages with no resolvable latest version are included with latestVersion set to 'unknown' and displayed as '? Version unknown'
|
||||
- needsUpdate is set to false when latest version is unknown
|
||||
|
||||
## 2026-02-03 - 3.1.2 - fix(scripts)
|
||||
make test script output verbose by using --verbose instead of --web
|
||||
|
||||
- package.json: change npm "test" script from "(tstest test/ --web)" to "(tstest test/ --verbose)" to enable verbose test output
|
||||
|
||||
## 2026-02-03 - 3.1.1 - fix(tools)
|
||||
no changes detected
|
||||
|
||||
- No files were modified in this diff
|
||||
- No code or documentation changes to include in a commit message
|
||||
|
||||
## 2026-02-03 - 3.1.0 - feat(cli)
|
||||
add update command to check and update globally installed @git.zone packages
|
||||
|
||||
- Adds a new mod_update module and PackageManagerUtil to detect installed @git.zone packages and fetch latest versions
|
||||
- Supports npm, yarn and pnpm: detection, listing, version comparison and executing updates
|
||||
- Interactive confirmation via @push.rocks/smartinteract with a -y flag to skip prompt
|
||||
- Uses @push.rocks/smartshell to run shell commands for listing and updating packages
|
||||
- Wires the new 'update' command into the CLI (tools.cli) and updates CLI version to 3.0.0
|
||||
- Adds CLI helper files (cli.child.js, adjusts cli.js and cli.ts.js) to run TypeScript CLI via tsrun/runPath
|
||||
- Updates package.json to include new dependencies and npmextra.json with release registries and @git.zone/cli metadata
|
||||
- Updates .gitignore to exclude local AI tool dirs (.claude, .serena)
|
||||
|
||||
## 2026-02-03 - 3.0.0 - BREAKING CHANGE(tools)
|
||||
replace install functionality with a minimal placeholder CLI; remove installer, logging, path utilities, and related assets
|
||||
|
||||
- Removed installer implementation (ts/tools.install.ts) and package library asset (assets/package_library.json).
|
||||
- Removed logging and path utilities (ts/tools.logging.ts, ts/tools.paths.ts) and simplified plugins (ts/tools.plugins.ts) to only export smartcli.
|
||||
- Reworked CLI (ts/tools.cli.ts) to a placeholder standard command that prints messages instead of running installs; internal CLI version set to 2.0.22.
|
||||
- Module export/behavior changed: ts/index.ts now exports runCli and no longer auto-runs; tests updated to expect runCli.
|
||||
- package.json updated: description changed, several dependencies removed (e.g. @push.rocks/smartlog, @push.rocks/smartshell), @types/node bumped to ^22.0.0, and assets removed from packaged files list.
|
||||
- Removed project policy and metadata files: .snyk deleted and readme.md added with usage and legal information.
|
||||
|
||||
## 2026-02-03 - 2.0.22 - core
|
||||
Core maintenance release.
|
||||
|
||||
- fix(core): update — minor core fixes and maintenance improvements.
|
||||
|
||||
## 2026-02-03 - 2.0.23 - release tag
|
||||
Release tag only — no recorded changelog details.
|
||||
|
||||
- 2.0.23: release tag with no additional commit message.
|
||||
|
||||
## 2020-10-05 - 2.0.3 - 2.0.21 - core (maintenance)
|
||||
Series of maintenance releases containing repeated small core fixes.
|
||||
|
||||
- Multiple commits "fix(core): update" applied across 2.0.3 through 2.0.21 addressing minor bugs and stability improvements.
|
||||
- These releases are maintenance-focused; no user-facing feature additions recorded.
|
||||
|
||||
## 2019-06-17 - 2.0.2 - core
|
||||
Maintenance update to core.
|
||||
|
||||
- fix(core): update — continued minor fixes and upkeep.
|
||||
|
||||
## 2018-06-07 - 2.0.0 - 2.0.1 - policy/package fixes
|
||||
Initial 2.0.x releases with policy and package fixes.
|
||||
|
||||
- 2.0.0: fix(snyk policy): update — update to Snyk policy.
|
||||
- 2.0.1: fix(package.json): fix private property — corrected package.json property.
|
||||
- 2.0.2: release tag (later maintenance continued in subsequent 2.0.x).
|
||||
|
||||
## 2018-06-07 - 1.0.9 - core (feature)
|
||||
New core scope added.
|
||||
|
||||
- feat(core): new tools scope — introduces a "tools" scope in core.
|
||||
|
||||
## 2017-06-04 - 1.0.8 - release tag
|
||||
Release tag only — no recorded changelog details.
|
||||
|
||||
- 1.0.8: release tag with no additional commit message.
|
||||
|
||||
## 2017-06-04 - 1.0.7 - cli
|
||||
CLI update.
|
||||
|
||||
- update cli — improvements/updates to the command-line interface.
|
||||
|
||||
## 2017-05-28 - 1.0.6 / 1.0.5 - docs & CLI
|
||||
Documentation and CLI additions.
|
||||
|
||||
- 1.0.6: update README — documentation updates.
|
||||
- 1.0.5: add cli.js — adds CLI entry point.
|
||||
|
||||
## 2017-05-28 - 1.0.4 - maintenance and cleanups
|
||||
Collection of maintenance, cleanup, and packaging updates for 1.0.4 over several commits.
|
||||
|
||||
- update / cleanup — general code cleanups.
|
||||
- Update packageLibrary.json — package metadata adjusted.
|
||||
- added cli.js, added .npmignore — packaging and CLI support added.
|
||||
- Consolidated multiple 1.0.4 changes across 2016–2017 into this release.
|
||||
|
||||
## 2016-03-03 - 1.0.3 - tests
|
||||
Test fix.
|
||||
|
||||
- fix test.ts — test suite correction.
|
||||
|
||||
## 2016-03-03 - 1.0.2 - CI
|
||||
Continuous integration fix.
|
||||
|
||||
- fix Travis — adjusts CI configuration.
|
||||
|
||||
## 2016-03-02 - 1.0.1 - install/tests
|
||||
Installer and tests improvements.
|
||||
|
||||
- now installing things correctly — fixes installation behavior.
|
||||
- create test / update — test additions and miscellaneous updates.
|
||||
|
||||
## 2016-03-02 - 1.0.0 - initial release
|
||||
Project initial structure and first release.
|
||||
|
||||
- add structure / set up initial structure / cleanup — initial project layout and housekeeping.
|
||||
- initial (2016-02-29 and 2016-02-24) — first commits establishing the repository.
|
||||
4
cli.child.ts
Normal file
4
cli.child.ts
Normal file
@@ -0,0 +1,4 @@
|
||||
#!/usr/bin/env node
|
||||
process.env.CLI_CALL = 'true';
|
||||
import * as cliTool from './ts/index.js';
|
||||
cliTool.runCli();
|
||||
4
cli.js
Normal file
4
cli.js
Normal file
@@ -0,0 +1,4 @@
|
||||
#!/usr/bin/env node
|
||||
process.env.CLI_CALL = 'true';
|
||||
const cliTool = await import('./dist_ts/index.js');
|
||||
cliTool.runCli();
|
||||
5
cli.ts.js
Normal file
5
cli.ts.js
Normal file
@@ -0,0 +1,5 @@
|
||||
#!/usr/bin/env node
|
||||
process.env.CLI_CALL = 'true';
|
||||
|
||||
import * as tsrun from '@git.zone/tsrun';
|
||||
tsrun.runPath('./cli.child.js', import.meta.url);
|
||||
7
dist/index.js
vendored
7
dist/index.js
vendored
@@ -1,7 +0,0 @@
|
||||
var install = require("./npmg.install");
|
||||
var npmg = {
|
||||
install: install
|
||||
};
|
||||
module.exports = npmg;
|
||||
|
||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImluZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLElBQU8sT0FBTyxXQUFXLGdCQUFnQixDQUFDLENBQUM7QUFDM0MsSUFBSSxJQUFJLEdBQUc7SUFDUCxPQUFPLEVBQUUsT0FBTztDQUNuQixDQUFDO0FBQ0YsaUJBQVMsSUFBSSxDQUFDIiwiZmlsZSI6ImluZGV4LmpzIiwic291cmNlc0NvbnRlbnQiOlsiLy8vIDxyZWZlcmVuY2UgcGF0aD1cIi4vdHlwaW5ncy9tYWluLmQudHNcIiAvPlxuaW1wb3J0IHBsdWdpbnMgPSByZXF1aXJlKFwiLi9ucG1nLnBsdWdpbnNcIik7XG5pbXBvcnQgaW5zdGFsbCA9IHJlcXVpcmUoXCIuL25wbWcuaW5zdGFsbFwiKTtcbmxldCBucG1nID0ge1xuICAgIGluc3RhbGw6IGluc3RhbGxcbn07XG5leHBvcnQgPSBucG1nOyJdLCJzb3VyY2VSb290IjoiL3NvdXJjZS8ifQ==
|
||||
3
dist/npmg.cli.js
vendored
3
dist/npmg.cli.js
vendored
@@ -1,3 +0,0 @@
|
||||
|
||||
|
||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsImZpbGUiOiJucG1nLmNsaS5qcyIsInNvdXJjZXNDb250ZW50IjpbXSwic291cmNlUm9vdCI6Ii9zb3VyY2UvIn0=
|
||||
25
dist/npmg.install.js
vendored
25
dist/npmg.install.js
vendored
@@ -1,25 +0,0 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
var plugins = require("./npmg.plugins");
|
||||
var paths = require("./npmg.paths");
|
||||
var installExec = function (packageNames) {
|
||||
for (var packageName in packageNames) {
|
||||
var execCommand = "npm install -g " + packageNames[packageName];
|
||||
plugins.beautylog.info("now installing " + packageNames[packageName]);
|
||||
plugins.shelljs.exec(execCommand);
|
||||
}
|
||||
;
|
||||
};
|
||||
var packageLibrary = plugins.smartfile.readFileToObject(plugins.path.join(paths.packageBase, "packageLibrary.json"));
|
||||
var install = function (packageSetArg) {
|
||||
switch (packageSetArg) {
|
||||
case "default":
|
||||
installExec(packageLibrary.default);
|
||||
break;
|
||||
default:
|
||||
plugins.beautylog.warn("no set has been specified");
|
||||
break;
|
||||
}
|
||||
};
|
||||
module.exports = install;
|
||||
|
||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5wbWcuaW5zdGFsbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw0Q0FBNEM7QUFDNUMsSUFBTyxPQUFPLFdBQVcsZ0JBQWdCLENBQUMsQ0FBQztBQUMzQyxJQUFPLEtBQUssV0FBVyxjQUFjLENBQUMsQ0FBQztBQUN2QyxJQUFJLFdBQVcsR0FBRyxVQUFTLFlBQXFCO0lBQzVDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxXQUFXLElBQUksWUFBWSxDQUFDLENBQUEsQ0FBQztRQUNsQyxJQUFJLFdBQVcsR0FBRyxpQkFBaUIsR0FBRyxZQUFZLENBQUMsV0FBVyxDQUFDLENBQUM7UUFDaEUsT0FBTyxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLEdBQUcsWUFBWSxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUM7UUFDdEUsT0FBTyxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDdEMsQ0FBQztJQUFBLENBQUM7QUFDTixDQUFDLENBQUM7QUFFRixJQUFJLGNBQWMsR0FBRyxPQUFPLENBQUMsU0FBUyxDQUFDLGdCQUFnQixDQUNuRCxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxFQUFDLHFCQUFxQixDQUFDLENBQzdELENBQUM7QUFFRixJQUFJLE9BQU8sR0FBRyxVQUFTLGFBQW9CO0lBQ3ZDLE1BQU0sQ0FBQyxDQUFDLGFBQWEsQ0FBQyxDQUFBLENBQUM7UUFDbkIsS0FBSyxTQUFTO1lBQ1YsV0FBVyxDQUFDLGNBQWMsQ0FBQyxPQUFPLENBQUMsQ0FBQztZQUNwQyxLQUFLLENBQUM7UUFDVjtZQUNJLE9BQU8sQ0FBQyxTQUFTLENBQUMsSUFBSSxDQUFDLDJCQUEyQixDQUFDLENBQUM7WUFDcEQsS0FBSyxDQUFDO0lBQ2QsQ0FBQztBQUNMLENBQUMsQ0FBQztBQUVGLGlCQUFTLE9BQU8sQ0FBQyIsImZpbGUiOiJucG1nLmluc3RhbGwuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvLy8gPHJlZmVyZW5jZSBwYXRoPVwiLi90eXBpbmdzL21haW4uZC50c1wiIC8+XG5pbXBvcnQgcGx1Z2lucyA9IHJlcXVpcmUoXCIuL25wbWcucGx1Z2luc1wiKTtcbmltcG9ydCBwYXRocyA9IHJlcXVpcmUoXCIuL25wbWcucGF0aHNcIik7XG5sZXQgaW5zdGFsbEV4ZWMgPSBmdW5jdGlvbihwYWNrYWdlTmFtZXM6c3RyaW5nW10pe1xuICAgIGZvciAobGV0IHBhY2thZ2VOYW1lIGluIHBhY2thZ2VOYW1lcyl7XG4gICAgICAgIGxldCBleGVjQ29tbWFuZCA9IFwibnBtIGluc3RhbGwgLWcgXCIgKyBwYWNrYWdlTmFtZXNbcGFja2FnZU5hbWVdO1xuICAgICAgICBwbHVnaW5zLmJlYXV0eWxvZy5pbmZvKFwibm93IGluc3RhbGxpbmcgXCIgKyBwYWNrYWdlTmFtZXNbcGFja2FnZU5hbWVdKTtcbiAgICAgICAgcGx1Z2lucy5zaGVsbGpzLmV4ZWMoZXhlY0NvbW1hbmQpO1xuICAgIH07XG59O1xuXG5sZXQgcGFja2FnZUxpYnJhcnkgPSBwbHVnaW5zLnNtYXJ0ZmlsZS5yZWFkRmlsZVRvT2JqZWN0KFxuICAgIHBsdWdpbnMucGF0aC5qb2luKHBhdGhzLnBhY2thZ2VCYXNlLFwicGFja2FnZUxpYnJhcnkuanNvblwiKVxuKTtcblxubGV0IGluc3RhbGwgPSBmdW5jdGlvbihwYWNrYWdlU2V0QXJnOlN0cmluZyl7XG4gICAgc3dpdGNoIChwYWNrYWdlU2V0QXJnKXtcbiAgICAgICAgY2FzZSBcImRlZmF1bHRcIjpcbiAgICAgICAgICAgIGluc3RhbGxFeGVjKHBhY2thZ2VMaWJyYXJ5LmRlZmF1bHQpO1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICBwbHVnaW5zLmJlYXV0eWxvZy53YXJuKFwibm8gc2V0IGhhcyBiZWVuIHNwZWNpZmllZFwiKTtcbiAgICAgICAgICAgIGJyZWFrO1xuICAgIH1cbn07XG5cbmV4cG9ydCA9IGluc3RhbGw7Il0sInNvdXJjZVJvb3QiOiIvc291cmNlLyJ9
|
||||
8
dist/npmg.paths.js
vendored
8
dist/npmg.paths.js
vendored
@@ -1,8 +0,0 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
var plugins = require("./npmg.plugins");
|
||||
var paths = {
|
||||
packageBase: plugins.path.join("__dirname", "../")
|
||||
};
|
||||
module.exports = paths;
|
||||
|
||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5wbWcucGF0aHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsNENBQTRDO0FBQzVDLElBQU8sT0FBTyxXQUFXLGdCQUFnQixDQUFDLENBQUM7QUFDM0MsSUFBSSxLQUFLLEdBQUc7SUFDUixXQUFXLEVBQUUsT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFDLEtBQUssQ0FBQztDQUNwRCxDQUFBO0FBQ0QsaUJBQVMsS0FBSyxDQUFDIiwiZmlsZSI6Im5wbWcucGF0aHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvLy8gPHJlZmVyZW5jZSBwYXRoPVwiLi90eXBpbmdzL21haW4uZC50c1wiIC8+XG5pbXBvcnQgcGx1Z2lucyA9IHJlcXVpcmUoXCIuL25wbWcucGx1Z2luc1wiKTtcbmxldCBwYXRocyA9IHtcbiAgICBwYWNrYWdlQmFzZTogcGx1Z2lucy5wYXRoLmpvaW4oXCJfX2Rpcm5hbWVcIixcIi4uL1wiKVxufVxuZXhwb3J0ID0gcGF0aHM7Il0sInNvdXJjZVJvb3QiOiIvc291cmNlLyJ9
|
||||
7
dist/npmg.plugins.js
vendored
7
dist/npmg.plugins.js
vendored
@@ -1,7 +0,0 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
exports.beautylog = require("beautylog");
|
||||
exports.path = require("path");
|
||||
exports.shelljs = require("shelljs");
|
||||
exports.smartfile = require("smartfile");
|
||||
|
||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5wbWcucGx1Z2lucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw0Q0FBNEM7QUFDakMsaUJBQVMsR0FBRyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUM7QUFDakMsWUFBSSxHQUFHLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQztBQUNwQixlQUFPLFdBQVcsU0FBUyxDQUFDLENBQUM7QUFDaEMsaUJBQVMsR0FBRyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUMiLCJmaWxlIjoibnBtZy5wbHVnaW5zLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLy8vIDxyZWZlcmVuY2UgcGF0aD1cIi4vdHlwaW5ncy9tYWluLmQudHNcIiAvPlxuZXhwb3J0IGxldCBiZWF1dHlsb2cgPSByZXF1aXJlKFwiYmVhdXR5bG9nXCIpO1xuZXhwb3J0IGxldCBwYXRoID0gcmVxdWlyZShcInBhdGhcIik7XG5leHBvcnQgaW1wb3J0IHNoZWxsanMgPSByZXF1aXJlKFwic2hlbGxqc1wiKTtcbmV4cG9ydCBsZXQgc21hcnRmaWxlID0gcmVxdWlyZShcInNtYXJ0ZmlsZVwiKTtcbiJdLCJzb3VyY2VSb290IjoiL3NvdXJjZS8ifQ==
|
||||
29
npmextra.json
Normal file
29
npmextra.json
Normal file
@@ -0,0 +1,29 @@
|
||||
{
|
||||
"npmts": {
|
||||
"cli": true
|
||||
},
|
||||
"@git.zone/cli": {
|
||||
"projectType": "npm",
|
||||
"module": {
|
||||
"githost": "gitlab.com",
|
||||
"gitscope": "gitzone",
|
||||
"gitrepo": "tools",
|
||||
"shortDescription": "setup your environment with the most important tools and update them easily.",
|
||||
"npmPackagename": "@git.zone/tools",
|
||||
"license": "MIT",
|
||||
"description": "manage git.zone tools"
|
||||
},
|
||||
"release": {
|
||||
"registries": [
|
||||
"https://verdaccio.lossless.digital",
|
||||
"https://registry.npmjs.org"
|
||||
],
|
||||
"accessLevel": "public"
|
||||
}
|
||||
},
|
||||
"@ship.zone/szci": {
|
||||
"globalNpmTools": [
|
||||
"npmts"
|
||||
]
|
||||
}
|
||||
}
|
||||
@@ -1,4 +0,0 @@
|
||||
{
|
||||
"mode": "default",
|
||||
"coveralls": true
|
||||
}
|
||||
11159
package-lock.json
generated
Normal file
11159
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load Diff
47
package.json
47
package.json
@@ -1,10 +1,17 @@
|
||||
{
|
||||
"name": "npmg",
|
||||
"version": "1.0.2",
|
||||
"description": "setup your environment with the most important tools and update them easily.",
|
||||
"main": "dist/index.js",
|
||||
"name": "@git.zone/tools",
|
||||
"version": "3.2.0",
|
||||
"private": false,
|
||||
"type": "module",
|
||||
"description": "A CLI tool placeholder for development utilities.",
|
||||
"main": "dist_ts/index.js",
|
||||
"typings": "dist_ts/index.d.ts",
|
||||
"scripts": {
|
||||
"test": "(npmts)"
|
||||
"test": "(tstest test/ --verbose)",
|
||||
"build": "(tsbuild --web)"
|
||||
},
|
||||
"bin": {
|
||||
"gtools": "cli.js"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
@@ -23,14 +30,28 @@
|
||||
},
|
||||
"homepage": "https://github.com/GitZoneTools/npmg#readme",
|
||||
"devDependencies": {
|
||||
"npmts": "^3.3.2",
|
||||
"smartenv": "^1.2.0"
|
||||
"@git.zone/tsbuild": "^4.1.2",
|
||||
"@git.zone/tstest": "^3.1.8"
|
||||
},
|
||||
"dependencies": {
|
||||
"beautylog": "^3.1.2",
|
||||
"q": "^1.4.1",
|
||||
"shelljs": "^0.6.0",
|
||||
"smartcli": "0.0.11",
|
||||
"smartfile": "0.0.11"
|
||||
}
|
||||
"@git.zone/tsrun": "^2.0.1",
|
||||
"@push.rocks/smartcli": "^4.0.20",
|
||||
"@push.rocks/smartinteract": "^2.0.16",
|
||||
"@push.rocks/smartshell": "^3.3.0",
|
||||
"@types/node": "^22.0.0"
|
||||
},
|
||||
"files": [
|
||||
"ts/**/*",
|
||||
"ts_web/**/*",
|
||||
"dist/**/*",
|
||||
"dist_*/**/*",
|
||||
"dist_ts/**/*",
|
||||
"dist_ts_web/**/*",
|
||||
"cli.js",
|
||||
"npmextra.json",
|
||||
"readme.md"
|
||||
],
|
||||
"browserslist": [
|
||||
"last 1 chrome versions"
|
||||
]
|
||||
}
|
||||
|
||||
@@ -1,7 +0,0 @@
|
||||
{
|
||||
"default":[
|
||||
"cash",
|
||||
"npm-check-updates",
|
||||
"typings"
|
||||
]
|
||||
}
|
||||
8365
pnpm-lock.yaml
generated
Normal file
8365
pnpm-lock.yaml
generated
Normal file
File diff suppressed because it is too large
Load Diff
42
readme.md
Normal file
42
readme.md
Normal file
@@ -0,0 +1,42 @@
|
||||
# @git.zone/tools 🛠️
|
||||
|
||||
A CLI tool placeholder for development utilities.
|
||||
|
||||
## Issue Reporting and Security
|
||||
|
||||
For reporting bugs, issues, or security vulnerabilities, please visit [community.foss.global/](https://community.foss.global/). This is the central community hub for all issue reporting. Developers who sign and comply with our contribution agreement and go through identification can also get a [code.foss.global/](https://code.foss.global/) account to submit Pull Requests directly.
|
||||
|
||||
## Installation 📦
|
||||
|
||||
```bash
|
||||
npm install -g @git.zone/tools
|
||||
```
|
||||
|
||||
## Usage 🚀
|
||||
|
||||
```bash
|
||||
gtools
|
||||
```
|
||||
|
||||
Currently a placeholder CLI with no commands implemented yet.
|
||||
|
||||
## License and Legal Information
|
||||
|
||||
This repository contains open-source code licensed under the MIT License. A copy of the license can be found in the [LICENSE](./LICENSE) file.
|
||||
|
||||
**Please note:** The MIT License does not grant permission to use the trade names, trademarks, service marks, or product names of the project, except as required for reasonable and customary use in describing the origin of the work and reproducing the content of the NOTICE file.
|
||||
|
||||
### Trademarks
|
||||
|
||||
This project is owned and maintained by Task Venture Capital GmbH. The names and logos associated with Task Venture Capital GmbH and any related products or services are trademarks of Task Venture Capital GmbH or third parties, and are not included within the scope of the MIT license granted herein.
|
||||
|
||||
Use of these trademarks must comply with Task Venture Capital GmbH's Trademark Guidelines or the guidelines of the respective third-party owners, and any usage must be approved in writing. Third-party trademarks used herein are the property of their respective owners and used only in a descriptive manner, e.g. for an implementation of an API or similar.
|
||||
|
||||
### Company Information
|
||||
|
||||
Task Venture Capital GmbH
|
||||
Registered at District Court Bremen HRB 35230 HB, Germany
|
||||
|
||||
For any legal inquiries or further information, please contact us via email at hello@task.vc.
|
||||
|
||||
By using this repository, you acknowledge that you have read this section, agree to comply with its terms, and understand that the licensing of the code does not imply endorsement by Task Venture Capital GmbH of any derivative works.
|
||||
17
test/test.js
17
test/test.js
@@ -1,17 +0,0 @@
|
||||
/// <reference path="../ts/typings/main.d.ts"
|
||||
var npmg = require("../dist/index.js");
|
||||
var smartenv = require("smartenv");
|
||||
var environment = smartenv.getEnv();
|
||||
describe("npmg", function () {
|
||||
describe(".install()", function () {
|
||||
it("should install default list globally when parsed 'default' as argument", function () {
|
||||
if (environment.isC9 || environment.isCI) {
|
||||
this.timeout(60000);
|
||||
npmg.install("default");
|
||||
}
|
||||
;
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsOENBQThDO0FBQzlDLElBQUksSUFBSSxHQUFHLE9BQU8sQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO0FBQ3ZDLElBQUksUUFBUSxHQUFHLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQztBQUNuQyxJQUFJLFdBQVcsR0FBRyxRQUFRLENBQUMsTUFBTSxFQUFFLENBQUM7QUFHaEMsUUFBUSxDQUFDLE1BQU0sRUFBQztJQUNaLFFBQVEsQ0FBQyxZQUFZLEVBQUM7UUFDbEIsRUFBRSxDQUFDLHdFQUF3RSxFQUFDO1lBQ3hFLEVBQUUsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxJQUFJLElBQUksV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFBLENBQUM7Z0JBQ3RDLElBQUksQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUM7Z0JBQ3BCLElBQUksQ0FBQyxPQUFPLENBQUMsU0FBUyxDQUFDLENBQUM7WUFDNUIsQ0FBQztZQUFBLENBQUM7UUFDTixDQUFDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQyxDQUFBO0FBQ04sQ0FBQyxDQUFDLENBQUMiLCJmaWxlIjoidGVzdC5qcyIsInNvdXJjZXNDb250ZW50IjpbIi8vLyA8cmVmZXJlbmNlIHBhdGg9XCIuLi90cy90eXBpbmdzL21haW4uZC50c1wiIFxubGV0IG5wbWcgPSByZXF1aXJlKFwiLi4vZGlzdC9pbmRleC5qc1wiKTtcbmxldCBzbWFydGVudiA9IHJlcXVpcmUoXCJzbWFydGVudlwiKTtcbmxldCBlbnZpcm9ubWVudCA9IHNtYXJ0ZW52LmdldEVudigpO1xuXG5cbiAgICBkZXNjcmliZShcIm5wbWdcIixmdW5jdGlvbigpe1xuICAgICAgICBkZXNjcmliZShcIi5pbnN0YWxsKClcIixmdW5jdGlvbigpe1xuICAgICAgICAgICAgaXQoXCJzaG91bGQgaW5zdGFsbCBkZWZhdWx0IGxpc3QgZ2xvYmFsbHkgd2hlbiBwYXJzZWQgJ2RlZmF1bHQnIGFzIGFyZ3VtZW50XCIsZnVuY3Rpb24oKXtcbiAgICAgICAgICAgICAgICBpZiAoZW52aXJvbm1lbnQuaXNDOSB8fCBlbnZpcm9ubWVudC5pc0NJKXtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy50aW1lb3V0KDYwMDAwKTtcbiAgICAgICAgICAgICAgICAgICAgbnBtZy5pbnN0YWxsKFwiZGVmYXVsdFwiKTtcbiAgICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfSk7XG4gICAgICAgIH0pXG4gICAgfSk7XG4iXSwic291cmNlUm9vdCI6Ii9zb3VyY2UvIn0=
|
||||
@@ -1 +0,0 @@
|
||||
{"version":3,"file":"test.js","sourceRoot":"","sources":["test.ts"],"names":[],"mappings":"AAAA,IAAI,KAAK,GAAG,OAAO,CAAC,kBAAkB,CAAC,CAAC"}
|
||||
9
test/test.node.ts
Normal file
9
test/test.node.ts
Normal file
@@ -0,0 +1,9 @@
|
||||
import { expect, tap } from '@git.zone/tstest/tapbundle';
|
||||
|
||||
import * as tools from '../ts/index.js';
|
||||
|
||||
tap.test('should export runCli function', async () => {
|
||||
expect(typeof tools.runCli).toEqual('function');
|
||||
});
|
||||
|
||||
export default tap.start();
|
||||
16
test/test.ts
16
test/test.ts
@@ -1,16 +0,0 @@
|
||||
/// <reference path="../ts/typings/main.d.ts"
|
||||
let npmg = require("../dist/index.js");
|
||||
let smartenv = require("smartenv");
|
||||
let environment = smartenv.getEnv();
|
||||
|
||||
|
||||
describe("npmg",function(){
|
||||
describe(".install()",function(){
|
||||
it("should install default list globally when parsed 'default' as argument",function(){
|
||||
if (environment.isC9 || environment.isCI){
|
||||
this.timeout(60000);
|
||||
npmg.install("default");
|
||||
};
|
||||
});
|
||||
})
|
||||
});
|
||||
8
ts/00_commitinfo_data.ts
Normal file
8
ts/00_commitinfo_data.ts
Normal file
@@ -0,0 +1,8 @@
|
||||
/**
|
||||
* autocreated commitinfo by @push.rocks/commitinfo
|
||||
*/
|
||||
export const commitinfo = {
|
||||
name: '@git.zone/tools',
|
||||
version: '3.2.0',
|
||||
description: 'A CLI tool placeholder for development utilities.'
|
||||
}
|
||||
12
ts/index.ts
12
ts/index.ts
@@ -1,7 +1,7 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
import plugins = require("./npmg.plugins");
|
||||
import install = require("./npmg.install");
|
||||
let npmg = {
|
||||
install: install
|
||||
import * as cli from './tools.cli.js';
|
||||
|
||||
export const runCli = async () => {
|
||||
await cli.run();
|
||||
};
|
||||
export = npmg;
|
||||
|
||||
runCli();
|
||||
|
||||
301
ts/mod_update/classes.packagemanager.ts
Normal file
301
ts/mod_update/classes.packagemanager.ts
Normal file
@@ -0,0 +1,301 @@
|
||||
import * as plugins from './mod.plugins.js';
|
||||
|
||||
export type TPackageManager = 'npm' | 'yarn' | 'pnpm';
|
||||
|
||||
export interface IInstalledPackage {
|
||||
name: string;
|
||||
version: string;
|
||||
packageManager: TPackageManager;
|
||||
}
|
||||
|
||||
export interface IPackageUpdateInfo {
|
||||
name: string;
|
||||
currentVersion: string;
|
||||
latestVersion: string;
|
||||
packageManager: TPackageManager;
|
||||
needsUpdate: boolean;
|
||||
}
|
||||
|
||||
export interface IPackageManagerInfo {
|
||||
name: TPackageManager;
|
||||
available: boolean;
|
||||
detectionMethod?: 'which' | 'version-command';
|
||||
path?: string;
|
||||
currentVersion?: string;
|
||||
latestVersion?: string | null;
|
||||
needsUpdate?: boolean;
|
||||
}
|
||||
|
||||
export class PackageManagerUtil {
|
||||
private shell = new plugins.smartshell.Smartshell({
|
||||
executor: 'bash',
|
||||
});
|
||||
|
||||
/**
|
||||
* Check if a package manager is available on the system
|
||||
* Uses multiple detection methods for robustness across different shell contexts
|
||||
*/
|
||||
public async isAvailable(pm: TPackageManager, verbose = false): Promise<boolean> {
|
||||
const info = await this.detectPackageManager(pm, verbose);
|
||||
return info.available;
|
||||
}
|
||||
|
||||
/**
|
||||
* Detect a package manager and return detailed info
|
||||
*/
|
||||
public async detectPackageManager(pm: TPackageManager, verbose = false): Promise<IPackageManagerInfo> {
|
||||
const info: IPackageManagerInfo = { name: pm, available: false };
|
||||
|
||||
// Primary method: try 'which' command
|
||||
try {
|
||||
const whichResult = await this.shell.execSilent(`which ${pm} 2>/dev/null`);
|
||||
if (whichResult.exitCode === 0 && whichResult.stdout.trim()) {
|
||||
info.available = true;
|
||||
info.detectionMethod = 'which';
|
||||
info.path = whichResult.stdout.trim();
|
||||
if (verbose) {
|
||||
console.log(` Checking ${pm}... found via 'which' at ${info.path}`);
|
||||
}
|
||||
return info;
|
||||
}
|
||||
} catch {
|
||||
// Continue to fallback
|
||||
}
|
||||
|
||||
// Fallback method: try running pm --version directly
|
||||
// This can find PMs that are available but not in PATH for 'which'
|
||||
try {
|
||||
const versionResult = await this.shell.execSilent(`${pm} --version 2>/dev/null`);
|
||||
if (versionResult.exitCode === 0 && versionResult.stdout.trim()) {
|
||||
info.available = true;
|
||||
info.detectionMethod = 'version-command';
|
||||
if (verbose) {
|
||||
console.log(` Checking ${pm}... found via '--version' (which failed)`);
|
||||
}
|
||||
return info;
|
||||
}
|
||||
} catch {
|
||||
// Not available
|
||||
}
|
||||
|
||||
if (verbose) {
|
||||
console.log(` Checking ${pm}... not found`);
|
||||
}
|
||||
return info;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the current and latest version of a package manager
|
||||
*/
|
||||
public async getPackageManagerVersion(pm: TPackageManager): Promise<{ current: string; latest: string | null }> {
|
||||
let current = 'unknown';
|
||||
let latest: string | null = null;
|
||||
|
||||
// Get current version
|
||||
try {
|
||||
const result = await this.shell.execSilent(`${pm} --version 2>/dev/null`);
|
||||
if (result.exitCode === 0 && result.stdout.trim()) {
|
||||
// Parse version from output - handle different formats
|
||||
const output = result.stdout.trim();
|
||||
// npm: "10.2.0", pnpm: "8.15.0", yarn: "1.22.19"
|
||||
// Some may include prefix like "v1.22.19"
|
||||
const versionMatch = output.match(/(\d+\.\d+\.\d+)/);
|
||||
if (versionMatch) {
|
||||
current = versionMatch[1];
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
// Keep as unknown
|
||||
}
|
||||
|
||||
// Get latest version from npm registry
|
||||
try {
|
||||
const result = await this.shell.execSilent(`npm view ${pm} version 2>/dev/null`);
|
||||
if (result.exitCode === 0 && result.stdout.trim()) {
|
||||
latest = result.stdout.trim();
|
||||
}
|
||||
} catch {
|
||||
// Keep as null
|
||||
}
|
||||
|
||||
return { current, latest };
|
||||
}
|
||||
|
||||
/**
|
||||
* Get all globally installed @git.zone packages for a package manager
|
||||
*/
|
||||
public async getInstalledPackages(pm: TPackageManager): Promise<IInstalledPackage[]> {
|
||||
const packages: IInstalledPackage[] = [];
|
||||
|
||||
try {
|
||||
let result;
|
||||
switch (pm) {
|
||||
case 'npm':
|
||||
result = await this.shell.execSilent('npm list -g --depth=0 --json 2>/dev/null || true');
|
||||
break;
|
||||
case 'yarn':
|
||||
result = await this.shell.execSilent('yarn global list --depth=0 --json 2>/dev/null || true');
|
||||
break;
|
||||
case 'pnpm':
|
||||
result = await this.shell.execSilent('pnpm list -g --depth=0 --json 2>/dev/null || true');
|
||||
break;
|
||||
}
|
||||
|
||||
const output = result.stdout.trim();
|
||||
if (!output) {
|
||||
return packages;
|
||||
}
|
||||
|
||||
if (pm === 'npm') {
|
||||
try {
|
||||
const data = JSON.parse(output);
|
||||
const deps = data.dependencies || {};
|
||||
for (const [name, info] of Object.entries(deps)) {
|
||||
if (name.startsWith('@git.zone/')) {
|
||||
packages.push({
|
||||
name,
|
||||
version: (info as any).version || 'unknown',
|
||||
packageManager: pm,
|
||||
});
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
// JSON parse failed
|
||||
}
|
||||
} else if (pm === 'pnpm') {
|
||||
// pnpm returns an array of objects
|
||||
try {
|
||||
const data = JSON.parse(output);
|
||||
// Handle array format from pnpm
|
||||
const dataArray = Array.isArray(data) ? data : [data];
|
||||
for (const item of dataArray) {
|
||||
const deps = item.dependencies || {};
|
||||
for (const [name, info] of Object.entries(deps)) {
|
||||
if (name.startsWith('@git.zone/')) {
|
||||
packages.push({
|
||||
name,
|
||||
version: (info as any).version || 'unknown',
|
||||
packageManager: pm,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
// JSON parse failed
|
||||
}
|
||||
} else if (pm === 'yarn') {
|
||||
// Yarn global list --json outputs multiple JSON lines
|
||||
const lines = output.split('\n').filter(l => l.trim());
|
||||
for (const line of lines) {
|
||||
try {
|
||||
const data = JSON.parse(line);
|
||||
if (data.type === 'tree' && data.data && data.data.trees) {
|
||||
for (const tree of data.data.trees) {
|
||||
const name = tree.name?.split('@')[0] || '';
|
||||
if (name.startsWith('@git.zone/')) {
|
||||
const version = tree.name?.split('@').pop() || 'unknown';
|
||||
packages.push({
|
||||
name,
|
||||
version,
|
||||
packageManager: pm,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
// Skip invalid JSON lines
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
// Command failed, return empty array
|
||||
}
|
||||
|
||||
return packages;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the latest version of a package from npm registry
|
||||
* Tries private registry (verdaccio.lossless.digital) first via API, then falls back to public npm
|
||||
*/
|
||||
public async getLatestVersion(packageName: string): Promise<string | null> {
|
||||
// URL-encode the package name for scoped packages (@scope/name -> @scope%2fname)
|
||||
const encodedName = packageName.replace('/', '%2f');
|
||||
|
||||
// Try private registry first via direct API call (npm view doesn't work reliably)
|
||||
try {
|
||||
const result = await this.shell.execSilent(
|
||||
`curl -sf "https://verdaccio.lossless.digital/${encodedName}" 2>/dev/null`
|
||||
);
|
||||
if (result.exitCode === 0 && result.stdout.trim()) {
|
||||
const data = JSON.parse(result.stdout.trim());
|
||||
if (data['dist-tags']?.latest) {
|
||||
return data['dist-tags'].latest;
|
||||
}
|
||||
}
|
||||
} catch {
|
||||
// Continue to public registry
|
||||
}
|
||||
|
||||
// Fall back to public npm
|
||||
try {
|
||||
const result = await this.shell.execSilent(`npm view ${packageName} version 2>/dev/null`);
|
||||
if (result.exitCode === 0 && result.stdout.trim()) {
|
||||
return result.stdout.trim();
|
||||
}
|
||||
} catch {
|
||||
// Command failed
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Execute an update for a package using the specified package manager
|
||||
*/
|
||||
public async executeUpdate(pm: TPackageManager, packageName: string): Promise<boolean> {
|
||||
let command: string;
|
||||
|
||||
switch (pm) {
|
||||
case 'npm':
|
||||
command = `npm install -g ${packageName}@latest`;
|
||||
break;
|
||||
case 'yarn':
|
||||
command = `yarn global add ${packageName}@latest`;
|
||||
break;
|
||||
case 'pnpm':
|
||||
command = `pnpm add -g ${packageName}@latest`;
|
||||
break;
|
||||
}
|
||||
|
||||
console.log(` Updating ${packageName} via ${pm}...`);
|
||||
|
||||
try {
|
||||
const result = await this.shell.exec(command);
|
||||
return result.exitCode === 0;
|
||||
} catch {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Compare two semver versions
|
||||
* Returns true if latest > current
|
||||
*/
|
||||
public isNewerVersion(current: string, latest: string): boolean {
|
||||
const cleanVersion = (v: string) => v.replace(/^[^\d]*/, '');
|
||||
const currentClean = cleanVersion(current);
|
||||
const latestClean = cleanVersion(latest);
|
||||
|
||||
const currentParts = currentClean.split('.').map(n => parseInt(n, 10) || 0);
|
||||
const latestParts = latestClean.split('.').map(n => parseInt(n, 10) || 0);
|
||||
|
||||
for (let i = 0; i < Math.max(currentParts.length, latestParts.length); i++) {
|
||||
const curr = currentParts[i] || 0;
|
||||
const lat = latestParts[i] || 0;
|
||||
if (lat > curr) return true;
|
||||
if (lat < curr) return false;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
||||
173
ts/mod_update/index.ts
Normal file
173
ts/mod_update/index.ts
Normal file
@@ -0,0 +1,173 @@
|
||||
import * as plugins from './mod.plugins.js';
|
||||
import { PackageManagerUtil, type TPackageManager, type IPackageUpdateInfo, type IPackageManagerInfo } from './classes.packagemanager.js';
|
||||
|
||||
const GITZONE_PACKAGES = [
|
||||
'@git.zone/cli',
|
||||
'@git.zone/tsdoc',
|
||||
'@git.zone/tsbuild',
|
||||
'@git.zone/tstest',
|
||||
'@git.zone/tspublish',
|
||||
'@git.zone/tsbundle',
|
||||
'@git.zone/tsdocker',
|
||||
'@git.zone/tsview',
|
||||
'@git.zone/tswatch',
|
||||
];
|
||||
|
||||
export interface IUpdateOptions {
|
||||
yes?: boolean;
|
||||
verbose?: boolean;
|
||||
}
|
||||
|
||||
export const run = async (options: IUpdateOptions = {}): Promise<void> => {
|
||||
const pmUtil = new PackageManagerUtil();
|
||||
const verbose = options.verbose === true;
|
||||
|
||||
console.log('Scanning for installed @git.zone packages...\n');
|
||||
|
||||
// Check which package managers are available
|
||||
if (verbose) {
|
||||
console.log('Detecting package managers:');
|
||||
}
|
||||
|
||||
const detectedPMs: IPackageManagerInfo[] = [];
|
||||
for (const pm of ['npm', 'yarn', 'pnpm'] as TPackageManager[]) {
|
||||
const info = await pmUtil.detectPackageManager(pm, verbose);
|
||||
if (info.available) {
|
||||
detectedPMs.push(info);
|
||||
}
|
||||
}
|
||||
|
||||
if (verbose) {
|
||||
console.log('');
|
||||
}
|
||||
|
||||
if (detectedPMs.length === 0) {
|
||||
console.log('No package managers found (npm, yarn, pnpm).');
|
||||
console.log('Tried detection via \'which\' command and direct version check.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Get version info for each PM and display status table
|
||||
console.log('Package managers:\n');
|
||||
console.log(' Name Current Latest Status');
|
||||
console.log(' ──────────────────────────────────────────────');
|
||||
|
||||
for (const pmInfo of detectedPMs) {
|
||||
const versionInfo = await pmUtil.getPackageManagerVersion(pmInfo.name);
|
||||
pmInfo.currentVersion = versionInfo.current;
|
||||
pmInfo.latestVersion = versionInfo.latest;
|
||||
pmInfo.needsUpdate = versionInfo.latest
|
||||
? pmUtil.isNewerVersion(versionInfo.current, versionInfo.latest)
|
||||
: false;
|
||||
|
||||
const name = pmInfo.name.padEnd(9);
|
||||
const current = versionInfo.current.padEnd(12);
|
||||
const latest = (versionInfo.latest || 'unknown').padEnd(12);
|
||||
const status = versionInfo.latest === null
|
||||
? '? Version unknown'
|
||||
: pmInfo.needsUpdate
|
||||
? '⬆️ Update available'
|
||||
: '✓ Up to date';
|
||||
console.log(` ${name}${current}${latest}${status}`);
|
||||
}
|
||||
|
||||
console.log('');
|
||||
|
||||
// Collect all installed @git.zone packages from all package managers
|
||||
const allPackages: IPackageUpdateInfo[] = [];
|
||||
|
||||
for (const pmInfo of detectedPMs) {
|
||||
const installed = await pmUtil.getInstalledPackages(pmInfo.name);
|
||||
for (const pkg of installed) {
|
||||
// Only include packages from our predefined list
|
||||
if (GITZONE_PACKAGES.includes(pkg.name)) {
|
||||
const latestVersion = await pmUtil.getLatestVersion(pkg.name);
|
||||
allPackages.push({
|
||||
name: pkg.name,
|
||||
currentVersion: pkg.version,
|
||||
latestVersion: latestVersion || 'unknown',
|
||||
packageManager: pmInfo.name,
|
||||
needsUpdate: latestVersion ? pmUtil.isNewerVersion(pkg.version, latestVersion) : false,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (allPackages.length === 0) {
|
||||
console.log('No @git.zone packages found installed globally.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Display package table
|
||||
console.log('Installed @git.zone packages:\n');
|
||||
console.log(' Package Current Latest PM Status');
|
||||
console.log(' ─────────────────────────────────────────────────────────────────────');
|
||||
|
||||
for (const pkg of allPackages) {
|
||||
const name = pkg.name.padEnd(28);
|
||||
const current = pkg.currentVersion.padEnd(12);
|
||||
const latest = pkg.latestVersion.padEnd(12);
|
||||
const pm = pkg.packageManager.padEnd(8);
|
||||
const status = pkg.latestVersion === 'unknown'
|
||||
? '? Version unknown'
|
||||
: pkg.needsUpdate
|
||||
? '⬆️ Update available'
|
||||
: '✓ Up to date';
|
||||
console.log(` ${name}${current}${latest}${pm}${status}`);
|
||||
}
|
||||
|
||||
console.log('');
|
||||
|
||||
// Filter packages that need updates
|
||||
const packagesToUpdate = allPackages.filter(p => p.needsUpdate);
|
||||
|
||||
if (packagesToUpdate.length === 0) {
|
||||
console.log('All packages are up to date!');
|
||||
return;
|
||||
}
|
||||
|
||||
console.log(`Found ${packagesToUpdate.length} package(s) with available updates.\n`);
|
||||
|
||||
// Ask for confirmation unless -y flag is provided
|
||||
let shouldUpdate = options.yes === true;
|
||||
|
||||
if (!shouldUpdate) {
|
||||
const smartinteractInstance = new plugins.smartinteract.SmartInteract();
|
||||
const answer = await smartinteractInstance.askQuestion({
|
||||
type: 'confirm',
|
||||
name: 'confirmUpdate',
|
||||
message: 'Do you want to update these packages?',
|
||||
default: true,
|
||||
});
|
||||
shouldUpdate = answer.value === true;
|
||||
}
|
||||
|
||||
if (!shouldUpdate) {
|
||||
console.log('Update cancelled.');
|
||||
return;
|
||||
}
|
||||
|
||||
// Execute updates
|
||||
console.log('\nUpdating packages...\n');
|
||||
|
||||
let successCount = 0;
|
||||
let failCount = 0;
|
||||
|
||||
for (const pkg of packagesToUpdate) {
|
||||
const success = await pmUtil.executeUpdate(pkg.packageManager, pkg.name);
|
||||
if (success) {
|
||||
console.log(` ✓ ${pkg.name} updated successfully`);
|
||||
successCount++;
|
||||
} else {
|
||||
console.log(` ✗ ${pkg.name} update failed`);
|
||||
failCount++;
|
||||
}
|
||||
}
|
||||
|
||||
console.log('');
|
||||
if (failCount === 0) {
|
||||
console.log(`All ${successCount} package(s) updated successfully!`);
|
||||
} else {
|
||||
console.log(`Updated ${successCount} package(s), ${failCount} failed.`);
|
||||
}
|
||||
};
|
||||
4
ts/mod_update/mod.plugins.ts
Normal file
4
ts/mod_update/mod.plugins.ts
Normal file
@@ -0,0 +1,4 @@
|
||||
import * as smartinteract from '@push.rocks/smartinteract';
|
||||
import * as smartshell from '@push.rocks/smartshell';
|
||||
|
||||
export { smartinteract, smartshell };
|
||||
@@ -1,27 +0,0 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
import plugins = require("./npmg.plugins");
|
||||
import paths = require("./npmg.paths");
|
||||
let installExec = function(packageNames:string[]){
|
||||
for (let packageName in packageNames){
|
||||
let execCommand = "npm install -g " + packageNames[packageName];
|
||||
plugins.beautylog.info("now installing " + packageNames[packageName]);
|
||||
plugins.shelljs.exec(execCommand);
|
||||
};
|
||||
};
|
||||
|
||||
let packageLibrary = plugins.smartfile.readFileToObject(
|
||||
plugins.path.join(paths.packageBase,"packageLibrary.json")
|
||||
);
|
||||
|
||||
let install = function(packageSetArg:String){
|
||||
switch (packageSetArg){
|
||||
case "default":
|
||||
installExec(packageLibrary.default);
|
||||
break;
|
||||
default:
|
||||
plugins.beautylog.warn("no set has been specified");
|
||||
break;
|
||||
}
|
||||
};
|
||||
|
||||
export = install;
|
||||
@@ -1,6 +0,0 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
import plugins = require("./npmg.plugins");
|
||||
let paths = {
|
||||
packageBase: plugins.path.join("__dirname","../")
|
||||
}
|
||||
export = paths;
|
||||
@@ -1,5 +0,0 @@
|
||||
/// <reference path="./typings/main.d.ts" />
|
||||
export let beautylog = require("beautylog");
|
||||
export let path = require("path");
|
||||
export import shelljs = require("shelljs");
|
||||
export let smartfile = require("smartfile");
|
||||
25
ts/tools.cli.ts
Normal file
25
ts/tools.cli.ts
Normal file
@@ -0,0 +1,25 @@
|
||||
import * as plugins from './tools.plugins.js';
|
||||
import * as modUpdate from './mod_update/index.js';
|
||||
|
||||
export const run = async () => {
|
||||
const toolsCli = new plugins.smartcli.Smartcli();
|
||||
|
||||
toolsCli.standardCommand().subscribe(async (argvArg) => {
|
||||
console.log('@git.zone/tools - CLI utility for managing @git.zone packages\n');
|
||||
console.log('Commands:');
|
||||
console.log(' update Check and update globally installed @git.zone packages');
|
||||
console.log(' update -y Update without confirmation prompt');
|
||||
console.log(' update --verbose Show detection diagnostics');
|
||||
console.log('');
|
||||
console.log('Use "gtools <command> --help" for more information about a command.');
|
||||
});
|
||||
|
||||
toolsCli.addCommand('update').subscribe(async (argvArg) => {
|
||||
const yesFlag = argvArg.y === true || argvArg.yes === true;
|
||||
const verboseFlag = argvArg.v === true || argvArg.verbose === true;
|
||||
await modUpdate.run({ yes: yesFlag, verbose: verboseFlag });
|
||||
});
|
||||
|
||||
toolsCli.addVersion('3.0.0');
|
||||
toolsCli.startParse();
|
||||
};
|
||||
6
ts/tools.plugins.ts
Normal file
6
ts/tools.plugins.ts
Normal file
@@ -0,0 +1,6 @@
|
||||
// push.rocks scope
|
||||
import * as smartcli from '@push.rocks/smartcli';
|
||||
import * as smartinteract from '@push.rocks/smartinteract';
|
||||
import * as smartshell from '@push.rocks/smartshell';
|
||||
|
||||
export { smartcli, smartinteract, smartshell };
|
||||
@@ -1,6 +0,0 @@
|
||||
{
|
||||
"ambientDependencies": {
|
||||
"node": "github:DefinitelyTyped/DefinitelyTyped/node/node.d.ts#48c1e3c1d6baefa4f1a126f188c27c4fefd36bff",
|
||||
"shelljs": "github:DefinitelyTyped/DefinitelyTyped/shelljs/shelljs.d.ts#ce14ae27a020194da3d35aa3468ca1e9e5296316"
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user