Ash: 1 - Resources
Application.put_env(:ash, :validate_domain_resource_inclusion?, false)
Application.put_env(:ash, :validate_domain_config_inclusion?, false)
Mix.install([{:ash, "~> 3.0"}], consolidate_protocols: false)What is a resource?
HomeActionsThe fundamental building blocks of Ash are resources, actions, and domains. The resource is Ash’s central concept, so we need to define it well for anything else to make sense, so let’s turn to the Ash Glossary.
> The central concept in Ash, a resource is a domain model object in your system, the nouns that your app revolves around. Resources contain definitions of the data they hold in the form of attributes, but also define actions that can be taken on that data and actors that are allowed to run them. > > It is not a strict requirement that resources contain data - they can be used purely to create a standard interface for performing tasks - but in practice, most resources will be used to manage data.
A domain simply groups together related resources. Some things can be defined in a domain instead of a resource. We will do this when we get to code interfaces later on.
Example Code
Here we have a minimal resource called Profile that has, aside from its primary key, a single string attribute name.
defmodule BitHub.User do
  use Ash.Resource,
    domain: BitHub
  attributes do
    uuid_primary_key :id
    attribute :name, :string
  end
end{:module, BitHub.User, <<70, 79, 82, 49, 0, 0, 86, ...>>,
 [Ash.Resource.Attribute.Helpers, Ash.Resource.Dsl.Attributes.UuidV7PrimaryKey,
  Ash.Resource.Dsl.Attributes.UuidPrimaryKey, Ash.Resource.Dsl.Attributes.IntegerPrimaryKey,
  Ash.Resource.Dsl.Attributes.UpdateTimestamp, Ash.Resource.Dsl.Attributes.CreateTimestamp, ...]}To be able to do anything with this resource, we also need a domain.
defmodule BitHub do
  use Ash.Domain
  resources do
    resource BitHub.User
  end
end{:module, BitHub, <<70, 79, 82, 49, 0, 0, 43, ...>>,
 [
   Ash.Domain.Dsl.Resources.Resource,
   Ash.Domain.Dsl.Resources.Options,
   %{opts: [], entities: [%Ash.Domain.Dsl.ResourceReference{...}]}
 ]}Now we can create a changeset for the resource.
BitHub.User
|> Ash.Changeset.new()
|> Ash.Changeset.change_attributes(%{name: "John Doe"})#Ash.Changeset<
  action_type: :create,
  action: nil,
  attributes: %{name: "John Doe"},
  relationships: %{},
  errors: [],
  data: #BitHub.User<
    __meta__: #Ecto.Schema.Metadata<:built, "">,
    id: nil,
    name: nil,
    aggregates: %{},
    calculations: %{},
    ...
  >,
  valid?: true
>Changesets are used to create and update data in Ash. (Note: without a data layer, you won’t be able to persist creations and changes. This will be shown in a later tutorial.)
HomeActions