]
+ int rc = ResultCode.CLIENT_ERROR;
+ String msg = response;
+
+ try {
+ String[] list = response.split(":");
+ String s = list[0];
+
+ if (s.regionMatches(true, 0, "FAIL", 0, 4) ||
+ s.regionMatches(true, 0, "ERROR", 0, 5)) {
+
+ if (list.length >= 3) {
+ msg = list[2].trim();
+ s = list[1].trim();
+
+ if (! s.isEmpty()) {
+ rc = Integer.parseInt(s);
+ }
+ }
+ else if (list.length == 2) {
+ s = list[1].trim();
+
+ if (! s.isEmpty()) {
+ try {
+ rc = Integer.parseInt(s);
+ }
+ catch (Throwable t) {
+ // Some error strings omit the code and just have a message.
+ msg = s;
+ }
+ }
+ }
+ }
+ }
+ catch (Throwable t) {
+ }
+ finally {
+ this.code = rc;
+ this.message = msg;
+ }
+ }
+ }
}
diff --git a/client/src/com/aerospike/client/async/AsyncBatch.java b/client/src/com/aerospike/client/async/AsyncBatch.java
index 5893e06ae..22d4fecf6 100644
--- a/client/src/com/aerospike/client/async/AsyncBatch.java
+++ b/client/src/com/aerospike/client/async/AsyncBatch.java
@@ -16,7 +16,6 @@
*/
package com.aerospike.client.async;
-import java.util.ArrayList;
import java.util.List;
import com.aerospike.client.AerospikeClient;
diff --git a/client/src/com/aerospike/client/cluster/Connection.java b/client/src/com/aerospike/client/cluster/Connection.java
index 1a420177f..0ed06c28a 100644
--- a/client/src/com/aerospike/client/cluster/Connection.java
+++ b/client/src/com/aerospike/client/cluster/Connection.java
@@ -327,6 +327,8 @@ public void close() {
}
public static final class ReadTimeout extends RuntimeException {
+ private static final long serialVersionUID = 1L;
+
public final byte[] buffer;
public final int offset;
public final int length;
diff --git a/client/src/com/aerospike/client/command/BatchAttr.java b/client/src/com/aerospike/client/command/BatchAttr.java
index 487a742d0..7f063537e 100644
--- a/client/src/com/aerospike/client/command/BatchAttr.java
+++ b/client/src/com/aerospike/client/command/BatchAttr.java
@@ -48,7 +48,7 @@ public BatchAttr(Policy policy, int rattr) {
public BatchAttr(Policy policy, int rattr, Operation[] ops) {
setRead(policy);
- this.readAttr |= rattr;
+ this.readAttr = rattr;
if (ops != null) {
adjustRead(ops);
diff --git a/client/src/com/aerospike/client/command/Command.java b/client/src/com/aerospike/client/command/Command.java
index 0e573594c..7ae5e92d1 100644
--- a/client/src/com/aerospike/client/command/Command.java
+++ b/client/src/com/aerospike/client/command/Command.java
@@ -242,7 +242,17 @@ public final void setExists(Policy policy, Key key) {
end();
}
- private final void setRead(Policy policy, Key key) {
+ public final void setRead(Policy policy, Key key, String[] binNames) {
+ int readAttr = Command.INFO1_READ;
+ int opCount = 0;
+
+ if (binNames != null && binNames.length > 0) {
+ opCount = binNames.length;
+ }
+ else {
+ readAttr |= Command.INFO1_GET_ALL;
+ }
+
begin();
int fieldCount = estimateKeySize(policy, key);
@@ -250,45 +260,27 @@ private final void setRead(Policy policy, Key key) {
dataOffset += policy.filterExp.size();
fieldCount++;
}
+
+ if (opCount != 0) {
+ for (String binName : binNames) {
+ estimateOperationSize(binName);
+ }
+ }
+
sizeBuffer();
- writeHeaderRead(policy, serverTimeout, Command.INFO1_READ | Command.INFO1_GET_ALL, 0, 0, fieldCount, 0);
+ writeHeaderRead(policy, serverTimeout, readAttr, 0, 0, fieldCount, opCount);
writeKey(policy, key);
if (policy.filterExp != null) {
policy.filterExp.write(this);
}
- end();
- }
-
- public final void setRead(Policy policy, Key key, String[] binNames) {
- if (binNames != null) {
- begin();
- int fieldCount = estimateKeySize(policy, key);
-
- if (policy.filterExp != null) {
- dataOffset += policy.filterExp.size();
- fieldCount++;
- }
-
- for (String binName : binNames) {
- estimateOperationSize(binName);
- }
- sizeBuffer();
- writeHeaderRead(policy, serverTimeout, Command.INFO1_READ, 0, 0, fieldCount, binNames.length);
- writeKey(policy, key);
-
- if (policy.filterExp != null) {
- policy.filterExp.write(this);
- }
+ if (opCount != 0) {
for (String binName : binNames) {
writeOperation(binName, Operation.Type.READ);
}
- end();
- }
- else {
- setRead(policy, key);
}
+ end();
}
public final void setRead(Policy policy, BatchRead br) {
@@ -852,7 +844,13 @@ public final void setBatchOperate(
}
if (br.binNames != null) {
- writeBatchBinNames(key, br.binNames, attr, attr.filterExp);
+ if (br.binNames.length > 0) {
+ writeBatchBinNames(key, br.binNames, attr, attr.filterExp);
+ }
+ else {
+ attr.adjustRead(true);
+ writeBatchRead(key, attr, attr.filterExp, 0);
+ }
}
else if (br.ops != null) {
attr.adjustRead(br.ops);
@@ -1331,9 +1329,8 @@ public final void setScan(
}
// Clusters that support partition queries also support not sending partition done messages.
- int infoAttr = cluster.hasPartitionQuery? Command.INFO3_PARTITION_DONE : 0;
int operationCount = (binNames == null)? 0 : binNames.length;
- writeHeaderRead(policy, totalTimeout, readAttr, 0, infoAttr, fieldCount, operationCount);
+ writeHeaderRead(policy, totalTimeout, readAttr, 0, Command.INFO3_PARTITION_DONE, fieldCount, operationCount);
if (namespace != null) {
writeField(namespace, FieldType.NAMESPACE);
@@ -1583,7 +1580,7 @@ else if (qp.expectedDuration == QueryDuration.LONG_RELAX_AP) {
writeAttr |= Command.INFO2_RELAX_AP_LONG_QUERY;
}
- int infoAttr = isNew? Command.INFO3_PARTITION_DONE : 0;
+ int infoAttr = (isNew || filter == null)? Command.INFO3_PARTITION_DONE : 0;
writeHeaderRead(policy, totalTimeout, readAttr, writeAttr, infoAttr, fieldCount, operationCount);
}
diff --git a/client/src/com/aerospike/client/metrics/LatencyBuckets.java b/client/src/com/aerospike/client/metrics/LatencyBuckets.java
index db61b2089..ca1b9df3e 100644
--- a/client/src/com/aerospike/client/metrics/LatencyBuckets.java
+++ b/client/src/com/aerospike/client/metrics/LatencyBuckets.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2023 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -33,7 +33,7 @@ public final class LatencyBuckets {
*
* @param latencyColumns number of latency buckets
* @param latencyShift power of 2 multiple between each range bucket in latency histograms starting at bucket 3.
- * The first 2 buckets are "<=1ms" and ">1ms".
+ * The first 2 buckets are "<=1ms" and ">1ms".
*/
public LatencyBuckets(int latencyColumns, int latencyShift) {
this.latencyShift = latencyShift;
diff --git a/client/src/com/aerospike/client/metrics/LatencyType.java b/client/src/com/aerospike/client/metrics/LatencyType.java
index 01fda6aa2..625a9ef07 100644
--- a/client/src/com/aerospike/client/metrics/LatencyType.java
+++ b/client/src/com/aerospike/client/metrics/LatencyType.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2023 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -16,6 +16,9 @@
*/
package com.aerospike.client.metrics;
+/**
+ * Latency group type.
+ */
public enum LatencyType {
CONN,
WRITE,
diff --git a/client/src/com/aerospike/client/metrics/MetricsPolicy.java b/client/src/com/aerospike/client/metrics/MetricsPolicy.java
index df6e285a3..b6119027e 100644
--- a/client/src/com/aerospike/client/metrics/MetricsPolicy.java
+++ b/client/src/com/aerospike/client/metrics/MetricsPolicy.java
@@ -69,7 +69,7 @@ public final class MetricsPolicy {
/**
* Power of 2 multiple between each range bucket in latency histograms starting at column 3. The bucket units
- * are in milliseconds. The first 2 buckets are "<=1ms" and ">1ms". Examples:
+ * are in milliseconds. The first 2 buckets are "<=1ms" and ">1ms". Examples:
* {@code
* // latencyColumns=7 latencyShift=1
* <=1ms >1ms >2ms >4ms >8ms >16ms >32ms
diff --git a/client/src/com/aerospike/client/policy/ReadModeSC.java b/client/src/com/aerospike/client/policy/ReadModeSC.java
index ff2c6dbd3..eed429b13 100644
--- a/client/src/com/aerospike/client/policy/ReadModeSC.java
+++ b/client/src/com/aerospike/client/policy/ReadModeSC.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2021 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -24,25 +24,25 @@
public enum ReadModeSC {
/**
* Ensures this client will only see an increasing sequence of record versions.
- * Server only reads from master. This is the default.
+ * Client only reads from master. This is the default.
*/
SESSION,
/**
- * Ensures ALL clients will only see an increasing sequence of record versions.
- * Server only reads from master.
+ * Ensures all clients will only see an increasing sequence of record versions.
+ * Client only reads from master.
*/
LINEARIZE,
/**
- * Server may read from master or any full (non-migrating) replica.
+ * Client may read from master or any full (non-migrating) replica.
* Increasing sequence of record versions is not guaranteed.
*/
ALLOW_REPLICA,
/**
- * Server may read from master or any full (non-migrating) replica or from unavailable
- * partitions. Increasing sequence of record versions is not guaranteed.
+ * Client may read from master or any full (non-migrating) replica or from unavailable
+ * partitions. Increasing sequence of record versions is not guaranteed.
*/
ALLOW_UNAVAILABLE
}
diff --git a/client/src/com/aerospike/client/task/IndexTask.java b/client/src/com/aerospike/client/task/IndexTask.java
index 98852e603..bdc469127 100644
--- a/client/src/com/aerospike/client/task/IndexTask.java
+++ b/client/src/com/aerospike/client/task/IndexTask.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2021 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -18,6 +18,7 @@
import com.aerospike.client.AerospikeException;
import com.aerospike.client.Info;
+import com.aerospike.client.ResultCode;
import com.aerospike.client.cluster.Cluster;
import com.aerospike.client.cluster.Node;
import com.aerospike.client.policy.Policy;
@@ -92,13 +93,14 @@ public static int parseStatusResponse(String command, String response, boolean i
int index = response.indexOf(find);
if (index < 0) {
- if (response.indexOf("FAIL:201") >= 0 || response.indexOf("FAIL:203") >= 0) {
- // Index not found or not readable.
+ Info.Error error = new Info.Error(response);
+
+ if (error.code == ResultCode.INDEX_NOTFOUND || error.code == ResultCode.INDEX_NOTREADABLE) {
return Task.NOT_FOUND;
}
else {
// Throw exception immediately.
- throw new AerospikeException(command + " failed: " + response);
+ throw new AerospikeException(error.code, command + " failed: " + error.message);
}
}
@@ -113,7 +115,9 @@ public static int parseStatusResponse(String command, String response, boolean i
}
else {
// Check if index has been dropped.
- if (response.indexOf("FAIL:201") < 0) {
+ Info.Error error = new Info.Error(response);
+
+ if (error.code != ResultCode.INDEX_NOTFOUND) {
// Index still exists.
return Task.IN_PROGRESS;
}
diff --git a/client/src/com/aerospike/client/util/Crypto.java b/client/src/com/aerospike/client/util/Crypto.java
index 22037c7c3..63ca13556 100644
--- a/client/src/com/aerospike/client/util/Crypto.java
+++ b/client/src/com/aerospike/client/util/Crypto.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2021 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
diff --git a/examples/pom.xml b/examples/pom.xml
index ad3807c82..b0ca631da 100644
--- a/examples/pom.xml
+++ b/examples/pom.xml
@@ -6,7 +6,7 @@
com.aerospike
aerospike-parent
- 8.1.1
+ 8.1.2
aerospike-examples
jar
diff --git a/pom.xml b/pom.xml
index 6c462c20b..416961f65 100644
--- a/pom.xml
+++ b/pom.xml
@@ -5,7 +5,7 @@
com.aerospike
aerospike-parent
aerospike-parent
- 8.1.1
+ 8.1.2
pom
https://github.com/aerospike/aerospike-client-java
@@ -39,12 +39,12 @@
2.18.1
3.2.0
- 4.1.108.Final
+ 4.1.110.Final
2.0.62.Final
1.59.0
3.0.1
0.4
- 1.7.0
+ 1.8.0
4.13.1
diff --git a/proxy/pom.xml b/proxy/pom.xml
index 84c64ea2d..8d980ca2c 100644
--- a/proxy/pom.xml
+++ b/proxy/pom.xml
@@ -6,7 +6,7 @@
com.aerospike
aerospike-parent
- 8.1.1
+ 8.1.2
aerospike-proxy-client
jar
diff --git a/proxy/src/com/aerospike/client/proxy/AerospikeClientProxy.java b/proxy/src/com/aerospike/client/proxy/AerospikeClientProxy.java
index 7274c0f5f..889e1c502 100644
--- a/proxy/src/com/aerospike/client/proxy/AerospikeClientProxy.java
+++ b/proxy/src/com/aerospike/client/proxy/AerospikeClientProxy.java
@@ -1335,7 +1335,9 @@ public void get(EventLoop eventLoop, RecordArrayListener listener, BatchPolicy p
policy = batchPolicyDefault;
}
- CommandProxy command = new BatchProxy.GetArrayCommand(executor, policy, listener, keys, binNames, null, Command.INFO1_READ, false);
+ int readAttr = (binNames == null || binNames.length == 0)? Command.INFO1_READ | Command.INFO1_GET_ALL : Command.INFO1_READ;
+
+ CommandProxy command = new BatchProxy.GetArrayCommand(executor, policy, listener, keys, binNames, null, readAttr, false);
command.execute();
}
@@ -1363,7 +1365,9 @@ public void get(EventLoop eventLoop, RecordSequenceListener listener, BatchPolic
policy = batchPolicyDefault;
}
- CommandProxy command = new BatchProxy.GetSequenceCommand(executor, policy, listener, keys, binNames, null, Command.INFO1_READ, false);
+ int readAttr = (binNames == null || binNames.length == 0)? Command.INFO1_READ | Command.INFO1_GET_ALL : Command.INFO1_READ;
+
+ CommandProxy command = new BatchProxy.GetSequenceCommand(executor, policy, listener, keys, binNames, null, readAttr, false);
command.execute();
}
diff --git a/test/pom.xml b/test/pom.xml
index 796c09656..48de52c7b 100644
--- a/test/pom.xml
+++ b/test/pom.xml
@@ -6,7 +6,7 @@
com.aerospike
aerospike-parent
- 8.1.1
+ 8.1.2
aerospike-client-test
jar
diff --git a/test/src/com/aerospike/test/sync/basic/TestBatch.java b/test/src/com/aerospike/test/sync/basic/TestBatch.java
index 90afd5765..82b7083ed 100644
--- a/test/src/com/aerospike/test/sync/basic/TestBatch.java
+++ b/test/src/com/aerospike/test/sync/basic/TestBatch.java
@@ -144,6 +144,30 @@ public void batchReads () {
}
}
+ @Test
+ public void batchReadsEmptyBinNames() {
+ Key[] keys = new Key[Size];
+ for (int i = 0; i < Size; i++) {
+ keys[i] = new Key(args.namespace, args.set, KeyPrefix + (i + 1));
+ }
+
+ String[] binNames = new String[] {};
+ Record[] records = client.get(null, keys, binNames);
+ assertEquals(Size, records.length);
+
+ for (int i = 0; i < records.length; i++) {
+ Key key = keys[i];
+ Record record = records[i];
+
+ if (i != 5) {
+ assertBinEqual(key, record, BinName, ValuePrefix + (i + 1));
+ }
+ else {
+ assertBinEqual(key, record, BinName, i + 1);
+ }
+ }
+ }
+
@Test
public void batchReadHeaders () {
Key[] keys = new Key[Size];
@@ -168,7 +192,7 @@ public void batchReadHeaders () {
}
@Test
- public void batchReadComplex () {
+ public void batchReadComplex() {
// Batch allows multiple namespaces in one call, but example test environment may only have one namespace.
// bin * 8
@@ -179,7 +203,7 @@ public void batchReadComplex () {
List records = new ArrayList();
records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 1), bins));
records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 2), true));
- records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 3), true));
+ records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 3), new String[] {}));
records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 4), false));
records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 5), true));
records.add(new BatchRead(new Key(args.namespace, args.set, KeyPrefix + 6), ops));
diff --git a/test/src/com/aerospike/test/sync/basic/TestPutGet.java b/test/src/com/aerospike/test/sync/basic/TestPutGet.java
index e2f99cd42..58830abee 100644
--- a/test/src/com/aerospike/test/sync/basic/TestPutGet.java
+++ b/test/src/com/aerospike/test/sync/basic/TestPutGet.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2023 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -41,6 +41,11 @@ public void putGet() {
Record record = client.get(null, key);
assertBinEqual(key, record, bin1);
assertBinEqual(key, record, bin2);
+
+ // Test empty binNames array.
+ record = client.get(null, key, new String[] {});
+ assertBinEqual(key, record, bin1);
+ assertBinEqual(key, record, bin2);
}
@Test
diff --git a/test/src/com/aerospike/test/sync/basic/TestScan.java b/test/src/com/aerospike/test/sync/basic/TestScan.java
index d0b3e1762..c6a29289f 100644
--- a/test/src/com/aerospike/test/sync/basic/TestScan.java
+++ b/test/src/com/aerospike/test/sync/basic/TestScan.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2023 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -35,6 +35,10 @@ public class TestScan extends TestSync implements ScanCallback {
public void scanParallel() {
ScanPolicy policy = new ScanPolicy();
client.scanAll(policy, args.namespace, args.set, this);
+
+ // Test empty binNames.
+ String[] binNames = new String[] {};
+ client.scanAll(policy, args.namespace, args.set, this, binNames);
}
@Test
diff --git a/test/src/com/aerospike/test/sync/basic/TestServerInfo.java b/test/src/com/aerospike/test/sync/basic/TestServerInfo.java
index f3f52424a..2d39bf2bf 100644
--- a/test/src/com/aerospike/test/sync/basic/TestServerInfo.java
+++ b/test/src/com/aerospike/test/sync/basic/TestServerInfo.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2012-2023 Aerospike, Inc.
+ * Copyright 2012-2024 Aerospike, Inc.
*
* Portions may be licensed to Aerospike, Inc. under one or more contributor
* license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
@@ -17,9 +17,11 @@
package com.aerospike.test.sync.basic;
import static org.junit.Assert.assertNotNull;
+import static org.junit.Assert.assertEquals;
import java.util.Map;
+import com.aerospike.client.ResultCode;
import org.junit.Test;
import com.aerospike.client.Info;
@@ -76,4 +78,33 @@ private void LogNameValueTokens(String tokens) {
assertNotNull(value);
}
}
+
+ @Test
+ public void errorResponse() {
+ Info.Error error;
+
+ error = new Info.Error("FaIL:201:index not found");
+ assertEquals(error.code, 201);
+ assertEquals(error.message, "index not found");
+
+ error = new Info.Error("ERRor:201:index not found");
+ assertEquals(error.code, 201);
+ assertEquals(error.message, "index not found");
+
+ error = new Info.Error("error::index not found ");
+ assertEquals(error.code, ResultCode.CLIENT_ERROR);
+ assertEquals(error.message, "index not found");
+
+ error = new Info.Error("error: index not found ");
+ assertEquals(error.code, ResultCode.CLIENT_ERROR);
+ assertEquals(error.message, "index not found");
+
+ error = new Info.Error("error:99");
+ assertEquals(error.code, 99);
+ assertEquals(error.message, "error:99");
+
+ error = new Info.Error("generic message");
+ assertEquals(error.code, ResultCode.CLIENT_ERROR);
+ assertEquals(error.message, "generic message");
+ }
}
diff --git a/test/src/com/aerospike/test/sync/query/TestQueryString.java b/test/src/com/aerospike/test/sync/query/TestQueryString.java
index 92a2ea96a..8ef779f0c 100644
--- a/test/src/com/aerospike/test/sync/query/TestQueryString.java
+++ b/test/src/com/aerospike/test/sync/query/TestQueryString.java
@@ -95,4 +95,33 @@ public void queryString() {
rs.close();
}
}
+
+ @Test
+ public void queryStringEmptyBinName() {
+ String filter = valuePrefix + 3;
+
+ Statement stmt = new Statement();
+ stmt.setNamespace(args.namespace);
+ stmt.setSetName(args.set);
+ stmt.setBinNames(new String[] {});
+ stmt.setFilter(Filter.equal(binName, filter));
+
+ RecordSet rs = client.query(null, stmt);
+
+ try {
+ int count = 0;
+
+ while (rs.next()) {
+ Record record = rs.getRecord();
+ String result = record.getString(binName);
+ assertEquals(filter, result);
+ count++;
+ }
+
+ assertNotEquals(0, count);
+ }
+ finally {
+ rs.close();
+ }
+ }
}