Hex Artifact Content

Artifact 9ed60c581dabcc5366305524de5f99066975b7a2:


0000: 23 20 52 65 6d 6f 76 65 64 20 70 72 6f 76 69 73  # Removed provis
0010: 69 6f 6e 20 6f 66 20 74 68 65 20 62 61 63 6b 77  ion of the backw
0020: 61 72 64 20 63 6f 6d 70 61 74 69 62 6c 65 20 6e  ard compatible n
0030: 61 6d 65 2e 20 4d 6f 76 65 64 20 74 6f 20 73 65  ame. Moved to se
0040: 70 61 72 61 74 65 0a 23 20 66 69 6c 65 2f 70 61  parate.# file/pa
0050: 63 6b 61 67 65 2e 0a 0a 70 61 63 6b 61 67 65 20  ckage...package 
0060: 72 65 71 75 69 72 65 20 76 66 73 0a 0a 23 20 55  require vfs..# U
0070: 73 69 6e 67 20 74 68 65 20 76 66 73 2c 20 6d 65  sing the vfs, me
0080: 6d 63 68 61 6e 20 61 6e 64 20 54 72 66 20 65 78  mchan and Trf ex
0090: 74 65 6e 73 69 6f 6e 73 2c 20 77 65 20 6f 75 67  tensions, we oug
00a0: 68 74 20 74 6f 20 62 65 20 61 62 6c 65 0a 23 20  ht to be able.# 
00b0: 74 6f 20 77 72 69 74 65 20 61 20 54 63 6c 2d 6f  to write a Tcl-o
00c0: 6e 6c 79 20 7a 69 70 20 76 69 72 74 75 61 6c 20  nly zip virtual 
00d0: 66 69 6c 65 73 79 73 74 65 6d 2e 20 20 57 68 61  filesystem.  Wha
00e0: 74 20 77 65 20 68 61 76 65 20 62 65 6c 6f 77 0a  t we have below.
00f0: 23 20 69 73 20 62 61 73 69 63 61 6c 6c 79 20 74  # is basically t
0100: 68 61 74 2e 0a 0a 6e 61 6d 65 73 70 61 63 65 20  hat...namespace 
0110: 65 76 61 6c 20 76 66 73 3a 3a 7a 69 70 20 7b 7d  eval vfs::zip {}
0120: 0a 0a 23 20 55 73 65 64 20 74 6f 20 65 78 65 63  ..# Used to exec
0130: 75 74 65 20 61 20 7a 69 70 20 61 72 63 68 69 76  ute a zip archiv
0140: 65 2e 20 20 54 68 69 73 20 69 73 20 72 61 74 68  e.  This is rath
0150: 65 72 20 6c 69 6b 65 20 61 20 6a 61 72 20 66 69  er like a jar fi
0160: 6c 65 0a 23 20 62 75 74 20 73 69 6d 70 6c 65 72  le.# but simpler
0170: 2e 20 20 57 65 20 73 69 6d 70 6c 79 20 6d 6f 75  .  We simply mou
0180: 6e 74 20 69 74 20 61 6e 64 20 74 68 65 6e 20 73  nt it and then s
0190: 6f 75 72 63 65 20 61 20 74 6f 70 6c 65 76 65 6c  ource a toplevel
01a0: 0a 23 20 66 69 6c 65 20 63 61 6c 6c 65 64 20 27  .# file called '
01b0: 6d 61 69 6e 2e 74 63 6c 27 2e 0a 70 72 6f 63 20  main.tcl'..proc 
01c0: 76 66 73 3a 3a 7a 69 70 3a 3a 45 78 65 63 75 74  vfs::zip::Execut
01d0: 65 20 7b 7a 69 70 66 69 6c 65 7d 20 7b 0a 20 20  e {zipfile} {.  
01e0: 20 20 4d 6f 75 6e 74 20 24 7a 69 70 66 69 6c 65    Mount $zipfile
01f0: 20 24 7a 69 70 66 69 6c 65 0a 20 20 20 20 73 6f   $zipfile.    so
0200: 75 72 63 65 20 5b 66 69 6c 65 20 6a 6f 69 6e 20  urce [file join 
0210: 24 7a 69 70 66 69 6c 65 20 6d 61 69 6e 2e 74 63  $zipfile main.tc
0220: 6c 5d 0a 7d 0a 0a 70 72 6f 63 20 76 66 73 3a 3a  l].}..proc vfs::
0230: 7a 69 70 3a 3a 4d 6f 75 6e 74 20 7b 7a 69 70 66  zip::Mount {zipf
0240: 69 6c 65 20 6c 6f 63 61 6c 7d 20 7b 0a 20 20 20  ile local} {.   
0250: 20 73 65 74 20 66 64 20 5b 3a 3a 7a 69 70 3a 3a   set fd [::zip::
0260: 6f 70 65 6e 20 5b 3a 3a 66 69 6c 65 20 6e 6f 72  open [::file nor
0270: 6d 61 6c 69 7a 65 20 24 7a 69 70 66 69 6c 65 5d  malize $zipfile]
0280: 5d 0a 20 20 20 20 76 66 73 3a 3a 66 69 6c 65 73  ].    vfs::files
0290: 79 73 74 65 6d 20 6d 6f 75 6e 74 20 24 6c 6f 63  ystem mount $loc
02a0: 61 6c 20 5b 6c 69 73 74 20 3a 3a 76 66 73 3a 3a  al [list ::vfs::
02b0: 7a 69 70 3a 3a 68 61 6e 64 6c 65 72 20 24 66 64  zip::handler $fd
02c0: 5d 0a 20 20 20 20 23 20 52 65 67 69 73 74 65 72  ].    # Register
02d0: 20 63 6f 6d 6d 61 6e 64 20 74 6f 20 75 6e 6d 6f   command to unmo
02e0: 75 6e 74 0a 20 20 20 20 76 66 73 3a 3a 52 65 67  unt.    vfs::Reg
02f0: 69 73 74 65 72 4d 6f 75 6e 74 20 24 6c 6f 63 61  isterMount $loca
0300: 6c 20 5b 6c 69 73 74 20 3a 3a 76 66 73 3a 3a 7a  l [list ::vfs::z
0310: 69 70 3a 3a 55 6e 6d 6f 75 6e 74 20 24 66 64 5d  ip::Unmount $fd]
0320: 0a 20 20 20 20 72 65 74 75 72 6e 20 24 66 64 0a  .    return $fd.
0330: 7d 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a 69 70  }..proc vfs::zip
0340: 3a 3a 55 6e 6d 6f 75 6e 74 20 7b 66 64 20 6c 6f  ::Unmount {fd lo
0350: 63 61 6c 7d 20 7b 0a 20 20 20 20 76 66 73 3a 3a  cal} {.    vfs::
0360: 66 69 6c 65 73 79 73 74 65 6d 20 75 6e 6d 6f 75  filesystem unmou
0370: 6e 74 20 24 6c 6f 63 61 6c 0a 20 20 20 20 3a 3a  nt $local.    ::
0380: 7a 69 70 3a 3a 5f 63 6c 6f 73 65 20 24 66 64 0a  zip::_close $fd.
0390: 7d 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a 69 70  }..proc vfs::zip
03a0: 3a 3a 68 61 6e 64 6c 65 72 20 7b 7a 69 70 66 64  ::handler {zipfd
03b0: 20 63 6d 64 20 72 6f 6f 74 20 72 65 6c 61 74 69   cmd root relati
03c0: 76 65 20 61 63 74 75 61 6c 70 61 74 68 20 61 72  ve actualpath ar
03d0: 67 73 7d 20 7b 0a 20 20 20 20 23 3a 3a 76 66 73  gs} {.    #::vfs
03e0: 3a 3a 6c 6f 67 20 5b 6c 69 73 74 20 24 7a 69 70  ::log [list $zip
03f0: 66 64 20 24 63 6d 64 20 24 72 6f 6f 74 20 24 72  fd $cmd $root $r
0400: 65 6c 61 74 69 76 65 20 24 61 63 74 75 61 6c 70  elative $actualp
0410: 61 74 68 20 24 61 72 67 73 5d 0a 20 20 20 20 69  ath $args].    i
0420: 66 20 7b 24 63 6d 64 20 3d 3d 20 22 6d 61 74 63  f {$cmd == "matc
0430: 68 69 6e 64 69 72 65 63 74 6f 72 79 22 7d 20 7b  hindirectory"} {
0440: 0a 09 65 76 61 6c 20 5b 6c 69 73 74 20 24 63 6d  ..eval [list $cm
0450: 64 20 24 7a 69 70 66 64 20 24 72 65 6c 61 74 69  d $zipfd $relati
0460: 76 65 20 24 61 63 74 75 61 6c 70 61 74 68 5d 20  ve $actualpath] 
0470: 24 61 72 67 73 0a 20 20 20 20 7d 20 65 6c 73 65  $args.    } else
0480: 20 7b 0a 09 65 76 61 6c 20 5b 6c 69 73 74 20 24   {..eval [list $
0490: 63 6d 64 20 24 7a 69 70 66 64 20 24 72 65 6c 61  cmd $zipfd $rela
04a0: 74 69 76 65 5d 20 24 61 72 67 73 0a 20 20 20 20  tive] $args.    
04b0: 7d 0a 7d 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a  }.}..proc vfs::z
04c0: 69 70 3a 3a 61 74 74 72 69 62 75 74 65 73 20 7b  ip::attributes {
04d0: 7a 69 70 66 64 7d 20 7b 20 72 65 74 75 72 6e 20  zipfd} { return 
04e0: 5b 6c 69 73 74 20 22 73 74 61 74 65 22 5d 20 7d  [list "state"] }
04f0: 0a 70 72 6f 63 20 76 66 73 3a 3a 7a 69 70 3a 3a  .proc vfs::zip::
0500: 73 74 61 74 65 20 7b 7a 69 70 66 64 20 61 72 67  state {zipfd arg
0510: 73 7d 20 7b 0a 20 20 20 20 76 66 73 3a 3a 61 74  s} {.    vfs::at
0520: 74 72 69 62 75 74 65 43 61 6e 74 43 6f 6e 66 69  tributeCantConfi
0530: 67 75 72 65 20 22 73 74 61 74 65 22 20 22 72 65  gure "state" "re
0540: 61 64 6f 6e 6c 79 22 20 24 61 72 67 73 0a 7d 0a  adonly" $args.}.
0550: 0a 23 20 49 66 20 77 65 20 69 6d 70 6c 65 6d 65  .# If we impleme
0560: 6e 74 20 74 68 65 20 63 6f 6d 6d 61 6e 64 73 20  nt the commands 
0570: 62 65 6c 6f 77 2c 20 77 65 20 77 69 6c 6c 20 68  below, we will h
0580: 61 76 65 20 61 20 70 65 72 66 65 63 74 0a 23 20  ave a perfect.# 
0590: 76 69 72 74 75 61 6c 20 66 69 6c 65 20 73 79 73  virtual file sys
05a0: 74 65 6d 20 66 6f 72 20 7a 69 70 20 66 69 6c 65  tem for zip file
05b0: 73 2e 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a 69  s...proc vfs::zi
05c0: 70 3a 3a 6d 61 74 63 68 69 6e 64 69 72 65 63 74  p::matchindirect
05d0: 6f 72 79 20 7b 7a 69 70 66 64 20 70 61 74 68 20  ory {zipfd path 
05e0: 61 63 74 75 61 6c 70 61 74 68 20 70 61 74 74 65  actualpath patte
05f0: 72 6e 20 74 79 70 65 7d 20 7b 0a 20 20 20 20 23  rn type} {.    #
0600: 3a 3a 76 66 73 3a 3a 6c 6f 67 20 5b 6c 69 73 74  ::vfs::log [list
0610: 20 6d 61 74 63 68 69 6e 64 69 72 65 63 74 6f 72   matchindirector
0620: 79 20 24 70 61 74 68 20 24 61 63 74 75 61 6c 70  y $path $actualp
0630: 61 74 68 20 24 70 61 74 74 65 72 6e 20 24 74 79  ath $pattern $ty
0640: 70 65 5d 0a 0a 20 20 20 20 23 20 54 68 69 73 20  pe]..    # This 
0650: 63 61 6c 6c 20 74 6f 20 7a 69 70 3a 3a 67 65 74  call to zip::get
0660: 64 69 72 20 68 61 6e 64 6c 65 73 20 65 6d 70 74  dir handles empt
0670: 79 20 70 61 74 74 65 72 6e 73 20 70 72 6f 70 65  y patterns prope
0680: 72 6c 79 20 61 73 20 61 73 6b 69 6e 67 0a 20 20  rly as asking.  
0690: 20 20 23 20 66 6f 72 20 74 68 65 20 65 78 69 73    # for the exis
06a0: 74 65 6e 63 65 20 6f 66 20 61 20 73 69 6e 67 6c  tence of a singl
06b0: 65 20 66 69 6c 65 20 24 70 61 74 68 20 6f 6e 6c  e file $path onl
06c0: 79 0a 20 20 20 20 73 65 74 20 72 65 73 20 5b 3a  y.    set res [:
06d0: 3a 7a 69 70 3a 3a 67 65 74 64 69 72 20 24 7a 69  :zip::getdir $zi
06e0: 70 66 64 20 24 70 61 74 68 20 24 70 61 74 74 65  pfd $path $patte
06f0: 72 6e 5d 0a 20 20 20 20 23 3a 3a 76 66 73 3a 3a  rn].    #::vfs::
0700: 6c 6f 67 20 22 67 6f 74 20 24 72 65 73 22 0a 20  log "got $res". 
0710: 20 20 20 69 66 20 7b 21 5b 73 74 72 69 6e 67 20     if {![string 
0720: 6c 65 6e 67 74 68 20 24 70 61 74 74 65 72 6e 5d  length $pattern]
0730: 7d 20 7b 0a 09 69 66 20 7b 21 5b 3a 3a 7a 69 70  } {..if {![::zip
0740: 3a 3a 65 78 69 73 74 73 20 24 7a 69 70 66 64 20  ::exists $zipfd 
0750: 24 70 61 74 68 5d 7d 20 7b 20 72 65 74 75 72 6e  $path]} { return
0760: 20 7b 7d 20 7d 0a 09 73 65 74 20 72 65 73 20 5b   {} }..set res [
0770: 6c 69 73 74 20 24 61 63 74 75 61 6c 70 61 74 68  list $actualpath
0780: 5d 0a 09 73 65 74 20 61 63 74 75 61 6c 70 61 74  ]..set actualpat
0790: 68 20 22 22 0a 20 20 20 20 7d 0a 0a 20 20 20 20  h "".    }..    
07a0: 73 65 74 20 6e 65 77 72 65 73 20 5b 6c 69 73 74  set newres [list
07b0: 5d 0a 20 20 20 20 66 6f 72 65 61 63 68 20 70 20  ].    foreach p 
07c0: 5b 3a 3a 76 66 73 3a 3a 6d 61 74 63 68 43 6f 72  [::vfs::matchCor
07d0: 72 65 63 74 54 79 70 65 73 20 24 74 79 70 65 20  rectTypes $type 
07e0: 24 72 65 73 20 24 61 63 74 75 61 6c 70 61 74 68  $res $actualpath
07f0: 5d 20 7b 0a 09 6c 61 70 70 65 6e 64 20 6e 65 77  ] {..lappend new
0800: 72 65 73 20 5b 66 69 6c 65 20 6a 6f 69 6e 20 24  res [file join $
0810: 61 63 74 75 61 6c 70 61 74 68 20 24 70 5d 0a 20  actualpath $p]. 
0820: 20 20 20 7d 0a 20 20 20 20 23 3a 3a 76 66 73 3a     }.    #::vfs:
0830: 3a 6c 6f 67 20 22 67 6f 74 20 24 6e 65 77 72 65  :log "got $newre
0840: 73 22 0a 20 20 20 20 72 65 74 75 72 6e 20 24 6e  s".    return $n
0850: 65 77 72 65 73 0a 7d 0a 0a 70 72 6f 63 20 76 66  ewres.}..proc vf
0860: 73 3a 3a 7a 69 70 3a 3a 73 74 61 74 20 7b 7a 69  s::zip::stat {zi
0870: 70 66 64 20 6e 61 6d 65 7d 20 7b 0a 20 20 20 20  pfd name} {.    
0880: 23 3a 3a 76 66 73 3a 3a 6c 6f 67 20 22 73 74 61  #::vfs::log "sta
0890: 74 20 24 6e 61 6d 65 22 0a 20 20 20 20 3a 3a 7a  t $name".    ::z
08a0: 69 70 3a 3a 73 74 61 74 20 24 7a 69 70 66 64 20  ip::stat $zipfd 
08b0: 24 6e 61 6d 65 20 73 62 0a 20 20 20 20 23 3a 3a  $name sb.    #::
08c0: 76 66 73 3a 3a 6c 6f 67 20 5b 61 72 72 61 79 20  vfs::log [array 
08d0: 67 65 74 20 73 62 5d 0a 20 20 20 20 61 72 72 61  get sb].    arra
08e0: 79 20 67 65 74 20 73 62 0a 7d 0a 0a 70 72 6f 63  y get sb.}..proc
08f0: 20 76 66 73 3a 3a 7a 69 70 3a 3a 61 63 63 65 73   vfs::zip::acces
0900: 73 20 7b 7a 69 70 66 64 20 6e 61 6d 65 20 6d 6f  s {zipfd name mo
0910: 64 65 7d 20 7b 0a 20 20 20 20 23 3a 3a 76 66 73  de} {.    #::vfs
0920: 3a 3a 6c 6f 67 20 22 7a 69 70 2d 61 63 63 65 73  ::log "zip-acces
0930: 73 20 24 6e 61 6d 65 20 24 6d 6f 64 65 22 0a 20  s $name $mode". 
0940: 20 20 20 69 66 20 7b 24 6d 6f 64 65 20 26 20 32     if {$mode & 2
0950: 7d 20 7b 0a 09 76 66 73 3a 3a 66 69 6c 65 73 79  } {..vfs::filesy
0960: 73 74 65 6d 20 70 6f 73 69 78 65 72 72 6f 72 20  stem posixerror 
0970: 24 3a 3a 76 66 73 3a 3a 70 6f 73 69 78 28 45 52  $::vfs::posix(ER
0980: 4f 46 53 29 0a 20 20 20 20 7d 0a 20 20 20 20 23  OFS).    }.    #
0990: 20 52 65 61 64 61 62 6c 65 2c 20 45 78 69 73 74   Readable, Exist
09a0: 73 20 61 6e 64 20 45 78 65 63 75 74 61 62 6c 65  s and Executable
09b0: 20 61 72 65 20 74 72 65 61 74 65 64 20 61 73 20   are treated as 
09c0: 27 65 78 69 73 74 73 27 0a 20 20 20 20 23 20 43  'exists'.    # C
09d0: 6f 75 6c 64 20 77 65 20 67 65 74 20 6d 6f 72 65  ould we get more
09e0: 20 69 6e 66 6f 72 6d 61 74 69 6f 6e 20 66 72 6f   information fro
09f0: 6d 20 74 68 65 20 61 72 63 68 69 76 65 3f 0a 20  m the archive?. 
0a00: 20 20 20 69 66 20 7b 5b 3a 3a 7a 69 70 3a 3a 65     if {[::zip::e
0a10: 78 69 73 74 73 20 24 7a 69 70 66 64 20 24 6e 61  xists $zipfd $na
0a20: 6d 65 5d 7d 20 7b 0a 09 72 65 74 75 72 6e 20 31  me]} {..return 1
0a30: 0a 20 20 20 20 7d 20 65 6c 73 65 20 7b 0a 09 65  .    } else {..e
0a40: 72 72 6f 72 20 22 4e 6f 20 73 75 63 68 20 66 69  rror "No such fi
0a50: 6c 65 22 0a 20 20 20 20 7d 0a 20 20 20 20 0a 7d  le".    }.    .}
0a60: 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a 69 70 3a  ..proc vfs::zip:
0a70: 3a 6f 70 65 6e 20 7b 7a 69 70 66 64 20 6e 61 6d  :open {zipfd nam
0a80: 65 20 6d 6f 64 65 20 70 65 72 6d 69 73 73 69 6f  e mode permissio
0a90: 6e 73 7d 20 7b 0a 20 20 20 20 23 3a 3a 76 66 73  ns} {.    #::vfs
0aa0: 3a 3a 6c 6f 67 20 22 6f 70 65 6e 20 24 6e 61 6d  ::log "open $nam
0ab0: 65 20 24 6d 6f 64 65 20 24 70 65 72 6d 69 73 73  e $mode $permiss
0ac0: 69 6f 6e 73 22 0a 20 20 20 20 23 20 72 65 74 75  ions".    # retu
0ad0: 72 6e 20 61 20 6c 69 73 74 20 6f 66 20 74 77 6f  rn a list of two
0ae0: 20 65 6c 65 6d 65 6e 74 73 3a 0a 20 20 20 20 23   elements:.    #
0af0: 20 31 2e 20 66 69 72 73 74 20 65 6c 65 6d 65 6e   1. first elemen
0b00: 74 20 69 73 20 74 68 65 20 54 63 6c 20 63 68 61  t is the Tcl cha
0b10: 6e 6e 65 6c 20 6e 61 6d 65 20 77 68 69 63 68 20  nnel name which 
0b20: 68 61 73 20 62 65 65 6e 20 6f 70 65 6e 65 64 0a  has been opened.
0b30: 20 20 20 20 23 20 32 2e 20 73 65 63 6f 6e 64 20      # 2. second 
0b40: 65 6c 65 6d 65 6e 74 20 28 6f 70 74 69 6f 6e 61  element (optiona
0b50: 6c 29 20 69 73 20 61 20 63 6f 6d 6d 61 6e 64 20  l) is a command 
0b60: 74 6f 20 65 76 61 6c 75 61 74 65 20 77 68 65 6e  to evaluate when
0b70: 0a 20 20 20 20 23 20 20 20 20 74 68 65 20 63 68  .    #    the ch
0b80: 61 6e 6e 65 6c 20 69 73 20 63 6c 6f 73 65 64 2e  annel is closed.
0b90: 0a 0a 20 20 20 20 73 77 69 74 63 68 20 2d 2d 20  ..    switch -- 
0ba0: 24 6d 6f 64 65 20 7b 0a 09 22 22 20 2d 0a 09 22  $mode {.."" -.."
0bb0: 72 22 20 7b 0a 09 20 20 20 20 69 66 20 7b 21 5b  r" {..    if {![
0bc0: 3a 3a 7a 69 70 3a 3a 65 78 69 73 74 73 20 24 7a  ::zip::exists $z
0bd0: 69 70 66 64 20 24 6e 61 6d 65 5d 7d 20 7b 0a 09  ipfd $name]} {..
0be0: 09 76 66 73 3a 3a 66 69 6c 65 73 79 73 74 65 6d  .vfs::filesystem
0bf0: 20 70 6f 73 69 78 65 72 72 6f 72 20 24 3a 3a 76   posixerror $::v
0c00: 66 73 3a 3a 70 6f 73 69 78 28 45 4e 4f 45 4e 54  fs::posix(ENOENT
0c10: 29 0a 09 20 20 20 20 7d 0a 09 20 20 20 20 0a 09  )..    }..    ..
0c20: 20 20 20 20 3a 3a 7a 69 70 3a 3a 73 74 61 74 20      ::zip::stat 
0c30: 24 7a 69 70 66 64 20 24 6e 61 6d 65 20 73 62 0a  $zipfd $name sb.
0c40: 0a 20 20 20 20 20 20 20 20 20 20 20 20 69 66 20  .            if 
0c50: 7b 24 73 62 28 69 6e 6f 29 20 3d 3d 20 2d 31 7d  {$sb(ino) == -1}
0c60: 20 7b 0a 20 20 20 20 20 20 20 20 20 20 20 20 20   {.             
0c70: 20 20 20 76 66 73 3a 3a 66 69 6c 65 73 79 73 74     vfs::filesyst
0c80: 65 6d 20 70 6f 73 69 78 65 72 72 6f 72 20 24 3a  em posixerror $:
0c90: 3a 76 66 73 3a 3a 70 6f 73 69 78 28 45 49 53 44  :vfs::posix(EISD
0ca0: 49 52 29 0a 20 20 20 20 20 20 20 20 20 20 20 20  IR).            
0cb0: 7d 0a 0a 09 20 20 20 20 73 65 74 20 6e 66 64 20  }...    set nfd 
0cc0: 5b 76 66 73 3a 3a 6d 65 6d 63 68 61 6e 5d 0a 09  [vfs::memchan]..
0cd0: 20 20 20 20 66 63 6f 6e 66 69 67 75 72 65 20 24      fconfigure $
0ce0: 6e 66 64 20 2d 74 72 61 6e 73 6c 61 74 69 6f 6e  nfd -translation
0cf0: 20 62 69 6e 61 72 79 0a 0a 09 20 20 20 20 73 65   binary...    se
0d00: 65 6b 20 24 7a 69 70 66 64 20 24 73 62 28 69 6e  ek $zipfd $sb(in
0d10: 6f 29 20 73 74 61 72 74 0a 09 20 20 20 20 7a 69  o) start..    zi
0d20: 70 3a 3a 44 61 74 61 20 24 7a 69 70 66 64 20 73  p::Data $zipfd s
0d30: 62 20 64 61 74 61 0a 0a 09 20 20 20 20 70 75 74  b data...    put
0d40: 73 20 2d 6e 6f 6e 65 77 6c 69 6e 65 20 24 6e 66  s -nonewline $nf
0d50: 64 20 24 64 61 74 61 0a 0a 09 20 20 20 20 66 63  d $data...    fc
0d60: 6f 6e 66 69 67 75 72 65 20 24 6e 66 64 20 2d 74  onfigure $nfd -t
0d70: 72 61 6e 73 6c 61 74 69 6f 6e 20 61 75 74 6f 0a  ranslation auto.
0d80: 09 20 20 20 20 73 65 65 6b 20 24 6e 66 64 20 30  .    seek $nfd 0
0d90: 0a 09 20 20 20 20 72 65 74 75 72 6e 20 5b 6c 69  ..    return [li
0da0: 73 74 20 24 6e 66 64 5d 0a 09 7d 0a 09 64 65 66  st $nfd]..}..def
0db0: 61 75 6c 74 20 7b 0a 09 20 20 20 20 76 66 73 3a  ault {..    vfs:
0dc0: 3a 66 69 6c 65 73 79 73 74 65 6d 20 70 6f 73 69  :filesystem posi
0dd0: 78 65 72 72 6f 72 20 24 3a 3a 76 66 73 3a 3a 70  xerror $::vfs::p
0de0: 6f 73 69 78 28 45 52 4f 46 53 29 0a 09 7d 0a 20  osix(EROFS)..}. 
0df0: 20 20 20 7d 0a 7d 0a 0a 70 72 6f 63 20 76 66 73     }.}..proc vfs
0e00: 3a 3a 7a 69 70 3a 3a 63 72 65 61 74 65 64 69 72  ::zip::createdir
0e10: 65 63 74 6f 72 79 20 7b 7a 69 70 66 64 20 6e 61  ectory {zipfd na
0e20: 6d 65 7d 20 7b 0a 20 20 20 20 23 3a 3a 76 66 73  me} {.    #::vfs
0e30: 3a 3a 6c 6f 67 20 22 63 72 65 61 74 65 64 69 72  ::log "createdir
0e40: 65 63 74 6f 72 79 20 24 6e 61 6d 65 22 0a 20 20  ectory $name".  
0e50: 20 20 76 66 73 3a 3a 66 69 6c 65 73 79 73 74 65    vfs::filesyste
0e60: 6d 20 70 6f 73 69 78 65 72 72 6f 72 20 24 3a 3a  m posixerror $::
0e70: 76 66 73 3a 3a 70 6f 73 69 78 28 45 52 4f 46 53  vfs::posix(EROFS
0e80: 29 0a 7d 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a  ).}..proc vfs::z
0e90: 69 70 3a 3a 72 65 6d 6f 76 65 64 69 72 65 63 74  ip::removedirect
0ea0: 6f 72 79 20 7b 7a 69 70 66 64 20 6e 61 6d 65 20  ory {zipfd name 
0eb0: 72 65 63 75 72 73 69 76 65 7d 20 7b 0a 20 20 20  recursive} {.   
0ec0: 20 23 3a 3a 76 66 73 3a 3a 6c 6f 67 20 22 72 65   #::vfs::log "re
0ed0: 6d 6f 76 65 64 69 72 65 63 74 6f 72 79 20 24 6e  movedirectory $n
0ee0: 61 6d 65 22 0a 20 20 20 20 76 66 73 3a 3a 66 69  ame".    vfs::fi
0ef0: 6c 65 73 79 73 74 65 6d 20 70 6f 73 69 78 65 72  lesystem posixer
0f00: 72 6f 72 20 24 3a 3a 76 66 73 3a 3a 70 6f 73 69  ror $::vfs::posi
0f10: 78 28 45 52 4f 46 53 29 0a 7d 0a 0a 70 72 6f 63  x(EROFS).}..proc
0f20: 20 76 66 73 3a 3a 7a 69 70 3a 3a 64 65 6c 65 74   vfs::zip::delet
0f30: 65 66 69 6c 65 20 7b 7a 69 70 66 64 20 6e 61 6d  efile {zipfd nam
0f40: 65 7d 20 7b 0a 20 20 20 20 23 3a 3a 76 66 73 3a  e} {.    #::vfs:
0f50: 3a 6c 6f 67 20 22 64 65 6c 65 74 65 66 69 6c 65  :log "deletefile
0f60: 20 24 6e 61 6d 65 22 0a 20 20 20 20 76 66 73 3a   $name".    vfs:
0f70: 3a 66 69 6c 65 73 79 73 74 65 6d 20 70 6f 73 69  :filesystem posi
0f80: 78 65 72 72 6f 72 20 24 3a 3a 76 66 73 3a 3a 70  xerror $::vfs::p
0f90: 6f 73 69 78 28 45 52 4f 46 53 29 0a 7d 0a 0a 70  osix(EROFS).}..p
0fa0: 72 6f 63 20 76 66 73 3a 3a 7a 69 70 3a 3a 66 69  roc vfs::zip::fi
0fb0: 6c 65 61 74 74 72 69 62 75 74 65 73 20 7b 7a 69  leattributes {zi
0fc0: 70 66 64 20 6e 61 6d 65 20 61 72 67 73 7d 20 7b  pfd name args} {
0fd0: 0a 20 20 20 20 23 3a 3a 76 66 73 3a 3a 6c 6f 67  .    #::vfs::log
0fe0: 20 22 66 69 6c 65 61 74 74 72 69 62 75 74 65 73   "fileattributes
0ff0: 20 24 61 72 67 73 22 0a 20 20 20 20 73 77 69 74   $args".    swit
1000: 63 68 20 2d 2d 20 5b 6c 6c 65 6e 67 74 68 20 24  ch -- [llength $
1010: 61 72 67 73 5d 20 7b 0a 09 30 20 7b 0a 09 20 20  args] {..0 {..  
1020: 20 20 23 20 6c 69 73 74 20 73 74 72 69 6e 67 73    # list strings
1030: 0a 09 20 20 20 20 72 65 74 75 72 6e 20 5b 6c 69  ..    return [li
1040: 73 74 5d 0a 09 7d 0a 09 31 20 7b 0a 09 20 20 20  st]..}..1 {..   
1050: 20 23 20 67 65 74 20 76 61 6c 75 65 0a 09 20 20   # get value..  
1060: 20 20 73 65 74 20 69 6e 64 65 78 20 5b 6c 69 6e    set index [lin
1070: 64 65 78 20 24 61 72 67 73 20 30 5d 0a 09 20 20  dex $args 0]..  
1080: 20 20 72 65 74 75 72 6e 20 22 22 0a 09 7d 0a 09    return ""..}..
1090: 32 20 7b 0a 09 20 20 20 20 23 20 73 65 74 20 76  2 {..    # set v
10a0: 61 6c 75 65 0a 09 20 20 20 20 73 65 74 20 69 6e  alue..    set in
10b0: 64 65 78 20 5b 6c 69 6e 64 65 78 20 24 61 72 67  dex [lindex $arg
10c0: 73 20 30 5d 0a 09 20 20 20 20 73 65 74 20 76 61  s 0]..    set va
10d0: 6c 20 5b 6c 69 6e 64 65 78 20 24 61 72 67 73 20  l [lindex $args 
10e0: 31 5d 0a 09 20 20 20 20 76 66 73 3a 3a 66 69 6c  1]..    vfs::fil
10f0: 65 73 79 73 74 65 6d 20 70 6f 73 69 78 65 72 72  esystem posixerr
1100: 6f 72 20 24 3a 3a 76 66 73 3a 3a 70 6f 73 69 78  or $::vfs::posix
1110: 28 45 52 4f 46 53 29 0a 09 7d 0a 20 20 20 20 7d  (EROFS)..}.    }
1120: 0a 7d 0a 0a 70 72 6f 63 20 76 66 73 3a 3a 7a 69  .}..proc vfs::zi
1130: 70 3a 3a 75 74 69 6d 65 20 7b 66 64 20 70 61 74  p::utime {fd pat
1140: 68 20 61 63 74 69 6d 65 20 6d 74 69 6d 65 7d 20  h actime mtime} 
1150: 7b 0a 20 20 20 20 76 66 73 3a 3a 66 69 6c 65 73  {.    vfs::files
1160: 79 73 74 65 6d 20 70 6f 73 69 78 65 72 72 6f 72  ystem posixerror
1170: 20 24 3a 3a 76 66 73 3a 3a 70 6f 73 69 78 28 45   $::vfs::posix(E
1180: 52 4f 46 53 29 0a 7d 0a 0a 23 20 42 65 6c 6f 77  ROFS).}..# Below
1190: 20 63 6f 70 69 65 64 20 66 72 6f 6d 20 54 63 6c   copied from Tcl
11a0: 4b 69 74 20 64 69 73 74 72 69 62 75 74 69 6f 6e  Kit distribution
11b0: 0a 0a 23 0a 23 20 5a 49 50 20 64 65 63 6f 64 65  ..#.# ZIP decode
11c0: 72 3a 0a 23 0a 23 20 46 6f 72 6d 61 74 20 6f 66  r:.#.# Format of
11d0: 20 7a 69 70 20 66 69 6c 65 3a 0a 23 20 5b 20 44   zip file:.# [ D
11e0: 61 74 61 20 5d 2a 20 5b 20 54 4f 43 20 5d 2a 20  ata ]* [ TOC ]* 
11f0: 45 6e 64 4f 66 41 72 63 68 69 76 65 0a 23 0a 23  EndOfArchive.#.#
1200: 20 4e 6f 74 65 3a 20 54 4f 43 20 69 73 20 72 65   Note: TOC is re
1210: 66 65 72 65 64 20 74 6f 20 69 6e 20 5a 49 50 20  fered to in ZIP 
1220: 64 6f 63 20 61 73 20 22 43 65 6e 74 72 61 6c 20  doc as "Central 
1230: 41 72 63 68 69 76 65 22 0a 23 0a 23 20 54 68 69  Archive".#.# Thi
1240: 73 20 6d 65 61 6e 73 20 74 68 65 72 65 20 61 72  s means there ar
1250: 65 20 74 77 6f 20 77 61 79 73 20 6f 66 20 61 63  e two ways of ac
1260: 63 65 73 73 69 6e 67 3a 0a 23 0a 23 20 31 29 20  cessing:.#.# 1) 
1270: 66 72 6f 6d 20 74 68 65 20 62 65 67 69 6e 69 6e  from the beginin
1280: 67 20 61 73 20 61 20 73 74 72 65 61 6d 20 2d 20  g as a stream - 
1290: 75 6e 74 69 6c 20 74 68 65 20 68 65 61 64 65 72  until the header
12a0: 0a 23 09 69 73 20 6e 6f 74 20 22 50 4b 5c 30 33  .#.is not "PK\03
12b0: 5c 30 34 22 20 2d 20 69 64 65 61 6c 20 66 6f 72  \04" - ideal for
12c0: 20 75 6e 7a 69 70 70 69 6e 67 2e 0a 23 0a 23 20   unzipping..#.# 
12d0: 32 29 20 66 6f 72 20 74 61 62 6c 65 20 6f 66 20  2) for table of 
12e0: 63 6f 6e 74 65 6e 74 73 20 77 69 74 68 6f 75 74  contents without
12f0: 20 72 65 61 64 69 6e 67 20 65 6e 74 69 72 65 0a   reading entire.
1300: 23 09 61 72 63 68 69 76 65 20 62 79 20 66 69 72  #.archive by fir
1310: 73 74 20 66 65 74 63 68 69 6e 67 20 45 6e 64 4f  st fetching EndO
1320: 66 41 72 63 68 69 76 65 2c 20 74 68 65 6e 0a 23  fArchive, then.#
1330: 09 6a 75 73 74 20 6c 6f 61 64 69 6e 67 20 74 68  .just loading th
1340: 65 20 54 4f 43 0a 23 0a 0a 6e 61 6d 65 73 70 61  e TOC.#..namespa
1350: 63 65 20 65 76 61 6c 20 7a 69 70 20 7b 0a 20 20  ce eval zip {.  
1360: 20 20 61 72 72 61 79 20 73 65 74 20 6d 65 74 68    array set meth
1370: 6f 64 73 20 7b 0a 09 30 09 7b 73 74 6f 72 65 64  ods {..0.{stored
1380: 20 2d 20 54 68 65 20 66 69 6c 65 20 69 73 20 73   - The file is s
1390: 74 6f 72 65 64 20 28 6e 6f 20 63 6f 6d 70 72 65  tored (no compre
13a0: 73 73 69 6f 6e 29 7d 0a 09 31 09 7b 73 68 72 75  ssion)}..1.{shru
13b0: 6e 6b 20 2d 20 54 68 65 20 66 69 6c 65 20 69 73  nk - The file is
13c0: 20 53 68 72 75 6e 6b 7d 0a 09 32 09 7b 72 65 64   Shrunk}..2.{red
13d0: 75 63 65 31 20 2d 20 54 68 65 20 66 69 6c 65 20  uce1 - The file 
13e0: 69 73 20 52 65 64 75 63 65 64 20 77 69 74 68 20  is Reduced with 
13f0: 63 6f 6d 70 72 65 73 73 69 6f 6e 20 66 61 63 74  compression fact
1400: 6f 72 20 31 7d 0a 09 33 09 7b 72 65 64 75 63 65  or 1}..3.{reduce
1410: 32 20 2d 20 54 68 65 20 66 69 6c 65 20 69 73 20  2 - The file is 
1420: 52 65 64 75 63 65 64 20 77 69 74 68 20 63 6f 6d  Reduced with com
1430: 70 72 65 73 73 69 6f 6e 20 66 61 63 74 6f 72 20  pression factor 
1440: 32 7d 0a 09 34 09 7b 72 65 64 75 63 65 33 20 2d  2}..4.{reduce3 -
1450: 20 54 68 65 20 66 69 6c 65 20 69 73 20 52 65 64   The file is Red
1460: 75 63 65 64 20 77 69 74 68 20 63 6f 6d 70 72 65  uced with compre
1470: 73 73 69 6f 6e 20 66 61 63 74 6f 72 20 33 7d 0a  ssion factor 3}.
1480: 09 35 09 7b 72 65 64 75 63 65 34 20 2d 20 54 68  .5.{reduce4 - Th
1490: 65 20 66 69 6c 65 20 69 73 20 52 65 64 75 63 65  e file is Reduce
14a0: 64 20 77 69 74 68 20 63 6f 6d 70 72 65 73 73 69  d with compressi
14b0: 6f 6e 20 66 61 63 74 6f 72 20 34 7d 0a 09 36 09  on factor 4}..6.
14c0: 7b 69 6d 70 6c 6f 64 65 20 2d 20 54 68 65 20 66  {implode - The f
14d0: 69 6c 65 20 69 73 20 49 6d 70 6c 6f 64 65 64 7d  ile is Imploded}
14e0: 0a 09 37 09 7b 72 65 73 65 72 76 65 64 20 2d 20  ..7.{reserved - 
14f0: 52 65 73 65 72 76 65 64 20 66 6f 72 20 54 6f 6b  Reserved for Tok
1500: 65 6e 69 7a 69 6e 67 20 63 6f 6d 70 72 65 73 73  enizing compress
1510: 69 6f 6e 20 61 6c 67 6f 72 69 74 68 6d 7d 0a 09  ion algorithm}..
1520: 38 09 7b 64 65 66 6c 61 74 65 20 2d 20 54 68 65  8.{deflate - The
1530: 20 66 69 6c 65 20 69 73 20 44 65 66 6c 61 74 65   file is Deflate
1540: 64 7d 0a 09 39 09 7b 72 65 73 65 72 76 65 64 20  d}..9.{reserved 
1550: 2d 20 52 65 73 65 72 76 65 64 20 66 6f 72 20 65  - Reserved for e
1560: 6e 68 61 6e 63 65 64 20 44 65 66 6c 61 74 69 6e  nhanced Deflatin
1570: 67 7d 0a 09 31 30 09 7b 70 6b 69 6d 70 6c 6f 64  g}..10.{pkimplod
1580: 65 20 2d 20 50 4b 57 41 52 45 20 44 61 74 65 20  e - PKWARE Date 
1590: 43 6f 6d 70 72 65 73 73 69 6f 6e 20 4c 69 62 72  Compression Libr
15a0: 61 72 79 20 49 6d 70 6c 6f 64 69 6e 67 7d 0a 20  ary Imploding}. 
15b0: 20 20 20 7d 0a 20 20 20 20 23 20 56 65 72 73 69     }.    # Versi
15c0: 6f 6e 20 74 79 70 65 73 20 28 68 69 67 68 2d 6f  on types (high-o
15d0: 72 64 65 72 20 62 79 74 65 29 0a 20 20 20 20 61  rder byte).    a
15e0: 72 72 61 79 20 73 65 74 20 73 79 73 74 65 6d 73  rray set systems
15f0: 20 7b 0a 09 30 09 7b 64 6f 73 7d 0a 09 31 09 7b   {..0.{dos}..1.{
1600: 61 6d 69 67 61 7d 0a 09 32 09 7b 76 6d 73 7d 0a  amiga}..2.{vms}.
1610: 09 33 09 7b 75 6e 69 78 7d 0a 09 34 09 7b 76 6d  .3.{unix}..4.{vm
1620: 20 63 6d 73 7d 0a 09 35 09 7b 61 74 61 72 69 7d   cms}..5.{atari}
1630: 0a 09 36 09 7b 6f 73 2f 32 7d 0a 09 37 09 7b 6d  ..6.{os/2}..7.{m
1640: 61 63 6f 73 7d 0a 09 38 09 7b 7a 20 73 79 73 74  acos}..8.{z syst
1650: 65 6d 20 38 7d 0a 09 39 09 7b 63 70 2f 6d 7d 0a  em 8}..9.{cp/m}.
1660: 09 31 30 09 7b 74 6f 70 73 32 30 7d 0a 09 31 31  .10.{tops20}..11
1670: 09 7b 77 69 6e 64 6f 77 73 7d 0a 09 31 32 09 7b  .{windows}..12.{
1680: 71 64 6f 73 7d 0a 09 31 33 09 7b 72 69 73 63 6f  qdos}..13.{risco
1690: 73 7d 0a 09 31 34 09 7b 76 66 61 74 7d 0a 09 31  s}..14.{vfat}..1
16a0: 35 09 7b 6d 76 73 7d 0a 09 31 36 09 7b 62 65 6f  5.{mvs}..16.{beo
16b0: 73 7d 0a 09 31 37 09 7b 74 61 6e 64 65 6d 7d 0a  s}..17.{tandem}.
16c0: 09 31 38 09 7b 74 68 65 6f 73 7d 0a 20 20 20 20  .18.{theos}.    
16d0: 7d 0a 20 20 20 20 23 20 44 4f 53 20 46 69 6c 65  }.    # DOS File
16e0: 20 41 74 74 72 73 0a 20 20 20 20 61 72 72 61 79   Attrs.    array
16f0: 20 73 65 74 20 64 6f 73 61 74 74 72 73 20 7b 0a   set dosattrs {.
1700: 09 31 09 7b 72 65 61 64 6f 6e 6c 79 7d 0a 09 32  .1.{readonly}..2
1710: 09 7b 68 69 64 64 65 6e 7d 0a 09 34 09 7b 73 79  .{hidden}..4.{sy
1720: 73 74 65 6d 7d 0a 09 38 09 7b 75 6e 6b 6e 6f 77  stem}..8.{unknow
1730: 6e 38 7d 0a 09 31 36 09 7b 64 69 72 65 63 74 6f  n8}..16.{directo
1740: 72 79 7d 0a 09 33 32 09 7b 61 72 63 68 69 76 65  ry}..32.{archive
1750: 7d 0a 09 36 34 09 7b 75 6e 6b 6e 6f 77 6e 36 34  }..64.{unknown64
1760: 7d 0a 09 31 32 38 09 7b 6e 6f 72 6d 61 6c 7d 0a  }..128.{normal}.
1770: 20 20 20 20 7d 0a 0a 20 20 20 20 70 72 6f 63 20      }..    proc 
1780: 75 5f 73 68 6f 72 74 20 7b 6e 7d 20 20 7b 20 72  u_short {n}  { r
1790: 65 74 75 72 6e 20 5b 65 78 70 72 20 7b 20 28 24  eturn [expr { ($
17a0: 6e 2b 30 78 31 30 30 30 30 29 25 30 78 31 30 30  n+0x10000)%0x100
17b0: 30 30 20 7d 5d 20 7d 0a 7d 0a 0a 70 72 6f 63 20  00 }] }.}..proc 
17c0: 7a 69 70 3a 3a 44 6f 73 54 69 6d 65 20 7b 64 61  zip::DosTime {da
17d0: 74 65 20 74 69 6d 65 7d 20 7b 0a 20 20 20 20 23  te time} {.    #
17e0: 20 54 68 65 20 70 72 65 2d 56 46 53 20 65 6e 76   The pre-VFS env
17f0: 69 72 6f 6e 6d 65 6e 74 20 77 69 6c 6c 20 6e 6f  ironment will no
1800: 74 20 68 61 76 65 20 61 63 63 65 73 73 20 74 6f  t have access to
1810: 20 22 63 6c 6f 63 6b 22 2c 20 73 6f 20 64 6f 6e   "clock", so don
1820: 27 74 20 65 76 65 6e 0a 20 20 20 20 23 20 62 6f  't even.    # bo
1830: 74 68 65 72 0a 20 20 20 20 72 65 74 75 72 6e 20  ther.    return 
1840: 30 0a 7d 0a 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a  0.}...proc zip::
1850: 44 61 74 61 20 7b 66 64 20 61 72 72 20 7b 76 61  Data {fd arr {va
1860: 72 50 74 72 20 22 22 7d 20 7b 76 65 72 69 66 79  rPtr ""} {verify
1870: 20 30 7d 7d 20 7b 0a 20 20 20 20 75 70 76 61 72   0}} {.    upvar
1880: 20 31 20 24 61 72 72 20 73 62 0a 0a 20 20 20 20   1 $arr sb..    
1890: 69 66 20 7b 20 24 76 61 72 50 74 72 20 21 3d 20  if { $varPtr != 
18a0: 22 22 20 7d 20 7b 0a 09 75 70 76 61 72 20 31 20  "" } {..upvar 1 
18b0: 24 76 61 72 50 74 72 20 64 61 74 61 0a 20 20 20  $varPtr data.   
18c0: 20 7d 0a 0a 20 20 20 20 73 65 74 20 62 75 66 20   }..    set buf 
18d0: 5b 72 65 61 64 20 24 66 64 20 33 30 5d 0a 20 20  [read $fd 30].  
18e0: 20 20 73 65 74 20 6e 20 5b 62 69 6e 61 72 79 20    set n [binary 
18f0: 73 63 61 6e 20 24 62 75 66 20 41 34 73 73 73 73  scan $buf A4ssss
1900: 73 69 69 69 73 73 20 5c 0a 09 09 68 64 72 20 73  siiiss \...hdr s
1910: 62 28 76 65 72 29 20 73 62 28 66 6c 61 67 73 29  b(ver) sb(flags)
1920: 20 73 62 28 6d 65 74 68 6f 64 29 20 5c 0a 09 09   sb(method) \...
1930: 74 69 6d 65 20 64 61 74 65 20 5c 0a 09 09 73 62  time date \...sb
1940: 28 63 72 63 29 20 73 62 28 63 73 69 7a 65 29 20  (crc) sb(csize) 
1950: 73 62 28 73 69 7a 65 29 20 66 6c 65 6e 20 65 6c  sb(size) flen el
1960: 65 6e 5d 0a 0a 20 20 20 20 69 66 20 7b 20 21 5b  en]..    if { ![
1970: 73 74 72 69 6e 67 20 65 71 75 61 6c 20 22 50 4b  string equal "PK
1980: 5c 30 33 5c 30 34 22 20 24 68 64 72 5d 20 7d 20  \03\04" $hdr] } 
1990: 7b 0a 09 62 69 6e 61 72 79 20 73 63 61 6e 20 24  {..binary scan $
19a0: 68 64 72 20 48 2a 20 78 0a 09 65 72 72 6f 72 20  hdr H* x..error 
19b0: 22 62 61 64 20 68 65 61 64 65 72 3a 20 24 78 22  "bad header: $x"
19c0: 0a 20 20 20 20 7d 0a 20 20 20 20 73 65 74 20 73  .    }.    set s
19d0: 62 28 76 65 72 29 09 09 5b 75 5f 73 68 6f 72 74  b(ver)..[u_short
19e0: 20 24 73 62 28 76 65 72 29 5d 0a 20 20 20 20 73   $sb(ver)].    s
19f0: 65 74 20 73 62 28 66 6c 61 67 73 29 09 5b 75 5f  et sb(flags).[u_
1a00: 73 68 6f 72 74 20 24 73 62 28 66 6c 61 67 73 29  short $sb(flags)
1a10: 5d 0a 20 20 20 20 73 65 74 20 73 62 28 6d 65 74  ].    set sb(met
1a20: 68 6f 64 29 09 5b 75 5f 73 68 6f 72 74 20 24 73  hod).[u_short $s
1a30: 62 28 6d 65 74 68 6f 64 29 5d 0a 20 20 20 20 73  b(method)].    s
1a40: 65 74 20 73 62 28 6d 74 69 6d 65 29 09 5b 44 6f  et sb(mtime).[Do
1a50: 73 54 69 6d 65 20 24 64 61 74 65 20 24 74 69 6d  sTime $date $tim
1a60: 65 5d 0a 0a 20 20 20 20 73 65 74 20 73 62 28 6e  e]..    set sb(n
1a70: 61 6d 65 29 20 5b 72 65 61 64 20 24 66 64 20 5b  ame) [read $fd [
1a80: 75 5f 73 68 6f 72 74 20 24 66 6c 65 6e 5d 5d 0a  u_short $flen]].
1a90: 20 20 20 20 73 65 74 20 73 62 28 65 78 74 72 61      set sb(extra
1aa0: 29 20 5b 72 65 61 64 20 24 66 64 20 5b 75 5f 73  ) [read $fd [u_s
1ab0: 68 6f 72 74 20 24 65 6c 65 6e 5d 5d 0a 0a 20 20  hort $elen]]..  
1ac0: 20 20 69 66 20 7b 20 24 76 61 72 50 74 72 20 3d    if { $varPtr =
1ad0: 3d 20 22 22 20 7d 20 7b 0a 09 73 65 65 6b 20 24  = "" } {..seek $
1ae0: 66 64 20 24 73 62 28 63 73 69 7a 65 29 20 63 75  fd $sb(csize) cu
1af0: 72 72 65 6e 74 0a 20 20 20 20 7d 20 65 6c 73 65  rrent.    } else
1b00: 20 7b 0a 09 23 20 41 64 64 65 64 20 62 79 20 43   {..# Added by C
1b10: 68 75 63 6b 20 46 65 72 72 69 6c 20 31 30 2d 32  huck Ferril 10-2
1b20: 36 2d 30 33 20 74 6f 20 66 69 78 20 72 65 61 64  6-03 to fix read
1b30: 69 6e 67 20 6f 66 20 4f 70 65 6e 4f 66 66 69 63  ing of OpenOffic
1b40: 65 0a 09 23 20 20 2e 73 78 77 20 66 69 6c 65 73  e..#  .sxw files
1b50: 2e 20 41 6e 79 20 66 69 6c 65 73 20 69 6e 20 74  . Any files in t
1b60: 68 65 20 7a 69 70 20 74 68 61 74 20 68 61 64 20  he zip that had 
1b70: 61 20 6d 65 74 68 6f 64 20 6f 66 20 38 0a 09 23  a method of 8..#
1b80: 20 20 28 64 65 66 6c 61 74 65 29 20 66 61 69 6c    (deflate) fail
1b90: 65 64 20 68 65 72 65 20 62 65 63 61 75 73 65 20  ed here because 
1ba0: 73 69 7a 65 20 61 6e 64 20 63 73 69 7a 65 20 77  size and csize w
1bb0: 65 72 65 20 7a 65 72 6f 2e 0a 09 23 20 20 49 27  ere zero...#  I'
1bc0: 6d 20 6e 6f 74 20 73 75 72 65 20 77 68 79 20 74  m not sure why t
1bd0: 68 65 20 61 62 6f 76 65 20 63 6f 6d 70 75 74 65  he above compute
1be0: 73 20 74 68 65 20 73 69 7a 65 20 61 6e 64 20 63  s the size and c
1bf0: 73 69 7a 65 0a 09 23 20 20 77 72 6f 6e 67 2c 20  size..#  wrong, 
1c00: 62 75 74 20 73 74 61 74 20 61 70 70 65 61 72 73  but stat appears
1c10: 20 77 6f 72 6b 73 20 70 72 6f 70 65 72 6c 79 2e   works properly.
1c20: 20 49 20 6f 72 69 67 69 6e 61 6c 6c 79 0a 09 23   I originally..#
1c30: 20 20 63 68 65 63 6b 65 64 20 66 6f 72 20 63 73    checked for cs
1c40: 69 7a 65 20 6f 66 20 7a 65 72 6f 2c 20 62 75 74  ize of zero, but
1c50: 20 61 64 64 69 6e 67 20 74 68 69 73 20 63 68 61   adding this cha
1c60: 6e 67 65 20 64 69 64 6e 27 74 0a 09 23 20 20 61  nge didn't..#  a
1c70: 70 70 65 61 72 20 74 6f 20 62 72 65 61 6b 20 74  ppear to break t
1c80: 68 65 20 6e 6f 6e 65 20 64 65 66 6c 61 74 65 64  he none deflated
1c90: 20 66 69 6c 65 20 61 63 63 65 73 73 20 61 6e 64   file access and
1ca0: 20 73 65 65 6d 65 64 0a 09 23 20 20 6d 6f 72 65   seemed..#  more
1cb0: 20 6e 61 74 75 72 61 6c 2e 0a 20 09 7a 69 70 3a   natural.. .zip:
1cc0: 3a 73 74 61 74 20 24 66 64 20 24 73 62 28 6e 61  :stat $fd $sb(na
1cd0: 6d 65 29 20 73 62 0a 0a 09 73 65 74 20 64 61 74  me) sb...set dat
1ce0: 61 20 5b 72 65 61 64 20 24 66 64 20 24 73 62 28  a [read $fd $sb(
1cf0: 63 73 69 7a 65 29 5d 0a 20 20 20 20 7d 0a 0a 20  csize)].    }.. 
1d00: 20 20 20 69 66 20 7b 20 24 73 62 28 66 6c 61 67     if { $sb(flag
1d10: 73 29 20 26 20 30 78 34 20 7d 20 7b 0a 09 23 20  s) & 0x4 } {..# 
1d20: 44 61 74 61 20 44 65 73 63 72 69 70 74 6f 72 20  Data Descriptor 
1d30: 75 73 65 64 0a 09 73 65 74 20 62 75 66 20 5b 72  used..set buf [r
1d40: 65 61 64 20 24 66 64 20 31 32 5d 0a 09 62 69 6e  ead $fd 12]..bin
1d50: 61 72 79 20 73 63 61 6e 20 24 62 75 66 20 69 69  ary scan $buf ii
1d60: 69 20 73 62 28 63 72 63 29 20 73 62 28 63 73 69  i sb(crc) sb(csi
1d70: 7a 65 29 20 73 62 28 73 69 7a 65 29 0a 20 20 20  ze) sb(size).   
1d80: 20 7d 0a 0a 0a 20 20 20 20 69 66 20 7b 20 24 76   }...    if { $v
1d90: 61 72 50 74 72 20 3d 3d 20 22 22 20 7d 20 7b 0a  arPtr == "" } {.
1da0: 09 72 65 74 75 72 6e 20 22 22 0a 20 20 20 20 7d  .return "".    }
1db0: 0a 0a 20 20 20 20 69 66 20 7b 20 24 73 62 28 6d  ..    if { $sb(m
1dc0: 65 74 68 6f 64 29 20 21 3d 20 30 20 7d 20 7b 0a  ethod) != 0 } {.
1dd0: 09 69 66 20 7b 20 5b 63 61 74 63 68 20 7b 0a 09  .if { [catch {..
1de0: 20 20 20 20 73 65 74 20 64 61 74 61 20 5b 76 66      set data [vf
1df0: 73 3a 3a 7a 69 70 20 2d 6d 6f 64 65 20 64 65 63  s::zip -mode dec
1e00: 6f 6d 70 72 65 73 73 20 2d 6e 6f 77 72 61 70 20  ompress -nowrap 
1e10: 31 20 24 64 61 74 61 5d 0a 09 7d 20 65 72 72 5d  1 $data]..} err]
1e20: 20 7d 20 7b 0a 09 20 20 20 20 3a 3a 76 66 73 3a   } {..    ::vfs:
1e30: 3a 6c 6f 67 20 22 24 73 62 28 6e 61 6d 65 29 3a  :log "$sb(name):
1e40: 20 69 6e 66 6c 61 74 65 20 65 72 72 6f 72 3a 20   inflate error: 
1e50: 24 65 72 72 22 0a 09 20 20 20 20 62 69 6e 61 72  $err"..    binar
1e60: 79 20 73 63 61 6e 20 24 64 61 74 61 20 48 2a 20  y scan $data H* 
1e70: 78 0a 09 20 20 20 20 3a 3a 76 66 73 3a 3a 6c 6f  x..    ::vfs::lo
1e80: 67 20 24 78 0a 09 7d 0a 20 20 20 20 7d 0a 20 20  g $x..}.    }.  
1e90: 20 20 72 65 74 75 72 6e 0a 20 20 20 20 69 66 20    return.    if 
1ea0: 7b 20 24 76 65 72 69 66 79 20 7d 20 7b 0a 09 73  { $verify } {..s
1eb0: 65 74 20 6e 63 72 63 20 5b 76 66 73 3a 3a 63 72  et ncrc [vfs::cr
1ec0: 63 20 24 64 61 74 61 5d 0a 09 69 66 20 7b 20 24  c $data]..if { $
1ed0: 6e 63 72 63 20 21 3d 20 24 73 62 28 63 72 63 29  ncrc != $sb(crc)
1ee0: 20 7d 20 7b 0a 09 20 20 20 20 74 63 6c 4c 6f 67   } {..    tclLog
1ef0: 20 5b 66 6f 72 6d 61 74 20 7b 25 73 3a 20 63 72   [format {%s: cr
1f00: 63 20 6d 69 73 6d 61 74 63 68 3a 20 65 78 70 65  c mismatch: expe
1f10: 63 74 65 64 20 30 78 25 78 2c 20 67 6f 74 20 30  cted 0x%x, got 0
1f20: 78 25 78 7d 20 5c 0a 09 09 20 20 20 20 24 73 62  x%x} \...    $sb
1f30: 28 6e 61 6d 65 29 20 24 73 62 28 63 72 63 29 20  (name) $sb(crc) 
1f40: 24 6e 63 72 63 5d 0a 09 7d 0a 20 20 20 20 7d 0a  $ncrc]..}.    }.
1f50: 7d 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a 45 6e 64  }..proc zip::End
1f60: 4f 66 41 72 63 68 69 76 65 20 7b 66 64 20 61 72  OfArchive {fd ar
1f70: 72 7d 20 7b 0a 20 20 20 20 75 70 76 61 72 20 31  r} {.    upvar 1
1f80: 20 24 61 72 72 20 63 62 0a 0a 20 20 20 20 23 20   $arr cb..    # 
1f90: 5b 53 46 20 54 63 6c 76 66 73 20 42 75 67 20 31  [SF Tclvfs Bug 1
1fa0: 30 30 33 35 37 34 5d 2e 20 44 6f 20 6e 6f 74 20  003574]. Do not 
1fb0: 73 65 65 6b 20 6f 76 65 72 20 62 65 67 69 6e 6e  seek over beginn
1fc0: 69 6e 67 20 6f 66 20 66 69 6c 65 2e 0a 20 20 20  ing of file..   
1fd0: 20 73 65 65 6b 20 24 66 64 20 30 20 65 6e 64 0a   seek $fd 0 end.
1fe0: 0a 20 20 20 20 23 20 4a 75 73 74 20 6c 6f 6f 6b  .    # Just look
1ff0: 69 6e 67 20 69 6e 20 74 68 65 20 6c 61 73 74 20  ing in the last 
2000: 35 31 32 20 62 79 74 65 73 20 6d 61 79 20 62 65  512 bytes may be
2010: 20 65 6e 6f 75 67 68 20 74 6f 20 68 61 6e 64 6c   enough to handl
2020: 65 20 7a 69 70 0a 20 20 20 20 23 20 61 72 63 68  e zip.    # arch
2030: 69 76 65 73 20 77 69 74 68 6f 75 74 20 63 6f 6d  ives without com
2040: 6d 65 6e 74 73 2c 20 68 6f 77 65 76 65 72 20 66  ments, however f
2050: 6f 72 20 61 72 63 68 69 76 65 73 20 77 68 69 63  or archives whic
2060: 68 20 68 61 76 65 0a 20 20 20 20 23 20 63 6f 6d  h have.    # com
2070: 6d 65 6e 74 73 20 74 68 65 20 63 68 75 6e 6b 20  ments the chunk 
2080: 6d 61 79 20 73 74 61 72 74 20 61 74 20 61 6e 20  may start at an 
2090: 61 72 62 69 74 72 61 72 79 20 64 69 73 74 61 6e  arbitrary distan
20a0: 63 65 20 66 72 6f 6d 20 74 68 65 0a 20 20 20 20  ce from the.    
20b0: 23 20 65 6e 64 20 6f 66 20 74 68 65 20 66 69 6c  # end of the fil
20c0: 65 2e 20 53 6f 20 69 66 20 77 65 20 64 6f 20 6e  e. So if we do n
20d0: 6f 74 20 66 69 6e 64 20 74 68 65 20 68 65 61 64  ot find the head
20e0: 65 72 20 69 6d 6d 65 64 69 61 74 65 6c 79 0a 20  er immediately. 
20f0: 20 20 20 23 20 77 65 20 68 61 76 65 20 74 6f 20     # we have to 
2100: 65 78 74 65 6e 64 20 74 68 65 20 72 61 6e 67 65  extend the range
2110: 20 6f 66 20 6f 75 72 20 73 65 61 72 63 68 2c 20   of our search, 
2120: 70 6f 73 73 69 62 6c 79 20 75 6e 74 69 6c 20 77  possibly until w
2130: 65 0a 20 20 20 20 23 20 68 61 76 65 20 61 20 6c  e.    # have a l
2140: 61 72 67 65 20 70 61 72 74 20 6f 66 20 74 68 65  arge part of the
2150: 20 61 72 63 68 69 76 65 20 69 6e 20 6d 65 6d 6f   archive in memo
2160: 72 79 2e 20 57 65 20 63 61 6e 20 66 61 69 6c 20  ry. We can fail 
2170: 6f 6e 6c 79 0a 20 20 20 20 23 20 61 66 74 65 72  only.    # after
2180: 20 74 68 65 20 77 68 6f 6c 65 20 66 69 6c 65 20   the whole file 
2190: 68 61 73 20 62 65 65 6e 20 73 65 61 72 63 68 65  has been searche
21a0: 64 2e 0a 0a 20 20 20 20 73 65 74 20 73 7a 20 20  d...    set sz  
21b0: 5b 74 65 6c 6c 20 24 66 64 5d 0a 20 20 20 20 69  [tell $fd].    i
21c0: 66 20 7b 5b 69 6e 66 6f 20 65 78 69 73 74 73 20  f {[info exists 
21d0: 3a 3a 7a 69 70 3a 3a 6d 61 78 5f 68 65 61 64 65  ::zip::max_heade
21e0: 72 5f 73 65 65 6b 5d 7d 20 7b 0a 20 20 20 20 20  r_seek]} {.     
21f0: 20 20 20 69 66 20 7b 24 3a 3a 7a 69 70 3a 3a 6d     if {$::zip::m
2200: 61 78 5f 68 65 61 64 65 72 5f 73 65 65 6b 20 3c  ax_header_seek <
2210: 20 24 73 7a 7d 20 7b 0a 20 20 20 20 20 20 20 20   $sz} {.        
2220: 20 20 20 20 73 65 74 20 73 7a 20 24 3a 3a 7a 69      set sz $::zi
2230: 70 3a 3a 6d 61 78 5f 68 65 61 64 65 72 5f 73 65  p::max_header_se
2240: 65 6b 0a 20 20 20 20 20 20 20 20 7d 0a 20 20 20  ek.        }.   
2250: 20 7d 20 0a 20 20 20 20 73 65 74 20 6c 65 6e 20   } .    set len 
2260: 35 31 32 0a 20 20 20 20 73 65 74 20 61 74 20 20  512.    set at  
2270: 35 31 32 0a 20 20 20 20 77 68 69 6c 65 20 7b 31  512.    while {1
2280: 7d 20 7b 0a 09 69 66 20 7b 24 73 7a 20 3c 20 24  } {..if {$sz < $
2290: 61 74 7d 20 7b 73 65 74 20 6e 20 2d 24 73 7a 7d  at} {set n -$sz}
22a0: 20 65 6c 73 65 20 7b 73 65 74 20 6e 20 2d 24 61   else {set n -$a
22b0: 74 7d 0a 0a 09 73 65 65 6b 20 24 66 64 20 24 6e  t}...seek $fd $n
22c0: 20 65 6e 64 0a 09 73 65 74 20 68 64 72 20 5b 72   end..set hdr [r
22d0: 65 61 64 20 24 66 64 20 24 6c 65 6e 5d 0a 09 73  ead $fd $len]..s
22e0: 65 74 20 70 6f 73 20 5b 73 74 72 69 6e 67 20 66  et pos [string f
22f0: 69 72 73 74 20 22 50 4b 5c 30 35 5c 30 36 22 20  irst "PK\05\06" 
2300: 24 68 64 72 5d 0a 09 69 66 20 7b 24 70 6f 73 20  $hdr]..if {$pos 
2310: 3d 3d 20 2d 31 7d 20 7b 0a 09 20 20 20 20 69 66  == -1} {..    if
2320: 20 7b 24 61 74 20 3e 3d 20 24 73 7a 7d 20 7b 0a   {$at >= $sz} {.
2330: 09 09 72 65 74 75 72 6e 20 2d 63 6f 64 65 20 65  ..return -code e
2340: 72 72 6f 72 20 22 6e 6f 20 68 65 61 64 65 72 20  rror "no header 
2350: 66 6f 75 6e 64 22 0a 09 20 20 20 20 7d 0a 09 20  found"..    }.. 
2360: 20 20 20 73 65 74 20 6c 65 6e 20 35 34 30 20 3b     set len 540 ;
2370: 20 23 20 61 66 74 65 72 20 31 73 74 20 69 74 65   # after 1st ite
2380: 72 61 74 69 6f 6e 20 77 65 20 66 6f 72 63 65 20  ration we force 
2390: 6f 76 65 72 6c 61 70 20 77 69 74 68 20 6c 61 73  overlap with las
23a0: 74 20 62 75 66 66 65 72 0a 09 20 20 20 20 69 6e  t buffer..    in
23b0: 63 72 20 61 74 20 35 31 32 20 3b 20 23 20 74 6f  cr at 512 ; # to
23c0: 20 65 6e 73 75 72 65 20 74 68 61 74 20 74 68 65   ensure that the
23d0: 20 70 61 74 74 65 72 6e 20 77 65 20 6c 6f 6f 6b   pattern we look
23e0: 20 66 6f 72 20 69 73 20 6e 6f 74 20 73 70 6c 69   for is not spli
23f0: 74 20 61 74 0a 09 20 20 20 20 23 20 20 20 20 20  t at..    #     
2400: 20 20 20 20 20 20 3b 20 23 20 61 20 62 75 66 66        ; # a buff
2410: 65 72 20 62 6f 75 6e 64 61 72 79 2c 20 6e 6f 72  er boundary, nor
2420: 20 74 68 65 20 68 65 61 64 65 72 20 69 74 73 65   the header itse
2430: 6c 66 0a 09 7d 20 65 6c 73 65 20 7b 0a 09 20 20  lf..} else {..  
2440: 20 20 62 72 65 61 6b 0a 09 7d 0a 20 20 20 20 7d    break..}.    }
2450: 0a 0a 20 20 20 20 73 65 74 20 68 64 72 20 5b 73  ..    set hdr [s
2460: 74 72 69 6e 67 20 72 61 6e 67 65 20 24 68 64 72  tring range $hdr
2470: 20 5b 65 78 70 72 20 24 70 6f 73 20 2b 20 34 5d   [expr $pos + 4]
2480: 20 5b 65 78 70 72 20 24 70 6f 73 20 2b 20 32 31   [expr $pos + 21
2490: 5d 5d 0a 0a 20 20 20 20 73 65 74 20 73 65 65 6b  ]]..    set seek
24a0: 73 74 61 72 74 20 5b 65 78 70 72 20 7b 5b 74 65  start [expr {[te
24b0: 6c 6c 20 24 66 64 5d 20 2d 20 35 31 32 7d 5d 0a  ll $fd] - 512}].
24c0: 20 20 20 20 69 66 20 7b 24 73 65 65 6b 73 74 61      if {$seeksta
24d0: 72 74 20 3c 20 30 7d 20 7b 0a 20 20 20 20 20 20  rt < 0} {.      
24e0: 20 20 73 65 74 20 73 65 65 6b 73 74 61 72 74 20    set seekstart 
24f0: 30 0a 20 20 20 20 7d 0a 20 20 20 20 73 65 74 20  0.    }.    set 
2500: 70 6f 73 20 5b 65 78 70 72 20 7b 24 73 65 65 6b  pos [expr {$seek
2510: 73 74 61 72 74 20 2b 20 24 70 6f 73 7d 5d 0a 0a  start + $pos}]..
2520: 20 20 20 20 62 69 6e 61 72 79 20 73 63 61 6e 20      binary scan 
2530: 24 68 64 72 20 73 73 73 73 69 69 73 20 5c 0a 09  $hdr ssssiis \..
2540: 63 62 28 6e 64 69 73 6b 29 20 63 62 28 63 64 69  cb(ndisk) cb(cdi
2550: 73 6b 29 20 5c 0a 09 63 62 28 6e 69 74 65 6d 73  sk) \..cb(nitems
2560: 29 20 63 62 28 6e 74 6f 74 61 6c 29 20 5c 0a 09  ) cb(ntotal) \..
2570: 63 62 28 63 73 69 7a 65 29 20 63 62 28 63 6f 66  cb(csize) cb(cof
2580: 66 29 20 5c 0a 09 63 62 28 63 6f 6d 6d 65 6e 74  f) \..cb(comment
2590: 29 0a 0a 20 20 20 20 73 65 74 20 63 62 28 6e 64  )..    set cb(nd
25a0: 69 73 6b 29 09 5b 75 5f 73 68 6f 72 74 20 24 63  isk).[u_short $c
25b0: 62 28 6e 64 69 73 6b 29 5d 0a 20 20 20 20 73 65  b(ndisk)].    se
25c0: 74 20 63 62 28 6e 69 74 65 6d 73 29 09 5b 75 5f  t cb(nitems).[u_
25d0: 73 68 6f 72 74 20 24 63 62 28 6e 69 74 65 6d 73  short $cb(nitems
25e0: 29 5d 0a 20 20 20 20 73 65 74 20 63 62 28 6e 74  )].    set cb(nt
25f0: 6f 74 61 6c 29 09 5b 75 5f 73 68 6f 72 74 20 24  otal).[u_short $
2600: 63 62 28 6e 74 6f 74 61 6c 29 5d 0a 20 20 20 20  cb(ntotal)].    
2610: 73 65 74 20 63 62 28 63 6f 6d 6d 65 6e 74 29 09  set cb(comment).
2620: 5b 75 5f 73 68 6f 72 74 20 24 63 62 28 63 6f 6d  [u_short $cb(com
2630: 6d 65 6e 74 29 5d 0a 0a 20 20 20 20 23 20 43 6f  ment)]..    # Co
2640: 6d 70 75 74 65 20 62 61 73 65 20 66 6f 72 20 73  mpute base for s
2650: 69 74 75 61 74 69 6f 6e 73 20 77 68 65 72 65 20  ituations where 
2660: 5a 49 50 20 66 69 6c 65 0a 20 20 20 20 23 20 68  ZIP file.    # h
2670: 61 73 20 62 65 65 6e 20 61 70 70 65 6e 64 65 64  as been appended
2680: 20 74 6f 20 61 6e 6f 74 68 65 72 20 6d 65 64 69   to another medi
2690: 61 20 28 65 2e 67 2e 20 45 58 45 29 0a 20 20 20  a (e.g. EXE).   
26a0: 20 73 65 74 20 62 61 73 65 20 20 20 20 20 20 20   set base       
26b0: 20 20 20 20 20 5b 65 78 70 72 20 7b 20 24 70 6f       [expr { $po
26c0: 73 20 2d 20 24 63 62 28 63 73 69 7a 65 29 20 2d  s - $cb(csize) -
26d0: 20 24 63 62 28 63 6f 66 66 29 20 7d 5d 0a 20 20   $cb(coff) }].  
26e0: 20 20 69 66 20 7b 24 62 61 73 65 20 3c 20 30 7d    if {$base < 0}
26f0: 20 7b 0a 20 20 20 20 20 20 20 20 73 65 74 20 62   {.        set b
2700: 61 73 65 20 30 0a 20 20 20 20 7d 0a 20 20 20 20  ase 0.    }.    
2710: 73 65 74 20 63 62 28 62 61 73 65 29 09 24 62 61  set cb(base).$ba
2720: 73 65 0a 7d 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a  se.}..proc zip::
2730: 54 4f 43 20 7b 66 64 20 61 72 72 7d 20 7b 0a 20  TOC {fd arr} {. 
2740: 20 20 20 75 70 76 61 72 20 23 30 20 7a 69 70 3a     upvar #0 zip:
2750: 3a 24 66 64 20 63 62 0a 20 20 20 20 75 70 76 61  :$fd cb.    upva
2760: 72 20 31 20 24 61 72 72 20 73 62 0a 0a 20 20 20  r 1 $arr sb..   
2770: 20 73 65 74 20 62 75 66 20 5b 72 65 61 64 20 24   set buf [read $
2780: 66 64 20 34 36 5d 0a 0a 20 20 20 20 62 69 6e 61  fd 46]..    bina
2790: 72 79 20 73 63 61 6e 20 24 62 75 66 20 41 34 73  ry scan $buf A4s
27a0: 73 73 73 73 73 69 69 69 73 73 73 73 73 69 69 20  sssssiiisssssii 
27b0: 68 64 72 20 5c 0a 20 20 20 20 20 20 73 62 28 76  hdr \.      sb(v
27c0: 65 6d 29 20 73 62 28 76 65 72 29 20 73 62 28 66  em) sb(ver) sb(f
27d0: 6c 61 67 73 29 20 73 62 28 6d 65 74 68 6f 64 29  lags) sb(method)
27e0: 20 74 69 6d 65 20 64 61 74 65 20 5c 0a 20 20 20   time date \.   
27f0: 20 20 20 73 62 28 63 72 63 29 20 73 62 28 63 73     sb(crc) sb(cs
2800: 69 7a 65 29 20 73 62 28 73 69 7a 65 29 20 5c 0a  ize) sb(size) \.
2810: 20 20 20 20 20 20 66 6c 65 6e 20 65 6c 65 6e 20        flen elen 
2820: 63 6c 65 6e 20 73 62 28 64 69 73 6b 29 20 73 62  clen sb(disk) sb
2830: 28 61 74 74 72 29 20 5c 0a 20 20 20 20 20 20 73  (attr) \.      s
2840: 62 28 61 74 78 29 20 73 62 28 69 6e 6f 29 0a 0a  b(atx) sb(ino)..
2850: 20 20 20 20 73 65 74 20 73 62 28 69 6e 6f 29 20      set sb(ino) 
2860: 5b 65 78 70 72 20 7b 24 63 62 28 62 61 73 65 29  [expr {$cb(base)
2870: 20 2b 20 24 73 62 28 69 6e 6f 29 7d 5d 0a 0a 20   + $sb(ino)}].. 
2880: 20 20 20 69 66 20 7b 20 21 5b 73 74 72 69 6e 67     if { ![string
2890: 20 65 71 75 61 6c 20 22 50 4b 5c 30 31 5c 30 32   equal "PK\01\02
28a0: 22 20 24 68 64 72 5d 20 7d 20 7b 0a 09 62 69 6e  " $hdr] } {..bin
28b0: 61 72 79 20 73 63 61 6e 20 24 68 64 72 20 48 2a  ary scan $hdr H*
28c0: 20 78 0a 09 65 72 72 6f 72 20 22 62 61 64 20 63   x..error "bad c
28d0: 65 6e 74 72 61 6c 20 68 65 61 64 65 72 3a 20 24  entral header: $
28e0: 78 22 0a 20 20 20 20 7d 0a 0a 20 20 20 20 66 6f  x".    }..    fo
28f0: 72 65 61 63 68 20 76 20 7b 76 65 6d 20 76 65 72  reach v {vem ver
2900: 20 66 6c 61 67 73 20 6d 65 74 68 6f 64 20 64 69   flags method di
2910: 73 6b 20 61 74 74 72 7d 20 7b 0a 09 73 65 74 20  sk attr} {..set 
2920: 63 62 28 24 76 29 20 5b 75 5f 73 68 6f 72 74 20  cb($v) [u_short 
2930: 5b 73 65 74 20 73 62 28 24 76 29 5d 5d 0a 20 20  [set sb($v)]].  
2940: 20 20 7d 0a 0a 20 20 20 20 73 65 74 20 73 62 28    }..    set sb(
2950: 6d 74 69 6d 65 29 20 5b 44 6f 73 54 69 6d 65 20  mtime) [DosTime 
2960: 24 64 61 74 65 20 24 74 69 6d 65 5d 0a 20 20 20  $date $time].   
2970: 20 73 65 74 20 73 62 28 6d 6f 64 65 29 20 5b 65   set sb(mode) [e
2980: 78 70 72 20 7b 20 28 24 73 62 28 61 74 78 29 20  xpr { ($sb(atx) 
2990: 3e 3e 20 31 36 29 20 26 20 30 78 66 66 66 66 20  >> 16) & 0xffff 
29a0: 7d 5d 0a 20 20 20 20 69 66 20 7b 20 28 20 24 73  }].    if { ( $s
29b0: 62 28 61 74 78 29 20 26 20 30 78 66 66 20 29 20  b(atx) & 0xff ) 
29c0: 26 20 31 36 20 7d 20 7b 0a 09 73 65 74 20 73 62  & 16 } {..set sb
29d0: 28 74 79 70 65 29 20 64 69 72 65 63 74 6f 72 79  (type) directory
29e0: 0a 20 20 20 20 7d 20 65 6c 73 65 20 7b 0a 09 73  .    } else {..s
29f0: 65 74 20 73 62 28 74 79 70 65 29 20 66 69 6c 65  et sb(type) file
2a00: 0a 20 20 20 20 7d 0a 20 20 20 20 73 65 74 20 73  .    }.    set s
2a10: 62 28 6e 61 6d 65 29 20 5b 72 65 61 64 20 24 66  b(name) [read $f
2a20: 64 20 5b 75 5f 73 68 6f 72 74 20 24 66 6c 65 6e  d [u_short $flen
2a30: 5d 5d 0a 20 20 20 20 73 65 74 20 73 62 28 65 78  ]].    set sb(ex
2a40: 74 72 61 29 20 5b 72 65 61 64 20 24 66 64 20 5b  tra) [read $fd [
2a50: 75 5f 73 68 6f 72 74 20 24 65 6c 65 6e 5d 5d 0a  u_short $elen]].
2a60: 20 20 20 20 73 65 74 20 73 62 28 63 6f 6d 6d 65      set sb(comme
2a70: 6e 74 29 20 5b 72 65 61 64 20 24 66 64 20 5b 75  nt) [read $fd [u
2a80: 5f 73 68 6f 72 74 20 24 63 6c 65 6e 5d 5d 0a 7d  _short $clen]].}
2a90: 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a 6f 70 65 6e  ..proc zip::open
2aa0: 20 7b 70 61 74 68 7d 20 7b 0a 20 20 20 20 73 65   {path} {.    se
2ab0: 74 20 66 64 20 5b 3a 3a 6f 70 65 6e 20 24 70 61  t fd [::open $pa
2ac0: 74 68 5d 0a 20 20 20 20 0a 20 20 20 20 69 66 20  th].    .    if 
2ad0: 7b 5b 63 61 74 63 68 20 7b 0a 09 75 70 76 61 72  {[catch {..upvar
2ae0: 20 23 30 20 7a 69 70 3a 3a 24 66 64 20 63 62 0a   #0 zip::$fd cb.
2af0: 09 75 70 76 61 72 20 23 30 20 7a 69 70 3a 3a 24  .upvar #0 zip::$
2b00: 66 64 2e 74 6f 63 20 74 6f 63 0a 0a 09 66 63 6f  fd.toc toc...fco
2b10: 6e 66 69 67 75 72 65 20 24 66 64 20 2d 74 72 61  nfigure $fd -tra
2b20: 6e 73 6c 61 74 69 6f 6e 20 62 69 6e 61 72 79 20  nslation binary 
2b30: 3b 23 2d 62 75 66 66 65 72 69 6e 67 20 6e 6f 6e  ;#-buffering non
2b40: 65 0a 09 0a 09 7a 69 70 3a 3a 45 6e 64 4f 66 41  e....zip::EndOfA
2b50: 72 63 68 69 76 65 20 24 66 64 20 63 62 0a 0a 09  rchive $fd cb...
2b60: 73 65 65 6b 20 24 66 64 20 5b 65 78 70 72 20 7b  seek $fd [expr {
2b70: 24 63 62 28 62 61 73 65 29 20 2b 20 24 63 62 28  $cb(base) + $cb(
2b80: 63 6f 66 66 29 7d 5d 20 73 74 61 72 74 0a 0a 09  coff)}] start...
2b90: 73 65 74 20 74 6f 63 28 5f 29 20 30 3b 20 75 6e  set toc(_) 0; un
2ba0: 73 65 74 20 74 6f 63 28 5f 29 3b 20 23 4d 61 6b  set toc(_); #Mak
2bb0: 65 41 72 72 61 79 0a 09 0a 09 66 6f 72 20 7b 20  eArray....for { 
2bc0: 73 65 74 20 69 20 30 20 7d 20 7b 20 24 69 20 3c  set i 0 } { $i <
2bd0: 20 24 63 62 28 6e 69 74 65 6d 73 29 20 7d 20 7b   $cb(nitems) } {
2be0: 20 69 6e 63 72 20 69 20 7d 20 7b 0a 09 20 20 20   incr i } {..   
2bf0: 20 7a 69 70 3a 3a 54 4f 43 20 24 66 64 20 73 62   zip::TOC $fd sb
2c00: 0a 09 20 20 20 20 0a 09 20 20 20 20 73 65 74 20  ..    ..    set 
2c10: 73 62 28 64 65 70 74 68 29 20 5b 6c 6c 65 6e 67  sb(depth) [lleng
2c20: 74 68 20 5b 66 69 6c 65 20 73 70 6c 69 74 20 24  th [file split $
2c30: 73 62 28 6e 61 6d 65 29 5d 5d 0a 09 20 20 20 20  sb(name)]]..    
2c40: 0a 09 20 20 20 20 73 65 74 20 6e 61 6d 65 20 5b  ..    set name [
2c50: 73 74 72 69 6e 67 20 74 6f 6c 6f 77 65 72 20 24  string tolower $
2c60: 73 62 28 6e 61 6d 65 29 5d 0a 09 20 20 20 20 73  sb(name)]..    s
2c70: 65 74 20 74 6f 63 28 24 6e 61 6d 65 29 20 5b 61  et toc($name) [a
2c80: 72 72 61 79 20 67 65 74 20 73 62 5d 0a 09 20 20  rray get sb]..  
2c90: 20 20 46 41 4b 45 44 49 52 20 74 6f 63 20 5b 66    FAKEDIR toc [f
2ca0: 69 6c 65 20 64 69 72 6e 61 6d 65 20 24 6e 61 6d  ile dirname $nam
2cb0: 65 5d 0a 09 7d 0a 20 20 20 20 7d 20 65 72 72 5d  e]..}.    } err]
2cc0: 7d 20 7b 0a 09 63 6c 6f 73 65 20 24 66 64 0a 09  } {..close $fd..
2cd0: 72 65 74 75 72 6e 20 2d 63 6f 64 65 20 65 72 72  return -code err
2ce0: 6f 72 20 24 65 72 72 0a 20 20 20 20 7d 0a 0a 20  or $err.    }.. 
2cf0: 20 20 20 72 65 74 75 72 6e 20 24 66 64 0a 7d 0a     return $fd.}.
2d00: 0a 70 72 6f 63 20 7a 69 70 3a 3a 46 41 4b 45 44  .proc zip::FAKED
2d10: 49 52 20 7b 61 72 72 20 70 61 74 68 7d 20 7b 0a  IR {arr path} {.
2d20: 20 20 20 20 75 70 76 61 72 20 31 20 24 61 72 72      upvar 1 $arr
2d30: 20 74 6f 63 0a 0a 20 20 20 20 69 66 20 7b 20 24   toc..    if { $
2d40: 70 61 74 68 20 3d 3d 20 22 2e 22 7d 20 7b 20 72  path == "."} { r
2d50: 65 74 75 72 6e 20 7d 0a 0a 0a 20 20 20 20 69 66  eturn }...    if
2d60: 20 7b 20 21 5b 69 6e 66 6f 20 65 78 69 73 74 73   { ![info exists
2d70: 20 74 6f 63 28 24 70 61 74 68 29 5d 20 7d 20 7b   toc($path)] } {
2d80: 0a 09 23 20 49 6d 70 6c 69 63 69 74 20 64 69 72  ..# Implicit dir
2d90: 65 63 74 6f 72 79 0a 09 6c 61 70 70 65 6e 64 20  ectory..lappend 
2da0: 74 6f 63 28 24 70 61 74 68 29 20 5c 0a 09 09 6e  toc($path) \...n
2db0: 61 6d 65 20 24 70 61 74 68 20 5c 0a 09 09 74 79  ame $path \...ty
2dc0: 70 65 20 64 69 72 65 63 74 6f 72 79 20 6d 74 69  pe directory mti
2dd0: 6d 65 20 30 20 73 69 7a 65 20 30 20 6d 6f 64 65  me 0 size 0 mode
2de0: 20 30 37 37 37 20 5c 0a 09 09 69 6e 6f 20 2d 31   0777 \...ino -1
2df0: 20 64 65 70 74 68 20 5b 6c 6c 65 6e 67 74 68 20   depth [llength 
2e00: 5b 66 69 6c 65 20 73 70 6c 69 74 20 24 70 61 74  [file split $pat
2e10: 68 5d 5d 0a 20 20 20 20 7d 0a 20 20 20 20 46 41  h]].    }.    FA
2e20: 4b 45 44 49 52 20 74 6f 63 20 5b 66 69 6c 65 20  KEDIR toc [file 
2e30: 64 69 72 6e 61 6d 65 20 24 70 61 74 68 5d 0a 7d  dirname $path].}
2e40: 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a 65 78 69 73  ..proc zip::exis
2e50: 74 73 20 7b 66 64 20 70 61 74 68 7d 20 7b 0a 20  ts {fd path} {. 
2e60: 20 20 20 23 3a 3a 76 66 73 3a 3a 6c 6f 67 20 22     #::vfs::log "
2e70: 24 66 64 20 24 70 61 74 68 22 0a 20 20 20 20 69  $fd $path".    i
2e80: 66 20 7b 24 70 61 74 68 20 3d 3d 20 22 22 7d 20  f {$path == ""} 
2e90: 7b 0a 09 72 65 74 75 72 6e 20 31 0a 20 20 20 20  {..return 1.    
2ea0: 7d 20 65 6c 73 65 20 7b 0a 09 75 70 76 61 72 20  } else {..upvar 
2eb0: 23 30 20 7a 69 70 3a 3a 24 66 64 2e 74 6f 63 20  #0 zip::$fd.toc 
2ec0: 74 6f 63 0a 09 69 6e 66 6f 20 65 78 69 73 74 73  toc..info exists
2ed0: 20 74 6f 63 28 5b 73 74 72 69 6e 67 20 74 6f 6c   toc([string tol
2ee0: 6f 77 65 72 20 24 70 61 74 68 5d 29 0a 20 20 20  ower $path]).   
2ef0: 20 7d 0a 7d 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a   }.}..proc zip::
2f00: 73 74 61 74 20 7b 66 64 20 70 61 74 68 20 61 72  stat {fd path ar
2f10: 72 7d 20 7b 0a 20 20 20 20 75 70 76 61 72 20 23  r} {.    upvar #
2f20: 30 20 7a 69 70 3a 3a 24 66 64 2e 74 6f 63 20 74  0 zip::$fd.toc t
2f30: 6f 63 0a 20 20 20 20 75 70 76 61 72 20 31 20 24  oc.    upvar 1 $
2f40: 61 72 72 20 73 62 0a 0a 20 20 20 20 73 65 74 20  arr sb..    set 
2f50: 6e 61 6d 65 20 5b 73 74 72 69 6e 67 20 74 6f 6c  name [string tol
2f60: 6f 77 65 72 20 24 70 61 74 68 5d 0a 20 20 20 20  ower $path].    
2f70: 69 66 20 7b 20 24 6e 61 6d 65 20 3d 3d 20 22 22  if { $name == ""
2f80: 20 7c 7c 20 24 6e 61 6d 65 20 3d 3d 20 22 2e 22   || $name == "."
2f90: 20 7d 20 7b 0a 09 61 72 72 61 79 20 73 65 74 20   } {..array set 
2fa0: 73 62 20 7b 0a 09 20 20 20 20 74 79 70 65 20 64  sb {..    type d
2fb0: 69 72 65 63 74 6f 72 79 20 6d 74 69 6d 65 20 30  irectory mtime 0
2fc0: 20 73 69 7a 65 20 30 20 6d 6f 64 65 20 30 37 37   size 0 mode 077
2fd0: 37 20 0a 09 20 20 20 20 69 6e 6f 20 2d 31 20 64  7 ..    ino -1 d
2fe0: 65 70 74 68 20 30 20 6e 61 6d 65 20 22 22 0a 09  epth 0 name ""..
2ff0: 7d 0a 20 20 20 20 7d 20 65 6c 73 65 69 66 20 7b  }.    } elseif {
3000: 21 5b 69 6e 66 6f 20 65 78 69 73 74 73 20 74 6f  ![info exists to
3010: 63 28 24 6e 61 6d 65 29 5d 20 7d 20 7b 0a 09 72  c($name)] } {..r
3020: 65 74 75 72 6e 20 2d 63 6f 64 65 20 65 72 72 6f  eturn -code erro
3030: 72 20 22 63 6f 75 6c 64 20 6e 6f 74 20 72 65 61  r "could not rea
3040: 64 20 5c 22 24 70 61 74 68 5c 22 3a 20 6e 6f 20  d \"$path\": no 
3050: 73 75 63 68 20 66 69 6c 65 20 6f 72 20 64 69 72  such file or dir
3060: 65 63 74 6f 72 79 22 0a 20 20 20 20 7d 20 65 6c  ectory".    } el
3070: 73 65 20 7b 0a 09 61 72 72 61 79 20 73 65 74 20  se {..array set 
3080: 73 62 20 24 74 6f 63 28 24 6e 61 6d 65 29 0a 20  sb $toc($name). 
3090: 20 20 20 7d 0a 20 20 20 20 73 65 74 20 73 62 28     }.    set sb(
30a0: 64 65 76 29 20 2d 31 0a 20 20 20 20 73 65 74 20  dev) -1.    set 
30b0: 73 62 28 75 69 64 29 09 2d 31 0a 20 20 20 20 73  sb(uid).-1.    s
30c0: 65 74 20 73 62 28 67 69 64 29 09 2d 31 0a 20 20  et sb(gid).-1.  
30d0: 20 20 73 65 74 20 73 62 28 6e 6c 69 6e 6b 29 20    set sb(nlink) 
30e0: 31 0a 20 20 20 20 73 65 74 20 73 62 28 61 74 69  1.    set sb(ati
30f0: 6d 65 29 20 24 73 62 28 6d 74 69 6d 65 29 0a 20  me) $sb(mtime). 
3100: 20 20 20 73 65 74 20 73 62 28 63 74 69 6d 65 29     set sb(ctime)
3110: 20 24 73 62 28 6d 74 69 6d 65 29 0a 20 20 20 20   $sb(mtime).    
3120: 72 65 74 75 72 6e 20 22 22 0a 7d 0a 0a 23 20 54  return "".}..# T
3130: 72 65 61 74 73 20 65 6d 70 74 79 20 70 61 74 74  reats empty patt
3140: 65 72 6e 20 61 73 20 61 73 6b 69 6e 67 20 66 6f  ern as asking fo
3150: 72 20 61 20 70 61 72 74 69 63 75 6c 61 72 20 66  r a particular f
3160: 69 6c 65 20 6f 6e 6c 79 0a 70 72 6f 63 20 7a 69  ile only.proc zi
3170: 70 3a 3a 67 65 74 64 69 72 20 7b 66 64 20 70 61  p::getdir {fd pa
3180: 74 68 20 7b 70 61 74 20 2a 7d 7d 20 7b 0a 20 20  th {pat *}} {.  
3190: 20 20 23 3a 3a 76 66 73 3a 3a 6c 6f 67 20 5b 6c    #::vfs::log [l
31a0: 69 73 74 20 67 65 74 64 69 72 20 24 66 64 20 24  ist getdir $fd $
31b0: 70 61 74 68 20 24 70 61 74 5d 0a 20 20 20 20 75  path $pat].    u
31c0: 70 76 61 72 20 23 30 20 7a 69 70 3a 3a 24 66 64  pvar #0 zip::$fd
31d0: 2e 74 6f 63 20 74 6f 63 0a 0a 20 20 20 20 69 66  .toc toc..    if
31e0: 20 7b 20 24 70 61 74 68 20 3d 3d 20 22 2e 22 20   { $path == "." 
31f0: 7c 7c 20 24 70 61 74 68 20 3d 3d 20 22 22 20 7d  || $path == "" }
3200: 20 7b 0a 09 73 65 74 20 70 61 74 68 20 5b 73 74   {..set path [st
3210: 72 69 6e 67 20 74 6f 6c 6f 77 65 72 20 24 70 61  ring tolower $pa
3220: 74 5d 0a 20 20 20 20 7d 20 65 6c 73 65 20 7b 0a  t].    } else {.
3230: 09 73 65 74 20 70 61 74 68 20 5b 73 74 72 69 6e  .set path [strin
3240: 67 20 74 6f 6c 6f 77 65 72 20 24 70 61 74 68 5d  g tolower $path]
3250: 0a 09 69 66 20 7b 24 70 61 74 20 21 3d 20 22 22  ..if {$pat != ""
3260: 7d 20 7b 0a 09 20 20 20 20 61 70 70 65 6e 64 20  } {..    append 
3270: 70 61 74 68 20 2f 5b 73 74 72 69 6e 67 20 74 6f  path /[string to
3280: 6c 6f 77 65 72 20 24 70 61 74 5d 0a 09 7d 0a 20  lower $pat]..}. 
3290: 20 20 20 7d 0a 20 20 20 20 73 65 74 20 64 65 70     }.    set dep
32a0: 74 68 20 5b 6c 6c 65 6e 67 74 68 20 5b 66 69 6c  th [llength [fil
32b0: 65 20 73 70 6c 69 74 20 24 70 61 74 68 5d 5d 0a  e split $path]].
32c0: 0a 20 20 20 20 23 70 75 74 73 20 73 74 64 65 72  .    #puts stder
32d0: 72 20 22 67 65 74 64 69 72 20 24 66 64 20 24 70  r "getdir $fd $p
32e0: 61 74 68 20 24 64 65 70 74 68 20 24 70 61 74 20  ath $depth $pat 
32f0: 5b 61 72 72 61 79 20 6e 61 6d 65 73 20 74 6f 63  [array names toc
3300: 20 24 70 61 74 68 5d 22 0a 20 20 20 20 69 66 20   $path]".    if 
3310: 7b 24 64 65 70 74 68 7d 20 7b 0a 09 73 65 74 20  {$depth} {..set 
3320: 72 65 74 20 7b 7d 0a 09 66 6f 72 65 61 63 68 20  ret {}..foreach 
3330: 6b 65 79 20 5b 61 72 72 61 79 20 6e 61 6d 65 73  key [array names
3340: 20 74 6f 63 20 24 70 61 74 68 5d 20 7b 0a 09 20   toc $path] {.. 
3350: 20 20 20 69 66 20 7b 5b 73 74 72 69 6e 67 20 69     if {[string i
3360: 6e 64 65 78 20 24 6b 65 79 20 65 6e 64 5d 20 3d  ndex $key end] =
3370: 3d 20 22 2f 22 7d 20 7b 0a 09 09 23 20 44 69 72  = "/"} {...# Dir
3380: 65 63 74 6f 72 69 65 73 20 61 72 65 20 6c 69 73  ectories are lis
3390: 74 65 64 20 74 77 69 63 65 3a 20 62 6f 74 68 20  ted twice: both 
33a0: 77 69 74 68 20 61 6e 64 20 77 69 74 68 6f 75 74  with and without
33b0: 0a 09 09 23 20 74 68 65 20 74 72 61 69 6c 69 6e  ...# the trailin
33c0: 67 20 27 2f 27 2c 20 73 6f 20 77 65 20 69 67 6e  g '/', so we ign
33d0: 6f 72 65 20 74 68 65 20 6f 6e 65 20 77 69 74 68  ore the one with
33e0: 0a 09 09 63 6f 6e 74 69 6e 75 65 0a 09 20 20 20  ...continue..   
33f0: 20 7d 0a 09 20 20 20 20 61 72 72 61 79 20 73 65   }..    array se
3400: 74 20 73 62 20 24 74 6f 63 28 24 6b 65 79 29 0a  t sb $toc($key).
3410: 0a 09 20 20 20 20 69 66 20 7b 20 24 73 62 28 64  ..    if { $sb(d
3420: 65 70 74 68 29 20 3d 3d 20 24 64 65 70 74 68 20  epth) == $depth 
3430: 7d 20 7b 0a 09 09 69 66 20 7b 5b 69 6e 66 6f 20  } {...if {[info 
3440: 65 78 69 73 74 73 20 74 6f 63 28 24 7b 6b 65 79  exists toc(${key
3450: 7d 2f 29 5d 7d 20 7b 0a 09 09 20 20 20 20 61 72  }/)]} {...    ar
3460: 72 61 79 20 73 65 74 20 73 62 20 24 74 6f 63 28  ray set sb $toc(
3470: 24 7b 6b 65 79 7d 2f 29 0a 09 09 7d 0a 09 09 6c  ${key}/)...}...l
3480: 61 70 70 65 6e 64 20 72 65 74 20 5b 66 69 6c 65  append ret [file
3490: 20 74 61 69 6c 20 24 73 62 28 6e 61 6d 65 29 5d   tail $sb(name)]
34a0: 0a 09 20 20 20 20 7d 20 65 6c 73 65 20 7b 0a 09  ..    } else {..
34b0: 09 23 3a 3a 76 66 73 3a 3a 6c 6f 67 20 22 24 73  .#::vfs::log "$s
34c0: 62 28 64 65 70 74 68 29 20 76 73 20 24 64 65 70  b(depth) vs $dep
34d0: 74 68 20 66 6f 72 20 24 73 62 28 6e 61 6d 65 29  th for $sb(name)
34e0: 22 0a 09 20 20 20 20 7d 0a 09 20 20 20 20 75 6e  "..    }..    un
34f0: 73 65 74 20 73 62 0a 09 7d 0a 09 72 65 74 75 72  set sb..}..retur
3500: 6e 20 24 72 65 74 0a 20 20 20 20 7d 20 65 6c 73  n $ret.    } els
3510: 65 20 7b 0a 09 23 20 6a 75 73 74 20 74 68 65 20  e {..# just the 
3520: 27 72 6f 6f 74 27 20 6f 66 20 74 68 65 20 7a 69  'root' of the zi
3530: 70 20 61 72 63 68 69 76 65 2e 20 20 54 68 69 73  p archive.  This
3540: 20 6f 62 76 69 6f 75 73 6c 79 20 65 78 69 73 74   obviously exist
3550: 73 20 61 6e 64 0a 09 23 20 69 73 20 61 20 64 69  s and..# is a di
3560: 72 65 63 74 6f 72 79 2e 0a 09 72 65 74 75 72 6e  rectory...return
3570: 20 5b 6c 69 73 74 20 7b 7d 5d 0a 20 20 20 20 7d   [list {}].    }
3580: 0a 7d 0a 0a 70 72 6f 63 20 7a 69 70 3a 3a 5f 63  .}..proc zip::_c
3590: 6c 6f 73 65 20 7b 66 64 7d 20 7b 0a 20 20 20 20  lose {fd} {.    
35a0: 76 61 72 69 61 62 6c 65 20 24 66 64 0a 20 20 20  variable $fd.   
35b0: 20 76 61 72 69 61 62 6c 65 20 24 66 64 2e 74 6f   variable $fd.to
35c0: 63 0a 20 20 20 20 75 6e 73 65 74 20 24 66 64 0a  c.    unset $fd.
35d0: 20 20 20 20 75 6e 73 65 74 20 24 66 64 2e 74 6f      unset $fd.to
35e0: 63 0a 20 20 20 20 3a 3a 63 6c 6f 73 65 20 24 66  c.    ::close $f
35f0: 64 0a 7d 0a 0a 23 20 75 73 65 20 7a 6c 69 62 20  d.}..# use zlib 
3600: 74 6f 20 64 65 66 69 6e 65 20 7a 69 70 20 61 6e  to define zip an
3610: 64 20 63 72 63 20 69 66 20 61 76 61 69 6c 61 62  d crc if availab
3620: 6c 65 0a 69 66 20 7b 5b 6c 6c 65 6e 67 74 68 20  le.if {[llength 
3630: 5b 69 6e 66 6f 20 63 6f 6d 6d 61 6e 64 20 76 66  [info command vf
3640: 73 3a 3a 7a 69 70 5d 5d 20 3d 3d 20 30 20 26 26  s::zip]] == 0 &&
3650: 20 5b 6c 6c 65 6e 67 74 68 20 5b 69 6e 66 6f 20   [llength [info 
3660: 63 6f 6d 6d 61 6e 64 20 7a 6c 69 62 5d 5d 20 7c  command zlib]] |
3670: 7c 20 21 5b 63 61 74 63 68 20 7b 6c 6f 61 64 20  | ![catch {load 
3680: 22 22 20 7a 6c 69 62 7d 5d 7d 20 7b 0a 09 70 72  "" zlib}]} {..pr
3690: 6f 63 20 76 66 73 3a 3a 7a 69 70 20 7b 66 6c 61  oc vfs::zip {fla
36a0: 67 20 76 61 6c 75 65 20 61 72 67 73 7d 20 7b 0a  g value args} {.
36b0: 09 09 73 77 69 74 63 68 20 2d 67 6c 6f 62 20 2d  ..switch -glob -
36c0: 2d 20 22 24 66 6c 61 67 20 24 76 61 6c 75 65 22  - "$flag $value"
36d0: 20 7b 0a 09 09 09 7b 2d 6d 6f 64 65 20 64 2a 7d   {....{-mode d*}
36e0: 20 7b 20 73 65 74 20 6d 6f 64 65 20 64 65 63 6f   { set mode deco
36f0: 6d 70 72 65 73 73 20 7d 0a 09 09 09 7b 2d 6d 6f  mpress }....{-mo
3700: 64 65 20 63 2a 7d 20 7b 20 73 65 74 20 6d 6f 64  de c*} { set mod
3710: 65 20 63 6f 6d 70 72 65 73 73 20 7d 0a 09 09 09  e compress }....
3720: 64 65 66 61 75 6c 74 20 7b 20 65 72 72 6f 72 20  default { error 
3730: 22 75 73 61 67 65 3a 20 7a 69 70 20 2d 6d 6f 64  "usage: zip -mod
3740: 65 20 7b 63 6f 6d 70 72 65 73 73 7c 64 65 63 6f  e {compress|deco
3750: 6d 70 72 65 73 73 7d 20 64 61 74 61 22 20 7d 0a  mpress} data" }.
3760: 09 09 7d 0a 0a 09 09 23 20 6b 6c 75 64 67 65 20  ..}....# kludge 
3770: 74 6f 20 61 6c 6c 6f 77 20 22 2d 6e 6f 77 72 61  to allow "-nowra
3780: 70 20 31 22 20 61 73 20 73 65 63 6f 6e 64 20 6f  p 1" as second o
3790: 70 74 69 6f 6e 2c 20 35 2d 39 2d 32 30 30 32 0a  ption, 5-9-2002.
37a0: 09 09 69 66 20 7b 5b 6c 6c 65 6e 67 74 68 20 24  ..if {[llength $
37b0: 61 72 67 73 5d 20 3e 20 32 20 26 26 20 5b 6c 72  args] > 2 && [lr
37c0: 61 6e 67 65 20 24 61 72 67 73 20 30 20 31 5d 20  ange $args 0 1] 
37d0: 65 71 20 22 2d 6e 6f 77 72 61 70 20 31 22 7d 20  eq "-nowrap 1"} 
37e0: 7b 0a 09 09 09 69 66 20 7b 24 6d 6f 64 65 20 65  {....if {$mode e
37f0: 71 20 22 63 6f 6d 70 72 65 73 73 22 7d 20 7b 0a  q "compress"} {.
3800: 09 09 09 09 73 65 74 20 6d 6f 64 65 20 64 65 66  ....set mode def
3810: 6c 61 74 65 0a 09 09 09 7d 20 65 6c 73 65 20 7b  late....} else {
3820: 0a 09 09 09 09 73 65 74 20 6d 6f 64 65 20 69 6e  .....set mode in
3830: 66 6c 61 74 65 0a 09 09 09 7d 0a 09 09 7d 0a 0a  flate....}...}..
3840: 09 09 72 65 74 75 72 6e 20 5b 7a 6c 69 62 20 24  ..return [zlib $
3850: 6d 6f 64 65 20 5b 6c 69 6e 64 65 78 20 24 61 72  mode [lindex $ar
3860: 67 73 20 65 6e 64 5d 5d 0a 09 7d 0a 7d 0a        gs end]]..}.}.