@dx-pkg/mksymlink 1.0.18 → 1.0.19
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/CHANGELOG.md +7 -0
- package/bin/index.js +2 -2
- package/package.json +2 -2
- package/src/index.d.mts +3 -3
- package/src/index.d.ts +3 -3
- package/src/index.js +2 -2
- package/src/index.mjs +2 -2
package/CHANGELOG.md
CHANGED
package/bin/index.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
var e=Object.create,t=Object.defineProperty,n=Object.getOwnPropertyDescriptor,r=Object.getOwnPropertyNames,i=Object.getPrototypeOf,a=Object.prototype.hasOwnProperty,o=(e,i,o,s)=>{if(i&&typeof i==`object`||typeof i==`function`)for(var c=r(i),l=0,u=c.length,d;l<u;l++)d=c[l],!a.call(e,d)&&d!==o&&t(e,d,{get:(e=>i[e]).bind(null,d),enumerable:!(s=n(i,d))||s.enumerable});return e},s=(n,r,a)=>(a=n==null?{}:e(i(n)),o(r||!n||!n.__esModule?t(a,`default`,{value:n,enumerable:!0}):a,n));let c=require(`commander`),l=require(`@dx-pkg/logger`),u=require(`path`);u=s(u);let d=require(`fs`);d=s(d);let f=require(`os`),p=require(`child_process`),m=require(`util`),h=require(`fs/promises`);var g=class{
|
|
2
|
+
var e=Object.create,t=Object.defineProperty,n=Object.getOwnPropertyDescriptor,r=Object.getOwnPropertyNames,i=Object.getPrototypeOf,a=Object.prototype.hasOwnProperty,o=(e,i,o,s)=>{if(i&&typeof i==`object`||typeof i==`function`)for(var c=r(i),l=0,u=c.length,d;l<u;l++)d=c[l],!a.call(e,d)&&d!==o&&t(e,d,{get:(e=>i[e]).bind(null,d),enumerable:!(s=n(i,d))||s.enumerable});return e},s=(n,r,a)=>(a=n==null?{}:e(i(n)),o(r||!n||!n.__esModule?t(a,`default`,{value:n,enumerable:!0}):a,n));let c=require(`commander`),l=require(`@dx-pkg/logger`),u=require(`path`);u=s(u);let d=require(`fs`);d=s(d);let f=require(`os`),p=require(`child_process`),m=require(`util`),h=require(`fs/promises`);var g=class{constructor(e){this.config={};let t=(0,f.homedir)();this.configPath=e||(0,u.resolve)(t,`.mksymlinkrc`),this.loadConfig()}get(e){return this.config[e]}set(e,t){this.config[e]=t,this.saveConfig()}unset(e){delete this.config[e],this.saveConfig()}list(){return{...this.config}}getDefaultSymlinkDir(){return this.get(`symlink.defaultDir`)}setDefaultSymlinkDir(e){this.set(`symlink.defaultDir`,e)}getConfigInfo(){let e=(0,d.existsSync)(this.configPath),t=Object.keys(this.config).length;if(!e)return{path:this.configPath,exists:!1,entries:t};let n=(0,d.statSync)(this.configPath);return{path:this.configPath,exists:!0,size:n.size,modified:n.mtime,entries:t}}loadConfig(){if(!(0,d.existsSync)(this.configPath)){this.config={};return}try{let e=(0,d.readFileSync)(this.configPath,`utf-8`).split(`
|
|
3
3
|
`).filter(e=>e.trim());this.config={};for(let t of e){let[e,...n]=t.split(`=`);e&&n.length>0&&(this.config[e.trim()]=n.join(`=`).trim())}}catch(e){throw Error(`Failed to load config from ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}saveConfig(){try{let e=(0,u.dirname)(this.configPath);(0,d.existsSync)(e)||(0,d.mkdirSync)(e,{recursive:!0});let t=Object.entries(this.config).filter(([,e])=>e!==void 0).map(([e,t])=>`${e}=${t}`).join(`
|
|
4
4
|
`);(0,d.writeFileSync)(this.configPath,t+`
|
|
5
|
-
`,`utf-8`)}catch(e){throw Error(`Failed to save config to ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}};const _=[`file`,`dir`,`junction`];var v=class{constructor(e,t,n,r,i=new g){this.optionValidator=e,this.logger=t,this.symlinkManager=n,this.options=r,this.configService=i}async execute(){this.optionValidator.validate(this.options);try{let{source:e,target:t,type:n,force:r}=this.resolveOptions();this.logger.info(`Creating symlink with options:`),this.logger.info(` Source: ${e}`),this.logger.info(` Target: ${t}`),this.logger.info(` Type: ${n}`),this.logger.info(` Force: ${r}`);let i=await this.symlinkManager.createSymlink({source:e,target:t,type:n,force:r});if(!i.success){if(i.error)throw i.error;this.logger.
|
|
5
|
+
`,`utf-8`)}catch(e){throw Error(`Failed to save config to ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}};const _=[`file`,`dir`,`junction`];var v=class{constructor(e,t,n,r,i=new g){this.optionValidator=e,this.logger=t,this.symlinkManager=n,this.options=r,this.configService=i}async execute(){this.optionValidator.validate(this.options);try{let{source:e,target:t,type:n,force:r}=this.resolveOptions();this.logger.info(`Creating symlink with options:`),this.logger.info(` Source: ${e}`),this.logger.info(` Target: ${t}`),this.logger.info(` Type: ${n}`),this.logger.info(` Force: ${r}`);let i=await this.symlinkManager.createSymlink({source:e,target:t,type:n,force:r});if(!i.success){if(i.error)throw i.error;this.logger.warn(i.message||`Symlink creation failed`);return}this.logger.success(`Symlink created successfully!`),this.logger.info(` Source: ${i.source}`),this.logger.info(` Target: ${i.target}`)}catch(e){throw this.logger.error(`Failed to create symlink: ${e instanceof Error?e.message:String(e)}`),e}}resolveOptions(){let e=process.cwd();return{source:this.options.source||e,target:this.resolveTarget(this.options)||this.generateDefaultTarget(e),type:this.options.type||`junction`,force:this.options.force||!1}}resolveTarget(e){if(e.target!==`.`&&e.target!==`./`)return e.target;let t=process.cwd();return(0,u.resolve)(t,(0,u.basename)(e.source||t))}generateDefaultTarget(e){let t=(0,u.basename)(e),n=`${(0,u.basename)((0,u.dirname)(e))}--${t}`;return(0,u.resolve)(this.configService.getDefaultSymlinkDir()||e,n)}},y=class extends Error{constructor(e,t){super(e),this.name=`SymlinkError`,this.code=t,Error.captureStackTrace(this,this.constructor)}};const b=(0,m.promisify)(p.exec);var x=class{constructor(e,t){this.logger=e,this.osDetector=t}async createSymlink(e){let{source:t,target:n,type:r=`dir`,force:i=!1}=e;try{if(await this.symlinkExists(n)){if(!i)return{success:!1,source:t,target:n,message:`Symlink already exists: ${n}`};this.logger.info(`Force mode enabled. Removing existing symlink: ${n}`),await this.removeSymlink(n)}return this.osDetector.isWindows()?await this.createWindowsSymlink(t,n,r):await this.createUnixSymlink(t,n),{success:!0,source:t,target:n,message:`Symlink created successfully`}}catch(e){return this.logger.error(`Failed to create symlink: ${e instanceof Error?e.message:String(e)}`),{success:!1,source:t,target:n,error:e instanceof Error?e:Error(String(e))}}}async createWindowsSymlink(e,t,n){let r;switch(n){case`junction`:r=`/J`;break;case`file`:r=``;break;default:r=`/D`;break}let i=r?`mklink ${r} "${t}" "${e}"`:`mklink "${t}" "${e}"`;this.logger.info(`Executing Windows command: ${i}`);try{let{stdout:e,stderr:t}=await b(i,{shell:`cmd.exe`});if(t&&!t.includes(`Junction`)&&!t.includes(`symbolic link`))throw new y(t,`WINDOWS_MKLINK_ERROR`);this.logger.info(`Windows mklink output: ${e}`)}catch(e){throw new y(`Failed to create Windows symlink: ${e instanceof Error?e.message:String(e)}`,`WINDOWS_MKLINK_FAILED`)}}async createUnixSymlink(e,t){try{await(0,h.symlink)(e,t),this.logger.info(`Unix symlink created: ln -s ${e} ${t}`)}catch(e){throw new y(`Failed to create Unix symlink: ${e instanceof Error?e.message:String(e)}`,`UNIX_SYMLINK_FAILED`)}}async symlinkExists(e){try{return(await(0,h.lstat)(e)).isSymbolicLink()}catch{return!1}}async removeSymlink(e){try{await(0,h.access)(e),await(0,h.unlink)(e),this.logger.info(`Symlink removed: ${e}`)}catch(e){throw new y(`Failed to remove symlink: ${e instanceof Error?e.message:String(e)}`,`REMOVE_SYMLINK_FAILED`)}}},S=class{constructor(){this._platform=this.normalizePlatform((0,f.platform)())}normalizePlatform(e){switch(e){case`darwin`:return`darwin`;case`win32`:return`win32`;case`linux`:return`linux`;default:return`unknown`}}getPlatform(){return this._platform}isWindows(){return this._platform===`win32`}isMacOS(){return this._platform===`darwin`}isLinux(){return this._platform===`linux`}},C=class{validate(e){this.validateType(e.type),this.validateSource(e.source),this.validateForce(e.force)}validateType(e){if(e&&!_.includes(e))throw new y(`Invalid symlink type: ${e}. Must be one of: ${_.join(`, `)}`,`INVALID_SYMLINK_TYPE`)}validateSource(e){if(!e)return;let t=(0,u.resolve)(e);if(!(0,d.existsSync)(t))throw new y(`Source path does not exist: ${t}`,`SOURCE_NOT_FOUND`)}validateForce(e){if(e!==void 0&&typeof e!=`boolean`)throw new y(`Invalid force option: ${e}. Must be a boolean.`,`INVALID_FORCE_OPTION`)}},w=class{static create(e){let t=new C,n=new l.ConsoleLogger;return new v(t,n,new x(n,new S),e)}},T=class{constructor(e,t,n){this.logger=e,this.configService=t,this.options=n}async execute(){try{let{action:e,key:t,value:n}=this.options;switch(e){case`get`:await this.handleGet(t);break;case`set`:await this.handleSet(t,n);break;case`unset`:await this.handleUnset(t);break;case`list`:await this.handleList();break;case`info`:await this.handleInfo();break;default:throw Error(`Unknown config action: ${e}`)}}catch(e){throw this.logger.error(`Config command failed: ${e instanceof Error?e.message:String(e)}`),e}}async handleGet(e){if(!e)throw Error(`Key is required for get action`);let t=this.configService.get(e);if(t===void 0){this.logger.warn(`Config key '${e}' not found`);return}this.logger.info(t)}async handleSet(e,t){if(!e)throw Error(`Key is required for set action`);if(!t)throw Error(`Value is required for set action`);this.configService.set(e,t),this.logger.success(`Config set: ${e} = ${t}`)}async handleUnset(e){if(!e)throw Error(`Key is required for unset action`);this.configService.unset(e),this.logger.success(`Config unset: ${e}`)}async handleList(){let e=this.configService.list(),t=Object.entries(e);if(t.length===0){this.logger.info(`No configuration found`);return}this.logger.info(`Current configuration:`);for(let[e,n]of t)this.logger.info(` ${e} = ${n}`)}async handleInfo(){let e=this.configService.getConfigInfo();this.logger.info(`Configuration file information:`),this.logger.info(` Path: ${e.path}`),this.logger.info(` Exists: ${e.exists?`Yes`:`No`}`),e.exists&&(this.logger.info(` Size: ${e.size} bytes`),this.logger.info(` Modified: ${e.modified?.toLocaleString()}`)),this.logger.info(` Entries: ${e.entries}`)}};const E=[`get`,`set`,`unset`,`list`,`info`];var D=class{validate(e){this.validateAction(e.action),this.validateKey(e.action,e.key),this.validateValue(e.action,e.value)}validateAction(e){if(!E.includes(e))throw new y(`Invalid config action: ${e}. Must be one of: ${E.join(`, `)}`,`INVALID_CONFIG_ACTION`)}validateKey(e,t){if(!(e===`list`||e===`info`)){if(!t)throw new y(`Key is required for '${e}' action`,`MISSING_CONFIG_KEY`);if(typeof t!=`string`||t.trim().length===0)throw new y(`Invalid config key: ${t}. Key must be a non-empty string.`,`INVALID_CONFIG_KEY`)}}validateValue(e,t){if(e===`set`){if(!t)throw new y(`Value is required for 'set' action`,`MISSING_CONFIG_VALUE`);if(typeof t!=`string`)throw new y(`Invalid config value: ${t}. Value must be a string.`,`INVALID_CONFIG_VALUE`)}}},O=class{static create(e){return new D().validate(e),new T(new l.ConsoleLogger,new g,e)}};const k=JSON.parse(d.default.readFileSync(u.default.resolve(__dirname,`../package.json`),`utf8`)),A=new c.Command;A.name(`mksymlink`).description(`Create symbolic links across platforms (macOS, Windows, Linux)`).version(k.version),A.command(`create`,{isDefault:!0}).description(`Create a symbolic link`).option(`-s, --source <path>`,`Source path (defaults to current directory)`).option(`-t, --target <path>`,`Target symlink path (auto-generated if not provided)`).option(`--type <type>`,`Symlink type for Windows: file, dir, junction (default: junction)`,`junction`).option(`-f, --force`,`Force overwrite existing symlink`,!0).action(async e=>{try{await w.create(e).execute(),process.exit(0)}catch(e){console.error(`Command failed:`,e instanceof Error?e.message:String(e)),process.exit(1)}});const j=A.command(`config`).description(`Manage mksymlink configuration`);j.command(`get <key>`).description(`Get a configuration value`).action(async e=>{try{await O.create({action:`get`,key:e}).execute(),process.exit(0)}catch(e){console.error(`Config get failed:`,e instanceof Error?e.message:String(e)),process.exit(1)}}),j.command(`set <key> <value>`).description(`Set a configuration value`).action(async(e,t)=>{try{await O.create({action:`set`,key:e,value:t}).execute(),process.exit(0)}catch(e){console.error(`Config set failed:`,e instanceof Error?e.message:String(e)),process.exit(1)}}),j.command(`unset <key>`).description(`Unset a configuration value`).action(async e=>{try{await O.create({action:`unset`,key:e}).execute(),process.exit(0)}catch(e){console.error(`Config unset failed:`,e instanceof Error?e.message:String(e)),process.exit(1)}}),j.command(`list`).description(`List all configuration values`).action(async()=>{try{await O.create({action:`list`}).execute(),process.exit(0)}catch(e){console.error(`Config list failed:`,e instanceof Error?e.message:String(e)),process.exit(1)}}),j.command(`info`).description(`Show configuration file information`).action(async()=>{try{await O.create({action:`info`}).execute(),process.exit(0)}catch(e){console.error(`Config info failed:`,e instanceof Error?e.message:String(e)),process.exit(1)}}),A.parse(process.argv);
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dx-pkg/mksymlink",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.19",
|
|
4
4
|
"description": "Create symbolic links across platforms (macOS, Windows)",
|
|
5
5
|
"private": false,
|
|
6
6
|
"type": "commonjs",
|
|
@@ -63,7 +63,7 @@
|
|
|
63
63
|
"cli"
|
|
64
64
|
],
|
|
65
65
|
"dependencies": {
|
|
66
|
-
"@dx-pkg/logger": "^1.0
|
|
66
|
+
"@dx-pkg/logger": "^1.1.0",
|
|
67
67
|
"chalk": "^5.6.2",
|
|
68
68
|
"commander": "^12.0.0"
|
|
69
69
|
},
|
package/src/index.d.mts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { CliLogger } from "@dx-pkg/logger";
|
|
2
2
|
type WindowsSymlinkType = 'file' | 'dir' | 'junction';
|
|
3
3
|
interface SymlinkOptions {
|
|
4
4
|
source: string;
|
|
@@ -69,7 +69,7 @@ declare class MkSymlinkCommand implements Command<MkSymlinkCommandOptions> {
|
|
|
69
69
|
private readonly symlinkManager;
|
|
70
70
|
private readonly options;
|
|
71
71
|
private readonly configService;
|
|
72
|
-
constructor(optionValidator: MkSymlinkOptionValidator, logger:
|
|
72
|
+
constructor(optionValidator: MkSymlinkOptionValidator, logger: CliLogger, symlinkManager: SymlinkOperations, options: MkSymlinkCommandOptions, configService?: ConfigService);
|
|
73
73
|
execute(): Promise<void>;
|
|
74
74
|
private resolveOptions;
|
|
75
75
|
private resolveTarget;
|
|
@@ -89,7 +89,7 @@ declare class ConfigCommand implements Command<ConfigCommandOptions> {
|
|
|
89
89
|
private readonly logger;
|
|
90
90
|
private readonly configService;
|
|
91
91
|
private readonly options;
|
|
92
|
-
constructor(logger:
|
|
92
|
+
constructor(logger: CliLogger, configService: ConfigService, options: ConfigCommandOptions);
|
|
93
93
|
execute(): Promise<void>;
|
|
94
94
|
private handleGet;
|
|
95
95
|
private handleSet;
|
package/src/index.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { CliLogger } from "@dx-pkg/logger";
|
|
2
2
|
type WindowsSymlinkType = 'file' | 'dir' | 'junction';
|
|
3
3
|
interface SymlinkOptions {
|
|
4
4
|
source: string;
|
|
@@ -69,7 +69,7 @@ declare class MkSymlinkCommand implements Command<MkSymlinkCommandOptions> {
|
|
|
69
69
|
private readonly symlinkManager;
|
|
70
70
|
private readonly options;
|
|
71
71
|
private readonly configService;
|
|
72
|
-
constructor(optionValidator: MkSymlinkOptionValidator, logger:
|
|
72
|
+
constructor(optionValidator: MkSymlinkOptionValidator, logger: CliLogger, symlinkManager: SymlinkOperations, options: MkSymlinkCommandOptions, configService?: ConfigService);
|
|
73
73
|
execute(): Promise<void>;
|
|
74
74
|
private resolveOptions;
|
|
75
75
|
private resolveTarget;
|
|
@@ -89,7 +89,7 @@ declare class ConfigCommand implements Command<ConfigCommandOptions> {
|
|
|
89
89
|
private readonly logger;
|
|
90
90
|
private readonly configService;
|
|
91
91
|
private readonly options;
|
|
92
|
-
constructor(logger:
|
|
92
|
+
constructor(logger: CliLogger, configService: ConfigService, options: ConfigCommandOptions);
|
|
93
93
|
execute(): Promise<void>;
|
|
94
94
|
private handleGet;
|
|
95
95
|
private handleSet;
|
package/src/index.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});let e=require(`@dx-pkg/logger`),t=require(`path`),n=require(`fs`),r=require(`os`),i=require(`child_process`),a=require(`util`),o=require(`fs/promises`);var s=class{
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});let e=require(`@dx-pkg/logger`),t=require(`path`),n=require(`fs`),r=require(`os`),i=require(`child_process`),a=require(`util`),o=require(`fs/promises`);var s=class{constructor(e){this.config={};let n=(0,r.homedir)();this.configPath=e||(0,t.resolve)(n,`.mksymlinkrc`),this.loadConfig()}get(e){return this.config[e]}set(e,t){this.config[e]=t,this.saveConfig()}unset(e){delete this.config[e],this.saveConfig()}list(){return{...this.config}}getDefaultSymlinkDir(){return this.get(`symlink.defaultDir`)}setDefaultSymlinkDir(e){this.set(`symlink.defaultDir`,e)}getConfigInfo(){let e=(0,n.existsSync)(this.configPath),t=Object.keys(this.config).length;if(!e)return{path:this.configPath,exists:!1,entries:t};let r=(0,n.statSync)(this.configPath);return{path:this.configPath,exists:!0,size:r.size,modified:r.mtime,entries:t}}loadConfig(){if(!(0,n.existsSync)(this.configPath)){this.config={};return}try{let e=(0,n.readFileSync)(this.configPath,`utf-8`).split(`
|
|
2
2
|
`).filter(e=>e.trim());this.config={};for(let t of e){let[e,...n]=t.split(`=`);e&&n.length>0&&(this.config[e.trim()]=n.join(`=`).trim())}}catch(e){throw Error(`Failed to load config from ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}saveConfig(){try{let e=(0,t.dirname)(this.configPath);(0,n.existsSync)(e)||(0,n.mkdirSync)(e,{recursive:!0});let r=Object.entries(this.config).filter(([,e])=>e!==void 0).map(([e,t])=>`${e}=${t}`).join(`
|
|
3
3
|
`);(0,n.writeFileSync)(this.configPath,r+`
|
|
4
|
-
`,`utf-8`)}catch(e){throw Error(`Failed to save config to ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}};const c=[`file`,`dir`,`junction`];var l=class{constructor(e,t,n,r,i=new s){this.optionValidator=e,this.logger=t,this.symlinkManager=n,this.options=r,this.configService=i}async execute(){this.optionValidator.validate(this.options);try{let{source:e,target:t,type:n,force:r}=this.resolveOptions();this.logger.info(`Creating symlink with options:`),this.logger.info(` Source: ${e}`),this.logger.info(` Target: ${t}`),this.logger.info(` Type: ${n}`),this.logger.info(` Force: ${r}`);let i=await this.symlinkManager.createSymlink({source:e,target:t,type:n,force:r});if(!i.success){if(i.error)throw i.error;this.logger.
|
|
4
|
+
`,`utf-8`)}catch(e){throw Error(`Failed to save config to ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}};const c=[`file`,`dir`,`junction`];var l=class{constructor(e,t,n,r,i=new s){this.optionValidator=e,this.logger=t,this.symlinkManager=n,this.options=r,this.configService=i}async execute(){this.optionValidator.validate(this.options);try{let{source:e,target:t,type:n,force:r}=this.resolveOptions();this.logger.info(`Creating symlink with options:`),this.logger.info(` Source: ${e}`),this.logger.info(` Target: ${t}`),this.logger.info(` Type: ${n}`),this.logger.info(` Force: ${r}`);let i=await this.symlinkManager.createSymlink({source:e,target:t,type:n,force:r});if(!i.success){if(i.error)throw i.error;this.logger.warn(i.message||`Symlink creation failed`);return}this.logger.success(`Symlink created successfully!`),this.logger.info(` Source: ${i.source}`),this.logger.info(` Target: ${i.target}`)}catch(e){throw this.logger.error(`Failed to create symlink: ${e instanceof Error?e.message:String(e)}`),e}}resolveOptions(){let e=process.cwd();return{source:this.options.source||e,target:this.resolveTarget(this.options)||this.generateDefaultTarget(e),type:this.options.type||`junction`,force:this.options.force||!1}}resolveTarget(e){if(e.target!==`.`&&e.target!==`./`)return e.target;let n=process.cwd();return(0,t.resolve)(n,(0,t.basename)(e.source||n))}generateDefaultTarget(e){let n=(0,t.basename)(e),r=`${(0,t.basename)((0,t.dirname)(e))}--${n}`;return(0,t.resolve)(this.configService.getDefaultSymlinkDir()||e,r)}},u=class extends Error{constructor(e,t){super(e),this.name=`SymlinkError`,this.code=t,Error.captureStackTrace(this,this.constructor)}};const d=(0,a.promisify)(i.exec);var f=class{constructor(e,t){this.logger=e,this.osDetector=t}async createSymlink(e){let{source:t,target:n,type:r=`dir`,force:i=!1}=e;try{if(await this.symlinkExists(n)){if(!i)return{success:!1,source:t,target:n,message:`Symlink already exists: ${n}`};this.logger.info(`Force mode enabled. Removing existing symlink: ${n}`),await this.removeSymlink(n)}return this.osDetector.isWindows()?await this.createWindowsSymlink(t,n,r):await this.createUnixSymlink(t,n),{success:!0,source:t,target:n,message:`Symlink created successfully`}}catch(e){return this.logger.error(`Failed to create symlink: ${e instanceof Error?e.message:String(e)}`),{success:!1,source:t,target:n,error:e instanceof Error?e:Error(String(e))}}}async createWindowsSymlink(e,t,n){let r;switch(n){case`junction`:r=`/J`;break;case`file`:r=``;break;default:r=`/D`;break}let i=r?`mklink ${r} "${t}" "${e}"`:`mklink "${t}" "${e}"`;this.logger.info(`Executing Windows command: ${i}`);try{let{stdout:e,stderr:t}=await d(i,{shell:`cmd.exe`});if(t&&!t.includes(`Junction`)&&!t.includes(`symbolic link`))throw new u(t,`WINDOWS_MKLINK_ERROR`);this.logger.info(`Windows mklink output: ${e}`)}catch(e){throw new u(`Failed to create Windows symlink: ${e instanceof Error?e.message:String(e)}`,`WINDOWS_MKLINK_FAILED`)}}async createUnixSymlink(e,t){try{await(0,o.symlink)(e,t),this.logger.info(`Unix symlink created: ln -s ${e} ${t}`)}catch(e){throw new u(`Failed to create Unix symlink: ${e instanceof Error?e.message:String(e)}`,`UNIX_SYMLINK_FAILED`)}}async symlinkExists(e){try{return(await(0,o.lstat)(e)).isSymbolicLink()}catch{return!1}}async removeSymlink(e){try{await(0,o.access)(e),await(0,o.unlink)(e),this.logger.info(`Symlink removed: ${e}`)}catch(e){throw new u(`Failed to remove symlink: ${e instanceof Error?e.message:String(e)}`,`REMOVE_SYMLINK_FAILED`)}}},p=class{constructor(){this._platform=this.normalizePlatform((0,r.platform)())}normalizePlatform(e){switch(e){case`darwin`:return`darwin`;case`win32`:return`win32`;case`linux`:return`linux`;default:return`unknown`}}getPlatform(){return this._platform}isWindows(){return this._platform===`win32`}isMacOS(){return this._platform===`darwin`}isLinux(){return this._platform===`linux`}},m=class{validate(e){this.validateType(e.type),this.validateSource(e.source),this.validateForce(e.force)}validateType(e){if(e&&!c.includes(e))throw new u(`Invalid symlink type: ${e}. Must be one of: ${c.join(`, `)}`,`INVALID_SYMLINK_TYPE`)}validateSource(e){if(!e)return;let r=(0,t.resolve)(e);if(!(0,n.existsSync)(r))throw new u(`Source path does not exist: ${r}`,`SOURCE_NOT_FOUND`)}validateForce(e){if(e!==void 0&&typeof e!=`boolean`)throw new u(`Invalid force option: ${e}. Must be a boolean.`,`INVALID_FORCE_OPTION`)}},h=class{static create(t){let n=new m,r=new e.ConsoleLogger;return new l(n,r,new f(r,new p),t)}},g=class{constructor(e,t,n){this.logger=e,this.configService=t,this.options=n}async execute(){try{let{action:e,key:t,value:n}=this.options;switch(e){case`get`:await this.handleGet(t);break;case`set`:await this.handleSet(t,n);break;case`unset`:await this.handleUnset(t);break;case`list`:await this.handleList();break;case`info`:await this.handleInfo();break;default:throw Error(`Unknown config action: ${e}`)}}catch(e){throw this.logger.error(`Config command failed: ${e instanceof Error?e.message:String(e)}`),e}}async handleGet(e){if(!e)throw Error(`Key is required for get action`);let t=this.configService.get(e);if(t===void 0){this.logger.warn(`Config key '${e}' not found`);return}this.logger.info(t)}async handleSet(e,t){if(!e)throw Error(`Key is required for set action`);if(!t)throw Error(`Value is required for set action`);this.configService.set(e,t),this.logger.success(`Config set: ${e} = ${t}`)}async handleUnset(e){if(!e)throw Error(`Key is required for unset action`);this.configService.unset(e),this.logger.success(`Config unset: ${e}`)}async handleList(){let e=this.configService.list(),t=Object.entries(e);if(t.length===0){this.logger.info(`No configuration found`);return}this.logger.info(`Current configuration:`);for(let[e,n]of t)this.logger.info(` ${e} = ${n}`)}async handleInfo(){let e=this.configService.getConfigInfo();this.logger.info(`Configuration file information:`),this.logger.info(` Path: ${e.path}`),this.logger.info(` Exists: ${e.exists?`Yes`:`No`}`),e.exists&&(this.logger.info(` Size: ${e.size} bytes`),this.logger.info(` Modified: ${e.modified?.toLocaleString()}`)),this.logger.info(` Entries: ${e.entries}`)}};const _=[`get`,`set`,`unset`,`list`,`info`];var v=class{validate(e){this.validateAction(e.action),this.validateKey(e.action,e.key),this.validateValue(e.action,e.value)}validateAction(e){if(!_.includes(e))throw new u(`Invalid config action: ${e}. Must be one of: ${_.join(`, `)}`,`INVALID_CONFIG_ACTION`)}validateKey(e,t){if(!(e===`list`||e===`info`)){if(!t)throw new u(`Key is required for '${e}' action`,`MISSING_CONFIG_KEY`);if(typeof t!=`string`||t.trim().length===0)throw new u(`Invalid config key: ${t}. Key must be a non-empty string.`,`INVALID_CONFIG_KEY`)}}validateValue(e,t){if(e===`set`){if(!t)throw new u(`Value is required for 'set' action`,`MISSING_CONFIG_VALUE`);if(typeof t!=`string`)throw new u(`Invalid config value: ${t}. Value must be a string.`,`INVALID_CONFIG_VALUE`)}}},y=class{static create(t){return new v().validate(t),new g(new e.ConsoleLogger,new s,t)}};exports.ConfigCommandFactory=y,exports.MkSymlinkCommandFactory=h;
|
package/src/index.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import{ConsoleLogger as e}from"@dx-pkg/logger";import{basename as t,dirname as n,resolve as r}from"path";import{existsSync as i,mkdirSync as a,readFileSync as o,statSync as s,writeFileSync as c}from"fs";import{homedir as l,platform as u}from"os";import{exec as d}from"child_process";import{promisify as f}from"util";import{access as p,lstat as m,symlink as h,unlink as g}from"fs/promises";var _=class{
|
|
1
|
+
import{ConsoleLogger as e}from"@dx-pkg/logger";import{basename as t,dirname as n,resolve as r}from"path";import{existsSync as i,mkdirSync as a,readFileSync as o,statSync as s,writeFileSync as c}from"fs";import{homedir as l,platform as u}from"os";import{exec as d}from"child_process";import{promisify as f}from"util";import{access as p,lstat as m,symlink as h,unlink as g}from"fs/promises";var _=class{constructor(e){this.config={};let t=l();this.configPath=e||r(t,`.mksymlinkrc`),this.loadConfig()}get(e){return this.config[e]}set(e,t){this.config[e]=t,this.saveConfig()}unset(e){delete this.config[e],this.saveConfig()}list(){return{...this.config}}getDefaultSymlinkDir(){return this.get(`symlink.defaultDir`)}setDefaultSymlinkDir(e){this.set(`symlink.defaultDir`,e)}getConfigInfo(){let e=i(this.configPath),t=Object.keys(this.config).length;if(!e)return{path:this.configPath,exists:!1,entries:t};let n=s(this.configPath);return{path:this.configPath,exists:!0,size:n.size,modified:n.mtime,entries:t}}loadConfig(){if(!i(this.configPath)){this.config={};return}try{let e=o(this.configPath,`utf-8`).split(`
|
|
2
2
|
`).filter(e=>e.trim());this.config={};for(let t of e){let[e,...n]=t.split(`=`);e&&n.length>0&&(this.config[e.trim()]=n.join(`=`).trim())}}catch(e){throw Error(`Failed to load config from ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}saveConfig(){try{let e=n(this.configPath);i(e)||a(e,{recursive:!0});let t=Object.entries(this.config).filter(([,e])=>e!==void 0).map(([e,t])=>`${e}=${t}`).join(`
|
|
3
3
|
`);c(this.configPath,t+`
|
|
4
|
-
`,`utf-8`)}catch(e){throw Error(`Failed to save config to ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}};const v=[`file`,`dir`,`junction`];var y=class{constructor(e,t,n,r,i=new _){this.optionValidator=e,this.logger=t,this.symlinkManager=n,this.options=r,this.configService=i}async execute(){this.optionValidator.validate(this.options);try{let{source:e,target:t,type:n,force:r}=this.resolveOptions();this.logger.info(`Creating symlink with options:`),this.logger.info(` Source: ${e}`),this.logger.info(` Target: ${t}`),this.logger.info(` Type: ${n}`),this.logger.info(` Force: ${r}`);let i=await this.symlinkManager.createSymlink({source:e,target:t,type:n,force:r});if(!i.success){if(i.error)throw i.error;this.logger.
|
|
4
|
+
`,`utf-8`)}catch(e){throw Error(`Failed to save config to ${this.configPath}: ${e instanceof Error?e.message:String(e)}`)}}};const v=[`file`,`dir`,`junction`];var y=class{constructor(e,t,n,r,i=new _){this.optionValidator=e,this.logger=t,this.symlinkManager=n,this.options=r,this.configService=i}async execute(){this.optionValidator.validate(this.options);try{let{source:e,target:t,type:n,force:r}=this.resolveOptions();this.logger.info(`Creating symlink with options:`),this.logger.info(` Source: ${e}`),this.logger.info(` Target: ${t}`),this.logger.info(` Type: ${n}`),this.logger.info(` Force: ${r}`);let i=await this.symlinkManager.createSymlink({source:e,target:t,type:n,force:r});if(!i.success){if(i.error)throw i.error;this.logger.warn(i.message||`Symlink creation failed`);return}this.logger.success(`Symlink created successfully!`),this.logger.info(` Source: ${i.source}`),this.logger.info(` Target: ${i.target}`)}catch(e){throw this.logger.error(`Failed to create symlink: ${e instanceof Error?e.message:String(e)}`),e}}resolveOptions(){let e=process.cwd();return{source:this.options.source||e,target:this.resolveTarget(this.options)||this.generateDefaultTarget(e),type:this.options.type||`junction`,force:this.options.force||!1}}resolveTarget(e){if(e.target!==`.`&&e.target!==`./`)return e.target;let n=process.cwd();return r(n,t(e.source||n))}generateDefaultTarget(e){let i=t(e),a=`${t(n(e))}--${i}`;return r(this.configService.getDefaultSymlinkDir()||e,a)}},b=class extends Error{constructor(e,t){super(e),this.name=`SymlinkError`,this.code=t,Error.captureStackTrace(this,this.constructor)}};const x=f(d);var S=class{constructor(e,t){this.logger=e,this.osDetector=t}async createSymlink(e){let{source:t,target:n,type:r=`dir`,force:i=!1}=e;try{if(await this.symlinkExists(n)){if(!i)return{success:!1,source:t,target:n,message:`Symlink already exists: ${n}`};this.logger.info(`Force mode enabled. Removing existing symlink: ${n}`),await this.removeSymlink(n)}return this.osDetector.isWindows()?await this.createWindowsSymlink(t,n,r):await this.createUnixSymlink(t,n),{success:!0,source:t,target:n,message:`Symlink created successfully`}}catch(e){return this.logger.error(`Failed to create symlink: ${e instanceof Error?e.message:String(e)}`),{success:!1,source:t,target:n,error:e instanceof Error?e:Error(String(e))}}}async createWindowsSymlink(e,t,n){let r;switch(n){case`junction`:r=`/J`;break;case`file`:r=``;break;default:r=`/D`;break}let i=r?`mklink ${r} "${t}" "${e}"`:`mklink "${t}" "${e}"`;this.logger.info(`Executing Windows command: ${i}`);try{let{stdout:e,stderr:t}=await x(i,{shell:`cmd.exe`});if(t&&!t.includes(`Junction`)&&!t.includes(`symbolic link`))throw new b(t,`WINDOWS_MKLINK_ERROR`);this.logger.info(`Windows mklink output: ${e}`)}catch(e){throw new b(`Failed to create Windows symlink: ${e instanceof Error?e.message:String(e)}`,`WINDOWS_MKLINK_FAILED`)}}async createUnixSymlink(e,t){try{await h(e,t),this.logger.info(`Unix symlink created: ln -s ${e} ${t}`)}catch(e){throw new b(`Failed to create Unix symlink: ${e instanceof Error?e.message:String(e)}`,`UNIX_SYMLINK_FAILED`)}}async symlinkExists(e){try{return(await m(e)).isSymbolicLink()}catch{return!1}}async removeSymlink(e){try{await p(e),await g(e),this.logger.info(`Symlink removed: ${e}`)}catch(e){throw new b(`Failed to remove symlink: ${e instanceof Error?e.message:String(e)}`,`REMOVE_SYMLINK_FAILED`)}}},C=class{constructor(){this._platform=this.normalizePlatform(u())}normalizePlatform(e){switch(e){case`darwin`:return`darwin`;case`win32`:return`win32`;case`linux`:return`linux`;default:return`unknown`}}getPlatform(){return this._platform}isWindows(){return this._platform===`win32`}isMacOS(){return this._platform===`darwin`}isLinux(){return this._platform===`linux`}},w=class{validate(e){this.validateType(e.type),this.validateSource(e.source),this.validateForce(e.force)}validateType(e){if(e&&!v.includes(e))throw new b(`Invalid symlink type: ${e}. Must be one of: ${v.join(`, `)}`,`INVALID_SYMLINK_TYPE`)}validateSource(e){if(!e)return;let t=r(e);if(!i(t))throw new b(`Source path does not exist: ${t}`,`SOURCE_NOT_FOUND`)}validateForce(e){if(e!==void 0&&typeof e!=`boolean`)throw new b(`Invalid force option: ${e}. Must be a boolean.`,`INVALID_FORCE_OPTION`)}},T=class{static create(t){let n=new w,r=new e;return new y(n,r,new S(r,new C),t)}},E=class{constructor(e,t,n){this.logger=e,this.configService=t,this.options=n}async execute(){try{let{action:e,key:t,value:n}=this.options;switch(e){case`get`:await this.handleGet(t);break;case`set`:await this.handleSet(t,n);break;case`unset`:await this.handleUnset(t);break;case`list`:await this.handleList();break;case`info`:await this.handleInfo();break;default:throw Error(`Unknown config action: ${e}`)}}catch(e){throw this.logger.error(`Config command failed: ${e instanceof Error?e.message:String(e)}`),e}}async handleGet(e){if(!e)throw Error(`Key is required for get action`);let t=this.configService.get(e);if(t===void 0){this.logger.warn(`Config key '${e}' not found`);return}this.logger.info(t)}async handleSet(e,t){if(!e)throw Error(`Key is required for set action`);if(!t)throw Error(`Value is required for set action`);this.configService.set(e,t),this.logger.success(`Config set: ${e} = ${t}`)}async handleUnset(e){if(!e)throw Error(`Key is required for unset action`);this.configService.unset(e),this.logger.success(`Config unset: ${e}`)}async handleList(){let e=this.configService.list(),t=Object.entries(e);if(t.length===0){this.logger.info(`No configuration found`);return}this.logger.info(`Current configuration:`);for(let[e,n]of t)this.logger.info(` ${e} = ${n}`)}async handleInfo(){let e=this.configService.getConfigInfo();this.logger.info(`Configuration file information:`),this.logger.info(` Path: ${e.path}`),this.logger.info(` Exists: ${e.exists?`Yes`:`No`}`),e.exists&&(this.logger.info(` Size: ${e.size} bytes`),this.logger.info(` Modified: ${e.modified?.toLocaleString()}`)),this.logger.info(` Entries: ${e.entries}`)}};const D=[`get`,`set`,`unset`,`list`,`info`];var O=class{validate(e){this.validateAction(e.action),this.validateKey(e.action,e.key),this.validateValue(e.action,e.value)}validateAction(e){if(!D.includes(e))throw new b(`Invalid config action: ${e}. Must be one of: ${D.join(`, `)}`,`INVALID_CONFIG_ACTION`)}validateKey(e,t){if(!(e===`list`||e===`info`)){if(!t)throw new b(`Key is required for '${e}' action`,`MISSING_CONFIG_KEY`);if(typeof t!=`string`||t.trim().length===0)throw new b(`Invalid config key: ${t}. Key must be a non-empty string.`,`INVALID_CONFIG_KEY`)}}validateValue(e,t){if(e===`set`){if(!t)throw new b(`Value is required for 'set' action`,`MISSING_CONFIG_VALUE`);if(typeof t!=`string`)throw new b(`Invalid config value: ${t}. Value must be a string.`,`INVALID_CONFIG_VALUE`)}}},k=class{static create(t){return new O().validate(t),new E(new e,new _,t)}};export{k as ConfigCommandFactory,T as MkSymlinkCommandFactory};
|