Compare commits

..

2 commits

Author SHA1 Message Date
e10e63518b
feat: all vlans to potos 2024-12-18 13:08:39 +01:00
0bb363716d
feat: vlan-hypervisor to potos 2024-12-16 22:54:52 +01:00
10 changed files with 189 additions and 249 deletions

View file

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

View file

@ -1,7 +1,6 @@
{
lib,
config,
xml,
...
}:
let
@ -11,6 +10,7 @@ let
;
inherit (lib.types)
bool
str
attrsOf
submodule
;
@ -33,8 +33,9 @@ in
./system.nix
];
options = {
netconf.xml = mkOption {
type = xml.type;
netconf.xmls.configuration = mkOption {
type = str;
readOnly = true;
description = ''
The full configuration to send to a JunOS.
'';
@ -64,4 +65,13 @@ in
mkIntf = _: _: { };
in
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,
xml,
...
}:
{ lib, config, ... }:
let
inherit (lib)
mkEnableOption
mkOption
optionalString
mapAttrsToList
mkMerge
mkIf
;
inherit (lib.types)
enum
@ -26,6 +19,7 @@ let
interface =
{ name, config, ... }:
let
intf-name = name;
unit =
{ name, config, ... }:
{
@ -80,36 +74,46 @@ let
};
};
xml = mkOption {
type = xml.type;
type = str;
visible = false;
readOnly = true;
};
};
config.xml =
let
eth = mkIf config.family.ethernet-switching.enable {
family.ethernet-switching = {
interface-mode = config.family.ethernet-switching.interface-mode;
vlan.members = map toString config.family.ethernet-switching.vlans;
storm-control.profile-name = "default";
};
};
members = map (
vlan: "<members>${builtins.toString vlan}</members>"
) config.family.ethernet-switching.vlans;
eth = optionalString config.family.ethernet-switching.enable ''
<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;
inet = mkIf config.family.inet.enable { family.inet.address = addr4; };
addr4 = map (addr: "<name>${addr}</name>") config.family.inet.addresses;
inet = optionalString config.family.inet.enable ''
<inet>
<address>${builtins.concatStringsSep "" addr4}</address>
</inet>
'';
addr6 = map (name: { inherit name; }) config.family.inet6.addresses;
inet6 = mkIf config.family.inet6.enable { family.inet6.address = addr6; };
addr6 = map (addr: "<name>${addr}</name>") config.family.inet6.addresses;
inet6 = optionalString config.family.inet6.enable ''
<inet6>
<address>${builtins.concatStringsSep "" addr6}</address>
</inet6>
'';
in
mkMerge [
{
inherit name;
}
(mkIf (!config.enable) { disable = { }; })
eth
inet
inet6
];
''
<unit>
<name>${name}</name>
${optionalString (!config.enable) "<disable/>"}
<family>
${eth}${inet}${inet6}
</family>
</unit>'';
};
in
{
@ -123,21 +127,22 @@ let
'';
};
xml = mkOption {
type = xml.type;
type = str;
visible = false;
readOnly = true;
};
};
config.xml =
let
unit = mapAttrsToList (_: unit: unit.xml) config.unit;
units = mapAttrsToList (_: unit: unit.xml) config.unit;
in
mkMerge [
{
inherit name unit;
}
(mkIf (!config.enable) { disable = { }; })
];
''
<interface>
<name>${name}</name>
${optionalString (!config.enable) "<disable/>"}
${builtins.concatStringsSep "" units}
</interface>
'';
};
in
{
@ -148,9 +153,15 @@ in
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
inherit (lib)
mkOption
mkEnableOption
mapAttrsToList
mkIf
mkMerge
optionalString
;
inherit (lib.types)
str
attrsOf
submodule
;
@ -18,15 +18,14 @@ let
options = {
enable = mkEnableOption "the PoE for this interface";
xml = mkOption {
type = xml.type;
type = str;
visible = false;
readOnly = true;
};
};
config.xml = mkMerge [
{ inherit name; }
(mkIf (!config.enable) { disable = { }; })
];
config.xml = ''
<interface><name>${name}</name>${optionalString (!config.enable) "<disable/>"}</interface>
'';
};
in
{
@ -38,9 +37,15 @@ in
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
inherit (lib)
mkOption
concatStringsSep
;
inherit (lib.types)
listOf
@ -16,9 +17,21 @@ in
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
hasPrefix
filter
concatStrings
concatStringsSep
;
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
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-keys = map (concatStringsSep " ") ssh-keys2;
ssh-edsca = map (name: { inherit name; }) (filter (hasPrefix "ssh-edsca ") ssh-keys);
ssh-rsa = map (name: { inherit name; }) (filter (hasPrefix "ssh-rsa ") ssh-keys);
ssh-ed25519 = map (name: { inherit name; }) (filter (hasPrefix "ssh-ed25519 ") ssh-keys);
edsca = map (key: "<ssh-edsca><name>${key}</name></ssh-edsca>") (
filter (hasPrefix "ssh-edsca ") 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
{
host-name = {
"@operation" = "replace";
"#text" = config.system.host-name;
};
root-authentication = {
"@operation" = "replace";
encrypted-password = config.system.root-authentication.hashedPasswd;
inherit ssh-edsca ssh-rsa ssh-ed25519;
};
services = {
"@operation" = "replace";
ssh.root-login = config.system.services.ssh.root-login;
netconf = {
ssh.port = config.system.services.netconf.port;
rfc-compliant = {};
yang-compliant = {};
};
};
};
''
<system>
<host-name operation="replace">${config.system.host-name}</host-name>
<root-authentication operation="replace">
<encrypted-password>${config.system.root-authentication.hashedPasswd}</encrypted-password>
${concatStrings (edsca ++ rsa ++ ed25519)}
</root-authentication>
<services operation="replace">
<ssh><root-login>${config.system.services.ssh.root-login}</root-login></ssh>
<netconf>
<ssh><port>${toString config.system.services.netconf.port}</port></ssh>
<rfc-compliant/><yang-compliant/>
</netconf>
</services>
</system>
'';
}

View file

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

@ -15,6 +15,7 @@ let
];
"ap-staging".id = 2000;
"hypervisor".id = 2001;
};
AP = {
poe = true;
@ -118,9 +119,9 @@ in
};
# netcore01 (Potos)
"xe-0/1/2".ethernet-switching = {
interface-mode = "access";
interface-mode = "trunk";
vlans = [
"ap-staging"
"all"
];
};
# uplink
@ -172,39 +173,25 @@ in
dgn-interfaces =
let
admin-intf.ethernet-switching = {
interface-mode = "trunk";
vlans = [ "all" ];
hypervisor.ethernet-switching = {
interface-mode = "access";
vlans = [ "hypervisor" ];
};
in
{
"xe-0/1/0" = admin-intf;
"ge-0/1/0" = admin-intf;
"et-0/1/0" = admin-intf;
"xe-0/1/1" = admin-intf;
"ge-0/1/1" = admin-intf;
"et-0/1/1" = admin-intf;
"xe-0/1/2" = admin-intf;
"ge-0/1/2" = admin-intf;
"et-0/1/2" = admin-intf;
"xe-0/1/3" = admin-intf;
"ge-0/1/3" = admin-intf;
"et-0/1/3" = admin-intf;
"xe-0/2/0" = admin-intf;
"ge-0/2/0" = admin-intf;
"xe-0/2/1" = admin-intf;
"ge-0/2/1" = admin-intf;
"xe-0/2/2" = admin-intf;
"ge-0/2/2" = admin-intf;
"xe-0/2/3" = admin-intf;
"ge-0/2/3" = admin-intf;
"ge-0/0/47".ethernet-switching = {
"xe-0/2/0".ethernet-switching = {
interface-mode = "trunk";
vlans = [ "all" ];
};
"ge-0/0/0" = hypervisor;
"ge-0/0/1" = hypervisor;
"ge-0/0/2" = hypervisor;
"ge-0/0/3" = hypervisor;
"ge-0/0/4" = hypervisor;
"ge-0/0/5" = hypervisor;
"ge-0/0/47" = hypervisor;
# management
"me0".inet.addresses = [ "192.168.2.2/24" ];
"irb".inet6.addresses = [ "fd26:baf9:d250:8000::200f/64" ];

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
}