Skip to content

Commit c76232b

Browse files
authored
Re-Implements the NextcloudConnectionValidator using the multple status retun (opf#18471)
* First Draft of the overall validator * Incorporates feedback
1 parent dcbda6d commit c76232b

12 files changed

+338
-72
lines changed

app/models/types/patterns/token_property_mapper.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ class TokenPropertyMapper
4747
creation_date: { fn: ->(wp) { wp.created_at }, label: -> { WorkPackage.human_attribute_name(:created_at) } },
4848
estimated_time: { fn: ->(wp) { wp.estimated_hours }, label: -> { WorkPackage.human_attribute_name(:estimated_hours) } },
4949
finish_date: { fn: ->(wp) { wp.due_date }, label: -> { WorkPackage.human_attribute_name(:due_date) } },
50-
parent_id: { fn: ->(wp) { wp.parent&.id }, label: -> { WorkPackage.human_attribute_name(:parent_id) } },
50+
parent_id: { fn: ->(wp) { wp.parent&.id }, label: -> { WorkPackage.human_attribute_name(:id) } },
5151
parent_assignee: { fn: ->(wp) { wp.parent&.assigned_to&.name }, label: -> {
5252
WorkPackage.human_attribute_name(:assigned_to)
5353
} },
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,82 @@
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+
module Storages
32+
module Peripherals
33+
module ConnectionValidators
34+
class BaseValidatorGroup
35+
include TaggedLogging
36+
37+
def initialize(storage)
38+
@storage = storage
39+
@results = ValidationGroupResult.new
40+
end
41+
42+
def call
43+
catch :interrupted do
44+
validate
45+
end
46+
47+
@results
48+
end
49+
50+
private
51+
52+
def validate = raise Errors::SubclassResponsibility
53+
54+
def register_checks(*keys)
55+
keys.each { @results.register_check(it) }
56+
end
57+
58+
def update_result(...)
59+
@results.update_result(...)
60+
end
61+
62+
def pass_check(key)
63+
update_result(key, CheckResult.success(key))
64+
end
65+
66+
def fail_check(key, message)
67+
update_result(key, CheckResult.failure(key, message))
68+
throw :interrupted
69+
end
70+
71+
def warn_check(key, message, halt_validation: false)
72+
update_result(key, CheckResult.warning(key, message))
73+
throw :interrupted if halt_validation
74+
end
75+
76+
def message(key, context = {})
77+
I18n.t("storages.health.connection_validation.#{key}", **context)
78+
end
79+
end
80+
end
81+
end
82+
end

modules/storages/app/common/storages/peripherals/connection_validators/nextcloud/ampf_connection_validator.rb modules/storages/app/common/storages/peripherals/connection_validators/nextcloud/ampf_configuration_validator.rb

+3-3
Original file line numberDiff line numberDiff line change
@@ -32,19 +32,19 @@ module Storages
3232
module Peripherals
3333
module ConnectionValidators
3434
module Nextcloud
35-
class AmpfConnectionValidator < BaseValidator
35+
class AmpfConfigurationValidator < BaseValidatorGroup
3636
using ServiceResultRefinements
3737

3838
private
3939

4040
def validate
4141
register_checks(
42-
:userless_access, :group_folder_presence, :files_request, :group_folder_contents
42+
:files_request, :userless_access, :group_folder_presence, :group_folder_contents
4343
)
4444

45+
files_request_failed_with_unknown_error
4546
userless_access_denied
4647
group_folder_not_found
47-
files_request_failed_with_unknown_error
4848
with_unexpected_content
4949
end
5050

modules/storages/app/common/storages/peripherals/connection_validators/nextcloud/authentication_validator.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ module Storages
3232
module Peripherals
3333
module ConnectionValidators
3434
module Nextcloud
35-
class AuthenticationValidator < BaseValidator
35+
class AuthenticationValidator < BaseValidatorGroup
3636
def initialize(storage)
3737
super
3838
@user = User.current

modules/storages/app/common/storages/peripherals/connection_validators/nextcloud/base_configuration_validator.rb modules/storages/app/common/storages/peripherals/connection_validators/nextcloud/storage_configuration_validator.rb

+13-3
Original file line numberDiff line numberDiff line change
@@ -32,21 +32,31 @@ module Storages
3232
module Peripherals
3333
module ConnectionValidators
3434
module Nextcloud
35-
class BaseConfigurationValidator < BaseValidator
35+
class StorageConfigurationValidator < BaseValidatorGroup
3636
private
3737

3838
def validate
39-
register_checks(:capabilities_request, :host_url_accessible, :dependencies_check, :dependencies_versions)
39+
register_checks(:storage_configured, :capabilities_request,
40+
:host_url_accessible, :dependencies_check, :dependencies_versions)
4041

42+
storage_configuration_status
4143
capabilities_request_status
4244
host_url_not_found
4345
missing_dependencies
4446
version_mismatch
4547
end
4648

49+
def storage_configuration_status
50+
if @storage.configured?
51+
pass_check(:storage_configured)
52+
else
53+
fail_check(:storage_configured, message(:not_configured))
54+
end
55+
end
56+
4757
def capabilities_request_status
4858
if capabilities.failure? && capabilities.result != :not_found
49-
fail_check(:capabilities_request, message(:error))
59+
fail_check(:capabilities_request, message(:unknown_error))
5060
else
5161
pass_check(:capabilities_request)
5262
end

modules/storages/app/common/storages/peripherals/connection_validators/nextcloud/base_validator.rb modules/storages/app/common/storages/peripherals/connection_validators/nextcloud_validator.rb

+25-40
Original file line numberDiff line numberDiff line change
@@ -31,56 +31,41 @@
3131
module Storages
3232
module Peripherals
3333
module ConnectionValidators
34-
module Nextcloud
35-
class BaseValidator
36-
include TaggedLogging
37-
38-
def initialize(storage)
39-
@storage = storage
34+
class NextcloudValidator
35+
# Class Level interface will be moved to a superclass/mixin once we do the OneDrive port
36+
class << self
37+
def validation_groups
38+
@validation_groups ||= {}
4039
end
4140

42-
def call
43-
catch :interrupted do
44-
validate
45-
end
46-
47-
@results
41+
def register_group(group_name, klass, when: ->(*) { true })
42+
validation_groups[group_name] = { klass:, when: }
4843
end
44+
end
4945

50-
private
51-
52-
def validate = raise Errors::SubclassResponsibility
46+
register_group :base_configuration, Nextcloud::StorageConfigurationValidator
47+
register_group :authentication, Nextcloud::AuthenticationValidator,
48+
when: ->(_, result) { result.group(:base_configuration).non_failure? }
49+
register_group :ampf_configuration, Nextcloud::AmpfConfigurationValidator,
50+
when: ->(storage, result) {
51+
result.group(:base_configuration).non_failure? && storage.automatic_management_enabled?
52+
}
5353

54-
def register_checks(*keys)
55-
@results = keys.to_h { [it, CheckResult.skipped(it)] }
56-
end
54+
def initialize(storage:)
55+
@storage = storage
56+
end
5757

58-
def update_result(key, value)
59-
if @results&.has_key?(key)
60-
@results[key] = value
61-
else
62-
raise ArgumentError, "Check #{key} not registered."
58+
def validate
59+
validation_groups.each_with_object(ValidatorResult.new) do |(key, group_metadata), result|
60+
if group_metadata[:when].call(@storage, result)
61+
result.add_group_result(key, group_metadata[:klass].new(@storage).call)
6362
end
6463
end
64+
end
6565

66-
def pass_check(key)
67-
update_result(key, CheckResult.success(key))
68-
end
69-
70-
def fail_check(key, message)
71-
update_result(key, CheckResult.failure(key, message))
72-
throw :interrupted
73-
end
66+
private
7467

75-
def warn_check(key, message, halt_validation: false)
76-
update_result(key, CheckResult.warning(key, message))
77-
throw :interrupted if halt_validation
78-
end
79-
80-
def message(key, context = {})
81-
I18n.t("storages.health.connection_validation.#{key}", **context)
82-
end
83-
end
68+
def validation_groups = self.class.validation_groups
8469
end
8570
end
8671
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
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+
module Storages
32+
module Peripherals
33+
module ConnectionValidators
34+
class ValidationGroupResult
35+
delegate :[], to: :@results
36+
37+
def initialize
38+
@results = {}
39+
end
40+
41+
def success? = @results.values.all?(&:success?)
42+
def non_failure? = @results.values.none?(&:failure?)
43+
def failure? = @results.values.any?(&:failure?)
44+
def warning? = @results.values.any?(&:warning?)
45+
46+
def tally
47+
@results.values.map(&:state).tally
48+
end
49+
50+
def register_checks(keys)
51+
Array(keys).each { register_check(it) }
52+
end
53+
54+
def register_check(key)
55+
warn("Overriding already defined check") if @results.key?(key)
56+
57+
@results[key] = CheckResult.skipped(key)
58+
end
59+
60+
def update_result(key, value)
61+
raise(ArgumentError, "Check #{key} not registered.") unless @results.key?(key)
62+
63+
@results[key] = value
64+
end
65+
end
66+
end
67+
end
68+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
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+
module Storages
32+
module Peripherals
33+
module ConnectionValidators
34+
class ValidatorResult
35+
private attr_reader :group_results
36+
37+
delegate :each_pair, :empty?, to: :group_results
38+
39+
def initialize
40+
@group_results = {}
41+
end
42+
43+
def healthy? = group_results.values.all?(&:success?)
44+
def unhealthy? = group_results.values.any?(&:failure?)
45+
def warning? = group_results.values.any?(&:warning?)
46+
47+
def group(key) = group_results.fetch(key)
48+
alias_method :fetch, :group
49+
50+
def add_group_result(key, result)
51+
Kernel.warn "Overwriting #{key} results" if group_results.key?(key)
52+
53+
group_results[key] = result
54+
end
55+
end
56+
end
57+
end
58+
end

modules/storages/spec/common/storages/peripherals/connection_validators/nextcloud/ampf_connection_validator_spec.rb modules/storages/spec/common/storages/peripherals/connection_validators/nextcloud/ampf_configuration_validator_spec.rb

+4-6
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ module Storages
3535
module Peripherals
3636
module ConnectionValidators
3737
module Nextcloud
38-
RSpec.describe AmpfConnectionValidator, :webmock do
38+
RSpec.describe AmpfConfigurationValidator, :webmock do
3939
let(:storage) { create(:nextcloud_storage_configured, :as_automatically_managed) }
4040
let(:project_folder_id) { "1337" }
4141
let!(:project_storage) do
@@ -57,9 +57,7 @@ module Nextcloud
5757
end
5858

5959
it "pass all checks" do
60-
results = validator.call
61-
62-
expect(results.values).to all(be_success)
60+
expect(validator.call).to be_success
6361
end
6462

6563
context "if userless authentication fails" do
@@ -68,8 +66,8 @@ module Nextcloud
6866
it "fails and skips the next checks" do
6967
results = validator.call
7068

71-
states = results.values.map { it.state }.tally
72-
expect(states).to eq({ failure: 1, skipped: 3 })
69+
states = results.tally
70+
expect(states).to eq({ success: 1, failure: 1, skipped: 2 })
7371
expect(results[:userless_access]).to be_failure
7472
expect(results[:userless_access].message).to eq(i18n_message(:userless_access_denied))
7573
end

0 commit comments

Comments
 (0)