Skip to content

Commit 1f57cdc

Browse files
authored
Implements the Authentication Configuration Checks (opf#18377)
* Implements SSO tests * Implements some rudimentary OAuth2 tests * Incorporates feedback * Adds a user bound request on SSO
1 parent 81a4788 commit 1f57cdc

File tree

7 files changed

+458
-56
lines changed

7 files changed

+458
-56
lines changed

modules/storages/app/common/storages/peripherals/connection_validators/check_result.rb

+6
Original file line numberDiff line numberDiff line change
@@ -45,8 +45,14 @@ def self.success(key)
4545
new(key:, state: :success, message: nil, timestamp: Time.zone.now)
4646
end
4747

48+
def self.warning(key, message)
49+
new(key:, state: :warning, message: message, timestamp: Time.zone.now)
50+
end
51+
4852
def success? = state == :success
4953
def failure? = state == :failure
54+
def warning? = state == :warning
55+
def skipped? = state == :skipped
5056
end
5157
end
5258
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,120 @@
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+
module Nextcloud
35+
class AuthenticationValidator < BaseValidator
36+
def initialize(storage)
37+
super
38+
@user = User.current
39+
end
40+
41+
private
42+
43+
def validate
44+
@storage.authenticate_via_idp? ? validate_sso : validate_oauth
45+
end
46+
47+
def validate_oauth
48+
register_checks(:existing_token, :user_bound_request)
49+
50+
oauth_token
51+
user_bound_request
52+
end
53+
54+
def oauth_token
55+
if OAuthClientToken.where(user: @user, oauth_client: @storage.oauth_client).any?
56+
pass_check(:existing_token)
57+
else
58+
warn_check(:existing_token, message(:oauth_token_missing), halt_validation: true)
59+
end
60+
end
61+
62+
def user_bound_request
63+
Registry["nextcloud.queries.user"].call(storage: @storage, auth_strategy:).on_failure do
64+
fail_check(:user_bound_request, message("oauth_request_#{it.result}"))
65+
end
66+
67+
pass_check(:user_bound_request)
68+
end
69+
70+
def auth_strategy = Registry["nextcloud.authentication.user_bound"].call(storage: @storage, user: @user)
71+
72+
def validate_sso
73+
register_checks(:non_provisioned_user, :provisioned_user_provider, :token_negotiable, :user_bound_request)
74+
75+
non_provisioned_user
76+
non_oidc_provisioned_user
77+
token_negotiable
78+
user_bound_request
79+
end
80+
81+
def non_provisioned_user
82+
if @user.identity_url.present?
83+
pass_check(:non_provisioned_user)
84+
else
85+
warn_check(:non_provisioned_user, message(:oidc_non_provisioned_user), halt_validation: true)
86+
end
87+
end
88+
89+
def non_oidc_provisioned_user
90+
if @user.authentication_provider.is_a?(OpenIDConnect::Provider)
91+
pass_check(:provisioned_user_provider)
92+
else
93+
warn_check(:provisioned_user_provider, message(:oidc_non_oidc_user), halt_validation: true)
94+
end
95+
end
96+
97+
def token_negotiable
98+
service = OpenIDConnect::UserTokens::FetchService.new(user: @user)
99+
100+
result = service.access_token_for(audience: @storage.audience)
101+
return pass_check(:token_negotiable) if result.success?
102+
103+
error_code = case result.failure
104+
in { code: /token_exchange/ | :unable_to_exchange_token }
105+
:oidc_cant_exchange_token
106+
in { code: /token_refresh/ }
107+
:oidc_cant_refresh_token
108+
in { code: :no_token_for_audience }
109+
:oidc_cant_acquire_token
110+
else
111+
:unknown_error
112+
end
113+
114+
fail_check(:token_negotiable, message(error_code))
115+
end
116+
end
117+
end
118+
end
119+
end
120+
end

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

+17-50
Original file line numberDiff line numberDiff line change
@@ -32,30 +32,23 @@ module Storages
3232
module Peripherals
3333
module ConnectionValidators
3434
module Nextcloud
35-
class BaseConfigurationValidator
36-
def initialize(storage)
37-
@storage = storage
38-
@results = build_result_list
39-
end
35+
class BaseConfigurationValidator < BaseValidator
36+
private
4037

41-
def call
42-
catch :interrupted do
43-
capabilities_request_failed
44-
host_url_not_found
45-
missing_dependencies
46-
version_mismatch
47-
end
38+
def validate
39+
register_checks(:capabilities_request, :host_url_accessible, :dependencies_check, :dependencies_versions)
4840

49-
@results
41+
capabilities_request_status
42+
host_url_not_found
43+
missing_dependencies
44+
version_mismatch
5045
end
5146

52-
private
53-
54-
def capabilities_request_failed
47+
def capabilities_request_status
5548
if capabilities.failure? && capabilities.result != :not_found
56-
fail_check(__method__, message(:error))
49+
fail_check(:capabilities_request, message(:error))
5750
else
58-
pass_check(__method__)
51+
pass_check(:capabilities_request)
5952
end
6053
end
6154

@@ -64,9 +57,9 @@ def version_mismatch
6457
capabilities_result = capabilities.result
6558

6659
if capabilities_result.app_version < min_app_version
67-
fail_check(__method__, message(:app_version_mismatch))
60+
fail_check(:dependencies_versions, message(:app_version_mismatch))
6861
else
69-
pass_check(__method__)
62+
pass_check(:dependencies_versions)
7063
end
7164
end
7265

@@ -75,24 +68,20 @@ def missing_dependencies
7568
app_name = I18n.t("storages.dependencies.nextcloud.integration_app")
7669

7770
if capabilities_result.app_disabled?
78-
fail_check(__method__, message(:missing_dependencies, dependency: app_name))
71+
fail_check(:dependencies_check, message(:missing_dependencies, dependency: app_name))
7972
else
80-
pass_check(__method__)
73+
pass_check(:dependencies_check)
8174
end
8275
end
8376

8477
def host_url_not_found
8578
if capabilities.result == :not_found
86-
fail_check(__method__, message(:host_not_found))
79+
fail_check(:host_url_accessible, message(:host_not_found))
8780
else
88-
pass_check(__method__)
81+
pass_check(:host_url_accessible)
8982
end
9083
end
9184

92-
def message(key, context = {})
93-
I18n.t("storages.health.connection_validation.#{key}", **context)
94-
end
95-
9685
def noop = StorageInteraction::AuthenticationStrategies::Noop.strategy
9786

9887
def capabilities
@@ -105,28 +94,6 @@ def nextcloud_dependencies
10594
end
10695

10796
def path_to_config = Rails.root.join("modules/storages/config/nextcloud_dependencies.yml")
108-
109-
def fail_check(key, message)
110-
update_result(key, CheckResult.failure(key, message))
111-
throw :interrupted
112-
end
113-
114-
def pass_check(key)
115-
update_result(key, CheckResult.success(key))
116-
end
117-
118-
def update_result(method, value)
119-
@results[method.to_sym] = value
120-
end
121-
122-
def build_result_list
123-
{
124-
capabilities_request_failed: CheckResult.skipped(:capabilities_request_failed),
125-
host_url_not_found: CheckResult.skipped(:host_url_not_found),
126-
missing_dependencies: CheckResult.skipped(:missing_dependencies),
127-
version_mismatch: CheckResult.skipped(:version_mismatch)
128-
}
129-
end
13097
end
13198
end
13299
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
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+
module Nextcloud
35+
class BaseValidator
36+
def initialize(storage)
37+
@storage = storage
38+
end
39+
40+
def call
41+
catch :interrupted do
42+
validate
43+
end
44+
45+
@results
46+
end
47+
48+
private
49+
50+
def validate = raise Errors::SubclassResponsibility
51+
52+
def register_checks(*keys)
53+
@results = keys.to_h { [it, CheckResult.skipped(it)] }
54+
end
55+
56+
def update_result(key, value)
57+
if @results&.has_key?(key)
58+
@results[key] = value
59+
else
60+
raise ArgumentError, "Check #{key} not registered."
61+
end
62+
end
63+
64+
def pass_check(key)
65+
update_result(key, CheckResult.success(key))
66+
end
67+
68+
def fail_check(key, message)
69+
update_result(key, CheckResult.failure(key, message))
70+
throw :interrupted
71+
end
72+
73+
def warn_check(key, message, halt_validation: false)
74+
update_result(key, CheckResult.warning(key, message))
75+
throw :interrupted if halt_validation
76+
end
77+
78+
def message(key, context = {})
79+
I18n.t("storages.health.connection_validation.#{key}", **context)
80+
end
81+
end
82+
end
83+
end
84+
end
85+
end

modules/storages/config/locales/en.yml

+3
Original file line numberDiff line numberDiff line change
@@ -243,6 +243,9 @@ en:
243243
host_not_found: No Nextcloud server found at the configured host url. Please check the configuration.
244244
missing_dependencies: 'A required dependency is missing on the file storage. Please add the following dependency: %{dependency}.'
245245
not_configured: The connection could not be validated. Please finish configuration first.
246+
oauth_request_not_found: OpenProject couldn't find the root folder of Nextcloud using the current logged-in user. Please check the server logs for further information.
247+
oauth_request_unauthorized: The current user isn't authorized to see the main folder of Nextcloud or we couldn't acquire a token. Please check the server logs for further information.
248+
oauth_token_missing: OpenProject cannot test the user level communication with Nextcloud as there's no token for the current user.
246249
oidc_cant_acquire_token: Your OpenID Connect setup doesn't provide the necessary audience, nor provides token exchange capabilities. Please check out our documentation for more information.
247250
oidc_cant_exchange_token: There seems to be a problem with the Token Exchange setup on your OpenID Connect Provider. Please check its configuration and try again.
248251
oidc_cant_refresh_token: There was an error while trying to check your access to the storage. Please check the server logs for further information.

0 commit comments

Comments
 (0)