如何从 Bash CGI 脚本中的 POST 数据获取文件?
How to get file from POST data in Bash CGI script?
我正在尝试 post 使用 cURL 的文件并通过 CGI Bash 脚本在另一端接收它并以相同的名称存储它。上传完成后,原文件和重构文件之间的diff
应该return归零。
cURL发送数据的方式:
curl --request POST --data-binary "@dummy.dat" 127.0.0.1/cgi-bin/upload-rpm
接收脚本:
#!/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
但它不起作用。文件不是在服务器端创建的。我怎样才能得到文件名?
curl 发送数据的方式
有字段:
curl --data "param1=value1¶m2=value2" https://example.com/resource.cgi
单独指定的字段:
curl --data "param1=value1" --data "param2=value2" https://example.com/resource.cgi
多部分:
curl --form "fileupload=@my-file.txt" https://example.com/resource.cgi
包含字段和文件名的多部分:
curl --form "fileupload=@my-file.txt;filename=desired-filename.txt" --form param1=value1 --form param2=value2 https://example.com/resource.cgi
对于包含 XML:
的 RESTful HTTP POST
curl -X POST -d @filename.txt http://example.com/path/to/resource --header "Content-Type:text/xml"
或 JSON,使用这个:
curl -X POST -d @filename.txt http://example.com/path/to/resource --header "Content-Type:application/json"
这将读取名为 filename.txt 的文件的内容并将其作为 post 请求发送。
有关详细信息,请参阅
使用 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
要保存二进制文件或文本文件:
boundary=$(export | \
sed '/CONTENT_TYPE/!d;s/^.*dary=//;s/.$//')
filename=$(echo "$QUERY_STRING" | \
sed -n '2!d;s/\(.*filename=\"\)\(.*\)\".*$//;p')
file=$(echo "$QUERY_STRING" | \
sed -n "1,/$boundary/p" | sed '1,4d;$d')
上传的文件现在包含在 $file 变量中,文件名包含在 $filename 变量中。
只要您始终使用 multipart/form-data
格式上传 1 个文件,并且客户端始终将 Content-Type
作为文件上传的最后一个 header(curl似乎总是如此),这似乎有效:
#!/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
示例上传调用(我用来调试上面的):
curl http://localhost:81/cgi-bin/wtf.sh -F "MyFile=@myfile.txt"
- 综上所述,这太疯狂了! bash绝对不适合处理二进制数据,比如http文件上传,用别的吧。 (PHP?Python?)
我正在尝试 post 使用 cURL 的文件并通过 CGI Bash 脚本在另一端接收它并以相同的名称存储它。上传完成后,原文件和重构文件之间的diff
应该return归零。
cURL发送数据的方式:
curl --request POST --data-binary "@dummy.dat" 127.0.0.1/cgi-bin/upload-rpm
接收脚本:
#!/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
但它不起作用。文件不是在服务器端创建的。我怎样才能得到文件名?
curl 发送数据的方式
有字段:
curl --data "param1=value1¶m2=value2" https://example.com/resource.cgi
单独指定的字段:
curl --data "param1=value1" --data "param2=value2" https://example.com/resource.cgi
多部分:
curl --form "fileupload=@my-file.txt" https://example.com/resource.cgi
包含字段和文件名的多部分:
curl --form "fileupload=@my-file.txt;filename=desired-filename.txt" --form param1=value1 --form param2=value2 https://example.com/resource.cgi
对于包含 XML:
的 RESTful HTTP POSTcurl -X POST -d @filename.txt http://example.com/path/to/resource --header "Content-Type:text/xml"
或 JSON,使用这个:
curl -X POST -d @filename.txt http://example.com/path/to/resource --header "Content-Type:application/json"
这将读取名为 filename.txt 的文件的内容并将其作为 post 请求发送。
有关详细信息,请参阅
使用 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
要保存二进制文件或文本文件:
boundary=$(export | \
sed '/CONTENT_TYPE/!d;s/^.*dary=//;s/.$//')
filename=$(echo "$QUERY_STRING" | \
sed -n '2!d;s/\(.*filename=\"\)\(.*\)\".*$//;p')
file=$(echo "$QUERY_STRING" | \
sed -n "1,/$boundary/p" | sed '1,4d;$d')
上传的文件现在包含在 $file 变量中,文件名包含在 $filename 变量中。
只要您始终使用 multipart/form-data
格式上传 1 个文件,并且客户端始终将 Content-Type
作为文件上传的最后一个 header(curl似乎总是如此),这似乎有效:
#!/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
示例上传调用(我用来调试上面的):
curl http://localhost:81/cgi-bin/wtf.sh -F "MyFile=@myfile.txt"
- 综上所述,这太疯狂了! bash绝对不适合处理二进制数据,比如http文件上传,用别的吧。 (PHP?Python?)