File HTTPS Upload

This is an example of uploading a file from a client to a server using https (SSL) through a squid proxy running on another host. The website is using a self-signed SSL certificate.

Server code in file upload.cgi:

#! /bin/sh
# \
exec tclsh $0 $@

set DIR /server_path_to/upload_directory

package require ncgi ;# in tcllib

# for message back to browser
puts "Content-type: text/html
<HTML>
 <HEAD>
  <TITLE>POST File</TITLE>
 </HEAD>
 <BODY>"

::ncgi::parse
set filedata [::ncgi::value file2upload]
set filedesc [::ncgi::value name]

if {[catch {open "$DIR/$filedesc" "w"} fh]} {
   puts "Failed - $fh"
} else {
   puts -nonewline $fh "$filedata"
   puts "<BR>File uploaded to $filedesc"
   close $fh
}

puts " </BODY>
</HTML>"

Here is an html file, form.htm, for testing upload.cgi from a web browser running on your client:

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<HTML>
 <HEAD>
    <META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=iso-8859-1">
    <TITLE>Form Test</TITLE>
 </HEAD>

 <BODY>
  <FORM METHOD="post" ACTION="https://www.your-site.org/to_cgi_bin/upload.cgi"
        ENCTYPE="multipart/form-data">
   <INPUT TYPE="text" NAME="name" VALUE="bytesSSL.dat" SIZE="25">
   <INPUT TYPE="file" NAME="file2upload">
   <INPUT TYPE="submit">
  </FORM>
 </BODY>
</HTML>

Once this is working,

Client code, upload.tcl:

#!/bin/sh
# \
exec /usr/bin/tclsh $0 $@

package require http
package require tls
package require base64

set host     www.your-site.org
set cgi      /to_cgi_bin/upload.cgi ;# relative location of script on $host
set cert     /client_path_to/self-signed-certificate.pem
set bound    "--HARD_TO_STUMBLE_ON_0123456789" ;# multipart separator
set file     "junk.dat" ;# client file to be uploaded
set new_name bytesSSL.dat ;# name to give the file on server
set ssl      443 ;# https port
set ssluser  your_ssl_user_name_on_server
set passwd   your_ssl_password_on_server
set up       [::base64::encode "$ssluser:$passwd"] ;# encoded SSL user:password
set phost    proxyhost ;# name or FQDN or numeric
set pport    3128 ;# proxy port

set ifid [open $file "r"]
fconfigure $ifid -translation binary
set data [read $ifid]
close $ifid

append content "--$bound\r\n"
append content "Content-Disposition: form-data; name=\"name\"\r\n"
append content "\r\n"
append content "$new_name\r\n"
append content "--$bound\r\n"
append content "Content-Disposition: form-data; name=\"file2upload\"; "
append content "filename=\"$file\"\r\n" 
append content "Content-Type: application/octet-stream\r\n"
append content "\r\n"
append content "$data\r\n"
append content "--${bound}--\r\n"

set psocket [socket $phost $pport] ;# create socket to proxy
puts $psocket "CONNECT $host:$ssl HTTP/1.1\r\n" ;# tunnel to https://$host
flush $psocket
while {[gets $psocket line] > 0} { ;# look for "200 Connection established"
    ;#puts $line
}
http::register https $ssl \
 [list ::tls::import $psocket -tls1 1 -require 1 -cafile $cert]

# if proxy is not used,
# http::register https $ssl [list ::tls::socket -tls1 1 -require 1 -cafile $cert]

set token [::http::geturl https://${host}$cgi \
 -type "multipart/form-data; boundary=$bound" -query $content \
 -headers [list Authorization "Basic $up"]]
# Depending on your data and platform, you might also need -binary 1 in the geturl command

# if no SSL needed:
# set token [::http::geturl http://${host}$cgi \
# -type "multipart/form-data; boundary=$bound" -query $content

puts "[::http::data $token]"

Test on Linux:

upload.tcl | lynx -stdin