161 lines
5.9 KiB
Elixir
161 lines
5.9 KiB
Elixir
|
defmodule Mix.Tasks.Phx.Gen.Json do
|
||
|
@shortdoc "Generates controller, views, and context for a JSON resource"
|
||
|
|
||
|
@moduledoc """
|
||
|
Generates controller, views, and context for a JSON resource.
|
||
|
|
||
|
mix phx.gen.json Accounts User users name:string age:integer
|
||
|
|
||
|
The first argument is the context module followed by the schema module
|
||
|
and its plural name (used as the schema table name).
|
||
|
|
||
|
The context is an Elixir module that serves as an API boundary for
|
||
|
the given resource. A context often holds many related resources.
|
||
|
Therefore, if the context already exists, it will be augmented with
|
||
|
functions for the given resource.
|
||
|
|
||
|
> Note: A resource may also be split
|
||
|
> over distinct contexts (such as `Accounts.User` and `Payments.User`).
|
||
|
|
||
|
The schema is responsible for mapping the database fields into an
|
||
|
Elixir struct. It is followed by an optional list of attributes,
|
||
|
with their respective names and types. See `mix phx.gen.schema`
|
||
|
for more information on attributes.
|
||
|
|
||
|
Overall, this generator will add the following files to `lib/`:
|
||
|
|
||
|
* a context module in `lib/app/accounts.ex` for the accounts API
|
||
|
* a schema in `lib/app/accounts/user.ex`, with an `users` table
|
||
|
* a view in `lib/app_web/views/user_view.ex`
|
||
|
* a controller in `lib/app_web/controllers/user_controller.ex`
|
||
|
|
||
|
A migration file for the repository and test files for the context and
|
||
|
controller features will also be generated.
|
||
|
|
||
|
## The context app
|
||
|
|
||
|
The location of the web files (controllers, views, templates, etc) in an
|
||
|
umbrella application will vary based on the `:context_app` config located
|
||
|
in your applications `:generators` configuration. When set, the Phoenix
|
||
|
generators will generate web files directly in your lib and test folders
|
||
|
since the application is assumed to be isolated to web specific functionality.
|
||
|
If `:context_app` is not set, the generators will place web related lib
|
||
|
and test files in a `web/` directory since the application is assumed
|
||
|
to be handling both web and domain specific functionality.
|
||
|
Example configuration:
|
||
|
|
||
|
config :my_app_web, :generators, context_app: :my_app
|
||
|
|
||
|
Alternatively, the `--context-app` option may be supplied to the generator:
|
||
|
|
||
|
mix phx.gen.json Sales User users --context-app warehouse
|
||
|
|
||
|
## Web namespace
|
||
|
|
||
|
By default, the controller and view will be namespaced by the schema name.
|
||
|
You can customize the web module namespace by passing the `--web` flag with a
|
||
|
module name, for example:
|
||
|
|
||
|
mix phx.gen.json Sales User users --web Sales
|
||
|
|
||
|
Which would generate a `lib/app_web/controllers/sales/user_controller.ex` and
|
||
|
`lib/app_web/views/sales/user_view.ex`.
|
||
|
|
||
|
## Customizing the context, schema, tables and migrations
|
||
|
|
||
|
In some cases, you may wish to bootstrap JSON views, controllers,
|
||
|
and controller tests, but leave internal implementation of the context
|
||
|
or schema to yourself. You can use the `--no-context` and `--no-schema`
|
||
|
flags for file generation control.
|
||
|
|
||
|
You can also change the table name or configure the migrations to
|
||
|
use binary ids for primary keys, see `mix phx.gen.schema` for more
|
||
|
information.
|
||
|
"""
|
||
|
|
||
|
use Mix.Task
|
||
|
|
||
|
alias Mix.Phoenix.Context
|
||
|
alias Mix.Tasks.Phx.Gen
|
||
|
|
||
|
@doc false
|
||
|
def run(args) do
|
||
|
if Mix.Project.umbrella?() do
|
||
|
Mix.raise "mix phx.gen.json must be invoked from within your *_web application root directory"
|
||
|
end
|
||
|
|
||
|
{context, schema} = Gen.Context.build(args)
|
||
|
Gen.Context.prompt_for_code_injection(context)
|
||
|
|
||
|
binding = [context: context, schema: schema]
|
||
|
paths = Mix.Phoenix.generator_paths()
|
||
|
|
||
|
prompt_for_conflicts(context)
|
||
|
|
||
|
context
|
||
|
|> copy_new_files(paths, binding)
|
||
|
|> print_shell_instructions()
|
||
|
end
|
||
|
|
||
|
defp prompt_for_conflicts(context) do
|
||
|
context
|
||
|
|> files_to_be_generated()
|
||
|
|> Kernel.++(context_files(context))
|
||
|
|> Mix.Phoenix.prompt_for_conflicts()
|
||
|
end
|
||
|
defp context_files(%Context{generate?: true} = context) do
|
||
|
Gen.Context.files_to_be_generated(context)
|
||
|
end
|
||
|
defp context_files(%Context{generate?: false}) do
|
||
|
[]
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
def files_to_be_generated(%Context{schema: schema, context_app: context_app}) do
|
||
|
web_prefix = Mix.Phoenix.web_path(context_app)
|
||
|
test_prefix = Mix.Phoenix.web_test_path(context_app)
|
||
|
web_path = to_string(schema.web_path)
|
||
|
|
||
|
[
|
||
|
{:eex, "controller.ex", Path.join([web_prefix, "controllers", web_path, "#{schema.singular}_controller.ex"])},
|
||
|
{:eex, "view.ex", Path.join([web_prefix, "views", web_path, "#{schema.singular}_view.ex"])},
|
||
|
{:eex, "controller_test.exs", Path.join([test_prefix, "controllers", web_path, "#{schema.singular}_controller_test.exs"])},
|
||
|
{:new_eex, "changeset_view.ex", Path.join([web_prefix, "views/changeset_view.ex"])},
|
||
|
{:new_eex, "fallback_controller.ex", Path.join([web_prefix, "controllers/fallback_controller.ex"])},
|
||
|
]
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
def copy_new_files(%Context{} = context, paths, binding) do
|
||
|
files = files_to_be_generated(context)
|
||
|
Mix.Phoenix.copy_from paths, "priv/templates/phx.gen.json", binding, files
|
||
|
if context.generate?, do: Gen.Context.copy_new_files(context, paths, binding)
|
||
|
|
||
|
context
|
||
|
end
|
||
|
|
||
|
@doc false
|
||
|
def print_shell_instructions(%Context{schema: schema, context_app: ctx_app} = context) do
|
||
|
if schema.web_namespace do
|
||
|
Mix.shell().info """
|
||
|
|
||
|
Add the resource to your #{schema.web_namespace} :api scope in #{Mix.Phoenix.web_path(ctx_app)}/router.ex:
|
||
|
|
||
|
scope "/#{schema.web_path}", #{inspect Module.concat(context.web_module, schema.web_namespace)}, as: :#{schema.web_path} do
|
||
|
pipe_through :api
|
||
|
...
|
||
|
resources "/#{schema.plural}", #{inspect schema.alias}Controller
|
||
|
end
|
||
|
"""
|
||
|
else
|
||
|
Mix.shell().info """
|
||
|
|
||
|
Add the resource to your :api scope in #{Mix.Phoenix.web_path(ctx_app)}/router.ex:
|
||
|
|
||
|
resources "/#{schema.plural}", #{inspect schema.alias}Controller, except: [:new, :edit]
|
||
|
"""
|
||
|
end
|
||
|
if context.generate?, do: Gen.Context.print_shell_instructions(context)
|
||
|
end
|
||
|
end
|