Skip to content

Commit

Permalink
valkey - fixing items by moving back to useRedisSets (#1205)
Browse files Browse the repository at this point in the history
* valkey - fixing items by moving back to useRedisSets

* misspelling

* fixing test
  • Loading branch information
jaredwray authored Nov 13, 2024
1 parent c067783 commit bb7bf89
Show file tree
Hide file tree
Showing 4 changed files with 26 additions and 26 deletions.
24 changes: 12 additions & 12 deletions packages/valkey/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@

[Valkey](https://valkey.io) storage adapter for [Keyv](https://github.com/jaredwray/keyv).

Valkey is the open source replacement to Redis which decided do a [dual license](https://redis.com/blog/redis-adopts-dual-source-available-licensing/) approach moving forward. Valkey is a drop-in replacement for Redis and is fully compatible with the Redis protocol.
Valkey is the open source replacement to Redis which decided to do a [dual license](https://redis.com/blog/redis-adopts-dual-source-available-licensing/) approach moving forward. Valkey is a drop-in replacement for Redis and is fully compatible with the Redis protocol.

We are using the [iovalkey](https://www.npmjs.com/package/iovalkey) which is a Node.js client for Valkey based on the `ioredis` client.

Expand Down Expand Up @@ -45,7 +45,7 @@ Or you can manually create a storage adapter instance and pass it to Keyv:

```js
import Keyv from 'keyv';
import KeyvValkey from '@keyv/redis';
import KeyvValkey from '@keyv/valkey';

const KeyvValkey = new KeyvValkey('redis://user:pass@localhost:6379');
const keyv = new Keyv({ store: KeyvValkey });
Expand All @@ -56,7 +56,7 @@ Or reuse a previous Redis instance:
```js
import Keyv from 'keyv';
import Redis from 'ioredis';
import KeyvValkey from '@keyv/redis';
import KeyvValkey from '@keyv/valkey';

const redis = new Redis('redis://user:pass@localhost:6379');
const KeyvValkey = new KeyvValkey(redis);
Expand All @@ -68,38 +68,38 @@ Or reuse a previous Redis cluster:
```js
import Keyv from 'keyv';
import Redis from 'ioredis';
import KeyvValkey from '@keyv/redis';
import KeyvValkey from '@keyv/valkey';

const redis = new Redis.Cluster('redis://user:pass@localhost:6379');
const KeyvValkey = new KeyvValkey(redis);
const keyv = new Keyv({ store: KeyvValkey });
```
# Options

## useSets
## useRedisSets

The `useSets` option lets you decide whether to use Redis sets for key management. By default, this option is set to `true`.
The `useRedisSets` option lets you decide whether to use Redis sets for key management. By default, this option is set to `true`.

When `useSets` is enabled (`true`):
When `useRedisSets` is enabled (`true`):

- A namespace for the Redis sets is created, and all created keys are added to this. This allows for group management of keys.
- When a key is deleted, it's removed not only from the main storage but also from the Redis set.
- When clearing all keys (using the `clear` function), all keys in the Redis set are looked up for deletion. The set itself is also deleted.

**Note**: In high-performance scenarios, enabling `useSets` might lead to memory leaks. If you're running a high-performance application or service, it is recommended to set `useSets` to `false`.
**Note**: In high-performance scenarios, enabling `useRedisSets` might lead to memory leaks. If you're running a high-performance application or service, it is recommended to set `useRedisSets` to `false`.

If you decide to set `useSets` as `false`, keys will be handled individually and Redis sets won't be utilized.
If you decide to set `useRedisSets` as `false`, keys will be handled individually and Redis sets won't be utilized.

However, please note that setting `useSets` to `false` could lead to performance issues in production when using the `clear` function, as it will need to iterate over all keys to delete them.
However, please note that setting `useRedisSets` to `false` could lead to performance issues in production when using the `clear` function, as it will need to iterate over all keys to delete them.

## Example

Here's how you can use the `useSets` option:
Here's how you can use the `useRedisSets` option:

```js
import Keyv from 'keyv';

const keyv = new Keyv(new KeyvValkey('redis://user:pass@localhost:6379', { useSets: false }));
const keyv = new Keyv(new KeyvValkey('redis://user:pass@localhost:6379', { useRedisSets: false }));
```

## License
Expand Down
14 changes: 7 additions & 7 deletions packages/valkey/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ class KeyvValkey extends EventEmitter implements KeyvStoreAdapter {
constructor(uri: KeyvValkeyOptions | KeyvUriOptions, options?: KeyvValkeyOptions) {
super();
this.opts = {};
this.opts.useSets = true;
this.opts.useRedisSets = true;
this.opts.dialect = 'redis';

if (typeof uri !== 'string' && uri.options && ('family' in uri.options || uri.isCluster)) {
Expand All @@ -27,8 +27,8 @@ class KeyvValkey extends EventEmitter implements KeyvStoreAdapter {
this.redis = new Redis(options.uri!, options);
}

if (options !== undefined && options.useSets === false) {
this.opts.useSets = false;
if (options !== undefined && options.useRedisSets === false) {
this.opts.useRedisSets = false;
}

this.redis.on('error', (error: Error) => this.emit('error', error));
Expand All @@ -39,7 +39,7 @@ class KeyvValkey extends EventEmitter implements KeyvStoreAdapter {
}

_getKeyName = (key: string): string => {
if (!this.opts.useSets) {
if (!this.opts.useRedisSets) {
return `sets:${this._getNamespace()}:${key}`;
}

Expand Down Expand Up @@ -77,7 +77,7 @@ class KeyvValkey extends EventEmitter implements KeyvStoreAdapter {
}
};

if (this.opts.useSets) {
if (this.opts.useRedisSets) {
const trx = await this.redis.multi();
await set(trx);
await trx.sadd(this._getNamespace(), key);
Expand All @@ -92,7 +92,7 @@ class KeyvValkey extends EventEmitter implements KeyvStoreAdapter {
let items = 0;
const unlink = async (redis: any) => redis.unlink(key);

if (this.opts.useSets) {
if (this.opts.useRedisSets) {
const trx = this.redis.multi();
await unlink(trx);
await trx.srem(this._getNamespace(), key);
Expand All @@ -113,7 +113,7 @@ class KeyvValkey extends EventEmitter implements KeyvStoreAdapter {
}

async clear() {
if (this.opts.useSets) {
if (this.opts.useRedisSets) {
const keys: string[] = await this.redis.smembers(this._getNamespace());
if (keys.length > 0) {
await Promise.all([
Expand Down
2 changes: 1 addition & 1 deletion packages/valkey/src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ export type KeyvValkeyOptions = {
} & {
uri?: string;
dialect?: string;
useSets?: boolean;
useRedisSets?: boolean;
};

export type KeyvUriOptions = string | KeyvValkeyOptions | Redis | Cluster;
12 changes: 6 additions & 6 deletions packages/valkey/test/test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -124,7 +124,7 @@ test.it('should handle KeyvOptions with family option', t => {
});

test.it('set method should use Redis sets when useSets is false', async t => {
const options = {useSets: false};
const options = {useRedisSets: false};
const keyv = new KeyvValkey(options);

await keyv.set('foo', 'bar');
Expand All @@ -134,7 +134,7 @@ test.it('set method should use Redis sets when useSets is false', async t => {
});

test.it('clear method when useSets is false', async t => {
const options = {useSets: false};
const options = {useRedisSets: false};
const keyv = new KeyvValkey(options);

await keyv.set('foo', 'bar');
Expand All @@ -149,21 +149,21 @@ test.it('clear method when useSets is false', async t => {
});

test.it('clear method when useSets is false and empty keys should not error', async t => {
const options = {useSets: false};
const options = {useRedisSets: false};
const keyv = new KeyvValkey(options);
t.expect(await keyv.clear()).toBeUndefined();
});

test.it('when passing in ioredis set the options.useSets', t => {
const options = {useSets: false};
const options = {useRedisSets: false};
const redis = new Redis(redisURI);
const keyv = new KeyvValkey(redis, options);

t.expect(keyv.opts.useSets).toBe(false);
t.expect(keyv.opts.useRedisSets).toBe(false);
});

test.it('del should work when not using useSets', async t => {
const options = {useSets: false};
const options = {useRedisSets: false};
const redis = new Redis(redisURI);
const keyv = new KeyvValkey(redis, options);

Expand Down

0 comments on commit bb7bf89

Please sign in to comment.