2023-02-16 17:41:37 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.services.opensearch;
|
|
|
|
|
|
|
|
settingsFormat = pkgs.formats.yaml {};
|
|
|
|
|
|
|
|
configDir = cfg.dataDir + "/config";
|
|
|
|
|
|
|
|
usingDefaultDataDir = cfg.dataDir == "/var/lib/opensearch";
|
|
|
|
usingDefaultUserAndGroup = cfg.user == "opensearch" && cfg.group == "opensearch";
|
|
|
|
|
|
|
|
opensearchYml = settingsFormat.generate "opensearch.yml" cfg.settings;
|
|
|
|
|
|
|
|
loggingConfigFilename = "log4j2.properties";
|
|
|
|
loggingConfigFile = pkgs.writeTextFile {
|
|
|
|
name = loggingConfigFilename;
|
|
|
|
text = cfg.logging;
|
|
|
|
};
|
|
|
|
in
|
|
|
|
{
|
|
|
|
|
|
|
|
options.services.opensearch = {
|
|
|
|
enable = mkEnableOption (lib.mdDoc "OpenSearch");
|
|
|
|
|
2024-01-02 11:29:13 +00:00
|
|
|
package = lib.mkPackageOption pkgs "OpenSearch" {
|
2023-02-16 17:41:37 +00:00
|
|
|
default = [ "opensearch" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
settings = lib.mkOption {
|
|
|
|
type = lib.types.submodule {
|
|
|
|
freeformType = settingsFormat.type;
|
|
|
|
|
|
|
|
options."network.host" = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
default = "127.0.0.1";
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
Which port this service should listen on.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
options."cluster.name" = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
default = "opensearch";
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
The name of the cluster.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
options."discovery.type" = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
default = "single-node";
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
The type of discovery to use.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
options."http.port" = lib.mkOption {
|
|
|
|
type = lib.types.port;
|
|
|
|
default = 9200;
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
The port to listen on for HTTP traffic.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
options."transport.port" = lib.mkOption {
|
|
|
|
type = lib.types.port;
|
|
|
|
default = 9300;
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
The port to listen on for transport traffic.
|
|
|
|
'';
|
|
|
|
};
|
2023-11-16 04:20:00 +00:00
|
|
|
|
|
|
|
options."plugins.security.disabled" = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
|
|
|
default = true;
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
Whether to enable the security plugin,
|
|
|
|
`plugins.security.ssl.transport.keystore_filepath` or
|
|
|
|
`plugins.security.ssl.transport.server.pemcert_filepath` and
|
|
|
|
`plugins.security.ssl.transport.client.pemcert_filepath`
|
|
|
|
must be set for this plugin to be enabled.
|
|
|
|
'';
|
|
|
|
};
|
2023-02-16 17:41:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
default = {};
|
|
|
|
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
OpenSearch configuration.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
logging = lib.mkOption {
|
|
|
|
description = lib.mdDoc "opensearch logging configuration.";
|
|
|
|
|
|
|
|
default = ''
|
|
|
|
logger.action.name = org.opensearch.action
|
|
|
|
logger.action.level = info
|
|
|
|
|
|
|
|
appender.console.type = Console
|
|
|
|
appender.console.name = console
|
|
|
|
appender.console.layout.type = PatternLayout
|
|
|
|
appender.console.layout.pattern = [%d{ISO8601}][%-5p][%-25c{1.}] %marker%m%n
|
|
|
|
|
|
|
|
rootLogger.level = info
|
|
|
|
rootLogger.appenderRef.console.ref = console
|
|
|
|
'';
|
|
|
|
type = types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
dataDir = lib.mkOption {
|
|
|
|
type = lib.types.path;
|
|
|
|
default = "/var/lib/opensearch";
|
|
|
|
apply = converge (removeSuffix "/");
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
Data directory for OpenSearch. If you change this, you need to
|
|
|
|
manually create the directory. You also need to create the
|
|
|
|
`opensearch` user and group, or change
|
|
|
|
[](#opt-services.opensearch.user) and
|
|
|
|
[](#opt-services.opensearch.group) to existing ones with
|
|
|
|
access to the directory.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
user = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
default = "opensearch";
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
The user OpenSearch runs as. Should be left at default unless
|
|
|
|
you have very specific needs.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
group = lib.mkOption {
|
|
|
|
type = lib.types.str;
|
|
|
|
default = "opensearch";
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
The group OpenSearch runs as. Should be left at default unless
|
|
|
|
you have very specific needs.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
extraCmdLineOptions = lib.mkOption {
|
|
|
|
description = lib.mdDoc "Extra command line options for the OpenSearch launcher.";
|
|
|
|
default = [ ];
|
|
|
|
type = lib.types.listOf lib.types.str;
|
|
|
|
};
|
|
|
|
|
|
|
|
extraJavaOptions = lib.mkOption {
|
|
|
|
description = lib.mdDoc "Extra command line options for Java.";
|
|
|
|
default = [ ];
|
|
|
|
type = lib.types.listOf lib.types.str;
|
|
|
|
example = [ "-Djava.net.preferIPv4Stack=true" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
restartIfChanged = lib.mkOption {
|
|
|
|
type = lib.types.bool;
|
|
|
|
description = lib.mdDoc ''
|
|
|
|
Automatically restart the service on config change.
|
|
|
|
This can be set to false to defer restarts on a server or cluster.
|
|
|
|
Please consider the security implications of inadvertently running an older version,
|
|
|
|
and the possibility of unexpected behavior caused by inconsistent versions across a cluster when disabling this option.
|
|
|
|
'';
|
|
|
|
default = true;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkIf cfg.enable {
|
|
|
|
systemd.services.opensearch = {
|
|
|
|
description = "OpenSearch Daemon";
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
after = [ "network.target" ];
|
|
|
|
path = [ pkgs.inetutils ];
|
|
|
|
inherit (cfg) restartIfChanged;
|
|
|
|
environment = {
|
|
|
|
OPENSEARCH_HOME = cfg.dataDir;
|
|
|
|
OPENSEARCH_JAVA_OPTS = toString cfg.extraJavaOptions;
|
|
|
|
OPENSEARCH_PATH_CONF = configDir;
|
|
|
|
};
|
|
|
|
serviceConfig = {
|
|
|
|
ExecStartPre =
|
|
|
|
let
|
|
|
|
startPreFullPrivileges = ''
|
|
|
|
set -o errexit -o pipefail -o nounset -o errtrace
|
|
|
|
shopt -s inherit_errexit
|
|
|
|
'' + (optionalString (!config.boot.isContainer) ''
|
|
|
|
# Only set vm.max_map_count if lower than ES required minimum
|
|
|
|
# This avoids conflict if configured via boot.kernel.sysctl
|
|
|
|
if [ $(${pkgs.procps}/bin/sysctl -n vm.max_map_count) -lt 262144 ]; then
|
|
|
|
${pkgs.procps}/bin/sysctl -w vm.max_map_count=262144
|
|
|
|
fi
|
|
|
|
'');
|
|
|
|
startPreUnprivileged = ''
|
|
|
|
set -o errexit -o pipefail -o nounset -o errtrace
|
|
|
|
shopt -s inherit_errexit
|
|
|
|
|
|
|
|
# Install plugins
|
2023-11-16 04:20:00 +00:00
|
|
|
|
|
|
|
# remove plugins directory if it is empty.
|
2024-01-02 11:29:13 +00:00
|
|
|
if [[ -d ${cfg.dataDir}/plugins && -z "$(ls -A ${cfg.dataDir}/plugins)" ]]; then
|
2023-11-16 04:20:00 +00:00
|
|
|
rm -r "${cfg.dataDir}/plugins"
|
|
|
|
fi
|
|
|
|
|
|
|
|
ln -sfT "${cfg.package}/plugins" "${cfg.dataDir}/plugins"
|
2023-02-16 17:41:37 +00:00
|
|
|
ln -sfT ${cfg.package}/lib ${cfg.dataDir}/lib
|
|
|
|
ln -sfT ${cfg.package}/modules ${cfg.dataDir}/modules
|
|
|
|
|
|
|
|
# opensearch needs to create the opensearch.keystore in the config directory
|
|
|
|
# so this directory needs to be writable.
|
|
|
|
mkdir -p ${configDir}
|
|
|
|
chmod 0700 ${configDir}
|
|
|
|
|
|
|
|
# Note that we copy config files from the nix store instead of symbolically linking them
|
|
|
|
# because otherwise X-Pack Security will raise the following exception:
|
|
|
|
# java.security.AccessControlException:
|
|
|
|
# access denied ("java.io.FilePermission" "/var/lib/opensearch/config/opensearch.yml" "read")
|
|
|
|
|
2023-03-04 12:14:45 +00:00
|
|
|
rm -f ${configDir}/opensearch.yml
|
2023-02-16 17:41:37 +00:00
|
|
|
cp ${opensearchYml} ${configDir}/opensearch.yml
|
|
|
|
|
|
|
|
# Make sure the logging configuration for old OpenSearch versions is removed:
|
|
|
|
rm -f "${configDir}/logging.yml"
|
2023-03-04 12:14:45 +00:00
|
|
|
rm -f ${configDir}/${loggingConfigFilename}
|
2023-02-16 17:41:37 +00:00
|
|
|
cp ${loggingConfigFile} ${configDir}/${loggingConfigFilename}
|
|
|
|
mkdir -p ${configDir}/scripts
|
2023-03-04 12:14:45 +00:00
|
|
|
|
|
|
|
rm -f ${configDir}/jvm.options
|
2023-02-16 17:41:37 +00:00
|
|
|
cp ${cfg.package}/config/jvm.options ${configDir}/jvm.options
|
|
|
|
|
|
|
|
# redirect jvm logs to the data directory
|
|
|
|
mkdir -p ${cfg.dataDir}/logs
|
|
|
|
chmod 0700 ${cfg.dataDir}/logs
|
|
|
|
sed -e '#logs/gc.log#${cfg.dataDir}/logs/gc.log#' -i ${configDir}/jvm.options
|
|
|
|
'';
|
|
|
|
in [
|
|
|
|
"+${pkgs.writeShellScript "opensearch-start-pre-full-privileges" startPreFullPrivileges}"
|
|
|
|
"${pkgs.writeShellScript "opensearch-start-pre-unprivileged" startPreUnprivileged}"
|
|
|
|
];
|
|
|
|
ExecStartPost = pkgs.writeShellScript "opensearch-start-post" ''
|
|
|
|
set -o errexit -o pipefail -o nounset -o errtrace
|
|
|
|
shopt -s inherit_errexit
|
|
|
|
|
|
|
|
# Make sure opensearch is up and running before dependents
|
|
|
|
# are started
|
|
|
|
while ! ${pkgs.curl}/bin/curl -sS -f http://${cfg.settings."network.host"}:${toString cfg.settings."http.port"} 2>/dev/null; do
|
|
|
|
sleep 1
|
|
|
|
done
|
|
|
|
'';
|
|
|
|
ExecStart = "${cfg.package}/bin/opensearch ${toString cfg.extraCmdLineOptions}";
|
|
|
|
User = cfg.user;
|
|
|
|
Group = cfg.group;
|
|
|
|
LimitNOFILE = "1024000";
|
|
|
|
Restart = "always";
|
|
|
|
TimeoutStartSec = "infinity";
|
|
|
|
DynamicUser = usingDefaultUserAndGroup && usingDefaultDataDir;
|
|
|
|
} // (optionalAttrs (usingDefaultDataDir) {
|
|
|
|
StateDirectory = "opensearch";
|
|
|
|
StateDirectoryMode = "0700";
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
environment.systemPackages = [ cfg.package ];
|
|
|
|
};
|
|
|
|
}
|