Collection of themes/skins for the Fossil SCM

โŒˆโŒ‹ โއ branch:  Fossil Skins Extra


Hex Artifact Content

Artifact 5debf965c095001629ffb9c1365d6c4dc6221e4c:

  • File parts/github/github.th1-setup — part of check-in [d933f24359] at 2021-04-04 22:51:56 on branch trunk — Enable upvar 1 for fx_stats query (user: mario size: 12801)

0000: 0a 23 2d 2d 20 50 72 65 2d 69 6e 63 72 65 6d 65  .#-- Pre-increme
0010: 6e 74 20 20 5b 2b 2b 20 76 61 72 6e 61 6d 65 5d  nt  [++ varname]
0020: 0a 70 72 6f 63 20 2b 2b 20 7b 76 61 72 6e 61 6d  .proc ++ {varnam
0030: 65 7d 20 7b 0a 20 20 20 75 70 76 61 72 20 31 20  e} {.   upvar 1 
0040: 24 76 61 72 6e 61 6d 65 20 69 0a 20 20 20 72 65  $varname i.   re
0050: 74 75 72 6e 20 5b 75 70 6c 65 76 65 6c 20 31 20  turn [uplevel 1 
0060: 22 73 65 74 20 7b 24 76 61 72 6e 61 6d 65 7d 20  "set {$varname} 
0070: 5b 65 78 70 72 20 31 2b 24 69 5d 22 5d 0a 7d 0a  [expr 1+$i]"].}.
0080: 0a 23 2d 2d 20 74 65 72 6e 61 72 79 20 2f 20 69  .#-- ternary / i
0090: 66 2d 73 68 6f 72 74 68 61 6e 64 20 28 63 6f 6e  f-shorthand (con
00a0: 64 2f 74 68 65 6e 2f 65 6c 73 65 20 6d 61 79 20  d/then/else may 
00b0: 62 65 20 6c 69 74 65 72 61 6c 73 2c 20 6f 72 20  be literals, or 
00c0: 7b 5b 65 78 70 72 65 73 73 69 6f 6e 73 5d 7d 20  {[expressions]} 
00d0: 74 68 65 6d 73 65 6c 76 65 73 29 0a 70 72 6f 63  themselves).proc
00e0: 20 3f 3a 20 7b 63 6f 6e 64 20 74 68 65 6e 20 65   ?: {cond then e
00f0: 6c 73 65 7d 20 7b 0a 20 20 20 75 70 6c 65 76 65  lse} {.   upleve
0100: 6c 20 31 20 22 69 66 20 7b 24 63 6f 6e 64 7d 20  l 1 "if {$cond} 
0110: 7b 20 72 65 74 75 72 6e 20 24 74 68 65 6e 3b 20  { return $then; 
0120: 7d 20 65 6c 73 65 20 7b 20 72 65 74 75 72 6e 20  } else { return 
0130: 24 65 6c 73 65 3b 20 7d 22 0a 7d 0a 0a 23 2d 2d  $else; }".}..#--
0140: 20 69 6e 66 6f 20 65 78 69 73 74 73 20 73 68 6f   info exists sho
0150: 72 74 68 61 6e 64 0a 70 72 6f 63 20 69 73 73 65  rthand.proc isse
0160: 74 20 7b 76 61 72 6e 61 6d 65 7d 20 7b 0a 20 20  t {varname} {.  
0170: 20 72 65 74 75 72 6e 20 5b 75 70 6c 65 76 65 6c   return [uplevel
0180: 20 31 20 22 69 6e 66 6f 20 65 78 69 73 74 73 20   1 "info exists 
0190: 7b 24 76 61 72 6e 61 6d 65 7d 22 5d 0a 7d 0a 0a  {$varname}"].}..
01a0: 23 2d 2d 20 73 74 72 69 6e 67 20 65 71 75 61 6c  #-- string equal
01b0: 69 74 79 20 73 68 6f 72 74 68 61 6e 64 0a 70 72  ity shorthand.pr
01c0: 6f 63 20 65 71 20 7b 73 74 72 31 20 73 74 72 32  oc eq {str1 str2
01d0: 7d 20 7b 0a 20 20 20 72 65 74 75 72 6e 20 5b 65  } {.   return [e
01e0: 78 70 72 20 7b 24 73 74 72 31 20 65 71 20 24 73  xpr {$str1 eq $s
01f0: 74 72 32 7d 5d 0a 7d 0a 0a 23 2d 2d 20 77 68 69  tr2}].}..#-- whi
0200: 6c 65 20 6c 6f 6f 70 0a 70 72 6f 63 20 77 68 69  le loop.proc whi
0210: 6c 65 20 7b 63 6f 6e 64 69 74 69 6f 6e 20 63 6f  le {condition co
0220: 64 65 7d 20 7b 0a 20 20 20 72 65 74 75 72 6e 20  de} {.   return 
0230: 5b 75 70 6c 65 76 65 6c 20 31 20 22 66 6f 72 20  [uplevel 1 "for 
0240: 7b 7d 20 7b 24 63 6f 6e 64 69 74 69 6f 6e 7d 20  {} {$condition} 
0250: 7b 7d 20 7b 24 63 6f 64 65 7d 22 5d 0a 7d 0a 0a  {} {$code}"].}..
0260: 23 2d 2d 20 66 6f 72 65 61 63 68 20 56 41 52 20  #-- foreach VAR 
0270: 22 61 62 63 20 78 79 7a 20 31 32 33 22 20 7b 20  "abc xyz 123" { 
0280: 70 75 74 73 20 22 28 24 56 41 52 29 20 22 20 7d  puts "($VAR) " }
0290: 0a 70 72 6f 63 20 66 6f 72 65 61 63 68 20 7b 76  .proc foreach {v
02a0: 61 72 6e 61 6d 65 20 6c 69 73 74 20 63 6f 64 65  arname list code
02b0: 7d 20 7b 0a 20 20 20 75 70 76 61 72 20 31 20 24  } {.   upvar 1 $
02c0: 76 61 72 6e 61 6d 65 20 76 61 6c 0a 20 20 20 66  varname val.   f
02d0: 6f 72 20 7b 73 65 74 20 69 20 30 7d 20 20 7b 24  or {set i 0}  {$
02e0: 69 20 3c 20 5b 6c 6c 65 6e 67 74 68 20 24 6c 69  i < [llength $li
02f0: 73 74 5d 7d 20 20 7b 2b 2b 20 69 7d 20 7b 0a 20  st]}  {++ i} {. 
0300: 20 20 20 20 20 73 65 74 20 76 61 6c 20 5b 6c 69       set val [li
0310: 6e 64 65 78 20 24 6c 69 73 74 20 24 69 5d 0a 20  ndex $list $i]. 
0320: 20 20 20 20 20 75 70 6c 65 76 65 6c 20 31 20 22       uplevel 1 "
0330: 24 63 6f 64 65 22 0a 20 20 20 7d 0a 7d 0a 0a 23  $code".   }.}..#
0340: 2d 2d 20 41 20 73 77 69 74 63 68 20 73 74 61 74  -- A switch stat
0350: 65 6d 65 6e 74 2e 0a 23 0a 23 20 73 77 69 74 63  ement..#.# switc
0360: 68 20 22 76 61 6c 22 20 7b 0a 23 20 20 20 20 20  h "val" {.#     
0370: 20 20 20 22 63 6d 70 31 22 20 7b 63 6f 64 65 31     "cmp1" {code1
0380: 7d 0a 23 20 20 20 20 20 20 20 20 22 63 6d 70 32  }.#        "cmp2
0390: 22 20 7b 63 6f 64 65 32 7d 0a 23 20 20 20 20 20  " {code2}.#     
03a0: 20 20 20 22 63 6d 70 33 22 20 7b 63 6f 64 65 33     "cmp3" {code3
03b0: 7d 0a 23 20 20 20 7b 7b 64 65 66 61 75 6c 74 7d  }.#   {{default}
03c0: 7d 20 7b 63 6f 64 65 4e 7d 0a 23 20 7d 0a 23 0a  } {codeN}.# }.#.
03d0: 70 72 6f 63 20 73 77 69 74 63 68 20 7b 63 6f 6d  proc switch {com
03e0: 70 61 72 65 5f 76 61 6c 75 65 20 76 61 6c 5f 63  pare_value val_c
03f0: 6f 64 65 5f 70 61 69 72 73 7d 20 7b 0a 20 20 20  ode_pairs} {.   
0400: 73 65 74 20 6c 65 6e 20 5b 6c 6c 65 6e 67 74 68  set len [llength
0410: 20 24 76 61 6c 5f 63 6f 64 65 5f 70 61 69 72 73   $val_code_pairs
0420: 5d 0a 20 20 20 23 20 6c 6f 6f 70 20 6f 76 65 72  ].   # loop over
0430: 20 63 6f 6d 70 61 72 65 20 76 61 6c 75 65 73 20   compare values 
0440: 2b 20 63 6f 64 65 20 70 61 69 72 73 0a 20 20 20  + code pairs.   
0450: 66 6f 72 20 20 7b 73 65 74 20 6e 20 30 7d 20 20  for  {set n 0}  
0460: 7b 24 6e 20 3c 20 24 6c 65 6e 7d 20 20 7b 2b 2b  {$n < $len}  {++
0470: 20 6e 7d 20 7b 0a 20 20 20 20 20 20 73 65 74 20   n} {.      set 
0480: 63 6d 70 20 5b 6c 69 6e 64 65 78 20 24 76 61 6c  cmp [lindex $val
0490: 5f 63 6f 64 65 5f 70 61 69 72 73 20 24 6e 5d 3b  _code_pairs $n];
04a0: 0a 20 20 20 20 20 20 69 66 20 7b 5b 65 78 70 72  .      if {[expr
04b0: 20 24 63 6d 70 20 65 71 20 24 63 6f 6d 70 61 72   $cmp eq $compar
04c0: 65 5f 76 61 6c 75 65 20 7c 7c 20 24 63 6d 70 20  e_value || $cmp 
04d0: 65 71 20 7b 7b 64 65 66 61 75 6c 74 7d 7d 20 5d  eq {{default}} ]
04e0: 7d 20 7b 0a 20 20 20 20 20 20 20 20 20 72 65 74  } {.         ret
04f0: 75 72 6e 20 5b 75 70 6c 65 76 65 6c 20 31 20 5b  urn [uplevel 1 [
0500: 6c 69 6e 64 65 78 20 24 76 61 6c 5f 63 6f 64 65  lindex $val_code
0510: 5f 70 61 69 72 73 20 5b 2b 2b 20 6e 5d 5d 5d 3b  _pairs [++ n]]];
0520: 0a 20 20 20 20 20 20 7d 0a 20 20 20 7d 0a 7d 0a  .      }.   }.}.
0530: 0a 23 2d 2d 20 73 65 74 20 61 20 63 6f 75 70 6c  .#-- set a coupl
0540: 65 20 6f 66 20 76 61 72 69 61 62 6c 65 73 0a 70  e of variables.p
0550: 72 6f 63 20 75 69 3a 3a 76 61 72 73 20 7b 7d 20  roc ui::vars {} 
0560: 7b 0a 20 20 20 23 2d 2d 20 69 6d 70 6f 72 74 0a  {.   #-- import.
0570: 20 20 20 73 65 74 20 63 75 72 72 65 6e 74 5f 70     set current_p
0580: 61 67 65 20 24 3a 3a 63 75 72 72 65 6e 74 5f 70  age $::current_p
0590: 61 67 65 0a 20 20 20 73 65 74 20 62 61 73 65 75  age.   set baseu
05a0: 72 6c 20 24 3a 3a 62 61 73 65 75 72 6c 0a 0a 20  rl $::baseurl.. 
05b0: 20 20 23 2d 2d 20 65 78 70 6f 72 74 20 76 61 72    #-- export var
05c0: 69 61 62 6c 65 73 0a 20 20 20 75 70 76 61 72 20  iables.   upvar 
05d0: 31 20 70 61 67 65 63 61 74 20 70 61 67 65 63 61  1 pagecat pageca
05e0: 74 20 20 20 20 20 20 20 20 20 20 20 20 23 20 69  t            # i
05f0: 6e 64 65 78 20 7c 20 77 63 6f 6e 74 65 6e 74 20  ndex | wcontent 
0600: 7c 20 77 69 6b 69 20 7c 20 74 72 65 65 20 7c 20  | wiki | tree | 
0610: 74 69 6d 65 6c 69 6e 65 0a 20 20 20 75 70 76 61  timeline.   upva
0620: 72 20 31 20 70 61 67 65 6e 61 6d 65 20 70 61 67  r 1 pagename pag
0630: 65 6e 61 6d 65 20 20 20 20 20 20 20 20 20 20 23  ename          #
0640: 20 73 72 63 2f 64 61 74 61 0a 20 20 20 75 70 76   src/data.   upv
0650: 61 72 20 31 20 62 61 73 65 64 6f 6d 61 69 6e 20  ar 1 basedomain 
0660: 62 61 73 65 64 6f 6d 61 69 6e 0a 20 20 20 75 70  basedomain.   up
0670: 76 61 72 20 31 20 73 74 61 74 73 5f 64 65 73 63  var 1 stats_desc
0680: 72 69 70 74 69 6f 6e 20 73 74 61 74 73 5f 64 65  ription stats_de
0690: 73 63 72 69 70 74 69 6f 6e 0a 20 20 20 75 70 76  scription.   upv
06a0: 61 72 20 31 20 73 74 61 74 73 5f 73 6f 63 69 61  ar 1 stats_socia
06b0: 6c 20 73 74 61 74 73 5f 73 6f 63 69 61 6c 0a 20  l stats_social. 
06c0: 20 20 75 70 76 61 72 20 31 20 73 74 61 74 73 5f    upvar 1 stats_
06d0: 66 6f 72 6b 73 20 73 74 61 74 73 5f 66 6f 72 6b  forks stats_fork
06e0: 73 0a 0a 20 20 20 23 2d 2d 20 44 65 74 65 72 6d  s..   #-- Determ
06f0: 69 6e 65 20 63 75 72 72 65 6e 74 20 70 61 67 65  ine current page
0700: 20 74 79 70 65 0a 20 20 20 73 65 74 20 70 61 67   type.   set pag
0710: 65 6e 61 6d 65 20 22 22 0a 20 20 20 69 66 20 7b  ename "".   if {
0720: 5b 72 65 67 65 78 70 20 7b 5e 28 69 6e 64 65 78  [regexp {^(index
0730: 7c 68 6f 6d 65 29 5b 3f 5d 3f 7d 20 24 63 75 72  |home)[?]?} $cur
0740: 72 65 6e 74 5f 70 61 67 65 5d 7d 20 7b 0a 20 20  rent_page]} {.  
0750: 20 20 20 20 73 65 74 20 70 61 67 65 63 61 74 20      set pagecat 
0760: 22 69 6e 64 65 78 22 0a 20 20 20 7d 20 65 6c 73  "index".   } els
0770: 65 20 7b 20 69 66 20 7b 5b 72 65 67 65 78 70 20  e { if {[regexp 
0780: 7b 5e 77 69 6b 69 5c 3f 6e 61 6d 65 3d 7d 20 24  {^wiki\?name=} $
0790: 63 75 72 72 65 6e 74 5f 70 61 67 65 5d 7d 20 7b  current_page]} {
07a0: 0a 20 20 20 20 20 20 20 20 20 73 65 74 20 70 61  .         set pa
07b0: 67 65 63 61 74 20 22 77 69 6b 69 22 0a 20 20 20  gecat "wiki".   
07c0: 20 20 20 20 20 20 73 65 74 20 70 61 67 65 6e 61        set pagena
07d0: 6d 65 20 5b 73 74 72 69 6e 67 20 72 61 6e 67 65  me [string range
07e0: 20 24 63 75 72 72 65 6e 74 5f 70 61 67 65 20 31   $current_page 1
07f0: 30 20 32 30 34 38 5d 0a 20 20 20 7d 20 65 6c 73  0 2048].   } els
0800: 65 20 7b 0a 20 20 20 20 20 20 73 65 74 20 70 61  e {.      set pa
0810: 67 65 63 61 74 20 24 63 75 72 72 65 6e 74 5f 70  gecat $current_p
0820: 61 67 65 0a 20 20 20 7d 20 7d 0a 0a 20 20 20 23  age.   } }..   #
0830: 2d 2d 20 53 70 6c 69 74 20 64 6f 6d 61 69 6e 20  -- Split domain 
0840: 6e 61 6d 65 20 66 72 6f 6d 20 62 61 73 65 75 72  name from baseur
0850: 6c 0a 20 20 20 73 65 74 20 62 61 73 65 64 6f 6d  l.   set basedom
0860: 61 69 6e 20 5b 73 74 72 69 6e 67 20 72 61 6e 67  ain [string rang
0870: 65 20 24 62 61 73 65 75 72 6c 20 5b 65 78 70 72  e $baseurl [expr
0880: 20 37 2b 5b 65 78 70 72 20 7b 5b 73 74 72 69 6e   7+[expr {[strin
0890: 67 20 72 61 6e 67 65 20 24 62 61 73 65 75 72 6c  g range $baseurl
08a0: 20 34 20 34 5d 20 65 71 20 22 73 22 7d 5d 5d 20   4 4] eq "s"}]] 
08b0: 5b 65 78 70 72 20 37 2b 5b 73 74 72 69 6e 67 20  [expr 7+[string 
08c0: 66 69 72 73 74 20 22 2f 22 20 5b 73 74 72 69 6e  first "/" [strin
08d0: 67 20 72 61 6e 67 65 20 22 24 62 61 73 65 75 72  g range "$baseur
08e0: 6c 2f 22 20 38 20 35 30 5d 5d 5d 5d 0a 0a 20 20  l/" 8 50]]]]..  
08f0: 20 23 2d 2d 20 50 72 6f 6a 65 63 74 20 73 74 61   #-- Project sta
0900: 74 73 0a 20 20 20 73 65 74 20 73 74 61 74 73 5f  ts.   set stats_
0910: 64 65 73 63 72 69 70 74 69 6f 6e 20 5b 73 65 74  description [set
0920: 74 69 6e 67 20 70 72 6f 6a 65 63 74 2d 64 65 73  ting project-des
0930: 63 72 69 70 74 69 6f 6e 5d 0a 20 20 20 73 65 74  cription].   set
0940: 20 73 74 61 74 73 5f 73 6f 63 69 61 6c 20 30 0a   stats_social 0.
0950: 20 20 20 73 65 74 20 73 74 61 74 73 5f 66 6f 72     set stats_for
0960: 6b 73 20 31 0a 20 20 20 63 61 74 63 68 20 7b 20  ks 1.   catch { 
0970: 71 75 65 72 79 20 7b 20 53 45 4c 45 43 54 20 6e  query { SELECT n
0980: 61 6d 65 2c 76 61 6c 75 65 20 46 52 4f 4d 20 66  ame,value FROM f
0990: 78 5f 73 74 61 74 73 20 57 48 45 52 45 20 6e 61  x_stats WHERE na
09a0: 6d 65 20 47 4c 4f 42 20 27 73 74 61 74 73 5f 2a  me GLOB 'stats_*
09b0: 27 20 7d 20 7b 0a 20 20 20 20 20 20 73 65 74 20  ' } {.      set 
09c0: 22 24 6e 61 6d 65 22 20 22 24 76 61 6c 75 65 22  "$name" "$value"
09d0: 0a 20 20 20 20 20 20 75 70 76 61 72 20 31 20 24  .      upvar 1 $
09e0: 6e 61 6d 65 20 24 6e 61 6d 65 0a 20 20 20 7d 20  name $name.   } 
09f0: 7d 0a 7d 0a 0a 0a 23 2d 2d 20 46 6f 72 20 6f 75  }.}...#-- For ou
0a00: 74 70 75 74 74 69 6e 67 20 63 6c 61 73 73 3d 63  tputting class=c
0a10: 75 72 72 65 6e 74 20 69 6e 20 23 73 69 64 65 62  urrent in #sideb
0a20: 61 72 0a 70 72 6f 63 20 63 75 72 72 65 6e 74 20  ar.proc current 
0a30: 7b 6e 61 6d 65 7d 20 7b 0a 20 20 20 75 70 76 61  {name} {.   upva
0a40: 72 20 31 20 70 61 67 65 63 61 74 20 70 61 67 65  r 1 pagecat page
0a50: 63 61 74 0a 20 20 20 69 66 20 5b 72 65 67 65 78  cat.   if [regex
0a60: 70 20 22 5e 28 24 6e 61 6d 65 29 22 20 24 70 61  p "^($name)" $pa
0a70: 67 65 63 61 74 5d 20 7b 20 70 75 74 73 20 7b 20  gecat] { puts { 
0a80: 63 6c 61 73 73 3d 63 75 72 72 65 6e 74 7d 20 7d  class=current} }
0a90: 0a 7d 0a 0a 0a 23 2d 2d 20 54 75 72 6e 20 63 6f  .}...#-- Turn co
0aa0: 6e 66 69 67 3a 73 69 74 65 6d 61 70 2d 2a 20 75  nfig:sitemap-* u
0ab0: 72 6c 73 20 69 6e 74 6f 20 62 75 74 74 6f 6e 20  rls into button 
0ac0: 6c 69 6e 6b 73 0a 70 72 6f 63 20 75 69 3a 3a 73  links.proc ui::s
0ad0: 69 74 65 6d 61 70 5f 6c 69 6e 6b 73 20 7b 7d 20  itemap_links {} 
0ae0: 7b 0a 20 20 20 73 65 74 20 63 6f 6c 6f 72 20 22  {.   set color "
0af0: 67 72 65 65 6e 22 0a 20 20 20 66 6f 72 65 61 63  green".   foreac
0b00: 68 20 6e 61 6d 65 20 22 64 6f 77 6e 6c 6f 61 64  h name "download
0b10: 20 64 6f 63 69 64 78 20 6c 69 63 65 6e 73 65 20   docidx license 
0b20: 63 6f 6e 74 61 63 74 20 63 68 61 6e 67 65 6c 6f  contact changelo
0b30: 67 20 6e 65 77 73 20 66 72 65 73 68 63 6f 64 65  g news freshcode
0b40: 20 72 65 70 6f 73 69 74 6f 72 79 20 64 6f 61 70   repository doap
0b50: 20 70 72 6f 6a 65 63 74 2e 6a 73 6f 6e 22 20 7b   project.json" {
0b60: 0a 20 20 20 20 20 20 73 65 74 20 75 72 6c 20 5b  .      set url [
0b70: 73 65 74 74 69 6e 67 20 73 69 74 65 6d 61 70 2d  setting sitemap-
0b80: 24 6e 61 6d 65 5d 0a 20 20 20 20 20 20 69 66 20  $name].      if 
0b90: 7b 5b 73 74 72 69 6e 67 20 6c 65 6e 67 74 68 20  {[string length 
0ba0: 24 75 72 6c 5d 7d 20 7b 0a 20 20 20 20 20 20 20  $url]} {.       
0bb0: 20 20 68 74 6d 6c 20 22 3c 61 20 68 72 65 66 3d    html "<a href=
0bc0: 27 24 75 72 6c 27 20 63 6c 61 73 73 3d 27 62 75  '$url' class='bu
0bd0: 74 74 6f 6e 20 24 63 6f 6c 6f 72 20 70 72 6f 6a  tton $color proj
0be0: 2d 73 69 74 65 6d 61 70 27 3e 24 6e 61 6d 65 3c  -sitemap'>$name<
0bf0: 2f 61 3e 22 0a 20 20 20 20 20 20 7d 0a 20 20 20  /a>".      }.   
0c00: 20 20 20 73 65 74 20 63 6f 6c 6f 72 20 22 77 68     set color "wh
0c10: 69 74 65 22 0a 20 20 20 7d 0a 7d 0a 0a 23 2d 2d  ite".   }.}..#--
0c20: 20 4f 72 64 65 72 65 64 20 6c 69 73 74 20 6f 66   Ordered list of
0c30: 20 70 72 6f 6a 65 63 74 20 73 74 61 74 69 73 74   project statist
0c40: 69 63 73 20 28 77 69 6c 6c 20 70 6f 70 75 6c 61  ics (will popula
0c50: 74 65 20 67 6c 6f 62 61 6c 20 24 73 74 61 74 73  te global $stats
0c60: 28 29 20 61 72 72 61 79 29 0a 70 72 6f 63 20 75  () array).proc u
0c70: 69 3a 3a 73 74 61 74 73 20 7b 7d 20 7b 0a 20 20  i::stats {} {.  
0c80: 20 75 70 6c 65 76 65 6c 20 31 20 7b 20 71 75 65   uplevel 1 { que
0c90: 72 79 20 7b 53 45 4c 45 43 54 0a 20 20 20 20 20  ry {SELECT.     
0ca0: 28 53 45 4c 45 43 54 20 63 6f 75 6e 74 28 6f 62  (SELECT count(ob
0cb0: 6a 69 64 29 20 46 52 4f 4d 20 65 76 65 6e 74 20  jid) FROM event 
0cc0: 57 48 45 52 45 20 74 79 70 65 3d 27 63 69 27 20  WHERE type='ci' 
0cd0: 4c 49 4d 49 54 20 31 29 20 41 53 20 60 73 74 61  LIMIT 1) AS `sta
0ce0: 74 73 5f 63 68 65 63 6b 69 6e 73 60 2c 0a 20 20  ts_checkins`,.  
0cf0: 20 20 20 28 53 45 4c 45 43 54 20 63 6f 75 6e 74     (SELECT count
0d00: 28 6e 61 6d 65 29 20 46 52 4f 4d 20 66 69 6c 65  (name) FROM file
0d10: 6e 61 6d 65 20 4c 49 4d 49 54 20 31 29 20 41 53  name LIMIT 1) AS
0d20: 20 60 73 74 61 74 73 5f 66 69 6c 65 73 60 2c 0a   `stats_files`,.
0d30: 20 20 20 20 20 28 53 45 4c 45 43 54 20 63 6f 75       (SELECT cou
0d40: 6e 74 28 73 74 61 74 75 73 29 20 46 52 4f 4d 20  nt(status) FROM 
0d50: 74 69 63 6b 65 74 20 4c 49 4d 49 54 20 31 29 20  ticket LIMIT 1) 
0d60: 41 53 20 60 73 74 61 74 73 5f 74 69 63 6b 65 74  AS `stats_ticket
0d70: 73 60 2c 0a 20 20 20 20 20 28 53 45 4c 45 43 54  s`,.     (SELECT
0d80: 20 63 6f 75 6e 74 28 44 49 53 54 49 4e 43 54 20   count(DISTINCT 
0d90: 75 73 65 72 29 20 46 52 4f 4d 20 65 76 65 6e 74  user) FROM event
0da0: 20 4c 49 4d 49 54 20 31 29 20 41 53 20 60 73 74   LIMIT 1) AS `st
0db0: 61 74 73 5f 64 65 76 65 6c 6f 70 65 72 73 60 2c  ats_developers`,
0dc0: 0a 20 20 20 20 20 28 53 45 4c 45 43 54 20 63 6f  .     (SELECT co
0dd0: 75 6e 74 28 44 49 53 54 49 4e 43 54 20 76 61 6c  unt(DISTINCT val
0de0: 75 65 29 20 46 52 4f 4d 20 74 61 67 78 72 65 66  ue) FROM tagxref
0df0: 20 57 48 45 52 45 20 74 61 67 69 64 3d 38 29 20   WHERE tagid=8) 
0e00: 41 53 20 60 73 74 61 74 73 5f 62 72 61 6e 63 68  AS `stats_branch
0e10: 65 73 60 2c 0a 20 20 20 20 20 28 53 45 4c 45 43  es`,.     (SELEC
0e20: 54 20 63 6f 75 6e 74 28 74 61 67 6e 61 6d 65 29  T count(tagname)
0e30: 20 46 52 4f 4d 20 74 61 67 20 57 48 45 52 45 20   FROM tag WHERE 
0e40: 74 61 67 6e 61 6d 65 20 4c 49 4b 45 20 27 73 79  tagname LIKE 'sy
0e50: 6d 2d 25 27 29 20 41 53 20 60 73 74 61 74 73 5f  m-%') AS `stats_
0e60: 74 61 67 73 60 2c 0a 20 20 20 20 20 28 53 45 4c  tags`,.     (SEL
0e70: 45 43 54 20 63 6f 75 6e 74 28 74 61 67 6e 61 6d  ECT count(tagnam
0e80: 65 29 20 46 52 4f 4d 20 74 61 67 20 57 48 45 52  e) FROM tag WHER
0e90: 45 20 74 61 67 6e 61 6d 65 20 52 45 47 45 58 50  E tagname REGEXP
0ea0: 20 27 5e 73 79 6d 5b 2d 61 2d 7a 30 2d 39 5f 2e   '^sym[-a-z0-9_.
0eb0: 5d 2b 5c 64 2b 5c 2e 5c 64 2b 27 29 20 41 53 20  ]+\d+\.\d+') AS 
0ec0: 60 73 74 61 74 73 5f 72 65 6c 65 61 73 65 73 60  `stats_releases`
0ed0: 0a 20 20 20 7d 20 7b 7d 20 7d 0a 7d 0a 0a 23 2d  .   } {} }.}..#-
0ee0: 2d 20 4c 61 6e 67 75 61 67 65 2f 43 6f 6e 74 65  - Language/Conte
0ef0: 6e 74 20 73 74 61 74 69 73 74 69 63 73 20 28 6f  nt statistics (o
0f00: 75 74 70 75 74 73 20 63 6f 6c 6f 72 65 64 20 62  utputs colored b
0f10: 61 72 20 67 72 61 70 68 29 0a 70 72 6f 63 20 75  ar graph).proc u
0f20: 69 3a 3a 6c 61 6e 67 5f 73 74 61 74 73 20 7b 7d  i::lang_stats {}
0f30: 20 7b 0a 20 20 20 23 20 66 65 74 63 68 20 24 6c   {.   # fetch $l
0f40: 61 6e 67 28 6a 73 2f 2e 2e 2e 29 2c 20 24 6c 61  ang(js/...), $la
0f50: 6e 67 5f 63 6f 6c 6f 72 28 6a 73 29 2c 20 24 6c  ng_color(js), $l
0f60: 61 6e 67 5f 6c 69 73 74 2c 20 24 74 6f 74 61 6c  ang_list, $total
0f70: 5f 73 69 7a 65 0a 20 20 20 71 75 65 72 79 20 7b  _size.   query {
0f80: 53 45 4c 45 43 54 20 6e 61 6d 65 2c 20 76 61 6c  SELECT name, val
0f90: 75 65 20 46 52 4f 4d 20 66 78 5f 73 74 61 74 73  ue FROM fx_stats
0fa0: 20 4f 52 44 45 52 20 62 79 20 56 41 4c 55 45 20   ORDER by VALUE 
0fb0: 44 45 53 43 7d 20 7b 0a 20 20 20 20 20 20 73 65  DESC} {.      se
0fc0: 74 20 24 6e 61 6d 65 20 24 76 61 6c 75 65 0a 20  t $name $value. 
0fd0: 20 20 7d 0a 20 20 20 23 20 6f 75 74 70 75 74 20    }.   # output 
0fe0: 63 6f 6c 6f 72 20 62 61 72 20 66 6f 72 20 6c 61  color bar for la
0ff0: 6e 67 75 61 67 65 20 70 72 6f 70 6f 72 74 69 6f  nguage proportio
1000: 6e 73 0a 20 20 20 23 68 74 6d 6c 20 22 3c 64 69  ns.   #html "<di
1010: 76 20 63 6c 61 73 73 3d 6c 61 6e 67 75 61 67 65  v class=language
1020: 2d 62 61 72 20 73 74 79 6c 65 3d 27 77 69 64 74  -bar style='widt
1030: 68 3a 31 30 30 25 3b 20 68 65 69 67 68 74 3a 33  h:100%; height:3
1040: 70 74 3b 20 62 6f 78 2d 73 69 7a 69 6e 67 3a 62  pt; box-sizing:b
1050: 6f 72 64 65 72 2d 62 6f 78 3b 27 3e 22 0a 20 20  order-box;'>".  
1060: 20 66 6f 72 65 61 63 68 20 6e 61 6d 65 20 24 6c   foreach name $l
1070: 61 6e 67 5f 6c 69 73 74 20 7b 0a 20 20 20 20 20  ang_list {.     
1080: 20 73 65 74 20 70 65 72 63 65 6e 74 20 22 5b 65   set percent "[e
1090: 78 70 72 20 24 6c 61 6e 67 28 24 6e 61 6d 65 29  xpr $lang($name)
10a0: 2a 31 30 30 5d 25 22 0a 20 20 20 20 20 20 68 74  *100]%".      ht
10b0: 6d 6c 20 22 3c 73 70 61 6e 20 63 6c 61 73 73 3d  ml "<span class=
10c0: 63 6f 64 65 2d 72 61 74 65 2d 24 6e 61 6d 65 20  code-rate-$name 
10d0: 73 74 79 6c 65 3d 27 68 65 69 67 68 74 3a 31 30  style='height:10
10e0: 30 25 3b 20 77 69 64 74 68 3a 24 70 65 72 63 65  0%; width:$perce
10f0: 6e 74 3b 20 64 69 73 70 6c 61 79 3a 69 6e 6c 69  nt; display:inli
1100: 6e 65 2d 62 6c 6f 63 6b 3b 20 62 61 63 6b 67 72  ne-block; backgr
1110: 6f 75 6e 64 2d 63 6f 6c 6f 72 3a 23 24 6c 61 6e  ound-color:#$lan
1120: 67 5f 63 6f 6c 6f 72 28 24 6e 61 6d 65 29 27 20  g_color($name)' 
1130: 74 69 74 6c 65 3d 27 24 70 65 72 63 65 6e 74 20  title='$percent 
1140: 24 6e 61 6d 65 27 3e 3c 2f 73 70 61 6e 3e 22 3b  $name'></span>";
1150: 0a 20 20 20 7d 0a 20 20 20 23 68 74 6d 6c 20 22  .   }.   #html "
1160: 3c 2f 64 69 76 3e 22 3b 0a 7d 0a 0a 23 2d 2d 20  </div>";.}..#-- 
1170: 70 72 69 6e 74 20 74 77 6f 20 74 61 62 6c 65 20  print two table 
1180: 72 6f 77 73 20 66 6f 72 20 6c 61 73 74 20 63 6f  rows for last co
1190: 6d 6d 69 74 0a 70 72 6f 63 20 75 69 3a 3a 6c 61  mmit.proc ui::la
11a0: 73 74 5f 63 6f 6d 6d 69 74 20 7b 7d 20 7b 0a 20  st_commit {} {. 
11b0: 20 20 71 75 65 72 79 20 7b 0a 20 20 20 20 20 20    query {.      
11c0: 20 53 45 4c 45 43 54 20 2a 2c 20 43 41 53 54 28   SELECT *, CAST(
11d0: 6a 75 6c 69 61 6e 64 61 79 28 27 6e 6f 77 27 29  julianday('now')
11e0: 2d 6d 74 69 6d 65 20 41 53 20 49 4e 54 29 20 41  -mtime AS INT) A
11f0: 53 20 61 67 65 2c 20 73 75 62 73 74 72 28 63 6f  S age, substr(co
1200: 6d 6d 65 6e 74 2c 30 2c 31 39 39 29 20 41 53 20  mment,0,199) AS 
1210: 6d 73 67 2c 20 73 75 62 73 74 72 28 75 75 69 64  msg, substr(uuid
1220: 2c 20 30 2c 20 31 30 29 20 41 53 20 73 68 6f 72  , 0, 10) AS shor
1230: 74 5f 75 75 69 64 0a 20 20 20 20 20 20 20 46 52  t_uuid.       FR
1240: 4f 4d 20 65 76 65 6e 74 20 4a 4f 49 4e 20 62 6c  OM event JOIN bl
1250: 6f 62 20 4f 4e 20 62 6c 6f 62 2e 72 69 64 3d 65  ob ON blob.rid=e
1260: 76 65 6e 74 2e 6f 62 6a 69 64 0a 20 20 20 20 20  vent.objid.     
1270: 20 20 57 48 45 52 45 20 74 79 70 65 3d 27 63 69    WHERE type='ci
1280: 27 20 4f 52 44 45 52 20 42 59 20 6d 74 69 6d 65  ' ORDER BY mtime
1290: 20 44 45 53 43 20 4c 49 4d 49 54 20 31 0a 20 20   DESC LIMIT 1.  
12a0: 20 7d 20 7b 0a 20 20 20 20 20 20 68 74 6d 6c 20   } {.      html 
12b0: 22 20 20 3c 74 72 3e 3c 74 68 20 63 6f 6c 73 70  "  <tr><th colsp
12c0: 61 6e 3d 33 3e 24 6d 73 67 3c 2f 74 68 3e 3c 2f  an=3>$msg</th></
12d0: 74 72 3e 22 3b 0a 20 20 20 20 20 20 68 74 6d 6c  tr>";.      html
12e0: 20 22 20 20 3c 74 72 3e 3c 74 68 20 63 6f 6c 73   "  <tr><th cols
12f0: 70 61 6e 3d 33 20 73 74 79 6c 65 3d 62 61 63 6b  pan=3 style=back
1300: 67 72 6f 75 6e 64 3a 23 66 66 66 3e 3c 61 20 68  ground:#fff><a h
1310: 72 65 66 3d 27 74 69 6d 65 6c 69 6e 65 3f 75 3d  ref='timeline?u=
1320: 24 75 73 65 72 27 20 63 6c 61 73 73 3d 75 73 65  $user' class=use
1330: 72 3e 24 75 73 65 72 3c 2f 61 3e 20 61 75 74 68  r>$user</a> auth
1340: 6f 72 65 64 20 24 61 67 65 20 64 61 79 73 20 61  ored $age days a
1350: 67 6f 0a 20 20 20 20 20 20 20 20 20 20 20 20 3c  go.            <
1360: 73 70 61 6e 20 73 74 79 6c 65 3d 66 6c 6f 61 74  span style=float
1370: 3a 72 69 67 68 74 3e 6c 61 73 74 20 63 68 65 63  :right>last chec
1380: 6b 69 6e 20 3c 61 20 68 72 65 66 3d 27 63 69 2f  kin <a href='ci/
1390: 24 75 75 69 64 27 3e 24 73 68 6f 72 74 5f 75 75  $uuid'>$short_uu
13a0: 69 64 20 3c 73 70 61 6e 20 63 6c 61 73 73 3d 67  id <span class=g
13b0: 6c 79 70 68 3e 26 23 78 32 33 39 38 3b 3c 2f 73  lyph>&#x2398;</s
13c0: 70 61 6e 3e 3c 2f 61 3e 3c 2f 73 70 61 6e 3e 3c  pan></a></span><
13d0: 2f 74 68 3e 3c 2f 74 72 3e 22 3b 0a 20 20 20 7d  /th></tr>";.   }
13e0: 0a 7d 0a 0a 23 2d 2d 20 6f 75 74 70 75 74 73 20  .}..#-- outputs 
13f0: 74 61 62 6c 65 20 72 6f 77 73 20 63 6f 6e 74 61  table rows conta
1400: 69 6e 69 6e 67 20 74 6f 70 2d 6c 65 76 65 6c 20  ining top-level 
1410: 66 69 6c 65 6e 61 6d 65 73 20 61 6e 64 20 72 65  filenames and re
1420: 63 65 6e 74 20 63 68 65 63 6b 69 6e 20 63 6f 6d  cent checkin com
1430: 6d 65 6e 74 73 0a 70 72 6f 63 20 75 69 3a 3a 72  ments.proc ui::r
1440: 65 63 65 6e 74 5f 66 69 6c 65 73 20 7b 64 69 72  ecent_files {dir
1450: 6e 61 6d 65 7d 20 7b 0a 20 20 20 73 65 74 20 73  name} {.   set s
1460: 65 65 6e 20 22 28 2e 67 69 74 69 67 6e 6f 72 65  een "(.gitignore
1470: 29 22 0a 20 20 20 0a 20 20 20 23 20 73 65 61 72  )".   .   # sear
1480: 63 68 20 66 69 6c 65 73 20 75 73 69 6e 67 20 64  ch files using d
1490: 69 72 65 63 74 6f 72 79 20 61 73 20 62 61 73 65  irectory as base
14a0: 20 70 61 74 68 0a 20 20 20 73 65 74 20 62 72 61   path.   set bra
14b0: 6e 63 68 20 22 74 72 75 6e 6b 22 0a 20 20 20 73  nch "trunk".   s
14c0: 65 74 20 63 75 74 6e 61 6d 65 20 30 0a 20 20 20  et cutname 0.   
14d0: 73 65 74 20 41 4e 44 5f 44 49 52 20 22 22 0a 20  set AND_DIR "". 
14e0: 20 20 69 66 20 7b 5b 73 74 72 69 6e 67 20 6c 65    if {[string le
14f0: 6e 67 74 68 20 24 64 69 72 6e 61 6d 65 5d 7d 20  ngth $dirname]} 
1500: 7b 0a 20 20 20 20 20 20 20 73 65 74 20 64 69 72  {.       set dir
1510: 6e 61 6d 65 20 22 24 64 69 72 6e 61 6d 65 2f 22  name "$dirname/"
1520: 0a 20 20 20 20 20 20 20 73 65 74 20 63 75 74 6e  .       set cutn
1530: 61 6d 65 20 5b 73 74 72 69 6e 67 20 6c 65 6e 67  ame [string leng
1540: 74 68 20 24 64 69 72 6e 61 6d 65 5d 0a 20 20 20  th $dirname].   
1550: 20 20 20 20 73 65 74 20 41 4e 44 5f 44 49 52 20      set AND_DIR 
1560: 22 20 41 4e 44 20 73 75 62 73 74 72 28 6e 61 6d  " AND substr(nam
1570: 65 2c 20 30 2c 20 5c 24 63 75 74 6e 61 6d 65 2b  e, 0, \$cutname+
1580: 31 29 20 3d 20 5c 24 64 69 72 6e 61 6d 65 20 22  1) = \$dirname "
1590: 0a 20 20 20 7d 0a 0a 20 20 20 23 20 66 69 6c 65  .   }..   # file
15a0: 73 20 20 20 2f 2f 20 76 63 61 63 68 65 2e 72 69  s   // vcache.ri
15b0: 64 3d 6d 6c 69 6e 6b 2e 66 69 64 20 77 6f 75 6c  d=mlink.fid woul
15c0: 64 20 62 65 20 65 61 73 69 65 72 20 74 6f 20 73  d be easier to s
15d0: 6b 69 70 20 64 65 6c 65 74 65 64 20 66 69 6c 65  kip deleted file
15e0: 73 2c 20 62 75 74 20 61 63 63 65 73 73 20 69 73  s, but access is
15f0: 20 70 72 6f 68 69 62 69 74 65 64 28 3f 29 0a 20   prohibited(?). 
1600: 20 20 71 75 65 72 79 20 22 0a 20 20 20 20 20 20    query ".      
1610: 20 53 45 4c 45 43 54 20 44 49 53 54 49 4e 43 54   SELECT DISTINCT
1620: 0a 20 20 20 20 20 20 20 20 20 20 6d 2e 66 6e 69  .          m.fni
1630: 64 2c 20 6d 2e 66 69 64 2c 20 4d 41 58 28 6d 2e  d, m.fid, MAX(m.
1640: 6d 69 64 29 2c 0a 20 20 20 20 20 20 20 20 20 20  mid),.          
1650: 49 4e 53 54 52 28 53 55 42 53 54 52 28 6e 61 6d  INSTR(SUBSTR(nam
1660: 65 2c 5c 24 63 75 74 6e 61 6d 65 2b 31 29 2c 27  e,\$cutname+1),'
1670: 2f 27 29 3e 30 20 20 41 53 20 20 69 73 5f 64 69  /')>0  AS  is_di
1680: 72 2c 0a 20 20 20 20 20 20 20 20 20 20 6e 61 6d  r,.          nam
1690: 65 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20  e               
16a0: 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20 20                  
16b0: 20 41 53 20 20 70 61 74 68 6e 61 6d 65 2c 0a 20   AS  pathname,. 
16c0: 20 20 20 20 20 20 20 20 20 62 66 2e 72 69 64 20           bf.rid 
16d0: 20 20 41 53 20 20 66 6e 5f 72 69 64 2c 20 20 20    AS  fn_rid,   
16e0: 20 20 62 66 2e 75 75 69 64 20 20 20 20 41 53 20    bf.uuid    AS 
16f0: 20 66 6e 5f 75 75 69 64 2c 0a 20 20 20 20 20 20   fn_uuid,.      
1700: 20 20 20 20 62 6d 2e 72 69 64 20 20 20 41 53 20      bm.rid   AS 
1710: 20 63 69 5f 72 69 64 2c 20 20 20 20 20 62 6d 2e   ci_rid,     bm.
1720: 75 75 69 64 20 20 20 20 41 53 20 20 63 69 5f 75  uuid    AS  ci_u
1730: 75 69 64 2c 0a 20 20 20 20 20 20 20 20 20 20 53  uid,.          S
1740: 55 42 53 54 52 28 63 6f 6d 6d 65 6e 74 2c 20 30  UBSTR(comment, 0
1750: 2c 20 37 30 29 20 20 20 20 20 20 20 20 20 20 20  , 70)           
1760: 20 20 20 41 53 20 20 63 6f 6d 6d 65 6e 74 2c 0a     AS  comment,.
1770: 20 20 20 20 20 20 20 20 20 20 43 41 53 54 28 4a            CAST(J
1780: 55 4c 49 41 4e 44 41 59 28 27 6e 6f 77 27 29 2d  ULIANDAY('now')-
1790: 65 2e 6d 74 69 6d 65 20 41 53 20 49 4e 54 29 20  e.mtime AS INT) 
17a0: 41 53 20 20 61 67 65 0a 20 20 20 20 20 20 20 46  AS  age.       F
17b0: 52 4f 4d 0a 20 20 20 20 20 20 20 20 20 20 66 69  ROM.          fi
17c0: 6c 65 6e 61 6d 65 0a 20 20 20 20 20 20 20 20 20  lename.         
17d0: 20 4c 45 46 54 20 4a 4f 49 4e 20 6d 6c 69 6e 6b   LEFT JOIN mlink
17e0: 20 6d 20 4f 4e 20 6d 2e 66 6e 69 64 20 3d 20 66   m ON m.fnid = f
17f0: 69 6c 65 6e 61 6d 65 2e 66 6e 69 64 0a 20 20 20  ilename.fnid.   
1800: 20 20 20 20 20 20 20 4c 45 46 54 20 4a 4f 49 4e         LEFT JOIN
1810: 20 74 61 67 78 72 65 66 20 4f 4e 20 6d 2e 6d 69   tagxref ON m.mi
1820: 64 20 3d 20 74 61 67 78 72 65 66 2e 72 69 64 0a  d = tagxref.rid.
1830: 20 20 20 20 20 20 20 20 20 20 4c 45 46 54 20 4a            LEFT J
1840: 4f 49 4e 20 62 6c 6f 62 20 62 66 20 4f 4e 20 62  OIN blob bf ON b
1850: 66 2e 72 69 64 20 3d 20 6d 2e 66 69 64 0a 20 20  f.rid = m.fid.  
1860: 20 20 20 20 20 20 20 20 4c 45 46 54 20 4a 4f 49          LEFT JOI
1870: 4e 20 62 6c 6f 62 20 62 6d 20 4f 4e 20 62 6d 2e  N blob bm ON bm.
1880: 72 69 64 20 3d 20 6d 2e 6d 69 64 0a 20 20 20 20  rid = m.mid.    
1890: 20 20 20 20 20 20 4c 45 46 54 20 4a 4f 49 4e 20        LEFT JOIN 
18a0: 65 76 65 6e 74 20 65 20 4f 4e 20 65 2e 6f 62 6a  event e ON e.obj
18b0: 69 64 20 3d 20 6d 2e 6d 69 64 0a 20 20 20 20 20  id = m.mid.     
18c0: 20 20 57 48 45 52 45 0a 20 20 20 20 20 20 20 20    WHERE.        
18d0: 20 20 74 61 67 78 72 65 66 2e 76 61 6c 75 65 20    tagxref.value 
18e0: 3d 20 5c 24 62 72 61 6e 63 68 0a 20 20 20 20 20  = \$branch.     
18f0: 20 20 20 20 20 24 41 4e 44 5f 44 49 52 0a 20 20       $AND_DIR.  
1900: 20 20 20 20 20 47 52 4f 55 50 20 42 59 0a 20 20       GROUP BY.  
1910: 20 20 20 20 20 20 20 20 6e 61 6d 65 0a 20 20 20          name.   
1920: 20 20 20 20 48 41 56 49 4e 47 0a 20 20 20 20 20      HAVING.     
1930: 20 20 20 20 20 66 69 64 20 3c 3e 20 30 0a 20 20       fid <> 0.  
1940: 20 20 20 20 20 4f 52 44 45 52 20 42 59 0a 20 20       ORDER BY.  
1950: 20 20 20 20 20 20 20 20 69 73 5f 64 69 72 20 44          is_dir D
1960: 45 53 43 2c 20 6e 61 6d 65 20 41 53 43 2c 20 65  ESC, name ASC, e
1970: 2e 6d 74 69 6d 65 20 44 45 53 43 0a 20 20 20 22  .mtime DESC.   "
1980: 20 7b 0a 0a 20 20 20 20 20 20 23 20 73 65 70 61   {..      # sepa
1990: 72 61 74 65 20 64 69 72 65 63 74 6f 72 69 65 73  rate directories
19a0: 20 61 6e 64 20 66 69 6c 65 73 0a 20 20 20 20 20   and files.     
19b0: 20 73 65 74 20 6e 61 6d 65 20 5b 73 74 72 69 6e   set name [strin
19c0: 67 20 72 61 6e 67 65 20 24 70 61 74 68 6e 61 6d  g range $pathnam
19d0: 65 20 24 63 75 74 6e 61 6d 65 20 32 30 34 38 5d  e $cutname 2048]
19e0: 0a 20 20 20 20 20 20 73 65 74 20 64 69 72 20 5b  .      set dir [
19f0: 73 74 72 69 6e 67 20 66 69 72 73 74 20 22 2f 22  string first "/"
1a00: 20 24 6e 61 6d 65 5d 0a 20 20 20 20 20 20 69 66   $name].      if
1a10: 20 7b 24 64 69 72 3e 30 7d 20 7b 20 73 65 74 20   {$dir>0} { set 
1a20: 6e 61 6d 65 20 5b 73 74 72 69 6e 67 20 72 61 6e  name [string ran
1a30: 67 65 20 24 6e 61 6d 65 20 30 20 5b 65 78 70 72  ge $name 0 [expr
1a40: 20 24 64 69 72 2d 31 5d 5d 20 7d 0a 0a 20 20 20   $dir-1]] }..   
1a50: 20 20 20 23 20 73 6b 69 70 20 73 65 65 6e 20 66     # skip seen f
1a60: 69 6c 65 73 0a 20 20 20 20 20 20 69 66 20 5b 73  iles.      if [s
1a70: 74 72 3a 3a 63 6f 6e 74 61 69 6e 73 20 22 28 24  tr::contains "($
1a80: 6e 61 6d 65 29 22 20 24 73 65 65 6e 5d 20 7b 20  name)" $seen] { 
1a90: 63 6f 6e 74 69 6e 75 65 20 7d 20 65 6c 73 65 20  continue } else 
1aa0: 7b 20 73 65 74 20 73 65 65 6e 20 22 28 24 6e 61  { set seen "($na
1ab0: 6d 65 29 2c 24 73 65 65 6e 22 20 7d 0a 20 20 20  me),$seen" }.   
1ac0: 20 20 20 0a 20 20 20 20 20 20 23 20 6f 75 74 70     .      # outp
1ad0: 75 74 20 74 61 62 6c 65 20 65 6e 74 72 69 65 73  ut table entries
1ae0: 0a 20 20 20 20 20 20 68 74 6d 6c 20 22 20 20 20  .      html "   
1af0: 20 20 20 20 20 20 20 20 20 20 20 20 3c 74 72 3e              <tr>
1b00: 3c 74 64 3e 22 3b 0a 20 20 20 20 20 20 69 66 20  <td>";.      if 
1b10: 7b 24 64 69 72 3e 30 7d 20 7b 0a 20 20 20 20 20  {$dir>0} {.     
1b20: 20 20 20 20 20 23 20 69 66 20 74 68 65 72 65 20       # if there 
1b30: 69 73 20 61 6e 20 65 71 75 69 76 61 6c 65 6e 74  is an equivalent
1b40: 20 77 69 6b 69 20 70 61 67 65 20 66 6f 72 20 61   wiki page for a
1b50: 20 64 69 72 65 63 74 6f 72 79 2c 20 74 68 65 6e   directory, then
1b60: 20 77 65 20 6d 69 78 20 66 69 6c 65 62 6f 78 20   we mix filebox 
1b70: 2b 20 77 69 6b 69 0a 20 20 20 20 20 20 20 20 20  + wiki.         
1b80: 20 73 65 74 20 64 69 73 70 6c 61 79 20 22 77 69   set display "wi
1b90: 6b 69 22 0a 20 20 20 20 20 20 20 20 20 20 23 73  ki".          #s
1ba0: 65 74 20 64 69 73 70 6c 61 79 20 5b 3f 3a 20 5b  et display [?: [
1bb0: 73 71 6c 3a 3a 70 61 67 65 5f 65 78 69 73 74 73  sql::page_exists
1bc0: 20 22 24 64 69 72 6e 61 6d 65 24 6e 61 6d 65 22   "$dirname$name"
1bd0: 5d 20 22 77 69 6b 69 22 20 22 74 72 65 65 22 5d  ] "wiki" "tree"]
1be0: 0a 20 20 20 20 20 20 20 20 20 20 68 74 6d 6c 20  .          html 
1bf0: 22 3c 61 20 63 6c 61 73 73 3d 64 69 72 20 68 72  "<a class=dir hr
1c00: 65 66 3d 27 24 64 69 73 70 6c 61 79 2f 5b 68 74  ef='$display/[ht
1c10: 6d 6c 69 7a 65 20 24 64 69 72 6e 61 6d 65 24 6e  mlize $dirname$n
1c20: 61 6d 65 5d 27 3e 3c 62 20 63 6c 61 73 73 3d 67  ame]'><b class=g
1c30: 6c 79 70 68 3e f0 9f 93 82 3c 2f 62 3e 20 5b 68  lyph>๐Ÿ“‚</b> [h
1c40: 74 6d 6c 69 7a 65 20 24 6e 61 6d 65 5d 3c 2f 61  tmlize $name]</a
1c50: 3e 22 3b 0a 20 20 20 20 20 20 7d 20 65 6c 73 65  >";.      } else
1c60: 20 7b 0a 20 20 20 20 20 20 20 20 20 20 68 74 6d   {.          htm
1c70: 6c 20 22 3c 61 20 63 6c 61 73 73 3d 66 69 6c 65  l "<a class=file
1c80: 20 68 72 65 66 3d 27 61 72 74 69 66 61 63 74 2f   href='artifact/
1c90: 24 66 6e 5f 75 75 69 64 27 3e 3c 62 20 63 6c 61  $fn_uuid'><b cla
1ca0: 73 73 3d 67 6c 79 70 68 3e f0 9f 93 84 3c 2f 62  ss=glyph>๐Ÿ“„</b
1cb0: 3e 20 5b 68 74 6d 6c 69 7a 65 20 24 6e 61 6d 65  > [htmlize $name
1cc0: 5d 3c 2f 61 3e 22 3b 0a 20 20 20 20 20 20 7d 0a  ]</a>";.      }.
1cd0: 20 20 20 20 20 20 68 74 6d 6c 20 22 3c 2f 74 64        html "</td
1ce0: 3e 20 3c 74 64 3e 5b 68 74 6d 6c 69 7a 65 20 24  > <td>[htmlize $
1cf0: 63 6f 6d 6d 65 6e 74 5d 3c 61 20 68 72 65 66 3d  comment]<a href=
1d00: 27 63 69 2f 24 63 69 5f 75 75 69 64 27 3e e2 80  'ci/$ci_uuid'>โ€
1d10: b9 e2 80 ba 3c 2f 61 3e 3c 2f 74 64 3e 20 3c 74  นโ€บ</a></td> <t
1d20: 64 3e 5b 68 74 6d 6c 69 7a 65 20 24 61 67 65 5d  d>[htmlize $age]
1d30: 20 64 61 79 73 20 61 67 6f 3c 2f 74 64 3e 3c 2f   days ago</td></
1d40: 74 72 3e 5c 6e 22 3b 0a 20 20 20 7d 0a 7d 0a 0a  tr>\n";.   }.}..
1d50: 23 2d 2d 20 73 6f 63 69 61 6c 20 6d 65 64 69 61  #-- social media
1d60: 20 73 68 61 72 65 20 6c 69 6e 6b 73 0a 70 72 6f   share links.pro
1d70: 63 20 75 69 3a 3a 73 6f 63 69 61 6c 5f 6c 69 6e  c ui::social_lin
1d80: 6b 73 20 7b 62 61 73 65 75 72 6c 7d 20 7b 0a 20  ks {baseurl} {. 
1d90: 20 68 74 6d 6c 20 22 0a 20 20 20 3c 61 20 63 6c   html ".   <a cl
1da0: 61 73 73 3d 73 6d 6c 2d 67 6f 20 68 72 65 66 3d  ass=sml-go href=
1db0: 27 68 74 74 70 73 3a 2f 2f 70 6c 75 73 2e 67 6f  'https://plus.go
1dc0: 6f 67 6c 65 2e 63 6f 6d 2f 73 68 61 72 65 3f 75  ogle.com/share?u
1dd0: 72 6c 3d 24 62 61 73 65 75 72 6c 27 20 74 69 74  rl=$baseurl' tit
1de0: 6c 65 3d 67 6f 6f 67 6c 65 2b 3e 67 2b 3c 2f 61  le=google+>g+</a
1df0: 3e 20 26 6d 69 64 64 6f 74 3b 0a 20 20 20 3c 61  > &middot;.   <a
1e00: 20 63 6c 61 73 73 3d 73 6d 6c 2d 66 62 20 68 72   class=sml-fb hr
1e10: 65 66 3d 27 68 74 74 70 73 3a 2f 2f 77 77 77 2e  ef='https://www.
1e20: 66 61 63 65 62 6f 6f 6b 2e 63 6f 6d 2f 73 68 61  facebook.com/sha
1e30: 72 65 72 2f 73 68 61 72 65 72 2e 70 68 70 3f 75  rer/sharer.php?u
1e40: 3d 24 62 61 73 65 75 72 6c 27 20 74 69 74 6c 65  =$baseurl' title
1e50: 3d 66 61 63 65 62 6f 6f 6b 3e 66 62 3c 2f 61 3e  =facebook>fb</a>
1e60: 20 26 6d 69 64 64 6f 74 3b 0a 20 20 20 3c 61 20   &middot;.   <a 
1e70: 63 6c 61 73 73 3d 73 6d 6c 2d 74 77 20 68 72 65  class=sml-tw hre
1e80: 66 3d 27 68 74 74 70 73 3a 2f 2f 74 77 69 74 74  f='https://twitt
1e90: 65 72 2e 63 6f 6d 2f 69 6e 74 65 6e 74 2f 74 77  er.com/intent/tw
1ea0: 65 65 74 3f 75 72 6c 3d 24 62 61 73 65 75 72 6c  eet?url=$baseurl
1eb0: 27 20 74 69 74 6c 65 3d 74 77 69 74 74 65 72 3e  ' title=twitter>
1ec0: 74 77 3c 2f 61 3e 20 26 6d 69 64 64 6f 74 3b 0a  tw</a> &middot;.
1ed0: 20 20 20 3c 61 20 63 6c 61 73 73 3d 73 6d 6c 2d     <a class=sml-
1ee0: 72 64 20 68 72 65 66 3d 27 68 74 74 70 3a 2f 2f  rd href='http://
1ef0: 72 65 64 64 69 74 2e 63 6f 6d 2f 73 75 62 6d 69  reddit.com/submi
1f00: 74 3f 75 72 6c 3d 24 62 61 73 65 75 72 6c 27 20  t?url=$baseurl' 
1f10: 74 69 74 6c 65 3d 72 65 64 64 69 74 3e 72 64 3c  title=reddit>rd<
1f20: 2f 61 3e 20 26 6d 69 64 64 6f 74 3b 0a 20 20 20  /a> &middot;.   
1f30: 3c 61 20 63 6c 61 73 73 3d 73 6d 6c 2d 69 6e 20  <a class=sml-in 
1f40: 68 72 65 66 3d 27 68 74 74 70 73 3a 2f 2f 77 77  href='https://ww
1f50: 77 2e 6c 69 6e 6b 65 64 69 6e 2e 63 6f 6d 2f 73  w.linkedin.com/s
1f60: 68 61 72 65 41 72 74 69 63 6c 65 3f 6d 69 6e 69  hareArticle?mini
1f70: 3d 74 72 75 65 26 61 6d 70 3b 75 72 6c 3d 24 62  =true&amp;url=$b
1f80: 61 73 65 75 72 6c 27 20 74 69 74 6c 65 3d 6c 69  aseurl' title=li
1f90: 6e 6b 65 64 69 6e 3e 69 6e 3c 2f 61 3e 20 26 6d  nkedin>in</a> &m
1fa0: 69 64 64 6f 74 3b 0a 20 20 20 3c 61 20 63 6c 61  iddot;.   <a cla
1fb0: 73 73 3d 73 6d 6c 2d 73 75 20 68 72 65 66 3d 27  ss=sml-su href='
1fc0: 68 74 74 70 73 3a 2f 2f 77 77 77 2e 73 74 75 6d  https://www.stum
1fd0: 62 6c 65 75 70 6f 6e 2e 63 6f 6d 2f 73 75 62 6d  bleupon.com/subm
1fe0: 69 74 3f 75 72 6c 3d 24 62 61 73 65 75 72 6c 27  it?url=$baseurl'
1ff0: 20 74 69 74 6c 65 3d 73 74 75 6d 62 6c 65 75 70   title=stumbleup
2000: 6f 6e 3e 73 75 3c 2f 61 3e 20 26 6d 69 64 64 6f  on>su</a> &middo
2010: 74 3b 0a 20 20 20 3c 61 20 63 6c 61 73 73 3d 73  t;.   <a class=s
2020: 6d 6c 2d 64 6c 20 68 72 65 66 3d 27 68 74 74 70  ml-dl href='http
2030: 73 3a 2f 2f 64 65 6c 2e 69 63 69 6f 2e 75 73 2f  s://del.icio.us/
2040: 70 6f 73 74 3f 75 72 6c 3d 24 62 61 73 65 75 72  post?url=$baseur
2050: 6c 27 20 74 69 74 6c 65 3d 64 65 6c 69 63 69 6f  l' title=delicio
2060: 75 73 3e 64 6c 3c 2f 61 3e 0a 20 20 22 3b 0a 7d  us>dl</a>.  ";.}
2070: 0a 0a 0a 23 20 4f 75 74 70 75 74 73 20 61 20 74  ...# Outputs a t
2080: 65 78 74 75 61 6c 20 2f 63 68 61 6e 67 65 6c 6f  extual /changelo
2090: 67 0a 70 72 6f 63 20 77 65 62 70 61 67 65 5f 63  g.proc webpage_c
20a0: 68 61 6e 67 65 6c 6f 67 20 7b 7d 20 7b 0a 20 20  hangelog {} {.  
20b0: 68 74 6d 6c 20 22 3c 21 2d 2d 20 4e 45 57 53 2d  html "<!-- NEWS-
20c0: 73 74 79 6c 65 20 74 69 6d 65 6c 69 6e 65 20 2d  style timeline -
20d0: 2d 3e 20 3c 6d 65 74 61 20 68 74 74 70 2d 65 71  -> <meta http-eq
20e0: 75 69 76 3d 5c 22 43 6f 6e 74 65 6e 74 2d 54 79  uiv=\"Content-Ty
20f0: 70 65 5c 22 20 63 6f 6e 74 65 6e 74 3d 5c 22 74  pe\" content=\"t
2100: 65 78 74 2f 70 6c 61 69 6e 5c 22 3e 20 3c 70 72  ext/plain\"> <pr
2110: 65 3e 5c 6e 5c 6e 22 3b 0a 20 20 73 65 74 20 76  e>\n\n";.  set v
2120: 65 72 73 69 6f 6e 20 22 74 72 75 6e 6b 22 0a 20  ersion "trunk". 
2130: 20 70 75 74 73 20 22 24 76 65 72 73 69 6f 6e 20   puts "$version 
2140: 28 75 6e 72 65 6c 65 61 73 65 64 29 5c 6e 22 3b  (unreleased)\n";
2150: 0a 20 20 71 75 65 72 79 20 7b 0a 20 20 20 20 20  .  query {.     
2160: 53 45 4c 45 43 54 20 65 76 65 6e 74 2e 6d 74 69  SELECT event.mti
2170: 6d 65 2c 20 74 61 67 2e 74 61 67 6e 61 6d 65 2c  me, tag.tagname,
2180: 20 4d 41 58 28 74 61 67 2e 74 61 67 69 64 29 2c   MAX(tag.tagid),
2190: 20 44 41 54 45 28 65 76 65 6e 74 2e 6d 74 69 6d   DATE(event.mtim
21a0: 65 29 20 41 53 20 64 2c 0a 20 20 20 20 20 20 20  e) AS d,.       
21b0: 20 20 20 20 20 52 45 50 4c 41 43 45 28 54 52 49       REPLACE(TRI
21c0: 4d 28 52 45 50 4c 41 43 45 28 65 76 65 6e 74 2e  M(REPLACE(event.
21d0: 63 6f 6d 6d 65 6e 74 2c 20 63 68 61 72 28 31 30  comment, char(10
21e0: 2c 31 30 29 2c 20 63 68 61 72 28 31 30 29 29 2c  ,10), char(10)),
21f0: 20 63 68 61 72 28 38 2c 31 30 2c 31 33 2c 33 32   char(8,10,13,32
2200: 29 29 2c 20 63 68 61 72 28 31 30 29 2c 20 63 68  )), char(10), ch
2210: 61 72 28 31 30 2c 33 32 2c 33 32 2c 33 32 29 29  ar(10,32,32,32))
2220: 20 41 53 20 63 6f 6d 6d 65 6e 74 0a 20 20 20 20   AS comment.    
2230: 20 46 52 4f 4d 20 65 76 65 6e 74 0a 20 20 20 20   FROM event.    
2240: 20 20 4c 45 46 54 20 4a 4f 49 4e 20 74 61 67 78    LEFT JOIN tagx
2250: 72 65 66 20 4f 4e 20 65 76 65 6e 74 2e 6f 62 6a  ref ON event.obj
2260: 69 64 3d 74 61 67 78 72 65 66 2e 72 69 64 0a 20  id=tagxref.rid. 
2270: 20 20 20 20 20 4c 45 46 54 20 4a 4f 49 4e 20 74       LEFT JOIN t
2280: 61 67 20 4f 4e 20 74 61 67 78 72 65 66 2e 74 61  ag ON tagxref.ta
2290: 67 69 64 3d 74 61 67 2e 74 61 67 69 64 0a 20 20  gid=tag.tagid.  
22a0: 20 20 20 57 48 45 52 45 20 74 79 70 65 3d 27 63     WHERE type='c
22b0: 69 27 0a 20 20 20 20 20 47 52 4f 55 50 20 42 59  i'.     GROUP BY
22c0: 20 6f 62 6a 69 64 0a 20 20 20 20 20 4f 52 44 45   objid.     ORDE
22d0: 52 20 42 59 20 65 76 65 6e 74 2e 6d 74 69 6d 65  R BY event.mtime
22e0: 20 44 45 53 43 0a 20 20 20 20 20 4c 49 4d 49 54   DESC.     LIMIT
22f0: 20 37 35 30 0a 20 20 7d 20 7b 0a 20 20 20 20 20   750.  } {.     
2300: 69 66 20 7b 5b 72 65 67 65 78 70 20 7b 5e 73 79  if {[regexp {^sy
2310: 6d 2d 2e 2a 5c 64 2b 5c 2e 5c 64 2b 7d 20 24 74  m-.*\d+\.\d+} $t
2320: 61 67 6e 61 6d 65 5d 7d 20 7b 0a 20 20 20 20 20  agname]} {.     
2330: 20 20 20 66 6f 72 20 7b 7d 20 7b 5b 73 74 72 69     for {} {[stri
2340: 6e 67 20 6c 65 6e 67 74 68 20 24 74 61 67 6e 61  ng length $tagna
2350: 6d 65 5d 20 3e 3d 20 33 20 26 26 20 5b 72 65 67  me] >= 3 && [reg
2360: 65 78 70 20 7b 5e 5c 64 2b 5c 2e 7d 20 24 74 61  exp {^\d+\.} $ta
2370: 67 6e 61 6d 65 5d 20 3d 3d 20 30 7d 20 7b 7d 20  gname] == 0} {} 
2380: 7b 0a 20 20 20 20 20 20 20 20 20 20 73 65 74 20  {.          set 
2390: 74 61 67 6e 61 6d 65 20 5b 73 74 72 69 6e 67 20  tagname [string 
23a0: 72 61 6e 67 65 20 24 74 61 67 6e 61 6d 65 20 31  range $tagname 1
23b0: 20 31 30 30 5d 0a 20 20 20 20 20 20 20 20 7d 0a   100].        }.
23c0: 20 20 20 20 20 20 20 20 70 75 74 73 20 22 5c 6e          puts "\n
23d0: 24 74 61 67 6e 61 6d 65 20 28 24 64 29 5c 6e 22  $tagname ($d)\n"
23e0: 3b 0a 20 20 20 20 20 7d 0a 20 20 20 20 20 70 75  ;.     }.     pu
23f0: 74 73 20 22 20 2a 20 24 63 6f 6d 6d 65 6e 74 5c  ts " * $comment\
2400: 6e 22 3b 0a 20 20 7d 0a 20 20 70 75 74 73 20 22  n";.  }.  puts "
2410: 5c 6e 5c 6e 22 3b 0a 7d 0a 0a 23 20 41 6c 74 65  \n\n";.}..# Alte
2420: 72 6e 61 74 69 76 65 20 74 6f 20 2f 72 61 77 20  rnative to /raw 
2430: 74 72 75 6e 6b 20 66 69 6c 65 20 61 63 63 65 73  trunk file acces
2440: 73 20 77 69 74 68 6f 75 74 20 3f 6e 61 6d 65 3d  s without ?name=
2450: 75 75 69 64 2c 0a 23 20 44 6f 65 73 6e 27 74 20  uuid,.# Doesn't 
2460: 77 6f 72 6b 20 77 69 74 68 20 43 4f 4e 54 45 4e  work with CONTEN
2470: 54 28 29 20 79 65 74 2e 0a 70 72 6f 63 20 77 65  T() yet..proc we
2480: 62 70 61 67 65 5f 63 61 74 20 7b 7d 20 7b 0a 20  bpage_cat {} {. 
2490: 20 73 65 74 20 6e 61 6d 65 20 5b 67 65 74 50 61   set name [getPa
24a0: 72 61 6d 65 74 65 72 20 6e 61 6d 65 20 22 22 5d  rameter name ""]
24b0: 0a 20 20 69 66 20 7b 21 5b 73 74 72 69 6e 67 20  .  if {![string 
24c0: 6c 65 6e 67 74 68 20 24 6e 61 6d 65 5d 7d 20 7b  length $name]} {
24d0: 20 70 75 74 73 20 22 4e 6f 20 66 69 6c 65 6e 61   puts "No filena
24e0: 6d 65 20 67 69 76 65 6e 2e 22 3b 20 62 72 65 61  me given."; brea
24f0: 6b 3b 20 7d 0a 20 20 71 75 65 72 79 20 7b 0a 20  k; }.  query {. 
2500: 20 20 20 20 53 45 4c 45 43 54 20 75 75 69 64 0a      SELECT uuid.
2510: 20 20 20 20 20 46 52 4f 4d 20 62 6c 6f 62 20 4c       FROM blob L
2520: 45 46 54 20 4a 4f 49 4e 20 6d 6c 69 6e 6b 20 4f  EFT JOIN mlink O
2530: 4e 20 62 6c 6f 62 2e 72 69 64 3d 6d 6c 69 6e 6b  N blob.rid=mlink
2540: 2e 66 69 64 0a 20 20 20 20 20 20 20 20 20 20 20  .fid.           
2550: 20 20 20 20 4c 45 46 54 20 4a 4f 49 4e 20 66 69      LEFT JOIN fi
2560: 6c 65 6e 61 6d 65 20 4f 4e 20 6d 6c 69 6e 6b 2e  lename ON mlink.
2570: 66 6e 69 64 3d 66 69 6c 65 6e 61 6d 65 2e 66 6e  fnid=filename.fn
2580: 69 64 0a 20 20 20 20 20 57 48 45 52 45 20 6e 61  id.     WHERE na
2590: 6d 65 20 3d 20 24 6e 61 6d 65 0a 20 20 20 20 20  me = $name.     
25a0: 4f 52 44 45 52 20 42 59 20 72 69 64 20 44 45 53  ORDER BY rid DES
25b0: 43 20 4c 49 4d 49 54 20 31 0a 20 20 7d 20 7b 20  C LIMIT 1.  } { 
25c0: 68 74 6d 6c 20 5b 61 72 74 69 66 61 63 74 20 22  html [artifact "
25d0: 24 75 75 69 64 22 5d 3b 20 7d 0a 7d 0a 0a 23 20  $uuid"]; }.}..# 
25e0: 47 65 6e 65 72 61 74 65 20 61 20 74 65 78 74 2f  Generate a text/
25f0: 75 72 69 2d 6c 69 73 74 20 66 6f 72 20 61 76 61  uri-list for ava
2600: 69 6c 61 62 6c 65 20 66 69 6c 65 73 0a 70 72 6f  ilable files.pro
2610: 63 20 77 65 62 70 61 67 65 5f 75 72 69 2d 6c 69  c webpage_uri-li
2620: 73 74 20 7b 7d 20 7b 0a 20 20 68 74 6d 6c 20 22  st {} {.  html "
2630: 23 20 46 6f 73 73 69 6c 20 6c 61 74 65 73 74 20  # Fossil latest 
2640: 66 69 6c 65 20 72 65 66 65 72 65 6e 63 65 73 20  file references 
2650: 3c 70 72 65 3e 5c 72 5c 6e 22 0a 20 20 71 75 65  <pre>\r\n".  que
2660: 72 79 20 7b 0a 20 20 20 20 20 53 45 4c 45 43 54  ry {.     SELECT
2670: 20 66 69 6c 65 6e 61 6d 65 2e 6e 61 6d 65 2c 20   filename.name, 
2680: 75 75 69 64 0a 20 20 20 20 20 46 52 4f 4d 20 62  uuid.     FROM b
2690: 6c 6f 62 20 4c 45 46 54 20 4a 4f 49 4e 20 6d 6c  lob LEFT JOIN ml
26a0: 69 6e 6b 20 4f 4e 20 62 6c 6f 62 2e 72 69 64 3d  ink ON blob.rid=
26b0: 6d 6c 69 6e 6b 2e 66 69 64 20 4c 45 46 54 20 4a  mlink.fid LEFT J
26c0: 4f 49 4e 20 66 69 6c 65 6e 61 6d 65 20 4f 4e 20  OIN filename ON 
26d0: 6d 6c 69 6e 6b 2e 66 6e 69 64 3d 66 69 6c 65 6e  mlink.fnid=filen
26e0: 61 6d 65 2e 66 6e 69 64 0a 20 20 20 20 20 47 52  ame.fnid.     GR
26f0: 4f 55 50 20 42 59 20 66 69 6c 65 6e 61 6d 65 2e  OUP BY filename.
2700: 6e 61 6d 65 20 20 4f 52 44 45 52 20 42 59 20 72  name  ORDER BY r
2710: 69 64 20 44 45 53 43 0a 20 20 7d 20 7b 20 68 74  id DESC.  } { ht
2720: 6d 6c 20 22 24 6e 61 6d 65 3f 6e 61 6d 65 3d 24  ml "$name?name=$
2730: 75 75 69 64 5c 72 5c 6e 22 20 7d 0a 7d 0a 0a 23  uuid\r\n" }.}..#
2740: 20 49 6e 76 6f 6b 65 73 20 77 65 62 20 72 65 71   Invokes web req
2750: 75 65 73 74 20 70 61 67 65 20 70 72 6f 63 73 0a  uest page procs.
2760: 70 72 6f 63 20 77 65 62 70 61 67 65 5f 68 6f 6f  proc webpage_hoo
2770: 6b 20 7b 7d 20 7b 0a 20 20 23 69 66 20 7b 21 20  k {} {.  #if {! 
2780: 5b 61 6e 79 63 61 70 20 72 6f 5d 7d 20 7b 20 62  [anycap ro]} { b
2790: 72 65 61 6b 20 7d 0a 20 20 63 61 74 63 68 20 7b  reak }.  catch {
27a0: 20 22 77 65 62 70 61 67 65 5f 24 3a 3a 77 65 62   "webpage_$::web
27b0: 5f 6e 61 6d 65 22 3b 20 72 65 74 75 72 6e 20 2d  _name"; return -
27c0: 63 6f 64 65 20 32 20 66 6f 75 6e 64 3b 20 7d 20  code 2 found; } 
27d0: 72 63 0a 20 20 69 66 20 7b 22 24 72 63 22 20 65  rc.  if {"$rc" e
27e0: 71 20 22 66 6f 75 6e 64 22 7d 20 7b 20 62 72 65  q "found"} { bre
27f0: 61 6b 20 63 6f 6e 74 69 6e 75 65 20 7d 0a 7d 0a  ak continue }.}.
2800: 0a 0a 23 2d 2d 20 57 68 69 74 65 6c 69 73 74 20  ..#-- Whitelist 
2810: 66 6f 72 20 53 51 4c 20 70 61 72 61 6d 73 0a 23  for SQL params.#
2820: 20 4a 75 73 74 20 72 65 61 6c 69 7a 65 64 20 74   Just realized t
2830: 68 69 73 20 69 73 20 72 65 64 75 6e 64 61 6e 74  his is redundant
2840: 3b 20 62 65 63 61 75 73 65 20 71 75 65 72 79 20  ; because query 
2850: 7b 7d 20 61 63 63 65 70 74 73 0a 23 20 75 6e 69  {} accepts.# uni
2860: 6e 74 65 72 70 6f 6c 61 74 65 64 20 5c 24 76 61  nterpolated \$va
2870: 72 6e 61 6d 65 73 20 61 73 20 70 61 72 61 6d 65  rnames as parame
2880: 74 65 72 20 70 6c 61 63 65 68 6f 6c 64 65 72 73  ter placeholders
2890: 2e 0a 70 72 6f 63 20 73 71 6c 3a 3a 61 6c 6c 6f  ..proc sql::allo
28a0: 77 65 64 20 7b 73 74 72 7d 20 7b 0a 20 20 20 72  wed {str} {.   r
28b0: 65 74 75 72 6e 20 5b 72 65 67 65 78 70 20 7b 5e  eturn [regexp {^
28c0: 5b 2d 61 2d 7a 41 2d 5a 30 2d 39 20 21 24 26 2f  [-a-zA-Z0-9 !$&/
28d0: 28 29 7b 7d 3d 3c 3e 2c 2e 3b 3a 5f 2b 23 2a 40  (){}=<>,.;:_+#*@
28e0: 5d 2b 24 7d 20 24 73 74 72 5d 0a 7d 0a 23 2d 2d  ]+$} $str].}.#--
28f0: 20 41 6c 73 6f 20 70 72 6f 68 69 62 69 74 20 72   Also prohibit r
2900: 65 67 65 78 20 73 70 65 63 69 61 6c 20 63 68 61  egex special cha
2910: 72 73 0a 70 72 6f 63 20 73 71 6c 3a 3a 61 6c 6c  rs.proc sql::all
2920: 6f 77 65 64 5f 72 65 67 65 78 70 20 7b 73 74 72  owed_regexp {str
2930: 7d 20 7b 0a 20 20 20 72 65 74 75 72 6e 20 5b 72  } {.   return [r
2940: 65 67 65 78 70 20 7b 5e 5b 2d 61 2d 7a 41 2d 5a  egexp {^[-a-zA-Z
2950: 30 2d 39 20 21 24 26 2f 20 20 20 20 3d 3c 3e 2c  0-9 !$&/    =<>,
2960: 2e 3b 3a 5f 20 23 20 40 5d 2b 24 7d 20 24 73 74  .;:_ # @]+$} $st
2970: 72 5d 0a 7d 0a 0a 0a 23 2d 2d 20 43 68 65 63 6b  r].}...#-- Check
2980: 20 66 6f 72 20 65 78 69 73 74 65 6e 63 65 20 6f   for existence o
2990: 66 20 77 69 6b 69 20 70 61 67 65 0a 70 72 6f 63  f wiki page.proc
29a0: 20 73 71 6c 3a 3a 70 61 67 65 5f 65 78 69 73 74   sql::page_exist
29b0: 73 20 7b 6e 61 6d 65 7d 20 7b 0a 20 20 20 71 75  s {name} {.   qu
29c0: 65 72 79 20 7b 53 45 4c 45 43 54 20 31 20 46 52  ery {SELECT 1 FR
29d0: 4f 4d 20 74 61 67 20 57 48 45 52 45 20 74 61 67  OM tag WHERE tag
29e0: 6e 61 6d 65 20 3d 20 28 27 77 69 6b 69 2d 27 20  name = ('wiki-' 
29f0: 7c 7c 20 24 6e 61 6d 65 29 7d 20 7b 20 72 65 74  || $name)} { ret
2a00: 75 72 6e 20 31 20 7d 0a 20 20 20 72 65 74 75 72  urn 1 }.   retur
2a10: 6e 20 30 0a 7d 0a 0a 0a 23 2d 2d 20 43 68 65 63  n 0.}...#-- Chec
2a20: 6b 20 69 66 20 65 78 61 63 74 20 66 69 6c 65 20  k if exact file 
2a30: 6e 61 6d 65 20 28 69 6e 63 6c 75 64 69 6e 67 20  name (including 
2a40: 70 61 74 68 29 20 65 78 69 73 74 73 20 69 6e 20  path) exists in 
2a50: 72 65 70 6f 73 69 74 6f 72 79 0a 70 72 6f 63 20  repository.proc 
2a60: 73 71 6c 3a 3a 66 69 6c 65 5f 65 78 69 73 74 73  sql::file_exists
2a70: 20 7b 6e 61 6d 65 7d 20 7b 0a 20 20 20 71 75 65   {name} {.   que
2a80: 72 79 20 7b 53 45 4c 45 43 54 20 31 20 46 52 4f  ry {SELECT 1 FRO
2a90: 4d 20 66 69 6c 65 6e 61 6d 65 20 57 48 45 52 45  M filename WHERE
2aa0: 20 6e 61 6d 65 20 3d 20 24 6e 61 6d 65 7d 20 7b   name = $name} {
2ab0: 20 72 65 74 75 72 6e 20 31 20 7d 0a 20 20 20 72   return 1 }.   r
2ac0: 65 74 75 72 6e 20 30 0a 7d 0a 0a 0a 23 2d 2d 20  eturn 0.}...#-- 
2ad0: 46 69 6e 64 20 66 69 6c 65 20 62 79 20 62 61 73  Find file by bas
2ae0: 65 6e 61 6d 65 0a 70 72 6f 63 20 73 71 6c 3a 3a  ename.proc sql::
2af0: 66 69 6e 64 5f 66 69 6c 65 20 7b 70 61 74 68 7d  find_file {path}
2b00: 20 7b 0a 20 20 20 69 66 20 7b 21 5b 73 71 6c 3a   {.   if {![sql:
2b10: 3a 61 6c 6c 6f 77 65 64 5f 72 65 67 65 78 70 20  :allowed_regexp 
2b20: 24 70 61 74 68 5d 7d 20 7b 20 72 65 74 75 72 6e  $path]} { return
2b30: 20 30 20 7d 0a 20 20 20 71 75 65 72 79 20 7b 53   0 }.   query {S
2b40: 45 4c 45 43 54 20 6e 61 6d 65 20 46 52 4f 4d 20  ELECT name FROM 
2b50: 66 69 6c 65 6e 61 6d 65 20 57 48 45 52 45 20 6e  filename WHERE n
2b60: 61 6d 65 20 52 45 47 45 58 50 20 28 27 28 5e 7c  ame REGEXP ('(^|
2b70: 2f 29 27 20 7c 7c 20 24 70 61 74 68 20 7c 7c 20  /)' || $path || 
2b80: 27 5c 24 27 29 7d 20 7b 20 72 65 74 75 72 6e 20  '\$')} { return 
2b90: 24 6e 61 6d 65 20 7d 0a 20 20 20 72 65 74 75 72  $name }.   retur
2ba0: 6e 20 22 22 0a 7d 0a 0a 0a 23 2d 2d 20 43 68 65  n "".}...#-- Che
2bb0: 63 6b 20 69 66 20 64 69 72 65 63 74 6f 72 79 20  ck if directory 
2bc0: 65 78 69 73 74 73 0a 70 72 6f 63 20 73 71 6c 3a  exists.proc sql:
2bd0: 3a 64 69 72 5f 65 78 69 73 74 73 20 7b 70 61 74  :dir_exists {pat
2be0: 68 7d 20 7b 0a 20 20 20 69 66 20 7b 21 5b 73 71  h} {.   if {![sq
2bf0: 6c 3a 3a 61 6c 6c 6f 77 65 64 5f 72 65 67 65 78  l::allowed_regex
2c00: 70 20 24 70 61 74 68 5d 7d 20 7b 20 72 65 74 75  p $path]} { retu
2c10: 72 6e 20 30 20 7d 0a 20 20 20 71 75 65 72 79 20  rn 0 }.   query 
2c20: 7b 53 45 4c 45 43 54 20 6e 61 6d 65 20 46 52 4f  {SELECT name FRO
2c30: 4d 20 66 69 6c 65 6e 61 6d 65 20 57 48 45 52 45  M filename WHERE
2c40: 20 6e 61 6d 65 20 52 45 47 45 58 50 20 28 27 5e   name REGEXP ('^
2c50: 27 20 7c 7c 20 24 70 61 74 68 20 7c 7c 20 27 2f  ' || $path || '/
2c60: 2e 2b 27 29 7d 20 7b 20 72 65 74 75 72 6e 20 31  .+')} { return 1
2c70: 20 7d 0a 20 20 20 72 65 74 75 72 6e 20 30 0a 7d   }.   return 0.}
2c80: 0a 0a 20 20 20 0a 23 2d 2d 20 72 65 74 75 72 6e  ..   .#-- return
2c90: 73 20 74 72 75 65 20 69 66 20 73 74 72 69 6e 67  s true if string
2ca0: 20 63 6f 6e 74 61 69 6e 65 64 20 69 6e 20 61 6e   contained in an
2cb0: 6f 74 68 65 72 20 73 74 72 69 6e 67 0a 70 72 6f  other string.pro
2cc0: 63 20 73 74 72 3a 3a 63 6f 6e 74 61 69 6e 73 20  c str::contains 
2cd0: 7b 6e 65 65 64 6c 65 20 68 61 79 73 74 61 63 6b  {needle haystack
2ce0: 7d 20 7b 0a 20 20 20 72 65 74 75 72 6e 20 5b 65  } {.   return [e
2cf0: 78 70 72 20 7b 2d 31 20 21 3d 20 5b 73 74 72 69  xpr {-1 != [stri
2d00: 6e 67 20 66 69 72 73 74 20 24 6e 65 65 64 6c 65  ng first $needle
2d10: 20 24 68 61 79 73 74 61 63 6b 5d 7d 5d 0a 7d 0a   $haystack]}].}.
2d20: 0a 23 2d 2d 20 77 72 61 70 70 65 72 20 66 6f 72  .#-- wrapper for
2d30: 20 5b 73 74 72 69 6e 67 20 66 69 72 73 74 20 2e   [string first .
2d40: 2e 2e 5d 20 74 6f 20 73 75 70 70 6f 72 74 20 73  ..] to support s
2d50: 74 61 72 74 69 6e 64 65 78 0a 70 72 6f 63 20 73  tartindex.proc s
2d60: 74 72 3a 3a 6e 65 78 74 20 7b 73 65 61 72 63 68  tr::next {search
2d70: 20 63 6f 6e 74 65 6e 74 20 73 74 61 72 74 7d 20   content start} 
2d80: 7b 0a 20 20 20 23 20 63 75 74 20 6f 75 74 20 24  {.   # cut out $
2d90: 63 6f 6e 74 65 6e 74 20 61 74 20 24 73 74 61 72  content at $star
2da0: 74 20 62 65 66 6f 72 65 20 73 65 61 72 63 68 69  t before searchi
2db0: 6e 67 0a 20 20 20 73 65 74 20 70 20 5b 73 74 72  ng.   set p [str
2dc0: 69 6e 67 20 66 69 72 73 74 20 24 73 65 61 72 63  ing first $searc
2dd0: 68 20 5b 73 74 72 69 6e 67 20 72 61 6e 67 65 20  h [string range 
2de0: 24 63 6f 6e 74 65 6e 74 20 24 73 74 61 72 74 20  $content $start 
2df0: 5b 73 74 72 69 6e 67 20 6c 65 6e 67 74 68 20 24  [string length $
2e00: 63 6f 6e 74 65 6e 74 5d 5d 5d 0a 20 20 20 69 66  content]]].   if
2e10: 20 5b 65 78 70 72 20 24 70 3e 3d 30 5d 20 7b 0a   [expr $p>=0] {.
2e20: 20 20 20 20 20 20 73 65 74 20 70 20 5b 65 78 70        set p [exp
2e30: 72 20 24 73 74 61 72 74 2b 24 70 5d 0a 20 20 20  r $start+$p].   
2e40: 7d 0a 20 20 20 72 65 74 75 72 6e 20 24 70 0a 7d  }.   return $p.}
2e50: 0a 0a 23 2d 2d 20 65 6e 63 6c 6f 73 65 20 73 74  ..#-- enclose st
2e60: 72 69 6e 67 20 69 6e 20 65 2e 67 2e 20 68 74 6d  ring in e.g. htm
2e70: 6c 20 74 61 67 73 0a 70 72 6f 63 20 73 74 72 3a  l tags.proc str:
2e80: 3a 77 72 61 70 20 7b 63 6f 6e 74 65 6e 74 20 73  :wrap {content s
2e90: 65 61 72 63 68 20 62 65 66 6f 72 65 20 61 66 74  earch before aft
2ea0: 65 72 7d 20 7b 0a 20 20 20 73 65 74 20 6c 65 6e  er} {.   set len
2eb0: 20 5b 73 74 72 69 6e 67 20 6c 65 6e 67 74 68 20   [string length 
2ec0: 24 73 65 61 72 63 68 5d 0a 20 20 20 73 65 74 20  $search].   set 
2ed0: 70 20 30 0a 20 20 20 77 68 69 6c 65 20 7b 5b 65  p 0.   while {[e
2ee0: 78 70 72 20 5b 73 65 74 20 70 20 5b 73 74 72 3a  xpr [set p [str:
2ef0: 3a 6e 65 78 74 20 24 73 65 61 72 63 68 20 24 63  :next $search $c
2f00: 6f 6e 74 65 6e 74 20 24 70 5d 5d 3e 3d 30 5d 7d  ontent $p]]>=0]}
2f10: 20 7b 0a 20 20 20 20 20 20 73 65 74 20 63 6f 6e   {.      set con
2f20: 74 65 6e 74 20 22 5b 73 74 72 69 6e 67 20 72 61  tent "[string ra
2f30: 6e 67 65 20 24 63 6f 6e 74 65 6e 74 20 30 20 5b  nge $content 0 [
2f40: 65 78 70 72 20 24 70 2d 31 5d 5d 24 62 65 66 6f  expr $p-1]]$befo
2f50: 72 65 24 73 65 61 72 63 68 24 61 66 74 65 72 5b  re$search$after[
2f60: 73 74 72 69 6e 67 20 72 61 6e 67 65 20 24 63 6f  string range $co
2f70: 6e 74 65 6e 74 20 5b 65 78 70 72 20 24 70 2b 24  ntent [expr $p+$
2f80: 6c 65 6e 5d 20 32 30 30 30 5d 22 3b 0a 20 20 20  len] 2000]";.   
2f90: 20 20 20 73 65 74 20 70 20 5b 65 78 70 72 20 24     set p [expr $
2fa0: 70 2b 5b 73 74 72 69 6e 67 20 6c 65 6e 67 74 68  p+[string length
2fb0: 20 22 24 62 65 66 6f 72 65 2b 24 73 65 61 72 63   "$before+$searc
2fc0: 68 2b 24 61 66 74 65 72 22 5d 5d 3b 20 23 20 73  h+$after"]]; # s
2fd0: 6b 69 70 20 61 20 6c 69 74 74 6c 65 20 66 75 72  kip a little fur
2fe0: 74 68 65 72 0a 20 20 20 7d 0a 20 20 20 72 65 74  ther.   }.   ret
2ff0: 75 72 6e 20 24 63 6f 6e 74 65 6e 74 0a 7d 0a 0a  urn $content.}..
3000: 23 2d 2d 20 53 70 6c 69 74 20 73 74 72 69 6e 67  #-- Split string
3010: 20 69 6e 74 6f 20 6c 69 73 74 20 6f 6e 20 64 65   into list on de
3020: 6c 69 6d 69 74 65 72 20 63 68 61 72 61 63 74 65  limiter characte
3030: 72 0a 23 20 28 62 61 73 69 63 61 6c 6c 79 20 6a  r.# (basically j
3040: 75 73 74 20 74 75 72 6e 73 20 64 65 6c 69 6d 69  ust turns delimi
3050: 74 65 72 20 69 6e 74 6f 20 73 70 61 63 65 29 0a  ter into space).
3060: 23 0a 70 72 6f 63 20 73 74 72 3a 3a 65 78 70 6c  #.proc str::expl
3070: 6f 64 65 20 7b 64 65 6c 69 6d 20 73 74 72 7d 20  ode {delim str} 
3080: 7b 0a 20 20 20 73 65 74 20 72 20 22 22 0a 20 20  {.   set r "".  
3090: 20 73 65 74 20 6c 65 6e 20 5b 73 74 72 69 6e 67   set len [string
30a0: 20 6c 65 6e 67 74 68 20 24 73 74 72 5d 0a 20 20   length $str].  
30b0: 20 77 68 69 6c 65 20 7b 2d 31 20 21 3d 20 5b 73   while {-1 != [s
30c0: 65 74 20 70 20 5b 73 74 72 69 6e 67 20 66 69 72  et p [string fir
30d0: 73 74 20 24 64 65 6c 69 6d 20 24 73 74 72 5d 5d  st $delim $str]]
30e0: 7d 20 7b 0a 20 20 20 20 20 20 73 65 74 20 72 20  } {.      set r 
30f0: 22 24 72 20 5b 73 74 72 69 6e 67 20 72 61 6e 67  "$r [string rang
3100: 65 20 24 73 74 72 20 30 20 5b 65 78 70 72 20 24  e $str 0 [expr $
3110: 70 2d 31 5d 5d 22 0a 20 20 20 20 20 20 73 65 74  p-1]]".      set
3120: 20 73 74 72 20 5b 73 74 72 69 6e 67 20 72 61 6e   str [string ran
3130: 67 65 20 24 73 74 72 20 5b 2b 2b 20 70 5d 20 24  ge $str [++ p] $
3140: 6c 65 6e 5d 0a 20 20 20 7d 0a 20 20 20 72 65 74  len].   }.   ret
3150: 75 72 6e 20 5b 6c 69 73 74 20 5b 73 74 72 69 6e  urn [list [strin
3160: 67 20 74 72 69 6d 20 22 24 72 20 24 73 74 72 22  g trim "$r $str"
3170: 5d 5d 0a 7d 20 20 20 0a 0a 23 2d 2d 20 45 78 74  ]].}   ..#-- Ext
3180: 72 61 63 74 20 64 69 72 6e 61 6d 65 20 66 72 6f  ract dirname fro
3190: 6d 20 70 61 74 68 2f 66 69 6c 65 2f 6e 61 6d 65  m path/file/name
31a0: 0a 70 72 6f 63 20 73 74 72 3a 3a 64 69 72 6e 61  .proc str::dirna
31b0: 6d 65 20 7b 70 61 74 68 7d 20 7b 0a 20 20 20 72  me {path} {.   r
31c0: 65 74 75 72 6e 20 5b 73 74 72 69 6e 67 20 72 61  eturn [string ra
31d0: 6e 67 65 20 24 70 61 74 68 20 30 20 5b 65 78 70  nge $path 0 [exp
31e0: 72 20 5b 73 74 72 69 6e 67 20 6c 61 73 74 20 22  r [string last "
31f0: 2f 22 20 24 70 61 74 68 5d 2d 31 5d 5d 0a 7d 0a  /" $path]-1]].}.
3200: 0a                                               .