com.cognite.client.Assets Maven / Gradle / Ivy
/*
* Copyright (c) 2020 Cognite AS
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.cognite.client;
import com.cognite.client.config.UpsertMode;
import com.cognite.client.dto.Aggregate;
import com.cognite.client.dto.Asset;
import com.cognite.client.config.ResourceType;
import com.cognite.client.dto.Item;
import com.cognite.client.servicesV1.ConnectorServiceV1;
import com.cognite.client.servicesV1.parser.AssetParser;
import com.cognite.client.stream.ListSource;
import com.cognite.client.stream.Publisher;
import com.cognite.client.util.Items;
import com.cognite.client.util.Partition;
import com.google.auto.value.AutoValue;
import com.google.common.base.Preconditions;
import org.apache.commons.lang3.RandomStringUtils;
import org.jgrapht.Graph;
import org.jgrapht.alg.connectivity.ConnectivityInspector;
import org.jgrapht.alg.cycle.CycleDetector;
import org.jgrapht.graph.DefaultEdge;
import org.jgrapht.graph.SimpleDirectedGraph;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.time.Duration;
import java.time.Instant;
import java.util.*;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;
/**
* This class represents the Cognite assets api endpoint.
*
* It provides methods for reading and writing {@link com.cognite.client.dto.Asset}.
*/
@AutoValue
public abstract class Assets extends ApiBase implements ListSource {
private final static int MAX_UPSERT_BATCH_SIZE = 200;
private static Builder builder() {
return new AutoValue_Assets.Builder();
}
protected static final Logger LOG = LoggerFactory.getLogger(Assets.class);
/**
* Constructs a new {@link Assets} object using the provided client configuration.
*
* This method is intended for internal use--SDK clients should always use {@link CogniteClient}
* as the entry point to this class.
*
* @param client The {@link CogniteClient} to use for configuration settings.
* @return the assets api object.
*/
public static Assets of(CogniteClient client) {
return Assets.builder()
.setClient(client)
.build();
}
/**
* Returns all {@link Asset} objects.
*
* @see #list(Request)
*/
public Iterator> list() throws Exception {
return this.list(Request.create());
}
/**
* Returns all {@link Asset} objects that matches the filters set in the {@link Request}.
*
* The results are paged through / iterated over via an {@link Iterator}--the entire results set is not buffered in
* memory, but streamed in "pages" from the Cognite api. If you need to buffer the entire results set, then you
* have to stream these results into your own data structure.
*
* The assets are retrieved using multiple, parallel request streams towards the Cognite api. The number of
* parallel streams are set in the {@link com.cognite.client.config.ClientConfig}.
*
* @param requestParameters the filters to use for retrieving the assets.
* @return an {@link Iterator} to page through the results set.
* @throws Exception
*/
public Iterator> list(Request requestParameters) throws Exception {
List partitions = buildPartitionsList(getClient().getClientConfig().getNoListPartitions());
return this.list(requestParameters, partitions.toArray(new String[0]));
}
/**
* Returns all {@link Asset} objects that matches the filters set in the {@link Request} for the
* specified partitions. This is method is intended for advanced use cases where you need direct control over
* the individual partitions. For example, when using the SDK in a distributed computing environment.
*
* The results are paged through / iterated over via an {@link Iterator}--the entire results set is not buffered in
* memory, but streamed in "pages" from the Cognite api. If you need to buffer the entire results set, then you
* have to stream these results into your own data structure.
*
* @param requestParameters the filters to use for retrieving the assets.
* @param partitions the partitions to include.
* @return an {@link Iterator} to page through the results set.
* @throws Exception
*/
public Iterator> list(Request requestParameters, String... partitions) throws Exception {
return AdapterIterator.of(listJson(ResourceType.ASSET, requestParameters, partitions), this::parseAsset);
}
/**
* Returns a {@link Publisher} that can stream {@link Asset} from Cognite Data Fusion.
*
* When an {@link Asset} is created or updated, it will be captured by the publisher and emitted to the registered
* consumer.
*
* @return The publisher producing the stream of assets. Call {@code start()} to start the stream.
*/
public Publisher stream() {
return Publisher.of(this);
}
/**
* Retrieve assets by {@code externalId}.
*
* @param externalId The {@code externalIds} to retrieve
* @return The retrieved assets.
* @throws Exception
*/
public List retrieve(String... externalId) throws Exception {
return retrieve(Items.parseItems(externalId));
}
/**
* Retrieve assets by {@code internal id}.
*
* @param id The {@code ids} to retrieve
* @return The retrieved assets.
* @throws Exception
*/
public List retrieve(long... id) throws Exception {
return retrieve(Items.parseItems(id));
}
/**
* Retrieve assets by {@code externalId / id}.
*
* @param items The item(s) {@code externalId / id} to retrieve.
* @return The retrieved assets.
* @throws Exception
*/
public List retrieve(List- items) throws Exception {
return retrieveJson(ResourceType.ASSET, items).stream()
.map(this::parseAsset)
.collect(Collectors.toList());
}
/**
* Performs an item aggregation request to Cognite Data Fusion.
*
* The default aggregation is a total item count based on the (optional) filters in the request.
* Multiple aggregation types are supported. Please refer to the Cognite API specification for more information
* on the possible settings.
*
* @param requestParameters The filtering and aggregates specification
* @return The aggregation results.
* @throws Exception
* @see Cognite API v1 specification
*/
public Aggregate aggregate(Request requestParameters) throws Exception {
return aggregate(ResourceType.ASSET, requestParameters);
}
/**
* Synchronizes the input collection of {@link Asset} (representing multiple, complete asset hierarchies)
* with existing asset hierarchies in CDF.
*
* This method will inspect the input collection of {@link Asset} and identify the various asset hierarchies. Each
* hierarchy is then processed by {@link #synchronizeHierarchy(Collection)}.
*
* @see #synchronizeHierarchy(Collection)
*
* @param assetHierarchies The input asset hierarchies--this represents the target state of the synchronization.
* @return the synchronized assets.
* @throws Exception
*/
public List
synchronizeMultipleHierarchies(Collection assetHierarchies) throws Exception {
String loggingRandomString = RandomStringUtils.randomAlphanumeric(5);
String loggingPrefix = "synchronizeMultipleHierarchies() - " + loggingRandomString + " - ";
Instant startInstant = Instant.now();
if (!checkExternalId(assetHierarchies, loggingPrefix)) {
String message = loggingPrefix + "Some input assets are missing externalId.";
LOG.error(message);
throw new Exception(message);
}
Map assetMap = new HashMap<>();
assetHierarchies
.forEach(asset -> assetMap.put(asset.getExternalId(), asset));
// Identify the hierarchies by loading all assets into a graph and identify the connected sub-graphs
Graph graph = new SimpleDirectedGraph<>(DefaultEdge.class);
// add vertices
for (Asset vertex : assetMap.values()) {
graph.addVertex(vertex);
}
// add edges
for (Asset asset : assetMap.values()) {
if (asset.hasParentExternalId() && assetMap.containsKey(asset.getParentExternalId())) {
graph.addEdge(assetMap.get(asset.getParentExternalId()), asset);
}
}
ConnectivityInspector connectivityInspector = new ConnectivityInspector<>(graph);
List> hierarchies = connectivityInspector.connectedSets();
LOG.info(loggingPrefix + "Identified {} hierarchies in the input collection. Duration: {}",
hierarchies.size(),
Duration.between(startInstant, Instant.now()));
List returnList = new ArrayList<>();
for (Set hierarchy : hierarchies) {
returnList.addAll(this.synchronizeHierarchy(hierarchy));
}
LOG.info(loggingPrefix + "Finished synchronizing {} hierarchies. Duration: {}",
hierarchies.size(),
Duration.between(startInstant, Instant.now()));
return returnList;
}
/**
* Synchronizes the input collection of {@link Asset} (representing a single, complete asset hierarchy)
* with an existing asset hierarchy in CDF. The input asset collection represents the target state.
* New asset nodes will be added, changed asset nodes will be updated
* and deleted asset nodes will be removed (from CDF).
*
* Algorithm:
* - Verify that the input collection satisfies the hierarchy constraints:
* - All assets must specify an {@code externalId}.
* - No duplicates (based on {@code externalId}).
* - The collection must contain one and only one asset object with no parent reference (representing the root node)
* - All other assets must contain a valid {@code parentExternalId} reference (no self-references).
* - No circular references.
* - Read the CDF asset hierarchy based on the identified root external id.
* - Compare the input collection with the existing CDF hierarchy. Identify creates, updates and deletes.
* - Write creates and updates in topological order.
* - Write deletes in reverse topological order.
*
* @param assetHierarchy The input asset hierarchy--this represents the target state of the synchronization.
* @return the synchronized assets.
* @throws Exception
*/
public List synchronizeHierarchy(Collection assetHierarchy) throws Exception {
String loggingRandomString = RandomStringUtils.randomAlphanumeric(5);
String loggingPrefix = "synchronizeHierarchy() - " + loggingRandomString + " - ";
Instant startInstant = Instant.now();
LOG.info(loggingPrefix + "Start hierarchy synchronization. Received {} assets in the input collection",
assetHierarchy.size());
LOG.debug(loggingPrefix + "Check input collection data integrity.");
if (!verifyAssetHierarchyIntegrity(assetHierarchy, loggingRandomString)) {
String message = loggingPrefix + "The input asset collection does not satisfy the integrity constraints.";
LOG.error(message);
throw new Exception(message);
}
LOG.debug(loggingPrefix + "Input collection data integrity is validated. Duration: {}",
Duration.between(startInstant, Instant.now()));
LOG.debug(loggingPrefix + "Identify root node.");
List rootNodes = identifyRootNodes(assetHierarchy);
if (rootNodes.size() != 1) {
String message = String.format("%s Error, found %d root nodes. Expected to find one.",
loggingPrefix,
rootNodes.size());
LOG.error(message);
throw new Exception(message);
}
String rootExternalId = rootNodes.get(0).getExternalId();
LOG.info(loggingPrefix + "Identified root node with external id [{}]. Duration: {}",
rootExternalId,
Duration.between(startInstant, Instant.now()));
LOG.debug(loggingPrefix + "Start downloading the existing asset hierarchy for root [{}]",
rootExternalId);
// Use the subtree query. The max 100k constraint does not count towards root assets. I.e. root assets
// subtrees will return the entire subtree every time.
Map cdfAssets = new HashMap<>();
Request assetRequest = Request.create()
.withFilterParameter("assetSubtreeIds", List.of(Map.of("externalId", rootExternalId)));
getClient().assets().list(assetRequest)
.forEachRemaining(assetBatch -> {
Map batchMap = assetBatch.stream()
.collect(Collectors.toMap(Asset::getExternalId, Function.identity()));
cdfAssets.putAll(batchMap);
});
LOG.debug(loggingPrefix + "Finished downloading the existing asset hierarchy for root [{}]. "
+ "No assets: {}. Duration {}",
rootExternalId,
cdfAssets.size(),
Duration.between(startInstant, Instant.now()));
// Identify upserts and deletes
List upsertList = new ArrayList<>();
List noChangeList = new ArrayList<>();
List- deleteList = new ArrayList<>();
int changedAssetCounter = 0;
int noChangeCounter = 0;
int newAssetCounter = 0;
for (Asset inputAsset : assetHierarchy) {
if (cdfAssets.containsKey(inputAsset.getExternalId())) {
// The asset exists from before. Check if it has changed.
if (!isEqual(inputAsset, cdfAssets.get(inputAsset.getExternalId()))) {
upsertList.add(inputAsset);
changedAssetCounter++;
} else {
noChangeList.add(cdfAssets.get(inputAsset.getExternalId())); // Add from CDF to get id fields++
noChangeCounter++;
}
cdfAssets.remove(inputAsset.getExternalId());
} else {
// A new asset, add it to the upserts list
upsertList.add(inputAsset);
newAssetCounter++;
}
}
// Check for leftover CDF assets. These should be deleted
if (!cdfAssets.isEmpty()) {
List
sortedAssets = topologicalSort(cdfAssets.values());
Collections.reverse(sortedAssets); // Should delete assets in reverse topological order
deleteList = sortedAssets.stream()
.map(asset ->
Item.newBuilder()
.setExternalId(asset.getExternalId())
.build())
.collect(Collectors.toList());
}
LOG.info(loggingPrefix + "Change detection complete. New assets: {}, changed assets: {}, deleted assets: {}, "
+ "assets with no change: {}. Duration {}",
newAssetCounter,
changedAssetCounter,
deleteList.size(),
noChangeCounter,
Duration.between(startInstant, Instant.now()));
// Write the changes to CDF
// The upserts must be written with REPLACE mode
List upsertedAssets = getClient()
.withClientConfig(getClient().getClientConfig()
.withUpsertMode(UpsertMode.REPLACE))
.assets()
.upsert(upsertList);
getClient().assets().delete(deleteList);
LOG.info(loggingPrefix + "Finished synchronizing hierarchy. Root asset external id: [{}]. "
+ "Total no assets: {}. Upserted assets: {}. Deleted assets: {}. Duration: {}",
rootExternalId,
assetHierarchy.size(),
upsertList.size(),
deleteList.size(),
Duration.between(startInstant, Instant.now()));
return Stream.concat(noChangeList.stream(), upsertedAssets.stream())
.collect(Collectors.toList());
}
/**
* Creates or updates a set of {@link Asset} objects.
*
* If it is a new {@link Asset} object (based on {@code id / externalId}, then it will be created.
*
* If an {@link Asset} object already exists in Cognite Data Fusion, it will be updated. The update behavior
* is specified via the update mode in the {@link com.cognite.client.config.ClientConfig} settings.
*
* The assets will be checked for integrity and topologically sorted before an ordered upsert operation
* is started.
*
* The following constraints will be evaluated:
* - All assets must specify an {@code externalId}.
* - No duplicates (based on {@code externalId}.
* - No self-reference.
* - No circular references.
*
* @param assets The assets to upsert.
* @return The upserted assets.
* @throws Exception
*/
public List upsert(Collection assets) throws Exception {
String loggingPrefix = "upsert() - " + RandomStringUtils.randomAlphanumeric(5) + " - ";
Instant startInstant = Instant.now();
// Check integrity and sort assets topologically
List sortedAssets = topologicalSort(assets);
// Setup writers and upsert manager
ConnectorServiceV1 connector = getClient().getConnectorService();
ConnectorServiceV1.ItemWriter createItemWriter = connector.writeAssets();
ConnectorServiceV1.ItemWriter updateItemWriter = connector.updateAssets();
UpsertItems upsertItems = UpsertItems.of(createItemWriter, this::toRequestInsertItem, getClient().buildAuthConfig())
.withUpdateItemWriter(updateItemWriter)
.withUpdateMappingFunction(this::toRequestUpdateItem)
.withIdFunction(this::getAssetId);
if (getClient().getClientConfig().getUpsertMode() == UpsertMode.REPLACE) {
upsertItems = upsertItems.withUpdateMappingFunction(this::toRequestReplaceItem);
}
// Write assets as a serial set of single-worker requests. Must do this to guarantee the order of upsert.
List> sortedBatches = Partition.ofSize(sortedAssets, MAX_UPSERT_BATCH_SIZE);
List assetUpsertResults = new ArrayList<>(assets.size());
for (List batch : sortedBatches) {
assetUpsertResults.addAll(upsertItems.upsertViaCreateAndUpdate(batch));
}
LOG.info(loggingPrefix + "Finished upserting {} assets across {} batches within a duration of {}",
sortedAssets.size(),
sortedBatches.size(),
Duration.between(startInstant, Instant.now()));
return assetUpsertResults.stream()
.map(this::parseAsset)
.collect(Collectors.toList());
}
/**
* Checks a collection of assets for integrity. The assets must represent a single, complete
* hierarchy.
*
* This verifies if the collection satisfies the
* constraints of the Cognite Data Fusion data model if you were to write them using the
* {@link #upsert(Collection) upsert} method.
*
* The following constraints will be evaluated:
* - All assets must specify an {@code externalId}.
* - No duplicates (based on {@code externalId}).
* - The collection must contain one and only one asset object with no parent reference (representing the root node)
* - All other assets must contain a valid {@code parentExternalId} reference (no self-references).
* - No circular references.
*
* @param assets A collection of {@link Asset} representing a single, complete asset hierarchy.
* @return
*/
public boolean verifyAssetHierarchyIntegrity(Collection assets) {
return verifyAssetHierarchyIntegrity(assets, RandomStringUtils.randomAlphanumeric(5));
}
/*
Private version of the verification logic which allows for setting a common logging identifier. This is useful in
debugging complex operations, like asset hierarchy synchronization.
*/
private boolean verifyAssetHierarchyIntegrity(Collection assets, String loggingIdentifier) {
if (!checkExternalId(assets, loggingIdentifier)) return false;
if (!checkDuplicates(assets, loggingIdentifier)) return false;
if (!checkSelfReference(assets, loggingIdentifier)) return false;
if (!checkCircularReferences(assets, loggingIdentifier)) return false;
if (!checkReferentialIntegrity(assets, loggingIdentifier)) return false;
return true;
}
/**
* Deletes a set of assets.
*
* The assets to delete are identified via their {@code externalId / id} by submitting a list of
* {@link Item}.
*
* This method will not delete assets recursively. Please use {@code delete(List- items, boolean recursive)}
* for recursive deletes.
*
* @param items a list of {@link Item} representing the assets (externalId / id) to be deleted
* @return The deleted events via {@link Item}
* @throws Exception
*/
public List
- delete(List
- items) throws Exception {
return delete(items, false);
}
/**
* Deletes a set of assets.
*
* The assets to delete are identified via their {@code externalId / id} by submitting a list of
* {@link Item}.
*
* @param items a list of {@link Item} representing the assets (externalId / id) to be deleted
* @param recursive Set to {@code true} to recursively delete all subtrees under the specified items.
* @return The deleted events via {@link Item}
* @throws Exception
*/
public List
- delete(List
- items, boolean recursive) throws Exception {
ConnectorServiceV1 connector = getClient().getConnectorService();
ConnectorServiceV1.ItemWriter deleteItemWriter = connector.deleteAssets();
DeleteItems deleteItems = DeleteItems.of(deleteItemWriter, getClient().buildAuthConfig())
.addParameter("ignoreUnknownIds", true)
.addParameter("recursive", recursive);
return deleteItems.deleteItems(items);
}
/**
* Find and return the root nodes (if any) in the assets collection. A root node is an asset that doesn't
* specify a parent external id reference.
*
* @param assets The assets to search for root nodes.
* @return a list of asset root nodes.
*/
private List
identifyRootNodes(Collection assets) {
String loggingPrefix = "identifyRootNodes() - " + RandomStringUtils.randomAlphanumeric(5) + " - ";
LOG.debug(loggingPrefix + "Start identifying root nodes.");
List rootNodeList = assets.stream()
.filter(asset -> asset.getParentExternalId().isBlank())
.collect(Collectors.toList());
LOG.debug(loggingPrefix + "Found {} root nodes.",
rootNodeList.size());
return rootNodeList;
}
/**
* Check if two asset objects are equal in terms of their main payload. Internal attributes like ids and
* internal timestamps are ignored.
*
* @param one The first asset to compare.
* @param other The second asset to compare.
* @return true if the assets carry an equal payload, false if not.
*/
private boolean isEqual(Asset one, Asset other) {
boolean result = true;
result = result && (one.hasExternalId() == other.hasExternalId());
if (one.hasExternalId()) {
result = result && one.getExternalId()
.equals(other.getExternalId());
}
result = result && one.getName()
.equals(other.getName());
result = result && (one.hasParentExternalId() == other.hasParentExternalId());
if (one.hasParentExternalId()) {
result = result && one.getParentExternalId()
.equals(other.getParentExternalId());
}
result = result && (one.hasDescription() == other.hasDescription());
if (one.hasDescription()) {
result = result && one.getDescription()
.equals(other.getDescription());
}
result = result && one.getMetadataMap().equals(
other.getMetadataMap());
result = result && (one.hasSource() == other.hasSource());
if (one.hasSource()) {
result = result && one.getSource()
.equals(other.getSource());
}
result = result && (one.hasDataSetId() == other.hasDataSetId());
if (one.hasDataSetId()) {
result = result && one.getDataSetId() == other.getDataSetId();
}
result = result && (one.getLabelsList().equals(other.getLabelsList()));
return result;
}
/**
* This function will sort a collection of assets into the correct order for upsert to CDF.
*
* Assets need to be written in a certain order to comply with the hierarchy constraints of CDF. In short, if an asset
* references a parent (either via id or externalId), then that parent must exist either in CDF or in the same write
* batch. Hence, a collection of assets must be written to CDF in topological order.
*
* This function requires that the input collection is complete. That is, all assets required to traverse the hierarchy
* from the root node to the leaves must either exist in CDF and/or in the input collection.
*
* The sorting algorithm employed is a naive breadth-first O(n * depth(n)):
* while (items in inputCollection) {
* for (items in inputCollection) {
* if (item references unknown OR item references null OR item references id) : write item and remove from inputCollection
* }
* }
*
* Other requirements for the input:
* - Assets must have externalId set.
* - No duplicates.
* - No self-references.
* - No circular references.
* - If both parentExternalId and parentId are set, then parentExternalId takes precedence in the sort.
*
* @param assets A collection of {@link Asset} to be topologically sorted.
* @return The sorted assets collection.
* @throws Exception if one (or more) of the constraints are not fulfilled.
*/
private List topologicalSort(Collection assets) throws Exception {
Instant startInstant = Instant.now();
String loggingRandomString = RandomStringUtils.randomAlphanumeric(5);
String loggingPrefix = "sortAssetsForUpsert - " + loggingRandomString + " - ";
Preconditions.checkArgument(checkExternalId(assets, loggingRandomString),
"Some assets do not have externalId. Please check the log for more information.");
Preconditions.checkArgument(checkDuplicates(assets, loggingRandomString),
"Found duplicates in the input assets. Please check the log for more information.");
Preconditions.checkArgument(checkSelfReference(assets, loggingRandomString),
"Some assets contain a self-reference. Please check the log for more information.");
Preconditions.checkArgument(checkCircularReferences(assets, loggingRandomString),
"Circular reference detected. Please check the log for more information.");
LOG.debug(loggingPrefix + "Constraints check passed. Starting sort.");
Map inputMap = assets.stream()
.collect(Collectors.toMap(Asset::getExternalId, Function.identity()));
List sortedAssets = new ArrayList<>();
while (!inputMap.isEmpty()) {
int startInputMapSize = inputMap.size();
LOG.debug(loggingPrefix + "Starting new sort iteration. Assets left to sort: {}", inputMap.size());
for (Iterator iterator = inputMap.values().iterator(); iterator.hasNext();) {
Asset asset = iterator.next();
if (asset.hasParentExternalId()) {
// Check if the parent asset exists in the input collection. If no, it is safe to write the asset.
if (!inputMap.containsKey(asset.getParentExternalId())) {
sortedAssets.add(asset);
iterator.remove();
}
} else {
// Asset either has no parent reference or references an (internal) id.
// Null parent is safe to write and (internal) id is assumed to already exist in cdf--safe to write.
sortedAssets.add(asset);
iterator.remove();
}
}
LOG.debug(loggingPrefix + "Finished sort iteration. Assets left to sort: {}", inputMap.size());
if (startInputMapSize == inputMap.size()) {
String message = loggingPrefix + "Possible circular reference detected when sorting assets, aborting.";
LOG.error(message);
throw new Exception(message);
}
}
LOG.info(loggingPrefix + "Sort assets finished. Sorted {} assets within a duration of {}.",
sortedAssets.size(),
Duration.between(startInstant, Instant.now()).toString());
return sortedAssets;
}
/**
* Checks the assets for {@code externalId}.
*
* @param assets The assets to check.
* @return true if all assets contain {@code externalId}. False if one or more assets do not have {@code externalId}.
*/
private boolean checkExternalId(Collection assets, String loggingIdentifier) {
Preconditions.checkNotNull(loggingIdentifier);
String loggingPrefix = "checkExternalId() - " + loggingIdentifier + " - ";
List missingExternalIdList = new ArrayList<>();
for (Asset asset : assets) {
if (!asset.hasExternalId()) {
missingExternalIdList.add(asset);
}
}
// Report on missing externalId
if (missingExternalIdList.size() > 0) {
StringBuilder message = new StringBuilder();
String errorMessage = loggingPrefix + "Found " + missingExternalIdList.size() + " assets missing externalId.";
message.append(errorMessage).append(System.lineSeparator());
if (missingExternalIdList.size() > 100) {
missingExternalIdList = missingExternalIdList.subList(0, 99);
}
message.append("Items with missing externalId (max 100 displayed): " + System.lineSeparator());
for (Asset item : missingExternalIdList) {
message.append("---------------------------").append(System.lineSeparator())
.append("name: [").append(item.getName()).append("]").append(System.lineSeparator())
.append("parentExternalId: [").append(item.getParentExternalId()).append("]").append(System.lineSeparator())
.append("description: [").append(item.getDescription()).append("]").append(System.lineSeparator())
.append("--------------------------");
}
LOG.warn(message.toString());
return false;
}
LOG.debug(loggingPrefix + "All assets contain an externalId.");
return true;
}
/**
* Checks the assets for duplicates. The duplicates check is based on {@code externalId}.
*
* @param assets The assets to check.
* @return true if no duplicates are detected. False if one or more duplicates are detected.
*/
private boolean checkDuplicates(Collection assets, String loggingIdentifier) {
Preconditions.checkNotNull(loggingIdentifier);
String loggingPrefix = "checkDuplicates() - " + loggingIdentifier + " - ";
List duplicatesList = new ArrayList<>();
Map inputMap = new HashMap<>();
for (Asset asset : assets) {
if (inputMap.containsKey(asset.getExternalId())) {
duplicatesList.add(asset);
}
inputMap.put(asset.getExternalId(), asset);
}
// Report on duplicates
if (duplicatesList.size() > 0) {
StringBuilder message = new StringBuilder();
String errorMessage = loggingPrefix + "Found " + duplicatesList.size() + " duplicates.";
message.append(errorMessage).append(System.lineSeparator());
if (duplicatesList.size() > 100) {
duplicatesList = duplicatesList.subList(0, 99);
}
message.append("Duplicate items (max 100 displayed): " + System.lineSeparator());
for (Asset item : duplicatesList) {
message.append("---------------------------").append(System.lineSeparator())
.append("externalId: [").append(item.getExternalId()).append("]").append(System.lineSeparator())
.append("name: [").append(item.getName()).append("]").append(System.lineSeparator())
.append("description: [").append(item.getDescription()).append("]").append(System.lineSeparator())
.append("--------------------------");
}
LOG.warn(message.toString());
return false;
}
LOG.debug(loggingPrefix + "No duplicates detected in the assets collection.");
return true;
}
/**
* Checks the assets for self-reference. That is, the asset's {@code parentExternalId} references
* its own {@code externalId}.
*
* @param assets The assets to check.
* @return true if no self-references are detected. False if one or more self-reference are detected.
*/
private boolean checkSelfReference(Collection assets, String loggingIdentifier) {
Preconditions.checkNotNull(loggingIdentifier);
String loggingPrefix = "checkExternalId() - " + loggingIdentifier + " - ";
List selfReferenceList = new ArrayList<>(50);
for (Asset asset : assets) {
if (asset.hasParentExternalId()
&& asset.getParentExternalId().equals(asset.getExternalId())) {
selfReferenceList.add(asset);
}
}
// Report on self-references
if (!selfReferenceList.isEmpty()) {
StringBuilder message = new StringBuilder();
String errorMessage = loggingPrefix + "Found " + selfReferenceList.size()
+ " items with self-referencing parentExternalId.";
message.append(errorMessage).append(System.lineSeparator());
if (selfReferenceList.size() > 100) {
selfReferenceList = selfReferenceList.subList(0, 99);
}
message.append("Items with self-reference (max 100 displayed): " + System.lineSeparator());
for (Asset item : selfReferenceList) {
message.append("---------------------------").append(System.lineSeparator())
.append("externalId: [").append(item.getExternalId()).append("]").append(System.lineSeparator())
.append("name: [").append(item.getName()).append("]").append(System.lineSeparator())
.append("description: [").append(item.getDescription()).append("]").append(System.lineSeparator())
.append("parentExternalId: [").append(item.getParentExternalId()).append("]").append(System.lineSeparator())
.append("--------------------------");
}
LOG.warn(message.toString());
return false;
}
LOG.debug(loggingPrefix + "No self-references detected in the assets collection.");
return true;
}
/**
* Checks the assets for circular references.
*
* @param assets The assets to check.
* @return True if circular references are detected. False if no circular references are detected.
*/
private boolean checkCircularReferences(Collection assets, String loggingIdentifier) {
Preconditions.checkNotNull(loggingIdentifier);
String loggingPrefix = "checkCircularReferences() - " + loggingIdentifier + " - ";
Map assetMap = new HashMap<>();
assets.forEach(asset -> assetMap.put(asset.getExternalId(), asset));
// Checking for circular references
Graph graph = new SimpleDirectedGraph<>(DefaultEdge.class);
// add vertices
for (Asset vertex : assetMap.values()) {
graph.addVertex(vertex);
}
// add edges
for (Asset asset : assetMap.values()) {
if (asset.hasParentExternalId() && assetMap.containsKey(asset.getParentExternalId())) {
graph.addEdge(assetMap.get(asset.getParentExternalId()), asset);
}
}
CycleDetector cycleDetector = new CycleDetector<>(graph);
if (cycleDetector.detectCycles()) {
List cycles = cycleDetector.findCycles().stream()
.map(Asset::getExternalId)
.collect(Collectors.toList());
String message = loggingPrefix + "Cycles detected. Number of asset in the cycle: " + cycles.size();
LOG.error(message);
LOG.error(loggingPrefix + "Cycles: " + cycles.toString());
return false;
}
LOG.debug(loggingPrefix + "No cycles detected in the assets collection.");
return true;
}
/**
* Checks the assets for referential integrity.
*
* This check is relevant for a collection of assets that represents a single, complete
* asset hierarchy. It performs the following checks:
* - One and only one root node (an asset with no parent reference).
* - All other assets reference an asset in this collection.
*
* @param assets The assets to check.
* @return True if integrity is intact. False otherwise.
*/
private boolean checkReferentialIntegrity(Collection assets, String loggingIdentifier) {
Preconditions.checkNotNull(loggingIdentifier);
String loggingPrefix = "checkReferentialIntegrity() - " + loggingIdentifier + " - ";
Set extIdSet = new HashSet<>();
List invalidReferenceList = new ArrayList<>();
List rootNodeList = new ArrayList<>();
LOG.debug(loggingPrefix + "Checking asset input table for integrity.");
for (Asset element : assets) {
if (!element.hasParentExternalId()) {
rootNodeList.add(element);
}
extIdSet.add(element.getExternalId());
}
for (Asset element : assets) {
if (element.hasParentExternalId()
&& !extIdSet.contains(element.getParentExternalId())) {
invalidReferenceList.add(element);
}
}
if (rootNodeList.size() != 1) {
String errorMessage = loggingPrefix + "Found " + rootNodeList.size() + " root nodes.";
StringBuilder message = new StringBuilder()
.append(errorMessage).append(System.lineSeparator());
if (rootNodeList.size() > 100) {
rootNodeList = rootNodeList.subList(0, 99);
}
message.append("Root nodes (max 100 displayed): " + System.lineSeparator());
for (Asset item : rootNodeList) {
message.append("---------------------------").append(System.lineSeparator())
.append("externalId: [").append(item.getExternalId()).append("]").append(System.lineSeparator())
.append("name: [").append(item.getName()).append("]").append(System.lineSeparator())
.append("parentExternalId: [").append(item.getParentExternalId()).append("]").append(System.lineSeparator())
.append("description: [").append(item.getDescription()).append("]").append(System.lineSeparator())
.append("--------------------------");
}
LOG.warn(message.toString());
return false;
}
if (invalidReferenceList.size() > 0) {
StringBuilder message = new StringBuilder();
String errorMessage = loggingPrefix + "Found " + invalidReferenceList.size() + " assets with invalid parent reference.";
message.append(errorMessage).append(System.lineSeparator());
if (invalidReferenceList.size() > 100) {
invalidReferenceList = invalidReferenceList.subList(0, 99);
}
message.append("Items with invalid parentExternalId (max 100 displayed): " + System.lineSeparator());
for (Asset item : invalidReferenceList) {
message.append("---------------------------").append(System.lineSeparator())
.append("externalId: [").append(item.getExternalId()).append("]").append(System.lineSeparator())
.append("name: [").append(item.getName()).append("]").append(System.lineSeparator())
.append("parentExternalId: [").append(item.getParentExternalId()).append("]").append(System.lineSeparator())
.append("description: [").append(item.getDescription()).append("]").append(System.lineSeparator())
.append("--------------------------");
}
LOG.warn(message.toString());
return false;
}
LOG.debug(loggingPrefix + "The asset collection contains a single root node and valid parent references.");
return true;
}
/*
Wrapping the parser because we need to handle the exception--an ugly workaround since lambdas don't
deal very well with exceptions.
*/
private Asset parseAsset(String json) {
try {
return AssetParser.parseAsset(json);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/*
Wrapping the parser because we need to handle the exception--an ugly workaround since lambdas don't
deal very well with exceptions.
*/
private Map toRequestInsertItem(Asset item) {
try {
return AssetParser.toRequestInsertItem(item);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/*
Wrapping the parser because we need to handle the exception--an ugly workaround since lambdas don't
deal very well with exceptions.
*/
private Map toRequestUpdateItem(Asset item) {
try {
return AssetParser.toRequestUpdateItem(item);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/*
Wrapping the parser because we need to handle the exception--an ugly workaround since lambdas don't
deal very well with exceptions.
*/
private Map toRequestReplaceItem(Asset item) {
try {
return AssetParser.toRequestReplaceItem(item);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/*
Returns the id of an event. It will first check for an externalId, second it will check for id.
If no id is found, it returns an empty Optional.
*/
private Optional getAssetId(Asset item) {
if (item.hasExternalId()) {
return Optional.of(item.getExternalId());
} else if (item.hasId()) {
return Optional.of(String.valueOf(item.getId()));
} else {
return Optional.empty();
}
}
@AutoValue.Builder
abstract static class Builder extends ApiBase.Builder {
abstract Assets build();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy