diff options
-rw-r--r-- | accounts/gkleen@sif/default.nix | 2 | ||||
-rw-r--r-- | accounts/gkleen@sif/zshrc | 403 |
2 files changed, 405 insertions, 0 deletions
diff --git a/accounts/gkleen@sif/default.nix b/accounts/gkleen@sif/default.nix index be16693e..d6e91447 100644 --- a/accounts/gkleen@sif/default.nix +++ b/accounts/gkleen@sif/default.nix | |||
@@ -78,6 +78,8 @@ in { | |||
78 | }; | 78 | }; |
79 | profiles = customUtils.recImport { dir = ./autorandr-profiles; }; | 79 | profiles = customUtils.recImport { dir = ./autorandr-profiles; }; |
80 | }; | 80 | }; |
81 | |||
82 | zsh.initExtra = "source ${./zshrc}"; | ||
81 | }; | 83 | }; |
82 | 84 | ||
83 | services = { | 85 | services = { |
diff --git a/accounts/gkleen@sif/zshrc b/accounts/gkleen@sif/zshrc new file mode 100644 index 00000000..34aa3772 --- /dev/null +++ b/accounts/gkleen@sif/zshrc | |||
@@ -0,0 +1,403 @@ | |||
1 | filebin() { | ||
2 | basePath=/srv/www/files | ||
3 | ssh ymir find "${basePath}" -type f -print0 \ | ||
4 | | while IFS= read -r -d $'\0' p; do | ||
5 | printf "https://f.141.li/%s\n" "${p#${basePath}/}" | ||
6 | done | ||
7 | } | ||
8 | |||
9 | genmail() { | ||
10 | local baseName="" | ||
11 | local target="" | ||
12 | if [[ ${#@} -ge 1 ]]; then | ||
13 | target=${1} | ||
14 | shift | ||
15 | fi | ||
16 | |||
17 | if [[ ${#@} -ge 1 ]]; then | ||
18 | baseName=$(pwgen ${@}) | ||
19 | else | ||
20 | baseName=$(pwgen -v -A -0 16 1) | ||
21 | fi | ||
22 | baseName=$(tr -cd $'[:alnum:]!#$%&*+-/=?^_{|}~.' <<<${baseName}) | ||
23 | address=${baseName}@141.li | ||
24 | insertAddr() { | ||
25 | echo "${baseName} gkleen" | ssh ymir tee -a /srv/mail/spm 1>/dev/null \ | ||
26 | } | ||
27 | |||
28 | printf "%s\n" ${address} | ||
29 | read -q 'cont?Continue [y/N]? ' || return | ||
30 | |||
31 | insertAddr | ||
32 | } | ||
33 | |||
34 | s() { | ||
35 | dir=$(pwd) | ||
36 | [[ ${#@} -ge 1 ]] && dir=$1 | ||
37 | |||
38 | shellFile=$(findNix ${@}) | ||
39 | [[ ${#@} -ge 1 ]] && shift | ||
40 | |||
41 | typeset -a cmd | ||
42 | if [[ -d ${dir}/.nix-gc-roots ]]; then | ||
43 | cmd=(persistent-nix-shell ${shellFile} ${S_EXTRA_ARGS} ${@}) | ||
44 | else | ||
45 | cmd=(nix-shell ${shellFile} ${S_EXTRA_ARGS} ${@}) | ||
46 | fi | ||
47 | |||
48 | if [[ -n "${S_SYSTEMD}" ]]; then | ||
49 | systemd-run --user --slice=development.slice --collect -E PATH=${PATH} -p WorkingDirectory=${dir} -- ${cmd} | ||
50 | else | ||
51 | ( | ||
52 | cd ${dir} | ||
53 | |||
54 | exec ${cmd} | ||
55 | ) | ||
56 | fi | ||
57 | } | ||
58 | |||
59 | sz() { | ||
60 | typeset -a S_EXTRA_ARGS | ||
61 | S_EXTRA_ARGS=(--run "env __ETC_ZSHENV_SOURCED=1 zsh") s ${@} | ||
62 | } | ||
63 | st() { | ||
64 | typeset -a S_EXTRA_ARGS | ||
65 | S_EXTRA_ARGS=(--run "tmux new-session env __ETC_ZSHENV_SOURCED=1 zsh") s ${@} | ||
66 | } | ||
67 | stt() { | ||
68 | typeset -a S_EXTRA_ARGS | ||
69 | S_SYSTEMD=true S_EXTRA_ARGS=(--run "urxvt -e tmux -S .tmux.sock new-session env __ETC_ZSHENV_SOURCED=1 zsh") s ${@} | ||
70 | } | ||
71 | se() { | ||
72 | typeset -a S_EXTRA_ARGS | ||
73 | S_SYSTEMD=true S_EXTRA_ARGS=(--run "emacs") s ${@} | ||
74 | } | ||
75 | |||
76 | findNix() { | ||
77 | if [[ $#@ -eq 0 ]]; then | ||
78 | findNix $(pwd) | ||
79 | elif [[ -f "$1" ]]; then | ||
80 | print ${1:a} | ||
81 | elif [[ -d "$1" && -f "$1"/shell.nix ]]; then | ||
82 | print ${1:a}/shell.nix | ||
83 | elif [[ -d "$1" && -f "$1"/default.nix ]]; then | ||
84 | print ${1:a}/default.nix | ||
85 | elif [[ -d "$1" && "$1" != "/" ]]; then | ||
86 | findNix ${1:h} | ||
87 | else | ||
88 | printf "Traversed directories to ‘/’ and found no shell specification\n" >&2 | ||
89 | return 1 | ||
90 | fi | ||
91 | } | ||
92 | |||
93 | dir() { | ||
94 | curlArchive=false | ||
95 | templateArchive="" | ||
96 | repoUrl="" | ||
97 | nixShell="" | ||
98 | findNix=false | ||
99 | dir="" | ||
100 | forceShell=false | ||
101 | wormhole=false | ||
102 | gitWorktree="" | ||
103 | notmuchMsg="" | ||
104 | quickserve=false | ||
105 | |||
106 | while getopts ':t:a:s:Sd:ir:wqg:n:' arg; do | ||
107 | case $arg in | ||
108 | "t") ;; | ||
109 | "a") | ||
110 | if [[ ${OPTARG} =~ "://" ]]; then | ||
111 | templateArchive=${OPTARG} | ||
112 | curlArchive=true | ||
113 | else | ||
114 | templateArchive=${OPTARG:a} | ||
115 | fi | ||
116 | ;; | ||
117 | "s") nixShell=${OPTARG:a} ;; | ||
118 | "S") findNix=true ;; | ||
119 | "d") dir=${OPTARG} ;; | ||
120 | "i") forceShell=true ;; | ||
121 | "r") repoUrl=${OPTARG} ;; | ||
122 | "w") wormhole=true ;; | ||
123 | "g") gitWorktree=${OPTARG} ;; | ||
124 | "n") notmuchMsg=${OPTARG} ;; | ||
125 | "q") quickserve=true ;; | ||
126 | *) printf "Invalid option: %s\n" $arg >&2; exit 2 ;; | ||
127 | esac | ||
128 | done | ||
129 | |||
130 | shift $((OPTIND - 1)) | ||
131 | |||
132 | if [[ -z ${dir} && ${#@} -ge 1 ]]; then | ||
133 | dir=${1} | ||
134 | shift | ||
135 | fi | ||
136 | |||
137 | [[ -n ${dir} ]] || return 2; | ||
138 | |||
139 | if [[ ! -e ${dir} ]]; then | ||
140 | if [[ -z "${gitWorktree}" ]]; then | ||
141 | mkdir -vp ${dir} | ||
142 | else | ||
143 | git -C ${gitWorktree} worktree add ${dir} | ||
144 | fi | ||
145 | else | ||
146 | gitWorktree="" | ||
147 | fi | ||
148 | |||
149 | ( | ||
150 | cd ${dir} | ||
151 | export dir; | ||
152 | |||
153 | ${findNix} && { nixShell=$(findNix) || return $? } | ||
154 | |||
155 | [[ -n ${repoUrl} ]] && git clone -- ${repoUrl} . | ||
156 | |||
157 | if [[ -n ${templateArchive} ]]; then | ||
158 | ( | ||
159 | archiveFile="" | ||
160 | cleanup() { | ||
161 | [[ -n "${archiveFile}" ]] && rm -fv ${archiveFile} | ||
162 | } | ||
163 | trap cleanup EXIT | ||
164 | |||
165 | if ${curlArchive}; then | ||
166 | archiveFile=$(mktemp -t "archive.XXXXXXXXXX.${templateArchive:t}") | ||
167 | |||
168 | curl -L -o ${archiveFile} ${templateArchive} | ||
169 | |||
170 | templateArchive=${archiveFile} | ||
171 | fi | ||
172 | |||
173 | case $(file --brief --mime-type ${templateArchive}) in | ||
174 | application/zip) unzip ${templateArchive} ;; | ||
175 | *) tar -xvaf ${templateArchive} ;; | ||
176 | esac | ||
177 | ) | ||
178 | fi | ||
179 | |||
180 | |||
181 | if [[ -n ${notmuchMsg} ]]; then | ||
182 | getMimeTypes() { | ||
183 | nix-shell -p mailcap --run "find \${buildInputs} -path '*/etc/mime.types' | head -n 1 | xargs -- cat" | ||
184 | } | ||
185 | |||
186 | typeset -a messages | ||
187 | messages=(${(z)$(notmuch search --output=messages ${notmuchMsg})}) | ||
188 | |||
189 | for message (${messages}); do | ||
190 | typeset -A notmuchAtts | ||
191 | notmuchAtts=() | ||
192 | |||
193 | while IFS= read -r -d $'\n' line; do | ||
194 | [[ ${line} =~ '(attachment|part)\{ ID: ([0-9]+)' ]] || continue | ||
195 | attId=${match[2]} | ||
196 | |||
197 | [[ ${line} =~ 'Content-type: multipart/' ]] && continue | ||
198 | |||
199 | fName="part_${attId}" | ||
200 | [[ ${line} =~ 'Filename: (([^,]|,[^ ])+)' ]] && fName=${match[1]} | ||
201 | |||
202 | if [[ ${#messages} -gt 1 ]]; then | ||
203 | fName="${message}/${fName}" | ||
204 | fi | ||
205 | |||
206 | fExt="${fName:e}" | ||
207 | [[ -n "${fExt}" ]] && fName="${fName:r}" | ||
208 | |||
209 | if [[ -z "${fExt}" && ${line} =~ 'Content-type: (([^,]|,[^ ])+)$' ]]; then | ||
210 | fExt=$(getMimeTypes | grep ${match[1]}$'\t' | head -n 1 | awk '{ print $2; }') | ||
211 | fi | ||
212 | |||
213 | mkdir -p ${fName:h} | ||
214 | if [[ -n "${fExt}" ]]; then | ||
215 | fName=$(mktemp -p . "${fName}.XXXXXX.${fExt}") | ||
216 | else | ||
217 | fName=$(mktemp -p . "${fName}.XXXXXX") | ||
218 | fi | ||
219 | |||
220 | notmuchAtts[${attId}]=${fName} | ||
221 | done <<(notmuch show --decrypt=false -- ${message} | tr -d $'\f') | ||
222 | |||
223 | for attId fName in ${(kv)notmuchAtts}; do | ||
224 | [[ -d ${fName:h} ]] || mkdir -p ${fName:h} | ||
225 | printf "#%d → ‘%s’\n" "${attId}" "${fName}" >&2 | ||
226 | |||
227 | notmuch show --decrypt=false --part=${attId} -- ${message} | pv -W -D 2 -i 0.1 >${fName} | ||
228 | done | ||
229 | done | ||
230 | fi | ||
231 | |||
232 | |||
233 | ${wormhole} && wormhole receive | ||
234 | |||
235 | if ${quickserve}; then | ||
236 | quickserve --root . --upload . --show-hidden --tar gz | ||
237 | fi | ||
238 | |||
239 | |||
240 | if [[ ${#@} -eq 0 ]] || ${forceShell}; then | ||
241 | if [[ ${#@} -gt 0 ]]; then | ||
242 | if [[ -z ${nixShell} ]]; then | ||
243 | ${@} | ||
244 | else | ||
245 | nix-shell ${nixShell} --run "${@}" | ||
246 | fi | ||
247 | fi | ||
248 | |||
249 | cd $(pwd) # Needed for mounting to work | ||
250 | |||
251 | isSingleDir() { | ||
252 | typeset -a contents | ||
253 | contents=(*(N) .*(N)) | ||
254 | |||
255 | if [[ ${#contents} -eq 1 && -d ${contents[1]} ]]; then | ||
256 | print ${contents[1]} | ||
257 | return 0 | ||
258 | else | ||
259 | return 1 | ||
260 | fi | ||
261 | } | ||
262 | while d=$(isSingleDir); do cd ${d}; done | ||
263 | |||
264 | |||
265 | if [[ -z ${nixShell} ]]; then | ||
266 | exec -- zsh | ||
267 | else | ||
268 | exec -- nix-shell ${nixShell} --run zsh | ||
269 | fi | ||
270 | else | ||
271 | if [[ -z ${nixShell} ]]; then | ||
272 | exec -- ${@} | ||
273 | else | ||
274 | exec -- nix-shell ${nixShell} --run "${@}" | ||
275 | fi | ||
276 | fi | ||
277 | ) | ||
278 | } | ||
279 | |||
280 | tmpdir() { | ||
281 | cleanup() | ||
282 | { | ||
283 | cd / | ||
284 | unmount() { | ||
285 | printf "Unmounting %s\n" ${1} >&2 | ||
286 | fusermount -u ${1} || umount ${1} || sudo umount ${1} | ||
287 | } | ||
288 | |||
289 | if mountpoint -q -- ${dir}; then | ||
290 | unmount ${dir} || return $? | ||
291 | else | ||
292 | while read -d $'\0' subDir; do | ||
293 | mountpoint -q -- ${subDir} || continue | ||
294 | unmount ${subDir} || return $? | ||
295 | done <<<$(find ${dir} -xdev -type d -print0 | sort -zr) | ||
296 | fi | ||
297 | |||
298 | rm -rfv --one-file-system -- ${dir} | ||
299 | } | ||
300 | |||
301 | local tmpdir="" | ||
302 | |||
303 | while getopts ':t:a:s:Sd:ir:wqg:n:' arg; do | ||
304 | case $arg in | ||
305 | "t") tmpdir="=${OPTARG}" ;; | ||
306 | "?"|":") printf "Invalid option: %s\n" $arg >&2; exit 2 ;; | ||
307 | esac | ||
308 | done | ||
309 | |||
310 | ( | ||
311 | trap cleanup EXIT | ||
312 | |||
313 | |||
314 | local dir=$(mktemp -ud --tmpdir${tmpdir} ${0}.XXXXXXXXXX || return $?) | ||
315 | |||
316 | dir -d ${dir} ${@} | ||
317 | ) | ||
318 | } | ||
319 | |||
320 | inhibit-sleep() { | ||
321 | if systemctl --user is-active prevent-suspend.service 1>/dev/null; then | ||
322 | echo "Allowing suspend" | ||
323 | systemctl --user stop prevent-suspend.service | ||
324 | else | ||
325 | echo "Inhibiting suspend" | ||
326 | systemctl --user start prevent-suspend.service | ||
327 | fi | ||
328 | } | ||
329 | |||
330 | qr() { | ||
331 | qrencode -l M -o - -t ANSIUTF8 $@ | ||
332 | } | ||
333 | |||
334 | clock() { | ||
335 | tty-clock -sSbc -C 7 -d 0 -a 100000000 | ||
336 | } | ||
337 | |||
338 | public-ip() { | ||
339 | curl -s -H 'Accept: application/json' $@ ifconfig.co | jq -r '.ip' | ||
340 | } | ||
341 | |||
342 | nix-ghci() { | ||
343 | pkgExpr="" | ||
344 | if [[ ${#@} -gt 0 ]]; then | ||
345 | pkgExpr="${1}" | ||
346 | shift | ||
347 | fi | ||
348 | |||
349 | nix-shell -p "with (import <nixpkgs> {}); pkgs.haskellPackages.ghcWithPackages (p: with p; [${pkgExpr}])" --run "ghci ${@}" | ||
350 | } | ||
351 | |||
352 | swap() { | ||
353 | f1=${1} | ||
354 | f2=${2} | ||
355 | |||
356 | if [[ -z "${f1}" || ! -e "${f1}" ]]; then | ||
357 | printf "‘%s’ does not exist\n" "${f1}" >&2 | ||
358 | return 2 | ||
359 | fi | ||
360 | if [[ -z "${f2}" || ! -e "${f2}" ]]; then | ||
361 | printf "‘%s’ does not exist\n" "${f2}" >&2 | ||
362 | return 2 | ||
363 | fi | ||
364 | |||
365 | tmpfile=$(mktemp --dry-run --tmpdir=${f1:h} .swap.XXXXXXXXXX) | ||
366 | mv -v ${f1} ${tmpfile} | ||
367 | mv -v ${f2} ${f1} | ||
368 | mv -v ${tmpfile} ${f2} | ||
369 | } | ||
370 | |||
371 | unalias l | ||
372 | l() { | ||
373 | exa --binary --git --time-style=iso --long --all --header --group-directories-first --colour=always $@ | ||
374 | } | ||
375 | |||
376 | unalias ll | ||
377 | ll() { | ||
378 | l $@ | less --mouse -FR | ||
379 | } | ||
380 | |||
381 | alias '..'='cd ..' | ||
382 | alias -g L='| less' | ||
383 | alias -g S='&> /dev/null' | ||
384 | alias -g G='| grep' | ||
385 | alias -g B='&> /dev/null &' | ||
386 | alias -g BB='&> /dev/null &!' | ||
387 | |||
388 | fancy-ctrl-z () { | ||
389 | emulate -LR zsh | ||
390 | if [[ $#BUFFER -eq 0 ]]; then | ||
391 | [[ -n $(jobs -s) ]] && bg | ||
392 | zle redisplay | ||
393 | else | ||
394 | zle push-input | ||
395 | fi | ||
396 | } | ||
397 | zle -N fancy-ctrl-z | ||
398 | bindkey '^Z' fancy-ctrl-z | ||
399 | |||
400 | export DEFAULT_USER=gkleen | ||
401 | export EDITOR=emacsclient | ||
402 | |||
403 | bindkey -e \ No newline at end of file | ||