@rosen-bridge/address-extractor 2.1.6 → 2.1.8
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/dist/actions/boxAction.d.ts +63 -0
- package/dist/actions/boxAction.d.ts.map +1 -0
- package/dist/actions/boxAction.js +165 -0
- package/dist/actions/db.d.ts +32 -0
- package/dist/actions/db.d.ts.map +1 -0
- package/dist/actions/db.js +149 -0
- package/dist/constants.d.ts +4 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +4 -0
- package/dist/entities/boxEntity.d.ts +12 -0
- package/dist/entities/boxEntity.d.ts.map +1 -0
- package/dist/entities/boxEntity.js +62 -0
- package/dist/extractor/ergoUtxoExtractor.d.ts +150 -0
- package/dist/extractor/ergoUtxoExtractor.d.ts.map +1 -0
- package/dist/extractor/ergoUtxoExtractor.js +242 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +4 -0
- package/dist/interfaces/ExtractedBox.d.ts +8 -0
- package/dist/interfaces/ExtractedBox.d.ts.map +1 -0
- package/dist/interfaces/ExtractedBox.js +8 -0
- package/dist/interfaces/types.d.ts +28 -0
- package/dist/interfaces/types.d.ts.map +1 -0
- package/dist/interfaces/types.js +2 -0
- package/dist/migrations/index.d.ts +7 -0
- package/dist/migrations/index.d.ts.map +1 -0
- package/dist/migrations/index.js +9 -0
- package/dist/migrations/postgres/1688558553044-migration.d.ts +7 -0
- package/dist/migrations/postgres/1688558553044-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1688558553044-migration.js +24 -0
- package/dist/migrations/postgres/1689151869889-migration.d.ts +7 -0
- package/dist/migrations/postgres/1689151869889-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1689151869889-migration.js +15 -0
- package/dist/migrations/sqlite/1688555566662-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1688555566662-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1688555566662-migration.js +23 -0
- package/dist/migrations/sqlite/1689143753142-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1689143753142-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1689143753142-migration.js +85 -0
- package/dist/network/ergoNetworkApi.d.ts +23 -0
- package/dist/network/ergoNetworkApi.d.ts.map +1 -0
- package/dist/network/ergoNetworkApi.js +55 -0
- package/dist/network/parser.d.ts +8 -0
- package/dist/network/parser.d.ts.map +1 -0
- package/dist/network/parser.js +5 -0
- package/dist/utils.d.ts +9 -0
- package/dist/utils.d.ts.map +1 -0
- package/dist/utils.js +6 -0
- package/package.json +2 -2
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import { DataSource } from 'typeorm';
|
|
2
|
+
import { AbstractLogger } from '@rosen-bridge/logger-interface';
|
|
3
|
+
import { BlockEntity } from '@rosen-bridge/scanner';
|
|
4
|
+
import { ExtractedBox } from '../interfaces/types';
|
|
5
|
+
export declare class BoxEntityAction {
|
|
6
|
+
private readonly datasource;
|
|
7
|
+
readonly logger: AbstractLogger;
|
|
8
|
+
private readonly repository;
|
|
9
|
+
constructor(dataSource: DataSource, logger: AbstractLogger);
|
|
10
|
+
/**
|
|
11
|
+
* insert new box into database
|
|
12
|
+
* @param box
|
|
13
|
+
* @param extractor
|
|
14
|
+
*/
|
|
15
|
+
insertBox: (box: ExtractedBox, extractor: string) => Promise<import("typeorm").InsertResult>;
|
|
16
|
+
/**
|
|
17
|
+
* Update an unspent box information in the database
|
|
18
|
+
* @param box
|
|
19
|
+
* @param extractor
|
|
20
|
+
*/
|
|
21
|
+
updateBox: (box: ExtractedBox, extractor: string) => Promise<import("typeorm").UpdateResult>;
|
|
22
|
+
/**
|
|
23
|
+
* It stores list of blocks in the dataSource with block id
|
|
24
|
+
* @param boxes
|
|
25
|
+
* @param spendBoxes
|
|
26
|
+
* @param block
|
|
27
|
+
* @param extractor
|
|
28
|
+
*/
|
|
29
|
+
storeBox: (boxes: Array<ExtractedBox>, block: BlockEntity, extractor: string) => Promise<boolean>;
|
|
30
|
+
/**
|
|
31
|
+
* Update spendBlock and spendHeight of boxes spent on the block
|
|
32
|
+
* @param spendIds
|
|
33
|
+
* @param block
|
|
34
|
+
* @param extractor
|
|
35
|
+
*/
|
|
36
|
+
spendBoxes: (spendIds: Array<string>, block: BlockEntity, extractor: string) => Promise<void>;
|
|
37
|
+
/**
|
|
38
|
+
* delete boxes in specific block from database. if box spend in this block marked as unspent
|
|
39
|
+
* and if created in this block remove it from database
|
|
40
|
+
* @param block
|
|
41
|
+
* @param extractor
|
|
42
|
+
*/
|
|
43
|
+
deleteBlockBoxes: (block: string, extractor: string) => Promise<void>;
|
|
44
|
+
/**
|
|
45
|
+
* Returns all stored box ids
|
|
46
|
+
*/
|
|
47
|
+
getAllBoxIds: (extractor: string) => Promise<Array<string>>;
|
|
48
|
+
/**
|
|
49
|
+
* Removes specified box
|
|
50
|
+
* @param boxId
|
|
51
|
+
* @param extractor
|
|
52
|
+
*/
|
|
53
|
+
removeBox: (boxId: string, extractor: string) => Promise<import("typeorm").DeleteResult>;
|
|
54
|
+
/**
|
|
55
|
+
* Update the box spending information
|
|
56
|
+
* @param boxId
|
|
57
|
+
* @param extractor
|
|
58
|
+
* @param blockId
|
|
59
|
+
* @param blockHeight
|
|
60
|
+
*/
|
|
61
|
+
updateSpendBlock: (boxId: string, extractor: string, blockId: string, blockHeight: number) => Promise<import("typeorm").UpdateResult>;
|
|
62
|
+
}
|
|
63
|
+
//# sourceMappingURL=boxAction.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"boxAction.d.ts","sourceRoot":"","sources":["../../lib/actions/boxAction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAkB,MAAM,SAAS,CAAC;AAErD,OAAO,EAAE,cAAc,EAAE,MAAM,gCAAgC,CAAC;AAChE,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAGpD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAGnD,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAa;IACxC,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;IAChC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAwB;gBAEvC,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,cAAc;IAM1D;;;;OAIG;IACH,SAAS,QAAe,YAAY,aAAa,MAAM,6CASrD;IAEF;;;;OAIG;IACH,SAAS,QAAe,YAAY,aAAa,MAAM,6CAYrD;IAEF;;;;;;OAMG;IACH,QAAQ,UACC,MAAM,YAAY,CAAC,SACnB,WAAW,aACP,MAAM,sBA6CjB;IAEF;;;;;OAKG;IACH,UAAU,aACE,MAAM,MAAM,CAAC,SAChB,WAAW,aACP,MAAM,KAChB,QAAQ,IAAI,CAAC,CAoBd;IAEF;;;;;OAKG;IACH,gBAAgB,UAAiB,MAAM,aAAa,MAAM,mBAYxD;IAEF;;OAEG;IACH,YAAY,cAAqB,MAAM,KAAG,QAAQ,MAAM,MAAM,CAAC,CAAC,CAU9D;IAEF;;;;OAIG;IACH,SAAS,UAAiB,MAAM,aAAa,MAAM,6CAEjD;IAEF;;;;;;OAMG;IACH,gBAAgB,UACP,MAAM,aACF,MAAM,WACR,MAAM,eACF,MAAM,6CAMnB;CACH"}
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import { In } from 'typeorm';
|
|
2
|
+
import { chunk } from 'lodash-es';
|
|
3
|
+
import { BoxEntity } from '../entities/boxEntity';
|
|
4
|
+
import { dbIdChunkSize } from '../constants';
|
|
5
|
+
export class BoxEntityAction {
|
|
6
|
+
datasource;
|
|
7
|
+
logger;
|
|
8
|
+
repository;
|
|
9
|
+
constructor(dataSource, logger) {
|
|
10
|
+
this.datasource = dataSource;
|
|
11
|
+
this.logger = logger;
|
|
12
|
+
this.repository = dataSource.getRepository(BoxEntity);
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* insert new box into database
|
|
16
|
+
* @param box
|
|
17
|
+
* @param extractor
|
|
18
|
+
*/
|
|
19
|
+
insertBox = async (box, extractor) => {
|
|
20
|
+
return this.repository.insert({
|
|
21
|
+
address: box.address,
|
|
22
|
+
boxId: box.boxId,
|
|
23
|
+
createBlock: box.blockId,
|
|
24
|
+
creationHeight: box.height,
|
|
25
|
+
serialized: box.serialized,
|
|
26
|
+
extractor: extractor,
|
|
27
|
+
});
|
|
28
|
+
};
|
|
29
|
+
/**
|
|
30
|
+
* Update an unspent box information in the database
|
|
31
|
+
* @param box
|
|
32
|
+
* @param extractor
|
|
33
|
+
*/
|
|
34
|
+
updateBox = async (box, extractor) => {
|
|
35
|
+
return this.repository.update({ boxId: box.boxId, extractor: extractor }, {
|
|
36
|
+
address: box.address,
|
|
37
|
+
createBlock: box.blockId,
|
|
38
|
+
creationHeight: box.height,
|
|
39
|
+
serialized: box.serialized,
|
|
40
|
+
spendBlock: null,
|
|
41
|
+
spendHeight: 0,
|
|
42
|
+
});
|
|
43
|
+
};
|
|
44
|
+
/**
|
|
45
|
+
* It stores list of blocks in the dataSource with block id
|
|
46
|
+
* @param boxes
|
|
47
|
+
* @param spendBoxes
|
|
48
|
+
* @param block
|
|
49
|
+
* @param extractor
|
|
50
|
+
*/
|
|
51
|
+
storeBox = async (boxes, block, extractor) => {
|
|
52
|
+
const boxIds = boxes.map((item) => item.boxId);
|
|
53
|
+
const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({
|
|
54
|
+
boxId: In(boxIds),
|
|
55
|
+
extractor: extractor,
|
|
56
|
+
});
|
|
57
|
+
let success = true;
|
|
58
|
+
const queryRunner = this.datasource.createQueryRunner();
|
|
59
|
+
await queryRunner.connect();
|
|
60
|
+
await queryRunner.startTransaction();
|
|
61
|
+
const repository = await queryRunner.manager.getRepository(BoxEntity);
|
|
62
|
+
try {
|
|
63
|
+
for (const box of boxes) {
|
|
64
|
+
const entity = {
|
|
65
|
+
address: box.address,
|
|
66
|
+
boxId: box.boxId,
|
|
67
|
+
createBlock: block.hash,
|
|
68
|
+
creationHeight: block.height,
|
|
69
|
+
spendBlock: undefined,
|
|
70
|
+
serialized: box.serialized,
|
|
71
|
+
extractor: extractor,
|
|
72
|
+
};
|
|
73
|
+
const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);
|
|
74
|
+
if (dbBox.length > 0) {
|
|
75
|
+
this.logger.info(`Updating box ${box.boxId} and extractor ${extractor}`);
|
|
76
|
+
this.logger.debug(`Entity: ${JSON.stringify(entity)}`);
|
|
77
|
+
await repository.update({ id: dbBoxes[0].id }, entity);
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
this.logger.info(`Storing box ${box.boxId}`);
|
|
81
|
+
this.logger.debug(JSON.stringify(entity));
|
|
82
|
+
await repository.insert(entity);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
await queryRunner.commitTransaction();
|
|
86
|
+
}
|
|
87
|
+
catch (e) {
|
|
88
|
+
this.logger.error(`An error occurred during store boxes action: ${e}`);
|
|
89
|
+
await queryRunner.rollbackTransaction();
|
|
90
|
+
success = false;
|
|
91
|
+
}
|
|
92
|
+
finally {
|
|
93
|
+
await queryRunner.release();
|
|
94
|
+
}
|
|
95
|
+
return success;
|
|
96
|
+
};
|
|
97
|
+
/**
|
|
98
|
+
* Update spendBlock and spendHeight of boxes spent on the block
|
|
99
|
+
* @param spendIds
|
|
100
|
+
* @param block
|
|
101
|
+
* @param extractor
|
|
102
|
+
*/
|
|
103
|
+
spendBoxes = async (spendIds, block, extractor) => {
|
|
104
|
+
const spendIdChunks = chunk(spendIds, dbIdChunkSize);
|
|
105
|
+
for (const spendIdChunk of spendIdChunks) {
|
|
106
|
+
const updateResult = await this.repository.update({ boxId: In(spendIdChunk), extractor: extractor }, { spendBlock: block.hash, spendHeight: block.height });
|
|
107
|
+
if (updateResult.affected && updateResult.affected > 0) {
|
|
108
|
+
const spentRows = await this.repository.findBy({
|
|
109
|
+
boxId: In(spendIdChunk),
|
|
110
|
+
spendBlock: block.hash,
|
|
111
|
+
});
|
|
112
|
+
for (const row of spentRows) {
|
|
113
|
+
this.logger.debug(`Spent box with boxId [${row.boxId}] at height ${block.height}`);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
};
|
|
118
|
+
/**
|
|
119
|
+
* delete boxes in specific block from database. if box spend in this block marked as unspent
|
|
120
|
+
* and if created in this block remove it from database
|
|
121
|
+
* @param block
|
|
122
|
+
* @param extractor
|
|
123
|
+
*/
|
|
124
|
+
deleteBlockBoxes = async (block, extractor) => {
|
|
125
|
+
this.logger.info(`Deleting boxes in block ${block} and extractor ${extractor}`);
|
|
126
|
+
await this.repository.delete({
|
|
127
|
+
extractor: extractor,
|
|
128
|
+
createBlock: block,
|
|
129
|
+
});
|
|
130
|
+
await this.repository.update({ spendBlock: block, extractor: extractor }, { spendBlock: null, spendHeight: 0 });
|
|
131
|
+
};
|
|
132
|
+
/**
|
|
133
|
+
* Returns all stored box ids
|
|
134
|
+
*/
|
|
135
|
+
getAllBoxIds = async (extractor) => {
|
|
136
|
+
const boxIds = await this.repository.find({
|
|
137
|
+
select: {
|
|
138
|
+
boxId: true,
|
|
139
|
+
},
|
|
140
|
+
where: {
|
|
141
|
+
extractor: extractor,
|
|
142
|
+
},
|
|
143
|
+
});
|
|
144
|
+
return boxIds.map((item) => item.boxId);
|
|
145
|
+
};
|
|
146
|
+
/**
|
|
147
|
+
* Removes specified box
|
|
148
|
+
* @param boxId
|
|
149
|
+
* @param extractor
|
|
150
|
+
*/
|
|
151
|
+
removeBox = async (boxId, extractor) => {
|
|
152
|
+
return await this.repository.delete({ boxId: boxId, extractor: extractor });
|
|
153
|
+
};
|
|
154
|
+
/**
|
|
155
|
+
* Update the box spending information
|
|
156
|
+
* @param boxId
|
|
157
|
+
* @param extractor
|
|
158
|
+
* @param blockId
|
|
159
|
+
* @param blockHeight
|
|
160
|
+
*/
|
|
161
|
+
updateSpendBlock = async (boxId, extractor, blockId, blockHeight) => {
|
|
162
|
+
return await this.repository.update({ boxId: boxId, extractor: extractor }, { spendBlock: blockId, spendHeight: blockHeight });
|
|
163
|
+
};
|
|
164
|
+
}
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"boxAction.js","sourceRoot":"","sources":["../../lib/actions/boxAction.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource, In, Repository } from 'typeorm';\nimport { chunk } from 'lodash-es';\nimport { AbstractLogger } from '@rosen-bridge/logger-interface';\nimport { BlockEntity } from '@rosen-bridge/scanner';\n\nimport { BoxEntity } from '../entities/boxEntity';\nimport { ExtractedBox } from '../interfaces/types';\nimport { dbIdChunkSize } from '../constants';\n\nexport class BoxEntityAction {\n  private readonly datasource: DataSource;\n  readonly logger: AbstractLogger;\n  private readonly repository: Repository<BoxEntity>;\n\n  constructor(dataSource: DataSource, logger: AbstractLogger) {\n    this.datasource = dataSource;\n    this.logger = logger;\n    this.repository = dataSource.getRepository(BoxEntity);\n  }\n\n  /**\n   * insert new box into database\n   * @param box\n   * @param extractor\n   */\n  insertBox = async (box: ExtractedBox, extractor: string) => {\n    return this.repository.insert({\n      address: box.address,\n      boxId: box.boxId,\n      createBlock: box.blockId,\n      creationHeight: box.height,\n      serialized: box.serialized,\n      extractor: extractor,\n    });\n  };\n\n  /**\n   * Update an unspent box information in the database\n   * @param box\n   * @param extractor\n   */\n  updateBox = async (box: ExtractedBox, extractor: string) => {\n    return this.repository.update(\n      { boxId: box.boxId, extractor: extractor },\n      {\n        address: box.address,\n        createBlock: box.blockId,\n        creationHeight: box.height,\n        serialized: box.serialized,\n        spendBlock: null,\n        spendHeight: 0,\n      }\n    );\n  };\n\n  /**\n   * It stores list of blocks in the dataSource with block id\n   * @param boxes\n   * @param spendBoxes\n   * @param block\n   * @param extractor\n   */\n  storeBox = async (\n    boxes: Array<ExtractedBox>,\n    block: BlockEntity,\n    extractor: string\n  ) => {\n    const boxIds = boxes.map((item) => item.boxId);\n    const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({\n      boxId: In(boxIds),\n      extractor: extractor,\n    });\n    let success = true;\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    const repository = await queryRunner.manager.getRepository(BoxEntity);\n    try {\n      for (const box of boxes) {\n        const entity = {\n          address: box.address,\n          boxId: box.boxId,\n          createBlock: block.hash,\n          creationHeight: block.height,\n          spendBlock: undefined,\n          serialized: box.serialized,\n          extractor: extractor,\n        };\n        const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);\n        if (dbBox.length > 0) {\n          this.logger.info(\n            `Updating box ${box.boxId} and extractor ${extractor}`\n          );\n          this.logger.debug(`Entity: ${JSON.stringify(entity)}`);\n          await repository.update({ id: dbBoxes[0].id }, entity);\n        } else {\n          this.logger.info(`Storing box ${box.boxId}`);\n          this.logger.debug(JSON.stringify(entity));\n          await repository.insert(entity);\n        }\n      }\n      await queryRunner.commitTransaction();\n    } catch (e) {\n      this.logger.error(`An error occurred during store boxes action: ${e}`);\n      await queryRunner.rollbackTransaction();\n      success = false;\n    } finally {\n      await queryRunner.release();\n    }\n    return success;\n  };\n\n  /**\n   * Update spendBlock and spendHeight of boxes spent on the block\n   * @param spendIds\n   * @param block\n   * @param extractor\n   */\n  spendBoxes = async (\n    spendIds: Array<string>,\n    block: BlockEntity,\n    extractor: string\n  ): Promise<void> => {\n    const spendIdChunks = chunk(spendIds, dbIdChunkSize);\n    for (const spendIdChunk of spendIdChunks) {\n      const updateResult = await this.repository.update(\n        { boxId: In(spendIdChunk), extractor: extractor },\n        { spendBlock: block.hash, spendHeight: block.height }\n      );\n\n      if (updateResult.affected && updateResult.affected > 0) {\n        const spentRows = await this.repository.findBy({\n          boxId: In(spendIdChunk),\n          spendBlock: block.hash,\n        });\n        for (const row of spentRows) {\n          this.logger.debug(\n            `Spent box with boxId [${row.boxId}] at height ${block.height}`\n          );\n        }\n      }\n    }\n  };\n\n  /**\n   * delete boxes in specific block from database. if box spend in this block marked as unspent\n   * and if created in this block remove it from database\n   * @param block\n   * @param extractor\n   */\n  deleteBlockBoxes = async (block: string, extractor: string) => {\n    this.logger.info(\n      `Deleting boxes in block ${block} and extractor ${extractor}`\n    );\n    await this.repository.delete({\n      extractor: extractor,\n      createBlock: block,\n    });\n    await this.repository.update(\n      { spendBlock: block, extractor: extractor },\n      { spendBlock: null, spendHeight: 0 }\n    );\n  };\n\n  /**\n   *  Returns all stored box ids\n   */\n  getAllBoxIds = async (extractor: string): Promise<Array<string>> => {\n    const boxIds = await this.repository.find({\n      select: {\n        boxId: true,\n      },\n      where: {\n        extractor: extractor,\n      },\n    });\n    return boxIds.map((item: { boxId: string }) => item.boxId);\n  };\n\n  /**\n   * Removes specified box\n   * @param boxId\n   * @param extractor\n   */\n  removeBox = async (boxId: string, extractor: string) => {\n    return await this.repository.delete({ boxId: boxId, extractor: extractor });\n  };\n\n  /**\n   * Update the box spending information\n   * @param boxId\n   * @param extractor\n   * @param blockId\n   * @param blockHeight\n   */\n  updateSpendBlock = async (\n    boxId: string,\n    extractor: string,\n    blockId: string,\n    blockHeight: number\n  ) => {\n    return await this.repository.update(\n      { boxId: boxId, extractor: extractor },\n      { spendBlock: blockId, spendHeight: blockHeight }\n    );\n  };\n}\n"]}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { DataSource } from 'typeorm';
|
|
2
|
+
import ExtractedBox from '../interfaces/ExtractedBox';
|
|
3
|
+
import { AbstractLogger } from '@rosen-bridge/logger-interface';
|
|
4
|
+
import { BlockEntity } from '@rosen-bridge/scanner';
|
|
5
|
+
export declare class BoxEntityAction {
|
|
6
|
+
private readonly datasource;
|
|
7
|
+
readonly logger: AbstractLogger;
|
|
8
|
+
constructor(dataSource: DataSource, logger: AbstractLogger);
|
|
9
|
+
/**
|
|
10
|
+
* stores initial extracted boxes to the database
|
|
11
|
+
* @param boxes
|
|
12
|
+
* @param initializationHeight
|
|
13
|
+
* @param extractor
|
|
14
|
+
*/
|
|
15
|
+
storeInitialBoxes: (boxes: Array<ExtractedBox>, initializationHeight: number, extractor: string) => Promise<boolean>;
|
|
16
|
+
/**
|
|
17
|
+
* It stores list of blocks in the dataSource with block id
|
|
18
|
+
* @param boxes
|
|
19
|
+
* @param spendBoxes
|
|
20
|
+
* @param block
|
|
21
|
+
* @param extractor
|
|
22
|
+
*/
|
|
23
|
+
storeBox: (boxes: Array<ExtractedBox>, spendBoxes: Array<string>, block: BlockEntity, extractor: string) => Promise<boolean>;
|
|
24
|
+
/**
|
|
25
|
+
* delete boxes in specific block from database. if box spend in this block marked as unspent
|
|
26
|
+
* and if created in this block remove it from database
|
|
27
|
+
* @param block
|
|
28
|
+
* @param extractor
|
|
29
|
+
*/
|
|
30
|
+
deleteBlockBoxes: (block: string, extractor: string) => Promise<void>;
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=db.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"db.d.ts","sourceRoot":"","sources":["../../lib/actions/db.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAgB,MAAM,SAAS,CAAC;AACnD,OAAO,YAAY,MAAM,4BAA4B,CAAC;AACtD,OAAO,EAAE,cAAc,EAAE,MAAM,gCAAgC,CAAC;AAChE,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAEpD,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAa;IACxC,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;gBAEpB,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,cAAc;IAK1D;;;;;OAKG;IACH,iBAAiB,UACR,MAAM,YAAY,CAAC,wBACJ,MAAM,aACjB,MAAM,sBAqCjB;IAEF;;;;;;OAMG;IACH,QAAQ,UACC,MAAM,YAAY,CAAC,cACd,MAAM,MAAM,CAAC,SAClB,WAAW,aACP,MAAM,sBA6DjB;IAEF;;;;;OAKG;IACH,gBAAgB,UAAiB,MAAM,aAAa,MAAM,mBAuBxD;CACH"}
|
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { BoxEntity } from '../entities/boxEntity';
|
|
2
|
+
import { In, LessThan } from 'typeorm';
|
|
3
|
+
export class BoxEntityAction {
|
|
4
|
+
datasource;
|
|
5
|
+
logger;
|
|
6
|
+
constructor(dataSource, logger) {
|
|
7
|
+
this.datasource = dataSource;
|
|
8
|
+
this.logger = logger;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* stores initial extracted boxes to the database
|
|
12
|
+
* @param boxes
|
|
13
|
+
* @param initializationHeight
|
|
14
|
+
* @param extractor
|
|
15
|
+
*/
|
|
16
|
+
storeInitialBoxes = async (boxes, initializationHeight, extractor) => {
|
|
17
|
+
const queryRunner = this.datasource.createQueryRunner();
|
|
18
|
+
await queryRunner.connect();
|
|
19
|
+
await queryRunner.startTransaction();
|
|
20
|
+
try {
|
|
21
|
+
const repository = queryRunner.manager.getRepository(BoxEntity);
|
|
22
|
+
await repository.delete({
|
|
23
|
+
creationHeight: LessThan(initializationHeight),
|
|
24
|
+
});
|
|
25
|
+
for (const box of boxes) {
|
|
26
|
+
const entity = {
|
|
27
|
+
address: box.address,
|
|
28
|
+
boxId: box.boxId,
|
|
29
|
+
createBlock: box.blockId,
|
|
30
|
+
creationHeight: box.height,
|
|
31
|
+
spendBlock: undefined,
|
|
32
|
+
serialized: box.serialized,
|
|
33
|
+
extractor: extractor,
|
|
34
|
+
};
|
|
35
|
+
this.logger.info(`Storing initial box ${box.boxId} with extractor ${extractor}`);
|
|
36
|
+
this.logger.debug(`Stored box entity: [${JSON.stringify(entity)}]`);
|
|
37
|
+
await repository.insert(entity);
|
|
38
|
+
}
|
|
39
|
+
await queryRunner.commitTransaction();
|
|
40
|
+
}
|
|
41
|
+
catch (e) {
|
|
42
|
+
this.logger.error(`An error occurred during store boxes action: ${e}`);
|
|
43
|
+
await queryRunner.rollbackTransaction();
|
|
44
|
+
throw new Error('Initialization failed while storing initial address boxes');
|
|
45
|
+
}
|
|
46
|
+
finally {
|
|
47
|
+
await queryRunner.release();
|
|
48
|
+
}
|
|
49
|
+
return true;
|
|
50
|
+
};
|
|
51
|
+
/**
|
|
52
|
+
* It stores list of blocks in the dataSource with block id
|
|
53
|
+
* @param boxes
|
|
54
|
+
* @param spendBoxes
|
|
55
|
+
* @param block
|
|
56
|
+
* @param extractor
|
|
57
|
+
*/
|
|
58
|
+
storeBox = async (boxes, spendBoxes, block, extractor) => {
|
|
59
|
+
const boxIds = boxes.map((item) => item.boxId);
|
|
60
|
+
const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({
|
|
61
|
+
boxId: In(boxIds),
|
|
62
|
+
extractor: extractor,
|
|
63
|
+
});
|
|
64
|
+
let success = true;
|
|
65
|
+
const queryRunner = this.datasource.createQueryRunner();
|
|
66
|
+
await queryRunner.connect();
|
|
67
|
+
await queryRunner.startTransaction();
|
|
68
|
+
try {
|
|
69
|
+
for (const box of boxes) {
|
|
70
|
+
const entity = {
|
|
71
|
+
address: box.address,
|
|
72
|
+
boxId: box.boxId,
|
|
73
|
+
createBlock: block.hash,
|
|
74
|
+
creationHeight: block.height,
|
|
75
|
+
spendBlock: undefined,
|
|
76
|
+
serialized: box.serialized,
|
|
77
|
+
extractor: extractor,
|
|
78
|
+
};
|
|
79
|
+
const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);
|
|
80
|
+
if (dbBox.length > 0) {
|
|
81
|
+
this.logger.info(`Updating box ${box.boxId} and extractor ${extractor}`);
|
|
82
|
+
this.logger.debug(`Entity: ${JSON.stringify(entity)}`);
|
|
83
|
+
await queryRunner.manager
|
|
84
|
+
.getRepository(BoxEntity)
|
|
85
|
+
.createQueryBuilder()
|
|
86
|
+
.update()
|
|
87
|
+
.set(entity)
|
|
88
|
+
.where({ id: dbBox[0].id })
|
|
89
|
+
.execute();
|
|
90
|
+
}
|
|
91
|
+
else {
|
|
92
|
+
this.logger.info(`Storing box ${box.boxId}`);
|
|
93
|
+
this.logger.debug(JSON.stringify(entity));
|
|
94
|
+
await queryRunner.manager.getRepository(BoxEntity).insert(entity);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
this.logger.debug(`Updating spendBlock for boxes ${spendBoxes}`);
|
|
98
|
+
await queryRunner.manager
|
|
99
|
+
.getRepository(BoxEntity)
|
|
100
|
+
.createQueryBuilder()
|
|
101
|
+
.update()
|
|
102
|
+
.set({ spendBlock: block.hash })
|
|
103
|
+
.where('boxId IN (:...boxes) AND extractor = :extractor', {
|
|
104
|
+
boxes: spendBoxes,
|
|
105
|
+
extractor: extractor,
|
|
106
|
+
})
|
|
107
|
+
.execute();
|
|
108
|
+
await queryRunner.commitTransaction();
|
|
109
|
+
}
|
|
110
|
+
catch (e) {
|
|
111
|
+
this.logger.error(`An error occurred during store boxes action: ${e}`);
|
|
112
|
+
await queryRunner.rollbackTransaction();
|
|
113
|
+
success = false;
|
|
114
|
+
}
|
|
115
|
+
finally {
|
|
116
|
+
await queryRunner.release();
|
|
117
|
+
}
|
|
118
|
+
return success;
|
|
119
|
+
};
|
|
120
|
+
/**
|
|
121
|
+
* delete boxes in specific block from database. if box spend in this block marked as unspent
|
|
122
|
+
* and if created in this block remove it from database
|
|
123
|
+
* @param block
|
|
124
|
+
* @param extractor
|
|
125
|
+
*/
|
|
126
|
+
deleteBlockBoxes = async (block, extractor) => {
|
|
127
|
+
this.logger.info(`Deleting boxes in block ${block} and extractor ${extractor}`);
|
|
128
|
+
await this.datasource
|
|
129
|
+
.createQueryBuilder()
|
|
130
|
+
.delete()
|
|
131
|
+
.from(BoxEntity)
|
|
132
|
+
.where('extractor = :extractor AND createBlock = :block', {
|
|
133
|
+
block: block,
|
|
134
|
+
extractor: extractor,
|
|
135
|
+
})
|
|
136
|
+
.execute();
|
|
137
|
+
await this.datasource
|
|
138
|
+
.getRepository(BoxEntity)
|
|
139
|
+
.createQueryBuilder()
|
|
140
|
+
.update()
|
|
141
|
+
.set({ spendBlock: null })
|
|
142
|
+
.where('spendBlock = :block AND extractor = :extractor', {
|
|
143
|
+
block: block,
|
|
144
|
+
extractor: extractor,
|
|
145
|
+
})
|
|
146
|
+
.execute();
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"db.js","sourceRoot":"","sources":["../../lib/actions/db.ts"],"names":[],"mappings":"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","sourcesContent":["import { BoxEntity } from '../entities/boxEntity';\nimport { DataSource, In, LessThan } from 'typeorm';\nimport ExtractedBox from '../interfaces/ExtractedBox';\nimport { AbstractLogger } from '@rosen-bridge/logger-interface';\nimport { BlockEntity } from '@rosen-bridge/scanner';\n\nexport class BoxEntityAction {\n  private readonly datasource: DataSource;\n  readonly logger: AbstractLogger;\n\n  constructor(dataSource: DataSource, logger: AbstractLogger) {\n    this.datasource = dataSource;\n    this.logger = logger;\n  }\n\n  /**\n   * stores initial extracted boxes to the database\n   * @param boxes\n   * @param initializationHeight\n   * @param extractor\n   */\n  storeInitialBoxes = async (\n    boxes: Array<ExtractedBox>,\n    initializationHeight: number,\n    extractor: string\n  ) => {\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    try {\n      const repository = queryRunner.manager.getRepository(BoxEntity);\n      await repository.delete({\n        creationHeight: LessThan(initializationHeight),\n      });\n      for (const box of boxes) {\n        const entity = {\n          address: box.address,\n          boxId: box.boxId,\n          createBlock: box.blockId,\n          creationHeight: box.height,\n          spendBlock: undefined,\n          serialized: box.serialized,\n          extractor: extractor,\n        };\n        this.logger.info(\n          `Storing initial box ${box.boxId} with extractor ${extractor}`\n        );\n        this.logger.debug(`Stored box entity: [${JSON.stringify(entity)}]`);\n        await repository.insert(entity);\n      }\n      await queryRunner.commitTransaction();\n    } catch (e) {\n      this.logger.error(`An error occurred during store boxes action: ${e}`);\n      await queryRunner.rollbackTransaction();\n      throw new Error(\n        'Initialization failed while storing initial address boxes'\n      );\n    } finally {\n      await queryRunner.release();\n    }\n    return true;\n  };\n\n  /**\n   * It stores list of blocks in the dataSource with block id\n   * @param boxes\n   * @param spendBoxes\n   * @param block\n   * @param extractor\n   */\n  storeBox = async (\n    boxes: Array<ExtractedBox>,\n    spendBoxes: Array<string>,\n    block: BlockEntity,\n    extractor: string\n  ) => {\n    const boxIds = boxes.map((item) => item.boxId);\n    const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({\n      boxId: In(boxIds),\n      extractor: extractor,\n    });\n    let success = true;\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    try {\n      for (const box of boxes) {\n        const entity = {\n          address: box.address,\n          boxId: box.boxId,\n          createBlock: block.hash,\n          creationHeight: block.height,\n          spendBlock: undefined,\n          serialized: box.serialized,\n          extractor: extractor,\n        };\n        const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);\n        if (dbBox.length > 0) {\n          this.logger.info(\n            `Updating box ${box.boxId} and extractor ${extractor}`\n          );\n          this.logger.debug(`Entity: ${JSON.stringify(entity)}`);\n          await queryRunner.manager\n            .getRepository(BoxEntity)\n            .createQueryBuilder()\n            .update()\n            .set(entity)\n            .where({ id: dbBox[0].id })\n            .execute();\n        } else {\n          this.logger.info(`Storing box ${box.boxId}`);\n          this.logger.debug(JSON.stringify(entity));\n          await queryRunner.manager.getRepository(BoxEntity).insert(entity);\n        }\n      }\n      this.logger.debug(`Updating spendBlock for boxes ${spendBoxes}`);\n      await queryRunner.manager\n        .getRepository(BoxEntity)\n        .createQueryBuilder()\n        .update()\n        .set({ spendBlock: block.hash })\n        .where('boxId IN (:...boxes) AND extractor = :extractor', {\n          boxes: spendBoxes,\n          extractor: extractor,\n        })\n        .execute();\n      await queryRunner.commitTransaction();\n    } catch (e) {\n      this.logger.error(`An error occurred during store boxes action: ${e}`);\n      await queryRunner.rollbackTransaction();\n      success = false;\n    } finally {\n      await queryRunner.release();\n    }\n    return success;\n  };\n\n  /**\n   * delete boxes in specific block from database. if box spend in this block marked as unspent\n   * and if created in this block remove it from database\n   * @param block\n   * @param extractor\n   */\n  deleteBlockBoxes = async (block: string, extractor: string) => {\n    this.logger.info(\n      `Deleting boxes in block ${block} and extractor ${extractor}`\n    );\n    await this.datasource\n      .createQueryBuilder()\n      .delete()\n      .from(BoxEntity)\n      .where('extractor = :extractor AND createBlock = :block', {\n        block: block,\n        extractor: extractor,\n      })\n      .execute();\n    await this.datasource\n      .getRepository(BoxEntity)\n      .createQueryBuilder()\n      .update()\n      .set({ spendBlock: null })\n      .where('spendBlock = :block AND extractor = :extractor', {\n        block: block,\n        extractor: extractor,\n      })\n      .execute();\n  };\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../lib/constants.ts"],"names":[],"mappings":"AAAA,QAAA,MAAM,eAAe,MAAM,CAAC;AAC5B,QAAA,MAAM,aAAa,MAAM,CAAC;AAE1B,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,CAAC"}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
const DefaultApiLimit = 100;
|
|
2
|
+
const dbIdChunkSize = 100;
|
|
3
|
+
export { DefaultApiLimit, dbIdChunkSize };
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLGVBQWUsR0FBRyxHQUFHLENBQUM7QUFDNUIsTUFBTSxhQUFhLEdBQUcsR0FBRyxDQUFDO0FBRTFCLE9BQU8sRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJjb25zdCBEZWZhdWx0QXBpTGltaXQgPSAxMDA7XG5jb25zdCBkYklkQ2h1bmtTaXplID0gMTAwO1xuXG5leHBvcnQgeyBEZWZhdWx0QXBpTGltaXQsIGRiSWRDaHVua1NpemUgfTtcbiJdfQ==
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
export declare class BoxEntity {
|
|
2
|
+
id: number;
|
|
3
|
+
address: string;
|
|
4
|
+
boxId: string;
|
|
5
|
+
createBlock: string;
|
|
6
|
+
creationHeight: number;
|
|
7
|
+
serialized: string;
|
|
8
|
+
spendBlock?: string | null;
|
|
9
|
+
spendHeight?: number;
|
|
10
|
+
extractor: string;
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=boxEntity.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"boxEntity.d.ts","sourceRoot":"","sources":["../../lib/entities/boxEntity.ts"],"names":[],"mappings":"AAEA,qBACa,SAAS;IAEpB,EAAE,EAAE,MAAM,CAAC;IAGX,OAAO,EAAE,MAAM,CAAC;IAGhB,KAAK,EAAE,MAAM,CAAC;IAGd,WAAW,EAAE,MAAM,CAAC;IAGpB,cAAc,EAAE,MAAM,CAAC;IAGvB,UAAU,EAAE,MAAM,CAAC;IAGnB,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAG3B,WAAW,CAAC,EAAE,MAAM,CAAC;IAGrB,SAAS,EAAE,MAAM,CAAC;CACnB"}
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
|
|
2
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
3
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
4
|
+
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
5
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
6
|
+
};
|
|
7
|
+
var __metadata = (this && this.__metadata) || function (k, v) {
|
|
8
|
+
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
|
|
9
|
+
};
|
|
10
|
+
import { Column, Entity, PrimaryGeneratedColumn } from 'typeorm';
|
|
11
|
+
let BoxEntity = class BoxEntity {
|
|
12
|
+
id;
|
|
13
|
+
address;
|
|
14
|
+
boxId;
|
|
15
|
+
createBlock;
|
|
16
|
+
creationHeight;
|
|
17
|
+
serialized;
|
|
18
|
+
spendBlock;
|
|
19
|
+
spendHeight;
|
|
20
|
+
extractor;
|
|
21
|
+
};
|
|
22
|
+
__decorate([
|
|
23
|
+
PrimaryGeneratedColumn(),
|
|
24
|
+
__metadata("design:type", Number)
|
|
25
|
+
], BoxEntity.prototype, "id", void 0);
|
|
26
|
+
__decorate([
|
|
27
|
+
Column(),
|
|
28
|
+
__metadata("design:type", String)
|
|
29
|
+
], BoxEntity.prototype, "address", void 0);
|
|
30
|
+
__decorate([
|
|
31
|
+
Column(),
|
|
32
|
+
__metadata("design:type", String)
|
|
33
|
+
], BoxEntity.prototype, "boxId", void 0);
|
|
34
|
+
__decorate([
|
|
35
|
+
Column(),
|
|
36
|
+
__metadata("design:type", String)
|
|
37
|
+
], BoxEntity.prototype, "createBlock", void 0);
|
|
38
|
+
__decorate([
|
|
39
|
+
Column(),
|
|
40
|
+
__metadata("design:type", Number)
|
|
41
|
+
], BoxEntity.prototype, "creationHeight", void 0);
|
|
42
|
+
__decorate([
|
|
43
|
+
Column(),
|
|
44
|
+
__metadata("design:type", String)
|
|
45
|
+
], BoxEntity.prototype, "serialized", void 0);
|
|
46
|
+
__decorate([
|
|
47
|
+
Column({ nullable: true, type: 'text' }),
|
|
48
|
+
__metadata("design:type", Object)
|
|
49
|
+
], BoxEntity.prototype, "spendBlock", void 0);
|
|
50
|
+
__decorate([
|
|
51
|
+
Column({ nullable: true }),
|
|
52
|
+
__metadata("design:type", Number)
|
|
53
|
+
], BoxEntity.prototype, "spendHeight", void 0);
|
|
54
|
+
__decorate([
|
|
55
|
+
Column(),
|
|
56
|
+
__metadata("design:type", String)
|
|
57
|
+
], BoxEntity.prototype, "extractor", void 0);
|
|
58
|
+
BoxEntity = __decorate([
|
|
59
|
+
Entity()
|
|
60
|
+
], BoxEntity);
|
|
61
|
+
export { BoxEntity };
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYm94RW50aXR5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vbGliL2VudGl0aWVzL2JveEVudGl0eS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFBQSxPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxzQkFBc0IsRUFBRSxNQUFNLFNBQVMsQ0FBQztBQUcxRCxJQUFNLFNBQVMsR0FBZixNQUFNLFNBQVM7SUFFcEIsRUFBRSxDQUFTO0lBR1gsT0FBTyxDQUFTO0lBR2hCLEtBQUssQ0FBUztJQUdkLFdBQVcsQ0FBUztJQUdwQixjQUFjLENBQVM7SUFHdkIsVUFBVSxDQUFTO0lBR25CLFVBQVUsQ0FBaUI7SUFHM0IsV0FBVyxDQUFVO0lBR3JCLFNBQVMsQ0FBUztDQUNuQixDQUFBO0FBMUJDO0lBQUMsc0JBQXNCLEVBQUU7O3FDQUNkO0FBRVg7SUFBQyxNQUFNLEVBQUU7OzBDQUNPO0FBRWhCO0lBQUMsTUFBTSxFQUFFOzt3Q0FDSztBQUVkO0lBQUMsTUFBTSxFQUFFOzs4Q0FDVztBQUVwQjtJQUFDLE1BQU0sRUFBRTs7aURBQ2M7QUFFdkI7SUFBQyxNQUFNLEVBQUU7OzZDQUNVO0FBRW5CO0lBQUMsTUFBTSxDQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLENBQUM7OzZDQUNkO0FBRTNCO0lBQUMsTUFBTSxDQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxDQUFDOzs4Q0FDTjtBQUVyQjtJQUFDLE1BQU0sRUFBRTs7NENBQ1M7QUExQlAsU0FBUztJQURyQixNQUFNLEVBQUU7R0FDSSxTQUFTLENBMkJyQjtTQTNCWSxTQUFTIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29sdW1uLCBFbnRpdHksIFByaW1hcnlHZW5lcmF0ZWRDb2x1bW4gfSBmcm9tICd0eXBlb3JtJztcblxuQEVudGl0eSgpXG5leHBvcnQgY2xhc3MgQm94RW50aXR5IHtcbiAgQFByaW1hcnlHZW5lcmF0ZWRDb2x1bW4oKVxuICBpZDogbnVtYmVyO1xuXG4gIEBDb2x1bW4oKVxuICBhZGRyZXNzOiBzdHJpbmc7XG5cbiAgQENvbHVtbigpXG4gIGJveElkOiBzdHJpbmc7XG5cbiAgQENvbHVtbigpXG4gIGNyZWF0ZUJsb2NrOiBzdHJpbmc7XG5cbiAgQENvbHVtbigpXG4gIGNyZWF0aW9uSGVpZ2h0OiBudW1iZXI7XG5cbiAgQENvbHVtbigpXG4gIHNlcmlhbGl6ZWQ6IHN0cmluZztcblxuICBAQ29sdW1uKHsgbnVsbGFibGU6IHRydWUsIHR5cGU6ICd0ZXh0JyB9KVxuICBzcGVuZEJsb2NrPzogc3RyaW5nIHwgbnVsbDtcblxuICBAQ29sdW1uKHsgbnVsbGFibGU6IHRydWUgfSlcbiAgc3BlbmRIZWlnaHQ/OiBudW1iZXI7XG5cbiAgQENvbHVtbigpXG4gIGV4dHJhY3Rvcjogc3RyaW5nO1xufVxuIl19
|