@caracal-lynx/sluice 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.
- package/CLAUDE.md +1822 -0
- package/LICENCE-FAQ.md +74 -0
- package/LICENSE +92 -0
- package/README.md +582 -0
- package/dist/adapters/source/csv.d.ts +10 -0
- package/dist/adapters/source/csv.d.ts.map +1 -0
- package/dist/adapters/source/csv.js +110 -0
- package/dist/adapters/source/csv.js.map +1 -0
- package/dist/adapters/source/index.d.ts +9 -0
- package/dist/adapters/source/index.d.ts.map +1 -0
- package/dist/adapters/source/index.js +26 -0
- package/dist/adapters/source/index.js.map +1 -0
- package/dist/adapters/source/mssql.d.ts +11 -0
- package/dist/adapters/source/mssql.d.ts.map +1 -0
- package/dist/adapters/source/mssql.js +230 -0
- package/dist/adapters/source/mssql.js.map +1 -0
- package/dist/adapters/source/pg.d.ts +11 -0
- package/dist/adapters/source/pg.d.ts.map +1 -0
- package/dist/adapters/source/pg.js +88 -0
- package/dist/adapters/source/pg.js.map +1 -0
- package/dist/adapters/source/registry.d.ts +10 -0
- package/dist/adapters/source/registry.d.ts.map +1 -0
- package/dist/adapters/source/registry.js +36 -0
- package/dist/adapters/source/registry.js.map +1 -0
- package/dist/adapters/source/rest.d.ts +16 -0
- package/dist/adapters/source/rest.d.ts.map +1 -0
- package/dist/adapters/source/rest.js +182 -0
- package/dist/adapters/source/rest.js.map +1 -0
- package/dist/adapters/source/rest.types.d.ts +15 -0
- package/dist/adapters/source/rest.types.d.ts.map +1 -0
- package/dist/adapters/source/rest.types.js +6 -0
- package/dist/adapters/source/rest.types.js.map +1 -0
- package/dist/adapters/source/types.d.ts +23 -0
- package/dist/adapters/source/types.d.ts.map +1 -0
- package/dist/adapters/source/types.js +4 -0
- package/dist/adapters/source/types.js.map +1 -0
- package/dist/adapters/source/xlsx.d.ts +10 -0
- package/dist/adapters/source/xlsx.d.ts.map +1 -0
- package/dist/adapters/source/xlsx.js +71 -0
- package/dist/adapters/source/xlsx.js.map +1 -0
- package/dist/adapters/target/bc.d.ts +21 -0
- package/dist/adapters/target/bc.d.ts.map +1 -0
- package/dist/adapters/target/bc.js +188 -0
- package/dist/adapters/target/bc.js.map +1 -0
- package/dist/adapters/target/bluecherry.d.ts +10 -0
- package/dist/adapters/target/bluecherry.d.ts.map +1 -0
- package/dist/adapters/target/bluecherry.js +127 -0
- package/dist/adapters/target/bluecherry.js.map +1 -0
- package/dist/adapters/target/csv.d.ts +10 -0
- package/dist/adapters/target/csv.d.ts.map +1 -0
- package/dist/adapters/target/csv.js +40 -0
- package/dist/adapters/target/csv.js.map +1 -0
- package/dist/adapters/target/ifs.d.ts +10 -0
- package/dist/adapters/target/ifs.d.ts.map +1 -0
- package/dist/adapters/target/ifs.js +55 -0
- package/dist/adapters/target/ifs.js.map +1 -0
- package/dist/adapters/target/index.d.ts +8 -0
- package/dist/adapters/target/index.d.ts.map +1 -0
- package/dist/adapters/target/index.js +22 -0
- package/dist/adapters/target/index.js.map +1 -0
- package/dist/adapters/target/pg.d.ts +11 -0
- package/dist/adapters/target/pg.d.ts.map +1 -0
- package/dist/adapters/target/pg.js +103 -0
- package/dist/adapters/target/pg.js.map +1 -0
- package/dist/adapters/target/registry.d.ts +9 -0
- package/dist/adapters/target/registry.d.ts.map +1 -0
- package/dist/adapters/target/registry.js +29 -0
- package/dist/adapters/target/registry.js.map +1 -0
- package/dist/adapters/target/types.d.ts +15 -0
- package/dist/adapters/target/types.d.ts.map +1 -0
- package/dist/adapters/target/types.js +4 -0
- package/dist/adapters/target/types.js.map +1 -0
- package/dist/cli.d.ts +25 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +354 -0
- package/dist/cli.js.map +1 -0
- package/dist/config/index.d.ts +4 -0
- package/dist/config/index.d.ts.map +1 -0
- package/dist/config/index.js +6 -0
- package/dist/config/index.js.map +1 -0
- package/dist/config/loader.d.ts +5 -0
- package/dist/config/loader.d.ts.map +1 -0
- package/dist/config/loader.js +135 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/config/schema.d.ts +4162 -0
- package/dist/config/schema.d.ts.map +1 -0
- package/dist/config/schema.js +263 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/config/types.d.ts +3 -0
- package/dist/config/types.d.ts.map +1 -0
- package/dist/config/types.js +4 -0
- package/dist/config/types.js.map +1 -0
- package/dist/dq/engine.d.ts +10 -0
- package/dist/dq/engine.d.ts.map +1 -0
- package/dist/dq/engine.js +114 -0
- package/dist/dq/engine.js.map +1 -0
- package/dist/dq/index.d.ts +6 -0
- package/dist/dq/index.d.ts.map +1 -0
- package/dist/dq/index.js +6 -0
- package/dist/dq/index.js.map +1 -0
- package/dist/dq/reporter.d.ts +5 -0
- package/dist/dq/reporter.d.ts.map +1 -0
- package/dist/dq/reporter.js +41 -0
- package/dist/dq/reporter.js.map +1 -0
- package/dist/dq/rules/allowedValues.d.ts +7 -0
- package/dist/dq/rules/allowedValues.d.ts.map +1 -0
- package/dist/dq/rules/allowedValues.js +26 -0
- package/dist/dq/rules/allowedValues.js.map +1 -0
- package/dist/dq/rules/email.d.ts +7 -0
- package/dist/dq/rules/email.d.ts.map +1 -0
- package/dist/dq/rules/email.js +24 -0
- package/dist/dq/rules/email.js.map +1 -0
- package/dist/dq/rules/index.d.ts +15 -0
- package/dist/dq/rules/index.d.ts.map +1 -0
- package/dist/dq/rules/index.js +30 -0
- package/dist/dq/rules/index.js.map +1 -0
- package/dist/dq/rules/maxLength.d.ts +7 -0
- package/dist/dq/rules/maxLength.d.ts.map +1 -0
- package/dist/dq/rules/maxLength.js +25 -0
- package/dist/dq/rules/maxLength.js.map +1 -0
- package/dist/dq/rules/minMax.d.ts +11 -0
- package/dist/dq/rules/minMax.d.ts.map +1 -0
- package/dist/dq/rules/minMax.js +52 -0
- package/dist/dq/rules/minMax.js.map +1 -0
- package/dist/dq/rules/notNull.d.ts +7 -0
- package/dist/dq/rules/notNull.d.ts.map +1 -0
- package/dist/dq/rules/notNull.js +21 -0
- package/dist/dq/rules/notNull.js.map +1 -0
- package/dist/dq/rules/pattern.d.ts +7 -0
- package/dist/dq/rules/pattern.d.ts.map +1 -0
- package/dist/dq/rules/pattern.js +31 -0
- package/dist/dq/rules/pattern.js.map +1 -0
- package/dist/dq/rules/types.d.ts +6 -0
- package/dist/dq/rules/types.d.ts.map +1 -0
- package/dist/dq/rules/types.js +4 -0
- package/dist/dq/rules/types.js.map +1 -0
- package/dist/dq/rules/ukPostcode.d.ts +7 -0
- package/dist/dq/rules/ukPostcode.d.ts.map +1 -0
- package/dist/dq/rules/ukPostcode.js +24 -0
- package/dist/dq/rules/ukPostcode.js.map +1 -0
- package/dist/dq/rules/unique.d.ts +14 -0
- package/dist/dq/rules/unique.d.ts.map +1 -0
- package/dist/dq/rules/unique.js +9 -0
- package/dist/dq/rules/unique.js.map +1 -0
- package/dist/dq/types.d.ts +29 -0
- package/dist/dq/types.d.ts.map +1 -0
- package/dist/dq/types.js +4 -0
- package/dist/dq/types.js.map +1 -0
- package/dist/enrich/types.d.ts +87 -0
- package/dist/enrich/types.d.ts.map +1 -0
- package/dist/enrich/types.js +4 -0
- package/dist/enrich/types.js.map +1 -0
- package/dist/index.d.ts +17 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +17 -0
- package/dist/index.js.map +1 -0
- package/dist/merge/conflict-log.d.ts +9 -0
- package/dist/merge/conflict-log.d.ts.map +1 -0
- package/dist/merge/conflict-log.js +28 -0
- package/dist/merge/conflict-log.js.map +1 -0
- package/dist/merge/engine.d.ts +7 -0
- package/dist/merge/engine.d.ts.map +1 -0
- package/dist/merge/engine.js +19 -0
- package/dist/merge/engine.js.map +1 -0
- package/dist/merge/index.d.ts +11 -0
- package/dist/merge/index.d.ts.map +1 -0
- package/dist/merge/index.js +34 -0
- package/dist/merge/index.js.map +1 -0
- package/dist/merge/sql-builder.d.ts +19 -0
- package/dist/merge/sql-builder.d.ts.map +1 -0
- package/dist/merge/sql-builder.js +148 -0
- package/dist/merge/sql-builder.js.map +1 -0
- package/dist/merge/strategies/coalesce.d.ts +17 -0
- package/dist/merge/strategies/coalesce.d.ts.map +1 -0
- package/dist/merge/strategies/coalesce.js +77 -0
- package/dist/merge/strategies/coalesce.js.map +1 -0
- package/dist/merge/strategies/index.d.ts +5 -0
- package/dist/merge/strategies/index.d.ts.map +1 -0
- package/dist/merge/strategies/index.js +7 -0
- package/dist/merge/strategies/index.js.map +1 -0
- package/dist/merge/strategies/intersect.d.ts +17 -0
- package/dist/merge/strategies/intersect.d.ts.map +1 -0
- package/dist/merge/strategies/intersect.js +75 -0
- package/dist/merge/strategies/intersect.js.map +1 -0
- package/dist/merge/strategies/priority-override.d.ts +16 -0
- package/dist/merge/strategies/priority-override.d.ts.map +1 -0
- package/dist/merge/strategies/priority-override.js +78 -0
- package/dist/merge/strategies/priority-override.js.map +1 -0
- package/dist/merge/strategies/registry.d.ts +8 -0
- package/dist/merge/strategies/registry.d.ts.map +1 -0
- package/dist/merge/strategies/registry.js +19 -0
- package/dist/merge/strategies/registry.js.map +1 -0
- package/dist/merge/strategies/union.d.ts +15 -0
- package/dist/merge/strategies/union.d.ts.map +1 -0
- package/dist/merge/strategies/union.js +75 -0
- package/dist/merge/strategies/union.js.map +1 -0
- package/dist/merge/types.d.ts +24 -0
- package/dist/merge/types.d.ts.map +1 -0
- package/dist/merge/types.js +4 -0
- package/dist/merge/types.js.map +1 -0
- package/dist/multi-source-runner.d.ts +22 -0
- package/dist/multi-source-runner.d.ts.map +1 -0
- package/dist/multi-source-runner.js +398 -0
- package/dist/multi-source-runner.js.map +1 -0
- package/dist/plugins/index.d.ts +4 -0
- package/dist/plugins/index.d.ts.map +1 -0
- package/dist/plugins/index.js +5 -0
- package/dist/plugins/index.js.map +1 -0
- package/dist/plugins/loader.d.ts +22 -0
- package/dist/plugins/loader.d.ts.map +1 -0
- package/dist/plugins/loader.js +151 -0
- package/dist/plugins/loader.js.map +1 -0
- package/dist/plugins/registry.d.ts +25 -0
- package/dist/plugins/registry.d.ts.map +1 -0
- package/dist/plugins/registry.js +42 -0
- package/dist/plugins/registry.js.map +1 -0
- package/dist/plugins/types.d.ts +61 -0
- package/dist/plugins/types.d.ts.map +1 -0
- package/dist/plugins/types.js +4 -0
- package/dist/plugins/types.js.map +1 -0
- package/dist/runner.d.ts +97 -0
- package/dist/runner.d.ts.map +1 -0
- package/dist/runner.js +520 -0
- package/dist/runner.js.map +1 -0
- package/dist/staging/index.d.ts +3 -0
- package/dist/staging/index.d.ts.map +1 -0
- package/dist/staging/index.js +5 -0
- package/dist/staging/index.js.map +1 -0
- package/dist/staging/schema.d.ts +19 -0
- package/dist/staging/schema.d.ts.map +1 -0
- package/dist/staging/schema.js +15 -0
- package/dist/staging/schema.js.map +1 -0
- package/dist/staging/store.d.ts +71 -0
- package/dist/staging/store.d.ts.map +1 -0
- package/dist/staging/store.js +270 -0
- package/dist/staging/store.js.map +1 -0
- package/dist/transform/cleanse.d.ts +2 -0
- package/dist/transform/cleanse.d.ts.map +1 -0
- package/dist/transform/cleanse.js +59 -0
- package/dist/transform/cleanse.js.map +1 -0
- package/dist/transform/engine.d.ts +10 -0
- package/dist/transform/engine.d.ts.map +1 -0
- package/dist/transform/engine.js +225 -0
- package/dist/transform/engine.js.map +1 -0
- package/dist/transform/expression.d.ts +5 -0
- package/dist/transform/expression.d.ts.map +1 -0
- package/dist/transform/expression.js +52 -0
- package/dist/transform/expression.js.map +1 -0
- package/dist/transform/index.d.ts +6 -0
- package/dist/transform/index.d.ts.map +1 -0
- package/dist/transform/index.js +7 -0
- package/dist/transform/index.js.map +1 -0
- package/dist/transform/lookup.d.ts +10 -0
- package/dist/transform/lookup.d.ts.map +1 -0
- package/dist/transform/lookup.js +66 -0
- package/dist/transform/lookup.js.map +1 -0
- package/dist/transform/types.d.ts +10 -0
- package/dist/transform/types.d.ts.map +1 -0
- package/dist/transform/types.js +4 -0
- package/dist/transform/types.js.map +1 -0
- package/dist/utils/env.d.ts +3 -0
- package/dist/utils/env.d.ts.map +1 -0
- package/dist/utils/env.js +26 -0
- package/dist/utils/env.js.map +1 -0
- package/dist/utils/errors.d.ts +26 -0
- package/dist/utils/errors.d.ts.map +1 -0
- package/dist/utils/errors.js +39 -0
- package/dist/utils/errors.js.map +1 -0
- package/dist/utils/index.d.ts +5 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +7 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/logger.d.ts +14 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +16 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/progress.d.ts +66 -0
- package/dist/utils/progress.d.ts.map +1 -0
- package/dist/utils/progress.js +283 -0
- package/dist/utils/progress.js.map +1 -0
- package/package.json +92 -0
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
export type PhaseKind = 'extract' | 'dq' | 'merge' | 'enrich' | 'transform' | 'load';
|
|
2
|
+
export type PhaseEndState = 'success' | 'warn' | 'fail';
|
|
3
|
+
export type ProgressLogLevel = 'debug' | 'info' | 'warn' | 'error';
|
|
4
|
+
export interface StartPhaseOpts {
|
|
5
|
+
total?: number;
|
|
6
|
+
}
|
|
7
|
+
export interface EndPhaseOpts {
|
|
8
|
+
state?: PhaseEndState;
|
|
9
|
+
message?: string;
|
|
10
|
+
}
|
|
11
|
+
export interface SummaryOpts {
|
|
12
|
+
pipeline: string;
|
|
13
|
+
elapsedMs: number;
|
|
14
|
+
rowsExtracted?: number;
|
|
15
|
+
rowsLoaded?: number;
|
|
16
|
+
warnings?: number;
|
|
17
|
+
state?: PhaseEndState;
|
|
18
|
+
}
|
|
19
|
+
export interface ProgressReporterOptions {
|
|
20
|
+
silent?: boolean;
|
|
21
|
+
totalPhases: number;
|
|
22
|
+
logLevel?: ProgressLogLevel;
|
|
23
|
+
/** Override for tests — defaults to `process.stdout.isTTY`. */
|
|
24
|
+
isTTY?: boolean;
|
|
25
|
+
/** Override for tests — defaults to `process.stdout`. */
|
|
26
|
+
stdout?: NodeJS.WritableStream;
|
|
27
|
+
/** Override for tests — defaults to `() => Date.now()`. */
|
|
28
|
+
now?: () => number;
|
|
29
|
+
}
|
|
30
|
+
export declare class ProgressReporter {
|
|
31
|
+
private readonly silent;
|
|
32
|
+
private readonly totalPhases;
|
|
33
|
+
private readonly stdout;
|
|
34
|
+
private readonly isTTY;
|
|
35
|
+
private readonly useBar;
|
|
36
|
+
private readonly now;
|
|
37
|
+
private phaseIndex;
|
|
38
|
+
private phase;
|
|
39
|
+
private bar;
|
|
40
|
+
private indeterminateTimer;
|
|
41
|
+
private indeterminateFrame;
|
|
42
|
+
private stopped;
|
|
43
|
+
constructor(opts: ProgressReporterOptions);
|
|
44
|
+
startPhase(kind: PhaseKind, label: string, opts?: StartPhaseOpts): void;
|
|
45
|
+
update(rows: number): void;
|
|
46
|
+
endPhase(opts?: EndPhaseOpts): void;
|
|
47
|
+
/** Final run summary banner. Safe to call when silent — no-ops. */
|
|
48
|
+
summary(opts: SummaryOpts): void;
|
|
49
|
+
stop(): void;
|
|
50
|
+
private startDeterminateBar;
|
|
51
|
+
private startIndeterminatePulse;
|
|
52
|
+
private renderIndeterminate;
|
|
53
|
+
private stopBar;
|
|
54
|
+
private renderStartLine;
|
|
55
|
+
private renderEndLine;
|
|
56
|
+
private renderPhaseLabel;
|
|
57
|
+
private phaseIcon;
|
|
58
|
+
private stateMarker;
|
|
59
|
+
private formatDeterminate;
|
|
60
|
+
private writeLine;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Convenience factory for a silent reporter (tests, library callers).
|
|
64
|
+
*/
|
|
65
|
+
export declare function createSilentProgress(totalPhases?: number): ProgressReporter;
|
|
66
|
+
//# sourceMappingURL=progress.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"progress.d.ts","sourceRoot":"","sources":["../../src/utils/progress.ts"],"names":[],"mappings":"AA2BA,MAAM,MAAM,SAAS,GAAG,SAAS,GAAG,IAAI,GAAG,OAAO,GAAG,QAAQ,GAAG,WAAW,GAAG,MAAM,CAAC;AAErF,MAAM,MAAM,aAAa,GAAG,SAAS,GAAG,MAAM,GAAG,MAAM,CAAC;AAExD,MAAM,MAAM,gBAAgB,GAAG,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC;AAEnE,MAAM,WAAW,cAAc;IAC7B,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,YAAY;IAC3B,KAAK,CAAC,EAAE,aAAa,CAAC;IACtB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,KAAK,CAAC,EAAE,aAAa,CAAC;CACvB;AAED,MAAM,WAAW,uBAAuB;IACtC,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,EAAE,gBAAgB,CAAC;IAC5B,+DAA+D;IAC/D,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,yDAAyD;IACzD,MAAM,CAAC,EAAE,MAAM,CAAC,cAAc,CAAC;IAC/B,2DAA2D;IAC3D,GAAG,CAAC,EAAE,MAAM,MAAM,CAAC;CACpB;AAiDD,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAU;IACjC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAS;IACrC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAwB;IAC/C,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAU;IAChC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAU;IACjC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAe;IAEnC,OAAO,CAAC,UAAU,CAAK;IACvB,OAAO,CAAC,KAAK,CAA2B;IACxC,OAAO,CAAC,GAAG,CAAsC;IACjD,OAAO,CAAC,kBAAkB,CAA+B;IACzD,OAAO,CAAC,kBAAkB,CAAK;IAC/B,OAAO,CAAC,OAAO,CAAS;gBAEZ,IAAI,EAAE,uBAAuB;IAWzC,UAAU,CAAC,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,GAAE,cAAmB,GAAG,IAAI;IA6B3E,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAU1B,QAAQ,CAAC,IAAI,GAAE,YAAiB,GAAG,IAAI;IAkBvC,mEAAmE;IACnE,OAAO,CAAC,IAAI,EAAE,WAAW,GAAG,IAAI;IAoBhC,IAAI,IAAI,IAAI;IAaZ,OAAO,CAAC,mBAAmB;IAoB3B,OAAO,CAAC,uBAAuB;IAY/B,OAAO,CAAC,mBAAmB;IAW3B,OAAO,CAAC,OAAO;IAaf,OAAO,CAAC,eAAe;IAKvB,OAAO,CAAC,aAAa;IAerB,OAAO,CAAC,gBAAgB;IASxB,OAAO,CAAC,SAAS;IAIjB,OAAO,CAAC,WAAW;IAenB,OAAO,CAAC,iBAAiB;IAWzB,OAAO,CAAC,SAAS;CAGlB;AAED;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,WAAW,SAAI,GAAG,gBAAgB,CAEtE"}
|
|
@@ -0,0 +1,283 @@
|
|
|
1
|
+
// SPDX-License-Identifier: Elastic-2.0
|
|
2
|
+
// Copyright (c) 2026 Caracal Lynx Ltd.
|
|
3
|
+
/**
|
|
4
|
+
* ProgressReporter — renders phase-by-phase progress feedback to stdout.
|
|
5
|
+
*
|
|
6
|
+
* Rendering decisions:
|
|
7
|
+
* silent → no-op (nothing on stdout)
|
|
8
|
+
* logLevel = 'debug' → bar disabled; phase transitions still logged by
|
|
9
|
+
* the caller via the pino logger
|
|
10
|
+
* !isTTY → one plain line per phase start + one per end
|
|
11
|
+
* (no escape codes, no emojis, no colour)
|
|
12
|
+
* otherwise → animated cli-progress bar with ETA for determinate
|
|
13
|
+
* phases, rotating spinner for indeterminate phases
|
|
14
|
+
*
|
|
15
|
+
* The caller drives the state machine:
|
|
16
|
+
* startPhase(kind, label, { total? })
|
|
17
|
+
* update(rows) // many times
|
|
18
|
+
* endPhase({ state, message? }) // success | warn | fail
|
|
19
|
+
* ...repeat for each phase...
|
|
20
|
+
* summary({ pipeline, elapsedMs, ... })
|
|
21
|
+
* stop() // always — put in try/finally
|
|
22
|
+
*/
|
|
23
|
+
import cliProgress from 'cli-progress';
|
|
24
|
+
import pc from 'picocolors';
|
|
25
|
+
const PHASE_ICON_EMOJI = {
|
|
26
|
+
extract: '🔎',
|
|
27
|
+
dq: '🛡️ ',
|
|
28
|
+
merge: '🔀',
|
|
29
|
+
enrich: '🌐',
|
|
30
|
+
transform: '🔧',
|
|
31
|
+
load: '📤',
|
|
32
|
+
};
|
|
33
|
+
// Plain-ASCII fallbacks used when the stream is not a TTY.
|
|
34
|
+
const PHASE_ICON_ASCII = {
|
|
35
|
+
extract: '[EX]',
|
|
36
|
+
dq: '[DQ]',
|
|
37
|
+
merge: '[MG]',
|
|
38
|
+
enrich: '[EN]',
|
|
39
|
+
transform: '[TX]',
|
|
40
|
+
load: '[LD]',
|
|
41
|
+
};
|
|
42
|
+
const PHASE_COLOUR = {
|
|
43
|
+
extract: pc.cyan,
|
|
44
|
+
dq: pc.yellow,
|
|
45
|
+
merge: pc.magenta,
|
|
46
|
+
enrich: pc.cyan,
|
|
47
|
+
transform: pc.blue,
|
|
48
|
+
load: pc.green,
|
|
49
|
+
};
|
|
50
|
+
const SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];
|
|
51
|
+
const INDETERMINATE_TICK_MS = 100;
|
|
52
|
+
function formatRows(n) {
|
|
53
|
+
return n.toLocaleString('en-US');
|
|
54
|
+
}
|
|
55
|
+
function formatElapsed(ms) {
|
|
56
|
+
return `${(ms / 1000).toFixed(1)}s`;
|
|
57
|
+
}
|
|
58
|
+
export class ProgressReporter {
|
|
59
|
+
silent;
|
|
60
|
+
totalPhases;
|
|
61
|
+
stdout;
|
|
62
|
+
isTTY;
|
|
63
|
+
useBar;
|
|
64
|
+
now;
|
|
65
|
+
phaseIndex = 0;
|
|
66
|
+
phase = null;
|
|
67
|
+
bar = null;
|
|
68
|
+
indeterminateTimer = null;
|
|
69
|
+
indeterminateFrame = 0;
|
|
70
|
+
stopped = false;
|
|
71
|
+
constructor(opts) {
|
|
72
|
+
this.silent = opts.silent ?? false;
|
|
73
|
+
this.totalPhases = opts.totalPhases;
|
|
74
|
+
this.stdout = opts.stdout ?? process.stdout;
|
|
75
|
+
this.isTTY = opts.isTTY ?? Boolean(this.stdout.isTTY);
|
|
76
|
+
this.now = opts.now ?? (() => Date.now());
|
|
77
|
+
const barDisabled = opts.logLevel === 'debug';
|
|
78
|
+
this.useBar = !this.silent && this.isTTY && !barDisabled;
|
|
79
|
+
}
|
|
80
|
+
startPhase(kind, label, opts = {}) {
|
|
81
|
+
if (this.stopped)
|
|
82
|
+
return;
|
|
83
|
+
// If a previous phase was left open, close it quietly so we never leak a bar.
|
|
84
|
+
if (this.phase)
|
|
85
|
+
this.endPhase({ state: 'success' });
|
|
86
|
+
this.phaseIndex++;
|
|
87
|
+
this.phase = {
|
|
88
|
+
kind,
|
|
89
|
+
label,
|
|
90
|
+
total: opts.total,
|
|
91
|
+
startMs: this.now(),
|
|
92
|
+
lastRows: 0,
|
|
93
|
+
};
|
|
94
|
+
if (this.silent)
|
|
95
|
+
return;
|
|
96
|
+
if (this.useBar) {
|
|
97
|
+
if (opts.total !== undefined && opts.total > 0) {
|
|
98
|
+
this.startDeterminateBar(opts.total);
|
|
99
|
+
}
|
|
100
|
+
else {
|
|
101
|
+
this.startIndeterminatePulse();
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
// Non-TTY / debug: single line on start. `endPhase` prints the closing line.
|
|
106
|
+
this.writeLine(this.renderStartLine());
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
update(rows) {
|
|
110
|
+
if (this.stopped || !this.phase)
|
|
111
|
+
return;
|
|
112
|
+
this.phase.lastRows = rows;
|
|
113
|
+
if (this.silent)
|
|
114
|
+
return;
|
|
115
|
+
if (this.useBar && this.bar && this.phase.total !== undefined) {
|
|
116
|
+
this.bar.update(Math.min(rows, this.phase.total));
|
|
117
|
+
}
|
|
118
|
+
// Indeterminate bar keeps its own animation timer; we just record lastRows.
|
|
119
|
+
}
|
|
120
|
+
endPhase(opts = {}) {
|
|
121
|
+
if (this.stopped || !this.phase)
|
|
122
|
+
return;
|
|
123
|
+
const state = opts.state ?? 'success';
|
|
124
|
+
const rows = this.phase.lastRows;
|
|
125
|
+
const elapsedMs = this.now() - this.phase.startMs;
|
|
126
|
+
if (!this.silent) {
|
|
127
|
+
if (this.useBar) {
|
|
128
|
+
this.stopBar();
|
|
129
|
+
this.writeLine(this.renderEndLine(state, rows, elapsedMs, opts.message));
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
this.writeLine(this.renderEndLine(state, rows, elapsedMs, opts.message));
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
this.phase = null;
|
|
136
|
+
}
|
|
137
|
+
/** Final run summary banner. Safe to call when silent — no-ops. */
|
|
138
|
+
summary(opts) {
|
|
139
|
+
if (this.silent)
|
|
140
|
+
return;
|
|
141
|
+
const state = opts.state ?? 'success';
|
|
142
|
+
const marker = this.stateMarker(state);
|
|
143
|
+
const bits = [];
|
|
144
|
+
bits.push(marker);
|
|
145
|
+
bits.push(pc.bold(opts.pipeline));
|
|
146
|
+
if (opts.rowsExtracted !== undefined) {
|
|
147
|
+
bits.push(`${formatRows(opts.rowsExtracted)} rows extracted`);
|
|
148
|
+
}
|
|
149
|
+
if (opts.warnings !== undefined && opts.warnings > 0) {
|
|
150
|
+
bits.push(pc.yellow(`${formatRows(opts.warnings)} warnings`));
|
|
151
|
+
}
|
|
152
|
+
if (opts.rowsLoaded !== undefined) {
|
|
153
|
+
bits.push(`${formatRows(opts.rowsLoaded)} loaded`);
|
|
154
|
+
}
|
|
155
|
+
bits.push(pc.dim(formatElapsed(opts.elapsedMs)));
|
|
156
|
+
this.writeLine(bits.join(pc.dim(' · ')));
|
|
157
|
+
}
|
|
158
|
+
stop() {
|
|
159
|
+
if (this.stopped)
|
|
160
|
+
return;
|
|
161
|
+
if (this.phase) {
|
|
162
|
+
// Caller aborted mid-phase (e.g. thrown error). Close with failure state.
|
|
163
|
+
// Set `stopped` after endPhase so the close line renders.
|
|
164
|
+
this.endPhase({ state: 'fail' });
|
|
165
|
+
}
|
|
166
|
+
this.stopped = true;
|
|
167
|
+
this.stopBar();
|
|
168
|
+
}
|
|
169
|
+
// ── Internal rendering ────────────────────────────────────────────────────
|
|
170
|
+
startDeterminateBar(total) {
|
|
171
|
+
const bar = new cliProgress.SingleBar({
|
|
172
|
+
format: this.formatDeterminate(),
|
|
173
|
+
barCompleteChar: '█',
|
|
174
|
+
barIncompleteChar: '░',
|
|
175
|
+
hideCursor: true,
|
|
176
|
+
clearOnComplete: false,
|
|
177
|
+
stopOnComplete: false,
|
|
178
|
+
stream: this.stdout,
|
|
179
|
+
fps: 10,
|
|
180
|
+
noTTYOutput: false,
|
|
181
|
+
forceRedraw: false,
|
|
182
|
+
}, cliProgress.Presets.shades_classic);
|
|
183
|
+
bar.start(total, 0);
|
|
184
|
+
this.bar = bar;
|
|
185
|
+
}
|
|
186
|
+
startIndeterminatePulse() {
|
|
187
|
+
this.indeterminateFrame = 0;
|
|
188
|
+
this.renderIndeterminate();
|
|
189
|
+
this.indeterminateTimer = setInterval(() => {
|
|
190
|
+
this.indeterminateFrame++;
|
|
191
|
+
this.renderIndeterminate();
|
|
192
|
+
}, INDETERMINATE_TICK_MS);
|
|
193
|
+
if (typeof this.indeterminateTimer.unref === 'function') {
|
|
194
|
+
this.indeterminateTimer.unref();
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
renderIndeterminate() {
|
|
198
|
+
if (!this.phase)
|
|
199
|
+
return;
|
|
200
|
+
const spin = SPINNER_FRAMES[this.indeterminateFrame % SPINNER_FRAMES.length] ?? '.';
|
|
201
|
+
const label = this.renderPhaseLabel();
|
|
202
|
+
const rows = formatRows(this.phase.lastRows);
|
|
203
|
+
const elapsed = formatElapsed(this.now() - this.phase.startMs);
|
|
204
|
+
const line = `${label} ${PHASE_COLOUR[this.phase.kind](spin)} ${rows} rows · ${pc.dim(elapsed)}`;
|
|
205
|
+
// Rewrite the current line: CR + clear-to-EOL.
|
|
206
|
+
this.stdout.write(`\r${line}\x1b[K`);
|
|
207
|
+
}
|
|
208
|
+
stopBar() {
|
|
209
|
+
if (this.bar) {
|
|
210
|
+
this.bar.stop();
|
|
211
|
+
this.bar = null;
|
|
212
|
+
}
|
|
213
|
+
if (this.indeterminateTimer) {
|
|
214
|
+
clearInterval(this.indeterminateTimer);
|
|
215
|
+
this.indeterminateTimer = null;
|
|
216
|
+
// Erase the in-flight indeterminate line.
|
|
217
|
+
if (this.isTTY)
|
|
218
|
+
this.stdout.write('\r\x1b[K');
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
renderStartLine() {
|
|
222
|
+
const label = this.renderPhaseLabel();
|
|
223
|
+
return `${label} starting…`;
|
|
224
|
+
}
|
|
225
|
+
renderEndLine(state, rows, elapsedMs, message) {
|
|
226
|
+
const label = this.renderPhaseLabel();
|
|
227
|
+
const marker = this.stateMarker(state);
|
|
228
|
+
const parts = [label, marker];
|
|
229
|
+
if (rows > 0)
|
|
230
|
+
parts.push(`${formatRows(rows)} rows`);
|
|
231
|
+
parts.push(pc.dim(formatElapsed(elapsedMs)));
|
|
232
|
+
if (message)
|
|
233
|
+
parts.push(pc.dim(message));
|
|
234
|
+
return parts.join(' ');
|
|
235
|
+
}
|
|
236
|
+
renderPhaseLabel() {
|
|
237
|
+
if (!this.phase)
|
|
238
|
+
return '';
|
|
239
|
+
const counter = pc.dim(`[${this.phaseIndex}/${this.totalPhases}]`);
|
|
240
|
+
const icon = this.phaseIcon(this.phase.kind);
|
|
241
|
+
const colourise = PHASE_COLOUR[this.phase.kind];
|
|
242
|
+
const name = colourise(pc.bold(this.phase.label));
|
|
243
|
+
return `${counter} ${icon} ${name}`;
|
|
244
|
+
}
|
|
245
|
+
phaseIcon(kind) {
|
|
246
|
+
return this.isTTY ? PHASE_ICON_EMOJI[kind] : PHASE_ICON_ASCII[kind];
|
|
247
|
+
}
|
|
248
|
+
stateMarker(state) {
|
|
249
|
+
if (this.isTTY) {
|
|
250
|
+
switch (state) {
|
|
251
|
+
case 'success': return pc.bold(pc.green('✅'));
|
|
252
|
+
case 'warn': return pc.yellow('⚠️ ');
|
|
253
|
+
case 'fail': return pc.bold(pc.red('❌'));
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
switch (state) {
|
|
257
|
+
case 'success': return '[ok]';
|
|
258
|
+
case 'warn': return '[warn]';
|
|
259
|
+
case 'fail': return '[fail]';
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
formatDeterminate() {
|
|
263
|
+
if (!this.phase)
|
|
264
|
+
return '';
|
|
265
|
+
const label = this.renderPhaseLabel();
|
|
266
|
+
const colourise = PHASE_COLOUR[this.phase.kind];
|
|
267
|
+
// cli-progress tokens: {bar} {value} {total} {eta_formatted} {duration_formatted}
|
|
268
|
+
const bar = `${colourise('{bar}')}`;
|
|
269
|
+
const counts = `{value}/{total} rows`;
|
|
270
|
+
const eta = pc.dim(`ETA {eta_formatted}`);
|
|
271
|
+
return `${label} ${bar} ${counts} · ${eta}`;
|
|
272
|
+
}
|
|
273
|
+
writeLine(s) {
|
|
274
|
+
this.stdout.write(`${s}\n`);
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
/**
|
|
278
|
+
* Convenience factory for a silent reporter (tests, library callers).
|
|
279
|
+
*/
|
|
280
|
+
export function createSilentProgress(totalPhases = 0) {
|
|
281
|
+
return new ProgressReporter({ silent: true, totalPhases });
|
|
282
|
+
}
|
|
283
|
+
//# sourceMappingURL=progress.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"progress.js","sourceRoot":"","sources":["../../src/utils/progress.ts"],"names":[],"mappings":"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"}
|
package/package.json
ADDED
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@caracal-lynx/sluice",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Config-driven ETL toolkit for ERP data migrations",
|
|
5
|
+
"license": "Elastic-2.0",
|
|
6
|
+
"author": "Caracal Lynx Ltd. <michael.scott@caracallynx.com> (https://caracallynx.com)",
|
|
7
|
+
"homepage": "https://github.com/caracal-lynx/sluice#readme",
|
|
8
|
+
"repository": {
|
|
9
|
+
"type": "git",
|
|
10
|
+
"url": "git+https://github.com/caracal-lynx/sluice.git"
|
|
11
|
+
},
|
|
12
|
+
"bugs": {
|
|
13
|
+
"url": "https://github.com/caracal-lynx/sluice/issues"
|
|
14
|
+
},
|
|
15
|
+
"keywords": [
|
|
16
|
+
"etl",
|
|
17
|
+
"data-migration",
|
|
18
|
+
"erp",
|
|
19
|
+
"yaml",
|
|
20
|
+
"duckdb",
|
|
21
|
+
"cli",
|
|
22
|
+
"data-quality",
|
|
23
|
+
"pipeline",
|
|
24
|
+
"typescript"
|
|
25
|
+
],
|
|
26
|
+
"type": "module",
|
|
27
|
+
"main": "dist/index.js",
|
|
28
|
+
"types": "dist/index.d.ts",
|
|
29
|
+
"bin": {
|
|
30
|
+
"sluice": "dist/cli.js"
|
|
31
|
+
},
|
|
32
|
+
"files": [
|
|
33
|
+
"dist",
|
|
34
|
+
"README.md",
|
|
35
|
+
"LICENSE",
|
|
36
|
+
"LICENCE-FAQ.md",
|
|
37
|
+
"CLAUDE.md"
|
|
38
|
+
],
|
|
39
|
+
"publishConfig": {
|
|
40
|
+
"access": "public"
|
|
41
|
+
},
|
|
42
|
+
"engines": {
|
|
43
|
+
"node": ">=24.0.0"
|
|
44
|
+
},
|
|
45
|
+
"scripts": {
|
|
46
|
+
"build": "tsc -p tsconfig.json",
|
|
47
|
+
"dev": "tsx watch src/cli.ts",
|
|
48
|
+
"lint": "eslint src tests",
|
|
49
|
+
"format": "prettier --write src tests",
|
|
50
|
+
"test": "vitest run",
|
|
51
|
+
"test:watch": "vitest",
|
|
52
|
+
"test:cov": "vitest run --coverage",
|
|
53
|
+
"typecheck": "tsc --noEmit",
|
|
54
|
+
"typecheck:tsgo": "tsgo --noEmit",
|
|
55
|
+
"sluice": "tsx src/cli.ts"
|
|
56
|
+
},
|
|
57
|
+
"dependencies": {
|
|
58
|
+
"@duckdb/node-api": "^1.5.0-r.1",
|
|
59
|
+
"axios": "^1.7.0",
|
|
60
|
+
"axios-retry": "^4.5.0",
|
|
61
|
+
"cli-progress": "^3.12.0",
|
|
62
|
+
"commander": "^12.1.0",
|
|
63
|
+
"csv-parse": "^5.5.6",
|
|
64
|
+
"csv-stringify": "^6.5.1",
|
|
65
|
+
"dayjs": "^1.11.13",
|
|
66
|
+
"dotenv": "^16.4.5",
|
|
67
|
+
"expr-eval": "^2.0.2",
|
|
68
|
+
"js-yaml": "^4.1.0",
|
|
69
|
+
"mssql": "^11.0.1",
|
|
70
|
+
"pg": "^8.13.0",
|
|
71
|
+
"picocolors": "^1.1.1",
|
|
72
|
+
"pino": "^9.5.0",
|
|
73
|
+
"xlsx": "^0.18.5",
|
|
74
|
+
"zod": "^3.23.8"
|
|
75
|
+
},
|
|
76
|
+
"devDependencies": {
|
|
77
|
+
"@types/cli-progress": "^3.11.6",
|
|
78
|
+
"@types/js-yaml": "^4.0.9",
|
|
79
|
+
"@types/mssql": "^9.1.5",
|
|
80
|
+
"@types/node": "^24.12.2",
|
|
81
|
+
"@types/pg": "^8.11.10",
|
|
82
|
+
"@typescript/native-preview": "^7.0.0-dev.20260504.1",
|
|
83
|
+
"@vitest/coverage-v8": "^2.1.0",
|
|
84
|
+
"eslint": "^9.17.0",
|
|
85
|
+
"pino-pretty": "^13.0.0",
|
|
86
|
+
"prettier": "^3.4.0",
|
|
87
|
+
"tsx": "^4.19.0",
|
|
88
|
+
"typescript": "^6.0.3",
|
|
89
|
+
"typescript-eslint": "^8.59.1",
|
|
90
|
+
"vitest": "^2.1.0"
|
|
91
|
+
}
|
|
92
|
+
}
|