From 52ebeee764a97ee94430e466a1354d9e03b0499c Mon Sep 17 00:00:00 2001 From: williamlardier Date: Fri, 17 May 2024 17:07:04 +0200 Subject: [PATCH] S3UTILS-163: adapt and add tests for inflights --- tests/unit/CountItems/CountManager.js | 232 +++++++++++++++++++++++ tests/unit/CountItems/utils/utils.js | 60 ++++++ tests/unit/utils/S3UtilsMongoClient.js | 253 +++++++++++++++++++++++-- 3 files changed, 532 insertions(+), 13 deletions(-) diff --git a/tests/unit/CountItems/CountManager.js b/tests/unit/CountItems/CountManager.js index d323b0ce..fd8b774f 100644 --- a/tests/unit/CountItems/CountManager.js +++ b/tests/unit/CountItems/CountManager.js @@ -199,6 +199,7 @@ describe('CountItems::CountManager', () => { usedCapacity: { current: 200, nonCurrent: 100, + _inflightsPreScan: 0, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 100, @@ -222,6 +223,7 @@ describe('CountItems::CountManager', () => { usedCapacity: { current: 200, nonCurrent: 100, + _inflightsPreScan: 0, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 100, @@ -249,6 +251,7 @@ describe('CountItems::CountManager', () => { usedCapacity: { current: 200, nonCurrent: 100, + _inflightsPreScan: 0, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 100, @@ -274,6 +277,235 @@ describe('CountItems::CountManager', () => { usedCapacity: { current: 200, nonCurrent: 100, + _inflightsPreScan: 0, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + }); + }); + + test('should update dataMetrics with inflights', () => { + const workers = createWorkers(1); + const m = new CountManager({ + log: new DummyLogger(), + workers, + maxConcurrent: 1, + }); + expect(m.dataMetrics).toEqual({ + account: {}, + bucket: {}, + location: {}, + }); + m._consolidateData({ + dataMetrics: { + account: { + account1: { + objectCount: { + current: 10, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 100, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + locations: { + location1: { + objectCount: { + current: 10, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 100, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + }, + }, + bucket: { + bucket1: { + objectCount: { + current: 10, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 100, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + location: { + location1: { + objectCount: { + current: 10, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 100, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + }, + }); + expect(m.dataMetrics).toEqual({ + account: { + account1: { + objectCount: { + current: 11, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + locations: { + location1: { + objectCount: { + current: 11, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + }, + }, + bucket: { + bucket1: { + objectCount: { + current: 11, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 100, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 100, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + location: { + location1: { + objectCount: { + current: 11, + deleteMarker: 0, + nonCurrent: 10, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 1, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 100, + _inflightsPreScan: 1000, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 100, diff --git a/tests/unit/CountItems/utils/utils.js b/tests/unit/CountItems/utils/utils.js index 812c8e06..12a8e913 100644 --- a/tests/unit/CountItems/utils/utils.js +++ b/tests/unit/CountItems/utils/utils.js @@ -5,6 +5,7 @@ describe('CountItems::utils::consolidateDataMetrics', () => { usedCapacity: { current: 0, nonCurrent: 0, + _inflightsPreScan: 0, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 0, @@ -29,6 +30,7 @@ describe('CountItems::utils::consolidateDataMetrics', () => { usedCapacity: { current: 10, nonCurrent: 10, + _inflightsPreScan: 0, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 0, @@ -53,6 +55,7 @@ describe('CountItems::utils::consolidateDataMetrics', () => { usedCapacity: { current: 20, nonCurrent: 20, + _inflightsPreScan: 0, _currentCold: 0, _nonCurrentCold: 0, _currentRestored: 0, @@ -73,6 +76,56 @@ describe('CountItems::utils::consolidateDataMetrics', () => { }, }; + const exampleWithInflights = { + usedCapacity: { + current: 20, + nonCurrent: 20, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + objectCount: { + current: 20, + nonCurrent: 20, + deleteMarker: 20, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }; + + const expectedResponseWithInflights = { + usedCapacity: { + current: 40, + nonCurrent: 40, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + objectCount: { + current: 40, + nonCurrent: 40, + deleteMarker: 40, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }; + test('should return zero-value if target and source are both undefined', () => { const res = consolidateDataMetrics(undefined, undefined); expect(res).toEqual(zeroValueRes); @@ -114,4 +167,11 @@ describe('CountItems::utils::consolidateDataMetrics', () => { const res = consolidateDataMetrics(target, source); expect(res).toEqual(zeroValueRes); }); + + test('should consolidate inflight delta metrics', () => { + const source = exampleWithInflights; + const target = example1; + const res = consolidateDataMetrics(target, source); + expect(res).toEqual(expectedResponseWithInflights); + }); }); diff --git a/tests/unit/utils/S3UtilsMongoClient.js b/tests/unit/utils/S3UtilsMongoClient.js index 2884623f..12bfbc06 100644 --- a/tests/unit/utils/S3UtilsMongoClient.js +++ b/tests/unit/utils/S3UtilsMongoClient.js @@ -1,5 +1,6 @@ global.TextEncoder = require('util').TextEncoder; global.TextDecoder = require('util').TextDecoder; +const sinon = require('sinon'); const async = require('async'); const assert = require('assert'); const werelogs = require('werelogs'); @@ -1003,6 +1004,10 @@ describe('S3UtilsMongoClient, tests', () => { next => repl.stop() .then(() => next()) .catch(next), + next => { + sinon.restore(); + next(); + }, ], done)); const nonVersionedObjectMdTemp = { @@ -2122,12 +2127,227 @@ describe('S3UtilsMongoClient, tests', () => { }, }, ], + [ + 'getObjectMDStats() should return correct results for buckets with inflights', + { + bucketName: 'test-bucket-inflights', + isVersioned: true, + inflights: 1000, + objectList: [ + // versioned object 1, + { + ...objectMdTemp, + versioning: true, + }, + // versioned object 2, + { + ...objectMdTemp, + versioning: true, + }, + // stalled object 1 + { + ...objectMdTemp, + versioning: true, + lastModified: new Date(Date.now() - hr), + repInfo: { + ...objectMdTemp.repInfo, + status: 'PENDING', + backends: [ + { + status: 'PENDING', + site: 'rep-loc-1', + }, + ], + }, + }, + // null versioned object + { + name: 'nullkey', + isNull: true, + ownerId: testAccountCanonicalId, + lastModified: new Date(Date.now() - hr), + }, + ], + }, + { + dataManaged: { + locations: { + 'rep-loc-1': { + curr: 0, + prev: 200, + }, + 'us-east-1': { + curr: 200, + prev: 200, + }, + }, + total: { + curr: 200, + prev: 400, + }, + }, + objects: 2, + stalled: 1, + versions: 2, + dataMetrics: { + account: { + [testAccountCanonicalId]: { + objectCount: { + current: 2, + deleteMarker: 0, + nonCurrent: 2, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 200, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + locations: { + 'rep-loc-1': { + objectCount: { + current: 0, + deleteMarker: 0, + nonCurrent: 2, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 0, + nonCurrent: 200, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + 'us-east-1': { + objectCount: { + current: 2, + deleteMarker: 0, + nonCurrent: 2, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 200, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + }, + }, + bucket: { + [`test-bucket-inflights_${testBucketCreationDate}`]: { + accountOwnerID: 'd1d40abd2bd8250962f7f5774af1bbbeaec9b77a0853749d41ec46f142e66fe4', + objectCount: { + current: 2, + deleteMarker: 0, + nonCurrent: 2, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 200, + _inflightsPreScan: 1000, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + location: { + 'rep-loc-1': { + objectCount: { + current: 0, + deleteMarker: 0, + nonCurrent: 2, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 0, + nonCurrent: 200, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + 'us-east-1': { + objectCount: { + current: 2, + deleteMarker: 0, + nonCurrent: 2, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + usedCapacity: { + current: 200, + nonCurrent: 200, + _currentCold: 0, + _nonCurrentCold: 0, + _currentRestored: 0, + _currentRestoring: 0, + _nonCurrentRestored: 0, + _nonCurrentRestoring: 0, + }, + }, + }, + }, + }, + ], ]; tests.forEach(([msg, testCase, expected]) => it(msg, done => { const { bucketName, isVersioned, objectList, + inflights, } = testCase; return async.waterfall([ next => createBucket(client, bucketName, isVersioned, err => next(err)), @@ -2144,19 +2364,26 @@ describe('S3UtilsMongoClient, tests', () => { ), next => uploadObjects(client, bucketName, objectList, err => next(err)), next => client.getBucketAttributes(bucketName, logger, next), - (bucketInfo, next) => client.getObjectMDStats( - bucketName, - BucketInfo.fromObj(bucketInfo), - false, - logger, - (err, res) => { - if (err) { - return next(err); - } - assert.deepStrictEqual(res, expected); - return next(); - }, - ), + (bucketInfo, next) => { + if (inflights) { + const mock = sinon.stub(client, 'readStorageConsumptionInflights'); + mock.onFirstCall().returns(Promise.resolve(inflights)); + mock.onSecondCall().returns(Promise.resolve(inflights * 1.5)); + } + return client.getObjectMDStats( + bucketName, + BucketInfo.fromObj(bucketInfo), + false, + logger, + (err, res) => { + if (err) { + return next(err); + } + assert.deepStrictEqual(res, expected); + return next(); + }, + ); + }, next => client.deleteBucket(bucketName, logger, next), ], done); }));