POST (не GET) запрос викиданных - PullRequest
2 голосов
/ 21 сентября 2019

Я следую этим инструкциям , чтобы отправить запрос SPARQL, предпочитая метод POST, так как запросы могут быть довольно длинными.Но, похоже, он не работает, хотя GET работает - есть ли способ заставить POST-запрос работать?

sparql = <<END
SELECT ?item ?itemLabel 
WHERE 
{
    ?item wdt:P31 wd:Q146.
    SERVICE wikibase:label { bd:serviceParam wikibase:language "[AUTO_LANGUAGE],en". }
}
END

# Fails with 405 Not writable
resp = Excon.post('https://query.wikidata.org/sparql', body: "query=#{URI::encode(sparql)}")
puts resp.status, resp.body

# Works with 200
resp = Excon.get("https://query.wikidata.org/sparql?query=#{URI::encode(sparql)}")
puts resp.status, resp.body

Я тоже пытался отправить "Content-Type" => "application/x-www-form-urlencoded", но безуспешно.

1 Ответ

3 голосов
/ 22 сентября 2019

Спасибо вышеупомянутым комментаторам за показ рабочих примеров.Я сделал еще несколько проверок, основанных на всех очевидных комбинациях, как показано ниже.

В итоге:

  • Используйте GET, если он короткий и, вероятно, будет использоваться повторно (так как запросы GET кэшируются)
  • Убедитесь, что вы отправили заголовок user-agent и отметили, что некоторые библиотеки не будут включать его по умолчанию (в противном случае он возвращает необъяснимое значение 403)
  • Для POST - лучший и самый простой (на мой взгляд)для отправки запроса SPARQL в виде всего тела с «типом содержимого» в виде «application / sparql-query», здесь нет необходимости кодировать запрос.Вы также можете использовать синтаксис формы с «типом содержимого» как «application / x-www-form-urlencoded» и убедиться, что запрос закодирован.
require 'excon'

# any arbitrary query   
sparql = 'SELECT ?item ?itemLabel WHERE { ?item wdt:P31 wd:Q146.  SERVICE wikibase:label { bd:serviceParam wikibase:language "[AUTO_LANGUAGE],en". } }'

# SUCCESS!
resp = Excon.get("https://query.wikidata.org/sparql?query=#{URI::encode(sparql)}")
puts "GET", resp.status, resp.body[0,100], "\n"

# FAIL! 403 (need user agent)
headers = { "Content-Type" => "application/sparql-query" }
resp= Excon.post('https://query.wikidata.org/sparql', body: sparql, headers: headers)
puts "POST sparql-query", resp.status, resp.body[0,100], "\n"

# SUCCESS!
headers = { "Content-Type" => "application/sparql-query", "User-Agent" => "Ruby 2.6.4" }
resp= Excon.post('https://query.wikidata.org/sparql', body: sparql, headers: headers)
puts "POST sparql-query with user-agent", resp.status, resp.body[0,100], "\n"

# FAIL! 403 (need user agent)
headers = { "Content-Type" => "application/x-www-form-urlencoded" }
resp = Excon.post('https://query.wikidata.org/sparql', body: "query=#{URI::encode(sparql)}", headers: headers)
puts "POST form", resp.status, resp.body[0,100], "\n"

# SUCCESS!
headers = { "Content-Type" => "application/x-www-form-urlencoded", "User-Agent" => "Ruby 2.6.4" }
resp = Excon.post('https://query.wikidata.org/sparql', body: "query=#{URI::encode(sparql)}", headers: headers)
puts "POST form with user-agent", resp.status, resp.body[0,100], "\n"

# FAIL! 405 (need encoding)
resp = Excon.post('https://query.wikidata.org/sparql', body: "query=#{URI::encode(sparql)}")
puts "POST plain", resp.status, resp.body[0,100], "\n"

# FAIL! 405 (need encoding)
headers = { "User-Agent" => "Ruby 2.6.4" }
resp = Excon.post('https://query.wikidata.org/sparql', body: "query=#{URI::encode(sparql)}", headers: headers)
puts "POST plain with user-agent", resp.status, resp.body[0,100], "\n"

Дополнительная информация

...