2019-03-12 11:52:28 +01:00
|
|
|
defmodule Mobilizon.Service.Geospatial.MapQuest do
|
|
|
|
@moduledoc """
|
|
|
|
[MapQuest](https://developer.mapquest.com/documentation) backend.
|
|
|
|
|
|
|
|
## Options
|
|
|
|
In addition to the [the shared options](Mobilizon.Service.Geospatial.Provider.html#module-shared-options),
|
|
|
|
MapQuest methods support the following options:
|
|
|
|
* `:open_data` Whether to use [Open Data or Licenced Data](https://developer.mapquest.com/documentation/open/).
|
|
|
|
Defaults to `true`
|
|
|
|
"""
|
2019-09-22 16:26:23 +02:00
|
|
|
|
2019-03-12 11:52:28 +01:00
|
|
|
alias Mobilizon.Addresses.Address
|
2019-10-11 17:20:03 +02:00
|
|
|
alias Mobilizon.Config
|
2020-01-28 20:15:59 +01:00
|
|
|
alias Mobilizon.Service.Geospatial.Provider
|
2019-09-22 16:26:23 +02:00
|
|
|
|
2019-03-12 11:52:28 +01:00
|
|
|
require Logger
|
|
|
|
|
|
|
|
@behaviour Provider
|
|
|
|
|
|
|
|
@api_key Application.get_env(:mobilizon, __MODULE__) |> get_in([:api_key])
|
|
|
|
|
|
|
|
@api_key_missing_message "API Key required to use MapQuest"
|
|
|
|
|
2020-02-18 08:57:00 +01:00
|
|
|
@http_options [
|
|
|
|
follow_redirect: true,
|
|
|
|
ssl: [{:versions, [:"tlsv1.2"]}]
|
|
|
|
]
|
|
|
|
|
2019-03-12 11:52:28 +01:00
|
|
|
@impl Provider
|
|
|
|
@doc """
|
|
|
|
MapQuest implementation for `c:Mobilizon.Service.Geospatial.Provider.geocode/3`.
|
|
|
|
"""
|
|
|
|
@spec geocode(String.t(), keyword()) :: list(Address.t())
|
|
|
|
def geocode(lon, lat, options \\ []) do
|
|
|
|
api_key = Keyword.get(options, :api_key, @api_key)
|
|
|
|
limit = Keyword.get(options, :limit, 10)
|
|
|
|
open_data = Keyword.get(options, :open_data, true)
|
2019-10-11 17:20:03 +02:00
|
|
|
user_agent = Keyword.get(options, :user_agent, Config.instance_user_agent())
|
|
|
|
headers = [{"User-Agent", user_agent}]
|
2019-03-12 11:52:28 +01:00
|
|
|
|
|
|
|
prefix = if open_data, do: "open", else: "www"
|
|
|
|
|
|
|
|
if is_nil(api_key), do: raise(ArgumentError, message: @api_key_missing_message)
|
|
|
|
|
|
|
|
with {:ok, %HTTPoison.Response{status_code: 200, body: body}} <-
|
|
|
|
HTTPoison.get(
|
|
|
|
"https://#{prefix}.mapquestapi.com/geocoding/v1/reverse?key=#{api_key}&location=#{
|
|
|
|
lat
|
2019-10-11 17:20:03 +02:00
|
|
|
},#{lon}&maxResults=#{limit}",
|
2020-02-18 08:57:00 +01:00
|
|
|
headers,
|
|
|
|
@http_options
|
2019-03-12 11:52:28 +01:00
|
|
|
),
|
|
|
|
{:ok, %{"results" => results, "info" => %{"statuscode" => 0}}} <- Poison.decode(body) do
|
2019-07-23 18:06:22 +02:00
|
|
|
results |> Enum.map(&process_data/1)
|
2019-03-12 11:52:28 +01:00
|
|
|
else
|
|
|
|
{:ok, %HTTPoison.Response{status_code: 403, body: err}} ->
|
|
|
|
raise(ArgumentError, message: err)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl Provider
|
|
|
|
@doc """
|
|
|
|
MapQuest implementation for `c:Mobilizon.Service.Geospatial.Provider.search/2`.
|
|
|
|
"""
|
|
|
|
@spec search(String.t(), keyword()) :: list(Address.t())
|
|
|
|
def search(q, options \\ []) do
|
2019-10-11 17:20:03 +02:00
|
|
|
user_agent = Keyword.get(options, :user_agent, Config.instance_user_agent())
|
|
|
|
headers = [{"User-Agent", user_agent}]
|
2019-03-12 11:52:28 +01:00
|
|
|
limit = Keyword.get(options, :limit, 10)
|
|
|
|
api_key = Keyword.get(options, :api_key, @api_key)
|
|
|
|
|
|
|
|
open_data = Keyword.get(options, :open_data, true)
|
|
|
|
|
|
|
|
prefix = if open_data, do: "open", else: "www"
|
|
|
|
|
|
|
|
if is_nil(api_key), do: raise(ArgumentError, message: @api_key_missing_message)
|
|
|
|
|
|
|
|
url =
|
|
|
|
"https://#{prefix}.mapquestapi.com/geocoding/v1/address?key=#{api_key}&location=#{
|
|
|
|
URI.encode(q)
|
|
|
|
}&maxResults=#{limit}"
|
|
|
|
|
|
|
|
Logger.debug("Asking MapQuest for addresses with #{url}")
|
|
|
|
|
|
|
|
with {:ok, %HTTPoison.Response{status_code: 200, body: body}} <-
|
2020-02-18 08:57:00 +01:00
|
|
|
HTTPoison.get(url, headers, @http_options),
|
2019-03-12 11:52:28 +01:00
|
|
|
{:ok, %{"results" => results, "info" => %{"statuscode" => 0}}} <- Poison.decode(body) do
|
2019-07-23 18:06:22 +02:00
|
|
|
results |> Enum.map(&process_data/1)
|
2019-03-12 11:52:28 +01:00
|
|
|
else
|
|
|
|
{:ok, %HTTPoison.Response{status_code: 403, body: err}} ->
|
|
|
|
raise(ArgumentError, message: err)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-23 18:06:22 +02:00
|
|
|
defp process_data(
|
2019-03-12 11:52:28 +01:00
|
|
|
%{
|
|
|
|
"locations" => addresses,
|
|
|
|
"providedLocation" => %{"latLng" => %{"lat" => lat, "lng" => lng}}
|
|
|
|
} = _body
|
|
|
|
) do
|
|
|
|
case addresses do
|
|
|
|
[] -> nil
|
2019-07-23 18:06:22 +02:00
|
|
|
addresses -> addresses |> hd |> produce_address(lat, lng)
|
2019-03-12 11:52:28 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-23 18:06:22 +02:00
|
|
|
defp process_data(%{"locations" => addresses}) do
|
2019-03-12 11:52:28 +01:00
|
|
|
case addresses do
|
|
|
|
[] -> nil
|
2019-07-23 18:06:22 +02:00
|
|
|
addresses -> addresses |> hd |> produce_address()
|
2019-03-12 11:52:28 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-07-23 18:06:22 +02:00
|
|
|
defp produce_address(%{"latLng" => %{"lat" => lat, "lng" => lng}} = address) do
|
|
|
|
produce_address(address, lat, lng)
|
2019-03-12 11:52:28 +01:00
|
|
|
end
|
|
|
|
|
2019-07-23 18:06:22 +02:00
|
|
|
defp produce_address(address, lat, lng) do
|
2019-03-12 11:52:28 +01:00
|
|
|
%Address{
|
2019-03-22 15:51:23 +01:00
|
|
|
country: Map.get(address, "adminArea1"),
|
|
|
|
locality: Map.get(address, "adminArea5"),
|
|
|
|
region: Map.get(address, "adminArea3"),
|
2019-03-12 11:52:28 +01:00
|
|
|
description: Map.get(address, "street"),
|
|
|
|
geom: [lng, lat] |> Provider.coordinates(),
|
2019-03-22 15:51:23 +01:00
|
|
|
postal_code: Map.get(address, "postalCode"),
|
|
|
|
street: Map.get(address, "street")
|
2019-03-12 11:52:28 +01:00
|
|
|
}
|
|
|
|
end
|
|
|
|
end
|