2022-01-08 10:32:32 +00:00
|
|
|
{ config, lib, pkgs, ... }:
|
2021-10-20 21:31:12 +01:00
|
|
|
with lib;
|
2018-05-21 02:09:31 +01:00
|
|
|
let
|
|
|
|
cfg = config.services.hadoop;
|
2022-01-08 10:32:32 +00:00
|
|
|
|
|
|
|
# Config files for hadoop services
|
2021-10-20 21:31:12 +01:00
|
|
|
hadoopConf = "${import ./conf.nix { inherit cfg pkgs lib; }}/";
|
2022-01-08 10:32:32 +00:00
|
|
|
|
|
|
|
# Generator for HDFS service options
|
2022-03-02 07:20:01 +00:00
|
|
|
hadoopServiceOption = { serviceName, firewallOption ? true, extraOpts ? null }: {
|
2022-08-28 20:18:44 +01:00
|
|
|
enable = mkEnableOption (lib.mdDoc serviceName);
|
2022-01-08 10:32:32 +00:00
|
|
|
restartIfChanged = mkOption {
|
|
|
|
type = types.bool;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2022-01-08 10:32:32 +00:00
|
|
|
Automatically restart the service on config change.
|
|
|
|
This can be set to false to defer restarts on clusters running critical applications.
|
|
|
|
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 = false;
|
|
|
|
};
|
2022-03-02 07:20:01 +00:00
|
|
|
extraFlags = mkOption{
|
|
|
|
type = with types; listOf str;
|
|
|
|
default = [];
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Extra command line flags to pass to ${serviceName}";
|
2022-03-02 07:20:01 +00:00
|
|
|
example = [
|
|
|
|
"-Dcom.sun.management.jmxremote"
|
|
|
|
"-Dcom.sun.management.jmxremote.port=8010"
|
|
|
|
];
|
|
|
|
};
|
|
|
|
extraEnv = mkOption{
|
|
|
|
type = with types; attrsOf str;
|
|
|
|
default = {};
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Extra environment variables for ${serviceName}";
|
2022-03-02 07:20:01 +00:00
|
|
|
};
|
2022-01-08 10:32:32 +00:00
|
|
|
} // (optionalAttrs firewallOption {
|
|
|
|
openFirewall = mkOption {
|
|
|
|
type = types.bool;
|
2022-01-08 10:49:15 +00:00
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Open firewall ports for ${serviceName}.";
|
2022-01-08 10:32:32 +00:00
|
|
|
};
|
2022-03-02 07:20:01 +00:00
|
|
|
}) // (optionalAttrs (extraOpts != null) extraOpts);
|
2022-01-08 10:32:32 +00:00
|
|
|
|
|
|
|
# Generator for HDFS service configs
|
|
|
|
hadoopServiceConfig =
|
|
|
|
{ name
|
|
|
|
, serviceOptions ? cfg.hdfs."${toLower name}"
|
|
|
|
, description ? "Hadoop HDFS ${name}"
|
|
|
|
, User ? "hdfs"
|
|
|
|
, allowedTCPPorts ? [ ]
|
|
|
|
, preStart ? ""
|
|
|
|
, environment ? { }
|
2022-03-02 07:20:01 +00:00
|
|
|
, extraConfig ? { }
|
2022-01-08 10:32:32 +00:00
|
|
|
}: (
|
|
|
|
|
2022-03-02 07:20:01 +00:00
|
|
|
mkIf serviceOptions.enable ( mkMerge [{
|
2022-01-08 10:32:32 +00:00
|
|
|
systemd.services."hdfs-${toLower name}" = {
|
2022-03-02 07:20:01 +00:00
|
|
|
inherit description preStart;
|
|
|
|
environment = environment // serviceOptions.extraEnv;
|
2022-01-08 10:32:32 +00:00
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
inherit (serviceOptions) restartIfChanged;
|
|
|
|
serviceConfig = {
|
|
|
|
inherit User;
|
|
|
|
SyslogIdentifier = "hdfs-${toLower name}";
|
2022-03-02 07:20:01 +00:00
|
|
|
ExecStart = "${cfg.package}/bin/hdfs --config ${hadoopConf} ${toLower name} ${escapeShellArgs serviceOptions.extraFlags}";
|
2022-01-08 10:32:32 +00:00
|
|
|
Restart = "always";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2022-02-27 06:22:18 +00:00
|
|
|
services.hadoop.gatewayRole.enable = true;
|
|
|
|
|
2022-01-08 10:32:32 +00:00
|
|
|
networking.firewall.allowedTCPPorts = mkIf
|
|
|
|
((builtins.hasAttr "openFirewall" serviceOptions) && serviceOptions.openFirewall)
|
|
|
|
allowedTCPPorts;
|
2022-03-02 07:20:01 +00:00
|
|
|
} extraConfig])
|
2022-01-08 10:32:32 +00:00
|
|
|
);
|
|
|
|
|
2018-05-21 02:09:31 +01:00
|
|
|
in
|
|
|
|
{
|
|
|
|
options.services.hadoop.hdfs = {
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2022-01-08 08:32:32 +00:00
|
|
|
namenode = hadoopServiceOption { serviceName = "HDFS NameNode"; } // {
|
2021-11-02 06:29:58 +00:00
|
|
|
formatOnInit = mkOption {
|
|
|
|
type = types.bool;
|
|
|
|
default = false;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2022-01-08 10:32:32 +00:00
|
|
|
Format HDFS namenode on first start. This is useful for quickly spinning up
|
|
|
|
ephemeral HDFS clusters with a single namenode.
|
|
|
|
For HA clusters, initialization involves multiple steps across multiple nodes.
|
|
|
|
Follow this guide to initialize an HA cluster manually:
|
2022-07-28 22:19:15 +01:00
|
|
|
<https://hadoop.apache.org/docs/stable/hadoop-project-dist/hadoop-hdfs/HDFSHighAvailabilityWithQJM.html>
|
2021-10-20 21:31:12 +01:00
|
|
|
'';
|
|
|
|
};
|
2018-05-21 02:09:31 +01:00
|
|
|
};
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2022-03-02 07:20:01 +00:00
|
|
|
datanode = hadoopServiceOption { serviceName = "HDFS DataNode"; } // {
|
|
|
|
dataDirs = mkOption {
|
|
|
|
default = null;
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Tier and path definitions for datanode storage.";
|
2022-03-02 07:20:01 +00:00
|
|
|
type = with types; nullOr (listOf (submodule {
|
|
|
|
options = {
|
|
|
|
type = mkOption {
|
|
|
|
type = enum [ "SSD" "DISK" "ARCHIVE" "RAM_DISK" ];
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc ''
|
2022-03-02 07:20:01 +00:00
|
|
|
Storage types ([SSD]/[DISK]/[ARCHIVE]/[RAM_DISK]) for HDFS storage policies.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
path = mkOption {
|
|
|
|
type = path;
|
|
|
|
example = [ "/var/lib/hadoop/hdfs/dn" ];
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "Determines where on the local filesystem a data node should store its blocks.";
|
2022-03-02 07:20:01 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
};
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2022-01-08 08:32:32 +00:00
|
|
|
journalnode = hadoopServiceOption { serviceName = "HDFS JournalNode"; };
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2022-01-08 08:32:32 +00:00
|
|
|
zkfc = hadoopServiceOption {
|
|
|
|
serviceName = "HDFS ZooKeeper failover controller";
|
|
|
|
firewallOption = false;
|
2021-11-02 06:29:58 +00:00
|
|
|
};
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2022-01-08 08:32:32 +00:00
|
|
|
httpfs = hadoopServiceOption { serviceName = "HDFS JournalNode"; } // {
|
2021-11-02 13:04:04 +00:00
|
|
|
tempPath = mkOption {
|
|
|
|
type = types.path;
|
|
|
|
default = "/tmp/hadoop/httpfs";
|
2022-07-28 22:19:15 +01:00
|
|
|
description = lib.mdDoc "HTTPFS_TEMP path used by HTTPFS";
|
2021-11-04 05:56:32 +00:00
|
|
|
};
|
2021-11-02 13:04:04 +00:00
|
|
|
};
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2018-05-21 02:09:31 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
config = mkMerge [
|
2022-01-08 10:32:32 +00:00
|
|
|
(hadoopServiceConfig {
|
|
|
|
name = "NameNode";
|
|
|
|
allowedTCPPorts = [
|
2021-10-20 21:31:12 +01:00
|
|
|
9870 # namenode.http-address
|
|
|
|
8020 # namenode.rpc-address
|
2022-02-27 05:11:05 +00:00
|
|
|
8022 # namenode.servicerpc-address
|
|
|
|
8019 # dfs.ha.zkfc.port
|
2022-01-08 10:32:32 +00:00
|
|
|
];
|
|
|
|
preStart = (mkIf cfg.hdfs.namenode.formatOnInit
|
|
|
|
"${cfg.package}/bin/hdfs --config ${hadoopConf} namenode -format -nonInteractive || true"
|
|
|
|
);
|
2018-05-21 02:09:31 +01:00
|
|
|
})
|
2021-10-20 21:31:12 +01:00
|
|
|
|
2022-01-08 10:32:32 +00:00
|
|
|
(hadoopServiceConfig {
|
|
|
|
name = "DataNode";
|
2022-02-27 05:11:05 +00:00
|
|
|
# port numbers for datanode changed between hadoop 2 and 3
|
|
|
|
allowedTCPPorts = if versionAtLeast cfg.package.version "3" then [
|
2021-10-20 21:31:12 +01:00
|
|
|
9864 # datanode.http.address
|
|
|
|
9866 # datanode.address
|
|
|
|
9867 # datanode.ipc.address
|
2022-02-27 05:11:05 +00:00
|
|
|
] else [
|
|
|
|
50075 # datanode.http.address
|
|
|
|
50010 # datanode.address
|
|
|
|
50020 # datanode.ipc.address
|
2022-01-08 10:32:32 +00:00
|
|
|
];
|
2022-05-13 06:25:16 +01:00
|
|
|
extraConfig.services.hadoop.hdfsSiteInternal."dfs.datanode.data.dir" = mkIf (cfg.hdfs.datanode.dataDirs!= null)
|
|
|
|
(concatMapStringsSep "," (x: "["+x.type+"]file://"+x.path) cfg.hdfs.datanode.dataDirs);
|
2018-05-21 02:09:31 +01:00
|
|
|
})
|
2021-11-02 06:29:58 +00:00
|
|
|
|
2022-01-08 10:32:32 +00:00
|
|
|
(hadoopServiceConfig {
|
|
|
|
name = "JournalNode";
|
|
|
|
allowedTCPPorts = [
|
2021-11-02 06:29:58 +00:00
|
|
|
8480 # dfs.journalnode.http-address
|
|
|
|
8485 # dfs.journalnode.rpc-address
|
2022-01-08 10:32:32 +00:00
|
|
|
];
|
2021-11-02 06:29:58 +00:00
|
|
|
})
|
2021-11-02 13:04:04 +00:00
|
|
|
|
2022-01-08 10:32:32 +00:00
|
|
|
(hadoopServiceConfig {
|
|
|
|
name = "zkfc";
|
|
|
|
description = "Hadoop HDFS ZooKeeper failover controller";
|
|
|
|
})
|
2021-11-02 13:04:04 +00:00
|
|
|
|
2022-01-08 10:32:32 +00:00
|
|
|
(hadoopServiceConfig {
|
|
|
|
name = "HTTPFS";
|
|
|
|
environment.HTTPFS_TEMP = cfg.hdfs.httpfs.tempPath;
|
|
|
|
preStart = "mkdir -p $HTTPFS_TEMP";
|
|
|
|
User = "httpfs";
|
|
|
|
allowedTCPPorts = [
|
2021-11-02 13:04:04 +00:00
|
|
|
14000 # httpfs.http.port
|
2022-01-08 10:32:32 +00:00
|
|
|
];
|
2021-11-02 13:04:04 +00:00
|
|
|
})
|
2022-01-08 10:32:32 +00:00
|
|
|
|
2022-02-27 06:22:18 +00:00
|
|
|
(mkIf cfg.gatewayRole.enable {
|
|
|
|
users.users.hdfs = {
|
|
|
|
description = "Hadoop HDFS user";
|
|
|
|
group = "hadoop";
|
|
|
|
uid = config.ids.uids.hdfs;
|
|
|
|
};
|
|
|
|
})
|
2021-11-02 16:16:48 +00:00
|
|
|
(mkIf cfg.hdfs.httpfs.enable {
|
2021-11-02 13:04:04 +00:00
|
|
|
users.users.httpfs = {
|
|
|
|
description = "Hadoop HTTPFS user";
|
|
|
|
group = "hadoop";
|
|
|
|
isSystemUser = true;
|
|
|
|
};
|
|
|
|
})
|
2022-02-27 06:22:18 +00:00
|
|
|
|
2018-05-21 02:09:31 +01:00
|
|
|
];
|
|
|
|
}
|