]> git.plutz.net Git - shellwiki/blob - handlers/40_search.sh
add license headers
[shellwiki] / handlers / 40_search.sh
1 #!/bin/sh
2
3 # Copyright 2023 Paul Hänsch
4
5 # Permission to use, copy, modify, and/or distribute this software for any
6 # purpose with or without fee is hereby granted, provided that the above
7 # copyright notice and this permission notice appear in all copies.
8
9 # THE SOFTWARE IS PROVIDED “AS IS” AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
12 # SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
15 # IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16
17 [ "${PATH_INFO%\[search\]}" = "$PATH_INFO" ] && return 1
18
19 . "$_EXEC/cgilite/storage.sh"
20
21 I="$_DATA/index"
22 words="$( GET q | awk '
23   BEGIN { # Field separator FS should include punctuation, including Unicode Block U+2000 - U+206F
24           if ( length("¡") == 1 )  # Utf-8 aware AWK
25           FS = "([] \\t\\n\\r!\"#'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '[\342\200\200-\342\201\257]')"')+";
26           else                     # UTF-8 Hack
27           FS = "([] \\t\\n\\r!\"#'\''()*+,./:;<=>?\\\\^_`{|}~[-]|%[0-9A-Fa-f]{2}|'"$(printf '\342\200[\200-\277]|\342\201[\201-\257]')"')+";
28           fi
29         }
30         { for (n = 1; n <= NF; n++) printf "%s  ", tolower($n); }
31 ')"
32
33 for w in ${words}; do
34   [ ! -f "$I/$w" ] && continue
35
36   while read date doc freq num total; do
37     P="$_DATA/pages$(UNSTRING "$doc")"
38     d="$(stat -c %Y -- "$P/#index.flag")"
39     [ "$d" -gt "$date" ] && continue
40
41     printf '%s  %f\n' "$doc" "$freq"
42   done <"$I/$w"
43 done \
44 | awk '
45       { cnt[$1]++; weight[$1] = weight[$1] ? weight[$1] * $2 : $2; }
46   END { m = 0; for (d in cnt) m = ( m < cnt[d] ) ? cnt[d] : m;
47         for (d in cnt) if ( cnt[d] == m ) printf "%f    %s\n", weight[d], d;
48       }
49 ' \
50 | sort -nr \
51 | while read freq doc; do
52   page="$(UNSTRING "$doc")"
53   [ "${page%*/\[*\]/*}" != "$page" ] && continue
54   if [ "$LANGUAGE_DEFAULT" ]; then
55     [ -d "${_DATA}/pages/${page}/:${LANGUAGE}/" ] && continue
56     [ "${page%/:*/}" = "${page%/:${LANGUAGE}/}" ] || continue
57   fi
58   acl_read "$page" || continue
59   printf '%s\n' "$page"
60 done \
61 | theme_search "${words%        }"