From e869a38d1350f253aa50402c4424e5a9d6b3ff28 Mon Sep 17 00:00:00 2001 From: Madelyn Olson Date: Wed, 3 Jul 2024 15:56:22 -0700 Subject: [PATCH] Add information about 'server' domain object for LUA (#85) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Documents the behavior for the new server top level domain object and the new Lua fields. Also updates all of the examples to use the server object. Also includes some cleanup to references to "Valkey serialization protocol" which doesn't exist. There will be some minimal conflicts with https://github.com/valkey-io/valkey-doc/pull/22, as it attempts to remove the versioning references which this file also does for the files that were touched. --------- Signed-off-by: Madelyn Olson Signed-off-by: Viktor Söderqvist Co-authored-by: Viktor Söderqvist --- topics/lua-api.md | 84 +++++++++++++++++++++++++++++++---------------- 1 file changed, 56 insertions(+), 28 deletions(-) diff --git a/topics/lua-api.md b/topics/lua-api.md index d75af62b..04720a70 100644 --- a/topics/lua-api.md +++ b/topics/lua-api.md @@ -75,11 +75,15 @@ The only libraries that Valkey ships with and that you can use in scripts are li While the sandbox prevents users from declaring globals, the execution context is pre-populated with several of these. -### The _redis_ singleton +### The _server_ singleton -The _redis_ singleton is an object instance that's accessible from all scripts. +The _server_ singleton is an object instance that's accessible from all scripts. It provides the API to interact with Valkey from scripts. -Its description follows [below](#redis_object). +Its description follows [below](#server_object). + +**Note:** +For compatibility with Redis, Valkey also exposes a _redis_ top level object, that exposes the exact same set of APIs as the _server_ object. +Valkey does not intend to drop compatibility for this _redis_ API, but it is recommended to use the _server_ object for newly developed scripts. ### The _KEYS_ global variable @@ -104,15 +108,15 @@ It is pre-populated with all key name input arguments. The _ARGV_ global variable is available only in [ephemeral scripts](eval-intro.md). It is pre-populated with all regular input arguments. -## _redis_ object +## _server_ object * Since version: 2.6.0 * Available in scripts: yes * Available in functions: yes -The Valkey Lua execution context always provides a singleton instance of an object named _redis_. -The _redis_ instance enables the script to interact with the Valkey server that's running it. -Following is the API provided by the _redis_ object instance. +The Valkey Lua execution context always provides a singleton instance of an object named _server_. +The _server_ instance enables the script to interact with the Valkey server that's running it. +Following is the API provided by the _server_ object instance. ### `server.call(command [,arg...])` @@ -159,7 +163,7 @@ The following demonstrates how to use `server.pcall()` to intercept and handle r local reply = server.pcall('ECHO', unpack(ARGV)) if reply['err'] ~= nil then -- Handle the error sometime, but for now just log it - server.log(redis.LOG_WARNING, reply['err']) + server.log(server.LOG_WARNING, reply['err']) reply['err'] = 'ERR Something is wrong, but no worries, everything is under control' end return reply @@ -262,10 +266,10 @@ It expects two input arguments: the log level and a message. The message is a string to write to the log file. Log level can be on of these: -* `redis.LOG_DEBUG` -* `redis.LOG_VERBOSE` -* `redis.LOG_NOTICE` -* `redis.LOG_WARNING` +* `server.LOG_DEBUG` +* `server.LOG_VERBOSE` +* `server.LOG_NOTICE` +* `server.LOG_WARNING` These levels map to the server's log levels. The log only records messages equal or greater in level than the server's `loglevel` configuration directive. @@ -273,7 +277,7 @@ The log only records messages equal or greater in level than the server's `logle The following snippet: ```lua -server.log(redis.LOG_WARNING, 'Something is terribly wrong') +server.log(server.LOG_WARNING, 'Something is terribly wrong') ``` will produce a line similar to the following in your server's log: @@ -288,7 +292,7 @@ will produce a line similar to the following in your server's log: * Available in scripts: yes * Available in functions: yes -This function allows the executing script to switch between [Valkey Serialization Protocol (RESP)](protocol.md) versions for the replies returned by [`server.call()`](#server.call) and [`server.pcall()`](#server.pcall). +This function allows the executing script to switch between [RESP](protocol.md) protocol versions for the replies returned by [`server.call()`](#server.call) and [`server.pcall()`](#server.pcall). It expects a single numerical argument as the protocol's version. The default protocol version is _2_, but it can be switched to version _3_. @@ -333,13 +337,13 @@ Replicating the `SUNIONSTORE` command and the `DEL`ition of the temporary key is The `server.set_repl()` function instructs the server how to treat subsequent write commands in terms of replication. It accepts a single input argument that only be one of the following: -* `redis.REPL_ALL`: replicates the effects to the AOF and replicas. -* `redis.REPL_AOF`: replicates the effects to the AOF alone. -* `redis.REPL_REPLICA`: replicates the effects to the replicas alone. -* `redis.REPL_SLAVE`: same as `REPL_REPLICA`, maintained for backward compatibility. -* `redis.REPL_NONE`: disables effect replication entirely. +* `server.REPL_ALL`: replicates the effects to the AOF and replicas. +* `server.REPL_AOF`: replicates the effects to the AOF alone. +* `server.REPL_REPLICA`: replicates the effects to the replicas alone. +* `server.REPL_SLAVE`: same as `REPL_REPLICA`, maintained for backward compatibility. +* `server.REPL_NONE`: disables effect replication entirely. -By default, the scripting engine is initialized to the `redis.REPL_ALL` setting when a script begins its execution. +By default, the scripting engine is initialized to the `server.REPL_ALL` setting when a script begins its execution. You can call the `server.set_repl()` function at any time during the script's execution to switch between the different replication modes. A simple example follows: @@ -347,9 +351,9 @@ A simple example follows: ```lua server.replicate_commands() -- Enable effects replication in versions lower than Redis OSS v7.0 server.call('SET', KEYS[1], ARGV[1]) -server.set_repl(redis.REPL_NONE) +server.set_repl(server.REPL_NONE) server.call('SET', KEYS[2], ARGV[2]) -server.set_repl(redis.REPL_ALL) +server.set_repl(server.REPL_ALL) server.call('SET', KEYS[3], ARGV[3]) ``` @@ -498,9 +502,8 @@ You can use the following flags and instruct the server to treat the scripts' ex Please refer to [Function Flags](functions-intro.md#function-flags) and [Eval Flags](eval-intro.md#eval-flags) for a detailed example. -### `redis.REDIS_VERSION` +### `server.SERVER_VERSION` -* Since version: 7.0.0 * Available in scripts: yes * Available in functions: yes @@ -511,9 +514,21 @@ The reply's format is `MM.mm.PP`, where: * **mm:** is the minor version. * **PP:** is the patch level. -### `redis.REDIS_VERSION_NUM` +### `server.REDIS_VERSION` + +* Since version: 7.0.0 (if accessed as `redis.REDIS_VERSION`) +* Available in scripts: yes +* Available in functions: yes + +Returns the current Redis compatibility version as a Lua string. +The reply's format is `MM.mm.PP`, where: + +* **MM:** is the major version. +* **mm:** is the minor version. +* **PP:** is the patch level. + +### `server.SERVER_VERSION_NUM` -* Since version: 7.0.0 * Available in scripts: yes * Available in functions: yes @@ -524,6 +539,19 @@ The reply is a hexadecimal value structured as `0x00MMmmPP`, where: * **mm:** is the minor version. * **PP:** is the patch level. +### `server.REDIS_VERSION_NUM` + +* Since version: 7.0.0 (if accessed as `redis.REDIS_VERSION_NUM`) +* Available in scripts: yes +* Available in functions: yes + +Returns the current Redis compatibility version as a number. +The reply is a hexadecimal value structured as `0x00MMmmPP`, where: + +* **MM:** is the major version. +* **mm:** is the minor version. +* **PP:** is the patch level. + ## Data type conversion Unless a runtime exception is raised, `server.call()` and `server.pcall()` return the reply from the executed command to the Lua script. @@ -535,7 +563,7 @@ that reply is automatically converted to Valkey' protocol. Put differently; there's a one-to-one mapping between Valkey' replies and Lua's data types and a one-to-one mapping between Lua's data types and the [Valkey Protocol](protocol.md) data types. The underlying design is such that if a Valkey type is converted into a Lua type and converted back into a Valkey type, the result is the same as the initial value. -Type conversion from Valkey protocol replies (i.e., the replies from `server.call()` and `server.pcall()`) to Lua data types depends on the Valkey Serialization Protocol version used by the script. +Type conversion from Valkey replies (i.e. the replies from `server.call()` and `server.pcall()`) to Lua data types depends on the RESP protocol version used by the script. The default protocol version during script executions is RESP2. The script may switch the replies' protocol versions by calling the `server.setresp()` function. @@ -613,7 +641,7 @@ As you can see, the float value of _3.333_ gets converted to an integer _3_, the ### RESP3 to Lua type conversion -RESP3 is a newer version of Valkey's [Serialization Protocol](protocol.md). +RESP3 is a newer version of the [protocol](protocol.md) used by Valkey. It is available as an opt-in choice. An executing script may call the [`server.setresp`](#server.setresp) function during its execution and switch the protocol version that's used for returning replies from Valkey' commands (that can be invoked via [`server.call()`](#server.call) or [`server.pcall()`](#server.pcall)).