Compare commits

..

1 commit

Author SHA1 Message Date
3881e8cecb
chore(catvayor): rotating ssh keys 2025-01-23 12:41:30 +01:00
10 changed files with 176 additions and 221 deletions

View file

@ -1,13 +1,13 @@
{
pkgs ? (import <nixpkgs> { }),
}:
let let
sources = import ./npins;
pkgs = import sources.kat-pkgs { };
lib = pkgs.lib; lib = pkgs.lib;
hive_mod = hive_mod =
{ {
lib, lib,
config, config,
name, name,
xml,
... ...
}: }:
let let
@ -32,17 +32,21 @@ let
}; };
}; };
config.deployment = rec { config.deployment = rec {
rpc = xml.generate "config-${name}_rpc.xml" { rpc = pkgs.writeText "config-${name}_rpc.xml" ''
rpc = [ <rpc>
{ <edit-config>
edit-config = { <config>
config.configuration = config.netconf.xml; ${config.netconf.xmls.configuration}
target.candidate = ""; </config>
}; <target>
} <candidate/>
{ commit = { }; } </target>
]; </edit-config>
}; </rpc>
<rpc>
<commit/>
</rpc>
'';
cmd = pkgs.writeShellApplication { cmd = pkgs.writeShellApplication {
name = "deploy-${name}.sh"; name = "deploy-${name}.sh";
runtimeInputs = with pkgs; [ openssh ]; runtimeInputs = with pkgs; [ openssh ];
@ -56,7 +60,6 @@ let
cfg = pkgs.lib.evalModules { cfg = pkgs.lib.evalModules {
specialArgs = { specialArgs = {
inherit name; inherit name;
xml = pkgs.formats.xml { };
}; };
modules = [ modules = [
./junos ./junos

View file

@ -1,7 +1,6 @@
{ {
lib, lib,
config, config,
xml,
... ...
}: }:
let let
@ -11,6 +10,7 @@ let
; ;
inherit (lib.types) inherit (lib.types)
bool bool
str
attrsOf attrsOf
submodule submodule
; ;
@ -33,8 +33,9 @@ in
./system.nix ./system.nix
]; ];
options = { options = {
netconf.xml = mkOption { netconf.xmls.configuration = mkOption {
type = xml.type; type = str;
readOnly = true;
description = '' description = ''
The full configuration to send to a JunOS. The full configuration to send to a JunOS.
''; '';
@ -64,4 +65,13 @@ in
mkIntf = _: _: { }; mkIntf = _: _: { };
in in
mapAttrs mkIntf config.netconf.mandatoryInterfaces; mapAttrs mkIntf config.netconf.mandatoryInterfaces;
config.netconf.xmls.configuration = ''
<configuration>
${config.netconf.xmls.system}
${config.netconf.xmls.interfaces}
${config.netconf.xmls.protocols}
${config.netconf.xmls.vlans}
${config.netconf.xmls.poe}
</configuration>
'';
} }

View file

@ -1,17 +1,10 @@
{ { lib, config, ... }:
lib,
config,
xml,
...
}:
let let
inherit (lib) inherit (lib)
mkEnableOption mkEnableOption
mkOption mkOption
optionalString optionalString
mapAttrsToList mapAttrsToList
mkMerge
mkIf
; ;
inherit (lib.types) inherit (lib.types)
enum enum
@ -26,6 +19,7 @@ let
interface = interface =
{ name, config, ... }: { name, config, ... }:
let let
intf-name = name;
unit = unit =
{ name, config, ... }: { name, config, ... }:
{ {
@ -80,36 +74,46 @@ let
}; };
}; };
xml = mkOption { xml = mkOption {
type = xml.type; type = str;
visible = false; visible = false;
readOnly = true; readOnly = true;
}; };
}; };
config.xml = config.xml =
let let
eth = mkIf config.family.ethernet-switching.enable { members = map (
family.ethernet-switching = { vlan: "<members>${builtins.toString vlan}</members>"
interface-mode = config.family.ethernet-switching.interface-mode; ) config.family.ethernet-switching.vlans;
vlan.members = map toString config.family.ethernet-switching.vlans; eth = optionalString config.family.ethernet-switching.enable ''
storm-control.profile-name = "default"; <ethernet-switching>
}; <interface-mode>${config.family.ethernet-switching.interface-mode}</interface-mode>
}; <vlan>${builtins.concatStringsSep "" members}</vlan>
<storm-control><profile-name>default</profile-name></storm-control>
</ethernet-switching>
'';
addr4 = map (name: { inherit name; }) config.family.inet.addresses; addr4 = map (addr: "<name>${addr}</name>") config.family.inet.addresses;
inet = mkIf config.family.inet.enable { family.inet.address = addr4; }; inet = optionalString config.family.inet.enable ''
<inet>
<address>${builtins.concatStringsSep "" addr4}</address>
</inet>
'';
addr6 = map (name: { inherit name; }) config.family.inet6.addresses; addr6 = map (addr: "<name>${addr}</name>") config.family.inet6.addresses;
inet6 = mkIf config.family.inet6.enable { family.inet6.address = addr6; }; inet6 = optionalString config.family.inet6.enable ''
<inet6>
<address>${builtins.concatStringsSep "" addr6}</address>
</inet6>
'';
in in
mkMerge [ ''
{ <unit>
inherit name; <name>${name}</name>
} ${optionalString (!config.enable) "<disable/>"}
(mkIf (!config.enable) { disable = { }; }) <family>
eth ${eth}${inet}${inet6}
inet </family>
inet6 </unit>'';
];
}; };
in in
{ {
@ -123,21 +127,22 @@ let
''; '';
}; };
xml = mkOption { xml = mkOption {
type = xml.type; type = str;
visible = false; visible = false;
readOnly = true; readOnly = true;
}; };
}; };
config.xml = config.xml =
let let
unit = mapAttrsToList (_: unit: unit.xml) config.unit; units = mapAttrsToList (_: unit: unit.xml) config.unit;
in in
mkMerge [ ''
{ <interface>
inherit name unit; <name>${name}</name>
} ${optionalString (!config.enable) "<disable/>"}
(mkIf (!config.enable) { disable = { }; }) ${builtins.concatStringsSep "" units}
]; </interface>
'';
}; };
in in
{ {
@ -148,9 +153,15 @@ in
The interfaces configuration. The interfaces configuration.
''; '';
}; };
netconf.xmls.interfaces = mkOption {
type = str;
visible = false;
readOnly = true;
}; };
config.netconf.xml.interfaces = {
"@operation" = "replace";
interface = mapAttrsToList (_: intf: intf.xml) config.interfaces;
}; };
config.netconf.xmls.interfaces = ''
<interfaces operation="replace">
${builtins.concatStringsSep "" (mapAttrsToList (_: intf: intf.xml) config.interfaces)}
</interfaces>
'';
} }

View file

@ -1,13 +1,13 @@
{ lib, config, xml, ... }: { lib, config, ... }:
let let
inherit (lib) inherit (lib)
mkOption mkOption
mkEnableOption mkEnableOption
mapAttrsToList mapAttrsToList
mkIf optionalString
mkMerge
; ;
inherit (lib.types) inherit (lib.types)
str
attrsOf attrsOf
submodule submodule
; ;
@ -18,15 +18,14 @@ let
options = { options = {
enable = mkEnableOption "the PoE for this interface"; enable = mkEnableOption "the PoE for this interface";
xml = mkOption { xml = mkOption {
type = xml.type; type = str;
visible = false; visible = false;
readOnly = true; readOnly = true;
}; };
}; };
config.xml = mkMerge [ config.xml = ''
{ inherit name; } <interface><name>${name}</name>${optionalString (!config.enable) "<disable/>"}</interface>
(mkIf (!config.enable) { disable = { }; }) '';
];
}; };
in in
{ {
@ -38,9 +37,15 @@ in
PoE configuration of interfaces. PoE configuration of interfaces.
''; '';
}; };
netconf.xmls.poe = mkOption {
type = str;
visible = false;
readOnly = true;
}; };
config.netconf.xml.poe = {
"@operation" = "replace";
interface = mapAttrsToList (_: intf: intf.xml) config.poe.interfaces;
}; };
config.netconf.xmls.poe = ''
<poe operation="replace">
${builtins.concatStringsSep "" (mapAttrsToList (_: intf: intf.xml) config.poe.interfaces)}
</poe>
'';
} }

View file

@ -2,6 +2,7 @@
let let
inherit (lib) inherit (lib)
mkOption mkOption
concatStringsSep
; ;
inherit (lib.types) inherit (lib.types)
listOf listOf
@ -16,9 +17,21 @@ in
List of interfaces on which Rapid Spanning Tree Protocol should be enabled. List of interfaces on which Rapid Spanning Tree Protocol should be enabled.
''; '';
}; };
netconf.xmls.protocols = mkOption {
type = str;
visible = false;
readOnly = true;
}; };
config.netconf.xml.protocols.rstp = {
"@operation" = "replace";
interface = map (name: { inherit name; }) config.protocols.rstp;
}; };
config.netconf.xmls.protocols =
let
rstps = map (intf: "<interface><name>${intf}</name></interface>") config.protocols.rstp;
in
''
<protocols>
<rstp operation="replace">
${concatStringsSep "" rstps}
</rstp>
</protocols>
'';
} }

View file

@ -6,6 +6,7 @@ let
length length
hasPrefix hasPrefix
filter filter
concatStrings
concatStringsSep concatStringsSep
; ;
inherit (lib.types) inherit (lib.types)
@ -49,34 +50,39 @@ in
}; };
}; };
}; };
netconf.xmls.system = mkOption {
type = str;
visible = false;
readOnly = true;
}; };
config.netconf.xml.system = };
config.netconf.xmls.system =
let let
ssh-keys1 = map (splitString " ") config.system.root-authentication.ssh-keys; ssh-keys1 = map (splitString " ") config.system.root-authentication.ssh-keys;
ssh-keys2 = map (key: if length key < 3 then key ++ [ "foo@bar" ] else key) ssh-keys1; ssh-keys2 = map (key: if length key < 3 then key ++ [ "foo@bar" ] else key) ssh-keys1;
ssh-keys = map (concatStringsSep " ") ssh-keys2; ssh-keys = map (concatStringsSep " ") ssh-keys2;
ssh-edsca = map (name: { inherit name; }) (filter (hasPrefix "ssh-edsca ") ssh-keys); edsca = map (key: "<ssh-edsca><name>${key}</name></ssh-edsca>") (
ssh-rsa = map (name: { inherit name; }) (filter (hasPrefix "ssh-rsa ") ssh-keys); filter (hasPrefix "ssh-edsca ") ssh-keys
ssh-ed25519 = map (name: { inherit name; }) (filter (hasPrefix "ssh-ed25519 ") ssh-keys); );
rsa = map (key: "<ssh-rsa><name>${key}</name></ssh-rsa>") (filter (hasPrefix "ssh-rsa ") ssh-keys);
ed25519 = map (key: "<ssh-ed25519><name>${key}</name></ssh-ed25519>") (
filter (hasPrefix "ssh-ed25519 ") ssh-keys
);
in in
{ ''
host-name = { <system>
"@operation" = "replace"; <host-name operation="replace">${config.system.host-name}</host-name>
"#text" = config.system.host-name; <root-authentication operation="replace">
}; <encrypted-password>${config.system.root-authentication.hashedPasswd}</encrypted-password>
root-authentication = { ${concatStrings (edsca ++ rsa ++ ed25519)}
"@operation" = "replace"; </root-authentication>
encrypted-password = config.system.root-authentication.hashedPasswd; <services operation="replace">
inherit ssh-edsca ssh-rsa ssh-ed25519; <ssh><root-login>${config.system.services.ssh.root-login}</root-login></ssh>
}; <netconf>
services = { <ssh><port>${toString config.system.services.netconf.port}</port></ssh>
"@operation" = "replace"; <rfc-compliant/><yang-compliant/>
ssh.root-login = config.system.services.ssh.root-login; </netconf>
netconf = { </services>
ssh.port = config.system.services.netconf.port; </system>
rfc-compliant = {}; '';
yang-compliant = {};
};
};
};
} }

View file

@ -1,11 +1,11 @@
{ lib, config, xml, ... }: { lib, config, ... }:
let let
inherit (lib) inherit (lib)
mkOption mkOption
optionalString
assertMsg assertMsg
concatStringsSep
mapAttrsToList mapAttrsToList
mkMerge
mkIf
; ;
inherit (lib.types) inherit (lib.types)
nullOr nullOr
@ -80,28 +80,30 @@ let
''; '';
}; };
xml = mkOption { xml = mkOption {
type = xml.type; type = str;
readOnly = true; readOnly = true;
visible = false; visible = false;
}; };
}; };
config.xml = config.xml =
let let
id = mkIf (!isNull config.id) { vlan-id = toString config.id; }; id = optionalString (!isNull config.id) "<vlan-id>${toString config.id}</vlan-id>";
id-list = mkIf (config.id-list != [ ]) { id-list = concatStringsSep "" (
vlan-id-list = map toString config.id-list; map (vlan: "<vlan-id-list>${toString vlan}</vlan-id-list>") config.id-list
}; );
l3-intf = mkIf (!isNull config.l3-interface) { inherit (config) l3-interface; }; l3-intf = optionalString (
!isNull config.l3-interface
) "<l3-interface>${config.l3-interface}</l3-interface>";
in in
assert assertMsg ( assert assertMsg (
config.id == null || config.id-list == [ ] config.id == null || config.id-list == [ ]
) "vlans.${name}.id and vlans.${name}.id-list are incompatible."; ) "vlans.${name}.id and vlans.${name}.id-list are incompatible.";
mkMerge [ ''
{ inherit name; } <vlan>
id <name>${name}</name>
id-list ${id}${id-list}${l3-intf}
l3-intf </vlan>
]; '';
}; };
in in
{ {
@ -113,9 +115,15 @@ in
instead of just using their IDs. instead of just using their IDs.
''; '';
}; };
netconf.xmls.vlans = mkOption {
type = str;
visible = false;
readOnly = true;
}; };
config.netconf.xml.vlans = {
"@operation" = "replace";
vlan = mapAttrsToList (_: vlan: vlan.xml) config.vlans;
}; };
config.netconf.xmls.vlans = ''
<vlans operation="replace">
${builtins.concatStringsSep "" (mapAttrsToList (_: vlan: vlan.xml) config.vlans)}
</vlans>
'';
} }

View file

@ -45,6 +45,8 @@ let
hashedPasswd = "$6$BKetIIfT$JVyE0B7F4O.fJwQFu5jVrVExAZROrEMLW5HkDkhjMShJ9cRIgxSm2VM9OThDowsnLmAewqDN7eAY.EQt4UR4U0"; hashedPasswd = "$6$BKetIIfT$JVyE0B7F4O.fJwQFu5jVrVExAZROrEMLW5HkDkhjMShJ9cRIgxSm2VM9OThDowsnLmAewqDN7eAY.EQt4UR4U0";
ssh-keys = [ ssh-keys = [
"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIAA16foz+XzwKwyIR4wFgNIAE3Y7AfXyEsUZFVVz8Rie catvayor@katvayor" "ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIAA16foz+XzwKwyIR4wFgNIAE3Y7AfXyEsUZFVVz8Rie catvayor@katvayor"
"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIFfIJ8BToZ9EDxBsEJXQhUju7gm+rUDjGCNMvFSZCl1o"
"ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAICdOxx4I8BSbYPdouvuzDepwTwzQzGSBCNIV8TB5dduT"
]; ];
}; };
}; };

View file

@ -1,81 +0,0 @@
# Generated by npins. Do not modify; will be overwritten regularly
let
data = builtins.fromJSON (builtins.readFile ./sources.json);
version = data.version;
mkSource =
spec:
assert spec ? type;
let
path =
if spec.type == "Git" then
mkGitSource spec
else if spec.type == "GitRelease" then
mkGitSource spec
else if spec.type == "PyPi" then
mkPyPiSource spec
else if spec.type == "Channel" then
mkChannelSource spec
else
builtins.throw "Unknown source type ${spec.type}";
in
spec // { outPath = path; };
mkGitSource =
{
repository,
revision,
url ? null,
hash,
branch ? null,
...
}:
assert repository ? type;
# At the moment, either it is a plain git repository (which has an url), or it is a GitHub/GitLab repository
# In the latter case, there we will always be an url to the tarball
if url != null then
(builtins.fetchTarball {
inherit url;
sha256 = hash; # FIXME: check nix version & use SRI hashes
})
else
assert repository.type == "Git";
let
urlToName =
url: rev:
let
matched = builtins.match "^.*/([^/]*)(\\.git)?$" repository.url;
short = builtins.substring 0 7 rev;
appendShort = if (builtins.match "[a-f0-9]*" rev) != null then "-${short}" else "";
in
"${if matched == null then "source" else builtins.head matched}${appendShort}";
name = urlToName repository.url revision;
in
builtins.fetchGit {
url = repository.url;
rev = revision;
inherit name;
allRefs = true;
# hash = hash;
};
mkPyPiSource =
{ url, hash, ... }:
builtins.fetchurl {
inherit url;
sha256 = hash;
};
mkChannelSource =
{ url, hash, ... }:
builtins.fetchTarball {
inherit url;
sha256 = hash;
};
in
if version == 3 then
builtins.mapAttrs (_: mkSource) data.pins
else
throw "Unsupported format version ${toString version} in sources.json. Try running `npins upgrade`"

View file

@ -1,22 +0,0 @@
{
"pins": {
"kat-pkgs": {
"type": "Git",
"repository": {
"type": "Git",
"url": "https://git.dgnum.eu/lbailly/kat-pkgs.git"
},
"branch": "master",
"revision": "191cb1dacb273078455b86006941386642681644",
"url": null,
"hash": "14rwcg92mnyb0azac27v05k72lm62pgl43cqj0np711sssjkmhv1"
},
"nixpkgs": {
"type": "Channel",
"name": "nixpkgs-unstable",
"url": "https://releases.nixos.org/nixpkgs/nixpkgs-25.05pre720687.5a48e3c2e435/nixexprs.tar.xz",
"hash": "06g8b0ga935dnziyzhxznwcx1vb2clc84hcxwrcqb26lgjgwsgbf"
}
},
"version": 3
}