com.microsoft.azure.toolkit.lib.cosmos.sql.SqlContainer Maven / Gradle / Ivy
/*
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*/
package com.microsoft.azure.toolkit.lib.cosmos.sql;
import com.azure.cosmos.CosmosClient;
import com.azure.cosmos.CosmosContainer;
import com.azure.cosmos.models.CosmosContainerResponse;
import com.azure.resourcemanager.cosmos.fluent.models.SqlContainerGetResultsInner;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.microsoft.azure.toolkit.lib.common.bundle.AzureString;
import com.microsoft.azure.toolkit.lib.common.messager.AzureMessager;
import com.microsoft.azure.toolkit.lib.common.model.AbstractAzResource;
import com.microsoft.azure.toolkit.lib.common.model.AbstractAzResourceModule;
import com.microsoft.azure.toolkit.lib.common.model.Deletable;
import com.microsoft.azure.toolkit.lib.cosmos.ICosmosCollection;
import com.microsoft.azure.toolkit.lib.cosmos.ICosmosDocumentContainer;
import lombok.Getter;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.util.Collections;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.UUID;
import static com.microsoft.azure.toolkit.lib.cosmos.sql.SqlDocumentModule.ID;
public class SqlContainer extends AbstractAzResource
implements Deletable, ICosmosCollection, ICosmosDocumentContainer {
private CosmosContainer container;
private CosmosContainerResponse containerResponse;
@Getter
private final SqlDocumentModule documentModule;
protected SqlContainer(@Nonnull String name, @Nonnull String resourceGroupName, @Nonnull SqlContainerModule module) {
super(name, resourceGroupName, module);
this.documentModule = new SqlDocumentModule(this);
}
protected SqlContainer(@Nonnull SqlContainerGetResultsInner remote, @Nonnull SqlContainerModule module) {
super(remote.name(), module);
this.documentModule = new SqlDocumentModule(this);
}
protected SqlContainer(@Nonnull SqlContainer collection) {
super(collection);
this.container = collection.container;
this.containerResponse = collection.containerResponse;
this.documentModule = new SqlDocumentModule(this);
}
@Nonnull
@Override
public List> getSubModules() {
return Collections.singletonList(documentModule);
}
@Override
public SqlDocument importDocument(@Nonnull final ObjectNode node) {
if (node.get(ID) == null) {
node.put(ID, UUID.randomUUID().toString());
}
final String id = node.get(ID).asText();
final String partitionKey = SqlDocumentModule.getSqlDocumentPartitionValue(node, getPartitionKey());
final String resourceName = SqlDocumentModule.getSqlDocumentResourceName(id, partitionKey);
final boolean existing = this.getDocumentModule().exists(resourceName, getResourceGroupName());
final SqlDocumentDraft documentDraft = existing ?
(SqlDocumentDraft) Objects.requireNonNull(this.documentModule.get(resourceName, getResourceGroupName())).update() :
this.documentModule.create(resourceName, getResourceGroupName());
documentDraft.setDraftDocument(node);
final SqlDocument result = documentDraft.commit();
final AzureString importMessage = AzureString.format("Import document to Cosmos container %s successfully.", this.getName());
final AzureString updateMessage = AzureString.format("Update document %s in Cosmos container %s successfully.", id, this.getName());
AzureMessager.getMessager().info(existing ? updateMessage : importMessage);
return result;
}
public String getPartitionKey() {
return Optional.ofNullable(this.containerResponse)
.map(CosmosContainerResponse::getProperties)
.map(p -> p.getPartitionKeyDefinition().getPaths().get(0))
.orElse(null);
}
public synchronized CosmosContainer getClient() {
if (Objects.isNull(this.container)) {
this.container = getDocumentClient();
this.containerResponse = Optional.ofNullable(container).map(CosmosContainer::read).orElse(null);
}
return this.container;
}
@Override
protected void updateAdditionalProperties(@Nullable SqlContainerGetResultsInner newRemote, @Nullable SqlContainerGetResultsInner oldRemote) {
super.updateAdditionalProperties(newRemote, oldRemote);
if (Objects.nonNull(newRemote)) {
this.container = getDocumentClient();
this.containerResponse = Optional.ofNullable(container).map(CosmosContainer::read).orElse(null);
} else {
this.container = null;
this.containerResponse = null;
}
}
private CosmosContainer getDocumentClient() {
try {
final SqlDatabase sqlDatabase = this.getParent();
final SqlCosmosDBAccount account = (SqlCosmosDBAccount) sqlDatabase.getParent();
final CosmosClient cosmosClient = account.getClient();
return cosmosClient.getDatabase(sqlDatabase.getName()).getContainer(this.getName());
} catch (Throwable e) {
// swallow exception to load data client
return null;
}
}
@Nonnull
@Override
protected String loadStatus(@Nonnull SqlContainerGetResultsInner remote) {
return Status.RUNNING;
}
}