Как получить файл из POST data в скрипте Bash CGI?
Я пытаюсь опубликовать файл с помощью cURL и получить его с другой стороны через скрипт CGI Bash и сохранить его с тем же именем. После завершения загрузки, diff
между исходным файлом и восстановленным должен возвращаться ноль.
способ 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
но это не работает. Файл не создается на стороне сервера. И как я могу получить имя файла?
2 ответов
пока вы всегда загружаете ровно 1 файл, используя multipart/form-data
формат и клиент всегда ставят Content-Type
как последний заголовок загрузки файла (который, похоже, всегда делает 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?)
пути curl отправляет данные
С полями:
curl --data "param1=value1¶m2=value2" https://example.com/resource.cgi
с полями, указанными индивидуально:
curl --data "param1=value1" --data "param2=value2" https://example.com/resource.cgi
Multipart:
curl --form "fileupload=@my-file.txt" https://example.com/resource.cgi
Multipart с полями и именем файла:
curl --form "fileupload=@my-file.txt;filename=desired-filename.txt" --form param1=value1 --form param2=value2 https://example.com/resource.cgi
для спокойного HTTP-сообщения, содержащего XML:
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 и отправить его как сообщение запрос.
дополнительная информация
чтение данных HTTP POST с помощью BASH
Как получить длину содержимого:
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, а имя файла в именем переменной.