Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(ui/backend/openapi/docs) : Add support for Business Attributes #9863

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
81 commits
Select commit Hold shift + click to select a range
d443b07
business-attribute: graphql crud resolvers and metadata models
deepgarg-visa Dec 17, 2023
830294d
business-attribute: add businessAttributeService
deepgarg-visa Dec 18, 2023
a3d81d1
business-attribute: add resolvers to add/remove businessattribute to …
deepgarg-visa Dec 21, 2023
0e84213
business-attribute: Created initial version of Business Attribute Scr…
PrithviVISA Jan 3, 2024
f02ffee
Merge pull request #3 from PrithviVISA/business-attributes-v1
PrithviVISA Jan 5, 2024
5606c91
Added lastModified and Created for business Attribute
PrithviVISA Jan 10, 2024
f988a10
Merge branch 'business-attributes-v1' of https://github.com/deepgarg-…
deepgarg-visa Jan 10, 2024
583d647
business-attribute: refactoring and added lastmodified
deepgarg-visa Jan 10, 2024
41b6731
business-attribute: junits for business attribute resolvers
deepgarg-visa Jan 11, 2024
730bb5a
Business Attribute Association
aditigup-visa Jan 16, 2024
618e691
Merge pull request #5 from aditigup-visa/business-attributes-v1
deepgarg-visa Jan 17, 2024
c1f7680
Business Attribute related entities and css
aditigup-visa Jan 19, 2024
fed0202
Merge pull request #6 from aditigup-visa/business-attributes-v1
deepgarg-visa Jan 21, 2024
58d364e
businessattribute: openApi support
deepgarg-visa Jan 22, 2024
b127d4f
businessattribute: businessattribute custom properties fetching
deepgarg-visa Jan 22, 2024
1ab94f2
Business Attribute Minor Issues
aditigup-visa Jan 23, 2024
fcbc9c1
Business Attribute Minor Issues
aditigup-visa Jan 24, 2024
b7aea86
Merge pull request #7 from aditigup-visa/business-attributes-v1
deepgarg-visa Jan 24, 2024
fa23494
businessattribute: generate platform events for business attributes
deepgarg-visa Jan 27, 2024
aaf5182
businessattribute: metadata access management for Business Attribute
deepgarg-visa Feb 1, 2024
24baf28
businessattribute: modifify policies.json
deepgarg-visa Feb 2, 2024
8bc9669
businessattribute: generate lifecycle platform events
deepgarg-visa Feb 2, 2024
249adec
Cypress Test Cases, Preview Test Case, updating delete BA api, Removi…
aditigup-visa Feb 5, 2024
70054ea
Enabling editing data type
aditigup-visa Feb 5, 2024
e0ff89d
Merge pull request #10 from aditigup-visa/business-attributes-v1
deepgarg-visa Feb 5, 2024
3aee567
Merge branch 'master' into business-attributes
deepgarg-visa Feb 7, 2024
0ba227c
businessattribute: resolve merge conflicts with master
deepgarg-visa Feb 7, 2024
8f5ea63
business-attribute: businessattributes change propagation platform ev…
aabharti-visa Feb 2, 2024
7877912
Business attribute UI changes merged with master
aditigup-visa Feb 12, 2024
5e78afd
Merge remote-tracking branch 'datahub/business-attributes' into busin…
aditigup-visa Feb 12, 2024
e6b9d24
Merge remote-tracking branch 'datahub/business-attributes' into busin…
aditigup-visa Feb 12, 2024
dbc0a40
Merge branch 'business-attributes' of https://github.com/deepgarg-vis…
deepgarg-visa Feb 12, 2024
950f40b
business-attribute: documentation
deepgarg-visa Feb 13, 2024
6b25744
Business Attribute : SearchableRef Annotation Elastic Insert and Search
hisingh-visa Feb 13, 2024
07b9494
business-attribute: update documentation
deepgarg-visa Feb 15, 2024
f5420e9
Merge branch 'master' into business-attributes
deepgarg-visa Feb 19, 2024
9b65c79
Merge branch 'master' into business-attributes
deepgarg-visa Feb 28, 2024
6c05227
business-attributes-propagation-tests
aabharti-visa Mar 1, 2024
3cec180
Merge pull request #23 from deepgarg-visa/business-attributes-propaga…
deepgarg-visa Mar 1, 2024
3b2c34c
business attributes: fixing dependency issue
deepgarg-visa Mar 4, 2024
c68bcbc
Merge branch 'master' into business-attributes
deepgarg-visa Mar 4, 2024
3669141
business-attribute: fix issues due to merge conflicts
deepgarg-visa Mar 6, 2024
13e5aee
Business Attribute : SearchableRef Unit Test
hisingh-visa Dec 20, 2023
6b242a0
fix(ui): business-attribute: Dulicate Gloassary term rendering
amit-visa Mar 6, 2024
77ba88f
Merge pull request #24 from hisingh-visa/business-attributes-ref-unit…
deepgarg-visa Mar 7, 2024
1c7c6e5
Bug Fix : SearchableRef Search Reference Field
hisingh-visa Mar 4, 2024
f394657
fix(ui): business-attribute: institutional memory support
deepgarg-visa Mar 7, 2024
7026939
Merge pull request #27 from hisingh-visa/business-attributes-searchab…
deepgarg-visa Mar 8, 2024
4562978
Fix : Business Attribute SearchableRef Search Depth
hisingh-visa Mar 19, 2024
a1294bf
business-attributes: review comments - business attributes lives in s…
deepgarg-visa Mar 19, 2024
7365b43
Merge branch 'master' into business-attributes
deepgarg-visa Mar 20, 2024
0bb7aa4
business-attributes: review comments - refactor businessAttribute pro…
deepgarg-visa Mar 20, 2024
0cc49c0
feature flag for business attribute GENAI=YES
aabharti-visa Mar 21, 2024
b7d827f
Merge branch 'business-attributes' of https://github.com/deepgarg-vis…
aabharti-visa Mar 21, 2024
ba03044
rebase
aabharti-visa Mar 21, 2024
b497277
Merge pull request #29 from deepgarg-visa/feature-flag-for-business-a…
deepgarg-visa Mar 21, 2024
8cb0811
Merge pull request #28 from hisingh-visa/business-attributes-depth_fix
deepgarg-visa Mar 21, 2024
e2651a8
business-attributes: support for custom urn with graphql
deepgarg-visa Mar 22, 2024
cb54236
Merge branch 'master' into business-attributes
deepgarg-visa Mar 26, 2024
8411393
business-attributes: Update Business Attribute changes as per current…
deepgarg-visa Mar 27, 2024
89d9e93
Merge branch 'master' into business-attributes
deepgarg-visa Mar 28, 2024
baa052b
Business Attributes: UI Schema Field Entity Changes
kartikey-visa Mar 28, 2024
56b3066
Merge pull request #30 from kartikey-visa/feature/business-attributes…
deepgarg-visa Apr 1, 2024
ee865dd
feat(search): Add SchemaFieldEntity to search functionality
amit-visa Apr 2, 2024
7997a12
Business Attributes: UI Show|Hide Feature Flag Implementation | Modif…
kartikey-visa Mar 28, 2024
d05f792
Merge pull request #31 from kartikey-visa/feature/ui-business-attribu…
deepgarg-visa Apr 3, 2024
c277123
Merge pull request #32 from amit-visa/schema-field-search
deepgarg-visa Apr 3, 2024
982f7d5
Business Attributes: Customized URNs Support for Business Attributes
kartikey-visa Apr 3, 2024
998aeb2
Merge pull request #33 from kartikey-visa/feature/added-support-for-c…
deepgarg-visa Apr 3, 2024
8195acf
Business Attributes: Fixed Schema Field Cypress Test Cases
kartikey-visa Apr 3, 2024
d43a0bc
Merge pull request #34 from kartikey-visa/test/fixed-cypress-test-cas…
deepgarg-visa Apr 3, 2024
799759e
business-attributes: introduce fieldNamealias for schemafieldentity
deepgarg-visa Apr 4, 2024
dafa1c7
feat(search/schema_field): Update schema field card in search results
amit-visa Apr 4, 2024
4e0b9e4
Merge pull request #38 from amit-visa/feature/schema-field-attr
deepgarg-visa Apr 4, 2024
ec2d7ea
Business Attributes: Feature Flag Cypress Test Cases Fix
kartikey-visa Apr 3, 2024
0e39770
Merge pull request #36 from kartikey-visa/test/feature-flag-cypress-t…
deepgarg-visa Apr 4, 2024
cd15208
business-attribute-flag-for-openapi
aabharti-visa Apr 4, 2024
b2854bd
Merge pull request #37 from aabharti-visa/feature-flag-for-business-a…
deepgarg-visa Apr 4, 2024
1c2ba53
Merge branch 'master' into business-attributes
deepgarg-visa Apr 14, 2024
3267fdf
business-attributes: changes due to merge resolve conflicts
deepgarg-visa Apr 14, 2024
6640d2b
business-attributes: fix failing frontend test
deepgarg-visa Apr 15, 2024
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
2 changes: 1 addition & 1 deletion buildSrc/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -25,4 +25,4 @@ dependencies {

compileOnly 'org.projectlombok:lombok:1.18.30'
annotationProcessor 'org.projectlombok:lombok:1.18.30'
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@ public class OpenApiEntities {
.add("dataProductProperties")
.add("institutionalMemory")
.add("forms").add("formInfo").add("dynamicFormAssignment")
.add("businessAttributeInfo")
.build();

private final static ImmutableSet<String> ENTITY_EXCLUSIONS = ImmutableSet.<String>builder()
Expand Down

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
import com.linkedin.metadata.graph.SiblingGraphService;
import com.linkedin.metadata.models.registry.EntityRegistry;
import com.linkedin.metadata.recommendation.RecommendationsService;
import com.linkedin.metadata.service.BusinessAttributeService;
import com.linkedin.metadata.service.DataProductService;
import com.linkedin.metadata.service.ERModelRelationshipService;
import com.linkedin.metadata.service.FormService;
Expand Down Expand Up @@ -81,6 +82,7 @@ public class GmsGraphQLEngineArgs {
RestrictedService restrictedService;
int graphQLQueryComplexityLimit;
int graphQLQueryDepthLimit;
BusinessAttributeService businessAttributeService;

// any fork specific args should go below this line
}
Original file line number Diff line number Diff line change
Expand Up @@ -19,4 +19,5 @@ public class FeatureFlags {
private boolean showAccessManagement = false;
private boolean nestedDomainsEnabled = false;
private boolean schemaFieldEntityFetchEnabled = false;
private boolean businessAttributeEntityEnabled = false;
}
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
import com.linkedin.datahub.graphql.generated.AuthenticatedUser;
import com.linkedin.datahub.graphql.generated.CorpUser;
import com.linkedin.datahub.graphql.generated.PlatformPrivileges;
import com.linkedin.datahub.graphql.resolvers.businessattribute.BusinessAttributeAuthorizationUtils;
import com.linkedin.datahub.graphql.types.corpuser.mappers.CorpUserMapper;
import com.linkedin.entity.EntityResponse;
import com.linkedin.entity.client.EntityClient;
Expand Down Expand Up @@ -86,7 +87,10 @@ public CompletableFuture<AuthenticatedUser> get(DataFetchingEnvironment environm
AuthorizationUtils.canManageOwnershipTypes(context));
platformPrivileges.setManageGlobalAnnouncements(
AuthorizationUtils.canManageGlobalAnnouncements(context));

platformPrivileges.setCreateBusinessAttributes(
BusinessAttributeAuthorizationUtils.canCreateBusinessAttribute(context));
platformPrivileges.setManageBusinessAttributes(
BusinessAttributeAuthorizationUtils.canManageBusinessAttribute(context));
// Construct and return authenticated user object.
final AuthenticatedUser authUser = new AuthenticatedUser();
authUser.setCorpUser(corpUser);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
package com.linkedin.datahub.graphql.resolvers.businessattribute;

import static com.linkedin.datahub.graphql.resolvers.ResolverUtils.bindArgument;
import static com.linkedin.datahub.graphql.resolvers.mutate.MutationUtils.buildMetadataChangeProposalWithUrn;
import static com.linkedin.metadata.Constants.BUSINESS_ATTRIBUTE_ASPECT;

import com.linkedin.businessattribute.BusinessAttributeAssociation;
import com.linkedin.businessattribute.BusinessAttributes;
import com.linkedin.common.urn.BusinessAttributeUrn;
import com.linkedin.common.urn.Urn;
import com.linkedin.common.urn.UrnUtils;
import com.linkedin.datahub.graphql.QueryContext;
import com.linkedin.datahub.graphql.generated.AddBusinessAttributeInput;
import com.linkedin.datahub.graphql.generated.ResourceRefInput;
import com.linkedin.metadata.entity.EntityService;
import com.linkedin.metadata.entity.EntityUtils;
import com.linkedin.mxe.MetadataChangeProposal;
import graphql.schema.DataFetcher;
import graphql.schema.DataFetchingEnvironment;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;

@Slf4j
@RequiredArgsConstructor
public class AddBusinessAttributeResolver implements DataFetcher<CompletableFuture<Boolean>> {
private final EntityService entityService;

@Override
public CompletableFuture<Boolean> get(DataFetchingEnvironment environment) throws Exception {
final QueryContext context = environment.getContext();
final AddBusinessAttributeInput input =
bindArgument(environment.getArgument("input"), AddBusinessAttributeInput.class);
final Urn businessAttributeUrn = UrnUtils.getUrn(input.getBusinessAttributeUrn());
final List<ResourceRefInput> resourceRefInputs = input.getResourceUrn();
validateBusinessAttribute(businessAttributeUrn);
return CompletableFuture.supplyAsync(
() -> {
try {
addBusinessAttributeToResource(
businessAttributeUrn,
resourceRefInputs,
UrnUtils.getUrn(context.getActorUrn()),
entityService);
return true;
} catch (Exception e) {
log.error(
String.format(
"Failed to add Business Attribute %s to resources %s",
businessAttributeUrn, resourceRefInputs));
throw new RuntimeException(
String.format(
"Failed to add Business Attribute %s to resources %s",
businessAttributeUrn, resourceRefInputs),
e);
}
});
}

private void validateBusinessAttribute(Urn businessAttributeUrn) {
if (!entityService.exists(businessAttributeUrn, true)) {
throw new IllegalArgumentException(
String.format("This urn does not exist: %s", businessAttributeUrn));
}
}

private void addBusinessAttributeToResource(
Urn businessAttributeUrn,
List<ResourceRefInput> resourceRefInputs,
Urn actorUrn,
EntityService entityService)
throws URISyntaxException {
List<MetadataChangeProposal> proposals = new ArrayList<>();
for (ResourceRefInput resourceRefInput : resourceRefInputs) {
proposals.add(
buildAddBusinessAttributeToEntityProposal(
businessAttributeUrn, resourceRefInput, entityService, actorUrn));
}
EntityUtils.ingestChangeProposals(proposals, entityService, actorUrn, false);
}

private MetadataChangeProposal buildAddBusinessAttributeToEntityProposal(
Urn businessAttributeUrn,
ResourceRefInput resource,
EntityService entityService,
Urn actorUrn)
throws URISyntaxException {
BusinessAttributes businessAttributes =
(BusinessAttributes)
EntityUtils.getAspectFromEntity(
resource.getResourceUrn(),
BUSINESS_ATTRIBUTE_ASPECT,
entityService,
new BusinessAttributes());
if (!businessAttributes.hasBusinessAttribute()) {
businessAttributes.setBusinessAttribute(new BusinessAttributeAssociation());
}
BusinessAttributeAssociation businessAttributeAssociation =
businessAttributes.getBusinessAttribute();
businessAttributeAssociation.setBusinessAttributeUrn(
BusinessAttributeUrn.createFromUrn(businessAttributeUrn));
businessAttributes.setBusinessAttribute(businessAttributeAssociation);
return buildMetadataChangeProposalWithUrn(
UrnUtils.getUrn(resource.getResourceUrn()), BUSINESS_ATTRIBUTE_ASPECT, businessAttributes);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
package com.linkedin.datahub.graphql.resolvers.businessattribute;

import com.datahub.authorization.AuthUtil;
import com.datahub.authorization.ConjunctivePrivilegeGroup;
import com.datahub.authorization.DisjunctivePrivilegeGroup;
import com.google.common.collect.ImmutableList;
import com.linkedin.datahub.graphql.QueryContext;
import com.linkedin.metadata.authorization.PoliciesConfig;
import javax.annotation.Nonnull;

public class BusinessAttributeAuthorizationUtils {
private BusinessAttributeAuthorizationUtils() {}

public static boolean canCreateBusinessAttribute(@Nonnull QueryContext context) {
final DisjunctivePrivilegeGroup orPrivilegeGroups =
new DisjunctivePrivilegeGroup(
ImmutableList.of(
new ConjunctivePrivilegeGroup(
ImmutableList.of(PoliciesConfig.CREATE_BUSINESS_ATTRIBUTE_PRIVILEGE.getType())),
new ConjunctivePrivilegeGroup(
ImmutableList.of(
PoliciesConfig.MANAGE_BUSINESS_ATTRIBUTE_PRIVILEGE.getType()))));
return AuthUtil.isAuthorized(
context.getAuthorizer(), context.getActorUrn(), orPrivilegeGroups, null);
}

public static boolean canManageBusinessAttribute(@Nonnull QueryContext context) {
final DisjunctivePrivilegeGroup orPrivilegeGroups =
new DisjunctivePrivilegeGroup(
ImmutableList.of(
new ConjunctivePrivilegeGroup(
ImmutableList.of(
PoliciesConfig.MANAGE_BUSINESS_ATTRIBUTE_PRIVILEGE.getType()))));
return AuthUtil.isAuthorized(
context.getAuthorizer(), context.getActorUrn(), orPrivilegeGroups, null);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
package com.linkedin.datahub.graphql.resolvers.businessattribute;

import static com.linkedin.datahub.graphql.resolvers.ResolverUtils.bindArgument;
import static com.linkedin.datahub.graphql.resolvers.mutate.MutationUtils.buildMetadataChangeProposalWithKey;
import static com.linkedin.metadata.Constants.BUSINESS_ATTRIBUTE_ENTITY_NAME;
import static com.linkedin.metadata.Constants.BUSINESS_ATTRIBUTE_INFO_ASPECT_NAME;

import com.linkedin.businessattribute.BusinessAttributeInfo;
import com.linkedin.businessattribute.BusinessAttributeKey;
import com.linkedin.common.AuditStamp;
import com.linkedin.common.urn.Urn;
import com.linkedin.common.urn.UrnUtils;
import com.linkedin.data.template.SetMode;
import com.linkedin.datahub.graphql.QueryContext;
import com.linkedin.datahub.graphql.exception.AuthorizationException;
import com.linkedin.datahub.graphql.exception.DataHubGraphQLErrorCode;
import com.linkedin.datahub.graphql.exception.DataHubGraphQLException;
import com.linkedin.datahub.graphql.generated.BusinessAttribute;
import com.linkedin.datahub.graphql.generated.CreateBusinessAttributeInput;
import com.linkedin.datahub.graphql.generated.OwnerEntityType;
import com.linkedin.datahub.graphql.resolvers.mutate.util.BusinessAttributeUtils;
import com.linkedin.datahub.graphql.resolvers.mutate.util.OwnerUtils;
import com.linkedin.datahub.graphql.types.businessattribute.mappers.BusinessAttributeMapper;
import com.linkedin.entity.client.EntityClient;
import com.linkedin.metadata.entity.EntityService;
import com.linkedin.metadata.service.BusinessAttributeService;
import com.linkedin.metadata.utils.EntityKeyUtils;
import com.linkedin.mxe.MetadataChangeProposal;
import graphql.schema.DataFetcher;
import graphql.schema.DataFetchingEnvironment;
import java.util.UUID;
import java.util.concurrent.CompletableFuture;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;

@Slf4j
@RequiredArgsConstructor
public class CreateBusinessAttributeResolver
implements DataFetcher<CompletableFuture<BusinessAttribute>> {
private final EntityClient _entityClient;
private final EntityService _entityService;
private final BusinessAttributeService businessAttributeService;

@Override
public CompletableFuture<BusinessAttribute> get(DataFetchingEnvironment environment)
throws Exception {
final QueryContext context = environment.getContext();
CreateBusinessAttributeInput input =
bindArgument(environment.getArgument("input"), CreateBusinessAttributeInput.class);
if (!BusinessAttributeAuthorizationUtils.canCreateBusinessAttribute(context)) {
throw new AuthorizationException(
"Unauthorized to perform this action. Please contact your DataHub administrator.");
}
return CompletableFuture.supplyAsync(
() -> {
try {
final BusinessAttributeKey businessAttributeKey = new BusinessAttributeKey();
String id = input.getId() != null ? input.getId() : UUID.randomUUID().toString();
businessAttributeKey.setId(id);

if (_entityClient.exists(
EntityKeyUtils.convertEntityKeyToUrn(
businessAttributeKey, BUSINESS_ATTRIBUTE_ENTITY_NAME),
context.getAuthentication())) {
throw new IllegalArgumentException("This Business Attribute already exists!");
}

if (BusinessAttributeUtils.hasNameConflict(input.getName(), context, _entityClient)) {
throw new DataHubGraphQLException(
String.format(
"\"%s\" already exists as Business Attribute. Please pick a unique name.",
input.getName()),
DataHubGraphQLErrorCode.CONFLICT);
}

// Create the MCP
final MetadataChangeProposal changeProposal =
buildMetadataChangeProposalWithKey(
businessAttributeKey,
BUSINESS_ATTRIBUTE_ENTITY_NAME,
BUSINESS_ATTRIBUTE_INFO_ASPECT_NAME,
mapBusinessAttributeInfo(input, context));

// Ingest the MCP
Urn businessAttributeUrn =
UrnUtils.getUrn(
_entityClient.ingestProposal(changeProposal, context.getAuthentication()));
OwnerUtils.addCreatorAsOwner(
context,
businessAttributeUrn.toString(),
OwnerEntityType.CORP_USER,
_entityService);
return BusinessAttributeMapper.map(
context,
businessAttributeService.getBusinessAttributeEntityResponse(
businessAttributeUrn, context.getAuthentication()));

} catch (DataHubGraphQLException e) {
throw e;
} catch (Exception e) {
log.error(
"Failed to create Business Attribute with name: {}: {}",
input.getName(),
e.getMessage());
throw new RuntimeException(
String.format("Failed to create Business Attribute with name: %s", input.getName()),
e);
}
});
}

private BusinessAttributeInfo mapBusinessAttributeInfo(
CreateBusinessAttributeInput input, QueryContext context) {
final BusinessAttributeInfo info = new BusinessAttributeInfo();
info.setFieldPath(input.getName(), SetMode.DISALLOW_NULL);
info.setName(input.getName(), SetMode.DISALLOW_NULL);
info.setDescription(input.getDescription(), SetMode.IGNORE_NULL);
info.setType(
BusinessAttributeUtils.mapSchemaFieldDataType(input.getType()), SetMode.IGNORE_NULL);
info.setCreated(
new AuditStamp()
.setActor(UrnUtils.getUrn(context.getActorUrn()))
.setTime(System.currentTimeMillis()));
info.setLastModified(
new AuditStamp()
.setActor(UrnUtils.getUrn(context.getActorUrn()))
.setTime(System.currentTimeMillis()));
return info;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
package com.linkedin.datahub.graphql.resolvers.businessattribute;

import com.linkedin.common.urn.Urn;
import com.linkedin.common.urn.UrnUtils;
import com.linkedin.datahub.graphql.QueryContext;
import com.linkedin.datahub.graphql.exception.AuthorizationException;
import com.linkedin.entity.client.EntityClient;
import graphql.schema.DataFetcher;
import graphql.schema.DataFetchingEnvironment;
import java.util.concurrent.CompletableFuture;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;

/** Resolver responsible for hard deleting a particular Business Attribute */
@Slf4j
@RequiredArgsConstructor
public class DeleteBusinessAttributeResolver implements DataFetcher<CompletableFuture<Boolean>> {
private final EntityClient _entityClient;

@Override
public CompletableFuture<Boolean> get(DataFetchingEnvironment environment) throws Exception {
final QueryContext context = environment.getContext();
final Urn businessAttributeUrn = UrnUtils.getUrn(environment.getArgument("urn"));
if (!BusinessAttributeAuthorizationUtils.canManageBusinessAttribute(context)) {
throw new AuthorizationException(
"Unauthorized to perform this action. Please contact your DataHub administrator.");
}
if (!_entityClient.exists(businessAttributeUrn, context.getAuthentication())) {
throw new RuntimeException(
String.format("This urn does not exist: %s", businessAttributeUrn));
}
return CompletableFuture.supplyAsync(
() -> {
try {
_entityClient.deleteEntity(businessAttributeUrn, context.getAuthentication());
CompletableFuture.runAsync(
() -> {
try {
_entityClient.deleteEntityReferences(
businessAttributeUrn, context.getAuthentication());
} catch (Exception e) {
log.error(
String.format(
"Exception while attempting to clear all entity references for Business Attribute with urn %s",
businessAttributeUrn),
e);
}
});
return true;
} catch (Exception e) {
throw new RuntimeException(
String.format(
"Failed to delete Business Attribute with urn %s", businessAttributeUrn),
e);
}
});
}
}
Loading
Loading