Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c0d5e848b8 | |||
| 65a717698b | |||
| df817a0b1b |
@@ -1,7 +1,6 @@
|
|||||||
name: pytest
|
name: pytest
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ jobs:
|
|||||||
lockfile:
|
lockfile:
|
||||||
runs-on: self-hosted
|
runs-on: self-hosted
|
||||||
permissions:
|
permissions:
|
||||||
actions: write
|
|
||||||
contents: write
|
contents: write
|
||||||
pull-requests: write
|
pull-requests: write
|
||||||
steps:
|
steps:
|
||||||
|
|||||||
Generated
+18
-18
@@ -8,11 +8,11 @@
|
|||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"dir": "pkgs/firefox-addons",
|
"dir": "pkgs/firefox-addons",
|
||||||
"lastModified": 1781270820,
|
"lastModified": 1780027372,
|
||||||
"narHash": "sha256-KuDdN/p4UtqUb1wnjo8a/YougmRXEaKvUoGHkYnuAq0=",
|
"narHash": "sha256-LQ3CUdVZoKQqWzS2eEpY0rp9bJuzqydNFJUiJ6De9r8=",
|
||||||
"owner": "rycee",
|
"owner": "rycee",
|
||||||
"repo": "nur-expressions",
|
"repo": "nur-expressions",
|
||||||
"rev": "5f4259c0c832a93e13c5ec481d3318ce1394a8f9",
|
"rev": "ef18b76eabdf4f9b2ce8e99e78ce698923693300",
|
||||||
"type": "gitlab"
|
"type": "gitlab"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -29,11 +29,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1781305496,
|
"lastModified": 1780099287,
|
||||||
"narHash": "sha256-g8Vv4Qfc7n+lgov97REu3X6BeJtvYY0hlSUZR1GrGQQ=",
|
"narHash": "sha256-efIPwVGtIWIjWcznhaop6XN6HxnOL8800hF6CBNvlqQ=",
|
||||||
"owner": "nix-community",
|
"owner": "nix-community",
|
||||||
"repo": "home-manager",
|
"repo": "home-manager",
|
||||||
"rev": "c87a39aa979acc4848016d2220c6238390d84779",
|
"rev": "7d8127d308c3fb9664f7e643eec944be74ebb37d",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -47,11 +47,11 @@
|
|||||||
"nixpkgs": "nixpkgs"
|
"nixpkgs": "nixpkgs"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1781168557,
|
"lastModified": 1780065812,
|
||||||
"narHash": "sha256-LOnLQ2tpYF9gqIDDr3+j3DbpJJr/QCH6zPRT2GzEUOE=",
|
"narHash": "sha256-SCSLUKBmwlSLGQ8Xbr8PjRFtiHNk0l9ktqkcmqdBkfE=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixos-hardware",
|
"repo": "nixos-hardware",
|
||||||
"rev": "6358ff76821101c178e3ab4919a62799bfe3652e",
|
"rev": "b76b5639c0593e0aeb0b5879ad62d4b30596c144",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -76,11 +76,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs-master": {
|
"nixpkgs-master": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1781306963,
|
"lastModified": 1780101106,
|
||||||
"narHash": "sha256-qWZ+XEwsf8pN4DnYyyLbmAyj1a74gOO/VNCff44MuB4=",
|
"narHash": "sha256-VcvUdRb9rzKBbF6oMaMiAt+6HZQ1gom9b2dUybhVTVY=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "1eee1a273e2e3fdc3db5b1a4a66f06cd5749db69",
|
"rev": "26b82d423c4f6fda4a8015182516c938f8104337",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -108,11 +108,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs_2": {
|
"nixpkgs_2": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1781074563,
|
"lastModified": 1779560665,
|
||||||
"narHash": "sha256-md8WlXOlfnIeHeOScMTTHFyf2d6iaTwPl2apR5EQ3P4=",
|
"narHash": "sha256-tpyBcxPpcQb8ukyNF7DoCwfSY3VPsxHoYwj00Cayv5o=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "9ae611a455b90cf061d8f332b977e387bda8e1ca",
|
"rev": "64c08a7ca051951c8eae34e3e3cb1e202fe36786",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@@ -141,11 +141,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1780547341,
|
"lastModified": 1777944972,
|
||||||
"narHash": "sha256-Gq8KNx5A7hBB3uGJaj6eQfLDIz5YdLu92gqBcvHvoUo=",
|
"narHash": "sha256-VfGRo1qTBKOe3s2gOv8LSoA6Fk19PvBlwQ1ECN0Evn8=",
|
||||||
"owner": "Mic92",
|
"owner": "Mic92",
|
||||||
"repo": "sops-nix",
|
"repo": "sops-nix",
|
||||||
"rev": "9ed65852b6257fbeae4355bc24ecfea307ca759a",
|
"rev": "c591bf665727040c6cc5cb409079acb22dcce33c",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
|||||||
@@ -4,12 +4,10 @@ from __future__ import annotations
|
|||||||
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import Self
|
from typing import Self
|
||||||
from urllib.parse import quote
|
|
||||||
|
|
||||||
import httpx
|
import httpx
|
||||||
|
|
||||||
DEFAULT_PAGE_SIZE = 100
|
DEFAULT_PAGE_SIZE = 100
|
||||||
EXPECTED_NO_CONTENT = 204
|
|
||||||
EXPECTED_CREATED = 201
|
EXPECTED_CREATED = 201
|
||||||
EXPECTED_OK = 200
|
EXPECTED_OK = 200
|
||||||
|
|
||||||
@@ -224,16 +222,6 @@ class GiteaClient:
|
|||||||
json=payload,
|
json=payload,
|
||||||
)
|
)
|
||||||
|
|
||||||
def dispatch_workflow(self, *, owner: str, repo: str, workflow_id: str, ref: str) -> None:
|
|
||||||
"""Trigger a workflow_dispatch run."""
|
|
||||||
workflow_path = quote(workflow_id, safe="")
|
|
||||||
self._request(
|
|
||||||
"POST",
|
|
||||||
f"/api/v1/repos/{owner}/{repo}/actions/workflows/{workflow_path}/dispatches",
|
|
||||||
expected_statuses={EXPECTED_OK, EXPECTED_NO_CONTENT},
|
|
||||||
json={"ref": ref},
|
|
||||||
)
|
|
||||||
|
|
||||||
def list_run_jobs(self, *, owner: str, repo: str, run_id: str | int) -> list[WorkflowJob]:
|
def list_run_jobs(self, *, owner: str, repo: str, run_id: str | int) -> list[WorkflowJob]:
|
||||||
"""List workflow jobs for a specific run."""
|
"""List workflow jobs for a specific run."""
|
||||||
jobs: list[WorkflowJob] = []
|
jobs: list[WorkflowJob] = []
|
||||||
|
|||||||
@@ -14,7 +14,6 @@ DEFAULT_BASE_BRANCH = "main"
|
|||||||
DEFAULT_BRANCH = "automation/update-flake-lock"
|
DEFAULT_BRANCH = "automation/update-flake-lock"
|
||||||
DEFAULT_GITEA_URL = "https://gitea.tmmworkshop.com"
|
DEFAULT_GITEA_URL = "https://gitea.tmmworkshop.com"
|
||||||
PR_LABELS = ["dependencies", "automated", "flake_lock_update"]
|
PR_LABELS = ["dependencies", "automated", "flake_lock_update"]
|
||||||
PR_CHECK_WORKFLOWS = ["build_systems.yml", "treefmt.yml", "pytest.yml"]
|
|
||||||
PR_TITLE = "Update flake.lock"
|
PR_TITLE = "Update flake.lock"
|
||||||
PR_BODY = "Automated flake.lock update."
|
PR_BODY = "Automated flake.lock update."
|
||||||
|
|
||||||
@@ -58,12 +57,6 @@ def find_flake_lock_pull_request(client: GiteaClient, *, owner: str, repo: str)
|
|||||||
return pull_requests[0]
|
return pull_requests[0]
|
||||||
|
|
||||||
|
|
||||||
def dispatch_pull_request_checks(client: GiteaClient, *, owner: str, repo: str, branch: str) -> None:
|
|
||||||
"""Dispatch the workflows that normally run for pull requests."""
|
|
||||||
for workflow in PR_CHECK_WORKFLOWS:
|
|
||||||
client.dispatch_workflow(owner=owner, repo=repo, workflow_id=workflow, ref=branch)
|
|
||||||
|
|
||||||
|
|
||||||
def has_worktree_changes() -> bool:
|
def has_worktree_changes() -> bool:
|
||||||
"""Return whether `flake.lock` has worktree changes."""
|
"""Return whether `flake.lock` has worktree changes."""
|
||||||
result = run_cmd(["git", "diff", "--quiet", "--", "flake.lock"], check=False)
|
result = run_cmd(["git", "diff", "--quiet", "--", "flake.lock"], check=False)
|
||||||
@@ -120,9 +113,6 @@ def update(
|
|||||||
branch=branch,
|
branch=branch,
|
||||||
base=base,
|
base=base,
|
||||||
)
|
)
|
||||||
# We can remove this if Gitea fixes the following issue:
|
|
||||||
# https://github.com/go-gitea/gitea/issues/33963
|
|
||||||
dispatch_pull_request_checks(client, owner=owner, repo=repo_name, branch=branch)
|
|
||||||
typer.echo(pull_request.html_url or f"Pull request #{pull_request.number}")
|
typer.echo(pull_request.html_url or f"Pull request #{pull_request.number}")
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -28,12 +28,7 @@
|
|||||||
networking = {
|
networking = {
|
||||||
hostName = "bob";
|
hostName = "bob";
|
||||||
hostId = "7c678a41";
|
hostId = "7c678a41";
|
||||||
firewall = {
|
firewall.enable = true;
|
||||||
enable = true;
|
|
||||||
allowedTCPPorts = [
|
|
||||||
8000
|
|
||||||
];
|
|
||||||
};
|
|
||||||
networkmanager.enable = true;
|
networkmanager.enable = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -30,11 +30,6 @@
|
|||||||
keyFile = "/dev/disk/by-id/usb-Samsung_Flash_Drive_FIT_0374620080067131-0:0";
|
keyFile = "/dev/disk/by-id/usb-Samsung_Flash_Drive_FIT_0374620080067131-0:0";
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
zfs.extraPools = [
|
|
||||||
"storage"
|
|
||||||
];
|
|
||||||
|
|
||||||
kernelModules = [ "kvm-amd" ];
|
kernelModules = [ "kvm-amd" ];
|
||||||
extraModulePackages = [ ];
|
extraModulePackages = [ ];
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,113 +0,0 @@
|
|||||||
"""Tests for Gitea flake.lock automation."""
|
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from python.gitea import PullRequest
|
|
||||||
from python.gitea_flake_lock import (
|
|
||||||
PR_CHECK_WORKFLOWS,
|
|
||||||
PR_LABELS,
|
|
||||||
dispatch_pull_request_checks,
|
|
||||||
ensure_flake_lock_pull_request,
|
|
||||||
find_flake_lock_pull_request,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def _pull_request(number=1, head_branch="automation/update-flake-lock"):
|
|
||||||
return PullRequest(
|
|
||||||
number=number,
|
|
||||||
title="Update flake.lock",
|
|
||||||
html_url=f"https://gitea.example.test/pulls/{number}",
|
|
||||||
labels=(),
|
|
||||||
head_branch=head_branch,
|
|
||||||
base_branch="main",
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class FakeGiteaClient:
|
|
||||||
def __init__(self, pull_requests=None):
|
|
||||||
self.pull_requests = pull_requests or []
|
|
||||||
self.dispatch_calls = []
|
|
||||||
self.list_calls = []
|
|
||||||
self.create_calls = []
|
|
||||||
|
|
||||||
def list_open_pull_requests(self, **kwargs):
|
|
||||||
self.list_calls.append(kwargs)
|
|
||||||
return self.pull_requests
|
|
||||||
|
|
||||||
def create_pull_request(self, **kwargs):
|
|
||||||
self.create_calls.append(kwargs)
|
|
||||||
return _pull_request()
|
|
||||||
|
|
||||||
def dispatch_workflow(self, **kwargs):
|
|
||||||
self.dispatch_calls.append(kwargs)
|
|
||||||
|
|
||||||
|
|
||||||
def test_ensure_flake_lock_pull_request_finds_by_branch():
|
|
||||||
pull_request = _pull_request()
|
|
||||||
client = FakeGiteaClient([pull_request])
|
|
||||||
|
|
||||||
result = ensure_flake_lock_pull_request(
|
|
||||||
client,
|
|
||||||
owner="Richie",
|
|
||||||
repo="dotfiles",
|
|
||||||
branch="automation/update-flake-lock",
|
|
||||||
base="main",
|
|
||||||
)
|
|
||||||
|
|
||||||
assert result == pull_request
|
|
||||||
assert client.list_calls == [
|
|
||||||
{"owner": "Richie", "repo": "dotfiles", "head": "automation/update-flake-lock"},
|
|
||||||
]
|
|
||||||
assert client.create_calls == []
|
|
||||||
|
|
||||||
|
|
||||||
def test_ensure_flake_lock_pull_request_creates_with_labels():
|
|
||||||
client = FakeGiteaClient()
|
|
||||||
|
|
||||||
ensure_flake_lock_pull_request(
|
|
||||||
client,
|
|
||||||
owner="Richie",
|
|
||||||
repo="dotfiles",
|
|
||||||
branch="automation/update-flake-lock",
|
|
||||||
base="main",
|
|
||||||
)
|
|
||||||
|
|
||||||
assert client.create_calls == [
|
|
||||||
{
|
|
||||||
"owner": "Richie",
|
|
||||||
"repo": "dotfiles",
|
|
||||||
"title": "Update flake.lock",
|
|
||||||
"body": "Automated flake.lock update.",
|
|
||||||
"head": "automation/update-flake-lock",
|
|
||||||
"base": "main",
|
|
||||||
"labels": PR_LABELS,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
def test_find_flake_lock_pull_request_finds_by_label():
|
|
||||||
pull_request = _pull_request()
|
|
||||||
client = FakeGiteaClient([pull_request])
|
|
||||||
|
|
||||||
result = find_flake_lock_pull_request(client, owner="Richie", repo="dotfiles")
|
|
||||||
|
|
||||||
assert result == pull_request
|
|
||||||
assert client.list_calls == [
|
|
||||||
{"owner": "Richie", "repo": "dotfiles", "labels": ["flake_lock_update"]},
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
def test_dispatch_pull_request_checks_runs_each_workflow():
|
|
||||||
client = FakeGiteaClient()
|
|
||||||
|
|
||||||
dispatch_pull_request_checks(client, owner="Richie", repo="dotfiles", branch="automation/update-flake-lock")
|
|
||||||
|
|
||||||
assert client.dispatch_calls == [
|
|
||||||
{
|
|
||||||
"owner": "Richie",
|
|
||||||
"repo": "dotfiles",
|
|
||||||
"workflow_id": workflow,
|
|
||||||
"ref": "automation/update-flake-lock",
|
|
||||||
}
|
|
||||||
for workflow in PR_CHECK_WORKFLOWS
|
|
||||||
]
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
{ pkgs, ... }:
|
{ pkgs, ... }:
|
||||||
{
|
{
|
||||||
home.packages = [
|
home.packages = [
|
||||||
pkgs.master.claude-code
|
pkgs.claude-code
|
||||||
pkgs.master.codex
|
pkgs.master.codex
|
||||||
pkgs.master.opencode
|
pkgs.master.opencode
|
||||||
pkgs.master.pi-coding-agent
|
pkgs.master.pi-coding-agent
|
||||||
|
|||||||
@@ -2,46 +2,46 @@
|
|||||||
programs.ssh = {
|
programs.ssh = {
|
||||||
enable = true;
|
enable = true;
|
||||||
enableDefaultConfig = false;
|
enableDefaultConfig = false;
|
||||||
settings = {
|
matchBlocks = {
|
||||||
jeeves = {
|
jeeves = {
|
||||||
HostName = "192.168.90.40";
|
hostname = "192.168.90.40";
|
||||||
User = "richie";
|
user = "richie";
|
||||||
IdentityFile = "~/.ssh/id_ed25519";
|
identityFile = "~/.ssh/id_ed25519";
|
||||||
Port = 629;
|
port = 629;
|
||||||
DynamicForward = [ { port = 9050; } ];
|
dynamicForwards = [ { port = 9050; } ];
|
||||||
Compression = true;
|
compression = true;
|
||||||
};
|
};
|
||||||
unlock-jeeves = {
|
unlock-jeeves = {
|
||||||
HostName = "192.168.99.14";
|
hostname = "192.168.99.14";
|
||||||
User = "root";
|
user = "root";
|
||||||
IdentityFile = "~/.ssh/id_ed25519";
|
identityFile = "~/.ssh/id_ed25519";
|
||||||
Port = 2222;
|
port = 2222;
|
||||||
};
|
};
|
||||||
brain = {
|
brain = {
|
||||||
HostName = "192.168.90.35";
|
hostname = "192.168.90.35";
|
||||||
User = "richie";
|
user = "richie";
|
||||||
IdentityFile = "~/.ssh/id_ed25519";
|
identityFile = "~/.ssh/id_ed25519";
|
||||||
Port = 129;
|
port = 129;
|
||||||
DynamicForward = [ { port = 9050; } ];
|
dynamicForwards = [ { port = 9050; } ];
|
||||||
};
|
};
|
||||||
unlock-brain = {
|
unlock-brain = {
|
||||||
HostName = "192.168.95.35";
|
hostname = "192.168.95.35";
|
||||||
User = "root";
|
user = "root";
|
||||||
IdentityFile = "~/.ssh/id_ed25519";
|
identityFile = "~/.ssh/id_ed25519";
|
||||||
Port = 2222;
|
port = 2222;
|
||||||
};
|
};
|
||||||
bob = {
|
bob = {
|
||||||
HostName = "192.168.90.25";
|
hostname = "192.168.90.25";
|
||||||
User = "richie";
|
user = "richie";
|
||||||
IdentityFile = "~/.ssh/id_ed25519";
|
identityFile = "~/.ssh/id_ed25519";
|
||||||
Port = 262;
|
port = 262;
|
||||||
DynamicForward = [ { port = 9050; } ];
|
dynamicForwards = [ { port = 9050; } ];
|
||||||
};
|
};
|
||||||
rhapsody-in-green = {
|
rhapsody-in-green = {
|
||||||
HostName = "192.168.90.221";
|
hostname = "192.168.90.221";
|
||||||
User = "richie";
|
user = "richie";
|
||||||
IdentityFile = "~/.ssh/id_ed25519";
|
identityFile = "~/.ssh/id_ed25519";
|
||||||
Port = 922;
|
port = 922;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user