]> Sergey Matveev's repositories - zk.zsh.git/blob - zk.zsh
a9b63e7a0b8b67efa6cb8dd4b4695bb5967cb0f4
[zk.zsh.git] / zk.zsh
1 #!/usr/bin/env zsh
2 # zk.zsh -- zettelkästen/wiki/static website helper/generator
3 # Copyright (C) 2022-2023 Sergey Matveev <stargrave@stargrave.org>
4
5 set -e
6 ZK_VERSION=ZKZSH1
7
8 usage() {
9     >&2 <<EOF
10 Usage:
11   \$ $0:t links PAGE
12     Print PAGE's links
13   \$ $0:t backs PAGE
14     Print PAGE's backlinks
15   \$ $0:t htmls DIR
16     Generate HTMLs in DIR
17 EOF
18     exit 1
19 }
20
21 [[ $# -eq 2 ]] || usage
22
23 separator="------------------------ >8 ------------------------"
24 setopt GLOB_STAR_SHORT
25 zmodload -F zsh/stat b:zstat
26 typeset -A pages
27 typeset -A sizes
28 for p (**(.)) {
29     [[ $p:t == "index" ]] && {
30         echo unacceptable filename: $p >&2
31         exit 1
32     }
33     zstat -A mtime -F "%F %T" +mtime $p
34     zstat -A size +size $p
35     pages[$p]=${mtime[1]}
36     sizes[$p]=${size[1]}
37 }
38 typeset -A cats
39 for p (**(/)) cats[$p]=1
40
41 zmodload zsh/mapfile
42 zmodload -F zsh/files b:zf_mkdir
43 typeset -A links
44 typeset -A backs
45 typeset -A cached
46 typeset -aU ws
47 for p (${(k)pages}) {
48     [[ $ZK_CACHE ]] && {
49         zstat -A inode +inode $p
50         zstat -A ctime +ctime $p
51         cache=(${(f)mapfile[$ZK_CACHE/$p]})
52         if [[ ( ${cache[1]} = $ZK_VERSION ) &&
53               ( ${cache[2]} = ${inode[1]} ) &&
54               ( ${cache[3]} = ${ctime[1]} ) ]]; then
55             ws=(${cache[4,-1]})
56             [[ $ws ]] && links[$p]=${(j: :)ws}
57             cached[$p]=1
58             continue
59         fi
60     }
61     ws=()
62     for w (${=mapfile[$p]}) {
63         [[ $w =~ "\[([^] ]+)\]" ]] || continue
64         w=${match[1]}
65         [[ ( $w =~ "/$" ) && ( ${cats[$w[1,-2]]} ) ]] && {
66             ws=($ws $w)
67             continue
68         }
69         [[ ${pages[$w]} ]] || {
70             [[ $ZK_SHOW_MISSING ]] && print "missing $w"
71             continue
72         }
73         ws=($ws $w)
74     }
75     [[ $ZK_CACHE ]] && {
76         zf_mkdir -p $ZK_CACHE/$p:h
77         ws=($ZK_VERSION ${inode[1]} ${ctime[1]} $ws)
78         print -l $ws > $ZK_CACHE/$p
79         ws=(${ws[4,-1]})
80     }
81     [[ $ws ]] && links[$p]=${(j: :)ws}
82 }
83 unset cache ws
84 for p ws (${(kv)links}) {
85     for w (${=ws}) backs[$w]="$p ${backs[$w]}"
86 }
87 for p ws (${(kv)backs}) backs[$p]=${(j: :)${(u)=ws}}
88
89 getrel() {
90     # nearly the copy-paste of Functions/Misc/relative
91     local dst=$2:a
92     local src=$1:h:a
93     local -a cur abs
94     cur=(${(s:/:)src})
95     abs=(${(s:/:)dst:h} $dst:t)
96     integer i=1
97     while [[ i -le $#abs && $abs[i] == $cur[i] ]] ; do
98         ((++i > $#cur)) && {
99             REPLY=${(j:/:)abs[i,-1]}
100             return
101         }
102     done
103     src=${(j:/:)cur[i,-1]/*/..}
104     dst=${(j:/:)abs[i,-1]}
105     REPLY=$src${dst:+/$dst}
106 }
107
108 genHTML() {
109     local page=$1
110     local data p
111     [[ $# -eq 1 ]] && data=${mapfile[$page]} || data=$2
112     local _links=(${(oi)=links[$page]})
113     if [[ ( ${cached[$page]} ) && ( -s $ZK_CACHE/${page}.html ) ]]; then
114         < $ZK_CACHE/${page}.html
115     else
116         data=${data//&/&amp;}
117         data=${data//</&lt;}
118         data=${data//>/&gt;}
119         for p ($_links) {
120             getrel $page $p
121             [[ -d $p ]] && REPLY=$REPLY/index
122             data="${data//\[${p}\]/<a href=\"${REPLY}.html\">[$p]</a>}"
123         }
124         data="<!DOCTYPE html>
125 <html><head>
126 <title>$page (${pages[$page]})</title>
127 <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">
128 </head><body><pre>
129 $data</pre>"
130         if [[ $ZK_CACHE ]]; then
131             print -r "$data" > $ZK_CACHE/${page}.html
132             < $ZK_CACHE/${page}.html
133         else
134             print -r "$data"
135         fi
136     fi
137     if [[ $_links ]]; then
138         print "<hr/>Links:<ul>"
139         for p ($_links) {
140             getrel $page $p
141             print "<li><a href=\"${REPLY}.html\">$p</a> <sup>${pages[$p]}</sup></li>"
142         }
143         print "</ul>"
144     fi
145     local bs=(${(oi)=${backs[$page]}})
146     if [[ $bs ]]; then
147         print "<hr/>Backlinks:<ul>"
148         for p ($bs) {
149             getrel $page $p
150             print "<li><a href=\"${REPLY}.html\">$p</a> <sup>${pages[$p]}</sup></li>"
151         }
152         print "</ul>"
153     fi
154     print "</body></html>"
155 }
156
157 zmodload -F zsh/datetime b:strftime
158 strftime -s now "%F %T"
159
160 genIndex() {
161     local p
162     local entries=()
163     local _links=()
164     typeset -aU _cats=()
165     local curdepth=${#${(s:/:)1}}
166     (( curdepth = curdepth + 1 ))
167     for p (${(k)pages[(I)$1*]}) {
168         case ${#${(As:/:)p}} in
169         ($curdepth) _links=($p $_links) ;;
170         ( $(( $curdepth + 1 )) ) _cats=(${1}${${p#$1}%%/*} $_cats) ;;
171         (*) continue ;;
172         esac
173     }
174     local page=${1}index
175     print "<!DOCTYPE html>
176 <html><head><title>$page ($now)</title></head><body><ul>"
177     for p (${(oi)_links}) {
178         getrel $page $p
179         print "<li><a href=\"${REPLY}.html\">$p</a> <sup>${pages[$p]}</sup> (${sizes[$p]} bytes)</li>"
180     }
181     print "</ul>"
182     if [[ $_cats ]]; then
183         print "<hr/>Subdirectories:<ul>"
184         for p (${(oi)_cats}) {
185             getrel $page $p/index
186             print "<li><a href=\"${REPLY}.html\">$p</a></li>"
187         }
188         print "</ul>"
189     fi
190     local bs=(${(oi)=${backs[$1]}})
191     if [[ $bs ]]; then
192         print "<hr/>Backlinks:<ul>"
193         for p ($bs) {
194             getrel $page $p
195             print "<li><a href=\"${REPLY}.html\">$p</a> <sup>${pages[$p]}</sup></li>"
196         }
197         print "</ul>"
198     fi
199     print "</body></html>"
200 }
201
202 case $1 in
203 (links) for w (${(oi)=${links[$2]}}) print $w ;;
204 (backs) for w (${(oi)=${backs[$2]}}) print $w ;;
205 (html) genHTML $2 ;;
206 (html-index) genIndex $2 ;;
207 (htmls)
208     for p (${(k)pages}) {
209         zf_mkdir -p $2/$p:h
210         genHTML $p > $2/$p.html
211         touch -r $p $2/$p.html
212     }
213     for p (${(k)cats}) genIndex $p/ > $2/$p/index.html
214     genIndex "" > $2/index.html
215     for p ("" ${(k)cats}) touch -d ${now/ /T} $2/$p/index.html
216     ;;
217 (*) usage ;;
218 esac