Ответ 1
Что делать, если вы сделали только :url => upload_photo_path(@photo)
?
Кажется немного странным, что вы загрузили бы участника. Является ли это просто методом создания (в этом случае вам нужно просто POST в путь коллекции)?
У меня возникли проблемы с маршрутизацией формы для пользовательского действия в Rails 3. Вот мои маршруты:
resources :photos do
resources :comments
collection do
get 'update_states'
end
member do
put 'upload'
end
end
Здесь form_for:
form_for @photo, :remote => true, :url => { :action => upload_photo_path(@photo) }, :html => { :multipart => :true, :method => 'put' } do |f|
И вот сообщение об ошибке:
No route matches {:action=>"/photos/42/upload", :controller=>"photos"}
... это особенно расстраивает, потому что "photos/: id/upload" - это точно правильное действие для этой формы.
Что мне не хватает?
РЕДАКТИРОВКА - вот оригинальные маршруты, связанные с фото:
photo_comments GET /photos/:photo_id/comments(.:format) {:action=>"index", :controller=>"comments"}
POST /photos/:photo_id/comments(.:format) {:action=>"create", :controller=>"comments"}
new_photo_comment GET /photos/:photo_id/comments/new(.:format) {:action=>"new", :controller=>"comments"}
edit_photo_comment GET /photos/:photo_id/comments/:id/edit(.:format) {:action=>"edit", :controller=>"comments"}
photo_comment GET /photos/:photo_id/comments/:id(.:format) {:action=>"show", :controller=>"comments"}
PUT /photos/:photo_id/comments/:id(.:format) {:action=>"update", :controller=>"comments"}
DELETE /photos/:photo_id/comments/:id(.:format) {:action=>"destroy", :controller=>"comments"}
update_states_photos GET /photos/update_states(.:format) {:action=>"update_states", :controller=>"photos"}
upload_photo PUT /photos/:id/upload(.:format) {:action=>"upload", :controller=>"photos"}
photos GET /photos(.:format) {:action=>"index", :controller=>"photos"}
POST /photos(.:format) {:action=>"create", :controller=>"photos"}
new_photo GET /photos/new(.:format) {:action=>"new", :controller=>"photos"}
edit_photo GET /photos/:id/edit(.:format) {:action=>"edit", :controller=>"photos"}
photo GET /photos/:id(.:format) {:action=>"show", :controller=>"photos"}
PUT /photos/:id(.:format) {:action=>"update", :controller=>"photos"}
DELETE /photos/:id(.:format) {:action=>"destroy", :controller=>"photos"}
Вот соответствующие маршруты, когда я изменил маршрут на match 'upload'
:
photo_comments GET /photos/:photo_id/comments(.:format) {:action=>"index", :controller=>"comments"}
POST /photos/:photo_id/comments(.:format) {:action=>"create", :controller=>"comments"}
}
new_photo_comment GET /photos/:photo_id/comments/new(.:format) {:action=>"new", :controller=>"comments"}
edit_photo_comment GET /photos/:photo_id/comments/:id/edit(.:format) {:action=>"edit", :controller=>"comments"}
photo_comment GET /photos/:photo_id/comments/:id(.:format) {:action=>"show", :controller=>"comments"}
PUT /photos/:photo_id/comments/:id(.:format) {:action=>"update", :controller=>"comments"}
DELETE /photos/:photo_id/comments/:id(.:format) {:action=>"destroy", :controller=>"comments"}
update_states_photos GET /photos/update_states(.:format) {:action=>"update_states", :controller=>"photos"}
upload_photo /photos/:id/upload(.:format) {:action=>"upload", :controller=>"photos"}
photos GET /photos(.:format) {:action=>"index", :controller=>"photos"}
POST /photos(.:format) {:action=>"create", :controller=>"photos"}
new_photo GET /photos/new(.:format) {:action=>"new", :controller=>"photos"}
edit_photo GET /photos/:id/edit(.:format) {:action=>"edit", :controller=>"photos"}
photo GET /photos/:id(.:format) {:action=>"show", :controller=>"photos"}
PUT /photos/:id(.:format) {:action=>"update", :controller=>"photos"}
DELETE /photos/:id(.:format) {:action=>"destroy", :controller=>"photos"}
К сожалению, "матч" не сработал лучше...
- EDIT -
Просто чтобы подтвердить другой сценарий здесь... с этим в маршрутах:
resources :photos do
resources :comments
collection do
get 'update_states'
end
member do
match 'upload'
end
end
и это в представлении:
form_for @photo, :remote => true, :url => { :action => 'upload' }, :html => { :multipart => :true, :id => 'photo_upload' } do |f|
Я все еще получаю:
No route matches {:action=>"upload", :controller=>"photos"}
Что делать, если вы сделали только :url => upload_photo_path(@photo)
?
Кажется немного странным, что вы загрузили бы участника. Является ли это просто методом создания (в этом случае вам нужно просто POST в путь коллекции)?
У меня была такая же проблема, и, наконец, я работал над решением, которое я не уверен в этом, потому что исходный плакат перешел к другому подходу.
Мои маршруты имели
resources :members do
member do
get "invite"
post 'register'
end
end
И включены "рейк-маршруты"
register_member POST /members/:id/register(.:format) {:protocol=>"http", :action=>"register", :controller=>"members"}
Тем не менее я продолжал получать ошибку
Started POST "/members/149/register" for 127.0.0.1 at 2012-04-13 13:18:35 -0700
ActionController::RoutingError (No route matches "/members/149/register"):
Rendered /Users/lisa/.rvm/gems/[email protected]/gems/actionpack-3.0.9/lib/action_dispatch/middleware/templates/rescues/routing_error.erb within rescues/layout (1.1ms)
и проблема была ограничена только формой, созданной Rails в соответствии с приведенной ниже формой_for (обратите внимание, что я подтвердил это с помощью HTTP-клиента, чтобы вручную отправить POST на URL-адрес и увидел, что он находит маршрут)
<%= form_for @account, :url => register_member_path(@account.id) do |account_form| %>
...
Я проверил метод, я проверил путь, все выглядело хорошо. Что я наконец заметил, очищая сгенерированную форму по строкам:
<form accept-charset="UTF-8" action="/members/149/register" class="edit_member" id="edit_member_149" method="post">
<div style="margin:0;padding:0;display:inline">
<input name="utf8" type="hidden" value="✓" />
<input name="_method" type="hidden" value="put" />
<input name="authenticity_token" type="hidden" value="74pkMgRHfdowSfzjJGMILkAsivVNrJZ0iWYXRUgxyF0=" />
</div>
...
Обратите внимание на скрытый ввод name= "_ method". Мне жаль, что Rails интерпретировал это как PUT, показанный в журналах, что сделало бы мою отладку намного быстрее. Я исправил это, явно указав форму, чтобы использовать метод POST, который, конечно, уже был, но сказал, что удалил скрытое переопределение параметра. Я предполагаю, что есть некоторые аспекты @account, которые вызвали Rails для использования параметра _method, но @account должен быть существующей записью.
Ваш параметр url
должен быть
: url = > {: action = > "upload" }
(исходный ответ)
Я бы поспорил, потому что ваш маршрут ожидает PUT
, и ваша форма отправляет POST
(возможно, потому, что @photo = Photo.new
). У вас есть несколько вариантов:
post 'upload'
form_for @photo, :as => :post
с остальными аргументами@photo
является существующей записью (например, вызывая create
вместо new
)Наиболее подходящий вариант, вероятно, является одним из первых 2.