如何从Bash CGI脚本中的POST数据中获取文件?

sor*_*h-r 14 bash curl cgi

我试图使用cURL发布文件,并通过CGI Bash脚本在另一侧接收它并使用相同的名称存储它.上传完成后,diff原始文件和重建文件之间应返回零.

cURL发送数据的方式:

curl --request POST --data-binary "@dummy.dat" 127.0.0.1/cgi-bin/upload-rpm
Run Code Online (Sandbox Code Playgroud)

接收器脚本:

#!/bin/bash

echo "Content-type: text/html"
echo ""

echo '<html>'
echo '<head>'
echo '<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">'
echo '<title>Foo</title>'
echo '</head>'
echo '<body>'

echo "<p>Start</p>"

if [ "$REQUEST_METHOD" = "POST" ]; then
    echo "<p>Post Method</p>"
    if [ "$CONTENT_LENGTH" -gt 0 ]; then
        echo "<p>Size=$CONTENT_LENGTH</p>"
        while read -n 1 byte -t 3; do echo -n -e "$byte" >> ./foo.dat ; done
    fi
fi
echo '</body>'
echo '</html>'

exit 0
Run Code Online (Sandbox Code Playgroud)

但它不起作用.文件不是在服务器端创建的.我怎样才能获得文件名?

han*_*rik 8

只要你总是使用multipart/form-data格式正好上传1个文件,并且客户端总是把它Content-Type作为文件上传的最后一个标题(似乎总是这样做),这似乎有效:

#!/bin/bash

echo "Content-type: text/html"
echo ""

echo '<html>'
echo '<head>'
echo '<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">'
echo '<title>Foo</title>'
echo '</head>'
echo '<body>'

echo "<p>Start</p>"

if [ "$REQUEST_METHOD" = "POST" ]; then
    echo "<p>Post Method</p>"
    if [ "$CONTENT_LENGTH" -gt 0 ]; then
    in_raw=`cat`
    boundary=$(echo -n "$in_raw" | head -1 | tr -d '\r\n');
    filename=$(echo -n "$in_raw" | grep --text --max-count=1 -oP "(?<=filename=\")[^\"]*");
    file_content=$(echo -n "$in_raw" | sed '1,/Content-Type:/d' | tail -c +3 | head --lines=-1 | head --bytes=-4  );
    echo "boundary: $boundary"
    echo "filename: $filename"
    #echo "file_content: $file_content"
    fi
fi
echo '</body>'
echo '</html>'

exit 0
Run Code Online (Sandbox Code Playgroud)

示例上传调用(我用来调试上面的):

curl http://localhost:81/cgi-bin/wtf.sh -F "MyFile=@myfile.txt"
Run Code Online (Sandbox Code Playgroud)
  • 尽管如此,这是不可靠的!bash绝对适合处理二进制数据,比如http文件上传,使用别的东西.(PHP?Python?)


Bha*_*ata 7

curl发送数据的方式

有字段:

curl --data "param1=value1&param2=value2" https://example.com/resource.cgi
Run Code Online (Sandbox Code Playgroud)

使用单独指定的字段:

curl --data "param1=value1" --data "param2=value2" https://example.com/resource.cgi
Run Code Online (Sandbox Code Playgroud)

多:

curl --form "fileupload=@my-file.txt" https://example.com/resource.cgi
Run Code Online (Sandbox Code Playgroud)

包含字段和文件名的多部分:

curl --form "fileupload=@my-file.txt;filename=desired-filename.txt" --form param1=value1 --form param2=value2 https://example.com/resource.cgi
Run Code Online (Sandbox Code Playgroud)

对于包含XML的RESTful HTTP POST:

curl -X POST -d @filename.txt http://example.com/path/to/resource --header "Content-Type:text/xml"
Run Code Online (Sandbox Code Playgroud)

或者对于JSON,使用此:

curl -X POST -d @filename.txt http://example.com/path/to/resource --header "Content-Type:application/json"
Run Code Online (Sandbox Code Playgroud)

这将读取名为filename.txt的文件的内容并将其作为发布请求发送.

有关详细信息,请参阅

使用BASH读取HTTP POST数据

如何获取内容长度:

if [ "$REQUEST_METHOD" = "POST" ]; then
    if [ "$CONTENT_LENGTH" -gt 0 ]; then
        read -n $CONTENT_LENGTH POST_DATA <&0
        echo "$CONTENT_LENGTH"
    fi
fi
Run Code Online (Sandbox Code Playgroud)

要保存二进制文件或文本文件:

boundary=$(export | \
    sed '/CONTENT_TYPE/!d;s/^.*dary=//;s/.$//')
filename=$(echo "$QUERY_STRING" | \
    sed -n '2!d;s/\(.*filename=\"\)\(.*\)\".*$/\2/;p')
file=$(echo "$QUERY_STRING" | \
    sed -n "1,/$boundary/p" | sed '1,4d;$d')
Run Code Online (Sandbox Code Playgroud)

上传的文件现在包含在$ file变量中,文件名包含在$ filename变量中.