diff options
| author | Toon Claes <toon@gitlab.com> | 2017-04-21 11:36:34 +0200 | 
|---|---|---|
| committer | Toon Claes <toon@iotcl.com> | 2017-08-03 16:31:05 +0200 | 
| commit | a723cba57493ec3220596ca8543a8b1b1ec118fe (patch) | |
| tree | fc1048fe60a21339cef711caf69aff59edf5a5fb | |
| parent | c39daf937bd62eedcbe7e3b44f107bb7e87452e7 (diff) | |
| download | gitlab-ce-a723cba57493ec3220596ca8543a8b1b1ec118fe.tar.gz | |
Avoid plucking Todo ids and use sub-queries instead
TodoService should not call `.select(&:id)` on todos, because this is
bad performance. So instead use sub-queries, which will result in a
single SQL query to the database.
https://docs.gitlab.com/ee/development/sql.html#plucking-ids
| -rw-r--r-- | app/controllers/dashboard/todos_controller.rb | 4 | ||||
| -rw-r--r-- | app/services/issuable_base_service.rb | 2 | ||||
| -rw-r--r-- | app/services/todo_service.rb | 16 | ||||
| -rw-r--r-- | changelogs/unreleased/tc-no-todo-service-select.yml | 4 | ||||
| -rw-r--r-- | lib/api/todos.rb | 6 | ||||
| -rw-r--r-- | lib/api/v3/todos.rb | 6 | ||||
| -rw-r--r-- | spec/services/todo_service_spec.rb | 33 | 
7 files changed, 51 insertions, 20 deletions
| diff --git a/app/controllers/dashboard/todos_controller.rb b/app/controllers/dashboard/todos_controller.rb index 59e5b5e4775..a8b2b93b458 100644 --- a/app/controllers/dashboard/todos_controller.rb +++ b/app/controllers/dashboard/todos_controller.rb @@ -13,7 +13,7 @@ class Dashboard::TodosController < Dashboard::ApplicationController    end    def destroy -    TodoService.new.mark_todos_as_done_by_ids([params[:id]], current_user) +    TodoService.new.mark_todos_as_done_by_ids(params[:id], current_user)      respond_to do |format|        format.html do @@ -37,7 +37,7 @@ class Dashboard::TodosController < Dashboard::ApplicationController    end    def restore -    TodoService.new.mark_todos_as_pending_by_ids([params[:id]], current_user) +    TodoService.new.mark_todos_as_pending_by_ids(params[:id], current_user)      render json: todos_counts    end diff --git a/app/services/issuable_base_service.rb b/app/services/issuable_base_service.rb index ea497729115..760a15e3ed0 100644 --- a/app/services/issuable_base_service.rb +++ b/app/services/issuable_base_service.rb @@ -288,7 +288,7 @@ class IssuableBaseService < BaseService        todo_service.mark_todo(issuable, current_user)      when 'done'        todo = TodosFinder.new(current_user).execute.find_by(target: issuable) -      todo_service.mark_todos_as_done([todo], current_user) if todo +      todo_service.mark_todos_as_done_by_ids(todo, current_user) if todo      end    end diff --git a/app/services/todo_service.rb b/app/services/todo_service.rb index 322c6286365..6ee96d6a0f8 100644 --- a/app/services/todo_service.rb +++ b/app/services/todo_service.rb @@ -170,20 +170,22 @@ class TodoService    # When user marks some todos as done    def mark_todos_as_done(todos, current_user) -    update_todos_state_by_ids(todos.select(&:id), current_user, :done) +    update_todos_state(todos, current_user, :done)    end    def mark_todos_as_done_by_ids(ids, current_user) -    update_todos_state_by_ids(ids, current_user, :done) +    todos = todos_by_ids(ids, current_user) +    mark_todos_as_done(todos, current_user)    end    # When user marks some todos as pending    def mark_todos_as_pending(todos, current_user) -    update_todos_state_by_ids(todos.select(&:id), current_user, :pending) +    update_todos_state(todos, current_user, :pending)    end    def mark_todos_as_pending_by_ids(ids, current_user) -    update_todos_state_by_ids(ids, current_user, :pending) +    todos = todos_by_ids(ids, current_user) +    mark_todos_as_pending(todos, current_user)    end    # When user marks an issue as todo @@ -198,9 +200,11 @@ class TodoService    private -  def update_todos_state_by_ids(ids, current_user, state) -    todos = current_user.todos.where(id: ids) +  def todos_by_ids(ids, current_user) +    current_user.todos.where(id: Array(ids)) +  end +  def update_todos_state(todos, current_user, state)      # Only update those that are not really on that state      todos = todos.where.not(state: state)      todos_ids = todos.pluck(:id) diff --git a/changelogs/unreleased/tc-no-todo-service-select.yml b/changelogs/unreleased/tc-no-todo-service-select.yml new file mode 100644 index 00000000000..ddcae334aa7 --- /dev/null +++ b/changelogs/unreleased/tc-no-todo-service-select.yml @@ -0,0 +1,4 @@ +--- +title: Avoid plucking Todo ids in TodoService +merge_request: 10845 +author: diff --git a/lib/api/todos.rb b/lib/api/todos.rb index d1f7e364029..55191169dd4 100644 --- a/lib/api/todos.rb +++ b/lib/api/todos.rb @@ -59,10 +59,10 @@ module API          requires :id, type: Integer, desc: 'The ID of the todo being marked as done'        end        post ':id/mark_as_done' do -        todo = current_user.todos.find(params[:id]) -        TodoService.new.mark_todos_as_done([todo], current_user) +        TodoService.new.mark_todos_as_done_by_ids(params[:id], current_user) +        todo = Todo.find(params[:id]) -        present todo.reload, with: Entities::Todo, current_user: current_user +        present todo, with: Entities::Todo, current_user: current_user        end        desc 'Mark all todos as done' diff --git a/lib/api/v3/todos.rb b/lib/api/v3/todos.rb index e3b311d61cd..2f2cf259987 100644 --- a/lib/api/v3/todos.rb +++ b/lib/api/v3/todos.rb @@ -11,10 +11,10 @@ module API            requires :id, type: Integer, desc: 'The ID of the todo being marked as done'          end          delete ':id' do -          todo = current_user.todos.find(params[:id]) -          TodoService.new.mark_todos_as_done([todo], current_user) +          TodoService.new.mark_todos_as_done_by_ids(params[:id], current_user) +          todo = Todo.find(params[:id]) -          present todo.reload, with: ::API::Entities::Todo, current_user: current_user +          present todo, with: ::API::Entities::Todo, current_user: current_user          end          desc 'Mark all todos as done' diff --git a/spec/services/todo_service_spec.rb b/spec/services/todo_service_spec.rb index 534d3e65be2..80d05451e09 100644 --- a/spec/services/todo_service_spec.rb +++ b/spec/services/todo_service_spec.rb @@ -336,7 +336,7 @@ describe TodoService do      describe '#mark_todos_as_done' do        it_behaves_like 'updating todos state', :mark_todos_as_done, :pending, :done do -        let(:collection) { [first_todo, second_todo] } +        let(:collection) { Todo.all }        end      end @@ -348,7 +348,7 @@ describe TodoService do      describe '#mark_todos_as_pending' do        it_behaves_like 'updating todos state', :mark_todos_as_pending, :done, :pending do -        let(:collection) { [first_todo, second_todo] } +        let(:collection) { Todo.all }        end      end @@ -880,14 +880,16 @@ describe TodoService do      it 'marks an array of todos as done' do        todo = create(:todo, :mentioned, user: john_doe, target: issue, project: project) -      expect { described_class.new.mark_todos_as_done([todo], john_doe) } +      todos = TodosFinder.new(john_doe, {}).execute +      expect { described_class.new.mark_todos_as_done(todos, john_doe) }          .to change { todo.reload.state }.from('pending').to('done')      end      it 'returns the ids of updated todos' do # Needed on API        todo = create(:todo, :mentioned, user: john_doe, target: issue, project: project) -      expect(described_class.new.mark_todos_as_done([todo], john_doe)).to eq([todo.id]) +      todos = TodosFinder.new(john_doe, {}).execute +      expect(described_class.new.mark_todos_as_done(todos, john_doe)).to eq([todo.id])      end      context 'when some of the todos are done already' do @@ -907,11 +909,32 @@ describe TodoService do          expect(described_class.new.mark_todos_as_done(Todo.all, john_doe)).to eq([])        end      end +  end + +  describe '#mark_todos_as_done_by_ids' do +    let(:issue) { create(:issue, project: project, author: author, assignees: [john_doe]) } +    let(:another_issue) { create(:issue, project: project, author: author, assignees: [john_doe]) } + +    it 'marks an array of todo ids as done' do +      todo = create(:todo, :mentioned, user: john_doe, target: issue, project: project) +      another_todo = create(:todo, :mentioned, user: john_doe, target: another_issue, project: project) + +      expect { described_class.new.mark_todos_as_done_by_ids([todo.id, another_todo.id], john_doe) } +        .to change { john_doe.todos.done.count }.from(0).to(2) +    end + +    it 'marks a single todo id as done' do +      todo = create(:todo, :mentioned, user: john_doe, target: issue, project: project) + +      expect { described_class.new.mark_todos_as_done_by_ids(todo.id, john_doe) } +        .to change { todo.reload.state }.from('pending').to('done') +    end      it 'caches the number of todos of a user', :use_clean_rails_memory_store_caching do        create(:todo, :mentioned, user: john_doe, target: issue, project: project)        todo = create(:todo, :mentioned, user: john_doe, target: issue, project: project) -      described_class.new.mark_todos_as_done([todo], john_doe) + +      described_class.new.mark_todos_as_done_by_ids(todo, john_doe)        expect_any_instance_of(TodosFinder).not_to receive(:execute) | 
