Commit 9b802ed9 authored by Simon Knox's avatar Simon Knox

Merge branch '2518-persisted-issue-boards-filter-be' of...

Merge branch '2518-persisted-issue-boards-filter-be' of gitlab.com:gitlab-org/gitlab-ee into edit-board
parents 51004d0e fa0d0bf2
class BoardFilter < ActiveRecord::Base
belongs_to :board
belongs_to :milestone
belongs_to :author, class_name: 'User'
belongs_to :assignee, class_name: 'User'
has_many :board_filter_labels
has_many :labels, through: :board_filter_labels
validates :board, presence: true
def milestone
return nil unless board.parent.feature_available?(:scoped_issue_board)
if milestone_id == ::Milestone::Upcoming.id
::Milestone::Upcoming
else
super
end
end
end
class BoardFilterLabel < ActiveRecord::Base class BoardFilterLabel < ActiveRecord::Base
belongs_to :board_filter belongs_to :board
belongs_to :label belongs_to :label
validates :board_filter, presence: true validates :board, presence: true
validates :label, presence: true validates :label, presence: true
validates :board_filter, uniqueness: { scope: :label_id }
end end
class CreateBoardFiltersTable < ActiveRecord::Migration
include Gitlab::Database::MigrationHelpers
DOWNTIME = false
disable_ddl_transaction!
def up
create_table :board_filters do |t|
t.integer :board_id, null: false, index: true
t.integer :milestone_id, index: true
t.integer :weight, index: true
t.integer :author_id, index: true
t.integer :assignee_id, index: true
end
add_concurrent_foreign_key :board_filters, :boards, column: :board_id, on_delete: :cascade
add_concurrent_foreign_key :board_filters, :milestones, column: :milestone_id, on_delete: :nullify
add_concurrent_foreign_key :board_filters, :users, column: :author_id, on_delete: :nullify
add_concurrent_foreign_key :board_filters, :users, column: :assignee_id, on_delete: :nullify
end
def down
drop_table :board_filters
end
end
class GenerateBoardFilters < ActiveRecord::Migration
DOWNTIME = false
def up
# Sub-query executed on production
# https://gitlab.com/gitlab-com/infrastructure/issues/2839#note_41023984
execute <<-SQL
INSERT INTO board_filters(board_id, milestone_id)
SELECT id as board_id, milestone_id FROM boards
WHERE (boards.milestone_id IS NOT NULL);
SQL
end
def down
execute <<-SQL
DELETE FROM board_filters;
SQL
end
end
class AddBoardFilterFields < ActiveRecord::Migration
include Gitlab::Database::MigrationHelpers
DOWNTIME = false
disable_ddl_transaction!
def up
add_column :boards, :weight, :integer, index: true
add_reference :boards, :author, index: true
add_reference :boards, :assignee, index: true
add_concurrent_foreign_key :boards, :users, column: :author_id, on_delete: :nullify
add_concurrent_foreign_key :boards, :users, column: :assignee_id, on_delete: :nullify
end
def down
remove_column :boards, :weight
remove_foreign_key :boards, column: :author_id
remove_reference :boards, :author
remove_foreign_key :boards, column: :assignee_id
remove_reference :boards, :assignee
end
end
...@@ -7,13 +7,13 @@ class CreateBoardFilterLabels < ActiveRecord::Migration ...@@ -7,13 +7,13 @@ class CreateBoardFilterLabels < ActiveRecord::Migration
def up def up
create_table :board_filter_labels do |t| create_table :board_filter_labels do |t|
t.integer :board_filter_id, null: false, index: true t.integer :board_id, null: false, index: true
t.integer :label_id, null: false, index: true t.integer :label_id, null: false, index: true
end end
add_index :board_filter_labels, [:board_filter_id, :label_id], unique: true add_index :board_filter_labels, [:board_id, :label_id], unique: true
add_concurrent_foreign_key :board_filter_labels, :board_filters, column: :board_filter_id, on_delete: :cascade add_concurrent_foreign_key :board_filter_labels, :boards, column: :board_id, on_delete: :cascade
add_concurrent_foreign_key :board_filter_labels, :labels, column: :label_id, on_delete: :cascade add_concurrent_foreign_key :board_filter_labels, :labels, column: :label_id, on_delete: :cascade
end end
......
...@@ -11,7 +11,7 @@ ...@@ -11,7 +11,7 @@
# #
# It's strongly recommended that you check this file into your version control system. # It's strongly recommended that you check this file into your version control system.
ActiveRecord::Schema.define(version: 20170921203824) do ActiveRecord::Schema.define(version: 20170926203418) do
# These are extensions that must be enabled in order to support this database # These are extensions that must be enabled in order to support this database
enable_extension "plpgsql" enable_extension "plpgsql"
...@@ -215,28 +215,14 @@ ActiveRecord::Schema.define(version: 20170921203824) do ...@@ -215,28 +215,14 @@ ActiveRecord::Schema.define(version: 20170921203824) do
add_index "award_emoji", ["user_id", "name"], name: "index_award_emoji_on_user_id_and_name", using: :btree add_index "award_emoji", ["user_id", "name"], name: "index_award_emoji_on_user_id_and_name", using: :btree
create_table "board_filter_labels", force: :cascade do |t| create_table "board_filter_labels", force: :cascade do |t|
t.integer "board_filter_id", null: false t.integer "board_id", null: false
t.integer "label_id", null: false t.integer "label_id", null: false
end end
add_index "board_filter_labels", ["board_filter_id", "label_id"], name: "index_board_filter_labels_on_board_filter_id_and_label_id", unique: true, using: :btree add_index "board_filter_labels", ["board_id", "label_id"], name: "index_board_filter_labels_on_board_id_and_label_id", unique: true, using: :btree
add_index "board_filter_labels", ["board_filter_id"], name: "index_board_filter_labels_on_board_filter_id", using: :btree add_index "board_filter_labels", ["board_id"], name: "index_board_filter_labels_on_board_id", using: :btree
add_index "board_filter_labels", ["label_id"], name: "index_board_filter_labels_on_label_id", using: :btree add_index "board_filter_labels", ["label_id"], name: "index_board_filter_labels_on_label_id", using: :btree
create_table "board_filters", force: :cascade do |t|
t.integer "board_id", null: false
t.integer "milestone_id"
t.integer "weight"
t.integer "author_id"
t.integer "assignee_id"
end
add_index "board_filters", ["assignee_id"], name: "index_board_filters_on_assignee_id", using: :btree
add_index "board_filters", ["author_id"], name: "index_board_filters_on_author_id", using: :btree
add_index "board_filters", ["board_id"], name: "index_board_filters_on_board_id", using: :btree
add_index "board_filters", ["milestone_id"], name: "index_board_filters_on_milestone_id", using: :btree
add_index "board_filters", ["weight"], name: "index_board_filters_on_weight", using: :btree
create_table "boards", force: :cascade do |t| create_table "boards", force: :cascade do |t|
t.integer "project_id" t.integer "project_id"
t.datetime "created_at", null: false t.datetime "created_at", null: false
...@@ -244,8 +230,13 @@ ActiveRecord::Schema.define(version: 20170921203824) do ...@@ -244,8 +230,13 @@ ActiveRecord::Schema.define(version: 20170921203824) do
t.string "name", default: "Development", null: false t.string "name", default: "Development", null: false
t.integer "milestone_id" t.integer "milestone_id"
t.integer "group_id" t.integer "group_id"
t.integer "weight"
t.integer "author_id"
t.integer "assignee_id"
end end
add_index "boards", ["assignee_id"], name: "index_boards_on_assignee_id", using: :btree
add_index "boards", ["author_id"], name: "index_boards_on_author_id", using: :btree
add_index "boards", ["group_id"], name: "index_boards_on_group_id", using: :btree add_index "boards", ["group_id"], name: "index_boards_on_group_id", using: :btree
add_index "boards", ["milestone_id"], name: "index_boards_on_milestone_id", using: :btree add_index "boards", ["milestone_id"], name: "index_boards_on_milestone_id", using: :btree
add_index "boards", ["project_id"], name: "index_boards_on_project_id", using: :btree add_index "boards", ["project_id"], name: "index_boards_on_project_id", using: :btree
...@@ -2069,14 +2060,12 @@ ActiveRecord::Schema.define(version: 20170921203824) do ...@@ -2069,14 +2060,12 @@ ActiveRecord::Schema.define(version: 20170921203824) do
add_foreign_key "approvals", "merge_requests", name: "fk_310d714958", on_delete: :cascade add_foreign_key "approvals", "merge_requests", name: "fk_310d714958", on_delete: :cascade
add_foreign_key "approver_groups", "namespaces", column: "group_id", on_delete: :cascade add_foreign_key "approver_groups", "namespaces", column: "group_id", on_delete: :cascade
add_foreign_key "board_filter_labels", "board_filters", name: "fk_ebc90d2f1a", on_delete: :cascade add_foreign_key "board_filter_labels", "boards", name: "fk_53e44f3a07", on_delete: :cascade
add_foreign_key "board_filter_labels", "labels", name: "fk_91e18fdcee", on_delete: :cascade add_foreign_key "board_filter_labels", "labels", name: "fk_91e18fdcee", on_delete: :cascade
add_foreign_key "board_filters", "boards", name: "fk_87e919b0eb", on_delete: :cascade
add_foreign_key "board_filters", "milestones", name: "fk_37d28eeebc", on_delete: :nullify
add_foreign_key "board_filters", "users", column: "assignee_id", name: "fk_e7893dfa6e", on_delete: :nullify
add_foreign_key "board_filters", "users", column: "author_id", name: "fk_b341da4d2b", on_delete: :nullify
add_foreign_key "boards", "namespaces", column: "group_id", name: "fk_1e9a074a35", on_delete: :cascade add_foreign_key "boards", "namespaces", column: "group_id", name: "fk_1e9a074a35", on_delete: :cascade
add_foreign_key "boards", "projects", name: "fk_f15266b5f9", on_delete: :cascade add_foreign_key "boards", "projects", name: "fk_f15266b5f9", on_delete: :cascade
add_foreign_key "boards", "users", column: "assignee_id", name: "fk_2a3450e77c", on_delete: :nullify
add_foreign_key "boards", "users", column: "author_id", name: "fk_58e8fc64f3", on_delete: :nullify
add_foreign_key "chat_teams", "namespaces", on_delete: :cascade add_foreign_key "chat_teams", "namespaces", on_delete: :cascade
add_foreign_key "ci_builds", "ci_pipelines", column: "auto_canceled_by_id", name: "fk_a2141b1522", on_delete: :nullify add_foreign_key "ci_builds", "ci_pipelines", column: "auto_canceled_by_id", name: "fk_a2141b1522", on_delete: :nullify
add_foreign_key "ci_builds", "ci_stages", column: "stage_id", name: "fk_3a9eaa254d", on_delete: :cascade add_foreign_key "ci_builds", "ci_stages", column: "stage_id", name: "fk_3a9eaa254d", on_delete: :cascade
......
...@@ -55,7 +55,7 @@ module EE ...@@ -55,7 +55,7 @@ module EE
end end
def board_params def board_params
params.require(:board).permit(:name, :milestone_id) params.require(:board).permit(:name, :weight, :milestone_id, :author_id, :assignee_id)
end end
def find_board def find_board
......
...@@ -3,9 +3,13 @@ module EE ...@@ -3,9 +3,13 @@ module EE
extend ActiveSupport::Concern extend ActiveSupport::Concern
prepended do prepended do
has_one :board_filter
belongs_to :group belongs_to :group
belongs_to :milestone
belongs_to :author, class_name: 'User'
belongs_to :assignee, class_name: 'User'
has_many :board_filter_labels
has_many :labels, through: :board_filter_labels
validates :name, presence: true validates :name, presence: true
validates :group, presence: true, unless: :project validates :group, presence: true, unless: :project
...@@ -23,6 +27,16 @@ module EE ...@@ -23,6 +27,16 @@ module EE
group_id.present? group_id.present?
end end
def milestone
return nil unless parent.feature_available?(:scoped_issue_board)
if milestone_id == ::Milestone::Upcoming.id
::Milestone::Upcoming
else
super
end
end
def as_json(options = {}) def as_json(options = {})
milestone_attrs = options.fetch(:include, {}) milestone_attrs = options.fetch(:include, {})
.extract!(:milestone) .extract!(:milestone)
......
...@@ -36,17 +36,35 @@ describe Projects::BoardsController do ...@@ -36,17 +36,35 @@ describe Projects::BoardsController do
end end
context 'with valid params' do context 'with valid params' do
let(:user) { create(:user) }
let(:milestone) { create(:milestone) }
let(:valid_params) do
{ name: 'Backend',
weight: 1,
milestone_id: milestone.id,
author_id: user.id,
assignee_id: user.id }
end
it 'returns a successful 200 response' do it 'returns a successful 200 response' do
create_board name: 'Backend' create_board valid_params
expect(response).to have_http_status(200) expect(response).to have_http_status(200)
end end
it 'returns the created board' do it 'returns the created board' do
create_board name: 'Backend' create_board valid_params
expect(response).to match_response_schema('board') expect(response).to match_response_schema('board')
end end
it 'valid board is created' do
create_board valid_params
expect(Board.count).to eq(1)
expect(Board.first).to have_attributes(valid_params)
end
end end
context 'with invalid params' do context 'with invalid params' do
...@@ -80,10 +98,10 @@ describe Projects::BoardsController do ...@@ -80,10 +98,10 @@ describe Projects::BoardsController do
expect(response).to have_http_status(404) expect(response).to have_http_status(404)
end end
def create_board(name:) def create_board(board_params)
post :create, namespace_id: project.namespace.to_param, post :create, namespace_id: project.namespace.to_param,
project_id: project.to_param, project_id: project.to_param,
board: { name: name }, board: board_params,
format: :json format: :json
end end
end end
......
require 'spec_helper' require 'spec_helper'
describe Board do describe Board do
let(:board) { create(:board) }
it { is_expected.to include_module(EE::Board) }
context 'validations' do context 'validations' do
context 'when group is present' do context 'when group is present' do
subject { described_class.new(group: create(:group)) } subject { described_class.new(group: create(:group)) }
...@@ -16,4 +20,39 @@ describe Board do ...@@ -16,4 +20,39 @@ describe Board do
it { is_expected.not_to validate_presence_of(:group) } it { is_expected.not_to validate_presence_of(:group) }
end end
end end
describe 'milestone' do
context 'when the feature is available' do
before do
stub_licensed_features(scoped_issue_board: true)
end
it 'returns Milestone::Upcoming for upcoming milestone id' do
board.milestone_id = Milestone::Upcoming.id
expect(board.milestone).to eq Milestone::Upcoming
end
it 'returns milestone for valid milestone id' do
milestone = create(:milestone)
board.milestone_id = milestone.id
expect(board.milestone).to eq milestone
end
it 'returns nil for invalid milestone id' do
board.milestone_id = -1
expect(board.milestone).to be_nil
end
end
it 'returns nil when the feature is not available' do
stub_licensed_features(scoped_issue_board: false)
milestone = create(:milestone)
board.milestone_id = milestone.id
expect(board.milestone).to be_nil
end
end
end end
FactoryGirl.define do FactoryGirl.define do
factory :board_filter_label do factory :board_filter_label do
association :board_filter association :board
association :label association :label
end end
end end
FactoryGirl.define do
factory :board_filter do
association :board
association :milestone
association :author, factory: :user
association :assignee, factory: :user
end
end
require 'spec_helper' require 'spec_helper'
describe BoardFilterLabel, type: :model do describe BoardFilterLabel do
describe 'validations' do describe 'validations' do
subject { create(:board_filter_label) } it { is_expected.to validate_presence_of(:board) }
it { is_expected.to validate_presence_of(:board_filter) }
it { is_expected.to validate_presence_of(:label) } it { is_expected.to validate_presence_of(:label) }
it { is_expected.to validate_uniqueness_of(:board_filter).scoped_to(:label_id) }
end end
describe 'associations' do describe 'associations' do
it { is_expected.to belong_to(:board_filter) }
it { is_expected.to belong_to(:label) } it { is_expected.to belong_to(:label) }
end end
end end
\ No newline at end of file
require 'spec_helper'
describe BoardFilter, type: :model do
describe 'validations' do
it { is_expected.to validate_presence_of(:board) }
end
describe 'associations' do
it { is_expected.to belong_to(:board) }
it { is_expected.to belong_to(:milestone) }
it { is_expected.to belong_to(:author).class_name('User') }
it { is_expected.to belong_to(:assignee).class_name('User') }
it { is_expected.to have_many(:board_filter_labels) }
it { is_expected.to have_many(:labels).through(:board_filter_labels) }
end
describe 'milestone' do
subject(:board_filter) { build(:board_filter) }
context 'when the feature is available' do
before do
stub_licensed_features(scoped_issue_board: true)
end
it 'returns Milestone::Upcoming for upcoming milestone id' do
board_filter.milestone_id = Milestone::Upcoming.id
expect(board_filter.milestone).to eq Milestone::Upcoming
end
it 'returns milestone for valid milestone id' do
milestone = create(:milestone)
board_filter.milestone_id = milestone.id
expect(board_filter.milestone).to eq milestone
end
it 'returns nil for invalid milestone id' do
board_filter.milestone_id = -1
expect(board_filter.milestone).to be_nil
end
end
it 'returns nil when the feature is not available' do
stub_licensed_features(scoped_issue_board: false)
milestone = create(:milestone)
board_filter.milestone_id = milestone.id
expect(board_filter.milestone).to be_nil
end
end
end
...@@ -4,6 +4,10 @@ describe Board do ...@@ -4,6 +4,10 @@ describe Board do
describe 'relationships' do describe 'relationships' do
it { is_expected.to belong_to(:project) } it { is_expected.to belong_to(:project) }
it { is_expected.to belong_to(:milestone) } it { is_expected.to belong_to(:milestone) }
it { is_expected.to belong_to(:author).class_name('User') }
it { is_expected.to belong_to(:assignee).class_name('User') }
it { is_expected.to have_many(:board_filter_labels) }
it { is_expected.to have_many(:labels).through(:board_filter_labels) }
it { is_expected.to have_many(:lists).order(list_type: :asc, position: :asc).dependent(:delete_all) } it { is_expected.to have_many(:lists).order(list_type: :asc, position: :asc).dependent(:delete_all) }
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