Rails 5 multiple nested attributes no Routing Erro

2019-07-27 12:25发布

Hello I have a little app with three nested models Client, Site and Damper. When I add a client or client_site all is fine... but when i come to add a damper I get

Routing Error
uninitialized constant Sites

in the console

Started GET "/clients/1/sites/1/dampers/new" for my ip at 2018-10-26 18:05:29 +1000

ActionController::RoutingError (uninitialized constant Sites):

app/controllers/clients/sites/dampers_controller.rb:1:in `<main>'

Routes

resources :clients do
  resources :sites, controller: 'clients/sites' do
    resources :dampers, controller: 'clients/sites/dampers'
  end
end

Models

app/models/client.rb

class Client < ApplicationRecord
  has_many :sites
end

app/models/site.rb

class Site < ApplicationRecord
  belongs_to :client
  has_many :dampers
end

app/models/damper.rb

class Damper < ApplicationRecord
  belongs_to :site
end

please note I had made a mistake and this was originally :sites but even after changing this the fault remained.

Controllers

app/controllers/clients_controller.rb

class ClientsController < ApplicationController
  before_action :set_client, only: [:show, :edit, :update, :destroy]

  # GET /clients
  # GET /clients.json
  def index
    @clients = Client.all
  end

  # GET /clients/1
  # GET /clients/1.json
  def show
    @client = Client.find(params[:id])
    @sites = @client.sites
  end

  # GET /clients/new
  def new
    @client = Client.new
  end

  # GET /clients/1/edit
  def edit
  end

  # POST /clients
  # POST /clients.json
  def create
    @client = Client.new(client_params)

    respond_to do |format|
      if @client.save
        format.html { redirect_to @client, notice: 'Client was successfully created.' }
        format.json { render :show, status: :created, location: @client }
      else
        format.html { render :new }
        format.json { render json: @client.errors, status: :unprocessable_entity }
      end
    end
  end

  # PATCH/PUT /clients/1
  # PATCH/PUT /clients/1.json
  def update
    respond_to do |format|
      if @client.update(client_params)
        format.html { redirect_to @client, notice: 'Client was successfully updated.' }
        format.json { render :show, status: :ok, location: @client }
      else
        format.html { render :edit }
        format.json { render json: @client.errors, status: :unprocessable_entity }
      end
    end
  end

  # DELETE /clients/1
  # DELETE /clients/1.json
  def destroy
    @client.destroy
    respond_to do |format|
      format.html { redirect_to clients_url, notice: 'Client was successfully destroyed.' }
      format.json { head :no_content }
    end
  end

  private
    # Use callbacks to share common setup or constraints between actions.
    def set_client
      @client = Client.find(params[:id])
    end

    # Never trust parameters from the scary internet, only allow the white list through.
    def client_params
      params.require(:client).permit(:name)
    end
end

app/controllers/clients/sites_controller.rb

class Clients::SitesController < ApplicationController
  before_action :set_client
  before_action :set_site, except: [:new, :create]

  # GET /sites
  # GET /sites.json
  def index
    @sites = Site.all
  end

  # GET /sites/1
  # GET /sites/1.json
  def show
    @client = Client.find(params[:client_id])
    @site = @client.sites.find(params[:id])
  end

  # GET /sites/new
  def new
    @site = Site.new
  end

  # GET /sites/1/edit
  def edit
  end

  # POST /sites
  # POST /sites.json
  def create
    @site = Site.new(site_params)
    @site.client = @client

    respond_to do |format|
      if @site.save
        format.html { redirect_to @client, notice: 'Site was successfully created.' }
        format.json { render :show, status: :created, location: @client }
      else
        format.html { render :new }
        format.json { render json: @client.errors, status: :unprocessable_entity }
      end
    end
  end

  # PATCH/PUT /sites/1
  # PATCH/PUT /sites/1.json
  def update
    respond_to do |format|
      if @site.update(site_params)
        format.html { redirect_to @client, notice: 'Site was successfully updated.' }
        format.json { render :show, status: :ok, location: @site }
      else
        format.html { render :edit }
        format.json { render json: @client.errors, status: :unprocessable_entity }
      end
    end
  end

  # DELETE /sites/1
  # DELETE /sites/1.json
  def destroy
    @site.destroy
    respond_to do |format|
      format.html { redirect_to sites_url, notice: 'Site was successfully destroyed.' }
      format.json { head :no_content }
    end
  end

  private
    # Use callbacks to share common setup or constraints between actions.
    def set_site
      @site = Site.find(params[:id])
    end

  def set_client
    @client = Client.find(params[:client_id])
  end

    # Never trust parameters from the scary internet, only allow the white list through.
    def site_params
      params.require(:site).permit(:name, :client_id)
    end
end

app/controllers/clients/sites/dampers_controller.rb

class Sites::DampersController < ApplicationController
  before_action :set_client
  before_action :set_site
  before_action :set_damper, except: [:new, :create]

  # GET /dampers
  # GET /dampers.json
  def index
    @dampers = Damper.all
  end

  # GET /dampers/1
  # GET /dampers/1.json
  def show
  end

  # GET /dampers/new
  def new
    @damper = Damper.new
  end

  # GET /dampers/1/edit
  def edit
  end

  # POST /dampers
  # POST /dampers.json
  def create
    @damper = Damper.new(damper_params)
    @damper.site = @site

    respond_to do |format|
      if @damper.save
        format.html { redirect_to @site, notice: 'Damper was successfully created.' }
        format.json { render :show, status: :created, location: @site }
      else
        format.html { render :new }
        format.json { render json: @site.errors, status: :unprocessable_entity }
      end
    end
  end

  # PATCH/PUT /dampers/1
  # PATCH/PUT /dampers/1.json
  def update
    respond_to do |format|
      if @damper.update(damper_params)
        format.html { redirect_to @site, notice: 'Damper was successfully updated.' }
        format.json { render :show, status: :ok, location: @site }
      else
        format.html { render :edit }
        format.json { render json: @site.errors, status: :unprocessable_entity }
      end
    end
  end

  # DELETE /dampers/1
  # DELETE /dampers/1.json
  def destroy
    @damper.destroy
    respond_to do |format|
      format.html { redirect_to dampers_url, notice: 'Damper was successfully destroyed.' }
      format.json { head :no_content }
    end
  end

  private
    # Use callbacks to share common setup or constraints between actions.
    def set_damper
      @damper = Damper.find(params[:id])
    end
  def set_site
    @site = Site.find(params[:site_id])
  end
  def set_client
    @client = Client.find(params[:client_id])
  end

    # Never trust parameters from the scary internet, only allow the white list through.
    def damper_params
      params.require(:damper).permit(:location, :number, :site_id, :client_id)
    end
end

2条回答
地球回转人心会变
2楼-- · 2019-07-27 12:55

ActionController::RoutingError (uninitialized constant Sites)

The dampers_controller.rb sits under controllers/clients/sites, so you need to change the class name to

class Clients::Sites::DampersController < ApplicationController

instead of

class Sites::DampersController < ApplicationController

for the sake of namespacing

Also, I recommend you to have a look at controller-namespaces-and-routing

查看更多
ら.Afraid
3楼-- · 2019-07-27 12:57

You created a controller named Sites::DampersController, which is a class DampersController defined inside the namespace (module, class,...) named Sites, but you forgot to define this last one.

You could create it this way:

module Sites
  class DampersController < ApplicationController
  end
end

Or just get rid of the Sites:: part.

You will also need to update your routes, to specify the correct controller name.

More generally, it is easier to follow rails default route generation:

resources :clients do
  resources :sites do
    resources :dampers
  end
end

Which will create routes pointing to the following controllers:

  • ClientsController
  • SitesController
  • DampersController

If you really intend on putting the other controllers in a sub folders, following your original routes, you will need to define the following:

  • controller ClientsController
  • module Clients
  • controller SitesController inside module Clients
  • module Sites inside module Clients
  • controller DampersController inside module Clients::Sites

Which, for autoloading to works, would have to be organized in subfolders too:

  • app/controllers
    • clients_controllers.rb
    • clients/
    • sites__controllers.rb
    • sites/
      • dampers_controllers.rb
查看更多
登录 后发表回答