1
0
mirror of https://github.com/redis/node-redis.git synced 2025-08-07 13:22:56 +03:00
This commit is contained in:
Leibale
2023-06-22 19:41:35 -04:00
parent 6059b1edd8
commit e95634b375
6 changed files with 281 additions and 202 deletions

View File

@@ -1,19 +1,16 @@
# Leibale # Client
- Does `close`/`destory` actually close the connection from the Redis POV? Works with OSS, but what about Redis Enterprie? - Does `close`/`destory` actually close the connection from the Redis POV? Works with OSS, but what about Redis Enterprie?
# Docs Docs:
- [v4 to v5](./v4-to-v5.md) - Legacy mode - [v4 to v5](./v4-to-v5.md) - Legacy mode
- [Command Options](./command-options.md) - [Command Options](./command-options.md)
- [RESP](./RESP.md) - [RESP](./RESP.md)
# Missing functionality # Server
- `HEXISTS`: accepts one field only, should be the same as `EXISTS` - `HEXISTS`: accepts one field only, should be the same as `EXISTS`
# Replies
`String` -> `Double`: `String` -> `Double`:
- `INCRBYFLOAT` - `INCRBYFLOAT`
- `HINCRBYFLOAT` - `HINCRBYFLOAT`

View File

@@ -1,195 +1,226 @@
// import { strict as assert } from 'assert'; // import { strict as assert } from 'assert';
// import { SinonSpy, spy } from 'sinon'; // import { SinonSpy, spy } from 'sinon';
// import RESP2Decoder from './decoder'; // import { Decoder, RESP_TYPES } from './decoder';
// import { ErrorReply } from '../../errors'; // import { ErrorReply } from '../errors';
// import { TypeMapping } from './types';
// interface DecoderAndSpies { // function createDecoderAndSpies(typeMapping: TypeMapping = {}) {
// decoder: RESP2Decoder; // const typeMappingSpy = spy(() => typeMapping),
// returnStringsAsBuffersSpy: SinonSpy; // onReplySpy = spy(),
// onReplySpy: SinonSpy; // onErrorReplySpy = spy(),
// onPushSpy = spy();
// return {
// decoder: new Decoder({
// getTypeMapping: typeMappingSpy,
// onReply: onReplySpy,
// onErrorReply: onErrorReplySpy,
// onPush: onPushSpy
// }),
// typeMappingSpy,
// onReplySpy,
// onErrorReplySpy,
// onPushSpy
// };
// } // }
// function createDecoderAndSpies(returnStringsAsBuffers: boolean): DecoderAndSpies { // function writeChunks(stream: Decoder, buffer: Buffer) {
// const returnStringsAsBuffersSpy = spy(() => returnStringsAsBuffers), // let i = 0;
// onReplySpy = spy(); // while (i < buffer.length) {
// stream.write(buffer.subarray(i, ++i));
// return { // }
// decoder: new RESP2Decoder({
// returnStringsAsBuffers: returnStringsAsBuffersSpy,
// onReply: onReplySpy
// }),
// returnStringsAsBuffersSpy,
// onReplySpy
// };
// }
// function writeChunks(stream: RESP2Decoder, buffer: Buffer) {
// let i = 0;
// while (i < buffer.length) {
// stream.write(buffer.slice(i, ++i));
// }
// } // }
// type Replies = Array<Array<unknown>>; // type Replies = Array<Array<unknown>>;
// interface TestsOptions { // function generateTests(toWrite: Buffer, tests: Array<Test & { name: string }>): void {
// toWrite: Buffer; // for (const test of tests) {
// returnStringsAsBuffers: boolean; // describe(test.name, () => {
// replies: Replies; // generateTests(toWrite, test);
// });
// }
// } // }
// function generateTests({ // interface Test {
// toWrite, // typeMapping?: TypeMapping;
// returnStringsAsBuffers, // replies?: Replies;
// replies // errorReplies?: Replies;
// }: TestsOptions): void { // pushReplies?: Replies;
// it('single chunk', () => {
// const { decoder, returnStringsAsBuffersSpy, onReplySpy } =
// createDecoderAndSpies(returnStringsAsBuffers);
// decoder.write(toWrite);
// assert.equal(returnStringsAsBuffersSpy.callCount, replies.length);
// testReplies(onReplySpy, replies);
// });
// it('multiple chunks', () => {
// const { decoder, returnStringsAsBuffersSpy, onReplySpy } =
// createDecoderAndSpies(returnStringsAsBuffers);
// writeChunks(decoder, toWrite);
// assert.equal(returnStringsAsBuffersSpy.callCount, replies.length);
// testReplies(onReplySpy, replies);
// });
// } // }
// function testReplies(spy: SinonSpy, replies: Replies): void { // function genetareTypeTests(toWrite: Buffer, { typeMapping, replies, errorReplies, pushReplies }: Test) {
// if (!replies) { // const total = (replies?.length ?? 0) + (errorReplies?.length ?? 0) + (pushReplies?.length ?? 0);
// assert.equal(spy.callCount, 0); // it('single chunk', () => {
// return; // const { decoder, typeMappingSpy, onReplySpy, onErrorReplySpy, onPushSpy } = createDecoderAndSpies(typeMapping);
// } // decoder.write(toWrite);
// assert.equal(typeMappingSpy.callCount, total);
// testReplies(onReplySpy, replies);
// testReplies(onErrorReplySpy, errorReplies);
// testReplies(onPushSpy, pushReplies);
// });
// assert.equal(spy.callCount, replies.length); // it('multiple chunks', () => {
// for (const [i, reply] of replies.entries()) { // const { decoder, typeMappingSpy, onReplySpy, onErrorReplySpy, onPushSpy } = createDecoderAndSpies(typeMapping);
// assert.deepEqual( // writeChunks(decoder, toWrite);
// spy.getCall(i).args, // assert.equal(typeMappingSpy.callCount, total);
// reply // testReplies(onReplySpy, replies);
// ); // testReplies(onErrorReplySpy, errorReplies);
// } // testReplies(onPushSpy, pushReplies);
// });
// }
// function testReplies(spy: SinonSpy, replies?: Replies): void {
// if (!replies) {
// assert.equal(spy.callCount, 0);
// return;
// }
// assert.equal(spy.callCount, replies.length);
// for (const [i, reply] of replies.entries()) {
// assert.deepEqual(
// spy.getCall(i).args,
// reply
// );
// }
// } // }
// describe('RESP2Parser', () => { // describe('RESP2Parser', () => {
// describe('Simple String', () => { // describe('Null', () => {
// describe('as strings', () => { // genetareTypeTests(Buffer.from('_\r\n'), {
// generateTests({ // replies: [[null]]
// toWrite: Buffer.from('+OK\r\n'),
// returnStringsAsBuffers: false,
// replies: [['OK']]
// });
// });
// describe('as buffers', () => {
// generateTests({
// toWrite: Buffer.from('+OK\r\n'),
// returnStringsAsBuffers: true,
// replies: [[Buffer.from('OK')]]
// });
// });
// }); // });
// });
// describe('Error', () => { // describe('Boolean', () => {
// genetareTypeTests(Buffer.from('#t\r\n'), {
// replies: [[null]]
// });
// });
// describe('Number', () => {
// generateTests(Buffer.from(':-1\r\n'))
// describe('as number', () => {
// describe('-1', () => {
// generateTests({ // generateTests({
// toWrite: Buffer.from('-ERR\r\n'), // toWrite: ,
// returnStringsAsBuffers: false, // replies: [[-1]]
// replies: [[new ErrorReply('ERR')]]
// }); // });
// });
// describe('0', () => {
// generateTests({
// toWrite: Buffer.from(':0\r\n'),
// replies: [[0]]
// });
// });
// describe('+1', () => {
// generateTests({
// toWrite: Buffer.from(':+1\r\n'),
// replies: [[1]]
// });
// });
// });
// });
// describe('Simple String', () => {
// describe('as strings', () => {
// generateTests({
// toWrite: Buffer.from('+OK\r\n'),
// replies: [['OK']]
// });
// }); // });
// describe('Integer', () => { // describe('as buffers', () => {
// describe('-1', () => { // generateTests({
// generateTests({ // toWrite: Buffer.from('+OK\r\n'),
// toWrite: Buffer.from(':-1\r\n'), // typeMapping: {
// returnStringsAsBuffers: false, // [RESP_TYPES.SIMPLE_STRING]: Buffer
// replies: [[-1]] // },
// }); // replies: [[Buffer.from('OK')]]
// }); // });
// });
// });
// describe('0', () => { // describe('Error', () => {
// generateTests({ // generateTests({
// toWrite: Buffer.from(':0\r\n'), // toWrite: Buffer.from('-ERR\r\n'),
// returnStringsAsBuffers: false, // errorReplies: [[new ErrorReply('ERR')]]
// replies: [[0]] // });
// }); // });
// });
// describe('Bulk String', () => {
// describe('null', () => {
// generateTests({
// toWrite: Buffer.from('$-1\r\n'),
// returnStringsAsBuffers: false,
// replies: [[null]]
// });
// }); // });
// describe('Bulk String', () => { // describe('as strings', () => {
// describe('null', () => { // generateTests({
// generateTests({ // toWrite: Buffer.from('$2\r\naa\r\n'),
// toWrite: Buffer.from('$-1\r\n'), // returnStringsAsBuffers: false,
// returnStringsAsBuffers: false, // replies: [['aa']]
// replies: [[null]] // });
// });
// });
// describe('as strings', () => {
// generateTests({
// toWrite: Buffer.from('$2\r\naa\r\n'),
// returnStringsAsBuffers: false,
// replies: [['aa']]
// });
// });
// describe('as buffers', () => {
// generateTests({
// toWrite: Buffer.from('$2\r\naa\r\n'),
// returnStringsAsBuffers: true,
// replies: [[Buffer.from('aa')]]
// });
// });
// }); // });
// describe('Array', () => { // describe('as buffers', () => {
// describe('null', () => { // generateTests({
// generateTests({ // toWrite: Buffer.from('$2\r\naa\r\n'),
// toWrite: Buffer.from('*-1\r\n'), // returnStringsAsBuffers: true,
// returnStringsAsBuffers: false, // replies: [[Buffer.from('aa')]]
// replies: [[null]] // });
// });
// });
// const arrayBuffer = Buffer.from(
// '*5\r\n' +
// '+OK\r\n' +
// '-ERR\r\n' +
// ':0\r\n' +
// '$1\r\na\r\n' +
// '*0\r\n'
// );
// describe('as strings', () => {
// generateTests({
// toWrite: arrayBuffer,
// returnStringsAsBuffers: false,
// replies: [[[
// 'OK',
// new ErrorReply('ERR'),
// 0,
// 'a',
// []
// ]]]
// });
// });
// describe('as buffers', () => {
// generateTests({
// toWrite: arrayBuffer,
// returnStringsAsBuffers: true,
// replies: [[[
// Buffer.from('OK'),
// new ErrorReply('ERR'),
// 0,
// Buffer.from('a'),
// []
// ]]]
// });
// });
// }); // });
// });
// describe('Array', () => {
// describe('null', () => {
// generateTests({
// toWrite: Buffer.from('*-1\r\n'),
// returnStringsAsBuffers: false,
// replies: [[null]]
// });
// });
// const arrayBuffer = Buffer.from(
// '*5\r\n' +
// '+OK\r\n' +
// '-ERR\r\n' +
// ':0\r\n' +
// '$1\r\na\r\n' +
// '*0\r\n'
// );
// describe('as strings', () => {
// generateTests({
// toWrite: arrayBuffer,
// returnStringsAsBuffers: false,
// replies: [[[
// 'OK',
// new ErrorReply('ERR'),
// 0,
// 'a',
// []
// ]]]
// });
// });
// describe('as buffers', () => {
// generateTests({
// toWrite: arrayBuffer,
// returnStringsAsBuffers: true,
// replies: [[[
// Buffer.from('OK'),
// new ErrorReply('ERR'),
// 0,
// Buffer.from('a'),
// []
// ]]]
// });
// });
// });
// }); // });

View File

@@ -0,0 +1,46 @@
import { strict as assert } from 'assert';
import testUtils, { GLOBAL } from '../test-utils';
import ZRANK_WITHSCORE from './ZRANK_WITHSCORE';
describe('ZRANK WITHSCORE', () => {
testUtils.isVersionGreaterThanHook([7, 2]);
it('transformArguments', () => {
assert.deepEqual(
ZRANK_WITHSCORE.transformArguments('key', 'member'),
['ZRANK', 'key', 'member', 'WITHSCORE']
);
});
testUtils.testAll('zRankWithScore - null', async client => {
assert.equal(
await client.zRankWithScore('key', 'member'),
null
);
}, {
client: GLOBAL.SERVERS.OPEN,
cluster: GLOBAL.CLUSTERS.OPEN
});
testUtils.testAll('zRankWithScore - with member', async client => {
const member = {
value: '1',
score: 1
}
const [, reply] = await Promise.all([
client.zAdd('key', member),
client.zRankWithScore('key', member.value)
])
assert.deepEqual(
reply,
{
rank: 0,
score: 1
}
);
}, {
client: GLOBAL.SERVERS.OPEN,
cluster: GLOBAL.CLUSTERS.OPEN
});
});

View File

@@ -0,0 +1,30 @@
import { NullReply, TuplesReply, NumberReply, BlobStringReply, DoubleReply, Command } from '../RESP/types';
import ZRANK from './ZRANK';
export default {
FIRST_KEY_INDEX: ZRANK.FIRST_KEY_INDEX,
IS_READ_ONLY: ZRANK.IS_READ_ONLY,
transformArguments(...args: Parameters<typeof ZRANK.transformArguments>) {
const redisArgs = ZRANK.transformArguments(...args);
redisArgs.push('WITHSCORE');
return redisArgs;
},
transformReply: {
2: (reply: NullReply | TuplesReply<[NumberReply, BlobStringReply]>) => {
if (reply === null) return null;
return {
rank: reply[0],
score: Number(reply[1])
};
},
3: (reply: NullReply | TuplesReply<[BlobStringReply, DoubleReply]>) => {
if (reply === null) return null;
return {
rank: reply[0],
score: reply[1]
};
}
}
} as const satisfies Command;

View File

@@ -286,6 +286,7 @@ import ZRANGEBYSCORE_WITHSCORES from './ZRANGEBYSCORE_WITHSCORES';
import ZRANGEBYSCORE from './ZRANGEBYSCORE'; import ZRANGEBYSCORE from './ZRANGEBYSCORE';
import ZRANGESTORE from './ZRANGESTORE'; import ZRANGESTORE from './ZRANGESTORE';
import ZREMRANGEBYSCORE from './ZREMRANGEBYSCORE'; import ZREMRANGEBYSCORE from './ZREMRANGEBYSCORE';
import ZRANK_WITHSCORE from './ZRANK_WITHSCORE';
import ZRANK from './ZRANK'; import ZRANK from './ZRANK';
import ZREM from './ZREM'; import ZREM from './ZREM';
import ZREMRANGEBYLEX from './ZREMRANGEBYLEX'; import ZREMRANGEBYLEX from './ZREMRANGEBYLEX';
@@ -586,6 +587,7 @@ type ZRANGEBYSCORE_WITHSCORES = typeof import('./ZRANGEBYSCORE_WITHSCORES').defa
type ZRANGEBYSCORE = typeof import('./ZRANGEBYSCORE').default; type ZRANGEBYSCORE = typeof import('./ZRANGEBYSCORE').default;
type ZRANGESTORE = typeof import('./ZRANGESTORE').default; type ZRANGESTORE = typeof import('./ZRANGESTORE').default;
type ZREMRANGEBYSCORE = typeof import('./ZREMRANGEBYSCORE').default; type ZREMRANGEBYSCORE = typeof import('./ZREMRANGEBYSCORE').default;
type ZRANK_WITHSCORE = typeof import('./ZRANK_WITHSCORE').default;
type ZRANK = typeof import('./ZRANK').default; type ZRANK = typeof import('./ZRANK').default;
type ZREM = typeof import('./ZREM').default; type ZREM = typeof import('./ZREM').default;
type ZREMRANGEBYLEX = typeof import('./ZREMRANGEBYLEX').default; type ZREMRANGEBYLEX = typeof import('./ZREMRANGEBYLEX').default;
@@ -1174,6 +1176,8 @@ type Commands = {
zRangeByScore: ZRANGEBYSCORE; zRangeByScore: ZRANGEBYSCORE;
ZRANGESTORE: ZRANGESTORE; ZRANGESTORE: ZRANGESTORE;
zRangeStore: ZRANGESTORE; zRangeStore: ZRANGESTORE;
ZRANK_WITHSCORE: ZRANK_WITHSCORE;
zRankWithScore: ZRANK_WITHSCORE;
ZRANK: ZRANK; ZRANK: ZRANK;
zRank: ZRANK; zRank: ZRANK;
ZREM: ZREM; ZREM: ZREM;
@@ -1775,6 +1779,8 @@ export default {
zRangeByScore: ZRANGEBYSCORE, zRangeByScore: ZRANGEBYSCORE,
ZRANGESTORE, ZRANGESTORE,
zRangeStore: ZRANGESTORE, zRangeStore: ZRANGESTORE,
ZRANK_WITHSCORE,
zRankWithScore: ZRANK_WITHSCORE,
ZRANK, ZRANK,
zRank: ZRANK, zRank: ZRANK,
ZREM, ZREM,

31
todo.md
View File

@@ -1,31 +0,0 @@
# return type \ missing documentation
- `XAUTOCLAIM`
- `XSETID`
- `ZUNION`
# create commands
- `ZREVRANGE`
- `ZREVRANGEBYLEX`
- `ZREVRANGEBYSCORE`
# waiting List categoreis
- Set
- Bitmap
# fot leiba
- `BZMPOP.ts`
- `BZPOPMAX.ts`
- `BZPOPMIN.ts`
- `ZMPOP.ts`
- `ZPOPMAX.ts`
- `ZPOPMIN.ts`
- `ZREVRANGE WITHSCORE`
- `ZRANK WITHSCORE`
- `MIGRATE`
- `RESTORE`
- `WAITAOF`
# other