diff --git a/app/lib/dal/indexDAL/leveldb/LevelDBSindex.ts b/app/lib/dal/indexDAL/leveldb/LevelDBSindex.ts index 7c9117cdbb98ab0044c1028ea2b2dcae2cd797b5..4fd8de92bfd2fa6ff76002c5059d67831a0139b4 100644 --- a/app/lib/dal/indexDAL/leveldb/LevelDBSindex.ts +++ b/app/lib/dal/indexDAL/leveldb/LevelDBSindex.ts @@ -16,9 +16,13 @@ import { CommonConstants } from "../../../common-libs/constants"; export class LevelDBSindex extends LevelDBTable<SindexEntry> implements SIndexDAO { + // Remembers what sources (identifier-pos, e.g. E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855-0) were either created and/or consumed at a given block number private indexForTrimming: LevelDBTable<string[]>; + // Remembers what sources (identifier-pos, e.g. E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855-0) were consumed at a given block number private indexForConsumed: LevelDBTable<string[]>; + // Remembers what sources (identifier-pos, e.g. E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855-0) are involved in a simple condition (e.g. "SIG(pubkey)") private indexForConditions: LevelDBTable<string[]>; + // Remembers what sources (identifier-pos, e.g. E3B0C44298FC1C149AFBF4C8996FB92427AE41E4649B934CA495991B7852B855-0) are involved in a complex condition (e.g. "SIG(pubkey) OR CSV(1000)") private indexOfComplexeConditionForPubkeys: LevelDBTable<string[]>; constructor(protected getLevelDB: (dbName: string) => Promise<LevelUp>) { @@ -171,6 +175,13 @@ export class LevelDBSindex extends LevelDBTable<SindexEntry> return entries; } + /** + * Trims (i.e. removes definitively) some sources from sindex. + * N.B: we only trim *consumed sources* because a source is exclusively a two entries operation: CREATE then UPDATE. + * So a source without UPDATE is not to be trimmed, because it has never been consumed. + * Looking into `indexForConsumed` is therefore an optimisation to avoid checking `CREATE` sources which cannot be trimmed. + * @param belowNumber Trim all the UPDATE sources written below `belowNumber` block. + */ async trimConsumedSource(belowNumber: number): Promise<void> { let belowNumberIds: string[] = []; const mapIds: { @@ -182,7 +193,7 @@ export class LevelDBSindex extends LevelDBTable<SindexEntry> } = {}; const mapIds2WrittenOn: { [k: string]: number } = {}; - // First: we look at what was written before `belowNumber` + // First: we look at what was consumed before `belowNumber` await this.indexForConsumed.readAllKeyValue( async (kv) => { belowNumberIds = belowNumberIds.concat(kv.value); @@ -211,12 +222,13 @@ export class LevelDBSindex extends LevelDBTable<SindexEntry> updatedOn: updateRecord.writtenOn, conditions: updateRecord.conditions, }; + // Remove both the source CREATE (availability) and UPDATE (consumption) from sindex await this.del(createKey); await this.del(updateKey); } } - // We update indexes + // We update sub-indexes for (const id of Underscore.keys(mapIds).map(String)) { const map = mapIds[id]; await this.trimConditions(map.conditions, id); diff --git a/test/dal/triming-dal.ts b/test/dal/triming-dal.ts index eafe4f0b62460a7b364fd8c6be56bc7ae96ee9fd..1fc86192e7d4018ed7ccc3c3cb6d5b4a98c26b86 100644 --- a/test/dal/triming-dal.ts +++ b/test/dal/triming-dal.ts @@ -145,7 +145,7 @@ describe("Triming", function(){ should.not.exist(await sindexDAL.getIndexForConsumed().getOrNull("0000000139")); // The only consumption at this block was trimmed let entriesAt0000000126 = await sindexDAL.getIndexForTrimming().getOrNull("0000000126"); - // FIXME another issue here + // https://git.duniter.org/nodes/typescript/duniter/-/issues/1447 entriesAt0000000126.should.not.containEql('SOURCE_1-0000000004'); // This CREATE entry should have been trimmed let entriesAt0000000139 = await sindexDAL.getIndexForTrimming().getOrNull("0000000139"); @@ -166,6 +166,7 @@ describe("Triming", function(){ // All sub index should be empty for (let index of sindexDAL.getInternalIndexes()) { const res = await index.findAllKeys(); + // https://git.duniter.org/nodes/typescript/duniter/-/issues/1447 res.should.have.length(0, `index ${index['name']} should have been trimmed`); } })