レール 3.2.12 および 3.2.11 でテスト済み。別のRails 3.2.11プロジェクトでは、この問題はありませんf.file_field
が、現在のプロジェクトではこの奇妙な動作の理由を見つけることができないため、ここに私の質問があります。
更新アクションに奇妙な問題があります。コードの関連部分は次のとおりです。
ルート:
get "signup" => "users#new", :as => "signup"
get "profile" => "users#profile", :as => "profile"
resources :users do
member do
get :activate
end
end
コントローラ:
def update
@user = User.find(params[:id])
if @user.update_attributes(params[:user])
redirect_to user_path(@user), :notice => t('users_controller.update.updated')
else
render :edit
end
end
ハムルのフォーム(簡略化されていますが、同じ動作をします):
= form_for @user do |f|
.field
= f.label :first_name
%br
= f.text_field :first_name, :size => 40
.actions
= f.submit
したがって、更新を押すと、すべてが期待どおりに機能し、ユーザーの属性が更新されます。ただし、次のようなファイルフィールドを追加すると:
= form_for @user do |f|
.field
= f.label :first_name
%br
= f.text_field :first_name, :size => 40
.field
= f.label :avatar
%br
= f.file_field :avatar
.actions
= f.submit
[更新] を押すと、ルーティング エラーが発生します。
No route matches [PUT] "/1"
メソッド/1
でパスに到達しようとする理由がわかりません。PUT
そのルーティング エラーを示すページで/users/1
、ブラウザのアドレス バーに表示されます。
フォーム用に生成された html は次のとおりです。
<form accept-charset="UTF-8" action="/users/1" class="edit_user" enctype="multipart/form-data" id="edit_user_1" 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="che8VLfDxDAoenma+TXwsA+0IQ7+/jbCIK+Q2xwr8uc=" /></div>
<div class='field'>
<label for="user_first_name">First name</label>
<br>
<input id="user_first_name" name="user[first_name]" size="40" type="text" value="Anton" />
</div>
<div class='field'>
<label for="user_avatar">Avatar</label>
<br>
<input id="user_avatar" name="user[avatar]" type="file" />
</div>
<div class='actions'>
<input name="commit" type="submit" value="Update User" />
</div>
</form>
というわけで、ここからが一番面白いです。フォームを次のように変更すると:
= form_for @user do |f|
.field
= f.label :first_name
%br
= f.text_field :first_name, :size => 40
.field
= f.label :avatar
%br
%input{:id => "user_avatar", :name => "user[avatar]", :type => "file"}
.actions
= f.submit
生成された html は前の場合とまったく同じです (私が見ることができる唯一の違いは、ファイル フィールド属性に二重引用符ではなく単一引用符が使用されていることです):
<form accept-charset="UTF-8" action="/users/1" class="edit_user" id="edit_user_1" 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="che8VLfDxDAoenma+TXwsA+0IQ7+/jbCIK+Q2xwr8uc=" /></div>
<div class='field'>
<label for="user_first_name">First name</label>
<br>
<input id="user_first_name" name="user[first_name]" size="40" type="text" value="Anton" />
</div>
<div class='field'>
<label for="user_avatar">Avatar</label>
<br>
<input id='user_avatar' name='user[avatar]' type='file'>
</div>
<div class='actions'>
<input name="commit" type="submit" value="Update User" />
</div>
</form>
しかし、このフォームを送信した後、ルーティング エラーは発生せず、すべて正常に機能します。
アップデート
実際には、思うように動作しません。params
ハッシュを見て、キー:avatar
が存在することを確認しましたが、後者の場合enctype="multipart/form-data"
、html のフォーム オープン タグに属性がないため、ファイルがアップロードされないことを見逃していました。属性を追加enctype=multipart/form-data
すると、ルーティング エラーが再び発生します。
マルチパートフォームを送信した後put ":id" => "users#update"
にルートを追加しようとすると(確かに、このルートでは のルーティングエラーはありません)、ルーティングエラーも発生することがわかりました。redirect_to user_path(@user)
PUT
No route matches [GET] "/users/users/1"
ここにいっぱいroutes.rb
です:
Myapp::Application.routes.draw do
match "oauth/callback" => "oauths#callback"
match "oauth/callback/:provider" => "oauths#callback"
match "oauth/:provider" => "oauths#oauth", :as => :auth_at_provider
resources :countries
resources :categories
resources :images
resources :collections
resources :items
put ":id" => "users#update"
get "signup" => "users#new", :as => "signup"
get "profile" => "users#profile", :as => "profile"
resources :users do
member do
get :activate
end
end
get "signout" => "sessions#destroy", :as => "signout"
get "signin" => "sessions#new", :as => "signin"
resources :sessions
get "site/index"
root :to => "site#index"
end
レーキルート
oauth_callback /oauth/callback(.:format) oauths#callback
/oauth/callback/:provider(.:format) oauths#callback
auth_at_provider /oauth/:provider(.:format) oauths#oauth
countries GET /countries(.:format) countries#index
POST /countries(.:format) countries#create
new_country GET /countries/new(.:format) countries#new
edit_country GET /countries/:id/edit(.:format) countries#edit
country GET /countries/:id(.:format) countries#show
PUT /countries/:id(.:format) countries#update
DELETE /countries/:id(.:format) countries#destroy
categories GET /categories(.:format) categories#index
POST /categories(.:format) categories#create
new_category GET /categories/new(.:format) categories#new
edit_category GET /categories/:id/edit(.:format) categories#edit
category GET /categories/:id(.:format) categories#show
PUT /categories/:id(.:format) categories#update
DELETE /categories/:id(.:format) categories#destroy
images GET /images(.:format) images#index
POST /images(.:format) images#create
new_image GET /images/new(.:format) images#new
edit_image GET /images/:id/edit(.:format) images#edit
image GET /images/:id(.:format) images#show
PUT /images/:id(.:format) images#update
DELETE /images/:id(.:format) images#destroy
collections GET /collections(.:format) collections#index
POST /collections(.:format) collections#create
new_collection GET /collections/new(.:format) collections#new
edit_collection GET /collections/:id/edit(.:format) collections#edit
collection GET /collections/:id(.:format) collections#show
PUT /collections/:id(.:format) collections#update
DELETE /collections/:id(.:format) collections#destroy
items GET /items(.:format) items#index
POST /items(.:format) items#create
new_item GET /items/new(.:format) items#new
edit_item GET /items/:id/edit(.:format) items#edit
item GET /items/:id(.:format) items#show
PUT /items/:id(.:format) items#update
DELETE /items/:id(.:format) items#destroy
PUT /:id(.:format) users#update
signup GET /signup(.:format) users#new
profile GET /profile(.:format) users#profile
activate_user GET /users/:id/activate(.:format) users#activate
users GET /users(.:format) users#index
POST /users(.:format) users#create
new_user GET /users/new(.:format) users#new
edit_user GET /users/:id/edit(.:format) users#edit
user GET /users/:id(.:format) users#show
PUT /users/:id(.:format) users#update
DELETE /users/:id(.:format) users#destroy
signout GET /signout(.:format) sessions#destroy
signin GET /signin(.:format) sessions#new
sessions GET /sessions(.:format) sessions#index
POST /sessions(.:format) sessions#create
new_session GET /sessions/new(.:format) sessions#new
edit_session GET /sessions/:id/edit(.:format) sessions#edit
session GET /sessions/:id(.:format) sessions#show
PUT /sessions/:id(.:format) sessions#update
DELETE /sessions/:id(.:format) sessions#destroy
site_index GET /site/index(.:format) site#index
root /
誰でも何か考えがありますか?
更新2
問題を明らかにすることは、同じ問題に関するこの投稿を見つけるのに役立つマルチパート フォームにあります - Post/Put リクエスト (パッセンジャー ヘッダー) のルーティング エラーですが、残念ながら解決策はありません...
更新3
面白いものを見つけました。にメソッドがあります/path_to_gemset_here/gem/journey-1.0.4/lib/journey/router.rb
:
def find_routes env
req = request_class.new env
routes = filter_routes(req.path_info) + custom_routes.find_all { |r|
r.path.match(req.path_info)
}
routes.sort_by(&:precedence).find_all { |r|
r.constraints.all? { |k,v| v === req.send(k) } &&
r.verb === req.request_method
}.reject { |r| req.ip && !(r.ip === req.ip) }.map { |r|
match_data = r.path.match(req.path_info)
match_names = match_data.names.map { |n| n.to_sym }
match_values = match_data.captures.map { |v| v && Utils.unescape_uri(v) }
info = Hash[match_names.zip(match_values).find_all { |_,y| y }]
[match_data, r.defaults.merge(info), r]
}
end
env
非マルチパートとマルチパートの両方のリクエストを確認したところ、次のことがわかりました。
非マルチパート:
"REQUEST_URI"=>"/users/1",
"SCRIPT_NAME"=>"",
"PATH_INFO"=>"/users/1"
マルチパート:
"REQUEST_URI"=>"/users/1",
"SCRIPT_NAME"=>"/users",
"PATH_INFO"=>"/1",
"SCRIPT_FILENAME"=>"/path_to_project_folder_here/public/users", - there is no such variable in a non-multipart request
ここに問題があります。メソッドの定義でわかるように:
match_data = r.path.match(req.path_info)
PATH_INFO
リクエストを処理するルートを見つけるために使用されますが、後者の場合、何かがREQUEST_URI
2 つの部分に分かれているために完全に間違っています。残念ながら、今日は調査を終える時間がありません。明日になれば幸いです。
私よりも早く問題の原因を突き止められるほどの好奇心を持っている人がいれば、大歓迎です :)
UPDATE4 (編集済み)
というわけで、調査の続きです。
メソッド:parse_native_request
ファイル内:/path_to_gemset_here/gems/passenger-3.0.17/lib/phusion_passenger/abstract_request_handler.rb
headers_data
この呼び出し後の変数:
headers_data = channel.read_scalar(buffer, MAX_HEADER_SIZE)
内容:
"SERVER_SOFTWARE\x00Apache/2.2.22 (Ubuntu)\x00
SERVER_PROTOCOL\x00HTTP/1.1\x00
SERVER_NAME\x00myapp.loc\x00
SERVER_ADMIN\x00[no address given]\x00
SERVER_ADDR\x00127.0.0.1\x00
SERVER_PORT\x0080\x00
REMOTE_ADDR\x00127.0.0.1\x00
REMOTE_PORT\x0033199\x00
REQUEST_METHOD\x00POST\x00
QUERY_STRING\x00\x00
CONTENT_TYPE\x00multipart/form-data; boundary=----WebKitFormBoundary8HlzQxocoOROMfRV\x00
DOCUMENT_ROOT\x00/path_to_project_folder_here/public\x00
REQUEST_URI\x00/users/1\x00
SCRIPT_NAME\x00\x00
PATH_INFO\x00/users/1\x00
HTTP_HOST\x00myapp.loc\x00
HTTP_CONNECTION\x00keep-alive\x00
HTTP_CONTENT_LENGTH\x00748\x00
HTTP_CACHE_CONTROL\x00max-age=0\x00
HTTP_ACCEPT\x00text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\x00
HTTP_ORIGIN\x00http://myapp.loc\x00
HTTP_USER_AGENT\x00Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.22 (KHTML, like Gecko) Chrome/25.0.1364.97 Safari/537.22\x00
HTTP_CONTENT_TYPE\x00multipart/form-data; boundary=----WebKitFormBoundary8HlzQxocoOROMfRV\x00
HTTP_REFERER\x00http://myapp.loc/profile\x00
HTTP_ACCEPT_ENCODING\x00gzip,deflate,sdch\x00
HTTP_ACCEPT_LANGUAGE\x00en-US,en;q=0.8\x00
HTTP_ACCEPT_CHARSET\x00ISO-8859-1,utf-8;q=0.7,*;q=0.3\x00
HTTP_COOKIE\x00_myapp_session=BAh7CEkiDHVzZXJfaWQGOgZFRmkGSSIPc2Vzc2lvbl9pZAY7AEZJIiVhMjU2ZjU5N2VmMTE0YTJiOGEwNGJiYzUyYjM2NDg0OQY7AFRJIhBfY3NyZl90b2tlbgY7AEZJIjFjaGU4VkxmRHhEQW9lbm1hK1RYd3NBKzBJUTcrL2piQ0lLK1EyeHdyOHVjPQY7AEY%3D--a6e5daff1334c083e54b2bcafba43b32e546af9c\x00
UNIQUE_ID\x00UTXfEX8AAQEAACWVEMoAAAAB\x00
GATEWAY_INTERFACE\x00CGI/1.1\x00
>>>> here seems to start a kind of redirect <<<<
SERVER_PROTOCOL\x00HTTP/1.1\x00
REQUEST_METHOD\x00POST\x00
QUERY_STRING\x00\x00
REQUEST_URI\x00/users/1\x00
SCRIPT_NAME\x00/users\x00
PATH_INFO\x00/1\x00
PATH_TRANSLATED\x00/path_to_project_folder_here/public/1\x00
HTTP_HOST\x00myapp.loc\x00
HTTP_CONNECTION\x00keep-alive\x00
CONTENT_LENGTH\x00748\x00HTTP_CACHE_CONTROL\x00max-age=0\x00
HTTP_ACCEPT\x00text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\x00
HTTP_ORIGIN\x00http://myapp.loc\x00
HTTP_USER_AGENT\x00Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.22 (KHTML, like Gecko) Chrome/25.0.1364.97 Safari/537.22\x00CONTENT_TYPE\x00multipart/form-data; boundary=----WebKitFormBoundary8HlzQxocoOROMfRV\x00
HTTP_REFERER\x00http://myapp.loc/profile\x00
HTTP_ACCEPT_ENCODING\x00gzip,deflate,sdch\x00
HTTP_ACCEPT_LANGUAGE\x00en-US,en;q=0.8\x00
HTTP_ACCEPT_CHARSET\x00ISO-8859-1,utf-8;q=0.7,*;q=0.3\x00
HTTP_COOKIE\x00_myapp_session=BAh7CEkiDHVzZXJfaWQGOgZFRmkGSSIPc2Vzc2lvbl9pZAY7AEZJIiVhMjU2ZjU5N2VmMTE0YTJiOGEwNGJiYzUyYjM2NDg0OQY7AFRJIhBfY3NyZl90b2tlbgY7AEZJIjFjaGU4VkxmRHhEQW9lbm1hK1RYd3NBKzBJUTcrL2piQ0lLK1EyeHdyOHVjPQY7AEY%3D--a6e5daff1334c083e54b2bcafba43b32e546af9c\x00
PATH\x00/usr/local/bin:/usr/bin:/bin\x00
SERVER_SIGNATURE\x00<address>Apache/2.2.22 (Ubuntu) Server at myapp.loc Port 80</address>\n\x00
SERVER_SOFTWARE\x00Apache/2.2.22 (Ubuntu)\x00
SERVER_NAME\x00myapp.loc\x00
SERVER_ADDR\x00127.0.0.1\x00
SERVER_PORT\x0080\x00
REMOTE_ADDR\x00127.0.0.1\x00
DOCUMENT_ROOT\x00/path_to_project_folder_here/public\x00
SERVER_ADMIN\x00[no address given]\x00
SCRIPT_FILENAME\x00/path_to_project_folder_here/public/users\x00
REMOTE_PORT\x0033199\x00
PATH_TRANSLATED\x00/bin/runAV\x00
REDIRECT_STATUS\x00302\x00
PASSENGER_CONNECT_PASSWORD\x00EElt7wIBLlliWGCYJJoezPvecsB2brraBWdiIbD4nul\x00_\x00_\x00"
その後、次の呼び出しが続きます。
headers = split_by_null_into_hash(headers_data)
含まれるものheaders
:
{"SERVER_SOFTWARE"=>"Apache/2.2.22 (Ubuntu)",
"SERVER_PROTOCOL"=>"HTTP/1.1",
"SERVER_NAME"=>"myapp.loc",
"SERVER_ADMIN"=>"[no address given]",
"SERVER_ADDR"=>"127.0.0.1",
"SERVER_PORT"=>"80",
"REMOTE_ADDR"=>"127.0.0.1",
"REMOTE_PORT"=>"33243",
"REQUEST_METHOD"=>"POST",
"QUERY_STRING"=>"",
"CONTENT_TYPE"=>"multipart/form-data; boundary=----WebKitFormBoundary8HlzQxocoOROMfRV",
"DOCUMENT_ROOT"=>"/path_to_project_folder_here/public",
"REQUEST_URI"=>"/users/1",
"SCRIPT_NAME"=>"/users",
"PATH_INFO"=>"/1",
"HTTP_HOST"=>"myapp.loc",
"HTTP_CONNECTION"=>"keep-alive",
"HTTP_CONTENT_LENGTH"=>"748",
"HTTP_CACHE_CONTROL"=>"max-age=0",
"HTTP_ACCEPT"=>"text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8",
"HTTP_ORIGIN"=>"http://myapp.loc",
"HTTP_USER_AGENT"=>"Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.22 (KHTML, like Gecko) Chrome/25.0.1364.97 Safari/537.22",
"HTTP_CONTENT_TYPE"=>"multipart/form-data; boundary=----WebKitFormBoundary8HlzQxocoOROMfRV",
"HTTP_REFERER"=>"http://myapp.loc/profile",
"HTTP_ACCEPT_ENCODING"=>"gzip,deflate,sdch",
"HTTP_ACCEPT_LANGUAGE"=>"en-US,en;q=0.8",
"HTTP_ACCEPT_CHARSET"=>"ISO-8859-1,utf-8;q=0.7,*;q=0.3",
"HTTP_COOKIE"=>"_myapp_session=BAh7CEkiDHVzZXJfaWQGOgZFRmkGSSIPc2Vzc2lvbl9pZAY7AEZJIiVhMjU2ZjU5N2VmMTE0YTJiOGEwNGJiYzUyYjM2NDg0OQY7AFRJIhBfY3NyZl90b2tlbgY7AEZJIjFjaGU4VkxmRHhEQW9lbm1hK1RYd3NBKzBJUTcrL2piQ0lLK1EyeHdyOHVjPQY7AEY%3D--a6e5daff1334c083e54b2bcafba43b32e546af9c",
"UNIQUE_ID"=>"UTXjXn8AAQEAACceEdgAAAAA",
"GATEWAY_INTERFACE"=>"CGI/1.1",
"PATH_TRANSLATED"=>"/bin/runAV",
"CONTENT_LENGTH"=>"748",
"PATH"=>"/usr/local/bin:/usr/bin:/bin",
"SERVER_SIGNATURE"=>"<address>Apache/2.2.22 (Ubuntu) Server at myapp.loc Port 80</address>\n",
"SCRIPT_FILENAME"=>"/path_to_project_folder_here/public/users",
"REDIRECT_STATUS"=>"302",
"PASSENGER_CONNECT_PASSWORD"=>"GgEqWssAcbBETWnFI7xzBfWRGibgB34OhfFSUVyOhPn", "_"=>"_"}
したがって、問題は明らかに、ヘッダーがハッシュにパックされる方法にあります- PATH_INFO
(および他のヘッダーにも)2つの値があり、後者の値(間違っている)が最初の値を書き換えます(実際、問題はこれらの理由にありますヘッダーが送信されていますが、これを処理する方法がわかりません)。メソッドでハッシュへのパッキングが行われていsplit_by_null_into_hash(headers_data)
ます。今そこに行く。
ファイル:/path_to_gemset_here/gems/passenger-3.0.17/lib/phusion_passenger/utils.rb
モジュールUtils
には次のコードが含まれています。
if defined?(PhusionPassenger::NativeSupport)
# Split the given string into an hash. Keys and values are obtained by splitting the
# string using the null character as the delimitor.
def split_by_null_into_hash(data)
return PhusionPassenger::NativeSupport.split_by_null_into_hash(data)
end
else
NULL = "\0".freeze
def split_by_null_into_hash(data)
args = data.split(NULL, -1)
args.pop
return Hash[*args]
end
end
私の場合、if
条件の部分が実行されているため、問題は次のようになります
PhusionPassenger::NativeSupport.split_by_null_into_hash(data)
そして、それはファイルに私たちを連れて行くようです:/path_to_gemset_here/gems/passenger-3.0.17/ext/ruby/passenger_native_support.c
つづく...
更新5
C
実際、このファイルはパッセンジャーのインストール中にコンパイルされ、デバッグするには、パッセンジャーを何度も再インストールして再インストールする必要があるため、そのファイルのデバッグ地獄に対処しないことにしました。else
したがって、条件の - 部分を使用することに切り替えることにしました。これは、まったく同じ目標を達成しているように見えますが、明らかにプリコンパイルされたC
-code よりも少し遅いためです。しかし、私の場合、それは本当に問題ではありません。/path_to_project_folder_here/lib
そのため、次のコードでフォルダーにファイルを含めることで、メソッドの定義をオーバーライドしました。
module PhusionPassenger
module Utils
protected
NULL = "\0".freeze
def split_by_null_into_hash(data)
args = data.split(NULL, -1)
args.pop
return Hash[*args]
end
end
end
動作を変更することはできませんHash[*args]
(より正確には、::[]
メソッドをオーバーライドすることでできると言っていますが、確実にしたくはありません) ので、コードを少し変更します。
module PhusionPassenger
module Utils
protected
NULL = "\0".freeze
def split_by_null_into_hash(data)
args = data.split(NULL, -1)
args.pop
headers_hash = Hash.new
args.each_slice(2).to_a.each do |pair|
headers_hash[pair.first] = pair.last unless headers_hash.keys.include? pair.first
end
return headers_hash
end
end
end
そしてビンゴ!今では動作します。
ただし、これを行うことで他の機能を壊していないかどうかはわかりません。そのため、このアプローチを使用するようにアドバイスすることはできません。この変更に関連する問題が発生するまで使用します。その場合は、問題を解決する別の方法を見つけようとします。
そして、なぜこれらの間違ったヘッダーが送信されているのかという主な疑問が残っています。