BitCommandsTest.java
package redis.clients.jedis.commands.jedis;
import java.util.List;
import io.redis.test.annotations.SinceRedisVersion;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.params.ParameterizedClass;
import org.junit.jupiter.params.provider.MethodSource;
import redis.clients.jedis.Protocol;
import redis.clients.jedis.RedisProtocol;
import redis.clients.jedis.args.BitCountOption;
import redis.clients.jedis.args.BitOP;
import redis.clients.jedis.exceptions.JedisDataException;
import redis.clients.jedis.params.BitPosParams;
import redis.clients.jedis.util.SafeEncoder;
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
@ParameterizedClass
@MethodSource("redis.clients.jedis.commands.CommandsTestsParameters#respVersions")
public class BitCommandsTest extends JedisCommandsTestBase {
public BitCommandsTest(RedisProtocol protocol) {
super(protocol);
}
@Test
public void setAndgetbit() {
assertFalse(jedis.setbit("foo", 0, true));
assertTrue(jedis.getbit("foo", 0));
// Binary
assertFalse(jedis.setbit("bfoo".getBytes(), 0, true));
assertTrue(jedis.getbit("bfoo".getBytes(), 0));
}
@Test
public void bitpos() {
String foo = "foo";
jedis.set(foo, String.valueOf(0));
// string "0" with bits: 0011 0000
jedis.setbit(foo, 3, true);
jedis.setbit(foo, 7, true);
jedis.setbit(foo, 13, true);
jedis.setbit(foo, 39, true);
/*
* bit: 00110001 / 00000100 / 00000000 / 00000000 / 00000001
* byte: 0 1 2 3 4
*/
long offset = jedis.bitpos(foo, true);
assertEquals(2, offset);
offset = jedis.bitpos(foo, false);
assertEquals(0, offset);
offset = jedis.bitpos(foo, true, new BitPosParams(1));
assertEquals(13, offset);
offset = jedis.bitpos(foo, false, new BitPosParams(1));
assertEquals(8, offset);
offset = jedis.bitpos(foo, true, new BitPosParams(2, 3));
assertEquals(-1, offset);
offset = jedis.bitpos(foo, false, new BitPosParams(2, 3));
assertEquals(16, offset);
offset = jedis.bitpos(foo, true, new BitPosParams(3, 4));
assertEquals(39, offset);
}
@Test
public void bitposBinary() {
// binary
byte[] bfoo = { 0x01, 0x02, 0x03, 0x04 };
jedis.set(bfoo, Protocol.toByteArray(0));
// bits: 0011 0000
jedis.setbit(bfoo, 3, true);
jedis.setbit(bfoo, 7, true);
jedis.setbit(bfoo, 13, true);
jedis.setbit(bfoo, 39, true);
/*
* bit: 00110001 / 00000100 / 00000000 / 00000000 / 00000001
* byte: 0 1 2 3 4
*/
long offset = jedis.bitpos(bfoo, true);
assertEquals(2, offset);
offset = jedis.bitpos(bfoo, false);
assertEquals(0, offset);
offset = jedis.bitpos(bfoo, true, new BitPosParams(1));
assertEquals(13, offset);
offset = jedis.bitpos(bfoo, false, new BitPosParams(1));
assertEquals(8, offset);
offset = jedis.bitpos(bfoo, true, new BitPosParams(2, 3));
assertEquals(-1, offset);
offset = jedis.bitpos(bfoo, false, new BitPosParams(2, 3));
assertEquals(16, offset);
offset = jedis.bitpos(bfoo, true, new BitPosParams(3, 4));
assertEquals(39, offset);
}
@Test
public void bitposWithNoMatchingBitExist() {
String foo = "foo";
jedis.set(foo, String.valueOf(0));
for (int idx = 0; idx < 8; idx++) {
jedis.setbit(foo, idx, true);
}
/*
* bit: 11111111
* byte: 0
*/
long offset = jedis.bitpos(foo, false);
// offset should be last index + 1
assertEquals(8, offset);
}
@Test
public void bitposWithNoMatchingBitExistWithinRange() {
String foo = "foo";
jedis.set(foo, String.valueOf(0));
for (int idx = 0; idx < 8 * 5; idx++) {
jedis.setbit(foo, idx, true);
}
/*
* bit: 11111111 / 11111111 / 11111111 / 11111111 / 11111111
* byte: 0 1 2 3 4
*/
long offset = jedis.bitpos(foo, false, new BitPosParams(2, 3));
// offset should be -1
assertEquals(-1, offset);
}
@Test
@SinceRedisVersion(value = "7.0.0", message = "7.0.0 Added the BYTE|BIT option.")
public void bitposModifier() {
jedis.set("mykey", "\\x00\\xff\\xf0");
assertEquals(0, jedis.bitpos("mykey", false));
assertEquals(1, jedis.bitpos("mykey", true));
assertEquals(1, jedis.bitpos("mykey", true, BitPosParams.bitPosParams()));
assertEquals(18, jedis.bitpos("mykey", true, BitPosParams.bitPosParams().start(2)));
assertEquals(18, jedis.bitpos("mykey", true, BitPosParams.bitPosParams().start(2).end(-1)));
assertEquals(18, jedis.bitpos("mykey", true, BitPosParams.bitPosParams().start(2).end(-1)
.modifier(BitCountOption.BYTE)));
assertEquals(9, jedis.bitpos("mykey", true, BitPosParams.bitPosParams().start(7).end(15)
.modifier(BitCountOption.BIT)));
}
@Test
@SinceRedisVersion("7.0.0")
public void setAndgetrange() {
jedis.set("key1", "Hello World");
assertEquals(11, jedis.setrange("key1", 6, "Jedis"));
assertEquals("Hello Jedis", jedis.get("key1"));
assertEquals("Hello", jedis.getrange("key1", 0, 4));
assertEquals("Jedis", jedis.getrange("key1", 6, 11));
}
@Test
public void bitCount() {
jedis.setbit("foo", 16, true);
jedis.setbit("foo", 24, true);
jedis.setbit("foo", 40, true);
jedis.setbit("foo", 56, true);
assertEquals(4, (long) jedis.bitcount("foo"));
assertEquals(4, (long) jedis.bitcount("foo".getBytes()));
assertEquals(3, (long) jedis.bitcount("foo", 2L, 5L));
assertEquals(3, (long) jedis.bitcount("foo".getBytes(), 2L, 5L));
}
@Test
@SinceRedisVersion("7.0.0")
public void bitCountModifier() {
jedis.setbit("foo", 16, true);
jedis.setbit("foo", 24, true);
jedis.setbit("foo", 40, true);
jedis.setbit("foo", 56, true);
assertEquals(3, (long) jedis.bitcount("foo", 2L, 5L, BitCountOption.BYTE));
assertEquals(3, (long) jedis.bitcount("foo".getBytes(), 2L, 5L, BitCountOption.BYTE));
assertEquals(0, (long) jedis.bitcount("foo", 2L, 5L, BitCountOption.BIT));
assertEquals(0, (long) jedis.bitcount("foo".getBytes(), 2L, 5L, BitCountOption.BIT));
}
@Test
public void bitOp() {
jedis.set("key1", "\u0060");
jedis.set("key2", "\u0044");
jedis.bitop(BitOP.AND, "resultAnd", "key1", "key2");
String resultAnd = jedis.get("resultAnd");
assertEquals("\u0040", resultAnd);
jedis.bitop(BitOP.OR, "resultOr", "key1", "key2");
String resultOr = jedis.get("resultOr");
assertEquals("\u0064", resultOr);
jedis.bitop(BitOP.XOR, "resultXor", "key1", "key2");
String resultXor = jedis.get("resultXor");
assertEquals("\u0024", resultXor);
}
@Test
public void bitOpNot() {
jedis.setbit("key", 0, true);
jedis.setbit("key", 4, true);
jedis.bitop(BitOP.NOT, "resultNot", "key");
String resultNot = jedis.get("resultNot");
assertEquals("\u0077", resultNot);
}
@Test
public void bitOpBinary() {
byte[] dest = {0x0};
byte[] key1 = {0x1};
byte[] key2 = {0x2};
jedis.set(key1, new byte[]{0x6});
jedis.set(key2, new byte[]{0x3});
jedis.bitop(BitOP.AND, dest, key1, key2);
assertArrayEquals(new byte[]{0x2}, jedis.get(dest));
jedis.bitop(BitOP.OR, dest, key1, key2);
assertArrayEquals(new byte[]{0x7}, jedis.get(dest));
jedis.bitop(BitOP.XOR, dest, key1, key2);
assertArrayEquals(new byte[]{0x5}, jedis.get(dest));
jedis.setbit(key1, 0, true);
jedis.bitop(BitOP.NOT, dest, key1);
assertArrayEquals(new byte[]{0x79}, jedis.get(dest));
}
@Test
public void bitOpNotMultiSourceShouldFail() {
Assertions.assertThrows(JedisDataException.class, () -> {
jedis.bitop(BitOP.NOT, "dest", "src1", "src2");
});
}
@Test
public void testBitfield() {
List<Long> responses = jedis.bitfield("mykey", "INCRBY", "i5", "100", "1", "GET", "u4", "0");
assertEquals(1L, responses.get(0).longValue());
assertEquals(0L, responses.get(1).longValue());
}
@Test
public void testBitfieldReadonly() {
List<Long> responses = jedis.bitfield("mykey", "INCRBY", "i5", "100", "1", "GET", "u4", "0");
assertEquals(1L, responses.get(0).longValue());
assertEquals(0L, responses.get(1).longValue());
List<Long> responses2 = jedis.bitfieldReadonly("mykey", "GET", "i5", "100");
assertEquals(1L, responses2.get(0).longValue());
try {
jedis.bitfieldReadonly("mykey", "INCRBY", "i5", "100", "1", "GET", "u4", "0");
fail("Readonly command shouldn't allow INCRBY");
} catch (JedisDataException e) {
}
}
@Test
public void testBinaryBitfield() {
List<Long> responses = jedis.bitfield(SafeEncoder.encode("mykey"),
SafeEncoder.encode("INCRBY"), SafeEncoder.encode("i5"), SafeEncoder.encode("100"),
SafeEncoder.encode("1"), SafeEncoder.encode("GET"), SafeEncoder.encode("u4"),
SafeEncoder.encode("0"));
assertEquals(1L, responses.get(0).longValue());
assertEquals(0L, responses.get(1).longValue());
}
@Test
public void testBinaryBitfieldReadonly() {
List<Long> responses = jedis.bitfield("mykey", "INCRBY", "i5", "100", "1", "GET", "u4", "0");
assertEquals(1L, responses.get(0).longValue());
assertEquals(0L, responses.get(1).longValue());
List<Long> responses2 = jedis.bitfieldReadonly(SafeEncoder.encode("mykey"),
SafeEncoder.encode("GET"), SafeEncoder.encode("i5"), SafeEncoder.encode("100"));
assertEquals(1L, responses2.get(0).longValue());
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiff() {
// Use single-byte values for simplicity
byte[] key1 = new byte[] { (byte) 0b00000111 }; // bits 0,1,2 set
byte[] key2 = new byte[] { (byte) 0b00000010 }; // bit 1 set
byte[] key3 = new byte[] { (byte) 0b00000100 }; // bit 2 set
String destKey = "resultDiff";
// Set keys using byte arrays
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// DIFF(key1, key2, key3) = key1 AND NOT(key2 OR key3)
// key1 = 00000111 (bits 0,1,2 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// key2 OR key3 = 00000110 (bits 1,2 set)
// NOT(key2 OR key3) = 11111001 (all bits except 1,2 set)
// key1 AND NOT(key2 OR key3) = 00000001 (only bit 0 set)
jedis.bitop(BitOP.DIFF, destKey, "key1", "key2", "key3");
// Get result as bytes
byte[] resultBytes = jedis.get(destKey).getBytes();
// Expected result: 00000001 (only bit 0 set)
byte[] expectedBytes = new byte[] { (byte) 0b00000001 };
// Verify the result
assertArrayEquals(expectedBytes, resultBytes);
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiff1() {
// Use single-byte values for simplicity
byte[] key1 = new byte[] { (byte) 0x07 }; // 00000111 - bits 0,1,2 set
byte[] key2 = new byte[] { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = new byte[] { (byte) 0x04 }; // 00000100 - bit 2 set
String destKey = "resultDiff1";
// Set keys using byte arrays
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// DIFF1(key1, key2, key3) = NOT(key1) AND (key2 OR key3)
// key1 = 00000111 (bits 0,1,2 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// key2 OR key3 = 00000110 (bits 1,2 set)
// NOT(key1) = 11111000 (all bits except 0,1,2 set)
// NOT(key1) AND (key2 OR key3) = 00000000 (no bits set)
jedis.bitop(BitOP.DIFF1, destKey, "key1", "key2", "key3");
// Get result as bytes
byte[] resultBytes = jedis.get(destKey).getBytes();
// Expected result: 00000000 (no bits set)
byte[] expectedBytes = new byte[] { (byte) 0x00 };
// Verify the result
assertArrayEquals(expectedBytes, resultBytes);
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpAndor() {
// Use single-byte values for simplicity
byte[] key1 = new byte[] { (byte) 0x07 }; // 00000111 - bits 0,1,2 set
byte[] key2 = new byte[] { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = new byte[] { (byte) 0x04 }; // 00000100 - bit 2 set
String destKey = "resultAndor";
// Set keys using byte arrays
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// ANDOR(key1, key2, key3) = key1 AND (key2 OR key3)
// key1 = 00000111 (bits 0,1,2 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// key2 OR key3 = 00000110 (bits 1,2 set)
// key1 AND (key2 OR key3) = 00000110 (bits 1,2 set)
jedis.bitop(BitOP.ANDOR, destKey, "key1", "key2", "key3");
// Get result as bytes
byte[] resultBytes = jedis.get(destKey).getBytes();
// Expected result: 00000110 (bits 1,2 set)
byte[] expectedBytes = new byte[] { (byte) 0x06 };
// Verify the result
assertArrayEquals(expectedBytes, resultBytes);
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpOne() {
// Use single-byte values for simplicity
byte[] key1 = new byte[] { (byte) 0x01 }; // 00000001 - bit 0 set
byte[] key2 = new byte[] { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = new byte[] { (byte) 0x04 }; // 00000100 - bit 2 set
String destKey = "resultOne";
// Set keys using byte arrays
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// ONE(key1, key2, key3) = bits set in exactly one of the inputs
// key1 = 00000001 (bit 0 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// Result = 00000111 (bits 0,1,2 set - each in exactly one input)
jedis.bitop(BitOP.ONE, destKey, "key1", "key2", "key3");
// Get result as bytes
byte[] resultBytes = jedis.get(destKey).getBytes();
// Expected result: 00000111 (bits 0,1,2 set)
byte[] expectedBytes = new byte[] { (byte) 0x07 };
// Verify the result
assertArrayEquals(expectedBytes, resultBytes);
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiffBinary() {
byte[] key1 = { (byte) 0x07 }; // 00000111 - bits 0,1,2 set
byte[] key2 = { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = { (byte) 0x04 }; // 00000100 - bit 2 set
byte[] dest = "resultDiffBinary".getBytes();
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// DIFF(key1, key2, key3) = key1 AND NOT(key2 OR key3)
// key1 = 00000111 (bits 0,1,2 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// key2 OR key3 = 00000110 (bits 1,2 set)
// NOT(key2 OR key3) = 11111001 (all bits except 1,2 set)
// key1 AND NOT(key2 OR key3) = 00000001 (only bit 0 set)
jedis.bitop(BitOP.DIFF, dest, "key1".getBytes(), "key2".getBytes(), "key3".getBytes());
// Expected result: 00000001 (only bit 0 set)
byte[] expectedBytes = new byte[] { (byte) 0x01 };
// Verify the result
assertArrayEquals(expectedBytes, jedis.get(dest));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiff1Binary() {
byte[] key1 = { (byte) 0x07 }; // 00000111 - bits 0,1,2 set
byte[] key2 = { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = { (byte) 0x04 }; // 00000100 - bit 2 set
byte[] dest = "resultDiff1Binary".getBytes();
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// DIFF1(key1, key2, key3) = NOT(key1) AND (key2 OR key3)
// key1 = 00000111 (bits 0,1,2 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// key2 OR key3 = 00000110 (bits 1,2 set)
// NOT(key1) = 11111000 (all bits except 0,1,2 set)
// NOT(key1) AND (key2 OR key3) = 00000000 (no bits set)
jedis.bitop(BitOP.DIFF1, dest, "key1".getBytes(), "key2".getBytes(), "key3".getBytes());
// Expected result: 00000000 (no bits set)
byte[] expectedBytes = new byte[] { (byte) 0x00 };
// Verify the result
assertArrayEquals(expectedBytes, jedis.get(dest));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpAndorBinary() {
byte[] key1 = { (byte) 0x07 }; // 00000111 - bits 0,1,2 set
byte[] key2 = { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = { (byte) 0x04 }; // 00000100 - bit 2 set
byte[] dest = "resultAndorBinary".getBytes();
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// ANDOR(key1, key2, key3) = key1 AND (key2 OR key3)
// key1 = 00000111 (bits 0,1,2 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// key2 OR key3 = 00000110 (bits 1,2 set)
// key1 AND (key2 OR key3) = 00000110 (bits 1,2 set)
jedis.bitop(BitOP.ANDOR, dest, "key1".getBytes(), "key2".getBytes(), "key3".getBytes());
// Expected result: 00000110 (bits 1,2 set)
byte[] expectedBytes = new byte[] { (byte) 0x06 };
// Verify the result
assertArrayEquals(expectedBytes, jedis.get(dest));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpOneBinary() {
byte[] key1 = { (byte) 0x01 }; // 00000001 - bit 0 set
byte[] key2 = { (byte) 0x02 }; // 00000010 - bit 1 set
byte[] key3 = { (byte) 0x04 }; // 00000100 - bit 2 set
byte[] dest = "resultOneBinary".getBytes();
jedis.set("key1".getBytes(), key1);
jedis.set("key2".getBytes(), key2);
jedis.set("key3".getBytes(), key3);
// ONE(key1, key2, key3) = bits set in exactly one of the inputs
// key1 = 00000001 (bit 0 set)
// key2 = 00000010 (bit 1 set)
// key3 = 00000100 (bit 2 set)
// Result = 00000111 (bits 0,1,2 set - each in exactly one input)
jedis.bitop(BitOP.ONE, dest, "key1".getBytes(), "key2".getBytes(), "key3".getBytes());
// Expected result: 00000111 (bits 0,1,2 set)
byte[] expectedBytes = new byte[] { (byte) 0x07 };
// Verify the result
assertArrayEquals(expectedBytes, jedis.get(dest));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiffSingleSourceShouldFail() {
assertThrows(JedisDataException.class, () -> jedis.bitop(BitOP.DIFF, "dest", "src1"));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiff1SingleSourceShouldFail() {
assertThrows(JedisDataException.class, () -> jedis.bitop(BitOP.DIFF1, "dest", "src1"));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpAndorSingleSourceShouldFail() {
assertThrows(JedisDataException.class, () -> jedis.bitop(BitOP.ANDOR, "dest", "src1"));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiffBinarySingleSourceShouldFail() {
byte[] dest = "dest".getBytes();
byte[] src1 = "src1".getBytes();
assertThrows(JedisDataException.class, () -> jedis.bitop(BitOP.DIFF, dest, src1));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpDiff1BinarySingleSourceShouldFail() {
byte[] dest = "dest".getBytes();
byte[] src1 = "src1".getBytes();
assertThrows(JedisDataException.class, () -> jedis.bitop(BitOP.DIFF1, dest, src1));
}
@Test
@SinceRedisVersion("8.1.240")
public void bitOpAndorBinarySingleSourceShouldFail() {
byte[] dest = "dest".getBytes();
byte[] src1 = "src1".getBytes();
assertThrows(JedisDataException.class, () -> jedis.bitop(BitOP.ANDOR, dest, src1));
}
}