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


method http write_header size tcp log -> status
  arg_rw HttpStreamDriver http ; arg Intn size ; arg_rw Stre
# Copyright  Hubert Tonneau  hubert.tonneau@pliant.cx
#
# This program is free software; you can redistribute it and
# modify it under the terms of the GNU General Public Licens
# as published by the Free Software Foundation.
#
# This program is distributed in the hope that it will be us
# but WITHOUT ANY WARRANTY; without even the implied warrant
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See 
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public 
# version 2 along with this program; if not, write to the Fr
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 


method http write_header size tcp log -> status
  arg_rw HttpStreamDriver http ; arg Intn size ; arg_rw Stre
  tcp writeline "PUT /"+http:path+" HTTP/1.0" log
  var Str option := http:options option "option" Str
  tcp writeline "PUT /"+http:path+(shunt option<>"" "?"+option "")+" HTTP/1.0" log
  tcp writeline "Host: "+http:server log
  tcp writeline "Content-Length: "+string:size log
  if http:datetime=defined
    tcp writeline "Last-Modified: "+(rfc1123_date http:datet
  var Str p_opt := ""
  if (http:options option "mode")
    p_opt += "mode "+string:(http:options option "mode" Int)
  if (http:options option "uid")
    p_opt += " uid "+string:(http:options option "uid" Int)
  if (http:options option "gid")
    p_opt += " gid "+string:(http:options option "gid" Int)
  if p_opt<>""
    tcp writeline "Pliant-Options: "+http_encode:p_opt log
  write_http_options "" http:options tcp log
  if http:initial_ack
    if debug
      console "wait for initial ack" eol
    var Str answer := tcp readline log
    if not (answer parse "HTTP/" (var Float protocol_level) 
      return (failure "HTTP server returned initial error "+
    if debug
      console "wait for initial options" eol
    while { var Str l := tcp readline log ; l<>"" }
      void
    if debug
      console "got initial answer" eol
  status := success


method fs list epath options flags files -> supported_flags
  oarg_rw HttpFileSystem fs ; arg Str epath options ; arg In
  supported_flags := extended
  if (epath eparse "//" any:(var Str server) "/" any:(var St
    void
  else
    return
  tcp writeline "Host: "+http:server log
  tcp writeline "Content-Length: "+string:size log
  if http:datetime=defined
    tcp writeline "Last-Modified: "+(rfc1123_date http:datet
  var Str p_opt := ""
  if (http:options option "mode")
    p_opt += "mode "+string:(http:options option "mode" Int)
  if (http:options option "uid")
    p_opt += " uid "+string:(http:options option "uid" Int)
  if (http:options option "gid")
    p_opt += " gid "+string:(http:options option "gid" Int)
  if p_opt<>""
    tcp writeline "Pliant-Options: "+http_encode:p_opt log
  write_http_options "" http:options tcp log
  if http:initial_ack
    if debug
      console "wait for initial ack" eol
    var Str answer := tcp readline log
    if not (answer parse "HTTP/" (var Float protocol_level) 
      return (failure "HTTP server returned initial error "+
    if debug
      console "wait for initial options" eol
    while { var Str l := tcp readline log ; l<>"" }
      void
    if debug
      console "got initial answer" eol
  status := success


method fs list epath options flags files -> supported_flags
  oarg_rw HttpFileSystem fs ; arg Str epath options ; arg In
  supported_flags := extended
  if (epath eparse "//" any:(var Str server) "/" any:(var St
    void
  else
    return
  if not (options option "no_http_encode")
    path := http_encode path
  var Int port
  if (server eparse any:(var Str server1) ":" port)
    server := server1
  else
    port := 80
  var Str channel := options option "channel" Str
  if channel=""
    channel := "tcp://"+server+"/client/"+string:port
  var Link:Stream tcp :> new Stream
  tcp open channel in+out+safe+cr+lf
  if tcp=failure
    http_trace trace "Failed to connect to " channel
    return failure
  (var TraceSession log) bind http_trace
  tcp writeline "GET /"+path+"?list HTTP/1.0" log
  write_http_options server options tcp log
  if not ((tcp readline log) parse "HTTP/" (var Float protoc
    return
  while { var Str l := tcp readline log ; l<>"" }
    void
  while not tcp:atend
    var Str l := tcp readline
    if (l eparse "<pfile name=[dq]" any:(var Str name) "[dq]
      var Link:FileInfo info :> new FileInfo 
      info name := epath+html_decode:name
      info size := size
      info datetime := dt
      info options := html_decode:opt
      files append addressof:info
      log trace "list" string:name " " size " " datetime " "


method fs open name options flags stream support -> status
  arg_rw HttpFileSystem fs ; arg Str name options ; arg Int 
  if (name eparse "//" any:(var Str server) "/" any:(var Str
    void
  else
    return failure
  var Int port
  if (server eparse any:(var Str server1) ":" port)
    server := server1
  else
    port := 80
  var Str channel := options option "channel" Str
  if channel=""
    channel := "tcp://"+server+"/client/"+string:port
  var Link:Stream tcp :> new Stream
  tcp open channel in+out+safe+cr+lf
  if tcp=failure
    http_trace trace "Failed to connect to " channel
    return failure
  (var TraceSession log) bind http_trace
  tcp writeline "GET /"+path+"?list HTTP/1.0" log
  write_http_options server options tcp log
  if not ((tcp readline log) parse "HTTP/" (var Float protoc
    return
  while { var Str l := tcp readline log ; l<>"" }
    void
  while not tcp:atend
    var Str l := tcp readline
    if (l eparse "<pfile name=[dq]" any:(var Str name) "[dq]
      var Link:FileInfo info :> new FileInfo 
      info name := epath+html_decode:name
      info size := size
      info datetime := dt
      info options := html_decode:opt
      files append addressof:info
      log trace "list" string:name " " size " " datetime " "


method fs open name options flags stream support -> status
  arg_rw HttpFileSystem fs ; arg Str name options ; arg Int 
  if (name eparse "//" any:(var Str server) "/" any:(var Str
    void
  else
    return failure
  if not (options option "no_http_encode")
    path := http_encode path
  var Int port
  if (server eparse any:(var Str server1) ":" port)
    server := server1
  else
    port := 80
  var Str channel := options option "channel" Str
  if channel=""
    channel := "tcp://"+server+"/client/"+string:port
  var Link:Stream tcp :> new Stream
  tcp open channel in+out+safe+cr+lf
  if tcp=failure
    http_trace trace "Failed to connect to " channel
    return (failure "Failed to connect to "+string:server+" 
  var Link:HttpStreamDriver http :> new HttpStreamDriver
  http tcp :> tcp
  http direct := false
  http initial_ack := options option "http_initial_ack"
  http server := server+(shunt port=80 "" ":"+string:port)
  http path := path
  http datetime := options option "datetime" DateTime
  http options := options
  http opened_on := datetime
  http:log bind http_trace
  var Str msg := ""
  if (flags .and. in_out)=in
    var CBool post := options option "form"
  var Int port
  if (server eparse any:(var Str server1) ":" port)
    server := server1
  else
    port := 80
  var Str channel := options option "channel" Str
  if channel=""
    channel := "tcp://"+server+"/client/"+string:port
  var Link:Stream tcp :> new Stream
  tcp open channel in+out+safe+cr+lf
  if tcp=failure
    http_trace trace "Failed to connect to " channel
    return (failure "Failed to connect to "+string:server+" 
  var Link:HttpStreamDriver http :> new HttpStreamDriver
  http tcp :> tcp
  http direct := false
  http initial_ack := options option "http_initial_ack"
  http server := server+(shunt port=80 "" ":"+string:port)
  http path := path
  http datetime := options option "datetime" DateTime
  http options := options
  http opened_on := datetime
  http:log bind http_trace
  var Str msg := ""
  if (flags .and. in_out)=in
    var CBool post := options option "form"
    tcp writeline (shunt post "POST" "GET")+" /"+path+" HTTP
    var Str option := options option "option" Str
    tcp writeline (shunt post "POST" "GET")+" /"+path+(shunt option<>"" "?"+option "")+" HTTP/1.0" http:log
    write_http_options server options tcp http:log
    if post
      tcp writechars (options option "form" Str)
    var Str answer := tcp readline http:log
    if not (answer parse "HTTP/" (var Float protocol_level) 
      return (failure "Unexpected answer "+string:answer+" f
    msg := "http_return_code "+string:retcode
    while { var Str l := tcp readline http:log ; l<>"" }
      msg += " http_answer_option "+string:l
      void
    if retcode>=300
      return failure:msg
  eif (flags .and. in_out)=out
    var Intn size := options option "file_size" Intn
    if size<>undefined
      if debug
        console "upload on the fly "+options eol
      http direct := true
      http remain := size
      status := http write_header size tcp http:log
      if status=failure
        return
    else
      if debug
        console "store then upload" eol
      http filename := file_temporary
      http temp :> new Stream
      http:temp open http:filename out+(flags .and. safe)
  else
    return failure
  stream stream_driver :> http
  status := success ; status message := msg
    
method fs configure name options command -> status
  arg_rw HttpFileSystem fs ; arg Str name options command ; 
  if command="delete"
    if (name eparse "//" any:(var Str server) "/" any:(var S
      void
    else
      return failure
    var Int port
    if (server eparse any:(var Str server1) ":" port)
      server := server1
    else
      port := 80
    var Str channel := options option "channel" Str
    if channel=""
      channel := "tcp://"+server+"/client/"+string:port
    (var Stream tcp) open channel in+out+safe+cr+lf
    if tcp=failure
      http_trace trace "Failed to connect to " channel
      return failure
    (var TraceSession log) bind http_trace
    tcp writeline "DELETE /"+path+" HTTP/1.0" log
    write_http_options server options tcp log
    var Str answer := tcp readline log
    status := shunt (answer parse "HTTP/" (var Float protoco
  else
    status := failure


    write_http_options server options tcp http:log
    if post
      tcp writechars (options option "form" Str)
    var Str answer := tcp readline http:log
    if not (answer parse "HTTP/" (var Float protocol_level) 
      return (failure "Unexpected answer "+string:answer+" f
    msg := "http_return_code "+string:retcode
    while { var Str l := tcp readline http:log ; l<>"" }
      msg += " http_answer_option "+string:l
      void
    if retcode>=300
      return failure:msg
  eif (flags .and. in_out)=out
    var Intn size := options option "file_size" Intn
    if size<>undefined
      if debug
        console "upload on the fly "+options eol
      http direct := true
      http remain := size
      status := http write_header size tcp http:log
      if status=failure
        return
    else
      if debug
        console "store then upload" eol
      http filename := file_temporary
      http temp :> new Stream
      http:temp open http:filename out+(flags .and. safe)
  else
    return failure
  stream stream_driver :> http
  status := success ; status message := msg
    
method fs configure name options command -> status
  arg_rw HttpFileSystem fs ; arg Str name options command ; 
  if command="delete"
    if (name eparse "//" any:(var Str server) "/" any:(var S
      void
    else
      return failure
    var Int port
    if (server eparse any:(var Str server1) ":" port)
      server := server1
    else
      port := 80
    var Str channel := options option "channel" Str
    if channel=""
      channel := "tcp://"+server+"/client/"+string:port
    (var Stream tcp) open channel in+out+safe+cr+lf
    if tcp=failure
      http_trace trace "Failed to connect to " channel
      return failure
    (var TraceSession log) bind http_trace
    tcp writeline "DELETE /"+path+" HTTP/1.0" log
    write_http_options server options tcp log
    var Str answer := tcp readline log
    status := shunt (answer parse "HTTP/" (var Float protoco
  else
    status := failure