Improve hello-world example service

This commit is contained in:
pinpox
2025-08-11 11:37:53 +02:00
parent 073027f7c6
commit e3ed9d7b4b
5 changed files with 115 additions and 29 deletions

View File

@@ -1,3 +1,9 @@
# Example clan service. See https://docs.clan.lol/guides/services/community/
# for more details
# The test for this module in ./tests/vm/default.nix shows an example of how
# the service is used.
{ packages }: { packages }:
{ ... }: { ... }:
{ {
@@ -5,30 +11,94 @@
manifest.name = "clan-core/hello-word"; manifest.name = "clan-core/hello-word";
manifest.description = "This is a test"; manifest.description = "This is a test";
roles.peer = { # This service provides two roles: "morning" and "evening". Roles can be
# defined in this file directly (e.g. the "morning" role) or split up into a
# separate file (e.g. the "evening" role)
roles.morning = {
interface = interface =
{ lib, ... }: { lib, ... }:
{ {
options.foo = lib.mkOption { # Here we define the settings for this role. They will be accessible
# via `roles.morning.settings` in the role
options.greeting = lib.mkOption {
type = lib.types.str; type = lib.types.str;
# default = ""; default = "Good morning";
description = "Some option"; description = "The greeting to use";
}; };
}; };
# Maps over all instances and produces one result per instance.
perInstance =
{
# Role settings for this machine/instance
settings,
# The name of this instance of the service
instanceName,
# The current machine
machine,
# All roles of this service, with their assigned machines
roles,
...
}:
{
# Analog to 'perSystem' of flake-parts.
# For every instance of this service we will add a nixosModule to a morning-machine
nixosModule =
{ config, ... }:
{
# Interaction examples what you could do here:
# - Get some settings of this machine
# settings.ipRanges
#
# - Get all evening names:
# allEveningNames = lib.attrNames roles.evening.machines
#
# - Get all roles of the machine:
# machine.roles
#
# - Get the settings that where applied to a specific evening machine:
# roles.evening.machines.peer1.settings
imports = [ ];
environment.etc.hello.text = "${settings.greeting} World!";
};
};
}; };
# The impnlementation of the evening role is in a separate file. We have kept
# the interface here, so we can see all settings of the service in one place,
# but you can also move it to the respective file
roles.evening = {
interface =
{ lib, ... }:
{
options.greeting = lib.mkOption {
type = lib.types.str;
default = "Good evening";
description = "The greeting to use";
};
};
};
imports = [ ./evening.nix ];
# This part gets applied to all machines, regardless of their role.
perMachine = perMachine =
{ machine, ... }: { machine, ... }:
{ {
nixosModule = { nixosModule =
clan.core.vars.generators.hello = { { pkgs, ... }:
files.hello = { {
secret = false; environment.systemPackages = [
}; (pkgs.writeShellScriptBin "greet-world" ''
script = '' #!${pkgs.bash}/bin/bash
echo "Hello world from ${machine.name}" > $out/hello set -euo pipefail
'';
cat /etc/hello
echo " I'm ${machine.name}"
'')
];
}; };
};
}; };
} }

View File

@@ -0,0 +1,12 @@
{
roles.evening.perInstance =
{ settings, ... }:
{
nixosModule =
{ ... }:
{
imports = [ ];
environment.etc.hello.text = "${settings.greeting} World!";
};
};
}

View File

@@ -27,20 +27,10 @@ let
module.name = "hello-world"; module.name = "hello-world";
module.input = "self"; module.input = "self";
roles.peer.machines.jon = { }; roles.evening.machines.jon = { };
}; };
}; };
}; };
# NOTE:
# If you wonder why 'self-zerotier-redux':
# A local module has prefix 'self', otherwise it is the name of the 'input'
# The rest is the name of the service as in the instance 'module.name';
#
# -> ${module.input}-${module.name}
# In this case it is 'self-zerotier-redux'
# This is usually only used internally, but we can use it to test the evaluation of service module in isolation
# evaluatedService =
# testFlake.clanInternals.inventoryClass.distributedServices.importedModulesEvaluated.self-zerotier-redux.config;
in in
{ {
test_simple = { test_simple = {

View File

@@ -5,22 +5,35 @@
directory = ./.; directory = ./.;
inventory = { inventory = {
machines.peer1 = { }; machines.peer1 = { };
machines.peer2 = { };
instances."test" = { instances."test" = {
module.name = "hello-service"; module.name = "hello-service";
module.input = "self"; module.input = "self";
roles.peer.machines.peer1 = { };
# Assign the roles to the two machines
roles.morning.machines.peer1 = { };
roles.evening.machines.peer2 = {
# Set roles settings for the peers, where we want to differ from
# the role defaults
settings = {
greeting = "Good night";
};
};
}; };
}; };
}; };
testScript = testScript =
{ nodes, ... }: { ... }:
'' ''
start_all() start_all()
# peer1 should have the 'hello' file value = peer1.succeed("greet-world")
value = peer1.succeed("cat ${nodes.peer1.clan.core.vars.generators.hello.files.hello.path}") assert value.strip() == "Good morning World! I'm peer1", value
assert value.strip() == "Hello world from peer1", value
value = peer2.succeed("greet-world")
assert value.strip() == "Good night World! I'm peer2", value
''; '';
} }

View File

@@ -61,6 +61,7 @@
"pkgs/clan-cli/clan_cli/tests/data/sshd_config" "pkgs/clan-cli/clan_cli/tests/data/sshd_config"
"pkgs/clan-vm-manager/.vscode/lhebendanz.weaudit" "pkgs/clan-vm-manager/.vscode/lhebendanz.weaudit"
"pkgs/clan-vm-manager/bin/clan-vm-manager" "pkgs/clan-vm-manager/bin/clan-vm-manager"
"clanServices/hello-world/default.nix"
"sops/secrets/test-backup-age.key/secret" "sops/secrets/test-backup-age.key/secret"
]; ];
treefmt.settings.formatter.ruff-format.includes = [ treefmt.settings.formatter.ruff-format.includes = [