Patch title: Release 93 bulk changes
Abstract:
File: /protocol/http/chunked.pli
Key:
    Removed line
    Added line
# Copyright  Hubert Tonneau  hubert.tonneau@pliant.cx
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 2
# as published by the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# version 2 along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.

module "/pliant/language/unsafe.pli"
module "/pliant/language/stream.pli"
module "/pliant/language/stream/filesystembase.pli"
module "/pliant/language/stream/multi.pli"
module "/pliant/language/stream/openmode.pli"


type ChunkedStreamDriver
  field Link:Stream s
  field Int flags
  field Int read_remain ; field CBool read_eof
  field Int read_counter write_counter
  field CBool browser
StreamDriver maybe ChunkedStreamDriver


function unhexa s -> i
  arg Str s ; arg Int i
  i := 0
  for (var Int j) 0 s:len-1
    var Int c := s:j number
    if c>="0":0:number and c<="9":0:number
      i := i*16+(c-"0":0:number)
    eif c>="A":0:number and c<="F":0:number
      i := i*16+(c-"A":0:number+10)
    eif c>="a":0:number and c<="f":0:number
      i := i*16+(c-"a":0:number+10)
    eif c=" ":number
      return
    else
      return undefined

method drv read buf mini maxi -> red
  oarg_rw ChunkedStreamDriver drv ; arg Address buf ; arg Int mini maxi red
  red := 0
  while red<mini
    if drv:read_eof
      return
    eif drv:read_remain>0
      drv:s read_available (var Address adr) (var Int size) (min maxi-red drv:read_remain)
      memory_copy adr (buf translate Byte red) size ; red += size ; drv read_remain -= size
      drv read_counter += size
    else
    eif drv:browser
      var Str l := drv:s readline
      if not (l parse word:"b" drv:read_remain)
        drv:read_remain=undefined
        drv:s unreadline l
    else # HTTP chunking
      if drv:read_remain<>undefined
        drv:s readline
      drv read_remain := unhexa drv:s:readline
      if drv:read_remain=0
        drv:s readline
      if drv:read_remain<=0
        drv read_eof := true


method drv write buf mini maxi -> written
  oarg_rw ChunkedStreamDriver drv ; arg Address buf ; arg Int mini maxi written
  drv:s writechars (string maxi "radix 16")+"[cr][lf]"
  drv:s raw_write buf maxi
  drv:s writechars "[cr][lf]"
  if drv:browser
    drv:s writechars "b "+string:maxi+"[lf]"
    drv:s raw_write buf maxi
  else
    drv:s writechars (string maxi "radix 16")+"[cr][lf]"
    drv:s raw_write buf maxi
    drv:s writechars "[cr][lf]"
  drv write_counter += maxi
  written := shunt drv:s=success maxi 0


method drv flush level -> status
  oarg_rw ChunkedStreamDriver drv ; arg Int level ; arg Status status
  drv:s flush level
  status := shunt drv:s=success success failure


method drv close -> status
  oarg_rw ChunkedStreamDriver drv ; arg ExtendedStatus status
  if (drv:flags .and. out)<>0
    drv:s writechars "0[cr][lf]"
    drv:s writechars "[cr][lf]"
    if drv:browser
      void
    else
      drv:s writechars "0[cr][lf]"
      drv:s writechars "[cr][lf]"
  status := success


method drv query command stream answer -> status
  oarg_rw ChunkedStreamDriver drv ; arg Str command ; arg_rw Stream stream ; arg_w Str answer ; arg ExtendedStatus status 
  if command="read_counter"
    answer := string drv:read_counter
    status := success
  eif command="write_counter"
    answer := string drv:write_counter
    status := success
  else
    status := drv:s:stream_driver query command drv:s answer


method drv configure command stream -> status
  oarg_rw ChunkedStreamDriver drv ; arg Str command ; arg_rw Stream stream ; arg ExtendedStatus status
  if (command parse word:"read_counter" (var Int counter))
    drv read_counter := counter
    status := success
  eif (command parse word:"write_counter" (var Int counter))
    drv write_counter := counter
    status := success
  else
    status := drv:s:stream_driver configure command drv:s


#----------------------------------------------------------------


type ChunkedFileSystem
  void
FileSystem maybe ChunkedFileSystem


method fs open name options flags stream support -> status
  oarg_rw ChunkedFileSystem fs ; arg Str name options ; arg Int flags ; arg_rw Stream stream support ; arg ExtendedStatus status
  var Link:Stream s
  if exists:support
    s :> support
  else
    var Link:Stream s :> new Stream
    s open name options (flags .and. in+out+safe)
    if s=failure
      return failure
  var Link:ChunkedStreamDriver drv :> new ChunkedStreamDriver
  drv s :> s
  drv flags := flags
  drv read_remain := undefined ; drv read_eof := false
  drv read_counter := 0
  drv write_counter := 0
  drv browser := options option "pliant_browser_chunking"
  stream stream_driver :> drv
  status := success


gvar ChunkedFileSystem chunked_file_system
pliant_multi_file_system mount "chunked:" "" chunked_file_system