switch to my nix-packages repo
This commit is contained in:
parent
309a493491
commit
53c9540338
5 changed files with 60 additions and 196 deletions
|
@ -1,10 +1,10 @@
|
||||||
{ lib, config, pkgs, ... }:
|
{ system, nix-packages, lib, config, pkgs, ... }:
|
||||||
let
|
let
|
||||||
listenIPs = (import ../../utils/getInternalIP.nix config).listenIPs;
|
listenIPs = (import ../../utils/getInternalIP.nix config).listenIPs;
|
||||||
listenStatements = lib.concatStringsSep "\n" (builtins.map (ip: "listen ${ip}:443 http3;") listenIPs) + ''
|
listenStatements = lib.concatStringsSep "\n" (builtins.map (ip: "listen ${ip}:443 http3;") listenIPs) + ''
|
||||||
add_header Alt-Svc 'h3=":443"';
|
add_header Alt-Svc 'h3=":443"';
|
||||||
'';
|
'';
|
||||||
clean-cache = pkgs.callPackage ../../packages/clean-s3-cache.nix { };
|
clean-cache = nix-packages.packages.${system}.clean-s3-cache;
|
||||||
machines = pkgs.writeText "machines" ''
|
machines = pkgs.writeText "machines" ''
|
||||||
localhost armv7l-linux,aarch64-linux,powerpc-linux,powerpc64-linux,powerpc64le-linux,riscv32-linux,riscv64-linux,wasm32-wasi,x86_64-linux,i686-linux - 12 1 kvm,nixos-test,big-parallel,benchmark,gccarch-znver1,gccarch-skylake,ca-derivations -
|
localhost armv7l-linux,aarch64-linux,powerpc-linux,powerpc64-linux,powerpc64le-linux,riscv32-linux,riscv64-linux,wasm32-wasi,x86_64-linux,i686-linux - 12 1 kvm,nixos-test,big-parallel,benchmark,gccarch-znver1,gccarch-skylake,ca-derivations -
|
||||||
'';
|
'';
|
||||||
|
|
55
flake.lock
55
flake.lock
|
@ -53,6 +53,22 @@
|
||||||
"url": "https://git.chir.rs/darkkirb/chir.rs.git"
|
"url": "https://git.chir.rs/darkkirb/chir.rs.git"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"clean-s3-cache": {
|
||||||
|
"flake": false,
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1654951411,
|
||||||
|
"narHash": "sha256-Om6LZGjdrhsIL2K8VwzyRrX3+fOHJ1Bct4pI7/ijDP4=",
|
||||||
|
"ref": "refs/heads/main",
|
||||||
|
"rev": "e4bef52b5b8b8427d7027c681c96ac52ea007421",
|
||||||
|
"revCount": 1,
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://gist.github.com/DarkKirb/533e9e8b9b6dd1badf9b103b4285dc4e"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://gist.github.com/DarkKirb/533e9e8b9b6dd1badf9b103b4285dc4e"
|
||||||
|
}
|
||||||
|
},
|
||||||
"dns": {
|
"dns": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"flake-utils": "flake-utils_2",
|
"flake-utils": "flake-utils_2",
|
||||||
|
@ -150,6 +166,21 @@
|
||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"flake-utils_4": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1653893745,
|
||||||
|
"narHash": "sha256-0jntwV3Z8//YwuOjzhV2sgJJPt+HY6KhU7VZUL0fKZQ=",
|
||||||
|
"owner": "numtide",
|
||||||
|
"repo": "flake-utils",
|
||||||
|
"rev": "1ed9fb1935d260de5fe1c2f7ee0ebaae17ed2fa1",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "numtide",
|
||||||
|
"repo": "flake-utils",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
"home-manager": {
|
"home-manager": {
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"nixpkgs": [
|
"nixpkgs": [
|
||||||
|
@ -297,6 +328,29 @@
|
||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"nix-packages": {
|
||||||
|
"inputs": {
|
||||||
|
"clean-s3-cache": "clean-s3-cache",
|
||||||
|
"flake-utils": "flake-utils_4",
|
||||||
|
"nixpkgs": [
|
||||||
|
"nixpkgs"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1654952251,
|
||||||
|
"narHash": "sha256-sa7L/6Hy5pIinOqGJBcmfqLJO5M3nKkR84Uah07XbbM=",
|
||||||
|
"owner": "DarkKirb",
|
||||||
|
"repo": "nix-packages",
|
||||||
|
"rev": "6b1156875efc37dc2b74c11a2a7f54228b185b22",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "DarkKirb",
|
||||||
|
"ref": "main",
|
||||||
|
"repo": "nix-packages",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
|
},
|
||||||
"nixos-hardware": {
|
"nixos-hardware": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1654057797,
|
"lastModified": 1654057797,
|
||||||
|
@ -498,6 +552,7 @@
|
||||||
"hydra": "hydra",
|
"hydra": "hydra",
|
||||||
"miifox-net": "miifox-net",
|
"miifox-net": "miifox-net",
|
||||||
"nix": "nix",
|
"nix": "nix",
|
||||||
|
"nix-packages": "nix-packages",
|
||||||
"nixos-hardware": "nixos-hardware",
|
"nixos-hardware": "nixos-hardware",
|
||||||
"nixpkgs": "nixpkgs_4",
|
"nixpkgs": "nixpkgs_4",
|
||||||
"nixpkgs-go116": "nixpkgs-go116",
|
"nixpkgs-go116": "nixpkgs-go116",
|
||||||
|
|
|
@ -25,6 +25,9 @@ rec {
|
||||||
miifox-net.flake = false;
|
miifox-net.flake = false;
|
||||||
nixpkgs-noto-variable.url = github:NixOS/nixpkgs/1988f9a17fc1c2ab11f5817adf34a4eb8d06454d;
|
nixpkgs-noto-variable.url = github:NixOS/nixpkgs/1988f9a17fc1c2ab11f5817adf34a4eb8d06454d;
|
||||||
nixpkgs-go116.url = github:NixOS/nixpkgs/dab5668f6be905a7f0de39a7d67fd8f78a13d600;
|
nixpkgs-go116.url = github:NixOS/nixpkgs/dab5668f6be905a7f0de39a7d67fd8f78a13d600;
|
||||||
|
|
||||||
|
nix-packages.url = github:DarkKirb/nix-packages/main;
|
||||||
|
nix-packages.inputs.nixpkgs.follows = "nixpkgs";
|
||||||
};
|
};
|
||||||
|
|
||||||
outputs = { self, nixpkgs, sops-nix, home-manager, chir-rs, nur, polymc, ... } @ args:
|
outputs = { self, nixpkgs, sops-nix, home-manager, chir-rs, nur, polymc, ... } @ args:
|
||||||
|
|
|
@ -1,12 +0,0 @@
|
||||||
{ writeTextFile, python3, python3Packages }:
|
|
||||||
let environment = python3.buildEnv.override {
|
|
||||||
extraLibs = with python3Packages; [
|
|
||||||
boto3
|
|
||||||
];
|
|
||||||
}; in
|
|
||||||
writeTextFile {
|
|
||||||
name = "clean-s3-cache.py";
|
|
||||||
executable = true;
|
|
||||||
destination = "/bin/clean-s3-cache.py";
|
|
||||||
text = builtins.replaceStrings [ "#SHEBANG#" ] [ "${environment}/bin/python" ] (builtins.readFile ./clean-s3-cache.py);
|
|
||||||
}
|
|
|
@ -1,182 +0,0 @@
|
||||||
#!#SHEBANG#
|
|
||||||
|
|
||||||
import asyncio
|
|
||||||
from concurrent.futures import ThreadPoolExecutor
|
|
||||||
import functools
|
|
||||||
from typing import Any, AsyncIterable, Awaitable, Callable, Optional, TypeVar, cast
|
|
||||||
from os import path, listdir
|
|
||||||
import datetime
|
|
||||||
import json
|
|
||||||
|
|
||||||
import boto3
|
|
||||||
from botocore.response import StreamingBody
|
|
||||||
|
|
||||||
ENDPOINT_URL: str = "https://s3.us-west-000.backblazeb2.com"
|
|
||||||
BUCKET_NAME: str = "cache-chir-rs"
|
|
||||||
|
|
||||||
executor: ThreadPoolExecutor = ThreadPoolExecutor()
|
|
||||||
|
|
||||||
F = TypeVar('F', bound=Callable[..., Any])
|
|
||||||
T = TypeVar('T')
|
|
||||||
|
|
||||||
|
|
||||||
def with_backoff(
|
|
||||||
f: Callable[..., Awaitable[T]]) -> Callable[..., Awaitable[T]]:
|
|
||||||
|
|
||||||
async def with_backoff_wrapper(*args: Any, **kwargs: Any) -> T:
|
|
||||||
last_delay = 2
|
|
||||||
while True:
|
|
||||||
try:
|
|
||||||
return await f(*args, **kwargs)
|
|
||||||
except Exception as e:
|
|
||||||
print(f"{e}")
|
|
||||||
if last_delay >= 120:
|
|
||||||
raise
|
|
||||||
await asyncio.sleep(last_delay)
|
|
||||||
last_delay *= last_delay
|
|
||||||
|
|
||||||
return with_backoff_wrapper
|
|
||||||
|
|
||||||
|
|
||||||
def aio(f: Callable[..., T]) -> Callable[..., Awaitable[T]]:
|
|
||||||
|
|
||||||
async def aio_wrapper(*args: Any, **kwargs: Any) -> T:
|
|
||||||
f_bound: Callable[[], T] = functools.partial(f, *args, **kwargs)
|
|
||||||
loop: asyncio.AbstractEventLoop = asyncio.get_running_loop()
|
|
||||||
return await loop.run_in_executor(executor, f_bound)
|
|
||||||
|
|
||||||
return aio_wrapper
|
|
||||||
|
|
||||||
|
|
||||||
@aio
|
|
||||||
def exists_locally(store_path: str) -> bool:
|
|
||||||
return path.exists(store_path)
|
|
||||||
|
|
||||||
|
|
||||||
class NarInfo(object):
|
|
||||||
|
|
||||||
def __init__(self, narinfo: str) -> None:
|
|
||||||
self.compression = "bzip2"
|
|
||||||
for narinfo_line in narinfo.splitlines():
|
|
||||||
key, value = narinfo_line.split(": ", 1)
|
|
||||||
if key == "StorePath":
|
|
||||||
self.store_path = value
|
|
||||||
elif key == "URL":
|
|
||||||
self.url = value
|
|
||||||
elif key == "Compression":
|
|
||||||
self.compression = value
|
|
||||||
elif key == "FileHash":
|
|
||||||
self.file_hash = value
|
|
||||||
elif key == "FileSize":
|
|
||||||
self.file_size = int(value)
|
|
||||||
elif key == "NarHash":
|
|
||||||
self.nar_hash = value
|
|
||||||
elif key == "NarSize":
|
|
||||||
self.nar_size = int(value)
|
|
||||||
elif key == "References":
|
|
||||||
self.references = value.split()
|
|
||||||
elif key == "Deriver":
|
|
||||||
self.deriver = value
|
|
||||||
elif key == "System":
|
|
||||||
self.system = value
|
|
||||||
elif key == "Sig":
|
|
||||||
self.sig = value
|
|
||||||
elif key == "CA":
|
|
||||||
self.ca = value
|
|
||||||
|
|
||||||
async def exists_locally(self) -> bool:
|
|
||||||
return await exists_locally(self.store_path)
|
|
||||||
|
|
||||||
|
|
||||||
s3 = boto3.client("s3", endpoint_url=ENDPOINT_URL)
|
|
||||||
|
|
||||||
|
|
||||||
@with_backoff
|
|
||||||
@aio
|
|
||||||
def get_object(Key: str) -> str:
|
|
||||||
obj = s3.get_object(Bucket=BUCKET_NAME, Key=Key)
|
|
||||||
if "Body" not in obj:
|
|
||||||
raise Exception("No Body")
|
|
||||||
if isinstance(obj["Body"], StreamingBody):
|
|
||||||
return obj["Body"].read().decode("utf-8")
|
|
||||||
raise Exception("Not StreamingBody")
|
|
||||||
|
|
||||||
|
|
||||||
async def list_cache_objects() -> AsyncIterable[str]:
|
|
||||||
|
|
||||||
@with_backoff
|
|
||||||
@aio
|
|
||||||
def list_objects_v2(ContinuationToken: Optional[str]) -> dict[str, Any]:
|
|
||||||
if ContinuationToken != None:
|
|
||||||
return s3.list_objects_v2(Bucket=BUCKET_NAME,
|
|
||||||
ContinuationToken=ContinuationToken)
|
|
||||||
else:
|
|
||||||
return s3.list_objects_v2(Bucket=BUCKET_NAME)
|
|
||||||
|
|
||||||
cont_token = None
|
|
||||||
while True:
|
|
||||||
objs = await list_objects_v2(cont_token)
|
|
||||||
if "Contents" not in objs:
|
|
||||||
raise Exception("No Contents")
|
|
||||||
if isinstance(objs["Contents"], list):
|
|
||||||
for obj in cast(list[Any], objs["Contents"]):
|
|
||||||
if not isinstance(obj, dict):
|
|
||||||
raise Exception("Not dict")
|
|
||||||
obj = cast(dict[str, Any], obj)
|
|
||||||
yield obj["Key"]
|
|
||||||
|
|
||||||
if "NextContinuationToken" not in objs:
|
|
||||||
break
|
|
||||||
cont_token = objs["NextContinuationToken"]
|
|
||||||
|
|
||||||
|
|
||||||
@with_backoff
|
|
||||||
@aio
|
|
||||||
def delete_object(key: str) -> None:
|
|
||||||
s3.delete_object(Bucket=BUCKET_NAME, Key=key)
|
|
||||||
|
|
||||||
|
|
||||||
def get_store_hashes() -> set[str]:
|
|
||||||
hashes = set()
|
|
||||||
for obj in listdir("/nix/store"):
|
|
||||||
hashes.add(obj.split("-")[0])
|
|
||||||
return hashes
|
|
||||||
|
|
||||||
|
|
||||||
async def main() -> None:
|
|
||||||
store_hashes = get_store_hashes()
|
|
||||||
nars_to_delete = set()
|
|
||||||
nars_to_keep = set()
|
|
||||||
async for obj_key in list_cache_objects():
|
|
||||||
if obj_key.endswith(".narinfo"):
|
|
||||||
# check if we have the hash locally
|
|
||||||
narinfo = await get_object(obj_key)
|
|
||||||
narinfo = NarInfo(narinfo)
|
|
||||||
if not await narinfo.exists_locally():
|
|
||||||
print(f"Found unused NAR for {narinfo.store_path}")
|
|
||||||
await delete_object(obj_key)
|
|
||||||
nars_to_delete.add(narinfo.url)
|
|
||||||
else:
|
|
||||||
nars_to_keep.add(narinfo.url)
|
|
||||||
if obj_key.startswith("realisations/"):
|
|
||||||
realisation = await get_object(obj_key)
|
|
||||||
realisation = json.loads(realisation)
|
|
||||||
if not isinstance(realisation, dict):
|
|
||||||
continue
|
|
||||||
if "outPath" not in realisation:
|
|
||||||
continue
|
|
||||||
if not await exists_locally("/nix/store/" +
|
|
||||||
realisation["outPath"]):
|
|
||||||
print(f"Found unused realisation for {realisation['outPath']}")
|
|
||||||
await delete_object(obj_key)
|
|
||||||
if obj_key.startswith("nar/"):
|
|
||||||
nars_to_delete.add(obj_key)
|
|
||||||
for nar in nars_to_delete:
|
|
||||||
if nar in nars_to_keep:
|
|
||||||
continue
|
|
||||||
print(f"Deleting unused NAR {nar}")
|
|
||||||
await delete_object(nar)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
asyncio.get_event_loop().run_until_complete(main())
|
|
Loading…
Reference in a new issue