From 1706018b6431b56cda5afbe91d9ffe0a743f732c Mon Sep 17 00:00:00 2001 From: Niall Thomson Date: Wed, 13 Dec 2023 22:24:22 +0000 Subject: [PATCH] Migrated carts service to AWS SDK v2 --- src/cart/pom.xml | 28 +++-- .../configuration/DynamoDBConfiguration.java | 48 ++++---- .../configuration/DynamoDBProperties.java | 1 - .../dynamo/entities/DynamoItemEntity.java | 18 ++- .../carts/services/DynamoDBCartService.java | 112 +++++++++--------- .../services/DynamoDBCartServiceTests.java | 11 +- 6 files changed, 113 insertions(+), 105 deletions(-) diff --git a/src/cart/pom.xml b/src/cart/pom.xml index eda2f0ab1..0f0c6abb1 100644 --- a/src/cart/pom.xml +++ b/src/cart/pom.xml @@ -25,6 +25,18 @@ 1.5.5.Final + + + + software.amazon.awssdk + bom + 2.20.125 + pom + import + + + + org.springframework.boot @@ -69,15 +81,17 @@ spring-boot-starter-data-mongodb - com.amazonaws - aws-java-sdk-dynamodb - 1.12.603 - + software.amazon.awssdk + dynamodb + - com.amazonaws - aws-java-sdk-sts - 1.12.578 + software.amazon.awssdk + dynamodb-enhanced + + software.amazon.awssdk + sts + org.projectlombok lombok diff --git a/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBConfiguration.java b/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBConfiguration.java index e7cdf3109..22afb3867 100644 --- a/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBConfiguration.java +++ b/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBConfiguration.java @@ -18,54 +18,46 @@ package com.amazon.sample.carts.configuration; -import com.amazonaws.client.builder.AwsClientBuilder; -import com.amazonaws.services.dynamodbv2.AmazonDynamoDB; -import com.amazonaws.services.dynamodbv2.AmazonDynamoDBClientBuilder; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBMapper; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBMapperConfig; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBTypeConverterFactory; +import software.amazon.awssdk.enhanced.dynamodb.DynamoDbEnhancedClient; +import software.amazon.awssdk.regions.Region; +import software.amazon.awssdk.services.dynamodb.DynamoDbClient; +import software.amazon.awssdk.services.dynamodb.DynamoDbClientBuilder; + import com.amazon.sample.carts.services.DynamoDBCartService; + +import java.net.URI; + import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Profile; import org.springframework.util.StringUtils; + import com.amazon.sample.carts.services.CartService; @Configuration @Profile("dynamodb") public class DynamoDBConfiguration { - @Bean - public AmazonDynamoDB amazonDynamoDB(DynamoDBProperties properties) { + @Bean DynamoDbClient dynamoDbClient(DynamoDBProperties properties) { + DynamoDbClientBuilder builder = DynamoDbClient.builder(); + if (!StringUtils.isEmpty(properties.getEndpoint())) { - return AmazonDynamoDBClientBuilder.standard().withEndpointConfiguration( - new AwsClientBuilder.EndpointConfiguration(properties.getEndpoint(), "us-west-2") - ).build(); + builder.region(Region.US_WEST_2); + builder.endpointOverride(URI.create(properties.getEndpoint())); } - return AmazonDynamoDBClientBuilder.standard().build(); - } - - @Bean - public DynamoDBMapperConfig dynamoDBMapperConfig(DynamoDBProperties properties) { - // Create empty DynamoDBMapperConfig builder - DynamoDBMapperConfig.Builder builder = new DynamoDBMapperConfig.Builder(); - // Inject missing defaults from the deprecated method - builder.withTypeConverterFactory(DynamoDBTypeConverterFactory.standard()); - builder.withTableNameResolver((aClass, dynamoDBMapperConfig) -> { - return properties.getTableName(); - }); - return builder.build(); } @Bean - public DynamoDBMapper dynamoDBMapper(AmazonDynamoDB amazonDynamoDB, DynamoDBMapperConfig config) { - return new DynamoDBMapper(amazonDynamoDB, config); + public DynamoDbEnhancedClient dynamoDbEnhancedClient(DynamoDbClient dynamoDbClient) { + return DynamoDbEnhancedClient.builder() + .dynamoDbClient(dynamoDbClient) + .build(); } @Bean - public CartService dynamoCartService(DynamoDBMapper mapper, AmazonDynamoDB amazonDynamoDB, DynamoDBProperties properties) { - return new DynamoDBCartService(mapper, amazonDynamoDB, properties.isCreateTable()); + public CartService dynamoCartService(DynamoDbClient dynamoDbClient, DynamoDbEnhancedClient dynamoDbEnhancedClient, DynamoDBProperties properties) { + return new DynamoDBCartService(dynamoDbClient, dynamoDbEnhancedClient, properties.isCreateTable()); } } diff --git a/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBProperties.java b/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBProperties.java index 80f90bac6..6b77efdd4 100644 --- a/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBProperties.java +++ b/src/cart/src/main/java/com/amazon/sample/carts/configuration/DynamoDBProperties.java @@ -19,7 +19,6 @@ package com.amazon.sample.carts.configuration; import lombok.Data; -import lombok.Getter; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.context.annotation.Configuration; diff --git a/src/cart/src/main/java/com/amazon/sample/carts/repositories/dynamo/entities/DynamoItemEntity.java b/src/cart/src/main/java/com/amazon/sample/carts/repositories/dynamo/entities/DynamoItemEntity.java index dce73bbfd..af892fd90 100644 --- a/src/cart/src/main/java/com/amazon/sample/carts/repositories/dynamo/entities/DynamoItemEntity.java +++ b/src/cart/src/main/java/com/amazon/sample/carts/repositories/dynamo/entities/DynamoItemEntity.java @@ -18,13 +18,13 @@ package com.amazon.sample.carts.repositories.dynamo.entities; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBAttribute; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBHashKey; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBIndexHashKey; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBTable; +import software.amazon.awssdk.enhanced.dynamodb.mapper.annotations.DynamoDbBean; +import software.amazon.awssdk.enhanced.dynamodb.mapper.annotations.DynamoDbPartitionKey; +import software.amazon.awssdk.enhanced.dynamodb.mapper.annotations.DynamoDbSecondaryPartitionKey; + import com.amazon.sample.carts.repositories.ItemEntity; -@DynamoDBTable(tableName="Items") +@DynamoDbBean() public class DynamoItemEntity implements ItemEntity { private String id; @@ -48,28 +48,24 @@ public DynamoItemEntity() { } - @DynamoDBHashKey + @DynamoDbPartitionKey public String getId() { return id; } - @DynamoDBAttribute - @DynamoDBIndexHashKey(globalSecondaryIndexName = "idx_global_customerId") + @DynamoDbSecondaryPartitionKey(indexNames = {"idx_global_customerId"}) public String getCustomerId() { return customerId; } - @DynamoDBAttribute public String getItemId() { return itemId; } - @DynamoDBAttribute public int getQuantity() { return quantity; } - @DynamoDBAttribute public int getUnitPrice() { return unitPrice; } diff --git a/src/cart/src/main/java/com/amazon/sample/carts/services/DynamoDBCartService.java b/src/cart/src/main/java/com/amazon/sample/carts/services/DynamoDBCartService.java index ec87f99c0..5ed9a1f22 100644 --- a/src/cart/src/main/java/com/amazon/sample/carts/services/DynamoDBCartService.java +++ b/src/cart/src/main/java/com/amazon/sample/carts/services/DynamoDBCartService.java @@ -18,66 +18,70 @@ package com.amazon.sample.carts.services; -import com.amazonaws.services.dynamodbv2.AmazonDynamoDB; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBMapper; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBQueryExpression; -import com.amazonaws.services.dynamodbv2.datamodeling.PaginatedQueryList; -import com.amazonaws.services.dynamodbv2.model.CreateTableRequest; -import com.amazonaws.services.dynamodbv2.model.DeleteTableRequest; -import com.amazonaws.services.dynamodbv2.model.Projection; -import com.amazonaws.services.dynamodbv2.model.ProvisionedThroughput; import com.amazon.sample.carts.repositories.CartEntity; import com.amazon.sample.carts.repositories.ItemEntity; import com.amazon.sample.carts.repositories.dynamo.entities.DynamoItemEntity; import lombok.extern.slf4j.Slf4j; - +import software.amazon.awssdk.enhanced.dynamodb.DynamoDbEnhancedClient; +import software.amazon.awssdk.enhanced.dynamodb.DynamoDbIndex; +import software.amazon.awssdk.enhanced.dynamodb.DynamoDbTable; +import software.amazon.awssdk.enhanced.dynamodb.Key; +import software.amazon.awssdk.enhanced.dynamodb.TableSchema; +import software.amazon.awssdk.enhanced.dynamodb.model.Page; +import software.amazon.awssdk.enhanced.dynamodb.model.QueryConditional; +import software.amazon.awssdk.services.dynamodb.DynamoDbClient; +import software.amazon.awssdk.services.dynamodb.model.DeleteTableRequest; +import software.amazon.awssdk.services.dynamodb.model.ProjectionType; +import software.amazon.awssdk.services.dynamodb.model.ResourceNotFoundException; import jakarta.annotation.PostConstruct; import java.util.ArrayList; +import java.util.Iterator; import java.util.List; import java.util.Optional; @Slf4j public class DynamoDBCartService implements CartService { - private final DynamoDBMapper mapper; - private final AmazonDynamoDB dynamoDB; + private final DynamoDbClient dynamoDBClient; private final boolean createTable; + private final DynamoDbTable table; + + static final TableSchema CART_TABLE_SCHEMA = TableSchema.fromClass(DynamoItemEntity.class); - public DynamoDBCartService(DynamoDBMapper mapper, AmazonDynamoDB dynamoDB, boolean createTable) { - this.mapper = mapper; - this.dynamoDB = dynamoDB; + public DynamoDBCartService(DynamoDbClient dynamoDBClient,DynamoDbEnhancedClient dynamoDbEnhancedClient, boolean createTable) { + this.dynamoDBClient = dynamoDBClient; this.createTable = createTable; + + this.table = dynamoDbEnhancedClient.table("Items", CART_TABLE_SCHEMA); } @PostConstruct public void init() { if(createTable) { - DeleteTableRequest deleteTableRequest = mapper.generateDeleteTableRequest(DynamoItemEntity.class); - try { - dynamoDB.describeTable(deleteTableRequest.getTableName()); - - log.warn("Dynamo table found, deleting to recreate...."); - dynamoDB.deleteTable(deleteTableRequest); + dynamoDBClient.deleteTable( + DeleteTableRequest.builder().tableName("Items").build() + ); } - catch (com.amazonaws.services.dynamodbv2.model.ResourceNotFoundException rnfe) { - log.warn("Dynamo "+deleteTableRequest.getTableName()+" table not found"); + catch (ResourceNotFoundException rnfe) { + log.warn("Dynamo table not found"); } - ProvisionedThroughput pt = new ProvisionedThroughput(1L, 1L); - - CreateTableRequest tableRequest = mapper - .generateCreateTableRequest(DynamoItemEntity.class); - tableRequest.setProvisionedThroughput(pt); - tableRequest.getGlobalSecondaryIndexes().get(0).setProvisionedThroughput(pt); - tableRequest.getGlobalSecondaryIndexes().get(0).setProjection(new Projection().withProjectionType("ALL")); - dynamoDB.createTable(tableRequest); - - try { - Thread.sleep(1000); - } catch (InterruptedException e) { - e.printStackTrace(); - } + this.table.createTable(builder -> builder + .globalSecondaryIndices(builder3 -> builder3 + .indexName("idx_global_customerId") + + .projection(builder2 -> builder2 + .projectionType(ProjectionType.ALL)) + .provisionedThroughput(builder4 -> builder4 + .writeCapacityUnits(1L) + .readCapacityUnits(1L))) + .provisionedThroughput(b -> b + .readCapacityUnits(1L) + .writeCapacityUnits(1L) + .build())); + + this.dynamoDBClient.waiter().waitUntilTableExists(b -> b.tableName("Items")); } } @@ -102,7 +106,9 @@ public List getItems() { public void delete(String customerId) { List items = items(customerId); - mapper.batchDelete(items); + items.forEach(item -> { + this.table.deleteItem(item); + }); } @Override @@ -114,7 +120,7 @@ public CartEntity merge(String sessionId, String customerId) { public ItemEntity add(String customerId, String itemId, int quantity, int unitPrice) { String hashKey = hashKey(customerId, itemId); - DynamoItemEntity item = this.mapper.load(DynamoItemEntity.class, hashKey); + DynamoItemEntity item = this.table.getItem(Key.builder().partitionValue(hashKey(customerId, itemId)).build()); if(item != null) { item.setQuantity(item.getQuantity() + quantity); @@ -123,34 +129,34 @@ public ItemEntity add(String customerId, String itemId, int quantity, int unitPr item = new DynamoItemEntity(hashKey, customerId, itemId, 1, unitPrice); } - this.mapper.save(item); + this.table.putItem(item); return item; } @Override public List items(String customerId) { - final DynamoItemEntity gsiKeyObj = new DynamoItemEntity(); - gsiKeyObj.setCustomerId(customerId); - final DynamoDBQueryExpression queryExpression = - new DynamoDBQueryExpression<>(); - queryExpression.setHashKeyValues(gsiKeyObj); - queryExpression.setIndexName("idx_global_customerId"); - queryExpression.setConsistentRead(false); // cannot use consistent read on GSI - final PaginatedQueryList results = - mapper.query(DynamoItemEntity.class, queryExpression); - - return new ArrayList<>(results); + DynamoDbIndex index = this.table.index("idx_global_customerId"); + QueryConditional q = QueryConditional.keyEqualTo(Key.builder().partitionValue(customerId).build()); + Iterator> result = index.query(q).iterator(); + List users = new ArrayList<>(); + + while (result.hasNext()) { + Page userPage = result.next(); + users.addAll(userPage.items()); + } + + return users; } @Override public Optional item(String customerId, String itemId) { - return Optional.of(mapper.load(DynamoItemEntity.class, hashKey(customerId, itemId))); + return Optional.of(this.table.getItem(Key.builder().partitionValue(hashKey(customerId, itemId)).build())); } @Override public void deleteItem(String customerId, String itemId) { - item(customerId, itemId).ifPresentOrElse(this.mapper::delete, + item(customerId, itemId).ifPresentOrElse(this.table::deleteItem, () -> log.warn("Item missing for delete {} -- {}", customerId, itemId)); } @@ -162,7 +168,7 @@ public Optional update(String customerId, String itemId, int q item.setQuantity(quantity); item.setUnitPrice(unitPrice); - this.mapper.save(item); + this.table.updateItem(item); return item; } diff --git a/src/cart/src/test/java/com/amazon/sample/carts/services/DynamoDBCartServiceTests.java b/src/cart/src/test/java/com/amazon/sample/carts/services/DynamoDBCartServiceTests.java index 6bb27e3e2..3cb2622dd 100644 --- a/src/cart/src/test/java/com/amazon/sample/carts/services/DynamoDBCartServiceTests.java +++ b/src/cart/src/test/java/com/amazon/sample/carts/services/DynamoDBCartServiceTests.java @@ -18,8 +18,9 @@ package com.amazon.sample.carts.services; -import com.amazonaws.services.dynamodbv2.AmazonDynamoDB; -import com.amazonaws.services.dynamodbv2.datamodeling.DynamoDBMapper; +import software.amazon.awssdk.enhanced.dynamodb.DynamoDbEnhancedClient; +import software.amazon.awssdk.services.dynamodb.DynamoDbClient; + import com.amazon.sample.carts.configuration.DynamoDBConfiguration; import com.amazon.sample.carts.configuration.DynamoDBProperties; import org.junit.jupiter.api.Tag; @@ -54,7 +55,7 @@ public class DynamoDBCartServiceTests extends AbstractServiceTests { private DynamoDBCartService service; @Autowired - private AmazonDynamoDB dynamodb; + private DynamoDbClient dynamoDbClient; @Container public static GenericContainer dynamodbContainer = @@ -72,10 +73,10 @@ public CartService getService() { static class TestConfiguration { @Autowired - private AmazonDynamoDB amazonDynamoDB; + private DynamoDbClient dynamoDbClient; @Autowired - private DynamoDBMapper mapper; + private DynamoDbEnhancedClient dynamoDbEnhancedClient; } public static class Initializer implements