@rosen-bridge/abstract-extractor 2.1.0 → 2.1.2-82c56984

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 (165) hide show
  1. package/CHANGELOG.md +30 -0
  2. package/dist/ergo/{AbstractErgoExtractor.d.ts → abstractErgoExtractor.d.ts} +3 -3
  3. package/dist/ergo/abstractErgoExtractor.d.ts.map +1 -0
  4. package/dist/ergo/{AbstractErgoExtractor.js → abstractErgoExtractor.js} +3 -3
  5. package/dist/ergo/{AbstractErgoExtractorAction.d.ts → abstractErgoExtractorAction.d.ts} +6 -4
  6. package/dist/ergo/abstractErgoExtractorAction.d.ts.map +1 -0
  7. package/dist/ergo/abstractErgoExtractorAction.js +219 -0
  8. package/dist/ergo/{AbstractErgoExtractorEntity.d.ts → abstractErgoExtractorEntity.d.ts} +1 -1
  9. package/dist/ergo/{AbstractErgoExtractorEntity.d.ts.map → abstractErgoExtractorEntity.d.ts.map} +1 -1
  10. package/dist/ergo/{AbstractErgoExtractorEntity.js → abstractErgoExtractorEntity.js} +1 -1
  11. package/dist/ergo/index.d.ts +6 -6
  12. package/dist/ergo/index.js +7 -7
  13. package/dist/ergo/initializable/{AbstractInitializable.d.ts → abstractInitializable.d.ts} +4 -4
  14. package/dist/ergo/initializable/{AbstractInitializable.d.ts.map → abstractInitializable.d.ts.map} +1 -1
  15. package/dist/ergo/initializable/abstractInitializable.js +162 -0
  16. package/dist/ergo/initializable/{AbstractInitializableAction.d.ts → abstractInitializableAction.d.ts} +4 -4
  17. package/dist/ergo/initializable/abstractInitializableAction.d.ts.map +1 -0
  18. package/dist/ergo/initializable/{AbstractInitializableAction.js → abstractInitializableAction.js} +2 -2
  19. package/dist/ergo/initializable/index.d.ts +2 -2
  20. package/dist/ergo/initializable/index.js +3 -3
  21. package/dist/{lib/ergo/network/AbstractNetwork.d.ts → ergo/network/abstractNetwork.d.ts} +1 -1
  22. package/dist/ergo/network/{AbstractNetwork.d.ts.map → abstractNetwork.d.ts.map} +1 -1
  23. package/dist/ergo/network/{AbstractNetwork.js → abstractNetwork.js} +1 -1
  24. package/dist/{lib/ergo/network/ExplorerNetwork.d.ts → ergo/network/explorerNetwork.d.ts} +2 -2
  25. package/dist/ergo/network/explorerNetwork.d.ts.map +1 -0
  26. package/dist/ergo/network/{ExplorerNetwork.js → explorerNetwork.js} +3 -3
  27. package/dist/ergo/network/{NodeNetwork.d.ts → nodeNetwork.d.ts} +2 -2
  28. package/dist/ergo/network/nodeNetwork.d.ts.map +1 -0
  29. package/dist/ergo/network/nodeNetwork.js +131 -0
  30. package/dist/ergo/utils.d.ts.map +1 -1
  31. package/dist/ergo/utils.js +1 -1
  32. package/package.json +24 -16
  33. package/.eslintignore +0 -1
  34. package/dist/ergo/AbstractErgoExtractor.d.ts.map +0 -1
  35. package/dist/ergo/AbstractErgoExtractorAction.d.ts.map +0 -1
  36. package/dist/ergo/AbstractErgoExtractorAction.js +0 -222
  37. package/dist/ergo/initializable/AbstractInitializable.js +0 -163
  38. package/dist/ergo/initializable/AbstractInitializableAction.d.ts.map +0 -1
  39. package/dist/ergo/network/AbstractNetwork.d.ts +0 -26
  40. package/dist/ergo/network/ExplorerNetwork.d.ts +0 -74
  41. package/dist/ergo/network/ExplorerNetwork.d.ts.map +0 -1
  42. package/dist/ergo/network/NodeNetwork.d.ts.map +0 -1
  43. package/dist/ergo/network/NodeNetwork.js +0 -131
  44. package/dist/lib/abstractExtractor.d.ts +0 -25
  45. package/dist/lib/abstractExtractor.d.ts.map +0 -1
  46. package/dist/lib/abstractExtractor.js +0 -3
  47. package/dist/lib/constants.d.ts +0 -4
  48. package/dist/lib/constants.d.ts.map +0 -1
  49. package/dist/lib/constants.js +0 -4
  50. package/dist/lib/ergo/AbstractErgoExtractor.d.ts +0 -80
  51. package/dist/lib/ergo/AbstractErgoExtractor.d.ts.map +0 -1
  52. package/dist/lib/ergo/AbstractErgoExtractor.js +0 -142
  53. package/dist/lib/ergo/AbstractErgoExtractorAction.d.ts +0 -92
  54. package/dist/lib/ergo/AbstractErgoExtractorAction.d.ts.map +0 -1
  55. package/dist/lib/ergo/AbstractErgoExtractorAction.js +0 -222
  56. package/dist/lib/ergo/AbstractErgoExtractorEntity.d.ts +0 -11
  57. package/dist/lib/ergo/AbstractErgoExtractorEntity.d.ts.map +0 -1
  58. package/dist/lib/ergo/AbstractErgoExtractorEntity.js +0 -57
  59. package/dist/lib/ergo/index.d.ts +0 -10
  60. package/dist/lib/ergo/index.d.ts.map +0 -1
  61. package/dist/lib/ergo/index.js +0 -10
  62. package/dist/lib/ergo/initializable/AbstractInitializable.d.ts +0 -48
  63. package/dist/lib/ergo/initializable/AbstractInitializable.d.ts.map +0 -1
  64. package/dist/lib/ergo/initializable/AbstractInitializable.js +0 -163
  65. package/dist/lib/ergo/initializable/AbstractInitializableAction.d.ts +0 -14
  66. package/dist/lib/ergo/initializable/AbstractInitializableAction.d.ts.map +0 -1
  67. package/dist/lib/ergo/initializable/AbstractInitializableAction.js +0 -16
  68. package/dist/lib/ergo/initializable/index.d.ts +0 -3
  69. package/dist/lib/ergo/initializable/index.d.ts.map +0 -1
  70. package/dist/lib/ergo/initializable/index.js +0 -3
  71. package/dist/lib/ergo/interfaces.d.ts +0 -47
  72. package/dist/lib/ergo/interfaces.d.ts.map +0 -1
  73. package/dist/lib/ergo/interfaces.js +0 -8
  74. package/dist/lib/ergo/network/AbstractNetwork.d.ts.map +0 -1
  75. package/dist/lib/ergo/network/AbstractNetwork.js +0 -3
  76. package/dist/lib/ergo/network/ExplorerNetwork.d.ts.map +0 -1
  77. package/dist/lib/ergo/network/ExplorerNetwork.js +0 -185
  78. package/dist/lib/ergo/network/NodeNetwork.d.ts +0 -60
  79. package/dist/lib/ergo/network/NodeNetwork.d.ts.map +0 -1
  80. package/dist/lib/ergo/network/NodeNetwork.js +0 -131
  81. package/dist/lib/ergo/utils.d.ts +0 -8
  82. package/dist/lib/ergo/utils.d.ts.map +0 -1
  83. package/dist/lib/ergo/utils.js +0 -16
  84. package/dist/lib/index.d.ts +0 -4
  85. package/dist/lib/index.d.ts.map +0 -1
  86. package/dist/lib/index.js +0 -4
  87. package/dist/tests/AbstractErgoExtractor.mock.d.ts +0 -11
  88. package/dist/tests/AbstractErgoExtractor.mock.d.ts.map +0 -1
  89. package/dist/tests/AbstractErgoExtractor.mock.js +0 -11
  90. package/dist/tests/AbstractErgoExtractor.spec.d.ts +0 -2
  91. package/dist/tests/AbstractErgoExtractor.spec.d.ts.map +0 -1
  92. package/dist/tests/AbstractErgoExtractor.spec.js +0 -240
  93. package/dist/tests/AbstractErgoExtractorAction.mock.d.ts +0 -15
  94. package/dist/tests/AbstractErgoExtractorAction.mock.d.ts.map +0 -1
  95. package/dist/tests/AbstractErgoExtractorAction.mock.js +0 -27
  96. package/dist/tests/AbstractErgoExtractorAction.spec.d.ts +0 -2
  97. package/dist/tests/AbstractErgoExtractorAction.spec.d.ts.map +0 -1
  98. package/dist/tests/AbstractErgoExtractorAction.spec.js +0 -221
  99. package/dist/tests/initializable/AbstractInitializable.mock.d.ts +0 -61
  100. package/dist/tests/initializable/AbstractInitializable.mock.d.ts.map +0 -1
  101. package/dist/tests/initializable/AbstractInitializable.mock.js +0 -18
  102. package/dist/tests/initializable/AbstractInitializable.spec.d.ts +0 -2
  103. package/dist/tests/initializable/AbstractInitializable.spec.d.ts.map +0 -1
  104. package/dist/tests/initializable/AbstractInitializable.spec.js +0 -299
  105. package/dist/tests/initializable/AbstractInitializableAction.mock.d.ts +0 -15
  106. package/dist/tests/initializable/AbstractInitializableAction.mock.d.ts.map +0 -1
  107. package/dist/tests/initializable/AbstractInitializableAction.mock.js +0 -27
  108. package/dist/tests/initializable/AbstractInitializableAction.spec.d.ts +0 -2
  109. package/dist/tests/initializable/AbstractInitializableAction.spec.d.ts.map +0 -1
  110. package/dist/tests/initializable/AbstractInitializableAction.spec.js +0 -54
  111. package/dist/tests/initializable/testData.d.ts +0 -94
  112. package/dist/tests/initializable/testData.d.ts.map +0 -1
  113. package/dist/tests/initializable/testData.js +0 -235
  114. package/dist/tests/network/ExplorerNetwork.spec.d.ts +0 -2
  115. package/dist/tests/network/ExplorerNetwork.spec.d.ts.map +0 -1
  116. package/dist/tests/network/ExplorerNetwork.spec.js +0 -61
  117. package/dist/tests/network/NodeNetwork.spec.d.ts +0 -2
  118. package/dist/tests/network/NodeNetwork.spec.d.ts.map +0 -1
  119. package/dist/tests/network/NodeNetwork.spec.js +0 -48
  120. package/dist/tests/network/testData.d.ts +0 -752
  121. package/dist/tests/network/testData.d.ts.map +0 -1
  122. package/dist/tests/network/testData.js +0 -1401
  123. package/dist/tests/testData.d.ts +0 -32
  124. package/dist/tests/testData.d.ts.map +0 -1
  125. package/dist/tests/testData.js +0 -121
  126. package/dist/tests/testUtils.d.ts +0 -9
  127. package/dist/tests/testUtils.d.ts.map +0 -1
  128. package/dist/tests/testUtils.js +0 -31
  129. package/dist/tsconfig.tsbuildinfo +0 -1
  130. package/dist/vitest.config.d.ts +0 -3
  131. package/dist/vitest.config.d.ts.map +0 -1
  132. package/dist/vitest.config.js +0 -18
  133. package/lib/abstractExtractor.ts +0 -31
  134. package/lib/constants.ts +0 -3
  135. package/lib/ergo/AbstractErgoExtractor.ts +0 -227
  136. package/lib/ergo/AbstractErgoExtractorAction.ts +0 -319
  137. package/lib/ergo/AbstractErgoExtractorEntity.ts +0 -32
  138. package/lib/ergo/index.ts +0 -9
  139. package/lib/ergo/initializable/AbstractInitializable.ts +0 -228
  140. package/lib/ergo/initializable/AbstractInitializableAction.ts +0 -33
  141. package/lib/ergo/initializable/index.ts +0 -2
  142. package/lib/ergo/interfaces.ts +0 -51
  143. package/lib/ergo/network/AbstractNetwork.ts +0 -29
  144. package/lib/ergo/network/ExplorerNetwork.ts +0 -237
  145. package/lib/ergo/network/NodeNetwork.ts +0 -169
  146. package/lib/ergo/utils.ts +0 -15
  147. package/lib/index.ts +0 -3
  148. package/tests/AbstractErgoExtractor.mock.ts +0 -33
  149. package/tests/AbstractErgoExtractor.spec.ts +0 -284
  150. package/tests/AbstractErgoExtractorAction.mock.ts +0 -45
  151. package/tests/AbstractErgoExtractorAction.spec.ts +0 -268
  152. package/tests/initializable/AbstractInitializable.mock.ts +0 -44
  153. package/tests/initializable/AbstractInitializable.spec.ts +0 -386
  154. package/tests/initializable/AbstractInitializableAction.mock.ts +0 -45
  155. package/tests/initializable/AbstractInitializableAction.spec.ts +0 -64
  156. package/tests/initializable/testData.ts +0 -283
  157. package/tests/network/ExplorerNetwork.spec.ts +0 -73
  158. package/tests/network/NodeNetwork.spec.ts +0 -56
  159. package/tests/network/testData.ts +0 -1708
  160. package/tests/testData.ts +0 -140
  161. package/tests/testUtils.ts +0 -22
  162. package/tsconfig.build.json +0 -9
  163. package/tsconfig.build.tsbuildinfo +0 -1
  164. package/tsconfig.json +0 -9
  165. package/vitest.config.ts +0 -18
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=AbstractErgoExtractor.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"AbstractErgoExtractor.spec.d.ts","sourceRoot":"","sources":["../../tests/AbstractErgoExtractor.spec.ts"],"names":[],"mappings":""}
@@ -1,240 +0,0 @@
1
- import { CallbackType, } from '../lib';
2
- import { block, extractedData, tx } from './testData';
3
- import { MockedErgoExtractor } from './AbstractErgoExtractor.mock';
4
- describe('AbstractErgoExtractor', () => {
5
- describe('processTransactions', () => {
6
- /**
7
- * @target processTransactions should initialize extractor with specified id and insert status into db
8
- * @dependencies
9
- * @scenario
10
- * - mock extractor
11
- * - mock `hasData` to return true for one box
12
- * - spy `extractBoxData` and `storeBoxes`
13
- * - run test (call `processTransactions`)
14
- * @expected
15
- * - to call `extractBoxData` for the specific box and all input extensions
16
- * - to insert the extracted box to database
17
- * - to return true when total procedure is successful
18
- * - to trigger `INSERT` callbacks with correct data
19
- */
20
- it('should process boxes with data and insert data into database', async () => {
21
- const extractor = new MockedErgoExtractor();
22
- const triggerCallbacks = vitest.fn();
23
- extractor['triggerCallbacks'] = triggerCallbacks;
24
- extractor.hasData = (box) => {
25
- if (box.boxId == tx.outputs[0].boxId)
26
- return true;
27
- return false;
28
- };
29
- const extractSpy = vitest.fn().mockReturnValue(extractedData);
30
- extractor.extractBoxData = extractSpy;
31
- const storeSpy = vitest.fn().mockResolvedValue(true);
32
- const spendSpy = vitest.fn().mockResolvedValue([]);
33
- extractor['actions'] = {
34
- storeBoxes: storeSpy,
35
- spendBoxes: spendSpy,
36
- };
37
- const result = await extractor.processTransactions([tx], block);
38
- expect(extractSpy).toBeCalledTimes(1);
39
- expect(extractSpy).toBeCalledWith(tx.outputs[0], [tx.inputs[0].extension, {}], {});
40
- expect(storeSpy).toBeCalledWith([extractedData], block, 'Test');
41
- expect(result).toEqual(true);
42
- expect(triggerCallbacks).toBeCalledWith(CallbackType.Insert, [
43
- extractedData,
44
- ]);
45
- });
46
- /**
47
- * @target processTransactions should extract spending information of all input boxes
48
- * @dependencies
49
- * @scenario
50
- * - mock extractor (hasData returns false as default)
51
- * - spy `extractBoxData`, `storeBoxes` and `spendBoxes`
52
- * - run test (call `processTransactions`)
53
- * @expected
54
- * - not to call `extractBoxData` and `storeBoxes` when there is not any box with data
55
- * - to extractor spend info of input boxes and call `spendBoxes`
56
- * - to return true when total procedure is successful
57
- * - to trigger `SPEND` callbacks with correct data
58
- */
59
- it('should extract spending information of all input boxes', async () => {
60
- const extractor = new MockedErgoExtractor();
61
- const triggerCallbacks = vitest.fn();
62
- extractor['triggerCallbacks'] = triggerCallbacks;
63
- const extractSpy = vitest.fn();
64
- extractor.extractBoxData = extractSpy;
65
- const storeSpy = vitest.fn().mockResolvedValue(true);
66
- const spendSpy = vitest
67
- .fn()
68
- .mockResolvedValue([
69
- { boxId: tx.inputs[0].boxId },
70
- { boxId: tx.inputs[1].boxId },
71
- ]);
72
- extractor['actions'] = {
73
- storeBoxes: storeSpy,
74
- spendBoxes: spendSpy,
75
- };
76
- const result = await extractor.processTransactions([tx], block);
77
- expect(extractSpy).not.toBeCalled();
78
- expect(storeSpy).not.toBeCalled();
79
- expect(spendSpy).toBeCalledWith([
80
- { boxId: tx.inputs[0].boxId, txId: tx.id, index: 1 },
81
- { boxId: tx.inputs[1].boxId, txId: tx.id, index: 2 },
82
- ], block, 'Test');
83
- expect(result).toEqual(true);
84
- expect(triggerCallbacks).toBeCalledWith(CallbackType.Spend, [
85
- { boxId: tx.inputs[0].boxId },
86
- { boxId: tx.inputs[1].boxId },
87
- ]);
88
- });
89
- /**
90
- * @target processTransactions should return false if data insertion fails
91
- * @dependencies
92
- * @scenario
93
- * - mock extractor
94
- * - mock `hasData` to return true for one box
95
- * - spy `extractBoxData` and `storeBoxes`
96
- * - run test (call `processTransactions`)
97
- * @expected
98
- * - to return false when `insertBoxes` returns false
99
- * - not to call `spendBoxes` if data insertion fails
100
- */
101
- it('should return false if data insertion fails', async () => {
102
- const extractor = new MockedErgoExtractor();
103
- extractor.hasData = (box) => {
104
- if (box.boxId == tx.outputs[0].boxId)
105
- return true;
106
- return false;
107
- };
108
- const extractSpy = vitest.fn().mockReturnValue(extractedData);
109
- extractor.extractBoxData = extractSpy;
110
- const storeSpy = vitest.fn().mockResolvedValue(false);
111
- const spendSpy = vitest.fn();
112
- extractor['actions'] = {
113
- storeBoxes: storeSpy,
114
- spendBoxes: spendSpy,
115
- };
116
- const result = await extractor.processTransactions([tx], block);
117
- expect(result).toEqual(false);
118
- expect(spendSpy).not.toBeCalled();
119
- });
120
- });
121
- describe('forkBlock', () => {
122
- /**
123
- * @target forkBlock should remove all data extracted from the specified block
124
- * @dependencies
125
- * @scenario
126
- * - mock extractor
127
- * - spy `deleteBlockBoxes`
128
- * - run test (call `forkBlock`)
129
- * @expected
130
- * - to call `deleteBlockBoxes` for the specific box
131
- * - to trigger `DELETE` callbacks for the deleted box
132
- * - to trigger `UPDATE` callbacks for the spent box
133
- */
134
- it('should remove all data extracted from the specified block', async () => {
135
- const extractor = new MockedErgoExtractor();
136
- const removeSpy = vitest.fn().mockResolvedValue({
137
- deletedData: [{ boxId: 'box1' }],
138
- updatedData: [{ boxId: 'box2' }],
139
- });
140
- extractor['actions'] = {
141
- deleteBlockBoxes: removeSpy,
142
- };
143
- const triggerCallbackSpy = vitest.fn().mockClear();
144
- extractor['triggerCallbacks'] = triggerCallbackSpy;
145
- await extractor.forkBlock(block.hash);
146
- expect(removeSpy).toBeCalledWith(block.hash, 'Test');
147
- expect(triggerCallbackSpy).toBeCalledWith(CallbackType.Delete, [
148
- { boxId: 'box1' },
149
- ]);
150
- expect(triggerCallbackSpy).toBeCalledWith(CallbackType.Update, [
151
- { boxId: 'box2' },
152
- ]);
153
- });
154
- });
155
- describe('hook', () => {
156
- /**
157
- * @target hook should hook a new callback on insert with the new id
158
- * @dependencies
159
- * @scenario
160
- * - mock extractor
161
- * - mock a callback for insert
162
- * - run test (call `hook`)
163
- * @expected
164
- * - hook the callback with the specified id
165
- * - return true
166
- */
167
- it('should hook a new callback on insert with the new id', async () => {
168
- const extractor = new MockedErgoExtractor();
169
- const insertCallback = vitest.fn();
170
- const id = await extractor.hook(CallbackType.Insert, insertCallback);
171
- expect(extractor['callbacks'][CallbackType.Insert]).toEqual(new Map().set(id, insertCallback));
172
- });
173
- });
174
- describe('unhook', () => {
175
- /**
176
- * @target unhook should unhook the callback on insert with the specified id
177
- * @dependencies
178
- * @scenario
179
- * - mock extractor
180
- * - mock a callback for insert
181
- * - hook the callback
182
- * - run test (call `unhook`)
183
- * @expected
184
- * - unhook the callback with the specified id
185
- * - return true
186
- */
187
- it('should unhook the callback on insert with the specified id', async () => {
188
- const extractor = new MockedErgoExtractor();
189
- const insertCallback = vitest.fn();
190
- const id = await extractor.hook(CallbackType.Insert, insertCallback);
191
- const result = await extractor.unhook(CallbackType.Insert, id);
192
- expect(result).toBeTruthy();
193
- expect(extractor['callbacks'][CallbackType.Insert].get(id)).toEqual(undefined);
194
- });
195
- /**
196
- * @target unhook should not unhook callbacks with the same id on other types
197
- * @dependencies
198
- * @scenario
199
- * - mock extractor
200
- * - mock two callbacks for insert
201
- * - hook the first callback
202
- * - run test (call `unhook` with the same id for second callback)
203
- * @expected
204
- * - not to change hooked callbacks when the callback with the id
205
- * doesn't exists on the specified type
206
- * - return false
207
- */
208
- it('should not unhook callbacks with the same id on other types', async () => {
209
- const extractor = new MockedErgoExtractor();
210
- const insertCallback = vitest.fn();
211
- const id = await extractor.hook(CallbackType.Insert, insertCallback);
212
- const result = await extractor.unhook(CallbackType.Update, id);
213
- expect(result).toBeFalsy();
214
- expect(extractor['callbacks'][CallbackType.Insert].get(id)).toEqual(insertCallback);
215
- expect(extractor['callbacks'][CallbackType.Update].get(id)).toEqual(undefined);
216
- });
217
- });
218
- describe('triggerCallbacks', () => {
219
- /**
220
- * @target triggerCallbacks should trigger all callbacks hooked on a type
221
- * @dependencies
222
- * @scenario
223
- * - mock extractor
224
- * - mock a callback for insert
225
- * - hook the callback
226
- * - run test (call `triggerCallbacks` for insert type)
227
- * @expected
228
- * - trigger all callbacks with the specified id
229
- */
230
- it('should trigger all callbacks hooked on a type', async () => {
231
- const extractor = new MockedErgoExtractor();
232
- const insertCallback = vitest.fn();
233
- await extractor.hook(CallbackType.Insert, insertCallback);
234
- const insertedData = [{ boxId: 'boxId', serialized: 'serialized' }];
235
- await extractor['triggerCallbacks'](CallbackType.Insert, insertedData);
236
- expect(insertCallback).toBeCalledWith(insertedData);
237
- });
238
- });
239
- });
240
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AbstractErgoExtractor.spec.js","sourceRoot":"","sources":["../../tests/AbstractErgoExtractor.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { V1 } from '@rosen-clients/ergo-explorer';\nimport { OutputBox } from '@rosen-bridge/scanner-interfaces';\n\nimport {\n  AbstractBoxData,\n  AbstractErgoExtractorAction,\n  CallbackType,\n  AbstractErgoExtractorEntity,\n} from '../lib';\nimport { block, extractedData, tx } from './testData';\nimport { MockedErgoExtractor } from './AbstractErgoExtractor.mock';\n\ndescribe('AbstractErgoExtractor', () => {\n  describe('processTransactions', () => {\n    /**\n     * @target processTransactions should initialize extractor with specified id and insert status into db\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - mock `hasData` to return true for one box\n     * - spy `extractBoxData` and `storeBoxes`\n     * - run test (call `processTransactions`)\n     * @expected\n     * - to call `extractBoxData` for the specific box and all input extensions\n     * - to insert the extracted box to database\n     * - to return true when total procedure is successful\n     * - to trigger `INSERT` callbacks with correct data\n     */\n    it('should process boxes with data and insert data into database', async () => {\n      const extractor = new MockedErgoExtractor();\n      const triggerCallbacks = vitest.fn();\n      extractor['triggerCallbacks'] = triggerCallbacks;\n      extractor.hasData = (box: V1.OutputInfo | OutputBox) => {\n        if (box.boxId == tx.outputs[0].boxId) return true;\n        return false;\n      };\n      const extractSpy = vitest.fn().mockReturnValue(extractedData);\n      extractor.extractBoxData = extractSpy;\n      const storeSpy = vitest.fn().mockResolvedValue(true);\n      const spendSpy = vitest.fn().mockResolvedValue([]);\n      extractor['actions'] = {\n        storeBoxes: storeSpy,\n        spendBoxes: spendSpy,\n      } as unknown as AbstractErgoExtractorAction<\n        AbstractBoxData,\n        AbstractErgoExtractorEntity\n      >;\n      const result = await extractor.processTransactions([tx], block);\n\n      expect(extractSpy).toBeCalledTimes(1);\n      expect(extractSpy).toBeCalledWith(\n        tx.outputs[0],\n        [tx.inputs[0].extension, {}],\n        {},\n      );\n      expect(storeSpy).toBeCalledWith([extractedData], block, 'Test');\n      expect(result).toEqual(true);\n      expect(triggerCallbacks).toBeCalledWith(CallbackType.Insert, [\n        extractedData,\n      ]);\n    });\n\n    /**\n     * @target processTransactions should extract spending information of all input boxes\n     * @dependencies\n     * @scenario\n     * - mock extractor (hasData returns false as default)\n     * - spy `extractBoxData`, `storeBoxes` and `spendBoxes`\n     * - run test (call `processTransactions`)\n     * @expected\n     * - not to call `extractBoxData` and `storeBoxes` when there is not any box with data\n     * - to extractor spend info of input boxes and call `spendBoxes`\n     * - to return true when total procedure is successful\n     * - to trigger `SPEND` callbacks with correct data\n     */\n    it('should extract spending information of all input boxes', async () => {\n      const extractor = new MockedErgoExtractor();\n      const triggerCallbacks = vitest.fn();\n      extractor['triggerCallbacks'] = triggerCallbacks;\n      const extractSpy = vitest.fn();\n      extractor.extractBoxData = extractSpy;\n      const storeSpy = vitest.fn().mockResolvedValue(true);\n      const spendSpy = vitest\n        .fn()\n        .mockResolvedValue([\n          { boxId: tx.inputs[0].boxId },\n          { boxId: tx.inputs[1].boxId },\n        ]);\n      extractor['actions'] = {\n        storeBoxes: storeSpy,\n        spendBoxes: spendSpy,\n      } as unknown as AbstractErgoExtractorAction<\n        AbstractBoxData,\n        AbstractErgoExtractorEntity\n      >;\n      const result = await extractor.processTransactions([tx], block);\n\n      expect(extractSpy).not.toBeCalled();\n      expect(storeSpy).not.toBeCalled();\n      expect(spendSpy).toBeCalledWith(\n        [\n          { boxId: tx.inputs[0].boxId, txId: tx.id, index: 1 },\n          { boxId: tx.inputs[1].boxId, txId: tx.id, index: 2 },\n        ],\n        block,\n        'Test',\n      );\n      expect(result).toEqual(true);\n      expect(triggerCallbacks).toBeCalledWith(CallbackType.Spend, [\n        { boxId: tx.inputs[0].boxId },\n        { boxId: tx.inputs[1].boxId },\n      ]);\n    });\n\n    /**\n     * @target processTransactions should return false if data insertion fails\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - mock `hasData` to return true for one box\n     * - spy `extractBoxData` and `storeBoxes`\n     * - run test (call `processTransactions`)\n     * @expected\n     * - to return false when `insertBoxes` returns false\n     * - not to call `spendBoxes` if data insertion fails\n     */\n    it('should return false if data insertion fails', async () => {\n      const extractor = new MockedErgoExtractor();\n      extractor.hasData = (box: V1.OutputInfo | OutputBox) => {\n        if (box.boxId == tx.outputs[0].boxId) return true;\n        return false;\n      };\n      const extractSpy = vitest.fn().mockReturnValue(extractedData);\n      extractor.extractBoxData = extractSpy;\n      const storeSpy = vitest.fn().mockResolvedValue(false);\n      const spendSpy = vitest.fn();\n      extractor['actions'] = {\n        storeBoxes: storeSpy,\n        spendBoxes: spendSpy,\n      } as unknown as AbstractErgoExtractorAction<\n        AbstractBoxData,\n        AbstractErgoExtractorEntity\n      >;\n      const result = await extractor.processTransactions([tx], block);\n\n      expect(result).toEqual(false);\n      expect(spendSpy).not.toBeCalled();\n    });\n  });\n\n  describe('forkBlock', () => {\n    /**\n     * @target forkBlock should remove all data extracted from the specified block\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - spy `deleteBlockBoxes`\n     * - run test (call `forkBlock`)\n     * @expected\n     * - to call `deleteBlockBoxes` for the specific box\n     * - to trigger `DELETE` callbacks for the deleted box\n     * - to trigger `UPDATE` callbacks for the spent box\n     */\n    it('should remove all data extracted from the specified block', async () => {\n      const extractor = new MockedErgoExtractor();\n      const removeSpy = vitest.fn().mockResolvedValue({\n        deletedData: [{ boxId: 'box1' }],\n        updatedData: [{ boxId: 'box2' }],\n      });\n      extractor['actions'] = {\n        deleteBlockBoxes: removeSpy,\n      } as unknown as AbstractErgoExtractorAction<\n        AbstractBoxData,\n        AbstractErgoExtractorEntity\n      >;\n      const triggerCallbackSpy = vitest.fn().mockClear();\n      extractor['triggerCallbacks'] = triggerCallbackSpy;\n      await extractor.forkBlock(block.hash);\n      expect(removeSpy).toBeCalledWith(block.hash, 'Test');\n      expect(triggerCallbackSpy).toBeCalledWith(CallbackType.Delete, [\n        { boxId: 'box1' },\n      ]);\n      expect(triggerCallbackSpy).toBeCalledWith(CallbackType.Update, [\n        { boxId: 'box2' },\n      ]);\n    });\n  });\n\n  describe('hook', () => {\n    /**\n     * @target hook should hook a new callback on insert with the new id\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - mock a callback for insert\n     * - run test (call `hook`)\n     * @expected\n     * - hook the callback with the specified id\n     * - return true\n     */\n    it('should hook a new callback on insert with the new id', async () => {\n      const extractor = new MockedErgoExtractor();\n      const insertCallback = vitest.fn();\n      const id = await extractor.hook(CallbackType.Insert, insertCallback);\n      expect(extractor['callbacks'][CallbackType.Insert]).toEqual(\n        new Map().set(id, insertCallback),\n      );\n    });\n  });\n\n  describe('unhook', () => {\n    /**\n     * @target unhook should unhook the callback on insert with the specified id\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - mock a callback for insert\n     * - hook the callback\n     * - run test (call `unhook`)\n     * @expected\n     * - unhook the callback with the specified id\n     * - return true\n     */\n    it('should unhook the callback on insert with the specified id', async () => {\n      const extractor = new MockedErgoExtractor();\n      const insertCallback = vitest.fn();\n      const id = await extractor.hook(CallbackType.Insert, insertCallback);\n      const result = await extractor.unhook(CallbackType.Insert, id);\n      expect(result).toBeTruthy();\n      expect(extractor['callbacks'][CallbackType.Insert].get(id)).toEqual(\n        undefined,\n      );\n    });\n\n    /**\n     * @target unhook should not unhook callbacks with the same id on other types\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - mock two callbacks for insert\n     * - hook the first callback\n     * - run test (call `unhook` with the same id for second callback)\n     * @expected\n     * - not to change hooked callbacks when the callback with the id\n     * doesn't exists on the specified type\n     * - return false\n     */\n    it('should not unhook callbacks with the same id on other types', async () => {\n      const extractor = new MockedErgoExtractor();\n      const insertCallback = vitest.fn();\n      const id = await extractor.hook(CallbackType.Insert, insertCallback);\n      const result = await extractor.unhook(CallbackType.Update, id);\n      expect(result).toBeFalsy();\n      expect(extractor['callbacks'][CallbackType.Insert].get(id)).toEqual(\n        insertCallback,\n      );\n      expect(extractor['callbacks'][CallbackType.Update].get(id)).toEqual(\n        undefined,\n      );\n    });\n  });\n\n  describe('triggerCallbacks', () => {\n    /**\n     * @target triggerCallbacks should trigger all callbacks hooked on a type\n     * @dependencies\n     * @scenario\n     * - mock extractor\n     * - mock a callback for insert\n     * - hook the callback\n     * - run test (call `triggerCallbacks` for insert type)\n     * @expected\n     * - trigger all callbacks with the specified id\n     */\n    it('should trigger all callbacks hooked on a type', async () => {\n      const extractor = new MockedErgoExtractor();\n      const insertCallback = vitest.fn();\n      await extractor.hook(CallbackType.Insert, insertCallback);\n      const insertedData = [{ boxId: 'boxId', serialized: 'serialized' }];\n      await extractor['triggerCallbacks'](CallbackType.Insert, insertedData);\n      expect(insertCallback).toBeCalledWith(insertedData);\n    });\n  });\n});\n"]}
@@ -1,15 +0,0 @@
1
- import { DataSource } from '@rosen-bridge/extended-typeorm';
2
- import { BlockInfo } from '@rosen-bridge/scanner-interfaces';
3
- import { AbstractErgoExtractorAction, AbstractErgoExtractorEntity, AbstractBoxData } from '../lib';
4
- export declare class TestErgoExtractorAction extends AbstractErgoExtractorAction<AbstractBoxData, AbstractErgoExtractorEntity> {
5
- constructor(dataSource: DataSource);
6
- /**
7
- * create the test database entity from data and block information
8
- */
9
- createEntity: (boxes: AbstractBoxData[], block: BlockInfo, extractor: string) => Omit<AbstractErgoExtractorEntity, "id">[];
10
- /**
11
- * convert the database entity back to raw data
12
- */
13
- convertEntityToData: (entities: AbstractErgoExtractorEntity[]) => AbstractBoxData[];
14
- }
15
- //# sourceMappingURL=AbstractErgoExtractorAction.mock.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"AbstractErgoExtractorAction.mock.d.ts","sourceRoot":"","sources":["../../tests/AbstractErgoExtractorAction.mock.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,gCAAgC,CAAC;AAE5D,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;AAE7D,OAAO,EACL,2BAA2B,EAC3B,2BAA2B,EAC3B,eAAe,EAChB,MAAM,QAAQ,CAAC;AAGhB,qBAAa,uBAAwB,SAAQ,2BAA2B,CACtE,eAAe,EACf,2BAA2B,CAC5B;gBACa,UAAU,EAAE,UAAU;IAIlC;;OAEG;IACH,YAAY,GACV,OAAO,eAAe,EAAE,EACxB,OAAO,SAAS,EAChB,WAAW,MAAM,KAChB,IAAI,CAAC,2BAA2B,EAAE,IAAI,CAAC,EAAE,CAQ1C;IAEF;;OAEG;IACH,mBAAmB,GACjB,UAAU,2BAA2B,EAAE,KACtC,eAAe,EAAE,CAElB;CACH"}
@@ -1,27 +0,0 @@
1
- import { pick } from 'lodash-es';
2
- import { AbstractErgoExtractorAction, } from '../lib';
3
- import { TestEntity } from './testUtils';
4
- export class TestErgoExtractorAction extends AbstractErgoExtractorAction {
5
- constructor(dataSource) {
6
- super(dataSource, TestEntity);
7
- }
8
- /**
9
- * create the test database entity from data and block information
10
- */
11
- createEntity = (boxes, block, extractor) => {
12
- return boxes.map((box) => ({
13
- boxId: box.boxId,
14
- block: block.hash,
15
- height: block.height,
16
- serialized: box.serialized,
17
- extractor: extractor,
18
- }));
19
- };
20
- /**
21
- * convert the database entity back to raw data
22
- */
23
- convertEntityToData = (entities) => {
24
- return entities.map((data) => pick(data, ['boxId', 'serialized']));
25
- };
26
- }
27
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQWJzdHJhY3RFcmdvRXh0cmFjdG9yQWN0aW9uLm1vY2suanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi90ZXN0cy9BYnN0cmFjdEVyZ29FeHRyYWN0b3JBY3Rpb24ubW9jay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsSUFBSSxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBR2pDLE9BQU8sRUFDTCwyQkFBMkIsR0FHNUIsTUFBTSxRQUFRLENBQUM7QUFDaEIsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUV6QyxNQUFNLE9BQU8sdUJBQXdCLFNBQVEsMkJBRzVDO0lBQ0MsWUFBWSxVQUFzQjtRQUNoQyxLQUFLLENBQUMsVUFBVSxFQUFFLFVBQVUsQ0FBQyxDQUFDO0lBQ2hDLENBQUM7SUFFRDs7T0FFRztJQUNILFlBQVksR0FBRyxDQUNiLEtBQXdCLEVBQ3hCLEtBQWdCLEVBQ2hCLFNBQWlCLEVBQzBCLEVBQUU7UUFDN0MsT0FBTyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsR0FBRyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBQ3pCLEtBQUssRUFBRSxHQUFHLENBQUMsS0FBSztZQUNoQixLQUFLLEVBQUUsS0FBSyxDQUFDLElBQUk7WUFDakIsTUFBTSxFQUFFLEtBQUssQ0FBQyxNQUFNO1lBQ3BCLFVBQVUsRUFBRSxHQUFHLENBQUMsVUFBVTtZQUMxQixTQUFTLEVBQUUsU0FBUztTQUNyQixDQUFDLENBQUMsQ0FBQztJQUNOLENBQUMsQ0FBQztJQUVGOztPQUVHO0lBQ0gsbUJBQW1CLEdBQUcsQ0FDcEIsUUFBdUMsRUFDcEIsRUFBRTtRQUNyQixPQUFPLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsQ0FBQyxPQUFPLEVBQUUsWUFBWSxDQUFDLENBQUMsQ0FBQyxDQUFDO0lBQ3JFLENBQUMsQ0FBQztDQUNIIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGF0YVNvdXJjZSB9IGZyb20gJ0Byb3Nlbi1icmlkZ2UvZXh0ZW5kZWQtdHlwZW9ybSc7XG5pbXBvcnQgeyBwaWNrIH0gZnJvbSAnbG9kYXNoLWVzJztcbmltcG9ydCB7IEJsb2NrSW5mbyB9IGZyb20gJ0Byb3Nlbi1icmlkZ2Uvc2Nhbm5lci1pbnRlcmZhY2VzJztcblxuaW1wb3J0IHtcbiAgQWJzdHJhY3RFcmdvRXh0cmFjdG9yQWN0aW9uLFxuICBBYnN0cmFjdEVyZ29FeHRyYWN0b3JFbnRpdHksXG4gIEFic3RyYWN0Qm94RGF0YSxcbn0gZnJvbSAnLi4vbGliJztcbmltcG9ydCB7IFRlc3RFbnRpdHkgfSBmcm9tICcuL3Rlc3RVdGlscyc7XG5cbmV4cG9ydCBjbGFzcyBUZXN0RXJnb0V4dHJhY3RvckFjdGlvbiBleHRlbmRzIEFic3RyYWN0RXJnb0V4dHJhY3RvckFjdGlvbjxcbiAgQWJzdHJhY3RCb3hEYXRhLFxuICBBYnN0cmFjdEVyZ29FeHRyYWN0b3JFbnRpdHlcbj4ge1xuICBjb25zdHJ1Y3RvcihkYXRhU291cmNlOiBEYXRhU291cmNlKSB7XG4gICAgc3VwZXIoZGF0YVNvdXJjZSwgVGVzdEVudGl0eSk7XG4gIH1cblxuICAvKipcbiAgICogY3JlYXRlIHRoZSB0ZXN0IGRhdGFiYXNlIGVudGl0eSBmcm9tIGRhdGEgYW5kIGJsb2NrIGluZm9ybWF0aW9uXG4gICAqL1xuICBjcmVhdGVFbnRpdHkgPSAoXG4gICAgYm94ZXM6IEFic3RyYWN0Qm94RGF0YVtdLFxuICAgIGJsb2NrOiBCbG9ja0luZm8sXG4gICAgZXh0cmFjdG9yOiBzdHJpbmcsXG4gICk6IE9taXQ8QWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5LCAnaWQnPltdID0+IHtcbiAgICByZXR1cm4gYm94ZXMubWFwKChib3gpID0+ICh7XG4gICAgICBib3hJZDogYm94LmJveElkLFxuICAgICAgYmxvY2s6IGJsb2NrLmhhc2gsXG4gICAgICBoZWlnaHQ6IGJsb2NrLmhlaWdodCxcbiAgICAgIHNlcmlhbGl6ZWQ6IGJveC5zZXJpYWxpemVkLFxuICAgICAgZXh0cmFjdG9yOiBleHRyYWN0b3IsXG4gICAgfSkpO1xuICB9O1xuXG4gIC8qKlxuICAgKiBjb252ZXJ0IHRoZSBkYXRhYmFzZSBlbnRpdHkgYmFjayB0byByYXcgZGF0YVxuICAgKi9cbiAgY29udmVydEVudGl0eVRvRGF0YSA9IChcbiAgICBlbnRpdGllczogQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5W10sXG4gICk6IEFic3RyYWN0Qm94RGF0YVtdID0+IHtcbiAgICByZXR1cm4gZW50aXRpZXMubWFwKChkYXRhKSA9PiBwaWNrKGRhdGEsIFsnYm94SWQnLCAnc2VyaWFsaXplZCddKSk7XG4gIH07XG59XG4iXX0=
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=AbstractErgoExtractorAction.spec.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"AbstractErgoExtractorAction.spec.d.ts","sourceRoot":"","sources":["../../tests/AbstractErgoExtractorAction.spec.ts"],"names":[],"mappings":""}
@@ -1,221 +0,0 @@
1
- import { pick } from 'lodash-es';
2
- import { TestErgoExtractorAction } from './AbstractErgoExtractorAction.mock';
3
- import { createDatabase, TestEntity } from './testUtils';
4
- import { block, block2, sampleEntities } from './testData';
5
- describe('AbstractErgoExtractorAction', () => {
6
- let dataSource;
7
- let action;
8
- let repository;
9
- beforeEach(async () => {
10
- dataSource = await createDatabase();
11
- action = new TestErgoExtractorAction(dataSource);
12
- repository = dataSource.getRepository(TestEntity);
13
- });
14
- describe('storeBoxes', () => {
15
- /**
16
- * @target storeBoxes should save the passed box entities to database
17
- * @dependencies
18
- * @scenario
19
- * - run test (call `storeBoxes` with 2 new boxes)
20
- * @expected
21
- * - to save 2 new entities
22
- * - to return 2 inserted entity data
23
- */
24
- it(`should save the passed box entities to database`, async () => {
25
- const result = await action.storeBoxes(sampleEntities.slice(0, 2), block, 'extractor1');
26
- const [rows, rowsCount] = await repository.findAndCount();
27
- expect(rowsCount).toEqual(2);
28
- expect(rows[0]).toMatchObject({
29
- ...sampleEntities[0],
30
- extractor: 'extractor1',
31
- block: block.hash,
32
- height: block.height,
33
- spendBlock: null,
34
- spendHeight: null,
35
- });
36
- expect(rows[1]).toMatchObject({
37
- ...sampleEntities[1],
38
- extractor: 'extractor1',
39
- block: block.hash,
40
- height: block.height,
41
- spendBlock: null,
42
- spendHeight: null,
43
- });
44
- expect(result).toEqual(true);
45
- });
46
- /**
47
- * @target storeBoxes should correctly save boxes with different extractors
48
- * @dependencies
49
- * @scenario
50
- * - insert 2 boxes belonging to first-extractor
51
- * - run test (call `storeBoxes` with same boxes for the second-extractor)
52
- * @expected
53
- * - to save 2 new entities belonging to second-extractor
54
- * - to return 2 new inserted entity data
55
- */
56
- it(`should correctly save boxes with different extractors`, async () => {
57
- await repository.insert([
58
- {
59
- ...sampleEntities[0],
60
- extractor: 'first-extractor',
61
- block: '1',
62
- height: 1,
63
- },
64
- {
65
- ...sampleEntities[1],
66
- extractor: 'first-extractor',
67
- block: '1',
68
- height: 1,
69
- },
70
- ]);
71
- const result = await action.storeBoxes([sampleEntities[0], sampleEntities[1]], block, 'second-extractor');
72
- const [insertedRows] = await repository.findAndCount();
73
- expect(insertedRows[2]).toMatchObject({
74
- ...sampleEntities[0],
75
- extractor: 'second-extractor',
76
- block: 'hash',
77
- height: block.height,
78
- spendBlock: null,
79
- spendHeight: null,
80
- });
81
- expect(insertedRows[3]).toMatchObject({
82
- ...sampleEntities[1],
83
- extractor: 'second-extractor',
84
- block: 'hash',
85
- height: block.height,
86
- spendBlock: null,
87
- spendHeight: null,
88
- });
89
- expect(result).toEqual(true);
90
- });
91
- /**
92
- * @target storeBoxes should update boxes correctly
93
- * @dependencies
94
- * @scenario
95
- * - insert 2 boxes
96
- * - run test (call `storeBoxes` with the same boxes and updated info)
97
- * @expected
98
- * - to update the existing box in database
99
- * - to return updated box id
100
- */
101
- it(`storeBoxes should update boxes correctly`, async () => {
102
- await repository.insert([
103
- {
104
- ...sampleEntities[0],
105
- extractor: 'extractor',
106
- block: '1',
107
- height: 1,
108
- },
109
- {
110
- ...sampleEntities[1],
111
- extractor: 'extractor',
112
- block: '1',
113
- height: 1,
114
- },
115
- ]);
116
- const result = await action.storeBoxes([
117
- {
118
- ...sampleEntities[0],
119
- serialized: 'updatedBoxSerialized',
120
- },
121
- ], block, 'extractor');
122
- const [secondInsertRows, secondInsertRowsCount] = await repository.findAndCount();
123
- expect(secondInsertRowsCount).toEqual(2);
124
- expect(secondInsertRows[0]).toMatchObject({
125
- ...sampleEntities[0],
126
- extractor: 'extractor',
127
- serialized: 'updatedBoxSerialized',
128
- block: block.hash,
129
- height: block.height,
130
- spendBlock: null,
131
- spendHeight: null,
132
- });
133
- expect(result).toEqual(true);
134
- });
135
- });
136
- describe('spendBoxes', () => {
137
- /**
138
- * @target spendBoxes should set spendBlock and spendHeight for a set of boxes
139
- * @dependencies
140
- * @scenario
141
- * - insert two boxes
142
- * - mock spending information for the first box
143
- * - run test (call `spendBoxes`)
144
- * @expected
145
- * - spend the first box
146
- * - return boxId and serialized of spent box
147
- */
148
- it(`should set spendBlock and spendHeight for a set of boxes`, async () => {
149
- await action.storeBoxes(sampleEntities.slice(0, 2), block, 'extractor1');
150
- const spendBlock = { ...block, hash: 'spendHash', height: 10006016 };
151
- const spendInfos = [
152
- { txId: 'txId', boxId: sampleEntities[0].boxId, index: 0 },
153
- ];
154
- const spentBoxIds = await action.spendBoxes(spendInfos, spendBlock, 'extractor1');
155
- const spentBoxes = await repository.findOneBy({
156
- boxId: sampleEntities[0].boxId,
157
- extractor: 'extractor1',
158
- });
159
- expect(spentBoxes).toMatchObject({
160
- ...sampleEntities[0],
161
- block: block.hash,
162
- height: block.height,
163
- extractor: 'extractor1',
164
- spendBlock: spendBlock.hash,
165
- spendHeight: spendBlock.height,
166
- });
167
- expect(spentBoxIds).toEqual([pick(sampleEntities[0], ['boxId'])]);
168
- });
169
- });
170
- describe('deleteBlockBoxes', () => {
171
- /**
172
- * @target deleteBlockBoxes should delete the boxes created in the specified block
173
- * @dependencies
174
- * @scenario
175
- * - insert four boxes created in two different blocks
176
- * - run test(call `deleteBlockBoxes` to delete block2)
177
- * @expected
178
- * - to delete the two boxes created in block2
179
- * - to return the deleted entity data
180
- */
181
- it(`should delete the boxes created in the specified block`, async () => {
182
- await action.storeBoxes(sampleEntities.slice(0, 2), block, 'extractor1');
183
- await action.storeBoxes(sampleEntities.slice(2), block2, 'extractor1');
184
- const result = await action.deleteBlockBoxes(block2.hash, 'extractor1');
185
- const [rows, rowsCount] = await repository.findAndCount();
186
- expect(rowsCount).toEqual(2);
187
- expect(rows.map((row) => row.boxId)).not.toContain(sampleEntities.slice(2).map((box) => box.boxId));
188
- expect(result).toEqual({
189
- deletedData: action.convertEntityToData(sampleEntities.slice(2)),
190
- updatedData: [],
191
- });
192
- });
193
- /**
194
- * @target deleteBlockBoxes should update the boxes spent in the specified block
195
- * @dependencies
196
- * @scenario
197
- * - insert four boxes created in a block
198
- * - spend one of the in the block2
199
- * - run test(call `deleteBlockBoxes` to delete block2)
200
- * @expected
201
- * - to update the box spent in block2
202
- * - to return the updated entity boxId and serialized
203
- */
204
- it(`should update the boxes spent in the specified block`, async () => {
205
- await action.storeBoxes(sampleEntities, block, 'extractor1');
206
- const spendInfos = [
207
- { txId: 'txId', boxId: sampleEntities[0].boxId, index: 0 },
208
- ];
209
- await action.spendBoxes(spendInfos, block2, 'extractor1');
210
- const result = await action.deleteBlockBoxes(block2.hash, 'extractor1');
211
- const [rows, rowsCount] = await repository.findAndCount();
212
- expect(rowsCount).toEqual(4);
213
- expect(rows.map((row) => row.boxId)).not.toContain(sampleEntities.slice(2).map((box) => box.boxId));
214
- expect(result).toEqual({
215
- deletedData: [],
216
- updatedData: [pick(sampleEntities[0], ['boxId'])],
217
- });
218
- });
219
- });
220
- });
221
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AbstractErgoExtractorAction.spec.js","sourceRoot":"","sources":["../../tests/AbstractErgoExtractorAction.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource, Repository } from '@rosen-bridge/extended-typeorm';\nimport { pick } from 'lodash-es';\n\nimport { TestErgoExtractorAction } from './AbstractErgoExtractorAction.mock';\nimport { createDatabase, TestEntity } from './testUtils';\nimport { block, block2, sampleEntities } from './testData';\nimport { SpendInfo } from '../lib';\n\ndescribe('AbstractErgoExtractorAction', () => {\n  let dataSource: DataSource;\n  let action: TestErgoExtractorAction;\n  let repository: Repository<TestEntity>;\n  beforeEach(async () => {\n    dataSource = await createDatabase();\n    action = new TestErgoExtractorAction(dataSource);\n    repository = dataSource.getRepository(TestEntity);\n  });\n\n  describe('storeBoxes', () => {\n    /**\n     * @target storeBoxes should save the passed box entities to database\n     * @dependencies\n     * @scenario\n     * - run test (call `storeBoxes` with 2 new boxes)\n     * @expected\n     * - to save 2 new entities\n     * - to return 2 inserted entity data\n     */\n    it(`should save the passed box entities to database`, async () => {\n      const result = await action.storeBoxes(\n        sampleEntities.slice(0, 2),\n        block,\n        'extractor1',\n      );\n\n      const [rows, rowsCount] = await repository.findAndCount();\n\n      expect(rowsCount).toEqual(2);\n      expect(rows[0]).toMatchObject({\n        ...sampleEntities[0],\n        extractor: 'extractor1',\n        block: block.hash,\n        height: block.height,\n        spendBlock: null,\n        spendHeight: null,\n      });\n      expect(rows[1]).toMatchObject({\n        ...sampleEntities[1],\n        extractor: 'extractor1',\n        block: block.hash,\n        height: block.height,\n        spendBlock: null,\n        spendHeight: null,\n      });\n      expect(result).toEqual(true);\n    });\n\n    /**\n     * @target storeBoxes should correctly save boxes with different extractors\n     * @dependencies\n     * @scenario\n     * - insert 2 boxes belonging to first-extractor\n     * - run test (call `storeBoxes` with same boxes for the second-extractor)\n     * @expected\n     * - to save 2 new entities belonging to second-extractor\n     * - to return 2 new inserted entity data\n     */\n    it(`should correctly save boxes with different extractors`, async () => {\n      await repository.insert([\n        {\n          ...sampleEntities[0],\n          extractor: 'first-extractor',\n          block: '1',\n          height: 1,\n        },\n        {\n          ...sampleEntities[1],\n          extractor: 'first-extractor',\n          block: '1',\n          height: 1,\n        },\n      ]);\n\n      const result = await action.storeBoxes(\n        [sampleEntities[0], sampleEntities[1]],\n        block,\n        'second-extractor',\n      );\n\n      const [insertedRows] = await repository.findAndCount();\n      expect(insertedRows[2]).toMatchObject({\n        ...sampleEntities[0],\n        extractor: 'second-extractor',\n        block: 'hash',\n        height: block.height,\n        spendBlock: null,\n        spendHeight: null,\n      });\n\n      expect(insertedRows[3]).toMatchObject({\n        ...sampleEntities[1],\n        extractor: 'second-extractor',\n        block: 'hash',\n        height: block.height,\n        spendBlock: null,\n        spendHeight: null,\n      });\n      expect(result).toEqual(true);\n    });\n\n    /**\n     * @target storeBoxes should update boxes correctly\n     * @dependencies\n     * @scenario\n     * - insert 2 boxes\n     * - run test (call `storeBoxes` with the same boxes and updated info)\n     * @expected\n     * - to update the existing box in database\n     * - to return updated box id\n     */\n    it(`storeBoxes should update boxes correctly`, async () => {\n      await repository.insert([\n        {\n          ...sampleEntities[0],\n          extractor: 'extractor',\n          block: '1',\n          height: 1,\n        },\n        {\n          ...sampleEntities[1],\n          extractor: 'extractor',\n          block: '1',\n          height: 1,\n        },\n      ]);\n\n      const result = await action.storeBoxes(\n        [\n          {\n            ...sampleEntities[0],\n            serialized: 'updatedBoxSerialized',\n          },\n        ],\n        block,\n        'extractor',\n      );\n      const [secondInsertRows, secondInsertRowsCount] =\n        await repository.findAndCount();\n      expect(secondInsertRowsCount).toEqual(2);\n      expect(secondInsertRows[0]).toMatchObject({\n        ...sampleEntities[0],\n        extractor: 'extractor',\n        serialized: 'updatedBoxSerialized',\n        block: block.hash,\n        height: block.height,\n        spendBlock: null,\n        spendHeight: null,\n      });\n      expect(result).toEqual(true);\n    });\n  });\n\n  describe('spendBoxes', () => {\n    /**\n     * @target spendBoxes should set spendBlock and spendHeight for a set of boxes\n     * @dependencies\n     * @scenario\n     * - insert two boxes\n     * - mock spending information for the first box\n     * - run test (call `spendBoxes`)\n     * @expected\n     * - spend the first box\n     * - return boxId and serialized of spent box\n     */\n    it(`should set spendBlock and spendHeight for a set of boxes`, async () => {\n      await action.storeBoxes(sampleEntities.slice(0, 2), block, 'extractor1');\n\n      const spendBlock = { ...block, hash: 'spendHash', height: 10006016 };\n      const spendInfos: Array<SpendInfo> = [\n        { txId: 'txId', boxId: sampleEntities[0].boxId, index: 0 },\n      ];\n\n      const spentBoxIds = await action.spendBoxes(\n        spendInfos,\n        spendBlock,\n        'extractor1',\n      );\n\n      const spentBoxes = await repository.findOneBy({\n        boxId: sampleEntities[0].boxId,\n        extractor: 'extractor1',\n      });\n\n      expect(spentBoxes).toMatchObject({\n        ...sampleEntities[0],\n        block: block.hash,\n        height: block.height,\n        extractor: 'extractor1',\n        spendBlock: spendBlock.hash,\n        spendHeight: spendBlock.height,\n      });\n      expect(spentBoxIds).toEqual([pick(sampleEntities[0], ['boxId'])]);\n    });\n  });\n\n  describe('deleteBlockBoxes', () => {\n    /**\n     * @target deleteBlockBoxes should delete the boxes created in the specified block\n     * @dependencies\n     * @scenario\n     * - insert four boxes created in two different blocks\n     * - run test(call `deleteBlockBoxes` to delete block2)\n     * @expected\n     * - to delete the two boxes created in block2\n     * - to return the deleted entity data\n     */\n    it(`should delete the boxes created in the specified block`, async () => {\n      await action.storeBoxes(sampleEntities.slice(0, 2), block, 'extractor1');\n      await action.storeBoxes(sampleEntities.slice(2), block2, 'extractor1');\n\n      const result = await action.deleteBlockBoxes(block2.hash, 'extractor1');\n\n      const [rows, rowsCount] = await repository.findAndCount();\n\n      expect(rowsCount).toEqual(2);\n      expect(rows.map((row) => row.boxId)).not.toContain(\n        sampleEntities.slice(2).map((box) => box.boxId),\n      );\n      expect(result).toEqual({\n        deletedData: action.convertEntityToData(\n          sampleEntities.slice(2) as TestEntity[],\n        ),\n        updatedData: [],\n      });\n    });\n\n    /**\n     * @target deleteBlockBoxes should update the boxes spent in the specified block\n     * @dependencies\n     * @scenario\n     * - insert four boxes created in a block\n     * - spend one of the in the block2\n     * - run test(call `deleteBlockBoxes` to delete block2)\n     * @expected\n     * - to update the box spent in block2\n     * - to return the updated entity boxId and serialized\n     */\n    it(`should update the boxes spent in the specified block`, async () => {\n      await action.storeBoxes(sampleEntities, block, 'extractor1');\n      const spendInfos: Array<SpendInfo> = [\n        { txId: 'txId', boxId: sampleEntities[0].boxId, index: 0 },\n      ];\n      await action.spendBoxes(spendInfos, block2, 'extractor1');\n      const result = await action.deleteBlockBoxes(block2.hash, 'extractor1');\n\n      const [rows, rowsCount] = await repository.findAndCount();\n\n      expect(rowsCount).toEqual(4);\n      expect(rows.map((row) => row.boxId)).not.toContain(\n        sampleEntities.slice(2).map((box) => box.boxId),\n      );\n      expect(result).toEqual({\n        deletedData: [],\n        updatedData: [pick(sampleEntities[0], ['boxId'])],\n      });\n    });\n  });\n});\n"]}