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.
- package/LICENSE +21 -0
- package/README.md +504 -0
- package/dist/configloder.bundle.js +8053 -0
- package/dist/conftool.bundle.js +11645 -0
- package/dist/finddifferences.bundle.js +396 -0
- package/dist/serializer.bundle.js +4248 -0
- package/dist/src/configloder/ConfigLoder.d.ts +92 -0
- package/dist/src/configloder/ConfigLoder.d.ts.map +1 -0
- package/dist/src/configloder/FindDifferences.d.ts +152 -0
- package/dist/src/configloder/FindDifferences.d.ts.map +1 -0
- package/dist/src/configloder/Serializer.d.ts +108 -0
- package/dist/src/configloder/Serializer.d.ts.map +1 -0
- package/dist/src/tools/index.d.ts +2 -0
- package/dist/src/tools/index.d.ts.map +1 -0
- package/dist/test/unit/configloder/ConfigLoder.test.d.ts +11 -0
- package/dist/test/unit/configloder/ConfigLoder.test.d.ts.map +1 -0
- package/dist/test/unit/configloder/FindDifferences.test.d.ts +11 -0
- package/dist/test/unit/configloder/FindDifferences.test.d.ts.map +1 -0
- package/dist/test/unit/configloder/Serializer.test.d.ts +11 -0
- package/dist/test/unit/configloder/Serializer.test.d.ts.map +1 -0
- package/package.json +69 -0
|
@@ -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":"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","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":""}
|