feat(dev): deploy claude-code config via systemd instead of home-manager
Home-manager fails on the dev microVM because nix-env --set needs writable nix state dirs, but the microVM shares /nix/store read-only via virtiofs. Extract shared claude-code settings into settings.nix, add a NixOS module (nixos.nix) that deploys the same files via a systemd oneshot service with RequiresMountsFor to handle virtiofs mount ordering. The nb host continues using home-manager unchanged.
This commit is contained in:
parent
4648d6b51a
commit
248534bc35
7 changed files with 114 additions and 1 deletions
22
utils/home-manager/claude-code/default.nix
Normal file
22
utils/home-manager/claude-code/default.nix
Normal file
|
|
@ -0,0 +1,22 @@
|
|||
{ config, lib, pkgs, ... }:
|
||||
let
|
||||
settings = import ./settings.nix { homeDir = config.home.homeDirectory; };
|
||||
in
|
||||
{
|
||||
home.file = {
|
||||
# Agents
|
||||
".claude/agents/devil-advocate.md".source = ./agents/devil-advocate.md;
|
||||
".claude/agents/lint-fixer.md".source = ./agents/lint-fixer.md;
|
||||
".claude/agents/secret-scanner.md".source = ./agents/secret-scanner.md;
|
||||
".claude/agents/test-runner.md".source = ./agents/test-runner.md;
|
||||
|
||||
# Statusline script
|
||||
".claude/statusline-command.sh" = {
|
||||
source = ./statusline-command.sh;
|
||||
executable = true;
|
||||
};
|
||||
|
||||
# Settings (local override — leaves settings.json writable for Claude)
|
||||
".claude/settings.local.json".text = builtins.toJSON settings;
|
||||
};
|
||||
}
|
||||
35
utils/home-manager/claude-code/nixos.nix
Normal file
35
utils/home-manager/claude-code/nixos.nix
Normal file
|
|
@ -0,0 +1,35 @@
|
|||
{ pkgs, ... }:
|
||||
let
|
||||
agentsDir = ./agents;
|
||||
statuslineScript = ./statusline-command.sh;
|
||||
settings = import ./settings.nix { homeDir = "/home/dominik"; };
|
||||
settingsJson = pkgs.writeText "claude-settings-local.json" (builtins.toJSON settings);
|
||||
|
||||
deployScript = pkgs.writeShellScript "deploy-claude-code" ''
|
||||
install -d -m 755 -o 1000 -g 100 /home/dominik/.claude
|
||||
install -d -m 755 -o 1000 -g 100 /home/dominik/.claude/agents
|
||||
install -m 644 -o 1000 -g 100 ${agentsDir}/devil-advocate.md /home/dominik/.claude/agents/
|
||||
install -m 644 -o 1000 -g 100 ${agentsDir}/lint-fixer.md /home/dominik/.claude/agents/
|
||||
install -m 644 -o 1000 -g 100 ${agentsDir}/secret-scanner.md /home/dominik/.claude/agents/
|
||||
install -m 644 -o 1000 -g 100 ${agentsDir}/test-runner.md /home/dominik/.claude/agents/
|
||||
install -m 755 -o 1000 -g 100 ${statuslineScript} /home/dominik/.claude/statusline-command.sh
|
||||
install -m 644 -o 1000 -g 100 ${settingsJson} /home/dominik/.claude/settings.local.json
|
||||
'';
|
||||
in
|
||||
{
|
||||
# Deploy claude-code config files via a systemd service instead of home-manager.
|
||||
# This avoids the nix-env --set call that fails on microVMs with read-only /nix/store.
|
||||
systemd.services.claude-code-dominik = {
|
||||
description = "Deploy Claude Code config for dominik";
|
||||
wantedBy = [ "multi-user.target" ];
|
||||
# Wait for /home to be mounted (virtiofs on microVMs)
|
||||
unitConfig.RequiresMountsFor = "/home/dominik";
|
||||
# Rerun on config changes during nixos-rebuild switch
|
||||
restartTriggers = [ deployScript ];
|
||||
serviceConfig = {
|
||||
Type = "oneshot";
|
||||
RemainAfterExit = true;
|
||||
ExecStart = deployScript;
|
||||
};
|
||||
};
|
||||
}
|
||||
44
utils/home-manager/claude-code/settings.nix
Normal file
44
utils/home-manager/claude-code/settings.nix
Normal file
|
|
@ -0,0 +1,44 @@
|
|||
{ homeDir }:
|
||||
{
|
||||
env = {
|
||||
CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS = "1";
|
||||
};
|
||||
statusLine = {
|
||||
type = "command";
|
||||
command = "${homeDir}/.claude/statusline-command.sh";
|
||||
};
|
||||
hooks.Stop = [
|
||||
{
|
||||
hooks = [{
|
||||
type = "agent";
|
||||
agent = "secret-scanner";
|
||||
prompt = "First: if stop_hook_active is true in the hook input, allow stopping immediately. Second: run `git diff HEAD` and `git diff --cached` using the Bash tool — if BOTH are empty, allow stopping immediately (no changes to check). Otherwise: Scan the diff for accidentally committed secrets. Check .claude/secret-scanner.md for project-specific allowlists. If secrets are found, they must be removed before the session can end. If no secrets found, allow stopping.";
|
||||
timeout = 120;
|
||||
}];
|
||||
}
|
||||
{
|
||||
hooks = [{
|
||||
type = "agent";
|
||||
agent = "lint-fixer";
|
||||
prompt = "First: if stop_hook_active is true in the hook input, allow stopping immediately. Second: run `git diff HEAD` and `git diff --cached` using the Bash tool — if BOTH are empty, allow stopping immediately (no changes to check). Otherwise: Run the project's linter/formatter. Check .claude/lint-fixer.md for project-specific config. If that file doesn't exist, auto-detect the linter and run it. Auto-fix what you can, report unfixable errors as blocking. If no linter detected, allow stopping.";
|
||||
timeout = 180;
|
||||
}];
|
||||
}
|
||||
{
|
||||
hooks = [{
|
||||
type = "agent";
|
||||
agent = "test-runner";
|
||||
prompt = "First: if stop_hook_active is true in the hook input, allow stopping immediately. Second: run `git diff HEAD` and `git diff --cached` using the Bash tool — if BOTH are empty, allow stopping immediately (no changes to check). Otherwise: Check if .claude/test-runner.md exists in the current working directory. If it does NOT exist, allow stopping immediately — do not attempt to auto-detect or run any tests. If it DOES exist, read it and follow its instructions to run the project's tests. If tests fail, they must be fixed before the session can end.";
|
||||
timeout = 300;
|
||||
}];
|
||||
}
|
||||
{
|
||||
hooks = [{
|
||||
type = "agent";
|
||||
agent = "devil-advocate";
|
||||
prompt = "First: if stop_hook_active is true in the hook input, allow stopping immediately. Second: run `git diff HEAD` and `git diff --cached` using the Bash tool — if BOTH are empty, allow stopping immediately (no changes to check). Otherwise: Review all code changes. Read the project's .claude/devil-advocate.md for project-specific conventions. Report any CRITICAL or HIGH issues found. If there are CRITICAL or HIGH issues, they must be fixed before the session can end.";
|
||||
timeout = 600;
|
||||
}];
|
||||
}
|
||||
];
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue