com.github.nosan.embedded.cassandra.CassandraBuilder Maven / Gradle / Ivy
/*
* Copyright 2020-2021 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.github.nosan.embedded.cassandra;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Supplier;
import com.github.nosan.embedded.cassandra.commons.Resource;
import com.github.nosan.embedded.cassandra.commons.StringUtils;
import com.github.nosan.embedded.cassandra.commons.function.IOSupplier;
import com.github.nosan.embedded.cassandra.commons.logging.Logger;
import com.github.nosan.embedded.cassandra.commons.web.JdkHttpClient;
/**
* A builder that can be used to configure and create {@link Cassandra}.
*
* This class is not thread safe and should not be shared across different threads!
*
*
* @author Dmytro Nosan
* @see #build()
* @see CassandraBuilderConfigurator
* @since 4.0.0
*/
public class CassandraBuilder {
/**
* Default Cassandra version.
*/
public static final Version DEFAULT_VERSION = Version.parse("4.1.7");
private static final AtomicInteger CASSANDRA_ID = new AtomicInteger();
private final Map environmentVariables = new LinkedHashMap<>();
private final Map configProperties = new LinkedHashMap<>();
private final Map systemProperties = new LinkedHashMap<>();
private final Set jvmOptions = new LinkedHashSet<>();
private final Set workingDirectoryCustomizers = new LinkedHashSet<>();
private boolean registerShutdownHook = true;
private String name;
private Version version;
private Duration startupTimeout;
private Logger logger;
private IOSupplier extends Path> workingDirectorySupplier;
private WorkingDirectoryDestroyer workingDirectoryDestroyer;
private WorkingDirectoryInitializer workingDirectoryInitializer;
/**
* Build a new {@link Cassandra} instance.
*
* @return a {@link Cassandra} instance.
*/
public Cassandra build() {
String name = (this.name != null) ? this.name : "cassandra-" + CASSANDRA_ID.getAndIncrement();
Version version = (this.version != null) ? this.version : DEFAULT_VERSION;
Path workingDirectory;
try {
IOSupplier extends Path> workingDirectorySupplier = this.workingDirectorySupplier;
if (workingDirectorySupplier != null) {
workingDirectory = workingDirectorySupplier.get();
Objects.requireNonNull(workingDirectory, "Working Directory must not be null");
}
else {
workingDirectory = Files.createTempDirectory("");
}
}
catch (IOException ex) {
throw new UncheckedIOException("Unable to get a working directory", ex);
}
WorkingDirectoryInitializer workingDirectoryInitializer = this.workingDirectoryInitializer;
if (workingDirectoryInitializer == null) {
workingDirectoryInitializer = new DefaultWorkingDirectoryInitializer(new WebCassandraDirectoryProvider(
new JdkHttpClient(Duration.ofMinutes(1), Duration.ofMinutes(1))));
}
WorkingDirectoryDestroyer workingDirectoryDestroyer = this.workingDirectoryDestroyer;
if (workingDirectoryDestroyer == null) {
workingDirectoryDestroyer = WorkingDirectoryDestroyer.deleteAll();
}
Duration startupTimeout = this.startupTimeout;
if (startupTimeout == null) {
startupTimeout = Duration.ofMinutes(2);
}
Logger logger = this.logger;
if (logger == null) {
logger = Logger.get(Cassandra.class);
}
Map environmentVariables = new LinkedHashMap<>(this.environmentVariables);
environmentVariables.values().removeIf(Objects::isNull);
Map systemProperties = new LinkedHashMap<>(this.systemProperties);
systemProperties.values().removeIf(Objects::isNull);
Set jvmOptions = new LinkedHashSet<>(this.jvmOptions);
jvmOptions.removeIf(Objects::isNull);
Set workingDirectoryCustomizers = new LinkedHashSet<>(
this.workingDirectoryCustomizers);
workingDirectoryCustomizers.removeIf(Objects::isNull);
Map configProperties = new LinkedHashMap<>(this.configProperties);
CassandraDatabaseFactory databaseFactory = new DefaultCassandraDatabaseFactory(name, version,
environmentVariables, configProperties, systemProperties, jvmOptions);
return new DefaultCassandra(name, version, workingDirectory.normalize().toAbsolutePath(),
this.registerShutdownHook, workingDirectoryInitializer, workingDirectoryDestroyer, startupTimeout,
workingDirectoryCustomizers, databaseFactory, logger);
}
/**
* Sets the Cassandra instance name.
* Defaults to cassandra-0, cassandra-1, and so on.
*
* @param name the Cassandra name
* @return this builder
*/
public CassandraBuilder name(String name) {
Objects.requireNonNull(name, "Name must not be null");
if (!StringUtils.hasText(name)) {
throw new IllegalArgumentException("Name must not be empty");
}
this.name = name;
return this;
}
/**
* Gets current configured Cassandra instance name.
*
* @return Cassandra instance name, or {@code null}
*/
public String getName() {
return this.name;
}
/**
* Sets the Cassandra version.
*
Defaults to {@link #DEFAULT_VERSION}
*
* @param version the Cassandra version
* @return this builder
*/
public CassandraBuilder version(String version) {
Objects.requireNonNull(version, "Version must not be null");
return version(Version.parse(version));
}
/**
* Sets the Cassandra version.
*
Defaults to {@link #DEFAULT_VERSION}
*
* @param version the Cassandra version
* @return this builder
*/
public CassandraBuilder version(Version version) {
Objects.requireNonNull(version, "Version must not be null");
this.version = version;
return this;
}
/**
* Gets current configured Cassandra version or {@link #DEFAULT_VERSION}.
*
* @return Cassandra version, never {@code null}
*/
public Version getVersion() {
Version version = this.version;
return (version != null) ? version : DEFAULT_VERSION;
}
/**
* Sets the Cassandra logger. This logger will consume Cassandra stdout and stderr outputs.
*
Defaults to {@code Logger.get(Cassandra.class)}
*
* @param logger the Cassandra logger
* @return this builder
*/
public CassandraBuilder logger(Logger logger) {
Objects.requireNonNull(logger, "Logger must not be null");
this.logger = logger;
return this;
}
/**
* Sets if the created {@link Cassandra} should have a shutdown hook registered.
*
Defaults to {@code true}.
*
* @param registerShutdownHook {@code true} if shutdown hook should be registered, otherwise {@code false}
* @return this builder
*/
public CassandraBuilder registerShutdownHook(boolean registerShutdownHook) {
this.registerShutdownHook = registerShutdownHook;
return this;
}
/**
* Sets the startup timeout.
*
Defaults to 2 minutes.
*
* @param startupTimeout the startup timeout
* @return this builder
*/
public CassandraBuilder startupTimeout(Duration startupTimeout) {
Objects.requireNonNull(startupTimeout, "Startup Timeout must not be null");
if (startupTimeout.isZero() || startupTimeout.isNegative()) {
throw new IllegalArgumentException("Startup Timeout must be positive");
}
this.startupTimeout = startupTimeout;
return this;
}
/**
* Sets the Cassandra configuration file.
*
* Equal to:
*
* {@code
* addSystemProperty("cassandra.config", configFile);
* }
*
*
* @param configFile the config file
* @return this builder
*/
public CassandraBuilder configFile(Resource configFile) {
return addSystemProperty("cassandra.config", configFile);
}
/**
* Set the {@code Supplier} of the working directory that should be called each time when {@link #build()} is
* called.
* The supplied directory will be initialized by {@link
* WorkingDirectoryInitializer} and used as Cassandra home directory.
*
* In the end, provided working directory may be deleted or partly deleted by {@link WorkingDirectoryDestroyer}.
*
Defaults to {@code Files.createTempDirectory("")}
*
* @param workingDirectorySupplier the working directory supplier
* @return this builder
* @see IOSupplier#wrap(Supplier)
*/
public CassandraBuilder workingDirectory(IOSupplier extends Path> workingDirectorySupplier) {
Objects.requireNonNull(workingDirectorySupplier, "Working Directory Supplier must not be null");
this.workingDirectorySupplier = workingDirectorySupplier;
return this;
}
/**
* Sets the {@link WorkingDirectoryInitializer}.
*
Defaults to {@link DefaultWorkingDirectoryInitializer} with the
* underlying {@link WebCassandraDirectoryProvider}.
*
* @param workingDirectoryInitializer the working directory initializer
* @return this builder
* @see DefaultWorkingDirectoryInitializer
*/
public CassandraBuilder workingDirectoryInitializer(WorkingDirectoryInitializer workingDirectoryInitializer) {
Objects.requireNonNull(workingDirectoryInitializer, "Working Directory Initializer must not be null");
this.workingDirectoryInitializer = workingDirectoryInitializer;
return this;
}
/**
* Sets the {@link WorkingDirectoryDestroyer}.
*
* Defaults to {@code WorkingDirectoryDestroyer.deleteAll()}
*
* @param workingDirectoryDestroyer the working directory destroyer
* @return this builder
* @see WorkingDirectoryDestroyer#doNothing()
* @see WorkingDirectoryDestroyer#deleteOnly(String...)
* @see WorkingDirectoryDestroyer#deleteAll()
*/
public CassandraBuilder workingDirectoryDestroyer(WorkingDirectoryDestroyer workingDirectoryDestroyer) {
Objects.requireNonNull(workingDirectoryDestroyer, "Working Directory Destroyer must not be null");
this.workingDirectoryDestroyer = workingDirectoryDestroyer;
return this;
}
/**
* Sets the {@link WorkingDirectoryCustomizer}. Setting this value will replace any previously configured
* customizers.
*
* @param workingDirectoryCustomizers the working directory customizers to set
* @return this builder
* @see WorkingDirectoryCustomizer#addResource(Resource, String)
* @see #workingDirectoryCustomizers(Collection)
* @see #addWorkingDirectoryCustomizers(WorkingDirectoryCustomizer...)
* @see #addWorkingDirectoryCustomizers(Collection)
*/
public CassandraBuilder workingDirectoryCustomizers(WorkingDirectoryCustomizer... workingDirectoryCustomizers) {
Objects.requireNonNull(workingDirectoryCustomizers, "Working Directory Customizers must not be null");
return workingDirectoryCustomizers(Arrays.asList(workingDirectoryCustomizers));
}
/**
* Sets the {@link WorkingDirectoryCustomizer}. Setting this value will replace any previously configured
* customizers.
*
* @param workingDirectoryCustomizers the working directory customizers to set
* @return this builder
* @see WorkingDirectoryCustomizer#addResource(Resource, String)
* @see #workingDirectoryCustomizers(WorkingDirectoryCustomizer...)
* @see #addWorkingDirectoryCustomizers(WorkingDirectoryCustomizer...)
* @see #addWorkingDirectoryCustomizers(Collection)
*/
public CassandraBuilder workingDirectoryCustomizers(
Collection extends WorkingDirectoryCustomizer> workingDirectoryCustomizers) {
Objects.requireNonNull(workingDirectoryCustomizers, "Working Directory Customizers must not be null");
this.workingDirectoryCustomizers.clear();
this.workingDirectoryCustomizers.addAll(deepCopy(workingDirectoryCustomizers));
return this;
}
/**
* Adds the {@link WorkingDirectoryCustomizer}.
*
* @param workingDirectoryCustomizers the working directory customizers to add
* @return this builder
* @see WorkingDirectoryCustomizer#addResource(Resource, String)
* @see #addWorkingDirectoryCustomizers(Collection)
* @see #workingDirectoryCustomizers(WorkingDirectoryCustomizer...)
* @see #workingDirectoryCustomizers(Collection)
*/
public CassandraBuilder addWorkingDirectoryCustomizers(WorkingDirectoryCustomizer... workingDirectoryCustomizers) {
Objects.requireNonNull(workingDirectoryCustomizers, "Working Directory Customizers must not be null");
return addWorkingDirectoryCustomizers(Arrays.asList(workingDirectoryCustomizers));
}
/**
* Adds the {@link WorkingDirectoryCustomizer}.
*
* @param workingDirectoryCustomizers the working directory customizers to add
* @return this builder
* @see WorkingDirectoryCustomizer#addResource(Resource, String)
* @see #addWorkingDirectoryCustomizers(WorkingDirectoryCustomizer...)
* @see #workingDirectoryCustomizers(WorkingDirectoryCustomizer...)
* @see #workingDirectoryCustomizers(Collection)
*/
public CassandraBuilder addWorkingDirectoryCustomizers(
Collection extends WorkingDirectoryCustomizer> workingDirectoryCustomizers) {
Objects.requireNonNull(workingDirectoryCustomizers, "Working Directory Customizers must not be null");
this.workingDirectoryCustomizers.addAll(deepCopy(workingDirectoryCustomizers));
return this;
}
/**
* Sets Cassandra environment variables. Setting this value will replace any previously configured environment
* variables. For example:
*
* {@code
* Map environmentVariables = new LinkedHashMap<>();
* environmentVariables.put("TZ", "Europe/London");
* builder.environmentVariables(environmentVariables);
* }
*
* @param environmentVariables Cassandra environment variables
* @return this builder
* @see #addEnvironmentVariable(String, Object)
* @see #addEnvironmentVariables(Map)
*/
public CassandraBuilder environmentVariables(Map environmentVariables) {
Objects.requireNonNull(environmentVariables, "Environment Variables must not be null");
this.environmentVariables.clear();
this.environmentVariables.putAll(deepCopy(environmentVariables));
return this;
}
/**
* Puts Cassandra environment variable. For example:
*
* {@code
* builder.addEnvironmentVariable("TZ", "Europe/London");
* }
*
*
* @param name Cassandra environment variable name
* @param value Cassandra environment variable value
* @return this builder
* @see #addEnvironmentVariables(Map)
* @see #environmentVariables(Map)
*/
public CassandraBuilder addEnvironmentVariable(String name, Object value) {
return addEnvironmentVariables(Collections.singletonMap(name, value));
}
/**
* Puts Cassandra environment variables. For example:
*
* {@code
* Map environmentVariables = new LinkedHashMap<>();
* environmentVariables.put("TZ", "Europe/London");
* builder.addEnvironmentVariables(environmentVariables);
* }
*
* @param environmentVariables Cassandra environment variables
* @return this builder
* @see #addEnvironmentVariable(String, Object)
* @see #environmentVariables(Map)
*/
public CassandraBuilder addEnvironmentVariables(Map environmentVariables) {
Objects.requireNonNull(environmentVariables, "Environment Variables must not be null");
this.environmentVariables.putAll(deepCopy(environmentVariables));
return this;
}
/**
* Sets Cassandra native Java Virtual Machine (JVM) system parameters. Setting this value will replace any
* previously configured system parameters. For example:
*
* {@code
* Map systemProperties = new LinkedHashMap<>();
* systemProperties.put("cassandra.config", new ClassPathResource("cassandra.yaml"));
* systemProperties.put("cassandra.native_transport_port", 9042);
* systemProperties.put("cassandra.jmx.local.port", 7199);
* builder.systemProperties(systemProperties);
* }
*
* @param systemProperties Cassandra system parameters
* @return this builder
* @see #addSystemProperty(String, Object)
*/
public CassandraBuilder systemProperties(Map systemProperties) {
Objects.requireNonNull(systemProperties, "System Properties must not be null");
this.systemProperties.clear();
this.systemProperties.putAll(deepCopy(systemProperties));
return this;
}
/**
* Puts Cassandra native Java Virtual Machine (JVM) system parameter, For example:
* {@code addSystemProperty("cassandra.config",new ClassPathResource("cassandra.yaml"))
* addSystemProperty("user.timezone","Europe/London")}.
*
*
* @param name Cassandra system parameter name
* @param value Cassandra system parameter value
* @return this builder
* @see #addSystemProperties(Map)
*/
public CassandraBuilder addSystemProperty(String name, Object value) {
return addSystemProperties(Collections.singletonMap(name, value));
}
/**
* Puts Cassandra native Java Virtual Machine (JVM) system parameters. For example:
*
* {@code
* Map systemProperties = new LinkedHashMap<>();
* systemProperties.put("cassandra.config", new ClassPathResource("cassandra.yaml"));
* systemProperties.put("cassandra.native_transport_port", 9042);
* systemProperties.put("cassandra.jmx.local.port", 7199);
* builder.addSystemProperties(systemProperties);
* }
*
* @param systemProperties Cassandra system parameters
* @return this builder
* @see #addSystemProperty(String, Object)
*/
public CassandraBuilder addSystemProperties(Map systemProperties) {
Objects.requireNonNull(systemProperties, "System Properties must not be null");
this.systemProperties.putAll(deepCopy(systemProperties));
return this;
}
/**
* Sets Cassandra native Java Virtual Machine (JVM) Options. Setting this value will replace any previously
* configured options.
*
* @param jvmOptions the JVM options to set
* @return this builder
* @see #jvmOptions(Collection)
* @see #addJvmOptions(Collection)
* @see #addJvmOptions(String...)
*/
public CassandraBuilder jvmOptions(String... jvmOptions) {
Objects.requireNonNull(jvmOptions, "JVM Options must not be null");
return jvmOptions(Arrays.asList(jvmOptions));
}
/**
* Sets Cassandra native Java Virtual Machine (JVM) Options. Setting this value will replace any previously
* configured options.
*
* @param jvmOptions the JVM options to set
* @return this builder
* @see #jvmOptions(Collection)
* @see #addJvmOptions(Collection)
* @see #addJvmOptions(String...)
*/
public CassandraBuilder jvmOptions(Collection jvmOptions) {
Objects.requireNonNull(jvmOptions, "JVM Options must not be null");
this.jvmOptions.clear();
this.jvmOptions.addAll(deepCopy(jvmOptions));
return this;
}
/**
* Adds Cassandra native Java Virtual Machine (JVM) Options.
*
* @param jvmOptions the JVM options to add
* @return this builder
* @see #addJvmOptions(Collection)
* @see #jvmOptions(Collection)
* @see #jvmOptions(String...)
*/
public CassandraBuilder addJvmOptions(String... jvmOptions) {
Objects.requireNonNull(jvmOptions, "JVM Options must not be null");
return addJvmOptions(Arrays.asList(jvmOptions));
}
/**
* Adds Cassandra native Java Virtual Machine (JVM) Options.
*
* @param jvmOptions the JVM options to add
* @return this builder
* @see #addJvmOptions(String...)
* @see #jvmOptions(Collection)
* @see #jvmOptions(String...)
*/
public CassandraBuilder addJvmOptions(Collection jvmOptions) {
Objects.requireNonNull(jvmOptions, "JVM Options must not be null");
this.jvmOptions.addAll(deepCopy(jvmOptions));
return this;
}
/**
* Sets Cassandra config properties, that should be merged with properties from cassandra.yaml. Setting this value
* will replace any previously configured config properties. For example:
*
* {@code
* Map properties = new LinkedHashMap<>();
* properties.put("client_encryption_options.enabled", true);
* properties.put("cluster_name", "MyCluster");
* builder.configProperties(properties);
* }
*
* Output Yaml:
*
* ...
* cluster_name: "MyCluster"
* client_encryption_options:
* enabled: true
* ...
* ...
*
*
* @param configProperties Cassandra config properties
* @return this builder
* @see #addConfigProperty(String, Object)
* @see #addConfigProperties(Map)
*/
public CassandraBuilder configProperties(Map configProperties) {
Objects.requireNonNull(configProperties, "Config Properties must not be null");
this.configProperties.clear();
this.configProperties.putAll(deepCopy(configProperties));
return this;
}
/**
* Puts Cassandra config property, that should be merged with a property from cassandra.yaml. For example:
*
* {@code
* builder.configProperty("client_encryption_options.enabled",true)
* .configProperty("cluster_name","MyCluster")}
*
* Output Yaml:
*
* ...
* cluster_name: "MyCluster"
* client_encryption_options:
* enabled: true
* ...
* ...
*
*
* @param name config property name (e.g. native_transport_port, client_encryption_options.enabled)
* @param value config property value
* @return this builder
* @see #addConfigProperties(Map)
* @see #configProperties(Map)
*/
public CassandraBuilder addConfigProperty(String name, Object value) {
return addConfigProperties(Collections.singletonMap(name, value));
}
/**
* Puts Cassandra config properties, that should be merged with properties from cassandra.yaml. For example:
*
* {@code
* Map properties = new LinkedHashMap<>();
* properties.put("client_encryption_options.enabled", true);
* properties.put("cluster_name", "MyCluster");
* builder.configProperties(properties);
* }
*
* Output Yaml:
*
* ...
* cluster_name: "MyCluster"
* client_encryption_options:
* enabled: true
* ...
* ...
*
*
* @param configProperties Cassandra config properties
* @return this builder
* @see #addConfigProperty(String, Object)
* @see #configProperties(Map)
*/
public CassandraBuilder addConfigProperties(Map configProperties) {
Objects.requireNonNull(configProperties, "Config Properties must not be null");
this.configProperties.putAll(deepCopy(configProperties));
return this;
}
/**
* Copy a resource to a target path within the working directory.
*
* Equal to:
*
* {@code
* addWorkingDirectoryCustomizers(WorkingDirectoryCustomizer.addResource(resource, path));
* }
*
*
* @param path path (file only) within the working directory (e.g conf/cassandra.yaml)
* @param resource the resource
* @return this builder
* @see WorkingDirectoryCustomizer#addResource(Resource, String)
*/
public CassandraBuilder addWorkingDirectoryResource(Resource resource, String path) {
return addWorkingDirectoryCustomizers(WorkingDirectoryCustomizer.addResource(resource, path));
}
/**
* Applies the provided configurator to this builder.
*
* @param configurator configurator to use
* @return this builder
* @see SimpleSeedProviderConfigurator
*/
public CassandraBuilder configure(CassandraBuilderConfigurator configurator) {
Objects.requireNonNull(configurator, "Cassandra Builder Configurator must not be null");
configurator.configure(this);
return this;
}
@SuppressWarnings("unchecked")
private static T deepCopy(T object) {
if (object instanceof Map, ?>) {
Map