Skip to content

Commit 6dfd60a

Browse files
Adds support for Redis setGet command
fixes gh-2853
1 parent ca8308c commit 6dfd60a

13 files changed

+221
-0
lines changed

src/main/java/org/springframework/data/redis/connection/DefaultStringRedisConnection.java

+5
Original file line numberDiff line numberDiff line change
@@ -770,6 +770,11 @@ public Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption op
770770
return convertAndReturn(delegate.set(key, value, expiration, option), Converters.identityConverter());
771771
}
772772

773+
@Override
774+
public byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option) {
775+
return convertAndReturn(delegate.setGet(key, value, expiration, option), Converters.identityConverter());
776+
}
777+
773778
@Override
774779
public Boolean setBit(byte[] key, long offset, boolean value) {
775780
return convertAndReturn(delegate.setBit(key, offset, value), Converters.identityConverter());

src/main/java/org/springframework/data/redis/connection/DefaultedRedisConnection.java

+7
Original file line numberDiff line numberDiff line change
@@ -326,6 +326,13 @@ default Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption o
326326
return stringCommands().set(key, value, expiration, option);
327327
}
328328

329+
/** @deprecated in favor of {@link RedisConnection#stringCommands()}}. */
330+
@Override
331+
@Deprecated
332+
default byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option) {
333+
return stringCommands().setGet(key, value, expiration, option);
334+
}
335+
329336
/** @deprecated in favor of {@link RedisConnection#stringCommands()}}. */
330337
@Override
331338
@Deprecated

src/main/java/org/springframework/data/redis/connection/ReactiveStringCommands.java

+42
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@
4747
*
4848
* @author Christoph Strobl
4949
* @author Mark Paluch
50+
* @author Marcin Grzejszczak
5051
* @since 2.0
5152
*/
5253
public interface ReactiveStringCommands {
@@ -62,15 +63,23 @@ class SetCommand extends KeyCommand {
6263
private @Nullable ByteBuffer value;
6364
private Expiration expiration;
6465
private SetOption option;
66+
private boolean get;
6567

6668
private SetCommand(ByteBuffer key, @Nullable ByteBuffer value, @Nullable Expiration expiration,
6769
@Nullable SetOption option) {
6870

71+
this(key, value, expiration, option, false);
72+
}
73+
74+
private SetCommand(ByteBuffer key, @Nullable ByteBuffer value, @Nullable Expiration expiration,
75+
@Nullable SetOption option, boolean get) {
76+
6977
super(key);
7078

7179
this.value = value;
7280
this.expiration = expiration;
7381
this.option = option;
82+
this.get = get;
7483
}
7584

7685
/**
@@ -125,6 +134,16 @@ public SetCommand withSetOption(SetOption option) {
125134
return new SetCommand(getKey(), value, expiration, option);
126135
}
127136

137+
/**
138+
* Applies GET option. Return the old string stored at key, or {@code null} if key did not exist.
139+
* An error is returned and SET aborted if the value stored at key is not a string.
140+
*
141+
* @return a new {@link SetCommand} with {@link SetOption} applied.
142+
*/
143+
public SetCommand withGet() {
144+
return new SetCommand(getKey(), value, expiration, option, true);
145+
}
146+
128147
/**
129148
* @return
130149
*/
@@ -146,6 +165,13 @@ public Optional<Expiration> getExpiration() {
146165
public Optional<SetOption> getOption() {
147166
return Optional.ofNullable(option);
148167
}
168+
169+
/**
170+
* @return
171+
*/
172+
public boolean isGet() {
173+
return get;
174+
}
149175
}
150176

151177
/**
@@ -184,6 +210,22 @@ default Mono<Boolean> set(ByteBuffer key, ByteBuffer value, Expiration expiratio
184210
.map(BooleanResponse::getOutput);
185211
}
186212

213+
/**
214+
* Set {@literal value} for {@literal key} with {@literal expiration} and {@literal options}. Return the old
215+
* string stored at key, or nil if key did not exist. An error is returned and SET aborted if the value
216+
* stored at key is not a string.
217+
*
218+
* @param key must not be {@literal null}.
219+
* @param value must not be {@literal null}.
220+
* @param expiration must not be {@literal null}. Use {@link Expiration#persistent()} for no expiration time or
221+
* {@link Expiration#keepTtl()} to keep the existing.
222+
* @param option must not be {@literal null}.
223+
* @return
224+
* @see <a href="https://redis.io/commands/set">Redis Documentation: SET</a>
225+
* @since 3.4
226+
*/
227+
Flux<ByteBufferResponse<SetCommand>> setGet(ByteBuffer key, ByteBuffer value, Expiration expiration, SetOption option);
228+
187229
/**
188230
* Set each and every item separately by invoking {@link SetCommand}.
189231
*

src/main/java/org/springframework/data/redis/connection/RedisStringCommands.java

+16
Original file line numberDiff line numberDiff line change
@@ -122,6 +122,22 @@ enum BitOperation {
122122
@Nullable
123123
Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption option);
124124

125+
/**
126+
* Set {@code value} for {@code key}. Return the old string stored at key, or nil if key did not exist.
127+
* An error is returned and SET aborted if the value stored at key is not a string.
128+
*
129+
* @param key must not be {@literal null}.
130+
* @param value must not be {@literal null}.
131+
* @param expiration must not be {@literal null}. Use {@link Expiration#persistent()} to not set any ttl or
132+
* {@link Expiration#keepTtl()} to keep the existing expiration.
133+
* @param option must not be {@literal null}. Use {@link SetOption#upsert()} to add non existing.
134+
* @return {@literal null} when used in pipeline / transaction.
135+
* @since 3.4
136+
* @see <a href="https://redis.io/commands/set">Redis Documentation: SET</a>
137+
*/
138+
@Nullable
139+
byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option);
140+
125141
/**
126142
* Set {@code value} for {@code key}, only if {@code key} does not exist.
127143
*

src/main/java/org/springframework/data/redis/connection/jedis/JedisClusterStringCommands.java

+18
Original file line numberDiff line numberDiff line change
@@ -150,6 +150,24 @@ public Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption op
150150
}
151151
}
152152

153+
@Override
154+
public byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option) {
155+
156+
Assert.notNull(key, "Key must not be null");
157+
Assert.notNull(value, "Value must not be null");
158+
Assert.notNull(expiration, "Expiration must not be null");
159+
Assert.notNull(option, "Option must not be null");
160+
161+
SetParams setParams = JedisConverters.toSetCommandExPxArgument(expiration,
162+
JedisConverters.toSetCommandNxXxArgument(option));
163+
164+
try {
165+
return connection.getCluster().setGet(key, value, setParams);
166+
} catch (Exception ex) {
167+
throw convertJedisAccessException(ex);
168+
}
169+
}
170+
153171
@Override
154172
public Boolean setNX(byte[] key, byte[] value) {
155173

src/main/java/org/springframework/data/redis/connection/jedis/JedisStringCommands.java

+14
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,20 @@ public Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption op
116116
.getOrElse(Converters.stringToBooleanConverter(), () -> false);
117117
}
118118

119+
@Override
120+
@Nullable
121+
public byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option) {
122+
Assert.notNull(key, "Key must not be null");
123+
Assert.notNull(value, "Value must not be null");
124+
Assert.notNull(expiration, "Expiration must not be null");
125+
Assert.notNull(option, "Option must not be null");
126+
127+
SetParams params = JedisConverters.toSetCommandExPxArgument(expiration,
128+
JedisConverters.toSetCommandNxXxArgument(option));
129+
130+
return connection.invoke().just(Jedis::setGet, PipelineBinaryCommands::setGet, key, value, params);
131+
}
132+
119133
@Override
120134
public Boolean setNX(byte[] key, byte[] value) {
121135

src/main/java/org/springframework/data/redis/connection/lettuce/LettuceReactiveStringCommands.java

+14
Original file line numberDiff line numberDiff line change
@@ -105,6 +105,20 @@ public Flux<BooleanResponse<SetCommand>> set(Publisher<SetCommand> commands) {
105105
}));
106106
}
107107

108+
@Override
109+
public Flux<ByteBufferResponse<SetCommand>> setGet(ByteBuffer key, ByteBuffer value, Expiration exp,
110+
RedisStringCommands.SetOption option) {
111+
return this.connection.execute(reactiveCommands -> Flux.from(Mono.just(SetCommand.set(key).value(value).withGet().expiring(exp).withSetOption(option))).concatMap((command) -> {
112+
113+
Assert.notNull(command.getKey(), "Key must not be null");
114+
Assert.notNull(command.getValue(), "Value must not be null");
115+
116+
return reactiveCommands.setGet(command.getKey(), command.getValue())
117+
.map(v -> new ByteBufferResponse<>(command, v))
118+
.defaultIfEmpty(new AbsentByteBufferResponse<>(command));
119+
}));
120+
}
121+
108122
@Override
109123
public Flux<ByteBufferResponse<SetCommand>> getSet(Publisher<SetCommand> commands) {
110124

src/main/java/org/springframework/data/redis/connection/lettuce/LettuceStringCommands.java

+12
Original file line numberDiff line numberDiff line change
@@ -115,6 +115,18 @@ public Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption op
115115
.orElse(LettuceConverters.stringToBooleanConverter(), false);
116116
}
117117

118+
@Override
119+
@Nullable
120+
public byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option) {
121+
Assert.notNull(key, "Key must not be null");
122+
Assert.notNull(value, "Value must not be null");
123+
Assert.notNull(expiration, "Expiration must not be null");
124+
Assert.notNull(option, "Option must not be null");
125+
126+
return connection.invoke()
127+
.just(RedisStringAsyncCommands::setGet, key, value, LettuceConverters.toSetArgs(expiration, option));
128+
}
129+
118130
@Override
119131
public Boolean setNX(byte[] key, byte[] value) {
120132

src/main/java/org/springframework/data/redis/core/DefaultValueOperations.java

+21
Original file line numberDiff line numberDiff line change
@@ -281,6 +281,27 @@ private boolean failsafeInvokePsetEx(RedisConnection connection) {
281281
});
282282
}
283283

284+
@Override
285+
public V setGet(K key, V value, long timeout, TimeUnit unit) {
286+
return doSetGet(key, value, Expiration.from(timeout, unit));
287+
}
288+
289+
@Override
290+
public V setGet(K key, V value, Duration duration) {
291+
return doSetGet(key, value, Expiration.from(duration));
292+
}
293+
294+
private V doSetGet(K key, V value, Expiration duration) {
295+
byte[] rawValue = rawValue(value);
296+
return execute( new ValueDeserializingRedisCallback(key) {
297+
298+
@Override
299+
protected byte[] inRedis(byte[] rawKey, RedisConnection connection) {
300+
return connection.stringCommands().setGet(rawKey, rawValue, duration, SetOption.UPSERT);
301+
}
302+
});
303+
}
304+
284305
@Override
285306
public Boolean setIfAbsent(K key, V value) {
286307

src/main/java/org/springframework/data/redis/core/ValueOperations.java

+28
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@
3232
* @author Christoph Strobl
3333
* @author Mark Paluch
3434
* @author Jiahe Cai
35+
* @author Marcin Grzejszczak
3536
*/
3637
public interface ValueOperations<K, V> {
3738

@@ -44,6 +45,33 @@ public interface ValueOperations<K, V> {
4445
*/
4546
void set(K key, V value);
4647

48+
/**
49+
* Set the {@code value} and expiration {@code timeout} for {@code key}. Return the old
50+
* string stored at key, or nil if key did not exist. An error is returned and SET aborted if the value
51+
* stored at key is not a string.
52+
*
53+
* @param key must not be {@literal null}.
54+
* @param value must not be {@literal null}.
55+
* @param timeout the key expiration timeout.
56+
* @param unit must not be {@literal null}.
57+
* @see <a href="https://redis.io/commands/set">Redis Documentation: SET</a>
58+
* @since 3.4
59+
*/
60+
V setGet(K key, V value, long timeout, TimeUnit unit);
61+
62+
/**
63+
* Set the {@code value} and expiration {@code timeout} for {@code key}. Return the old
64+
* string stored at key, or nil if key did not exist. An error is returned and SET aborted if the value
65+
* stored at key is not a string.
66+
*
67+
* @param key must not be {@literal null}.
68+
* @param value must not be {@literal null}.
69+
* @param duration expiration duration
70+
* @see <a href="https://redis.io/commands/set">Redis Documentation: SET</a>
71+
* @since 3.4
72+
*/
73+
V setGet(K key, V value, Duration duration);
74+
4775
/**
4876
* Set the {@code value} and expiration {@code timeout} for {@code key}.
4977
*

src/test/java/org/springframework/data/redis/connection/RedisConnectionUnitTests.java

+5
Original file line numberDiff line numberDiff line change
@@ -1093,6 +1093,11 @@ public Boolean set(byte[] key, byte[] value, Expiration expiration, SetOption op
10931093
return delegate.set(key, value, expiration, options);
10941094
}
10951095

1096+
@Override
1097+
public byte[] setGet(byte[] key, byte[] value, Expiration expiration, SetOption option) {
1098+
return delegate.setGet(key, value, expiration, option);
1099+
}
1100+
10961101
@Override
10971102
public List<Long> bitField(byte[] key, BitFieldSubCommands subCommands) {
10981103
return delegate.bitField(key, subCommands);

src/test/java/org/springframework/data/redis/connection/lettuce/LettuceReactiveStringCommandsIntegrationTests.java

+13
Original file line numberDiff line numberDiff line change
@@ -556,4 +556,17 @@ void setKeepTTL() {
556556
assertThat(nativeBinaryCommands.ttl(KEY_1_BBUFFER)).isCloseTo(expireSeconds, Offset.offset(5L));
557557
assertThat(nativeCommands.get(KEY_1)).isEqualTo(VALUE_2);
558558
}
559+
560+
@ParameterizedRedisTest // GH-2853
561+
void setWithGetOption() {
562+
nativeCommands.set(KEY_1, VALUE_1);
563+
564+
connection.stringCommands().setGet(KEY_1_BBUFFER, VALUE_2_BBUFFER, Expiration.keepTtl(), SetOption.upsert())
565+
.map(CommandResponse::getOutput)
566+
.as(StepVerifier::create) //
567+
.expectNext(VALUE_1_BBUFFER) //
568+
.verifyComplete();
569+
570+
assertThat(nativeCommands.get(KEY_1)).isEqualTo(VALUE_2);
571+
}
559572
}

src/test/java/org/springframework/data/redis/core/DefaultValueOperationsIntegrationTests.java

+26
Original file line numberDiff line numberDiff line change
@@ -316,6 +316,32 @@ void testSetWithExpirationWithTimeUnitMilliseconds() {
316316
await().atMost(Duration.ofMillis(500L)).until(() -> !redisTemplate.hasKey(key));
317317
}
318318

319+
@ParameterizedRedisTest
320+
void testSetGetWithExpiration() {
321+
322+
K key = keyFactory.instance();
323+
V value1 = valueFactory.instance();
324+
V value2 = valueFactory.instance();
325+
326+
valueOps.set(key, value1);
327+
328+
assertThat(valueOps.setGet(key, value2, 1, TimeUnit.SECONDS)).isEqualTo(value1);
329+
assertThat(valueOps.get(key)).isEqualTo(value2);
330+
}
331+
332+
@ParameterizedRedisTest
333+
void testSetGetWithExpirationDuration() {
334+
335+
K key = keyFactory.instance();
336+
V value1 = valueFactory.instance();
337+
V value2 = valueFactory.instance();
338+
339+
valueOps.set(key, value1);
340+
341+
assertThat(valueOps.setGet(key, value2, Duration.ofMillis(1000))).isEqualTo(value1);
342+
assertThat(valueOps.get(key)).isEqualTo(value2);
343+
}
344+
319345
@ParameterizedRedisTest
320346
void testAppend() {
321347

0 commit comments

Comments
 (0)