Как изменить карту в эликсире

Я создал JSON API, используя Elixir и Phoenix

У меня есть конечная точка для действия создания в моем контроллере, который принимает данные JSON, которые выглядят так:

     [{"opens_detail"=>
          [{"ua"=>"Linux/Ubuntu/Chrome/Chrome 28.0.1500.53",
              "ip"=>"55.55.55.55",
              "ts"=>1365190001,
              "location"=>"Georgia, US"}],
         "template"=>"example-template",
         "metadata"=>{"user_id"=>"123", "website"=>"www.example.com"},
         "clicks"=>42,
         "ts"=>1365190000,
         "state"=>"sent",
         "clicks_detail"=>
          [{"ua"=>"Linux/Ubuntu/Chrome/Chrome 28.0.1500.53",
              "ip"=>"55.55.55.55",
              "ts"=>1365190001,
              "url"=>"http://www.example.com",
              "location"=>"Georgia, US"}],
         "email"=>"recipient.email@example.com",
         "subject"=>"example subject",
         "sender"=>"sender@example.com",
         "_id"=>"abc123abc123abc123abc123",
         "tags"=>["password-reset"],
         "opens"=>42}]

Моя цель - взять этот json и создать из него новый, где некоторые ключи и значения будут переименованы, чтобы соответствовать моей схеме ниже:

в сети / models / messages.ex

   ...
      schema "messages" do
        field :sender, :string
        field :uniq_id, :string # equal to '_id' in the payload
        field :ts, :utc_datetime
        field :template, :string
        field :subject, :string
        field :email, :string
        field :tags, {:array, :string}
        field :opens, :integer
        field :opens_ip, :string # equal to nested 'ip' value in 'open_details'
        field :opens_location, :string # equal to nested 'location' value in 'open_details'
        field :clicks, :integer
        field :clicks_ip, :string # equal to nested 'ip' value in 'click_details'
        field :clicks_location, :string # equal to nested 'location' value in 'click_details'
        field :status, :string # equal to the "state" in the payload

        timestamps()
      end
  ...

Вот что я попробовал:

в web / controller / message_controller.ex:

  def create(conn, payload) do

    %{ payload |
      "uniq_id" => payload["_id"],
      "status" => payload["type"]
      "open_ips" =>  Enum.at(payload["opens_detail"], 0)['ip'],
      "open_location" => Enum.at(payload["opens_detail"], 0)['location'],
      "click_ips" =>  Enum.at(payload["clicks_detail"], 0)['ip'],
      "click_location" => Enum.at(payload["clicks_detail"], 0)['location'],
    }

    changeset = Message.changeset(%Message{}, payload)

   ...

  end

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

Я из Ruby/Python (Rails/Django) и не хочу начинать загрязнять свое изучение функционального программирования, в частности, эликсир / феникс, своими знаниями о OO.

Как бы вы решили эту проблему?

1 ответ

Решение

Как бы вы решили эту проблему?

Я бы создал новую карту с нуля вместо обновления исходной карты. Ты можешь использовать get_in упростить логику для доступа к вложенным полям. Вот пример:

map = %{
  uniq_id:        get_in(payload, ["_id"]),
  open_ips:       get_in(payload, ["opens_detail", Access.at(0), "ip"]),
  open_locations: get_in(payload, ["opens_detail", Access.at(0), "location"]),
}

Если вы хотите выбрать подмножество полей на исходной карте, вы можете использовать Map.merge а также Map.take:

Map.merge(Map.take(payload, [:sender, ...]), %{uniq_id: ...})

Но если это всего лишь пара полей, я бы лучше выписал их вручную.

Другие вопросы по тегам