nodeconfigloder 1.0.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.
@@ -0,0 +1,396 @@
1
+ #!/usr/bin/env node
2
+ (function webpackUniversalModuleDefinition(root, factory) {
3
+ if(typeof exports === 'object' && typeof module === 'object')
4
+ module.exports = factory();
5
+ else if(typeof define === 'function' && define.amd)
6
+ define([], factory);
7
+ else if(typeof exports === 'object')
8
+ exports["configloder"] = factory();
9
+ else
10
+ root["configloder"] = factory();
11
+ })(global, () => {
12
+ return /******/ (() => { // webpackBootstrap
13
+ /******/ "use strict";
14
+ /******/ var __webpack_modules__ = ({
15
+
16
+ /***/ "crypto"
17
+ /*!*************************!*\
18
+ !*** external "crypto" ***!
19
+ \*************************/
20
+ (module) {
21
+
22
+ module.exports = require("crypto");
23
+
24
+ /***/ },
25
+
26
+ /***/ "events"
27
+ /*!*************************!*\
28
+ !*** external "events" ***!
29
+ \*************************/
30
+ (module) {
31
+
32
+ module.exports = require("events");
33
+
34
+ /***/ }
35
+
36
+ /******/ });
37
+ /************************************************************************/
38
+ /******/ // The module cache
39
+ /******/ var __webpack_module_cache__ = {};
40
+ /******/
41
+ /******/ // The require function
42
+ /******/ function __webpack_require__(moduleId) {
43
+ /******/ // Check if module is in cache
44
+ /******/ var cachedModule = __webpack_module_cache__[moduleId];
45
+ /******/ if (cachedModule !== undefined) {
46
+ /******/ return cachedModule.exports;
47
+ /******/ }
48
+ /******/ // Check if module exists (development only)
49
+ /******/ if (__webpack_modules__[moduleId] === undefined) {
50
+ /******/ var e = new Error("Cannot find module '" + moduleId + "'");
51
+ /******/ e.code = 'MODULE_NOT_FOUND';
52
+ /******/ throw e;
53
+ /******/ }
54
+ /******/ // Create a new module (and put it into the cache)
55
+ /******/ var module = __webpack_module_cache__[moduleId] = {
56
+ /******/ // no module.id needed
57
+ /******/ // no module.loaded needed
58
+ /******/ exports: {}
59
+ /******/ };
60
+ /******/
61
+ /******/ // Execute the module function
62
+ /******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
63
+ /******/
64
+ /******/ // Return the exports of the module
65
+ /******/ return module.exports;
66
+ /******/ }
67
+ /******/
68
+ /************************************************************************/
69
+ /******/ /* webpack/runtime/compat get default export */
70
+ /******/ (() => {
71
+ /******/ // getDefaultExport function for compatibility with non-harmony modules
72
+ /******/ __webpack_require__.n = (module) => {
73
+ /******/ var getter = module && module.__esModule ?
74
+ /******/ () => (module['default']) :
75
+ /******/ () => (module);
76
+ /******/ __webpack_require__.d(getter, { a: getter });
77
+ /******/ return getter;
78
+ /******/ };
79
+ /******/ })();
80
+ /******/
81
+ /******/ /* webpack/runtime/define property getters */
82
+ /******/ (() => {
83
+ /******/ // define getter functions for harmony exports
84
+ /******/ __webpack_require__.d = (exports, definition) => {
85
+ /******/ for(var key in definition) {
86
+ /******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
87
+ /******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
88
+ /******/ }
89
+ /******/ }
90
+ /******/ };
91
+ /******/ })();
92
+ /******/
93
+ /******/ /* webpack/runtime/hasOwnProperty shorthand */
94
+ /******/ (() => {
95
+ /******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
96
+ /******/ })();
97
+ /******/
98
+ /******/ /* webpack/runtime/make namespace object */
99
+ /******/ (() => {
100
+ /******/ // define __esModule on exports
101
+ /******/ __webpack_require__.r = (exports) => {
102
+ /******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
103
+ /******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
104
+ /******/ }
105
+ /******/ Object.defineProperty(exports, '__esModule', { value: true });
106
+ /******/ };
107
+ /******/ })();
108
+ /******/
109
+ /************************************************************************/
110
+ var __webpack_exports__ = {};
111
+ // This entry needs to be wrapped in an IIFE because it needs to be isolated against other modules in the chunk.
112
+ (() => {
113
+ /*!****************************************!*\
114
+ !*** ./configloder/FindDifferences.ts ***!
115
+ \****************************************/
116
+ __webpack_require__.r(__webpack_exports__);
117
+ /* harmony export */ __webpack_require__.d(__webpack_exports__, {
118
+ /* harmony export */ FindDifferences: () => (/* binding */ FindDifferences),
119
+ /* harmony export */ "default": () => (__WEBPACK_DEFAULT_EXPORT__)
120
+ /* harmony export */ });
121
+ /* harmony import */ var events__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! events */ "events");
122
+ /* harmony import */ var events__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(events__WEBPACK_IMPORTED_MODULE_0__);
123
+ /* harmony import */ var crypto__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! crypto */ "crypto");
124
+ /* harmony import */ var crypto__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(crypto__WEBPACK_IMPORTED_MODULE_1__);
125
+
126
+
127
+ /**
128
+ * 処理名: 差分検出エンジン
129
+ *
130
+ * 処理概要:
131
+ * ハッシュツリーを使用して2つのJSON構造の差分を効率的に検出する。
132
+ * EventEmitterを継承し、差分を'difference'イベントで通知
133
+ *
134
+ * 実装理由:
135
+ * 大規模な設定ファイルの変更検出において、全体比較ではなく
136
+ * ハッシュツリーの活用で計算量を最小化するため
137
+ */
138
+ class FindDifferences extends events__WEBPACK_IMPORTED_MODULE_0__.EventEmitter {
139
+ /**
140
+ * 処理名: コンストラクタ
141
+ *
142
+ * 処理概要:
143
+ * FindDifferencesインスタンスを初期化し、前回のハッシュツリーを空の状態で保持
144
+ *
145
+ * 実装理由:
146
+ * EventEmitterの初期化と、差分検出のための状態を保持するため
147
+ */
148
+ constructor() {
149
+ super();
150
+ this.previousHashTree = { __hash: '' };
151
+ }
152
+ /**
153
+ * 処理名: ハッシュ値計算
154
+ *
155
+ * 処理概要:
156
+ * 与えられた値のSHA256ハッシュを計算する
157
+ *
158
+ * 実装理由:
159
+ * オブジェクトの内容をコンパクトに表現し、変更検出の基盤とするため
160
+ * @param {string} value ハッシュ対象の値
161
+ * @returns {string} ハッシュ文字列
162
+ * @private
163
+ */
164
+ calculateHash(value) {
165
+ return crypto__WEBPACK_IMPORTED_MODULE_1__.createHash('sha256').update(value).digest('hex');
166
+ }
167
+ /**
168
+ * 処理名: ハッシュツリー構築
169
+ *
170
+ * 処理概要:
171
+ * JSONデータから再帰的にハッシュツリーを構築する。
172
+ * 各ノードには__hashキーで当該ノード以下の変更を示すハッシュを保持
173
+ *
174
+ * 実装理由:
175
+ * 差分検出時に__hashの比較だけで下層の探索を短縮するため
176
+ * @param {unknown} data 構築対象のデータ
177
+ * @returns {HashTreeNode} ハッシュツリー
178
+ * @private
179
+ */
180
+ buildHashTree(data) {
181
+ if (typeof data !== 'object' || data === null) {
182
+ // プリミティブ値の場合、そのままハッシュ化
183
+ return {
184
+ __hash: this.calculateHash(JSON.stringify(data)),
185
+ };
186
+ }
187
+ if (Array.isArray(data)) {
188
+ // 配列の場合、各要素のハッシュ値を結合してハッシュ化
189
+ const childHashes = data.map((item) => this.buildHashTree(item));
190
+ const combinedHash = this.calculateHash(childHashes.map((h) => h.__hash).join(''));
191
+ return {
192
+ __hash: combinedHash,
193
+ ...Object.fromEntries(childHashes.map((h, i) => [i.toString(), h])),
194
+ };
195
+ }
196
+ if (typeof data === 'object' && data.type) {
197
+ // type値を持っているobjectの場合、そのままハッシュ化
198
+ return {
199
+ __hash: this.calculateHash(JSON.stringify(data)),
200
+ };
201
+ }
202
+ // オブジェクトの場合、キーと値をソートしてハッシュ化
203
+ const childHashes = Object.keys(data)
204
+ .sort()
205
+ .reduce((tree, key) => {
206
+ tree[key] = this.buildHashTree(data[key]);
207
+ return tree;
208
+ }, {});
209
+ const combinedHash = this.calculateHash(Object.entries(childHashes)
210
+ .map(([key, value]) => `${key}:${value.__hash}`)
211
+ .join(''));
212
+ return { __hash: combinedHash, ...childHashes };
213
+ }
214
+ /**
215
+ * 処理名: 削除・更新キーの処理
216
+ * @param {HashTreeNode} hashTree ハッシュツリー
217
+ * @param {Record<string, unknown>} jsonData JSON データ
218
+ * @param {string} path パス
219
+ * @returns {void}
220
+ * @private
221
+ */
222
+ processExistingKeys(hashTree, jsonData, path) {
223
+ for (const key in hashTree) {
224
+ if (key === '__hash')
225
+ continue;
226
+ const currentPath = path ? `${path}.${key}` : key;
227
+ if (!(key in jsonData)) {
228
+ this.emit('difference', {
229
+ type: 'removed',
230
+ path: currentPath,
231
+ });
232
+ delete hashTree[key];
233
+ }
234
+ else {
235
+ this.processKeyValue(hashTree, jsonData, key, currentPath);
236
+ }
237
+ }
238
+ }
239
+ /**
240
+ * 処理名: キー値の処理
241
+ * @param {HashTreeNode} hashTree ハッシュツリー
242
+ * @param {Record<string, unknown>} jsonData JSON データ
243
+ * @param {string} key キー
244
+ * @param {string} currentPath 現在のパス
245
+ * @returns {void}
246
+ * @private
247
+ */
248
+ processKeyValue(hashTree, jsonData, key, currentPath) {
249
+ const jsonValue = jsonData[key];
250
+ if (typeof jsonValue === 'object' && jsonValue !== null) {
251
+ this.processObjectValue(hashTree, jsonValue, key, currentPath);
252
+ }
253
+ else {
254
+ this.processPrimitiveValue(hashTree, jsonValue, key, currentPath);
255
+ }
256
+ }
257
+ /**
258
+ * 処理名: オブジェクト値の処理
259
+ * @param {HashTreeNode} hashTree ハッシュツリー
260
+ * @param {Record<string, unknown>} jsonValue JSON 値
261
+ * @param {string} key キー
262
+ * @param {string} currentPath 現在のパス
263
+ * @returns {void}
264
+ * @private
265
+ */
266
+ processObjectValue(hashTree, jsonValue, key, currentPath) {
267
+ if (jsonValue.type) {
268
+ const newPrimitiveHash = this.calculateHash(JSON.stringify(jsonValue));
269
+ if (hashTree[key].__hash !== newPrimitiveHash) {
270
+ this.emit('difference', {
271
+ type: 'modified',
272
+ path: currentPath,
273
+ });
274
+ hashTree[key].__hash = newPrimitiveHash;
275
+ }
276
+ }
277
+ else {
278
+ if (!hashTree[key] || typeof hashTree[key] !== 'object') {
279
+ hashTree[key] = { __hash: '' };
280
+ }
281
+ this.findDifferencesAndUpdate(hashTree[key], jsonValue, currentPath);
282
+ }
283
+ }
284
+ /**
285
+ * 処理名: プリミティブ値の処理
286
+ * @param {HashTreeNode} hashTree ハッシュツリー
287
+ * @param {unknown} jsonValue JSON 値
288
+ * @param {string} key キー
289
+ * @param {string} currentPath 現在のパス
290
+ * @returns {void}
291
+ * @private
292
+ */
293
+ processPrimitiveValue(hashTree, jsonValue, key, currentPath) {
294
+ const newPrimitiveHash = this.calculateHash(JSON.stringify(jsonValue));
295
+ if (hashTree[key].__hash !== newPrimitiveHash) {
296
+ this.emit('difference', {
297
+ type: 'modified',
298
+ path: currentPath,
299
+ });
300
+ hashTree[key].__hash = newPrimitiveHash;
301
+ }
302
+ }
303
+ /**
304
+ * 処理名: 追加されたキーの処理
305
+ * @param {HashTreeNode} hashTree ハッシュツリー
306
+ * @param {Record<string, unknown>} jsonData JSON データ
307
+ * @param {string} path パス
308
+ * @returns {void}
309
+ * @private
310
+ */
311
+ processAddedKeys(hashTree, jsonData, path) {
312
+ for (const key in jsonData) {
313
+ const currentPath = path ? `${path}.${key}` : key;
314
+ if (!(key in hashTree)) {
315
+ this.emit('difference', {
316
+ type: 'added',
317
+ path: currentPath,
318
+ });
319
+ const jsonValue = jsonData[key];
320
+ hashTree[key] =
321
+ typeof jsonValue === 'object' && jsonValue !== null
322
+ ? this.buildHashTree(jsonValue)
323
+ : { __hash: this.calculateHash(JSON.stringify(jsonValue)) };
324
+ }
325
+ }
326
+ }
327
+ /**
328
+ * 処理名: 差分検出と更新
329
+ *
330
+ * 処理概要:
331
+ * 前回のハッシュツリーと新しいJSONデータを比較し、
332
+ * 追加・削除・更新されたパスを'difference'イベントで通知
333
+ *
334
+ * 実装理由:
335
+ * ハッシュ値の比較で不要な再帰を避け、変更箇所のみを検出するため
336
+ * @param {HashTreeNode} hashTree ハッシュツリー
337
+ * @param {unknown} jsonData JSON データ
338
+ * @param {string} path パス
339
+ * @returns {void}
340
+ * @private
341
+ */
342
+ findDifferencesAndUpdate(hashTree, jsonData, path = '') {
343
+ const newHash = this.calculateHash(JSON.stringify(jsonData));
344
+ if (hashTree.__hash === newHash) {
345
+ return;
346
+ }
347
+ const data = jsonData;
348
+ this.processExistingKeys(hashTree, data, path);
349
+ this.processAddedKeys(hashTree, data, path);
350
+ hashTree.__hash = this.calculateHash(Object.entries(hashTree)
351
+ .filter(([key]) => key !== '__hash')
352
+ .map(([key, value]) => `${key}:${value.__hash}`)
353
+ .join(''));
354
+ }
355
+ /**
356
+ * 処理名: ハッシュツリー初期化
357
+ *
358
+ * 処理概要:
359
+ * 初回読み込み時にハッシュツリーを構築し、前回状態として保持
360
+ *
361
+ * 実装理由:
362
+ * 次回の差分検出の基準となるベースラインを作成するため
363
+ * @param {unknown} jsonData 初回のJSONデータ
364
+ * @returns {HashTreeNode} 構築されたハッシュツリー
365
+ */
366
+ initialize(jsonData) {
367
+ return (this.previousHashTree = this.buildHashTree(jsonData)); // 初期ハッシュツリー構築
368
+ }
369
+ /**
370
+ * 処理名: 差分検出
371
+ *
372
+ * 処理概要:
373
+ * 新しいJSONデータと前回のハッシュツリーを比較し、
374
+ * 差分を検出して'difference'イベントで通知。
375
+ * 内部的にハッシュツリーを更新して次回検出に備える
376
+ *
377
+ * 実装理由:
378
+ * ファイル監視やリアルタイム変更検出のため、継続的に差分を検出するため
379
+ * @param {unknown} jsonData 新しいJSONデータ
380
+ * @returns {HashTreeNode} 更新上のハッシュツリー
381
+ */
382
+ detectChanges(jsonData) {
383
+ this.findDifferencesAndUpdate(this.previousHashTree, jsonData);
384
+ return this.previousHashTree;
385
+ }
386
+ }
387
+ /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (FindDifferences);
388
+
389
+ })();
390
+
391
+ __webpack_exports__ = __webpack_exports__["default"];
392
+ /******/ return __webpack_exports__;
393
+ /******/ })()
394
+ ;
395
+ });
396
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"finddifferences.bundle.js","mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD,O;;;;;;;;;;ACVA,mC;;;;;;;;;;ACAA,mC;;;;;;UCAA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WC5BA;WACA;WACA;WACA;WACA;WACA,iCAAiC,WAAW;WAC5C;WACA,E;;;;;WCPA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA,E;;;;;WCPA,wF;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D,E;;;;;;;;;;;;;;;;;;;ACNsC;AACL;AAkBjC;;;;;;;;;;GAUG;AACI,MAAM,eAAgB,SAAQ,gDAAY;IAG/C;;;;;;;;OAQG;IACH;QACE,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,gBAAgB,GAAG,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;IACzC,CAAC;IAED;;;;;;;;;;;OAWG;IACK,aAAa,CAAC,KAAa;QACjC,OAAO,8CAAiB,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IACjE,CAAC;IAED;;;;;;;;;;;;OAYG;IACK,aAAa,CAAC,IAAa;QACjC,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;YAC9C,uBAAuB;YACvB,OAAO;gBACL,MAAM,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;aACjD,CAAC;QACJ,CAAC;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;YACxB,4BAA4B;YAC5B,MAAM,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC;YACjE,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,CAAC;YACnF,OAAO;gBACL,MAAM,EAAE,YAAY;gBACpB,GAAG,MAAM,CAAC,WAAW,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC;aACpD,CAAC;QACpB,CAAC;QAED,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAK,IAAgC,CAAC,IAAI,EAAE,CAAC;YACvE,iCAAiC;YACjC,OAAO;gBACL,MAAM,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;aACjD,CAAC;QACJ,CAAC;QAED,4BAA4B;QAC5B,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC;aAClC,IAAI,EAAE;aACN,MAAM,CAAC,CAAC,IAAkC,EAAE,GAAG,EAAE,EAAE;YAClD,IAAI,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,aAAa,CAAE,IAAgC,CAAC,GAAG,CAAC,CAAC,CAAC;YACvE,OAAO,IAAI,CAAC;QACd,CAAC,EAAE,EAAE,CAAC,CAAC;QAET,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CACrC,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC;aACxB,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;aAC/C,IAAI,CAAC,EAAE,CAAC,CACZ,CAAC;QACF,OAAO,EAAE,MAAM,EAAE,YAAY,EAAE,GAAG,WAAW,EAAE,CAAC;IAClD,CAAC;IAED;;;;;;;OAOG;IACK,mBAAmB,CACzB,QAAsB,EACtB,QAAiC,EACjC,IAAY;QAEZ,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;YAC3B,IAAI,GAAG,KAAK,QAAQ;gBAAE,SAAS;YAC/B,MAAM,WAAW,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,IAAI,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC;YAElD,IAAI,CAAC,CAAC,GAAG,IAAI,QAAQ,CAAC,EAAE,CAAC;gBACvB,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;oBACtB,IAAI,EAAE,SAAS;oBACf,IAAI,EAAE,WAAW;iBACC,CAAC,CAAC;gBACtB,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC;YACvB,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,QAAQ,EAAE,GAAG,EAAE,WAAW,CAAC,CAAC;YAC7D,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;;;;;OAQG;IACK,eAAe,CACrB,QAAsB,EACtB,QAAiC,EACjC,GAAW,EACX,WAAmB;QAEnB,MAAM,SAAS,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;QAChC,IAAI,OAAO,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,IAAI,EAAE,CAAC;YACxD,IAAI,CAAC,kBAAkB,CAAC,QAAQ,EAAE,SAAoC,EAAE,GAAG,EAAE,WAAW,CAAC,CAAC;QAC5F,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,qBAAqB,CAAC,QAAQ,EAAE,SAAS,EAAE,GAAG,EAAE,WAAW,CAAC,CAAC;QACpE,CAAC;IACH,CAAC;IAED;;;;;;;;OAQG;IACK,kBAAkB,CACxB,QAAsB,EACtB,SAAkC,EAClC,GAAW,EACX,WAAmB;QAEnB,IAAI,SAAS,CAAC,IAAI,EAAE,CAAC;YACnB,MAAM,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC;YACvE,IAAK,QAAQ,CAAC,GAAG,CAAkB,CAAC,MAAM,KAAK,gBAAgB,EAAE,CAAC;gBAChE,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;oBACtB,IAAI,EAAE,UAAU;oBAChB,IAAI,EAAE,WAAW;iBACC,CAAC,CAAC;gBACrB,QAAQ,CAAC,GAAG,CAAkB,CAAC,MAAM,GAAG,gBAAgB,CAAC;YAC5D,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,OAAO,QAAQ,CAAC,GAAG,CAAC,KAAK,QAAQ,EAAE,CAAC;gBACxD,QAAQ,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;YACjC,CAAC;YACD,IAAI,CAAC,wBAAwB,CAC3B,QAAQ,CAAC,GAAG,CAAiB,EAC7B,SAAS,EACT,WAAW,CACZ,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;;;;;;;OAQG;IACK,qBAAqB,CAC3B,QAAsB,EACtB,SAAkB,EAClB,GAAW,EACX,WAAmB;QAEnB,MAAM,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC;QACvE,IAAK,QAAQ,CAAC,GAAG,CAAkB,CAAC,MAAM,KAAK,gBAAgB,EAAE,CAAC;YAChE,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;gBACtB,IAAI,EAAE,UAAU;gBAChB,IAAI,EAAE,WAAW;aACC,CAAC,CAAC;YACrB,QAAQ,CAAC,GAAG,CAAkB,CAAC,MAAM,GAAG,gBAAgB,CAAC;QAC5D,CAAC;IACH,CAAC;IAED;;;;;;;OAOG;IACK,gBAAgB,CACtB,QAAsB,EACtB,QAAiC,EACjC,IAAY;QAEZ,KAAK,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC;YAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,IAAI,IAAI,GAAG,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC;YAClD,IAAI,CAAC,CAAC,GAAG,IAAI,QAAQ,CAAC,EAAE,CAAC;gBACvB,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE;oBACtB,IAAI,EAAE,OAAO;oBACb,IAAI,EAAE,WAAW;iBACC,CAAC,CAAC;gBACtB,MAAM,SAAS,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC;gBAChC,QAAQ,CAAC,GAAG,CAAC;oBACX,OAAO,SAAS,KAAK,QAAQ,IAAI,SAAS,KAAK,IAAI;wBACjD,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC;wBAC/B,CAAC,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC;YAClE,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;;;;;;;;;;;OAcG;IACK,wBAAwB,CAC9B,QAAsB,EACtB,QAAiB,EACjB,IAAI,GAAG,EAAE;QAET,MAAM,OAAO,GAAG,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC;QAC7D,IAAI,QAAQ,CAAC,MAAM,KAAK,OAAO,EAAE,CAAC;YAChC,OAAO;QACT,CAAC;QAED,MAAM,IAAI,GAAG,QAAmC,CAAC;QACjD,IAAI,CAAC,mBAAmB,CAAC,QAAQ,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;QAC/C,IAAI,CAAC,gBAAgB,CAAC,QAAQ,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;QAE3C,QAAoC,CAAC,MAAM,GAAG,IAAI,CAAC,aAAa,CAC/D,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC;aACrB,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,GAAG,KAAK,QAAQ,CAAC;aACnC,GAAG,CAAC,CAAC,CAAC,GAAG,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,IAAK,KAAsB,CAAC,MAAM,EAAE,CAAC;aACjE,IAAI,CAAC,EAAE,CAAC,CACZ,CAAC;IACJ,CAAC;IAED;;;;;;;;;;OAUG;IACH,UAAU,CAAC,QAAiB;QAC1B,OAAO,CAAC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,cAAc;IAC/E,CAAC;IAED;;;;;;;;;;;;OAYG;IACH,aAAa,CAAC,QAAiB;QAC7B,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,gBAAgB,EAAE,QAAQ,CAAC,CAAC;QAC/D,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;CACF;AAED,iEAAe,eAAe,EAAC","sources":["webpack://configloder/webpack/universalModuleDefinition","webpack://configloder/external node-commonjs \"crypto\"","webpack://configloder/external node-commonjs \"events\"","webpack://configloder/webpack/bootstrap","webpack://configloder/webpack/runtime/compat get default export","webpack://configloder/webpack/runtime/define property getters","webpack://configloder/webpack/runtime/hasOwnProperty shorthand","webpack://configloder/webpack/runtime/make namespace object","webpack://configloder/./configloder/FindDifferences.ts"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"configloder\"] = factory();\n\telse\n\t\troot[\"configloder\"] = factory();\n})(global, () => {\nreturn ","module.exports = require(\"crypto\");","module.exports = require(\"events\");","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Check if module exists (development only)\n\tif (__webpack_modules__[moduleId] === undefined) {\n\t\tvar e = new Error(\"Cannot find module '\" + moduleId + \"'\");\n\t\te.code = 'MODULE_NOT_FOUND';\n\t\tthrow e;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","import { EventEmitter } from 'events';\r\nimport * as crypto from 'crypto';\r\n\r\n/**\r\n * ハッシュツリーのノード型\r\n */\r\ninterface HashTreeNode {\r\n  __hash: string;\r\n  [key: string]: unknown;\r\n}\r\n\r\n/**\r\n * 差分イベントの型\r\n */\r\ninterface DifferenceEvent {\r\n  type: 'added' | 'removed' | 'modified';\r\n  path: string;\r\n}\r\n\r\n/**\r\n * 処理名: 差分検出エンジン\r\n *\r\n * 処理概要:\r\n * ハッシュツリーを使用して2つのJSON構造の差分を効率的に検出する。\r\n * EventEmitterを継承し、差分を'difference'イベントで通知\r\n *\r\n * 実装理由:\r\n * 大規模な設定ファイルの変更検出において、全体比較ではなく\r\n * ハッシュツリーの活用で計算量を最小化するため\r\n */\r\nexport class FindDifferences extends EventEmitter {\r\n  private previousHashTree: HashTreeNode;\r\n\r\n  /**\r\n   * 処理名: コンストラクタ\r\n   *\r\n   * 処理概要:\r\n   * FindDifferencesインスタンスを初期化し、前回のハッシュツリーを空の状態で保持\r\n   *\r\n   * 実装理由:\r\n   * EventEmitterの初期化と、差分検出のための状態を保持するため\r\n   */\r\n  constructor() {\r\n    super();\r\n    this.previousHashTree = { __hash: '' };\r\n  }\r\n\r\n  /**\r\n   * 処理名: ハッシュ値計算\r\n   *\r\n   * 処理概要:\r\n   * 与えられた値のSHA256ハッシュを計算する\r\n   *\r\n   * 実装理由:\r\n   * オブジェクトの内容をコンパクトに表現し、変更検出の基盤とするため\r\n   * @param {string} value ハッシュ対象の値\r\n   * @returns {string} ハッシュ文字列\r\n   * @private\r\n   */\r\n  private calculateHash(value: string): string {\r\n    return crypto.createHash('sha256').update(value).digest('hex');\r\n  }\r\n\r\n  /**\r\n   * 処理名: ハッシュツリー構築\r\n   *\r\n   * 処理概要:\r\n   * JSONデータから再帰的にハッシュツリーを構築する。\r\n   * 各ノードには__hashキーで当該ノード以下の変更を示すハッシュを保持\r\n   *\r\n   * 実装理由:\r\n   * 差分検出時に__hashの比較だけで下層の探索を短縮するため\r\n   * @param {unknown} data 構築対象のデータ\r\n   * @returns {HashTreeNode} ハッシュツリー\r\n   * @private\r\n   */\r\n  private buildHashTree(data: unknown): HashTreeNode {\r\n    if (typeof data !== 'object' || data === null) {\r\n      // プリミティブ値の場合、そのままハッシュ化\r\n      return {\r\n        __hash: this.calculateHash(JSON.stringify(data)),\r\n      };\r\n    }\r\n\r\n    if (Array.isArray(data)) {\r\n      // 配列の場合、各要素のハッシュ値を結合してハッシュ化\r\n      const childHashes = data.map((item) => this.buildHashTree(item));\r\n      const combinedHash = this.calculateHash(childHashes.map((h) => h.__hash).join(''));\r\n      return {\r\n        __hash: combinedHash,\r\n        ...Object.fromEntries(childHashes.map((h, i) => [i.toString(), h])),\r\n      } as HashTreeNode;\r\n    }\r\n\r\n    if (typeof data === 'object' && (data as Record<string, unknown>).type) {\r\n      // type値を持っているobjectの場合、そのままハッシュ化\r\n      return {\r\n        __hash: this.calculateHash(JSON.stringify(data)),\r\n      };\r\n    }\r\n\r\n    // オブジェクトの場合、キーと値をソートしてハッシュ化\r\n    const childHashes = Object.keys(data)\r\n      .sort()\r\n      .reduce((tree: Record<string, HashTreeNode>, key) => {\r\n        tree[key] = this.buildHashTree((data as Record<string, unknown>)[key]);\r\n        return tree;\r\n      }, {});\r\n\r\n    const combinedHash = this.calculateHash(\r\n      Object.entries(childHashes)\r\n        .map(([key, value]) => `${key}:${value.__hash}`)\r\n        .join('')\r\n    );\r\n    return { __hash: combinedHash, ...childHashes };\r\n  }\r\n\r\n  /**\r\n   * 処理名: 削除・更新キーの処理\r\n   * @param {HashTreeNode} hashTree ハッシュツリー\r\n   * @param {Record<string, unknown>} jsonData JSON データ\r\n   * @param {string} path パス\r\n   * @returns {void}\r\n   * @private\r\n   */\r\n  private processExistingKeys(\r\n    hashTree: HashTreeNode,\r\n    jsonData: Record<string, unknown>,\r\n    path: string\r\n  ): void {\r\n    for (const key in hashTree) {\r\n      if (key === '__hash') continue;\r\n      const currentPath = path ? `${path}.${key}` : key;\r\n\r\n      if (!(key in jsonData)) {\r\n        this.emit('difference', {\r\n          type: 'removed',\r\n          path: currentPath,\r\n        } as DifferenceEvent);\r\n        delete hashTree[key];\r\n      } else {\r\n        this.processKeyValue(hashTree, jsonData, key, currentPath);\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * 処理名: キー値の処理\r\n   * @param {HashTreeNode} hashTree ハッシュツリー\r\n   * @param {Record<string, unknown>} jsonData JSON データ\r\n   * @param {string} key キー\r\n   * @param {string} currentPath 現在のパス\r\n   * @returns {void}\r\n   * @private\r\n   */\r\n  private processKeyValue(\r\n    hashTree: HashTreeNode,\r\n    jsonData: Record<string, unknown>,\r\n    key: string,\r\n    currentPath: string\r\n  ): void {\r\n    const jsonValue = jsonData[key];\r\n    if (typeof jsonValue === 'object' && jsonValue !== null) {\r\n      this.processObjectValue(hashTree, jsonValue as Record<string, unknown>, key, currentPath);\r\n    } else {\r\n      this.processPrimitiveValue(hashTree, jsonValue, key, currentPath);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * 処理名: オブジェクト値の処理\r\n   * @param {HashTreeNode} hashTree ハッシュツリー\r\n   * @param {Record<string, unknown>} jsonValue JSON 値\r\n   * @param {string} key キー\r\n   * @param {string} currentPath 現在のパス\r\n   * @returns {void}\r\n   * @private\r\n   */\r\n  private processObjectValue(\r\n    hashTree: HashTreeNode,\r\n    jsonValue: Record<string, unknown>,\r\n    key: string,\r\n    currentPath: string\r\n  ): void {\r\n    if (jsonValue.type) {\r\n      const newPrimitiveHash = this.calculateHash(JSON.stringify(jsonValue));\r\n      if ((hashTree[key] as HashTreeNode).__hash !== newPrimitiveHash) {\r\n        this.emit('difference', {\r\n          type: 'modified',\r\n          path: currentPath,\r\n        } as DifferenceEvent);\r\n        (hashTree[key] as HashTreeNode).__hash = newPrimitiveHash;\r\n      }\r\n    } else {\r\n      if (!hashTree[key] || typeof hashTree[key] !== 'object') {\r\n        hashTree[key] = { __hash: '' };\r\n      }\r\n      this.findDifferencesAndUpdate(\r\n        hashTree[key] as HashTreeNode,\r\n        jsonValue,\r\n        currentPath\r\n      );\r\n    }\r\n  }\r\n\r\n  /**\r\n   * 処理名: プリミティブ値の処理\r\n   * @param {HashTreeNode} hashTree ハッシュツリー\r\n   * @param {unknown} jsonValue JSON 値\r\n   * @param {string} key キー\r\n   * @param {string} currentPath 現在のパス\r\n   * @returns {void}\r\n   * @private\r\n   */\r\n  private processPrimitiveValue(\r\n    hashTree: HashTreeNode,\r\n    jsonValue: unknown,\r\n    key: string,\r\n    currentPath: string\r\n  ): void {\r\n    const newPrimitiveHash = this.calculateHash(JSON.stringify(jsonValue));\r\n    if ((hashTree[key] as HashTreeNode).__hash !== newPrimitiveHash) {\r\n      this.emit('difference', {\r\n        type: 'modified',\r\n        path: currentPath,\r\n      } as DifferenceEvent);\r\n      (hashTree[key] as HashTreeNode).__hash = newPrimitiveHash;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * 処理名: 追加されたキーの処理\r\n   * @param {HashTreeNode} hashTree ハッシュツリー\r\n   * @param {Record<string, unknown>} jsonData JSON データ\r\n   * @param {string} path パス\r\n   * @returns {void}\r\n   * @private\r\n   */\r\n  private processAddedKeys(\r\n    hashTree: HashTreeNode,\r\n    jsonData: Record<string, unknown>,\r\n    path: string\r\n  ): void {\r\n    for (const key in jsonData) {\r\n      const currentPath = path ? `${path}.${key}` : key;\r\n      if (!(key in hashTree)) {\r\n        this.emit('difference', {\r\n          type: 'added',\r\n          path: currentPath,\r\n        } as DifferenceEvent);\r\n        const jsonValue = jsonData[key];\r\n        hashTree[key] =\r\n          typeof jsonValue === 'object' && jsonValue !== null\r\n            ? this.buildHashTree(jsonValue)\r\n            : { __hash: this.calculateHash(JSON.stringify(jsonValue)) };\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * 処理名: 差分検出と更新\r\n   *\r\n   * 処理概要:\r\n   * 前回のハッシュツリーと新しいJSONデータを比較し、\r\n   * 追加・削除・更新されたパスを'difference'イベントで通知\r\n   *\r\n   * 実装理由:\r\n   * ハッシュ値の比較で不要な再帰を避け、変更箇所のみを検出するため\r\n   * @param {HashTreeNode} hashTree ハッシュツリー\r\n   * @param {unknown} jsonData JSON データ\r\n   * @param {string} path パス\r\n   * @returns {void}\r\n   * @private\r\n   */\r\n  private findDifferencesAndUpdate(\r\n    hashTree: HashTreeNode,\r\n    jsonData: unknown,\r\n    path = ''\r\n  ): void {\r\n    const newHash = this.calculateHash(JSON.stringify(jsonData));\r\n    if (hashTree.__hash === newHash) {\r\n      return;\r\n    }\r\n\r\n    const data = jsonData as Record<string, unknown>;\r\n    this.processExistingKeys(hashTree, data, path);\r\n    this.processAddedKeys(hashTree, data, path);\r\n\r\n    (hashTree as Record<string, unknown>).__hash = this.calculateHash(\r\n      Object.entries(hashTree)\r\n        .filter(([key]) => key !== '__hash')\r\n        .map(([key, value]) => `${key}:${(value as HashTreeNode).__hash}`)\r\n        .join('')\r\n    );\r\n  }\r\n\r\n  /**\r\n   * 処理名: ハッシュツリー初期化\r\n   *\r\n   * 処理概要:\r\n   * 初回読み込み時にハッシュツリーを構築し、前回状態として保持\r\n   *\r\n   * 実装理由:\r\n   * 次回の差分検出の基準となるベースラインを作成するため\r\n   * @param {unknown} jsonData 初回のJSONデータ\r\n   * @returns {HashTreeNode} 構築されたハッシュツリー\r\n   */\r\n  initialize(jsonData: unknown): HashTreeNode {\r\n    return (this.previousHashTree = this.buildHashTree(jsonData)); // 初期ハッシュツリー構築\r\n  }\r\n\r\n  /**\r\n   * 処理名: 差分検出\r\n   *\r\n   * 処理概要:\r\n   * 新しいJSONデータと前回のハッシュツリーを比較し、\r\n   * 差分を検出して'difference'イベントで通知。\r\n   * 内部的にハッシュツリーを更新して次回検出に備える\r\n   *\r\n   * 実装理由:\r\n   * ファイル監視やリアルタイム変更検出のため、継続的に差分を検出するため\r\n   * @param {unknown} jsonData 新しいJSONデータ\r\n   * @returns {HashTreeNode} 更新上のハッシュツリー\r\n   */\r\n  detectChanges(jsonData: unknown): HashTreeNode {\r\n    this.findDifferencesAndUpdate(this.previousHashTree, jsonData);\r\n    return this.previousHashTree;\r\n  }\r\n}\r\n\r\nexport default FindDifferences;\r\n"],"names":[],"ignoreList":[],"sourceRoot":""}