sheldonify 0.1.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 (107) hide show
  1. package/.claude-plugin/plugin.json +8 -0
  2. package/.clinerules +24 -0
  3. package/.cursor/rules/sheldonify.md +24 -0
  4. package/.github/copilot-instructions.md +16 -0
  5. package/.kiro/steering/sheldonify.md +24 -0
  6. package/.openclaw/skills/sheldonify +83 -0
  7. package/.windsurf/rules/sheldonify.md +24 -0
  8. package/AGENTS.md +56 -0
  9. package/LICENSE +191 -0
  10. package/README.md +388 -0
  11. package/bin/sheldonify.js +2 -0
  12. package/codex.md +32 -0
  13. package/dist/classifier.d.ts +7 -0
  14. package/dist/classifier.d.ts.map +1 -0
  15. package/dist/classifier.js +19 -0
  16. package/dist/classifier.js.map +1 -0
  17. package/dist/cli.d.ts +2 -0
  18. package/dist/cli.d.ts.map +1 -0
  19. package/dist/cli.js +158 -0
  20. package/dist/cli.js.map +1 -0
  21. package/dist/config.d.ts +10 -0
  22. package/dist/config.d.ts.map +1 -0
  23. package/dist/config.js +121 -0
  24. package/dist/config.js.map +1 -0
  25. package/dist/duplicates.d.ts +3 -0
  26. package/dist/duplicates.d.ts.map +1 -0
  27. package/dist/duplicates.js +59 -0
  28. package/dist/duplicates.js.map +1 -0
  29. package/dist/executor.d.ts +3 -0
  30. package/dist/executor.d.ts.map +1 -0
  31. package/dist/executor.js +99 -0
  32. package/dist/executor.js.map +1 -0
  33. package/dist/index-generator.d.ts +3 -0
  34. package/dist/index-generator.d.ts.map +1 -0
  35. package/dist/index-generator.js +22 -0
  36. package/dist/index-generator.js.map +1 -0
  37. package/dist/index.d.ts +12 -0
  38. package/dist/index.d.ts.map +1 -0
  39. package/dist/index.js +11 -0
  40. package/dist/index.js.map +1 -0
  41. package/dist/output.d.ts +7 -0
  42. package/dist/output.d.ts.map +1 -0
  43. package/dist/output.js +129 -0
  44. package/dist/output.js.map +1 -0
  45. package/dist/planner.d.ts +3 -0
  46. package/dist/planner.d.ts.map +1 -0
  47. package/dist/planner.js +83 -0
  48. package/dist/planner.js.map +1 -0
  49. package/dist/protected.d.ts +4 -0
  50. package/dist/protected.d.ts.map +1 -0
  51. package/dist/protected.js +74 -0
  52. package/dist/protected.js.map +1 -0
  53. package/dist/scanner.d.ts +8 -0
  54. package/dist/scanner.d.ts.map +1 -0
  55. package/dist/scanner.js +65 -0
  56. package/dist/scanner.js.map +1 -0
  57. package/dist/strategies/by-context.d.ts +9 -0
  58. package/dist/strategies/by-context.d.ts.map +1 -0
  59. package/dist/strategies/by-context.js +104 -0
  60. package/dist/strategies/by-context.js.map +1 -0
  61. package/dist/strategies/by-date.d.ts +6 -0
  62. package/dist/strategies/by-date.d.ts.map +1 -0
  63. package/dist/strategies/by-date.js +23 -0
  64. package/dist/strategies/by-date.js.map +1 -0
  65. package/dist/strategies/by-type.d.ts +6 -0
  66. package/dist/strategies/by-type.d.ts.map +1 -0
  67. package/dist/strategies/by-type.js +65 -0
  68. package/dist/strategies/by-type.js.map +1 -0
  69. package/dist/strategies/custom.d.ts +7 -0
  70. package/dist/strategies/custom.d.ts.map +1 -0
  71. package/dist/strategies/custom.js +31 -0
  72. package/dist/strategies/custom.js.map +1 -0
  73. package/dist/strategies/index.d.ts +3 -0
  74. package/dist/strategies/index.d.ts.map +1 -0
  75. package/dist/strategies/index.js +18 -0
  76. package/dist/strategies/index.js.map +1 -0
  77. package/dist/types.d.ts +125 -0
  78. package/dist/types.d.ts.map +1 -0
  79. package/dist/types.js +2 -0
  80. package/dist/types.js.map +1 -0
  81. package/dist/undo-generator.d.ts +4 -0
  82. package/dist/undo-generator.d.ts.map +1 -0
  83. package/dist/undo-generator.js +65 -0
  84. package/dist/undo-generator.js.map +1 -0
  85. package/dist/utils/fs-helpers.d.ts +6 -0
  86. package/dist/utils/fs-helpers.d.ts.map +1 -0
  87. package/dist/utils/fs-helpers.js +54 -0
  88. package/dist/utils/fs-helpers.js.map +1 -0
  89. package/dist/utils/hash.d.ts +2 -0
  90. package/dist/utils/hash.d.ts.map +1 -0
  91. package/dist/utils/hash.js +12 -0
  92. package/dist/utils/hash.js.map +1 -0
  93. package/dist/utils/media.d.ts +6 -0
  94. package/dist/utils/media.d.ts.map +1 -0
  95. package/dist/utils/media.js +63 -0
  96. package/dist/utils/media.js.map +1 -0
  97. package/dist/utils/platform.d.ts +5 -0
  98. package/dist/utils/platform.d.ts.map +1 -0
  99. package/dist/utils/platform.js +15 -0
  100. package/dist/utils/platform.js.map +1 -0
  101. package/hooks/check-install.js +32 -0
  102. package/hooks/hooks.json +17 -0
  103. package/package.json +67 -0
  104. package/skills/sheldonify/SKILL.md +110 -0
  105. package/skills/sheldonify/references/config-format.md +55 -0
  106. package/skills/sheldonify-scan/SKILL.md +87 -0
  107. package/skills/sheldonify-undo/SKILL.md +45 -0
package/dist/config.js ADDED
@@ -0,0 +1,121 @@
1
+ import { z } from 'zod';
2
+ import fs from 'node:fs/promises';
3
+ import path from 'node:path';
4
+ const CustomRuleSchema = z.object({
5
+ name: z.string(),
6
+ match: z.object({
7
+ extensions: z.array(z.string()).optional(),
8
+ patterns: z.array(z.string()).optional(),
9
+ regex: z.string().optional(),
10
+ }),
11
+ priority: z.number().optional(),
12
+ });
13
+ const ConfigFileSchema = z.object({
14
+ strategy: z.enum(['type', 'context', 'date', 'custom']).optional(),
15
+ depth: z.number().int().min(1).optional(),
16
+ protected: z.object({
17
+ useDefaults: z.boolean().optional(),
18
+ include: z.array(z.string()).optional(),
19
+ exclude: z.array(z.string()).optional(),
20
+ }).optional(),
21
+ typeStrategy: z.object({
22
+ extraMappings: z.record(z.string()).optional(),
23
+ categoryRenames: z.record(z.string()).optional(),
24
+ }).optional(),
25
+ contextStrategy: z.object({
26
+ extraKeywords: z.record(z.array(z.string())).optional(),
27
+ }).optional(),
28
+ dateStrategy: z.object({
29
+ dateSource: z.enum(['modified', 'created']).optional(),
30
+ }).optional(),
31
+ customRules: z.array(CustomRuleSchema).optional(),
32
+ });
33
+ const DEFAULTS = {
34
+ strategy: 'type',
35
+ depth: 1,
36
+ dryRun: false,
37
+ verbose: false,
38
+ jsonOutput: false,
39
+ targetDir: '.',
40
+ protected: {
41
+ useDefaults: true,
42
+ include: [],
43
+ exclude: [],
44
+ },
45
+ typeStrategy: {
46
+ extraMappings: {},
47
+ categoryRenames: {},
48
+ },
49
+ contextStrategy: {
50
+ extraKeywords: {},
51
+ },
52
+ dateStrategy: {
53
+ dateSource: 'modified',
54
+ },
55
+ customRules: [],
56
+ };
57
+ export async function loadConfig(targetDir, cliOptions) {
58
+ const configFile = await findAndParseConfig(targetDir, cliOptions.config);
59
+ const config = {
60
+ ...DEFAULTS,
61
+ targetDir: path.resolve(targetDir),
62
+ strategy: (cliOptions.strategy ?? configFile?.strategy ?? DEFAULTS.strategy),
63
+ depth: cliOptions.depth ? parseInt(cliOptions.depth, 10) : (configFile?.depth ?? DEFAULTS.depth),
64
+ dryRun: cliOptions.dryRun ?? DEFAULTS.dryRun,
65
+ verbose: cliOptions.verbose ?? DEFAULTS.verbose,
66
+ jsonOutput: cliOptions.json ?? DEFAULTS.jsonOutput,
67
+ protected: {
68
+ useDefaults: configFile?.protected?.useDefaults ?? DEFAULTS.protected.useDefaults,
69
+ include: configFile?.protected?.include ?? DEFAULTS.protected.include,
70
+ exclude: configFile?.protected?.exclude ?? DEFAULTS.protected.exclude,
71
+ },
72
+ typeStrategy: {
73
+ extraMappings: configFile?.typeStrategy?.extraMappings ?? DEFAULTS.typeStrategy.extraMappings,
74
+ categoryRenames: configFile?.typeStrategy?.categoryRenames ?? DEFAULTS.typeStrategy.categoryRenames,
75
+ },
76
+ contextStrategy: {
77
+ extraKeywords: configFile?.contextStrategy?.extraKeywords ?? DEFAULTS.contextStrategy.extraKeywords,
78
+ },
79
+ dateStrategy: {
80
+ dateSource: configFile?.dateStrategy?.dateSource ?? DEFAULTS.dateStrategy.dateSource,
81
+ },
82
+ customRules: configFile?.customRules ?? DEFAULTS.customRules,
83
+ };
84
+ return config;
85
+ }
86
+ async function findAndParseConfig(targetDir, explicitPath) {
87
+ const candidates = explicitPath
88
+ ? [path.resolve(explicitPath)]
89
+ : [
90
+ path.join(path.resolve(targetDir), 'sheldonify.config.json'),
91
+ path.join(homedir(), 'sheldonify.config.json'),
92
+ ];
93
+ for (const candidate of candidates) {
94
+ try {
95
+ const raw = await fs.readFile(candidate, 'utf-8');
96
+ const parsed = JSON.parse(raw);
97
+ const validated = ConfigFileSchema.parse(parsed);
98
+ return validated;
99
+ }
100
+ catch (err) {
101
+ if (err instanceof z.ZodError) {
102
+ const issues = err.issues.map(i => ` ${i.path.join('.')}: ${i.message}`).join('\n');
103
+ throw new Error(`Invalid config file ${candidate}:\n${issues}`);
104
+ }
105
+ if (isNodeError(err) && err.code === 'ENOENT') {
106
+ continue;
107
+ }
108
+ if (explicitPath) {
109
+ throw new Error(`Failed to read config file ${candidate}: ${err}`);
110
+ }
111
+ }
112
+ }
113
+ return null;
114
+ }
115
+ function homedir() {
116
+ return process.env.HOME ?? process.env.USERPROFILE ?? '';
117
+ }
118
+ function isNodeError(err) {
119
+ return err instanceof Error && 'code' in err;
120
+ }
121
+ //# sourceMappingURL=config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"config.js","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { FileEntry, DuplicateGroup } from './types.js';
2
+ export declare function findDuplicates(files: FileEntry[]): Promise<DuplicateGroup[]>;
3
+ //# sourceMappingURL=duplicates.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"duplicates.d.ts","sourceRoot":"","sources":["../src/duplicates.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAGvD,wBAAsB,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC,CAyDlF"}
@@ -0,0 +1,59 @@
1
+ import { hashFile } from './utils/hash.js';
2
+ export async function findDuplicates(files) {
3
+ // Phase 1: group by size — unique sizes can't be duplicates
4
+ const sizeGroups = new Map();
5
+ for (const file of files) {
6
+ if (file.isDirectory)
7
+ continue;
8
+ const group = sizeGroups.get(file.size);
9
+ if (group) {
10
+ group.push(file);
11
+ }
12
+ else {
13
+ sizeGroups.set(file.size, [file]);
14
+ }
15
+ }
16
+ const duplicateGroups = [];
17
+ // Phase 2: hash files that share a size
18
+ for (const group of sizeGroups.values()) {
19
+ if (group.length < 2)
20
+ continue;
21
+ const hashGroups = new Map();
22
+ for (const file of group) {
23
+ try {
24
+ const hash = await hashFile(file.absolutePath);
25
+ file.hash = hash;
26
+ const hg = hashGroups.get(hash);
27
+ if (hg) {
28
+ hg.push(file);
29
+ }
30
+ else {
31
+ hashGroups.set(hash, [file]);
32
+ }
33
+ }
34
+ catch {
35
+ // skip files we can't hash
36
+ }
37
+ }
38
+ for (const [hash, hashGroup] of hashGroups) {
39
+ if (hashGroup.length < 2)
40
+ continue;
41
+ // Keep the file with the shortest path; ties broken by earliest modification
42
+ const sorted = [...hashGroup].sort((a, b) => {
43
+ const pathDiff = a.relativePath.length - b.relativePath.length;
44
+ if (pathDiff !== 0)
45
+ return pathDiff;
46
+ return a.modifiedAt.getTime() - b.modifiedAt.getTime();
47
+ });
48
+ const kept = sorted[0];
49
+ const dupes = sorted.slice(1);
50
+ duplicateGroups.push({
51
+ hash,
52
+ kept: kept.absolutePath,
53
+ duplicates: dupes.map(d => d.absolutePath),
54
+ });
55
+ }
56
+ }
57
+ return duplicateGroups;
58
+ }
59
+ //# sourceMappingURL=duplicates.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"duplicates.js","sourceRoot":"","sources":["../src/duplicates.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAE3C,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,KAAkB;IACrD,4DAA4D;IAC5D,MAAM,UAAU,GAAG,IAAI,GAAG,EAAuB,CAAC;IAClD,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,IAAI,IAAI,CAAC,WAAW;YAAE,SAAS;QAC/B,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACxC,IAAI,KAAK,EAAE,CAAC;YACV,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACnB,CAAC;aAAM,CAAC;YACN,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAED,MAAM,eAAe,GAAqB,EAAE,CAAC;IAE7C,wCAAwC;IACxC,KAAK,MAAM,KAAK,IAAI,UAAU,CAAC,MAAM,EAAE,EAAE,CAAC;QACxC,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC;YAAE,SAAS;QAE/B,MAAM,UAAU,GAAG,IAAI,GAAG,EAAuB,CAAC;QAClD,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBAC/C,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;gBACjB,MAAM,EAAE,GAAG,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;gBAChC,IAAI,EAAE,EAAE,CAAC;oBACP,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAChB,CAAC;qBAAM,CAAC;oBACN,UAAU,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;gBAC/B,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,2BAA2B;YAC7B,CAAC;QACH,CAAC;QAED,KAAK,MAAM,CAAC,IAAI,EAAE,SAAS,CAAC,IAAI,UAAU,EAAE,CAAC;YAC3C,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC;gBAAE,SAAS;YAEnC,6EAA6E;YAC7E,MAAM,MAAM,GAAG,CAAC,GAAG,SAAS,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;gBAC1C,MAAM,QAAQ,GAAG,CAAC,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,CAAC,YAAY,CAAC,MAAM,CAAC;gBAC/D,IAAI,QAAQ,KAAK,CAAC;oBAAE,OAAO,QAAQ,CAAC;gBACpC,OAAO,CAAC,CAAC,UAAU,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;YACzD,CAAC,CAAC,CAAC;YAEH,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;YACvB,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAE9B,eAAe,CAAC,IAAI,CAAC;gBACnB,IAAI;gBACJ,IAAI,EAAE,IAAI,CAAC,YAAY;gBACvB,UAAU,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC;aAC3C,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,eAAe,CAAC;AACzB,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { MovePlan, SheldonifyConfig, ExecutionResult } from './types.js';
2
+ export declare function executePlan(plan: MovePlan, config: SheldonifyConfig): Promise<ExecutionResult>;
3
+ //# sourceMappingURL=executor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.d.ts","sourceRoot":"","sources":["../src/executor.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,QAAQ,EAAiB,gBAAgB,EAAE,eAAe,EAAkB,MAAM,YAAY,CAAC;AAKxG,wBAAsB,WAAW,CAC/B,IAAI,EAAE,QAAQ,EACd,MAAM,EAAE,gBAAgB,GACvB,OAAO,CAAC,eAAe,CAAC,CAmF1B"}
@@ -0,0 +1,99 @@
1
+ import fs from 'node:fs/promises';
2
+ import path from 'node:path';
3
+ import { safeMove, ensureDir } from './utils/fs-helpers.js';
4
+ import { generateUndoSh, generateUndoPs1 } from './undo-generator.js';
5
+ import { generateIndex } from './index-generator.js';
6
+ export async function executePlan(plan, config) {
7
+ const errors = [];
8
+ let operationsExecuted = 0;
9
+ let aborted = false;
10
+ const completedOps = [];
11
+ const onAbort = () => { aborted = true; };
12
+ process.on('SIGINT', onAbort);
13
+ process.on('SIGTERM', onAbort);
14
+ try {
15
+ // Create all target directories
16
+ const dirs = new Set();
17
+ for (const op of plan.operations) {
18
+ dirs.add(path.dirname(op.destination));
19
+ }
20
+ for (const dir of dirs) {
21
+ await ensureDir(dir);
22
+ }
23
+ // Execute moves with progress
24
+ const total = plan.operations.length;
25
+ for (const op of plan.operations) {
26
+ if (aborted) {
27
+ errors.push('Aborted by user (SIGINT). Partial undo script generated.');
28
+ break;
29
+ }
30
+ try {
31
+ await safeMove(op.source, op.destination);
32
+ completedOps.push(op);
33
+ operationsExecuted++;
34
+ if (!config.jsonOutput && total > 10 && operationsExecuted % 50 === 0) {
35
+ process.stderr.write(` Moving files... ${operationsExecuted}/${total}\r`);
36
+ }
37
+ }
38
+ catch (err) {
39
+ const msg = err instanceof Error ? err.message : String(err);
40
+ errors.push(`Failed to move ${op.source} → ${op.destination}: ${msg}`);
41
+ }
42
+ }
43
+ if (!config.jsonOutput && total > 10) {
44
+ process.stderr.write('\n');
45
+ }
46
+ // Write duplicate metadata sidecars
47
+ if (!aborted) {
48
+ for (const group of plan.duplicateGroups) {
49
+ await writeDuplicateMetadata(group, config);
50
+ }
51
+ }
52
+ // Generate undo scripts (always — even on abort, for completed operations)
53
+ const opsForUndo = aborted ? completedOps : plan.operations;
54
+ const undoSh = generateUndoSh(opsForUndo, config.targetDir);
55
+ const undoPs1 = generateUndoPs1(opsForUndo, config.targetDir);
56
+ const undoShPath = path.join(config.targetDir, '_sheldonify-undo.sh');
57
+ const undoPs1Path = path.join(config.targetDir, '_sheldonify-undo.ps1');
58
+ await fs.writeFile(undoShPath, undoSh, 'utf-8');
59
+ await fs.writeFile(undoPs1Path, undoPs1, 'utf-8');
60
+ // Generate index
61
+ const partialPlan = aborted
62
+ ? { ...plan, operations: completedOps }
63
+ : plan;
64
+ const index = generateIndex(partialPlan, config);
65
+ const indexPath = path.join(config.targetDir, '_sheldonify-index.json');
66
+ await fs.writeFile(indexPath, JSON.stringify(index, null, 2), 'utf-8');
67
+ return {
68
+ success: errors.length === 0,
69
+ operationsExecuted,
70
+ errors,
71
+ indexPath,
72
+ undoShPath,
73
+ undoPs1Path,
74
+ };
75
+ }
76
+ finally {
77
+ process.removeListener('SIGINT', onAbort);
78
+ process.removeListener('SIGTERM', onAbort);
79
+ }
80
+ }
81
+ async function writeDuplicateMetadata(group, config) {
82
+ for (const dupPath of group.duplicates) {
83
+ const dupRelPath = path.relative(config.targetDir, dupPath);
84
+ const movedTo = path.join(config.targetDir, '_duplicates', dupRelPath);
85
+ const metaPath = movedTo + '.sheldonify-meta.json';
86
+ try {
87
+ await ensureDir(path.dirname(metaPath));
88
+ await fs.writeFile(metaPath, JSON.stringify({
89
+ originalPath: dupRelPath,
90
+ hash: group.hash,
91
+ keptCopy: path.relative(config.targetDir, group.kept),
92
+ }, null, 2), 'utf-8');
93
+ }
94
+ catch {
95
+ // non-critical
96
+ }
97
+ }
98
+ }
99
+ //# sourceMappingURL=executor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"executor.js","sourceRoot":"","sources":["../src/executor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { MovePlan, SheldonifyConfig, SheldonifyIndex, UncertainResolution } from './types.js';
2
+ export declare function generateIndex(plan: MovePlan, config: SheldonifyConfig, uncertainResolutions?: UncertainResolution[]): SheldonifyIndex;
3
+ //# sourceMappingURL=index-generator.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index-generator.d.ts","sourceRoot":"","sources":["../src/index-generator.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,eAAe,EAAE,mBAAmB,EAAE,MAAM,YAAY,CAAC;AAE9F,wBAAgB,aAAa,CAC3B,IAAI,EAAE,QAAQ,EACd,MAAM,EAAE,gBAAgB,EACxB,oBAAoB,GAAE,mBAAmB,EAAO,GAC/C,eAAe,CAmBjB"}
@@ -0,0 +1,22 @@
1
+ import path from 'node:path';
2
+ export function generateIndex(plan, config, uncertainResolutions = []) {
3
+ return {
4
+ version: 1,
5
+ timestamp: new Date().toISOString(),
6
+ strategy: config.strategy,
7
+ targetDir: config.targetDir,
8
+ stats: plan.stats,
9
+ operations: plan.operations.map(op => ({
10
+ source: path.relative(config.targetDir, op.source),
11
+ destination: path.relative(config.targetDir, op.destination),
12
+ reason: op.reason,
13
+ type: op.type,
14
+ })),
15
+ undo: plan.operations.map(op => ({
16
+ from: path.relative(config.targetDir, op.destination),
17
+ to: path.relative(config.targetDir, op.source),
18
+ })),
19
+ uncertainResolutions,
20
+ };
21
+ }
22
+ //# sourceMappingURL=index-generator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index-generator.js","sourceRoot":"","sources":["../src/index-generator.ts"],"names":[],"mappings":"AAAA,OAAO,IAAI,MAAM,WAAW,CAAC;AAG7B,MAAM,UAAU,aAAa,CAC3B,IAAc,EACd,MAAwB,EACxB,uBAA8C,EAAE;IAEhD,OAAO;QACL,OAAO,EAAE,CAAC;QACV,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;QACnC,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,SAAS,EAAE,MAAM,CAAC,SAAS;QAC3B,KAAK,EAAE,IAAI,CAAC,KAAK;QACjB,UAAU,EAAE,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YACrC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,EAAE,CAAC,MAAM,CAAC;YAClD,WAAW,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,EAAE,CAAC,WAAW,CAAC;YAC5D,MAAM,EAAE,EAAE,CAAC,MAAM;YACjB,IAAI,EAAE,EAAE,CAAC,IAAI;SACd,CAAC,CAAC;QACH,IAAI,EAAE,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;YAC/B,IAAI,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,EAAE,CAAC,WAAW,CAAC;YACrD,EAAE,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,EAAE,CAAC,MAAM,CAAC;SAC/C,CAAC,CAAC;QACH,oBAAoB;KACrB,CAAC;AACJ,CAAC"}
@@ -0,0 +1,12 @@
1
+ export { loadConfig } from './config.js';
2
+ export { scanDirectory, getScanWarnings } from './scanner.js';
3
+ export { createProtectedChecker, hasProjectMarker } from './protected.js';
4
+ export { getStrategy } from './strategies/index.js';
5
+ export { classify } from './classifier.js';
6
+ export { findDuplicates } from './duplicates.js';
7
+ export { buildMovePlan } from './planner.js';
8
+ export { executePlan } from './executor.js';
9
+ export { generateIndex } from './index-generator.js';
10
+ export { generateUndoSh, generateUndoPs1 } from './undo-generator.js';
11
+ export type { SheldonifyConfig, StrategyName, FileEntry, Classification, MoveOperation, MovePlan, DuplicateGroup, ExecutionResult, SheldonifyIndex, SortStrategy, CustomRule, UncertainFile, UncertainResolution, PlanStats, } from './types.js';
12
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,aAAa,EAAE,eAAe,EAAE,MAAM,cAAc,CAAC;AAC9D,OAAO,EAAE,sBAAsB,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAC1E,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AACpD,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtE,YAAY,EACV,gBAAgB,EAChB,YAAY,EACZ,SAAS,EACT,cAAc,EACd,aAAa,EACb,QAAQ,EACR,cAAc,EACd,eAAe,EACf,eAAe,EACf,YAAY,EACZ,UAAU,EACV,aAAa,EACb,mBAAmB,EACnB,SAAS,GACV,MAAM,YAAY,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,11 @@
1
+ export { loadConfig } from './config.js';
2
+ export { scanDirectory, getScanWarnings } from './scanner.js';
3
+ export { createProtectedChecker, hasProjectMarker } from './protected.js';
4
+ export { getStrategy } from './strategies/index.js';
5
+ export { classify } from './classifier.js';
6
+ export { findDuplicates } from './duplicates.js';
7
+ export { buildMovePlan } from './planner.js';
8
+ export { executePlan } from './executor.js';
9
+ export { generateIndex } from './index-generator.js';
10
+ export { generateUndoSh, generateUndoPs1 } from './undo-generator.js';
11
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,aAAa,EAAE,eAAe,EAAE,MAAM,cAAc,CAAC;AAC9D,OAAO,EAAE,sBAAsB,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAC1E,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AACpD,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,7 @@
1
+ import { MovePlan, ExecutionResult, SheldonifyConfig } from './types.js';
2
+ export declare function printDryRun(plan: MovePlan, config: SheldonifyConfig): void;
3
+ export declare function printResult(result: ExecutionResult, config: SheldonifyConfig): void;
4
+ export declare function printUncertainFiles(plan: MovePlan, config: SheldonifyConfig): void;
5
+ export declare function printError(message: string, jsonOutput: boolean): void;
6
+ export declare function printVerbose(message: string, config: SheldonifyConfig): void;
7
+ //# sourceMappingURL=output.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"output.d.ts","sourceRoot":"","sources":["../src/output.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,QAAQ,EAAiB,eAAe,EAAE,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAGxF,wBAAgB,WAAW,CAAC,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,gBAAgB,GAAG,IAAI,CAqC1E;AAED,wBAAgB,WAAW,CAAC,MAAM,EAAE,eAAe,EAAE,MAAM,EAAE,gBAAgB,GAAG,IAAI,CAyBnF;AAED,wBAAgB,mBAAmB,CAAC,IAAI,EAAE,QAAQ,EAAE,MAAM,EAAE,gBAAgB,GAAG,IAAI,CAwBlF;AAED,wBAAgB,UAAU,CAAC,OAAO,EAAE,MAAM,EAAE,UAAU,EAAE,OAAO,GAAG,IAAI,CAMrE;AAED,wBAAgB,YAAY,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,gBAAgB,GAAG,IAAI,CAI5E"}
package/dist/output.js ADDED
@@ -0,0 +1,129 @@
1
+ import chalk from 'chalk';
2
+ import path from 'node:path';
3
+ export function printDryRun(plan, config) {
4
+ if (config.jsonOutput) {
5
+ printJson({ status: 'dry-run', plan: sanitizePlan(plan, config) });
6
+ return;
7
+ }
8
+ const header = chalk.bold(`sheldonify — preview for ${chalk.cyan(config.targetDir)} using "${config.strategy}" strategy`);
9
+ stderr(header);
10
+ stderr('');
11
+ if (plan.operations.length === 0) {
12
+ stderr(chalk.yellow(' Nothing to do — directory is already organized.'));
13
+ return;
14
+ }
15
+ const grouped = groupByDestFolder(plan.operations, config.targetDir);
16
+ for (const [folder, ops] of Object.entries(grouped)) {
17
+ stderr(chalk.green(` 📁 ${folder}/`));
18
+ for (const op of ops) {
19
+ const from = path.relative(config.targetDir, op.source);
20
+ const to = path.relative(config.targetDir, op.destination);
21
+ const icon = op.type === 'duplicate' ? chalk.yellow('⚠') : ' ';
22
+ stderr(` ${icon} ${chalk.dim(from)} → ${chalk.white(to)}`);
23
+ }
24
+ }
25
+ stderr('');
26
+ stderr(chalk.bold(' Stats:'));
27
+ stderr(` Files to move: ${plan.stats.filesToMove}`);
28
+ stderr(` Duplicates: ${plan.stats.duplicatesFound}`);
29
+ stderr(` Folders to create: ${plan.stats.foldersToCreate}`);
30
+ stderr(` Protected skipped: ${plan.stats.protectedSkipped}`);
31
+ if (plan.stats.uncertainCount > 0) {
32
+ stderr(` Uncertain: ${chalk.yellow(plan.stats.uncertainCount.toString())}`);
33
+ }
34
+ stderr('');
35
+ stderr(chalk.dim(' Run without --dry-run to apply.'));
36
+ }
37
+ export function printResult(result, config) {
38
+ if (config.jsonOutput) {
39
+ printJson({
40
+ status: result.success ? 'completed' : 'partial',
41
+ operationsExecuted: result.operationsExecuted,
42
+ errors: result.errors,
43
+ indexPath: result.indexPath,
44
+ undoShPath: result.undoShPath,
45
+ undoPs1Path: result.undoPs1Path,
46
+ });
47
+ return;
48
+ }
49
+ if (result.success) {
50
+ stderr(chalk.green.bold(`\n ✓ Organized ${result.operationsExecuted} files successfully.`));
51
+ }
52
+ else {
53
+ stderr(chalk.red.bold(`\n ✗ Completed with ${result.errors.length} error(s).`));
54
+ for (const err of result.errors) {
55
+ stderr(chalk.red(` • ${err}`));
56
+ }
57
+ }
58
+ stderr(chalk.dim(`\n Index: ${result.indexPath}`));
59
+ stderr(chalk.dim(` Undo: ${result.undoShPath} (bash) / ${result.undoPs1Path} (powershell)`));
60
+ stderr('');
61
+ }
62
+ export function printUncertainFiles(plan, config) {
63
+ if (config.jsonOutput) {
64
+ printJson({
65
+ status: 'needs-review',
66
+ uncertainFiles: plan.uncertainFiles.map(u => ({
67
+ file: u.file.relativePath,
68
+ candidates: u.candidates.map(c => ({
69
+ category: c.category,
70
+ confidence: c.confidence,
71
+ reason: c.reason,
72
+ })),
73
+ })),
74
+ });
75
+ return;
76
+ }
77
+ stderr(chalk.yellow.bold('\n ⚠ The following files need your input:\n'));
78
+ for (const uncertain of plan.uncertainFiles) {
79
+ stderr(` ${chalk.white(uncertain.file.relativePath)}`);
80
+ for (const candidate of uncertain.candidates) {
81
+ stderr(` → ${chalk.cyan(candidate.category)} (${(candidate.confidence * 100).toFixed(0)}%: ${candidate.reason})`);
82
+ }
83
+ stderr('');
84
+ }
85
+ }
86
+ export function printError(message, jsonOutput) {
87
+ if (jsonOutput) {
88
+ printJson({ status: 'error', error: message });
89
+ }
90
+ else {
91
+ stderr(chalk.red.bold(`\n Error: ${message}\n`));
92
+ }
93
+ }
94
+ export function printVerbose(message, config) {
95
+ if (config.verbose && !config.jsonOutput) {
96
+ stderr(chalk.dim(` [verbose] ${message}`));
97
+ }
98
+ }
99
+ function printJson(data) {
100
+ process.stdout.write(JSON.stringify(data, null, 2) + '\n');
101
+ }
102
+ function stderr(message) {
103
+ process.stderr.write(message + '\n');
104
+ }
105
+ function groupByDestFolder(operations, targetDir) {
106
+ const groups = {};
107
+ for (const op of operations) {
108
+ const rel = path.relative(targetDir, op.destination);
109
+ const folder = rel.split(path.sep)[0] ?? 'root';
110
+ if (!groups[folder])
111
+ groups[folder] = [];
112
+ groups[folder].push(op);
113
+ }
114
+ return groups;
115
+ }
116
+ function sanitizePlan(plan, config) {
117
+ return {
118
+ stats: plan.stats,
119
+ operations: plan.operations.map(op => ({
120
+ from: path.relative(config.targetDir, op.source),
121
+ to: path.relative(config.targetDir, op.destination),
122
+ reason: op.reason,
123
+ type: op.type,
124
+ })),
125
+ skippedPaths: plan.skippedPaths,
126
+ warnings: plan.warnings,
127
+ };
128
+ }
129
+ //# sourceMappingURL=output.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"output.js","sourceRoot":"","sources":["../src/output.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { FileEntry, Classification, DuplicateGroup, UncertainFile, MovePlan, SheldonifyConfig } from './types.js';
2
+ export declare function buildMovePlan(files: FileEntry[], classifications: Map<string, Classification>, duplicateGroups: DuplicateGroup[], uncertainFiles: UncertainFile[], config: SheldonifyConfig): MovePlan;
3
+ //# sourceMappingURL=planner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"planner.d.ts","sourceRoot":"","sources":["../src/planner.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,cAAc,EAAE,cAAc,EAAE,aAAa,EAAE,QAAQ,EAAiB,gBAAgB,EAAE,MAAM,YAAY,CAAC;AAGjI,wBAAgB,aAAa,CAC3B,KAAK,EAAE,SAAS,EAAE,EAClB,eAAe,EAAE,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,EAC5C,eAAe,EAAE,cAAc,EAAE,EACjC,cAAc,EAAE,aAAa,EAAE,EAC/B,MAAM,EAAE,gBAAgB,GACvB,QAAQ,CAsFV"}
@@ -0,0 +1,83 @@
1
+ import path from 'node:path';
2
+ import { resolveConflict } from './utils/fs-helpers.js';
3
+ export function buildMovePlan(files, classifications, duplicateGroups, uncertainFiles, config) {
4
+ const operations = [];
5
+ const usedDests = new Set();
6
+ const duplicateSet = new Set();
7
+ const warnings = [];
8
+ // Collect all duplicate file paths
9
+ for (const group of duplicateGroups) {
10
+ for (const dup of group.duplicates) {
11
+ duplicateSet.add(dup);
12
+ }
13
+ }
14
+ // Build organize operations from classifications
15
+ for (const file of files) {
16
+ if (duplicateSet.has(file.absolutePath))
17
+ continue;
18
+ const classification = classifications.get(file.absolutePath);
19
+ if (!classification)
20
+ continue;
21
+ const destDir = path.join(config.targetDir, classification.category);
22
+ let destPath = path.join(destDir, file.name);
23
+ // Skip if file is already in the right place (case-insensitive on Windows)
24
+ if (path.dirname(file.absolutePath).toLowerCase() === destDir.toLowerCase())
25
+ continue;
26
+ destPath = resolveConflict(destPath, usedDests);
27
+ if (destPath.toLowerCase() === file.absolutePath.toLowerCase())
28
+ continue;
29
+ usedDests.add(destPath);
30
+ operations.push({
31
+ source: file.absolutePath,
32
+ destination: destPath,
33
+ reason: classification.reason,
34
+ type: 'organize',
35
+ });
36
+ }
37
+ // Build duplicate move operations
38
+ for (const group of duplicateGroups) {
39
+ for (const dup of group.duplicates) {
40
+ const dupFile = files.find(f => f.absolutePath === dup);
41
+ if (!dupFile)
42
+ continue;
43
+ const dupRelPath = path.relative(config.targetDir, dup);
44
+ if (dupRelPath.startsWith('_duplicates')) {
45
+ warnings.push(`Skipping duplicate already in _duplicates: ${dupRelPath}`);
46
+ continue;
47
+ }
48
+ let destPath = path.join(config.targetDir, '_duplicates', dupRelPath);
49
+ destPath = resolveConflict(destPath, usedDests);
50
+ usedDests.add(destPath);
51
+ operations.push({
52
+ source: dup,
53
+ destination: destPath,
54
+ reason: `duplicate of ${path.relative(config.targetDir, group.kept)}`,
55
+ type: 'duplicate',
56
+ });
57
+ }
58
+ }
59
+ // Calculate folders to create
60
+ const foldersToCreate = new Set();
61
+ for (const op of operations) {
62
+ const dir = path.dirname(op.destination);
63
+ if (dir !== config.targetDir) {
64
+ foldersToCreate.add(dir);
65
+ }
66
+ }
67
+ return {
68
+ operations,
69
+ duplicateGroups,
70
+ skippedPaths: [],
71
+ uncertainFiles,
72
+ warnings,
73
+ stats: {
74
+ totalFiles: files.length,
75
+ filesToMove: operations.filter(o => o.type === 'organize').length,
76
+ duplicatesFound: duplicateGroups.reduce((sum, g) => sum + g.duplicates.length, 0),
77
+ foldersToCreate: foldersToCreate.size,
78
+ protectedSkipped: 0,
79
+ uncertainCount: uncertainFiles.length,
80
+ },
81
+ };
82
+ }
83
+ //# sourceMappingURL=planner.js.map