io.craft.atom.redis.api.RedisCommand Maven / Gradle / Ivy
package io.craft.atom.redis.api;
import io.craft.atom.redis.api.handler.RedisMonitorHandler;
import io.craft.atom.redis.api.handler.RedisPsubscribeHandler;
import io.craft.atom.redis.api.handler.RedisSubscribeHandler;
import java.util.List;
import java.util.Map;
import java.util.Set;
/**
* All redis commands.
*
* @author mindwind
* @version 1.0, Jun 26, 2013
*/
public interface RedisCommand {
// ~ --------------------------------------------------------------------------------------------------------- Keys
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the number of keys that will be removed.
*
*
* When a key to remove holds a value other than a string,
* the individual complexity for this key is O(M)
* where M is the number of elements in the list, set, sorted set or hash.
* Removing a single key that holds a string value is O(1).
*
*
* Removes the specified keys. A key is ignored if it does not exist.
*
*
* @param keys
* @return The number of keys that were removed.
*/
Long del(String... keys);
/**
* Available since 2.6.0
* Time complexity: O(1) to access the key and additional O(N*M) to serialized it,
* where N is the number of Redis objects composing the value and M their average size.
* For small string values the time complexity is thus O(1)+O(1*M) where M is small, so simply O(1).
*
*
* Serialize the value stored at key in a Redis-specific format and return it to the user.
* The returned value can be synthesized back into a Redis key using the RESTORE command.
*
* The serialization format is opaque and non-standard, however it has a few semantical characteristics:
* - It contains a 64-bit checksum that is used to make sure errors will be detected.
* The RESTORE command makes sure to check the checksum before synthesizing a key using the serialized value.
* - Values are encoded in the same format used by RDB.
* - An RDB version is encoded inside the serialized value,
* so that different Redis versions with incompatible RDB formats will refuse to process the serialized value.
*
* The serialized value does NOT contain expire information.
* In order to capture the time to live of the current value the PTTL command should be used.
*
* If key does not exist a null bulk reply is returned.
*
* @param key
* @return the serialized value
*/
byte[] dump(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Returns trueif key exists.
*
* @param key
* @return true if the key exists.
* false if the key does not exist.
*/
Boolean exists(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Set a timeout on key. After the timeout has expired, the key will automatically be deleted.
* A key with an associated timeout is often said to be volatile in Redis terminology.
*
* The timeout is cleared only when the key is removed using the DEL command or
* overwritten using the SET or GETSET commands.
* This means that all the operations that conceptually alter the value stored at the key
* without replacing it with a new one will leave the timeout untouched.
* For instance, incrementing the value of a key with INCR, pushing a new value into a list with LPUSH,
* or altering the field value of a hash with HSET are all operations that will leave the timeout untouched.
*
* The timeout can also be cleared, turning the key back into a persistent key, using the PERSIST command.
* If a key is renamed with RENAME, the associated time to live is transferred to the new key name.
* If a key is overwritten by RENAME, like in the case of an existing key Key_A that is overwritten
* by a call like RENAME Key_B Key_A, it does not matter if the original Key_A had a timeout associated or not,
* the new key Key_A will inherit all the characteristics of Key_B.
*
* It is possible to call EXPIRE using as argument a key that already has an existing expire set.
* In this case the time to live of a key is updated to the new value. There are many useful applications for this.
*
* @param key
* @param seconds
* @return 1 if the timeout was set.
* 0 if key does not exist or the timeout could not be set.
*/
Long expire(String key, int seconds);
/**
* Available since 1.2.0
* Time complexity: O(1)
*
*
* EXPIREAT has the same effect and semantic as EXPIRE, but instead of specifying the number of seconds representing
* the TTL (time to live), it takes an absolute Unix timestamp (seconds since January 1, 1970).
*
* EXPIREAT was introduced in order to convert relative timeouts to absolute timeouts for the AOF persistence mode.
* Of course, it can be used directly to specify that a given key should expire at a given time in the future.
*
* @param key
* @param timestamp a unix timestamp
* @return 1 if the timeout was set.
* 0 if key does not exist or the timeout could not be set .
*/
Long expireat(String key, long timestamp);
/**
* Available since 1.0.0
* Time complexity: O(N) with N being the number of keys in the database,
* under the assumption that the key names in the database and the given pattern have limited length.
*
*
* Returns all keys matching pattern.
* While the time complexity for this operation is O(N), the constant times are fairly low.
* For example, Redis running on an entry level laptop can scan a 1 million key database in 40 milliseconds.
* Warning: consider KEYS as a command that should only be used in production environments with extreme care.
* It may ruin performance when it is executed against large databases.
* This command is intended for debugging and special operations, such as changing your keyspace layout.
* Don't use KEYS in your regular application code. If you're looking for a way to find keys in a subset of your keyspace, consider using sets.
* Supported glob-style patterns:
*
* h?llo matches hello, hallo and hxllo
* h*llo matches hllo and heeeello
* h[ae]llo matches hello and hallo, but not hillo
* Use \ to escape special characters if you want to match them verbatim.
*
*
* @param pattern
* @return ist of keys matching pattern.
*/
Set keys(String pattern);
/**
* Available since 2.6.0
* Time complexity: This command actually executes a DUMP+DEL in the source instance, and a RESTORE in the target instance.
* See the pages of these commands for time complexity. Also an O(N) data transfer between the two instances is performed.
*
*
* Atomically transfer a key from a source Redis instance to a destination Redis instance.
* On success the key is deleted from the original instance and is guaranteed to exist in the target instance.
* The command is atomic and blocks the two instances for the time required to transfer the key,
* at any given time the key will appear to exist in a given instance or in the other instance, unless a timeout error occurs.
*
* The command internally uses DUMP to generate the serialized version of the key value, and RESTORE in order
* to synthesize the key in the target instance. The source instance acts as a client for the target instance.
* If the target instance returns OK to the RESTORE command, the source instance deletes the key using DEL.
*
* The timeout specifies the maximum idle time in any moment of the communication with the destination instance in milliseconds.
* This means that the operation does not need to be completed within the specified amount of milliseconds,
* but that the transfer should make progresses without blocking for more than the specified amount of milliseconds.
*
* MIGRATE needs to perform I/O operations and to honor the specified timeout.
* When there is an I/O error during the transfer or if the timeout is reached the operation is aborted and the special error - IOERR returned.
* When this happens the following two cases are possible:
* - The key may be on both the instances.
* - The key may be only in the source instance.
* It is not possible for the key to get lost in the event of a timeout, but the client calling MIGRATE,
* in the event of a timeout error, should check if the key is also present in the target instance and act accordingly.
* When any other error is returned (starting with ERR) MIGRATE guarantees that the key is still only present in the
* originating instance (unless a key with the same name was also already present on the target instance).
*
* On success OK is returned.
*
* @param host
* @param port
* @param key
* @param destinationdb
* @param timeout in milliseconds
* @return OK
*/
String migrate(String host, int port, String key, int destinationdb, int timeout);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Move key from the currently selected database (see SELECT) to the specified destination database.
* When key already exists in the destination database, or it does not exist in the source database, it does nothing.
* It is possible to use MOVE as a locking primitive because of this.
*
* @param key
* @param db
* @return 1 if key was moved.
* 0 if key was not moved.
*/
Long move(String key, int db);
/**
* Available since 2.2.3
* Time complexity: O(1)
*
*
* The OBJECT command allows to inspect the internals of Redis Objects associated with keys.
* It is useful for debugging or to understand if your keys are using the specially encoded data types to save space.
* Your application may also use the information reported by the OBJECT command to implement application level
* key eviction policies when using Redis as a Cache.
*
* The OBJECT command supports multiple sub commands:
* OBJECT REFCOUNT key
* returns the number of references of the value associated with the specified key.
* This command is mainly useful for debugging.
*
* OBJECT ENCODING key
* returns the kind of internal representation used in order to store the value associated with a key.
*
* OBJECT IDLETIME key
* returns the number of seconds since the object stored at the specified key is idle (not requested by read or write operations).
* While the value is returned in seconds the actual resolution of this timer is 10 seconds, but may vary in future implementations.
*
*
* Objects can be encoded in different ways:
* - Strings can be encoded as raw (normal string encoding) or int (strings representing integers in a 64 bit signed
* interval are encoded in this way in order to save space).
* - Lists can be encoded as ziplist or linkedlist. The ziplist is the special representation that is used to save space for small lists.
* - Sets can be encoded as intset or hashtable. The intset is a special encoding used for small sets composed solely of integers.
* - Hashes can be encoded as zipmap or hashtable. The zipmap is a special encoding used for small hashes.
* - Sorted Sets can be encoded as ziplist or skiplist format. As for the List type small sorted sets can be specially
* - encoded using ziplist, while the skiplist encoding is the one that works with sorted sets of any size.
*
*
* All the specially encoded types are automatically converted to the general type once you perform an operation
* that makes it no possible for Redis to retain the space saving encoding.
*
* @param key
* @return Different return values are used for different subcommands.
* Subcommands refcount and idletime returns integers.
* Subcommand encoding returns a bulk reply.
* If the object you try to inspect is missing, a null bulk reply is returned.
*/
Long objectrefcount(String key);
String objectencoding(String key);
Long objectidletime(String key);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* Remove the existing timeout on key, turning the key from volatile (a key with an expire set)
* to persistent (a key that will never expire as no timeout is associated).
*
* @param key
* @return 1 if the timeout was removed.
* 0 if key does not exist or does not have an associated timeout.
*/
Long persist(String key);
/**
* Available since 2.6.0
* Time complexity: O(1)
*
*
* This command works exactly like EXPIRE but the time to live of the key is specified in milliseconds instead of seconds.
* @param key
* @param milliseconds
* @return 1 if the timeout was set.
* 0 if key does not exist or the timeout could not be set.
*/
Long pexpire(String key, long milliseconds);
/**
* Available since 2.6.0
* Time complexity: O(1)
*
*
* PEXPIREAT has the same effect and semantic as EXPIREAT,
* but the Unix time at which the key will expire is specified in milliseconds instead of seconds.
*
* @param key
* @param millisecondstimestamp
* @return 1 if the timeout was set.
* 0 if key does not exist or the timeout could not be set
*/
Long pexpireat(String key, long millisecondstimestamp);
/**
* Available since 2.6.0
* Time complexity: O(1)
*
*
* Like TTL this command returns the remaining time to live of a key that has an expire set,
* with the sole difference that TTL returns the amount of remaining time in seconds while PTTL returns it in milliseconds.
*
* @param key
* @return Time to live in milliseconds or -1 when key does not exist or does not have a timeout.
*/
Long pttl(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Return a random key from the currently selected database.
*
* @return the random key, or nil when the database is empty.
*/
String randomkey();
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Renames key to newkey.
* It returns an error when the source and destination names are the same, or when key does not exist.
* If newkey already exists it is overwritten.
*
* @param key
* @param newkey
* @return The command returns OK on success.
*/
String rename(String key, String newkey);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Renames key to newkey if newkey does not yet exist. It returns an error under the same conditions as RENAME.
*
* @param key
* @param newkey
* @return 1 if key was renamed to newkey.
* 0 if newkey already exists.
*/
Long renamenx(String key, String newkey);
/**
* Available since 2.6.0
* Time complexity: O(1) to create the new key and additional O(N*M) to recostruct the serialized value,
* where N is the number of Redis objects composing the value and M their average size.
* For small string values the time complexity is thus O(1)+O(1*M) where M is small, so simply O(1).
* However for sorted set values the complexity is O(N*M*log(N)) because inserting values into sorted sets is O(log(N)).
*
*
* Create a key associated with a value that is obtained by deserializing the provided serialized value (obtained via DUMP).
* If ttl is 0 the key is created without any expire, otherwise the specified expire time (in milliseconds) is set.
* RESTORE checks the RDB version and data checksum. If they don't match an error is returned.
*
* @param key
* @param ttl in milliseconds
* @param serializedvalue
* @return The command returns OK on success.
*/
String restore(String key, int ttl, byte[] serializedvalue);
/**
* Available since 1.0.0
* Time complexity: O(N+M*log(M)) where N is the number of elements in the list or set to sort, and M the number of returned elements.
* When the elements are not sorted, complexity is currently O(N) as there is a copy step that will be avoided in next releases.
*
*
* Returns or stores the elements contained in the list, set or sorted set at key.
* By default, sorting is numeric and elements are compared by their value interpreted as double precision floating point number.
* This is SORT in its simplest form:
*
* SORT mylist
*
*
* Assuming mylist is a list of numbers, this command will return the same list with the elements sorted from small to large.
* In order to sort the numbers from large to small, use the DESC modifier:
*
* SORT mylist DESC
*
*
* When mylist contains string values and you want to sort them lexicographically, use the ALPHA modifier:
*
* SORT mylist ALPHA
*
*
* Redis is UTF-8 aware, assuming you correctly set the !LC_COLLATE environment variable.
* The number of returned elements can be limited using the LIMIT modifier.
* This modifier takes the offset argument, specifying the number of elements to skip and the count argument,
* specifying the number of elements to return from starting at offset.
* The following example will return 10 elements of the sorted version of mylist, starting at element 0 (offset is zero-based):
*
* SORT mylist LIMIT 0 10
*
*
* Almost all modifiers can be used together. The following example will return the first 5 elements, lexicographically sorted in descending order:
*
* SORT mylist LIMIT 0 5 ALPHA DESC
*
*
* Sometimes you want to sort elements using external keys as weights to compare instead of comparing the actual elements in the list, set or sorted set.
* Let's say the list mylist contains the elements 1, 2 and 3 representing unique IDs of objects stored in object_1, object_2 and object_3.
* When these objects have associated weights stored in weight_1, weight_2 and weight_3,
* SORT can be instructed to use these weights to sort mylist with the following statement:
*
* SORT mylist BY weight_*
*
* The BY option takes a pattern (equal to weight_* in this example) that is used to generate the keys that are used for sorting.
* These key names are obtained substituting the first occurrence of * with the actual value of the element in the list (1, 2 and 3 in this example).
*
*
* The BY option can also take a non-existent key, which causes SORT to skip the sorting operation.
* This is useful if you want to retrieve external keys (see the GET option below) without the overhead of sorting.
*
* SORT mylist BY nosort
*
*
* Our previous example returns just the sorted IDs. In some cases, it is more useful to get the actual objects
* instead of their IDs (object_1, object_2 and object_3).
* Retrieving external keys based on the elements in a list,
* set or sorted set can be done with the following command:
*
* SORT mylist BY weight_* GET object_*
*
*
* The GET option can be used multiple times in order to get more keys for every element of the original list, set or sorted set.
* It is also possible to GET the element itself using the special pattern #:
*
* SORT mylist BY weight_* GET object_* GET #
*
*
* By default, SORT returns the sorted elements to the client. With the STORE option, the result will be stored as
* a list at the specified key instead of being returned to the client.
*
* SORT mylist BY weight_* STORE resultkey
*
* An interesting pattern using SORT ... STORE consists in associating an EXPIRE timeout to the resulting key
* so that in applications where the result of a SORT operation can be cached for some time.
* Other clients will use the cached list instead of calling SORT for every request. When the key will timeout,
* an updated version of the cache can be created by calling SORT ... STORE again.
* Note that for correctly implementing this pattern it is important to avoid multiple clients rebuilding the cache
* at the same time. Some kind of locking is needed here (for instance using SETNX).
*
*
* It is possible to use BY and GET options against hash fields with the following syntax:
*
* SORT mylist BY weight_*->fieldname GET object_*->fieldname
*
* The string -> is used to separate the key name from the hash field name.
* The key is substituted as documented above, and the hash stored at the resulting key is accessed to retrieve the specified hash field.
*
*
* For example:
*
* SORT mylist
* SORT mylist DESC
* SORT mylist ALPHA
* SORT mylist ALPHA DESC
* SORT mylist LIMIT 0 10
* SORT mylist LIMIT 0 10 ALPHA DESC
*
* SORT mylist BY weight_*
* SORT mylist BY nosort
* SORT mylist BY weight_* DESC
* SORT mylist BY weight_* ALPHA DESC
* SORT mylist BY weight_* LIMIT 0 10
* SORT mylist BY weight_* LIMIT 0 10 ALPHA DESC
*
* SORT mylist BY weight_* GET object_*
* SORT mylist BY weight_*->fieldname GET object_*->fieldname
* SORT mylist BY weight_* GET object_* GET #
* SORT mylist BY weight_* GET object_* GET # DESC
* SORT mylist BY weight_* GET object_* GET # ALPHA DESC
* SORT mylist BY weight_* GET object_* GET # LIMIT 0 10
* SORT mylist BY weight_* GET object_* GET # LIMIT 0 10 ALPHA DESC
*
* SORT mylist STORE resultkey
* SORT mylist DESC STORE resultkey
* SORT mylist ALPHA STORE resultkey
* SORT mylist ALPHA DESC STORE resultkey
* SORT mylist LIMIT 0 10 STORE resultkey
* SORT mylist LIMIT 0 10 ALPHA DESC STORE resultkey
*
* SORT mylist BY weight_* STORE resultkey
* SORT mylist BY weight_* DESC STORE resultkey
* SORT mylist BY weight_* ALPHA DESC STORE resultkey
* SORT mylist BY weight_* LIMIT 0 10 STORE resultkey
* SORT mylist BY weight_* LIMIT 0 10 ALPHA DESC STORE resultkey
*
* SORT mylist BY weight_* GET object_* GET # STORE resultkey
* SORT mylist BY weight_* GET object_* GET # DESC STORE resultkey
* SORT mylist BY weight_* GET object_* GET # ALPHA DESC STORE resultkey
* SORT mylist BY weight_* GET object_* GET # LIMIT 0 10 STORE resultkey
* SORT mylist BY weight_* GET object_* GET # LIMIT 0 10 ALPHA DESC STORE resultkey
*
*
* @param key
* @return list of sorted elements.
*/
List sort(String key);
List sort(String key, boolean desc);
List sort(String key, boolean alpha, boolean desc);
List sort(String key, int offset, int count);
List sort(String key, int offset, int count, boolean alpha, boolean desc);
List sort(String key, String bypattern, String... getpatterns);
List sort(String key, String bypattern, boolean desc, String... getpatterns);
List sort(String key, String bypattern, boolean alpha, boolean desc, String... getpatterns);
List sort(String key, String bypattern, int offset, int count, String... getpatterns);
List sort(String key, String bypattern, int offset, int count, boolean alpha, boolean desc, String... getpatterns);
/**
* @see #sort(String)
* @return The number of elements of the list at destination.
*/
Long sort(String key, String destination);
Long sort(String key, boolean desc, String destination);
Long sort(String key, boolean alpha, boolean desc, String destination);
Long sort(String key, int offset, int count, String destination);
Long sort(String key, int offset, int count, boolean alpha, boolean desc, String destination);
Long sort(String key, String bypattern, String destination, String... getpatterns);
Long sort(String key, String bypattern, boolean desc, String destination, String... getpatterns);
Long sort(String key, String bypattern, boolean alpha, boolean desc, String destination, String... getpatterns);
Long sort(String key, String bypattern, int offset, int count, String destination, String... getpatterns);
Long sort(String key, String bypattern, int offset, int count, boolean alpha, boolean desc, String destination, String... getpatterns);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Returns the remaining time to live of a key that has a timeout.
* This introspection capability allows a Redis client to check how many seconds a given key will continue to be part of the dataset.
*
*
* >=2.6, return -1 when key does not have a timeout or not exist
*
* @param key
* @return TTL in seconds, -2 when key does not exist or -1 when key does not have a timeout.
*/
Long ttl(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Returns the string representation of the type of the value stored at key.
* The different types that can be returned are: string, list, set, zset and hash.
*
* @param key
* @return type of key, or none when key does not exist.
*/
String type(String key);
/**
* Available since 2.8.0
* Time complexity: O(1) for every call. O(N) for a complete iteration, including enough command calls for the cursor
* to return back to 0. N is the number of elements inside the collection..
*
*
* The SCAN command and the closely related commands SSCAN, HSCAN and ZSCAN are used in order to incrementally iterate over a collection of elements.
* SCAN iterates the set of keys in the currently selected Redis database.
* SSCAN iterates elements of Sets types.
* HSCAN iterates fields of Hash types and their associated values.
* ZSCAN iterates elements of Sorted Set types and their associated scores.
* Since these commands allow for incremental iteration, returning only a small number of elements per call, they can be used in production without the downside of commands like KEYS or SMEMBERS that may block the server for a long time (even several seconds) when called against big collections of keys or elements.
* However while blocking commands like SMEMBERS are able to provide all the elements that are part of a Set in a given moment, The SCAN family of commands only offer limited guarantees about the returned elements since the collection that we incrementally iterate can change during the iteration process.
* Note that SCAN, SSCAN, HSCAN and ZSCAN all work very similarly, so this documentation covers all the four commands. However an obvious difference is that in the case of SSCAN, HSCAN and ZSCAN the first argument is the name of the key holding the Set, Hash or Sorted Set value. The SCAN command does not need any key name argument as it iterates keys in the current database, so the iterated object is the database itself.
*
*
* SCAN basic usage
* SCAN is a cursor based iterator. This means that at every call of the command, the server returns an updated cursor that the user needs to use as the cursor argument in the next call.
* An iteration starts when the cursor is set to 0, and terminates when the cursor returned by the server is 0. The following is an example of SCAN iteration:
*
* redis 127.0.0.1:6379> scan 0
* 1) "17"
* 2) 1) "key:12"
* 2) "key:8"
* 3) "key:4"
* 4) "key:14"
* 5) "key:16"
* 6) "key:17"
* 7) "key:15"
* 8) "key:10"
* 9) "key:3"
* 10) "key:7"
* 11) "key:1"
* redis 127.0.0.1:6379> scan 17
* 1) "0"
* 2) 1) "key:5"
* 2) "key:18"
* 3) "key:0"
* 4) "key:2"
* 5) "key:19"
* 6) "key:13"
* 7) "key:6"
* 8) "key:9"
* 9) "key:11"
*
* In the example above, the first call uses zero as a cursor, to start the iteration. The second call uses the cursor returned by the previous call as the first element of the reply, that is, 17.
* As you can see the SCAN return value is an array of two values: the first value is the new cursor to use in the next call, the second value is an array of elements.
* Since in the second call the returned cursor is 0, the server signaled to the caller that the iteration finished, and the collection was completely explored. Starting an iteration with a cursor value of 0, and calling SCAN until the returned cursor is 0 again is called a full iteration.
*
*
* Scan guarantees
* The SCAN command, and the other commands in the SCAN family, are able to provide to the user a set of guarantees associated to full iterations.
* A full iteration always retrieves all the elements that were present in the collection from the start to the end of a full iteration. This means that if a given element is inside the collection when an iteration is started, and is still there when an iteration terminates, then at some point SCAN returned it to the user.
* A full iteration never returns any element that was NOT present in the collection from the start to the end of a full iteration. So if an element was removed before the start of an iteration, and is never added back to the collection for all the time an iteration lasts, SCAN ensures that this element will never be returned.
* However because SCAN has very little state associated (just the cursor) it has the following drawbacks:
* A given element may be returned multiple times. It is up to the application to handle the case of duplicated elements, for example only using the returned elements in order to perform operations that are safe when re-applied multiple times.
* Elements that were not constantly present in the collection during a full iteration, may be returned or not: it is undefined.
*
*
* Number of elements returned at every SCAN call
* SCAN family functions do not guarantee that the number of elements returned per call are in a given range. The commands are also allowed to return zero elements, and the client should not consider the iteration complete as long as the returned cursor is not zero.
* However the number of returned elements is reasonable, that is, in practical terms SCAN may return a maximum number of elements in the order of a few tens of elements when iterating a large collection, or may return all the elements of the collection in a single call when the iterated collection is small enough to be internally represented as an encoded data structure (this happens for small sets, hashes and sorted sets).
* However there is a way for the user to tune the order of magnitude of the number of returned elements per call using the COUNT option.
*
*
* The COUNT option
* While SCAN does not provide guarantees about the number of elements returned at every iteration, it is possible to empirically adjust the behavior of SCAN using the COUNT option. Basically with COUNT the user specified the amount of work that should be done at every call in order to retrieve elements from the collection. This is just an hint for the implementation, however generally speaking this is what you could expect most of the times from the implementation.
* The default COUNT value is 10.
* When iterating the key space, or a Set, Hash or Sorted Set that is big enough to be represented by an hash table, assuming no MATCH option is used, the server will usually return count or a bit more than count elements per call.
* When iterating Sets encoded as intsets (small sets composed of just integers), or Hashes and Sorted Sets encoded as ziplists (small hashes and sets composed of small individual values), usually all the elements are returned in the first SCAN call regardless of the COUNT value.
* Important: there is no need to use the same COUNT value for every iteration. The caller is free to change the count from one iteration to the other as required, as long as the cursor passed in the next call is the one obtained in the previous call to the command.
*
*
* The MATCH option
* It is possible to only iterate elements matching a given glob-style pattern, similarly to the behavior of the KEYS command that takes a pattern as only argument.
* To do so, just append the MATCH arguments at the end of the SCAN command (it works with all the SCAN family commands).
* This is an example of iteration using MATCH:
*
*
* redis 127.0.0.1:6379> sadd myset 1 2 3 foo foobar feelsgood
* (integer) 6
* redis 127.0.0.1:6379> sscan myset 0 match f*
* 1) "0"
* 2) 1) "foo"
* 2) "feelsgood"
* 3) "foobar"
* redis 127.0.0.1:6379>
*
*
* It is important to note that the MATCH filter is applied after elements are retrieved from the collection, just before returning data to the client. This means that if the pattern matches very little elements inside the collection, SCAN will likely return no elements in most iterations. An example is shown below:
*
* redis 127.0.0.1:6379> scan 0 MATCH *11*
* 1) "288"
* 2) 1) "key:911"
* redis 127.0.0.1:6379> scan 288 MATCH *11*
* 1) "224"
* 2) (empty list or set)
* redis 127.0.0.1:6379> scan 224 MATCH *11*
* 1) "80"
* 2) (empty list or set)
* redis 127.0.0.1:6379> scan 80 MATCH *11*
* 1) "176"
* 2) (empty list or set)
* redis 127.0.0.1:6379> scan 176 MATCH *11* COUNT 1000
* 1) "0"
* 2) 1) "key:611"
* 2) "key:711"
* 3) "key:118"
* 4) "key:117"
* 5) "key:311"
* 6) "key:112"
* 7) "key:111"
* 8) "key:110"
* 9) "key:113"
* 10) "key:211"
* 11) "key:411"
* 12) "key:115"
* 13) "key:116"
* 14) "key:114"
* 15) "key:119"
* 16) "key:811"
* 17) "key:511"
* 18) "key:11"
* redis 127.0.0.1:6379>
*
* As you can see most of the calls returned zero elements, but the last call where a COUNT of 1000 was used in order to force the command to do more scanning for that iteration.
*
*
* Multiple parallel iterations
* It is possible for an infinite number of clients to iterate the same collection at the same time, as the full state of the iterator is in the cursor, that is obtained and returned to the client at every call. Server side no state is taken at all.
*
*
* Terminating iterations in the middle
* Since there is no state server side, but the full state is captured by the cursor, the caller is free to terminate an iteration half-way without signaling this to the server in any way. An infinite number of iterations can be started and never terminated without any issue.
*
*
* Calling SCAN with a corrupted cursor
* Calling SCAN with a broken, negative, out of range, or otherwise invalid cursor, will result into undefined behavior but never into a crash. What will be undefined is that the guarantees about the returned elements can no longer be ensured by the SCAN implementation.
* The only valid cursors to use are:
* The cursor value of 0 when starting an iteration.
* The cursor returned by the previous call to SCAN in order to continue the iteration.
*
* Guarantee of termination
* The SCAN algorithm is guaranteed to terminate only if the size of the iterated collection remains bounded to a given maximum size, otherwise iterating a collection that always grows may result into SCAN to never terminate a full iteration.
* This is easy to see intuitively: if the collection grows there is more and more work to do in order to visit all the possible elements, and the ability to terminate the iteration depends on the number of calls to SCAN and its COUNT option value compared with the rate at which the collection grows.
*
*
* Additional examples
* Iteration of an Hash value.
*
* redis 127.0.0.1:6379> hmset hash name Jack age 33
* OK
* redis 127.0.0.1:6379> hscan hash 0
* 1) "0"
* 2) 1) "name"
* 2) "Jack"
* 3) "age"
* 4) "33"
*
*
* @param cursor
* @return
* SCAN, SSCAN, HSCAN and ZSCAN return a two elements multi-bulk reply, where the first element is a string representing an unsigned 64 bit number (the cursor), and the second element is a multi-bulk with an array of elements.
* SCAN array of elements is a list of keys.
* SSCAN array of elements is a list of Set members.
* HSCAN array of elements contain two elements, a field and a value, for every returned element of the Hash.
* ZSCAN array of elements contain two elements, a member and its associated score, for every returned element of the sorted set.
*/
ScanResult scan(String cursor);
ScanResult scan(String cursor, int count);
ScanResult scan(String cursor, String pattern);
ScanResult scan(String cursor, String pattern, int count);
// ~ ------------------------------------------------------------------------------------------------------ Strings
/**
* Available since 2.0.0
* Time complexity: O(1). The amortized time complexity is O(1) assuming the appended value is small
* and the already present value is of any size, since the dynamic string library used by Redis will double the
* free space available on every reallocation.
*
*
* If key already exists and is a string, this command appends the value at the end of the string.
* If key does not exist it is created and set as an empty string, so APPEND will be similar to SET in this special case.
*
* @param key
* @param value
* @return the length of the string after the append operation.
*/
Long append(String key, String value);
/**
* Available since 2.6.0
* Time complexity: O(N)
*
*
* Count the number of set bits (population counting) in a string.
* By default all the bytes contained in the string are examined. It is possible to specify the counting operation
* only in an interval passing the additional arguments start and end.
* Like for the GETRANGE command start and end can contain negative values in order to index bytes starting from the end of the string,
* where -1 is the last byte, -2 is the penultimate, and so forth.
* Non-existent keys are treated as empty strings, so the command will return zero.
*
* @param key
* @param start inclusive
* @param end inclusive
* @return The number of bits set to 1.
*/
Long bitcount(String key, long start, long end);
Long bitcount(String key);
/**
* Available since 2.6.0
* Time complexity: O(N)
*
*
* Perform a bitwise operation between multiple keys (containing string values) and store the result in the destination key.
* The BITOP command supports four bitwise operations: AND, OR, XOR and NOT, thus the valid forms to call the command are:
*
* BITOP AND destkey srckey1 srckey2 srckey3 ... srckeyN
* BITOP OR destkey srckey1 srckey2 srckey3 ... srckeyN
* BITOP XOR destkey srckey1 srckey2 srckey3 ... srckeyN
* BITOP NOT destkey srckey
*
* As you can see NOT is special as it only takes an input key, because it performs inversion of bits so it only makes sense as an unary operator.
* The result of the operation is always stored at destkey.
*
* Handling of strings with different lengths
* When an operation is performed between strings having different lengths, all the strings shorter than the longest string
* in the set are treated as if they were zero-padded up to the length of the Longest string.
* The same holds true for non-existent keys, that are considered as a stream of zero bytes up to the length of the longest string.
*
* @param destkey
* @param key
* @return The size of the string stored in the destination key, that is equal to the size of the longest input string.
*/
Long bitnot(String destkey, String key);
Long bitand(String destkey, String... keys);
Long bitor(String destkey, String... keys);
Long bitxor(String destkey, String... keys);
/**
* Available since 2.8.7
* Time complexity: O(N)
*
*
* Return the position of the first bit set to 1 or 0 in a string.
* The position is returned thinking at the string as an array of bits from left to right where the first byte most significant bit is at position 0, the second byte most significant big is at position 8 and so forth.
* The same bit position convention is followed by GETBIT and SETBIT.
* By default all the bytes contained in the string are examined. It is possible to look for bits only in a specified interval passing the additional arguments start and end (it is possible to just pass start, the operation will assume that the end if the last byte of the string. However there are semantical differences as explained later). The range is interpreted as a range of bytes and not a range of bits, so start=0 and end=2 means to look at the first three bytes.
* Note that bit positions are returned always as absolute values starting from bit zero even when start and end are used to specify a range.
* Like for the GETRANGE command start and end can contain negative values in order to index bytes starting from the end of the string, where -1 is the last byte, -2 is the penultimate, and so forth.
* Non-existent keys are treated as empty strings.
*
* @param key
* @param value
* @return Integer reply
* The command returns the position of the first bit set to 1 or 0 according to the request.
* If we look for set bits (the bit argument is 1) and the string is empty or composed of just zero bytes, -1 is returned.
* If we look for clear bits (the bit argument is 0) and the string only contains bit set to 1, the function returns the first bit not part of the string on the right. So if the string is tree bytes set to the value 0xff the command BITPOS key 0 will return 24, since up to bit 23 all the bits are 1.
* Basically the function consider the right of the string as padded with zeros if you look for clear bits and specify no range or the start argument only.
* However this behavior changes if you are looking for clear bits and specify a range with both start and end. If no clear bit is found in the specified range, the function returns -1 as the user specified a clear range and there are no 0 bits in that range.
*/
Long bitpos(String key, boolean value);
Long bitpos(String key, boolean value, long start);
Long bitpos(String key, boolean value, long start, long end);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Decrements the number stored at key by one. If the key does not exist, it is set to 0 before performing the operation.
* An error is returned if the key contains a value of the wrong type or contains a string that can not be represented as integer.
* This operation is limited to 64 bit signed integers.
* See INCR for extra information on increment/decrement operations.
*
* @param key
* @return the value of key after the decrement
*/
Long decr(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Decrements the number stored at key by decrement. If the key does not exist, it is set to 0 before performing the operation.
* An error is returned if the key contains a value of the wrong type or contains a string that can not be represented as integer.
* This operation is limited to 64 bit signed integers.
* See INCR for extra information on increment/decrement operations.
*
* @param key
* @param decrement
* @return the value of key after the decrement
*/
Long decrby(String key, long decrement);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Get the value of key. If the key does not exist the special value null is returned.
* An error is returned if the value stored at key is not a string, because GET only handles string values.
*
* @param key
* @return the value of key, or null when key does not exist.
*/
String get(String key);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* Returns the bit value at offset in the string value stored at key.
* When offset is beyond the string length, the string is assumed to be a contiguous space with 0 bits.
* When key does not exist it is assumed to be an empty string, so offset is always out of range and the value is also
* assumed to be a contiguous space with 0 bits.
*
* @param key
* @param offset
* @return the bit value stored at offset.
*/
Boolean getbit(String key, long offset);
/**
* Available since 2.4.0
* Time complexity: O(N) where N is the length of the returned string.
* The complexity is ultimately determined by the returned length,
* but because creating a substring from an existing string is very cheap, it can be considered O(1) for small strings.
*
*
* Warning: this command was renamed to GETRANGE, it is called SUBSTR in Redis versions <= 2.0.
* Returns the substring of the string value stored at key, determined by the offsets start and end (both are inclusive).
* Negative offsets can be used in order to provide an offset starting from the end of the string. So -1 means the last character,
* -2 the penultimate and so forth.
* The function handles out of range requests by limiting the resulting range to the actual length of the string.
*
* @param key
* @param start inclusive
* @param end inclusive
* @return a substring of the string stored at a key
*/
String getrange(String key, long start, long end);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Atomically sets key to value and returns the old value stored at key.
* Returns an error when key exists but does not hold a string value.
*
* @param key
* @param value
* @return the old value stored at key, or null when key did not exist.
*/
String getset(String key, String value);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Increments the number stored at key by one. If the key does not exist, it is set to 0 before performing the operation.
* An error is returned if the key contains a value of the wrong type or contains a string that can not be represented as integer.
* This operation is limited to 64 bit signed integers.
*
* Note: this is a string operation because Redis does not have a dedicated integer type.
* The string stored at the key is interpreted as a base-10 64 bit signed integer to execute the operation.
* Redis stores integers in their integer representation, so for string values that actually hold an integer,
* there is no overhead for storing the string representation of the integer.
*
* @param key
* @return the value of key after the increment
*/
Long incr(String key);
/**
* available since 1.0.0
* Time complexity: O(1)
*
*
* Increments the number stored at key by increment. If the key does not exist, it is set to 0 before performing the operation.
* An error is returned if the key contains a value of the wrong type or contains a string that can not be represented as integer.
* This operation is limited to 64 bit signed integers.
* See INCR for extra information on increment/decrement operations.
*
* @param key
* @param increment
* @return the value of key after the increment
*/
Long incrby(String key, long increment);
/**
* Available since 2.6.0
* Time complexity: O(1)
*
*
* Increment the string representing a floating point number stored at key by the specified increment.
* If the key does not exist, it is set to 0 before performing the operation.
* An error is returned if one of the following conditions occur:
* - The key contains a value of the wrong type (not a string).
* - The current key content or the specified increment are not parsable as a double precision floating point number.
*
* If the command is successful the new incremented value is stored as the new value of the key (replacing the old one),
* and returned to the caller as a string.
* Both the value already contained in the string key and the increment argument can be optionally provided in exponential notation,
* however the value computed after the increment is stored consistently in the same format, that is,
* an integer number followed (if needed) by a dot, and a variable number of digits representing the decimal part of the number.
* Trailing zeroes are always removed.
* The precision of the output is fixed at 17 digits after the decimal point regardless of the actual internal precision of the computation.
*
* @param key
* @param increment
* @return the value of key after the increment.
*/
Double incrbyfloat(String key, double increment);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the number of keys to retrieve.
*
*
* Returns the values of all specified keys.
* For every key that does not hold a string value or does not exist,
* the special value null is returned. Because of this, the operation never fails.
*
* @param keys
* @return the values of all the given keys
*/
List mget(String... keys);
/**
* Available since 1.0.1
* Time complexity: O(N) where N is the number of keys to set.
*
*
* Sets the given keys to their respective values.
* MSET replaces existing values with new values, just as regular SET.
* See MSETNX if you don't want to overwrite existing values.
* MSET is atomic, so all given keys are set at once.
* It is not possible for clients to see that some of the keys were updated while others are unchanged.
*
* @param keysvalues
* @return always OK.
*/
String mset(String... keysvalues);
/**
* Available since 1.0.1.
* Time complexity: O(N) where N is the number of keys to set.
*
*
* Sets the given keys to their respective values.
* MSETNX will not perform any operation at all even if just a single key already exists.
* Because of this semantic MSETNX can be used in order to set different keys representing different fields of an unique logic object
* in a way that ensures that either all the fields or none at all are set.
* MSETNX is atomic, so all given keys are set at once.
* It is not possible for clients to see that some of the keys were updated while others are unchanged.
*
* @param keysvalues
* @return 1 if the all the keys were set.
* 0 if no key was set (at least one key already existed).
*/
Long msetnx(String... keysvalues);
/**
* Available since 2.6.0
* Time complexity: O(1)
*
*
* PSETEX works exactly like SETEX with the sole difference that the expire time is specified in milliseconds instead of seconds.
*
* @param key
* @param milliseconds
* @param value
* @return OK
*/
String psetex(String key, int milliseconds, String value);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Set key to hold the string value.
* If key already holds a value, it is overwritten, regardless of its type.
* Any previous time to live associated with the key is discarded on successful SET operation.
*
* @param key
* @param value
* @return OK if SET was executed correctly.
*/
String set(String key, String value);
/**
* Available since 2.6.12
*
*
* Options
* Starting with Redis 2.6.12 SET supports a set of options that modify its behavior:
* SET key value [EX seconds] [PX milliseconds] [NX|XX]
* EX seconds -- Set the specified expire time, in seconds.
* PX milliseconds -- Set the specified expire time, in milliseconds.
* NX -- Only set the key if it does not already exist.
* XX -- Only set the key if it already exist.
* Note: Since the SET command options can replace SETNX, SETEX, PSETEX, it is possible that in future versions of
* Redis these three commands will be deprecated and finally removed.
*
* @param key
* @param value
* @return OK if SET was executed correctly.
* Null multi-bulk reply: a Null Bulk Reply is returned if the SET operation was not performed becase the
* user specified the NX or XX option but the condition was not met.
*/
String setxx(String key, String value);
String setnxex(String key, String value, int seconds);
String setnxpx(String key, String value, int milliseconds);
String setxxex(String key, String value, int seconds);
String setxxpx(String key, String value, int milliseconds);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* Sets or clears the bit at offset in the string value stored at key.
* The bit is either set or cleared depending on value, which can be either 0 or 1. When key does not exist,
* a new string value is created.
* The string is grown to make sure it can hold a bit at offset.
* The offset argument is required to be greater than or equal to 0, and smaller than 232 (this limits bitmaps to 512MB).
* When the string at key is grown, added bits are set to 0.
*
* Warning:
* When setting the last possible bit (offset equal to 232 -1) and the string value stored at key does not yet hold a string value,
* or holds a small string value, Redis needs to allocate all intermediate memory which can block the server for some time.
* On a 2010 MacBook Pro, setting bit number 232 -1 (512MB allocation) takes ~300ms, setting bit number 230 -1 (128MB allocation) takes ~80ms,
* setting bit number 228 -1 (32MB allocation) takes ~30ms and setting bit number 226 -1 (8MB allocation) takes ~8ms.
* Note that once this first allocation is done, subsequent calls to SETBIT for the same key will not have the allocation overhead.
*
* @param key
* @param offset
* @param value true means bit is set 1, otherwise 0
* @return the original bit value stored at offset
*/
Boolean setbit(String key, long offset, boolean value);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Set key to hold the string value and set key to timeout after a given number of seconds.
* This command is equivalent to executing the following commands:
*
* SET mykey value
* EXPIRE mykey seconds
*
* SETEX is atomic, and can be reproduced by using the previous two commands inside an MULTI / EXEC block.
* It is provided as a faster alternative to the given sequence of operations, because this operation is very common when Redis is used as a cache.
* An error is returned when seconds is invalid.
*
* @param key
* @param seconds
* @param value
* @return Status code reply, e.g. OK
*/
String setex(String key, int seconds, String value);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Set key to hold string value if key does not exist. In that case, it is equal to SET.
* When key already holds a value, no operation is performed. SETNX is short for "SET if N ot e X ists".
*
* @param key
* @param value
* @return 1 if the key was set
* 0 if the key was not set
*/
Long setnx(String key, String value);
/**
* Available since 2.2.0
* Time complexity: O(1)
* not counting the time taken to copy the new string in place.
* Usually, this string is very small so the amortized complexity is O(1). Otherwise, complexity is O(M) with M being the length of the value argument.
*
*
* Overwrites part of the string stored at key, starting at the specified offset, for the entire length of value.
* If the offset is larger than the current length of the string at key, the string is padded with zero-bytes to make offset fit.
* Non-existing keys are considered as empty strings, so this command will make sure it holds a string large enough to be able to set value at offset.
* Note that the maximum offset that you can set is 229 -1 (536870911), as Redis Strings are limited to 512 megabytes.
* If you need to grow beyond this size, you can use multiple keys.
*
* Warning: When setting the last possible byte and the string value stored at key does not yet hold a string value,
* or holds a small string value, Redis needs to allocate all intermediate memory which can block the server for some time.
* On a 2010 MacBook Pro, setting byte number 536870911 (512MB allocation) takes ~300ms,
* setting byte number 134217728 (128MB allocation) takes ~80ms, setting bit number 33554432 (32MB allocation) takes ~30ms
* and setting bit number 8388608 (8MB allocation) takes ~8ms. Note that once this first allocation is done,
* subsequent calls to SETRANGE for the same key will not have the allocation overhead.
*
* @param key
* @param offset
* @param value
* @return Integer reply: the length of the string after it was modified by the command.
*/
Long setrange(String key, long offset, String value);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* Returns the length of the string value stored at key. An error is returned when key holds a non-string value.
*
* @param key
* @return the length of the string at key, or 0 when key does not exist.
*/
Long strlen(String key);
// ~ ------------------------------------------------------------------------------------------------------- Hashes
/**
* Available since 2.0.0
* Time complexity: O(N)
* where N is the number of fields to be removed.
*
*
* Removes the specified fields from the hash stored at key.
* Specified fields that do not exist within this hash are ignored.
* If key does not exist, it is treated as an empty hash and this command returns 0.
*
* History
* >= 2.4: Accepts multiple field arguments. Redis versions older than 2.4 can only remove a field per call.
* To remove multiple fields from a hash in an atomic fashion in earlier versions, use a MULTI / EXEC block.
*
* @param key
* @param fields
* @return the number of fields that were removed from the hash, not including specified but non existing fields.
*/
Long hdel(String key, String... fields);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Returns if field is an existing field in the hash stored at key.
*
* @param key
* @param field
* @return true if the hash contains field.
* false if the hash does not contain field, or key does not exist.
*/
Boolean hexists(String key, String field);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Returns the value associated with field in the hash stored at key.
*
* @param key
* @param field
* @return the value associated with field, or null when field is not present in the hash or key does not exist.
*/
String hget(String key, String field);
/**
* Available since 2.0.0
* Time complexity: O(N)
* where N is the size of the hash.
*
*
* Returns all fields and values of the hash stored at key.
* In the returned value, every field name is followed by its value, so the length of the reply is twice the size of the hash.
*
* @param key
* @return list of fields and their values stored in the hash, or an empty list when key does not exist.
*/
Map hgetall(String key);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Increments the number stored at field in the hash stored at key by increment.
* If key does not exist, a new key holding a hash is created.
* If field does not exist the value is set to 0 before the operation is performed.
* The range of values supported by HINCRBY is limited to 64 bit signed integers.
*
* @param key
* @param field
* @param increment
* @return the value at field after the increment operation.
*/
Long hincrby(String key, String field, long increment);
/**
* Available since 2.6.0
* Time complexity: O(1)
*
*
* Increment the specified field of an hash stored at key, and representing a floating point number, by the specified increment.
* If the field does not exist, it is set to 0 before performing the operation.
* An error is returned if one of the following conditions occur:
* - The field contains a value of the wrong type (not a string).
* - The current field content or the specified increment are not parsable as a double precision floating point number.
* The exact behavior of this command is identical to the one of the INCRBYFLOAT command,
* please refer to the documentation of INCRBYFLOAT for further information.
*
* @param key
* @param field
* @param increment
* @return the value of field after the increment.
*/
Double hincrbyfloat(String key, String field, double increment);
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the size of the hash.
*
*
* Returns all field names in the hash stored at key.
*
* @param key
* @return list of fields in the hash, or an empty list when key does not exist.
*/
Set hkeys(String key);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Returns the number of fields contained in the hash stored at key.
*
* @param key
* @return number of fields in the hash, or 0 when key does not exist.
*/
Long hlen(String key);
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the number of fields being requested.
*
*
* Returns the values associated with the specified fields in the hash stored at key.
* For every field that does not exist in the hash, a null value is returned.
* Because a non-existing keys are treated as empty hashes, running HMGET against a non-existing key will return a list of nil values.
*
* @param key
* @param fields
* @return list of values associated with the given fields, in the same order as they are requested.
*/
List hmget(String key, String... fields);
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the number of fields being set.
*
*
* Sets the specified fields to their respective values in the hash stored at key.
* This command overwrites any existing fields in the hash. If key does not exist, a new key holding a hash is created.
*
* @param key
* @param fieldvalues
* @return Status code reply, e.g. OK
*/
String hmset(String key, Map fieldvalues);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Sets field in the hash stored at key to value.
* If key does not exist, a new key holding a hash is created.
* If field already exists in the hash, it is overwritten.
*
* @param key
* @param field
* @param value
* @return 1 if field is a new field in the hash and value was set.
* 0 if field already exists in the hash and the value was updated.
*/
Long hset(String key, String field, String value);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* Sets field in the hash stored at key to value, only if field does not yet exist.
* If key does not exist, a new key holding a hash is created.
* If field already exists, this operation has no effect.
*
* @param key
* @param field
* @param value
* @return 1 if field is a new field in the hash and value was set.
* 0 if field already exists in the hash and no operation was performed.
*/
Long hsetnx(String key, String field, String value);
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the size of the hash.
*
*
* Returns all values in the hash stored at key.
*
* @param key
* @return list of values in the hash, or an empty list when key does not exist.
*/
List hvals(String key);
/**
* @see #scan(String)
*/
ScanResult> hscan(String key, String cursor);
ScanResult> hscan(String key, String cursor, int count);
ScanResult> hscan(String key, String cursor, String pattern);
ScanResult> hscan(String key, String cursor, String pattern, int count);
// ~ ------------------------------------------------------------------------------------------------------- Lists
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* BLPOP is a blocking list pop primitive.
* It is the blocking version of LPOP because it blocks the connection when
* there are no elements to pop from any of the given lists. An element is popped from the head of the first list that is non-empty,
* with the given keys being checked in the order that they are given.
*
*
* Non-blocking behavior
* When BLPOP is called, if at least one of the specified keys contains a non-empty list,
* an element is popped from the head of the list and returned to the caller together with the key it was popped from.
* Keys are checked in the order that they are given. Let's say that the key list1 doesn't exist and list2 and list3 hold non-empty lists.
* Consider the following command:
*
* BLPOP list1 list2 list3 0
*
* BLPOP guarantees to return an element from the list stored at list2 (since it is the first non empty list when
* checking list1, list2 and list3 in that order).
*
*
* Blocking behavior
* If none of the specified keys exist, BLPOP blocks the connection until another client performs an LPUSH or RPUSH
* operation against one of the keys.
* Once new data is present on one of the lists, the client returns with the name of the key unblocking it and the popped value.
* When BLPOP causes a client to block and a non-zero timeout is specified, the client will unblock returning a nil
* multi-bulk value when the specified timeout has expired without a push operation against at least one of the specified keys.
* The timeout argument is interpreted as an integer value specifying the maximum number of seconds to block.
* A timeout of zero can be used to block infinitely.
*
*
* What key is served first? What client? What element? Priority ordering details.
* If the client tries to blocks for multiple keys, but at least one key contains elements,
* the returned key / element pair is the first key from left to right that has one or more elements.
* In this case the client is not blocked. So for instance BLPOP key1 key2 key3 key4 0, assuming that both key2 and key4 are non-empty,
* will always return an element from key2.
* If multiple clients are blocked for the same key, the first client to be served is the one that was waiting for
* more time (the first that blocked for the key).
* Once a client is unblocked it does not retain any priority,
* when it blocks again with the next call to BLPOP it will be served accordingly to the number of clients already
* blocked for the same key, that will all be served before it (from the first to the last that blocked).
* When a client is blocking for multiple keys at the same time, and elements are available at the same time in
* multiple keys (because of a transaction or a Lua script added elements to multiple lists),
* the client will be unblocked using the first key that received a push operation (assuming it has enough elements to
* serve our client, as there may be other clients as well waiting for this key).
* Basically after the execution of every
* command Redis will run a list of all the keys that received data AND that have at least a client blocked.
* The list is ordered by new element arrival time, from the first key that received data to the last.
* For every key processed, Redis will serve all the clients waiting for that key in a FIFO fashion, as long as there
* are elements in this key. When the key is empty or there are no longer clients waiting for this key,
* the next key that received new data in the previous command / transaction / script is processed, and so forth.
*
*
* Behavior of BLPOP when multiple elements are pushed inside a list.
* There are times when a list can receive multiple elements in the context of the same conceptual command:
* Variadic push operations such as LPUSH mylist a b c.
* After an EXEC of a MULTI block with multiple push operations against the same list.
* Executing a Lua Script with Redis 2.6 or newer.
* When multiple elements are pushed inside a list where there are clients blocking, the behavior is different for Redis 2.4 and Redis 2.6 or newer.
* For Redis 2.6 what happens is that the command performing multiple pushes is executed,
* and only after the execution of the command the blocked clients are served. Consider this sequence of commands.
*
* Client A: BLPOP foo 0
* Client B: LPUSH foo a b c
*
* If the above condition happens using a Redis 2.6 server or greater, Client A will be served with the c element,
* because after the LPUSH command the list contains c,b,a, so taking an element from the left means to return c.
* Instead Redis 2.4 works in a different way: clients are served in the context of the push operation, so as long as
* LPUSH foo a b c starts pushing the first element to the list, it will be delivered to the Client A, that will receive a (the first element pushed).
* The behavior of Redis 2.4 creates a lot of problems when replicating or persisting data into the AOF file,
* so the much more generic and semantically simpler behaviour was introduced into Redis 2.6 to prevent problems.
* Note that for the same reason a Lua script or a MULTI/EXEC block may push elements into a list and afterward delete the list.
* In this case the blocked clients will not be served at all and will continue to be blocked as long as no data is
* present on the list after the execution of a single command, transaction, or script.
*
*
* BLPOP inside a MULTI / EXEC transaction
* BLPOP can be used with pipelining (sending multiple commands and reading the replies in batch),
* however this setup makes sense almost solely when it is the last command of the pipeline.
* Using BLPOP inside a MULTI / EXEC block does not make a lot of sense as it would require blocking the entire
* server in order to execute the block atomically, which in turn does not allow other clients to perform a push operation.
* For this reason the behavior of BLPOP inside MULTI / EXEC when the list is empty is to return a nil multi-bulk reply,
* which is the same thing that happens when the timeout is reached.
* If you like science fiction, think of time flowing at infinite speed inside a MULTI / EXEC block...
*
* @param key
* @return A null when no element could be popped and the timeout expired.
* A popped element.
*/
String blpop(String key);
String blpop(String key, int timeout);
/**
* @see #blpop(String)
* @param timeout max seconds to block
* @param keys
* @return A empty map(nil multi-bulk) when no element could be popped and the timeout expired.
* A map (two-element multi-bulk) with the key (first element) being the name of the key where an element was popped
* and the value (second element) being the value of the popped element.
*/
Map blpop(int timeout, String... keys);
Map blpop(String... keys);
/**
* Available since 2.0.0
* Time complexity: O(1)
*
*
* BRPOP is a blocking list pop primitive.
* It is the blocking version of RPOP because it blocks the connection when there are no elements to pop from any of the given lists.
* An element is popped from the tail of the first list that is non-empty,
* with the given keys being checked in the order that they are given.
* See the BLPOP documentation for the exact semantics, since BRPOP is identical to BLPOP with the only difference being that
* it pops elements from the tail of a list instead of popping from the head.
*
* @see #blpop(String)
* @param key
* @return A null when no element could be popped and the timeout expired.
* A popped element.
*/
String brpop(String key);
String brpop(String key, int timeout);
/**
* @see #brpop(String)
* @param timeout
* @param keys
* @return A empty map(nil multi-bulk) when no element could be popped and the timeout expired.
* A map (two-element multi-bulk) with the key (first element) being the name of the key where an element was popped
* and the value (second element) being the value of the popped element.
*/
Map brpop(int timeout, String... keys);
Map brpop(String... keys);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* BRPOPLPUSH is the blocking variant of RPOPLPUSH. When source contains elements, this command behaves exactly like RPOPLPUSH.
* When source is empty, Redis will block the connection until another client pushes to it or until timeout is reached.
* A timeout of zero can be used to block infinitely.
* See RPOPLPUSH for more information.
*
* @param source
* @param destination
* @param timeout
* @return the element being popped from source and pushed to destination.
* If timeout is reached, a null reply is returned.
*/
String brpoplpush(String source, String destination, int timeout);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the number of elements to traverse to get to the element at index.
* This makes asking for the first or the last element of the list O(1).
*
*
* Returns the element at index index in the list stored at key.
* The index is zero-based, so 0 means the first element, 1 the second element and so on.
* Negative indices can be used to designate elements starting at the tail of the list.
* Here, -1 means the last element, -2 means the penultimate and so forth.
* When the value at key is not a list, an error is returned.
*
* @param key
* @param index zero-based
* @return an element from a list by its index
*/
String lindex(String key, long index);
/**
* Available since 2.2.0
* Time complexity: O(N) where N is the number of elements to traverse before seeing the value pivot.
* This means that inserting somewhere on the left end on the list (head) can be considered O(1)
* and inserting somewhere on the right end (tail) is O(N).
*
*
* Inserts value in the list stored at key either before or after the reference value pivot.
* When key does not exist, it is considered an empty list and no operation is performed.
* When pivot can not be found and no operation is performed.
* An error is returned when key exists but does not hold a list value.
*
* @param key
* @param pivot
* @param value
* @return the length of the list after the insert operation, or -1 when the value pivot was not found.
*/
Long linsertbefore(String key, String pivot, String value);
Long linsertafter(String key, String pivot, String value);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Returns the length of the list stored at key. If key does not exist, it is interpreted as an empty list and 0 is returned.
* An error is returned when the value stored at key is not a list.
*
* @param key
* @return the length of the list at key.
*/
Long llen(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Removes and returns the first element of the list stored at key.
*
* @param key
* @return the value of the first element, or null when key does not exist.
*/
String lpop(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Insert all the specified values at the head of the list stored at key.
* If key does not exist, it is created as empty list before performing the push operations.
* When key holds a value that is not a list, an error is returned.
* It is possible to push multiple elements using a single command call just specifying multiple arguments at the end of the command.
* Elements are inserted one after the other to the head of the list, from the leftmost element to the rightmost element.
* So for instance the command LPUSH mylist a b c will result into a list containing c as first element, b as second element and a as third element.
*
*
* History
* >= 2.4: Accepts multiple value arguments.
* In Redis versions older than 2.4 it was possible to push a single value per command.
*
* @param key
* @param values
* @return the length of the list after the push operations.
*/
Long lpush(String key, String... values);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* Inserts value at the head of the list stored at key, only if key already exists and holds a list.
* In contrary to LPUSH, no operation will be performed when key does not yet exist.
*
* @param key
* @param value
* @return the length of the list after the push operation.
*/
Long lpushx(String key, String value);
/**
* Available since 1.0.0
* Time complexity: O(S+N) where S is the start offset and N is the number of elements in the specified range.
*
*
* Returns the specified elements of the list stored at key.
* The offsets start and stop are zero-based indexes,
* with 0 being the first element of the list (the head of the list), 1 being the next element and so on.
* These offsets can also be negative numbers indicating offsets starting at the end of the list.
* For example, -1 is the last element of the list, -2 the penultimate, and so on.
*
*
* Consistency with range functions in various programming languages
* Note that if you have a list of numbers from 0 to 100, LRANGE list 0 10 will return 11 elements, that is, the rightmost item is included.
* This may or may not be consistent with behavior of range-related functions in your programming language of choice
* (think Ruby's Range.new, Array#slice or Python's range() function).
*
*
* Out-of-range indexes
* Out of range indexes will not produce an error.
* If start is larger than the end of the list, an empty list is returned.
* If stop is larger than the actual end of the list, Redis will treat it like the last element of the list.
*
* @param key
* @param start
* @param stop
* @return list of elements in the specified range.
*/
List lrange(String key, long start, long stop);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the length of the list.
*
*
* Removes the first count occurrences of elements equal to value from the list stored at key.
* The count argument influences the operation in the following ways:
* count > 0: Remove elements equal to value moving from head to tail.
* count < 0: Remove elements equal to value moving from tail to head.
* count = 0: Remove all elements equal to value.
* For example, LREM list -2 "hello" will remove the last two occurrences of "hello" in the list stored at list.
* Note that non-existing keys are treated like empty lists, so when key does not exist, the command will always return 0.
*
* @param key
* @param count
* @param value
* @return the number of removed elements.
*/
Long lrem(String key, long count, String value);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the length of the list. Setting either the first or the last element of the list is O(1).
*
*
* Sets the list element at index to value.
* For more information on the index argument, see LINDEX.
* An error is returned for out of range indexes.
*
* @param key
* @param index
* @param value
* @return Status code reply, e.g. OK
*/
String lset(String key, long index, String value);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the number of elements to be removed by the operation.
*
*
* Trim an existing list so that it will contain only the specified range of elements specified.
* Both start and stop are zero-based indexes, where 0 is the first element of the list (the head), 1 the next element and so on.
* For example: LTRIM foobar 0 2 will modify the list stored at foobar so that only the first three elements of the list will remain.
* start and end can also be negative numbers indicating offsets from the end of the list, where -1 is the last element of the list,
* -2 the penultimate element and so on.
* Out of range indexes will not produce an error: if start is larger than the end of the list, or start > end,
* the result will be an empty list (which causes key to be removed).
* If end is larger than the end of the list, Redis will treat it like the last element of the list.
* A common use of LTRIM is together with LPUSH / RPUSH. For example:
*
* LPUSH mylist someelement
* LTRIM mylist 0 99
*
* This pair of commands will push a new element on the list, while making sure that the list will not grow larger than 100 elements.
* This is very useful when using Redis to store logs for example.
* It is important to note that when used in this way LTRIM is an O(1) operation because in the average case just one element
* is removed from the tail of the list.
*
* @param key
* @param start inclusive
* @param stop inclusive
* @return Status code reply, e.g. OK
*/
String ltrim(String key, long start, long stop);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Removes and returns the last element of the list stored at key.
*
* @param key
* @return the value of the last element, or null when key does not exist.
*/
String rpop(String key);
/**
* Available since 1.2.0
* Time complexity: O(1)
*
*
* Atomically returns and removes the last element (tail) of the list stored at source,
* and pushes the element at the first element (head) of the list stored at destination.
* For example: consider source holding the list a,b,c, and destination holding the list x,y,z.
* Executing RPOPLPUSH results in source holding a,b and destination holding c,x,y,z.
* If source does not exist, the value null is returned and no operation is performed.
* If source and destination are the same, the operation is equivalent to removing the last element from the list and
* pushing it as first element of the list, so it can be considered as a list rotation command.
*
* @param source
* @param destination
* @return the element being popped and pushed.
*/
String rpoplpush(String source, String destination);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Insert all the specified values at the tail of the list stored at key.
* If key does not exist, it is created as empty list before performing the push operation.
* When key holds a value that is not a list, an error is returned.
* It is possible to push multiple elements using a single command call just specifying multiple arguments at the end of the command.
* Elements are inserted one after the other to the tail of the list, from the leftmost element to the rightmost element.
* So for instance the command RPUSH mylist a b c will result into a list containing a as first element, b as second element and c as third element.
*
*
* History
* >= 2.4: Accepts multiple value arguments.
* In Redis versions older than 2.4 it was possible to push a single value per command.
*
* @param key
* @param values
* @return the length of the list after the push operation.
*/
Long rpush(String key, String... values);
/**
* Available since 2.2.0
* Time complexity: O(1)
*
*
* Inserts value at the tail of the list stored at key, only if key already exists and holds a list.
* In contrary to RPUSH, no operation will be performed when key does not yet exist.
*
* @param key
* @param value
* @return the length of the list after the push operation.
*/
Long rpushx(String key, String value);
// ~ ------------------------------------------------------------------------------------------------------- Sets
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the number of members to be added.
*
*
* Add the specified members to the set stored at key.
* Specified members that are already a member of this set are ignored.
* If key does not exist, a new set is created before adding the specified members.
* An error is returned when the value stored at key is not a set.
*
*
* History
* >= 2.4: Accepts multiple member arguments.
* Redis versions before 2.4 are only able to add a single member per call.
*
* @param key
* @param members
* @return the number of elements that were added to the set, not including all the elements already present into the set.
*/
Long sadd(String key, String... members);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Returns the set cardinality (number of elements) of the set stored at key.
*
* @param key
* @return the cardinality (number of elements) of the set, or 0 if key does not exist.
*/
Long scard(String key);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the total number of elements in all given sets.
*
*
* Returns the members of the set resulting from the difference between the first set and all the successive sets.
* For example:
*
* key1 = {a,b,c,d}
* key2 = {c}
* key3 = {a,c,e}
* SDIFF key1 key2 key3 = {b,d}
*
* Keys that do not exist are considered to be empty sets.
*
* @param keys
* @return list with members of the resulting set.
*/
Set sdiff(String... keys);
/**
* Available since 1.0.0.
* Time complexity: O(N) where N is the total number of elements in all given sets.
*
*
* This command is equal to SDIFF, but instead of returning the resulting set, it is stored in destination.
* If destination already exists, it is overwritten.
*
* @param destination
* @param keys
* @return the number of elements in the resulting set.
*/
Long sdiffstore(String destination, String... keys);
/**
* Available since 1.0.0
* Time complexity: O(N*M) worst case where N is the cardinality of the smallest set and M is the number of sets.
*
*
* Returns the members of the set resulting from the intersection of all the given sets.
* For example:
*
* key1 = {a,b,c,d}
* key2 = {c}
* key3 = {a,c,e}
* SINTER key1 key2 key3 = {c}
*
* Keys that do not exist are considered to be empty sets.
* With one of the keys being an empty set, the resulting set is also empty (since set intersection with an empty set always results in an empty set).
*
* @param keys
* @return list with members of the resulting set.
*/
Set sinter(String... keys);
/**
* Available since 1.0.0
* Time complexity: O(N*M) worst case where N is the cardinality of the smallest set and M is the number of sets.
*
*
* This command is equal to SINTER, but instead of returning the resulting set, it is stored in destination.
* If destination already exists, it is overwritten.
*
* @param destination
* @param keys
* @return the number of elements in the resulting set.
*/
Long sinterstore(String destination, String... keys);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Returns if member is a member of the set stored at key.
*
* @param key
* @param member
* @return true if the element is a member of the set.
* false if the element is not a member of the set, or if key does not exist.
*/
Boolean sismember(String key, String member);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the set cardinality.
*
*
* Returns all the members of the set value stored at key.
* This has the same effect as running SINTER with one argument key.
*
* @param key
* @return all the members in a set
*/
Set smembers(String key);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Move member from the set at source to the set at destination. This operation is atomic.
* In every given moment the element will appear to be a member of source or destination for other clients.
* If the source set does not exist or does not contain the specified element, no operation is performed and 0 is returned.
* Otherwise, the element is removed from the source set and added to the destination set.
* When the specified element already exists in the destination set, it is only removed from the source set.
* An error is returned if source or destination does not hold a set value.
*
* @param source
* @param destination
* @param member
* @return 1 if the element is moved.
* 0 if the element is not a member of source and no operation was performed.
*/
Long smove(String source, String destination, String member);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
*
* Removes and returns a random element from the set value stored at key.
* This operation is similar to SRANDMEMBER, that returns a random element from a set but does not remove it.
*
* @param key
* @return the removed element, or null when key does not exist.
*/
String spop(String key);
/**
* Available since 2.6.0
* Time complexity: O(N) where N is the absolute value of the passed count.
*
*
* When called with just the key argument, return a random element from the set value stored at key.
* Starting from Redis version 2.6, when called with the additional count argument,
* return an array of count distinct elements if count is positive.
* If called with a negative count the behavior changes and the command is allowed to return the same element multiple times.
* In this case the numer of returned elements is the absolute value of the specified count.
* When called with just the key argument, the operation is similar to SPOP, however while SPOP also removes the randomly selected element from the set,
* SRANDMEMBER will just return a random element without altering the original set in any way.
* @param key
* @param count
* @return returns an set of elements, or an empty set when key does not exist.
*/
List srandmember(String key, int count);
/**
* Available since 1.0.0
* Time complexity: O(1)
*
* @see #srandmember(String, int)
* @param key
* @return returns the randomly selected element, or null when key does not exist
*/
String srandmember(String key);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the number of members to be removed.
*
*
* Remove the specified members from the set stored at key.
* Specified members that are not a member of this set are ignored.
* If key does not exist, it is treated as an empty set and this command returns 0.
* An error is returned when the value stored at key is not a set.
*
*
* History
* >= 2.4: Accepts multiple member arguments.
* Redis versions older than 2.4 can only remove a set member per call.
*
* @param key
* @param members
* @return the number of members that were removed from the set, not including non existing members.
*/
Long srem(String key, String... members);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the total number of elements in all given sets.
*
*
* Returns the members of the set resulting from the union of all the given sets.
* For example:
*
* key1 = {a,b,c,d}
* key2 = {c}
* key3 = {a,c,e}
* SUNION key1 key2 key3 = {a,b,c,d,e}
*
* Keys that do not exist are considered to be empty sets.
*
* @param keys
* @return list with members of the resulting set.
*/
Set sunion(String... keys);
/**
* Available since 1.0.0
* Time complexity: O(N) where N is the total number of elements in all given sets.
*
*
* This command is equal to SUNION, but instead of returning the resulting set, it is stored in destination.
* If destination already exists, it is overwritten.
*
* @param destination
* @param keys
* @return the number of elements in the resulting set
*/
Long sunionstore(String destination, String... keys);
/**
* @see #scan(String)
*/
ScanResult sscan(String key, String cursor);
ScanResult sscan(String key, String cursor, int count);
ScanResult sscan(String key, String cursor, String pattern);
ScanResult sscan(String key, String cursor, String pattern, int count);
// ~ -------------------------------------------------------------------------------------------------- Sorted Sets
/**
* Available since 1.2.0
* Time complexity: O(log(N)) where N is the number of elements in the sorted set.
*
*
* Adds all the specified members with the specified scores to the sorted set stored at key.
* It is possible to specify multiple score/member pairs.
* If a specified member is already a member of the sorted set,
* the score is updated and the element reinserted at the right position to ensure the correct ordering.
* If key does not exist, a new sorted set with the specified members as sole members is created, like if the sorted set was empty.
* If the key exists but does not hold a sorted set, an error is returned.
* The score values should be the string representation of a numeric value, and accepts double precision floating point numbers.
*
*
* History
* >= 2.4: Accepts multiple elements.
* In Redis versions older than 2.4 it was possible to add or update a single member per call.
*
* @param key
* @param score
* @param member
* @return The number of elements added to the sorted sets, not including elements already existing for which the score was updated.
*/
Long zadd(String key, double score, String member);
Long zadd(String key, Map scoremembers);
/**
* Available since 1.2.0
* Time complexity: O(1)
*
*
* Returns the sorted set cardinality (number of elements) of the sorted set stored at key.
*
* @param key
* @return the cardinality (number of elements) of the sorted set, or 0 if key does not exist.
*/
Long zcard(String key);
/**
* Available since 2.0.0
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M being the number of elements between min and max.
*
*
* Returns the number of elements in the sorted set at key with a score between min and max.
* The min and max arguments have the same semantic as described for ZRANGEBYSCORE.
*
* @param key
* @param min inclusive
* @param max inclusive
* @return the number of members in a sorted set with scores within the given values
*/
Long zcount(String key, double min, double max);
Long zcount(String key, String min, String max);
/**
* Available since 1.2.0
* Time complexity: O(log(N)) where N is the number of elements in the sorted set.
*
*
* Increments the score of member in the sorted set stored at key by increment.
* If member does not exist in the sorted set, it is added with increment as its score (as if its previous score was 0.0).
* If key does not exist, a new sorted set with the specified member as its sole member is created.
* An error is returned when key exists but does not hold a sorted set.
* The score value should be the string representation of a numeric value, and accepts double precision floating point numbers.
* It is possible to provide a negative value to decrement the score.
*
* @param key
* @param score
* @param member
* @return the new score of member (a double precision floating point number), represented as string.
*/
Double zincrby(String key, double score, String member);
/**
* Available since 2.0.0
* Time complexity: O(N*K)+O(M*log(M)) worst case with N being the smallest input sorted set,
* K being the number of input sorted sets and M being the number of elements in the resulting sorted set.
*
*
* Computes the intersection of sorted sets given by the specified keys, and stores the result in destination.
* By default, the resulting score of an element is the sum of its scores in the sorted sets where it exists.
* Because intersection requires an element to be a member of every given sorted set,
* this results in the score of every element in the resulting sorted set to be equal to the number of input sorted sets.
* For a description of the WEIGHTS and AGGREGATE options, see ZUNIONSTORE.
* If destination already exists, it is overwritten.
*
* @param destination
* @param keys
* @return the number of elements in the resulting sorted set at destination.
*/
Long zinterstore(String destination, String... keys);
Long zinterstoremax(String destination, String... keys);
Long zinterstoremin(String destination, String... keys);
Long zinterstore(String destination, Map weightkeys);
Long zinterstoremax(String destination, Map weightkeys);
Long zinterstoremin(String destination, Map weightkeys);
/**
* Available since 2.8.9
* Time complexity: O(log(N)) with N being the number of elements in the sorted set.
*
* When all the elements in a sorted set are inserted with the same score, in order to force lexicographical ordering,
* this command returns the number of elements in the sorted set at key with a value between min and max.
* The min and max arguments have the same meaning as described for ZRANGEBYLEX.
* Note: the command has a complexity of just O(log(N)) because it uses elements ranks (see ZRANK) to get an idea of the range. Because of this there is no need to do a work proportional to the size of the range.
*
* Return value
* Integer reply: the number of elements in the specified score range.
*
* @param key
* @param min
* @param max
* @return the number of elements in the specified score range.
*/
Long zlexcount(String key, String min, String max);
/**
* Available since 1.2.0
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements returned.
*
*
* Returns the specified range of elements in the sorted set stored at key.
* The elements are considered to be ordered from the lowest to the highest score.
* Lexicographical order is used for elements with equal score.
* See ZREVRANGE when you need the elements ordered from highest to lowest score (and descending lexicographical order for elements with equal score).
* Both start and stop are zero-based indexes, where 0 is the first element, 1 is the next element and so on.
* They can also be negative numbers indicating offsets from the end of the sorted set, with -1 being the last element of the sorted set,
* -2 the penultimate element and so on.
* Out of range indexes will not produce an error.
* If start is larger than the largest index in the sorted set, or start > stop, an empty list is returned.
* If stop is larger than the end of the sorted set Redis will treat it like it is the last element of the sorted set.
* It is possible to pass the WITHSCORES option in order to return the scores of the elements together with the elements.
* The returned list will contain value1,score1,...,valueN,scoreN instead of value1,...,valueN.
* Client libraries are free to return a more appropriate data type (suggestion: an array with (value, score) arrays/tuples).
*
* @param key
* @param start inclusive
* @param stop invlusive
* @return list of elements in the specified range (optionally with their scores).
*/
Set zrange(String key, long start, long stop);
Map zrangewithscores(String key, long start, long stop);
/**
* Available since 2.8.9
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements being returned.
* If M is constant (e.g. always asking for the first 10 elements with LIMIT), you can consider it O(log(N)).
*
*
* When all the elements in a sorted set are inserted with the same score, in order to force lexicographical ordering,
* this command returns all the elements in the sorted set at key with a value between min and max.
* If the elements in the sorted set have different scores, the returned elements are unspecified.
* The elements are considered to be ordered from lower to higher strings as compared byte-by-byte using the memcmp() C function.
* Longer strings are considered greater than shorter strings if the common part is identical.
* The optional LIMIT argument can be used to only get a range of the matching elements (similar to SELECT LIMIT offset, count in SQL).
* Keep in mind that if offset is large, the sorted set needs to be traversed for offset elements before getting to the elements to return,
* which can add up to O(N) time complexity.
*
*
* How to specify intervals
* Valid start and stop must start with ( or [, in order to specify if the range item is respectively exclusive or inclusive.
* The special values of + or - for start and stop have the special meaning or positively infinite and negatively infinite strings,
* so for instance the command ZRANGEBYLEX myzset - + is guaranteed to return all the elements in the sorted set, if all the elements have the same score.
*
*
* Details on strings comparison
* Strings are compared as binary array of bytes. Because of how the ASCII character set is specified,
* this means that usually this also have the effect of comparing normal ASCII characters in an obvious dictionary way.
* However this is not true if non plain ASCII strings are used (for example utf8 strings).
* However the user can apply a transformation to the encoded string so that the first part of the element inserted in the sorted set
* will compare as the user requires for the specific application. For example if I want to add strings that will be compared in a case-insensitive way,
* but I still want to retrieve the real case when querying, I can add strings in the following way:
*
*
* ZADD autocomplete 0 foo:Foo 0 bar:BAR 0 zap:zap
*
*
* Because of the first normalized part in every element (before the colon character),
* we are forcing a given comparison, however after the range is queries using ZRANGEBYLEX the application can display
* to the user the second part of the string, after the colon.
* The binary nature of the comparison allows to use sorted sets as a general purpose index,
* for example the first part of the element can be a 64 bit big endian number: since big endian numbers have the most significant bytes in the initial positions,
* the binary comparison will match the numerical comparison of the numbers. This can be used in order to implement range queries on 64 bit values.
* As in the example below, after the first 8 bytes we can store the value of the element we are actually indexing.
*
* @param key
* @param min
* @param max
* @return list of elements in the specified score range.
*/
Set zrangebylex(String key, String min, String max);
Set zrangebylex(String key, String min, String max, int offset, int count);
/**
* Available since 1.0.5
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements being returned.
* If M is constant (e.g. always asking for the first 10 elements with LIMIT), you can consider it O(log(N)).
*
*
* Returns all the elements in the sorted set at key with a score between min and max (including elements with score equal to min or max).
* The elements are considered to be ordered from low to high scores.
* The elements having the same score are returned in lexicographical order (this follows from a property of the sorted set
* implementation in Redis and does not involve further computation).
* The optional LIMIT argument can be used to only get a range of the matching elements (similar to SELECT LIMIT offset, count in SQL).
* Keep in mind that if offset is large, the sorted set needs to be traversed for offset elements before getting to the elements to return,
* which can add up to O(N) time complexity.
* The optional WITHSCORES argument makes the command return both the element and its score, instead of the element alone.
* This option is available since Redis 2.0.
*
* Exclusive intervals and infinity
* min and max can be -inf and +inf, so that you are not required to know the highest or lowest score in the sorted
* set to get all elements from or up to a certain score.
* By default, the interval specified by min and max is closed (inclusive).
* It is possible to specify an open interval
* (exclusive) by prefixing the score with the character (. For example:
*
* ZRANGEBYSCORE zset (1 5
*
* Will return all elements with 1 < score <= 5 while:
*
* ZRANGEBYSCORE zset (5 (10
*
* Will return all the elements with 5 < score < 10 (5 and 10 excluded).
*
* @param key
* @param min inclusive
* @param max inclusive
* @return list of elements in the specified score range (optionally with their scores).
*/
Set zrangebyscore(String key, double min, double max);
Set zrangebyscore(String key, String min, String max);
Set zrangebyscore(String key, double min, double max, int offset, int count);
Set zrangebyscore(String key, String min, String max, int offset, int count);
Map zrangebyscorewithscores(String key, double min, double max);
Map zrangebyscorewithscores(String key, String min, String max);
Map zrangebyscorewithscores(String key, double min, double max, int offset, int count);
Map zrangebyscorewithscores(String key, String min, String max, int offset, int count);
/**
* Available since 2.0.0
* Time complexity: O(log(N))
*
*
* Returns the rank of member in the sorted set stored at key, with the scores ordered from low to high.
* The rank (or index) is 0-based, which means that the member with the lowest score has rank 0.
* Use ZREVRANK to get the rank of an element with the scores ordered from high to low.
*
* @param key
* @param member
* @return If member exists in the sorted set, return the rank of member.
* If member does not exist in the sorted set or key does not exist, return null.
*/
Long zrank(String key, String member);
/**
* Available since 1.2.0
* Time complexity: O(M*log(N)) with N being the number of elements in the sorted set and M the number of elements to be removed.
*
*
* Removes the specified members from the sorted set stored at key. Non existing members are ignored.
* An error is returned when key exists and does not hold a sorted set.
*
*
* History
* >= 2.4: Accepts multiple elements.
* In Redis versions older than 2.4 it was possible to remove a single member per call.
*
* @param key
* @param members
* @return The number of members removed from the sorted set, not including non existing members.
*/
Long zrem(String key, String... members);
/**
* Available since 2.8.9
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements removed by the operation.
*
* When all the elements in a sorted set are inserted with the same score, in order to force lexicographical ordering,
* this command removes all elements in the sorted set stored at key between the lexicographical range specified by min and max.
* The meaining of min and max are the same of the ZRANGEBYLEX command.
* Similarly, this command actually returns the same elements that ZRANGEBYLEX would return if called with the same min and max arguments.
*
* @param key
* @param min
* @param max
* @return the number of elements removed.
*/
Long zremrangebylex(String key, String min, String max);
/**
* Available since 2.0.0
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements removed by the operation.
*
*
* Removes all elements in the sorted set stored at key with rank between start and stop.
* Both start and stop are 0 -based indexes with 0 being the element with the lowest score.
* These indexes can be negative numbers, where they indicate offsets starting at the element with the highest score.
* For example: -1 is the element with the highest score, -2 the element with the second highest score and so forth.
*
* @param key
* @param start inclusive
* @param stop inclusive
* @return the number of elements removed.
*/
Long zremrangebyrank(String key, long start, long stop);
/**
* Available since 1.2.0
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements removed by the operation.
*
*
* Removes all elements in the sorted set stored at key with a score between min and max (inclusive).
* Since version 2.1.6, min and max can be exclusive, e.g.
*
* ZREMRANGEBYSOCRE test (1 (3
*
* following the syntax of ZRANGEBYSCORE.
*
* @param key
* @param min
* @param max
* @return the number of elements removed.
*/
Long zremrangebyscore(String key, double min, double max);
Long zremrangebyscore(String key, String min, String max);
/**
* Available since 1.2.0
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements returned.
*
*
* Returns the specified range of elements in the sorted set stored at key.
* The elements are considered to be ordered from the highest to the lowest score.
* Descending lexicographical order is used for elements with equal score.
* Apart from the reversed ordering, ZREVRANGE is similar to ZRANGE.
*
* @param key
* @param start
* @param stop
* @return list of elements in the specified range (optionally with their scores)
*/
Set zrevrange(String key, long start, long stop);
Map zrevrangewithscores(String key, long start, long stop);
/**
* Available since 2.2.0
* Time complexity: O(log(N)+M) with N being the number of elements in the sorted set and M the number of elements being returned.
* If M is constant (e.g. always asking for the first 10 elements with LIMIT), you can consider it O(log(N)).
*
*
* Returns all the elements in the sorted set at key with a score between max and min (including elements with score equal to max or min).
* In contrary to the default ordering of sorted sets, for this command the elements are considered to be ordered from high to low scores.
* The elements having the same score are returned in reverse lexicographical order.
* Apart from the reversed ordering, ZREVRANGEBYSCORE is similar to ZRANGEBYSCORE.
*
* @param key
* @param max
* @param min
* @return list of elements in the specified score range (optionally with their scores)
*/
Set zrevrangebyscore(String key, double max, double min);
Set zrevrangebyscore(String key, String max, String min);
Set zrevrangebyscore(String key, double max, double min, int offset, int count);
Set zrevrangebyscore(String key, String max, String min, int offset, int count);
Map zrevrangebyscorewithscores(String key, double max, double min);
Map zrevrangebyscorewithscores(String key, String max, String min);
Map zrevrangebyscorewithscores(String key, double max, double min, int offset, int count);
Map zrevrangebyscorewithscores(String key, String max, String min, int offset, int count);
/**
* Available since 2.0.0
* Time complexity: O(log(N))
*
*
* Returns the rank of member in the sorted set stored at key, with the scores ordered from high to low.
* The rank (or index) is 0-based, which means that the member with the highest score has rank 0.
* Use ZRANK to get the rank of an element with the scores ordered from low to high.
*
* @param key
* @param member
* @return If member exists in the sorted set, return the rank of member.
* If member does not exist in the sorted set or key does not exist, return null.
*/
Long zrevrank(String key, String member);
/**
* Available since 1.2.0
* Time complexity: O(1)
*
*
* Returns the score of member in the sorted set at key.
* If member does not exist in the sorted set, or key does not exist, null is returned.
*
* @param key
* @param member
* @return the score of member (a double precision floating point number), represented as string.
*/
Double zscore(String key, String member);
/**
* Available since 2.0.0
* Time complexity: O(N)+O(M log(M)) with N being the sum of the sizes of the input sorted sets,
* and M being the number of elements in the resulting sorted set.
*
*
* Computes the union of sorted sets given by the specified keys, and stores the result in destination.
* By default, the resulting score of an element is the sum of its scores in the sorted sets where it exists.
* Using the WEIGHTS option, it is possible to specify a multiplication factor for each input sorted set.
* This means that the score of every element in every input sorted set is multiplied by this factor before being
* passed to the aggregation function.
* When WEIGHTS is not given, the multiplication factors default to 1.
* With the AGGREGATE option, it is possible to specify how the results of the union are aggregated.
* This option defaults to SUM, where the score of an element is summed across the inputs where it exists.
* When this option is set to either MIN or MAX, the resulting set will contain the minimum or maximum score of an
* element across the inputs where it exists.
* If destination already exists, it is overwritten.
*
* @param destination
* @param keys
* @return the number of elements in the resulting sorted set at destination.
*/
Long zunionstore(String destination, String... keys);
Long zunionstoremax(String destination, String... keys);
Long zunionstoremin(String destination, String... keys);
Long zunionstore(String destination, Map weightkeys);
Long zunionstoremax(String destination, Map weightkeys);
Long zunionstoremin(String destination, Map weightkeys);
/**
* @see #scan(String)
*/
ScanResult> zscan(String key, String cursor);
ScanResult> zscan(String key, String cursor, int count);
ScanResult> zscan(String key, String cursor, String pattern);
ScanResult> zscan(String key, String cursor, String pattern, int count);
// ~ -------------------------------------------------------------------------------------------------- HyperLogLog
/**
* Available since 2.8.9.
* Time complexity: O(1) to add every element.
*
*
* Adds all the element arguments to the HyperLogLog data structure stored at the variable name specified as first argument.
* As a side effect of this command the HyperLogLog internals may be updated to reflect a different estimation of the number of unique items added so far (the cardinality of the set).
* If the approximated cardinality estimated by the HyperLogLog changed after executing the command, PFADD returns 1, otherwise 0 is returned. The command automatically creates an empty HyperLogLog structure (that is, a Redis String of a specified length and with a given encoding) if the specified key does not exist.
* To call the command without elements but just the variable name is valid, this will result into no operation performed if the variable already exists, or just the creation of the data structure if the key does not exist (in the latter case 1 is returned).
* For an introduction to HyperLogLog data structure check the PFCOUNT command page.
*
*
* Return value
* Integer reply, specifically:
* 1 if at least 1 HyperLogLog internal register was altered. 0 otherwise.
*
* @param key
* @param elements
* @return 1 if at least 1 HyperLogLog internal register was altered. 0 otherwise.
*/
Long pfadd(String key, String... elements);
/**
* Available since 2.8.9.
* Time complexity: O(1) with every small average constant times when called with a single key. O(N) with N being the number of keys, and much bigger constant times, when called with multiple keys.
*
*
* When called with a single key, returns the approximated cardinality computed by the HyperLogLog data structure stored at the specified variable, which is 0 if the variable does not exist.
* When called with multiple keys, returns the approximated cardinality of the union of the HyperLogLogs passed, by internally merging the HyperLogLogs stored at the provided keys into a temporary hyperLogLog.
* The HyperLogLog data structure can be used in order to count unique elements in a set using just a small constant amount of memory, specifically 12k bytes for every HyperLogLog (plus a few bytes for the key itself).
* The returned cardinality of the observed set is not exact, but approximated with a standard error of 0.81%.
* For example in order to take the count of all the unique search queries performed in a day, a program needs to call PFADD every time a query is processed. The estimated number of unique queries can be retrieved with PFCOUNT at any time.
* Note: as a side effect of calling this function, it is possible that the HyperLogLog is modified, since the last 8 bytes encode the latest computed cardinality for caching purposes. So PFCOUNT is technically a write command.
*
*
* Examples
*
* redis> PFADD hll foo bar zap
* (integer) 1
* redis> PFADD hll zap zap zap
* (integer) 0
* redis> PFADD hll foo bar
* (integer) 0
* redis> PFCOUNT hll
* (integer) 3
* redis> PFADD some-other-hll 1 2 3
* (integer) 1
* redis> PFCOUNT hll some-other-hll
* (integer) 6
* redis>
*
*
* Performances
* When PFCOUNT is called with a single key, performances as excellent even if in theory constant times to process a dense HyperLogLog are high. This is possible because the PFCOUNT uses caching in order to remember the cardinality previously computed, that rarely changes because most PFADD operations will not update any register. Hundreds of operations per second are possible.
* When PFCOUNT is called with multiple keys, an on-the-fly merge of the HyperLogLogs is performed, which is slow, moreover the cardinality of the union can't be cached, so when used with multiple keys PFCOUNT may take a time in the order of magnitude of the millisecond, and should be not abused.
* The user should take in mind that single-key and multiple-keys executions of this command are semantically different and have different performances.
*
* HyperLogLog representation
* Redis HyperLogLogs are represented using a double representation: the sparse representation suitable for HLLs counting a small number of elements (resulting in a small number of registers set to non-zero value), and a dense representation suitable for higher cardinalities. Redis automatically switches from the sparse to the dense representation when needed.
* The sparse representation uses a run-length encoding optimized to store efficiently a big number of registers set to zero. The dense representation is a Redis string of 12288 bytes in order to store 16384 6-bit counters. The need for the double representation comes from the fact that using 12k (which is the dense representation memory requirement) to encode just a few registers for smaller cardinalities is extremely suboptimal.
* Both representations are prefixed with a 16 bytes header, that includes a magic, an encoding / version fiend, and the cached cardinality estimation computed, stored in little endian format (the most significant bit is 1 if the estimation is invalid since the HyperLogLog was updated since the cardinality was computed).
* The HyperLogLog, being a Redis string, can be retrieved with GET and restored with SET. Calling PFADD, PFCOUNT or PFMERGE commands with a corrupted HyperLogLog is never a problem, it may return random values but does not affect the stability of the server. Most of the times when corrupting a sparse representation, the server recognizes the corruption and returns an error.
* The representation is neutral from the point of view of the processor word size and endianess, so the same representation is used by 32 bit and 64 bit processor, big endian or little endian.
* More details about the Redis HyperLogLog implementation can be found in this blog post. The source code of the implementation in the hyperloglog.c file is also easy to read and understand, and includes a full specification for the exact encoding used for the sparse and dense representations.
*
* @param keys
* @return Integer reply, specifically:
* The approximated number of unique elements observed via PFADD.
*/
Long pfcount(String... keys);
/**
* Available since 2.8.9
* Time complexity: O(N) to merge N HyperLogLogs, but with high constant times.
*
*
* Merge multiple HyperLogLog values into an unique value that will approximate the cardinality of the union of the observed Sets of the source HyperLogLog structures.
* The computed merged HyperLogLog is set to the destination variable, which is created if does not exist (defauling to an empty HyperLogLog).
*
*
* Examples
*
* redis> PFADD hll1 foo bar zap a
* (integer) 1
* redis> PFADD hll2 a b c foo
* (integer) 1
* redis> PFMERGE hll3 hll1 hll2
* OK
* redis> PFCOUNT hll3
* (integer) 6
* redis>
*
*
* @param destkey
* @param sourcekeys
* @return Simple string reply: The command just returns OK.
*/
String pfmerge(String destkey, String... sourcekeys);
// ~ ------------------------------------------------------------------------------------------------------ Pub/Sub
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the number of patterns the client is already subscribed to.
*
*
* Subscribes the client to the given patterns.
*
* @param handler
* @param patterns
* @return a redis pub/sub object
*/
RedisPubSub psubscribe(RedisPsubscribeHandler handler, String... patterns);
/**
* Available since 2.0.0
* Time complexity: O(N+M) where N is the number of clients subscribed to the receiving channel
* and M is the total number of subscribed patterns (by any client).
*
*
* Posts a message to the given channel.
*
* @param channel
* @param message
* @return the number of clients that received the message.
*/
Long publish(String channel, String message);
/**
* Available since 2.0.0.
* Time complexity: O(N+M) where N is the number of patterns the client is already subscribed
* and M is the number of total patterns subscribed in the system (by any client).
*
*
* Unsubscribes the client from the given patterns, or from all of them if none is given.
* When no patters are specified, the client is unsubscribed from all the previously subscribed patterns.
* In this case, a message for every unsubscribed pattern will be sent to the client.
*
* @param pubsub
* @param patterns
*/
void punsubscribe(RedisPubSub pubsub, String... patterns);
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the number of channels to subscribe to.
*
*
* Subscribes the client to the specified channels.
* Once the client enters the subscribed state it is not supposed to issue any other commands,
* except for additional SUBSCRIBE, PSUBSCRIBE, UNSUBSCRIBE and PUNSUBSCRIBE commands.
*
* @param handler
* @param channels
* @return redis pub/sub object.
*/
RedisPubSub subscribe(RedisSubscribeHandler handler, String... channels);
/**
* Available since 2.0.0
* Time complexity: O(N) where N is the number of clients already subscribed to a channel.
*
*
* Unsubscribes the client from the given channels, or from all of them if none is given.
* When no channels are specified, the client is unsubscribed from all the previously subscribed channels.
* In this case, a message for every unsubscribed channel will be sent to the client.
*
* @param pubsub
* @param channels
*/
void unsubscribe(RedisPubSub pubsub, String... channels);
/**
* Available since 2.8.0
* Time complexity: O(N) for the CHANNELS subcommand, where N is the number of active channels, and assuming constant
* time pattern matching (relatively short channels and patterns). O(N) for the NUMSUB subcommand,
* where N is the number of requested channels. O(1) for the NUMPAT subcommand.
*
*
* The PUBSUB command is an introspection command that allows to inspect the state of the Pub/Sub subsystem.
* It is composed of subcommands that are documented separately. The general form is:
*
* PUBSUB subcommand ... args ...
*
*
*
* PUBSUB CHANNELS [pattern]
* Lists the currently active channels. An active channel is a Pub/Sub channel with one ore more subscribers (not including clients subscribed to patterns).
* If no pattern is specified, all the channels are listed, otherwise if pattern is specified only channels matching the specified glob-style pattern are listed.
* Return value
* Array reply: a list of active channels, optionally matching the specified pattern.
*
*
* PUBSUB NUMSUB [channel-1 ... channel-N]
* Returns the number of subscribers (not counting clients subscribed to patterns) for the specified channels.
* Return value
* Array reply: a list of channels and number of subscribers for every channel. The format is channel, count, channel, count, ..., so the list is flat. The order in which the channels are listed is the same as the order of the channels specified in the command call.
* Note that it is valid to call this command without channels. In this case it will just return an empty list.
*
*
* PUBSUB NUMPAT
* Returns the number of subscriptions to patterns (that are performed using the PSUBSCRIBE command). Note that this is not just the count of clients subscribed to patterns but the total number of patterns all the clients are subscribed to.
*
* @param pattern
* @return Integer reply: the number of patterns all the clients are subscribed to.
*/
List pubsubchannels(String pattern);
Long pubsubnumpat();
Map pubsubnumsub(String... channels);
// ~ ------------------------------------------------------------------------------------------------ Transactions
/**
* Available since 2.0.0
*
*
* Flushes all previously queued commands in a transaction and restores the connection state to normal.
* If WATCH was used, DISCARD unwatches all keys.
*
* @param t
* @return always OK.
*/
String discard(RedisTransaction t);
/**
* Available since 1.2.0
*
*
* Executes all previously queued commands in a transaction and restores the connection state to normal.
* When using WATCH, EXEC will execute commands only if the watched keys were not modified, allowing for a check-and-set mechanism.
*
* @param t
* @return each element being the reply to each of the commands in the atomic transaction.
* When using WATCH, EXEC can return a empty list if the execution was aborted.
*/
List