Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Boxiang Sun
gitlab-ce
Commits
2bdf6ede
Commit
2bdf6ede
authored
May 16, 2018
by
Rémy Coutable
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Simplify Gitlab::CurrentSettings now that the logic is in CacheableAttributes
Signed-off-by:
Rémy Coutable
<
remy@rymai.me
>
parent
a46929ea
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
69 additions
and
76 deletions
+69
-76
lib/gitlab/current_settings.rb
lib/gitlab/current_settings.rb
+18
-26
spec/lib/gitlab/current_settings_spec.rb
spec/lib/gitlab/current_settings_spec.rb
+51
-50
No files found.
lib/gitlab/current_settings.rb
View file @
2bdf6ede
...
...
@@ -9,8 +9,8 @@ module Gitlab
end
end
def
fake_application_settings
(
defaults
=
::
ApplicationSetting
.
defaults
)
Gitlab
::
FakeApplicationSettings
.
new
(
defaults
)
def
fake_application_settings
(
attributes
=
{}
)
Gitlab
::
FakeApplicationSettings
.
new
(
::
ApplicationSetting
.
defaults
.
merge
(
attributes
||
{})
)
end
def
method_missing
(
name
,
*
args
,
&
block
)
...
...
@@ -25,43 +25,35 @@ module Gitlab
def
ensure_application_settings!
return
in_memory_application_settings
if
ENV
[
'IN_MEMORY_APPLICATION_SETTINGS'
]
==
'true'
cached_application_settings
||
uncached_application_settings
end
def
cached_application_settings
begin
::
ApplicationSetting
.
cached
rescue
::
Redis
::
BaseError
,
::
Errno
::
ENOENT
,
::
Errno
::
EADDRNOTAVAIL
# In case Redis isn't running or the Redis UNIX socket file is not available
end
end
def
uncached_application_settings
return
fake_application_settings
unless
connect_to_db?
db_settings
=
::
ApplicationSetting
.
current
current_settings
=
::
ApplicationSetting
.
current
# If there are pending migrations, it's possible there are columns that
# need to be added to the application settings. To prevent Rake tasks
# and other callers from failing, use any loaded settings and return
# defaults for missing columns.
if
ActiveRecord
::
Migrator
.
needs_migration?
defaults
=
::
ApplicationSetting
.
defaults
defaults
.
merge!
(
db_settings
.
attributes
.
symbolize_keys
)
if
db_settings
.
present?
return
fake_application_settings
(
defaults
)
return
fake_application_settings
(
current_settings
&
.
attributes
)
end
return
db_settings
if
db
_settings
.
present?
return
current_settings
if
current
_settings
.
present?
::
ApplicationSetting
.
create_from_defaults
||
in_memory_application_settings
with_fallback_to_fake_application_settings
do
::
ApplicationSetting
.
create_from_defaults
||
in_memory_application_settings
end
end
def
in_memory_application_settings
@in_memory_application_settings
||=
::
ApplicationSetting
.
new
(
::
ApplicationSetting
.
defaults
)
# rubocop:disable Gitlab/ModuleWithInstanceVariables
rescue
ActiveRecord
::
StatementInvalid
,
ActiveRecord
::
UnknownAttributeError
# In case migrations the application_settings table is not created yet,
# we fallback to a simple OpenStruct
with_fallback_to_fake_application_settings
do
@in_memory_application_settings
||=
::
ApplicationSetting
.
build_from_defaults
# rubocop:disable Gitlab/ModuleWithInstanceVariables
end
end
def
with_fallback_to_fake_application_settings
(
&
block
)
yield
rescue
# In case the application_settings table is not created yet, or if a new
# ApplicationSetting column is not yet migrated we fallback to a simple OpenStruct
fake_application_settings
end
...
...
spec/lib/gitlab/current_settings_spec.rb
View file @
2bdf6ede
require
'spec_helper'
describe
Gitlab
::
CurrentSettings
do
include
StubENV
before
do
stub_env
(
'IN_MEMORY_APPLICATION_SETTINGS'
,
'false'
)
end
describe
'#current_application_settings'
do
describe
'#current_application_settings'
,
:use_clean_rails_memory_store_caching
do
it
'allows keys to be called directly'
do
db_settings
=
create
(
:application_setting
,
home_page_url:
'http://mydomain.com'
,
signup_enabled:
false
)
home_page_url:
'http://mydomain.com'
,
signup_enabled:
false
)
expect
(
described_class
.
home_page_url
).
to
eq
(
db_settings
.
home_page_url
)
expect
(
described_class
.
signup_enabled?
).
to
be_falsey
...
...
@@ -19,46 +17,54 @@ describe Gitlab::CurrentSettings do
expect
(
described_class
.
metrics_sample_interval
).
to
be
(
15
)
end
context
'w
ith DB availabl
e'
do
context
'w
hen ENV["IN_MEMORY_APPLICATION_SETTINGS"] is tru
e'
do
before
do
# For some reason, `allow(described_class).to receive(:connect_to_db?).and_return(true)` causes issues
# during the initialization phase of the test suite, so instead let's mock the internals of it
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:active?
).
and_return
(
true
)
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:table_exists?
).
and_call_original
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:table_exists?
).
with
(
'application_settings'
).
and_return
(
true
)
stub_env
(
'IN_MEMORY_APPLICATION_SETTINGS'
,
'true'
)
end
it
'
attempts to use cached values firs
t'
do
expect
(
ApplicationSetting
).
to
receive
(
:cached
)
it
'
returns an in-memory ApplicationSetting objec
t'
do
expect
(
ApplicationSetting
).
not_to
receive
(
:current
)
expect
(
described_class
.
current_application_settings
).
to
be_a
(
ApplicationSetting
)
expect
(
described_class
.
current_application_settings
).
not_to
be_persisted
end
end
it
'falls back to DB if Redis returns an empty valu
e'
do
expect
(
ApplicationSetting
).
to
receive
(
:cached
).
and_return
(
nil
)
expect
(
ApplicationSetting
).
to
receive
(
:last
).
and_call_original
.
twice
expect
(
described_class
.
current_application_settings
).
to
be_a
(
ApplicationSetting
)
context
'with DB unavailabl
e'
do
before
do
# For some reason, `allow(described_class).to receive(:connect_to_db?).and_return(false)` causes issues
# during the initialization phase of the test suite, so instead let's mock the internals of it
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:active?
).
and_return
(
false
)
end
it
'
falls back to DB if Redis fails
'
do
db_settings
=
ApplicationSetting
.
create!
(
ApplicationSetting
.
defaults
)
it
'
returns an in-memory ApplicationSetting object
'
do
expect
(
ApplicationSetting
).
not_to
receive
(
:current
)
expect
(
ApplicationSetting
).
to
receive
(
:cached
).
and_raise
(
::
Redis
::
BaseError
)
expect
(
Rails
.
cache
).
to
receive
(
:fetch
).
with
(
ApplicationSetting
::
CACHE_KEY
).
and_raise
(
Redis
::
BaseError
)
expect
(
described_class
.
current_application_settings
).
to
be_a
(
Gitlab
::
FakeApplicationSettings
)
end
end
expect
(
described_class
.
current_application_settings
).
to
eq
(
db_settings
)
context
'with DB available'
do
# This method returns the ::ApplicationSetting.defaults hash
# but with respect of custom attribute accessors of ApplicationSetting model
def
settings_from_defaults
ar_wrapped_defaults
=
::
ApplicationSetting
.
build_from_defaults
.
attributes
ar_wrapped_defaults
.
slice
(
*::
ApplicationSetting
.
defaults
.
keys
)
end
it
'creates default ApplicationSettings if none are present'
do
expect
(
ApplicationSetting
).
to
receive
(
:cached
).
and_raise
(
::
Redis
::
BaseError
)
expect
(
Rails
.
cache
).
to
receive
(
:fetch
).
with
(
ApplicationSetting
::
CACHE_KEY
).
and_raise
(
Redis
::
BaseError
)
before
do
# For some reason, `allow(described_class).to receive(:connect_to_db?).and_return(true)` causes issues
# during the initialization phase of the test suite, so instead let's mock the internals of it
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:active?
).
and_return
(
true
)
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:cached_table_exists?
).
with
(
'application_settings'
).
and_return
(
true
)
end
it
'creates default ApplicationSettings if none are present'
do
settings
=
described_class
.
current_application_settings
expect
(
settings
).
to
be_a
(
ApplicationSetting
)
expect
(
settings
).
to
be_persisted
expect
(
settings
).
to
have_attributes
(
ApplicationSetting
.
defaults
)
expect
(
settings
).
to
have_attributes
(
settings_from_
defaults
)
end
context
'with migrations pending'
do
...
...
@@ -69,7 +75,7 @@ describe Gitlab::CurrentSettings do
it
'returns an in-memory ApplicationSetting object'
do
settings
=
described_class
.
current_application_settings
expect
(
settings
).
to
be_a
(
OpenStruct
)
expect
(
settings
).
to
be_a
(
Gitlab
::
FakeApplicationSettings
)
expect
(
settings
.
sign_in_enabled?
).
to
eq
(
settings
.
sign_in_enabled
)
expect
(
settings
.
sign_up_enabled?
).
to
eq
(
settings
.
sign_up_enabled
)
end
...
...
@@ -81,7 +87,7 @@ describe Gitlab::CurrentSettings do
settings
=
described_class
.
current_application_settings
app_defaults
=
ApplicationSetting
.
last
expect
(
settings
).
to
be_a
(
OpenStruct
)
expect
(
settings
).
to
be_a
(
Gitlab
::
FakeApplicationSettings
)
expect
(
settings
.
home_page_url
).
to
eq
(
db_settings
.
home_page_url
)
expect
(
settings
.
signup_enabled?
).
to
be_falsey
expect
(
settings
.
signup_enabled
).
to
be_falsey
...
...
@@ -91,34 +97,29 @@ describe Gitlab::CurrentSettings do
settings
.
each
{
|
key
,
_
|
expect
(
settings
[
key
]).
to
eq
(
app_defaults
[
key
])
}
end
end
end
context
'with DB unavailable'
do
before
do
# For some reason, `allow(described_class).to receive(:connect_to_db?).and_return(false)` causes issues
# during the initialization phase of the test suite, so instead let's mock the internals of it
allow
(
ActiveRecord
::
Base
.
connection
).
to
receive
(
:active?
).
and_return
(
false
)
end
it
'returns an in-memory ApplicationSetting objec
t'
do
expect
(
ApplicationSetting
).
not_to
receive
(
:current
)
expect
(
ApplicationSetting
).
not_to
receive
(
:last
)
context
'when ApplicationSettings.current is presen
t'
do
it
'returns the existing application settings'
do
expect
(
ApplicationSetting
).
to
receive
(
:current
).
and_return
(
:current_settings
)
expect
(
described_class
.
current_application_settings
).
to
be_a
(
OpenStruct
)
expect
(
described_class
.
current_application_settings
).
to
eq
(
:current_settings
)
end
end
end
context
'when ENV["IN_MEMORY_APPLICATION_SETTINGS"] is true'
do
before
do
stub_env
(
'IN_MEMORY_APPLICATION_SETTINGS'
,
'true'
)
context
'when the application_settings table does not exists'
do
it
'returns an in-memory ApplicationSetting object'
do
expect
(
ApplicationSetting
).
to
receive
(
:create_from_defaults
).
and_raise
(
ActiveRecord
::
StatementInvalid
)
expect
(
described_class
.
current_application_settings
).
to
be_a
(
Gitlab
::
FakeApplicationSettings
)
end
end
it
'returns an in-memory ApplicationSetting object
'
do
expect
(
ApplicationSetting
).
not_to
receive
(
:current
)
expect
(
ApplicationSetting
).
not_to
receive
(
:last
)
context
'when the application_settings table is not fully migrated
'
do
it
'returns an in-memory ApplicationSetting object'
do
expect
(
ApplicationSetting
).
to
receive
(
:create_from_defaults
).
and_raise
(
ActiveRecord
::
UnknownAttributeError
)
expect
(
described_class
.
current_application_settings
).
to
be_a
(
ApplicationSetting
)
e
xpect
(
described_class
.
current_application_settings
).
not_to
be_persiste
d
expect
(
described_class
.
current_application_settings
).
to
be_a
(
Gitlab
::
FakeApplicationSettings
)
e
n
d
end
end
end
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment