zachdaniel
Ash 3.0 Teasers!
Hey folks! I’ll be teasing some interesting bits going into Ash 3.0 while I work on it, and this is post #1!
You can follow along with the changes, if you are interested in the 3.0 branch of Ash. Please do not use this branch
. If you find issues, I will not help you with them until an actual release candidate is published.
So, on to teaser #1!
Timeline
The current plan is to have a release candidate ready in March, so we are not far off from 3.0!
picosat_elixir Installation Issues
Some of our users (especially those on windows) experience installation issues with picosat_elixir. picosat_elixir is an excellent package, and we suggest that everyone starts there, we have published an alternative, called simple_sat. You can now select one of the two when following the getting started guide, and Ash will use whichever one is present.
We strongly suggest that folks eventually figure out their installation issues with picosat_elixir (our experience is that this is primarily isolated to windows users, and that folks eventually find a workaround). However, we didn’t want that to get in the way of people trying out Ash!
Registries are no more
Registries are no longer necessary, and as such are being removed for 3.0. This simplifies set up overall, and helps avoid a point of confusion that existed in 2.0, as registries were still supported but provided no tangible benefit.
Thats it!
There is tons more already done for 3.0, and more to come, but I’ll talk more about those in upcoming posts.
I hope you’re all as excited as I am!
Teaser #2: Ash 3.0 Teasers! - #9 by zachdaniel
Teaser #3: Ash 3.0 Teasers! - #23 by zachdaniel
Teaser #4: Ash 3.0 Teasers! - #28 by zachdaniel
Teaser #5: Ash 3.0 Teasers! - #30 by zachdaniel
Teaser #6: Ash 3.0 Teasers! - #36 by zachdaniel
Most Liked
zachdaniel
Hey everyone, time for Teaser #2!
If you missed teaser #1, check out out here: Ash 3.0 Teasers!
This is a big one ![]()
Ash.Api → Ash.Domain
Ash.Api as a name has caused users lots of confusion in the past. It is an overloaded term. Also, while we considered it the “API to a given bounded context” as in, you always interacted with a resource through an API, we’ve ultimately decided to change the conceptual role of what we call an API. With that change, comes a name that is more representative of its function. Shoutout to @lukasender for the suggestion that stuck.
Ash.Domain represents the configuration of a domain, which includes things like “what resources are available to this domain”, as well as other high level configurations. Instead of calling to an API module, we provide one standard interface, but that interface must always be able to determine what Domain it is interacting with, in addition to what Resource it is working with. Keep reading for more on what this looks like.
domain option to use Ash.Resource
When creating a resource, you pass the domain option. For example:
defmodule MyApp.Accounts.User do
use Ash.Resource,
domain: MyApp.Accounts
end
This static configuration can be used in various places where you would have previously had to specify an api. For example, you no longer need to specify define_for in the code_interface block.
You will get a warning if you don’t pass the domain option, as well as if the configured domain doesn’t know about the resource.
Which leads us to one of the more significant changes of 3.0:
Using Ash instead of MyApp.MyApi
We will be deprecating the functions that we typically on MyApp.MyApi, in favor of those same functions defined in the Ash module. This allows us to refactor resources and move them from one domain to another without having to hunt down the calling code that interacts with it. It also helps reduce the cognitive overhead of having to remember what Api you’re working with for any given resource in order to call an action on it.
For example:
MyApp.Helpdesk.Ticket
|> Ash.Changeset.for_create(:open, %{title: "halp"})
|> MyApp.Helpdesk.create
MyApp.Helpdesk.count!(MyApp.Helpdesk.Ticket)
MyApp.Helpdesk.Ticket
|> Ash.Query.for_read(:open)
|> MyApp.Helpdesk.read!()
Would become
MyApp.Helpdesk.Ticket
|> Ash.Changeset.for_create(:open, %{title: "halp"})
|> Ash.create
Ash.count!(MyApp.Helpdesk.Ticket)
MyApp.Helpdesk.Ticket
|> Ash.Query.for_read(:open)
|> Ash.read!()
How to make this change
Good news! You can make this change before upgrading to 3.0. You can do everything stated above by configuring the api option when calling use Ash.Resource, and switching your calls to the api to call Ash! The difference between 2.0 and 3.0 is that in 3.0 the functions defined on the api module will be deprecated, and api has been renamed to domain.
Teaser #3: Ash 3.0 Teasers! - #23 by zachdaniel
zachdaniel
Ash 3.0 Teaser #6: A cherry on top
This will be the final teaser before the 3.0 release candidates come out!
Ash.ToTenant
A common case is to have a tenant represented by a resource, like %Organization{} or %Tenant{}, but a tenant is always identified by a simple value, like a string or an integer. Because of this, there is often code that looks like this:
Ash.Changeset.for_update(%Record{}, tenant: "org_#{org.id}")
There is also complexity when you have a mix of multi tenancy strategies, like if one resource uses schema-based multi tenancy, and the rest use attribute. The Ash.ToTenant protocol simplifies this, and allows you to use the same tenant everywhere, but have a different derived tenant value. Here is an example of how you might use it:
# in Organization resource
defimpl Ash.ToTenant do
def to_tenant(resource, %MyApp.Accounts.Organization{id: id}) do
if Ash.Resource.Info.data_layer(resource) == AshPostgres.DataLayer
&& Ash.Resource.Info.multitenancy_strategy(resource) == :context do
"org_#{id}"
else
id
end
end
end
Sensitive Calculations & Aggregates
In the same way that you could specify attributes as sensitive?: true, you can now specify calculations and aggregates as sensitive. These will be redacted when inspecting records, and will also be redacted when inspecting filters inside of queries.
Code interfaces support atomic & bulk actions
In 2.0, you need to look up a record before you can update or destroy it, unless you change your code to use YourApi.bulk_update or YourApi.bulk_destroy (in 3.0, Ash.bulk_update and Ash.bulk_destroy). This can be quite verbose. For example, let’s say you have the id of a thing, and you want to update it. The most idiomatic way would have been something like this:
MyApp.Blog.Post
|> Ash.get!(id)
|> Post.archive!()
Or alternatively, you could have opted not to use the code interface, and used Ash.bulk_update. For example:
MyApp.Blog.Post
|> Ash.Query.filter(id == ^id)
|> Ash.bulk_update(:archive, %{....})
But then you don’t get to use your nicely defined code interface, which acts like a context function which fills the role of a context function in Phoenix.
In Ash 3.0, code interfaces have been updated to support bulk operations, which makes cases like the above much more seamless!
For updates/destroys, you can pass identifiers, queries, and lists/streams of inputs directly instead of a record or a changeset. From here on, we’ll also be using code interface functions defined on our domain instead of our resources, which is the recommended way in 3.0. See previous teasers for more.
Update/Destroy examples:
# If the action can be done atomically (i.e without looking up the record), it will be.
# Otherwise, we will look up the record and update it.
# => MyApp.Blog.archive_post!(post.id)
# queries can be provided, which will return an `Ash.BulkResult`
Post
|> Ash.Query.filter(author_id == ^author_id)
|> MyApp.Blog.archive_post!()
# => %Ash.BulkResult{}
# lists of records can also be provided, also returning an `Ash.BulkResult`
[%Post{}, %Post{}]
|> MyApp.Blog.archive_post!()
# => %Ash.BulkResult{}
Create
For creates, we detect if the input is a list (and not a keyword list), and opt into bulk create behavior:
# no need for the additional `define ..., bulk?: true`
Blog.create_post!([%{...inputs}, %{...inputs}]
EDIT: Below was the original section on bulk creates. Feel free to ignore it, but leaving it for posterity. @vonagam has made a good point in a discussion on discord, there is no reason that we can’t just detect a list of inputs in the input argument, and use that as a bulk create. This allows us not to need the bulk? true option, and they can function the same as the others, adapting their behavior based on the input. For example:
Create’s don’t take a first argument like updates/destroys, and so a bulk create must be explicitly defined. For example:
# in the domain
resource Post do
define :create_posts do
action :create
bulk? true
end
And it can be used like so:
Blog.create_posts!([%{...inputs}, %{...inputs}]
Streaming reads
You can also now ask read action code interfaces to return streams. Keep in mind that Ash streams are not (yet) based on data-layer-native streams, but rather will use your actions pagination functionality (preferring keyset vs offset pagination). Only the raising version (!) supports the :stream option, because streams can only raise errors, not return them.
For example:
Blog.active_posts!(stream?: true)
# => returns a lazily enumerable stream
The light at the end of the tunnel
3.0 is very close, and I’m so excited! Thanks again to everyone who has been a part of it. For those adventurous folks, the release candidates will be out soon for you to have a play with ![]()
zachdaniel
Ash 3.0 Teaser #3: Better Defaults, Less Surprises, Part 1
I meant to make this post a few days ago, but I got a bit busy handling a heisenbug that a user found when using aggregates in policies
. That is all sorted now so I can focus on making progress on 3.0 again! ![]()
You may also notice that, from here on out, we will be exchanging the term api with domain. See teaser #2 for more ![]()
A big part of the purpose of 3.0 is changing default behaviors. Ash 2.0 had a lot of “permissive” defaults that could make things extremely quick to get started but could easily bite you later down the road. Ultimately, with Ash, we care more about “year five” than “day one”, so we don’t really want to make design choices that will be a foot-gun later down the road for the sake of easy initial adoption.
There are a lot of these changes, so this will be a two parter!
domain.authorization.authorize now defaults to :by_default
The original default was :when_requested, which would cause authorization to trigger when an actor option was provided, or when authorize?: true was set. This makes it very easy to accidentally forget to authorize an action invocation. :by_default always sets authorize? true (has no effect if you are not using authorizers), unless you explicitly say authorize?: false..
We’ve known for a very long time that this was not the ideal default behavior, but it was a significant breaking change and so had to wait for 3.0. You can revert to the old behavior (but should eventually update) by setting the value in each domain module back to :when_requested.
unknown action inputs now produce errors
When passing parameters to actions, we would previously ignore unknown parameters. This made it very easy to misspell an input and not realize it, or otherwise believe that an input was being used when it wasn’t.
Bulk update/destroy strategy defaults to :atomic
When calling a bulk update/destroy, it may not be able to be done atomically (i.e one single UPDATE query + after hooks). You can specify a list of allowed strategies when calling a bulk action. The strategies available are
atomic- Must be doable as a single operation at the data layer (plus after action/after batch logic)atomic_batches- Can be a series of atomic operations (as above). This can be great for massive inputs where you want to do them in batches. What we do is stream the provided query or list into batches, and update by primary key as a single atomic operation.stream- We stream each record and do the logic to update each record one at a time.
Ash will use the “best” strategy that it can use (i.e the first one that can be used in the list above starting at the top). In 2.0, the default for the strategy option is [:atomic, :atomic_batches, :stream], allowing all three by default.
In 3.0 the default for the strategy option is [:atomic]. This is in line with making defaults something that will help you choose the safest option. If an action cannot be done atomically, you will be told a reason, and you can adjust the strategy option or modify the action.
require_atomic? on update/destroy actions defaults to true
When writing actions, we want them to be concurrency safe. What this means is that no update/destroy action will be performed non atomically (in this context, to be done atomically means that all changes, validations, and attribute changes can be done atomically. You will get a warning at compile time if your action is known not to be able to be done atomically, and an error at runtime if you attempt to run them.
For example, what might happen is something like this:
update :update do
# I add an anonymous function change
change fn changeset, _ ->
Ash.Changeset.change_attribute(changeset, :attribute, :value)
end
end
I get an error at compile time, like this:
warning: [Resource]
actions -> update:
update cannot be done atomically, because the changes `[Ash.Resource.Change.Function]` cannot be done atomically
So I adjust the action to use a builtin change that has an atomic implementation. I could also extract the logic into its own module and use Ash.Resource.Change and add the atomic/2 callback.
update :update do
# I add an anonymous function change
change set_attribute(:attribute, :value)
end
And now we’re good! I can also add require_atomic? false to the action if I know that the changes on this action are safe to run non-atomically.
update :update do
require_atomic? false
# I add an anonymous function change
change fn changeset, _ ->
Ash.Changeset.change_attribute(changeset, :attribute, :value)
end
end
Closing
This one was a lot, and there are more to come! Keep in mind that not all breaking changes will be included in these teasers, but the goal is to include all major/significant changes. Can’t wait for 3.0 to get out there.
Teaser #4: Ash 3.0 Teasers! - #28 by zachdaniel







