diff --git a/CHANGELOG.md b/CHANGELOG.md index 7575a797c0..32f2da2aff 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,4 +1,5 @@ #### Changes +* Node: Added FUNCTION STATS commands ([#2082](https://github.com/valkey-io/valkey-glide/pull/2082)) * Node: Added XCLAIM command ([#2092](https://github.com/valkey-io/valkey-glide/pull/2092)) * Node: Added EXPIRETIME and PEXPIRETIME commands ([#2063](https://github.com/valkey-io/valkey-glide/pull/2063)) * Node: Added SORT commands ([#2028](https://github.com/valkey-io/valkey-glide/pull/2028)) diff --git a/node/npm/glide/index.ts b/node/npm/glide/index.ts index ef1b324437..992bc06190 100644 --- a/node/npm/glide/index.ts +++ b/node/npm/glide/index.ts @@ -106,6 +106,7 @@ function initialize() { GlideClientConfiguration, FunctionListOptions, FunctionListResponse, + FunctionStatsResponse, SlotIdTypes, SlotKeyTypes, RouteByAddress, @@ -195,6 +196,7 @@ function initialize() { GlideClientConfiguration, FunctionListOptions, FunctionListResponse, + FunctionStatsResponse, SlotIdTypes, SlotKeyTypes, RouteByAddress, diff --git a/node/src/Commands.ts b/node/src/Commands.ts index 8353f79c29..08d6b61312 100644 --- a/node/src/Commands.ts +++ b/node/src/Commands.ts @@ -2149,6 +2149,19 @@ export function createFunctionList( return createCommand(RequestType.FunctionList, args); } +/** Type of the response of `FUNCTION STATS` command. */ +export type FunctionStatsResponse = Record< + string, + | null + | Record + | Record> +>; + +/** @internal */ +export function createFunctionStats(): command_request.Command { + return createCommand(RequestType.FunctionStats, []); +} + /** * Represents offsets specifying a string interval to analyze in the {@link BaseClient.bitcount|bitcount} command. The offsets are * zero-based indexes, with `0` being the first index of the string, `1` being the next index and so on. diff --git a/node/src/GlideClient.ts b/node/src/GlideClient.ts index 1c62b34506..daedf2935d 100644 --- a/node/src/GlideClient.ts +++ b/node/src/GlideClient.ts @@ -14,6 +14,7 @@ import { FlushMode, FunctionListOptions, FunctionListResponse, + FunctionStatsResponse, InfoOptions, LolwutOptions, SortOptions, @@ -33,6 +34,7 @@ import { createFunctionFlush, createFunctionList, createFunctionLoad, + createFunctionStats, createInfo, createLastSave, createLolwut, @@ -545,6 +547,58 @@ export class GlideClient extends BaseClient { return this.createWritePromise(createFunctionList(options)); } + /** + * Returns information about the function that's currently running and information about the + * available execution engines. + * + * See https://valkey.io/commands/function-stats/ for details. + * + * since Valkey version 7.0.0. + * + * @returns A `Record` with two keys: + * - `"running_script"` with information about the running script. + * - `"engines"` with information about available engines and their stats. + * + * See example for more details. + * + * @example + * ```typescript + * const response = await client.functionStats(); + * console.log(response); // Output: + * // { + * // "running_script": + * // { + * // "name": "deep_thought", + * // "command": ["fcall", "deep_thought", "0"], + * // "duration_ms": 5008 + * // }, + * // "engines": + * // { + * // "LUA": + * // { + * // "libraries_count": 2, + * // "functions_count": 3 + * // } + * // } + * // } + * // Output if no scripts running: + * // { + * // "running_script": null + * // "engines": + * // { + * // "LUA": + * // { + * // "libraries_count": 2, + * // "functions_count": 3 + * // } + * // } + * // } + * ``` + */ + public async functionStats(): Promise { + return this.createWritePromise(createFunctionStats()); + } + /** * Deletes all the keys of all the existing databases. This command never fails. * diff --git a/node/src/GlideClusterClient.ts b/node/src/GlideClusterClient.ts index 14f5e0f7fb..bfd00dbc5f 100644 --- a/node/src/GlideClusterClient.ts +++ b/node/src/GlideClusterClient.ts @@ -14,6 +14,7 @@ import { FlushMode, FunctionListOptions, FunctionListResponse, + FunctionStatsResponse, InfoOptions, LolwutOptions, SortClusterOptions, @@ -35,6 +36,7 @@ import { createFunctionFlush, createFunctionList, createFunctionLoad, + createFunctionStats, createInfo, createLastSave, createLolwut, @@ -834,7 +836,7 @@ export class GlideClusterClient extends BaseClient { * since Valkey version 7.0.0. * * @param mode - The flushing mode, could be either {@link FlushMode.SYNC} or {@link FlushMode.ASYNC}. - * @param route - The command will be routed to all primary node, unless `route` is provided, in which + * @param route - The command will be routed to all primary nodes, unless `route` is provided, in which * case the client will route the command to the nodes defined by `route`. * @returns A simple OK response. * @@ -892,6 +894,65 @@ export class GlideClusterClient extends BaseClient { ); } + /** + * Returns information about the function that's currently running and information about the + * available execution engines. + * + * See https://valkey.io/commands/function-stats/ for details. + * + * since Valkey version 7.0.0. + * + * @param route - The client will route the command to the nodes defined by `route`. + * If not defined, the command will be routed to all primary nodes. + * @returns A `Record` with two keys: + * - `"running_script"` with information about the running script. + * - `"engines"` with information about available engines and their stats. + * + * See example for more details. + * + * @example + * ```typescript + * const response = await client.functionStats("randomNode"); + * console.log(response); // Output: + * // { + * // "running_script": + * // { + * // "name": "deep_thought", + * // "command": ["fcall", "deep_thought", "0"], + * // "duration_ms": 5008 + * // }, + * // "engines": + * // { + * // "LUA": + * // { + * // "libraries_count": 2, + * // "functions_count": 3 + * // } + * // } + * // } + * // Output if no scripts running: + * // { + * // "running_script": null + * // "engines": + * // { + * // "LUA": + * // { + * // "libraries_count": 2, + * // "functions_count": 3 + * // } + * // } + * // } + * ``` + */ + public async functionStats( + route?: Routes, + ): Promise> { + return this.createWritePromise( + createFunctionStats(), + toProtobufRoute(route), + ); + } + /** * Deletes all the keys of all the existing databases. This command never fails. * diff --git a/node/src/Transaction.ts b/node/src/Transaction.ts index 20e35b49ee..c39763be84 100644 --- a/node/src/Transaction.ts +++ b/node/src/Transaction.ts @@ -24,6 +24,7 @@ import { FlushMode, FunctionListOptions, FunctionListResponse, // eslint-disable-line @typescript-eslint/no-unused-vars + FunctionStatsResponse, // eslint-disable-line @typescript-eslint/no-unused-vars GeoAddOptions, GeoBoxShape, // eslint-disable-line @typescript-eslint/no-unused-vars GeoCircleShape, // eslint-disable-line @typescript-eslint/no-unused-vars @@ -87,6 +88,7 @@ import { createFunctionFlush, createFunctionList, createFunctionLoad, + createFunctionStats, createGeoAdd, createGeoDist, createGeoHash, @@ -2492,6 +2494,23 @@ export class BaseTransaction> { return this.addAndReturn(createFunctionList(options)); } + /** + * Returns information about the function that's currently running and information about the + * available execution engines. + * + * See https://valkey.io/commands/function-stats/ for details. + * + * since Valkey version 7.0.0. + * + * Command Response - A `Record` of type {@link FunctionStatsResponse} with two keys: + * + * - `"running_script"` with information about the running script. + * - `"engines"` with information about available engines and their stats. + */ + public functionStats(): T { + return this.addAndReturn(createFunctionStats()); + } + /** * Deletes all the keys of all the existing databases. This command never fails. * diff --git a/node/tests/GlideClient.test.ts b/node/tests/GlideClient.test.ts index 11b75ef82e..a356812ece 100644 --- a/node/tests/GlideClient.test.ts +++ b/node/tests/GlideClient.test.ts @@ -25,6 +25,7 @@ import { command_request } from "../src/ProtobufMessage"; import { runBaseTests } from "./SharedTests"; import { checkFunctionListResponse, + checkFunctionStatsResponse, convertStringArrayToBuffer, flushAndCloseClient, generateLuaLibCode, @@ -507,7 +508,7 @@ describe("GlideClient", () => { ); it.each([ProtocolVersion.RESP2, ProtocolVersion.RESP3])( - "function load test_%p", + "function load function list function stats test_%p", async (protocol) => { if (cluster.checkIfServerVersionLessThan("7.0.0")) return; @@ -534,6 +535,9 @@ describe("GlideClient", () => { await client.fcallReadonly(funcName, [], ["one", "two"]), ).toEqual("one"); + let functionStats = await client.functionStats(); + checkFunctionStatsResponse(functionStats, [], 1, 1); + let functionList = await client.functionList({ libNamePattern: libName, }); @@ -592,6 +596,9 @@ describe("GlideClient", () => { newCode, ); + functionStats = await client.functionStats(); + checkFunctionStatsResponse(functionStats, [], 1, 2); + expect( await client.fcall(func2Name, [], ["one", "two"]), ).toEqual(2); @@ -600,6 +607,8 @@ describe("GlideClient", () => { ).toEqual(2); } finally { expect(await client.functionFlush()).toEqual("OK"); + const functionStats = await client.functionStats(); + checkFunctionStatsResponse(functionStats, [], 0, 0); client.close(); } }, diff --git a/node/tests/GlideClusterClient.test.ts b/node/tests/GlideClusterClient.test.ts index 1edb82e84b..04943619f2 100644 --- a/node/tests/GlideClusterClient.test.ts +++ b/node/tests/GlideClusterClient.test.ts @@ -25,11 +25,17 @@ import { ScoreFilter, } from ".."; import { RedisCluster } from "../../utils/TestUtils.js"; -import { FlushMode, GeoUnit, SortOrder } from "../build-ts/src/Commands"; +import { + FlushMode, + FunctionStatsResponse, + GeoUnit, + SortOrder, +} from "../build-ts/src/Commands"; import { runBaseTests } from "./SharedTests"; import { checkClusterResponse, checkFunctionListResponse, + checkFunctionStatsResponse, flushAndCloseClient, generateLuaLibCode, getClientConfigurationOption, @@ -739,7 +745,7 @@ describe("GlideClusterClient", () => { "Single node route = %s", (singleNodeRoute) => { it( - "function load and function list", + "function load function list function stats", async () => { if (cluster.checkIfServerVersionLessThan("7.0.0")) return; @@ -775,6 +781,21 @@ describe("GlideClusterClient", () => { singleNodeRoute, (value) => expect(value).toEqual([]), ); + + let functionStats = + await client.functionStats(route); + checkClusterResponse( + functionStats as object, + singleNodeRoute, + (value) => + checkFunctionStatsResponse( + value as FunctionStatsResponse, + [], + 0, + 0, + ), + ); + // load the library expect(await client.functionLoad(code)).toEqual( libName, @@ -803,6 +824,19 @@ describe("GlideClusterClient", () => { expectedFlags, ), ); + functionStats = + await client.functionStats(route); + checkClusterResponse( + functionStats as object, + singleNodeRoute, + (value) => + checkFunctionStatsResponse( + value as FunctionStatsResponse, + [], + 1, + 1, + ), + ); // call functions from that library to confirm that it works let fcall = await client.fcallWithRoute( @@ -881,6 +915,19 @@ describe("GlideClusterClient", () => { newCode, ), ); + functionStats = + await client.functionStats(route); + checkClusterResponse( + functionStats as object, + singleNodeRoute, + (value) => + checkFunctionStatsResponse( + value as FunctionStatsResponse, + [], + 1, + 2, + ), + ); fcall = await client.fcallWithRoute( func2Name, diff --git a/node/tests/TestUtilities.ts b/node/tests/TestUtilities.ts index 22283738f7..4d4d6d07d6 100644 --- a/node/tests/TestUtilities.ts +++ b/node/tests/TestUtilities.ts @@ -19,6 +19,7 @@ import { ClusterTransaction, FlushMode, FunctionListResponse, + FunctionStatsResponse, GeoUnit, GeospatialData, GlideClient, @@ -396,6 +397,43 @@ export function checkFunctionListResponse( expect(hasLib).toBeTruthy(); } +/** + * Validate whether `FUNCTION STATS` response contains required info. + * + * @param response - The response from server. + * @param runningFunction - Command line of running function expected. Empty, if nothing expected. + * @param libCount - Expected libraries count. + * @param functionCount - Expected functions count. + */ +export function checkFunctionStatsResponse( + response: FunctionStatsResponse, + runningFunction: string[], + libCount: number, + functionCount: number, +) { + if (response.running_script === null && runningFunction.length > 0) { + fail("No running function info"); + } + + if (response.running_script !== null && runningFunction.length == 0) { + fail( + "Unexpected running function info: " + + (response.running_script.command as string[]).join(" "), + ); + } + + if (response.running_script !== null) { + expect(response.running_script.command).toEqual(runningFunction); + // command line format is: + // fcall|fcall_ro * * + expect(response.running_script.name).toEqual(runningFunction[1]); + } + + expect(response.engines).toEqual({ + LUA: { libraries_count: libCount, functions_count: functionCount }, + }); +} + /** * Check transaction response. * @param response - Transaction result received from `exec` call. @@ -1183,6 +1221,8 @@ export async function transactionTest( ); if (gte(version, "7.0.0")) { + baseTransaction.functionFlush(); + responseData.push(["functionFlush()", "OK"]); baseTransaction.functionLoad(code); responseData.push(["functionLoad(code)", libName]); baseTransaction.functionLoad(code, true); @@ -1196,6 +1236,14 @@ export async function transactionTest( 'fcallReadonly(funcName, [], ["one", "two"]', "one", ]); + baseTransaction.functionStats(); + responseData.push([ + "functionStats()", + { + running_script: null, + engines: { LUA: { libraries_count: 1, functions_count: 1 } }, + }, + ]); baseTransaction.functionDelete(libName); responseData.push(["functionDelete(libName)", "OK"]); baseTransaction.functionFlush();