cloud.localstack.docker.LocalstackDocker Maven / Gradle / Ivy
package cloud.localstack.docker;
import cloud.localstack.LocalstackTestRunner;
import cloud.localstack.ServiceName;
import cloud.localstack.docker.command.RegexStream;
import cloud.localstack.docker.exception.LocalstackDockerException;
import lombok.Getter;
import lombok.Setter;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Logger;
import java.util.regex.Pattern;
import java.util.stream.Collectors;
/**
* Localstack Docker instance
*
* @author Alan Bevier
* @author fabianoo
*/
public class LocalstackDocker {
private static final Logger LOG = Logger.getLogger(LocalstackDocker.class.getName());
private static final String PORT_CONFIG_FILENAME = "/opt/code/localstack/.venv/lib/python2.7/site-packages/localstack_client/config.py";
private static final Pattern READY_TOKEN = Pattern.compile("Ready\\.");
//Regular expression used to parse localstack config to determine default ports for services
private static final Pattern DEFAULT_PORT_PATTERN = Pattern.compile("'(\\w+)'\\Q: '{proto}://{host}:\\E(\\d+)'");
private static final int SERVICE_NAME_GROUP = 1;
private static final int PORT_GROUP = 2;
private Container localStackContainer;
/**
* This is a mapping from service name to internal ports. In order to use them, the
* internal port must be resolved to an external docker port via Container.getExternalPortFor()
*/
private static Map serviceToPortMap;
private static boolean locked = false;
@Setter
private String externalHostName = "localhost";
@Setter
private boolean pullNewImage = true;
@Setter
private boolean randomizePorts = false;
@Setter
private Map environmentVariables = new HashMap<>();
@Getter
private static LocalstackDocker localstackDocker = new LocalstackDocker();
private LocalstackDocker() {}
public void startup() {
// make sure the local infrastructure is not running, to avoid port conflicts
LocalstackTestRunner.teardownInfrastructure();
// now create the container
if (locked) {
throw new IllegalStateException("A docker instance is starting or already started.");
}
locked = true;
try {
localStackContainer = Container.createLocalstackContainer(externalHostName, pullNewImage, randomizePorts, environmentVariables);
loadServiceToPortMap();
LOG.info("Waiting for localstack container to be ready...");
localStackContainer.waitForLogToken(READY_TOKEN);
} catch (Throwable t) {
locked = false;
throw new LocalstackDockerException("Could not start the localstack docker container.", t);
}
}
public void stop() {
localStackContainer.stop();
locked = false;
}
private void loadServiceToPortMap() {
String localStackPortConfig = localStackContainer.executeCommand(Arrays.asList("cat", PORT_CONFIG_FILENAME));
Map ports = new RegexStream(DEFAULT_PORT_PATTERN.matcher(localStackPortConfig)).stream()
.collect(Collectors.toMap(match -> match.group(SERVICE_NAME_GROUP),
match -> Integer.parseInt(match.group(PORT_GROUP))));
serviceToPortMap = Collections.unmodifiableMap(ports);
}
public String getEndpointS3() {
String s3Endpoint = endpointForService(ServiceName.S3);
/*
* Use the domain name wildcard *.localhost.atlassian.io which maps to 127.0.0.1
* We need to do this because S3 SDKs attempt to access a domain .
* which by default would result in .localhost, but that name cannot be resolved
* (unless hardcoded in /etc/hosts)
*/
s3Endpoint = s3Endpoint.replace("localhost", "test.localhost.atlassian.io");
return s3Endpoint;
}
public String getEndpointKinesis() {
return endpointForService(ServiceName.KINESIS);
}
public String getEndpointLambda() {
return endpointForService(ServiceName.LAMBDA);
}
public String getEndpointDynamoDB() {
return endpointForService(ServiceName.DYNAMO);
}
public String getEndpointDynamoDBStreams() {
return endpointForService(ServiceName.DYNAMO_STREAMS);
}
public String getEndpointAPIGateway() {
return endpointForService(ServiceName.API_GATEWAY);
}
public String getEndpointElasticsearch() {
return endpointForService(ServiceName.ELASTICSEARCH);
}
public String getEndpointElasticsearchService() {
return endpointForService(ServiceName.ELASTICSEARCH_SERVICE);
}
public String getEndpointFirehose() {
return endpointForService(ServiceName.FIREHOSE);
}
public String getEndpointSNS() {
return endpointForService(ServiceName.SNS);
}
public String getEndpointSQS() {
return endpointForService(ServiceName.SQS);
}
public String getEndpointRedshift() {
return endpointForService(ServiceName.REDSHIFT);
}
public String getEndpointSES() {
return endpointForService(ServiceName.SES);
}
public String getEndpointRoute53() {
return endpointForService(ServiceName.ROUTE53);
}
public String getEndpointCloudFormation() {
return endpointForService(ServiceName.CLOUDFORMATION);
}
public String getEndpointCloudWatch() {
return endpointForService(ServiceName.CLOUDWATCH);
}
public String getEndpointSSM() {
return endpointForService(ServiceName.SSM);
}
public String endpointForService(String serviceName) {
if (serviceToPortMap == null) {
throw new IllegalStateException("Service to port mapping has not been determined yet.");
}
if (!serviceToPortMap.containsKey(serviceName)) {
throw new IllegalArgumentException("Unknown port mapping for service");
}
int internalPort = serviceToPortMap.get(serviceName);
return endpointForPort(internalPort);
}
public String endpointForPort(int port) {
if (localStackContainer != null) {
int externalPort = localStackContainer.getExternalPortFor(port);
return String.format("http://%s:%s", externalHostName, externalPort);
}
throw new RuntimeException("Container not started");
}
public Container getLocalStackContainer() {
return localStackContainer;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy