cookutils view lighttpd/index.cgi @ rev 924

A bunch of small improvements, incl.: cook: not touching main log when packing; find lastcooktime in other logs; summary for packing log; index.cgi: return back after changing the theme; show directories too among the files list, quick jump links; new download page (src & pkgs); new toolchain page; show packing log too.
author Aleksej Bobylev <al.bobylev@gmail.com>
date Sun Jun 11 21:39:35 2017 +0300 (2017-06-11)
parents 5d6cbb571217
children 0b3395599e6f
line source
1 #!/bin/sh
2 #
3 # SliTaz Cooker CGI + Lighttpd web interface.
4 #
6 # Make request URI relative to the script name
7 base="$(dirname "$SCRIPT_NAME")"; [ "$base" == '/' ] && base=''
8 REQUEST_URI=$(echo "$REQUEST_URI" | sed "s|^$base/*|/|; s|\?.*||")
10 # Split the URI request to /pkg/cmd/arg
11 export pkg=$(echo "$REQUEST_URI" | cut -d/ -f2)
12 export cmd=$(echo "$REQUEST_URI" | cut -d/ -f3)
13 export arg=$(echo "$REQUEST_URI" | sed 's|^/[^/]*/[^/]*/||')
16 . /usr/lib/slitaz/httphelper.sh
18 [ -f "/etc/slitaz/cook.conf" ] && . /etc/slitaz/cook.conf
19 [ -f "./cook.conf" ] && . ./cook.conf
20 wok="$WOK"
21 title=${title:-SliTaz Cooker}
22 # Cooker DB files.
23 activity="$CACHE/activity"
24 commits="$CACHE/commits"
25 cooklist="$CACHE/cooklist"
26 cookorder="$CACHE/cookorder"
27 command="$CACHE/command"; touch $command
28 blocked="$CACHE/blocked"
29 broken="$CACHE/broken"
30 cooknotes="$CACHE/cooknotes"
31 cooktime="$CACHE/cooktime"
32 wokrev="$CACHE/wokrev"
34 # Path to markdown to html convertor
35 cmark_opts='--smart -e table -e strikethrough -e autolink -e tagfilter'
36 if [ -n "$(which cmark 2>/dev/null)" ]; then
37 md2html="$(which cmark) $cmark_opts"
38 elif [ -x "./cmark" ]; then
39 md2html="./cmark $cmark_opts"
40 elif [ -n "$(which sundown 2>/dev/null)" ]; then
41 md2html=$(which sundown)
42 elif [ -x "./sundown" ]; then
43 md2html="./sundown"
44 fi
49 # Search form redirection
50 if [ -n "$(GET search)" ]; then
51 echo -e "HTTP/1.1 301 Moved Permanently\nLocation: $base/$(GET q)\n\n"
52 exit 0
53 fi
56 # Show the running command and it's progression
58 running_command() {
59 state="$(cat $command)"
60 local pct=''
61 if [ -n "$state" ];then
62 echo -n "$state</td></tr><tr><td>Completion</td>"
63 set -- $(grep "^$state" $cooktime)
64 [ -n "$1" -a $2 -ne 0 ] && pct=$((($(date +%s)-$3)*100/$2))
65 [ -n "$pct" ] && max="max='100'"
66 echo -n "<td><progress id='gauge' $max value='$pct' title='Click to stop updating' onclick='stopUpdating()'>"
67 echo -n "</progress> <span id='pct'>${pct:-?}%</span>"
68 [ "$2" -gt 60 ] &&
69 echo -n "</td></tr><tr><td>Estimated end time</td><td>$(date +%H:%M -ud @$(($2+$3)))"
70 else
71 echo 'not running'
72 fi
73 }
76 # HTML page header
78 page_header() {
79 local theme t='' css
80 theme=$(COOKIE theme)
81 [ "$theme" == 'default' ] && theme=''
82 [ -n "$theme" ] && theme="-$theme"
83 css="cooker$theme.css"
85 echo -e 'Content-Type: text/html; charset=UTF-8\n'
87 cat <<EOT
88 <!DOCTYPE html>
89 <html lang="en">
90 <head>
91 <title>$([ -n "$pkg" ] && echo "$pkg - ")$title</title>
92 <link rel="stylesheet" href="/$css">
93 <link rel="icon" type="image/png" href="/slitaz-cooker.png">
94 <link rel="search" href="$base/os.xml" title="$title" type="application/opensearchdescription+xml">
95 <!-- mobile -->
96 <meta name="viewport" content="width=device-width, initial-scale=1.0">
97 <meta name="theme-color" content="#222">
98 <!-- rss -->
99 <link rel="alternate" type="application/rss+xml" title="$title Feed" href="?rss">
100 </head>
101 <body>
102 <div id="container">
103 <header>
104 <h1><a href="$base/">$title</a></h1>
105 <div class="network">
106 <a href="http://www.slitaz.org/">Home</a>
107 <a href="http://bugs.slitaz.org/">Bugs</a>
108 <a href="http://hg.slitaz.org/wok-next/">Hg</a>
109 <a href="http://roadmap.slitaz.org/">Roadmap</a>
110 <a href="http://pizza.slitaz.me/">Pizza</a>
111 <a href="http://tank.slitaz.org/">Tank</a>
112 |
113 <a href="$base/cross/">Cross</a>
114 <a href="$base/i486.cgi">i486</a>
115 <a href="$base/cookiso.cgi">ISO</a>
116 <select onChange="window.location.href=this.value" style="display: none">
117 <option value=".">Go to…</option>
118 <option value="http://www.slitaz.org/">Home</option>
119 <option value="http://bugs.slitaz.org/">Bug tracker</option>
120 <option value="http://hg.slitaz.org/wok/">Hg wok</option>
121 <option value="http://roadmap.slitaz.org/">Roadmap</option>
122 <option value="http://pizza.slitaz.me/">Pizza</option>
123 <option value="http://tank.slitaz.org/">Tank</option>
124 <option disabled>---------</option>
125 <option value="cross/">Cross</option>
126 <option value="i486.cgi">i486</option>
127 <option value="cookiso.cgi">ISO</option>
128 </select>
129 </div>
130 </header>
132 <main>
133 EOT
135 [ -n "$(GET debug)" ] && echo "<pre><code class='language-ini'>$(env | sort)</code></pre>"
136 }
139 # HTML page footer
141 page_footer() {
142 date_now=$(date +%s)
143 sec_now=$(date +%S); sec_now=${sec_now#0} # remove one leading zero
144 wait_sec=$(( 60 - $sec_now ))
145 cat <<EOT
146 </main>
148 <footer>
149 <a href="http://www.slitaz.org/">SliTaz Website</a>
150 <a href="$base/">Cooker</a>
151 <a href="$base/doc/cookutils/cookutils.html">Documentation</a>
152 <a href="$base/?theme">Theme</a>
153 </footer>
154 </div>
155 <script src="/cooker.js"></script>
156 <script>refreshDate(${wait_sec}000, ${date_now}000)</script>
157 </body>
158 </html>
159 EOT
160 }
163 show_note() {
164 echo "<div class='bigicon-$1'>$2</div>"
165 }
168 not_found() {
169 local file="${1#$PKGS/}"; file="${file#$LOGS/}"; file="${file#$WOK/}"
170 echo "HTTP/1.1 404 Not Found"
171 page_header
172 echo "<h2>Not Found</h2>"
173 case $2 in
174 pkg)
175 show_note e "The requested package “$(basename "$(dirname "$file")")” was not found." ;;
176 *)
177 show_note e "The requested file “$file” was not found." ;;
178 esac
179 page_footer
180 }
183 manage_modified() {
184 local file="$1" option="$2" nul day mon year time hh mm ss date_s
185 if [ ! -f "$file" ]; then
186 if [ "$option" == 'silently-absent' ]; then
187 echo "HTTP/1.1 404 Not Found"
188 return
189 else
190 not_found "$file" "$2"
191 exit
192 fi
193 fi
194 [ "$option" == 'no-last-modified' ] && return
195 if [ -n "$HTTP_IF_MODIFIED_SINCE" ]; then
196 echo "$HTTP_IF_MODIFIED_SINCE" | \
197 while read nul day mon year time nul; do
198 case $mon in
199 Jan) mon='01';; Feb) mon='02';; Mar) mon='03';; Apr) mon='04';;
200 May) mon='05';; Jun) mon='06';; Jul) mon='07';; Aug) mon='08';;
201 Sep) mon='09';; Oct) mon='10';; Nov) mon='11';; Dec) mon='12';;
202 esac
203 hh=$(echo $time | cut -d: -f1)
204 mm=$(echo $time | cut -d: -f2)
205 ss=$(echo $time | cut -d: -f3)
206 date_s=$(date -ud "$year$mon$day$hh$mm.$ss" +%s)
207 # if [ "$date_s" -ge "$(date -ur "$file" +%s)" ]; then
208 # echo -e 'HTTP/1.1 304 Not Modified\n'
209 # exit
210 # fi
211 # TODO: improve caching control
212 done
213 fi
214 echo "Last-Modified: $(date -Rur "$file" | sed 's|UTC|GMT|')"
215 echo "Cache-Control: public, max-age=3600"
216 }
219 # Query '?pct=<package>': update percentage
221 if [ -n "$(GET pct)" ]; then
222 pkg="$(GET pct)"
223 state="$(cat $command)"
224 if [ "$state" == "cook:$pkg" ]; then
225 set -- $(grep "^$state" $cooktime)
226 [ -n "$1" ] && pct=$(( ($(date +%s) - $3) * 100 / $2 ))
227 echo "${pct:-?}"
228 else
229 echo 'reload'
230 fi
231 exit 0
232 fi
235 # Query '?poke': poke cooker
237 if [ -n "$(GET poke)" ]; then
238 touch $CACHE/cooker-request
239 echo -e "Location: ${HTTP_REFERER:-${REQUEST_URI%\?*}}\n"
240 exit
241 fi
244 # Query '?recook=<package>': query to recook package
246 if [ -n "$(GET recook)" ]; then
247 pkg="$(GET recook)"
248 case "$HTTP_USER_AGENT" in
249 *SliTaz*)
250 grep -qs "^$pkg$" $CACHE/recook-packages ||
251 echo "$pkg" >> $CACHE/recook-packages
252 esac
253 echo -e "Location: ${HTTP_REFERER:-${REQUEST_URI%\?*}}\n"
254 exit
255 fi
258 # Query '/i/<log>/<pkg>': show indicator icon
259 # Can't use ?query - not able to change '+' to '%2B' in the sed rules (see log handler)
261 if [ "$pkg" == 'i' ]; then
262 echo -en "Content-Type: image/svg+xml\n\n<svg xmlns='http://www.w3.org/2000/svg' height='12' width='8'><path d='"
263 if [ $LOGS/$cmd -nt $PKGS/$arg.tazpkg ]; then
264 echo "m1 2-1 1v8l1 1h6l1-1v-8l-1-1z' fill='#090'/></svg>"
265 else
266 echo "m0 3v8l1 1h6l1-1v-8l-1-1h-6zm3 0h2v5h-2zm0 6h2v2h-2z' fill='#d00'/></svg>"
267 fi
268 exit
269 fi
272 # Query '?theme[=<theme>]': change UI theme
274 if [ -n "$(GET theme)" ]; then
275 theme="$(GET theme)"
276 ref="$(echo "$HTTP_REFERER" | sed 's|:|%3A|g; s|/|%2F|g; s|\?|%3F|g; s|\+|%2B|g;')"
277 case $theme in
278 theme)
279 current=$(COOKIE theme)
280 page_header
281 cat <<EOT
282 <section>
283 <h2>Change theme</h2>
284 <p>Current theme: “${current:-default}”. Select other:</p>
285 <ul>
286 $(
287 for i in default emerald sky goldenrod midnight like2016 terminal; do
288 [ "$i" == "${current:-default}" ] || echo "<li><a href=\"$base/?theme=$i&amp;ref=$ref\">$i</a></li>"
289 done
290 )
291 </ul>
292 </section>
293 EOT
294 page_footer
295 exit 0
296 ;;
297 default|emerald|sky|goldenrod|midnight|like2016|terminal)
298 ref="$(GET ref)"
299 [ -n "$ref" ] || ref="$base/"
300 # Expires in a year
301 expires=$(date -uRd @$(($(date +%s)+31536000)) | sed 's|UTC|GMT|')
302 echo -e "HTTP/1.1 302 Found\nLocation: $ref\nCache-Control: no-cache\nSet-Cookie: theme=$theme; expires=$expires\n\n"
303 exit 0
304 ;;
305 esac
306 fi
309 #case "$QUERY_STRING" in
310 # stuff*)
311 # file="$wok/$(GET stuff)"
312 # manage_modified "$file"
313 # ;;
314 #
315 # pkg=*|receipt=*|description=*|files=*|log=*|man=*|doc=*|info=*)
316 # type=${QUERY_STRING%%=*}
317 # pkg=$(GET $type)
318 # case "$type" in
319 # description)
320 # manage_modified "$wok/$pkg/receipt" 'no-last-modified'
321 # manage_modified "$wok/$pkg/description.txt" 'silently-absent'
322 # ;;
323 # log)
324 # manage_modified "$wok/${pkg%%.log*}/receipt" 'no-last-modified'
325 # manage_modified "$LOGS/$pkg"
326 # ;;
327 # *)
328 # manage_modified "$wok/$pkg/receipt" pkg
329 # ;;
330 # esac
331 # ;;
332 #esac
335 # RSS feed generator
336 # URI: ?rss[&limit={1..100}]
338 if [ -n "$(GET rss)" ]; then
339 limit=$(GET limit); limit="${limit:-12}"; [ "$limit" -gt 100 ] && limit='100'
340 pubdate=$(date -Rur$(ls -t $FEEDS/*.xml | head -n1) | sed 's|UTC|GMT|')
341 cooker_url="http://$HTTP_HOST$base/"
342 cat <<EOT
343 Content-Type: application/rss+xml
345 <?xml version="1.0" encoding="utf-8" ?>
346 <rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
347 <channel>
348 <title>$title</title>
349 <description>The SliTaz packages cooker feed</description>
350 <link>$cooker_url</link>
351 <lastBuildDate>$pubdate</lastBuildDate>
352 <pubDate>$pubdate</pubDate>
353 <atom:link href="$cooker_url?rss" rel="self" type="application/rss+xml" />
354 EOT
355 for rss in $(ls -t $FEEDS/*.xml | head -n$limit); do
356 sed "s|http[^=]*=|$cooker_url|; s|<guid|& isPermaLink=\"false\"|g; s|</pubDate| GMT&|g" $rss
357 done
358 cat <<EOT
359 </channel>
360 </rss>
361 EOT
362 exit 0
363 fi
366 ### OpenSearch ###
368 # Query '/os.xml': get OpenSearch Description
370 if [ "$pkg" == 'os.xml' ]; then
371 cat <<EOT
372 Content-Type: application/xml; charset=UTF-8
374 <?xml version="1.0" encoding="UTF-8"?>
375 <OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/">
376 <ShortName>$title</ShortName>
377 <Description>SliTaz packages search</Description>
378 <Image width="16" height="16" type="image/png">http://$HTTP_HOST/images/logo.png</Image>
379 <Url type="text/html" method="GET" template="http://$HTTP_HOST$base/{searchTerms}"/>
380 <Url type="application/x-suggestions+json" method="GET" template="http://$HTTP_HOST$base/">
381 <Param name="oss" value="{searchTerms}"/>
382 </Url>
383 <SearchForm>http://$HTTP_HOST$base/</SearchForm>
384 <InputEncoding>UTF-8</InputEncoding>
385 </OpenSearchDescription>
386 EOT
387 exit 0
388 fi
390 # Query '?oss[=<term>]': OpenSearch suggestions
392 if [ -n "$(GET oss)" ]; then
393 term="$(GET oss | tr -cd '[:alnum:]+-')"
394 echo -e 'Content-Type: application/x-suggestions+json; charset=UTF-8\n'
395 cd $wok
396 ls | fgrep "$term" | head -n10 | awk -vterm="$term" '
397 BEGIN{printf("[\"%s\",[", term)}
398 {printf("%s\"%s\"", NR != 1 ? "," : "", $0)}
399 END {printf("]]")}
400 '
401 exit 0
402 fi
407 #
408 # Functions
409 #
412 # Unpack to stdout
414 docat() {
415 case "$1" in
416 *gz) zcat ;;
417 *bz2) bzcat ;;
418 *xz) xzcat ;;
419 *) cat
420 esac < $1
421 }
424 # Tiny texinfo converter
426 info2html() {
427 sed \
428 -e 's|&|\&amp;|g; s|<|\&lt;|g; s|>|\&gt;|g' \
429 -e 's|^\* \(.*\)::|* <a href="#\1">\1</a> |' \
430 -e 's|\*note \(.*\)::|<a href="#\1">\1</a>|' \
431 -e '/^File: / s|(dir)|Top|g' \
432 -e '/^File: / s|Next: \([^,]*\)|<a class="button" href="#\1">Next: \1</a>|' \
433 -e '/^File: / s|Prev: \([^,]*\)|<a class="button" href="#\1">Prev: \1</a>|' \
434 -e '/^File: / s|Up: \([^,]*\)|<a class="button" href="#\1">Up: \1</a>|' \
435 -e '/^File: / s|^.* Node: \([^,]*\), *\(.*\)$|<pre id="\1">\2|' \
436 -e '/^<pre id=/ s|^\([^>]*>\)\(<a[^>]*>Next: [^,]*\), *\(<a[^>]*>Prev: [^,]*\), *\(<a[^>]*>Up: .*\)|\1 \3 \4 \2|' \
437 -e '/^Tag Table:$/,/^End Tag Table$/d' \
438 -e '/INFO-DIR/,/^END-INFO-DIR/d' \
439 -e "s|https*://[^>),'\"\`’ ]*|<a href=\"&\">&</a>|g" \
440 -e "s|ftp://[^>),\"\` ]*|<a href=\"&\">&</a>|g" \
441 -e 's|^\* Menu:|<b>Menu:</b>|' \
442 -e "s|^|</pre>|"
443 }
446 # Put some colors into log and DB files.
448 syntax_highlighter() {
449 case $1 in
450 log)
451 # If variables not defined - define them with some rare values
452 : ${_src=#_#_#}
453 : ${_install=#_#_#}
454 : ${_fs=#_#_#}
455 : ${_stuff=#_#_#}
456 # Use one-letter html tags to save some bytes :)
457 # <b>is error (red)</b> <u>is warning (orange)</u> <i>is informal (green)</i>
458 sed -e 's/&/\&amp;/g; s/</\&lt;/g; s/>/\&gt;/g' \
459 -e 's#OK$#<i>OK</i>#' \
460 -e 's#\([Dd]one\)$#<i>\1</i>#' \
461 -e 's#Success$#<i>Success</i>#' \
462 -e 's#\([^a-z]\)ok$#\1<i>ok</i>#' \
463 -e 's#\([^a-z]\)yes$#\1<i>yes</i>#' \
464 -e 's#\([^a-z]\)ON$#\1<i>ON</i>#' \
465 -e 's#\([^a-z]\)no$#\1<u>no</u>#' \
466 -e 's#\([^a-z]\)none$#\1<u>none</u>#' \
467 -e 's#\([^a-z]\)false$#\1<u>false</u>#' \
468 -e 's#\([^a-z]\)OFF$#\1<u>OFF</u>#' \
469 -e 's#\(^checking .*\.\.\. \)\(.*\)$#\1<i>\2</i>#' \
470 \
471 -e 's#\( \[Y[nm/]\?\] n\)$# <u>\1</u>#' \
472 -e 's#\( \[N[ym/]\?\] y\)$# <i>\1</i>#' \
473 -e 's# y$# <i>y</i>#' \
474 -e 's# n$# <u>n</u>#' \
475 -e 's#(NEW) *$#<b>(NEW)</b>#' \
476 \
477 -e 's#.*(pkg/local).*#<i>\0</i>#' \
478 -e 's#.*(web/cache).*#<u>\0</u>#' \
479 \
480 -e 's#\([^a-zA-Z]\)\([Ee]rror\)$#\1<b>\2</b>#' \
481 -e 's#ERROR:#<b>ERROR:</b>#g' \
482 \
483 -e 's#^.*[Ff][Aa][Ii][Ll][Ee][Dd].*#<b>\0</b>#' \
484 -e 's#^.*[Ff]atal.*#<b>\0</b>#' \
485 -e 's#^.*[Nn]ot found.*#<b>\0</b>#' \
486 -e 's#^.*[Nn]o such file.*#<b>\0</b>#' \
487 -e 's#^.*No package .* found.*#<b>\0</b>#' \
488 -e 's#^.*Unable to find.*#<b>\0</b>#' \
489 -e 's#^.*[Ii]nvalid.*#<b>\0</b>#' \
490 -e 's#\([Nn][Oo][Tt] found\.*\)$#<b>\1</b>#' \
491 -e 's#\(found\.*\)$#<i>\1</i>#' \
492 \
493 -e 's#^.*WARNING:.*#<u>\0</u>#' \
494 -e 's#^.*warning:.*#<u>\0</u>#' \
495 -e 's#^.* [Ee]rror:* .*#<b>\0</b>#' \
496 -e 's#^.*terminated.*#<b>\0</b>#' \
497 -e 's#\(missing\)#<b>\1</b>#g' \
498 -e 's#^.*[Cc]annot find.*#<b>\0</b>#' \
499 -e 's#^.*unrecognized options.*#<u>\0</u>#' \
500 -e 's#^.*does not.*#<u>\0</u>#' \
501 -e 's#^.*[Ii]gnoring.*#<u>\0</u>#' \
502 -e 's#^.*note:.*#<u>\0</u>#' \
503 \
504 -e 's#^.* will not .*#<u>\0</u>#' \
505 -e 's!^Hunk .* succeeded at .*!<u>\0</u>!' \
506 -e 's#^.* Warning: .*#<u>\0</u>#' \
507 \
508 -e "s#^Executing:\([^']*\).#<em>\0</em>#" \
509 -e "s#^Making.*#<em>\0</em>#" \
510 -e "s#^Scanning dependencies of target .*#<em>\0</em>#" \
511 -e "s#^====\([^']*\).#<span class='span-line'>\0</span>#g" \
512 -e "s#^[a-zA-Z0-9]\([^']*\) :: #<span class='span-sky'>\0</span>#g" \
513 -e "s#[fh]tt*ps*://[^ '\"]*#<a href='\0'>\0</a>#g" \
514 \
515 -e "s|$_src|<span class='var'>\${src}</span>|g;
516 s|$_install|<span class='var'>\${install}</span>|g;
517 s|$_fs|<span class='var'>\${fs}</span>|g;
518 s|$_stuff|<span class='var'>\${stuff}</span>|g" \
519 -e "s|\[9\([1-6]\)m|<span class='c\10'>|;
520 s|\[39m|</span>|;"
521 ;;
523 files)
524 # Highlight the Busybox's `ls` output
525 awk '{
526 part1 = substr($0, 0, 16);
527 part2 = substr($0, 17, 9);
528 part3 = substr($0, 26, 9);
529 part4 = substr($0, 35);
530 if (part2 != "root ") part2 = "<span class=\"c11\">" part2 "</span>";
531 if (part3 != "root ") part3 = "<span class=\"c11\">" part3 "</span>";
532 print part1 part2 part3 part4;
533 }' | \
534 sed "s|\[0m/|/\[0m|g;
535 s|\[\([01]\);3\([1-7]\)m|<a class='c\2\1'>|g;
536 s|\[\([01]\);0m|<a class='c0\1'>|g;
537 s|\[0m|</a>|g;
538 s|^\(lrwxrwxrwx\)|<span class='c61'>\1</span>|;
539 s|^\(-rwxr-xr-x\)|<span class='c21'>\1</span>|;
540 s|^\(-rw-r--r--\)|<span class='c31'>\1</span>|;
541 s|^\(drwxr-xr-x\)|<span class='c41'>\1</span>|;
542 s|^\([lrwxs-]*\)|<span class='c11'>\1</span>|;
543 "
544 ;;
545 esac
546 }
549 show_code() {
550 echo -n "<pre><code class=\"language-$1\">"
551 sed 's|&|\&amp;|g; s|<|\&lt;|g; s|>|\&gt;|g'
552 echo '</code></pre>'
553 }
556 datalist() {
557 (
558 cd $wok
560 ls | awk '
561 BEGIN{printf("<datalist id=\"packages\">")}
562 {printf("<option>%s</option>",$1)}
563 END {printf("</datalist>")}
564 '
565 )
566 }
569 mklog() {
570 awk '
571 BEGIN { printf("<pre class=\"log dog\">\n") }
572 { print }
573 END { print "</pre>" }'
574 }
577 summary() {
578 log="$1"
579 pkg="$(basename ${log%%.log*})"
581 if [ -f "$log" ]; then
582 if grep -q "cook:$pkg$" $command; then
583 show_note i "The Cooker is currently building $pkg"
584 elif fgrep -q "Summary for:" $log; then
585 sed '/^Summary for:/,$!d' $log | awk '
586 BEGIN { print "<section>" }
587 function row(line) {
588 split(line, s, " : ");
589 printf("\t<tr><td>%s</td><td>%s</td></tr>\n", s[1], s[2]);
590 }
591 function row2(line, rowNum) {
592 split(line, s, " : ");
593 if (rowNum == 1)
594 printf("\t<tr><th>%s</th><th>%s</th><th>%s</th><th>%s</th><th>%s</th></tr>\n", s[1], s[2], s[3], s[4], s[5]);
595 else
596 printf("\t<tr><td>%s</td><td>%s</td><td>%s</td><td>%s</td><td>%s</td></tr>\n", s[1], s[2], s[3], s[4], s[5]);
597 }
598 {
599 if (NR==1) { printf("<h3>%s</h3>\n<table>\n", $0); next }
600 if ($0 ~ "===") { seen++; if (seen == 1) next; else exit; }
601 if ($0 ~ "---") {
602 seen2++;
603 if (seen2 == 1) printf("</table>\n<table class=\"pkgslist\">\n")
604 next
605 }
606 if (seen2) row2($0, seen2); else row($0);
607 }
608 END { print "</table></section>" }
609 '
610 elif fgrep -q "Debug information" $log; then
611 echo -e '<section>\n<h3>Debug information</h3>'
612 sed -e '/^Debug information/,$!d; /^===/d; /^$/d' $log | sed -n '1!p' | \
613 if [ -n "$2" ]; then
614 syntax_highlighter log | sed 's|\([0-9][0-9]*\):|<a href="#l\1">\1</a>:|'
615 else
616 sed 's|^[0-9][0-9]*:||' | syntax_highlighter log
617 fi | mklog
618 echo '</section>'
619 fi
620 else
621 [ -n "$pkg" -a -d "$wok/$pkg" ] && show_note e "No log for $pkg"
622 fi
623 }
626 active() {
627 [ "$cmd" == "$1" -o "$cmd" == "${2:-$1}" ] && echo -n ' active'
628 }
631 pkg_info() {
632 local log active bpkg
633 log="$LOGS/$pkg.log"
635 echo "<h2>$pkg</h2>"
636 echo '<div id="info">'
637 echo "<a class='button icon receipt$(active receipt stuff)' href='$base/$pkg/receipt'>receipt &amp; stuff</a>"
639 unset WEB_SITE WANTED
640 . $wok/$pkg/receipt
642 [ -n "$WEB_SITE" ] &&
643 echo "<a class='button icon website' href='$WEB_SITE' target='_blank' rel='noopener noreferrer'>web site</a>"
645 if [ -f "$wok/$pkg/taz/$PACKAGE-$VERSION/receipt" ]; then
646 echo "<a class='button icon files$(active files)' href='$base/$pkg/files'>files</a>"
648 [ -n "$(ls $wok/$pkg/description*.txt)" ] &&
649 echo "<a class='button icon desc$(active description)' href='$base/$pkg/description'>description</a>"
651 [ -n "$TARBALL" -a -s "$SRC/$TARBALL" -o -d "$wok/$pkg/taz" ] &&
652 echo "<a class='button icon download' href='$base/$pkg/download'>download</a>"
653 fi
655 echo "<a class='button icon browse' href='$base/$pkg/browse/'>browse</a>"
657 [ -x ./man2html -a -d "$wok/$pkg/install/usr/share/man" ] &&
658 echo "<a class='button icon doc$(active man)' href='$base/$pkg/man/'>man</a>"
660 [ -d "$wok/$pkg/install/usr/share/doc" -o -d "$wok/$pkg/install/usr/share/gtk-doc" ] &&
661 echo "<a class='button icon doc$(active doc)' href='$base/$pkg/doc/'>doc</a>"
663 [ -d "$wok/$pkg/install/usr/share/info" ] &&
664 echo "<a class='button icon doc$(active info)' href='$base/$pkg/info/#Top'>info</a>"
666 [ -s "$log" ] &&
667 echo "<a class='button icon log$(active log)' href='$base/$pkg/log/'>logs</a>"
669 echo '</div>'
670 }
673 mktable() {
674 sed 's# : #|#' | awk -vc="$1" '
675 BEGIN { printf("<table class=\"%s\">\n", c); FS="|" }
676 { printf("<tr><td>%s</td>", $1);
677 if (NF == 2) printf("<td>%s</td>", $2);
678 printf("</tr>\n", $2) }
679 END { print "</table>" }'
680 }
683 section() {
684 local i=$(basename "$1")
685 echo -e '\n\n<section>'
686 [ $(wc -l < $1) -gt $2 ] && echo "<a class='button icon more r' href='?$i'>${3#*|}</a>"
687 echo "<h2>${3%|*}</h2>"
688 mktable "$i"
689 echo '</section>'
690 }
693 show_desc() {
694 echo "<section><h3>Description of “$1”</h3>"
695 if [ -n "$md2html" ]; then
696 $md2html $2
697 else
698 show_code markdown < $2
699 fi
700 echo "</section>"
701 }
704 # Return all the names of packages bundled in this receipt
706 all_names() {
707 local split=" $SPLIT "
708 if [ "${split/ $PACKAGE /}" != "$split" ]; then
709 # $PACKAGE included somewhere in $SPLIT (probably in the end).
710 # We should build packages in the order defined in the $SPLIT.
711 echo $SPLIT
712 else
713 # We'll build the $PACKAGE, then all defined in the $SPLIT.
714 echo $PACKAGE $SPLIT
715 fi
716 }
719 toolchain_version() {
720 echo "<tr><td><a href='$base/$1'>$1</a></td><td>"
721 if [ -e "$WOK/$1/receipt" ]; then
722 grep ^VERSION $WOK/$1/receipt | cut -d '"' -f2
723 echo '</td><td>'
724 grep ^SHORT_DESC $WOK/$1/receipt | cut -d '"' -f2
725 else
726 echo -n '---</td><td>---'
727 fi
728 echo "</td></tr>"
729 }
732 files_header() {
733 echo '<section><h3>Available downloads:</h3>'
734 echo '<table><thead><tr><th>File</th><th>Size</th><th>Description</th></tr></thead><tbody>'
735 }
740 #
741 # Load requested page
742 #
744 if [ -z "$pkg" ]; then
746 page_header
747 if [ -n "$QUERY_STRING" -a "$QUERY_STRING" != 'debug' ]; then
749 for list in activity cooknotes cooklist; do
750 [ -n "$(GET $list)" ] || continue
751 [ "$list" == 'cooklist' ] && nb="- Packages: $(wc -l < $cooklist)"
752 echo '<section id="content2">'
753 echo "<h2>DB: $list $nb</h2>"
754 tac $CACHE/$list | sed 's|cooker.cgi?pkg=||; s|%2B|+|g;
755 s|\[ Done|<span class="r c20">Done|;
756 s|\[ Failed|<span class="r c10">Failed|;
757 s| \]|</span>|' | mktable $list
758 echo '</section>'
759 done
761 if [ -n "$(GET broken)" ]; then
762 echo '<div id="content2">'
763 echo "<h2>DB: broken - Packages: $(wc -l < $broken)</h2>"
764 sort $CACHE/broken | sed "s|^[^']*|<a href='$base/\0'>\0</a>|g" | mktable
765 echo '</div>'
766 fi
768 case "$QUERY_STRING" in
769 *.log)
770 log=$LOGS/$QUERY_STRING
771 name=$(basename $log)
772 if [ -f "$log" ]; then
773 echo "<h2>Log for: ${name%.log}</h2>"
774 if fgrep -q "Summary" $log; then
775 echo '<pre class="log">'
776 grep -A 20 '^Summary' $log | syntax_highlighter log
777 echo '</pre>'
778 fi
779 echo '<pre class="log">'
780 syntax_highlighter log < $log
781 echo '</pre>'
782 else
783 show_note e "No log file: $log"
784 fi
785 ;;
786 toolchain)
787 cat <<EOT
788 <div id="content2">
789 <section>
790 <h2>SliTaz GNU/Linux toolchain</h2>
792 <table>
793 <tr><td>Build date</td> <td colspan="2">$(sed -n '/^Cook date/s|[^:]*: \(.*\)|\1|p' $LOGS/slitaz-toolchain.log)</td></tr>
794 <tr><td>Build duration</td> <td colspan="2">$(sed -n '/^Cook time/s|[^:]*: \(.*\)|\1|p' $LOGS/slitaz-toolchain.log)</td></tr>
795 <tr><td>Architecture</td> <td colspan="2">$ARCH</td></tr>
796 <tr><td>Build system</td> <td colspan="2">$BUILD_SYSTEM</td></tr>
797 <tr><td>Host system</td> <td colspan="2">$HOST_SYSTEM</td></tr>
798 <tr><th>Package</th><th>Version</th><th>Description</th></tr>
799 $(toolchain_version slitaz-toolchain)
800 $(toolchain_version binutils)
801 $(toolchain_version linux-api-headers)
802 $(toolchain_version gcc)
803 $(toolchain_version glibc)
804 </table>
806 <p>Toolchain documentation: <a target="_blank" rel="noopener noreferrer"
807 href="http://doc.slitaz.org/en:cookbook:toolchain">http://doc.slitaz.org/en:cookbook:toolchain</a>
808 </p>
810 </section>
811 </div>
812 EOT
813 ;;
814 esac
815 page_footer
816 exit 0
817 fi
820 # We may have a toolchain.cgi script for cross cooker's
821 if [ -f "toolchain.cgi" ]; then
822 toolchain="toolchain.cgi"
823 else
824 toolchain="?toolchain"
825 fi
826 # Main page with summary. Count only packages included in ARCH,
827 # use 'cooker arch-db' to manually create arch.$ARCH files.
828 inwok=$(ls $WOK/*/arch.$ARCH | wc -l)
829 cooked=$(ls -d $WOK/*/taz | wc -l)
830 unbuilt=$(($inwok - $cooked))
831 pct=0; [ $inwok -gt 0 ] && pct=$(( ($cooked * 100) / $inwok ))
832 cat <<EOT
833 <div id="content2">
835 <section>
836 <form method="get" action="" class="search r">
837 <input type="hidden" name="search" value="pkg"/>
838 <button type="submit" title="Search">Search</button>
839 <input type="search" name="q" placeholder="Package" list="packages" autocorrect="off" autocapitalize="off"/>
840 </form>
842 <h2>Summary</h2>
843 EOT
845 mktable <<EOT
846 Cooker state : $(running_command)
847 Wok revision : <a href='$WOK_URL' target='_blank' rel='noopener noreferrer'>$(cat $wokrev)</a>
848 Commits to cook : $(wc -l < $commits)
849 Current cooklist : $(wc -l < $cooklist)
850 Broken packages : $(wc -l < $broken)
851 Blocked packages : $(wc -l < $blocked)
852 Architecture : $ARCH, <a href="$toolchain">toolchain</a>
853 Server date : <span id='date'>$(date -u '+%F %R %Z')</span>
854 EOT
856 # If command is "cook:*", update gauge and percentage periodically.
857 # If different package is cooking, reload the page (with new settings)
858 cmd="$(cat $command)"
859 case "$cmd" in
860 cook:*)
861 pkg=${cmd#*:}
862 echo "<script>updatePkg = '${pkg//+/%2B}';</script>"
863 ;;
864 esac
866 if [ -e "$CACHE/cooker-request" -a ! -s $command ]; then
867 if [ "$activity" -nt "$CACHE/cooker-request" ]; then
868 echo '<a class="button icon bell r" href="?poke">Wake up</a>'
869 else
870 show_note i 'Cooker will be launched in the next 5 minutes.'
871 fi
872 fi
874 cat <<EOT
875 <p>Receipts in the wok: $inwok total · $cooked cooked · $unbuilt unbuilt</p>
877 <div class="meter"><progress max="100" value="$pct">${pct}%</progress><span>${pct}%</span></div>
879 <p>
880 Service logs:
881 <a href="?cookorder.log">cookorder</a> ·
882 <a href="?commits.log">commits</a> ·
883 <a href="?pkgdb.log">pkgdb</a>
884 </p>
885 </section>
886 EOT
888 tac $activity | head -n12 | sed 's|cooker.cgi?pkg=||;
889 s|\[ Done|<span class="r c20">Done|;
890 s|\[ Failed|<span class="r c10">Failed|;
891 s| \]|</span>|;
892 s|%2B|\+|g' | \
893 section $activity 12 "Activity|More activity"
895 [ -s "$cooknotes" ] && tac $cooknotes | head -n12 | \
896 section $cooknotes 12 "Cooknotes|More notes"
898 [ -s "$commits" ] &&
899 section $commits 20 "Commits|More commits" < $commits
901 [ -s "$cooklist" ] && head -n 20 $cooklist | \
902 section $cooklist 20 "Cooklist|Full cooklist"
904 [ -s "$broken" ] && head -n20 $broken | sed "s|^[^']*|<a href='\0'>\0</a>|g" | \
905 section $broken 20 "Broken|All broken packages"
907 [ -s "$blocked" ] && sed "s|^[^']*|<a href='\0'>\0</a>|g" $blocked | \
908 section $blocked 12 "Blocked|All blocked packages"
910 cd $PKGS
911 ls -let *.tazpkg | awk '
912 (NR<=20){
913 sub(/:[0-9][0-9]$/, "", $9);
914 mon = index(" JanFebMarAprMayJunJulAugSepOctNovDec", $7) / 3;
915 printf("%d-%02d-%02d %s : <a href=\"get/%s\">%s</a>\n", $10, mon, $8, $9, $11, $11);
916 }' | \
917 section $activity 1000 "Latest cook"
919 echo '</div>'
920 datalist
921 page_footer
922 exit 0
923 fi
926 case "$cmd" in
927 '')
928 page_header
929 log=$LOGS/$pkg.log
931 # Package info.
932 if [ -f "$wok/$pkg/receipt" ]; then
933 pkg_info
934 else
935 if [ $(ls $wok/*$pkg*/receipt 2>/dev/null | wc -l) -eq 0 ]; then
936 echo "<h2>Not Found</h2>"
937 show_note e "The requested package <b>$pkg</b> was not found on this server."
938 else
939 # Search page
940 echo "<section><h2>Package names matching “$pkg”</h2>"
941 echo "<table><thead><tr><th>Name</th><th>Description</th><th>Category</th></tr></thead><tbody>"
942 for i in $(cd $wok; ls *$pkg*/receipt); do
943 pkg=$(dirname $i)
944 unset SHORT_DESC CATEGORY
945 . $wok/$pkg/receipt
946 echo -n "<tr><td><a href="$base/$pkg">$pkg</a></td>"
947 echo -n "<td>$SHORT_DESC</td><td>$CATEGORY</td></tr>"
948 done
949 echo '</tbody></table></section>'
950 unset pkg
951 fi
952 page_footer
953 exit 0
954 fi
956 # Check for a log file and display summary if it exists.
957 summary "$log"
959 # Display <Recook> button only for SliTaz web browser
960 if [ -f "$log" ]; then
961 case "$HTTP_USER_AGENT" in
962 *SliTaz*)
963 if [ -f $CACHE/cooker-request -a -n "$HTTP_REFERER" ]; then
964 if grep -qs "^$pkg$" $CACHE/recook-packages; then
965 show_note i "The package “$pkg” has been requested for recook"
966 else
967 echo "<a class='button' href='$base/?recook=${pkg//+/%2B}'>Recook $pkg</a>"
968 fi
969 fi
970 ;;
971 esac
972 fi
973 ;;
975 receipt)
976 page_header
977 pkg_info
978 echo "<a class='button receipt' href='$base/$pkg/receipt'>receipt</a>"
979 ( cd $wok/$pkg; find stuff -type f 2>/dev/null ) | sort | \
980 awk -vb="$base/$pkg" '{printf("<a class=\"button\" href=\"%s/%s\">%s</a>\n", b, $0, $0)}'
982 show_code bash < $wok/$pkg/receipt
983 ;;
985 stuff)
986 page_header
987 pkg_info
988 file="$pkg/stuff/$arg"
989 echo "<a class='button' href='$base/$pkg/receipt'>receipt</a>"
990 ( cd $wok/$pkg; find stuff -type f 2>/dev/null ) | sort | \
991 awk -vb="$base/$pkg" -va="stuff/$arg" '{
992 printf("<a class=\"button%s\" href=\"%s/%s\">%s</a>\n", a==$0 ? " receipt" : "", b, $0, $0)
993 }'
995 if [ -f "$wok/$file" ]; then
996 case $file in
997 *.desktop|*.theme) class="ini" ;;
998 *.patch|*.diff|*.u) class="diff" ;;
999 *.sh) class="bash" ;;
1000 *.conf*|*.ini)
1001 class="bash"
1002 [ -n "$(cut -c1 < $wok/$file | fgrep '[')" ] && class="ini"
1003 ;;
1004 *.pl) class="perl" ;;
1005 *.c|*.h|*.awk) class="clike" ;;
1006 *.svg) class="svg" ;;
1007 *Makefile*) class="makefile" ;;
1008 *.po|*.pot) class="bash" ;;
1009 *.css) class="css" ;;
1010 *.htm|*.html) class="html" ;;
1011 *.js) class="js" ;;
1012 *.txt) class="asciidoc" ;;
1013 *)
1014 case $(head -n1 $wok/$file) in
1015 *!/bin/sh*|*!/bin/bash*) class="bash" ;;
1016 esac
1017 if [ -z "$class" -a "$(head -n1 $wok/$file | cut -b1)" == '#' ]; then
1018 class="bash"
1019 fi
1020 if [ -z "$class" ]; then
1021 # Follow Busybox restrictions. Search for non-printable chars
1022 if [ $(tr -d '[:alnum:][:punct:][:blank:][:cntrl:]' < "$wok/$file" | wc -c) -gt 0 ]; then
1023 raw="true"
1024 fi
1025 fi
1026 ;;
1027 esac
1029 # Display image
1030 case $file in
1031 *.png|*.svg|*.jpg|*.jpeg|*.ico)
1032 echo "<img src='$base/$pkg/browse/stuff/$arg' style='display: block; max-width: 100%; margin: auto'/>"
1033 ;;
1034 esac
1036 # Display colored listing for all text-based documents (also for *.svg)
1037 case $file in
1038 *.png|*.jpg|*.jpeg|*.ico) ;;
1039 *)
1040 if [ -z "$raw" ]; then
1041 cat $wok/$file | show_code $class
1042 fi
1043 ;;
1044 esac
1046 # Display hex dump for binary files
1047 if [ -n "$raw" ]; then
1048 hexdump -C $wok/$file | show_code #| sed 's|^\([0-9a-f][0-9a-f]*\)|<span class="c2">\1</span>|'
1049 fi
1050 else
1051 show_note e "File “$file” absent!"
1052 fi
1053 ;;
1055 files)
1056 page_header
1057 pkg_info
1059 packaged=$(mktemp)
1061 # find main package
1062 wanted=$(. $wok/$pkg/receipt; echo $WANTED)
1063 main=${wanted:-$pkg}
1065 # identify split packages
1066 split="$main $(. $wok/$main/receipt; echo $SPLIT)"
1067 [ -d "$wok/$main-dev" ] && split="$split $main-dev"
1068 split="$(echo $split | tr ' ' '\n' | sort -u)"
1070 # finally we need the version
1071 if [ -f "$WOK/linux/receipt" ]; then
1072 kvers=$(. $WOK/linux/receipt; echo $VERSION)
1073 kbasevers=$(echo $kvers | cut -d. -f1,2)
1074 elif [ -f "$INSTALLED/linux-api-headers/receipt" ]; then
1075 kvers=$(. $INSTALLED/linux-api-headers/receipt; echo $VERSION)
1076 kbasevers=$(echo $kvers | cut -d. -f1,2)
1077 fi
1078 ver=$(. $wok/$main/receipt; echo $VERSION$EXTRAVERSION)
1081 echo "<section><h3>Quick jump:</h3><ul>"
1082 echo "$split" | sed 'p' | xargs printf "<li><a href='#id-%s'>%s</a></li>\n"
1083 echo "<li><a href='#id-repeats'>repeatedly packaged files</a> (if any)</li>"
1084 echo "<li><a href='#id-orphans'>unpackaged files</a> (if any)</li>"
1085 echo "</ul></section>"
1087 for p in $split; do
1088 namever="$p-$ver"
1089 if [ -d "$wok/$p/taz/$p-$ver" ]; then
1090 indir=$p
1091 elif [ -d "$wok/$main/taz/$p-$ver" ]; then
1092 indir=$main
1093 fi
1094 dir="$wok/$indir/taz/$p-$ver/fs"
1096 size=$(du -hs $dir | awk '{ sub(/\.0/, ""); print $1 }')
1098 echo "<section><h3 id='id-$p'>Content of package “$namever” (${size:-empty}):</h3>"
1099 echo '<pre class="files">'
1100 if [ -s "$wok/$indir/taz/$p-$ver/files.list" ]; then
1101 echo -n '<span class="underline">permissions·lnk·user ·'
1102 echo -en 'group · size·date &amp; time ·name\n</span>'
1103 cd $dir
1104 find . -print0 | sort -z | xargs -0 ls -ldp --color=always | \
1105 syntax_highlighter files | \
1106 sed "s|\([^>]*\)>\.\([^<]*\)\(<.*\)$|\1 href='$base/$indir/browse/taz/$p-$ver/fs\2'>\2\3|;" | \
1107 awk 'BEGIN { FS="\""; }
1108 { gsub("+", "%2B", $2); print; }'
1109 else
1110 echo 'No files'
1111 fi
1112 echo '</pre></section>'
1113 cat $wok/$indir/taz/$p-$ver/files.list >> $packaged
1114 done
1116 # find repeatedly packaged files
1117 repeats="$(sort $packaged | uniq -d)"
1118 if [ -n "$repeats" ]; then
1119 echo -n '<section><h3 id="id-repeats">Repeatedly packaged files:</h3><pre class="files">'
1120 echo "$repeats" | sed 's|^|<span class="c11">!!!</span> |'
1121 echo "</pre></section>"
1122 fi
1124 # find unpackaged files
1125 all_files=$(mktemp)
1126 cd $wok/$main/install; find ! -type d | sed 's|\.||' > $all_files
1127 orphans="$(sort $all_files $packaged | uniq -u)"
1128 if [ -d "$wok/$main/install" -a -n "$orphans" ]; then
1129 echo '<section><h3 id="id-orphans">Unpackaged files:</h3>'
1130 table=$(mktemp)
1131 echo "$orphans" | awk -vi="$base/$indir/browse/install" '
1132 function tag(text, color) {
1133 printf("<span class=\"c%s1\">%s</span> ", color, text);
1134 printf("<a class=\"c00\" href=\"%s\">%s</a>\n", i $0, $0);
1136 /\/perllocal.pod$/ || /\/\.packlist$/ || /\/share\/bash-completion\// ||
1137 /\/lib\/systemd\// || /\.pyc$/ || /\.pyo$/ { tag("---", 0); next }
1138 /\.pod$/ { tag("pod", 5); next }
1139 /\/share\/man\// { tag("man", 5); next }
1140 /\/share\/doc\// || /\/share\/gtk-doc\// || /\/share\/info\// ||
1141 /\/share\/devhelp\// { tag("doc", 5); next }
1142 /\/share\/icons\// { tag("ico", 2); next }
1143 /\/share\/locale\// { tag("loc", 4); next }
1144 /\.h$/ || /\.a$/ || /\.la$/ || /\.pc$/ || /\/bin\/.*-config$/ ||
1145 /\/Makefile.*$/ { tag("dev", 3); next }
1146 { tag("???", 1) }
1147 ' > $table
1149 # Summary table
1150 for i in head body; do
1151 case $i in
1152 head) echo -n '<table class="summary"><tr>';;
1153 body) echo -n '<th> </th></tr><tr>';;
1154 esac
1155 for j in '???1' dev3 loc4 ico2 doc5 man5 pod5 '---0'; do
1156 tag=${j:0:3}; class="c${j:3:1}0"; [ "$class" == 'c00' ] && class='c01'
1157 case $i in
1158 head) echo -n "<th class='$class'>$tag</th>";;
1159 body) printf '<td>%s</td>' "$(grep ">$tag<" $table | wc -l)";;
1160 esac
1161 done
1162 done
1163 echo '<td> </td></tr></table>'
1165 echo -n '<pre class="files">'
1166 cat $table
1167 echo '</pre></section>'
1168 rm $table
1169 fi
1170 rm $packaged $all_files
1171 ;;
1173 description)
1174 page_header
1175 pkg_info
1176 descs="$(ls $WOK/$pkg/description*.txt)"
1177 if [ -n "$descs" ]; then
1178 echo '<div id="content2">'
1179 [ -f "$WOK/$pkg/description.txt" ] && show_desc "$pkg" "$WOK/$pkg/description.txt"
1180 for i in $descs; do
1181 case $i in
1182 */description.txt) continue ;;
1183 *) package=$(echo $i | cut -d. -f2) ;;
1184 esac
1185 show_desc "$package" "$i"
1186 done
1187 echo '</div>'
1188 else
1189 show_note w "No description of $pkg"
1190 fi
1191 ;;
1193 log)
1194 page_header
1195 pkg_info
1196 [ -z "$arg" ] && arg=$(stat -c %Y $LOGS/$pkg.log)
1198 echo '<div class="btnList">'
1199 acc='l' # access key for the latest log is 'L'
1200 while read log; do
1201 # for all $pkg.log, $pkg.log.0 .. $pkg.log.9, $pkg-pack.log (if any)
1202 timestamp=$(stat -c %Y $log)
1203 class=''
1204 if [ "$arg" == "$timestamp" ]; then
1205 class=' log'
1206 logfile="$log"
1207 fi
1208 case $log in *-pack.log) acc='p';; esac # access key for the packing log is 'P'
1209 echo -n "<a class='button$class' data-acc='$acc' accesskey='$acc' href='$base/$pkg/log/$timestamp'>"
1210 echo "$(stat -c %y $log | cut -d: -f1,2)</a>"
1211 case $acc in
1212 l) acc=0;;
1213 *) acc=$((acc+1));;
1214 esac
1215 done <<EOT
1216 $(find $LOGS -name "$pkg.log*" | sort)
1217 $(find $LOGS -name "$pkg-pack.log")
1218 EOT
1219 echo '</div>'
1221 if [ -z "$logfile" ]; then
1222 show_note e "Requested log is absent"
1223 page_footer
1224 exit 0
1225 fi
1227 # Define cook variables for syntax highlighter
1228 if [ -s "$WOK/$pkg/receipt" ]; then
1229 . "$WOK/$pkg/receipt"
1230 _wok='/home/slitaz/wok'
1231 _src="$_wok/$pkg/source/$PACKAGE-$VERSION"
1232 _install="$_wok/$pkg/install"
1233 _fs="$_wok/$pkg/taz/$PACKAGE-$VERSION/fs"
1234 _stuff="$_wok/$pkg/stuff"
1235 fi
1237 # if [ ! -f "gzlog/$pkg.$arg" ]; then
1238 # {
1239 # summary "$logfile" links
1241 # syntax_highlighter log < $logfile | awk '
1242 # BEGIN { print "<pre class=\"log\">"; }
1243 # { printf("<a name=\"l%d\" href=\"#l%d\">%5d</a> %s\n", NR, NR, NR, $0); }
1244 # END { print "</pre>"; }
1245 # '
1247 # page_footer
1248 # } | gzip > gzlog/$pkg.$arg
1249 # fi
1251 blog=$(basename $logfile)
1252 summary "$logfile" links
1254 # disable next `sed` for the 'like2016' theme
1255 theme=$(COOKIE theme); theme=${theme:-default}; [ "$theme" != 'like2016' ] && theme=''
1256 cat $logfile | syntax_highlighter log | \
1257 sed -e "/(pkg\/local$theme):/ s|: \([^<]*\)|<img src='$base/i/$blog/\1'> \1|" | \
1258 awk '
1259 BEGIN { print "<pre class=\"log\">"; }
1260 { printf("<a name=\"l%d\" href=\"#l%d\">%5d</a> %s\n", NR, NR, NR, $0); }
1261 END { print "</pre>"; }
1263 ;;
1266 man|doc|info)
1267 page_header
1268 pkg_info
1269 echo '<div style="max-height: 6.4em; overflow: auto; padding: 0 4px">'
1271 dir="wok/$pkg/install/usr/share/$cmd"
1272 [ "$cmd" == 'doc' ] && dir="$dir wok/$pkg/install/usr/share/gtk-doc"
1273 if [ "$cmd" == 'doc' -a -z "$arg" ]; then
1274 try=$(for i in $dir; do find $i -name 'index.htm*'; done | sed q)
1275 [ -n "$try" ] && arg="$try"
1276 fi
1277 while read i; do
1278 [ -s "$i" ] || continue
1279 case "$i" in
1280 *.jp*g|*.png|*.gif|*.svg|*.css) continue
1281 esac
1282 i=${i#$dir/}
1283 [ -n "$arg" ] || arg="$i"
1284 class=''; [ "$arg" == "$i" ] && class=" doc"
1285 case "$cmd" in
1286 man)
1287 case $i in
1288 man*) lang='';;
1289 *) lang="${i%%/*}: ";;
1290 esac
1291 man=$(basename $i .gz)
1292 echo "<a class='button$class' href='$base/$pkg/man/$i'>$lang${man%.*} (${man##*.})</a>"
1293 ;;
1294 doc)
1295 echo "<a class='button$class' href='$base/$pkg/doc/$i'>$(basename $i .gz)</a>"
1296 ;;
1297 info)
1298 info=$(basename $i)
1299 echo "<a class='button$class' href='$base/$pkg/info/$i#Top'>${info/.info/}</a>"
1300 ;;
1301 esac
1302 done <<EOT
1303 $(for i in $dir; do find $i -type f; done | sort)
1304 EOT
1305 echo '</div>'
1307 [ -f "$arg" ] || arg="$dir/$arg"
1308 if [ -f "$arg" ]; then
1309 tmp="$(mktemp)"
1310 docat "$arg" > $tmp
1311 [ -s "$tmp" ] &&
1312 case "$cmd" in
1313 info)
1314 echo '<div id="content2" class="texinfo"><pre class="first">'
1315 info2html < "$tmp"
1316 echo '</pre></div>'
1317 ;;
1318 doc)
1319 case "$arg" in
1320 *.sgml|*.devhelp2) class='xml';;
1321 *.py) class='python';; # pycurl package
1322 *.css) class='css';;
1323 *) class='asciidoc';;
1324 esac
1325 case "$arg" in
1326 *.htm*)
1327 case $arg in
1328 wok/*) page="${arg#wok/}"; page="$base/$pkg/browse/${page#*/}";;
1329 *) page="$base/$pkg/browse/install/usr/share/$cmd/$arg";;
1330 esac
1331 # make the iframe height so long to contain its content without scrollbar
1332 echo "<iframe id='idoc' src='$page' width='100%' onload='resizeIframe(this)'></iframe>"
1333 ;;
1334 *.pdf)
1335 case $arg in
1336 wok/*) page="${arg#wok/}"; page="$base/$pkg/browse/${page#*/}";;
1337 *) page="$base/$pkg/browse/install/usr/share/$cmd/$arg";;
1338 esac
1339 cat <<EOT
1340 <object id="idoc" data="$page" width="100%" height="100%" type="application/pdf" style="min-height: 600px">
1341 $(show_note w "Missing PDF plugin.<br/>Get the file <a href="$page">$(basename "$page")</a>.")
1342 </object>
1343 EOT
1344 ;;
1345 *)
1346 show_code $class < "$tmp"
1347 ;;
1348 esac
1349 ;;
1350 man)
1351 #export TEXTDOMAIN='man2html'
1352 echo "<div id='content2'>"
1354 html=$(./man2html "$tmp" | sed -e '1,/<header>/d' -e '/<footer>/,$d' \
1355 -e 's|<a href="file:///[^>]*>\([^<]*\)</a>|\1|g' \
1356 -e 's|<a href="?[1-9]\+[^>]*>\([^<]*\)</a>|\1|g')
1358 if [ -n "$(echo "$html" | fgrep 'The requested file /tmp/tmp.')" ]; then
1359 # Process the pre-formatted man-cat page
1360 # (for example see sudo package without groff in build dependencies)
1361 sed -i '
1362 s|M-bM-^@M-^S|—|g;
1363 s|M-bM-^@M-^\\|<b>|g;
1364 s|M-bM-^@M-^]|</b>|g
1365 s|M-bM-^@M-^X|<u>|g;
1366 s|M-bM-^@M-^Y|</u>|g;
1367 s|M-BM-||g;
1368 s|++oo|•|g;
1369 s|&|\&amp;|g; s|<|\&lt;|g; s|>|\&gt;|g;
1370 ' "$tmp"
1371 for i in a b c d e f g h i j k l m n o p q r s t u v w x y z \
1372 A B C D E F G H I J K L M N O P Q R S T U V W X Y Z \
1373 0 1 2 3 4 5 6 7 8 9 _ - '\\+' '\.' /; do
1374 sed -i "s|$i$i|<b>$i</b>|g; s|_$i|<u>$i</u>|g" "$tmp"
1375 done
1376 echo '<pre class="catman">'
1377 sed 's|</b><b>||g; s|</u><u>||g; s|</u><b>_</b><u>|_|g; s|</b> <b>| |g;' "$tmp"
1378 echo '</pre>'
1379 else
1380 echo "$html"
1381 fi
1382 echo "</div>"
1383 ;;
1384 esac
1385 rm -f $tmp
1386 else
1387 show_note e "File “$arg” not exists!"
1388 fi
1389 ;;
1391 download)
1392 page_header
1393 pkg_info
1394 show=0
1396 . $wok/$pkg/receipt
1398 if [ -n "$TARBALL" -a -s "$SRC/$TARBALL" ]; then
1399 files_header
1400 echo "<tr><td><a href='$base/src/$TARBALL'>$TARBALL</a></td>"
1401 ls -lh "$SRC/$TARBALL" | awk '{printf("<td>%sB</td>", $5)}'
1402 echo "<td>Sources for building the package “$pkg”</td></tr>"
1403 show=1
1404 fi
1406 if [ -d "$wok/$pkg/taz" ]; then
1407 [ "$show" -eq 1 ] || files_header
1409 for i in $(all_names); do
1410 [ -e "$wok/$pkg/taz/$i-$VERSION$EXTRAVERSION/receipt" ] || continue
1411 . $wok/$pkg/taz/$i-$VERSION$EXTRAVERSION/receipt
1413 for filename in "$PACKAGE-$VERSION$EXTRAVERSION.tazpkg" "$PACKAGE-$VERSION$EXTRAVERSION-$ARCH.tazpkg"; do
1414 [ -f "$PKGS/$filename" ] &&
1415 cat <<EOT
1416 <tr>
1417 <td><a href="$base/get/$filename">$filename</a></td>
1418 <td>$(ls -lh ./packages/$filename | awk '{printf("%sB", $5)}')</td>
1419 <td>$SHORT_DESC</td>
1420 </tr>
1421 EOT
1422 done
1423 done
1424 show=1
1425 fi
1427 if [ "$show" -eq 1 ]; then
1428 echo '</tbody></table></section>'
1429 else
1430 show_note w "Sorry, there's nothing to download…"
1431 fi
1432 ;;
1434 esac
1437 page_footer
1438 exit 0