ash_graphql/lib/resource/info.ex

194 lines
6.2 KiB
Elixir
Raw Normal View History

2022-08-31 13:08:16 +12:00
defmodule AshGraphql.Resource.Info do
@moduledoc "Introspection helpers for AshGraphql.Resource"
alias Spark.Dsl.Extension
@doc "The queries exposed for the resource"
def queries(resource, domain_or_domains \\ []) do
module =
if is_atom(resource) do
resource
else
Spark.Dsl.Extension.get_persisted(resource, :module)
end
domain_or_domains
|> List.wrap()
|> Enum.flat_map(&AshGraphql.Domain.Info.queries/1)
|> Enum.filter(&(&1.resource == module))
|> Enum.concat(Extension.get_entities(resource, [:graphql, :queries]))
2022-08-31 13:08:16 +12:00
end
@doc "The mutations exposed for the resource"
def mutations(resource, domain_or_domains \\ []) do
module =
if is_atom(resource) do
resource
else
Spark.Dsl.Extension.get_persisted(resource, :module)
end
domain_or_domains
|> List.wrap()
|> Enum.flat_map(&AshGraphql.Domain.Info.mutations/1)
|> Enum.filter(&(&1.resource == module))
|> Enum.concat(Extension.get_entities(resource, [:graphql, :mutations]) || [])
2022-08-31 13:08:16 +12:00
end
@doc "Wether or not to encode the primary key as a single `id` field when reading and getting"
def encode_primary_key?(resource) do
Extension.get_opt(resource, [:graphql], :encode_primary_key?, true)
end
2022-08-31 13:08:16 +12:00
@doc "The managed_relationship configurations"
def managed_relationships(resource) do
Extension.get_entities(resource, [:graphql, :managed_relationships]) || []
end
def managed_relationships_auto?(resource) do
Extension.get_opt(resource, [:graphql, :managed_relationships], :auto?, true)
end
@doc "The managed_relationshi configuration for a given action/argument"
def managed_relationship(resource, action, argument) do
resource
|> Extension.get_entities([:graphql, :managed_relationships])
|> List.wrap()
|> Enum.find(fn managed_relationship ->
managed_relationship.argument == argument.name and
managed_relationship.action == action.name
end)
|> then(fn managed_relationship ->
if managed_relationship && managed_relationship.ignore? do
nil
else
if managed_relationships_auto?(resource) do
managed_relationship || default_managed_relationship(action, argument)
else
managed_relationship
end
end
end)
end
defp default_managed_relationship(action, argument) do
if Enum.any?(Map.get(action, :changes, []), fn
%{change: {Ash.Resource.Change.ManageRelationship, opts}} ->
opts[:argument] == argument.name
_ ->
nil
end) && map_type?(argument.type) do
%AshGraphql.Resource.ManagedRelationship{
argument: argument.name,
action: action,
types: [],
type_name: nil,
lookup_with_primary_key?: true,
lookup_identities: []
}
end
end
defp map_type?({:array, type}), do: map_type?(type)
defp map_type?(Ash.Type.Map), do: true
defp map_type?(:map), do: true
defp map_type?(_), do: false
2022-08-31 13:08:16 +12:00
@doc "The graphql type of the resource"
def type(resource) do
Extension.get_opt(resource, [:graphql], :type, nil)
end
@doc "Wether or not to derive a filter input for the resource automatically"
def derive_filter?(resource) do
Extension.get_opt(resource, [:graphql], :derive_filter?, true)
end
@doc "Wether or not to derive a sort input for the resource automatically"
def derive_sort?(resource) do
Extension.get_opt(resource, [:graphql], :derive_sort?, true)
end
@doc "Graphql type overrides for the resource"
def attribute_types(resource) do
Extension.get_opt(resource, [:graphql], :attribute_types, [])
end
@doc "The field name to place the keyset of a result in"
def keyset_field(resource) do
Extension.get_opt(resource, [:graphql], :keyset_field, nil)
end
@doc "Graphql field name (attribute/relationship/calculation/arguments) overrides for the resource"
def field_names(resource) do
Extension.get_opt(resource, [:graphql], :field_names, [])
end
improvement!: port AshGraphql to Ash 3.0 (#123) Step 1: update Ash Step 2: mass rename Api to Domain Step 3: Ash.Query.expr -> Ash.Expr.expr Also change ref interpolation Step 4: remove all warnings Step 5: remove registries from tests Step 6: fix filter Step 7: private? -> !public? Step 8: Ash.Calculation -> Ash.Resource.Calculation Step 9: use depend_on_resources/1 -> resources/1 Step 10: add Domain to all resources Step 11: use Ash module for all actions Step 12: add public? true all around Step 13: remove verbose? from options passed during Domain calls Step 14: add simple_sat Step 15: Ash.ErrorKind is no more, so remove code from errors Step 16: sprinkle default_accept :* around tests Step 17: replace Ash.Changeset.new/2 with Ash.Changeset.for_* Step 18: calculation fixups - Context is now a struct and arguments go under the arguments key - Function based calculations receive a list of records - Add a select to query-based loads - select -> load Step 19: pass the correct name to pass the policy in tests Step 20: Ash.Query.new/2 is no more Step 21: add AshGraphql.Resource.embedded? utility function Use that instead of Ash.Type.embedded_type?(resource_or_type) since resources are not types anymore Step 22: handle struct + instance_of: Resource in unions Resources are not type anymore so they need to be passed this way in unions Step 23: ensure we only check GraphQL actions for pagination All reads are now paginated by default, so this triggered a compilation error Step 24: swap arguments for sort on calculations Step 25: remove unused debug? option
2024-04-02 07:03:06 +13:00
@doc "Fields to hide from the graphql domain"
2023-04-27 15:40:25 +12:00
def hide_fields(resource) do
Extension.get_opt(resource, [:graphql], :hide_fields, [])
end
improvement!: port AshGraphql to Ash 3.0 (#123) Step 1: update Ash Step 2: mass rename Api to Domain Step 3: Ash.Query.expr -> Ash.Expr.expr Also change ref interpolation Step 4: remove all warnings Step 5: remove registries from tests Step 6: fix filter Step 7: private? -> !public? Step 8: Ash.Calculation -> Ash.Resource.Calculation Step 9: use depend_on_resources/1 -> resources/1 Step 10: add Domain to all resources Step 11: use Ash module for all actions Step 12: add public? true all around Step 13: remove verbose? from options passed during Domain calls Step 14: add simple_sat Step 15: Ash.ErrorKind is no more, so remove code from errors Step 16: sprinkle default_accept :* around tests Step 17: replace Ash.Changeset.new/2 with Ash.Changeset.for_* Step 18: calculation fixups - Context is now a struct and arguments go under the arguments key - Function based calculations receive a list of records - Add a select to query-based loads - select -> load Step 19: pass the correct name to pass the policy in tests Step 20: Ash.Query.new/2 is no more Step 21: add AshGraphql.Resource.embedded? utility function Use that instead of Ash.Type.embedded_type?(resource_or_type) since resources are not types anymore Step 22: handle struct + instance_of: Resource in unions Resources are not type anymore so they need to be passed this way in unions Step 23: ensure we only check GraphQL actions for pagination All reads are now paginated by default, so this triggered a compilation error Step 24: swap arguments for sort on calculations Step 25: remove unused debug? option
2024-04-02 07:03:06 +13:00
@doc "Fields to show in the graphql domain"
def show_fields(resource) do
Extension.get_opt(resource, [:graphql], :show_fields, nil)
end
2023-04-27 15:40:25 +12:00
@doc "Wether or not a given field will be shown"
def show_field?(resource, field) do
hide_fields = hide_fields(resource)
show_fields = show_fields(resource) || [field]
field not in hide_fields and field in show_fields
2023-04-27 15:40:25 +12:00
end
@doc "Which relationships should be included in the generated type"
def relationships(resource) do
Extension.get_opt(resource, [:graphql], :relationships, nil) ||
resource |> Ash.Resource.Info.public_relationships() |> Enum.map(& &1.name)
end
2024-05-27 01:50:54 +12:00
@doc "Pagination configuration for list relationships"
def paginate_relationship_with(resource) do
Extension.get_opt(resource, [:graphql], :paginate_relationship_with, [])
end
@doc "Graphql argument name overrides for the resource"
def argument_names(resource) do
Extension.get_opt(resource, [:graphql], :argument_names, [])
end
@doc "Graphql attribute input type overrides for the resource"
def attribute_input_types(resource) do
Extension.get_opt(resource, [:graphql], :attribute_input_types, [])
end
@doc "Graphql argument type overrides for the resource"
def argument_input_types(resource) do
Extension.get_opt(resource, [:graphql], :argument_input_types, [])
end
2022-08-31 13:08:16 +12:00
@doc "The delimiter for a resource with a composite primary key"
def primary_key_delimiter(resource) do
Extension.get_opt(resource, [:graphql], :primary_key_delimiter, "-")
2022-08-31 13:08:16 +12:00
end
@doc "Wether or not an object should be generated, or if one with the type name for this resource should be used."
def generate_object?(resource) do
Extension.get_opt(resource, [:graphql], :generate_object?, true)
2022-08-31 13:08:16 +12:00
end
@doc "Fields that may be filtered on"
def filterable_fields(resource) do
Extension.get_opt(resource, [:graphql], :filterable_fields, nil)
end
@doc "May the specified field be filtered on?"
def filterable_field?(resource, field) do
filterable_fields = AshGraphql.Resource.Info.filterable_fields(resource)
is_nil(filterable_fields) or field.name in filterable_fields
end
2022-08-31 13:08:16 +12:00
end