Skip to content

Commit

Permalink
opensearch keystore used for key/truststore passwords (#20923)
Browse files Browse the repository at this point in the history
* opensearch keystore used for key/truststore passwords

* code cleanup

* Add changelog

* Added integration test for opensearch keystore cli

* add `-x` to add OS keystore secrets, bypassing the prompt

---------

Co-authored-by: Matthias Oesterheld <matthias.oesterheld@graylog.com>
  • Loading branch information
todvora and moesterheld authored Nov 15, 2024
1 parent 3a3cc4b commit ec1fd11
Show file tree
Hide file tree
Showing 11 changed files with 168 additions and 37 deletions.
5 changes: 5 additions & 0 deletions changelog/unreleased/issue-20926.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
type = "c"
message = "Data node: store keystore and truststore passwords in Opensearch keystore"

issues = ["20926"]
pulls = ["20923"]
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ public OpensearchDistributionProvider(final Configuration localConfiguration) {
this(Path.of(localConfiguration.getOpensearchDistributionRoot()), OpensearchArchitecture.fromOperatingSystem());
}

OpensearchDistributionProvider(final Path opensearchDistributionRoot, OpensearchArchitecture architecture) {
public OpensearchDistributionProvider(final Path opensearchDistributionRoot, OpensearchArchitecture architecture) {
this.distribution = Suppliers.memoize(() -> detectInDirectory(opensearchDistributionRoot, architecture));
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,14 +18,15 @@

import com.github.joschi.jadconfig.Parameter;
import com.github.joschi.jadconfig.converters.BooleanConverter;
import com.google.common.collect.ImmutableMap;
import org.apache.commons.lang3.StringUtils;
import org.graylog.datanode.configuration.variants.KeystoreContributor;
import org.graylog2.configuration.Documentation;

import java.util.Arrays;
import java.util.Map;
import java.util.Set;

public class S3RepositoryConfiguration {
public class S3RepositoryConfiguration implements KeystoreContributor {

@Documentation("S3 repository access key for searchable snapshots")
@Parameter(value = "s3_client_default_access_key")
Expand Down Expand Up @@ -100,4 +101,15 @@ private boolean noneBlank(String... properties) {
private boolean allBlank(String... properties) {
return Arrays.stream(properties).allMatch(StringUtils::isBlank);
}

@Override
public Map<String, String> getKeystoreItems() {
final ImmutableMap.Builder<String, String> config = ImmutableMap.builder();
if (isRepositoryEnabled()) {
config.put("s3.client.default.access_key", getS3ClientDefaultAccessKey());
config.put("s3.client.default.secret_key", getS3ClientDefaultSecretKey());

}
return config.build();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
/*
* Copyright (C) 2020 Graylog, Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the Server Side Public License, version 1,
* as published by MongoDB, Inc.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* Server Side Public License for more details.
*
* You should have received a copy of the Server Side Public License
* along with this program. If not, see
* <http://www.mongodb.com/licensing/server-side-public-license>.
*/
package org.graylog.datanode.configuration.variants;

import java.util.Map;

public interface KeystoreContributor {
/**
* @return collection of key-value pairs that should be added to the opensearch keystore (holding secrets)
*/
Map<String, String> getKeystoreItems();
}
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@
import java.util.Optional;
import java.util.stream.Collectors;

public class OpensearchSecurityConfiguration {
public class OpensearchSecurityConfiguration implements KeystoreContributor {

private static final Logger LOG = LoggerFactory.getLogger(OpensearchSecurityConfiguration.class);

Expand Down Expand Up @@ -106,23 +106,19 @@ public Map<String, String> getProperties() throws GeneralSecurityException, IOEx

config.put("plugins.security.ssl.transport.keystore_type", KEYSTORE_FORMAT);
config.put("plugins.security.ssl.transport.keystore_filepath", transportCertificate.location().getFileName().toString()); // todo: this should be computed as a relative path
config.put("plugins.security.ssl.transport.keystore_password", new String(transportCertificate.password()));
config.put("plugins.security.ssl.transport.keystore_alias", CertConstants.DATANODE_KEY_ALIAS);

config.put("plugins.security.ssl.transport.truststore_type", TRUSTSTORE_FORMAT);
config.put("plugins.security.ssl.transport.truststore_filepath", TRUSTSTORE_FILE.toString());
config.put("plugins.security.ssl.transport.truststore_password", new String(truststore.password()));

config.put("plugins.security.ssl.http.enabled", "true");

config.put("plugins.security.ssl.http.keystore_type", KEYSTORE_FORMAT);
config.put("plugins.security.ssl.http.keystore_filepath", httpCertificate.location().getFileName().toString()); // todo: this should be computed as a relative path
config.put("plugins.security.ssl.http.keystore_password", new String(httpCertificate.password()));
config.put("plugins.security.ssl.http.keystore_alias", CertConstants.DATANODE_KEY_ALIAS);

config.put("plugins.security.ssl.http.truststore_type", TRUSTSTORE_FORMAT);
config.put("plugins.security.ssl.http.truststore_filepath", TRUSTSTORE_FILE.toString());
config.put("plugins.security.ssl.http.truststore_password", new String(truststore.password()));

// enable client cert auth
config.put("plugins.security.ssl.http.clientauth_mode", "OPTIONAL");
Expand All @@ -133,6 +129,18 @@ public Map<String, String> getProperties() throws GeneralSecurityException, IOEx
return config.build();
}

@Override
public Map<String, String> getKeystoreItems() {
final ImmutableMap.Builder<String, String> config = ImmutableMap.builder();
if (securityEnabled()) {
config.put("plugins.security.ssl.transport.keystore_password_secure", new String(transportCertificate.password()));
config.put("plugins.security.ssl.transport.truststore_password_secure", new String(truststore.password()));
config.put("plugins.security.ssl.http.keystore_password_secure", new String(httpCertificate.password()));
config.put("plugins.security.ssl.http.truststore_password_secure", new String(truststore.password()));
}
return config.build();
}

private Map<String, Object> filterConfigurationMap(final Map<String, Object> map, final String... keys) {
Map<String, Object> result = map;
for (final String key : List.of(keys)) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@
import org.apache.commons.exec.DefaultExecuteResultHandler;
import org.apache.commons.exec.DefaultExecutor;
import org.apache.commons.exec.PumpStreamHandler;
import org.graylog.datanode.opensearch.configuration.OpensearchConfiguration;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
Expand All @@ -41,19 +40,17 @@ public abstract class AbstractOpensearchCli {
private final Path binPath;

/**
*
* @param configPath Opensearch CLI tools adapt configuration stored under OPENSEARCH_PATH_CONF env property.
* This is why wealways want to set this configPath for each CLI tool.
* @param bin location of the actual executable binary that this wrapper handles
* @param bin location of the actual executable binary that this wrapper handles
*/
private AbstractOpensearchCli(Path configPath, Path bin) {
this.configPath = configPath;
this.binPath = bin;
}

public AbstractOpensearchCli(OpensearchConfiguration config, String binName) {
this(config.datanodeDirectories().getOpensearchProcessConfigurationDir(),
checkExecutable(config.opensearchDistribution().getOpensearchBinDirPath().resolve(binName)));
protected AbstractOpensearchCli(Path configDir, Path binDir, String binName) {
this(configDir, checkExecutable(binDir.resolve(binName)));
}

private static Path checkExecutable(Path path) {
Expand All @@ -72,7 +69,7 @@ protected String runBatch(String... args) {
* to questions the tool is asking. It's scripting unfriendly. Our way around this is to
* provide an input stream of expected responses, each delimited by \n. There is no validation
* and no logic, just the expected order of responses.
* @param args arguments of the command, in opensearch-keystore create, the create is the first argument
* @param args arguments of the command, in opensearch-keystore create, the create is the first argument
* @return All the STDOUT and STDERR of the process merged into one String.
*/
protected String runWithStdin(List<String> answersToPrompts, String... args) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@

import org.graylog.datanode.opensearch.configuration.OpensearchConfiguration;

import java.nio.file.Path;

/**
* Collection of opensearch CLI tools. All of them need to have OPENSEARCH_PATH_CONF preconfigured, so they operate
* on the correct version of configuration.
Expand All @@ -27,7 +29,14 @@ public class OpensearchCli {
private final OpensearchKeystoreCli keystore;

public OpensearchCli(OpensearchConfiguration config) {
this.keystore = new OpensearchKeystoreCli(config);
this(
config.datanodeDirectories().getOpensearchProcessConfigurationDir(),
config.opensearchDistribution().getOpensearchBinDirPath()
);
}

public OpensearchCli(Path configDir, Path binDir) {
this.keystore = new OpensearchKeystoreCli(configDir, binDir);
}

public OpensearchKeystoreCli keystore() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -100,31 +100,24 @@ private void writeOpenSearchConfig(final OpensearchConfiguration config) {

public OpensearchCommandLineProcess(OpensearchConfiguration config, ProcessListener listener) {
fixJdkOnMac(config);
configureS3RepositoryPlugin(config);
configureOpensearchKeystoreSecrets(config);
final Path executable = config.opensearchDistribution().getOpensearchExecutable();
writeOpenSearchConfig(config);
resultHandler = new CommandLineProcessListener(listener);
commandLineProcess = new CommandLineProcess(executable, List.of(), resultHandler, config.getEnv());
}

private void configureS3RepositoryPlugin(OpensearchConfiguration config) {
if (config.s3RepositoryConfiguration().isRepositoryEnabled()) {
final OpensearchCli opensearchCli = new OpensearchCli(config);
configureS3Credentials(opensearchCli, config);
} else {
LOG.info("No S3 repository configuration provided, skipping plugin initialization");
}
}

private void configureS3Credentials(OpensearchCli opensearchCli, OpensearchConfiguration config) {
private void configureOpensearchKeystoreSecrets(OpensearchConfiguration config) {
final OpensearchCli opensearchCli = new OpensearchCli(config);
LOG.info("Creating opensearch keystore");
final String createdMessage = opensearchCli.keystore().create();
LOG.info(createdMessage);
LOG.info("Setting opensearch s3 repository keystore secrets");
opensearchCli.keystore().add("s3.client.default.access_key", config.s3RepositoryConfiguration().getS3ClientDefaultAccessKey());
opensearchCli.keystore().add("s3.client.default.secret_key", config.s3RepositoryConfiguration().getS3ClientDefaultSecretKey());
final Map<String, String> keystoreItems = config.getKeystoreItems();
keystoreItems.forEach((key, value) -> opensearchCli.keystore().add(key, value));
LOG.info("Added {} keystore items", keystoreItems.size());
}


private static Map<String, Object> getOpensearchConfigurationArguments(OpensearchConfiguration config) {
Map<String, Object> allArguments = new LinkedHashMap<>(config.asMap());

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,30 +16,38 @@
*/
package org.graylog.datanode.opensearch.cli;

import org.graylog.datanode.opensearch.configuration.OpensearchConfiguration;

import java.nio.file.Path;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

public class OpensearchKeystoreCli extends AbstractOpensearchCli {

OpensearchKeystoreCli(OpensearchConfiguration config) {
super(config, "opensearch-keystore");
public OpensearchKeystoreCli(Path configDir, Path binDir) {
super(configDir, binDir, "opensearch-keystore");
}

/**
* Create a new opensearch keystore. This command expects that there is no keystore. If there is a keystore,
* it will respond YES to override existing.
*
* @return STDOUT/STDERR of the execution as one String
*/
public String create() {
return runWithStdin(Collections.singletonList("Y"),"create");
return runWithStdin(Collections.singletonList("Y"), "create");
}

/**
* Add secrets to the store. The command is interactive, it will ask for the secret value (to avoid recording the value
* in the command line history). So we have to work around that and provide the value in STDIN.
*/
public void add(String key, String secretValue) {
runWithStdin(Collections.singletonList(secretValue), "add", key);
runWithStdin(List.of(secretValue), "add", "-x", key); // -x allows input from stdin, bypassing the prompt
}

public List<String> list() {
final String rawResponse = runWithStdin(Collections.emptyList(), "list");
final String[] items = rawResponse.split("\n");
return Arrays.asList(items);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
import org.graylog.datanode.OpensearchDistribution;
import org.graylog.datanode.configuration.DatanodeDirectories;
import org.graylog.datanode.configuration.S3RepositoryConfiguration;
import org.graylog.datanode.configuration.variants.KeystoreContributor;
import org.graylog.datanode.configuration.variants.OpensearchSecurityConfiguration;
import org.graylog.datanode.process.Environment;
import org.graylog.shaded.opensearch2.org.apache.http.HttpHost;
Expand All @@ -28,6 +29,8 @@
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;
import java.util.stream.Stream;

public record OpensearchConfiguration(
OpensearchDistribution opensearchDistribution,
Expand All @@ -45,7 +48,7 @@ public record OpensearchConfiguration(

String nodeSearchCacheSize,
Map<String, Object> additionalConfiguration
) {
) implements KeystoreContributor {
public Map<String, Object> asMap() {

Map<String, Object> config = new LinkedHashMap<>();
Expand Down Expand Up @@ -123,4 +126,12 @@ public HttpHost getClusterBaseUrl() {
public boolean securityConfigured() {
return opensearchSecurityConfiguration() != null;
}


@Override
public Map<String, String> getKeystoreItems() {
Stream<KeystoreContributor> keystoreContributorStream = Stream.of(opensearchSecurityConfiguration, s3RepositoryConfiguration);
return keystoreContributorStream.flatMap(config -> config.getKeystoreItems().entrySet().stream())
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
/*
* Copyright (C) 2020 Graylog, Inc.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the Server Side Public License, version 1,
* as published by MongoDB, Inc.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* Server Side Public License for more details.
*
* You should have received a copy of the Server Side Public License
* along with this program. If not, see
* <http://www.mongodb.com/licensing/server-side-public-license>.
*/
package org.graylog.datanode.opensearch.cli;

import jakarta.annotation.Nonnull;
import org.assertj.core.api.Assertions;
import org.graylog.datanode.OpensearchDistribution;
import org.graylog.datanode.configuration.OpensearchArchitecture;
import org.graylog.datanode.configuration.OpensearchDistributionProvider;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.io.TempDir;

import java.net.URISyntaxException;
import java.nio.file.Path;
import java.util.List;

class OpensearchKeystoreCommandLineIT {

@Test
void testKeystoreLifecycle(@TempDir Path tempDir) throws URISyntaxException {
final OpensearchCli cli = createCli(tempDir);
final String createdResponse = cli.keystore().create();

Assertions.assertThat(createdResponse).contains("Created opensearch keystore");

cli.keystore().add("s3.client.default.access_key", "foo");
cli.keystore().add("s3.client.default.secret_key", "bar");

final List<String> response = cli.keystore().list();
Assertions.assertThat(response)
.hasSize(3) // two keys and one internal seed
.contains("s3.client.default.access_key")
.contains("s3.client.default.secret_key");
}

private OpensearchCli createCli(Path tempDir) throws URISyntaxException {
final Path binDirPath = detectOpensearchBinDir();
return new OpensearchCli(tempDir, binDirPath);
}

@Nonnull
private Path detectOpensearchBinDir() throws URISyntaxException {
final Path opensearchDistRoot = Path.of(getClass().getResource("/").toURI()).getParent().resolve("opensearch");
final OpensearchDistributionProvider distributionProvider = new OpensearchDistributionProvider(opensearchDistRoot, OpensearchArchitecture.fromOperatingSystem());
final OpensearchDistribution opensearchDistribution = distributionProvider.get();
return opensearchDistribution.getOpensearchBinDirPath();
}
}

0 comments on commit ec1fd11

Please sign in to comment.