Ash.DataLayer behaviour (ash v3.7.2)
View SourceThe behaviour for backing resource actions with persistence layers.
Summary
Functions
The data layer of the resource, or nil if it does not have one
Whether or not the data layer supports a specific feature
Custom functions supported by the data layer of the resource
Whether or not lateral joins should be used for many to many relationships by default
Rolls back the current transaction
Wraps the execution of the function in a transaction with the resource's data_layer.
Types
@type bulk_create_options() :: %{ batch_size: pos_integer(), return_records?: boolean(), upsert?: boolean(), action_select: [atom()], upsert_keys: nil | [atom()], upsert_condition: Ash.Expr.t() | nil, return_skipped_upsert?: boolean(), identity: Ash.Resource.Identity.t() | nil, select: [atom()], upsert_fields: nil | [atom()] | :replace_all | {:replace, [atom()]} | {:replace_all_except, [atom()]}, tenant: term() }
@type bulk_update_options() :: %{ return_records?: boolean(), action_select: [atom()], calculations: [{Ash.Query.Calculation.t(), Ash.Expr.t()}], select: [atom()], tenant: term() }
@type combination_type() :: :union | :union_all | :intersection
      @type data_layer_query() :: struct()
@type feature() :: :transact | :multitenancy | :combine | {:combine, combination_type()} | {:atomic, :update} | {:atomic, :upsert} | {:exists, :unrelated} | {:lateral_join, [Ash.Resource.t()]} | {:join, Ash.Resource.t()} | {:aggregate, :unrelated} | {:aggregate, Ash.Query.Aggregate.kind()} | {:aggregate_relationship, Ash.Resource.Relationships.relationship()} | {:query_aggregate, Ash.Query.Aggregate.kind()} | :select | :expr_error | :calculate | :expression_calculation | :expression_calculation_sort | :aggregate_filter | :aggregate_sort | :boolean_filter | :async_engine | :bulk_create | :update_query | :destroy_query | :create | :read | :update | :destroy | :limit | :offset | :transact | :filter | :composite_type | {:lock, lock_type()} | {:filter_expr, struct()} | {:filter_relationship, Ash.Resource.Relationships.relationship()} | :sort | {:sort, Ash.Type.t()} | :upsert | :composite_primary_key | :bulk_upsert_return_skipped
@type lateral_join_link() :: {Ash.Resource.t(), atom(), atom(), Ash.Resource.Relationships.relationship()}
@type lock_type() :: :for_update | term()
@type t() :: module()
@type transaction_reason() :: %{ :type => :create, :metadata => %{resource: Ash.Resource.t(), action: atom()}, optional(:data_layer_context) => %{} } | %{ :type => :update, :metadata => %{ resource: Ash.Resource.t(), action: atom(), record: Ash.Resource.record(), actor: term() }, optional(:data_layer_context) => %{} } | %{ :type => :destroy, :metadata => %{ resource: Ash.Resource.t(), action: atom(), record: Ash.Resource.record(), actor: term() }, optional(:data_layer_context) => %{} } | %{ :type => :read, :metadata => %{ resource: Ash.Resource.t(), query: Ash.Query.t(), actor: term() }, optional(:data_layer_context) => %{} } | %{ :type => :flow_transaction, :metadata => %{ resource: Ash.Resource.t(), input: Ash.ActionInput.t(), action: atom(), actor: term() }, optional(:data_layer_context) => %{} } | %{ :type => :generic, :metadata => %{ step_name: atom() | [term()], flow: module(), actor: term() }, optional(:data_layer_context) => %{} } | %{type: :custom, metadata: map()} | %{type: atom() | tuple(), metadata: map()}
Callbacks
@callback add_aggregate( data_layer_query(), Ash.Query.Aggregate.t(), Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback add_aggregates( data_layer_query(), [Ash.Query.Aggregate.t()], Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback add_calculation( data_layer_query(), Ash.Query.Calculation.t(), expression :: any(), Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback add_calculations( data_layer_query(), [{Ash.Query.Calculation.t(), expression :: any()}], Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback bulk_create( Ash.Resource.t(), Enumerable.t(Ash.Changeset.t()), options :: bulk_create_options() ) :: :ok | {:ok, Enumerable.t(Ash.Resource.record())} | {:error, Ash.Error.t()} | {:error, :no_rollback, Ash.Error.t()}
@callback calculate(Ash.Resource.t(), [Ash.Expr.t()], context :: map()) :: {:ok, term()} | {:error, term()}
@callback can?(Ash.Resource.t() | Spark.Dsl.t(), feature()) :: boolean()
@callback combination_acc(data_layer_query()) :: any()
@callback combination_of( combine :: [{combination_type(), data_layer_query()}], resource :: Ash.Resource.t(), domain :: Ash.Domain.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback create(Ash.Resource.t(), Ash.Changeset.t()) :: {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()}
@callback destroy(Ash.Resource.t(), Ash.Changeset.t()) :: :ok | {:error, term()}
@callback destroy_query( data_layer_query(), Ash.Changeset.t(), Ash.Resource.t(), opts :: bulk_update_options() ) :: :ok | {:ok, Enumerable.t(Ash.Resource.record())} | {:error, Ash.Error.t()} | {:error, :no_rollback, Ash.Error.t()}
@callback distinct(data_layer_query(), [atom()], resource :: Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@callback distinct_sort(data_layer_query(), Ash.Sort.t(), resource :: Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@callback filter(data_layer_query(), Ash.Filter.t(), resource :: Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@callback functions(Ash.Resource.t()) :: [module()]
@callback in_transaction?(Ash.Resource.t()) :: boolean()
@callback limit( data_layer_query(), limit :: non_neg_integer(), resource :: Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback lock(data_layer_query(), lock_type(), resource :: Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@callback offset( data_layer_query(), offset :: non_neg_integer(), resource :: Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback prefer_lateral_join_for_many_to_many?() :: boolean()
@callback prefer_transaction?(Ash.Resource.t()) :: boolean()
@callback prefer_transaction_for_atomic_updates?(Ash.Resource.t()) :: boolean()
@callback resource_to_query(Ash.Resource.t(), Ash.Domain.t()) :: data_layer_query()
@callback return_query(data_layer_query(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@callback rollback(Ash.Resource.t(), term()) :: no_return()
@callback run_aggregate_query( data_layer_query(), [Ash.Query.Aggregate.t()], Ash.Resource.t() ) :: {:ok, map()} | {:error, term()}
@callback run_aggregate_query_with_lateral_join( data_layer_query(), [Ash.Query.Aggregate.t()], [Ash.Resource.record()], destination_resource :: Ash.Resource.t(), [lateral_join_link()] ) :: {:ok, [Ash.Resource.t()]} | {:error, term()}
@callback run_query(data_layer_query(), Ash.Resource.t()) :: {:ok, [Ash.Resource.record()]} | {:error, term()} | {:error, :no_rollback, term()}
@callback run_query_with_lateral_join( data_layer_query(), [Ash.Resource.record()], source_resource :: Ash.Resource.t(), [lateral_join_link()] ) :: {:ok, [Ash.Resource.record()]} | {:error, term()}
@callback select( data_layer_query(), select :: [atom()], resource :: Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@callback set_context(Ash.Resource.t(), data_layer_query(), map()) :: {:ok, data_layer_query()} | {:error, term()}
@callback set_tenant(Ash.Resource.t(), data_layer_query(), term()) :: {:ok, data_layer_query()} | {:error, term()}
@callback sort(data_layer_query(), Ash.Sort.t(), resource :: Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@callback source(Ash.Resource.t()) :: String.t()
@callback transaction( Ash.Resource.t(), (-> term()), nil | pos_integer(), reason :: transaction_reason() ) :: {:ok, term()} | {:error, term()}
@callback transform_query(Ash.Query.t()) :: Ash.Query.t()
@callback update(Ash.Resource.t(), Ash.Changeset.t()) :: {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()}
@callback update_query( data_layer_query(), Ash.Changeset.t(), Ash.Resource.t(), opts :: bulk_update_options() ) :: :ok | {:ok, Enumerable.t(Ash.Resource.record())} | {:error, Ash.Error.t()} | {:error, :no_rollback, Ash.Error.t()}
@callback upsert(Ash.Resource.t(), Ash.Changeset.t(), [atom()]) :: {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()}
@callback upsert( Ash.Resource.t(), Ash.Changeset.t(), [atom()], Ash.Resource.Identity.t() | nil ) :: {:ok, Ash.Resource.record() | {:upsert_skipped, Ash.Query.t(), (-> {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()})}} | {:error, term()} | {:error, :no_rollback, term()}
Functions
@spec add_aggregates(data_layer_query(), [Ash.Query.Aggregate.t()], Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec add_calculations( data_layer_query(), [{Ash.Query.Calculation.t(), expression :: term()}], Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@spec bulk_create( Ash.Resource.t(), Enumerable.t(Ash.Changeset.t()), options :: bulk_create_options() ) :: :ok | {:ok, Enumerable.t(Ash.Resource.record())} | {:error, Ash.Error.t()} | {:error, :no_rollback, Ash.Error.t()}
@spec calculate(Ash.Resource.t(), [Ash.Expr.t()], context :: map()) :: {:ok, [term()]} | {:error, Ash.Error.t()}
@spec can?(feature(), Ash.Resource.t() | Spark.Dsl.t()) :: boolean()
@spec combination_acc(data_layer_query(), Ash.Resource.t()) :: any()
@spec combination_of( [{combination_type(), data_layer_query()}], resource :: Ash.Resource.t(), domain :: Ash.Domain.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@spec create(Ash.Resource.t(), Ash.Changeset.t()) :: {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()}
@spec data_layer(Ash.Resource.t() | Spark.Dsl.t()) :: t() | nil
The data layer of the resource, or nil if it does not have one
@spec data_layer_can?(Ash.Resource.t() | Spark.Dsl.t(), feature()) :: boolean()
Whether or not the data layer supports a specific feature
@spec data_layer_functions(Ash.Resource.t()) :: map()
Custom functions supported by the data layer of the resource
@spec destroy(Ash.Resource.t(), Ash.Changeset.t()) :: :ok | {:error, term()} | {:error, :no_rollback, term()}
@spec destroy_query( data_layer_query(), Ash.Changeset.t(), opts :: bulk_update_options() ) :: :ok | {:ok, Enumerable.t(Ash.Resource.record())} | {:error, Ash.Error.t()} | {:error, :no_rollback, Ash.Error.t()}
@spec distinct(data_layer_query(), Ash.Sort.t(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec distinct_sort(data_layer_query(), Ash.Sort.t(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec filter(data_layer_query(), Ash.Filter.t(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec limit(data_layer_query(), limit :: non_neg_integer(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec lock( data_layer_query(), lock_type :: lock_type() | nil, resource :: Ash.Resource.t() ) :: {:ok, data_layer_query()} | {:error, term()}
@spec offset(data_layer_query(), offset :: non_neg_integer(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
Whether or not lateral joins should be used for many to many relationships by default
@spec prefer_transaction?(Ash.Resource.t()) :: boolean()
@spec prefer_transaction_for_atomic_updates?(Ash.Resource.t()) :: boolean()
@spec resource_to_query(Ash.Resource.t(), Ash.Domain.t()) :: data_layer_query()
@spec return_query(data_layer_query(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec rollback(Ash.Resource.t() | [Ash.Resource.t()], term()) :: no_return()
Rolls back the current transaction
@spec run_aggregate_query( data_layer_query(), [Ash.Query.Aggregate.t()], Ash.Resource.t() ) :: {:ok, map()} | {:error, term()}
@spec run_query(data_layer_query(), central_resource :: Ash.Resource.t()) :: {:ok, [Ash.Resource.record()]} | {:error, term()} | {:error, :no_rollback, term()}
@spec select(data_layer_query(), select :: [atom()], Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec set_context(Ash.Resource.t(), data_layer_query(), map()) :: {:ok, data_layer_query()} | {:error, term()}
@spec set_tenant(Ash.Resource.t(), data_layer_query(), String.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec sort(data_layer_query(), Ash.Sort.t(), Ash.Resource.t()) :: {:ok, data_layer_query()} | {:error, term()}
@spec source(Ash.Resource.t()) :: String.t()
@spec transaction( Ash.Resource.t() | [Ash.Resource.t()], (-> term()), nil | pos_integer(), reason :: transaction_reason(), opts :: Keyword.t() ) :: term()
Wraps the execution of the function in a transaction with the resource's data_layer.
@spec update(Ash.Resource.t(), Ash.Changeset.t()) :: {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()}
@spec update_query( data_layer_query(), Ash.Changeset.t(), opts :: bulk_update_options() ) :: :ok | {:ok, Enumerable.t(Ash.Resource.record())} | {:error, Ash.Error.t()} | {:error, :no_rollback, Ash.Error.t()}
@spec upsert( Ash.Resource.t(), Ash.Changeset.t(), [atom()], identity :: Ash.Resource.Identity.t() | nil ) :: {:ok, Ash.Resource.record() | {:upsert_skipped, Ash.Query.t(), (-> {:ok, Ash.Resource.record()} | {:error, term()} | {:error, :no_rollback, term()})}} | {:error, term()} | {:error, :no_rollback, term()}