dotfiles/deployer/index.ts

189 lines
5.7 KiB
TypeScript

#!/usr/bin/env -S tsx
import { ssh, SSH } from './ssh.js'
import { Expression, BuiltOutput } from './nix.js'
class Cmd {
registry: {[index: string]: (() => Promise<void>)} = {};
constructor() {
}
register(obj: string, action: string, fn: () => Promise<void>) {
this.registry[`${obj}.${action}`] = fn
}
async run() {
const opt = argv._[0]
if (opt && this.registry[opt]) await this.registry[opt]()
else {
console.log("Possible options: \n")
for (const k of Object.keys(this.registry)) {
console.log("-", k)
}
}
}
}
const action = new Cmd()
async function ping(hostname: string) {
return await $`ping -Anqc3 ${hostname}`.exitCode == 0
}
class Machine {
name: string;
hasHome: boolean = false;
hostname?: string;
constructor(args: {name?: string, hasHome?: boolean, hostname?: string}) {
Object.assign(this, args)
// todo
this.name = ""
}
isLocal() {
return os.hostname() == this.name
}
get attr(): Expression {
return new Expression(`.#nixosConfigurations.${this.name}.toplevel`)
}
private async sshTarget() {
if (this.isLocal()) return "localhost"
if (this.hostname && await ping(this.hostname)) {
return this.hostname
} else {
// todo: directify
return `${name}.vpn.yori.cc`
}
}
async findIP(): Promise<typeof Machine.prototype.ssh> {
const host = await this.sshTarget()
return <R>(user?: string, cb?: () => Promise<R>): Promise<SSH|R> => {
const sshTarget = user ? `${user}@${host}` : host
if (cb !== undefined) return ssh(host, cb)
return ssh(host)
}
}
ssh(user?: string): Promise<SSH>
ssh<R>(user?: string, cb?: () => Promise<R>): Promise<R>
async ssh<R>(user?: string, cb?: () => Promise<R>) {
return (await this.findIP())(user, cb)
}
get targets() {
if (this.hasHome) return [this.attr, Home]
else return [this.attr]
}
}
const machines = {
frumar: new Machine({hostname: "frumar.home.yori.cc"}),
pennyworth: new Machine({hostname: "pennyworth.yori.cc"}),
blackadder: new Machine({hasHome: true}),
jarvis: new Machine({hasHome: true}),
smithers: new Machine({hasHome: true}),
}
for (const [name, machine] of Object.entries(machines))
machine.name = name
for (const machine of Object.values(machines)) {
action.register(machine.name, "ssh", async function() {
(await machine.ssh()).interactive()
})
action.register(machine.name, "gc", async () => {
machine.ssh("root", async () => {
await $`nix-collect-garbage -d`
})
})
action.register(machine.name, "eval", async () => {
await Promise.all(machine.targets.map(x => x.derive()))
})
action.register(machine.name, "build", async () => {
await Promise.all(machine.targets.map(x => x.build()))
})
action.register(machine.name, "copy", async () => {
const machineSSH = await machine.findIP()
// todo merge builds
const outputs = (await Promise.all(machine.targets.map(x => x.build()))).flat()
if (machine.isLocal()) {
console.log("skipping copy, is localhost")
return
}
const conn = await machineSSH()
await Promise.all(outputs.map(x => x.copy(conn)))
// machine.toplevel.buildAndCopy(machine.ssh)
// Home.buildAndCopy(machine.ssh)
})
action.register(machine.name, "boot-deploy", async () => {
const machineSSH = await machine.findIP()
const path = (await machine.attr.build()).out
if (!machine.isLocal()) {
const conn = await machineSSH()
await path.copy(conn)
}
// machine.toplevel.buildAndCopy(machine.ssh)
// machine.ssh.within(machine.toplevel.activate("boot"))
await machineSSH("root", async () => {
await $`nix-env -p /nix/var/nix/profiles/system --set ${path}`
await $`${path}/bin/switch-to-configuration boot`
})
})
action.register(machine.name, "switch", async () => {
const machineSSH = await machine.findIP()
const path = (await machine.attr.build()).out
if (!machine.isLocal()) {
const conn = await machineSSH()
await path.copy(conn)
}
const new_kernel = (await $`readlink ${path.path}/kernel`).stdout
await machineSSH("root", async () => {
const old_kernel = (await $`readlink /run/booted-system/kernel`).stdout
if (new_kernel !== old_kernel) {
console.error(`[${machine.name}] requires reboot because of kernel update`)
process.exit(1)
}
await $`nix-env -p /nix/var/nix/profiles/system --set ${path.path}`
await $`${path}/bin/switch-to-configuration switch`
})
})
if (machine.hasHome) {
action.register(machine.name, "update-home", async () => {
const new_path = (await Home.build()).out
if (machine.isLocal()) {
await $`${new_path.path}/activate`
} else {
const conn = await machine.ssh()
await new_path.copy(conn)
await conn.within(() => $`${new_path.path}/activate`)
}
})
}
action.register(machine.name, "status", async () => {
await machine.ssh(undefined, async () => {
await $`systemctl is-system-running`
await $`zpool status -x`
await $`realpath /run/current-system /nix/var/nix/profiles/system`
})
})
}
action.register("all", "build", async () => {
console.log(await (new Expression(".#")).build())
})
const Home = new Expression(".#yorick-home")
action.register("home", "build", async () => {
console.log(await Home.build())
})
action.register("home", "eval", async () => {
console.log(await Home.derive())
})
action.register("local", "boot-deploy", async () => {
await action.registry[`${os.hostname()}.boot-deploy`]()
})
action.register("local", "status", async () => {
await action.registry[`${os.hostname()}.status`]()
})
await action.run()