Claude Code Plugins

Community-maintained marketplace

Feedback

Rules for working with AshPhoenix

Install Skill

1Download skill
2Enable skills in Claude

Open claude.ai/settings/capabilities and find the "Skills" section

3Upload to Claude

Click "Upload skill" and select the downloaded ZIP file

Note: Please verify skill by going through its instructions before using it.

SKILL.md

name ash-phoenix
description Rules for working with AshPhoenix

Rules for working with AshPhoenix

Understanding AshPhoenix

AshPhoenix is a package for integrating Ash Framework with Phoenix Framework. It provides tools for integrating with Phoenix forms (AshPhoenix.Form), Phoenix LiveViews (AshPhoenix.LiveView), and more. AshPhoenix makes it seamless to use Phoenix's powerful UI capabilities with Ash's data management features.

Form Integration

AshPhoenix provides AshPhoenix.Form, a powerful module for creating and handling forms backed by Ash resources.

Creating Forms

# For creating a new resource
form = AshPhoenix.Form.for_create(MyApp.Blog.Post, :create) |> to_form()

# For updating an existing resource
post = MyApp.Blog.get_post!(post_id)
form = AshPhoenix.Form.for_update(post, :update) |> to_form()

# Form with initial value
form = AshPhoenix.Form.for_create(MyApp.Blog.Post, :create,
  params: %{title: "Draft Title"}
) |> to_form()

Code Interfaces

Using the AshPhoenix extension in domains gets you special functions in a resource's code interface called form_to_*. Use this whenever possible.

First, add the AshPhoenix extension to our domains and resources, like so:

use Ash.Domain,
  extensions: [AshPhoenix]

which will cause another function to be generated for each definition, beginning with form_to_.

For example, if you had the following,

# in MyApp.Accounts
resources do
  resource MyApp.Accounts.User do
    define :register_with_password, args: [:email, :password]
  end
end

you could then make a form with:

MyApp.Accounts.form_to_register_with_password(...opts)

By default, the args option in define is ignored when building forms. If you want to have positional arguments, configure that in the forms section which is added by the AshPhoenix section. For example:

forms do
  form :register_with_password, args: [:email]
end

Which could then be used as:

MyApp.Accounts.register_with_password(email, ...)

These positional arguments are very important for certain cases, because there may be values you do not want the form to be able to set. For example, when updating a user's settings, maybe the action takes a user_id, but the form is on a page for a specific user's id and so this should therefore not be editable in the form. Use positional arguments for this.

Handling Form Submission

In your LiveView:

def handle_event("validate", %{"form" => params}, socket) do
  form = AshPhoenix.Form.validate(socket.assigns.form, params)
  {:noreply, assign(socket, :form, form)}
end

def handle_event("submit", %{"form" => params}, socket) do
  case AshPhoenix.Form.submit(socket.assigns.form, params: params) do
    {:ok, post} ->
      socket =
        socket
        |> put_flash(:info, "Post created successfully")
        |> push_navigate(to: ~p"/posts/#{post.id}")
      {:noreply, socket}

    {:error, form} ->
      {:noreply, assign(socket, :form, form)}
  end
end

Nested Forms

AshPhoenix supports forms with nested relationships, such as creating or updating related resources in a single form.

Automatically Inferred Nested Forms

If your action has manage_relationship, AshPhoenix automatically infers nested forms:

# In your resource:
create :create do
  accept [:name]
  argument :locations, {:array, :map}
  change manage_relationship(:locations, type: :create)
end

# In your template:
<.simple_form for={@form} phx-change="validate" phx-submit="submit">
  <.input field={@form[:name]} />

  <.inputs_for :let={location} field={@form[:locations]}>
    <.input field={location[:name]} />
  </.inputs_for>
</.simple_form>

Adding and Removing Nested Forms

To add a nested form with a button:

<.button type="button" phx-click="add-form" phx-value-path={@form.name <> "[locations]"}>
  <.icon name="hero-plus" />
</.button>

In your LiveView:

def handle_event("add-form", %{"path" => path}, socket) do
  form = AshPhoenix.Form.add_form(socket.assigns.form, path)
  {:noreply, assign(socket, :form, form)}
end

To remove a nested form:

<.button type="button" phx-click="remove-form" phx-value-path={location.name}>
  <.icon name="hero-x-mark" />
</.button>
def handle_event("remove-form", %{"path" => path}, socket) do
  form = AshPhoenix.Form.remove_form(socket.assigns.form, path)
  {:noreply, assign(socket, :form, form)}
end

Union Forms

AshPhoenix supports forms for union types, allowing different inputs based on the selected type.

<.inputs_for :let={fc} field={@form[:content]}>
  <.input
    field={fc[:_union_type]}
    phx-change="type-changed"
    type="select"
    options={[Normal: "normal", Special: "special"]}
  />

  <%= case fc.params["_union_type"] do %>
    <% "normal" -> %>
      <.input type="text" field={fc[:body]} />
    <% "special" -> %>
      <.input type="text" field={fc[:text]} />
  <% end %>
</.inputs_for>

In your LiveView:

def handle_event("type-changed", %{"_target" => path} = params, socket) do
  new_type = get_in(params, path)
  path = :lists.droplast(path)

  form =
    socket.assigns.form
    |> AshPhoenix.Form.remove_form(path)
    |> AshPhoenix.Form.add_form(path, params: %{"_union_type" => new_type})

  {:noreply, assign(socket, :form, form)}
end

Error Handling

AshPhoenix provides helpful error handling mechanisms:

# In your LiveView
def handle_event("submit", %{"form" => params}, socket) do
  case AshPhoenix.Form.submit(socket.assigns.form, params: params) do
    {:ok, post} ->
      # Success path
      {:noreply, success_path(socket, post)}

    {:error, form} ->
      # Show validation errors
      {:noreply, assign(socket, form: form)}
  end
end

Debugging Form Submission

Errors on forms are only shown when they implement the AshPhoenix.FormData.Error protocol and have a field or fields set. Most Phoenix applications are set up to show errors for <.inputs. This can some times lead to errors happening in the action that are not displayed because they don't implement the protocol, have field/fields, or for a field that is not shown in the form.

To debug these situations, you can use AshPhoenix.Form.raw_errors(form, for_path: :all) on a failed form submission to see what is going wrong, and potentially add custom error handling, or resolve whatever error is occurring. If the action has errors that can go wrong that aren't tied to fields, you will need to detect those error scenarios and display that with some other UI, like a flash message or a notice at the top/bottom of the form, etc.

If you want to see what errors the form will see (that implement the protocl and have fields) use AshPhoenix.Form.errors(form, for_path: :all).

Best Practices

  1. Let the Resource guide the UI: Your Ash resource configuration determines a lot about how forms and inputs will work. Well-defined resources with appropriate validations and changes make AshPhoenix more effective.

  2. Leverage code interfaces: Define code interfaces on your domains for a clean and consistent API to call your resource actions.

  3. Update resources before editing: When building forms for updating resources, load the resource with all required relationships using Ash.load!/2 before creating the form.