From c43adf52b62d5438f367fceebb5e1a417bc4bba4 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 10:57:50 -0600 Subject: [PATCH 01/17] Added readTouchTtlPercent to various policies CLIENT-2829 readTouchTtlPolicies was added to the following policies: operate read, batch_read batch Unit tests added to specified test files Added API documentation --- .gitmodules | 2 +- aerospike-client-c | 2 +- lib/policies/batch_policy.js | 16 +++ lib/policies/batch_read_policy.js | 15 +++ lib/policies/operate_policy.js | 15 +++ lib/policies/read_policy.js | 15 +++ src/main/policy.cc | 23 +++- test/batch_read.js | 174 ++++++++++++++++++++++++++++++ test/get.js | 84 +++++++++++++++ test/operate.js | 82 ++++++++++++++ 10 files changed, 425 insertions(+), 3 deletions(-) diff --git a/.gitmodules b/.gitmodules index fdf1dc411..be8c9ebcd 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,4 +1,4 @@ [submodule "aerospike-client-c"] path = aerospike-client-c url = https://github.com/aerospike/aerospike-client-c.git - branch = master + branch = stage diff --git a/aerospike-client-c b/aerospike-client-c index 029db7ac6..49b8298df 160000 --- a/aerospike-client-c +++ b/aerospike-client-c @@ -1 +1 @@ -Subproject commit 029db7ac63ba3533150c359e0dec5a51e54914ab +Subproject commit 49b8298dfeb65f812be03544b907800054e827b7 diff --git a/lib/policies/batch_policy.js b/lib/policies/batch_policy.js index 245812dba..416ead24b 100644 --- a/lib/policies/batch_policy.js +++ b/lib/policies/batch_policy.js @@ -58,6 +58,22 @@ class BatchPolicy extends BasePolicy { * @see {@link module:aerospike/policy.readModeSC} for supported policy values. */ this.readModeSC = props.readModeSC + + /** + * Determine how record TTL (time to live) is affected on reads. When enabled, the server can + * efficiently operate as a read-based LRU cache where the least recently used records are expired. + * The value is expressed as a percentage of the TTL sent on the most recent write such that a read + * within this interval of the record’s end of life will generate a touch. + * + * For example, if the most recent write had a TTL of 10 hours and read_touch_ttl_percent is set to + * 80, the next read within 8 hours of the record's end of life (equivalent to 2 hours after the most + * recent write) will result in a touch, resetting the TTL to another 10 hours. + * * + * @type number + * @default 0 + */ + this.readTouchTtlPercent = props.readTouchTtlPercent + /** * Determine if batch commands to each server are run in parallel threads. * diff --git a/lib/policies/batch_read_policy.js b/lib/policies/batch_read_policy.js index 21225d9c9..dd11f3f7d 100644 --- a/lib/policies/batch_read_policy.js +++ b/lib/policies/batch_read_policy.js @@ -52,6 +52,21 @@ class BatchReadPolicy { * @see {@link module:aerospike/policy.readModeSC} for supported policy values. */ this.readModeSC = props.readModeSC + + /** + * Determine how record TTL (time to live) is affected on reads. When enabled, the server can + * efficiently operate as a read-based LRU cache where the least recently used records are expired. + * The value is expressed as a percentage of the TTL sent on the most recent write such that a read + * within this interval of the record’s end of life will generate a touch. + * + * For example, if the most recent write had a TTL of 10 hours and read_touch_ttl_percent is set to + * 80, the next read within 8 hours of the record's end of life (equivalent to 2 hours after the most + * recent write) will result in a touch, resetting the TTL to another 10 hours. + * * + * @type number + * @default 0 + */ + this.readTouchTtlPercent = props.readTouchTtlPercent } } diff --git a/lib/policies/operate_policy.js b/lib/policies/operate_policy.js index 611c28c5c..32ff72289 100644 --- a/lib/policies/operate_policy.js +++ b/lib/policies/operate_policy.js @@ -113,6 +113,21 @@ class OperatePolicy extends BasePolicy { * @see {@link module:aerospike/policy.readModeSC} for supported policy values. */ this.readModeSC = props.readModeSC + + /** + * Determine how record TTL (time to live) is affected on reads. When enabled, the server can + * efficiently operate as a read-based LRU cache where the least recently used records are expired. + * The value is expressed as a percentage of the TTL sent on the most recent write such that a read + * within this interval of the record’s end of life will generate a touch. + * + * For example, if the most recent write had a TTL of 10 hours and read_touch_ttl_percent is set to + * 80, the next read within 8 hours of the record's end of life (equivalent to 2 hours after the most + * recent write) will result in a touch, resetting the TTL to another 10 hours. + * * + * @type number + * @default 0 + */ + this.readTouchTtlPercent = props.readTouchTtlPercent } } diff --git a/lib/policies/read_policy.js b/lib/policies/read_policy.js index 5241cd155..afe556c79 100644 --- a/lib/policies/read_policy.js +++ b/lib/policies/read_policy.js @@ -68,6 +68,21 @@ class ReadPolicy extends BasePolicy { */ this.readModeSC = props.readModeSC + /** + * Determine how record TTL (time to live) is affected on reads. When enabled, the server can + * efficiently operate as a read-based LRU cache where the least recently used records are expired. + * The value is expressed as a percentage of the TTL sent on the most recent write such that a read + * within this interval of the record’s end of life will generate a touch. + * + * For example, if the most recent write had a TTL of 10 hours and read_touch_ttl_percent is set to + * 80, the next read within 8 hours of the record's end of life (equivalent to 2 hours after the most + * recent write) will result in a touch, resetting the TTL to another 10 hours. + * * + * @type number + * @default 0 + */ + this.readTouchTtlPercent = props.readTouchTtlPercent + /** * Should CDT data types (Lists / Maps) be deserialized to JS data types * (Arrays / Objects) or returned as raw bytes (Buffer). diff --git a/src/main/policy.cc b/src/main/policy.cc index a4ba8bd4e..bb56cd74f 100644 --- a/src/main/policy.cc +++ b/src/main/policy.cc @@ -130,6 +130,11 @@ int readpolicy_from_jsobject(as_policy_read *policy, Local obj, AS_NODE_PARAM_OK) { return rc; } + if ((rc = get_optional_int_property((int *)&policy->read_touch_ttl_percent, + NULL, obj, "readTouchTtlPercent", log)) != + AS_NODE_PARAM_OK) { + return rc; + } if ((rc = get_optional_bool_property(&policy->deserialize, NULL, obj, "deserialize", log)) != AS_NODE_PARAM_OK) { @@ -249,6 +254,11 @@ int operatepolicy_from_jsobject(as_policy_operate *policy, Local obj, AS_NODE_PARAM_OK) { return rc; } + if ((rc = get_optional_int_property((int *)&policy->read_touch_ttl_percent, + NULL, obj, "readTouchTtlPercent", log)) != + AS_NODE_PARAM_OK) { + return rc; + } if ((rc = get_optional_uint32_property((uint32_t *)&policy->commit_level, NULL, obj, "commitLevel", log)) != AS_NODE_PARAM_OK) { @@ -342,6 +352,13 @@ int batchpolicy_from_jsobject(as_policy_batch *policy, Local obj, AS_NODE_PARAM_OK) { return rc; } + printf("read_touch_ttl_percent: %d\n\n", policy->read_touch_ttl_percent); + if ((rc = get_optional_int_property((int *)&policy->read_touch_ttl_percent, + NULL, obj, "readTouchTtlPercent", log)) != + AS_NODE_PARAM_OK) { + return rc; + } + printf("read_touch_ttl_percent: %d\n\n", policy->read_touch_ttl_percent); if ((rc = get_optional_bool_property(&policy->concurrent, NULL, obj, "concurrent", log)) != AS_NODE_PARAM_OK) { @@ -405,7 +422,11 @@ int batchread_policy_from_jsobject(as_policy_batch_read *policy, AS_NODE_PARAM_OK) { return rc; } - + if ((rc = get_optional_int_property((int *)&policy->read_touch_ttl_percent, + NULL, obj, "readTouchTtlPercent", log)) != + AS_NODE_PARAM_OK) { + return rc; + } return rc; } diff --git a/test/batch_read.js b/test/batch_read.js index fa988caac..bf61c9c77 100644 --- a/test/batch_read.js +++ b/test/batch_read.js @@ -181,6 +181,180 @@ describe('client.batchRead()', function () { }) }) }) + + context('readTouchTtlPercent policy', function () { + let batch = [{ + key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), + readAllBins: true + }] + context('BatchReadPolicy policy', function () { + it('100% touches record', async function () { + const policy = new Aerospike.BatchReadPolicy({ + readTouchTtlPercent: 100 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch, policy) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + + it('91% touches record', async function () { + const policy = new Aerospike.BatchReadPolicy({ + readTouchTtlPercent: 91 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch, policy) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + + it('70% doesnt touch record', async function () { + const policy = new Aerospike.BatchReadPolicy({ + readTouchTtlPercent: 70 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch, policy) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + + it('0% doesnt touch record', async function () { + const policy = new Aerospike.BatchReadPolicy({ + readTouchTtlPercent: 0 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch, policy) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + }) + + context('BatchReadPolicy policy', function () { + it('100% touches record', async function () { + batch = [{ + key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), + readAllBins: true, + policy: new Aerospike.BatchPolicy({ + readTouchTtlPercent: 100 + }) + }] + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + + it('91% touches record', async function () { + batch = [{ + key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), + readAllBins: true, + policy: new Aerospike.BatchPolicy({ + readTouchTtlPercent: 91 + }) + }] + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + + it('70% doesnt touch record', async function () { + batch = [{ + key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), + readAllBins: true, + policy: new Aerospike.BatchPolicy({ + readTouchTtlPercent: 70 + }) + }] + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + + it('0% doesnt touch record', async function () { + batch = [{ + key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), + readAllBins: true, + policy: new Aerospike.BatchPolicy({ + readTouchTtlPercent: 0 + }) + }] + await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + const batchResult = await client.batchRead(batch) + expect(batchResult[0].record.bins).to.eql({ i: 2 }) + expect(batchResult[0].record.ttl).to.equal(9) + + const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) + }) + }) + }) }) it('returns a Promise that resolves to the batch results', function () { diff --git a/test/get.js b/test/get.js index 3c398c835..a85c4ca31 100644 --- a/test/get.js +++ b/test/get.js @@ -82,6 +82,90 @@ describe('client.get()', function () { }) }) }) + + context('readTouchTtlPercent policy', function () { + it('100% touches record', async function () { + const key = keygen.integer(helper.namespace, helper.set)() + const policy = new Aerospike.ReadPolicy({ + readTouchTtlPercent: 100 + }) + + await client.put(key, { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + let record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(key) + }) + + it('91% touches record', async function () { + const key = keygen.integer(helper.namespace, helper.set)() + const policy = new Aerospike.ReadPolicy({ + readTouchTtlPercent: 91 + }) + + await client.put(key, { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + let record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(key) + }) + + it('70% never touches record', async function () { + const key = keygen.integer(helper.namespace, helper.set)() + const policy = new Aerospike.ReadPolicy({ + readTouchTtlPercent: 70 + }) + await client.put(key, { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + let record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + await client.remove(key) + }) + + it('0% never touches record', async function () { + const key = keygen.integer(helper.namespace, helper.set)() + const policy = new Aerospike.ReadPolicy({ + readTouchTtlPercent: 0 + }) + await client.put(key, { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + let record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(key, policy) + + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + await client.remove(key) + }) + }) }) it('should return the TTL for a never expiring record as Aerospike.ttl.NEVER_EXPIRE', function (done) { diff --git a/test/operate.js b/test/operate.js index 681d1f9fc..607f46fd7 100644 --- a/test/operate.js +++ b/test/operate.js @@ -281,6 +281,88 @@ context('Operations', function () { }) }) + context('readTouchTtlPercent policy', function () { + it('100% touches record', async function () { + const ops = [op.read('i')] + const policy = new Aerospike.OperatePolicy({ + readTouchTtlPercent: 100 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) + }) + + it('91% touches record', async function () { + const ops = [op.read('i')] + const policy = new Aerospike.OperatePolicy({ + readTouchTtlPercent: 91 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(10) + + await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) + }) + + it('70% does not touch record', async function () { + const ops = [op.read('i')] + const policy = new Aerospike.OperatePolicy({ + readTouchTtlPercent: 70 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) + }) + + it('0% does not touch record', async function () { + const ops = [op.read('i')] + const policy = new Aerospike.OperatePolicy({ + readTouchTtlPercent: 0 + }) + + await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) + await new Promise(resolve => setTimeout(resolve, 1000)) + + let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) + expect(record.bins).to.eql({ i: 2 }) + expect(record.ttl).to.equal(9) + + await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) + }) + }) + context('gen policy', function () { context('policy.gen.EQ', function () { const policy = new Aerospike.OperatePolicy({ From 55f4404ed11820243c05930e26dd7f766628208d Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 11:10:53 -0600 Subject: [PATCH 02/17] Added expectedDuration as QueryPolicy member CLIENT-2822 Added expectedDuration as QueryPolicy member Added queryDuration enum to aerospike.policy Added unit testing for expectedDuration in queries --- .gitmodules | 2 +- aerospike-client-c | 2 +- binding.gyp | 3 +- lib/policies/query_policy.js | 10 +++++ lib/policy.js | 8 ++++ lib/query_duration.js | 71 ++++++++++++++++++++++++++++++++ src/include/enums.h | 1 + src/main/aerospike.cc | 1 + src/main/enums/query_duration.cc | 38 +++++++++++++++++ src/main/policy.cc | 5 +++ test/query.js | 37 +++++++++++++++++ typings/index.d.ts | 5 +++ 12 files changed, 180 insertions(+), 3 deletions(-) create mode 100644 lib/query_duration.js create mode 100644 src/main/enums/query_duration.cc diff --git a/.gitmodules b/.gitmodules index fdf1dc411..be8c9ebcd 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,4 +1,4 @@ [submodule "aerospike-client-c"] path = aerospike-client-c url = https://github.com/aerospike/aerospike-client-c.git - branch = master + branch = stage diff --git a/aerospike-client-c b/aerospike-client-c index 029db7ac6..49b8298df 160000 --- a/aerospike-client-c +++ b/aerospike-client-c @@ -1 +1 @@ -Subproject commit 029db7ac63ba3533150c359e0dec5a51e54914ab +Subproject commit 49b8298dfeb65f812be03544b907800054e827b7 diff --git a/binding.gyp b/binding.gyp index e9fcc6dc5..cebead72c 100644 --- a/binding.gyp +++ b/binding.gyp @@ -144,13 +144,14 @@ 'src/main/enums/config_enum.cc', 'src/main/enums/exp_enum.cc', 'src/main/enums/batch_type.cc', + 'src/main/enums/query_duration.cc', 'src/main/enums/privilege_code.cc', 'src/main/enums/exp_read_flags.cc', 'src/main/enums/exp_write_flags.cc', 'src/main/stats.cc', 'src/main/util/conversions.cc', 'src/main/util/conversions_batch.cc', - 'src/main/util/log.cc' + 'src/main/util/log.cc', ], 'configurations': { 'Release': { diff --git a/lib/policies/query_policy.js b/lib/policies/query_policy.js index bc9da5a81..4c94c62af 100644 --- a/lib/policies/query_policy.js +++ b/lib/policies/query_policy.js @@ -95,6 +95,16 @@ class QueryPolicy extends BasePolicy { * @override */ this.totalTimeout = props.totalTimeout + + /** + * Expected query duration. The server treats the query in different ways depending on the expected duration. + * This field is ignored for aggregation queries, background queries and server versions < 6.0. + * + * @see {@link module:aerospike/policy.queryDuration} for supported policy values. + * @type {@link module:aerospike/policy.queryDuration} + * @default {@link module:aerospike/policy.queryDuration.LONG} + */ + this.expectedDuration = props.expectedDuration } } diff --git a/lib/policy.js b/lib/policy.js index b91e02976..5414512c1 100644 --- a/lib/policy.js +++ b/lib/policy.js @@ -296,6 +296,14 @@ exports.readModeSC = as.policy.readModeSC */ exports.commitLevel = as.policy.commitLevel +/** + * The {@link module:aerospike/policy.queryDuration|aerospike/policy.query_duration} + * module contains a list of query duration enumerations. + * + * @summary {@link module:aerospike/policy.queryDuration|aerospike/policy.query_duration} module + */ +exports.queryDuration = require('./query_duration') + /** * A base class extended to client policies. * diff --git a/lib/query_duration.js b/lib/query_duration.js new file mode 100644 index 000000000..bf8624f74 --- /dev/null +++ b/lib/query_duration.js @@ -0,0 +1,71 @@ +// ***************************************************************************** +// Copyright 2022-2023 Aerospike, Inc. +// +// Licensed under the Apache License, Version 2.0 (the 'License') +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an 'AS IS' BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// ***************************************************************************** + +'use strict' + +const as = require('bindings')('aerospike.node') +const queryDuration = as.queryDuration + +/** + * @module aerospike/queryDuration + * + * @description Expected query duration. The server treats the query in different ways depending on the expected duration. + * This enum is ignored for aggregation (UDF) queries, background queries and server versions < 6.0. + */ + +// ======================================================================== +// Constants +// ======================================================================== +module.exports = { + /** + * The query is expected to return more than 100 records per node. The server optimizes for a + * large record set in the following ways: + *
    + *
  • Allow query to be run in multiple threads using the server's query threading configuration.
  • + *
  • Do not relax read consistency for AP namespaces.
  • + *
  • Add the query to the server's query monitor.
  • + *
  • Do not add the overall latency to the server's latency histogram.
  • + *
  • Do not allow server timeouts.
  • + *
+ * + * @const {number} + */ + LONG: queryDuration.LONG, + + /** + * The query is expected to return less than 100 records per node. The server optimizes for a + * small record set in the following ways: + *
    + *
  • Always run the query in one thread and ignore the server's query threading configuration.
  • + *
  • Allow query to be inlined directly on the server's service thread.
  • + *
  • Relax read consistency for AP namespaces.
  • + *
  • Do not add the query to the server's query monitor.
  • + *
  • Add the overall latency to the server's latency histogram.
  • + *
  • Allow server timeouts. The default server timeout for a short query is 1 second.
  • + *
+ * + * @const {number} + */ + SHORT: queryDuration.SHORT, + + /** + * Treat query as a LONG query, but relax read consistency for AP namespaces. + * This value is treated exactly like LONG for server versions < 7.1. + * + * @const {number} + */ + LONG_RELAX_AP: queryDuration.LONG_RELAX_AP +} diff --git a/src/include/enums.h b/src/include/enums.h index 2a5ac0d64..a6fdb3be7 100644 --- a/src/include/enums.h +++ b/src/include/enums.h @@ -41,6 +41,7 @@ v8::Local retry_policy_values(); v8::Local status(); v8::Local ttl_enum_values(); v8::Local batchTypes(); +v8::Local queryDuration(); v8::Local privilegeCode(); v8::Local expReadFlags(); v8::Local expWriteFlags(); diff --git a/src/main/aerospike.cc b/src/main/aerospike.cc index a4f2b4758..6f5fe76f0 100644 --- a/src/main/aerospike.cc +++ b/src/main/aerospike.cc @@ -156,6 +156,7 @@ NAN_MODULE_INIT(Aerospike) export("ttl", ttl_enum_values()); export("auth", auth_mode_enum_values()); export("batchTypes", batchTypes()); + export("queryDuration", queryDuration()); export("privilegeCode", privilegeCode()); export("expReadFlags", expReadFlags()); export("expWriteFlags", expWriteFlags()); diff --git a/src/main/enums/query_duration.cc b/src/main/enums/query_duration.cc new file mode 100644 index 000000000..0da8c4115 --- /dev/null +++ b/src/main/enums/query_duration.cc @@ -0,0 +1,38 @@ +/******************************************************************************* + * Copyright 2023 Aerospike, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + ******************************************************************************/ + +#include +#include + +extern "C" { +#include +} + +using namespace v8; + +#define set(__obj, __name, __value) \ + Nan::Set(__obj, Nan::New(__name).ToLocalChecked(), Nan::New(__value)) + +Local queryDuration() +{ + Nan::EscapableHandleScope scope; + Local obj = Nan::New(); + set(obj, "LONG", AS_QUERY_DURATION_LONG); + set(obj, "SHORT", AS_QUERY_DURATION_SHORT); + set(obj, "LONG_RELAX_AP", AS_QUERY_DURATION_LONG_RELAX_AP); + + return scope.Escape(obj); +} \ No newline at end of file diff --git a/src/main/policy.cc b/src/main/policy.cc index a4ba8bd4e..d5694c64b 100644 --- a/src/main/policy.cc +++ b/src/main/policy.cc @@ -581,6 +581,11 @@ int querypolicy_from_jsobject(as_policy_query *policy, Local obj, AS_NODE_PARAM_OK) { return rc; } + if ((rc = get_optional_uint32_property((uint32_t *)&policy->expected_duration, NULL, obj, + "expectedDuration", log)) != + AS_NODE_PARAM_OK) { + return rc; + } if ((rc = get_optional_bool_property(&policy->fail_on_cluster_change, NULL, obj, "failOnClusterChange", log)) != AS_NODE_PARAM_OK) { return rc; } diff --git a/test/query.js b/test/query.js index 51fec8e0c..2c2eee88e 100644 --- a/test/query.js +++ b/test/query.js @@ -274,6 +274,43 @@ describe('Queries', function () { }) }) + it('Should run a regular primary index query with expectedDuration=LONG', function (done) { + const query = client.query(helper.namespace, testSet) + + const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG }) + const results = [] + stream.on('error', error => { throw error }) + stream.on('data', record => results.push(record.bins)) + stream.on('end', () => { + expect(results.length).to.be.above(samples.length) + done() + }) + }) + + it('Should run a regular primary index query with expectedDuration=SHORT', function (done) { + const query = client.query(helper.namespace, testSet) + const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.SHORT }) + const results = [] + stream.on('error', error => { throw error }) + stream.on('data', record => results.push(record.bins)) + stream.on('end', () => { + expect(results.length).to.be.above(samples.length) + done() + }) + }) + + it('Should run a regular primary index query with expectedDuration=LONG_RELAX_AP', function (done) { + const query = client.query(helper.namespace, testSet) + const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG_RELAX_AP }) + const results = [] + stream.on('error', error => { throw error }) + stream.on('data', record => results.push(record.bins)) + stream.on('end', () => { + expect(results.length).to.be.above(samples.length) + done() + }) + }) + it('Should run a paginated primary index query', async function () { let recordTotal = 0 let recordsReceived = 0 diff --git a/typings/index.d.ts b/typings/index.d.ts index 28dde3ccc..aa3b3e301 100644 --- a/typings/index.d.ts +++ b/typings/index.d.ts @@ -1653,6 +1653,11 @@ declare module 'query' { import RecordStream = require("record_stream"); } +declare module 'query_duration' { + export const LONG: any; + export const SHORT: any; + export const LONG_RELAX_AP: any; +} declare module 'record' { export = Record; class Record { From e6578f7e5c6b31defb0bcf4fd154338176c0ecdf Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 11:52:28 -0600 Subject: [PATCH 03/17] Fixed Type error for Mac compilation Fixed Type error for Mac compilation Ran npm update --- package-lock.json | 977 +++++++++++++++++++++---------------- package.json | 12 +- src/main/bit_operations.cc | 2 +- typings/index.d.ts | 1 + 4 files changed, 564 insertions(+), 428 deletions(-) diff --git a/package-lock.json b/package-lock.json index b82858307..39fad55d7 100644 --- a/package-lock.json +++ b/package-lock.json @@ -22,24 +22,24 @@ "@mapbox/node-pre-gyp": "^1.0.11", "bindings": "^1.5.0", "minimatch": "^3.1.2", - "nan": "^2.18.0", - "node-gyp": "^10.0.1", + "nan": "^2.19.0", + "node-gyp": "^10.1.0", "npm-run-all": "^4.1.5" }, "devDependencies": { - "@types/node": "^20.11.18", - "chai": "^4.3.7", + "@types/node": "^20.12.7", + "chai": "^4.4.1", "choma": "^1.2.1", "codecov": "^3.8.3", "deep-eql": "^4.1.3", "husky": "^9.0.11", - "mocha": "^10.3.0", + "mocha": "^10.4.0", "mocha-clean": "^1.0.0", "nyc": "^15.1.0", "p-throttle": "^3.1.0", "semver": "^7.6.0", "standard": "^17.1.0", - "tmp": "^0.2.1", + "tmp": "^0.2.3", "yargs": "^17.7.2" }, "engines": { @@ -56,56 +56,56 @@ } }, "node_modules/@ampproject/remapping": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.2.1.tgz", - "integrity": "sha512-lFMjJTrFL3j7L9yBxwYfCq2k6qqwHyzuUl/XBnif78PWTJYyL/dfowQHWE3sp6U6ZzqWiiIZnpTMO96zhkjwtg==", + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.3.0.tgz", + "integrity": "sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw==", "dev": true, "dependencies": { - "@jridgewell/gen-mapping": "^0.3.0", - "@jridgewell/trace-mapping": "^0.3.9" + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.24" }, "engines": { "node": ">=6.0.0" } }, "node_modules/@babel/code-frame": { - "version": "7.23.5", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.23.5.tgz", - "integrity": "sha512-CgH3s1a96LipHCmSUmYFPwY7MNx8C3avkq7i4Wl3cfa662ldtUe4VM1TPXX70pfmrlWTb6jLqTYrZyT2ZTJBgA==", + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.2.tgz", + "integrity": "sha512-y5+tLQyV8pg3fsiln67BVLD1P13Eg4lh5RW9mF0zUuvLrv9uIQ4MCL+CRT+FTsBlBjcIan6PGsLcBN0m3ClUyQ==", "dev": true, "dependencies": { - "@babel/highlight": "^7.23.4", - "chalk": "^2.4.2" + "@babel/highlight": "^7.24.2", + "picocolors": "^1.0.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/compat-data": { - "version": "7.23.5", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.23.5.tgz", - "integrity": "sha512-uU27kfDRlhfKl+w1U6vp16IuvSLtjAxdArVXPa9BvLkrr7CYIsxH5adpHObeAGY/41+syctUWOZ140a2Rvkgjw==", + "version": "7.24.4", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.24.4.tgz", + "integrity": "sha512-vg8Gih2MLK+kOkHJp4gBEIkyaIi00jgWot2D9QOmmfLC8jINSOzmCLta6Bvz/JSBCqnegV0L80jhxkol5GWNfQ==", "dev": true, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/core": { - "version": "7.23.9", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.23.9.tgz", - "integrity": "sha512-5q0175NOjddqpvvzU+kDiSOAk4PfdO6FvwCWoQ6RO7rTzEe8vlo+4HVfcnAREhD4npMs0e9uZypjTwzZPCf/cw==", + "version": "7.24.4", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.24.4.tgz", + "integrity": "sha512-MBVlMXP+kkl5394RBLSxxk/iLTeVGuXTV3cIDXavPpMMqnSnt6apKgan/U8O3USWZCWZT/TbgfEpKa4uMgN4Dg==", "dev": true, "dependencies": { "@ampproject/remapping": "^2.2.0", - "@babel/code-frame": "^7.23.5", - "@babel/generator": "^7.23.6", + "@babel/code-frame": "^7.24.2", + "@babel/generator": "^7.24.4", "@babel/helper-compilation-targets": "^7.23.6", "@babel/helper-module-transforms": "^7.23.3", - "@babel/helpers": "^7.23.9", - "@babel/parser": "^7.23.9", - "@babel/template": "^7.23.9", - "@babel/traverse": "^7.23.9", - "@babel/types": "^7.23.9", + "@babel/helpers": "^7.24.4", + "@babel/parser": "^7.24.4", + "@babel/template": "^7.24.0", + "@babel/traverse": "^7.24.1", + "@babel/types": "^7.24.0", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", @@ -136,14 +136,14 @@ } }, "node_modules/@babel/generator": { - "version": "7.23.6", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.23.6.tgz", - "integrity": "sha512-qrSfCYxYQB5owCmGLbl8XRpX1ytXlpueOb0N0UmQwA073KZxejgQTzAmJezxvpwQD9uGtK2shHdi55QT+MbjIw==", + "version": "7.24.4", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.24.4.tgz", + "integrity": "sha512-Xd6+v6SnjWVx/nus+y0l1sxMOTOMBkyL4+BIdbALyatQnAe/SRVjANeDPSCYaX+i1iJmuGSKf3Z+E+V/va1Hvw==", "dev": true, "dependencies": { - "@babel/types": "^7.23.6", - "@jridgewell/gen-mapping": "^0.3.2", - "@jridgewell/trace-mapping": "^0.3.17", + "@babel/types": "^7.24.0", + "@jridgewell/gen-mapping": "^0.3.5", + "@jridgewell/trace-mapping": "^0.3.25", "jsesc": "^2.5.1" }, "engines": { @@ -225,12 +225,12 @@ } }, "node_modules/@babel/helper-module-imports": { - "version": "7.22.15", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.22.15.tgz", - "integrity": "sha512-0pYVBnDKZO2fnSPCrgM/6WMc7eS20Fbok+0r88fp+YtWVLZrp4CkafFGIp+W0VKw4a22sgebPT99y+FDNMdP4w==", + "version": "7.24.3", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.3.tgz", + "integrity": "sha512-viKb0F9f2s0BCS22QSF308z/+1YWKV/76mwt61NBzS5izMzDPwdq1pTrzf+Li3npBWX9KdQbkeCt1jSAM7lZqg==", "dev": true, "dependencies": { - "@babel/types": "^7.22.15" + "@babel/types": "^7.24.0" }, "engines": { "node": ">=6.9.0" @@ -280,9 +280,9 @@ } }, "node_modules/@babel/helper-string-parser": { - "version": "7.23.4", - "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.23.4.tgz", - "integrity": "sha512-803gmbQdqwdf4olxrX4AJyFBV/RTr3rSmOj0rKwesmzlfhYNDEs+/iOcznzpNWlJlIlTJC2QfPFcHB6DlzdVLQ==", + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.1.tgz", + "integrity": "sha512-2ofRCjnnA9y+wk8b9IAREroeUP02KHp431N2mhKniy2yKIDKpbrHv9eXwm8cBeWQYcJmzv5qKCu65P47eCF7CQ==", "dev": true, "engines": { "node": ">=6.9.0" @@ -307,37 +307,38 @@ } }, "node_modules/@babel/helpers": { - "version": "7.23.9", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.23.9.tgz", - "integrity": "sha512-87ICKgU5t5SzOT7sBMfCOZQ2rHjRU+Pcb9BoILMYz600W6DkVRLFBPwQ18gwUVvggqXivaUakpnxWQGbpywbBQ==", + "version": "7.24.4", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.24.4.tgz", + "integrity": "sha512-FewdlZbSiwaVGlgT1DPANDuCHaDMiOo+D/IDYRFYjHOuv66xMSJ7fQwwODwRNAPkADIO/z1EoF/l2BCWlWABDw==", "dev": true, "dependencies": { - "@babel/template": "^7.23.9", - "@babel/traverse": "^7.23.9", - "@babel/types": "^7.23.9" + "@babel/template": "^7.24.0", + "@babel/traverse": "^7.24.1", + "@babel/types": "^7.24.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/highlight": { - "version": "7.23.4", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.23.4.tgz", - "integrity": "sha512-acGdbYSfp2WheJoJm/EBBBLh/ID8KDc64ISZ9DYtBmC8/Q204PZJLHyzeB5qMzJ5trcOkybd78M4x2KWsUq++A==", + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.24.2.tgz", + "integrity": "sha512-Yac1ao4flkTxTteCDZLEvdxg2fZfz1v8M4QpaGypq/WPDqg3ijHYbDfs+LG5hvzSoqaSZ9/Z9lKSP3CjZjv+pA==", "dev": true, "dependencies": { "@babel/helper-validator-identifier": "^7.22.20", "chalk": "^2.4.2", - "js-tokens": "^4.0.0" + "js-tokens": "^4.0.0", + "picocolors": "^1.0.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/parser": { - "version": "7.23.9", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.23.9.tgz", - "integrity": "sha512-9tcKgqKbs3xGJ+NtKF2ndOBBLVwPjl1SHxPQkd36r3Dlirw3xWUeGaTbqr7uGZcTaxkVNwc+03SVP7aCdWrTlA==", + "version": "7.24.4", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.24.4.tgz", + "integrity": "sha512-zTvEBcghmeBma9QIGunWevvBAp4/Qu9Bdq+2k0Ot4fVMD6v3dsC9WOcRSKk7tRRyBM/53yKMJko9xOatGQAwSg==", "dev": true, "bin": { "parser": "bin/babel-parser.js" @@ -347,33 +348,33 @@ } }, "node_modules/@babel/template": { - "version": "7.23.9", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.23.9.tgz", - "integrity": "sha512-+xrD2BWLpvHKNmX2QbpdpsBaWnRxahMwJjO+KZk2JOElj5nSmKezyS1B4u+QbHMTX69t4ukm6hh9lsYQ7GHCKA==", + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.24.0.tgz", + "integrity": "sha512-Bkf2q8lMB0AFpX0NFEqSbx1OkTHf0f+0j82mkw+ZpzBnkk7e9Ql0891vlfgi+kHwOk8tQjiQHpqh4LaSa0fKEA==", "dev": true, "dependencies": { "@babel/code-frame": "^7.23.5", - "@babel/parser": "^7.23.9", - "@babel/types": "^7.23.9" + "@babel/parser": "^7.24.0", + "@babel/types": "^7.24.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/traverse": { - "version": "7.23.9", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.23.9.tgz", - "integrity": "sha512-I/4UJ9vs90OkBtY6iiiTORVMyIhJ4kAVmsKo9KFc8UOxMeUfi2hvtIBsET5u9GizXE6/GFSuKCTNfgCswuEjRg==", + "version": "7.24.1", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.24.1.tgz", + "integrity": "sha512-xuU6o9m68KeqZbQuDt2TcKSxUw/mrsvavlEqQ1leZ/B+C9tk6E4sRWy97WaXgvq5E+nU3cXMxv3WKOCanVMCmQ==", "dev": true, "dependencies": { - "@babel/code-frame": "^7.23.5", - "@babel/generator": "^7.23.6", + "@babel/code-frame": "^7.24.1", + "@babel/generator": "^7.24.1", "@babel/helper-environment-visitor": "^7.22.20", "@babel/helper-function-name": "^7.23.0", "@babel/helper-hoist-variables": "^7.22.5", "@babel/helper-split-export-declaration": "^7.22.6", - "@babel/parser": "^7.23.9", - "@babel/types": "^7.23.9", + "@babel/parser": "^7.24.1", + "@babel/types": "^7.24.0", "debug": "^4.3.1", "globals": "^11.1.0" }, @@ -382,9 +383,9 @@ } }, "node_modules/@babel/types": { - "version": "7.23.9", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.23.9.tgz", - "integrity": "sha512-dQjSq/7HaSjRM43FFGnv5keM2HsxpmyV1PfaSVm0nzzjwwTmjOe6J4bC8e3+pTEIgHaHj+1ZlLThRJ2auc/w1Q==", + "version": "7.24.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.24.0.tgz", + "integrity": "sha512-+j7a5c253RfKh8iABBhywc8NSfP5LURe7Uh4qpsh6jc+aLJguvmIUBdjSdEMQv2bENrCR5MfRdjGo7vzS/ob7w==", "dev": true, "dependencies": { "@babel/helper-string-parser": "^7.23.4", @@ -488,9 +489,9 @@ } }, "node_modules/@eslint/js": { - "version": "8.56.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.56.0.tgz", - "integrity": "sha512-gMsVel9D7f2HLkBma9VbtzZRehRogVRfbr++f06nL2vnCGCNlzOD+/MUov/F4p8myyAHspEhVobgjpX64q5m6A==", + "version": "8.57.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-8.57.0.tgz", + "integrity": "sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g==", "dev": true, "engines": { "node": "^12.22.0 || ^14.17.0 || >=16.0.0" @@ -524,9 +525,9 @@ } }, "node_modules/@humanwhocodes/object-schema": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.2.tgz", - "integrity": "sha512-6EwiSjwWYP7pTckG6I5eyFANjPhmPjUX9JRLUSfNPC7FX7zK9gyZAfUEaECL6ALTpGX5AjnBq3C9XmVWPitNpw==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz", + "integrity": "sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA==", "dev": true }, "node_modules/@isaacs/cliui": { @@ -696,14 +697,14 @@ } }, "node_modules/@jridgewell/gen-mapping": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.3.tgz", - "integrity": "sha512-HLhSWOLRi875zjjMG/r+Nv0oCW8umGb0BgEhyX3dDX3egwZtB8PqLnjz3yedt8R5StBrzcg4aBpnh8UA9D1BoQ==", + "version": "0.3.5", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz", + "integrity": "sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg==", "dev": true, "dependencies": { - "@jridgewell/set-array": "^1.0.1", + "@jridgewell/set-array": "^1.2.1", "@jridgewell/sourcemap-codec": "^1.4.10", - "@jridgewell/trace-mapping": "^0.3.9" + "@jridgewell/trace-mapping": "^0.3.24" }, "engines": { "node": ">=6.0.0" @@ -719,9 +720,9 @@ } }, "node_modules/@jridgewell/set-array": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.2.tgz", - "integrity": "sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.2.1.tgz", + "integrity": "sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A==", "dev": true, "engines": { "node": ">=6.0.0" @@ -734,9 +735,9 @@ "dev": true }, "node_modules/@jridgewell/trace-mapping": { - "version": "0.3.22", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.22.tgz", - "integrity": "sha512-Wf963MzWtA2sjrNt+g18IAln9lKnlRp+K2eH4jjIoF1wYeq3aMREpG09xhlhdzS0EjwU7qmUJYangWa+151vZw==", + "version": "0.3.25", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", "dev": true, "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", @@ -798,24 +799,24 @@ } }, "node_modules/@npmcli/agent": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/@npmcli/agent/-/agent-2.2.1.tgz", - "integrity": "sha512-H4FrOVtNyWC8MUwL3UfjOsAihHvT1Pe8POj3JvjXhSTJipsZMtgUALCT4mGyYZNxymkUfOw3PUj6dE4QPp6osQ==", + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/@npmcli/agent/-/agent-2.2.2.tgz", + "integrity": "sha512-OrcNPXdpSl9UX7qPVRWbmWMCSXrcDa2M9DvrbOTj7ao1S4PlqVFYv9/yLKMkrJKZ/V5A/kDBC690or307i26Og==", "dependencies": { "agent-base": "^7.1.0", "http-proxy-agent": "^7.0.0", "https-proxy-agent": "^7.0.1", "lru-cache": "^10.0.1", - "socks-proxy-agent": "^8.0.1" + "socks-proxy-agent": "^8.0.3" }, "engines": { "node": "^16.14.0 || >=18.0.0" } }, "node_modules/@npmcli/agent/node_modules/agent-base": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.0.tgz", - "integrity": "sha512-o/zjMZRhJxny7OyEF+Op8X+efiELC7k7yOjMzgfzVqOzXqkBkWI79YoTdOtsuWd5BWhAGAuOY/Xa6xpiaWXiNg==", + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.1.tgz", + "integrity": "sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==", "dependencies": { "debug": "^4.3.4" }, @@ -871,9 +872,9 @@ "dev": true }, "node_modules/@types/node": { - "version": "20.11.19", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.11.19.tgz", - "integrity": "sha512-7xMnVEcZFu0DikYjWOlRq7NTPETrm7teqUT2WkQjrTIkEgUyyGdWsj/Zg8bEJt5TNklzbPD1X3fqfsHw3SpapQ==", + "version": "20.12.7", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.12.7.tgz", + "integrity": "sha512-wq0cICSkRLVaf3UGLMGItu/PtdY7oaXaI/RVU+xliKVOtRna3PRY57ZDfztpDL0n11vfymMUnXv8QwYCO7L1wg==", "dev": true, "dependencies": { "undici-types": "~5.26.4" @@ -1061,15 +1062,16 @@ } }, "node_modules/array-includes": { - "version": "3.1.7", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.7.tgz", - "integrity": "sha512-dlcsNBIiWhPkHdOEEKnehA+RNUWDc4UqFtnIXU4uuYDPtA4LDkr7qip2p0VvFAEXNDr0yWZ9PJyIRiGjRLQzwQ==", + "version": "3.1.8", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.8.tgz", + "integrity": "sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "get-intrinsic": "^1.2.1", + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", "is-string": "^1.0.7" }, "engines": { @@ -1079,17 +1081,18 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/array.prototype.filter": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/array.prototype.filter/-/array.prototype.filter-1.0.3.tgz", - "integrity": "sha512-VizNcj/RGJiUyQBgzwxzE5oHdeuXY5hSbbmKMlphj1cy1Vl7Pn2asCGbSrru6hSQjmCzqTBPVWAF/whmEOVHbw==", + "node_modules/array.prototype.findlast": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz", + "integrity": "sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "es-array-method-boxes-properly": "^1.0.0", - "is-string": "^1.0.7" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "es-shim-unscopables": "^1.0.2" }, "engines": { "node": ">= 0.4" @@ -1099,15 +1102,16 @@ } }, "node_modules/array.prototype.findlastindex": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.4.tgz", - "integrity": "sha512-hzvSHUshSpCflDR1QMUBLHGHP1VIEBegT4pix9H/Z92Xw3ySoy6c2qh7lJWTJnRJ8JCZ9bJNCgTyYaJGcJu6xQ==", + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.5.tgz", + "integrity": "sha512-zfETvRFA8o7EiNn++N5f/kaCw221hrpGsDmcpndVupkPzEc1Wuf3VgC0qby1BbHs7f5DVYjgtEU2LLh5bqeGfQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.5", + "call-bind": "^1.0.7", "define-properties": "^1.2.1", - "es-abstract": "^1.22.3", + "es-abstract": "^1.23.2", "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", "es-shim-unscopables": "^1.0.2" }, "engines": { @@ -1153,6 +1157,18 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/array.prototype.toreversed": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/array.prototype.toreversed/-/array.prototype.toreversed-1.1.2.tgz", + "integrity": "sha512-wwDCoT4Ck4Cz7sLtgUmzR5UV3YF5mFHUlbChCzZBQZ+0m2cl/DH3tKgvphv1nKgFsJ48oCSg6p91q2Vm0I/ZMA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "es-shim-unscopables": "^1.0.0" + } + }, "node_modules/array.prototype.tosorted": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/array.prototype.tosorted/-/array.prototype.tosorted-1.1.3.tgz", @@ -1196,19 +1212,13 @@ "node": "*" } }, - "node_modules/asynciterator.prototype": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/asynciterator.prototype/-/asynciterator.prototype-1.0.0.tgz", - "integrity": "sha512-wwHYEIS0Q80f5mosx3L/dfG5t5rjEa9Ft51GTaNt862EnpyGHpgz2RkZvLPp1oF5TnAiTohkEKVEu8pQPJI7Vg==", - "dev": true, - "dependencies": { - "has-symbols": "^1.0.3" - } - }, "node_modules/available-typed-arrays": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.6.tgz", - "integrity": "sha512-j1QzY8iPNPG4o4xmO3ptzpRxTciqD3MgEHtifP/YnJpIo58Xu+ne4BejlbkuaLfXn/nz6HFiw29bLpj2PNMdGg==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, "engines": { "node": ">= 0.4" }, @@ -1222,12 +1232,15 @@ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" }, "node_modules/binary-extensions": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", - "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", "dev": true, "engines": { "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, "node_modules/bindings": { @@ -1298,9 +1311,9 @@ } }, "node_modules/builtins": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/builtins/-/builtins-5.0.1.tgz", - "integrity": "sha512-qwVpFEHNfhYJIzNRBvd2C1kyo6jz3ZSMPyyuR47OPdiKWlbYnZNyDWuyR175qDnAJLiCo5fBBqPb3RiXgWlkOQ==", + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/builtins/-/builtins-5.1.0.tgz", + "integrity": "sha512-SW9lzGTLvWTP1AY8xeAMZimqDrIaSdLQUcVr9DMef51niJ022Ri87SwRRKYm4A6iHfkPaiVUu/Duw2Wc4J7kKg==", "dev": true, "dependencies": { "semver": "^7.0.0" @@ -1365,15 +1378,15 @@ } }, "node_modules/cacache/node_modules/glob": { - "version": "10.3.10", - "resolved": "https://registry.npmjs.org/glob/-/glob-10.3.10.tgz", - "integrity": "sha512-fa46+tv1Ak0UPK1TOy/pZrIybNNt4HCv7SDzwyfiOZkvZLEbjsZkJBPtDHVshZjbecAoAGSC20MjLDG/qr679g==", + "version": "10.3.12", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.3.12.tgz", + "integrity": "sha512-TCNv8vJ+xz4QiqTpfOJA7HvYv+tNIRHKfUWw/q+v2jdgN4ebz+KY9tGx5J4rHP0o84mNP+ApH66HRX8us3Khqg==", "dependencies": { "foreground-child": "^3.1.0", - "jackspeak": "^2.3.5", + "jackspeak": "^2.3.6", "minimatch": "^9.0.1", - "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0", - "path-scurry": "^1.10.1" + "minipass": "^7.0.4", + "path-scurry": "^1.10.2" }, "bin": { "glob": "dist/esm/bin.mjs" @@ -1391,9 +1404,9 @@ "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==" }, "node_modules/cacache/node_modules/minimatch": { - "version": "9.0.3", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", - "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "version": "9.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.4.tgz", + "integrity": "sha512-KqWh+VchfxcMNRAJjj2tnsSJdNbHsVgnkBhTNrW7AjVo6OvLtxw8zfT9oLw1JSohlFzJ8jCoTgaoXvJ+kHt6fw==", "dependencies": { "brace-expansion": "^2.0.1" }, @@ -1508,9 +1521,9 @@ } }, "node_modules/caniuse-lite": { - "version": "1.0.30001587", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001587.tgz", - "integrity": "sha512-HMFNotUmLXn71BQxg8cijvqxnIAofforZOwGsxyXJ0qugTdspUF4sPSJ2vhgprHCB996tIDzEq1ubumPDV8ULA==", + "version": "1.0.30001611", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001611.tgz", + "integrity": "sha512-19NuN1/3PjA3QI8Eki55N8my4LzfkMCRLgCVfrl/slbSAchQfV0+GwjPrK3rq37As4UCLlM/DHajbKkAqbv92Q==", "dev": true, "funding": [ { @@ -1742,6 +1755,54 @@ "which": "bin/which" } }, + "node_modules/data-view-buffer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-buffer/-/data-view-buffer-1.0.1.tgz", + "integrity": "sha512-0lht7OugA5x3iJLOWFhWK/5ehONdprk0ISXqVFn/NFrDu+cuc8iADFrGQz5BnRK7LLU3JmkbXSxaqX+/mXYtUA==", + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/data-view-byte-length/-/data-view-byte-length-1.0.1.tgz", + "integrity": "sha512-4J7wRJD3ABAzr8wP+OcIcqq2dlUKp4DVflx++hs5h5ZKydWMI6/D/fAot+yh6g2tHh8fLFTvNOaVN357NvSrOQ==", + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/data-view-byte-offset": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/data-view-byte-offset/-/data-view-byte-offset-1.0.0.tgz", + "integrity": "sha512-t/Ygsytq+R995EJ5PZlD4Cu56sWa8InXySaViRzw9apusqsOO2bQP+SbYzAhR0pFKoB+43lYy8rWban9JSuXnA==", + "dependencies": { + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "is-data-view": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/debug": { "version": "4.3.4", "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", @@ -1838,9 +1899,9 @@ "integrity": "sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ==" }, "node_modules/detect-libc": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.2.tgz", - "integrity": "sha512-UX6sGumvvqSaXgdKGUsgZWqcUyIXZ/vZTrlRT/iobiKhGL0zL4d3osHj3uqllWJK+i+sixDS/3COVEOFbupFyw==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", + "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==", "engines": { "node": ">=8" } @@ -1872,9 +1933,9 @@ "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==" }, "node_modules/electron-to-chromium": { - "version": "1.4.672", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.672.tgz", - "integrity": "sha512-YYCy+goe3UqZqa3MOQCI5Mx/6HdBLzXL/mkbGCEWL3sP3Z1BP9zqAzeD3YEmLZlespYGFtyM8tRp5i2vfaUGCA==", + "version": "1.4.744", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.744.tgz", + "integrity": "sha512-nAGcF0yeKKfrP13LMFr5U1eghfFSvFLg302VUFzWlcjPOnUYd52yU5x6PBYrujhNbc4jYmZFrGZFK+xasaEzVA==", "dev": true }, "node_modules/emoji-regex": { @@ -1913,17 +1974,21 @@ } }, "node_modules/es-abstract": { - "version": "1.22.4", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.22.4.tgz", - "integrity": "sha512-vZYJlk2u6qHYxBOTjAeg7qUxHdNfih64Uu2J8QqWgXZ2cri0ZpJAkzDUK/q593+mvKwlxyaxr6F1Q+3LKoQRgg==", + "version": "1.23.3", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.23.3.tgz", + "integrity": "sha512-e+HfNH61Bj1X9/jLc5v1owaLYuHdeHHSQlkhCBiTK8rBvKaULl/beGMxwrMXjpYrv4pz22BlY570vVePA2ho4A==", "dependencies": { "array-buffer-byte-length": "^1.0.1", "arraybuffer.prototype.slice": "^1.0.3", - "available-typed-arrays": "^1.0.6", + "available-typed-arrays": "^1.0.7", "call-bind": "^1.0.7", + "data-view-buffer": "^1.0.1", + "data-view-byte-length": "^1.0.1", + "data-view-byte-offset": "^1.0.0", "es-define-property": "^1.0.0", "es-errors": "^1.3.0", - "es-set-tostringtag": "^2.0.2", + "es-object-atoms": "^1.0.0", + "es-set-tostringtag": "^2.0.3", "es-to-primitive": "^1.2.1", "function.prototype.name": "^1.1.6", "get-intrinsic": "^1.2.4", @@ -1931,15 +1996,16 @@ "globalthis": "^1.0.3", "gopd": "^1.0.1", "has-property-descriptors": "^1.0.2", - "has-proto": "^1.0.1", + "has-proto": "^1.0.3", "has-symbols": "^1.0.3", - "hasown": "^2.0.1", + "hasown": "^2.0.2", "internal-slot": "^1.0.7", "is-array-buffer": "^3.0.4", "is-callable": "^1.2.7", - "is-negative-zero": "^2.0.2", + "is-data-view": "^1.0.1", + "is-negative-zero": "^2.0.3", "is-regex": "^1.1.4", - "is-shared-array-buffer": "^1.0.2", + "is-shared-array-buffer": "^1.0.3", "is-string": "^1.0.7", "is-typed-array": "^1.1.13", "is-weakref": "^1.0.2", @@ -1947,17 +2013,17 @@ "object-keys": "^1.1.1", "object.assign": "^4.1.5", "regexp.prototype.flags": "^1.5.2", - "safe-array-concat": "^1.1.0", + "safe-array-concat": "^1.1.2", "safe-regex-test": "^1.0.3", - "string.prototype.trim": "^1.2.8", - "string.prototype.trimend": "^1.0.7", - "string.prototype.trimstart": "^1.0.7", - "typed-array-buffer": "^1.0.1", - "typed-array-byte-length": "^1.0.0", - "typed-array-byte-offset": "^1.0.0", - "typed-array-length": "^1.0.4", + "string.prototype.trim": "^1.2.9", + "string.prototype.trimend": "^1.0.8", + "string.prototype.trimstart": "^1.0.8", + "typed-array-buffer": "^1.0.2", + "typed-array-byte-length": "^1.0.1", + "typed-array-byte-offset": "^1.0.2", + "typed-array-length": "^1.0.6", "unbox-primitive": "^1.0.2", - "which-typed-array": "^1.1.14" + "which-typed-array": "^1.1.15" }, "engines": { "node": ">= 0.4" @@ -1966,12 +2032,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/es-array-method-boxes-properly": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/es-array-method-boxes-properly/-/es-array-method-boxes-properly-1.0.0.tgz", - "integrity": "sha512-wd6JXUmyHmt8T5a2xreUwKcGPq6f1f+WwIJkijUqiGcJz1qqnZgP6XIK+QyIWU5lT7imeNxUll48bziG+TSYcA==", - "dev": true - }, "node_modules/es-define-property": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", @@ -1992,39 +2052,49 @@ } }, "node_modules/es-iterator-helpers": { - "version": "1.0.17", - "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.17.tgz", - "integrity": "sha512-lh7BsUqelv4KUbR5a/ZTaGGIMLCjPGPqJ6q+Oq24YP0RdyptX1uzm4vvaqzk7Zx3bpl/76YLTTDj9L7uYQ92oQ==", + "version": "1.0.18", + "resolved": "https://registry.npmjs.org/es-iterator-helpers/-/es-iterator-helpers-1.0.18.tgz", + "integrity": "sha512-scxAJaewsahbqTYrGKJihhViaM6DDZDDoucfvzNbK0pOren1g/daDQ3IAhzn+1G14rBG7w+i5N+qul60++zlKA==", "dev": true, "dependencies": { - "asynciterator.prototype": "^1.0.0", "call-bind": "^1.0.7", "define-properties": "^1.2.1", - "es-abstract": "^1.22.4", + "es-abstract": "^1.23.0", "es-errors": "^1.3.0", - "es-set-tostringtag": "^2.0.2", + "es-set-tostringtag": "^2.0.3", "function-bind": "^1.1.2", "get-intrinsic": "^1.2.4", "globalthis": "^1.0.3", "has-property-descriptors": "^1.0.2", - "has-proto": "^1.0.1", + "has-proto": "^1.0.3", "has-symbols": "^1.0.3", "internal-slot": "^1.0.7", "iterator.prototype": "^1.1.2", - "safe-array-concat": "^1.1.0" + "safe-array-concat": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-object-atoms": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.0.0.tgz", + "integrity": "sha512-MZ4iQ6JwHOBQjahnjwaC1ZtIBH+2ohjamzAO3oaHcXYup7qxjF2fixyH+Q71voWHeOkI2q/TnJao/KfXYIZWbw==", + "dependencies": { + "es-errors": "^1.3.0" }, "engines": { "node": ">= 0.4" } }, "node_modules/es-set-tostringtag": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.2.tgz", - "integrity": "sha512-BuDyupZt65P9D2D2vA/zqcI3G5xRsklm5N3xCwuiy+/vKy8i0ifdsQP1sLgO4tZDSCaQUSnmC48khknGMV3D2Q==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.3.tgz", + "integrity": "sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==", "dependencies": { - "get-intrinsic": "^1.2.2", - "has-tostringtag": "^1.0.0", - "hasown": "^2.0.0" + "get-intrinsic": "^1.2.4", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.1" }, "engines": { "node": ">= 0.4" @@ -2079,16 +2149,16 @@ } }, "node_modules/eslint": { - "version": "8.56.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.56.0.tgz", - "integrity": "sha512-Go19xM6T9puCOWntie1/P997aXxFsOi37JIHRWI514Hc6ZnaHGKY9xFhrU65RT6CcBEzZoGG1e6Nq+DT04ZtZQ==", + "version": "8.57.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.57.0.tgz", + "integrity": "sha512-dZ6+mexnaTIbSBZWgou51U6OmzIhYM2VcNdtiTtI7qPNZm35Akpr0f6vtw3w1Kmn5PYo+tZVfh13WrhpS6oLqQ==", "dev": true, "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", "@eslint-community/regexpp": "^4.6.1", "@eslint/eslintrc": "^2.1.4", - "@eslint/js": "8.56.0", - "@humanwhocodes/config-array": "^0.11.13", + "@eslint/js": "8.57.0", + "@humanwhocodes/config-array": "^0.11.14", "@humanwhocodes/module-importer": "^1.0.1", "@nodelib/fs.walk": "^1.2.8", "@ungap/structured-clone": "^1.2.0", @@ -2207,9 +2277,9 @@ } }, "node_modules/eslint-module-utils": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.8.0.tgz", - "integrity": "sha512-aWajIYfsqCKRDgUfjEXNN/JlrzauMuSEy5sbd7WXbtW3EH6A6MpwEh42c7qD+MqQo9QMJ6fWLAeIJynx0g6OAw==", + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.8.1.tgz", + "integrity": "sha512-rXDXR3h7cs7dy9RNpUlQf80nX31XWJEyGq1tRMo+6GsO5VmTe4UTwtmonAD4ZkAsrfMVDA2wlGJ3790Ys+D49Q==", "dev": true, "dependencies": { "debug": "^3.2.7" @@ -2374,27 +2444,29 @@ } }, "node_modules/eslint-plugin-react": { - "version": "7.33.2", - "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.33.2.tgz", - "integrity": "sha512-73QQMKALArI8/7xGLNI/3LylrEYrlKZSb5C9+q3OtOewTnMQi5cT+aE9E41sLCmli3I9PGGmD1yiZydyo4FEPw==", + "version": "7.34.1", + "resolved": "https://registry.npmjs.org/eslint-plugin-react/-/eslint-plugin-react-7.34.1.tgz", + "integrity": "sha512-N97CxlouPT1AHt8Jn0mhhN2RrADlUAsk1/atcT2KyA/l9Q/E6ll7OIGwNumFmWfZ9skV3XXccYS19h80rHtgkw==", "dev": true, "dependencies": { - "array-includes": "^3.1.6", - "array.prototype.flatmap": "^1.3.1", - "array.prototype.tosorted": "^1.1.1", + "array-includes": "^3.1.7", + "array.prototype.findlast": "^1.2.4", + "array.prototype.flatmap": "^1.3.2", + "array.prototype.toreversed": "^1.1.2", + "array.prototype.tosorted": "^1.1.3", "doctrine": "^2.1.0", - "es-iterator-helpers": "^1.0.12", + "es-iterator-helpers": "^1.0.17", "estraverse": "^5.3.0", "jsx-ast-utils": "^2.4.1 || ^3.0.0", "minimatch": "^3.1.2", - "object.entries": "^1.1.6", - "object.fromentries": "^2.0.6", - "object.hasown": "^1.1.2", - "object.values": "^1.1.6", + "object.entries": "^1.1.7", + "object.fromentries": "^2.0.7", + "object.hasown": "^1.1.3", + "object.values": "^1.1.7", "prop-types": "^15.8.1", - "resolve": "^2.0.0-next.4", + "resolve": "^2.0.0-next.5", "semver": "^6.3.1", - "string.prototype.matchall": "^4.0.8" + "string.prototype.matchall": "^4.0.10" }, "engines": { "node": ">=4" @@ -2905,9 +2977,9 @@ } }, "node_modules/flatted": { - "version": "3.2.9", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.9.tgz", - "integrity": "sha512-36yxDn5H7OFZQla0/jFJmbIKTdZAQHngCedGxiMmpNfEZM0sdEeT+WczLQrjK6D7o2aiyLYDnkw0R3JK0Qv1RQ==", + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz", + "integrity": "sha512-X8cqMLLie7KsNUDSdzeN8FYK9rEt4Dt67OsG/DNGnYTSDBG4uFAJFBnUeiV+zCVAvwFy56IjM9sH51jVaEhNxw==", "dev": true }, "node_modules/for-each": { @@ -3305,9 +3377,9 @@ } }, "node_modules/has-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz", - "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", + "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", "engines": { "node": ">= 0.4" }, @@ -3362,9 +3434,9 @@ } }, "node_modules/hasown": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.1.tgz", - "integrity": "sha512-1/th4MHjnwncwXsIW6QMzlvYL9kG5e/CpVvLRZe4XPa8TOUNbCELqmvhDmnkNsAjwaG4+I8gJJL0JBvTTLO9qA==", + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", + "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", "dependencies": { "function-bind": "^1.1.2" }, @@ -3410,9 +3482,9 @@ } }, "node_modules/http-proxy-agent/node_modules/agent-base": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.0.tgz", - "integrity": "sha512-o/zjMZRhJxny7OyEF+Op8X+efiELC7k7yOjMzgfzVqOzXqkBkWI79YoTdOtsuWd5BWhAGAuOY/Xa6xpiaWXiNg==", + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.1.tgz", + "integrity": "sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==", "dependencies": { "debug": "^4.3.4" }, @@ -3657,6 +3729,20 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/is-data-view": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-data-view/-/is-data-view-1.0.1.tgz", + "integrity": "sha512-AHkaJrsUVW6wq6JS8y3JnM/GJF/9cf+k20+iDzlSaJrinEo5+7vRiteOSwBhHRiAyQATN1AmY4hwzxJKPmYf+w==", + "dependencies": { + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/is-date-object": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", @@ -3733,18 +3819,21 @@ "integrity": "sha512-z7CMFGNrENq5iFB9Bqo64Xk6Y9sg+epq1myIcdHaGnbMTYOxvzsEtdYqQUylB7LxfkvgrrjP32T6Ywciio9UIQ==" }, "node_modules/is-map": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.2.tgz", - "integrity": "sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.3.tgz", + "integrity": "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw==", "dev": true, + "engines": { + "node": ">= 0.4" + }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/is-negative-zero": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", - "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", "engines": { "node": ">= 0.4" }, @@ -3809,20 +3898,26 @@ } }, "node_modules/is-set": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.2.tgz", - "integrity": "sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.3.tgz", + "integrity": "sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg==", "dev": true, + "engines": { + "node": ">= 0.4" + }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/is-shared-array-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", - "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.3.tgz", + "integrity": "sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==", "dependencies": { - "call-bind": "^1.0.2" + "call-bind": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -3901,10 +3996,13 @@ } }, "node_modules/is-weakmap": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.1.tgz", - "integrity": "sha512-NSBR4kH5oVj1Uwvv970ruUkCV7O1mzgVFO4/rev2cLRda9Tm9HrL70ZPut4rOHgY0FNrUu9BCbXA2sdQ+x0chA==", + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.2.tgz", + "integrity": "sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w==", "dev": true, + "engines": { + "node": ">= 0.4" + }, "funding": { "url": "https://github.com/sponsors/ljharb" } @@ -3921,13 +4019,16 @@ } }, "node_modules/is-weakset": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.2.tgz", - "integrity": "sha512-t2yVvttHkQktwnNNmBQ98AhENLdPUTDTE21uPqAQ0ARwQfGeQKRVS0NNurH7bTf7RrvcVn1OOge45CnBeHCSmg==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.3.tgz", + "integrity": "sha512-LvIm3/KWzS9oRFHugab7d+M/GcBXuXX5xZkzPmN+NxihdQlZUQ4dWuSV1xR/sq6upL1TJEDrfBgRepHFdBtSNQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -4159,9 +4260,9 @@ } }, "node_modules/istanbul-reports": { - "version": "3.1.6", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.6.tgz", - "integrity": "sha512-TLgnMkKg3iTDsQ9PbPTdpfAK2DzjF9mqUG7RMgcQl8oFjad8ob4laGxv5XV5U9MAfx8D6tSJiUyuAwzLicaxlg==", + "version": "3.1.7", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.7.tgz", + "integrity": "sha512-BewmUXImeuRk2YY0PVbxgKAysvhRPUQE0h5QRM++nVWyubKGV0l8qQ5op8+B2DOmwSe63Jivj0BjkPQVf8fP5g==", "dev": true, "dependencies": { "html-escaper": "^2.0.0", @@ -4680,9 +4781,9 @@ } }, "node_modules/mocha": { - "version": "10.3.0", - "resolved": "https://registry.npmjs.org/mocha/-/mocha-10.3.0.tgz", - "integrity": "sha512-uF2XJs+7xSLsrmIvn37i/wnc91nw7XjOQB8ccyx5aEgdnohr7n+rEiZP23WkCYHjilR6+EboEnbq/ZQDz4LSbg==", + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-10.4.0.tgz", + "integrity": "sha512-eqhGB8JKapEYcC4ytX/xrzKforgEc3j1pGlAXVy3eRwrtAy5/nIfT1SvgGzfN0XZZxeLq0aQWkOUAmqIJiv+bA==", "dev": true, "dependencies": { "ansi-colors": "4.1.1", @@ -4836,9 +4937,9 @@ "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" }, "node_modules/nan": { - "version": "2.18.0", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.18.0.tgz", - "integrity": "sha512-W7tfG7vMOGtD30sHoZSSc/JVYiyDPEyQVso/Zz+/uQd0B0L46gtC+pHha5FFMRpil6fm/AoEcRWyOVi4+E/f8w==" + "version": "2.19.0", + "resolved": "https://registry.npmjs.org/nan/-/nan-2.19.0.tgz", + "integrity": "sha512-nO1xXxfh/RWNxfd/XPfbIfFk5vgLsAxUR9y5O0cHMJu/AW9U95JLXqthYHjEp+8gQ5p96K9jUp8nbVOxCdRbtw==" }, "node_modules/natural-compare": { "version": "1.4.0", @@ -4879,9 +4980,9 @@ } }, "node_modules/node-gyp": { - "version": "10.0.1", - "resolved": "https://registry.npmjs.org/node-gyp/-/node-gyp-10.0.1.tgz", - "integrity": "sha512-gg3/bHehQfZivQVfqIyy8wTdSymF9yTyP4CJifK73imyNMU8AIGQE2pUa7dNWfmMeG9cDVF2eehiRMv0LC1iAg==", + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/node-gyp/-/node-gyp-10.1.0.tgz", + "integrity": "sha512-B4J5M1cABxPc5PwfjhbV5hoy2DP9p8lFXASnEN6hugXOa61416tnTZ29x9sSwAd0o99XNIcpvDDy1swAExsVKA==", "dependencies": { "env-paths": "^2.2.0", "exponential-backoff": "^3.1.1", @@ -4960,15 +5061,15 @@ } }, "node_modules/node-gyp/node_modules/glob": { - "version": "10.3.10", - "resolved": "https://registry.npmjs.org/glob/-/glob-10.3.10.tgz", - "integrity": "sha512-fa46+tv1Ak0UPK1TOy/pZrIybNNt4HCv7SDzwyfiOZkvZLEbjsZkJBPtDHVshZjbecAoAGSC20MjLDG/qr679g==", + "version": "10.3.12", + "resolved": "https://registry.npmjs.org/glob/-/glob-10.3.12.tgz", + "integrity": "sha512-TCNv8vJ+xz4QiqTpfOJA7HvYv+tNIRHKfUWw/q+v2jdgN4ebz+KY9tGx5J4rHP0o84mNP+ApH66HRX8us3Khqg==", "dependencies": { "foreground-child": "^3.1.0", - "jackspeak": "^2.3.5", + "jackspeak": "^2.3.6", "minimatch": "^9.0.1", - "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0", - "path-scurry": "^1.10.1" + "minipass": "^7.0.4", + "path-scurry": "^1.10.2" }, "bin": { "glob": "dist/esm/bin.mjs" @@ -4986,9 +5087,9 @@ "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==" }, "node_modules/node-gyp/node_modules/minimatch": { - "version": "9.0.3", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.3.tgz", - "integrity": "sha512-RHiac9mvaRw0x3AYRgDC1CxAP7HTcNrrECeA8YYJeWnpo+2Q5CegtZjaotWTWxDG3UeGA1coE05iH1mPjT/2mg==", + "version": "9.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.4.tgz", + "integrity": "sha512-KqWh+VchfxcMNRAJjj2tnsSJdNbHsVgnkBhTNrW7AjVo6OvLtxw8zfT9oLw1JSohlFzJ8jCoTgaoXvJ+kHt6fw==", "dependencies": { "brace-expansion": "^2.0.1" }, @@ -5412,28 +5513,29 @@ } }, "node_modules/object.entries": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.7.tgz", - "integrity": "sha512-jCBs/0plmPsOnrKAfFQXRG2NFjlhZgjjcBLSmTnEhU8U6vVTsVe8ANeQJCHTl3gSsI4J+0emOoCgoKlmQPMgmA==", + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/object.entries/-/object.entries-1.1.8.tgz", + "integrity": "sha512-cmopxi8VwRIAw/fkijJohSfpef5PdN0pMQJN6VC/ZKvn0LIknWD8KtgY6KlQdEc4tIjcQ3HxSMmnvtzIscdaYQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" }, "engines": { "node": ">= 0.4" } }, "node_modules/object.fromentries": { - "version": "2.0.7", - "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.7.tgz", - "integrity": "sha512-UPbPHML6sL8PI/mOqPwsH4G6iyXcCGzLin8KvEPenOZN5lpCNBZZQ+V62vdjB1mQHrmqGQt5/OJzemUA+KJmEA==", + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/object.fromentries/-/object.fromentries-2.0.8.tgz", + "integrity": "sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" }, "engines": { "node": ">= 0.4" @@ -5443,40 +5545,45 @@ } }, "node_modules/object.groupby": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.2.tgz", - "integrity": "sha512-bzBq58S+x+uo0VjurFT0UktpKHOZmv4/xePiOA1nbB9pMqpGK7rUPNgf+1YC+7mE+0HzhTMqNUuCqvKhj6FnBw==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/object.groupby/-/object.groupby-1.0.3.tgz", + "integrity": "sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ==", "dev": true, "dependencies": { - "array.prototype.filter": "^1.0.3", - "call-bind": "^1.0.5", + "call-bind": "^1.0.7", "define-properties": "^1.2.1", - "es-abstract": "^1.22.3", - "es-errors": "^1.0.0" + "es-abstract": "^1.23.2" + }, + "engines": { + "node": ">= 0.4" } }, "node_modules/object.hasown": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.3.tgz", - "integrity": "sha512-fFI4VcYpRHvSLXxP7yiZOMAd331cPfd2p7PFDVbgUsYOfCT3tICVqXWngbjr4m49OvsBwUBQ6O2uQoJvy3RexA==", + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/object.hasown/-/object.hasown-1.1.4.tgz", + "integrity": "sha512-FZ9LZt9/RHzGySlBARE3VF+gE26TxR38SdmqOqliuTnl9wrKulaQs+4dee1V+Io8VfxqzAfHu6YuRgUy8OHoTg==", "dev": true, "dependencies": { - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/object.values": { - "version": "1.1.7", - "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.7.tgz", - "integrity": "sha512-aU6xnDFYT3x17e/f0IiiwlGPTy2jzMySGfUB4fq6z7CV8l85CWHDk5ErhyhpfDHhrOMwGFhSQkhMGHaIotA6Ng==", + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.2.0.tgz", + "integrity": "sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" }, "engines": { "node": ">= 0.4" @@ -5642,11 +5749,11 @@ "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==" }, "node_modules/path-scurry": { - "version": "1.10.1", - "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.10.1.tgz", - "integrity": "sha512-MkhCqzzBEpPvxxQ71Md0b1Kk51W01lrYvlMzSUaIzNsODdd7mqhiimSZlr+VegAz5Z6Vzt9Xg2ttE//XBhH3EQ==", + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/path-scurry/-/path-scurry-1.10.2.tgz", + "integrity": "sha512-7xTavNy5RQXnsjANvVvMkEjvloOinkAjv/Z6Ildz9v2RinZ4SBKTWFOVRbaF8p0vpHnyjV/UwNDdKuUv6M5qcA==", "dependencies": { - "lru-cache": "^9.1.1 || ^10.0.0", + "lru-cache": "^10.2.0", "minipass": "^5.0.0 || ^6.0.2 || ^7.0.0" }, "engines": { @@ -5894,6 +6001,14 @@ "node": ">=8" } }, + "node_modules/possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "engines": { + "node": ">= 0.4" + } + }, "node_modules/prelude-ls": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", @@ -6029,16 +6144,16 @@ } }, "node_modules/reflect.getprototypeof": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.5.tgz", - "integrity": "sha512-62wgfC8dJWrmxv44CA36pLDnP6KKl3Vhxb7PL+8+qrrFMMoJij4vgiMP8zV4O8+CBMXY1mHxI5fITGHXFHVmQQ==", + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/reflect.getprototypeof/-/reflect.getprototypeof-1.0.6.tgz", + "integrity": "sha512-fmfw4XgoDke3kdI6h4xcUz1dG8uaiv5q9gcEwLS4Pnth2kxT+GZ7YehS1JTMGBQmtV7Y4GFGbs2re2NqhdozUg==", "dev": true, "dependencies": { - "call-bind": "^1.0.5", + "call-bind": "^1.0.7", "define-properties": "^1.2.1", - "es-abstract": "^1.22.3", - "es-errors": "^1.0.0", - "get-intrinsic": "^1.2.3", + "es-abstract": "^1.23.1", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", "globalthis": "^1.0.3", "which-builtin-type": "^1.1.3" }, @@ -6205,12 +6320,12 @@ } }, "node_modules/safe-array-concat": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.0.tgz", - "integrity": "sha512-ZdQ0Jeb9Ofti4hbt5lX3T2JcAamT9hfzYU1MNB+z/jaEbB6wfFfPIR/zEORmZqobkCCJhSjodobH6WHNmJ97dg==", + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.2.tgz", + "integrity": "sha512-vj6RsCsWBCf19jIeHEfkRMw8DPiBb+DMXklQ/1SGDHOMlHdPUkZXFQ2YdplS23zESTijAcurb1aSgJA3AgMu1Q==", "dependencies": { - "call-bind": "^1.0.5", - "get-intrinsic": "^1.2.2", + "call-bind": "^1.0.7", + "get-intrinsic": "^1.2.4", "has-symbols": "^1.0.3", "isarray": "^2.0.5" }, @@ -6308,29 +6423,30 @@ "integrity": "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw==" }, "node_modules/set-function-length": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.1.tgz", - "integrity": "sha512-j4t6ccc+VsKwYHso+kElc5neZpjtq9EnRICFZtWyBsLojhmeF/ZBd/elqm22WJh/BziDe/SBiOeAt0m2mfLD0g==", + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.2.tgz", + "integrity": "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==", "dependencies": { - "define-data-property": "^1.1.2", + "define-data-property": "^1.1.4", "es-errors": "^1.3.0", "function-bind": "^1.1.2", - "get-intrinsic": "^1.2.3", + "get-intrinsic": "^1.2.4", "gopd": "^1.0.1", - "has-property-descriptors": "^1.0.1" + "has-property-descriptors": "^1.0.2" }, "engines": { "node": ">= 0.4" } }, "node_modules/set-function-name": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.1.tgz", - "integrity": "sha512-tMNCiqYVkXIZgc2Hnoy2IvC/f8ezc5koaRFkCjrpWzGpCd3qbZXPzVy9MAZzK1ch/X0jvSkojys3oqJN0qCmdA==", + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", "dependencies": { - "define-data-property": "^1.0.1", + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", "functions-have-names": "^1.2.3", - "has-property-descriptors": "^1.0.0" + "has-property-descriptors": "^1.0.2" }, "engines": { "node": ">= 0.4" @@ -6364,11 +6480,11 @@ } }, "node_modules/side-channel": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.5.tgz", - "integrity": "sha512-QcgiIWV4WV7qWExbN5llt6frQB/lBven9pqliLXfGPB+K9ZYXxDozp0wLkHS24kWCm+6YXH/f0HhnObZnZOBnQ==", + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.6.tgz", + "integrity": "sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==", "dependencies": { - "call-bind": "^1.0.6", + "call-bind": "^1.0.7", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.4", "object-inspect": "^1.13.1" @@ -6395,9 +6511,9 @@ } }, "node_modules/socks": { - "version": "2.7.3", - "resolved": "https://registry.npmjs.org/socks/-/socks-2.7.3.tgz", - "integrity": "sha512-vfuYK48HXCTFD03G/1/zkIls3Ebr2YNa4qU9gHDZdblHLiqhJrJGkY3+0Nx0JpN9qBhJbVObc1CNciT1bIZJxw==", + "version": "2.8.3", + "resolved": "https://registry.npmjs.org/socks/-/socks-2.8.3.tgz", + "integrity": "sha512-l5x7VUUWbjVFbafGLxPWkYsHIhEvmF85tbIeFZWc8ZPtoMyybuEhL7Jye/ooC4/d48FgOjSJXgsF/AJPYCW8Zw==", "dependencies": { "ip-address": "^9.0.5", "smart-buffer": "^4.2.0" @@ -6408,11 +6524,11 @@ } }, "node_modules/socks-proxy-agent": { - "version": "8.0.2", - "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-8.0.2.tgz", - "integrity": "sha512-8zuqoLv1aP/66PHF5TqwJ7Czm3Yv32urJQHrVyhD7mmA6d61Zv8cIXQYPTWwmg6qlupnPvs/QKDmfa4P/qct2g==", + "version": "8.0.3", + "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-8.0.3.tgz", + "integrity": "sha512-VNegTZKhuGq5vSD6XNKlbqWhyt/40CgoEw8XxD6dhnm8Jq9IEa3nIa4HwnM8XOqU0CdB0BwWVXusqiFXfHB3+A==", "dependencies": { - "agent-base": "^7.0.2", + "agent-base": "^7.1.1", "debug": "^4.3.4", "socks": "^2.7.1" }, @@ -6421,9 +6537,9 @@ } }, "node_modules/socks-proxy-agent/node_modules/agent-base": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.0.tgz", - "integrity": "sha512-o/zjMZRhJxny7OyEF+Op8X+efiELC7k7yOjMzgfzVqOzXqkBkWI79YoTdOtsuWd5BWhAGAuOY/Xa6xpiaWXiNg==", + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-7.1.1.tgz", + "integrity": "sha512-H0TSyFNDMomMNJQBn8wFV5YC/2eJ+VXECwOadZJT554xP6cODZHPX3H9QMQECxvrgiSOP1pHjy1sMWQVYJOUOA==", "dependencies": { "debug": "^4.3.4" }, @@ -6634,33 +6750,40 @@ } }, "node_modules/string.prototype.matchall": { - "version": "4.0.10", - "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.10.tgz", - "integrity": "sha512-rGXbGmOEosIQi6Qva94HUjgPs9vKW+dkG7Y8Q5O2OYkWL6wFaTRZO8zM4mhP94uX55wgyrXzfS2aGtGzUL7EJQ==", + "version": "4.0.11", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.11.tgz", + "integrity": "sha512-NUdh0aDavY2og7IbBPenWqR9exH+E26Sv8e0/eTe1tltDGZL+GtBkDAnnyBtmekfK6/Dq3MkcGtzXFEd1LQrtg==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1", - "get-intrinsic": "^1.2.1", + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-errors": "^1.3.0", + "es-object-atoms": "^1.0.0", + "get-intrinsic": "^1.2.4", + "gopd": "^1.0.1", "has-symbols": "^1.0.3", - "internal-slot": "^1.0.5", - "regexp.prototype.flags": "^1.5.0", - "set-function-name": "^2.0.0", - "side-channel": "^1.0.4" + "internal-slot": "^1.0.7", + "regexp.prototype.flags": "^1.5.2", + "set-function-name": "^2.0.2", + "side-channel": "^1.0.6" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/string.prototype.padend": { - "version": "3.1.5", - "resolved": "https://registry.npmjs.org/string.prototype.padend/-/string.prototype.padend-3.1.5.tgz", - "integrity": "sha512-DOB27b/2UTTD+4myKUFh+/fXWcu/UDyASIXfg+7VzoCNNGOfWvoyU/x5pvVHr++ztyt/oSYI1BcWBBG/hmlNjA==", + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/string.prototype.padend/-/string.prototype.padend-3.1.6.tgz", + "integrity": "sha512-XZpspuSB7vJWhvJc9DLSlrXl1mcA2BdoY5jjnS135ydXqLoqhs96JjDtCkjJEQHvfqZIp9hBuBMgI589peyx9Q==", "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.2", + "es-object-atoms": "^1.0.0" }, "engines": { "node": ">= 0.4" @@ -6670,13 +6793,14 @@ } }, "node_modules/string.prototype.trim": { - "version": "1.2.8", - "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.8.tgz", - "integrity": "sha512-lfjY4HcixfQXOfaqCvcBuOIapyaroTXhbkfJN3gcB1OtyupngWK4sEET9Knd0cXd28kTUqu/kHoV4HKSJdnjiQ==", + "version": "1.2.9", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.9.tgz", + "integrity": "sha512-klHuCNxiMZ8MlsOihJhJEBJAiMVqU3Z2nEXWfWnIqjN0gEFS9J9+IxKozWWtQGcgoa1WUZzLjKPTr4ZHNFTFxw==", "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-abstract": "^1.23.0", + "es-object-atoms": "^1.0.0" }, "engines": { "node": ">= 0.4" @@ -6686,26 +6810,29 @@ } }, "node_modules/string.prototype.trimend": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.7.tgz", - "integrity": "sha512-Ni79DqeB72ZFq1uH/L6zJ+DKZTkOtPIHovb3YZHQViE+HDouuU4mBrLOLDn5Dde3RF8qw5qVETEjhu9locMLvA==", + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.8.tgz", + "integrity": "sha512-p73uL5VCHCO2BZZ6krwwQE3kCzM7NKmis8S//xEC6fQonchbum4eP6kR4DLEjQFO3Wnj3Fuo8NM0kOSjVdHjZQ==", "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/string.prototype.trimstart": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.7.tgz", - "integrity": "sha512-NGhtDFu3jCEm7B4Fy0DpLewdJQOZcQ0rGbwQ/+stjnrp2i+rlKeCvos9hOIeCmqwratM47OBxY7uFZzjxHXmrg==", + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz", + "integrity": "sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg==", "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.2.0", - "es-abstract": "^1.22.1" + "call-bind": "^1.0.7", + "define-properties": "^1.2.1", + "es-object-atoms": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -6784,9 +6911,9 @@ } }, "node_modules/tar": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/tar/-/tar-6.2.0.tgz", - "integrity": "sha512-/Wo7DcT0u5HUV486xg675HtjNd3BXZ6xDbzsCUZPt5iw8bTQ63bP0Raut3mvro9u+CUyq7YQd8Cx55fsZXxqLQ==", + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.2.1.tgz", + "integrity": "sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A==", "dependencies": { "chownr": "^2.0.0", "fs-minipass": "^2.0.0", @@ -6900,15 +7027,12 @@ "dev": true }, "node_modules/tmp": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz", - "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==", + "version": "0.2.3", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.3.tgz", + "integrity": "sha512-nZD7m9iCPC5g0pYmcaxogYKggSfLsdxl8of3Q/oIbqCqLLIO9IAF0GWjX1z9NZRHPiXv8Wex4yDCaZsgEw0Y8w==", "dev": true, - "dependencies": { - "rimraf": "^3.0.0" - }, "engines": { - "node": ">=8.17.0" + "node": ">=14.14" } }, "node_modules/to-fast-properties": { @@ -7001,11 +7125,11 @@ } }, "node_modules/typed-array-buffer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.1.tgz", - "integrity": "sha512-RSqu1UEuSlrBhHTWC8O9FnPjOduNs4M7rJ4pRKoEjtx1zUNOPN2sSXHLDX+Y2WPbHIxbvg4JFo2DNAEfPIKWoQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.2.tgz", + "integrity": "sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==", "dependencies": { - "call-bind": "^1.0.6", + "call-bind": "^1.0.7", "es-errors": "^1.3.0", "is-typed-array": "^1.1.13" }, @@ -7014,14 +7138,15 @@ } }, "node_modules/typed-array-byte-length": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.0.tgz", - "integrity": "sha512-Or/+kvLxNpeQ9DtSydonMxCx+9ZXOswtwJn17SNLvhptaXYDJvkFFP5zbfU/uLmvnBJlI4yrnXRxpdWH/M5tNA==", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.1.tgz", + "integrity": "sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==", "dependencies": { - "call-bind": "^1.0.2", + "call-bind": "^1.0.7", "for-each": "^0.3.3", - "has-proto": "^1.0.1", - "is-typed-array": "^1.1.10" + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" }, "engines": { "node": ">= 0.4" @@ -7031,15 +7156,16 @@ } }, "node_modules/typed-array-byte-offset": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.0.tgz", - "integrity": "sha512-RD97prjEt9EL8YgAgpOkf3O4IF9lhJFr9g0htQkm0rchFp/Vx7LW5Q8fSXXub7BXAODyUQohRMyOc3faCPd0hg==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.2.tgz", + "integrity": "sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==", "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", "for-each": "^0.3.3", - "has-proto": "^1.0.1", - "is-typed-array": "^1.1.10" + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" }, "engines": { "node": ">= 0.4" @@ -7049,13 +7175,19 @@ } }, "node_modules/typed-array-length": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz", - "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==", + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.6.tgz", + "integrity": "sha512-/OxDN6OtAk5KBpGb28T+HZc2M+ADtvRxXrKKbUwtsLgdoxgX13hyy7ek6bFRl5+aBs2yZzB0c4CnQfAtVypW/g==", "dependencies": { - "call-bind": "^1.0.2", + "call-bind": "^1.0.7", "for-each": "^0.3.3", - "is-typed-array": "^1.1.9" + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -7184,9 +7316,9 @@ } }, "node_modules/version-guard": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/version-guard/-/version-guard-1.1.1.tgz", - "integrity": "sha512-MGQLX89UxmYHgDvcXyjBI0cbmoW+t/dANDppNPrno64rYr8nH4SHSuElQuSYdXGEs0mUzdQe1BY+FhVPNsAmJQ==", + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/version-guard/-/version-guard-1.1.2.tgz", + "integrity": "sha512-D8d+YxCUpoqtCnQzDxm6SF7DLU3gr2535T4khAtMq4osBahsQnmSxuwXFdrbAdDGG8Uokzfis/jvyeFPdmlc7w==", "dev": true, "engines": { "node": ">=0.10.48" @@ -7262,15 +7394,18 @@ } }, "node_modules/which-collection": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.1.tgz", - "integrity": "sha512-W8xeTUwaln8i3K/cY1nGXzdnVZlidBcagyNFtBdD5kxnb4TvGKR7FfSIS3mYpwWS1QUCutfKz8IY8RjftB0+1A==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.2.tgz", + "integrity": "sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw==", "dev": true, "dependencies": { - "is-map": "^2.0.1", - "is-set": "^2.0.1", - "is-weakmap": "^2.0.1", - "is-weakset": "^2.0.1" + "is-map": "^2.0.3", + "is-set": "^2.0.3", + "is-weakmap": "^2.0.2", + "is-weakset": "^2.0.3" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -7283,15 +7418,15 @@ "dev": true }, "node_modules/which-typed-array": { - "version": "1.1.14", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.14.tgz", - "integrity": "sha512-VnXFiIW8yNn9kIHN88xvZ4yOWchftKDsRJ8fEPacX/wl1lOvBrhsJ/OeJCXq7B0AaijRuqgzSKalJoPk+D8MPg==", + "version": "1.1.15", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.15.tgz", + "integrity": "sha512-oV0jmFtUky6CXfkqehVvBP/LSWJ2sy4vWMioiENyJLePrBO/yKyV9OyJySfAKosh+RYkIl5zJCNZ8/4JncrpdA==", "dependencies": { - "available-typed-arrays": "^1.0.6", - "call-bind": "^1.0.5", + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", "for-each": "^0.3.3", "gopd": "^1.0.1", - "has-tostringtag": "^1.0.1" + "has-tostringtag": "^1.0.2" }, "engines": { "node": ">= 0.4" diff --git a/package.json b/package.json index 7ec7667bf..b5dd3d546 100644 --- a/package.json +++ b/package.json @@ -59,24 +59,24 @@ "@mapbox/node-pre-gyp": "^1.0.11", "bindings": "^1.5.0", "minimatch": "^3.1.2", - "nan": "^2.18.0", - "node-gyp": "^10.0.1", + "nan": "^2.19.0", + "node-gyp": "^10.1.0", "npm-run-all": "^4.1.5" }, "devDependencies": { - "@types/node": "^20.11.18", - "chai": "^4.3.7", + "@types/node": "^20.12.7", + "chai": "^4.4.1", "choma": "^1.2.1", "codecov": "^3.8.3", "deep-eql": "^4.1.3", "husky": "^9.0.11", - "mocha": "^10.3.0", + "mocha": "^10.4.0", "mocha-clean": "^1.0.0", "nyc": "^15.1.0", "p-throttle": "^3.1.0", "semver": "^7.6.0", "standard": "^17.1.0", - "tmp": "^0.2.1", + "tmp": "^0.2.3", "yargs": "^17.7.2" }, "standard": { diff --git a/src/main/bit_operations.cc b/src/main/bit_operations.cc index 283fc4c62..5338b3a7a 100644 --- a/src/main/bit_operations.cc +++ b/src/main/bit_operations.cc @@ -254,7 +254,7 @@ bool add_bit_rshift_op(as_operations *ops, char *bin, as_bit_policy *policy, typedef bool (*AsBitMath)(as_operations *ops, const char *bin, as_cdt_ctx *ctx, as_bit_policy *policy, int bit_offset, - uint32_t bit_size, int64_t value, bool sign, + uint32_t bit_size, uint64_t value, bool sign, as_bit_overflow_action action); bool add_bit_math_op(as_operations *ops, AsBitMath math_op, char *bin, diff --git a/typings/index.d.ts b/typings/index.d.ts index aa3b3e301..dd3cb105e 100644 --- a/typings/index.d.ts +++ b/typings/index.d.ts @@ -1504,6 +1504,7 @@ declare module 'policies/query_policy' { deserialize: boolean; failOnClusterChange: boolean; infoTimeout: number; + expectedDuration: number; } import BasePolicy = require("policies/base_policy"); From 3ba6b77dba94cd7588760b9aba7cdf863b44cec0 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 12:21:19 -0600 Subject: [PATCH 04/17] Modifying unit test to be more latency consistent --- test/batch_read.js | 46 +++++++++++++++++++++++----------------------- test/get.js | 24 ++++++++++++------------ test/operate.js | 24 ++++++++++++------------ 3 files changed, 47 insertions(+), 47 deletions(-) diff --git a/test/batch_read.js b/test/batch_read.js index bf61c9c77..1dd3c968c 100644 --- a/test/batch_read.js +++ b/test/batch_read.js @@ -194,15 +194,15 @@ describe('client.batchRead()', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.equal(8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -213,15 +213,15 @@ describe('client.batchRead()', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.equal(8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -232,15 +232,15 @@ describe('client.batchRead()', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -251,7 +251,7 @@ describe('client.batchRead()', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) @@ -259,7 +259,7 @@ describe('client.batchRead()', function () { const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -275,15 +275,15 @@ describe('client.batchRead()', function () { }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.equal(8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -297,15 +297,15 @@ describe('client.batchRead()', function () { }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.equal(8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -319,15 +319,15 @@ describe('client.batchRead()', function () { }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.equal(8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -341,15 +341,15 @@ describe('client.batchRead()', function () { }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.equal(8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) diff --git a/test/get.js b/test/get.js index a85c4ca31..5a47c67f7 100644 --- a/test/get.js +++ b/test/get.js @@ -91,16 +91,16 @@ describe('client.get()', function () { }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(key) }) @@ -112,16 +112,16 @@ describe('client.get()', function () { }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(key) }) @@ -132,17 +132,17 @@ describe('client.get()', function () { readTouchTtlPercent: 70 }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(key) }) @@ -152,17 +152,17 @@ describe('client.get()', function () { readTouchTtlPercent: 0 }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(key) }) }) diff --git a/test/operate.js b/test/operate.js index 607f46fd7..760973ff9 100644 --- a/test/operate.js +++ b/test/operate.js @@ -289,15 +289,15 @@ context('Operations', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) @@ -309,15 +309,15 @@ context('Operations', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(10) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) @@ -329,15 +329,15 @@ context('Operations', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) @@ -349,15 +349,15 @@ context('Operations', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 1000)) + await new Promise(resolve => setTimeout(resolve, 2000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.equal(8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(9) + expect(record.ttl).to.be.within(9, 10); await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) From 055413c4c29ca3ea3acda568b7c9ff78318d0cff Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 13:09:32 -0600 Subject: [PATCH 05/17] Made timing more consistent for ttl unit tests --- src/main/policy.cc | 2 -- test/batch_read.js | 66 ++++++++++++++++++++++++---------------------- test/get.js | 33 ++++++++++++----------- test/operate.js | 33 ++++++++++++----------- 4 files changed, 68 insertions(+), 66 deletions(-) diff --git a/src/main/policy.cc b/src/main/policy.cc index 177c3649d..88ac4d92c 100644 --- a/src/main/policy.cc +++ b/src/main/policy.cc @@ -352,13 +352,11 @@ int batchpolicy_from_jsobject(as_policy_batch *policy, Local obj, AS_NODE_PARAM_OK) { return rc; } - printf("read_touch_ttl_percent: %d\n\n", policy->read_touch_ttl_percent); if ((rc = get_optional_int_property((int *)&policy->read_touch_ttl_percent, NULL, obj, "readTouchTtlPercent", log)) != AS_NODE_PARAM_OK) { return rc; } - printf("read_touch_ttl_percent: %d\n\n", policy->read_touch_ttl_percent); if ((rc = get_optional_bool_property(&policy->concurrent, NULL, obj, "concurrent", log)) != AS_NODE_PARAM_OK) { diff --git a/test/batch_read.js b/test/batch_read.js index 1dd3c968c..47607043b 100644 --- a/test/batch_read.js +++ b/test/batch_read.js @@ -187,6 +187,8 @@ describe('client.batchRead()', function () { key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), readAllBins: true }] + this.timeout(4000) + context('BatchReadPolicy policy', function () { it('100% touches record', async function () { const policy = new Aerospike.BatchReadPolicy({ @@ -194,53 +196,53 @@ describe('client.batchRead()', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(8) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) - it('91% touches record', async function () { + it('71% touches record', async function () { const policy = new Aerospike.BatchReadPolicy({ - readTouchTtlPercent: 91 + readTouchTtlPercent: 71 }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(8) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) - it('70% doesnt touch record', async function () { + it('60% doesnt touch record', async function () { const policy = new Aerospike.BatchReadPolicy({ - readTouchTtlPercent: 70 + readTouchTtlPercent: 60 }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -251,15 +253,15 @@ describe('client.batchRead()', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch, policy) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(9) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -275,59 +277,59 @@ describe('client.batchRead()', function () { }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(8) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) - it('91% touches record', async function () { + it('71% touches record', async function () { batch = [{ key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), readAllBins: true, policy: new Aerospike.BatchPolicy({ - readTouchTtlPercent: 91 + readTouchTtlPercent: 71 }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(8) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) - it('70% doesnt touch record', async function () { + it('60% doesnt touch record', async function () { batch = [{ key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), readAllBins: true, policy: new Aerospike.BatchPolicy({ - readTouchTtlPercent: 70 + readTouchTtlPercent: 60 }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(8) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) @@ -341,15 +343,15 @@ describe('client.batchRead()', function () { }) }] await client.put(new Aerospike.Key('test', 'demo', 'batchReadTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) const batchResult = await client.batchRead(batch) expect(batchResult[0].record.bins).to.eql({ i: 2 }) - expect(batchResult[0].record.ttl).to.equal(8) + expect(batchResult[0].record.ttl).to.be.within(7, 8) const record = await client.get(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(new Aerospike.Key('test', 'demo', 'batchReadTtl1')) }) diff --git a/test/get.js b/test/get.js index 5a47c67f7..1b04cb2ab 100644 --- a/test/get.js +++ b/test/get.js @@ -84,6 +84,7 @@ describe('client.get()', function () { }) context('readTouchTtlPercent policy', function () { + this.timeout(4000) it('100% touches record', async function () { const key = keygen.integer(helper.namespace, helper.set)() const policy = new Aerospike.ReadPolicy({ @@ -91,58 +92,58 @@ describe('client.get()', function () { }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(key) }) - it('91% touches record', async function () { + it('71% touches record', async function () { const key = keygen.integer(helper.namespace, helper.set)() const policy = new Aerospike.ReadPolicy({ - readTouchTtlPercent: 91 + readTouchTtlPercent: 71 }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(key) }) - it('70% never touches record', async function () { + it('60% never touches record', async function () { const key = keygen.integer(helper.namespace, helper.set)() const policy = new Aerospike.ReadPolicy({ - readTouchTtlPercent: 70 + readTouchTtlPercent: 60 }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(key) }) @@ -152,17 +153,17 @@ describe('client.get()', function () { readTouchTtlPercent: 0 }) await client.put(key, { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(key, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(key) }) }) diff --git a/test/operate.js b/test/operate.js index 760973ff9..0abef399c 100644 --- a/test/operate.js +++ b/test/operate.js @@ -282,6 +282,7 @@ context('Operations', function () { }) context('readTouchTtlPercent policy', function () { + this.timeout(4000) it('100% touches record', async function () { const ops = [op.read('i')] const policy = new Aerospike.OperatePolicy({ @@ -289,55 +290,55 @@ context('Operations', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) - it('91% touches record', async function () { + it('71% touches record', async function () { const ops = [op.read('i')] const policy = new Aerospike.OperatePolicy({ - readTouchTtlPercent: 91 + readTouchTtlPercent: 71 }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(9, 10) await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) - it('70% does not touch record', async function () { + it('60% does not touch record', async function () { const ops = [op.read('i')] const policy = new Aerospike.OperatePolicy({ - readTouchTtlPercent: 70 + readTouchTtlPercent: 60 }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) @@ -349,15 +350,15 @@ context('Operations', function () { }) await client.put(new Aerospike.Key('test', 'demo', 'operateTtl1'), { i: 2 }, { ttl: 10 }) - await new Promise(resolve => setTimeout(resolve, 2000)) + await new Promise(resolve => setTimeout(resolve, 3000)) let record = await client.operate(new Aerospike.Key('test', 'demo', 'operateTtl1'), ops, null, policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.equal(8) + expect(record.ttl).to.be.within(7, 8) record = await client.get(new Aerospike.Key('test', 'demo', 'operateTtl1'), policy) expect(record.bins).to.eql({ i: 2 }) - expect(record.ttl).to.be.within(9, 10); + expect(record.ttl).to.be.within(7, 8) await client.remove(new Aerospike.Key('test', 'demo', 'operateTtl1')) }) From ff1badc29ef56eda698a42e34fbf7f57c95d778d Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 13:24:11 -0600 Subject: [PATCH 06/17] Added logic to skip unless version is 7.1 or above --- test/batch_read.js | 4 +++- test/get.js | 2 ++ test/operate.js | 2 ++ test/query.js | 3 +++ 4 files changed, 10 insertions(+), 1 deletion(-) diff --git a/test/batch_read.js b/test/batch_read.js index 47607043b..d7eb75051 100644 --- a/test/batch_read.js +++ b/test/batch_read.js @@ -189,7 +189,9 @@ describe('client.batchRead()', function () { }] this.timeout(4000) - context('BatchReadPolicy policy', function () { + context('BatchPolicy policy', function () { + helper.skipUnlessVersion('>= 7.1.0', this) + it('100% touches record', async function () { const policy = new Aerospike.BatchReadPolicy({ readTouchTtlPercent: 100 diff --git a/test/get.js b/test/get.js index 1b04cb2ab..437d9c9be 100644 --- a/test/get.js +++ b/test/get.js @@ -84,6 +84,8 @@ describe('client.get()', function () { }) context('readTouchTtlPercent policy', function () { + helper.skipUnlessVersion('>= 7.1.0', this) + this.timeout(4000) it('100% touches record', async function () { const key = keygen.integer(helper.namespace, helper.set)() diff --git a/test/operate.js b/test/operate.js index 0abef399c..d6ff44f8b 100644 --- a/test/operate.js +++ b/test/operate.js @@ -282,6 +282,8 @@ context('Operations', function () { }) context('readTouchTtlPercent policy', function () { + helper.skipUnlessVersion('>= 7.1.0', this) + this.timeout(4000) it('100% touches record', async function () { const ops = [op.read('i')] diff --git a/test/query.js b/test/query.js index 2c2eee88e..76ac15448 100644 --- a/test/query.js +++ b/test/query.js @@ -275,6 +275,7 @@ describe('Queries', function () { }) it('Should run a regular primary index query with expectedDuration=LONG', function (done) { + helper.skipUnlessVersion('>= 7.1.0', this) const query = client.query(helper.namespace, testSet) const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG }) @@ -288,6 +289,7 @@ describe('Queries', function () { }) it('Should run a regular primary index query with expectedDuration=SHORT', function (done) { + helper.skipUnlessVersion('>= 7.1.0', this) const query = client.query(helper.namespace, testSet) const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.SHORT }) const results = [] @@ -300,6 +302,7 @@ describe('Queries', function () { }) it('Should run a regular primary index query with expectedDuration=LONG_RELAX_AP', function (done) { + helper.skipUnlessVersion('>= 7.1.0', this) const query = client.query(helper.namespace, testSet) const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG_RELAX_AP }) const results = [] From aa042a1343935a1e9086b972e36c99c446f52d87 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 13:33:09 -0600 Subject: [PATCH 07/17] Added test to context function so it can be properly skipped Added test to context function so it can be properly skipped Added additional skip for version 7.1 --- test/batch_read.js | 1 + test/query.js | 63 +++++++++++++++++++++++----------------------- 2 files changed, 32 insertions(+), 32 deletions(-) diff --git a/test/batch_read.js b/test/batch_read.js index d7eb75051..5382b4f14 100644 --- a/test/batch_read.js +++ b/test/batch_read.js @@ -270,6 +270,7 @@ describe('client.batchRead()', function () { }) context('BatchReadPolicy policy', function () { + helper.skipUnlessVersion('>= 7.1.0', this) it('100% touches record', async function () { batch = [{ key: new Aerospike.Key('test', 'demo', 'batchReadTtl1'), diff --git a/test/query.js b/test/query.js index 76ac15448..f202c5ca5 100644 --- a/test/query.js +++ b/test/query.js @@ -273,44 +273,43 @@ describe('Queries', function () { done() }) }) - - it('Should run a regular primary index query with expectedDuration=LONG', function (done) { + context('expectedDuration', function () { helper.skipUnlessVersion('>= 7.1.0', this) - const query = client.query(helper.namespace, testSet) - const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG }) - const results = [] - stream.on('error', error => { throw error }) - stream.on('data', record => results.push(record.bins)) - stream.on('end', () => { - expect(results.length).to.be.above(samples.length) - done() + it('Should run a regular primary index query with expectedDuration=LONG', function (done) { + const query = client.query(helper.namespace, testSet) + const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG }) + const results = [] + stream.on('error', error => { throw error }) + stream.on('data', record => results.push(record.bins)) + stream.on('end', () => { + expect(results.length).to.be.above(samples.length) + done() + }) }) - }) - it('Should run a regular primary index query with expectedDuration=SHORT', function (done) { - helper.skipUnlessVersion('>= 7.1.0', this) - const query = client.query(helper.namespace, testSet) - const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.SHORT }) - const results = [] - stream.on('error', error => { throw error }) - stream.on('data', record => results.push(record.bins)) - stream.on('end', () => { - expect(results.length).to.be.above(samples.length) - done() + it('Should run a regular primary index query with expectedDuration=SHORT', function (done) { + const query = client.query(helper.namespace, testSet) + const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.SHORT }) + const results = [] + stream.on('error', error => { throw error }) + stream.on('data', record => results.push(record.bins)) + stream.on('end', () => { + expect(results.length).to.be.above(samples.length) + done() + }) }) - }) - it('Should run a regular primary index query with expectedDuration=LONG_RELAX_AP', function (done) { - helper.skipUnlessVersion('>= 7.1.0', this) - const query = client.query(helper.namespace, testSet) - const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG_RELAX_AP }) - const results = [] - stream.on('error', error => { throw error }) - stream.on('data', record => results.push(record.bins)) - stream.on('end', () => { - expect(results.length).to.be.above(samples.length) - done() + it('Should run a regular primary index query with expectedDuration=LONG_RELAX_AP', function (done) { + const query = client.query(helper.namespace, testSet) + const stream = query.foreach({ expectedDuration: Aerospike.policy.queryDuration.LONG_RELAX_AP }) + const results = [] + stream.on('error', error => { throw error }) + stream.on('data', record => results.push(record.bins)) + stream.on('end', () => { + expect(results.length).to.be.above(samples.length) + done() + }) }) }) From cf8c78a106120babe1cf0ba16ce9cddeb73747f6 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 13:43:51 -0600 Subject: [PATCH 08/17] Debugging pipeline Debugging pipeline --- .github/workflows/tests.yml | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 2b2ab7ef7..495f483b1 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -257,7 +257,7 @@ jobs: - name: Run lowest supported server run: | - SERVER_VERSION=$(curl -s "https://registry.hub.docker.com/v2/repositories/aerospike/aerospike-server/tags?page_size=100" | jq '.results[] | select(.name | startswith("6.0")).name' -r | head -n 1) + SERVER_VERSION=$(curl -s "https://registry.hub.docker.com/v2/repositories/aerospike/aerospike-server/tags?page_size=100" | jq '.results[] | select(.name | startswith("6.1")).name' -r | head -n 1) docker run -d --name aerospike -p 3000-3002:3000-3002 aerospike/aerospike-server:$SERVER_VERSION - name: Wait for database to be ready @@ -305,7 +305,8 @@ jobs: use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} docker-hub-password: ${{ secrets.DOCKER_HUB_BOT_PW }} - + working-directory: aerospike-client-python + - name: Wait for server to start run: sleep 5 @@ -344,7 +345,10 @@ jobs: run: npm install mocha - name: install valgrind - run: sudo apt install valgrind + run: + sudo apt-get update + sudo apt update + sudo apt install valgrind - name: Install client #fix the convention here From 542d48ac6b24ac77ea18f87768494b3d460319b7 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 13:50:22 -0600 Subject: [PATCH 09/17] Update tests.yml --- .github/workflows/tests.yml | 1 - 1 file changed, 1 deletion(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 495f483b1..debd75da9 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -305,7 +305,6 @@ jobs: use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} docker-hub-password: ${{ secrets.DOCKER_HUB_BOT_PW }} - working-directory: aerospike-client-python - name: Wait for server to start From 2610220a9a8c9c32595f99422b41754a1378a2ff Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 13:57:58 -0600 Subject: [PATCH 10/17] Debugging pipeline --- .github/workflows/requirements.txt | 3 +++ .github/workflows/tests.yml | 8 ++++---- 2 files changed, 7 insertions(+), 4 deletions(-) create mode 100644 .github/workflows/requirements.txt diff --git a/.github/workflows/requirements.txt b/.github/workflows/requirements.txt new file mode 100644 index 000000000..836636e9c --- /dev/null +++ b/.github/workflows/requirements.txt @@ -0,0 +1,3 @@ +parver==0.5 +crudini==0.9.4 +delocate==0.10.4 \ No newline at end of file diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index debd75da9..d61cd516f 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -344,10 +344,10 @@ jobs: run: npm install mocha - name: install valgrind - run: - sudo apt-get update - sudo apt update - sudo apt install valgrind + run: | + sudo apt-get update; + sudo apt update; + sudo apt install valgrind; - name: Install client #fix the convention here From ff7d5546232838442217cfb63e67bfbeb53a0070 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:11:35 -0600 Subject: [PATCH 11/17] Modified implementation of ee-server action --- .github/actions/run-ee-server/action.yml | 62 ++++++++++++++++++++++++ .github/workflows/requirements.txt | 3 -- .github/workflows/tests.yml | 2 +- 3 files changed, 63 insertions(+), 4 deletions(-) create mode 100644 .github/actions/run-ee-server/action.yml delete mode 100644 .github/workflows/requirements.txt diff --git a/.github/actions/run-ee-server/action.yml b/.github/actions/run-ee-server/action.yml new file mode 100644 index 000000000..f7d4379fb --- /dev/null +++ b/.github/actions/run-ee-server/action.yml @@ -0,0 +1,62 @@ +name: 'Run EE Server' +description: 'Run EE server' +inputs: + # All inputs in composite actions are strings + use-server-rc: + required: true + default: false + server-tag: + required: true + default: 'latest' + # Github Composite Actions can't access secrets + # so we need to pass them in as inputs + docker-hub-username: + required: false + docker-hub-password: + required: false + +runs: + using: "composite" + steps: + - name: Create config folder to store configs in + run: mkdir configs + shell: bash + + - name: Use release server + if: ${{ inputs.use-server-rc == 'false' }} + run: echo "SERVER_IMAGE=aerospike/aerospike-server-enterprise" >> $GITHUB_ENV + shell: bash + + - name: Use release candidate server + if: ${{ inputs.use-server-rc == 'true' }} + run: echo "SERVER_IMAGE=aerospike/aerospike-server-enterprise-rc" >> $GITHUB_ENV + shell: bash + + - name: Log into Docker Hub to get server RC + if: ${{ inputs.use-server-rc == 'true' }} + run: docker login --username ${{ inputs.docker-hub-username }} --password ${{ inputs.docker-hub-password }} + shell: bash + + - name: Get default aerospike.conf from Docker server EE container + run: | + docker run -d --name aerospike -p 3000-3002:3000-3002 $SERVER_IMAGE:${{ inputs.server-tag }} + sleep 5 + docker cp aerospike:/etc/aerospike/aerospike.conf ./configs/aerospike.conf + docker container stop aerospike + docker container rm aerospike + shell: bash + + - name: Enable security features using aerospike.conf + # Security stanza + run: echo -e "security {\n\tenable-quotas true\n}\n" >> ./aerospike.conf + working-directory: ./configs + shell: bash + + - name: Run enterprise edition server + run: docker run -tid -v $(pwd)/configs:/opt/aerospike/etc -p 3000:3000 --name aerospike $SERVER_IMAGE:${{ inputs.server-tag }} asd --config-file /opt/aerospike/etc/aerospike.conf + shell: bash + + - name: Create user in database for tests + # Use default admin user to create another user for testing + run: docker exec aerospike asadm --user admin --password admin --enable -e "manage acl create user superuser password superuser roles read-write-udf sys-admin user-admin data-admin" + shell: bash \ No newline at end of file diff --git a/.github/workflows/requirements.txt b/.github/workflows/requirements.txt deleted file mode 100644 index 836636e9c..000000000 --- a/.github/workflows/requirements.txt +++ /dev/null @@ -1,3 +0,0 @@ -parver==0.5 -crudini==0.9.4 -delocate==0.10.4 \ No newline at end of file diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index d61cd516f..326b5d300 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -300,7 +300,7 @@ jobs: run: git clone https://github.com/aerospike/aerospike-client-python.git - name: Run ee server - uses: ./aerospike-client-python/.github/actions/run-ee-server/ + uses: .github/actions/run-ee-server/ with: use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} From d97239b8085b131f608a68de346bbead9e3eafff Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:22:55 -0600 Subject: [PATCH 12/17] debugging pipeline --- .github/workflows/build-bindings.yml | 4 ++-- .github/workflows/tests.yml | 5 +++++ 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/.github/workflows/build-bindings.yml b/.github/workflows/build-bindings.yml index 7b846a3f0..56fba8dd2 100644 --- a/.github/workflows/build-bindings.yml +++ b/.github/workflows/build-bindings.yml @@ -429,14 +429,14 @@ jobs: - name: Modify the package.json run: | - npm install -g json + sudo npm install -g json json -I -f package.json -e "this.scripts.install=\"npm-run-all removeExtraBinaries build\"" - name: Run tests run: | mkdir -p testDir cd testDir - npm install .. + sudo npm install .. test-yarn-install: runs-on: ubuntu-latest diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 326b5d300..cab7dfa9b 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -299,6 +299,11 @@ jobs: - name: Download aerospike-client-python repository run: git clone https://github.com/aerospike/aerospike-client-python.git + - name: debugging + run: | + ls; + ls .github; + - name: Run ee server uses: .github/actions/run-ee-server/ with: From 79d2b2b5ffa88ae559e1008f1a696d76a48bfe56 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:25:20 -0600 Subject: [PATCH 13/17] Update tests.yml --- .github/workflows/tests.yml | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index cab7dfa9b..95055b5f8 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -304,12 +304,7 @@ jobs: ls; ls .github; - - name: Run ee server - uses: .github/actions/run-ee-server/ - with: - use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} - docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} - docker-hub-password: ${{ secrets.DOCKER_HUB_BOT_PW }} + - name: Wait for server to start From 1b3279f62ea2539dcb4a7f64cee68410dea51278 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:33:34 -0600 Subject: [PATCH 14/17] Debugging pipeline --- .github/workflows/build-bindings.yml | 8 ++++---- .github/workflows/tests.yml | 7 ++++++- 2 files changed, 10 insertions(+), 5 deletions(-) diff --git a/.github/workflows/build-bindings.yml b/.github/workflows/build-bindings.yml index 56fba8dd2..1666c76eb 100644 --- a/.github/workflows/build-bindings.yml +++ b/.github/workflows/build-bindings.yml @@ -455,7 +455,7 @@ jobs: - name: Modify the package.json run: | - npm install -g json + sudo npm install -g json json -I -f package.json -e "this.scripts.install=\"npm-run-all removeExtraBinaries build\"" - name: Run tests @@ -483,7 +483,7 @@ jobs: - name: Modify the package.json run: | - npm install -g json + sudo npm install -g json json -I -f package.json -e "this.scripts.install=\"npm-run-all removeExtraBinaries build\"" - name: Run tests @@ -512,7 +512,7 @@ jobs: - name: Modify the package.json run: | - npm install -g json + sudo npm install -g json json -I -f package.json -e "this.scripts.install=\"npm-run-all removeExtraBinaries build\"" - name: Run tests @@ -555,7 +555,7 @@ jobs: - name: Modify the package.json run: | - npm install -g json + sudo npm install -g json json -I -f package.json -e "this.scripts.install=\"npm-run-all removeExtraBinaries build\"" - name: Run tests diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 95055b5f8..f3c47c1e3 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -303,8 +303,13 @@ jobs: run: | ls; ls .github; - + - name: Run ee server + uses: .github/actions/run-ee-server + with: + use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} + docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} + docker-hub-password: ${{ secrets.DOCKER_HUB_BOT_PW }} - name: Wait for server to start From 617610114c188b7f32055f2962099ef39a996de4 Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:37:30 -0600 Subject: [PATCH 15/17] Update tests.yml --- .github/workflows/tests.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index f3c47c1e3..0d6b79eef 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -303,6 +303,7 @@ jobs: run: | ls; ls .github; + docker ps; - name: Run ee server uses: .github/actions/run-ee-server From 7f1416e18f93393f392310a5195345ef65e0814e Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:43:40 -0600 Subject: [PATCH 16/17] Update tests.yml --- .github/workflows/tests.yml | 6 ------ 1 file changed, 6 deletions(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 0d6b79eef..2d228d89d 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -305,12 +305,6 @@ jobs: ls .github; docker ps; - - name: Run ee server - uses: .github/actions/run-ee-server - with: - use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} - docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} - docker-hub-password: ${{ secrets.DOCKER_HUB_BOT_PW }} - name: Wait for server to start From d6255ae90274f4df7a749f4bc58ce7d6aab4bc1f Mon Sep 17 00:00:00 2001 From: Dominic Pelini <111786059+DomPeliniAerospike@users.noreply.github.com> Date: Fri, 19 Apr 2024 14:58:10 -0600 Subject: [PATCH 17/17] Update tests.yml --- .github/workflows/tests.yml | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 2d228d89d..584319390 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -305,6 +305,12 @@ jobs: ls .github; docker ps; + - name: Run ee server + uses: ./.github/actions/run-ee-server + with: + use-server-rc: ${{ contains(github.event.pull_request.labels.*.name, 'new-server-features') }} + docker-hub-username: ${{ secrets.DOCKER_HUB_BOT_USERNAME }} + docker-hub-password: ${{ secrets.DOCKER_HUB_BOT_PW }} - name: Wait for server to start