You've already forked node-redis
mirror of
https://github.com/redis/node-redis.git
synced 2025-08-04 15:02:09 +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
126 lines
4.0 KiB
TypeScript
126 lines
4.0 KiB
TypeScript
import { CommandParser } from '@redis/client/dist/lib/client/parser';
|
|
import { Command, ArrayReply, BlobStringReply, Resp2Reply, MapReply, TuplesReply, TypeMapping, RedisArgument, TuplesToMapReply, UnwrapReply } from '@redis/client/dist/lib/RESP/types';
|
|
import { RedisVariadicArgument } from '@redis/client/dist/lib/commands/generic-transformers';
|
|
import { resp2MapToValue, resp3MapToValue, SampleRawReply, Timestamp, transformSamplesReply } from './helpers';
|
|
import { TsRangeOptions, parseRangeArguments } from './RANGE';
|
|
import { parseFilterArgument } from './MGET';
|
|
|
|
export const TIME_SERIES_REDUCERS = {
|
|
AVG: 'AVG',
|
|
SUM: 'SUM',
|
|
MIN: 'MIN',
|
|
MAX: 'MAX',
|
|
RANGE: 'RANGE',
|
|
COUNT: 'COUNT',
|
|
STD_P: 'STD.P',
|
|
STD_S: 'STD.S',
|
|
VAR_P: 'VAR.P',
|
|
VAR_S: 'VAR.S'
|
|
} as const;
|
|
|
|
export type TimeSeriesReducer = typeof TIME_SERIES_REDUCERS[keyof typeof TIME_SERIES_REDUCERS];
|
|
|
|
export interface TsMRangeGroupBy {
|
|
label: RedisArgument;
|
|
REDUCE: TimeSeriesReducer;
|
|
}
|
|
|
|
/**
|
|
* Adds GROUPBY arguments to command
|
|
* @param parser - The command parser
|
|
* @param groupBy - Group by parameters
|
|
*/
|
|
export function parseGroupByArguments(parser: CommandParser, groupBy: TsMRangeGroupBy) {
|
|
parser.push('GROUPBY', groupBy.label, 'REDUCE', groupBy.REDUCE);
|
|
}
|
|
|
|
export type TsMRangeGroupByRawReply2 = ArrayReply<
|
|
TuplesReply<[
|
|
key: BlobStringReply,
|
|
labels: never, // empty array without WITHLABELS or SELECTED_LABELS
|
|
samples: ArrayReply<Resp2Reply<SampleRawReply>>
|
|
]>
|
|
>;
|
|
|
|
export type TsMRangeGroupByRawMetadataReply3 = TuplesToMapReply<[
|
|
[BlobStringReply<'sources'>, ArrayReply<BlobStringReply>]
|
|
]>;
|
|
|
|
export type TsMRangeGroupByRawReply3 = MapReply<
|
|
BlobStringReply,
|
|
TuplesReply<[
|
|
labels: never, // empty hash without WITHLABELS or SELECTED_LABELS
|
|
metadata1: never, // ?!
|
|
metadata2: TsMRangeGroupByRawMetadataReply3,
|
|
samples: ArrayReply<SampleRawReply>
|
|
]>
|
|
>;
|
|
|
|
/**
|
|
* Creates a function that parses arguments for multi-range commands with grouping
|
|
* @param command - The command name to use (TS.MRANGE or TS.MREVRANGE)
|
|
*/
|
|
export function createTransformMRangeGroupByArguments(command: RedisArgument) {
|
|
return (
|
|
parser: CommandParser,
|
|
fromTimestamp: Timestamp,
|
|
toTimestamp: Timestamp,
|
|
filter: RedisVariadicArgument,
|
|
groupBy: TsMRangeGroupBy,
|
|
options?: TsRangeOptions
|
|
) => {
|
|
parser.push(command);
|
|
parseRangeArguments(parser, fromTimestamp, toTimestamp, options)
|
|
|
|
parseFilterArgument(parser, filter);
|
|
|
|
parseGroupByArguments(parser, groupBy);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Extracts source keys from RESP3 metadata reply
|
|
* @param raw - Raw metadata from RESP3 reply
|
|
*/
|
|
export function extractResp3MRangeSources(raw: TsMRangeGroupByRawMetadataReply3) {
|
|
const unwrappedMetadata2 = raw as unknown as UnwrapReply<typeof raw>;
|
|
if (unwrappedMetadata2 instanceof Map) {
|
|
return unwrappedMetadata2.get('sources')!;
|
|
} else if (unwrappedMetadata2 instanceof Array) {
|
|
return unwrappedMetadata2[1];
|
|
} else {
|
|
return unwrappedMetadata2.sources;
|
|
}
|
|
}
|
|
|
|
export default {
|
|
IS_READ_ONLY: true,
|
|
/**
|
|
* Gets samples for time series matching a filter within a time range with grouping
|
|
* @param parser - The command parser
|
|
* @param fromTimestamp - Start timestamp for range
|
|
* @param toTimestamp - End timestamp for range
|
|
* @param filter - Filter to match time series keys
|
|
* @param groupBy - Group by parameters
|
|
* @param options - Optional parameters for the command
|
|
*/
|
|
parseCommand: createTransformMRangeGroupByArguments('TS.MRANGE'),
|
|
transformReply: {
|
|
2(reply: TsMRangeGroupByRawReply2, _?: any, typeMapping?: TypeMapping) {
|
|
return resp2MapToValue(reply, ([_key, _labels, samples]) => {
|
|
return {
|
|
samples: transformSamplesReply[2](samples)
|
|
};
|
|
}, typeMapping);
|
|
},
|
|
3(reply: TsMRangeGroupByRawReply3) {
|
|
return resp3MapToValue(reply, ([_labels, _metadata1, metadata2, samples]) => {
|
|
return {
|
|
sources: extractResp3MRangeSources(metadata2),
|
|
samples: transformSamplesReply[3](samples)
|
|
};
|
|
});
|
|
}
|
|
},
|
|
} as const satisfies Command;
|