Skip to content

Commit 1469653

Browse files
Merge remote-tracking branch 'origin/release/15.4' into dev
2 parents db873de + 1fb6c87 commit 1469653

File tree

9 files changed

+340
-24
lines changed

9 files changed

+340
-24
lines changed

.github/dangerfiles/release_migrations/Dangerfile

+2-2
Original file line numberDiff line numberDiff line change
@@ -29,10 +29,10 @@
2929
#++
3030
CORE_OR_MODULE_MIGRATIONS_REGEX = %r{(modules/.*)?db/migrate/.*\.rb}
3131

32-
def added_or_modified_migrations?
32+
def added_or_modified_migrations
3333
(git.modified_files + git.added_files).grep(CORE_OR_MODULE_MIGRATIONS_REGEX)
3434
end
3535

36-
if github.branch_for_base.match?(/^release/) && added_or_modified_migrations?
36+
if github.branch_for_base.match?(/^release/) && added_or_modified_migrations.any?
3737
warn "This PR has migration-related changes on a release branch. Ping @opf/operations"
3838
end

app/controllers/admin/settings/enumerations_controller_base.rb

-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,6 @@
3131
module Admin
3232
module Settings
3333
class EnumerationsControllerBase < ApplicationController
34-
extend ActiveSupport::Concern
3534
include OpTurbo::ComponentStream
3635

3736
before_action :require_admin
+9-8
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,10 @@
1-
<div class="advanced-filters--filter-value <%= value_visibility %>">
1+
<%= content_tag :div,
2+
class: "advanced-filters--filter-value #{value_visibility}",
3+
data: {
4+
filter_autocomplete: true,
5+
filter__filters_form_target: "filterValueContainer",
6+
filter_name: filter.name
7+
} do %>
28
<%= angular_component_tag autocomplete_options[:component],
39
inputs: {
410
inputName: "value",
@@ -13,11 +19,6 @@
1319
# the action can't be registered on the input field at the time
1420
# of the #connect lifecycle hook of the filter--filters-form
1521
# Stimulus controller.
16-
}.merge(autocomplete_options.except(:component)),
17-
data: {
18-
"filter-autocomplete": true,
19-
"filter--filters-form-target": "filterValueContainer",
20-
"filter-name": filter.name
21-
}
22+
}.merge(autocomplete_options.except(:component))
2223
%>
23-
</div>
24+
<% end %>

modules/documents/spec/requests/admin/settings/document_categories_spec.rb

+9
Original file line numberDiff line numberDiff line change
@@ -88,4 +88,13 @@
8888
end
8989
end
9090
end
91+
92+
describe "PUT /admin/settings/document_categories/:id/move" do
93+
it "moves the category to the bottom" do
94+
put move_admin_settings_document_category_path(category), params: { move_to: "lowest" }, as: :turbo_stream
95+
96+
expect(response).to have_http_status(:ok)
97+
expect(category.reload.position).to be > other_category.reload.position
98+
end
99+
end
91100
end

modules/meeting/app/models/queries/meetings/filters/attended_user_filter.rb

+27-7
Original file line numberDiff line numberDiff line change
@@ -40,23 +40,43 @@ def type_strategy
4040
@type_strategy ||= ::Queries::Filters::Strategies::IntegerListOptional.new(self)
4141
end
4242

43-
def where
44-
"meeting_participants.user_id IN (#{values.join(',')}) AND meeting_participants.attended"
43+
def where # rubocop:disable Metrics/AbcSize
44+
condition = "#{MeetingParticipant.table_name}.attended"
45+
46+
case operator
47+
when "="
48+
[operator_strategy.sql_for_field(values, MeetingParticipant.table_name, "user_id"), condition].join(" AND ")
49+
when "!"
50+
<<~SQL.squish
51+
NOT EXISTS (
52+
SELECT 1 FROM #{MeetingParticipant.table_name}
53+
WHERE #{MeetingParticipant.table_name}.meeting_id = meetings.id
54+
AND #{MeetingParticipant.table_name}.user_id = '#{MeetingParticipant.connection.quote_string(values.first)}'
55+
AND #{condition}
56+
)
57+
SQL
58+
when "*"
59+
["#{MeetingParticipant.table_name}.user_id IS NOT NULL", condition].join(" AND ")
60+
when "!*"
61+
<<~SQL.squish
62+
NOT EXISTS (
63+
SELECT 1 FROM #{MeetingParticipant.table_name}
64+
WHERE #{MeetingParticipant.table_name}.meeting_id = meetings.id
65+
AND #{condition}
66+
)
67+
SQL
68+
end
4569
end
4670

4771
def human_name
4872
I18n.t(:label_attended_user)
4973
end
5074

51-
def joins
75+
def left_outer_joins
5276
:participants
5377
end
5478

5579
def self.key
5680
:attended_user_id
5781
end
58-
59-
def available_operators
60-
[::Queries::Operators::Equals]
61-
end
6282
end

modules/meeting/app/models/queries/meetings/filters/invited_user_filter.rb

+27-6
Original file line numberDiff line numberDiff line change
@@ -40,18 +40,39 @@ def type_strategy
4040
@type_strategy ||= ::Queries::Filters::Strategies::IntegerListOptional.new(self)
4141
end
4242

43-
def where
44-
[
45-
operator_strategy.sql_for_field(values, MeetingParticipant.table_name, "user_id"),
46-
"#{MeetingParticipant.table_name}.invited"
47-
].join(" AND ")
43+
def where # rubocop:disable Metrics/AbcSize
44+
condition = "#{MeetingParticipant.table_name}.invited"
45+
46+
case operator
47+
when "="
48+
[operator_strategy.sql_for_field(values, MeetingParticipant.table_name, "user_id"), condition].join(" AND ")
49+
when "!"
50+
<<~SQL.squish
51+
NOT EXISTS (
52+
SELECT 1 FROM #{MeetingParticipant.table_name}
53+
WHERE #{MeetingParticipant.table_name}.meeting_id = meetings.id
54+
AND #{MeetingParticipant.table_name}.user_id = '#{MeetingParticipant.connection.quote_string(values.first)}'
55+
AND #{condition}
56+
)
57+
SQL
58+
when "*"
59+
["#{MeetingParticipant.table_name}.user_id IS NOT NULL", condition].join(" AND ")
60+
when "!*"
61+
<<~SQL.squish
62+
NOT EXISTS (
63+
SELECT 1 FROM #{MeetingParticipant.table_name}
64+
WHERE #{MeetingParticipant.table_name}.meeting_id = meetings.id
65+
AND #{condition}
66+
)
67+
SQL
68+
end
4869
end
4970

5071
def human_name
5172
I18n.t(:label_invited_user)
5273
end
5374

54-
def joins
75+
def left_outer_joins
5576
:participants
5677
end
5778

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,129 @@
1+
# frozen_string_literal: true
2+
3+
#-- copyright
4+
# OpenProject is an open source project management software.
5+
# Copyright (C) the OpenProject GmbH
6+
#
7+
# This program is free software; you can redistribute it and/or
8+
# modify it under the terms of the GNU General Public License version 3.
9+
#
10+
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
11+
# Copyright (C) 2006-2013 Jean-Philippe Lang
12+
# Copyright (C) 2010-2013 the ChiliProject Team
13+
#
14+
# This program is free software; you can redistribute it and/or
15+
# modify it under the terms of the GNU General Public License
16+
# as published by the Free Software Foundation; either version 2
17+
# of the License, or (at your option) any later version.
18+
#
19+
# This program is distributed in the hope that it will be useful,
20+
# but WITHOUT ANY WARRANTY; without even the implied warranty of
21+
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22+
# GNU General Public License for more details.
23+
#
24+
# You should have received a copy of the GNU General Public License
25+
# along with this program; if not, write to the Free Software
26+
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
27+
#
28+
# See COPYRIGHT and LICENSE files for more details.
29+
#++
30+
31+
require "spec_helper"
32+
33+
RSpec.describe Queries::Meetings::Filters::AttendedUserFilter do
34+
it_behaves_like "basic query filter" do
35+
let(:type) { :list_optional }
36+
let(:class_key) { :attended_user_id }
37+
let(:human_name) { I18n.t(:label_attended_user) }
38+
39+
describe "#available?" do
40+
it "is true" do
41+
expect(instance).to be_available
42+
end
43+
end
44+
45+
describe "#allowed_values" do
46+
it "is nil" do
47+
expect(instance.allowed_values).to be_nil
48+
end
49+
end
50+
end
51+
52+
describe "#where clause" do
53+
let(:user1) { create(:user) }
54+
let(:user2) { create(:user) }
55+
let(:project) { create(:project) }
56+
let(:meeting1) { create(:meeting, project:) }
57+
let(:meeting2) { create(:meeting, project:) }
58+
let(:meeting3) { create(:meeting, project:) }
59+
let(:meeting4) { create(:meeting, project:) }
60+
let!(:empty_meeting) do
61+
create(:meeting, project:).tap { |meeting| meeting.participants.delete_all }
62+
end
63+
64+
let!(:participant1) { create(:meeting_participant, :attendee, meeting: meeting1, user: user1) }
65+
let!(:participant2) { create(:meeting_participant, :attendee, meeting: meeting2, user: user2) }
66+
let!(:participant3) { create(:meeting_participant, :invitee, meeting: meeting3, user: user1) }
67+
let!(:participant4) { create(:meeting_participant, :invitee, meeting: meeting4, user: user2) }
68+
69+
let(:instance) do
70+
described_class.create!(name: :attended_user_id, operator:, values:)
71+
end
72+
73+
context 'for "="' do
74+
let(:operator) { "=" }
75+
let(:values) { [user1.id.to_s] }
76+
77+
it "finds meetings where the user attended" do
78+
expect(instance.where).to include("#{MeetingParticipant.table_name}.user_id")
79+
80+
meetings = Meeting.left_outer_joins(:participants).where(instance.where)
81+
82+
expect(meetings).to include(meeting1)
83+
expect(meetings).not_to include(meeting2)
84+
expect(meetings).not_to include(meeting3)
85+
expect(meetings).not_to include(meeting4)
86+
end
87+
end
88+
89+
context 'for "!"' do
90+
let(:operator) { "!" }
91+
let(:values) { [user1.id.to_s] }
92+
93+
it "finds meetings where the user did not attend" do
94+
expect(instance.where).to include("NOT")
95+
96+
meetings = Meeting.left_outer_joins(:participants).where(instance.where)
97+
expect(meetings.pluck(:id)).to contain_exactly(meeting2.id, empty_meeting.id, meeting3.id, meeting4.id)
98+
end
99+
end
100+
101+
context 'for "*"' do
102+
let(:operator) { "*" }
103+
let(:values) { [] }
104+
105+
it "finds meetings with any attendee" do
106+
meetings = Meeting.left_outer_joins(:participants).where(instance.where)
107+
108+
expect(meetings).to include(meeting1)
109+
expect(meetings).to include(meeting2)
110+
expect(meetings).not_to include(meeting3)
111+
expect(meetings).not_to include(meeting4)
112+
end
113+
end
114+
115+
context 'for "!*"' do
116+
let(:operator) { "!*" }
117+
let(:values) { [] }
118+
119+
it "finds meetings with no attendees" do
120+
meetings = Meeting.left_outer_joins(:participants).where(instance.where)
121+
122+
expect(meetings).not_to include(meeting1)
123+
expect(meetings).not_to include(meeting2)
124+
expect(meetings).to include(meeting3)
125+
expect(meetings).to include(meeting4)
126+
end
127+
end
128+
end
129+
end

0 commit comments

Comments
 (0)