Commit 934eb348 authored by Stan Hu's avatar Stan Hu

Fix merge request redirects going to /commits page

When a merge request is redirected to a new project, Rails attempts to
construct the proper URL in `url_for` by finding the first route that
matches the given parameters:

1. `controller` -> `projects/merge_requests`
2. `action` -> `show`
3. `namespace_id`
4. `project_id`
5. `id`

In this example, `/*namespace_id/:project_id/commits/*id` matched
instead of `/*namespace_id/:project_id/merge_requests/:id`.

This happens because the Rails `resources` block defines all the
standard #index, #show, #update, etc. routes after the block runs.

We fixed this issue by explicitly specifying the merge request #show
as the first route.

This is similar to https://gitlab.com/gitlab-org/gitlab/issues/31357,
except the solution is a bit simpler because the backend appears to set
a `tab` query string, which allows the redirected route to end up in the
right place. For example, visiting the old `/merge_requests/:id/commits`
appears to cause a redirection to `/merge_requests/:id/?tab=commits`,
which does the right thing.
parent 69cb23c6
...@@ -239,8 +239,9 @@ constraints(::Constraints::ProjectUrlConstrainer.new) do ...@@ -239,8 +239,9 @@ constraints(::Constraints::ProjectUrlConstrainer.new) do
end end
end end
resources :merge_requests, concerns: :awardable, except: [:new, :create], constraints: { id: /\d+/ } do resources :merge_requests, concerns: :awardable, except: [:new, :create, :show], constraints: { id: /\d+/ } do
member do member do
get :show # Insert this first to ensure redirections using merge_requests#show match this route
get :commit_change_content get :commit_change_content
post :merge post :merge
post :cancel_auto_merge post :cancel_auto_merge
......
---
title: Fix merge request redirects going to /commits page
merge_request: 16705
author:
type: fixed
...@@ -78,6 +78,40 @@ describe Projects::MergeRequestsController do ...@@ -78,6 +78,40 @@ describe Projects::MergeRequestsController do
expect(response).to be_successful expect(response).to be_successful
end end
end end
context 'when project has moved' do
let(:new_project) { create(:project) }
before do
project.route.destroy
new_project.redirect_routes.create!(path: project.full_path)
new_project.add_developer(user)
end
it 'redirects from an old merge request correctly' do
get :show,
params: {
namespace_id: project.namespace,
project_id: project,
id: merge_request
}
expect(response).to redirect_to(project_merge_request_path(new_project, merge_request))
expect(response).to have_gitlab_http_status(302)
end
it 'redirects from an old merge request commits correctly' do
get :commits,
params: {
namespace_id: project.namespace,
project_id: project,
id: merge_request
}
expect(response).to redirect_to(commits_project_merge_request_path(new_project, merge_request))
expect(response).to have_gitlab_http_status(302)
end
end
end end
context 'when user is setting notes filters' do context 'when user is setting notes filters' do
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment