When using the phoenix generators to create a new context (i.e. mix phx.gen.context ContextName SchemaName table_name column_name:column_type), I continually run into an issue where the default schema being used in the new module is Ecto.Schema. The below code is an example of the code that would be generated using the command above:
defmodule MyApp.ContextName.SchemaName do
use Ecto.Schema
import Ecto.Changeset
...
end
My question is: is there a way to set the default schema to MyApp.Schema, instead of Ecto.Schema? If any app specific schema configurations have been implemented (such as primary_keys defaulting to uuid instead of integer_id), the generated schema will throw errors when you try to use it. It winds up being one of those problems that you donât fix until you remember that you forgot to change use Ecto.Schema to use MyApp.Schema.
When you run a generator like phx.gen.context the content is obtained from a template. You can find the template in Phoenixâs source at /priv/templates/phx.gen.schema/schema.ex. When the generator is run, there is dynamic content generated such as the module name, fields, etc. But use Ecto.Schema is not a dynamically generated. Check it out:
# /priv/templates/phg.gen.schema/schema.ex
defmodule <%= inspect schema.module %> do
use Ecto.Schema
import Ecto.Changeset
<%= if schema.prefix do %>
@schema_prefix :<%= schema.prefix %><% end %><%= if schema.binary_id do %>
@primary_key {:id, :binary_id, autogenerate: true}
@foreign_key_type :binary_id<% end %>
schema <%= inspect schema.table %> do
<%= for {k, v} <- schema.types do %> field <%= inspect k %>, <%= Mix.Phoenix.Schema.type_and_opts_for_schema(v) %><%= schema.defaults[k] %><%= Mix.Phoenix.Schema.maybe_redact_field(k in schema.redacts) %>
<% end %><%= for {_, k, _, _} <- schema.assocs do %> field <%= inspect k %>, <%= if schema.binary_id do %>:binary_id<% else %>:id<% end %>
<% end %>
timestamps()
end
@doc false
def changeset(<%= schema.singular %>, attrs) do
<%= schema.singular %>
|> cast(attrs, [<%= Enum.map_join(schema.attrs, ", ", &inspect(elem(&1, 0))) %>])
|> validate_required([<%= Enum.map_join(schema.attrs, ", ", &inspect(elem(&1, 0))) %>])
<%= for k <- schema.uniques do %> |> unique_constraint(<%= inspect k %>)
<% end %> end
end
There is no configuration variable or command line argument that will achieve what youâre looking for. Some of your options may be:
Edit as you generate,
Automate the changes within your editor or a command line program like sed, or
Clone Phoenix and then edit the templates. You can then generate/move your project as you would if creating a project from an unreleased version. â this seems messy to me.
If I was in your position, I would create an interactive Emacs command to take care of renaming Ecto.Schema to MyApp.Schema after generating a schema.
I vaguely remember that you can copy the Phoenix source file /priv/templates/phx.gen.schema/schema.ex to your own project and edit the file and this version will be used by the generators.