2 votes

Erreur d'argument dans le fichier de modifications

J'apprends le cadre Phoenix à partir du livre Programming Phoenix et je suis confronté à une erreur que je ne parviens pas à comprendre. En fait, le livre en est au stade de l'ajout d'un formulaire pour la création de nouveaux utilisateurs. Ainsi, lorsque je dirige le navigateur vers http://localhost:4000/users/new J'obtiens la page suivante (veuillez agrandir l'image) :

enter image description here

Dans la sortie de la ligne de commande, le serveur dit :

[info] GET /users/new
[debug] Processing by Rumbl.UserController.new/2
  Parameters: %{}
  Pipelines: [:browser]
[info] Sent 500 in 9ms
[error] #PID<0.543.0> running Rumbl.Endpoint terminated
Server: localhost:4000 (http)
Request: GET /users/new
** (exit) an exception was raised:
    ** (ArgumentError) argument error
        :erlang.binary_to_existing_atom("name,", :utf8)
        (ecto) lib/ecto/changeset.ex:402: Ecto.Changeset.process_empty_fields/2
        (elixir) lib/enum.ex:1184: Enum."-map/2-lists^map/1-0-"/2
        (ecto) lib/ecto/changeset.ex:373: Ecto.Changeset.do_cast/7
        (rumbl) web/models/user.ex:15: Rumbl.User.changeset/2
        (rumbl) web/controllers/user_controller.ex:17: Rumbl.UserController.new/2
        (rumbl) web/controllers/user_controller.ex:1: Rumbl.UserController.action/2
        (rumbl) web/controllers/user_controller.ex:1: Rumbl.UserController.phoenix_controller_pipeline/2
        (rumbl) lib/rumbl/endpoint.ex:1: Rumbl.Endpoint.instrument/4
        (rumbl) lib/phoenix/router.ex:261: Rumbl.Router.dispatch/2
        (rumbl) web/router.ex:1: Rumbl.Router.do_call/2
        (rumbl) lib/rumbl/endpoint.ex:1: Rumbl.Endpoint.phoenix_pipeline/1
        (rumbl) lib/plug/debugger.ex:122: Rumbl.Endpoint."call (overridable 3)"/2
        (rumbl) lib/rumbl/endpoint.ex:1: Rumbl.Endpoint.call/2
        (plug) lib/plug/adapters/cowboy/handler.ex:15: Plug.Adapters.Cowboy.Handler.upgrade/4
        (cowboy) src/cowboy_protocol.erl:442: :cowboy_protocol.execute/4

Puisque l'erreur indique quelque chose comme binary_to_existing_atom J'ai pensé que j'avais peut-être mal saisi les paramètres, mais cela ne semble pas être le cas.

Mon fichier modèle :

defmodule Rumbl.User do
    use Rumbl.Web, :model

    schema "users" do
        field :name, :string
        field :username, :string
        field :password, :string, virtual: true
        field :password_hash, :string

        timestamps
    end

    def changeset(model, params \\ :invalid) do
        model
        |> cast(params, ~w(name, username), [])
        |> validate_length(:username, min: 1, max: 20)
    end 
end

Mon fichier de migration :

defmodule Rumbl.Repo.Migrations.CreateUser do
    use Ecto.Migration

    def change do
        create table(:users) do
            add :name, :string
            add :username, :string, null: false
            add :password_hash, :string

            timestamps
        end

        create unique_index(:users, [:username])
    end
end

Le contrôleur :

defmodule Rumbl.UserController do
    use Rumbl.Web, :controller

    def index(conn, _params) do
        users = Repo.all(Rumbl.User)
        render conn, "index.html", users: users
    end

    def show(conn, %{"id" => id}) do
        user = Repo.get(Rumbl.User, id)
        render conn, "show.html", user: user
    end

    alias Rumbl.User

    def new(conn, _params) do
        changeset = User.changeset(%User{})
        render conn, "new.html", changeset: changeset 
    end
end

Et enfin, le modèle :

<h1>New User</h1>

<%= form_for @changeset, user_path(@conn, :create), fn f -> %>
    <div class="form-group">
        <%= text_input f, :name, placeholder: "Username", class: "form-control" %>        
    </div>
    <div class="form-group">
        <%= password_input f, :password, placeholder: "Password", class: "form-control" %>        
    </div>
    <%= submit "Create User", class: "btn btn-primary" %>
<% end %>

Faites-moi savoir si un autre code est également nécessaire. Et merci d'avance !

6voto

JustMichael Points 110

En ~w sigil n'exige pas de virgules après chaque élément, mais lorsque vous les ajoutez, elles sont concaténées aux éléments de la liste, donc dans votre cas

~w(name, username)

a généré la liste suivante :

["name,", "username"]

ce qui n'est évidemment pas ce que vous vouliez.

Vous devriez donc supprimer la virgule et utiliser ceci à la place :

~w(name username)

qui génère exactement ce que vous attendez :

["name", "username"]

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X