Skip to content

Commit 504224f

Browse files
(feat) add Localstack SNS (#639)
* (feat) add Localstack SNS * fix (simplify) test * specify build enabled after 2.6.0
1 parent 5b5d13b commit 504224f

File tree

10 files changed

+161
-1
lines changed

10 files changed

+161
-1
lines changed

gradle/libs.versions.toml

+2
Original file line numberDiff line numberDiff line change
@@ -113,9 +113,11 @@ spock = { module = "org.spockframework:spock-core", version.ref = "spock" }
113113
amazon-awssdk-v1-dynamodb = { module = "com.amazonaws:aws-java-sdk-dynamodb", version.ref = "amazon-awssdk-v1" }
114114
amazon-awssdk-v1-s3 = { module = "com.amazonaws:aws-java-sdk-s3", version.ref = "amazon-awssdk-v1" }
115115
amazon-awssdk-v1-sqs = { module = "com.amazonaws:aws-java-sdk-sqs", version.ref = "amazon-awssdk-v1" }
116+
amazon-awssdk-v1-sns = { module = "com.amazonaws:aws-java-sdk-sns", version.ref = "amazon-awssdk-v1" }
116117
amazon-awssdk-v2-dynamodb = { module = "software.amazon.awssdk:dynamodb", version.ref = "amazon-awssdk-v2" }
117118
amazon-awssdk-v2-s3 = { module = "software.amazon.awssdk:s3", version.ref = "amazon-awssdk-v2" }
118119
amazon-awssdk-v2-sqs = { module = "software.amazon.awssdk:sqs", version.ref = "amazon-awssdk-v2" }
120+
amazon-awssdk-v2-sns = { module = "software.amazon.awssdk:sns", version.ref = "amazon-awssdk-v2" }
119121

120122
jansi = { module = "org.fusesource.jansi:jansi", version.ref = "jansi" }
121123
#

settings.gradle

+2-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,8 @@ def localstackModules = [
5757
'core',
5858
's3',
5959
'dynamodb',
60-
'sqs'
60+
'sqs',
61+
'sns'
6162
]
6263

6364
include 'test-resources-bom'

src/main/docs/guide/modules-localstack.adoc

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ The following services are supported:
77
- S3, by providing the `aws.services.s3.endpoint-override` property
88
- DynamoDB, by providing the `aws.services.dynamodb.endpoint-override` property
99
- SQS, by providing the `aws.services.sqs.endpoint-override` property
10+
- SNS, by providing the `aws.services.sns.endpoint-override` property
1011
1112
In addition, the following properties will be resolved by test resources:
1213

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
plugins {
2+
id 'io.micronaut.build.internal.localstack-module'
3+
}
4+
5+
micronautBuild {
6+
binaryCompatibility {
7+
enabledAfter '2.6.0'
8+
}
9+
}
10+
11+
description = """
12+
Provides support for Localstack SNS.
13+
"""
14+
15+
dependencies {
16+
implementation(project(":micronaut-test-resources-localstack-core"))
17+
runtimeOnly(libs.amazon.awssdk.v1.sns) {
18+
because "Localstack requires the AWS SDK in version 1 at runtime"
19+
}
20+
testImplementation(testFixtures(project(":micronaut-test-resources-localstack-core")))
21+
testImplementation(libs.amazon.awssdk.v2.sns)
22+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
/*
2+
* Copyright 2017-2024 original authors
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
package io.micronaut.testresources.localstack.sns;
17+
18+
import io.micronaut.testresources.localstack.LocalStackService;
19+
import org.testcontainers.containers.localstack.LocalStackContainer;
20+
21+
import java.util.Collections;
22+
import java.util.List;
23+
import java.util.Optional;
24+
25+
/**
26+
* Adds support for Localstack SNS.
27+
*/
28+
public class LocalStackSNSService implements LocalStackService {
29+
30+
private static final String AWS_SNS_ENDPOINT_OVERRIDE = "aws.services.sns.endpoint-override";
31+
32+
@Override
33+
public LocalStackContainer.Service getServiceKind() {
34+
return LocalStackContainer.Service.SNS;
35+
}
36+
37+
@Override
38+
public List<String> getResolvableProperties() {
39+
return Collections.singletonList(AWS_SNS_ENDPOINT_OVERRIDE);
40+
}
41+
42+
@Override
43+
public Optional<String> resolveProperty(String propertyName, LocalStackContainer container) {
44+
if (AWS_SNS_ENDPOINT_OVERRIDE.equals(propertyName)) {
45+
return Optional.of(container.getEndpointOverride(LocalStackContainer.Service.SNS).toString());
46+
}
47+
return Optional.empty();
48+
}
49+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
io.micronaut.testresources.localstack.sns.LocalStackSNSService
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
package io.micronaut.testresources.localstack.sns
2+
3+
import io.micronaut.test.extensions.spock.annotation.MicronautTest
4+
import io.micronaut.testresources.localstack.AbstractLocalStackSpec
5+
import jakarta.inject.Inject
6+
import org.junit.Assert
7+
import software.amazon.awssdk.services.sns.SnsClient
8+
import software.amazon.awssdk.services.sns.model.Topic
9+
10+
@MicronautTest
11+
class LocalStackSNSTest extends AbstractLocalStackSpec {
12+
13+
@Inject
14+
SnsClient client
15+
16+
def "automatically starts a SNS container"() {
17+
when:
18+
client.createTopic {
19+
it.name("test-topic")
20+
}
21+
22+
then:
23+
List<Topic> topics = client.listTopics().topics()
24+
Assert.assertEquals(1, topics.size())
25+
Assert.assertTrue(topics[0].topicArn().endsWith("test-topic"))
26+
}
27+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
package io.micronaut.testresources.localstack.sns
2+
3+
import io.micronaut.context.annotation.Factory
4+
import jakarta.inject.Singleton
5+
import software.amazon.awssdk.auth.credentials.AwsBasicCredentials
6+
import software.amazon.awssdk.auth.credentials.StaticCredentialsProvider
7+
import software.amazon.awssdk.regions.Region
8+
import software.amazon.awssdk.services.sns.SnsClient
9+
10+
@Factory
11+
class TestSnsClientFactory {
12+
13+
@Singleton
14+
SnsClient snsClient(TestSnsConfig testSnsConfig) {
15+
return SnsClient.builder()
16+
.endpointOverride(new URI(testSnsConfig.sns.endpointOverride))
17+
.credentialsProvider(
18+
StaticCredentialsProvider.create(
19+
AwsBasicCredentials.create(testSnsConfig.accessKeyId, testSnsConfig.secretKey)
20+
)
21+
)
22+
.region(Region.of(testSnsConfig.region))
23+
.build()
24+
}
25+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
package io.micronaut.testresources.localstack.sns
2+
3+
import io.micronaut.context.annotation.ConfigurationBuilder
4+
import io.micronaut.context.annotation.ConfigurationProperties
5+
6+
@ConfigurationProperties("aws")
7+
class TestSnsConfig {
8+
String accessKeyId
9+
String secretKey
10+
String region
11+
12+
@ConfigurationBuilder(configurationPrefix = "services.sns")
13+
final Sns sns = new Sns()
14+
15+
static class Sns {
16+
String endpointOverride
17+
}
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
<configuration>
2+
3+
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
4+
<!-- encoders are assigned the type
5+
ch.qos.logback.classic.encoder.PatternLayoutEncoder by default -->
6+
<encoder>
7+
<pattern>%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n</pattern>
8+
</encoder>
9+
</appender>
10+
11+
<root level="info">
12+
<appender-ref ref="STDOUT" />
13+
</root>
14+
</configuration>

0 commit comments

Comments
 (0)