@cldmv/slothlet 3.3.0 → 3.3.2

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 (127) hide show
  1. package/README.md +6 -8
  2. package/REFERENCE.md +23 -0
  3. package/dist/lib/builders/api-assignment.mjs +1 -589
  4. package/dist/lib/builders/api_builder.mjs +1 -1385
  5. package/dist/lib/builders/builder.mjs +1 -78
  6. package/dist/lib/builders/modes-processor.mjs +1 -1800
  7. package/dist/lib/errors.mjs +9 -211
  8. package/dist/lib/factories/component-base.mjs +1 -80
  9. package/dist/lib/factories/context.mjs +1 -22
  10. package/dist/lib/handlers/api-cache-manager.mjs +1 -200
  11. package/dist/lib/handlers/api-manager.mjs +1 -2536
  12. package/dist/lib/handlers/context-async.mjs +1 -172
  13. package/dist/lib/handlers/context-live.mjs +1 -173
  14. package/dist/lib/handlers/hook-manager.mjs +1 -667
  15. package/dist/lib/handlers/lifecycle-token.mjs +1 -28
  16. package/dist/lib/handlers/lifecycle.mjs +1 -115
  17. package/dist/lib/handlers/materialize-manager.mjs +1 -48
  18. package/dist/lib/handlers/metadata.mjs +1 -501
  19. package/dist/lib/handlers/ownership.mjs +1 -322
  20. package/dist/lib/handlers/permission-manager.mjs +1 -392
  21. package/dist/lib/handlers/unified-wrapper.mjs +1 -3110
  22. package/dist/lib/handlers/version-manager.mjs +1 -885
  23. package/dist/lib/helpers/class-instance-wrapper.mjs +1 -109
  24. package/dist/lib/helpers/config.mjs +1 -439
  25. package/dist/lib/helpers/eventemitter-context.mjs +1 -349
  26. package/dist/lib/helpers/hint-detector.mjs +1 -47
  27. package/dist/lib/helpers/modes-utils.mjs +1 -37
  28. package/dist/lib/helpers/pattern-matcher.mjs +1 -125
  29. package/dist/lib/helpers/resolve-from-caller.mjs +1 -169
  30. package/dist/lib/helpers/sanitize.mjs +1 -340
  31. package/dist/lib/helpers/utilities.mjs +1 -70
  32. package/dist/lib/i18n/translations.mjs +1 -126
  33. package/dist/lib/modes/eager.mjs +1 -59
  34. package/dist/lib/modes/lazy.mjs +1 -81
  35. package/dist/lib/processors/flatten.mjs +1 -437
  36. package/dist/lib/processors/loader.mjs +1 -339
  37. package/dist/lib/processors/type-generator.mjs +1 -275
  38. package/dist/lib/processors/typescript.mjs +1 -172
  39. package/dist/lib/runtime/runtime-asynclocalstorage.mjs +1 -113
  40. package/dist/lib/runtime/runtime-livebindings.mjs +1 -78
  41. package/dist/lib/runtime/runtime.mjs +1 -102
  42. package/dist/slothlet.mjs +1 -817
  43. package/package.json +34 -31
  44. package/types/dist/lib/builders/api-assignment.d.mts +3 -92
  45. package/types/dist/lib/builders/api-assignment.d.mts.map +1 -1
  46. package/types/dist/lib/builders/api_builder.d.mts +102 -91
  47. package/types/dist/lib/builders/api_builder.d.mts.map +1 -1
  48. package/types/dist/lib/builders/builder.d.mts +1 -55
  49. package/types/dist/lib/builders/builder.d.mts.map +1 -1
  50. package/types/dist/lib/builders/modes-processor.d.mts +3 -27
  51. package/types/dist/lib/builders/modes-processor.d.mts.map +1 -1
  52. package/types/dist/lib/errors.d.mts +19 -109
  53. package/types/dist/lib/errors.d.mts.map +1 -1
  54. package/types/dist/lib/factories/component-base.d.mts +7 -177
  55. package/types/dist/lib/factories/component-base.d.mts.map +1 -1
  56. package/types/dist/lib/factories/context.d.mts +4 -22
  57. package/types/dist/lib/factories/context.d.mts.map +1 -1
  58. package/types/dist/lib/handlers/api-cache-manager.d.mts +20 -203
  59. package/types/dist/lib/handlers/api-cache-manager.d.mts.map +1 -1
  60. package/types/dist/lib/handlers/api-manager.d.mts +33 -408
  61. package/types/dist/lib/handlers/api-manager.d.mts.map +1 -1
  62. package/types/dist/lib/handlers/context-async.d.mts +23 -61
  63. package/types/dist/lib/handlers/context-async.d.mts.map +1 -1
  64. package/types/dist/lib/handlers/context-live.d.mts +22 -59
  65. package/types/dist/lib/handlers/context-live.d.mts.map +1 -1
  66. package/types/dist/lib/handlers/hook-manager.d.mts +46 -185
  67. package/types/dist/lib/handlers/hook-manager.d.mts.map +1 -1
  68. package/types/dist/lib/handlers/lifecycle-token.d.mts +3 -48
  69. package/types/dist/lib/handlers/lifecycle-token.d.mts.map +1 -1
  70. package/types/dist/lib/handlers/lifecycle.d.mts +5 -82
  71. package/types/dist/lib/handlers/lifecycle.d.mts.map +1 -1
  72. package/types/dist/lib/handlers/materialize-manager.d.mts +8 -70
  73. package/types/dist/lib/handlers/materialize-manager.d.mts.map +1 -1
  74. package/types/dist/lib/handlers/metadata.d.mts +17 -221
  75. package/types/dist/lib/handlers/metadata.d.mts.map +1 -1
  76. package/types/dist/lib/handlers/ownership.d.mts +44 -160
  77. package/types/dist/lib/handlers/ownership.d.mts.map +1 -1
  78. package/types/dist/lib/handlers/permission-manager.d.mts +37 -141
  79. package/types/dist/lib/handlers/permission-manager.d.mts.map +1 -1
  80. package/types/dist/lib/handlers/unified-wrapper.d.mts +26 -239
  81. package/types/dist/lib/handlers/unified-wrapper.d.mts.map +1 -1
  82. package/types/dist/lib/handlers/version-manager.d.mts +28 -225
  83. package/types/dist/lib/handlers/version-manager.d.mts.map +1 -1
  84. package/types/dist/lib/helpers/class-instance-wrapper.d.mts +2 -52
  85. package/types/dist/lib/helpers/class-instance-wrapper.d.mts.map +1 -1
  86. package/types/dist/lib/helpers/config.d.mts +125 -139
  87. package/types/dist/lib/helpers/config.d.mts.map +1 -1
  88. package/types/dist/lib/helpers/eventemitter-context.d.mts +3 -29
  89. package/types/dist/lib/helpers/eventemitter-context.d.mts.map +1 -1
  90. package/types/dist/lib/helpers/hint-detector.d.mts +2 -15
  91. package/types/dist/lib/helpers/hint-detector.d.mts.map +1 -1
  92. package/types/dist/lib/helpers/modes-utils.d.mts +3 -30
  93. package/types/dist/lib/helpers/modes-utils.d.mts.map +1 -1
  94. package/types/dist/lib/helpers/pattern-matcher.d.mts +3 -43
  95. package/types/dist/lib/helpers/pattern-matcher.d.mts.map +1 -1
  96. package/types/dist/lib/helpers/resolve-from-caller.d.mts +3 -27
  97. package/types/dist/lib/helpers/resolve-from-caller.d.mts.map +1 -1
  98. package/types/dist/lib/helpers/sanitize.d.mts +4 -92
  99. package/types/dist/lib/helpers/sanitize.d.mts.map +1 -1
  100. package/types/dist/lib/helpers/utilities.d.mts +4 -52
  101. package/types/dist/lib/helpers/utilities.d.mts.map +1 -1
  102. package/types/dist/lib/i18n/translations.d.mts +4 -37
  103. package/types/dist/lib/i18n/translations.d.mts.map +1 -1
  104. package/types/dist/lib/modes/eager.d.mts +8 -30
  105. package/types/dist/lib/modes/eager.d.mts.map +1 -1
  106. package/types/dist/lib/modes/lazy.d.mts +10 -43
  107. package/types/dist/lib/modes/lazy.d.mts.map +1 -1
  108. package/types/dist/lib/processors/flatten.d.mts +56 -107
  109. package/types/dist/lib/processors/flatten.d.mts.map +1 -1
  110. package/types/dist/lib/processors/loader.d.mts +6 -41
  111. package/types/dist/lib/processors/loader.d.mts.map +1 -1
  112. package/types/dist/lib/processors/type-generator.d.mts +2 -16
  113. package/types/dist/lib/processors/type-generator.d.mts.map +1 -1
  114. package/types/dist/lib/processors/typescript.d.mts +6 -53
  115. package/types/dist/lib/processors/typescript.d.mts.map +1 -1
  116. package/types/dist/lib/runtime/runtime-asynclocalstorage.d.mts +3 -71
  117. package/types/dist/lib/runtime/runtime-asynclocalstorage.d.mts.map +1 -1
  118. package/types/dist/lib/runtime/runtime-livebindings.d.mts +2 -37
  119. package/types/dist/lib/runtime/runtime-livebindings.d.mts.map +1 -1
  120. package/types/dist/lib/runtime/runtime.d.mts +3 -39
  121. package/types/dist/lib/runtime/runtime.d.mts.map +1 -1
  122. package/types/dist/slothlet.d.mts +3 -249
  123. package/types/dist/slothlet.d.mts.map +1 -1
  124. package/types/index.d.mts +36 -16
  125. package/types/index.d.mts.map +1 -0
  126. package/AGENT-USAGE.md +0 -736
  127. package/docs/API-RULES.md +0 -712
@@ -14,214 +14,12 @@
14
14
  limitations under the License.
15
15
  */
16
16
 
17
-
18
-
19
-
20
- import { translate } from "@cldmv/slothlet/i18n";
21
- import { detectHint } from "@cldmv/slothlet/helpers/hint-detector";
22
-
23
-
24
- export class SlothletError extends Error {
25
-
26
- constructor(code, context = {}, originalError = null, options = {}) {
27
-
28
-
29
- const { validationError: ctxValidation, stub: ctxStub, ...contextData } = context;
30
- const { validationError: optsValidation, stub: optsStub } = options;
31
- const validationError = ctxValidation ?? optsValidation;
32
- const stub = ctxStub ?? optsStub;
33
-
34
-
35
- const enrichedContext = originalError ? { ...contextData, error: originalError.message } : contextData;
36
-
37
-
38
- const translatedMessage = translate(code, enrichedContext);
39
-
40
-
41
- const skipHint = stub || validationError;
42
- const hintKey = originalError && !skipHint ? detectHint(originalError, code) : undefined;
43
- let translatedHint = hintKey ? translate(hintKey, enrichedContext) : undefined;
44
-
45
-
46
-
47
-
48
-
49
-
50
- if (!translatedHint && validationError) {
51
- const staticHintKey = `HINT_${code}`;
52
- const staticHint = translate(staticHintKey, enrichedContext);
53
-
54
- if (staticHint && !staticHint.startsWith("Error:")) {
55
- translatedHint = staticHint;
56
- }
57
- }
58
-
59
-
60
- const messageWithCode = `[${code}] ${translatedMessage}`;
61
- super(messageWithCode);
62
- this.name = "SlothletError";
63
-
64
-
65
- Object.defineProperty(this, "code", {
66
- value: code,
67
- enumerable: false,
68
- writable: false
69
- });
70
-
71
- Object.defineProperty(this, "context", {
72
- value: context,
73
- enumerable: false,
74
- writable: false
75
- });
76
-
77
- Object.defineProperty(this, "originalError", {
78
- value: originalError,
79
- enumerable: false,
80
- writable: false
81
- });
82
-
83
- Object.defineProperty(this, "hint", {
84
- value: translatedHint,
85
- enumerable: false,
86
- writable: false
87
- });
88
-
89
- Error.captureStackTrace(this, SlothletError);
90
- }
91
-
92
-
93
- toString() {
94
- let output = `\n[${this.code}] ${this.name}\n`;
95
-
96
-
97
- if (this.hint) {
98
- output += `\n💡 ${this.hint}\n`;
99
- }
100
-
101
-
102
- const criticalKeys = ["modulePath", "moduleID", "apiPath"];
103
- const detailKeys = Object.keys(this.context).filter((k) => criticalKeys.includes(k));
104
-
105
- if (detailKeys.length > 0) {
106
- output += "\nDetails:\n";
107
- for (const key of detailKeys) {
108
- output += ` ${key}: ${this.context[key]}\n`;
109
- }
110
- }
111
-
112
- return output;
113
- }
114
-
115
-
116
- [Symbol.for("nodejs.util.inspect.custom")]() {
117
- return this.toString() + `\n${this.stack}`;
118
- }
119
-
120
-
121
- toJSON() {
122
- return {
123
- name: this.name,
124
- code: this.code,
125
- message: this.message,
126
- hint: this.context.hint
127
- };
128
- }
129
- }
130
-
131
-
132
- export class SlothletWarning {
133
-
134
- static captured = [];
135
- static suppressConsole = false;
136
-
137
-
138
- constructor(code, context = {}) {
139
-
140
- const { key: msgKey, ...contextData } = context;
141
- const translationKey = msgKey ?? code;
142
-
143
-
144
- const translatedMessage = translate(translationKey, contextData);
145
-
146
- this.name = "SlothletWarning";
147
- this.code = code;
148
- this.message = translatedMessage;
149
- this.context = context;
150
-
151
-
152
- if (!SlothletWarning.suppressConsole) {
153
- console.warn(`\n⚠️ [${this.code}] ${this.name}\n${this.message}`);
154
-
155
-
156
- if (Object.keys(contextData).length > 0) {
157
- console.warn("Context:", contextData);
158
- }
159
- } else {
160
-
161
- SlothletWarning.captured.push(this);
162
- }
163
- }
164
-
165
-
166
- toString() {
167
- return `[${this.code}] ${this.name}: ${this.message}`;
168
- }
169
-
170
-
171
- static clearCaptured() {
172
- SlothletWarning.captured = [];
173
- }
174
- }
175
-
176
-
177
- export class SlothletDebug {
178
-
179
- constructor(config = {}) {
180
- this.config = config;
181
- this.debugFlags = config.debug || {};
182
- }
183
-
184
-
185
- log(code, context = {}) {
186
-
187
-
188
-
189
-
190
- if (!this.debugFlags || !this.debugFlags[code]) {
191
- return;
192
- }
193
-
194
-
195
- const { key: msgKey, message, ...contextParams } = context;
196
-
197
-
198
- const translationKey = msgKey ?? `DEBUG_${code.toUpperCase()}`;
199
- const translatedMessage = translate(translationKey, contextParams);
200
- const hasTranslation = translatedMessage && !translatedMessage.startsWith("Error:");
201
-
202
-
203
- const label = `[DEBUG:${code.toUpperCase()}]`;
204
-
205
- if (hasTranslation) {
206
- console.log(`${label} ${translatedMessage}`);
207
- } else if (message) {
208
-
209
- console.log(`${label} ${message}`);
210
- } else {
211
-
212
- console.log(label, contextParams);
213
- }
214
-
215
-
216
- if ((hasTranslation || message) && Object.keys(contextParams).length > 0) {
217
- console.log(`${label} Context:`, contextParams);
218
- }
219
- }
220
-
221
-
222
- toString() {
223
- return `[SlothletDebug] flags: ${Object.keys(this.debugFlags)
224
- .filter((k) => this.debugFlags[k])
225
- .join(", ")}`;
226
- }
227
- }
17
+ import{translate}from"@cldmv/slothlet/i18n";import{detectHint}from"@cldmv/slothlet/helpers/hint-detector";class SlothletError extends Error{constructor(code,context={},originalError=null,options={}){const{validationError:ctxValidation,stub:ctxStub,...contextData}=context;const{validationError:optsValidation,stub:optsStub}=options;const validationError=ctxValidation??optsValidation;const stub=ctxStub??optsStub;const enrichedContext=originalError?{...contextData,error:originalError.message}:contextData;const translatedMessage=translate(code,enrichedContext);const skipHint=stub||validationError;const hintKey=originalError&&!skipHint?detectHint(originalError,code):void 0;let translatedHint=hintKey?translate(hintKey,enrichedContext):void 0;if(!translatedHint&&validationError){const staticHintKey=`HINT_${code}`;const staticHint=translate(staticHintKey,enrichedContext);if(staticHint&&!staticHint.startsWith("Error:")){translatedHint=staticHint}}const messageWithCode=`[${code}] ${translatedMessage}`;super(messageWithCode);this.name="SlothletError";Object.defineProperty(this,"code",{value:code,enumerable:false,writable:false});Object.defineProperty(this,"context",{value:context,enumerable:false,writable:false});Object.defineProperty(this,"originalError",{value:originalError,enumerable:false,writable:false});Object.defineProperty(this,"hint",{value:translatedHint,enumerable:false,writable:false});Error.captureStackTrace(this,SlothletError)}toString(){let output=`
18
+ [${this.code}] ${this.name}
19
+ `;if(this.hint){output+=`
20
+ \u{1F4A1} ${this.hint}
21
+ `}const criticalKeys=["modulePath","moduleID","apiPath"];const detailKeys=Object.keys(this.context).filter(k=>criticalKeys.includes(k));if(detailKeys.length>0){output+="\nDetails:\n";for(const key of detailKeys){output+=` ${key}: ${this.context[key]}
22
+ `}}return output}[Symbol.for("nodejs.util.inspect.custom")](){return this.toString()+`
23
+ ${this.stack}`}toJSON(){return{name:this.name,code:this.code,message:this.message,hint:this.context.hint}}}class SlothletWarning{static captured=[];static suppressConsole=false;constructor(code,context={}){const{key:msgKey,...contextData}=context;const translationKey=msgKey??code;const translatedMessage=translate(translationKey,contextData);this.name="SlothletWarning";this.code=code;this.message=translatedMessage;this.context=context;if(!SlothletWarning.suppressConsole){console.warn(`
24
+ \u26A0\uFE0F [${this.code}] ${this.name}
25
+ ${this.message}`);if(Object.keys(contextData).length>0){console.warn("Context:",contextData)}}else{SlothletWarning.captured.push(this)}}toString(){return`[${this.code}] ${this.name}: ${this.message}`}static clearCaptured(){SlothletWarning.captured=[]}}class SlothletDebug{constructor(config={}){this.config=config;this.debugFlags=config.debug||{}}log(code,context={}){if(!this.debugFlags||!this.debugFlags[code]){return}const{key:msgKey,message,...contextParams}=context;const translationKey=msgKey??`DEBUG_${code.toUpperCase()}`;const translatedMessage=translate(translationKey,contextParams);const hasTranslation=translatedMessage&&!translatedMessage.startsWith("Error:");const label=`[DEBUG:${code.toUpperCase()}]`;if(hasTranslation){console.log(`${label} ${translatedMessage}`)}else if(message){console.log(`${label} ${message}`)}else{console.log(label,contextParams)}if((hasTranslation||message)&&Object.keys(contextParams).length>0){console.log(`${label} Context:`,contextParams)}}toString(){return`[SlothletDebug] flags: ${Object.keys(this.debugFlags).filter(k=>this.debugFlags[k]).join(", ")}`}}export{SlothletDebug,SlothletError,SlothletWarning};
@@ -14,83 +14,4 @@
14
14
  limitations under the License.
15
15
  */
16
16
 
17
-
18
-
19
-
20
-
21
-
22
- export class ComponentBase {
23
-
24
- #slothlet;
25
-
26
-
27
- constructor(slothlet) {
28
- this.#slothlet = slothlet;
29
- }
30
-
31
-
32
- get ____slothlet() {
33
- return this.#slothlet;
34
- }
35
-
36
-
37
- get slothlet() {
38
- return this.#slothlet;
39
- }
40
-
41
-
42
- get ____config() {
43
- return this.____slothlet.config;
44
- }
45
-
46
-
47
- get instanceID() {
48
- return this.____slothlet.instanceID;
49
- }
50
-
51
-
52
- get SlothletError() {
53
- return this.____slothlet.SlothletError;
54
- }
55
-
56
-
57
- get SlothletWarning() {
58
- return this.____slothlet.SlothletWarning;
59
- }
60
-
61
-
62
- static INTERNAL_KEYS = new Set([
63
-
64
- "____slothletInternal",
65
- "____slothlet",
66
-
67
- "___getState",
68
- "___setImpl",
69
- "___resetLazy",
70
- "___invalidate",
71
-
72
- "__state",
73
- "__invalid",
74
- "__mode",
75
- "__apiPath",
76
- "__slothletPath",
77
- "__isCallable",
78
- "__materializeOnCreate",
79
- "__displayName",
80
- "__type",
81
- "__metadata",
82
- "__filePath",
83
- "__sourceFolder",
84
- "__moduleID",
85
- "__materialized",
86
- "__inFlight",
87
- "__impl",
88
-
89
- "_impl",
90
- "_materialize",
91
-
92
- "slothlet",
93
- "shutdown",
94
- "destroy"
95
- ]);
96
- }
17
+ class ComponentBase{#slothlet;constructor(slothlet){this.#slothlet=slothlet}get ____slothlet(){return this.#slothlet}get slothlet(){return this.#slothlet}get ____config(){return this.____slothlet.config}get instanceID(){return this.____slothlet.instanceID}get SlothletError(){return this.____slothlet.SlothletError}get SlothletWarning(){return this.____slothlet.SlothletWarning}static INTERNAL_KEYS=new Set(["____slothletInternal","____slothlet","___getState","___setImpl","___resetLazy","___invalidate","__state","__invalid","__mode","__apiPath","__slothletPath","__isCallable","__materializeOnCreate","__displayName","__type","__metadata","__filePath","__sourceFolder","__moduleID","__materialized","__inFlight","__impl","_impl","_materialize","slothlet","shutdown","destroy"])}export{ComponentBase};
@@ -14,25 +14,4 @@
14
14
  limitations under the License.
15
15
  */
16
16
 
17
-
18
-
19
-
20
- import { asyncContextManager } from "@cldmv/slothlet/handlers/context-async";
21
- import { liveContextManager } from "@cldmv/slothlet/handlers/context-live";
22
-
23
-
24
- export function getContextManager(runtime = "async") {
25
- return runtime === "live" ? liveContextManager : asyncContextManager;
26
- }
27
-
28
-
29
- export const contextManager = asyncContextManager;
30
-
31
-
32
- export const asyncRuntime = asyncContextManager;
33
-
34
-
35
- export const liveRuntime = liveContextManager;
36
-
37
-
38
- export { asyncContextManager, liveContextManager };
17
+ import{asyncContextManager}from"@cldmv/slothlet/handlers/context-async";import{liveContextManager}from"@cldmv/slothlet/handlers/context-live";function getContextManager(runtime="async"){return runtime==="live"?liveContextManager:asyncContextManager}const contextManager=asyncContextManager;const asyncRuntime=asyncContextManager;const liveRuntime=liveContextManager;export{asyncContextManager,asyncRuntime,contextManager,getContextManager,liveContextManager,liveRuntime};
@@ -14,203 +14,4 @@
14
14
  limitations under the License.
15
15
  */
16
16
 
17
-
18
-
19
-
20
- import { ComponentBase } from "@cldmv/slothlet/factories/component-base";
21
-
22
-
23
-
24
-
25
- export class ApiCacheManager extends ComponentBase {
26
- static slothletProperty = "apiCacheManager";
27
-
28
-
29
- constructor(slothlet) {
30
- super(slothlet);
31
-
32
-
33
- this.caches = new Map();
34
- }
35
-
36
-
37
- set(moduleID, entry) {
38
- if (!moduleID || typeof moduleID !== "string") {
39
- throw new this.SlothletError("INVALID_ARGUMENT", {
40
- argument: "moduleID",
41
- expected: "non-empty string",
42
- received: typeof moduleID,
43
- validationError: true
44
- });
45
- }
46
-
47
- if (!entry || !entry.api) {
48
- throw new this.SlothletError("INVALID_ARGUMENT", {
49
- argument: "entry.api",
50
- expected: "API tree object",
51
- received: typeof entry?.api,
52
- validationError: true
53
- });
54
- }
55
-
56
-
57
- if (entry.moduleID && entry.moduleID !== moduleID) {
58
- throw new this.SlothletError("CACHE_MODULEID_MISMATCH", {
59
- cacheKey: moduleID,
60
- entryModuleID: entry.moduleID,
61
- validationError: true
62
- });
63
- }
64
-
65
- this.caches.set(moduleID, entry);
66
-
67
- this.slothlet.debug("cache", {
68
- key: "DEBUG_MODE_CACHE_ENTRY_STORED",
69
- moduleID,
70
- endpoint: entry.endpoint,
71
- mode: entry.mode,
72
- timestamp: entry.timestamp
73
- });
74
- }
75
-
76
-
77
- get(moduleID) {
78
- return this.caches.get(moduleID);
79
- }
80
-
81
-
82
- has(moduleID) {
83
- return this.caches.has(moduleID);
84
- }
85
-
86
-
87
- delete(moduleID) {
88
- const deleted = this.caches.delete(moduleID);
89
-
90
- if (deleted) {
91
- this.slothlet.debug("cache", {
92
- key: "DEBUG_MODE_CACHE_ENTRY_DELETED",
93
- moduleID
94
- });
95
- }
96
-
97
- return deleted;
98
- }
99
-
100
-
101
- getAllModuleIDs() {
102
- return Array.from(this.caches.keys());
103
- }
104
-
105
-
106
- getCacheDiagnostics() {
107
- const caches = [];
108
-
109
- for (const [moduleID, entry] of this.caches.entries()) {
110
-
111
- const pathCount = this._countPaths(entry.api);
112
-
113
- caches.push({
114
- moduleID,
115
- endpoint: entry.endpoint,
116
- folderPath: entry.folderPath,
117
- mode: entry.mode,
118
- pathCount,
119
- timestamp: entry.timestamp
120
- });
121
- }
122
-
123
- return {
124
- totalCaches: this.caches.size,
125
- caches
126
- };
127
- }
128
-
129
-
130
- _countPaths(api, visited = new WeakSet()) {
131
- if (!api || typeof api !== "object" || visited.has(api)) {
132
- return 0;
133
- }
134
-
135
- visited.add(api);
136
- let count = 0;
137
-
138
- for (const [key, value] of Object.entries(api)) {
139
-
140
- if (key.startsWith("__") || key.startsWith("_")) {
141
- continue;
142
- }
143
-
144
- count += 1;
145
-
146
- if (value && typeof value === "object") {
147
- count += this._countPaths(value, visited);
148
- }
149
- }
150
-
151
- return count;
152
- }
153
-
154
-
155
- clear() {
156
- this.caches.clear();
157
-
158
- this.slothlet.debug("cache", {
159
- key: "DEBUG_MODE_ALL_CACHES_CLEARED"
160
- });
161
- }
162
-
163
-
164
- async rebuildCache(moduleID) {
165
- const entry = this.get(moduleID);
166
-
167
- if (!entry) {
168
- throw new this.SlothletError("CACHE_NOT_FOUND", {
169
- moduleID,
170
- operation: "rebuild",
171
- validationError: true
172
- });
173
- }
174
-
175
- this.slothlet.debug("cache", {
176
- key: "DEBUG_MODE_REBUILDING_CACHE_FROM_DISK",
177
- moduleID,
178
- folderPath: entry.folderPath,
179
- mode: entry.mode
180
- });
181
-
182
-
183
-
184
-
185
-
186
-
187
-
188
-
189
-
190
-
191
-
192
-
193
-
194
-
195
-
196
-
197
- const freshApi = await this.slothlet.builders.builder.buildAPI({
198
- dir: entry.folderPath,
199
- mode: entry.mode,
200
- sanitize: entry.sanitizeOptions,
201
- moduleID: moduleID,
202
- apiPathPrefix: entry.endpoint === "." ? "" : entry.endpoint,
203
- collisionMode: entry.collisionMode,
204
- collisionContext: entry.endpoint === "." ? "initial" : "addApi",
205
- cacheBust: Date.now()
206
- });
207
-
208
- this.slothlet.debug("cache", {
209
- key: "DEBUG_MODE_CACHE_REBUILT_SUCCESSFULLY",
210
- moduleID,
211
- timestamp: Date.now()
212
- });
213
-
214
- return freshApi;
215
- }
216
- }
17
+ import{ComponentBase}from"@cldmv/slothlet/factories/component-base";class ApiCacheManager extends ComponentBase{static slothletProperty="apiCacheManager";constructor(slothlet){super(slothlet);this.caches=new Map}set(moduleID,entry){if(!moduleID||typeof moduleID!=="string"){throw new this.SlothletError("INVALID_ARGUMENT",{argument:"moduleID",expected:"non-empty string",received:typeof moduleID,validationError:true})}if(!entry||!entry.api){throw new this.SlothletError("INVALID_ARGUMENT",{argument:"entry.api",expected:"API tree object",received:typeof entry?.api,validationError:true})}if(entry.moduleID&&entry.moduleID!==moduleID){throw new this.SlothletError("CACHE_MODULEID_MISMATCH",{cacheKey:moduleID,entryModuleID:entry.moduleID,validationError:true})}this.caches.set(moduleID,entry);this.slothlet.debug("cache",{key:"DEBUG_MODE_CACHE_ENTRY_STORED",moduleID,endpoint:entry.endpoint,mode:entry.mode,timestamp:entry.timestamp})}get(moduleID){return this.caches.get(moduleID)}has(moduleID){return this.caches.has(moduleID)}delete(moduleID){const deleted=this.caches.delete(moduleID);if(deleted){this.slothlet.debug("cache",{key:"DEBUG_MODE_CACHE_ENTRY_DELETED",moduleID})}return deleted}getAllModuleIDs(){return Array.from(this.caches.keys())}getCacheDiagnostics(){const caches=[];for(const[moduleID,entry]of this.caches.entries()){const pathCount=this._countPaths(entry.api);caches.push({moduleID,endpoint:entry.endpoint,folderPath:entry.folderPath,mode:entry.mode,pathCount,timestamp:entry.timestamp})}return{totalCaches:this.caches.size,caches}}_countPaths(api,visited=new WeakSet){if(!api||typeof api!=="object"||visited.has(api)){return 0}visited.add(api);let count=0;for(const[key,value]of Object.entries(api)){if(key.startsWith("__")||key.startsWith("_")){continue}count+=1;if(value&&typeof value==="object"){count+=this._countPaths(value,visited)}}return count}clear(){this.caches.clear();this.slothlet.debug("cache",{key:"DEBUG_MODE_ALL_CACHES_CLEARED"})}async rebuildCache(moduleID){const entry=this.get(moduleID);if(!entry){throw new this.SlothletError("CACHE_NOT_FOUND",{moduleID,operation:"rebuild",validationError:true})}this.slothlet.debug("cache",{key:"DEBUG_MODE_REBUILDING_CACHE_FROM_DISK",moduleID,folderPath:entry.folderPath,mode:entry.mode});const freshApi=await this.slothlet.builders.builder.buildAPI({dir:entry.folderPath,mode:entry.mode,sanitize:entry.sanitizeOptions,moduleID,apiPathPrefix:entry.endpoint==="."?"":entry.endpoint,collisionMode:entry.collisionMode,collisionContext:entry.endpoint==="."?"initial":"addApi",cacheBust:Date.now()});this.slothlet.debug("cache",{key:"DEBUG_MODE_CACHE_REBUILT_SUCCESSFULLY",moduleID,timestamp:Date.now()});return freshApi}}export{ApiCacheManager};