Installed diffview
This commit is contained in:
Generated
+17
@@ -774,6 +774,22 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"oidcwarden": {
|
||||||
|
"flake": false,
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1778081807,
|
||||||
|
"narHash": "sha256-tHacn9RtoByWpqnWX2/gWwODDSeXJa4mk4MfxHiiJ8A=",
|
||||||
|
"owner": "Timshel",
|
||||||
|
"repo": "OIDCWarden",
|
||||||
|
"rev": "48edfc7ba54372074befa1d62c63c4babfaadc77",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "Timshel",
|
||||||
|
"repo": "OIDCWarden",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
"pre-commit-hooks": {
|
"pre-commit-hooks": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"flake-compat": "flake-compat",
|
"flake-compat": "flake-compat",
|
||||||
@@ -810,6 +826,7 @@
|
|||||||
"nixos-generators": "nixos-generators",
|
"nixos-generators": "nixos-generators",
|
||||||
"nixpkgs": "nixpkgs",
|
"nixpkgs": "nixpkgs",
|
||||||
"nixvim": "nixvim",
|
"nixvim": "nixvim",
|
||||||
|
"oidcwarden": "oidcwarden",
|
||||||
"sops-nix": "sops-nix",
|
"sops-nix": "sops-nix",
|
||||||
"spicetify-nix": "spicetify-nix"
|
"spicetify-nix": "spicetify-nix"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -72,6 +72,11 @@
|
|||||||
url = "github:catppuccin/nix";
|
url = "github:catppuccin/nix";
|
||||||
inputs.nixpkgs.follows = "nixpkgs";
|
inputs.nixpkgs.follows = "nixpkgs";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
oidcwarden = {
|
||||||
|
url = "github:Timshel/OIDCWarden";
|
||||||
|
flake = false;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs =
|
outputs =
|
||||||
|
|||||||
@@ -26,6 +26,14 @@
|
|||||||
|
|
||||||
user = "git";
|
user = "git";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
"git.cyperpunk.de" = {
|
||||||
|
hostname = "git.cyperpunk.de";
|
||||||
|
port = 12222;
|
||||||
|
user = "gitea";
|
||||||
|
identityFile =
|
||||||
|
if isDarwin then "/Users/${primaryUser}/.ssh/ssh" else "/home/${primaryUser}/.ssh/ssh";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,8 @@
|
|||||||
../../nixos/roles/frontpage
|
../../nixos/roles/frontpage
|
||||||
../../nixos/roles/paperless-ngx.nix
|
../../nixos/roles/paperless-ngx.nix
|
||||||
../../nixos/roles/octoprint.nix
|
../../nixos/roles/octoprint.nix
|
||||||
|
../../nixos/roles/matrix/postgres-backup.nix
|
||||||
|
../../nixos/roles/kanidm.nix
|
||||||
];
|
];
|
||||||
|
|
||||||
networking = {
|
networking = {
|
||||||
|
|||||||
@@ -17,8 +17,20 @@
|
|||||||
prefixLength = 23;
|
prefixLength = 23;
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
ipv6.addresses = [
|
||||||
|
{
|
||||||
|
address = "2a00:6800:3:1094::1";
|
||||||
|
prefixLength = 64;
|
||||||
|
}
|
||||||
|
];
|
||||||
};
|
};
|
||||||
defaultGateway = "178.254.8.1";
|
defaultGateway = "178.254.8.1";
|
||||||
|
|
||||||
|
defaultGateway6 = {
|
||||||
|
address = "2a00:6800:3::1";
|
||||||
|
interface = "ens3";
|
||||||
|
};
|
||||||
|
|
||||||
nameservers = [
|
nameservers = [
|
||||||
"178.254.16.151"
|
"178.254.16.151"
|
||||||
"178.254.16.141"
|
"178.254.16.141"
|
||||||
@@ -34,6 +46,11 @@
|
|||||||
"cyperpunk.de"
|
"cyperpunk.de"
|
||||||
"matrix.cyperpunk.de"
|
"matrix.cyperpunk.de"
|
||||||
];
|
];
|
||||||
|
|
||||||
|
"2a00:6800:3:1094::1" = [
|
||||||
|
"cyperpunk.de"
|
||||||
|
"matrix.cyperpunk.de"
|
||||||
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|||||||
+20
-20
@@ -2,6 +2,7 @@
|
|||||||
pkgs,
|
pkgs,
|
||||||
inputs,
|
inputs,
|
||||||
lib,
|
lib,
|
||||||
|
config,
|
||||||
primaryUser,
|
primaryUser,
|
||||||
isServer,
|
isServer,
|
||||||
...
|
...
|
||||||
@@ -25,6 +26,10 @@
|
|||||||
|
|
||||||
nix = {
|
nix = {
|
||||||
settings = {
|
settings = {
|
||||||
|
trusted-users = [
|
||||||
|
"root"
|
||||||
|
primaryUser
|
||||||
|
];
|
||||||
experimental-features = [
|
experimental-features = [
|
||||||
"nix-command"
|
"nix-command"
|
||||||
"flakes"
|
"flakes"
|
||||||
@@ -37,11 +42,13 @@
|
|||||||
"https://cache.nixos.org"
|
"https://cache.nixos.org"
|
||||||
"https://hyprland.cachix.org"
|
"https://hyprland.cachix.org"
|
||||||
"https://nix-community.cachix.org"
|
"https://nix-community.cachix.org"
|
||||||
|
"https://cyper-cache.cachix.org"
|
||||||
];
|
];
|
||||||
trusted-public-keys = [
|
trusted-public-keys = [
|
||||||
"cache.nixos.org-1:6NCHdD59X431o0gWypbMrAURkbJ16ZPMQFGspcDShjY="
|
"cache.nixos.org-1:6NCHdD59X431o0gWypbMrAURkbJ16ZPMQFGspcDShjY="
|
||||||
"hyprland.cachix.org-1:a7pgxzMz7+chwVL3/pzj6jIBMioiJM7ypFP8PwtkuGc="
|
"hyprland.cachix.org-1:a7pgxzMz7+chwVL3/pzj6jIBMioiJM7ypFP8PwtkuGc="
|
||||||
"nix-community.cachix.org-1:mB9FSh9qf2dCimDSUo8Zy7bkq5CX+/rkCWyvRCYg3Fs="
|
"nix-community.cachix.org-1:mB9FSh9qf2dCimDSUo8Zy7bkq5CX+/rkCWyvRCYg3Fs="
|
||||||
|
"cyper-cache.cachix.org-1:pOpeWFEjGHg9XvqRg+DQpYnGRQNp+z+QEF8Ev2mbSoM="
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
gc = {
|
gc = {
|
||||||
@@ -117,32 +124,25 @@
|
|||||||
port = 9002;
|
port = 9002;
|
||||||
};
|
};
|
||||||
|
|
||||||
alloy = {
|
|
||||||
enable = true;
|
|
||||||
extraFlags = [ "--stability.level=public-preview" ];
|
|
||||||
configPath = pkgs.writeText "config.alloy" ''
|
|
||||||
loki.write "default" {
|
|
||||||
endpoint {
|
|
||||||
url = "http://192.168.2.2:3100/loki/api/v1/push"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
loki.source.journal "journal" {
|
|
||||||
forward_to = [loki.write.default.receiver]
|
|
||||||
labels = {
|
|
||||||
job = "systemd-journal",
|
|
||||||
host = sys.env("HOSTNAME"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
gnome = lib.mkIf (!isServer) {
|
gnome = lib.mkIf (!isServer) {
|
||||||
tinysparql.enable = true;
|
tinysparql.enable = true;
|
||||||
localsearch.enable = true;
|
localsearch.enable = true;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
sops.secrets.cachix_auth_token = { };
|
||||||
|
|
||||||
|
systemd.services.cachix-push = {
|
||||||
|
description = "Push new store paths to Cachix";
|
||||||
|
after = [ "multi-user.target" ];
|
||||||
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
serviceConfig = {
|
||||||
|
Type = "oneshot";
|
||||||
|
RemainAfterExit = true;
|
||||||
|
ExecStart = "${pkgs.bash}/bin/bash -c 'CACHIX_AUTH_TOKEN=$(cat ${config.sops.secrets.cachix_auth_token.path}) ${pkgs.nix}/bin/nix path-info --recursive /run/current-system | CACHIX_AUTH_TOKEN=$(cat ${config.sops.secrets.cachix_auth_token.path}) ${pkgs.cachix}/bin/cachix push cyper-cache'";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
networking.firewall.allowedTCPPorts = [
|
networking.firewall.allowedTCPPorts = [
|
||||||
9002
|
9002
|
||||||
3100
|
3100
|
||||||
|
|||||||
@@ -0,0 +1,12 @@
|
|||||||
|
{ pkgs, oidcwarden-src, ... }:
|
||||||
|
|
||||||
|
pkgs.vaultwarden.overrideAttrs (old: {
|
||||||
|
pname = "oidcwarden";
|
||||||
|
src = oidcwarden-src;
|
||||||
|
cargoDeps = pkgs.rustPlatform.importCargoLock {
|
||||||
|
lockFile = "${oidcwarden-src}/Cargo.lock";
|
||||||
|
};
|
||||||
|
postInstall = (old.postInstall or "") + ''
|
||||||
|
mv $out/bin/oidcwarden $out/bin/vaultwarden
|
||||||
|
'';
|
||||||
|
})
|
||||||
@@ -5,19 +5,18 @@
|
|||||||
|
|
||||||
let
|
let
|
||||||
primaryInterface = config.systemd.network.networks."10-ethernet".matchConfig.Name;
|
primaryInterface = config.systemd.network.networks."10-ethernet".matchConfig.Name;
|
||||||
adguardPort = 3000;
|
adguardPort = 3010;
|
||||||
in
|
in
|
||||||
{
|
{
|
||||||
services = {
|
services = {
|
||||||
resolved.enable = false;
|
resolved.enable = false;
|
||||||
adguardhome = {
|
adguardhome = {
|
||||||
enable = true;
|
enable = true;
|
||||||
mutableSettings = true;
|
mutableSettings = false;
|
||||||
allowDHCP = true;
|
allowDHCP = true;
|
||||||
|
port = adguardPort;
|
||||||
|
|
||||||
settings = {
|
settings = {
|
||||||
http.address = "0.0.0.0:${toString adguardPort}";
|
|
||||||
|
|
||||||
users = [
|
users = [
|
||||||
{
|
{
|
||||||
name = "DerGrumpf";
|
name = "DerGrumpf";
|
||||||
@@ -50,7 +49,7 @@ in
|
|||||||
};
|
};
|
||||||
|
|
||||||
dhcp = {
|
dhcp = {
|
||||||
enabled = true;
|
enabled = false;
|
||||||
interface_name = primaryInterface;
|
interface_name = primaryInterface;
|
||||||
local_domain_name = "lan";
|
local_domain_name = "lan";
|
||||||
dhcpv4 = {
|
dhcpv4 = {
|
||||||
|
|||||||
+12
-1
@@ -13,7 +13,7 @@ let
|
|||||||
stripRoot = false;
|
stripRoot = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
domain = "git.cyperpunk.de"; # swap to git.cyperpunk.de for prod
|
domain = "git.cyperpunk.de";
|
||||||
httpPort = 9000;
|
httpPort = 9000;
|
||||||
sshPort = 12222;
|
sshPort = 12222;
|
||||||
in
|
in
|
||||||
@@ -36,6 +36,11 @@ in
|
|||||||
owner = "gitea";
|
owner = "gitea";
|
||||||
group = "gitea";
|
group = "gitea";
|
||||||
};
|
};
|
||||||
|
"kanidm_gitea_secret" = {
|
||||||
|
owner = "gitea";
|
||||||
|
group = "gitea";
|
||||||
|
mode = "0444";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
services.postgresql = {
|
services.postgresql = {
|
||||||
@@ -79,6 +84,11 @@ in
|
|||||||
user = "gitea";
|
user = "gitea";
|
||||||
group = "gitea";
|
group = "gitea";
|
||||||
|
|
||||||
|
lfs = {
|
||||||
|
enable = true;
|
||||||
|
contentDir = "/storage/fast/lfs";
|
||||||
|
};
|
||||||
|
|
||||||
database = {
|
database = {
|
||||||
type = "postgres";
|
type = "postgres";
|
||||||
host = "127.0.0.1";
|
host = "127.0.0.1";
|
||||||
@@ -98,6 +108,7 @@ in
|
|||||||
ROOT_URL = "https://${domain}/";
|
ROOT_URL = "https://${domain}/";
|
||||||
DISABLE_SSH = false;
|
DISABLE_SSH = false;
|
||||||
START_SSH_SERVER = true;
|
START_SSH_SERVER = true;
|
||||||
|
LFS_START_SERVER = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
metrics = {
|
metrics = {
|
||||||
|
|||||||
@@ -1,117 +0,0 @@
|
|||||||
{
|
|
||||||
pkgs,
|
|
||||||
...
|
|
||||||
}:
|
|
||||||
|
|
||||||
let
|
|
||||||
domain = "jitsi.cyperpunk.de";
|
|
||||||
in
|
|
||||||
{
|
|
||||||
nixpkgs.config.permittedInsecurePackages = [
|
|
||||||
"jitsi-meet-1.0.8792"
|
|
||||||
];
|
|
||||||
|
|
||||||
services.jitsi-meet = {
|
|
||||||
enable = true;
|
|
||||||
hostName = domain;
|
|
||||||
|
|
||||||
config = {
|
|
||||||
enableWelcomePage = true;
|
|
||||||
prejoinPageEnabled = true;
|
|
||||||
enableInsecureRoomNameWarning = true;
|
|
||||||
disableAudioLevels = false;
|
|
||||||
enableLayerSuspension = true;
|
|
||||||
p2p.enabled = true;
|
|
||||||
analytics.disabled = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
interfaceConfig = {
|
|
||||||
SHOW_JITSI_WATERMARK = false;
|
|
||||||
SHOW_WATERMARK_FOR_GUESTS = false;
|
|
||||||
DEFAULT_REMOTE_DISPLAY_NAME = "Meeting @ Virtual";
|
|
||||||
TOOLBAR_BUTTONS = [
|
|
||||||
"microphone"
|
|
||||||
"camera"
|
|
||||||
"desktop"
|
|
||||||
"fullscreen"
|
|
||||||
"fodeviceselection"
|
|
||||||
"hangup"
|
|
||||||
"profile"
|
|
||||||
"chat"
|
|
||||||
"recording"
|
|
||||||
"livestreaming"
|
|
||||||
"etherpad"
|
|
||||||
"sharedvideo"
|
|
||||||
"settings"
|
|
||||||
"raisehand"
|
|
||||||
"videoquality"
|
|
||||||
"filmstrip"
|
|
||||||
"invite"
|
|
||||||
"feedback"
|
|
||||||
"stats"
|
|
||||||
"shortcuts"
|
|
||||||
"tileview"
|
|
||||||
"select-background"
|
|
||||||
"mute-everyone"
|
|
||||||
"security"
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
# Enable Jibri for recording/livestreaming support
|
|
||||||
jibri = {
|
|
||||||
enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
# Enable Jigasi for SIP/telephony support (optional, comment out if not needed)
|
|
||||||
# jigasi.enable = true;
|
|
||||||
|
|
||||||
nginx.enable = true;
|
|
||||||
prosody.enable = true;
|
|
||||||
};
|
|
||||||
|
|
||||||
# Jitsi Videobridge — handles the actual media routing
|
|
||||||
services.jitsi-videobridge = {
|
|
||||||
enable = true;
|
|
||||||
openFirewall = true;
|
|
||||||
|
|
||||||
config = {
|
|
||||||
videobridge = {
|
|
||||||
ice.udp.port = 10000;
|
|
||||||
apis.rest.enabled = true;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
networking.firewall = {
|
|
||||||
allowedTCPPorts = [
|
|
||||||
5222 # XMPP client (Prosody)
|
|
||||||
5269 # XMPP federation (Prosody)
|
|
||||||
];
|
|
||||||
allowedUDPPorts = [
|
|
||||||
10000 # Jitsi Videobridge RTP media
|
|
||||||
];
|
|
||||||
allowedUDPPortRanges = [
|
|
||||||
{
|
|
||||||
from = 49152;
|
|
||||||
to = 65535;
|
|
||||||
} # WebRTC ephemeral ports
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
# Prosody needs this for XMPP
|
|
||||||
networking.extraHosts = ''
|
|
||||||
127.0.0.1 ${domain}
|
|
||||||
127.0.0.1 auth.${domain}
|
|
||||||
127.0.0.1 focus.${domain}
|
|
||||||
127.0.0.1 jitsi-videobridge.${domain}
|
|
||||||
'';
|
|
||||||
|
|
||||||
# Jibri requires Chromium for recording
|
|
||||||
environment.systemPackages = with pkgs; [
|
|
||||||
chromium
|
|
||||||
ffmpeg
|
|
||||||
];
|
|
||||||
|
|
||||||
# ALSA loopback device — required by Jibri for audio capture during recording
|
|
||||||
boot.kernelModules = [ "snd-aloop" ];
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
{ pkgs, ... }:
|
||||||
|
let
|
||||||
|
domain = "auth.cyperpunk.de";
|
||||||
|
port = 8444;
|
||||||
|
certDir = "/var/lib/kanidm/tls";
|
||||||
|
in
|
||||||
|
{
|
||||||
|
systemd.services.kanidm-selfsigned-cert = {
|
||||||
|
description = "Generate self-signed TLS certificate for Kanidm";
|
||||||
|
wantedBy = [ "kanidm.service" ];
|
||||||
|
before = [ "kanidm.service" ];
|
||||||
|
serviceConfig = {
|
||||||
|
Type = "oneshot";
|
||||||
|
RemainAfterExit = true;
|
||||||
|
};
|
||||||
|
script = ''
|
||||||
|
if [ ! -f ${certDir}/cert.pem ]; then
|
||||||
|
mkdir -p ${certDir}
|
||||||
|
${pkgs.openssl}/bin/openssl req -x509 \
|
||||||
|
-newkey rsa:4096 \
|
||||||
|
-keyout ${certDir}/key.pem \
|
||||||
|
-out ${certDir}/cert.pem \
|
||||||
|
-days 3650 \
|
||||||
|
-nodes \
|
||||||
|
-subj "/CN=${domain}"
|
||||||
|
chown -R kanidm:kanidm ${certDir}
|
||||||
|
chmod 750 ${certDir}
|
||||||
|
chmod 640 ${certDir}/cert.pem ${certDir}/key.pem
|
||||||
|
fi
|
||||||
|
'';
|
||||||
|
};
|
||||||
|
|
||||||
|
services.kanidm = {
|
||||||
|
package = pkgs.kanidm_1_10;
|
||||||
|
|
||||||
|
server = {
|
||||||
|
enable = true;
|
||||||
|
settings = {
|
||||||
|
inherit domain;
|
||||||
|
origin = "https://${domain}";
|
||||||
|
|
||||||
|
tls_chain = "${certDir}/cert.pem";
|
||||||
|
tls_key = "${certDir}/key.pem";
|
||||||
|
|
||||||
|
bindaddress = "0.0.0.0:${toString port}";
|
||||||
|
|
||||||
|
log_level = "info";
|
||||||
|
|
||||||
|
online_backup = {
|
||||||
|
versions = 7;
|
||||||
|
path = "/var/lib/kanidm/backups";
|
||||||
|
schedule = "00 22 * * *";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
client = {
|
||||||
|
enable = true;
|
||||||
|
settings.uri = "https://${domain}";
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
networking.firewall.allowedTCPPorts = [ port ];
|
||||||
|
}
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
{ config, ... }:
|
|
||||||
{
|
|
||||||
services = {
|
|
||||||
nginx.virtualHosts."www.cyperpunk.de".locations."/cloak" = {
|
|
||||||
proxyPass = "http://localhost:${toString config.services.keycloak.settings.http-port}/cloak/";
|
|
||||||
};
|
|
||||||
|
|
||||||
keycloak = {
|
|
||||||
enable = true;
|
|
||||||
|
|
||||||
database = {
|
|
||||||
type = "postgresql";
|
|
||||||
createLocally = true;
|
|
||||||
|
|
||||||
username = "keycloak";
|
|
||||||
passwordFile = "/etc/nixos/secrets/keycloak_psql_pass";
|
|
||||||
};
|
|
||||||
|
|
||||||
settings = {
|
|
||||||
hostname = "cyperpunk.de";
|
|
||||||
http-relative-path = "/cloak";
|
|
||||||
http-port = 38080;
|
|
||||||
proxy = "passthrough";
|
|
||||||
http-enabled = true;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -133,15 +133,31 @@ let
|
|||||||
'';
|
'';
|
||||||
});
|
});
|
||||||
|
|
||||||
|
cinnyConfigured = pkgs.cinny-unwrapped.overrideAttrs (_: {
|
||||||
|
postInstall = ''
|
||||||
|
cp ${
|
||||||
|
builtins.toFile "cinny-config.json" (
|
||||||
|
builtins.toJSON {
|
||||||
|
defaultHomeserver = 0;
|
||||||
|
homeserverList = [ "cyperpunk.de" ];
|
||||||
|
allowCustomHomeservers = false;
|
||||||
|
}
|
||||||
|
)
|
||||||
|
} $out/config.json
|
||||||
|
'';
|
||||||
|
});
|
||||||
|
|
||||||
in
|
in
|
||||||
{
|
{
|
||||||
services.nginx.virtualHosts = {
|
services.nginx.virtualHosts = {
|
||||||
"cinny.cyperpunk.de" = {
|
"cinny.cyperpunk.de" = {
|
||||||
forceSSL = true;
|
forceSSL = true;
|
||||||
enableACME = true;
|
enableACME = true;
|
||||||
root = "${pkgs.cinny}";
|
root = "${cinnyConfigured}";
|
||||||
|
locations."/" = {
|
||||||
|
tryFiles = "$uri $uri/ /index.html";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
"element.cyperpunk.de" = {
|
"element.cyperpunk.de" = {
|
||||||
forceSSL = true;
|
forceSSL = true;
|
||||||
enableACME = true;
|
enableACME = true;
|
||||||
|
|||||||
@@ -7,96 +7,108 @@
|
|||||||
{
|
{
|
||||||
sops.secrets = {
|
sops.secrets = {
|
||||||
pg_replication_password = {
|
pg_replication_password = {
|
||||||
owner = "postgres";
|
owner = "root";
|
||||||
group = "postgres";
|
group = "root";
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
services.postgresql = {
|
virtualisation.docker.enable = true;
|
||||||
enable = true;
|
|
||||||
package = pkgs.postgresql_15; # must match cyper-proxy's PG version exactly
|
|
||||||
|
|
||||||
settings = {
|
systemd.services.postgresql-replica = {
|
||||||
hot_standby = true;
|
description = "PostgreSQL WAL streaming replica (Docker)";
|
||||||
hot_standby_feedback = true;
|
|
||||||
max_standby_streaming_delay = "30s";
|
|
||||||
listen_addresses = "127.0.0.1"; # only local, no external exposure
|
|
||||||
wal_receiver_timeout = "60s";
|
|
||||||
recovery_min_apply_delay = "0"; # set e.g. "1h" for a delayed safety replica
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
# Writes standby.signal and primary_conninfo before PostgreSQL starts.
|
|
||||||
systemd.services.postgresql = {
|
|
||||||
preStart = lib.mkAfter ''
|
|
||||||
DATADIR="${config.services.postgresql.dataDir}"
|
|
||||||
PG_PASS=$(cat ${config.sops.secrets.pg_replication_password.path})
|
|
||||||
|
|
||||||
# Tell Postgres to start as a hot standby
|
|
||||||
if [ ! -f "$DATADIR/standby.signal" ]; then
|
|
||||||
touch "$DATADIR/standby.signal"
|
|
||||||
chown postgres:postgres "$DATADIR/standby.signal"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# primary_conninfo via Tailscale — no public IP involved
|
|
||||||
CONNINFO="host=100.109.10.91 port=5432 user=replicator password=$PG_PASS application_name=cyper-controller sslmode=require"
|
|
||||||
|
|
||||||
grep -v "^primary_conninfo" "$DATADIR/postgresql.auto.conf" 2>/dev/null > /tmp/auto.conf.tmp || true
|
|
||||||
echo "primary_conninfo = '$CONNINFO'" >> /tmp/auto.conf.tmp
|
|
||||||
mv /tmp/auto.conf.tmp "$DATADIR/postgresql.auto.conf"
|
|
||||||
chown postgres:postgres "$DATADIR/postgresql.auto.conf"
|
|
||||||
'';
|
|
||||||
};
|
|
||||||
|
|
||||||
# Run once manually to seed the standby: systemctl start postgresql-basebackup
|
|
||||||
# Do NOT add it to wantedBy — it would wipe the data dir on every reboot.
|
|
||||||
systemd.services.postgresql-basebackup = {
|
|
||||||
description = "Bootstrap PostgreSQL standby via pg_basebackup from cyper-proxy";
|
|
||||||
requires = [
|
requires = [
|
||||||
"network-online.target"
|
"docker.service"
|
||||||
"tailscaled.service"
|
"tailscaled.service"
|
||||||
|
"network-online.target"
|
||||||
];
|
];
|
||||||
after = [
|
after = [
|
||||||
"network-online.target"
|
"docker.service"
|
||||||
"tailscaled.service"
|
"tailscaled.service"
|
||||||
|
"network-online.target"
|
||||||
];
|
];
|
||||||
|
wantedBy = [ "multi-user.target" ];
|
||||||
|
|
||||||
serviceConfig = {
|
serviceConfig = {
|
||||||
Type = "oneshot";
|
Type = "simple";
|
||||||
User = "postgres";
|
Restart = "on-failure";
|
||||||
RemainAfterExit = false;
|
RestartSec = "10s";
|
||||||
};
|
};
|
||||||
|
|
||||||
|
preStart = ''
|
||||||
|
DATADIR="/storage/backup/postgresql-replica"
|
||||||
|
PG_PASS=$(cat ${config.sops.secrets.pg_replication_password.path})
|
||||||
|
|
||||||
|
${pkgs.docker}/bin/docker pull postgres:17
|
||||||
|
|
||||||
|
if [ ! -f "$DATADIR/PG_VERSION" ]; then
|
||||||
|
echo "No data dir found — running pg_basebackup..."
|
||||||
|
rm -rf "$DATADIR"
|
||||||
|
|
||||||
|
${pkgs.docker}/bin/docker run --rm \
|
||||||
|
-e PGPASSWORD="$PG_PASS" \
|
||||||
|
-v "/storage/backup:/out" \
|
||||||
|
--network host \
|
||||||
|
postgres:17 \
|
||||||
|
pg_basebackup \
|
||||||
|
--host=100.109.10.91 \
|
||||||
|
--port=5432 \
|
||||||
|
--username=replicator \
|
||||||
|
--pgdata=/out/postgresql-replica \
|
||||||
|
--wal-method=stream \
|
||||||
|
--checkpoint=fast \
|
||||||
|
--progress \
|
||||||
|
--verbose
|
||||||
|
|
||||||
|
# standby signal
|
||||||
|
touch "$DATADIR/standby.signal"
|
||||||
|
|
||||||
|
# primary conninfo
|
||||||
|
cat > "$DATADIR/postgresql.auto.conf" <<EOF
|
||||||
|
primary_conninfo = 'host=100.109.10.91 port=5432 user=replicator password=$PG_PASS application_name=cyper-controller sslmode=require'
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# minimal postgresql.conf
|
||||||
|
cat > "$DATADIR/postgresql.conf" <<EOF
|
||||||
|
hot_standby = on
|
||||||
|
hot_standby_feedback = on
|
||||||
|
wal_receiver_timeout = '60s'
|
||||||
|
port = 5434
|
||||||
|
listen_addresses = '*'
|
||||||
|
EOF
|
||||||
|
|
||||||
|
chown -R 999:999 "$DATADIR"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# ensure postgresql.conf exists on subsequent starts
|
||||||
|
if [ ! -f "$DATADIR/postgresql.conf" ]; then
|
||||||
|
cat > "$DATADIR/postgresql.conf" <<EOF
|
||||||
|
hot_standby = on
|
||||||
|
hot_standby_feedback = on
|
||||||
|
wal_receiver_timeout = '60s'
|
||||||
|
port = 5434
|
||||||
|
listen_addresses = '*'
|
||||||
|
EOF
|
||||||
|
chown 999:999 "$DATADIR/postgresql.conf"
|
||||||
|
fi
|
||||||
|
'';
|
||||||
|
|
||||||
script = ''
|
script = ''
|
||||||
DATADIR="${config.services.postgresql.dataDir}"
|
${pkgs.docker}/bin/docker run --rm \
|
||||||
PG_PASS=$(cat ${config.sops.secrets.pg_replication_password.path})
|
--name postgresql-replica \
|
||||||
|
-v /storage/backup/postgresql-replica:/var/lib/postgresql/data \
|
||||||
|
--network host \
|
||||||
|
postgres:17
|
||||||
|
'';
|
||||||
|
|
||||||
if [ -d "$DATADIR/global" ]; then
|
postStop = ''
|
||||||
echo "Data directory already exists — skipping."
|
${pkgs.docker}/bin/docker rm -f postgresql-replica 2>/dev/null || true
|
||||||
echo "Remove $DATADIR manually to force a re-initialise."
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "Running pg_basebackup from cyper-proxy via Tailscale..."
|
|
||||||
PGPASSWORD="$PG_PASS" ${config.services.postgresql.package}/bin/pg_basebackup \
|
|
||||||
--host=100.109.10.91 \
|
|
||||||
--port=5432 \
|
|
||||||
--username=replicator \
|
|
||||||
--pgdata="$DATADIR" \
|
|
||||||
--wal-method=stream \
|
|
||||||
--write-recovery-conf \
|
|
||||||
--checkpoint=fast \
|
|
||||||
--progress \
|
|
||||||
--verbose
|
|
||||||
|
|
||||||
chown -R postgres:postgres "$DATADIR"
|
|
||||||
chmod 0700 "$DATADIR"
|
|
||||||
echo "Done. Start postgresql.service to begin streaming replication."
|
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
|
|
||||||
# Block any external access to postgres on the public interface
|
services.prometheus.exporters.postgres = {
|
||||||
networking.firewall = {
|
enable = true;
|
||||||
interfaces."tailscale0".allowedTCPPorts = [ ]; # replication is outbound only
|
port = 9188;
|
||||||
|
runAsLocalSuperUser = false;
|
||||||
|
dataSourceName = "postgresql://postgres@localhost:5434/postgres?sslmode=disable";
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -34,7 +34,14 @@ in
|
|||||||
owner = "matrix-synapse";
|
owner = "matrix-synapse";
|
||||||
group = "matrix-synapse";
|
group = "matrix-synapse";
|
||||||
};
|
};
|
||||||
pg_replication_password = { };
|
pg_replication_password = {
|
||||||
|
owner = "postgres";
|
||||||
|
group = "postgres";
|
||||||
|
};
|
||||||
|
kanidm_synapse_secret = {
|
||||||
|
owner = "matrix-synapse";
|
||||||
|
group = "matrix-synapse";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
services = {
|
services = {
|
||||||
@@ -57,14 +64,21 @@ in
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
#experimental_features = {
|
|
||||||
# msc3266_enabled = true;
|
rc_login = {
|
||||||
# msc3779_enabled = true;
|
address = {
|
||||||
# msc3401_enabled = true;
|
per_second = 0.17;
|
||||||
# msc4143_enabled = true;
|
burst_count = 10;
|
||||||
# msc4195_enabled = true;
|
};
|
||||||
# msc4222_enabled = true;
|
account = {
|
||||||
#};
|
per_second = 0.17;
|
||||||
|
burst_count = 10;
|
||||||
|
};
|
||||||
|
failed_attempts = {
|
||||||
|
per_second = 0.17;
|
||||||
|
burst_count = 10;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
listeners = [
|
listeners = [
|
||||||
{
|
{
|
||||||
@@ -91,16 +105,38 @@ in
|
|||||||
port = 9009;
|
port = 9009;
|
||||||
tls = false;
|
tls = false;
|
||||||
type = "metrics";
|
type = "metrics";
|
||||||
bind_addresses = [ "127.0.0.1" ];
|
bind_addresses = [
|
||||||
|
"127.0.0.1"
|
||||||
|
"100.109.10.91"
|
||||||
|
];
|
||||||
resources = [ ];
|
resources = [ ];
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
enable_metrics = true;
|
enable_metrics = true;
|
||||||
|
|
||||||
|
oidc_providers = [
|
||||||
|
{
|
||||||
|
idp_id = "kanidm";
|
||||||
|
idp_name = "Kanidm";
|
||||||
|
issuer = "https://auth.cyperpunk.de/oauth2/openid/synapse";
|
||||||
|
client_id = "synapse";
|
||||||
|
client_secret_path = config.sops.secrets.kanidm_synapse_secret.path;
|
||||||
|
scopes = [
|
||||||
|
"openid"
|
||||||
|
"profile"
|
||||||
|
"email"
|
||||||
|
];
|
||||||
|
allow_existing_users = true;
|
||||||
|
user_mapping_provider.config = {
|
||||||
|
localpart_template = "{{ user.preferred_username.split('@')[0] }}";
|
||||||
|
display_name_template = "{{ user.displayname }}";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
nginx.virtualHosts = {
|
nginx.virtualHosts = {
|
||||||
# Matrix homeserver
|
|
||||||
"cyperpunk.de" = {
|
"cyperpunk.de" = {
|
||||||
forceSSL = true;
|
forceSSL = true;
|
||||||
enableACME = true;
|
enableACME = true;
|
||||||
@@ -155,14 +191,22 @@ in
|
|||||||
|
|
||||||
settings = {
|
settings = {
|
||||||
wal_level = "replica";
|
wal_level = "replica";
|
||||||
max_wal_senders = 3;
|
max_wal_senders = 5;
|
||||||
wal_keep_size = "512MB";
|
wal_keep_size = "512MB";
|
||||||
|
listen_addresses = lib.mkForce "127.0.0.1,100.109.10.91";
|
||||||
};
|
};
|
||||||
|
|
||||||
authentication = lib.mkAfter ''
|
authentication = lib.mkAfter ''
|
||||||
host replication replicator 100.0.0.0/8 scram-sha-256
|
host replication replicator 100.0.0.0/8 scram-sha-256
|
||||||
'';
|
'';
|
||||||
};
|
};
|
||||||
|
|
||||||
|
prometheus.exporters.postgres = {
|
||||||
|
enable = true;
|
||||||
|
port = 9188;
|
||||||
|
runAsLocalSuperUser = true;
|
||||||
|
dataSourceName = "postgresql:///postgres?host=/run/postgresql&sslmode=disable";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
systemd.services = {
|
systemd.services = {
|
||||||
|
|||||||
+47
-96
@@ -21,44 +21,6 @@ let
|
|||||||
"cyper-node-2" = "192.168.2.31";
|
"cyper-node-2" = "192.168.2.31";
|
||||||
"cyper-proxy" = "178.254.8.35";
|
"cyper-proxy" = "178.254.8.35";
|
||||||
};
|
};
|
||||||
|
|
||||||
mkWeatherScrapeConfigs =
|
|
||||||
cities:
|
|
||||||
map (city: {
|
|
||||||
job_name = "weather_${lib.strings.toLower (lib.strings.replaceStrings [ " " ] [ "_" ] city)}";
|
|
||||||
scrape_interval = "5m";
|
|
||||||
scrape_timeout = "30s";
|
|
||||||
metrics_path = "/${city}";
|
|
||||||
params.format = [ "p1" ];
|
|
||||||
static_configs = [ { targets = [ "wttr.in" ]; } ];
|
|
||||||
scheme = "https";
|
|
||||||
metric_relabel_configs = [
|
|
||||||
{
|
|
||||||
target_label = "location";
|
|
||||||
replacement = city;
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}) cities;
|
|
||||||
|
|
||||||
weatherCities = [
|
|
||||||
# Braunschweig itself
|
|
||||||
"Braunschweig"
|
|
||||||
|
|
||||||
# Immediate surroundings (~15km)
|
|
||||||
"Wolfenbuettel"
|
|
||||||
"Salzgitter"
|
|
||||||
"Peine"
|
|
||||||
"Cremlingen"
|
|
||||||
"Wendeburg"
|
|
||||||
"Sickte"
|
|
||||||
|
|
||||||
# Greater region (~50km)
|
|
||||||
"Wolfsburg"
|
|
||||||
"Gifhorn"
|
|
||||||
"Goslar"
|
|
||||||
"Hildesheim"
|
|
||||||
"Hannover"
|
|
||||||
];
|
|
||||||
in
|
in
|
||||||
{
|
{
|
||||||
sops.secrets = {
|
sops.secrets = {
|
||||||
@@ -66,7 +28,10 @@ in
|
|||||||
owner = "grafana";
|
owner = "grafana";
|
||||||
group = "grafana";
|
group = "grafana";
|
||||||
};
|
};
|
||||||
|
kanidm_grafana_secret = {
|
||||||
|
owner = "grafana";
|
||||||
|
group = "grafana";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
services = {
|
services = {
|
||||||
@@ -83,12 +48,6 @@ in
|
|||||||
url = "http://127.0.0.1:${toString config.services.prometheus.port}";
|
url = "http://127.0.0.1:${toString config.services.prometheus.port}";
|
||||||
isDefault = true;
|
isDefault = true;
|
||||||
}
|
}
|
||||||
{
|
|
||||||
name = "Loki";
|
|
||||||
type = "loki";
|
|
||||||
url = "http://127.0.0.1:${toString config.services.loki.configuration.server.http_listen_port}";
|
|
||||||
isDefault = false;
|
|
||||||
}
|
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
@@ -97,7 +56,7 @@ in
|
|||||||
domain = "www.cyperpunk.de"; # serverIP; # "grafana.cyperpunk.de";
|
domain = "www.cyperpunk.de"; # serverIP; # "grafana.cyperpunk.de";
|
||||||
http_port = 2342;
|
http_port = 2342;
|
||||||
http_addr = "0.0.0.0";
|
http_addr = "0.0.0.0";
|
||||||
root_url = "http://www.cyperpunk.de/grafana/";
|
root_url = "https://www.cyperpunk.de/grafana/";
|
||||||
serve_from_sub_path = true;
|
serve_from_sub_path = true;
|
||||||
};
|
};
|
||||||
security = {
|
security = {
|
||||||
@@ -106,6 +65,23 @@ in
|
|||||||
};
|
};
|
||||||
auth = {
|
auth = {
|
||||||
disable_login_form = false;
|
disable_login_form = false;
|
||||||
|
oauth_allow_insecure_email_lookup = true;
|
||||||
|
};
|
||||||
|
"auth.generic_oauth" = {
|
||||||
|
enabled = true;
|
||||||
|
name = "Kanidm";
|
||||||
|
client_id = "grafana";
|
||||||
|
client_secret = "$__file{${config.sops.secrets.kanidm_grafana_secret.path}}";
|
||||||
|
scopes = "openid profile email";
|
||||||
|
auth_url = "https://auth.cyperpunk.de/ui/oauth2";
|
||||||
|
token_url = "https://auth.cyperpunk.de/oauth2/token";
|
||||||
|
api_url = "https://auth.cyperpunk.de/oauth2/openid/grafana/userinfo";
|
||||||
|
use_pkce = false;
|
||||||
|
allow_sign_up = true;
|
||||||
|
auto_assign_org = true;
|
||||||
|
auto_assign_org_id = 1;
|
||||||
|
auto_assign_org_role = "Admin";
|
||||||
|
skip_org_role_sync = true;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
@@ -134,60 +110,39 @@ in
|
|||||||
metrics_path = "/_synapse/metrics";
|
metrics_path = "/_synapse/metrics";
|
||||||
static_configs = [
|
static_configs = [
|
||||||
{
|
{
|
||||||
targets = [ "127.0.0.1:9009" ];
|
targets = [ "100.109.10.91:9009" ];
|
||||||
labels = {
|
labels = {
|
||||||
instance = config.networking.hostName;
|
instance = "cyper-proxy";
|
||||||
job = "master";
|
job = "master";
|
||||||
index = "1";
|
index = "1";
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
job_name = "postgresql-replica";
|
||||||
|
static_configs = [
|
||||||
|
{
|
||||||
|
targets = [ "localhost:9188" ];
|
||||||
|
labels = {
|
||||||
|
instance = config.networking.hostName;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
{
|
||||||
|
job_name = "postgresql-proxy";
|
||||||
|
static_configs = [
|
||||||
|
{
|
||||||
|
targets = [ "100.109.10.91:9188" ];
|
||||||
|
labels = {
|
||||||
|
instance = "cyper-proxy";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
]
|
]
|
||||||
++ (lib.mapAttrsToList mkNodeJob extraNodes)
|
++ (lib.mapAttrsToList mkNodeJob extraNodes);
|
||||||
++ (mkWeatherScrapeConfigs weatherCities);
|
|
||||||
};
|
|
||||||
|
|
||||||
loki = {
|
|
||||||
enable = true;
|
|
||||||
configuration = {
|
|
||||||
auth_enabled = false;
|
|
||||||
server.http_listen_port = 3100;
|
|
||||||
ingester = {
|
|
||||||
lifecycler = {
|
|
||||||
address = "127.0.0.1";
|
|
||||||
ring = {
|
|
||||||
kvstore.store = "inmemory";
|
|
||||||
replication_factor = 1;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
chunk_idle_period = "5m";
|
|
||||||
chunk_retain_period = "30s";
|
|
||||||
};
|
|
||||||
schema_config.configs = [
|
|
||||||
{
|
|
||||||
from = "2024-01-01";
|
|
||||||
store = "tsdb";
|
|
||||||
object_store = "filesystem";
|
|
||||||
schema = "v13";
|
|
||||||
index = {
|
|
||||||
prefix = "index_";
|
|
||||||
period = "24h";
|
|
||||||
};
|
|
||||||
}
|
|
||||||
];
|
|
||||||
storage_config = {
|
|
||||||
tsdb_shipper = {
|
|
||||||
active_index_directory = "/var/lib/loki/tsdb-index";
|
|
||||||
cache_location = "/var/lib/loki/tsdb-cache";
|
|
||||||
};
|
|
||||||
filesystem.directory = "/var/lib/loki/chunks";
|
|
||||||
};
|
|
||||||
limits_config = {
|
|
||||||
reject_old_samples = true;
|
|
||||||
reject_old_samples_max_age = "168h";
|
|
||||||
};
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -196,8 +151,4 @@ in
|
|||||||
9001
|
9001
|
||||||
3100
|
3100
|
||||||
];
|
];
|
||||||
|
|
||||||
systemd.tmpfiles.rules = [
|
|
||||||
"d /var/loki 0700 loki loki -"
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|||||||
+24
-1
@@ -19,11 +19,21 @@ let
|
|||||||
proxyWebsockets = true;
|
proxyWebsockets = true;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
mkHttpsProxy = port: {
|
||||||
|
forceSSL = true;
|
||||||
|
enableACME = true;
|
||||||
|
locations."/" = {
|
||||||
|
proxyPass = "https://${upstream}:${toString port}";
|
||||||
|
extraConfig = "proxy_ssl_verify off;";
|
||||||
|
};
|
||||||
|
};
|
||||||
in
|
in
|
||||||
{
|
{
|
||||||
networking.firewall.allowedTCPPorts = [
|
networking.firewall.allowedTCPPorts = [
|
||||||
80
|
80
|
||||||
443
|
443
|
||||||
|
12222
|
||||||
];
|
];
|
||||||
|
|
||||||
security.acme = {
|
security.acme = {
|
||||||
@@ -38,14 +48,27 @@ in
|
|||||||
recommendedOptimisation = true;
|
recommendedOptimisation = true;
|
||||||
recommendedGzipSettings = true;
|
recommendedGzipSettings = true;
|
||||||
|
|
||||||
|
# Git ssh
|
||||||
|
streamConfig = ''
|
||||||
|
server {
|
||||||
|
listen 12222;
|
||||||
|
proxy_pass ${upstream}:12222;
|
||||||
|
}
|
||||||
|
'';
|
||||||
|
|
||||||
virtualHosts = {
|
virtualHosts = {
|
||||||
# controller services (proxied to upstream tailscale node)
|
# controller services (proxied to upstream tailscale node)
|
||||||
"git.cyperpunk.de" = mkProxy 9000;
|
"git.cyperpunk.de" = (mkProxy 9000) // {
|
||||||
|
extraConfig = ''
|
||||||
|
client_max_body_size 500m;
|
||||||
|
'';
|
||||||
|
};
|
||||||
"search.cyperpunk.de" = mkProxy 11080;
|
"search.cyperpunk.de" = mkProxy 11080;
|
||||||
"file.cyperpunk.de" = mkProxy 10000;
|
"file.cyperpunk.de" = mkProxy 10000;
|
||||||
"ngx.cyperpunk.de" = mkWsProxy 28101;
|
"ngx.cyperpunk.de" = mkWsProxy 28101;
|
||||||
"vault.cyperpunk.de" = mkWsProxy 8222;
|
"vault.cyperpunk.de" = mkWsProxy 8222;
|
||||||
"calvin.cyperpunk.de" = mkWsProxy 15006;
|
"calvin.cyperpunk.de" = mkWsProxy 15006;
|
||||||
|
"auth.cyperpunk.de" = mkHttpsProxy 8444;
|
||||||
|
|
||||||
"www.cyperpunk.de" = {
|
"www.cyperpunk.de" = {
|
||||||
forceSSL = true;
|
forceSSL = true;
|
||||||
|
|||||||
@@ -1,9 +1,15 @@
|
|||||||
{ config, ... }:
|
{ config, ... }:
|
||||||
{
|
{
|
||||||
|
|
||||||
sops.secrets.paperless_admin = {
|
sops.secrets = {
|
||||||
owner = "paperless";
|
paperless_admin = {
|
||||||
|
owner = "paperless";
|
||||||
|
};
|
||||||
|
paperless_oidc_secret = {
|
||||||
|
owner = "paperless";
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
services.paperless = {
|
services.paperless = {
|
||||||
enable = true;
|
enable = true;
|
||||||
address = "0.0.0.0";
|
address = "0.0.0.0";
|
||||||
@@ -23,6 +29,7 @@
|
|||||||
];
|
];
|
||||||
PAPERLESS_OCR_LANGUAGE = "deu+eng";
|
PAPERLESS_OCR_LANGUAGE = "deu+eng";
|
||||||
PAPERLESS_CONSUMER_POLLING = 60;
|
PAPERLESS_CONSUMER_POLLING = 60;
|
||||||
|
PAPERLESS_APPS = "allauth.socialaccount.providers.openid_connect";
|
||||||
};
|
};
|
||||||
|
|
||||||
exporter = {
|
exporter = {
|
||||||
@@ -41,9 +48,14 @@
|
|||||||
"d /storage/backup/paperless 0775 root users -"
|
"d /storage/backup/paperless 0775 root users -"
|
||||||
];
|
];
|
||||||
|
|
||||||
services.paperless-scheduler = {
|
services = {
|
||||||
after = [ "systemd-tmpfiles-setup.service" ];
|
paperless-scheduler = {
|
||||||
requires = [ "systemd-tmpfiles-setup.service" ];
|
after = [ "systemd-tmpfiles-setup.service" ];
|
||||||
|
requires = [ "systemd-tmpfiles-setup.service" ];
|
||||||
|
};
|
||||||
|
paperless-web = {
|
||||||
|
serviceConfig.EnvironmentFiles = [ config.sops.secrets.paperless_oidc_secret.path ];
|
||||||
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
networking.firewall.allowedTCPPorts = [ 28101 ];
|
networking.firewall.allowedTCPPorts = [ 28101 ];
|
||||||
|
|||||||
+19
-11
@@ -1,36 +1,44 @@
|
|||||||
{
|
{
|
||||||
config,
|
config,
|
||||||
pkgs,
|
pkgs,
|
||||||
lib,
|
inputs,
|
||||||
...
|
...
|
||||||
}:
|
}:
|
||||||
|
|
||||||
let
|
let
|
||||||
address = config.systemd.network.networks."10-ethernet".networkConfig.Address;
|
|
||||||
ip = builtins.elemAt (lib.splitString "/" address) 0;
|
|
||||||
port = 8222;
|
port = 8222;
|
||||||
|
oidcwarden = import ../packages/oidcwarden.nix {
|
||||||
|
inherit pkgs;
|
||||||
|
oidcwarden-src = inputs.oidcwarden;
|
||||||
|
};
|
||||||
in
|
in
|
||||||
{
|
{
|
||||||
|
sops.secrets.vaultwarden_env = {
|
||||||
|
owner = "vaultwarden";
|
||||||
|
group = "vaultwarden";
|
||||||
|
};
|
||||||
|
|
||||||
services.vaultwarden = {
|
services.vaultwarden = {
|
||||||
enable = true;
|
enable = true;
|
||||||
environmentFile = config.sops.secrets.vaultwarden_admin_token.path;
|
package = oidcwarden;
|
||||||
|
environmentFile = config.sops.secrets.vaultwarden_env.path;
|
||||||
backupDir = "/var/local/vaultwarden/backup";
|
backupDir = "/var/local/vaultwarden/backup";
|
||||||
|
|
||||||
config = {
|
config = {
|
||||||
DOMAIN = "https://vault.cyperpunk.de"; # "http://${ip}:${toString port}";
|
DOMAIN = "https://vault.cyperpunk.de";
|
||||||
ROCKET_ADDRESS = "0.0.0.0";
|
ROCKET_ADDRESS = "0.0.0.0";
|
||||||
ROCKET_PORT = port;
|
ROCKET_PORT = port;
|
||||||
ROCKET_LOG = "critical";
|
ROCKET_LOG = "critical";
|
||||||
SIGNUPS_ALLOWED = true;
|
SIGNUPS_ALLOWED = false;
|
||||||
WEBSOCKET_ENABLED = true;
|
WEBSOCKET_ENABLED = true;
|
||||||
|
SSO_ENABLED = true;
|
||||||
|
SSO_ONLY = false;
|
||||||
|
SSO_AUTHORITY = "https://auth.cyperpunk.de/oauth2/openid/vaultwarden";
|
||||||
|
SSO_SCOPES = "openid profile email";
|
||||||
|
SSO_PKCE = false;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
sops.secrets.vaultwarden_admin_token = {
|
|
||||||
owner = "vaultwarden";
|
|
||||||
group = "vaultwarden";
|
|
||||||
};
|
|
||||||
|
|
||||||
networking.firewall.allowedTCPPorts = [ port ];
|
networking.firewall.allowedTCPPorts = [ port ];
|
||||||
|
|
||||||
systemd = {
|
systemd = {
|
||||||
|
|||||||
@@ -1,19 +1,24 @@
|
|||||||
|
cachix_auth_token: ENC[AES256_GCM,data:nR7e2ZOA3q5DmkrqFEzINpKFEHVD5nyzc3DQ3QgD42fdyABV+r1Ela3iEcbU8SWj5JMRq8T1r7QxqcYW+VSMsT2cjQV2e4ZrpUmkX2QnhfmLqQBdJLhgNKBnu+x8QGJpQ3j7mG23atJ3BDTYBEKlI8y6wLEgpTX8GIVzHJVwfbqewTX4EfFyh3mVMtxAK9II/w==,iv:CSMcUdsqC97fmu1Po3cRrUj9h51Wv+KaUPfEToE7qVs=,tag:s1XHG2eyZYJJ5xd9CZb+Pw==,type:str]
|
||||||
GROQ_API_KEY: ENC[AES256_GCM,data:OyuC4jfw67sCDa0XBGr78S6pzPV1ruy7KiIqPMgWWcOCVm3Y/khXEYPMjUTGrq9YLOw1MLso0OE=,iv:0y9klMYVtGsqAaLc2JidjZYSLhhbcbWbnBf8sZiC3rM=,tag:r6G2pzZn2d9JIaS+ozKnmg==,type:str]
|
GROQ_API_KEY: ENC[AES256_GCM,data:OyuC4jfw67sCDa0XBGr78S6pzPV1ruy7KiIqPMgWWcOCVm3Y/khXEYPMjUTGrq9YLOw1MLso0OE=,iv:0y9klMYVtGsqAaLc2JidjZYSLhhbcbWbnBf8sZiC3rM=,tag:r6G2pzZn2d9JIaS+ozKnmg==,type:str]
|
||||||
OPENWEATHER_API_KEY: ENC[AES256_GCM,data:bcuLz70u40nZfNgPTaeNRXdR/zjx0SQjwMbMNNFqROI=,iv:VCzse1a1/k1ZDIpFPL1QhjuS6YaDyohWi61JZaoc0Ws=,tag:UJSNyniNNLfGGRY/uiJcRA==,type:str]
|
OPENWEATHER_API_KEY: ENC[AES256_GCM,data:bcuLz70u40nZfNgPTaeNRXdR/zjx0SQjwMbMNNFqROI=,iv:VCzse1a1/k1ZDIpFPL1QhjuS6YaDyohWi61JZaoc0Ws=,tag:UJSNyniNNLfGGRY/uiJcRA==,type:str]
|
||||||
smb_passwd: ENC[AES256_GCM,data:+9RYomcnCZSME5DzuJWTLbS3IGJHhIYWZ5SmsgOn6YQ=,iv:VRPVR7DD+swjeUZKe54XYm3wn/KB4RqvQAyYXQbS+A8=,tag:NnA89efo6HVL0scHgyTZMQ==,type:str]
|
smb_passwd: ENC[AES256_GCM,data:+9RYomcnCZSME5DzuJWTLbS3IGJHhIYWZ5SmsgOn6YQ=,iv:VRPVR7DD+swjeUZKe54XYm3wn/KB4RqvQAyYXQbS+A8=,tag:NnA89efo6HVL0scHgyTZMQ==,type:str]
|
||||||
grafana_secret_key: ENC[AES256_GCM,data:d6tu4kL7flfbdeOYk21zkSRmVe+NvVwd14jgr9Ds0adfgYetN852G25Z8/g=,iv:uWuwGBZVK1syhEfO9nLZUWwa801759tNJx+Pmnz3xeg=,tag:X6/NcdGZHAdIlOwxNPo/Ew==,type:str]
|
grafana_secret_key: ENC[AES256_GCM,data:d6tu4kL7flfbdeOYk21zkSRmVe+NvVwd14jgr9Ds0adfgYetN852G25Z8/g=,iv:uWuwGBZVK1syhEfO9nLZUWwa801759tNJx+Pmnz3xeg=,tag:X6/NcdGZHAdIlOwxNPo/Ew==,type:str]
|
||||||
|
kanidm_grafana_secret: ENC[AES256_GCM,data:rqK3hkpvtg24P4UVV0pFuabUhxoTgtZvK+h8EiSfVyOSrnRy6jtyGvOMMlIOAQDr,iv:I6e++KrpxAFhM8VDQkx8YsrQSvT4brc+Fh1cY5PlwKY=,tag:sEq91Yucb2pUTDbRir+NyA==,type:str]
|
||||||
matrix_macaroon_secret: ENC[AES256_GCM,data:a9nMar+p+FXIsxxSqO/to2OJOvD1erfwLwwBeKOcWBu7xykHxqD+pCmrGhg=,iv:rp4ZDVIlZ7SN1RFHB2CfSV5ISPMl9pC4U8Jgqpz48Qs=,tag:LxmWUZE3mG4acagQmlieag==,type:str]
|
matrix_macaroon_secret: ENC[AES256_GCM,data:a9nMar+p+FXIsxxSqO/to2OJOvD1erfwLwwBeKOcWBu7xykHxqD+pCmrGhg=,iv:rp4ZDVIlZ7SN1RFHB2CfSV5ISPMl9pC4U8Jgqpz48Qs=,tag:LxmWUZE3mG4acagQmlieag==,type:str]
|
||||||
matrix_registration_secret: ENC[AES256_GCM,data:KhKkJZqwE8xk4/tuQ7NYTv/Ot1qCAiy8yUbDyVvRa0H5BT4amCBIdATfR4Q=,iv:HBN+GorT1VpWCVkDugk4UxYLEYKJIoDZh2d+oUDLc8g=,tag:hHus458yVnH0qaQ4u37IZg==,type:str]
|
matrix_registration_secret: ENC[AES256_GCM,data:KhKkJZqwE8xk4/tuQ7NYTv/Ot1qCAiy8yUbDyVvRa0H5BT4amCBIdATfR4Q=,iv:HBN+GorT1VpWCVkDugk4UxYLEYKJIoDZh2d+oUDLc8g=,tag:hHus458yVnH0qaQ4u37IZg==,type:str]
|
||||||
vaultwarden_admin_token: ENC[AES256_GCM,data:yoBs4CaIEJXB5b3PEwTpXFgxpX39hR9A4r9yamwDV7cTSRRp3n3O2VjDKTcI5Vo6RP2QUjcqUqYf98cZ09wDMc+6+oHHJke7+O0FgRgOC0vOQFs4bfZCBJBLxogrGiwtLGkyykR6VYhrT64AN3CbrXflj82OED2Hl8WwEdruBzGIcfnh6FqQowDx6vDR/kXXJHk=,iv:PJQo5V7FaKPQ+GzZNsy3KB+xyjcDKJ1UBHErrqgn/1U=,tag:BRIDJEDOAeToqio/DHMQaA==,type:str]
|
kanidm_synapse_secret: ENC[AES256_GCM,data:F770siYcYLm3RAQ+3epfVTyp5mv0OJfiOdFiHD8CudjceNkkSuXIX7pxQYkhS3VY,iv:hqYMKLS5m+o3leFE0gBS05Npjy9uyqgSe7yJpPzxvQY=,tag:lLjVZ7/iYoIZh06VyF8zSw==,type:str]
|
||||||
|
vaultwarden_env: ENC[AES256_GCM,data:tGAphb8y+9sdo29z1/zRe4IKKKAsKfJZIiRGSDe1Vbpn24ucjnvrDpbuI8AIf145mVckCZ6nsdB88q+XNr4jrywx3b5gGdacZn/wLx6rVpa24VdJOI685nu++CtJjUiPCOiNVRza7MGFStCWTekRGosVWwwjmu8IRHEtywO/+qkVa2tFx9SGxB9ayVHWJVJNI0V3EoWgXp0Sf+gYZvMww0Ew05xeeCdICe9Tz6ExVsI/2feq2QPBgcR4M+g47UrixWf0HmR+WU/HQygSYZ0vxau+88PyfrlOUWBZY0WtETNd0pymVHnvHyFIuBjwa8zg,iv:Oytbdz/9KS21sy2tWAws2DTKTmW8IOJcSso1K1tYm3M=,tag:iWphR6xUegNrCEsqL2o0yQ==,type:str]
|
||||||
flame_phil_password: ENC[AES256_GCM,data:Xy2ixMeRlnzC2gjKGrjfSbz/ee4=,iv:WFuBS8jn7WYRxEDG3XBzCMnm4eNkHQpSs5+GUwq/dcg=,tag:1zzj0eB9/4KrmYAqcxJMlg==,type:str]
|
flame_phil_password: ENC[AES256_GCM,data:Xy2ixMeRlnzC2gjKGrjfSbz/ee4=,iv:WFuBS8jn7WYRxEDG3XBzCMnm4eNkHQpSs5+GUwq/dcg=,tag:1zzj0eB9/4KrmYAqcxJMlg==,type:str]
|
||||||
flame_calvin_password: ENC[AES256_GCM,data:P5ppyqTjAJ1TL4hXtx5WyoS9a+g=,iv:sq98P3Oqud2FXfqsD76YS/p5NEF2xlN0MfG+ukCB9B0=,tag:AeKnu4Hg4xQ3tII0y6oNpQ==,type:str]
|
flame_calvin_password: ENC[AES256_GCM,data:P5ppyqTjAJ1TL4hXtx5WyoS9a+g=,iv:sq98P3Oqud2FXfqsD76YS/p5NEF2xlN0MfG+ukCB9B0=,tag:AeKnu4Hg4xQ3tII0y6oNpQ==,type:str]
|
||||||
paperless_admin: ENC[AES256_GCM,data:sVvlMQ3dDE2XsDfpwpCTbzPCEKdUMNTFtRXDIuBbgyf1gd6oiJzE23Ytc57plNUGg5h5aEtgxZ7NXeuK5vrhQw==,iv:x+QNAzY9k9t23UYlM9GcAke0urEA5jlV0VzHaBQkm7M=,tag:D/bMtjuwrX6pquZfJLwdkQ==,type:str]
|
paperless_admin: ENC[AES256_GCM,data:sVvlMQ3dDE2XsDfpwpCTbzPCEKdUMNTFtRXDIuBbgyf1gd6oiJzE23Ytc57plNUGg5h5aEtgxZ7NXeuK5vrhQw==,iv:x+QNAzY9k9t23UYlM9GcAke0urEA5jlV0VzHaBQkm7M=,tag:D/bMtjuwrX6pquZfJLwdkQ==,type:str]
|
||||||
|
paperless_oidc_secret: ENC[AES256_GCM,data:+oCeHlky5FgUR8Toue1iQiTaqVX3ZgenEv74S5Cg6XHBObyDLUgUkZNYKj7p88nR,iv:29CBYlOjpyegGwEl5lSlAirkCWtG0+4oOupOTfeI1yM=,tag:KckKGdFa9tNwigWEukLN3w==,type:str]
|
||||||
livekit_key_file: ENC[AES256_GCM,data:wOtJhwDtZNEY+QjHyLL1FTOtkmzkNA5BoTsx0+ZMij9uUaKC28uFIkMAq2ZzIU7Nyvk8+4YjbK/Rrsoy,iv:UTDuItr0XsG+/4HFkEHDpxXy41QiVgPCisHeMMY2dQo=,tag:SkoeLWClO9I/V2sn27Y2uw==,type:str]
|
livekit_key_file: ENC[AES256_GCM,data:wOtJhwDtZNEY+QjHyLL1FTOtkmzkNA5BoTsx0+ZMij9uUaKC28uFIkMAq2ZzIU7Nyvk8+4YjbK/Rrsoy,iv:UTDuItr0XsG+/4HFkEHDpxXy41QiVgPCisHeMMY2dQo=,tag:SkoeLWClO9I/V2sn27Y2uw==,type:str]
|
||||||
mjolnir_access_token: ENC[AES256_GCM,data:vvrAY9CAkEIGEzah+TQiwa6PahGuXVvU7wzBpTnqeSLqe3mqtw120GRj,iv:J+/VJ40BsImr832eGUHShhDVWYC7KsEwQUH9AE6Rs9c=,tag:n+y0flxfqY47rB4yv9TnBw==,type:str]
|
mjolnir_access_token: ENC[AES256_GCM,data:vvrAY9CAkEIGEzah+TQiwa6PahGuXVvU7wzBpTnqeSLqe3mqtw120GRj,iv:J+/VJ40BsImr832eGUHShhDVWYC7KsEwQUH9AE6Rs9c=,tag:n+y0flxfqY47rB4yv9TnBw==,type:str]
|
||||||
coturn_static_auth_secret: ENC[AES256_GCM,data:7AI0E8Hu4WxI5q4j1GqBMSQ+evE006uPMtwIfGn4eFz+XB2JA6fhhiGMPPxSkqOyK+3eZJ5ahiG05JpmBmmAbw==,iv:hQJQQDVo43U7lvV754PC1THeFCpZZEyag+BslXyoDos=,tag:Vkm+IXr1h8ZNpah6UYaKng==,type:str]
|
coturn_static_auth_secret: ENC[AES256_GCM,data:7AI0E8Hu4WxI5q4j1GqBMSQ+evE006uPMtwIfGn4eFz+XB2JA6fhhiGMPPxSkqOyK+3eZJ5ahiG05JpmBmmAbw==,iv:hQJQQDVo43U7lvV754PC1THeFCpZZEyag+BslXyoDos=,tag:Vkm+IXr1h8ZNpah6UYaKng==,type:str]
|
||||||
discord_bot_token: ENC[AES256_GCM,data:j37Qo3FCyRwNFqWSWpnQKCs+AxH5HlQ8U5If7ylHilQoORp8Pb3TtNETTJSjZyvUXllldevAbHrbAEEKnNfoUJx1U8/wl6H0,iv:WQqxFXTE+0LIB2lSvVcnr4LNXPE7uzNc0Kk8NU6Z/aE=,tag:fNeQLhoThEgfa4sSGKLZCw==,type:str]
|
discord_bot_token: ENC[AES256_GCM,data:j37Qo3FCyRwNFqWSWpnQKCs+AxH5HlQ8U5If7ylHilQoORp8Pb3TtNETTJSjZyvUXllldevAbHrbAEEKnNfoUJx1U8/wl6H0,iv:WQqxFXTE+0LIB2lSvVcnr4LNXPE7uzNc0Kk8NU6Z/aE=,tag:fNeQLhoThEgfa4sSGKLZCw==,type:str]
|
||||||
discord_client_id: ENC[AES256_GCM,data:U/iUKXT6Nsl6LRN9lPh1xaIaqw==,iv:k7kQ8rJBrMs3YwD9aDfZ6qhd7H3aVsSPTOwEIxVTw2Y=,tag:2wKhxGbf+P+h3BYeWUSczA==,type:str]
|
discord_client_id: ENC[AES256_GCM,data:U/iUKXT6Nsl6LRN9lPh1xaIaqw==,iv:k7kQ8rJBrMs3YwD9aDfZ6qhd7H3aVsSPTOwEIxVTw2Y=,tag:2wKhxGbf+P+h3BYeWUSczA==,type:str]
|
||||||
pg_replication_password: ENC[AES256_GCM,data:w2h07D+j3LNkcbvoKQ2Qp3HSvC2Wf5HRAPAo/HNhmUkHBOaDyILNxo7IDjqajv0jytpG7q4joCJQhS7tEUlA9Q==,iv:26ZurAq61IDqGdAl0yPpoTJElo93hJJIEUlza4DGDNc=,tag:a46FOKgeqEEZE+rC+H9NbQ==,type:str]
|
pg_replication_password: ENC[AES256_GCM,data:w2h07D+j3LNkcbvoKQ2Qp3HSvC2Wf5HRAPAo/HNhmUkHBOaDyILNxo7IDjqajv0jytpG7q4joCJQhS7tEUlA9Q==,iv:26ZurAq61IDqGdAl0yPpoTJElo93hJJIEUlza4DGDNc=,tag:a46FOKgeqEEZE+rC+H9NbQ==,type:str]
|
||||||
|
kanidm_gitea_secret: ENC[AES256_GCM,data:RavtSb5BaJGwwLB/oGzG/KK2AyV+IzEjihVxnD3/dVnxmxcG+CITIYPLvFUJjmvY,iv:Cg8dAhtJXDvRGULIkpWAyuhhlLEdvN+4lyjHPR/740I=,tag:8kMGrOjXEA4GWSLlP7oIkA==,type:str]
|
||||||
gitea:
|
gitea:
|
||||||
dbPassword: ENC[AES256_GCM,data:S6VvRgkdYk1AzXljyQEEq68UJ9zrFy6+INBMIAspXNcqcM6o+es19o0mcXA=,iv:/pHYpkZZq+9Md+75uSCb2YXfSvaDzUh6mMfH53wb7eg=,tag:ZnbyCQwrK2JnbO5HFqgJYw==,type:str]
|
dbPassword: ENC[AES256_GCM,data:S6VvRgkdYk1AzXljyQEEq68UJ9zrFy6+INBMIAspXNcqcM6o+es19o0mcXA=,iv:/pHYpkZZq+9Md+75uSCb2YXfSvaDzUh6mMfH53wb7eg=,tag:ZnbyCQwrK2JnbO5HFqgJYw==,type:str]
|
||||||
internalToken: ENC[AES256_GCM,data:7N8TkPNb1YdCk2uAcCvVd2pKRVOf85//DYxAvz0UCg1E8ccEI5630xVyKafDFiSTM4ER7xiYelartzXL0jLWSf3QNOjSHUP8TIAz4bJRAZUJPxO917bURSLGGe7WEOfONzqy3Ts5QhrJ,iv:DiIs1ytlwLvqD/Ejep6m2fmpSqdFZkxBcgLNt6+29jY=,tag:8jsEcOkH0p+1mP9cnVjiDQ==,type:str]
|
internalToken: ENC[AES256_GCM,data:7N8TkPNb1YdCk2uAcCvVd2pKRVOf85//DYxAvz0UCg1E8ccEI5630xVyKafDFiSTM4ER7xiYelartzXL0jLWSf3QNOjSHUP8TIAz4bJRAZUJPxO917bURSLGGe7WEOfONzqy3Ts5QhrJ,iv:DiIs1ytlwLvqD/Ejep6m2fmpSqdFZkxBcgLNt6+29jY=,tag:8jsEcOkH0p+1mP9cnVjiDQ==,type:str]
|
||||||
@@ -32,7 +37,7 @@ sops:
|
|||||||
N3I5dzUwc3JtYzczMUhyT04vSHlZamMKT+FzYcDLmlEFYxm/XoBpJb8XaZzBH1v9
|
N3I5dzUwc3JtYzczMUhyT04vSHlZamMKT+FzYcDLmlEFYxm/XoBpJb8XaZzBH1v9
|
||||||
6fuez+zApathZfl14w41kAUojPWBznnxDqYtNvzVVLXwnpp3BMx+7w==
|
6fuez+zApathZfl14w41kAUojPWBznnxDqYtNvzVVLXwnpp3BMx+7w==
|
||||||
-----END AGE ENCRYPTED FILE-----
|
-----END AGE ENCRYPTED FILE-----
|
||||||
lastmodified: "2026-05-09T09:12:42Z"
|
lastmodified: "2026-05-15T12:46:25Z"
|
||||||
mac: ENC[AES256_GCM,data:sTVJcBb8cBzixOBQNlx44/m8W3smfwP5fhmnm9hlr5iwMuPJ7JeKTUqqlQaeL4RX/MpEuLc+Rm4thromJ11M/aA5yiqgWOY7vn8xYPoScGzx6HfV1cRJTofmrWmpxrDICQULwOaO+c8vwFBPy7fVqF/AacRtejx5sEOxsMzrYR8=,iv:/Fc5//8coI/rdQIyGcxCTgXPzOS9xNd0ChDHNs4yffw=,tag:8w6bbZcWMBZQWkujhXQY0w==,type:str]
|
mac: ENC[AES256_GCM,data:KeBccgMJ2PLvLjKqnGcZTdVCkR60XO/H7Wy+3JvucPqRP+vdjXvSGzadgJ/d+ML04ytKk5Ffp0APnK//CLaiD+mcKlwAavtI4qKuvhtssxwsDbzYwvI/TsxFUD+BIVnvxUMDNfI//Wx1qSLb5jDdUWmiexF+bFtCwHfZFsEh4Sg=,iv:auXVwnkaCGpZcr8Jx1GSEdxab2/Y6jJhdfD4wjGBHBM=,tag:GCG7gWeAJ5nJCyY2ofZBGA==,type:str]
|
||||||
unencrypted_suffix: _unencrypted
|
unencrypted_suffix: _unencrypted
|
||||||
version: 3.12.2
|
version: 3.12.2
|
||||||
|
|||||||
Reference in New Issue
Block a user