lib/word/related_words.ex

defmodule Wordnik.Word.RelatedWords do
  @moduledoc """
  find related words organized by type of relationship (synonym, antonym, etc.)

  [Wordnik Docs](https://developer.wordnik.com/docs#!/word/getRelatedWords)
  """
  alias Wordnik.Formatter.Query

  @typedoc """
  map of optional parameters that can be passed to `get_related_words/2` query
  """
  @type related_words_params ::
          %{
            optional(:use_canonical) => boolean(),
            optional(:limit) => integer(),
            optional(:relationship_types) => Formatter.Enums.relationship_types()
          }

  @typedoc """
  parsed JSON response to `get_related_words/2` query
  """
  @type related_words ::
          list(%{
            relationshipType: String.t(),
            words: list(String.t())
          })

  @valid_params [
    "limit",
    "relationship_types",
    "use_canonical"
  ]

  defp format_url(word) do
    "http://api.wordnik.com/v4/word.json/#{word}/relatedWords"
  end

  @doc """
  get related_words for requested word

  ### Parameters
  - use_canonical: boolean
  - limit: integer
  - relationship_types: string

  ### Example
  ```elixir
  iex> get_related_words("verbose", %{use_canonical: true, relationship_types: "synonym"})
  ```

   ### Response
  `t:Word.RelatedWords.related_words/0`

  ### Wordnik Docs
  https://developer.wordnik.com/docs#!/word/getRelatedWords
  """
  @spec get_related_words(String.t(), related_words_params()) ::
          {:error, String.t()} | {:ok, related_words()}
  def get_related_words(word, params \\ %{}) do
    {fn_name, _} = __ENV__.function

    word
    |> format_url
    |> Query.validate_and_fetch_query(params, @valid_params, fn_name)
  end
end