Как мне отправить данные JSON с помощью cURL?


2834

Я использую Ubuntu и установил cURL на нем. Я хочу проверить мое приложение Spring REST с помощью cURL. Я написал свой POST-код на стороне Java. Тем не менее, я хочу проверить это с помощью cURL. Я пытаюсь опубликовать данные в формате JSON. Пример данных выглядит так:

{"value":"30","type":"Tip 3","targetModule":"Target 3","configurationGroup":null,"name":"Configuration Deneme 3","description":null,"identity":"Configuration Deneme 3","version":0,"systemId":3,"active":true}

Я использую эту команду:

curl -i \
    -H "Accept: application/json" \
    -H "X-HTTP-Method-Override: PUT" \
    -X POST -d "value":"30","type":"Tip 3","targetModule":"Target 3","configurationGroup":null,"name":"Configuration Deneme 3","description":null,"identity":"Configuration Deneme 3","version":0,"systemId":3,"active":true \
    http://localhost:8080/xx/xxx/xxxx

Возвращает эту ошибку:

HTTP/1.1 415 Unsupported Media Type
Server: Apache-Coyote/1.1
Content-Type: text/html;charset=utf-8
Content-Length: 1051
Date: Wed, 24 Aug 2011 08:50:17 GMT

Описание ошибки таково:

Сервер отклонил этот запрос, потому что объект запроса находится в формате, который не поддерживается запрошенным ресурсом для запрошенного метода ().

Журнал Tomcat: «POST / ui / webapp / conf / clear HTTP / 1.1» 415 1051

Какой правильный формат команды cURL?

Это мой PUTкод на стороне Java (я проверил GET и DELETE, и они работают):

@RequestMapping(method = RequestMethod.PUT)
public Configuration updateConfiguration(HttpServletResponse response, @RequestBody Configuration configuration) { //consider @Valid tag
    configuration.setName("PUT worked");
    //todo If error occurs response.sendError(HttpServletResponse.SC_NOT_FOUND);
    return configuration;
}

6
Оформить заказ на ссылку весной 3.2.0.
AmirHd

8
Есть хороший пост Использование Curl для специального тестирования RESTful Microservices, который освещает это с несколькими примерами.
upitau

Ответы:


4344

Вам необходимо установить тип контента для application / json. Но -dотправляет Content-Type application/x-www-form-urlencoded, который не принят на стороне Spring.

Глядя на справочную страницу curl , я думаю, что вы можете использовать -H:

-H "Content-Type: application/json"

Полный пример:

curl --header "Content-Type: application/json" \
  --request POST \
  --data '{"username":"xyz","password":"xyz"}' \
  http://localhost:3000/api/login

( -Hсокращенно --header, -dдля --data)

Обратите внимание, что -request POSTэто необязательно, если вы используете -d, так как -dфлаг подразумевает запрос POST.


В Windows все немного по-другому. Смотрите ветку комментариев.


262
Для окон одинарные кавычки вокруг json не работали, и в итоге я избежал двойных кавычек. curl -X POST -H "Content-Type: application/json" -d "{ \"key1\": \"value1\" }" http://localhost:3000/api/method
СЧАСТЛИВЫЙ

37
Для меня под Windows мне нужно было избегать кавычек, используя кавычки в этом формате "{ """key1""": """value1""" }". Также этот ответ: stackoverflow.com/questions/18314796/…
chodorowicz

4
@chodorowicz это ужасно! Я знаю только этот синтаксис из VB!
Шон Патрик Флойд

3
У меня были проблемы с запросами POST, но я решил их с помощью заглавной буквы «Application / json», поэтому, если вы получили ошибку 415, проверьте заглавные буквы.
WiteCastle

5
@ostrokach извините, это потратило впустую ваше время. Синтаксис работал нормально для меня на OSX, когда я опубликовал его (не повторил). Думаю, это было / было просто разница платформы. Я полагаю, что голосование от людей, которым это помогло.
Адам Таттл

565

Попробуйте поместить ваши данные в файл, скажем, body.jsonа затем использовать

curl -H "Content-Type: application/json" --data @body.json http://localhost:8080/ui/webapp/conf

12
Вы, вероятно, должны использовать --data-binaryопцию вместо --data. Можно было бы ожидать, что клиент отправляет данные как есть, но --dataудаляет CR и LF из входных данных.
h2stein

14
Использование cUrl со встроенными json-строками кажется кошмаром. Есть необходимость отбросить символ двойной кавычки. Работать с таким файлом приятнее.
экстремальный байкер

46
Важно добавить @символ перед именем файла, иначе он не будет работать. Я просто потратил 20 минут, стуча головой по этому
дерьму

3
Таким образом, вы также можете запустить JSON lint для файла, чтобы увидеть, есть ли ошибка при разборе JSON.
Даташаман

4
В Windows вам нужны двойные кавычки вокруг имени файла "@ body.json"
Stomf

100

Вам может пригодиться отдых: https://github.com/micha/resty

Это оболочка CURL, которая упрощает запросы REST командной строки. Вы указываете это на свою конечную точку API, и она дает вам команды PUT и POST. (Примеры адаптированы с домашней страницы)

$ resty http://127.0.0.1:8080/data #Sets up resty to point at your endpoing
$ GET /blogs.json                  #Gets http://127.0.0.1:8080/data/blogs.json
                                   #Put JSON
$ PUT /blogs/2.json '{"id" : 2, "title" : "updated post", "body" : "This is the new."}'
                                   # POST JSON from a file
$ POST /blogs/5.json < /tmp/blog.json

Кроме того, часто все еще необходимо добавлять заголовки типа контента. Вы можете сделать это один раз, чтобы установить по умолчанию добавление файлов конфигурации для каждого метода для каждого сайта: Настройка параметров RESTY по умолчанию


93

Для Windows наличие единственной кавычки для -dзначения не работало для меня, но оно работало после перехода на двойную кавычку. Также мне нужно было избегать двойных кавычек в фигурных скобках.

То есть следующее не сработало:

curl -i -X POST -H "Content-Type: application/json" -d '{"key":"val"}' http://localhost:8080/appname/path

Но сработало следующее:

curl -i -X POST -H "Content-Type: application/json" -d "{\"key\":\"val\"}" http://localhost:8080/appname/path

5
К вашему сведению - похоже, вам не хватает закрывающей двойной кавычки вокруг тела json
acanby

3
Для меня в Windows «вокруг данных не работает, вместо кавычек ничего не работает
rodedo

3
Если вы используете PowerShell, посмотрите этот ответ.
rsenna

86

Это сработало для меня, используя:

curl -H "Accept: application/json" -H "Content-type: application/json" -X POST -d '{"id":100}' http://localhost/api/postJsonReader.do

Это было счастливо сопоставлено с контроллером Spring:

@RequestMapping(value = "/postJsonReader", method = RequestMethod.POST)
public @ResponseBody String processPostJsonData(@RequestBody IdOnly idOnly) throws Exception {
        logger.debug("JsonReaderController hit! Reading JSON data!"+idOnly.getId());
        return "JSON Received";
}

IdOnlyпростой POJO со свойством id


56

Например, создайте файл JSON, params.json, и добавьте в него следующее содержимое:

[
    {
        "environment": "Devel",
        "description": "Machine for test, please do not delete!"
    }
]

Затем вы запускаете эту команду:

curl -v -H "Content-Type: application/json" -X POST --data @params.json -u your_username:your_password http://localhost:8000/env/add_server

42

Я просто столкнулся с той же проблемой. Я мог бы решить это, указав

-H "Content-Type: application/json; charset=UTF-8"

37

Это хорошо сработало для меня.

curl -X POST --data @json_out.txt http://localhost:8080/

Куда,

-X Означает глагол http.

--data Означает данные, которые вы хотите отправить.


5
В -X POSTданном примере это избыточно
инженер-программист

31

Вы можете использовать Postman с интуитивно понятным графическим интерфейсом для сборки вашей cURLкоманды.

  1. Установить и запустить Почтальон
  2. Введите свой URL, текст сообщения, заголовки запроса и т. П.
  3. Нажмите на Code
  4. Выберите cURLиз выпадающего списка
  5. скопируйте и вставьте вашу cURLкоманду

Примечание: в раскрывающемся списке есть несколько вариантов автоматической генерации запросов, поэтому я решил, что мой пост был необходим в первую очередь.


6
Не понял, что эта функция была включена в Почтальон. Спасибо за указание на это!
аристав

29

Используя CURL Windows, попробуйте это:

curl -X POST -H "Content-Type:application/json" -d "{\"firstName\": \"blablabla\",\"lastName\": \"dummy\",\"id\": \"123456\"}" http-host/_ah/api/employeeendpoint/v1/employee

28

Вы можете использовать почтальон для конвертации в CURLвведите описание изображения здесь

введите описание изображения здесь


1
Большое спасибо @ forever-LA, это была подсказка для меня
Vibs2006

24

HTTPie является рекомендуемой альтернативой, curlпотому что вы можете сделать только

$ http POST http://example.com/some/endpoint name=value name1=value1

По умолчанию он говорит на JSON и будет обрабатывать как установку необходимого вам заголовка, так и кодирование данных в качестве допустимого JSON. Есть также:

Some-Header:value

для заголовков и

name==value

для параметров строки запроса. Если у вас большой кусок данных, вы также можете прочитать его из файла, если он будет закодирован в JSON:

 field=@file.txt

20

Если вы тестируете много JSON-сообщений / ответов в отношении интерфейса RESTful, вы можете попробовать подключаемый модуль Postman для Chrome (который позволяет вручную определять тесты веб-служб) и его команду Newman на основе Node.js. -линейный компаньон (который позволяет автоматизировать тесты по «коллекциям» тестов Почтальона.) И бесплатно, и открыто!


18

Это работало хорошо для меня, дополнительно используя базовую аутентификацию:

curl -v --proxy '' --basic -u Administrator:password -X POST -H "Content-Type: application/json"
        --data-binary '{"value":"30","type":"Tip 3","targetModule":"Target 3","configurationGroup":null,"name":"Configuration Deneme 3","description":null,"identity":"Configuration Deneme 3","version":0,"systemId":3,"active":true}'
        http://httpbin.org/post

Конечно, вы никогда не должны использовать BASIC-аутентификацию без SSL и проверенного сертификата.

Сегодня я столкнулся с этим снова, используя Cygwin cURL 7.49.1 для Windows ... И при использовании --dataили --data-binaryс аргументом JSON cURL запутался и интерпретировал бы {}JSON как шаблон URL. Добавление -gаргумента для отключения cURL globbing исправило это.

Смотрите также Передача URL в квадратных скобках для скручивания .


17

Вы также можете поместить содержимое JSON в файл и передать его в curl, используя --file-uploadопцию через стандартный ввод, например:

 echo 'my.awesome.json.function({"do" : "whatever"})' | curl -X POST "http://url" -T -


10

Я использую приведенный ниже формат для тестирования с веб-сервером.

use -F 'json data'

Давайте предположим, что этот формат JSON dict:

{
    'comment': {
        'who':'some_one',
        'desc' : 'get it'
    }
}

Полный пример

curl -XPOST your_address/api -F comment='{"who":"some_one", "desc":"get it"}'

6

Используйте опцию -d для добавления полезной нагрузки

curl -X POST \
http://<host>:<port>/<path> \
-H 'Accept: application/json' \
-H 'Content-Type: application/json' \
-d '{
"foo": "bar",
"lorem": "ipsum"
}'

К тому же:

использовать -X POST для использования метода POST

используйте -H 'Accept: application / json', чтобы добавить заголовок типа принятия

используйте -H 'Content-Type: application / json', чтобы добавить заголовок типа контента


Я пытался использовать его, но получаю сообщение об ошибке {"errors": ["данные не предоставлены"]}.
Суреш

6

Пожалуйста, проверьте этот инструмент . Это поможет вам легко создавать фрагменты скручивания.

curl -XGET -H "Accept: application/json" -d "{\"value\":\"30\",\"type\":\"Tip 3\",\"targetModule\":\"Target 3\",\"configurationGroup\":null,\"name\":\"Configuration Deneme 3\",\"description\":null,\"identity\":\"Configuration Deneme 3\",\"version\":0,\"systemId\":3,\"active\":true}" "http://localhost:8080/xx/xxx/xxxx"


5

Вот еще один способ сделать это, если у вас есть динамические данные, которые будут включены.

#!/bin/bash

version=$1
text=$2
branch=$(git rev-parse --abbrev-ref HEAD)
repo_full_name=$(git config --get remote.origin.url | sed 's/.*:\/\/github.com\///;s/.git$//')
token=$(git config --global github.token)

generate_post_data()
{
  cat <<EOF
{
  "tag_name": "$version",
  "target_commitish": "$branch",
  "name": "$version",
  "body": "$text",
  "draft": false,
  "prerelease": false
}
EOF
}

echo "Create release $version for repo: $repo_full_name branch: $branch"
curl --data "$(generate_post_data)" "https://api.github.com/repos/$repo_full_name/releases?access_token=$token"

1
Вы спасли мою жизнь!
Абхиманью

4

Я знаю, что на этот вопрос было дано много ответов, но я хотел бы поделиться тем, где у меня возникла проблема:

curl -X POST http: // your-server-end-point -H "Content-Type: application / json" -d @ путь-вашего-json-file.json

Видишь, я все сделал правильно, только одно - "@" я пропустил до пути к файлу JSON.

Я нашел один соответствующий документ в Интернете - https://gist.github.com/subfuzion/08c5d85437d5d4f00e58

Надеюсь, что это может помочь немногим. Спасибо


1

Если вы сконфигурируете SWAGGER для своего приложения с весенней загрузкой и вызовите любой API из своего приложения, вы также сможете увидеть этот CURL-запрос.

Я думаю, что это простой способ генерации запросов через CURL.


-6

Вы можете передать расширение нужного формата в качестве конца URL. например http: // localhost: 8080 / xx / xxx / xxxx .json

или

http: // localhost: 8080 / xx / xxx / xxxx .xml

Примечание: вам нужно добавить зависимости Jackson и Jaxb Maven в вашем POM.


4
Это будет работать только в том случае, если серверная сторона принимает URL-адреса с этими расширениями.
Марк Стосберг
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.