Używanie curl do przesyłania danych POST z plikami

414

Chciałbym użyć cURL nie tylko do wysyłania parametrów danych w HTTP POST, ale także do przesyłania plików o określonej nazwie formularza. Jak mam to zrobić?

Parametry HTTP Post:

userid = 12345 filecomment = To jest plik obrazu

Przesyłanie pliku HTTP: Lokalizacja pliku = /home/user1/Desktop/test.jpg Nazwa formularza dla pliku = obraz (odpowiada $ _FILES ['obraz'] po stronie PHP)

Wymyśliłem część polecenia cURL w następujący sposób:

curl -d "userid=1&filecomment=This is an image file" --data-binary @"/home/user1/Desktop/test.jpg" localhost/uploader.php

Problem, który otrzymuję, jest następujący:

Notice: Undefined index: image in /var/www/uploader.php

Problem polega na tym, że używam $ _FILES ['image'] do pobierania plików w skrypcie PHP.

Jak odpowiednio dostosować polecenia CURL?

thotheolh
źródło

Odpowiedzi:

655

Musisz użyć -Fopcji:
-F/--form <name=content> Specify HTTP multipart POST data (H)

Spróbuj tego:

curl \
  -F "userid=1" \
  -F "filecomment=This is an image file" \
  -F "image=@/home/user1/Desktop/test.jpg" \
  localhost/uploader.php
jimp
źródło
1
Jestem zdezorientowany tą częścią dotyczącą kodowania adresu URL pliku. Przesłałem takie pliki JPG i PNG bez modyfikacji, bez żadnych problemów.
Deanna Gelbart
1
@DavidGelbart Masz rację. Moja pierwsza odpowiedź -domyłkowo odniosła się do opcji, która wymaga wejściowego zakodowanego adresu URL. Powinienem to usunąć, kiedy zaktualizowałem odpowiedź na -Fopcję. Dzięki za złapanie tego.
jimp
3
@ user956424 W tym przykładzie ustaw „image” na nazwę swojego pola. A niektóre języki, takie jak PHP, zbudują tablicę, jeśli określisz coś takiego jak „image []” dla danych wejściowych, które należy zgrupować.
jimp
1
Co jest @w image=@/..środku?
Timo,
2
@ Timo Oznacza to, że zawartość nazwanego pola formularza powinna zostać załadowana ze ścieżki pliku. Bez niego sam argument ciągu jest przekazywany.
jimp
93

Łapanie identyfikatora użytkownika jako zmiennej ścieżki (zalecane):

curl -i -X POST -H "Content-Type: multipart/form-data" 
-F "[email protected]" http://mysuperserver/media/1234/upload/

Łapanie identyfikatora użytkownika jako części formularza:

curl -i -X POST -H "Content-Type: multipart/form-data" 
-F "[email protected];userid=1234" http://mysuperserver/media/upload/

lub:

curl -i -X POST -H "Content-Type: multipart/form-data" 
-F "[email protected]" -F "userid=1234" http://mysuperserver/media/upload/
r1ckr
źródło
16
użyj -F nie trzeba ustawiać"Content-Type: multipart/form-data"
William Hu
10
Nie mogłem sprawić, aby -F działał poprawnie z tym separatorem średników, który wskazałeś. Zamiast tego musiałem podać dwa zbędne argumenty -F. Jak: -F "[email protected]" -F "userid = 1234"
robbpriestley,
22

Oto moje rozwiązanie, czytałem wiele postów i były one bardzo pomocne. W końcu napisałem trochę kodu dla małych plików, z CURL i PHP, które moim zdaniem są naprawdę przydatne.

public function postFile()
{    
        $file_url = "test.txt";  //here is the file route, in this case is on same directory but you can set URL too like "http://examplewebsite.com/test.txt"
        $eol = "\r\n"; //default line-break for mime type
        $BOUNDARY = md5(time()); //random boundaryid, is a separator for each param on my post curl function
        $BODY=""; //init my curl body
        $BODY.= '--'.$BOUNDARY. $eol; //start param header
        $BODY .= 'Content-Disposition: form-data; name="sometext"' . $eol . $eol; // last Content with 2 $eol, in this case is only 1 content.
        $BODY .= "Some Data" . $eol;//param data in this case is a simple post data and 1 $eol for the end of the data
        $BODY.= '--'.$BOUNDARY. $eol; // start 2nd param,
        $BODY.= 'Content-Disposition: form-data; name="somefile"; filename="test.txt"'. $eol ; //first Content data for post file, remember you only put 1 when you are going to add more Contents, and 2 on the last, to close the Content Instance
        $BODY.= 'Content-Type: application/octet-stream' . $eol; //Same before row
        $BODY.= 'Content-Transfer-Encoding: base64' . $eol . $eol; // we put the last Content and 2 $eol,
        $BODY.= chunk_split(base64_encode(file_get_contents($file_url))) . $eol; // we write the Base64 File Content and the $eol to finish the data,
        $BODY.= '--'.$BOUNDARY .'--' . $eol. $eol; // we close the param and the post width "--" and 2 $eol at the end of our boundary header.



        $ch = curl_init(); //init curl
        curl_setopt($ch, CURLOPT_HTTPHEADER, array(
                         'X_PARAM_TOKEN : 71e2cb8b-42b7-4bf0-b2e8-53fbd2f578f9' //custom header for my api validation you can get it from $_SERVER["HTTP_X_PARAM_TOKEN"] variable
                         ,"Content-Type: multipart/form-data; boundary=".$BOUNDARY) //setting our mime type for make it work on $_FILE variable
                    );
        curl_setopt($ch, CURLOPT_USERAGENT, 'Mozilla/1.0 (Windows NT 6.1; WOW64; rv:28.0) Gecko/20100101 Firefox/28.0'); //setting our user agent
        curl_setopt($ch, CURLOPT_URL, "api.endpoint.post"); //setting our api post url
        curl_setopt($ch, CURLOPT_COOKIEJAR, $BOUNDARY.'.txt'); //saving cookies just in case we want
        curl_setopt ($ch, CURLOPT_RETURNTRANSFER, 1); // call return content
        curl_setopt ($ch, CURLOPT_FOLLOWLOCATION, 1); navigate the endpoint
        curl_setopt($ch, CURLOPT_POST, true); //set as post
        curl_setopt($ch, CURLOPT_POSTFIELDS, $BODY); // set our $BODY 


        $response = curl_exec($ch); // start curl navigation

     print_r($response); //print response

}

Dzięki temu powinniśmy dostać się na „api.endpoint.post” zamieszczone następujące vars. Możesz łatwo przetestować za pomocą tego skryptu i powinieneś otrzymać ten debugowanie funkcji postFile()w ostatnim wierszu.

print_r($response); //print response

public function getPostFile()
{

    echo "\n\n_SERVER\n";
    echo "<pre>";
    print_r($_SERVER['HTTP_X_PARAM_TOKEN']);
    echo "/<pre>";
    echo "_POST\n";
    echo "<pre>";
    print_r($_POST['sometext']);
    echo "/<pre>";
    echo "_FILES\n";
    echo "<pre>";
    print_r($_FILEST['somefile']);
    echo "/<pre>";
}

Powinno działać dobrze, mogą to być lepsze rozwiązania, ale to działa i jest naprawdę pomocne, aby zrozumieć, jak mime Boundary i multipart / from-data działa na bibliotece PHP i cURL.

Libertański
źródło
jeśli chcesz wysłać plik niezakodowany, zmień te wiersze $ BODY. = 'Content-Transfer-Encoding: multipart / form-data'. $ eol. $ eol; // wstawiamy ostatnią zawartość i 2 $ eol, $ BODY. = file_get_contents ($ file_url). $ eol; // piszemy zawartość pliku Base64 i $ eol, aby dokończyć dane,
andreah
8

jeśli przesyłasz plik binarny, taki jak csv, użyj poniższego formatu, aby przesłać plik

curl -X POST \
    'http://localhost:8080/workers' \
    -H 'authorization: eyJhbGciOiJIUzI1NiIsInR5cCI6ImFjY2VzcyIsInR5cGUiOiJhY2Nlc3MifQ.eyJ1c2VySWQiOjEsImFjY291bnRJZCI6MSwiaWF0IjoxNTExMzMwMzg5LCJleHAiOjE1MTM5MjIzODksImF1ZCI6Imh0dHBzOi8veW91cmRvbWFpbi5jb20iLCJpc3MiOiJmZWF0aGVycyIsInN1YiI6ImFub255bW91cyJ9.HWk7qJ0uK6SEi8qSeeB6-TGslDlZOTpG51U6kVi8nYc' \
    -H 'content-type: application/x-www-form-urlencoded' \
    --data-binary '@/home/limitless/Downloads/iRoute Masters - Workers.csv'
KARTHIKEYAN.A
źródło
4
Chciałbym zobaczyć przykład binarnego pliku csv.
polis
4
@polis opcja --data-binarynakazuje, curlaby nie przeprowadzać wstępnego przetwarzania danych, w przeciwieństwie do --dataflagi. aby bezpośrednio odnieść się do komentarza, pamiętaj, że tekst jest również binarny, ale możemy interpretować go jako znaki ASCII. Jeśli naprawdę chcesz wyraźnego przykładu, pomyśl o pliku CSV, którego pola zawierają emoji. Ich bajty nie są bezpośrednio mapowane na tekst
Ciprian Tomoiagă
3

Problem, który mnie tu doprowadził, okazał się podstawowym błędem użytkownika - nie wpisałem @znaku do ścieżki pliku, więc curl publikował ścieżkę / nazwę pliku, a nie jego zawartość. W Content-Lengthzwiązku z tym wartość wynosiła 8 zamiast 479, której spodziewałem się zobaczyć, biorąc pod uwagę legnth mojego pliku testowego.

Content-LengthNagłówek zostanie automatycznie obliczana po curl odczytuje i posty plik.

curl -i -H "Content-Type: application/xml" --data "@test.xml" -v -X POST https://<url>/<uri/

... <Content-Length: 479 ...

Opublikowanie tego tutaj, aby pomóc innym początkującym.

shonky użytkownik Linux
źródło
2

Jako alternatywę curlmożesz użyć HTTPie , it'a CLI, narzędzia podobnego do cURL dla ludzi.

  1. Instrukcje instalacji: https://github.com/jakubroztocil/httpie#installation

  2. Następnie uruchomić:

    http -f POST http://localhost:4040/api/users username=johnsnow photo@images/avatar.jpg
    
    HTTP/1.1 200 OK
    Access-Control-Expose-Headers: X-Frontend
    Cache-control: no-store
    Connection: keep-alive
    Content-Encoding: gzip
    Content-Length: 89
    Content-Type: text/html; charset=windows-1251
    Date: Tue, 26 Jun 2018 11:11:55 GMT
    Pragma: no-cache
    Server: Apache
    Vary: Accept-Encoding
    X-Frontend: front623311
    
    ...
Gianfranco P.
źródło
2

Po wielu próbach to polecenie zadziałało dla mnie:

curl -v -F filename=image.jpg -F upload=@image.jpg http://localhost:8080/api/upload
Evandro Pomatti
źródło
1

Oto, jak poprawnie uciec przed dowolnymi nazwami przesyłanych plików za pomocą bash:

#!/bin/bash
set -eu

f="$1"
f=${f//\\/\\\\}
f=${f//\"/\\\"}
f=${f//;/\\;}

curl --silent --form "uploaded=@\"$f\"" "$2"
Vladimir Panteleev
źródło
0

Mam to działa z tym poleceniem curl -F 'filename=@/home/yourhomedirextory/file.txt' http://yourserver/upload

Shraavan Hebbar
źródło