aboutsummaryrefslogtreecommitdiff
path: root/nixops/modules
diff options
context:
space:
mode:
authorIsmaël Bouya <ismael.bouya@normalesup.org>2019-03-04 23:39:18 +0100
committerIsmaël Bouya <ismael.bouya@normalesup.org>2019-03-04 23:39:18 +0100
commit22149d17ff17e9870b3245f07abe05e5d026fdd3 (patch)
treebf4aaef2ddaccb7cf17f83ee6b7f2c0d3eb2223c /nixops/modules
parent1be1a523cb04a5079e2212f3ab5a09b6591a4340 (diff)
downloadNix-22149d17ff17e9870b3245f07abe05e5d026fdd3.tar.gz
Nix-22149d17ff17e9870b3245f07abe05e5d026fdd3.tar.zst
Nix-22149d17ff17e9870b3245f07abe05e5d026fdd3.zip
Add task server
Related issue: https://git.immae.eu/mantisbt/view.php?id=67
Diffstat (limited to 'nixops/modules')
-rw-r--r--nixops/modules/databases/immae.schema16
-rw-r--r--nixops/modules/task/default.nix131
-rw-r--r--nixops/modules/task/www/index.php135
3 files changed, 280 insertions, 2 deletions
diff --git a/nixops/modules/databases/immae.schema b/nixops/modules/databases/immae.schema
index f0e12bc..f5ee5d5 100644
--- a/nixops/modules/databases/immae.schema
+++ b/nixops/modules/databases/immae.schema
@@ -149,7 +149,19 @@ objectclass ( ImmaeobjectClass:8 NAME 'immaePuppetClass'
149 MUST ( immaePuppetJson ) 149 MUST ( immaePuppetJson )
150 ) 150 )
151 151
152attributetype (ImmaeattributeType:19 NAME 'immaeTaskId'
153 DESC 'Taskwarrior server Org:Name:Key'
154 EQUALITY caseIgnoreMatch
155 SUBSTR caseIgnoreSubstringsMatch
156 SYNTAX 1.3.6.1.4.1.1466.115.121.1.15{256} )
157
158objectclass ( ImmaeobjectClass:9 NAME 'immaeTaskClass'
159 DESC 'Expansion of the existing object classes for Task'
160 SUP top AUXILIARY
161 MUST ( immaeTaskId )
162 )
163
152# Last: 164# Last:
153# attributetype (ImmaeattributeType:18 NAME 'immaeAccessReadSubtree' 165# attributetype (ImmaeattributeType:19 NAME 'immaeTaskId'
154# objectclass ( ImmaeobjectClass:8 NAME 'immaePuppetClass' 166# objectclass ( ImmaeobjectClass:9 NAME 'immaeTaskClass'
155 167
diff --git a/nixops/modules/task/default.nix b/nixops/modules/task/default.nix
new file mode 100644
index 0000000..3dc3299
--- /dev/null
+++ b/nixops/modules/task/default.nix
@@ -0,0 +1,131 @@
1{ lib, pkgs, config, myconfig, mylibs, ... }:
2let
3 cfg = config.services.myTasks;
4 vardir = config.services.taskserver.dataDir;
5 fqdn = "task.immae.eu";
6 user = config.services.taskserver.user;
7 env = myconfig.env.tools.task;
8 group = config.services.taskserver.group;
9in {
10 options.services.myTasks = {
11 enable = lib.mkEnableOption "my tasks service";
12 };
13
14 config = lib.mkIf cfg.enable {
15 security.acme.certs."eldiron".extraDomains.${fqdn} = null;
16 services.myWebsites.tools.modules = [ "proxy_fcgi" ];
17 services.myWebsites.tools.vhostConfs.task = {
18 certName = "eldiron";
19 hosts = [ "task.immae.eu" ];
20 root = "/run/current-system/webapps/_task";
21 extraConfig = [ ''
22 <Directory /run/current-system/webapps/_task>
23 DirectoryIndex index.php
24 Use LDAPConnect
25 Require ldap-group cn=users,cn=taskwarrior,ou=services,dc=immae,dc=eu
26 <FilesMatch "\.php$">
27 SetHandler "proxy:unix:/var/run/phpfpm/task.sock|fcgi://localhost"
28 </FilesMatch>
29 SetEnv TASKD_HOST "${fqdn}:${toString config.services.taskserver.listenPort}"
30 SetEnv TASKD_VARDIR "${vardir}"
31 SetEnv TASKD_LDAP_HOST "ldaps://${env.ldap.host}"
32 SetEnv TASKD_LDAP_DN "${env.ldap.dn}"
33 SetEnv TASKD_LDAP_PASSWORD "${env.ldap.password}"
34 SetEnv TASKD_LDAP_BASE "${env.ldap.base}"
35 SetEnv TASKD_LDAP_FILTER "${env.ldap.search}"
36 </Directory>
37 '' ];
38 };
39 services.myPhpfpm.poolConfigs = {
40 tasks = ''
41 listen = /var/run/phpfpm/task.sock
42 user = ${user}
43 group = ${group}
44 listen.owner = wwwrun
45 listen.group = wwwrun
46 pm = dynamic
47 pm.max_children = 60
48 pm.start_servers = 2
49 pm.min_spare_servers = 1
50 pm.max_spare_servers = 10
51
52 ; Needed to avoid clashes in browser cookies (same domain)
53 env[PATH] = "/etc/profiles/per-user/${user}/bin"
54 php_value[session.name] = TaskPHPSESSID
55 php_admin_value[open_basedir] = "${./www}:/tmp:${vardir}:/etc/profiles/per-user/${user}/bin/"
56 '';
57 };
58
59 system.extraSystemBuilderCmds = ''
60 ln -s ${./www} $out/webapps/_task
61 '';
62
63 security.acme.certs."task" = config.services.myCertificates.certConfig // {
64 inherit user group;
65 plugins = [ "fullchain.pem" "key.pem" "cert.pem" "account_key.json" ];
66 domain = fqdn;
67 postRun = ''
68 systemctl restart taskserver.service
69 '';
70 };
71
72 users.users.${user}.packages = [
73 (pkgs.runCommand "taskserver-user-certs" {} ''
74 mkdir -p $out/bin
75 cat > $out/bin/taskserver-user-certs <<"EOF"
76 #!/usr/bin/env bash
77
78 user=$1
79
80 silent_certtool() {
81 if ! output="$("${pkgs.gnutls.bin}/bin/certtool" "$@" 2>&1)"; then
82 echo "GNUTLS certtool invocation failed with output:" >&2
83 echo "$output" >&2
84 fi
85 }
86
87 silent_certtool -p \
88 --bits 4096 \
89 --outfile "${vardir}/userkeys/$user.key.pem"
90 ${pkgs.gnused}/bin/sed -i -n -e '/^-----BEGIN RSA PRIVATE KEY-----$/,$p' "${vardir}/userkeys/$user.key.pem"
91
92 silent_certtool -c \
93 --template "${pkgs.writeText "taskserver-ca.template" ''
94 tls_www_client
95 encryption_key
96 signing_key
97 expiration_days = 3650
98 ''}" \
99 --load-ca-certificate "${vardir}/keys/ca.cert" \
100 --load-ca-privkey "${vardir}/keys/ca.key" \
101 --load-privkey "${vardir}/userkeys/$user.key.pem" \
102 --outfile "${vardir}/userkeys/$user.cert.pem"
103 EOF
104 chmod a+x $out/bin/taskserver-user-certs
105 patchShebangs $out/bin/taskserver-user-certs
106 '')
107 ];
108
109 systemd.services.taskserver-ca.postStart = ''
110 chown :${group} "${vardir}/keys/ca.key"
111 chmod g+r "${vardir}/keys/ca.key"
112 '';
113
114 system.activationScripts.taskserver = {
115 deps = [ "users" ];
116 text = ''
117 install -m 0750 -o ${user} -g ${group} -d ${vardir}
118 install -m 0750 -o ${user} -g ${group} -d ${vardir}/userkeys
119 install -m 0750 -o ${user} -g ${group} -d ${vardir}/keys
120 '';
121 };
122
123 services.taskserver = {
124 enable = true;
125 allowedClientIDs = [ "^task [2-9]" "^Mirakel [1-9]" ];
126 inherit fqdn;
127 listenHost = "::";
128 requestLimit = 104857600;
129 };
130 };
131}
diff --git a/nixops/modules/task/www/index.php b/nixops/modules/task/www/index.php
new file mode 100644
index 0000000..829cdd0
--- /dev/null
+++ b/nixops/modules/task/www/index.php
@@ -0,0 +1,135 @@
1<?php
2if (!isset($_SERVER["REMOTE_USER"])) {
3 die("please login");
4}
5$ldap_user = $_SERVER["REMOTE_USER"];
6$ldap_host = getenv("TASKD_LDAP_HOST");
7$ldap_dn = getenv('TASKD_LDAP_DN');
8$ldap_password = getenv('TASKD_LDAP_PASSWORD');
9$ldap_base = getenv('TASKD_LDAP_BASE');
10$ldap_filter = getenv('TASKD_LDAP_FILTER');
11$host = getenv('TASKD_HOST');
12$vardir = getenv('TASKD_VARDIR');
13
14$connect = ldap_connect($ldap_host);
15ldap_set_option($connect, LDAP_OPT_PROTOCOL_VERSION, 3);
16if (!$connect || !ldap_bind($connect, $ldap_dn, $ldap_password)) {
17 die("impossible to connect to LDAP");
18}
19
20$search_query = str_replace('%login%', ldap_escape($ldap_user), $ldap_filter);
21
22$search = ldap_search($connect, $ldap_base, $search_query);
23$info = ldap_get_entries($connect, $search);
24
25if (ldap_count_entries($connect, $search) != 1) {
26 die("Impossible to find user in LDAP");
27}
28
29$entries = [];
30foreach($info[0]["immaetaskid"] as $key => $value) {
31 if ($key !== "count") {
32 $entries[] = explode(":", $value);
33 }
34}
35
36if (isset($_GET["file"])) {
37 $basecert = $vardir . "/userkeys/" . $ldap_user;
38 if (!file_exists($basecert . ".cert.pem")) {
39 exec("taskserver-user-certs $ldap_user");
40 }
41 $certificate = file_get_contents($basecert . ".cert.pem");
42 $cert_key = file_get_contents($basecert . ".key.pem");
43 $server_cert = file_get_contents($vardir . "/keys/server.cert");
44
45 $file = $_GET["file"];
46 switch($file) {
47 case "ca.cert.pem":
48 $content = $server_cert;
49 $name = "ca.cert.pem";
50 $type = "application/x-x509-ca-cert";
51 break;
52 case "cert.pem":
53 $content = $certificate;
54 $name = $ldap_user . ".cert.pem";
55 $type = "application/x-x509-ca-cert";
56 break;
57 case "key.pem":
58 $content = $cert_key;
59 $name = $ldap_user . ".key.pem";
60 $type = "application/x-x509-ca-cert";
61 break;
62 case "mirakel";
63 foreach ($entries as $entry) {
64 list($org, $user, $key) = $entry;
65 if ($key == $_GET["key"]) { break; }
66 }
67 $name = $user . ".mirakel";
68 $type = "text/plain";
69 $content = "username: $user
70org: $org
71user key: $key
72server: $host
73client.cert:
74$certificate
75Client.key:
76$cert_key
77ca.cert:
78$server_cert
79";
80 break;
81 default:
82 die("invalid file name");
83 break;
84 }
85
86 header("Content-Type: $type");
87 header('Content-Disposition: attachment; filename="' . $name . '"');
88 header('Content-Transfer-Encoding: binary');
89 header('Accept-Ranges: bytes');
90 header('Cache-Control: private');
91 header('Pragma: private');
92 echo $content;
93 exit;
94}
95?>
96<html>
97<header>
98 <title>Taskwarrior configuration</title>
99</header>
100<body>
101<ul>
102 <li><a href="?file=ca.cert.pem">ca.cert.pem</a></li>
103 <li><a href="?file=cert.pem"><?php echo $ldap_user; ?>.cert.pem</a></li>
104 <li><a href="?file=key.pem"><?php echo $ldap_user; ?>.key.pem</a></li>
105</ul>
106For command line interface, download the files, put them near your Taskwarrior
107configuration files, and add that to your Taskwarrior configuration:
108<pre>
109taskd.certificate=/path/to/<?php echo $ldap_user; ?>.cert.pem
110taskd.key=/path/to/<?php echo $ldap_user; ?>.key.pem
111taskd.server=<?php echo $host ."\n"; ?>
112<?php if (count($entries) > 1) {
113 echo "# Chose one of them\n";
114 foreach($entries as $entry) {
115 list($org, $user, $key) = $entry;
116 echo "# taskd.credentials=$org/$user/$key\n";
117 }
118} else { ?>
119taskd.credentials=<?php echo $entries[0][0]; ?>/<?php echo $entries[0][1]; ?>/<?php echo $entries[0][2]; ?>
120<?php } ?>
121taskd.ca=/path/to/ca.cert.pem
122</pre>
123For Mirakel, download and import the file:
124<ul>
125<?php
126foreach ($entries as $entry) {
127 list($org, $user, $key) = $entry;
128 echo '<li><a href="?file=mirakel&key='.$key.'">' . $user . '.mirakel</a></li>';
129}
130?>
131</ul>
132For Android Taskwarrior app, see instructions <a href="https://bitbucket.org/kvorobyev/taskwarriorandroid/wiki/Configuration">here</a>.
133</body>
134</html>
135