anibal
Having trouble understanding the Centralized Code Interface code example in the Ash Docs
I have been studying the the beta of the Ash Book and going through the docs, and I am having some difficulties grasping this code example from the docs
defmodule MyApp.Tweets do
use Ash.Domain
resources do
resource MyApp.Tweets.Tweet do
# define a function called `tweet` that uses
# the `:create` action on MyApp.Tweets.Tweet
define :tweet, action: :create, args: [:text]
end
resource MyApp.Tweets.Comment do
# define a function called `comment` that uses
# the `:create` action on MyApp.Tweets.Comment
define :comment, action: :create, args: [:tweet_id, :text]
end
end
end
Which is supposed to allow:
tweet = MyApp.Tweets.tweet!("My first tweet!", actor: user1)
comment = MyApp.Tweets.comment!(tweet.id, "What a cool tweet!", actor: user2)
In my mind the define is operating nested inside a Resource definition and the tweet! and comment! functions should be defined at the Resource level and not at the Domain level.
Again in my mind the define “should” work something like:
define :tweet, module: :MyApp.Tweets.Tweet, action: :create, args: [:text]
Not sure if this is related to me trying to learn Elixir at the same time that I am learning Ash, for sure that is not helping and I have a deep dive on macros and alikes pending.
Thanks in advance on your guidance!
Most Liked
ken-kost
Imagine if you had a 100 interfaces for a resource (extreme example), in your example you would have to repeate same module/resource for each. The way it is now, you just put them in the resource block; or define them in the resource.
But I think I understand what you mean: it’s more intuitive to define it as an option. But in the case of defining an interface in a resource, what then? You wouldn’t mention the module? It would be less consistent.
sodapopcan
Ya it’s funny—I think it does have to do with being new to Elixir because I see what OP means now that they’ve pointed it out but this never once occurred to me before!
@anibal maybe thinking of it as with_resource do would help but as mentioned, it’s really just about getting a clean way to define things. Macro-heavy DSLs can take some getting used to. It’s probably easier to not think of resources and resource as function calls but as a simplified language for creating data structures that are then fed into “real” functions. Incidentally, this is exactly what is happening behind the scenes!
For example, would the following be as surprising? (Note, this is a completely invented simplified example):
defmodule MyApp.Tweets do
@behaviour Ash.Domain
@impl true
def domain do
%{
resources: [
%{
module: MyApp.Tweets.Tweet,
define: [
:tweet, action: :create, args: [:text]
]
},
%{
module: MyApp.Tweets.Comment,
define: [
:comment, action: :create, args: [:tweet_id, :text]
]
}
]
}
end
end
Ash could have been done in a manner like that, but the DSL is much nicer.
zachdaniel
By defining the interface on the domain, you get something more akin to a “service”. i.e
MyApp.Accounts.register_user!(...)
vs
MyApp.Accounts.User.register_user!()
Defining them on the domain for the most part decouples the caller from the resource in charge of registering said user.
Popular in Questions
Other popular topics
Categories:
Sub Categories:
Forums
Popular Tags
- #ecto
- #liveview
- #troubleshooting
- #learning-elixir
- #deployment
- #library
- #erlang
- #testing
- #genserver
- #mix
- #absinthe
- #remote-other
- #otp
- #plug
- #how-to-question
- #macros
- #postgres
- #channels
- #elixirconf
- #exunit
- #discussion
- #javascript
- #code-sync
- #podcasts
- #onsite
- #dialyzer
- #docker
- #authentication
- #umbrella
- #full-time-contract
- #podcasts-by-brainlid
- #ecto-query
- #elixir-ls
- #phoenix_html
- #iex
- #blog-post
- #graphql
- #genstage
- #ai
- #websockets
- #supervisor
- #advent-of-code
- #elixirconf-us
- #distillery
- #processes
- #forms
- #api
- #metaprogramming
- #security
- #performance








