2019-08-31 20:51:14 -07:00
|
|
|
package com.linkedin.metadata.dao;
|
|
|
|
|
|
|
|
import com.linkedin.common.AuditStamp;
|
|
|
|
import com.linkedin.common.urn.Urn;
|
|
|
|
import com.linkedin.data.schema.validation.CoercionMode;
|
|
|
|
import com.linkedin.data.schema.validation.RequiredMode;
|
|
|
|
import com.linkedin.data.schema.validation.UnrecognizedFieldMode;
|
|
|
|
import com.linkedin.data.schema.validation.ValidateDataAgainstSchema;
|
|
|
|
import com.linkedin.data.schema.validation.ValidationOptions;
|
|
|
|
import com.linkedin.data.schema.validation.ValidationResult;
|
|
|
|
import com.linkedin.data.template.RecordTemplate;
|
|
|
|
import com.linkedin.data.template.UnionTemplate;
|
|
|
|
import com.linkedin.metadata.dao.equality.DefaultEqualityTester;
|
|
|
|
import com.linkedin.metadata.dao.equality.EqualityTester;
|
|
|
|
import com.linkedin.metadata.dao.exception.ModelValidationException;
|
|
|
|
import com.linkedin.metadata.dao.producer.BaseMetadataEventProducer;
|
|
|
|
import com.linkedin.metadata.dao.retention.IndefiniteRetention;
|
|
|
|
import com.linkedin.metadata.dao.retention.Retention;
|
|
|
|
import com.linkedin.metadata.dao.retention.TimeBasedRetention;
|
|
|
|
import com.linkedin.metadata.dao.retention.VersionBasedRetention;
|
2020-08-19 16:06:29 -07:00
|
|
|
import com.linkedin.metadata.dao.storage.LocalDAOStorageConfig;
|
2019-08-31 20:51:14 -07:00
|
|
|
import com.linkedin.metadata.query.ExtraInfo;
|
2020-08-19 16:06:29 -07:00
|
|
|
import com.linkedin.metadata.query.IndexFilter;
|
2019-08-31 20:51:14 -07:00
|
|
|
import java.time.Clock;
|
|
|
|
import java.util.HashMap;
|
2020-06-11 10:21:51 -07:00
|
|
|
import java.util.LinkedList;
|
|
|
|
import java.util.List;
|
2019-08-31 20:51:14 -07:00
|
|
|
import java.util.Map;
|
|
|
|
import java.util.Optional;
|
2020-08-19 16:06:29 -07:00
|
|
|
import java.util.Set;
|
2019-08-31 20:51:14 -07:00
|
|
|
import java.util.UUID;
|
2020-06-11 10:21:51 -07:00
|
|
|
import java.util.function.BiConsumer;
|
2019-08-31 20:51:14 -07:00
|
|
|
import java.util.function.Function;
|
|
|
|
import java.util.function.Supplier;
|
|
|
|
import javax.annotation.Nonnull;
|
|
|
|
import javax.annotation.Nullable;
|
|
|
|
import lombok.Value;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A base class for all Local DAOs.
|
|
|
|
*
|
|
|
|
* Local DAO is a standardized interface to store and retrieve aspects from a document store.
|
|
|
|
*
|
|
|
|
* @param <ASPECT_UNION> must be a valid aspect union type defined in com.linkedin.metadata.aspect
|
|
|
|
* @param <URN> must be the entity URN type in {@code ASPECT_UNION}
|
|
|
|
*/
|
|
|
|
public abstract class BaseLocalDAO<ASPECT_UNION extends UnionTemplate, URN extends Urn>
|
|
|
|
extends BaseReadDAO<ASPECT_UNION, URN> {
|
|
|
|
|
|
|
|
@Value
|
2020-06-11 10:21:51 -07:00
|
|
|
static class AspectEntry<ASPECT extends RecordTemplate> {
|
|
|
|
ASPECT aspect;
|
2019-08-31 20:51:14 -07:00
|
|
|
ExtraInfo extraInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Value
|
2020-06-11 10:21:51 -07:00
|
|
|
static class AddResult<ASPECT extends RecordTemplate> {
|
|
|
|
ASPECT oldValue;
|
|
|
|
ASPECT newValue;
|
2019-08-31 20:51:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private static final String DEFAULT_ID_NAMESPACE = "global";
|
|
|
|
|
|
|
|
private static final IndefiniteRetention INDEFINITE_RETENTION = new IndefiniteRetention();
|
|
|
|
|
|
|
|
private static final int DEFAULT_MAX_TRANSACTION_RETRY = 3;
|
|
|
|
|
|
|
|
protected final BaseMetadataEventProducer _producer;
|
2020-08-19 16:06:29 -07:00
|
|
|
protected final LocalDAOStorageConfig _storageConfig;
|
2019-08-31 20:51:14 -07:00
|
|
|
|
|
|
|
// Maps an aspect class to the corresponding retention policy
|
|
|
|
private final Map<Class<? extends RecordTemplate>, Retention> _aspectRetentionMap = new HashMap<>();
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
// Maps as aspect class to a list of post-update hooks
|
|
|
|
private final Map<Class<? extends RecordTemplate>, List<BiConsumer<Urn, RecordTemplate>>> _aspectPostUpdateHooksMap =
|
|
|
|
new HashMap<>();
|
|
|
|
|
2019-08-31 20:51:14 -07:00
|
|
|
// Maps an aspect class to the corresponding equality tester
|
|
|
|
private final Map<Class<? extends RecordTemplate>, EqualityTester<? extends RecordTemplate>>
|
|
|
|
_aspectEqualityTesterMap = new HashMap<>();
|
|
|
|
|
|
|
|
private boolean _modelValidationOnWrite = true;
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
// Always emit MAE on every update regardless if there's any actual change in value
|
|
|
|
private boolean _alwaysEmitAuditEvent = false;
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
// Opt in to emit Aspect Specific MAE, at initial migration stage, always emit the event
|
|
|
|
private boolean _emitAspectSpecificAuditEvent = false;
|
|
|
|
|
|
|
|
// Flag for enabling reads and writes to local secondary index
|
|
|
|
private boolean _enableLocalSecondaryIndex = false;
|
|
|
|
|
|
|
|
// Flag for backfilling local secondary index
|
|
|
|
private boolean _backfillLocalSecondaryIndex = false;
|
|
|
|
|
2019-08-31 20:51:14 -07:00
|
|
|
private Clock _clock = Clock.systemUTC();
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
/**
|
|
|
|
* Constructor for BaseLocalDAO
|
|
|
|
*
|
|
|
|
* @param aspectUnionClass containing union of all supported aspects. Must be a valid aspect union defined in com.linkedin.metadata.aspect
|
|
|
|
* @param producer {@link BaseMetadataEventProducer} for the metadata event producer
|
|
|
|
*/
|
2019-08-31 20:51:14 -07:00
|
|
|
public BaseLocalDAO(@Nonnull Class<ASPECT_UNION> aspectUnionClass, @Nonnull BaseMetadataEventProducer producer) {
|
|
|
|
super(aspectUnionClass);
|
|
|
|
_producer = producer;
|
2020-08-19 16:06:29 -07:00
|
|
|
_storageConfig = LocalDAOStorageConfig.builder().build();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor for BaseLocalDAO
|
|
|
|
*
|
|
|
|
* @param producer {@link BaseMetadataEventProducer} for the metadata event producer
|
|
|
|
* @param storageConfig {@link LocalDAOStorageConfig} containing storage config of full list of supported aspects
|
|
|
|
*/
|
|
|
|
public BaseLocalDAO(@Nonnull BaseMetadataEventProducer producer, @Nonnull LocalDAOStorageConfig storageConfig) {
|
|
|
|
super(storageConfig.getAspectStorageConfigMap().keySet());
|
|
|
|
_producer = producer;
|
|
|
|
_storageConfig = storageConfig;
|
2019-08-31 20:51:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For tests to override the internal clock
|
|
|
|
*/
|
|
|
|
public void setClock(@Nonnull Clock clock) {
|
|
|
|
_clock = clock;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets {@link Retention} for a specific aspect type.
|
|
|
|
*/
|
|
|
|
public <ASPECT extends RecordTemplate> void setRetention(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull Retention retention) {
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
_aspectRetentionMap.put(aspectClass, retention);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the {@link Retention} for an aspect type, or {@link IndefiniteRetention} if none is registered.
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public <ASPECT extends RecordTemplate> Retention getRetention(@Nonnull Class<ASPECT> aspectClass) {
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
return _aspectRetentionMap.getOrDefault(aspectClass, INDEFINITE_RETENTION);
|
|
|
|
}
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
/**
|
|
|
|
* Registers a post-update hook for a specific aspect.
|
|
|
|
*
|
|
|
|
* The hook will be invoked with the latest value of an aspect after it's updated. There's no guarantee on the order
|
|
|
|
* of invocation when multiple hooks are added for a single aspect. Adding the same hook again will result in
|
|
|
|
* {@link IllegalArgumentException} thrown. Hooks are invoked in the order they're registered.
|
|
|
|
*/
|
|
|
|
public <URN extends Urn, ASPECT extends RecordTemplate> void addPostUpdateHook(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull BiConsumer<URN, ASPECT> postUpdateHook) {
|
|
|
|
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
// TODO: Also validate Urn once we convert all aspect models to PDL with proper annotation
|
|
|
|
|
|
|
|
final List<BiConsumer<Urn, RecordTemplate>> hooks =
|
|
|
|
_aspectPostUpdateHooksMap.getOrDefault(aspectClass, new LinkedList<>());
|
|
|
|
|
|
|
|
if (hooks.contains(postUpdateHook)) {
|
|
|
|
throw new IllegalArgumentException("Adding an already-registered hook");
|
|
|
|
}
|
|
|
|
|
|
|
|
hooks.add((BiConsumer<Urn, RecordTemplate>) postUpdateHook);
|
|
|
|
_aspectPostUpdateHooksMap.put(aspectClass, hooks);
|
|
|
|
}
|
|
|
|
|
2019-08-31 20:51:14 -07:00
|
|
|
/**
|
|
|
|
* Sets the {@link EqualityTester} for a specific aspect type.
|
|
|
|
*/
|
|
|
|
public <ASPECT extends RecordTemplate> void setEqualityTester(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull EqualityTester<ASPECT> tester) {
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
_aspectEqualityTesterMap.put(aspectClass, tester);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the {@link EqualityTester} for an aspect, or {@link DefaultEqualityTester} if none is registered.
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public <ASPECT extends RecordTemplate> EqualityTester<ASPECT> getEqualityTester(@Nonnull Class<ASPECT> aspectClass) {
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
return (EqualityTester<ASPECT>) _aspectEqualityTesterMap.computeIfAbsent(aspectClass,
|
|
|
|
key -> new DefaultEqualityTester<ASPECT>());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enables or disables model validation before persisting.
|
|
|
|
*/
|
|
|
|
public void enableModelValidationOnWrite(boolean enabled) {
|
|
|
|
_modelValidationOnWrite = enabled;
|
|
|
|
}
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
/**
|
|
|
|
* Sets if MAE should be always emitted after each update even if there's no actual value change.
|
|
|
|
*/
|
|
|
|
public void setAlwaysEmitAuditEvent(boolean alwaysEmitAuditEvent) {
|
|
|
|
_alwaysEmitAuditEvent = alwaysEmitAuditEvent;
|
|
|
|
}
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
/**
|
|
|
|
* Sets if aspect specific MAE should be always emitted after each update even if there's no actual value change.
|
|
|
|
*/
|
|
|
|
public void setEmitAspectSpecificAuditEvent(boolean emitAspectSpecificAuditEvent) {
|
|
|
|
_emitAspectSpecificAuditEvent = emitAspectSpecificAuditEvent;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets if writes to local secondary index enabled
|
|
|
|
* @deprecated Use {@link #enableLocalSecondaryIndex(boolean)} instead
|
|
|
|
*/
|
|
|
|
public void setWriteToLocalSecondaryIndex(boolean writeToLocalSecondaryIndex) {
|
|
|
|
_enableLocalSecondaryIndex = writeToLocalSecondaryIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enables reads from and writes to local secondary index
|
|
|
|
*/
|
|
|
|
public void enableLocalSecondaryIndex(boolean enableLocalSecondaryIndex) {
|
|
|
|
_enableLocalSecondaryIndex = enableLocalSecondaryIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets if reads and writes to local secondary index are enabled
|
|
|
|
*/
|
|
|
|
public boolean isLocalSecondaryIndexEnabled() {
|
|
|
|
return _enableLocalSecondaryIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets if local secondary index backfilling is enabled
|
|
|
|
*/
|
|
|
|
public void setBackfillLocalSecondaryIndex(boolean backfillLocalSecondaryIndex) {
|
|
|
|
_backfillLocalSecondaryIndex = backfillLocalSecondaryIndex;
|
|
|
|
}
|
|
|
|
|
2019-08-31 20:51:14 -07:00
|
|
|
/**
|
|
|
|
* Adds a new version of aspect for an entity.
|
|
|
|
*
|
|
|
|
* The new aspect will have an automatically assigned version number, which is guaranteed to be positive and
|
|
|
|
* monotonically increasing. Older versions of aspect will be purged automatically based on the retention setting.
|
|
|
|
* A MetadataAuditEvent is also emitted if there's an actual update.
|
|
|
|
*
|
|
|
|
* @param urn the URN for the entity the aspect is attached to
|
|
|
|
* @param auditStamp the audit stamp for the operation
|
|
|
|
* @param updateLambda a lambda expression that takes the previous version of aspect and returns the new version
|
2020-03-26 21:53:29 -07:00
|
|
|
* @return {@link RecordTemplate} of the new value of aspect
|
2019-08-31 20:51:14 -07:00
|
|
|
*/
|
2019-10-02 11:08:52 -07:00
|
|
|
@Nonnull
|
2020-06-11 10:21:51 -07:00
|
|
|
public <ASPECT extends RecordTemplate> ASPECT add(@Nonnull URN urn, @Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull Function<Optional<ASPECT>, ASPECT> updateLambda, @Nonnull AuditStamp auditStamp,
|
2019-08-31 20:51:14 -07:00
|
|
|
int maxTransactionRetry) {
|
|
|
|
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
|
|
|
|
final EqualityTester<ASPECT> equalityTester = getEqualityTester(aspectClass);
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
final AddResult<ASPECT> result = runInTransactionWithRetry(() -> {
|
2019-08-31 20:51:14 -07:00
|
|
|
// 1. Compute newValue based on oldValue
|
2020-06-11 10:21:51 -07:00
|
|
|
AspectEntry<ASPECT> latest = getLatest(urn, aspectClass);
|
|
|
|
final ASPECT oldValue = latest == null ? null : latest.getAspect();
|
|
|
|
final ASPECT newValue = updateLambda.apply(Optional.ofNullable(oldValue));
|
2019-08-31 20:51:14 -07:00
|
|
|
checkValidAspect(newValue.getClass());
|
|
|
|
if (_modelValidationOnWrite) {
|
|
|
|
validateAgainstSchema(newValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2. Skip saving if there's no actual change
|
|
|
|
if (oldValue != null && equalityTester.equals(oldValue, newValue)) {
|
2020-06-11 10:21:51 -07:00
|
|
|
return new AddResult<>(oldValue, oldValue);
|
2019-08-31 20:51:14 -07:00
|
|
|
}
|
|
|
|
|
2019-12-13 11:46:49 -08:00
|
|
|
// 3. Save the newValue as the latest version
|
|
|
|
long largestVersion =
|
|
|
|
saveLatest(urn, aspectClass, oldValue, latest == null ? null : latest.getExtraInfo().getAudit(), newValue,
|
|
|
|
auditStamp);
|
2019-08-31 20:51:14 -07:00
|
|
|
|
2019-12-13 11:46:49 -08:00
|
|
|
// 4. Apply retention policy
|
2019-08-31 20:51:14 -07:00
|
|
|
applyRetention(urn, aspectClass, getRetention(aspectClass), largestVersion);
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
// 5. Save to local secondary index
|
|
|
|
if (_enableLocalSecondaryIndex) {
|
|
|
|
saveToLocalSecondaryIndex(urn, newValue, largestVersion);
|
|
|
|
}
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
return new AddResult<>(oldValue, newValue);
|
2019-08-31 20:51:14 -07:00
|
|
|
}, maxTransactionRetry);
|
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
final ASPECT oldValue = result.getOldValue();
|
|
|
|
final ASPECT newValue = result.getNewValue();
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
// 6. Produce MAE after a successful update
|
2020-06-11 10:21:51 -07:00
|
|
|
if (_alwaysEmitAuditEvent || oldValue != newValue) {
|
|
|
|
_producer.produceMetadataAuditEvent(urn, oldValue, newValue);
|
|
|
|
}
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
// TODO: Replace step 6 with step 6.1 with diff option after pipeline is fully migrated to aspect specific events.
|
|
|
|
// 6.1. Produce aspect specific MAE after a successful update
|
|
|
|
if (_emitAspectSpecificAuditEvent) {
|
|
|
|
_producer.produceAspectSpecificMetadataAuditEvent(urn, oldValue, newValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
// 7. Invoke post-update hooks if there's any
|
2020-06-11 10:21:51 -07:00
|
|
|
if (_aspectPostUpdateHooksMap.containsKey(aspectClass)) {
|
|
|
|
_aspectPostUpdateHooksMap.get(aspectClass).forEach(hook -> hook.accept(urn, newValue));
|
|
|
|
}
|
2020-03-26 21:53:29 -07:00
|
|
|
|
2020-06-11 10:21:51 -07:00
|
|
|
return newValue;
|
2019-08-31 20:51:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to {@link #add(Urn, Class, Function, AuditStamp, int)} but uses the default maximum transaction retry.
|
|
|
|
*/
|
2019-10-02 11:08:52 -07:00
|
|
|
@Nonnull
|
2020-06-11 10:21:51 -07:00
|
|
|
public <ASPECT extends RecordTemplate> ASPECT add(@Nonnull URN urn, @Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull Function<Optional<ASPECT>, ASPECT> updateLambda, @Nonnull AuditStamp auditStamp) {
|
2019-10-02 11:08:52 -07:00
|
|
|
return add(urn, aspectClass, updateLambda, auditStamp, DEFAULT_MAX_TRANSACTION_RETRY);
|
2019-08-31 20:51:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to {@link #add(Urn, Class, Function, AuditStamp)} but takes the new value directly.
|
|
|
|
*/
|
2019-10-02 11:08:52 -07:00
|
|
|
@Nonnull
|
2020-06-11 10:21:51 -07:00
|
|
|
public <ASPECT extends RecordTemplate> ASPECT add(@Nonnull URN urn, @Nonnull ASPECT newValue,
|
2019-08-31 20:51:14 -07:00
|
|
|
@Nonnull AuditStamp auditStamp) {
|
2020-06-11 10:21:51 -07:00
|
|
|
return add(urn, (Class<ASPECT>) newValue.getClass(), ignored -> newValue, auditStamp);
|
2019-08-31 20:51:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
private <ASPECT extends RecordTemplate> void applyRetention(@Nonnull URN urn, @Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull Retention retention, long largestVersion) {
|
|
|
|
if (retention instanceof IndefiniteRetention) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (retention instanceof VersionBasedRetention) {
|
|
|
|
applyVersionBasedRetention(aspectClass, urn, (VersionBasedRetention) retention, largestVersion);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (retention instanceof TimeBasedRetention) {
|
|
|
|
applyTimeBasedRetention(aspectClass, urn, (TimeBasedRetention) retention, _clock.millis());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-13 11:46:49 -08:00
|
|
|
/**
|
|
|
|
* Saves the latest aspect
|
|
|
|
*
|
|
|
|
* @param urn the URN for the entity the aspect is attached to
|
|
|
|
* @param aspectClass the aspectClass of the aspect being saved
|
|
|
|
* @param oldEntry {@link RecordTemplate} of the previous latest value of aspect, null if new value is the first version
|
|
|
|
* @param oldAuditStamp the audit stamp of the previous latest aspect, null if new value is the first version
|
|
|
|
* @param newEntry {@link RecordTemplate} of the new latest value of aspect
|
|
|
|
* @param newAuditStamp the audit stamp for the operation
|
|
|
|
* @return the largestVersion
|
|
|
|
*/
|
2020-06-11 10:21:51 -07:00
|
|
|
protected abstract <ASPECT extends RecordTemplate> long saveLatest(@Nonnull URN urn,
|
|
|
|
@Nonnull Class<ASPECT> aspectClass, @Nullable ASPECT oldEntry, @Nullable AuditStamp oldAuditStamp,
|
|
|
|
@Nonnull ASPECT newEntry, @Nonnull AuditStamp newAuditStamp);
|
2019-12-13 11:46:49 -08:00
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
/**
|
|
|
|
* Saves the new value of an aspect to local secondary index
|
|
|
|
*
|
|
|
|
* @param urn the URN for the entity the aspect is attached to
|
|
|
|
* @param newValue {@link RecordTemplate} of the new value of aspect
|
|
|
|
* @param version version of the aspect
|
|
|
|
*/
|
|
|
|
protected abstract <ASPECT extends RecordTemplate> void saveToLocalSecondaryIndex(@Nonnull URN urn,
|
|
|
|
@Nullable ASPECT newValue, long version);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns list of urns from local secondary index that satisfy the given filter conditions.
|
|
|
|
*
|
|
|
|
* @param indexFilter {@link IndexFilter} containing filter conditions to be applied
|
|
|
|
* @param lastUrn last urn of the previous fetched page. For the first page, this should be set as NULL
|
|
|
|
* @param pageSize maximum number of distinct urns to return
|
|
|
|
* @return {@link ListResult} of urns from local secondary index that satisfy the given filter conditions
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public abstract ListResult<Urn> listUrns(@Nonnull IndexFilter indexFilter, @Nullable URN lastUrn, int pageSize);
|
|
|
|
|
2019-08-31 20:51:14 -07:00
|
|
|
/**
|
|
|
|
* Runs the given lambda expression in a transaction with a limited number of retries.
|
|
|
|
*
|
|
|
|
* @param block the lambda expression to run
|
|
|
|
* @param maxTransactionRetry maximum number of transaction retries before throwing an exception
|
|
|
|
* @param <T> type for the result object
|
|
|
|
* @return the result object from a successfully committed transaction
|
|
|
|
*/
|
2020-03-26 21:53:29 -07:00
|
|
|
@Nonnull
|
2019-08-31 20:51:14 -07:00
|
|
|
protected abstract <T> T runInTransactionWithRetry(@Nonnull Supplier<T> block, int maxTransactionRetry);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the latest version of a specific aspect type for an entity
|
|
|
|
*
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param aspectClass the type of aspect to get
|
|
|
|
* @return the latest version for the aspect type, or null if there's none
|
|
|
|
*/
|
|
|
|
@Nullable
|
2020-06-11 10:21:51 -07:00
|
|
|
protected abstract <ASPECT extends RecordTemplate> AspectEntry<ASPECT> getLatest(@Nonnull URN urn,
|
2019-08-31 20:51:14 -07:00
|
|
|
@Nonnull Class<ASPECT> aspectClass);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the next version to use for an entity's specific aspect type.
|
|
|
|
*
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param aspectClass the type of aspect to get
|
|
|
|
* @return the next version number to use, or {@link #LATEST_VERSION} if there's no previous versions
|
|
|
|
*/
|
|
|
|
protected abstract <ASPECT extends RecordTemplate> long getNextVersion(@Nonnull URN urn,
|
|
|
|
@Nonnull Class<ASPECT> aspectClass);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Saves an aspect for an entity with specific version & {@link AuditStamp}.
|
|
|
|
*
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param value the aspect to save
|
|
|
|
* @param auditStamp the {@link AuditStamp} for the aspect
|
|
|
|
* @param version the version for the aspect
|
|
|
|
* @param insert use insert, instead of update, operation to save
|
|
|
|
*/
|
|
|
|
protected abstract void save(@Nonnull URN urn, @Nonnull RecordTemplate value, @Nonnull AuditStamp auditStamp,
|
|
|
|
long version, boolean insert);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Applies version-based retention against a specific aspect type for an entity
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of aspect to apply retention to
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param retention the retention configuration
|
|
|
|
* @param largestVersion the largest version number for the aspect type
|
|
|
|
*/
|
|
|
|
protected abstract <ASPECT extends RecordTemplate> void applyVersionBasedRetention(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull URN urn, @Nonnull VersionBasedRetention retention, long largestVersion);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Applies time-based retention against a specific aspect type for an entity
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of aspect to apply retention to
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param retention the retention configuration
|
|
|
|
* @param currentTime the current timestamp
|
|
|
|
*/
|
|
|
|
protected abstract <ASPECT extends RecordTemplate> void applyTimeBasedRetention(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull URN urn, @Nonnull TimeBasedRetention retention, long currentTime);
|
|
|
|
|
|
|
|
/**
|
2020-08-19 16:06:29 -07:00
|
|
|
* Emits backfill MAE for the latest version of an aspect of an entity and also backfills local
|
|
|
|
* secondary index if writes & backfill enabled
|
2019-08-31 20:51:14 -07:00
|
|
|
*
|
|
|
|
* @param aspectClass the type of aspect to backfill
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
* @return the aspect emitted in the backfill message
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public <ASPECT extends RecordTemplate> Optional<ASPECT> backfill(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull URN urn) {
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
Optional<ASPECT> aspect = get(aspectClass, urn, LATEST_VERSION);
|
2020-08-19 16:06:29 -07:00
|
|
|
aspect.ifPresent(value -> backfill(value, urn));
|
2019-08-31 20:51:14 -07:00
|
|
|
return aspect;
|
|
|
|
}
|
|
|
|
|
2020-08-19 16:06:29 -07:00
|
|
|
/**
|
|
|
|
* Similar to {@link #backfill(Class, URN)} but gets a set of aspect classes and do a batch backfill
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public Map<Class<? extends RecordTemplate>, Optional<? extends RecordTemplate>> backfill(
|
|
|
|
@Nonnull Set<Class<? extends RecordTemplate>> aspectClasses, @Nonnull URN urn) {
|
|
|
|
checkValidAspects(aspectClasses);
|
|
|
|
Map<Class<? extends RecordTemplate>, Optional<? extends RecordTemplate>> aspects = get(aspectClasses, urn);
|
|
|
|
aspects.forEach((aspectClass, aspect) -> aspect.ifPresent(value -> backfill(value, urn)));
|
|
|
|
return aspects;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to {@link #backfill(Class, URN)} but gets a set of urns and do a batch backfill
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public <ASPECT extends RecordTemplate> Map<URN, Optional<ASPECT>> backfill(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull Set<URN> urns) {
|
|
|
|
checkValidAspect(aspectClass);
|
|
|
|
final Map<URN, Optional<ASPECT>> urnToAspects = get(aspectClass, urns);
|
|
|
|
urnToAspects.forEach((urn, aspect) -> aspect.ifPresent(value -> backfill(value, urn)));
|
|
|
|
return urnToAspects;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to {@link #backfill(Class, URN)} but gets a set of aspect classes and a set of URNs and do a batch backfill
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public Map<URN, Map<Class<? extends RecordTemplate>, Optional<? extends RecordTemplate>>> backfill(
|
|
|
|
@Nonnull Set<Class<? extends RecordTemplate>> aspectClasses, @Nonnull Set<URN> urns) {
|
|
|
|
checkValidAspects(aspectClasses);
|
|
|
|
final Map<URN, Map<Class<? extends RecordTemplate>, Optional<? extends RecordTemplate>>> urnToAspects = get(aspectClasses, urns);
|
|
|
|
urnToAspects.forEach((urn, aspects) -> {
|
|
|
|
aspects.forEach((aspectClass, aspect) -> aspect.ifPresent(value -> backfill(value, urn)));
|
|
|
|
});
|
|
|
|
return urnToAspects;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emits backfill MAE for an aspect of an entity and also backfills local secondary index if writes & backfill enabled
|
|
|
|
*
|
|
|
|
* @param aspect aspect to backfill
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
*/
|
|
|
|
private <ASPECT extends RecordTemplate> void backfill(@Nonnull ASPECT aspect, @Nonnull URN urn) {
|
|
|
|
// Backfill local secondary index as well if writes & backfill enabled
|
|
|
|
if (_enableLocalSecondaryIndex && _backfillLocalSecondaryIndex) {
|
|
|
|
saveToLocalSecondaryIndex(urn, aspect, FIRST_VERSION);
|
|
|
|
}
|
|
|
|
_producer.produceMetadataAuditEvent(urn, aspect, aspect);
|
|
|
|
}
|
|
|
|
|
2019-08-31 20:51:14 -07:00
|
|
|
/**
|
|
|
|
* Paginates over all available versions of an aspect for an entity.
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of the aspect to query
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param start the starting offset of the page
|
|
|
|
* @param pageSize the size of the page
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
* @return a {@link ListResult} containing a list of version numbers and other pagination information
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public abstract <ASPECT extends RecordTemplate> ListResult<Long> listVersions(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull URN urn, int start, int pageSize);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Paginates over all URNs for entities that have a specific aspect.
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of the aspect to query
|
|
|
|
* @param start the starting offset of the page
|
|
|
|
* @param pageSize the size of the page
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
* @return a {@link ListResult} containing a list of URN and other pagination information
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public abstract <ASPECT extends RecordTemplate> ListResult<Urn> listUrns(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
int start, int pageSize);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Paginates over all versions of an aspect for a specific Urn.
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of the aspect to query
|
|
|
|
* @param urn {@link Urn} for the entity
|
|
|
|
* @param start the starting offset of the page
|
|
|
|
* @param pageSize the size of the page
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
* @return a {@link ListResult} containing a list of aspects and other pagination information
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public abstract <ASPECT extends RecordTemplate> ListResult<ASPECT> list(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
@Nonnull URN urn, int start, int pageSize);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Paginates over a specific version of a specific aspect for all Urns
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of the aspect to query
|
|
|
|
* @param version the version of the aspect
|
|
|
|
* @param start the starting offset of the page
|
|
|
|
* @param pageSize the size of the page
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
* @return a {@link ListResult} containing a list of aspects and other pagination information
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public abstract <ASPECT extends RecordTemplate> ListResult<ASPECT> list(@Nonnull Class<ASPECT> aspectClass,
|
|
|
|
long version, int start, int pageSize);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Paginates over the latest version of a specific aspect for all Urns
|
|
|
|
*
|
|
|
|
* @param aspectClass the type of the aspect to query
|
|
|
|
* @param start the starting offset of the page
|
|
|
|
* @param pageSize the size of the page
|
|
|
|
* @param <ASPECT> must be a supported aspect type in {@code ASPECT_UNION}.
|
|
|
|
* @return a {@link ListResult} containing a list of aspects and other pagination information
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public abstract <ASPECT extends RecordTemplate> ListResult<ASPECT> list(@Nonnull Class<ASPECT> aspectClass, int start,
|
|
|
|
int pageSize);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates a new string ID that's guaranteed to be globally unique.
|
|
|
|
*/
|
|
|
|
@Nonnull
|
|
|
|
public String newStringId() {
|
|
|
|
return UUID.randomUUID().toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Generates a new numeric ID that's guaranteed to increase monotonically within the given namespace.
|
|
|
|
*/
|
|
|
|
public abstract long newNumericId(@Nonnull String namespace, int maxTransactionRetry);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to {@link #newNumericId(String, int)} but uses default maximum transaction retry count.
|
|
|
|
*/
|
|
|
|
public long newNumericId(@Nonnull String namespace) {
|
|
|
|
return newNumericId(namespace, DEFAULT_MAX_TRANSACTION_RETRY);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Similar to {@link #newNumericId(String, int)} but uses a single global namespace
|
|
|
|
*/
|
|
|
|
public long newNumericId() {
|
|
|
|
return newNumericId(DEFAULT_ID_NAMESPACE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validates a model against its schema.
|
|
|
|
*/
|
|
|
|
protected void validateAgainstSchema(@Nonnull RecordTemplate model) {
|
|
|
|
ValidationResult result = ValidateDataAgainstSchema.validate(model,
|
|
|
|
new ValidationOptions(RequiredMode.CAN_BE_ABSENT_IF_HAS_DEFAULT, CoercionMode.NORMAL,
|
|
|
|
UnrecognizedFieldMode.DISALLOW));
|
|
|
|
|
|
|
|
if (!result.isValid()) {
|
|
|
|
throw new ModelValidationException(result.getMessages().toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|