You've already forked node-redis
mirror of
https://github.com/redis/node-redis.git
synced 2025-08-03 04:01:40 +03:00
* (docs) bloom: add jsdocs for all commands * (docs) json: add jsdocs * (docs) search: add jsdocs for all commands * (docs) jsdocs for std commands * (docs) jsdoc comments to time series commands
115 lines
3.9 KiB
TypeScript
115 lines
3.9 KiB
TypeScript
import { CommandParser } from '../client/parser';
|
|
import { RedisArgument, TuplesToMapReply, BlobStringReply, NumberReply, NullReply, TuplesReply, ArrayReply, UnwrapReply, Command } from '../RESP/types';
|
|
import { isNullReply, transformTuplesReply } from './generic-transformers';
|
|
|
|
/**
|
|
* Reply structure for XINFO STREAM command containing detailed information about a stream
|
|
*
|
|
* @property length - Number of entries in the stream
|
|
* @property radix-tree-keys - Number of radix tree keys
|
|
* @property radix-tree-nodes - Number of radix tree nodes
|
|
* @property last-generated-id - Last generated message ID
|
|
* @property max-deleted-entry-id - Highest message ID deleted (Redis 7.2+)
|
|
* @property entries-added - Total number of entries added (Redis 7.2+)
|
|
* @property recorded-first-entry-id - ID of the first recorded entry (Redis 7.2+)
|
|
* @property groups - Number of consumer groups
|
|
* @property first-entry - First entry in the stream
|
|
* @property last-entry - Last entry in the stream
|
|
*/
|
|
export type XInfoStreamReply = TuplesToMapReply<[
|
|
[BlobStringReply<'length'>, NumberReply],
|
|
[BlobStringReply<'radix-tree-keys'>, NumberReply],
|
|
[BlobStringReply<'radix-tree-nodes'>, NumberReply],
|
|
[BlobStringReply<'last-generated-id'>, BlobStringReply],
|
|
/** added in 7.2 */
|
|
[BlobStringReply<'max-deleted-entry-id'>, BlobStringReply],
|
|
/** added in 7.2 */
|
|
[BlobStringReply<'entries-added'>, NumberReply],
|
|
/** added in 7.2 */
|
|
[BlobStringReply<'recorded-first-entry-id'>, BlobStringReply],
|
|
[BlobStringReply<'groups'>, NumberReply],
|
|
[BlobStringReply<'first-entry'>, ReturnType<typeof transformEntry>],
|
|
[BlobStringReply<'last-entry'>, ReturnType<typeof transformEntry>]
|
|
]>;
|
|
|
|
export default {
|
|
IS_READ_ONLY: true,
|
|
/**
|
|
* Constructs the XINFO STREAM command to get detailed information about a stream
|
|
*
|
|
* @param parser - The command parser
|
|
* @param key - The stream key
|
|
* @returns Detailed information about the stream including its length, structure, and entries
|
|
* @see https://redis.io/commands/xinfo-stream/
|
|
*/
|
|
parseCommand(parser: CommandParser, key: RedisArgument) {
|
|
parser.push('XINFO', 'STREAM');
|
|
parser.pushKey(key);
|
|
},
|
|
transformReply: {
|
|
// TODO: is there a "type safe" way to do it?
|
|
2(reply: any) {
|
|
const parsedReply: Partial<XInfoStreamReply['DEFAULT']> = {};
|
|
|
|
for (let i = 0; i < reply.length; i += 2) {
|
|
switch (reply[i]) {
|
|
case 'first-entry':
|
|
case 'last-entry':
|
|
parsedReply[reply[i] as ('first-entry' | 'last-entry')] = transformEntry(reply[i + 1]) as any;
|
|
break;
|
|
|
|
default:
|
|
parsedReply[reply[i] as keyof typeof parsedReply] = reply[i + 1];
|
|
break;
|
|
}
|
|
}
|
|
|
|
return parsedReply as XInfoStreamReply['DEFAULT'];
|
|
},
|
|
3(reply: any) {
|
|
if (reply instanceof Map) {
|
|
reply.set(
|
|
'first-entry',
|
|
transformEntry(reply.get('first-entry'))
|
|
);
|
|
reply.set(
|
|
'last-entry',
|
|
transformEntry(reply.get('last-entry'))
|
|
);
|
|
} else if (reply instanceof Array) {
|
|
reply[17] = transformEntry(reply[17]);
|
|
reply[19] = transformEntry(reply[19]);
|
|
} else {
|
|
reply['first-entry'] = transformEntry(reply['first-entry']);
|
|
reply['last-entry'] = transformEntry(reply['last-entry']);
|
|
}
|
|
|
|
return reply as XInfoStreamReply;
|
|
}
|
|
}
|
|
} as const satisfies Command;
|
|
|
|
/**
|
|
* Raw entry structure from Redis stream
|
|
*/
|
|
type RawEntry = TuplesReply<[
|
|
id: BlobStringReply,
|
|
message: ArrayReply<BlobStringReply>
|
|
]> | NullReply;
|
|
|
|
/**
|
|
* Transforms a raw stream entry into a structured object
|
|
*
|
|
* @param entry - Raw entry from Redis
|
|
* @returns Structured object with id and message, or null if entry is null
|
|
*/
|
|
function transformEntry(entry: RawEntry) {
|
|
if (isNullReply(entry)) return entry;
|
|
|
|
const [id, message] = entry as unknown as UnwrapReply<typeof entry>;
|
|
return {
|
|
id,
|
|
message: transformTuplesReply(message)
|
|
};
|
|
}
|