@jujulego/jill 1.2.0 → 2.0.0-alpha.1

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 (138) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +28 -0
  3. package/dist/commands/list.d.ts +23 -0
  4. package/dist/commands/list.d.ts.map +1 -0
  5. package/dist/commands/list.js +169 -0
  6. package/dist/commands/list.js.map +1 -0
  7. package/dist/filters/affected.filter.d.ts +11 -0
  8. package/dist/filters/affected.filter.d.ts.map +1 -0
  9. package/dist/filters/affected.filter.js +82 -0
  10. package/dist/filters/affected.filter.js.map +1 -0
  11. package/dist/filters/index.d.ts +4 -0
  12. package/dist/filters/index.d.ts.map +1 -0
  13. package/dist/filters/index.js +21 -0
  14. package/dist/filters/index.js.map +1 -0
  15. package/dist/filters/pipeline.d.ts +11 -0
  16. package/dist/filters/pipeline.d.ts.map +1 -0
  17. package/dist/filters/pipeline.js +34 -0
  18. package/dist/filters/pipeline.js.map +1 -0
  19. package/dist/filters/private.filter.d.ts +7 -0
  20. package/dist/filters/private.filter.d.ts.map +1 -0
  21. package/dist/filters/private.filter.js +20 -0
  22. package/dist/filters/private.filter.js.map +1 -0
  23. package/dist/filters/scripts.filter.d.ts +7 -0
  24. package/dist/filters/scripts.filter.d.ts.map +1 -0
  25. package/dist/filters/scripts.filter.js +21 -0
  26. package/dist/filters/scripts.filter.js.map +1 -0
  27. package/dist/git.d.ts +16 -0
  28. package/dist/git.d.ts.map +1 -0
  29. package/dist/git.js +94 -0
  30. package/dist/git.js.map +1 -0
  31. package/dist/index.d.ts +7 -4
  32. package/dist/index.d.ts.map +1 -1
  33. package/dist/index.js +19 -33
  34. package/dist/index.js.map +1 -1
  35. package/dist/main.d.ts.map +1 -1
  36. package/dist/main.js +23 -52
  37. package/dist/main.js.map +1 -1
  38. package/dist/modifiers/global-config.d.ts +6 -0
  39. package/dist/modifiers/global-config.d.ts.map +1 -0
  40. package/dist/modifiers/global-config.js +24 -0
  41. package/dist/modifiers/global-config.js.map +1 -0
  42. package/dist/modifiers/index.d.ts +3 -0
  43. package/dist/modifiers/index.d.ts.map +1 -0
  44. package/dist/modifiers/index.js +20 -0
  45. package/dist/modifiers/index.js.map +1 -0
  46. package/dist/modifiers/load-project.d.ts +7 -0
  47. package/dist/modifiers/load-project.d.ts.map +1 -0
  48. package/dist/modifiers/load-project.js +39 -0
  49. package/dist/modifiers/load-project.js.map +1 -0
  50. package/dist/modifiers/setup-ink.d.ts +2 -0
  51. package/dist/modifiers/setup-ink.d.ts.map +1 -0
  52. package/dist/modifiers/setup-ink.js +19 -0
  53. package/dist/modifiers/setup-ink.js.map +1 -0
  54. package/dist/project/index.d.ts +2 -0
  55. package/dist/project/index.d.ts.map +1 -0
  56. package/dist/project/index.js +19 -0
  57. package/dist/project/index.js.map +1 -0
  58. package/dist/project/project.d.ts +25 -0
  59. package/dist/project/project.d.ts.map +1 -0
  60. package/dist/project/project.js +190 -0
  61. package/dist/project/project.js.map +1 -0
  62. package/dist/project/workspace.d.ts +32 -0
  63. package/dist/project/workspace.d.ts.map +1 -0
  64. package/dist/project/workspace.js +159 -0
  65. package/dist/project/workspace.js.map +1 -0
  66. package/dist/services/index.d.ts +4 -0
  67. package/dist/services/index.d.ts.map +1 -0
  68. package/dist/services/index.js +21 -0
  69. package/dist/services/index.js.map +1 -0
  70. package/dist/services/inversify.config.d.ts +11 -0
  71. package/dist/services/inversify.config.d.ts.map +1 -0
  72. package/dist/services/inversify.config.js +34 -0
  73. package/dist/services/inversify.config.js.map +1 -0
  74. package/dist/services/logger.service.d.ts +5 -0
  75. package/dist/services/logger.service.d.ts.map +1 -0
  76. package/dist/services/logger.service.js +58 -0
  77. package/dist/services/logger.service.js.map +1 -0
  78. package/dist/services/spinner.service.d.ts +14 -0
  79. package/dist/services/spinner.service.d.ts.map +1 -0
  80. package/dist/services/spinner.service.js +55 -0
  81. package/dist/services/spinner.service.js.map +1 -0
  82. package/dist/services/task-manager.service.d.ts +1 -0
  83. package/dist/services/task-manager.service.d.ts.map +1 -0
  84. package/dist/services/task-manager.service.js +18 -0
  85. package/dist/services/task-manager.service.js.map +1 -0
  86. package/dist/types.d.ts +1 -0
  87. package/dist/types.d.ts.map +1 -0
  88. package/dist/types.js +7 -0
  89. package/dist/types.js.map +1 -0
  90. package/dist/ui/cli-list.d.ts +15 -0
  91. package/dist/ui/cli-list.d.ts.map +1 -0
  92. package/dist/ui/cli-list.js +66 -0
  93. package/dist/ui/cli-list.js.map +1 -0
  94. package/dist/ui/global-spinner.d.ts +2 -0
  95. package/dist/ui/global-spinner.d.ts.map +1 -0
  96. package/dist/ui/global-spinner.js +43 -0
  97. package/dist/ui/global-spinner.js.map +1 -0
  98. package/dist/ui/index.d.ts +4 -0
  99. package/dist/ui/index.d.ts.map +1 -0
  100. package/dist/ui/index.js +21 -0
  101. package/dist/ui/index.js.map +1 -0
  102. package/dist/ui/layout.d.ts +2 -0
  103. package/dist/ui/layout.d.ts.map +1 -0
  104. package/dist/ui/layout.js +19 -0
  105. package/dist/ui/layout.js.map +1 -0
  106. package/dist/ui/static-logs.d.ts +2 -0
  107. package/dist/ui/static-logs.d.ts.map +1 -0
  108. package/dist/ui/static-logs.js +79 -0
  109. package/dist/ui/static-logs.js.map +1 -0
  110. package/dist/utils.d.ts +7 -0
  111. package/dist/utils.d.ts.map +1 -0
  112. package/dist/utils.js +61 -0
  113. package/dist/utils.js.map +1 -0
  114. package/package.json +56 -48
  115. package/dist/commands/each.command.d.ts +0 -16
  116. package/dist/commands/each.command.d.ts.map +0 -1
  117. package/dist/commands/each.command.js +0 -203
  118. package/dist/commands/each.command.js.map +0 -1
  119. package/dist/commands/info.command.d.ts +0 -8
  120. package/dist/commands/info.command.d.ts.map +0 -1
  121. package/dist/commands/info.command.js +0 -178
  122. package/dist/commands/info.command.js.map +0 -1
  123. package/dist/commands/list.command.d.ts +0 -21
  124. package/dist/commands/list.command.d.ts.map +0 -1
  125. package/dist/commands/list.command.js +0 -251
  126. package/dist/commands/list.command.js.map +0 -1
  127. package/dist/commands/run.command.d.ts +0 -12
  128. package/dist/commands/run.command.d.ts.map +0 -1
  129. package/dist/commands/run.command.js +0 -111
  130. package/dist/commands/run.command.js.map +0 -1
  131. package/dist/core.plugin.d.ts +0 -2
  132. package/dist/core.plugin.d.ts.map +0 -1
  133. package/dist/core.plugin.js +0 -22
  134. package/dist/core.plugin.js.map +0 -1
  135. package/dist/task-logger.d.ts +0 -14
  136. package/dist/task-logger.d.ts.map +0 -1
  137. package/dist/task-logger.js +0 -76
  138. package/dist/task-logger.js.map +0 -1
@@ -0,0 +1,190 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "Project", {
6
+ enumerable: true,
7
+ get: ()=>Project
8
+ });
9
+ const _asyncLock = /*#__PURE__*/ _interopRequireDefault(require("async-lock"));
10
+ const _promises = /*#__PURE__*/ _interopRequireDefault(require("node:fs/promises"));
11
+ const _nodePath = /*#__PURE__*/ _interopRequireDefault(require("node:path"));
12
+ const _normalizePackageData = /*#__PURE__*/ _interopRequireDefault(require("normalize-package-data"));
13
+ const _tinyGlob = /*#__PURE__*/ _interopRequireDefault(require("tiny-glob"));
14
+ const _services = require("../services");
15
+ const _workspace = require("./workspace");
16
+ function _interopRequireDefault(obj) {
17
+ return obj && obj.__esModule ? obj : {
18
+ default: obj
19
+ };
20
+ }
21
+ var __decorate = (void 0) && (void 0).__decorate || function(decorators, target, key, desc) {
22
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
23
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
24
+ else for(var i = decorators.length - 1; i >= 0; i--)if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
25
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
26
+ };
27
+ class Project {
28
+ // Constructor
29
+ constructor(_root, opts = {}){
30
+ this._root = _root;
31
+ this._names = new Map();
32
+ this._workspaces = new Map();
33
+ this._isFullyLoaded = false;
34
+ this._lock = new _asyncLock.default();
35
+ if (opts.packageManager) {
36
+ this._logger.debug(`Forced use of ${opts.packageManager} in ${_nodePath.default.relative(process.cwd(), this.root) || '.'}`);
37
+ this._packageManager = opts.packageManager;
38
+ }
39
+ }
40
+ // Statics
41
+ static async searchProjectRoot(dir) {
42
+ const logger = _services.container.get(_services.Logger);
43
+ // Will process directories from dir to root
44
+ let found = false;
45
+ let last = dir;
46
+ dir = _nodePath.default.resolve(dir);
47
+ do {
48
+ const files = await _promises.default.readdir(dir);
49
+ if (files.includes('package.json')) {
50
+ last = dir;
51
+ found = true;
52
+ }
53
+ if ([
54
+ 'package-lock.json',
55
+ 'yarn.lock'
56
+ ].some((lock)=>files.includes(lock))) {
57
+ logger.debug(`Project root found at ${_nodePath.default.relative(process.cwd(), dir) || '.'}`);
58
+ return dir;
59
+ }
60
+ dir = _nodePath.default.dirname(dir);
61
+ }while (dir !== _nodePath.default.dirname(dir))
62
+ if (found) {
63
+ logger.debug(`Project root found at ${_nodePath.default.relative(process.cwd(), last) || '.'}`);
64
+ } else {
65
+ logger.debug(`Project root not found, keeping ${_nodePath.default.relative(process.cwd(), last) || '.'}`);
66
+ }
67
+ return last;
68
+ }
69
+ // Methods
70
+ async _loadManifest(dir) {
71
+ const file = _nodePath.default.resolve(this.root, dir, 'package.json');
72
+ const log = this._logger.child({
73
+ label: _nodePath.default.relative(this.root, _nodePath.default.dirname(file)) || '.'
74
+ });
75
+ log.verbose('Loading package.json ...');
76
+ const data = await _promises.default.readFile(file, 'utf-8');
77
+ const mnf = JSON.parse(data);
78
+ (0, _normalizePackageData.default)(mnf, (msg)=>log.verbose(msg));
79
+ return mnf;
80
+ }
81
+ async _loadWorkspace(dir) {
82
+ return await this._lock.acquire('workspaces', async ()=>{
83
+ let wks = this._workspaces.get(dir);
84
+ if (!wks) {
85
+ const manifest = await this._loadManifest(dir);
86
+ wks = new _workspace.Workspace(dir, manifest, this);
87
+ this._workspaces.set(dir, wks);
88
+ this._names.set(wks.name, wks);
89
+ }
90
+ return wks;
91
+ });
92
+ }
93
+ async packageManager() {
94
+ if (!this._packageManager) {
95
+ const files = await _promises.default.readdir(this.root);
96
+ if (files.includes('yarn.lock')) {
97
+ this._logger.debug(`Detected yarn in ${_nodePath.default.relative(process.cwd(), this.root) || '.'}`);
98
+ this._packageManager = 'yarn';
99
+ } else if (files.includes('package-lock.json')) {
100
+ this._logger.debug(`Detected npm in ${_nodePath.default.relative(process.cwd(), this.root) || '.'}`);
101
+ this._packageManager = 'npm';
102
+ } else {
103
+ this._logger.debug(`No package manager recognized in ${_nodePath.default.relative(process.cwd(), this.root) || '.'}, defaults to npm`);
104
+ this._packageManager = 'npm';
105
+ }
106
+ }
107
+ return this._packageManager;
108
+ }
109
+ async mainWorkspace() {
110
+ if (!this._mainWorkspace) {
111
+ const manifest = await this._loadManifest('.');
112
+ this._mainWorkspace = new _workspace.Workspace('.', manifest, this);
113
+ this._names.set(this._mainWorkspace.name, this._mainWorkspace);
114
+ }
115
+ return this._mainWorkspace;
116
+ }
117
+ async currentWorkspace(cwd = process.cwd()) {
118
+ let workspace = null;
119
+ cwd = _nodePath.default.normalize(cwd);
120
+ for await (const wks of this.workspaces()){
121
+ if (cwd.startsWith(_nodePath.default.normalize(wks.cwd))) {
122
+ workspace = wks;
123
+ if (wks.cwd !== this.root) return wks;
124
+ }
125
+ }
126
+ return workspace;
127
+ }
128
+ async *workspaces() {
129
+ const main = await this.mainWorkspace();
130
+ yield main;
131
+ if (this._isFullyLoaded) {
132
+ for (const wks of this._names.values()){
133
+ if (wks.name !== main.name) yield wks;
134
+ }
135
+ } else {
136
+ // Load child workspaces
137
+ const { workspaces =[] } = main.manifest;
138
+ for (const pattern of workspaces){
139
+ for (const dir of (await (0, _tinyGlob.default)(pattern, {
140
+ cwd: this.root
141
+ }))){
142
+ try {
143
+ // Check if dir is a directory exists
144
+ const file = _nodePath.default.resolve(this.root, dir);
145
+ const stat = await _promises.default.stat(file);
146
+ if (stat.isDirectory()) {
147
+ yield await this._loadWorkspace(dir);
148
+ }
149
+ } catch (error) {
150
+ if (error.code === 'ENOENT') {
151
+ continue;
152
+ }
153
+ throw error;
154
+ }
155
+ }
156
+ }
157
+ this._isFullyLoaded = true;
158
+ }
159
+ }
160
+ async workspace(name) {
161
+ // With current directory
162
+ if (!name) {
163
+ const dir = _nodePath.default.relative(this.root, process.cwd());
164
+ return this._loadWorkspace(dir);
165
+ }
166
+ // Try name index
167
+ const wks = this._names.get(name);
168
+ if (wks) return wks;
169
+ if (this._isFullyLoaded) {
170
+ return null;
171
+ }
172
+ // Load workspaces
173
+ for await (const ws of this.workspaces()){
174
+ if (ws.name === name) {
175
+ return ws;
176
+ }
177
+ }
178
+ this._isFullyLoaded = true;
179
+ return null;
180
+ }
181
+ // Properties
182
+ get root() {
183
+ return _nodePath.default.resolve(this._root);
184
+ }
185
+ }
186
+ __decorate([
187
+ (0, _services.lazyInject)(_services.Logger)
188
+ ], Project.prototype, "_logger", void 0);
189
+
190
+ //# sourceMappingURL=project.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["project/project.js"],"sourcesContent":["import AsyncLock from 'async-lock';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\nimport normalize, { Package } from 'normalize-package-data';\nimport glob from 'tiny-glob';\n\nimport { container, lazyInject, Logger } from '../services';\nimport { Workspace } from './workspace';\n\n// Types\nexport type PackageManager = 'npm' | 'yarn';\nexport interface ProjectOptions {\n packageManager?: PackageManager | undefined;\n}\n\n// Class\nexport class Project {\n // Attributes\n private _mainWorkspace?: Workspace;\n private readonly _names = new Map<string, Workspace>();\n private readonly _workspaces = new Map<string, Workspace>();\n\n private _packageManager?: PackageManager;\n private _isFullyLoaded = false;\n private _lock = new AsyncLock();\n\n @lazyInject(Logger)\n private readonly _logger: Logger;\n\n // Constructor\n constructor(\n private readonly _root: string,\n opts: ProjectOptions = {}\n ) {\n if (opts.packageManager) {\n this._logger.debug(`Forced use of ${opts.packageManager} in ${path.relative(process.cwd(), this.root) || '.'}`);\n this._packageManager = opts.packageManager;\n }\n }\n\n // Statics\n static async searchProjectRoot(dir: string): Promise<string> {\n const logger = container.get(Logger);\n\n // Will process directories from dir to root\n let found = false;\n let last = dir;\n dir = path.resolve(dir);\n\n do {\n const files = await fs.readdir(dir);\n\n if (files.includes('package.json')) {\n last = dir;\n found = true;\n }\n\n if (['package-lock.json', 'yarn.lock'].some(lock => files.includes(lock))) {\n logger.debug(`Project root found at ${path.relative(process.cwd(), dir) || '.'}`);\n return dir;\n }\n\n dir = path.dirname(dir);\n } while (dir !== path.dirname(dir));\n\n if (found) {\n logger.debug(`Project root found at ${path.relative(process.cwd(), last) || '.'}`);\n } else {\n logger.debug(`Project root not found, keeping ${path.relative(process.cwd(), last) || '.'}`);\n }\n\n return last;\n }\n\n // Methods\n private async _loadManifest(dir: string): Promise<Package> {\n const file = path.resolve(this.root, dir, 'package.json');\n const log = this._logger.child({ label: path.relative(this.root, path.dirname(file)) || '.' });\n log.verbose('Loading package.json ...');\n\n const data = await fs.readFile(file, 'utf-8');\n const mnf = JSON.parse(data);\n normalize(mnf, (msg) => log.verbose(msg));\n\n return mnf;\n }\n\n private async _loadWorkspace(dir: string): Promise<Workspace> {\n return await this._lock.acquire('workspaces', async () => {\n let wks = this._workspaces.get(dir);\n\n if (!wks) {\n const manifest = await this._loadManifest(dir);\n wks = new Workspace(dir, manifest, this);\n\n this._workspaces.set(dir, wks);\n this._names.set(wks.name, wks);\n }\n\n return wks;\n });\n }\n\n async packageManager(): Promise<PackageManager> {\n if (!this._packageManager) {\n const files = await fs.readdir(this.root);\n\n if (files.includes('yarn.lock')) {\n this._logger.debug(`Detected yarn in ${path.relative(process.cwd(), this.root) || '.'}`);\n this._packageManager = 'yarn';\n } else if (files.includes('package-lock.json')) {\n this._logger.debug(`Detected npm in ${path.relative(process.cwd(), this.root) || '.'}`);\n this._packageManager = 'npm';\n } else {\n this._logger.debug(`No package manager recognized in ${path.relative(process.cwd(), this.root) || '.'}, defaults to npm`);\n this._packageManager = 'npm';\n }\n }\n\n return this._packageManager;\n }\n\n async mainWorkspace(): Promise<Workspace> {\n if (!this._mainWorkspace) {\n const manifest = await this._loadManifest('.');\n this._mainWorkspace = new Workspace('.', manifest, this);\n\n this._names.set(this._mainWorkspace.name, this._mainWorkspace);\n }\n\n return this._mainWorkspace;\n }\n\n async currentWorkspace(cwd = process.cwd()): Promise<Workspace | null> {\n let workspace: Workspace | null = null;\n cwd = path.normalize(cwd);\n\n for await (const wks of this.workspaces()) {\n if (cwd.startsWith(path.normalize(wks.cwd))) {\n workspace = wks;\n\n if (wks.cwd !== this.root) return wks;\n }\n }\n\n return workspace;\n }\n\n async* workspaces(): AsyncGenerator<Workspace, void> {\n const main = await this.mainWorkspace();\n yield main;\n\n if (this._isFullyLoaded) {\n for (const wks of this._names.values()) {\n if (wks.name !== main.name) yield wks;\n }\n } else {\n // Load child workspaces\n const { workspaces = [] } = main.manifest;\n\n for (const pattern of workspaces) {\n for (const dir of await glob(pattern, { cwd: this.root })) {\n try {\n // Check if dir is a directory exists\n const file = path.resolve(this.root, dir);\n const stat = await fs.stat(file);\n\n if (stat.isDirectory()) {\n yield await this._loadWorkspace(dir);\n }\n\n } catch (error) {\n if (error.code === 'ENOENT') {\n continue;\n }\n\n throw error;\n }\n }\n }\n\n this._isFullyLoaded = true;\n }\n }\n\n async workspace(name?: string): Promise<Workspace | null> {\n // With current directory\n if (!name) {\n const dir = path.relative(this.root, process.cwd());\n return this._loadWorkspace(dir);\n }\n\n // Try name index\n const wks = this._names.get(name);\n if (wks) return wks;\n\n if (this._isFullyLoaded) {\n return null;\n }\n\n // Load workspaces\n for await (const ws of this.workspaces()) {\n if (ws.name === name) {\n return ws;\n }\n }\n\n this._isFullyLoaded = true;\n return null;\n }\n\n // Properties\n get root(): string {\n return path.resolve(this._root);\n }\n}\n"],"names":["Project","constructor","_root","opts","_names","Map","_workspaces","_isFullyLoaded","_lock","AsyncLock","packageManager","_logger","debug","path","relative","process","cwd","root","_packageManager","searchProjectRoot","dir","logger","container","get","Logger","found","last","resolve","files","fs","readdir","includes","some","lock","dirname","_loadManifest","file","log","child","label","verbose","data","readFile","mnf","JSON","parse","normalize","msg","_loadWorkspace","acquire","wks","manifest","Workspace","set","name","mainWorkspace","_mainWorkspace","currentWorkspace","workspace","workspaces","startsWith","main","values","pattern","glob","stat","isDirectory","error","code","ws","lazyInject"],"mappings":"AAAA;;;;+BAgBaA;;aAAAA;;gEAhBS;+DACP;+DACE;2EACkB;+DAClB;0BAE6B;2BACpB;;;;;;;;;;;;AASnB,MAAMA;IAaX,cAAc;IACdC,YACmBC,OACjBC,OAAuB,CAAC,CAAC,CACzB;qBAFiBD;aAZFE,SAAS,IAAIC;aACbC,cAAc,IAAID;aAG3BE,iBAAiB,KAAK;aACtBC,QAAQ,IAAIC,kBAAS;QAU3B,IAAIN,KAAKO,cAAc,EAAE;YACvB,IAAI,CAACC,OAAO,CAACC,KAAK,CAAC,CAAC,cAAc,EAAET,KAAKO,cAAc,CAAC,IAAI,EAAEG,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAI,IAAI,CAACC,IAAI,KAAK,IAAI,CAAC;YAC9G,IAAI,CAACC,eAAe,GAAGf,KAAKO,cAAc;QAC5C,CAAC;IACH;IAEA,UAAU;IACV,aAAaS,kBAAkBC,GAAW,EAAmB;QAC3D,MAAMC,SAASC,mBAAS,CAACC,GAAG,CAACC,gBAAM;QAEnC,4CAA4C;QAC5C,IAAIC,QAAQ,KAAK;QACjB,IAAIC,OAAON;QACXA,MAAMP,iBAAI,CAACc,OAAO,CAACP;QAEnB,GAAG;YACD,MAAMQ,QAAQ,MAAMC,iBAAE,CAACC,OAAO,CAACV;YAE/B,IAAIQ,MAAMG,QAAQ,CAAC,iBAAiB;gBAClCL,OAAON;gBACPK,QAAQ,IAAI;YACd,CAAC;YAED,IAAI;gBAAC;gBAAqB;aAAY,CAACO,IAAI,CAACC,CAAAA,OAAQL,MAAMG,QAAQ,CAACE,QAAQ;gBACzEZ,OAAOT,KAAK,CAAC,CAAC,sBAAsB,EAAEC,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAII,QAAQ,IAAI,CAAC;gBAChF,OAAOA;YACT,CAAC;YAEDA,MAAMP,iBAAI,CAACqB,OAAO,CAACd;QACrB,QAASA,QAAQP,iBAAI,CAACqB,OAAO,CAACd,KAAM;QAEpC,IAAIK,OAAO;YACTJ,OAAOT,KAAK,CAAC,CAAC,sBAAsB,EAAEC,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAIU,SAAS,IAAI,CAAC;QACnF,OAAO;YACLL,OAAOT,KAAK,CAAC,CAAC,gCAAgC,EAAEC,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAIU,SAAS,IAAI,CAAC;QAC7F,CAAC;QAED,OAAOA;IACT;IAEA,UAAU;IACV,MAAcS,cAAcf,GAAW,EAAoB;QACzD,MAAMgB,OAAOvB,iBAAI,CAACc,OAAO,CAAC,IAAI,CAACV,IAAI,EAAEG,KAAK;QAC1C,MAAMiB,MAAM,IAAI,CAAC1B,OAAO,CAAC2B,KAAK,CAAC;YAAEC,OAAO1B,iBAAI,CAACC,QAAQ,CAAC,IAAI,CAACG,IAAI,EAAEJ,iBAAI,CAACqB,OAAO,CAACE,UAAU;QAAI;QAC5FC,IAAIG,OAAO,CAAC;QAEZ,MAAMC,OAAO,MAAMZ,iBAAE,CAACa,QAAQ,CAACN,MAAM;QACrC,MAAMO,MAAMC,KAAKC,KAAK,CAACJ;QACvBK,IAAAA,6BAAS,EAACH,KAAK,CAACI,MAAQV,IAAIG,OAAO,CAACO;QAEpC,OAAOJ;IACT;IAEA,MAAcK,eAAe5B,GAAW,EAAsB;QAC5D,OAAO,MAAM,IAAI,CAACZ,KAAK,CAACyC,OAAO,CAAC,cAAc,UAAY;YACxD,IAAIC,MAAM,IAAI,CAAC5C,WAAW,CAACiB,GAAG,CAACH;YAE/B,IAAI,CAAC8B,KAAK;gBACR,MAAMC,WAAW,MAAM,IAAI,CAAChB,aAAa,CAACf;gBAC1C8B,MAAM,IAAIE,oBAAS,CAAChC,KAAK+B,UAAU,IAAI;gBAEvC,IAAI,CAAC7C,WAAW,CAAC+C,GAAG,CAACjC,KAAK8B;gBAC1B,IAAI,CAAC9C,MAAM,CAACiD,GAAG,CAACH,IAAII,IAAI,EAAEJ;YAC5B,CAAC;YAED,OAAOA;QACT;IACF;IAEA,MAAMxC,iBAA0C;QAC9C,IAAI,CAAC,IAAI,CAACQ,eAAe,EAAE;YACzB,MAAMU,QAAQ,MAAMC,iBAAE,CAACC,OAAO,CAAC,IAAI,CAACb,IAAI;YAExC,IAAIW,MAAMG,QAAQ,CAAC,cAAc;gBAC/B,IAAI,CAACpB,OAAO,CAACC,KAAK,CAAC,CAAC,iBAAiB,EAAEC,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAI,IAAI,CAACC,IAAI,KAAK,IAAI,CAAC;gBACvF,IAAI,CAACC,eAAe,GAAG;YACzB,OAAO,IAAIU,MAAMG,QAAQ,CAAC,sBAAsB;gBAC9C,IAAI,CAACpB,OAAO,CAACC,KAAK,CAAC,CAAC,gBAAgB,EAAEC,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAI,IAAI,CAACC,IAAI,KAAK,IAAI,CAAC;gBACtF,IAAI,CAACC,eAAe,GAAG;YACzB,OAAO;gBACL,IAAI,CAACP,OAAO,CAACC,KAAK,CAAC,CAAC,iCAAiC,EAAEC,iBAAI,CAACC,QAAQ,CAACC,QAAQC,GAAG,IAAI,IAAI,CAACC,IAAI,KAAK,IAAI,iBAAiB,CAAC;gBACxH,IAAI,CAACC,eAAe,GAAG;YACzB,CAAC;QACH,CAAC;QAED,OAAO,IAAI,CAACA,eAAe;IAC7B;IAEA,MAAMqC,gBAAoC;QACxC,IAAI,CAAC,IAAI,CAACC,cAAc,EAAE;YACxB,MAAML,WAAW,MAAM,IAAI,CAAChB,aAAa,CAAC;YAC1C,IAAI,CAACqB,cAAc,GAAG,IAAIJ,oBAAS,CAAC,KAAKD,UAAU,IAAI;YAEvD,IAAI,CAAC/C,MAAM,CAACiD,GAAG,CAAC,IAAI,CAACG,cAAc,CAACF,IAAI,EAAE,IAAI,CAACE,cAAc;QAC/D,CAAC;QAED,OAAO,IAAI,CAACA,cAAc;IAC5B;IAEA,MAAMC,iBAAiBzC,MAAMD,QAAQC,GAAG,EAAE,EAA6B;QACrE,IAAI0C,YAA8B,IAAI;QACtC1C,MAAMH,iBAAI,CAACiC,SAAS,CAAC9B;QAErB,WAAW,MAAMkC,OAAO,IAAI,CAACS,UAAU,GAAI;YACzC,IAAI3C,IAAI4C,UAAU,CAAC/C,iBAAI,CAACiC,SAAS,CAACI,IAAIlC,GAAG,IAAI;gBAC3C0C,YAAYR;gBAEZ,IAAIA,IAAIlC,GAAG,KAAK,IAAI,CAACC,IAAI,EAAE,OAAOiC;YACpC,CAAC;QACH;QAEA,OAAOQ;IACT;IAEA,OAAOC,aAA8C;QACnD,MAAME,OAAO,MAAM,IAAI,CAACN,aAAa;QACrC,MAAMM;QAEN,IAAI,IAAI,CAACtD,cAAc,EAAE;YACvB,KAAK,MAAM2C,OAAO,IAAI,CAAC9C,MAAM,CAAC0D,MAAM,GAAI;gBACtC,IAAIZ,IAAII,IAAI,KAAKO,KAAKP,IAAI,EAAE,MAAMJ;YACpC;QACF,OAAO;YACL,wBAAwB;YACxB,MAAM,EAAES,YAAa,EAAE,CAAA,EAAE,GAAGE,KAAKV,QAAQ;YAEzC,KAAK,MAAMY,WAAWJ,WAAY;gBAChC,KAAK,MAAMvC,OAAO,CAAA,MAAM4C,IAAAA,iBAAI,EAACD,SAAS;oBAAE/C,KAAK,IAAI,CAACC,IAAI;gBAAC,EAAC,EAAG;oBACzD,IAAI;wBACF,qCAAqC;wBACrC,MAAMmB,OAAOvB,iBAAI,CAACc,OAAO,CAAC,IAAI,CAACV,IAAI,EAAEG;wBACrC,MAAM6C,OAAO,MAAMpC,iBAAE,CAACoC,IAAI,CAAC7B;wBAE3B,IAAI6B,KAAKC,WAAW,IAAI;4BACtB,MAAM,MAAM,IAAI,CAAClB,cAAc,CAAC5B;wBAClC,CAAC;oBAEH,EAAE,OAAO+C,OAAO;wBACd,IAAIA,MAAMC,IAAI,KAAK,UAAU;4BAC3B,QAAS;wBACX,CAAC;wBAED,MAAMD,MAAM;oBACd;gBACF;YACF;YAEA,IAAI,CAAC5D,cAAc,GAAG,IAAI;QAC5B,CAAC;IACH;IAEA,MAAMmD,UAAUJ,IAAa,EAA6B;QACxD,yBAAyB;QACzB,IAAI,CAACA,MAAM;YACT,MAAMlC,MAAMP,iBAAI,CAACC,QAAQ,CAAC,IAAI,CAACG,IAAI,EAAEF,QAAQC,GAAG;YAChD,OAAO,IAAI,CAACgC,cAAc,CAAC5B;QAC7B,CAAC;QAED,iBAAiB;QACjB,MAAM8B,MAAM,IAAI,CAAC9C,MAAM,CAACmB,GAAG,CAAC+B;QAC5B,IAAIJ,KAAK,OAAOA;QAEhB,IAAI,IAAI,CAAC3C,cAAc,EAAE;YACvB,OAAO,IAAI;QACb,CAAC;QAED,kBAAkB;QAClB,WAAW,MAAM8D,MAAM,IAAI,CAACV,UAAU,GAAI;YACxC,IAAIU,GAAGf,IAAI,KAAKA,MAAM;gBACpB,OAAOe;YACT,CAAC;QACH;QAEA,IAAI,CAAC9D,cAAc,GAAG,IAAI;QAC1B,OAAO,IAAI;IACb;IAEA,aAAa;IACb,IAAIU,OAAe;QACjB,OAAOJ,iBAAI,CAACc,OAAO,CAAC,IAAI,CAACzB,KAAK;IAChC;AACF;;IA7LGoE,IAAAA,oBAAU,EAAC9C,gBAAM;GAVPxB","file":"project.js"}
@@ -0,0 +1,32 @@
1
+ import { SpawnTask, SpawnTaskOptions, TaskContext } from '@jujulego/tasks';
2
+ import { Package } from 'normalize-package-data';
3
+ import { Project } from './project';
4
+ export declare type WorkspaceDepsMode = 'all' | 'prod' | 'none';
5
+ export interface WorkspaceContext extends TaskContext {
6
+ workspace: Workspace;
7
+ }
8
+ export interface WorkspaceRunOptions extends Omit<SpawnTaskOptions, 'cwd'> {
9
+ buildDeps?: WorkspaceDepsMode;
10
+ }
11
+ export declare class Workspace {
12
+ private readonly _cwd;
13
+ readonly manifest: Package;
14
+ readonly project: Project;
15
+ private readonly _logger;
16
+ private readonly _affectedCache;
17
+ private readonly _tasks;
18
+ constructor(_cwd: string, manifest: Package, project: Project);
19
+ private _satisfies;
20
+ private _buildDependencies;
21
+ private _isAffected;
22
+ isAffected(reference: string): Promise<boolean>;
23
+ private _loadDependencies;
24
+ dependencies(): AsyncGenerator<Workspace, void>;
25
+ devDependencies(): AsyncGenerator<Workspace, void>;
26
+ run(script: string, args?: string[], opts?: WorkspaceRunOptions): Promise<SpawnTask<WorkspaceContext>>;
27
+ build(opts?: WorkspaceRunOptions): Promise<SpawnTask | null>;
28
+ get name(): string;
29
+ get version(): string;
30
+ get reference(): string;
31
+ get cwd(): string;
32
+ }
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["project/workspace.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,gBAAgB,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAE3E,OAAO,EAAE,OAAO,EAAE,MAAM,wBAAwB,CAAC;AAOjD,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAGpC,oBAAY,iBAAiB,GAAG,KAAK,GAAG,MAAM,GAAG,MAAM,CAAC;AAExD,MAAM,WAAW,gBAAiB,SAAQ,WAAW;IACnD,SAAS,EAAE,SAAS,CAAC;CACtB;AAED,MAAM,WAAW,mBAAoB,SAAQ,IAAI,CAAC,gBAAgB,EAAE,KAAK,CAAC;IACxE,SAAS,CAAC,EAAE,iBAAiB,CAAC;CAC/B;AAGD,qBAAa,SAAS;IAQlB,OAAO,CAAC,QAAQ,CAAC,IAAI;IACrB,QAAQ,CAAC,QAAQ,EAAE,OAAO;IAC1B,QAAQ,CAAC,OAAO,EAAE,OAAO;IAR3B,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAiB;IACzC,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAuC;IACtE,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAkD;gBAItD,IAAI,EAAE,MAAM,EACpB,QAAQ,EAAE,OAAO,EACjB,OAAO,EAAE,OAAO;IAO3B,OAAO,CAAC,UAAU;YAYJ,kBAAkB;YAuBlB,WAAW;IAqBnB,UAAU,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;YAWtC,iBAAiB;IAczB,YAAY,IAAI,cAAc,CAAC,SAAS,EAAE,IAAI,CAAC;IAQ/C,eAAe,IAAI,cAAc,CAAC,SAAS,EAAE,IAAI,CAAC;IAQnD,GAAG,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,GAAE,MAAM,EAAO,EAAE,IAAI,GAAE,mBAAwB,GAAG,OAAO,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC;IAuB9G,KAAK,CAAC,IAAI,CAAC,EAAE,mBAAmB,GAAG,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC;IAYlE,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,IAAI,OAAO,IAAI,MAAM,CAEpB;IAED,IAAI,SAAS,IAAI,MAAM,CAEtB;IAED,IAAI,GAAG,IAAI,MAAM,CAEhB;CACF","file":"workspace.d.ts","sourcesContent":["import { SpawnTask, SpawnTaskOptions, TaskContext } from '@jujulego/tasks';\nimport path from 'node:path';\nimport { Package } from 'normalize-package-data';\nimport { satisfies } from 'semver';\nimport winston from 'winston';\n\nimport { Git } from '../git';\nimport { container, Logger } from '../services';\nimport { combine } from '../utils';\nimport { Project } from './project';\n\n// Types\nexport type WorkspaceDepsMode = 'all' | 'prod' | 'none';\n\nexport interface WorkspaceContext extends TaskContext {\n workspace: Workspace;\n}\n\nexport interface WorkspaceRunOptions extends Omit<SpawnTaskOptions, 'cwd'> {\n buildDeps?: WorkspaceDepsMode;\n}\n\n// Class\nexport class Workspace {\n // Attributes\n private readonly _logger: winston.Logger;\n private readonly _affectedCache = new Map<string, Promise<boolean>>();\n private readonly _tasks = new Map<string, SpawnTask<WorkspaceContext>>();\n\n // Constructor\n constructor(\n private readonly _cwd: string,\n readonly manifest: Package,\n readonly project: Project\n ) {\n const logger = container.get(Logger);\n this._logger = logger.child({ label: this.manifest.name });\n }\n\n // Methods\n private _satisfies(from: Workspace, range: string): boolean {\n if (range.startsWith('file:')) {\n return path.resolve(from.cwd, range.substring(5)) === this.cwd;\n }\n\n if (range.startsWith('workspace:')) {\n range = range.substring(10);\n }\n\n return !this.version || satisfies(this.version, range);\n }\n\n private async _buildDependencies(task: SpawnTask, deps: WorkspaceDepsMode = 'all') {\n // Generators\n const generators: AsyncGenerator<Workspace, void>[] = [];\n\n switch (deps) {\n case 'all':\n generators.unshift(this.devDependencies());\n\n // eslint-disable-next no-fallthrough\n case 'prod':\n generators.unshift(this.dependencies());\n }\n\n // Build deps\n for await (const dep of combine(...generators)) {\n const build = await dep.build();\n\n if (build) {\n task.dependsOn(build);\n }\n }\n }\n\n private async _isAffected(reference: string): Promise<boolean> {\n const isAffected = await Git.isAffected(reference, ['--', this.cwd], {\n cwd: this.project.root,\n logger: this._logger,\n });\n\n if (isAffected) {\n return true;\n }\n\n // Test dependencies\n const proms: Promise<boolean>[] = [];\n\n for await (const dep of combine(this.dependencies(), this.devDependencies())) {\n proms.push(dep.isAffected(reference));\n }\n\n const results = await Promise.all(proms);\n return results.some(r => r);\n }\n\n async isAffected(reference: string): Promise<boolean> {\n let isAffected = this._affectedCache.get(reference);\n\n if (!isAffected) {\n isAffected = this._isAffected(reference);\n this._affectedCache.set(reference, isAffected);\n }\n\n return await isAffected;\n }\n\n private async* _loadDependencies(dependencies: Record<string, string>, kind: string): AsyncGenerator<Workspace, void> {\n for (const [dep, range] of Object.entries(dependencies)) {\n const ws = await this.project.workspace(dep);\n\n if (ws) {\n if (ws._satisfies(this, range)) {\n yield ws;\n } else {\n this._logger.verbose(`Ignoring ${kind} ${ws.reference} as it does not match requirement ${range}`);\n }\n }\n }\n }\n\n async* dependencies(): AsyncGenerator<Workspace, void> {\n if (!this.manifest.dependencies) return;\n\n for await (const ws of this._loadDependencies(this.manifest.dependencies, 'dependency')) {\n yield ws;\n }\n }\n\n async* devDependencies(): AsyncGenerator<Workspace, void> {\n if (!this.manifest.devDependencies) return;\n\n for await (const ws of this._loadDependencies(this.manifest.devDependencies, 'devDependency')) {\n yield ws;\n }\n }\n\n async run(script: string, args: string[] = [], opts: WorkspaceRunOptions = {}): Promise<SpawnTask<WorkspaceContext>> {\n let task = this._tasks.get(script);\n\n if (!task) {\n const pm = await this.project.packageManager();\n\n task = new SpawnTask(pm, ['run', script, ...args], { workspace: this }, {\n ...opts,\n cwd: this.cwd,\n logger: this._logger,\n env: {\n FORCE_COLOR: '1',\n ...opts.env\n }\n });\n await this._buildDependencies(task, opts.buildDeps);\n\n this._tasks.set(script, task);\n }\n\n return task;\n }\n\n async build(opts?: WorkspaceRunOptions): Promise<SpawnTask | null> {\n const { scripts = {} } = this.manifest;\n\n if (!scripts.build) {\n this._logger.warn('Will not be built (no build script)');\n return null;\n }\n\n return await this.run('build', [], opts);\n }\n\n // Properties\n get name(): string {\n return this.manifest.name;\n }\n\n get version(): string {\n return this.manifest.version;\n }\n\n get reference(): string {\n return this.version ? `${this.name}@${this.version}` : this.name;\n }\n\n get cwd(): string {\n return path.resolve(this.project.root, this._cwd);\n }\n}\n"]}
@@ -0,0 +1,159 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "Workspace", {
6
+ enumerable: true,
7
+ get: ()=>Workspace
8
+ });
9
+ const _tasks = require("@jujulego/tasks");
10
+ const _nodePath = /*#__PURE__*/ _interopRequireDefault(require("node:path"));
11
+ const _semver = require("semver");
12
+ const _git = require("../git");
13
+ const _services = require("../services");
14
+ const _utils = require("../utils");
15
+ function _interopRequireDefault(obj) {
16
+ return obj && obj.__esModule ? obj : {
17
+ default: obj
18
+ };
19
+ }
20
+ class Workspace {
21
+ // Constructor
22
+ constructor(_cwd, manifest, project){
23
+ this._cwd = _cwd;
24
+ this.manifest = manifest;
25
+ this.project = project;
26
+ this._affectedCache = new Map();
27
+ this._tasks = new Map();
28
+ const logger = _services.container.get(_services.Logger);
29
+ this._logger = logger.child({
30
+ label: this.manifest.name
31
+ });
32
+ }
33
+ // Methods
34
+ _satisfies(from, range) {
35
+ if (range.startsWith('file:')) {
36
+ return _nodePath.default.resolve(from.cwd, range.substring(5)) === this.cwd;
37
+ }
38
+ if (range.startsWith('workspace:')) {
39
+ range = range.substring(10);
40
+ }
41
+ return !this.version || (0, _semver.satisfies)(this.version, range);
42
+ }
43
+ async _buildDependencies(task, deps = 'all') {
44
+ // Generators
45
+ const generators = [];
46
+ switch(deps){
47
+ case 'all':
48
+ generators.unshift(this.devDependencies());
49
+ // eslint-disable-next no-fallthrough
50
+ case 'prod':
51
+ generators.unshift(this.dependencies());
52
+ }
53
+ // Build deps
54
+ for await (const dep of (0, _utils.combine)(...generators)){
55
+ const build = await dep.build();
56
+ if (build) {
57
+ task.dependsOn(build);
58
+ }
59
+ }
60
+ }
61
+ async _isAffected(reference) {
62
+ const isAffected = await _git.Git.isAffected(reference, [
63
+ '--',
64
+ this.cwd
65
+ ], {
66
+ cwd: this.project.root,
67
+ logger: this._logger
68
+ });
69
+ if (isAffected) {
70
+ return true;
71
+ }
72
+ // Test dependencies
73
+ const proms = [];
74
+ for await (const dep of (0, _utils.combine)(this.dependencies(), this.devDependencies())){
75
+ proms.push(dep.isAffected(reference));
76
+ }
77
+ const results = await Promise.all(proms);
78
+ return results.some((r)=>r);
79
+ }
80
+ async isAffected(reference) {
81
+ let isAffected = this._affectedCache.get(reference);
82
+ if (!isAffected) {
83
+ isAffected = this._isAffected(reference);
84
+ this._affectedCache.set(reference, isAffected);
85
+ }
86
+ return await isAffected;
87
+ }
88
+ async *_loadDependencies(dependencies, kind) {
89
+ for (const [dep, range] of Object.entries(dependencies)){
90
+ const ws = await this.project.workspace(dep);
91
+ if (ws) {
92
+ if (ws._satisfies(this, range)) {
93
+ yield ws;
94
+ } else {
95
+ this._logger.verbose(`Ignoring ${kind} ${ws.reference} as it does not match requirement ${range}`);
96
+ }
97
+ }
98
+ }
99
+ }
100
+ async *dependencies() {
101
+ if (!this.manifest.dependencies) return;
102
+ for await (const ws of this._loadDependencies(this.manifest.dependencies, 'dependency')){
103
+ yield ws;
104
+ }
105
+ }
106
+ async *devDependencies() {
107
+ if (!this.manifest.devDependencies) return;
108
+ for await (const ws of this._loadDependencies(this.manifest.devDependencies, 'devDependency')){
109
+ yield ws;
110
+ }
111
+ }
112
+ async run(script, args = [], opts = {}) {
113
+ let task = this._tasks.get(script);
114
+ if (!task) {
115
+ const pm = await this.project.packageManager();
116
+ task = new _tasks.SpawnTask(pm, [
117
+ 'run',
118
+ script,
119
+ ...args
120
+ ], {
121
+ workspace: this
122
+ }, {
123
+ ...opts,
124
+ cwd: this.cwd,
125
+ logger: this._logger,
126
+ env: {
127
+ FORCE_COLOR: '1',
128
+ ...opts.env
129
+ }
130
+ });
131
+ await this._buildDependencies(task, opts.buildDeps);
132
+ this._tasks.set(script, task);
133
+ }
134
+ return task;
135
+ }
136
+ async build(opts) {
137
+ const { scripts ={} } = this.manifest;
138
+ if (!scripts.build) {
139
+ this._logger.warn('Will not be built (no build script)');
140
+ return null;
141
+ }
142
+ return await this.run('build', [], opts);
143
+ }
144
+ // Properties
145
+ get name() {
146
+ return this.manifest.name;
147
+ }
148
+ get version() {
149
+ return this.manifest.version;
150
+ }
151
+ get reference() {
152
+ return this.version ? `${this.name}@${this.version}` : this.name;
153
+ }
154
+ get cwd() {
155
+ return _nodePath.default.resolve(this.project.root, this._cwd);
156
+ }
157
+ }
158
+
159
+ //# sourceMappingURL=workspace.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["project/workspace.js"],"sourcesContent":["import { SpawnTask, SpawnTaskOptions, TaskContext } from '@jujulego/tasks';\nimport path from 'node:path';\nimport { Package } from 'normalize-package-data';\nimport { satisfies } from 'semver';\nimport winston from 'winston';\n\nimport { Git } from '../git';\nimport { container, Logger } from '../services';\nimport { combine } from '../utils';\nimport { Project } from './project';\n\n// Types\nexport type WorkspaceDepsMode = 'all' | 'prod' | 'none';\n\nexport interface WorkspaceContext extends TaskContext {\n workspace: Workspace;\n}\n\nexport interface WorkspaceRunOptions extends Omit<SpawnTaskOptions, 'cwd'> {\n buildDeps?: WorkspaceDepsMode;\n}\n\n// Class\nexport class Workspace {\n // Attributes\n private readonly _logger: winston.Logger;\n private readonly _affectedCache = new Map<string, Promise<boolean>>();\n private readonly _tasks = new Map<string, SpawnTask<WorkspaceContext>>();\n\n // Constructor\n constructor(\n private readonly _cwd: string,\n readonly manifest: Package,\n readonly project: Project\n ) {\n const logger = container.get(Logger);\n this._logger = logger.child({ label: this.manifest.name });\n }\n\n // Methods\n private _satisfies(from: Workspace, range: string): boolean {\n if (range.startsWith('file:')) {\n return path.resolve(from.cwd, range.substring(5)) === this.cwd;\n }\n\n if (range.startsWith('workspace:')) {\n range = range.substring(10);\n }\n\n return !this.version || satisfies(this.version, range);\n }\n\n private async _buildDependencies(task: SpawnTask, deps: WorkspaceDepsMode = 'all') {\n // Generators\n const generators: AsyncGenerator<Workspace, void>[] = [];\n\n switch (deps) {\n case 'all':\n generators.unshift(this.devDependencies());\n\n // eslint-disable-next no-fallthrough\n case 'prod':\n generators.unshift(this.dependencies());\n }\n\n // Build deps\n for await (const dep of combine(...generators)) {\n const build = await dep.build();\n\n if (build) {\n task.dependsOn(build);\n }\n }\n }\n\n private async _isAffected(reference: string): Promise<boolean> {\n const isAffected = await Git.isAffected(reference, ['--', this.cwd], {\n cwd: this.project.root,\n logger: this._logger,\n });\n\n if (isAffected) {\n return true;\n }\n\n // Test dependencies\n const proms: Promise<boolean>[] = [];\n\n for await (const dep of combine(this.dependencies(), this.devDependencies())) {\n proms.push(dep.isAffected(reference));\n }\n\n const results = await Promise.all(proms);\n return results.some(r => r);\n }\n\n async isAffected(reference: string): Promise<boolean> {\n let isAffected = this._affectedCache.get(reference);\n\n if (!isAffected) {\n isAffected = this._isAffected(reference);\n this._affectedCache.set(reference, isAffected);\n }\n\n return await isAffected;\n }\n\n private async* _loadDependencies(dependencies: Record<string, string>, kind: string): AsyncGenerator<Workspace, void> {\n for (const [dep, range] of Object.entries(dependencies)) {\n const ws = await this.project.workspace(dep);\n\n if (ws) {\n if (ws._satisfies(this, range)) {\n yield ws;\n } else {\n this._logger.verbose(`Ignoring ${kind} ${ws.reference} as it does not match requirement ${range}`);\n }\n }\n }\n }\n\n async* dependencies(): AsyncGenerator<Workspace, void> {\n if (!this.manifest.dependencies) return;\n\n for await (const ws of this._loadDependencies(this.manifest.dependencies, 'dependency')) {\n yield ws;\n }\n }\n\n async* devDependencies(): AsyncGenerator<Workspace, void> {\n if (!this.manifest.devDependencies) return;\n\n for await (const ws of this._loadDependencies(this.manifest.devDependencies, 'devDependency')) {\n yield ws;\n }\n }\n\n async run(script: string, args: string[] = [], opts: WorkspaceRunOptions = {}): Promise<SpawnTask<WorkspaceContext>> {\n let task = this._tasks.get(script);\n\n if (!task) {\n const pm = await this.project.packageManager();\n\n task = new SpawnTask(pm, ['run', script, ...args], { workspace: this }, {\n ...opts,\n cwd: this.cwd,\n logger: this._logger,\n env: {\n FORCE_COLOR: '1',\n ...opts.env\n }\n });\n await this._buildDependencies(task, opts.buildDeps);\n\n this._tasks.set(script, task);\n }\n\n return task;\n }\n\n async build(opts?: WorkspaceRunOptions): Promise<SpawnTask | null> {\n const { scripts = {} } = this.manifest;\n\n if (!scripts.build) {\n this._logger.warn('Will not be built (no build script)');\n return null;\n }\n\n return await this.run('build', [], opts);\n }\n\n // Properties\n get name(): string {\n return this.manifest.name;\n }\n\n get version(): string {\n return this.manifest.version;\n }\n\n get reference(): string {\n return this.version ? `${this.name}@${this.version}` : this.name;\n }\n\n get cwd(): string {\n return path.resolve(this.project.root, this._cwd);\n }\n}\n"],"names":["Workspace","constructor","_cwd","manifest","project","_affectedCache","Map","_tasks","logger","container","get","Logger","_logger","child","label","name","_satisfies","from","range","startsWith","path","resolve","cwd","substring","version","satisfies","_buildDependencies","task","deps","generators","unshift","devDependencies","dependencies","dep","combine","build","dependsOn","_isAffected","reference","isAffected","Git","root","proms","push","results","Promise","all","some","r","set","_loadDependencies","kind","Object","entries","ws","workspace","verbose","run","script","args","opts","pm","packageManager","SpawnTask","env","FORCE_COLOR","buildDeps","scripts","warn"],"mappings":"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","file":"workspace.js"}
@@ -0,0 +1,4 @@
1
+ export * from './inversify.config';
2
+ export * from './logger.service';
3
+ export * from './spinner.service';
4
+ export * from './task-manager.service';
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/index.ts"],"names":[],"mappings":"AAAA,cAAc,oBAAoB,CAAC;AACnC,cAAc,kBAAkB,CAAC;AACjC,cAAc,mBAAmB,CAAC;AAClC,cAAc,wBAAwB,CAAC","file":"index.d.ts","sourcesContent":["export * from './inversify.config';\nexport * from './logger.service';\nexport * from './spinner.service';\nexport * from './task-manager.service';\n"]}
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ _exportStar(require("./inversify.config"), exports);
6
+ _exportStar(require("./logger.service"), exports);
7
+ _exportStar(require("./spinner.service"), exports);
8
+ _exportStar(require("./task-manager.service"), exports);
9
+ function _exportStar(from, to) {
10
+ Object.keys(from).forEach(function(k) {
11
+ if (k !== "default" && !Object.prototype.hasOwnProperty.call(to, k)) Object.defineProperty(to, k, {
12
+ enumerable: true,
13
+ get: function() {
14
+ return from[k];
15
+ }
16
+ });
17
+ });
18
+ return from;
19
+ }
20
+
21
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/index.js"],"sourcesContent":["export * from './inversify.config';\nexport * from './logger.service';\nexport * from './spinner.service';\nexport * from './task-manager.service';\n"],"names":[],"mappings":"AAAA;;;;oBAAc;oBACA;oBACA;oBACA","file":"index.js"}
@@ -0,0 +1,11 @@
1
+ import { Container } from 'inversify';
2
+ import 'reflect-metadata';
3
+ export declare const INK_APP: unique symbol;
4
+ export declare const GLOBAL_CONFIG: unique symbol;
5
+ export declare const CURRENT_PROJECT: unique symbol;
6
+ export interface GlobalConfig {
7
+ jobs?: number;
8
+ verbose: number;
9
+ }
10
+ export declare const container: Container;
11
+ export declare const lazyInject: (serviceIdentifier: string | symbol | import("inversify/lib/interfaces/interfaces").interfaces.Newable<any> | import("inversify/lib/interfaces/interfaces").interfaces.Abstract<any>) => (proto: any, key: string) => void;
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/inversify.config.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAGtC,OAAO,kBAAkB,CAAC;AAG1B,eAAO,MAAM,OAAO,eAAuB,CAAC;AAC5C,eAAO,MAAM,aAAa,eAA6B,CAAC;AACxD,eAAO,MAAM,eAAe,eAA+B,CAAC;AAG5D,MAAM,WAAW,YAAY;IAC3B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;CACjB;AAGD,eAAO,MAAM,SAAS,WAEpB,CAAC;AAGH,eAAO,MAAQ,UAAU,4NAA6B,CAAC","file":"inversify.config.d.ts","sourcesContent":["import { Container } from 'inversify';\nimport getDecorators from 'inversify-inject-decorators';\n\nimport 'reflect-metadata';\n\n// Constants\nexport const INK_APP = Symbol.for('InkApp');\nexport const GLOBAL_CONFIG = Symbol.for('GlobalConfig');\nexport const CURRENT_PROJECT = Symbol.for('CurrentProject');\n\n// Types\nexport interface GlobalConfig {\n jobs?: number;\n verbose: number;\n}\n\n// Container\nexport const container = new Container({\n skipBaseClassChecks: true,\n});\n\n// Utilities\nexport const { lazyInject } = getDecorators(container);\n"]}
@@ -0,0 +1,34 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ function _export(target, all) {
6
+ for(var name in all)Object.defineProperty(target, name, {
7
+ enumerable: true,
8
+ get: all[name]
9
+ });
10
+ }
11
+ _export(exports, {
12
+ INK_APP: ()=>INK_APP,
13
+ GLOBAL_CONFIG: ()=>GLOBAL_CONFIG,
14
+ CURRENT_PROJECT: ()=>CURRENT_PROJECT,
15
+ container: ()=>container,
16
+ lazyInject: ()=>lazyInject
17
+ });
18
+ const _inversify = require("inversify");
19
+ const _inversifyInjectDecorators = /*#__PURE__*/ _interopRequireDefault(require("inversify-inject-decorators"));
20
+ require("reflect-metadata");
21
+ function _interopRequireDefault(obj) {
22
+ return obj && obj.__esModule ? obj : {
23
+ default: obj
24
+ };
25
+ }
26
+ const INK_APP = Symbol.for('InkApp');
27
+ const GLOBAL_CONFIG = Symbol.for('GlobalConfig');
28
+ const CURRENT_PROJECT = Symbol.for('CurrentProject');
29
+ const container = new _inversify.Container({
30
+ skipBaseClassChecks: true
31
+ });
32
+ const { lazyInject } = (0, _inversifyInjectDecorators.default)(container);
33
+
34
+ //# sourceMappingURL=inversify.config.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/inversify.config.js"],"sourcesContent":["import { Container } from 'inversify';\nimport getDecorators from 'inversify-inject-decorators';\n\nimport 'reflect-metadata';\n\n// Constants\nexport const INK_APP = Symbol.for('InkApp');\nexport const GLOBAL_CONFIG = Symbol.for('GlobalConfig');\nexport const CURRENT_PROJECT = Symbol.for('CurrentProject');\n\n// Types\nexport interface GlobalConfig {\n jobs?: number;\n verbose: number;\n}\n\n// Container\nexport const container = new Container({\n skipBaseClassChecks: true,\n});\n\n// Utilities\nexport const { lazyInject } = getDecorators(container);\n"],"names":["INK_APP","GLOBAL_CONFIG","CURRENT_PROJECT","container","lazyInject","Symbol","for","Container","skipBaseClassChecks","getDecorators"],"mappings":"AAAA;;;;;;;;;;;IAMaA,OAAO,MAAPA;IACAC,aAAa,MAAbA;IACAC,eAAe,MAAfA;IASAC,SAAS,MAATA;IAKEC,UAAU,MAAVA;;2BAtBW;gFACA;QAEnB;;;;;;AAGA,MAAMJ,UAAUK,OAAOC,GAAG,CAAC;AAC3B,MAAML,gBAAgBI,OAAOC,GAAG,CAAC;AACjC,MAAMJ,kBAAkBG,OAAOC,GAAG,CAAC;AASnC,MAAMH,YAAY,IAAII,oBAAS,CAAC;IACrCC,qBAAqB,IAAI;AAC3B;AAGO,MAAM,EAAEJ,WAAU,EAAE,GAAGK,IAAAA,kCAAa,EAACN","file":"inversify.config.js"}
@@ -0,0 +1,5 @@
1
+ import winston from 'winston';
2
+ export interface Logger extends winston.Logger {
3
+ }
4
+ export declare class Logger {
5
+ }
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/logger.service.ts"],"names":[],"mappings":"AAEA,OAAO,OAAO,MAAM,SAAS,CAAC;AAY9B,MAAM,WAAW,MAAO,SAAQ,OAAO,CAAC,MAAM;CAAG;AAGjD,qBACa,MAAM;CAAG","file":"logger.service.d.ts","sourcesContent":["import chalk from 'chalk';\nimport { injectable } from 'inversify';\nimport winston from 'winston';\n\nimport { type GlobalConfig, GLOBAL_CONFIG, container } from './inversify.config';\n\n// Constants\nconst VERBOSITY_LEVEL: Record<number, string> = {\n 1: 'verbose',\n 2: 'debug',\n};\n\n// Types\n// eslint-disable-next-line @typescript-eslint/no-empty-interface\nexport interface Logger extends winston.Logger {}\n\n// Service\n@injectable()\nexport class Logger {}\n\ncontainer.bind(Logger)\n .toDynamicValue((context) => {\n const config = context.container.get<GlobalConfig>(GLOBAL_CONFIG);\n\n return winston.createLogger({\n level: VERBOSITY_LEVEL[Math.min(config.verbose, 2)],\n format: winston.format.combine(\n winston.format.timestamp(),\n winston.format.ms(),\n ),\n transports: [\n new winston.transports.Console({\n format: winston.format.combine(\n winston.format.errors(),\n winston.format.colorize({\n message: true,\n colors: { debug: 'grey', verbose: 'blue', info: 'white', error: 'red' }\n }),\n winston.format.printf(({ label, message }) => message.split('\\n').map(line => [label && chalk.grey(`[${label}]`), line].filter(p => p).join(' ')).join('\\n')),\n )\n })\n ]\n });\n })\n .inSingletonScope();\n"]}
@@ -0,0 +1,58 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", {
3
+ value: true
4
+ });
5
+ Object.defineProperty(exports, "Logger", {
6
+ enumerable: true,
7
+ get: ()=>Logger
8
+ });
9
+ const _chalk = /*#__PURE__*/ _interopRequireDefault(require("chalk"));
10
+ const _inversify = require("inversify");
11
+ const _winston = /*#__PURE__*/ _interopRequireDefault(require("winston"));
12
+ const _inversifyConfig = require("./inversify.config");
13
+ function _interopRequireDefault(obj) {
14
+ return obj && obj.__esModule ? obj : {
15
+ default: obj
16
+ };
17
+ }
18
+ var __decorate = (void 0) && (void 0).__decorate || function(decorators, target, key, desc) {
19
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
20
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
21
+ else for(var i = decorators.length - 1; i >= 0; i--)if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
22
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
23
+ };
24
+ // Constants
25
+ const VERBOSITY_LEVEL = {
26
+ 1: 'verbose',
27
+ 2: 'debug'
28
+ };
29
+ let Logger = class Logger {
30
+ };
31
+ Logger = __decorate([
32
+ (0, _inversify.injectable)()
33
+ ], Logger);
34
+ _inversifyConfig.container.bind(Logger).toDynamicValue((context)=>{
35
+ const config = context.container.get(_inversifyConfig.GLOBAL_CONFIG);
36
+ return _winston.default.createLogger({
37
+ level: VERBOSITY_LEVEL[Math.min(config.verbose, 2)],
38
+ format: _winston.default.format.combine(_winston.default.format.timestamp(), _winston.default.format.ms()),
39
+ transports: [
40
+ new _winston.default.transports.Console({
41
+ format: _winston.default.format.combine(_winston.default.format.errors(), _winston.default.format.colorize({
42
+ message: true,
43
+ colors: {
44
+ debug: 'grey',
45
+ verbose: 'blue',
46
+ info: 'white',
47
+ error: 'red'
48
+ }
49
+ }), _winston.default.format.printf(({ label , message })=>message.split('\n').map((line)=>[
50
+ label && _chalk.default.grey(`[${label}]`),
51
+ line
52
+ ].filter((p)=>p).join(' ')).join('\n')))
53
+ })
54
+ ]
55
+ });
56
+ }).inSingletonScope();
57
+
58
+ //# sourceMappingURL=logger.service.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/logger.service.js"],"sourcesContent":["import chalk from 'chalk';\nimport { injectable } from 'inversify';\nimport winston from 'winston';\n\nimport { type GlobalConfig, GLOBAL_CONFIG, container } from './inversify.config';\n\n// Constants\nconst VERBOSITY_LEVEL: Record<number, string> = {\n 1: 'verbose',\n 2: 'debug',\n};\n\n// Types\n// eslint-disable-next-line @typescript-eslint/no-empty-interface\nexport interface Logger extends winston.Logger {}\n\n// Service\n@injectable()\nexport class Logger {}\n\ncontainer.bind(Logger)\n .toDynamicValue((context) => {\n const config = context.container.get<GlobalConfig>(GLOBAL_CONFIG);\n\n return winston.createLogger({\n level: VERBOSITY_LEVEL[Math.min(config.verbose, 2)],\n format: winston.format.combine(\n winston.format.timestamp(),\n winston.format.ms(),\n ),\n transports: [\n new winston.transports.Console({\n format: winston.format.combine(\n winston.format.errors(),\n winston.format.colorize({\n message: true,\n colors: { debug: 'grey', verbose: 'blue', info: 'white', error: 'red' }\n }),\n winston.format.printf(({ label, message }) => message.split('\\n').map(line => [label && chalk.grey(`[${label}]`), line].filter(p => p).join(' ')).join('\\n')),\n )\n })\n ]\n });\n })\n .inSingletonScope();\n"],"names":["Logger","VERBOSITY_LEVEL","injectable","container","bind","toDynamicValue","context","config","get","GLOBAL_CONFIG","winston","createLogger","level","Math","min","verbose","format","combine","timestamp","ms","transports","Console","errors","colorize","message","colors","debug","info","error","printf","label","split","map","line","chalk","grey","filter","p","join","inSingletonScope"],"mappings":"AAAA;;;;+BAkBaA;;aAAAA;;4DAlBK;2BACS;8DACP;iCAEwC;;;;;;;;;;;;AAE5D,YAAY;AACZ,MAAMC,kBAA0C;IAC9C,GAAG;IACH,GAAG;AACL;IAQaD,SAAN;AAAc;AAARA;IADZE,IAAAA,qBAAU;GACEF;AAEbG,0BAAS,CAACC,IAAI,CAACJ,QACZK,cAAc,CAAC,CAACC,UAAY;IAC3B,MAAMC,SAASD,QAAQH,SAAS,CAACK,GAAG,CAAeC,8BAAa;IAEhE,OAAOC,gBAAO,CAACC,YAAY,CAAC;QAC1BC,OAAOX,eAAe,CAACY,KAAKC,GAAG,CAACP,OAAOQ,OAAO,EAAE,GAAG;QACnDC,QAAQN,gBAAO,CAACM,MAAM,CAACC,OAAO,CAC5BP,gBAAO,CAACM,MAAM,CAACE,SAAS,IACxBR,gBAAO,CAACM,MAAM,CAACG,EAAE;QAEnBC,YAAY;YACV,IAAIV,gBAAO,CAACU,UAAU,CAACC,OAAO,CAAC;gBAC7BL,QAAQN,gBAAO,CAACM,MAAM,CAACC,OAAO,CAC5BP,gBAAO,CAACM,MAAM,CAACM,MAAM,IACrBZ,gBAAO,CAACM,MAAM,CAACO,QAAQ,CAAC;oBACtBC,SAAS,IAAI;oBACbC,QAAQ;wBAAEC,OAAO;wBAAQX,SAAS;wBAAQY,MAAM;wBAASC,OAAO;oBAAM;gBACxE,IACAlB,gBAAO,CAACM,MAAM,CAACa,MAAM,CAAC,CAAC,EAAEC,MAAK,EAAEN,QAAO,EAAE,GAAKA,QAAQO,KAAK,CAAC,MAAMC,GAAG,CAACC,CAAAA,OAAQ;4BAACH,SAASI,cAAK,CAACC,IAAI,CAAC,CAAC,CAAC,EAAEL,MAAM,CAAC,CAAC;4BAAGG;yBAAK,CAACG,MAAM,CAACC,CAAAA,IAAKA,GAAGC,IAAI,CAAC,MAAMA,IAAI,CAAC;YAE3J;SACD;IACH;AACF,GACCC,gBAAgB","file":"logger.service.js"}
@@ -0,0 +1,14 @@
1
+ export interface SpinnerState {
2
+ spin: boolean;
3
+ label: string;
4
+ }
5
+ export declare type SpinnerStateListener = (state: SpinnerState) => void;
6
+ export declare class SpinnerService {
7
+ private _spin;
8
+ private _label;
9
+ private readonly _listeners;
10
+ subscribe(listener: SpinnerStateListener): () => void;
11
+ spin(label: string): void;
12
+ stop(): void;
13
+ get state(): SpinnerState;
14
+ }
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["services/spinner.service.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,OAAO,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;CACf;AAED,oBAAY,oBAAoB,GAAG,CAAC,KAAK,EAAE,YAAY,KAAK,IAAI,CAAC;AAGjE,qBACa,cAAc;IAEzB,OAAO,CAAC,KAAK,CAAS;IACtB,OAAO,CAAC,MAAM,CAAM;IAEpB,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAmC;IAG9D,SAAS,CAAC,QAAQ,EAAE,oBAAoB;IAQxC,IAAI,CAAC,KAAK,EAAE,MAAM;IASlB,IAAI;IASJ,IAAI,KAAK,IAAI,YAAY,CAKxB;CACF","file":"spinner.service.d.ts","sourcesContent":["import { injectable } from 'inversify';\nimport { container } from './inversify.config';\n\n// Interface\nexport interface SpinnerState {\n spin: boolean;\n label: string;\n}\n\nexport type SpinnerStateListener = (state: SpinnerState) => void;\n\n// Service\n@injectable()\nexport class SpinnerService {\n // Attributes\n private _spin = false;\n private _label = '';\n\n private readonly _listeners = new Set<SpinnerStateListener>();\n\n // Methods\n subscribe(listener: SpinnerStateListener) {\n this._listeners.add(listener);\n\n return () => {\n this._listeners.delete(listener);\n };\n }\n\n spin(label: string) {\n this._spin = true;\n this._label = label;\n\n for (const listener of this._listeners) {\n listener(this.state);\n }\n }\n\n stop() {\n this._spin = false;\n\n for (const listener of this._listeners) {\n listener(this.state);\n }\n }\n\n // Properties\n get state(): SpinnerState {\n return {\n spin: this._spin,\n label: this._label,\n };\n }\n}\n\ncontainer.bind(SpinnerService)\n .toSelf()\n .inSingletonScope();\n"]}