Commit 3d3e441c authored by James Lopez's avatar James Lopez

refactor code based on feedback

parent 92ec58ab
module Gitlab module Gitlab
module ImportExport module ImportExport
class GroupProjectFinder # Given a class, it finds or creates a new object
def self.find_or_new(*args) # (initializes in the case of Label) at group or project level
# If it does not exist in the group, it creates it at project level
#
# For example:
# `GroupProjectObjectBuilder.build(Label, label_attributes)`
# finds or initializes a label with the given attributes.
#
# It also adds some logic around Group Labels/Milestones for edge cases.
class GroupProjectObjectBuilder
def self.build(*args)
Project.transaction do Project.transaction do
new(*args).find_or_new new(*args).find
end
end
def self.find_or_create(*args)
Project.transaction do
new(*args).find_or_create
end end
end end
...@@ -20,30 +23,38 @@ module Gitlab ...@@ -20,30 +23,38 @@ module Gitlab
@project = @attributes[:project] @project = @attributes[:project]
end end
def find_or_new def find
@klass.where(where_clause).first || @klass.new(project_attributes) find_or_action do
label? ? @klass.new(project_attributes) : @klass.create(project_attributes)
end end
def find_or_create
@klass.where(where_clause).first || @klass.create(project_attributes)
end end
private private
def find_or_action(&block)
@klass.where(where_clause).first || yield
end
def where_clause def where_clause
return { project_id: @project.id } unless milestone? || label? return { project_id: @project.id } unless milestone? || label?
@attributes.slice('title').map do |key, value| @attributes.slice('title').map do |key, value|
project_clause = table[key].eq(value).and(table[:project_id].eq(@project.id))
if @group if @group
project_clause.or(table[key].eq(value).and(table[:group_id].eq(@group.id))) project_clause(key, value).or(group_clause(key, value))
else else
project_clause project_clause(key, value)
end end
end.reduce(:or) end.reduce(:or)
end end
def group_clause(key, value)
table[key].eq(value).and(table[:group_id].eq(@group.id))
end
def project_clause(key, value)
table[key].eq(value).and(table[:project_id].eq(@project.id))
end
def table def table
@table ||= @klass.arel_table @table ||= @klass.arel_table
end end
...@@ -71,7 +82,7 @@ module Gitlab ...@@ -71,7 +82,7 @@ module Gitlab
@klass == Milestone @klass == Milestone
end end
# If an existing group milesone used the IIID # If an existing group milesone used the IID
# claim the IID back and set the group milestone to use one available # claim the IID back and set the group milestone to use one available
# This is neccessary to fix situations like the following: # This is neccessary to fix situations like the following:
# - Importing into a user namespace project with exported group milestones # - Importing into a user namespace project with exported group milestones
......
...@@ -266,7 +266,7 @@ module Gitlab ...@@ -266,7 +266,7 @@ module Gitlab
hash.delete('project_id') hash.delete('project_id')
end end
GroupProjectFinder.find_or_create(relation_class, finder_hash) GroupProjectObjectBuilder.build(relation_class, finder_hash)
end end
end end
end end
......
require 'spec_helper'
describe Gitlab::ImportExport::GroupProjectFinder do
let(:project) do
create(:project,
:builds_disabled,
:issues_disabled,
name: 'project',
path: 'project',
group: create(:group))
end
context 'labels' do
it 'finds the right group label' do
group_label = create(:group_label, 'name': 'group label', 'group': project.group)
expect(described_class.find_or_new(Label,
title: 'group label',
project: project,
group: project.group)).to eq(group_label)
end
it 'initializes a new label' do
label = described_class.find_or_new(Label,
title: 'group label',
project: project,
group: project.group)
expect(label.persisted?).to be false
end
it 'creates a new label' do
label = described_class.find_or_create(Label,
title: 'group label',
project: project,
group: project.group)
expect(label.persisted?).to be true
end
end
context 'milestones' do
it 'finds the right group milestone' do
milestone = create(:milestone, 'name' => 'group milestone', 'group' => project.group)
expect(described_class.find_or_new(Milestone,
title: 'group milestone',
project: project,
group: project.group)).to eq(milestone)
end
it 'initializes a new milestone' do
milestone = described_class.find_or_new(Milestone,
title: 'group milestone',
project: project,
group: project.group)
expect(milestone.persisted?).to be false
end
it 'creates a new milestone' do
milestone = described_class.find_or_create(Milestone,
title: 'group milestone',
project: project,
group: project.group)
expect(milestone.persisted?).to be true
end
end
end
require 'spec_helper'
describe Gitlab::ImportExport::GroupProjectObjectBuilder do
let(:project) do
create(:project,
:builds_disabled,
:issues_disabled,
name: 'project',
path: 'project',
group: create(:group))
end
context 'labels' do
it 'finds the right group label' do
group_label = create(:group_label, 'name': 'group label', 'group': project.group)
expect(described_class.build(Label,
title: 'group label',
project: project,
group: project.group)).to eq(group_label)
end
it 'initializes a new label' do
label = described_class.build(Label,
title: 'group label',
project: project,
group: project.group)
expect(label.persisted?).to be false
end
end
context 'milestones' do
it 'finds the right group milestone' do
milestone = create(:milestone, 'name' => 'group milestone', 'group' => project.group)
expect(described_class.build(Milestone,
title: 'group milestone',
project: project,
group: project.group)).to eq(milestone)
end
it 'creates a new milestone' do
milestone = described_class.build(Milestone,
title: 'group milestone',
project: project,
group: project.group)
expect(milestone.persisted?).to be true
end
end
end
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