feat: nixos module
All checks were successful
build / build (push) Successful in 8s

This commit was merged in pull request #42.
This commit is contained in:
2026-04-18 20:12:20 -06:00
parent 7e6fb22d03
commit 7aaec7388d
8 changed files with 152 additions and 25 deletions

View File

@@ -33,8 +33,8 @@ let
]; ];
doerg-config = writeText "doerg-extra-config.edn" '' doerg-config = writeText "doerg-extra-config.edn" ''
#:net.deertopia.doerg #:net.deertopia.doerg.config
{:ibm-plex-web "${ibm-plex-web}" {:ibm-plex-web "${ibm-plex-web}/share/ibm-plex-web"
:latex "${lib.getExe' our-tex "xelatex"}" :latex "${lib.getExe' our-tex "xelatex"}"
:dvisvgm "${lib.getExe' our-tex "dvisvgm"}" :dvisvgm "${lib.getExe' our-tex "dvisvgm"}"
:doerg-temml-worker "${lib.getExe doerg-temml-worker}" :doerg-temml-worker "${lib.getExe doerg-temml-worker}"
@@ -43,15 +43,21 @@ let
in mkCljBin' { in mkCljBin' {
name = "net.deertopia/doerg"; name = "net.deertopia/doerg";
version = "0.1.0"; version = "0.1.0";
projectSrc = lib.cleanSource ./.; projectSrc = lib.cleanSourceWith {
filter = path: type:
lib.sources.cleanSourceFilter path type
|| lib.hasSuffix ".nix" path;
src = ./.;
};
lockfile = ./deps-lock.json; lockfile = ./deps-lock.json;
main-ns = "net.deertopia.doerg.main"; main-ns = "net.deertopia.doerg.main";
nativeBuildInputs = [ nativeBuildInputs = [
plex plex
makeWrapper makeWrapper
]; ];
buildInputs = [ propagatedBuildInputs = [
doerg-parser doerg-parser
ibm-plex-web
doerg-temml-worker doerg-temml-worker
plex plex
our-tex our-tex
@@ -73,5 +79,6 @@ in mkCljBin' {
XDG_STATE_HOME=$(mktemp -d "state-home-XXXXXX") XDG_STATE_HOME=$(mktemp -d "state-home-XXXXXX")
clojure -M:test clojure -M:test
''; '';
passthru = { inherit plex our-tex test-emacs; }; passthru = { inherit plex our-tex test-emacs doerg-config; };
meta.mainProgram = "doerg";
} }

View File

@@ -14,12 +14,15 @@
"x86_64-linux" "x86_64-linux"
]; ];
each-system = f: nixpkgs.lib.genAttrs supportedSystems (system: f rec {
pkgs = import nixpkgs {
inherit system;
overlays = [ overlays = [
inputs.sydpkgs.overlays.default inputs.sydpkgs.overlays.default
clj-nix.overlays.default clj-nix.overlays.default
];
each-system = f: nixpkgs.lib.genAttrs supportedSystems (system: f rec {
pkgs = import nixpkgs {
inherit system;
overlays = overlays ++ [
self.overlays.default self.overlays.default
]; ];
}; };
@@ -36,13 +39,17 @@
}); });
overlays.default = final: prev: overlays.default = final: prev:
let graal = x: final.mkGraalBin { cljDrv = x; }; let
# is this really the correct way to satisfy our
# dependencies? lmfao
final' = final.appendOverlays overlays;
graal = x: final'.mkGraalBin { cljDrv = x; };
in { in {
ibm-plex-web = final.callPackage ./ibm-plex-web.nix {}; ibm-plex-web = final'.callPackage ./ibm-plex-web.nix {};
doerg = final.callPackage ./. {}; doerg = final'.callPackage ./. {};
doerg-parser = final.callPackage ./doerg-parser {}; doerg-parser = final'.callPackage ./doerg-parser {};
doerg-temml-worker = final.callPackage ./doerg-temml-worker {}; doerg-temml-worker = final'.callPackage ./doerg-temml-worker {};
our-tex = final.callPackage ./our-tex.nix {}; our-tex = final'.callPackage ./our-tex.nix {};
}; };
checks = each-system ({ pkgs, system, ... }: { checks = each-system ({ pkgs, system, ... }: {
@@ -52,6 +59,8 @@
(pkgs.lib.attrValues self.packages.${system}); (pkgs.lib.attrValues self.packages.${system});
}); });
nixosModules.default = import ./module.nix { inherit self; };
devShells = each-system ({ pkgs, system, ... }: { devShells = each-system ({ pkgs, system, ... }: {
default = pkgs.mkShell { default = pkgs.mkShell {
inputsFrom = [ inputsFrom = [

107
module.nix Normal file
View File

@@ -0,0 +1,107 @@
{ self, ... }:
{ config, lib, pkgs, ... }:
let
cfg = config.services.doerg;
doerg-config = pkgs.writeText "doerg-config.edn" ''
#:net.deertopia.doerg.config
{:org-roam-db-path "${cfg.databasePath}"
:state-directory "${cfg.stateDir}"
:port ${builtins.toString cfg.port}}
'';
inherit (lib) types;
org-roam-db-sync = pkgs.writeText "org-roam-db-sync.el" ''
#!/usr/bin/env -S emacs -Q -x
(require 'org-roam)
(setq org-roam-directory (expand-file-name (car command-line-args-left)))
(setq org-roam-db-location (expand-file-name (cadr command-line-args-left)))
(org-roam-db-sync)
'';
in {
options.services.doerg = {
enable = lib.mkEnableOption "Doerg";
org-roam-db-sync.enable = lib.mkEnableOption "Org-roam db sync";
port = lib.mkOption {
default = 21984;
type = lib.types.port;
description = ''
The port on which Doerg will listen.
'';
};
stateDir = lib.mkOption {
type = types.path;
default = "/var/lib/private/doerg";
description = "Daemon's state directory.";
};
orgDir = lib.mkOption {
type = types.path;
description = "Org roam directory.";
};
package = lib.mkPackageOption pkgs "doerg" {};
databasePath = lib.mkOption {
type = types.path;
description = "Org roam database path";
default = cfg.orgDir + "org-roam.db";
};
openFirewall = lib.mkOption {
type = types.bool;
description = "Open doerg ports?";
default = false;
};
};
config = lib.mkIf cfg.enable {
nixpkgs.overlays = [ self.overlays.default ];
systemd.services.org-roam-db-sync = lib.mkIf cfg.org-roam-db-sync.enable {
script = lib.escapeShellArgs [
(lib.getExe cfg.package.test-emacs)
"-Q" "-x" org-roam-db-sync cfg.orgDir cfg.databasePath
];
serviceConfig = {
Type = "oneshot";
ReadOnlyBindPaths = [
cfg.orgDir
];
};
};
systemd.timers.org-roam-db-sync = lib.mkIf cfg.org-roam-db-sync.enable {
unitConfig.StopWhenUnneeded = true;
timerConfig = {
OnActiveSec = "1h";
RandomizedDelaySec = "30m";
Persistent = true;
};
};
networking.firewall.allowedTCPPorts = lib.mkIf cfg.openFirewall [
cfg.port
];
systemd.services.doerg = {
after = [ "network-online.target" ];
wants = [ "network-online.target" "org-roam-db-sync.timer" ];
wantedBy = [ "multi-user.target" ];
environment.DOERG_CONFIG = doerg-config;
serviceConfig = {
# WorkingDirectory = cfg.stateDir;
StateDirectory = "doerg";
ExecStart = lib.getExe cfg.package;
DynamicUser = true;
ProtectSystem = "strict";
PrivateTmp = true;
BindReadOnlyPaths = [
cfg.orgDir
# cfg.databasePath
"/nix"
];
};
};
};
}

View File

@@ -1,9 +1,8 @@
#:net.deertopia.doerg.config #:net.deertopia.doerg.config
{:ibm-plex-web #or [#xdg-data-dir "ibm-plex-web" {:ibm-plex-web #or [#env IBM_PLEX_WEB
#env IBM_PLEX_WEB] #xdg-data-dir "ibm-plex-web"]
:latex "xelatex" :latex "xelatex"
:dvisvgm "dvisvgm" :dvisvgm "dvisvgm"
:debug-unimplemented? #profile {:dev true :default false}
:doerg-temml-worker :doerg-temml-worker
#profile {:dev #file "../../../../doerg-temml-worker/index.js" #profile {:dev #file "../../../../doerg-temml-worker/index.js"
:default "doerg-temml-worker"} :default "doerg-temml-worker"}

View File

@@ -18,8 +18,10 @@
calling `compute` with no arguments only if stale? is logical true." calling `compute` with no arguments only if stale? is logical true."
[& {:keys [file stale? compute]}] [& {:keys [file stale? compute]}]
(when (or (not *use-cache?*) stale?) (when (or (not *use-cache?*) stale?)
(let [r (compute)] (let [r (compute)
dir (fs/parent file)]
(assert (string? r)) (assert (string? r))
(fs/create-dirs (fs/parent file)) (when-not (fs/exists? dir)
(fs/create-dirs dir))
(spit file r))) (spit file r)))
file) file)

View File

@@ -46,7 +46,8 @@
"Aero tag to search for a directory on $XDG_DATA_DIRS." "Aero tag to search for a directory on $XDG_DATA_DIRS."
(some #(let [x (fs/path % value)] (some #(let [x (fs/path % value)]
(and (fs/exists? x) x)) (and (fs/exists? x) x))
(fs/split-paths (System/getenv "XDG_DATA_DIRS")))) (some-> (System/getenv "XDG_DATA_DIRS")
fs/split-paths)))
(defmethod aero/reader 'file (defmethod aero/reader 'file
[{:keys [source]} tag value] [{:keys [source]} tag value]

View File

@@ -135,6 +135,10 @@
(defn public? [node] (defn public? [node]
(some-> (properties node) (get "DEERTOPIAVISIBILITY") (= "public"))) (some-> (properties node) (get "DEERTOPIAVISIBILITY") (= "public")))
(defn get-public-node [node]
(when-let [n (get-node node)]
(when (public? n) n)))
(defn graph-visible? [node] (defn graph-visible? [node]
(#{"public" "graphonly"} (#{"public" "graphonly"}
(some-> (properties node) (get "DEERTOPIAVISIBILITY")))) (some-> (properties node) (get "DEERTOPIAVISIBILITY"))))

View File

@@ -23,9 +23,7 @@
[net.deertopia.doerg.cached-file :as cached-file] [net.deertopia.doerg.cached-file :as cached-file]
[babashka.fs :as fs] [babashka.fs :as fs]
[aero.core :as aero] [aero.core :as aero]
[clojure.string :as str] [clojure.string :as str]))
[net.deertopia.doerg :as-alias doerg]
[net.deertopia.doerg.config :as doerg-config]))
;;; Routes ;;; Routes
@@ -105,7 +103,7 @@
(roam/title n))])]]))) (roam/title n))])]])))
(defn node-by-slug [{{:keys [slug]} :path-params :as req}] (defn node-by-slug [{{:keys [slug]} :path-params :as req}]
(if-some [node (some-> slug slug/from-string roam/get-node)] (if-some [node (some-> slug slug/from-string roam/get-public-node)]
(let [org-file (roam/org-file node) (let [org-file (roam/org-file node)
html-file (org-file->html-file org-file)] html-file (org-file->html-file org-file)]
(cached-file/cached-file (cached-file/cached-file
@@ -142,7 +140,7 @@
(if-some [[_ resource] (re-matches #"^/resource/ibm-plex-web/(.*)" uri)] (if-some [[_ resource] (re-matches #"^/resource/ibm-plex-web/(.*)" uri)]
(-> resource (-> resource
(response/file-response (response/file-response
{:root (-> doerg-config/*cfg* ::cfg/ibm-plex-web str)})) {:root (-> cfg/*cfg* ::cfg/ibm-plex-web str)}))
(-> uri (-> uri
(str/replace-first #"^/resource/" "") (str/replace-first #"^/resource/" "")
(response/resource-response (response/resource-response