lib/glific/contacts/contacts_field.ex

defmodule Glific.Contacts.ContactsField do
  @moduledoc """
  The minimal wrapper for the base Contact structure
  """
  use Ecto.Schema
  import Ecto.Changeset

  alias __MODULE__
  alias Glific.Partners.Organization
  alias Glific.Enums.{ContactFieldScope, ContactFieldValueType}

  @required_fields [
    :name,
    :shortcode,
    :organization_id
  ]
  @optional_fields [
    :value_type,
    :scope
  ]

  @type t() :: %ContactsField{
          __meta__: Ecto.Schema.Metadata.t(),
          id: non_neg_integer | nil,
          name: String.t() | nil,
          shortcode: String.t() | nil,
          value_type: String.t() | nil,
          scope: String.t() | nil,
          organization_id: non_neg_integer | nil,
          organization: Organization.t() | Ecto.Association.NotLoaded.t() | nil,
          inserted_at: :utc_datetime | nil,
          updated_at: :utc_datetime | nil
        }

  schema "contacts_fields" do
    field :name, :string
    field :shortcode, :string
    field :value_type, ContactFieldValueType, default: :text
    field :scope, ContactFieldScope, default: :contact
    field :variable, :string, virtual: true

    belongs_to :organization, Organization
    timestamps(type: :utc_datetime)
  end

  @doc """
  Standard changeset pattern we use for all data types
  """
  # Not sure why dialyzer is giving so many errors with type.
  # Will come back on this and improve the specs.
  @spec changeset(any(), map()) :: Ecto.Changeset.t()
  def changeset(contact_field, attrs) do
    contact_field
    |> cast(attrs, @required_fields ++ @optional_fields)
    |> validate_required(@required_fields)
    |> unique_constraint([:name, :organization_id])
    |> unique_constraint([:shortcode, :organization_id])
  end
end