Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
8ab6ed8
Changes to run tests for multihash collections
SrinikhilReddy Dec 14, 2020
d5d3b47
Changes.
SrinikhilReddy Jan 7, 2021
5459104
Parititionkey builder model as suggested by Mo.
SrinikhilReddy Jan 15, 2021
e1c3001
Apply suggestions from code review
SrinikhilReddy Jan 15, 2021
29f2cea
Merge remote-tracking branch 'upstream/master'
SrinikhilReddy Feb 16, 2021
dd77aaf
Pushing to remote. Cleanup
SrinikhilReddy Feb 16, 2021
a65e716
Merge branch 'master' of https://github.com/Azure/azure-sdk-for-java
SrinikhilReddy Feb 16, 2021
58c2746
commit local changes
SrinikhilReddy Apr 6, 2021
4cb8cb1
Merge branch 'master' of https://github.com/Azure/azure-sdk-for-java
SrinikhilReddy Apr 6, 2021
036b7ec
Address PR comments.
SrinikhilReddy Apr 6, 2021
f2905f4
Address code comments.
SrinikhilReddy Apr 6, 2021
24610d6
Resolve code comments
SrinikhilReddy Apr 6, 2021
5a741c5
Add tests.
SrinikhilReddy Apr 8, 2021
29bcfc9
Well, when does this stop.
SrinikhilReddy Apr 8, 2021
14bd7a2
No more comments.
SrinikhilReddy Apr 8, 2021
8f77d0d
Update CosmosMultiHashTest.java
SrinikhilReddy May 6, 2021
741ce81
Merge remote-tracking branch 'upstream/master'
SrinikhilReddy May 6, 2021
0b45b66
Fix test failures
SrinikhilReddy May 10, 2021
22fe137
Merge remote-tracking branch 'upstream/master'
SrinikhilReddy May 10, 2021
5bf9d26
Fixing code suggestions
SrinikhilReddy May 11, 2021
0c67ba6
Merge remote-tracking branch 'upstream/master'
SrinikhilReddy May 11, 2021
00d5f8e
Resolve code comments
SrinikhilReddy May 12, 2021
f008411
Merge remote-tracking branch 'upstream/master'
SrinikhilReddy May 12, 2021
28926cf
Merge branch 'master' of https://github.com/Azure/azure-sdk-for-java
SrinikhilReddy May 20, 2021
ed325f9
Merge remote-tracking branch 'upstream/master'
SrinikhilReddy Jun 10, 2021
b6e452e
Fixing a test
SrinikhilReddy Jun 10, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -273,7 +273,7 @@ public static class A_IMHeaderValues {
}

public static class Versions {
public static final String CURRENT_VERSION = "2018-12-31";
public static final String CURRENT_VERSION = "2020-07-15";
public static final String QUERY_VERSION = "1.0";
public static final String AZURE_COSMOS_PROPERTIES_FILE_NAME = "azure-cosmos.properties";

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1310,20 +1310,35 @@ public static PartitionKeyInternal extractPartitionKeyValueFromDocument(
InternalObjectNode document,
PartitionKeyDefinition partitionKeyDefinition) {
if (partitionKeyDefinition != null) {
String path = partitionKeyDefinition.getPaths().iterator().next();
List<String> parts = PathParser.getPathParts(path);
if (parts.size() >= 1) {
Object value = ModelBridgeInternal.getObjectByPathFromJsonSerializable(document, parts);
if (value == null || value.getClass() == ObjectNode.class) {
value = ModelBridgeInternal.getNonePartitionKey(partitionKeyDefinition);
}
switch (partitionKeyDefinition.getKind()) {
case HASH:
String path = partitionKeyDefinition.getPaths().iterator().next();
List<String> parts = PathParser.getPathParts(path);
if (parts.size() >= 1) {
Object value = ModelBridgeInternal.getObjectByPathFromJsonSerializable(document, parts);
if (value == null || value.getClass() == ObjectNode.class) {
value = ModelBridgeInternal.getNonePartitionKey(partitionKeyDefinition);
}

if (value instanceof PartitionKeyInternal) {
return (PartitionKeyInternal) value;
} else {
return PartitionKeyInternal.fromObjectArray(Collections.singletonList(value), false);
if (value instanceof PartitionKeyInternal) {
return (PartitionKeyInternal) value;
} else {
return PartitionKeyInternal.fromObjectArray(Collections.singletonList(value), false);
}
}
break;
case MULTI_HASH:
Object[] partitionKeyValues = new Object[partitionKeyDefinition.getPaths().size()];
for(int pathIter = 0 ; pathIter < partitionKeyDefinition.getPaths().size(); pathIter++){
String partitionPath = partitionKeyDefinition.getPaths().get(pathIter);
List<String> partitionPathParts = PathParser.getPathParts(partitionPath);
partitionKeyValues[pathIter] = ModelBridgeInternal.getObjectByPathFromJsonSerializable(document, partitionPathParts);
}
return PartitionKeyInternal.fromObjectArray(partitionKeyValues, false);

default:
throw new IllegalArgumentException("Unrecognized Partition kind: " + partitionKeyDefinition.getKind());
}
}
}

return null;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,8 @@ public class PartitionKeyInternalHelper {
(byte) 0x3F, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF,
(byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF, (byte) 0xFF});

private static final int HASH_V2_EPK_LENGTH = 32;

static byte[] uIntToBytes(UInt128 unit) {
ByteBuffer buffer = ByteBuffer.allocate(Long.BYTES * 2);
buffer.putLong(unit.low);
Expand Down Expand Up @@ -92,6 +94,31 @@ static public String getEffectivePartitionKeyForHashPartitioningV2(PartitionKeyI
}
}

static String getEffectivePartitionKeyForMultiHashPartitioning(PartitionKeyInternal partitionKeyInternal) {
StringBuilder stringBuilder = new StringBuilder(partitionKeyInternal.components.size() * HASH_V2_EPK_LENGTH);
for (int i = 0; i < partitionKeyInternal.components.size(); i++) {
try(ByteBufferOutputStream byteArrayBuffer = new ByteBufferOutputStream()) {
partitionKeyInternal.components.get(i).writeForHashingV2(byteArrayBuffer);

ByteBuffer byteBuffer = byteArrayBuffer.asByteBuffer();
UInt128 hashAsUnit128 = MurmurHash3_128.hash128(byteBuffer.array(), byteBuffer.limit());

byte[] hash = uIntToBytes(hashAsUnit128);
Bytes.reverse(hash);

// Reset 2 most significant bits, as max exclusive value is 'FF'.
// Plus one more just in case.
hash[0] &= 0x3F;

stringBuilder.append(HexConvert.bytesToHex(hash));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Q: How about allocate byte[] and then convert to hex at the end?
Purely from performance wise. .NET micro-benchmark will be useful for quick prototyping.

} catch (IOException e) {
throw new IllegalArgumentException(e);
}
}

return stringBuilder.toString();
}

static String getEffectivePartitionKeyForHashPartitioning(PartitionKeyInternal partitionKeyInternal) {
IPartitionKeyComponent[] truncatedComponents = new IPartitionKeyComponent[partitionKeyInternal.components.size()];

Expand Down Expand Up @@ -138,7 +165,7 @@ public static String getEffectivePartitionKeyString(PartitionKeyInternal partiti
return MaximumExclusiveEffectivePartitionKey;
}

if (partitionKeyInternal.components.size() < partitionKeyDefinition.getPaths().size()) {
if (partitionKeyInternal.components.size() < partitionKeyDefinition.getPaths().size() && partitionKeyDefinition.getKind() != PartitionKind.MULTI_HASH) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this to cover only for query?

If so do we need two variations?
Like what happens for point operations partial pk is specified?

throw new IllegalArgumentException(RMResources.TooFewPartitionKeyComponents);
}

Expand All @@ -161,6 +188,9 @@ public static String getEffectivePartitionKeyString(PartitionKeyInternal partiti
return getEffectivePartitionKeyForHashPartitioning(partitionKeyInternal);
}

case MULTI_HASH:
return getEffectivePartitionKeyForMultiHashPartitioning(partitionKeyInternal);

default:
return toHexEncodedBinaryString(partitionKeyInternal.components);
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.

package com.azure.cosmos.models;

import com.azure.cosmos.implementation.Undefined;
import com.azure.cosmos.implementation.routing.PartitionKeyInternal;
import com.azure.cosmos.util.Beta;
import com.azure.cosmos.util.Beta.SinceVersion;

import java.util.ArrayList;
import java.util.List;

@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public final class PartitionKeyBuilder {
private final List<Object> partitionKeyValues;

/**
* Constructor. CREATE a new instance of the PartitionKeyBuilder object.
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKeyBuilder() {
this.partitionKeyValues = new ArrayList<Object>();
}

/**
* Adds partition value of type string
* @param value The value of type string to be used as partition key
* @return The current PartitionKeyBuilder object
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKeyBuilder add(String value) {
this.partitionKeyValues.add(value);
return this;
}

/**
* Adds partition value of type double
* @param value The value of type double to be used as partition key
* @return The current PartitionKeyBuilder object
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKeyBuilder add(double value) {
this.partitionKeyValues.add(value);
return this;
}

/**
* Adds partition value of type boolean
* @param value The value of type boolean to be used as partition key
* @return The current PartitionKeyBuilder object
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKeyBuilder add(boolean value) {
this.partitionKeyValues.add(value);
return this;
}

/**
* Adds a null partition key value
* @return The current PartitionKeyBuilder object
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKeyBuilder addNullValue() {
this.partitionKeyValues.add(null);
return this;
}

/**
* Adds a None Partition Key
* @return The current PartitionKeyBuilder object
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKeyBuilder addNoneValue() {
this.partitionKeyValues.add(PartitionKey.NONE);
return this;
}

/**
* Builds a new instance of the type PartitionKey with the specified Partition Key values.
* @return PartitionKey object
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
public PartitionKey build() {
// Why these checks?
// These changes are being added for SDK to support multiple paths in a partition key.
//
// Currently, when a resource does not specify a value for the PartitionKey,
// we assign a temporary value `PartitionKey.None` and later discern whether
// it is a PartitionKey.Undefined or PartitionKey.Empty based on the Collection Type.
// We retain this behaviour for single path partition keys.
//
// For collections with multiple path keys, absence of a partition key values is
// always treated as a PartitionKey.Undefined.
if(this.partitionKeyValues.size() == 0) {
throw new IllegalArgumentException("No partition key value has been specified");
}

if(this.partitionKeyValues.size() == 1 && PartitionKey.NONE.equals(this.partitionKeyValues.get(0))) {
return PartitionKey.NONE;
}

PartitionKeyInternal partitionKeyInternal;
Object[] valueArray = new Object[this.partitionKeyValues.size()];
for(int i = 0; i < this.partitionKeyValues.size(); i++) {
Object val = this.partitionKeyValues.get(i);
if(PartitionKey.NONE.equals(val)) {
valueArray[i] = Undefined.value();
}
else {
valueArray[i] = val;
}
}

partitionKeyInternal = PartitionKeyInternal.fromObjectArray(valueArray, true);
return new PartitionKey(partitionKeyInternal);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,24 @@

package com.azure.cosmos.models;

import com.azure.cosmos.util.Beta;
import com.azure.cosmos.util.Beta.SinceVersion;

/**
* Specifies the partition scheme for an multiple-partitioned container in the Azure Cosmos DB database service.
*/
public enum PartitionKind {
/**
* The Partition of a item is calculated based on the hash value of the PartitionKey.
*/
HASH("Hash");
HASH("Hash"),

RANGE("Range"),
/**
* The Partition of a item is calculated based on the hash value of multiple PartitionKeys.
*/
@Beta(value = SinceVersion.V4_16_0, warningText = Beta.PREVIEW_SUBJECT_TO_CHANGE_WARNING)
MULTI_HASH("MultiHash");

PartitionKind(String overWireValue) {
this.overWireValue = overWireValue;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,13 +10,14 @@
import java.lang.annotation.Target;

import static java.lang.annotation.ElementType.CONSTRUCTOR;
import static java.lang.annotation.ElementType.FIELD;
import static java.lang.annotation.ElementType.METHOD;
import static java.lang.annotation.ElementType.PARAMETER;
import static java.lang.annotation.ElementType.TYPE;

@Documented
@Retention(RetentionPolicy.CLASS)
@Target({ TYPE, METHOD, PARAMETER, CONSTRUCTOR })
@Target({ TYPE, METHOD, PARAMETER, CONSTRUCTOR, FIELD })
@Inherited
/**
* Indicates functionality that is in preview and as such is subject to change in non-backwards compatible ways in future releases,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,8 @@
import com.azure.cosmos.models.IndexingPolicy;
import com.azure.cosmos.models.PartitionKey;
import com.azure.cosmos.models.PartitionKeyDefinition;
import com.azure.cosmos.models.PartitionKeyDefinitionVersion;
import com.azure.cosmos.models.PartitionKind;
import com.azure.cosmos.models.SqlQuerySpec;
import com.azure.cosmos.models.ThroughputProperties;
import com.azure.cosmos.rx.TestSuiteBase;
Expand All @@ -47,6 +49,7 @@
import java.util.Base64;
import java.util.List;
import java.util.UUID;
import java.util.ArrayList;

import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.fail;
Expand Down Expand Up @@ -764,6 +767,40 @@ public void readAllContainers() throws Exception{
assertThat(feedResponseIterator1.iterator().hasNext()).isTrue();
}

@Test(groups = { "emulator" }, timeOut = TIMEOUT)
public void crudMultiHashContainer() throws Exception {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also a test case covering the NONE explicitly for above variants.

String collectionName = UUID.randomUUID().toString();

PartitionKeyDefinition partitionKeyDefinition = new PartitionKeyDefinition();
partitionKeyDefinition.setKind(PartitionKind.MULTI_HASH);
partitionKeyDefinition.setVersion(PartitionKeyDefinitionVersion.V2);
ArrayList<String> paths = new ArrayList<>();
paths.add("/city");
paths.add("/zipcode");
partitionKeyDefinition.setPaths(paths);

CosmosContainerProperties containerProperties = getCollectionDefinition(collectionName, partitionKeyDefinition);

//MultiHash collection create
CosmosContainerResponse containerResponse = createdDatabase.createContainer(containerProperties);
validateContainerResponse(containerProperties, containerResponse);
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getKind() == PartitionKind.MULTI_HASH);
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getPaths().size() == paths.size());
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getPaths().get(0) == paths.get(0));
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getPaths().get(1) == paths.get(1));

//MultiHash collection read
CosmosContainer multiHashContainer = createdDatabase.getContainer(collectionName);
containerResponse = multiHashContainer.read();
validateContainerResponse(containerProperties, containerResponse);
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getKind() == PartitionKind.MULTI_HASH);
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getPaths().size() == paths.size());
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getPaths().get(0) == paths.get(0));
assertThat(containerResponse.getProperties().getPartitionKeyDefinition().getPaths().get(1) == paths.get(1));

//MultiHash collection delete
CosmosContainerResponse deleteResponse = multiHashContainer.delete();
}

@Test(groups = { "emulator" }, timeOut = TIMEOUT)
public void queryContainer() throws Exception{
Expand Down
Loading