diff --git a/docs/guide/src/docs/asciidoc/ses.adoc b/docs/guide/src/docs/asciidoc/ses.adoc index 9e08c471a..dc0c64745 100644 --- a/docs/guide/src/docs/asciidoc/ses.adoc +++ b/docs/guide/src/docs/asciidoc/ses.adoc @@ -55,11 +55,12 @@ include::{root-dir}/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/ <3> Define the from address <4> Define one or more recipients <5> Define HTML body (alternatively you can declare plain text body as well) -<6> Build an attachment -<7> Define the location of the file to be sent -<8> Define the file name (optional - deduced from the file) -<9> Define the mime type (usually optional - deduced from the file) -<10> Define the description of the file (optional) +<6> Define tags for the email, they will be included in SES events +<7> Build an attachment +<8> Define the location of the file to be sent +<9> Define the file name (optional - deduced from the file) +<10> Define the mime type (usually optional - deduced from the file) +<11> Define the description of the file (optional) [source,java,indent=0,options="nowrap",role="secondary"] @@ -72,11 +73,12 @@ include::{root-dir}/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/ <3> Define the from address <4> Define one or more recipients <5> Define HTML body (alternatively you can declare plain text body as well) -<6> Build an attachment -<7> Define the location of the file to be sent -<8> Define the file name (optional - deduced from the file) -<9> Define the mime type (usually optional - deduced from the file) -<10> Define the description of the file (optional) +<6> Define tags for the email, they will be included in SES events +<7> Build an attachment +<8> Define the location of the file to be sent +<9> Define the file name (optional - deduced from the file) +<10> Define the mime type (usually optional - deduced from the file) +<11> Define the description of the file (optional) Please, see https://agorapulse.github.io/micronaut-aws-sdk/api/com/agorapulse/micronaut/amazon/awsses/SimpleEmailService.html[SimpleEmailService AWS SDK 2.x] diff --git a/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/DefaultSimpleEmailService.java b/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/DefaultSimpleEmailService.java index 7e294bc3d..4217f3c92 100644 --- a/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/DefaultSimpleEmailService.java +++ b/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/DefaultSimpleEmailService.java @@ -23,6 +23,7 @@ import software.amazon.awssdk.awscore.exception.AwsServiceException; import software.amazon.awssdk.core.SdkBytes; import software.amazon.awssdk.services.ses.SesClient; +import software.amazon.awssdk.services.ses.model.MessageTag; import software.amazon.awssdk.services.ses.model.SendEmailRequest; import software.amazon.awssdk.services.ses.model.SendRawEmailRequest; @@ -41,8 +42,11 @@ import java.io.FileInputStream; import java.io.IOException; import java.nio.charset.StandardCharsets; +import java.util.Collections; +import java.util.List; import java.util.Optional; import java.util.Properties; +import java.util.stream.Collectors; import static javax.mail.Message.RecipientType.TO; @@ -155,7 +159,9 @@ private EmailDeliveryStatus sendEmailWithAttachment(TransactionalEmail email) th SendRawEmailRequest rawEmailRequest = SendRawEmailRequest.builder() .rawMessage(b -> b.data(SdkBytes.fromByteArray(outputStream.toByteArray()))) .destinations(email.getRecipients()) - .source(Optional.ofNullable(email.getFrom()).orElseGet(() -> configuration.getSourceEmail().orElse(null))).build(); + .source(Optional.ofNullable(email.getFrom()).orElseGet(() -> configuration.getSourceEmail().orElse(null))) + .tags(getCustomTags(email)) + .build(); return handleSend(email, () -> client.sendRawEmail(rawEmailRequest)); } @@ -178,7 +184,20 @@ private EmailDeliveryStatus sendWithoutAttachments(TransactionalEmail email) { builder.replyToAddresses(email.getReplyTo()); } + builder.tags(getCustomTags(email)); + return handleSend(email, () -> client.sendEmail(builder.build())); } + private List getCustomTags(TransactionalEmail email) { + if (email.getTags() == null || email.getTags().isEmpty()) { + return Collections.emptyList(); + } + return email.getTags().entrySet().stream() + .map(entry -> MessageTag.builder() + .name(entry.getKey()) + .value(entry.getValue()).build()) + .collect(Collectors.toList()); + } + } diff --git a/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/TransactionalEmail.java b/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/TransactionalEmail.java index b3c77f38d..922f9a327 100644 --- a/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/TransactionalEmail.java +++ b/subprojects/micronaut-amazon-awssdk-ses/src/main/java/com/agorapulse/micronaut/amazon/awssdk/ses/TransactionalEmail.java @@ -20,7 +20,9 @@ import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; +import java.util.HashMap; import java.util.List; +import java.util.Map; import java.util.function.Consumer; /** @@ -32,6 +34,7 @@ public class TransactionalEmail { private String subject = ""; private String htmlBody = ""; private String replyTo = ""; + private Map tags = new HashMap<>(); private final List recipients = new ArrayList<>(); private final List attachments = new ArrayList<>(); @@ -76,6 +79,11 @@ public TransactionalEmail replyTo(String str) { return this; } + public TransactionalEmail tags(Map customTags) { + this.tags = customTags; + return this; + } + public String getFrom() { return from; } @@ -100,6 +108,10 @@ public List getAttachments() { return Collections.unmodifiableList(attachments); } + public Map getTags() { + return tags; + } + // CHECKSTYLE:OFF @Override public String toString() { @@ -110,6 +122,7 @@ public String toString() { ", replyTo='" + replyTo + '\'' + ", recipients=" + recipients + ", attachments=" + attachments + + ", tags=" + tags + '}'; } // CHECKSTYLE:ON diff --git a/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailSpec.groovy b/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailSpec.groovy index 8ec5d9b55..1f71db710 100644 --- a/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailSpec.groovy +++ b/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailSpec.groovy @@ -43,6 +43,7 @@ class SendEmailSpec extends Specification { file.createNewFile() file.text = 'not a real PDF' String thePath = file.canonicalPath + Map mapOfTags = [myTagKey: 'myTagValue'] when: // tag::builder[] EmailDeliveryStatus status = service.send { // <1> @@ -50,11 +51,12 @@ class SendEmailSpec extends Specification { from 'subscribe@groovycalamari.com' // <3> to 'me@sergiodelamo.com' // <4> htmlBody '

This is an example body

' // <5> - attachment { // <6> - filepath thePath // <7> - filename 'test.pdf' // <8> - mimeType 'application/pdf' // <9> - description 'An example pdf' // <10> + tags mapOfTags // <6> + attachment { // <7> + filepath thePath // <8> + filename 'test.pdf' // <9> + mimeType 'application/pdf' // <10> + description 'An example pdf' // <11> } } // end::builder[] diff --git a/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailTest.java b/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailTest.java index 49e5be79d..2b04fb511 100644 --- a/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailTest.java +++ b/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SendEmailTest.java @@ -29,6 +29,8 @@ import java.io.IOException; import java.nio.file.Files; import java.util.Collections; +import java.util.HashMap; +import java.util.Map; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; @@ -54,6 +56,8 @@ public void testSendEmail() throws IOException { Files.write(file.toPath(), Collections.singletonList("not a real PDF")); String filepath = file.getCanonicalPath(); + Map mapOfTags = new HashMap<>(); + mapOfTags.put("myTagKey", "myTagValue"); // tag::builder[] EmailDeliveryStatus status = service.send(e -> // <1> @@ -61,11 +65,12 @@ public void testSendEmail() throws IOException { .from("subscribe@groovycalamari.com") // <3> .to("me@sergiodelamo.com") // <4> .htmlBody("

This is an example body

") // <5> - .attachment(a -> // <6> - a.filepath(filepath) // <7> - .filename("test.pdf") // <8> - .mimeType("application/pdf") // <9> - .description("An example pdf") // <10> + .tags(mapOfTags) // <6> + .attachment(a -> // <7> + a.filepath(filepath) // <8> + .filename("test.pdf") // <9> + .mimeType("application/pdf") // <10> + .description("An example pdf") // <11> ) ); // end::builder[] diff --git a/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SimpleEmailServiceSpec.groovy b/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SimpleEmailServiceSpec.groovy index 770afac69..c11e99b02 100644 --- a/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SimpleEmailServiceSpec.groovy +++ b/subprojects/micronaut-amazon-awssdk-ses/src/test/groovy/com/agorapulse/micronaut/amazon/awssdk/ses/SimpleEmailServiceSpec.groovy @@ -44,12 +44,15 @@ class SimpleEmailServiceSpec extends Specification { ) void "test transactionalEmailWithClosure"() { + given: + Map customTags = [key1: 'value1', key2: 'value2'] when: TransactionalEmail transactionalEmail = SimpleEmailService.email { subject 'Hi Paul' from 'subscribe@groovycalamari.com' to 'me@sergiodelamo.com' htmlBody '

This is an example body

' + tags customTags attachment { filename 'test.pdf' filepath '/tmp/test.pdf' @@ -64,6 +67,7 @@ class SimpleEmailServiceSpec extends Specification { transactionalEmail.htmlBody == '

This is an example body

' transactionalEmail.from == 'subscribe@groovycalamari.com' transactionalEmail.recipients == ['me@sergiodelamo.com'] + transactionalEmail.tags == [key1: 'value1', key2: 'value2'] transactionalEmail.attachments.size() == 1 transactionalEmail.attachments.first().filename == 'test.pdf' transactionalEmail.attachments.first().filepath == '/tmp/test.pdf' @@ -82,6 +86,7 @@ class SimpleEmailServiceSpec extends Specification { htmlBody '

This is an example body

' to 'me@sergiodelamo.com' from 'subscribe@groovycalamari.com' + tags customTags attachment { filepath f.absolutePath } @@ -93,6 +98,7 @@ class SimpleEmailServiceSpec extends Specification { transactionalEmail.htmlBody == '

This is an example body

' transactionalEmail.from == 'subscribe@groovycalamari.com' transactionalEmail.recipients == ['me@sergiodelamo.com'] + transactionalEmail.tags == [key1: 'value1', key2: 'value2'] transactionalEmail.attachments.size() == 1 transactionalEmail.attachments.first().filename == 'groovylogo.png' transactionalEmail.attachments.first().filepath == f.absolutePath diff --git a/subprojects/micronaut-aws-sdk-ag-ws/src/test/groovy/com/agorapulse/micronaut/aws/apigateway/ws/LambdaEchoFunctionSpec.groovy b/subprojects/micronaut-aws-sdk-ag-ws/src/test/groovy/com/agorapulse/micronaut/aws/apigateway/ws/LambdaEchoFunctionSpec.groovy index 00408738d..f49137c8c 100644 --- a/subprojects/micronaut-aws-sdk-ag-ws/src/test/groovy/com/agorapulse/micronaut/aws/apigateway/ws/LambdaEchoFunctionSpec.groovy +++ b/subprojects/micronaut-aws-sdk-ag-ws/src/test/groovy/com/agorapulse/micronaut/aws/apigateway/ws/LambdaEchoFunctionSpec.groovy @@ -33,7 +33,7 @@ import spock.lang.Specification */ @Ignore @Remember( - value = '2023-06-01', + value = '2024-06-01', description = 'Try to fix this test or remove this feature completely' ) class LambdaEchoFunctionSpec extends Specification { diff --git a/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/DefaultSimpleEmailService.groovy b/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/DefaultSimpleEmailService.groovy index a20c6d566..c37539996 100644 --- a/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/DefaultSimpleEmailService.groovy +++ b/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/DefaultSimpleEmailService.groovy @@ -149,6 +149,7 @@ class DefaultSimpleEmailService implements SimpleEmailService { rawEmailRequest.destinations = email.recipients rawEmailRequest.source = email.from ?: configuration.sourceEmail.orElse(null) + rawEmailRequest.tags = getTags(email) return handleSend(email) { client.sendRawEmail(rawEmailRequest) @@ -184,8 +185,20 @@ class DefaultSimpleEmailService implements SimpleEmailService { if (email.replyTo) { sendEmailRequest.replyToAddresses = singletonList(email.replyTo) } + sendEmailRequest.tags = getTags(email) client.sendEmail(sendEmailRequest) } } + private List getTags(TransactionalEmail email) { + if (!email.tags) { + return [] + } + return email.tags.collect { entry -> + new MessageTag() + .withName(entry.key) + .withValue(entry.value) + } + } + } diff --git a/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/TransactionalEmail.java b/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/TransactionalEmail.java index ee588e618..52ebb4b23 100644 --- a/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/TransactionalEmail.java +++ b/subprojects/micronaut-aws-sdk-ses/src/main/groovy/com/agorapulse/micronaut/aws/ses/TransactionalEmail.java @@ -23,7 +23,9 @@ import java.util.ArrayList; import java.util.Arrays; +import java.util.HashMap; import java.util.List; +import java.util.Map; import java.util.function.Consumer; /** @@ -35,6 +37,7 @@ public class TransactionalEmail { private String subject = ""; private String htmlBody = ""; private String replyTo = ""; + private Map tags = new HashMap<>(); private final List recipients = new ArrayList<>(); private final List attachments = new ArrayList<>(); @@ -83,6 +86,11 @@ public TransactionalEmail replyTo(String str) { return this; } + public TransactionalEmail tags(Map customTags) { + this.tags = customTags; + return this; + } + public String getFrom() { return from; } @@ -107,6 +115,10 @@ public List getAttachments() { return attachments; } + public Map getTags() { + return tags; + } + // CHECKSTYLE:OFF @Override public String toString() { @@ -117,6 +129,7 @@ public String toString() { ", replyTo='" + replyTo + '\'' + ", recipients=" + recipients + ", attachments=" + attachments + + ", tags=" + tags + '}'; } // CHECKSTYLE:ON diff --git a/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailSpec.groovy b/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailSpec.groovy index 80a662f0b..f43f50a73 100644 --- a/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailSpec.groovy +++ b/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailSpec.groovy @@ -46,6 +46,7 @@ class SendEmailSpec extends Specification { file.createNewFile() file.text = 'not a real PDF' String thePath = file.canonicalPath + Map mapOfTags = [myTagKey: 'myTagValue'] when: // tag::builder[] EmailDeliveryStatus status = service.send { // <1> @@ -53,11 +54,12 @@ class SendEmailSpec extends Specification { from 'subscribe@groovycalamari.com' // <3> to 'me@sergiodelamo.com' // <4> htmlBody '

This is an example body

' // <5> - attachment { // <6> - filepath thePath // <7> - filename 'test.pdf' // <8> - mimeType 'application/pdf' // <9> - description 'An example pdf' // <10> + tags mapOfTags // <6> + attachment { // <7> + filepath thePath // <8> + filename 'test.pdf' // <9> + mimeType 'application/pdf' // <10> + description 'An example pdf' // <11> } } // end::builder[] diff --git a/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailTest.java b/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailTest.java index 667f40a36..da378ba04 100644 --- a/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailTest.java +++ b/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SendEmailTest.java @@ -28,6 +28,8 @@ import java.io.IOException; import java.nio.file.Files; import java.util.Collections; +import java.util.HashMap; +import java.util.Map; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; @@ -55,6 +57,8 @@ public void testSendEmail() throws IOException { file.createNewFile(); Files.write(file.toPath(), Collections.singletonList("not a real PDF")); String filepath = file.getCanonicalPath(); + Map mapOfTags = new HashMap<>(); + mapOfTags.put("myTagKey", "myTagValue"); // tag::builder[] EmailDeliveryStatus status = service.send(e -> // <1> @@ -62,11 +66,12 @@ public void testSendEmail() throws IOException { .from("subscribe@groovycalamari.com") // <3> .to("me@sergiodelamo.com") // <4> .htmlBody("

This is an example body

") // <5> - .attachment(a -> // <6> - a.filepath(filepath) // <7> - .filename("test.pdf") // <8> - .mimeType("application/pdf") // <9> - .description("An example pdf") // <10> + .tags(mapOfTags) // <6> + .attachment(a -> // <7> + a.filepath(filepath) // <8> + .filename("test.pdf") // <9> + .mimeType("application/pdf") // <10> + .description("An example pdf") // <11> ) ); // end::builder[] diff --git a/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SimpleEmailServiceSpec.groovy b/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SimpleEmailServiceSpec.groovy index 6ceab28a0..fc1614bfa 100644 --- a/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SimpleEmailServiceSpec.groovy +++ b/subprojects/micronaut-aws-sdk-ses/src/test/groovy/com/agorapulse/micronaut/aws/ses/SimpleEmailServiceSpec.groovy @@ -41,12 +41,15 @@ class SimpleEmailServiceSpec extends Specification { )) void "test transactionalEmailWithClosure"() { + given: + Map customTags = [key1: 'value1', key2: 'value2'] when: TransactionalEmail transactionalEmail = SimpleEmailService.email { subject 'Hi Paul' from 'subscribe@groovycalamari.com' to 'me@sergiodelamo.com' htmlBody '

This is an example body

' + tags customTags attachment { filename 'test.pdf' filepath '/tmp/test.pdf' @@ -61,6 +64,7 @@ class SimpleEmailServiceSpec extends Specification { transactionalEmail.htmlBody == '

This is an example body

' transactionalEmail.from == 'subscribe@groovycalamari.com' transactionalEmail.recipients == ['me@sergiodelamo.com'] + transactionalEmail.tags == [key1: 'value1', key2: 'value2'] transactionalEmail.attachments.size() == 1 transactionalEmail.attachments.first().filename == 'test.pdf' transactionalEmail.attachments.first().filepath == '/tmp/test.pdf' @@ -79,6 +83,7 @@ class SimpleEmailServiceSpec extends Specification { htmlBody '

This is an example body

' to 'me@sergiodelamo.com' from 'subscribe@groovycalamari.com' + tags customTags attachment { filepath f.absolutePath } @@ -90,6 +95,7 @@ class SimpleEmailServiceSpec extends Specification { transactionalEmail.htmlBody == '

This is an example body

' transactionalEmail.from == 'subscribe@groovycalamari.com' transactionalEmail.recipients == ['me@sergiodelamo.com'] + transactionalEmail.tags == [key1: 'value1', key2: 'value2'] transactionalEmail.attachments.size() == 1 transactionalEmail.attachments.first().filename == 'groovylogo.png' transactionalEmail.attachments.first().filepath == f.absolutePath