-
-
Notifications
You must be signed in to change notification settings - Fork 168
/
elisp.nix
102 lines (96 loc) · 3.6 KB
/
elisp.nix
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/*
Parse an emacs lisp configuration file to derive packages from
use-package declarations.
*/
{ pkgs }:
let
parse = pkgs.callPackage ./parse.nix { };
inherit (pkgs) lib;
in
{ config
# bool to use the value of config or a derivation whose name is default.el
, defaultInitFile ? false
# emulate `use-package-always-ensure` behavior (defaulting to false)
, alwaysEnsure ? false
# emulate `#+PROPERTY: header-args:emacs-lisp :tangle yes`
, alwaysTangle ? false
, extraEmacsPackages ? epkgs: [ ]
, package ? pkgs.emacs
, override ? (self: super: { })
}:
let
isOrgModeFile =
let
ext = lib.last (builtins.split "\\." (builtins.toString config));
type = builtins.typeOf config;
in
(type == "path" || lib.hasPrefix "/" config) && ext == "org";
configText =
let
type = builtins.typeOf config;
in # configText can be sourced from either:
# - A string with context { config = "${hello}/config.el"; }
if type == "string" && builtins.hasContext config && lib.hasPrefix builtins.storeDir config then builtins.readFile config
# - A config literal { config = "(use-package foo)"; }
else if type == "string" then config
# - A config path { config = ./config.el; }
else if type == "path" then builtins.readFile config
# - A derivation { config = pkgs.writeText "config.el" "(use-package foo)"; }
else if lib.isDerivation config then builtins.readFile "${config}"
else throw "Unsupported type for config: \"${type}\"";
packages = parse.parsePackagesFromUsePackage {
inherit configText isOrgModeFile alwaysTangle alwaysEnsure;
};
emacsPackages = (pkgs.emacsPackagesFor package).overrideScope (self: super:
# for backward compatibility: override was a function with one parameter
if builtins.isFunction (override super)
then override self super
else override super
);
emacsWithPackages = emacsPackages.emacsWithPackages;
mkPackageError = name:
let
errorFun = if (alwaysEnsure != null && alwaysEnsure) then builtins.trace else throw;
in
errorFun "Emacs package ${name}, declared wanted with use-package, not found." null;
in
emacsWithPackages (epkgs:
let
usePkgs = map (name: epkgs.${name} or (mkPackageError name)) packages;
extraPkgs = extraEmacsPackages epkgs;
defaultInitFilePkg =
if !((builtins.isBool defaultInitFile) || (lib.isDerivation defaultInitFile))
then throw "defaultInitFile must be bool or derivation"
else
if defaultInitFile == false
then null
else
let
# name of the default init file must be default.el according to elisp manual
defaultInitFileName = "default.el";
configFile = pkgs.writeText defaultInitFileName configText;
orgModeConfigFile = pkgs.runCommand defaultInitFileName {
nativeBuildInputs = [ package ];
} ''
cp ${configFile} config.org
emacs -Q --batch ./config.org -f org-babel-tangle
mv config.el $out
'';
in
epkgs.trivialBuild {
pname = "default";
src =
if defaultInitFile == true
then
if isOrgModeFile
then orgModeConfigFile
else configFile
else
if defaultInitFile.name == defaultInitFileName
then defaultInitFile
else throw "name of defaultInitFile must be ${defaultInitFileName}";
version = "0.1.0";
packageRequires = usePkgs ++ extraPkgs;
};
in
usePkgs ++ extraPkgs ++ [ defaultInitFilePkg ])