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

1575 - implementation of ingest relationships and GET Relationships API #1591

Closed
wants to merge 56 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
56 commits
Select commit Hold shift + click to select a range
3914a3e
1536 - MVP for look-ups
Vladysl Dec 6, 2023
a6156ac
Merge branch 'main' into 1536
AndreyNenashev Dec 6, 2023
2444b8b
1536 - MVP for look-ups
Vladysl Dec 7, 2023
4db6a55
1536 - update spec
Vladysl Dec 8, 2023
819dbcd
1536 - update spec and first implementation of adding data to custom …
Vladysl Dec 11, 2023
a542367
feature(fe): tab and router for master-data tab and lookup tables page
AndreyNenashev Dec 6, 2023
37fa832
feature(fe): lookup tables page
AndreyNenashev Dec 6, 2023
12bffe8
feature(fe): lookup table input search component
AndreyNenashev Dec 7, 2023
66ea1b0
feature(fe): lookup table form
AndreyNenashev Dec 7, 2023
c164ce6
feature(fe): lookup tables management
AndreyNenashev Dec 8, 2023
cdae0b8
feature(fe): link to lookup table`s data entity
AndreyNenashev Dec 8, 2023
aa28c99
feature(fe): rename tableName to name
AndreyNenashev Dec 11, 2023
73e6bf1
1536 implementation of update lookup table and it's columns
Vladysl Dec 13, 2023
4d9660a
feature(fe): add column button if dataset is lookup table
AndreyNenashev Dec 14, 2023
5faf71b
feature(fe): add column form
AndreyNenashev Dec 14, 2023
6bc9f40
feature(chore): adjust fe to the new openapi
AndreyNenashev Dec 14, 2023
581f0b4
1536 added lookup table to data Entities details and Added remove APIS
Vladysl Dec 15, 2023
1dfa721
1536 added lookup table to data Entities details and Added remove APIS
Vladysl Dec 15, 2023
ea0f9ed
1536 added remove lookup table data and get lookupTableDefinition API's
Vladysl Dec 15, 2023
f8f8021
feature(fe): create column form
AndreyNenashev Dec 15, 2023
4b4e238
feature(fe): column deletion
AndreyNenashev Dec 15, 2023
0925b73
feature(fe): delete column button
AndreyNenashev Dec 15, 2023
3eafc05
feature(fe): edit column form
AndreyNenashev Dec 15, 2023
8549e51
1536 added default values for is_nullable and
Vladysl Dec 15, 2023
4fd8f6d
1536 - implementation of lookups APIs
Vladysl Dec 18, 2023
6f6ab05
1536 - implementation of getLookupTable API
Vladysl Dec 20, 2023
1ea86d2
1536 - added unit test and ordering by id for table lookup definitions
Vladysl Dec 21, 2023
3183f17
1536 - added unit test
Vladysl Dec 21, 2023
dd86ecb
1536 - is_nullable default value is true
Vladysl Dec 22, 2023
333d919
1536 - remove ENUM type
Vladysl Dec 22, 2023
cbefe6c
1536 - added row_id to LookupTableRow
Vladysl Dec 26, 2023
c063864
refactoring(fe): shared SearchInput component
AndreyNenashev Dec 20, 2023
f7a901e
refactoring(fe): query examples list component
AndreyNenashev Dec 20, 2023
bc2e6f1
feature(fe): show reference data componeny
AndreyNenashev Dec 21, 2023
e8638e4
feature(fe): show dataset data
AndreyNenashev Dec 22, 2023
04ea601
feature(fe): delete reference data row
AndreyNenashev Dec 26, 2023
75bf156
feature(fe): use table instead of flexbox for unkown table data
AndreyNenashev Dec 27, 2023
fbbd4d8
feature(fe): render form
AndreyNenashev Dec 29, 2023
a262514
feature(fe): create reference data row form action
AndreyNenashev Dec 29, 2023
939b376
1536 - update demo.yaml
Vladysl Jan 4, 2024
5a934f4
1536 - updated /api/referencedata/table/{lookup_table_id}/data/{row_i…
Vladysl Jan 10, 2024
df401b4
add .vscode in .gitignore
AndreyNenashev Jan 8, 2024
e723590
feature(fe): table row cell component
AndreyNenashev Jan 8, 2024
f35ac7a
feature(fe): force reload virtualized table
AndreyNenashev Jan 9, 2024
5b01408
feature(fe): update row
AndreyNenashev Jan 9, 2024
b1f518e
fix(fe): fix table cell rendering
AndreyNenashev Jan 9, 2024
065d19a
fix(fe): fix table cell rendering
AndreyNenashev Jan 9, 2024
6ffc015
chore(fe): minor refactoring
AndreyNenashev Jan 9, 2024
e39547a
chore(fe): fix inputs glitches
AndreyNenashev Jan 9, 2024
e228ee5
feature(fe): decompose table cell into different components
AndreyNenashev Jan 10, 2024
6e388c7
feature(fe): simplify DatasetDataTableRowCellDate component
AndreyNenashev Jan 10, 2024
280c402
chore(fe): edit table row
AndreyNenashev Jan 10, 2024
a848939
chore(fe): fix create row form
AndreyNenashev Jan 10, 2024
e7b54d9
1576 - MVP implementation of ingest relationships
Vladysl Jan 12, 2024
e5619fa
1576 - MVP implementation of ingest relationships
Vladysl Jan 12, 2024
ae4ad82
1576 - implementation of ingest relationships and GET Relationships API
Vladysl Jan 18, 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
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -23,3 +23,4 @@ node_modules/
/tests/dist/
/playwright/.cache/
.tool-versions
.vscode
3 changes: 3 additions & 0 deletions docker/demo.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,9 @@ services:
- SPRING_DATASOURCE_URL=jdbc:postgresql://database:5432/${POSTGRES_DATABASE}
- SPRING_DATASOURCE_USERNAME=${POSTGRES_USER}
- SPRING_DATASOURCE_PASSWORD=${POSTGRES_PASSWORD}
- SPRING_CUSTOM-DATASOURCE_URL=jdbc:postgresql://database:5432/${POSTGRES_DATABASE}?schema=lookup_tables_schema
- SPRING_CUSTOM-DATASOURCE_USERNAME=${POSTGRES_USER}
- SPRING_CUSTOM-DATASOURCE_PASSWORD=${POSTGRES_PASSWORD}
depends_on:
- database
ports:
Expand Down
6 changes: 3 additions & 3 deletions gradle/libs.versions.toml
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,8 @@
spring-webflux = '6.0.9'
reactor-extra = '3.5.1'
micrometer-registry-prometheus = '1.9.0'
ingestion-contract-server = '0.1.32'
oddrn-generator-java = '0.1.20'
ingestion-contract-server = '0.1.34'
oddrn-generator-java = '0.1.21'
odd-integration-manifests = '0.0.6'
apache-collections = '4.4'
apache-lang = '3.12.0'
Expand Down Expand Up @@ -33,7 +33,7 @@ shedlock-version = '4.42.0'
json-schema-validator = '1.0.84'
caffeine = '3.1.1'
lombok = '1.18.24'
testcontainers = '1.18.3'
testcontainers = '1.19.3'
slf4j-api = '1.7.30'
logback = '1.2.11'
easy-random-core = '5.0.0'
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package org.opendatadiscovery.oddplatform.annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import org.springframework.transaction.annotation.Transactional;

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Transactional("customTransactionManager")
public @interface ReactiveCustomTransactional {
}
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,9 @@ public enum AuthorizationManagerType {
DATASET_FIELD,
TERM,
QUERY_EXAMPLE,
LOOKUP_TABLE,
LOOKUP_TABLE_DEFINITION,
LOOKUP_TABLE_DATA,
ALERT,
DEG
}
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,9 @@ public final class ReactiveAuthorizationManagerFactory {
private static final String DATASET_FIELD_ID = "dataset_field_id";
private static final String DATA_ENTITY_GROUP_ID = "data_entity_group_id";
private static final String QUERY_EXAMPLE_ID = "example_id";
private static final String LOOKUP_TABLE_ID = "lookup_table_id";
private static final String LOOKUP_TABLE_DEFINITION_ID = "column_id";
private static final String LOOKUP_TABLE_DATA_ID = "row_id";
private static final String ALERT_ID = "alert_id";

private ReactiveAuthorizationManagerFactory() {
Expand Down Expand Up @@ -46,6 +49,9 @@ private static String resourceExtractorVariableName(final AuthorizationManagerTy
case ALERT -> ALERT_ID;
case DEG -> DATA_ENTITY_GROUP_ID;
case QUERY_EXAMPLE -> QUERY_EXAMPLE_ID;
case LOOKUP_TABLE -> LOOKUP_TABLE_ID;
case LOOKUP_TABLE_DEFINITION -> LOOKUP_TABLE_DEFINITION_ID;
case LOOKUP_TABLE_DATA -> LOOKUP_TABLE_DATA_ID;
default -> throw new IllegalArgumentException("Unsupported resource type: " + type);
};
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,15 @@
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.DATA_SOURCE_DELETE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.DATA_SOURCE_TOKEN_REGENERATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.DATA_SOURCE_UPDATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_CREATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DATA_CREATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DATA_DELETE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DATA_UPDATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DEFINITION_CREATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DEFINITION_DELETE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DEFINITION_UPDATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_DELETE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.LOOKUP_TABLE_UPDATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.NAMESPACE_CREATE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.NAMESPACE_DELETE;
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.NAMESPACE_UPDATE;
Expand Down Expand Up @@ -73,6 +82,7 @@
import static org.opendatadiscovery.oddplatform.dto.policy.PolicyPermissionDto.TERM_UPDATE;
import static org.springframework.http.HttpMethod.DELETE;
import static org.springframework.http.HttpMethod.GET;
import static org.springframework.http.HttpMethod.PATCH;
import static org.springframework.http.HttpMethod.POST;
import static org.springframework.http.HttpMethod.PUT;

Expand All @@ -97,6 +107,8 @@ public final class SecurityConstants {
new SecurityRule(NO_CONTEXT, new PathPatternParserServerWebExchangeMatcher("/api/terms", POST), TERM_CREATE),
new SecurityRule(NO_CONTEXT, new PathPatternParserServerWebExchangeMatcher("/api/queryexample", POST),
QUERY_EXAMPLE_CREATE),
new SecurityRule(NO_CONTEXT, new PathPatternParserServerWebExchangeMatcher("/api/referencedata/table", POST),
LOOKUP_TABLE_CREATE),
new SecurityRule(NO_CONTEXT, new PathPatternParserServerWebExchangeMatcher("/api/datasources", POST),
DATA_SOURCE_CREATE),
new SecurityRule(NO_CONTEXT,
Expand Down Expand Up @@ -290,6 +302,36 @@ DATA_ENTITY, new PathPatternParserServerWebExchangeMatcher(
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/queryexample/{example_id}/dataset/{data_entity_id}", DELETE),
QUERY_EXAMPLE_DATASET_DELETE)
QUERY_EXAMPLE_DATASET_DELETE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher("/api/referencedata/table/{lookup_table_id}", PUT),
LOOKUP_TABLE_UPDATE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher("/api/referencedata/table/{lookup_table_id}", DELETE),
LOOKUP_TABLE_DELETE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/referencedata/table/{lookup_table_id}/columns", POST),
LOOKUP_TABLE_DEFINITION_CREATE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/referencedata/table/{lookup_table_id}/column/{column_id}", PATCH),
LOOKUP_TABLE_DEFINITION_UPDATE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/referencedata/table/{lookup_table_id}/column/{column_id}", DELETE),
LOOKUP_TABLE_DEFINITION_DELETE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/referencedata/table/{lookup_table_id}/data", POST),
LOOKUP_TABLE_DATA_CREATE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/referencedata/table/{lookup_table_id}/data/{row_id}", PUT),
LOOKUP_TABLE_DATA_UPDATE),
new SecurityRule(NO_CONTEXT,
new PathPatternParserServerWebExchangeMatcher(
"/api/referencedata/table/{lookup_table_id}/data/{row_id}", DELETE),
LOOKUP_TABLE_DATA_DELETE)
);
}
Original file line number Diff line number Diff line change
Expand Up @@ -5,18 +5,23 @@
import io.r2dbc.spi.ConnectionFactories;
import io.r2dbc.spi.ConnectionFactory;
import io.r2dbc.spi.ConnectionFactoryOptions;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
import org.springframework.boot.autoconfigure.r2dbc.R2dbcProperties;
import org.springframework.boot.context.properties.PropertyMapper;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.r2dbc.connection.R2dbcTransactionManager;
import org.springframework.r2dbc.core.DatabaseClient;
import org.springframework.transaction.ReactiveTransactionManager;

@Configuration
public class R2DBCConfiguration {

@Bean(destroyMethod = "dispose")
@Primary
public ConnectionPool connectionFactory(final DataSourceProperties dataSourceProperties,
final R2dbcProperties properties) {
final String r2dbcUrl = dataSourceProperties.getUrl().replace("jdbc", "r2dbc");
Expand All @@ -42,8 +47,59 @@ public ConnectionPool connectionFactory(final DataSourceProperties dataSourcePro
return new ConnectionPool(builder.build());
}

@Bean(destroyMethod = "dispose")
@Qualifier("customConnectionPool")
public ConnectionPool databaseClientForCustomSchema(
@Value("${spring.custom-datasource.url}") final String url,
@Value("${spring.custom-datasource.username}") final String username,
@Value("${spring.custom-datasource.password}") final String password,
final R2dbcProperties properties) {
final String r2dbcUrl = url.replace("jdbc", "r2dbc");
final ConnectionFactory factory = ConnectionFactories.get(ConnectionFactoryOptions.parse(r2dbcUrl).mutate()
.option(ConnectionFactoryOptions.PROTOCOL, "postgresql")
.option(ConnectionFactoryOptions.USER, username)
.option(ConnectionFactoryOptions.PASSWORD, password)
.build());

final R2dbcProperties.Pool pool = properties.getPool();
final PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
final ConnectionPoolConfiguration.Builder builder = ConnectionPoolConfiguration.builder(factory);
map.from(pool.getMaxIdleTime()).to(builder::maxIdleTime);
map.from(pool.getMaxLifeTime()).to(builder::maxLifeTime);
map.from(pool.getMaxAcquireTime()).to(builder::maxAcquireTime);
map.from(pool.getMaxCreateConnectionTime()).to(builder::maxCreateConnectionTime);
map.from(pool.getInitialSize()).to(builder::initialSize);
map.from(pool.getMaxSize()).to(builder::maxSize);
map.from(pool.getValidationQuery()).whenHasText().to(builder::validationQuery);
map.from(pool.getValidationDepth()).to(builder::validationDepth);
map.from(pool.getMinIdle()).to(builder::minIdle);
map.from(pool.getMaxValidationTime()).to(builder::maxValidationTime);

return new ConnectionPool(builder.build());
}

@Bean
public DatabaseClient databaseClient(final ConnectionFactory schema1ConnectionFactory) {
return DatabaseClient.create(schema1ConnectionFactory);
}

@Bean
@Qualifier("customDataClient")
public DatabaseClient databaseClientCustomTables(
@Qualifier("customConnectionPool") final ConnectionFactory connectionFactory) {
return DatabaseClient.create(connectionFactory);
}

@Bean
@Primary
public ReactiveTransactionManager reactiveTransactionManager(final ConnectionFactory connectionFactory) {
return new R2dbcTransactionManager(connectionFactory);
}

@Bean
@Qualifier("customTransactionManager")
public ReactiveTransactionManager reactiveCustomTransactionManager(
@Qualifier("customConnectionPool") final ConnectionFactory connectionFactory) {
return new R2dbcTransactionManager(connectionFactory);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,13 @@
import org.opendatadiscovery.oddplatform.api.contract.api.DataSetApi;
import org.opendatadiscovery.oddplatform.api.contract.model.DataSetStructure;
import org.opendatadiscovery.oddplatform.api.contract.model.DataSetVersionDiffList;
import org.opendatadiscovery.oddplatform.api.contract.model.ERDRelationshipDetailsList;
import org.opendatadiscovery.oddplatform.api.contract.model.GraphRelationshipDetailsList;
import org.opendatadiscovery.oddplatform.api.contract.model.RelationshipList;
import org.opendatadiscovery.oddplatform.service.DatasetVersionService;
import org.opendatadiscovery.oddplatform.service.ERDRelationshipsService;
import org.opendatadiscovery.oddplatform.service.GraphRelationshipsService;
import org.opendatadiscovery.oddplatform.service.RelationshipsService;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.server.ServerWebExchange;
Expand All @@ -14,6 +20,9 @@
@RequiredArgsConstructor
public class DatasetController implements DataSetApi {
private final DatasetVersionService datasetVersionService;
private final RelationshipsService relationshipsService;
private final ERDRelationshipsService erdRelationshipsService;
private final GraphRelationshipsService graphRelationshipsService;

@Override
public Mono<ResponseEntity<DataSetStructure>> getDataSetStructureByVersionId(
Expand Down Expand Up @@ -44,4 +53,29 @@ public Mono<ResponseEntity<DataSetVersionDiffList>> getDataSetStructureDiff(fina
return datasetVersionService.getDatasetVersionDiff(dataEntityId, firstVersionId, secondVersionId)
.map(ResponseEntity::ok);
}

@Override
public Mono<ResponseEntity<RelationshipList>> getDataSetRelationships(final Long dataEntityId,
final ServerWebExchange exchange) {
return relationshipsService.getRelationsByDatasetId(dataEntityId)
.map(ResponseEntity::ok);
}

@Override
public Mono<ResponseEntity<ERDRelationshipDetailsList>>
getDataSetErdRelationshipsById(final Long dataEntityId,
final Long relationshipsId,
final ServerWebExchange exchange) {
return erdRelationshipsService.getDataSetErdRelationshipsById(dataEntityId, relationshipsId)
.map(ResponseEntity::ok);
}

@Override
public Mono<ResponseEntity<GraphRelationshipDetailsList>>
getDataSetGraphRelationshipsById(final Long dataEntityId,
final Long relationshipsId,
final ServerWebExchange exchange) {
return graphRelationshipsService.getDataSetGraphRelationshipsById(relationshipsId)
.map(ResponseEntity::ok);
}
}
Loading