2019-05-18 00:54:41 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
describe ApplicationSetting do
|
2019-05-18 00:54:41 +05:30
|
|
|
subject(:setting) { described_class.create_from_defaults }
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it { include(CacheableAttributes) }
|
2019-05-18 00:54:41 +05:30
|
|
|
it { include(ApplicationSettingImplementation) }
|
2018-11-08 19:23:39 +05:30
|
|
|
it { expect(described_class.current_without_cache).to eq(described_class.last) }
|
|
|
|
|
2015-11-26 14:37:03 +05:30
|
|
|
it { expect(setting).to be_valid }
|
2017-08-17 22:00:37 +05:30
|
|
|
it { expect(setting.uuid).to be_present }
|
2018-03-17 18:26:18 +05:30
|
|
|
it { expect(setting).to have_db_column(:auto_devops_enabled) }
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
describe 'validations' do
|
|
|
|
let(:http) { 'http://example.com' }
|
|
|
|
let(:https) { 'https://example.com' }
|
|
|
|
let(:ftp) { 'ftp://example.com' }
|
|
|
|
|
|
|
|
it { is_expected.to allow_value(nil).for(:home_page_url) }
|
|
|
|
it { is_expected.to allow_value(http).for(:home_page_url) }
|
|
|
|
it { is_expected.to allow_value(https).for(:home_page_url) }
|
|
|
|
it { is_expected.not_to allow_value(ftp).for(:home_page_url) }
|
|
|
|
|
|
|
|
it { is_expected.to allow_value(nil).for(:after_sign_out_path) }
|
|
|
|
it { is_expected.to allow_value(http).for(:after_sign_out_path) }
|
|
|
|
it { is_expected.to allow_value(https).for(:after_sign_out_path) }
|
|
|
|
it { is_expected.not_to allow_value(ftp).for(:after_sign_out_path) }
|
2016-04-02 18:10:28 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
it { is_expected.to allow_value("dev.gitlab.com").for(:commit_email_hostname) }
|
|
|
|
it { is_expected.not_to allow_value("@dev.gitlab").for(:commit_email_hostname) }
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
describe 'default_artifacts_expire_in' do
|
|
|
|
it 'sets an error if it cannot parse' do
|
|
|
|
setting.update(default_artifacts_expire_in: 'a')
|
|
|
|
|
|
|
|
expect_invalid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sets an error if it is blank' do
|
|
|
|
setting.update(default_artifacts_expire_in: ' ')
|
|
|
|
|
|
|
|
expect_invalid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sets the value if it is valid' do
|
|
|
|
setting.update(default_artifacts_expire_in: '30 days')
|
|
|
|
|
|
|
|
expect(setting).to be_valid
|
|
|
|
expect(setting.default_artifacts_expire_in).to eq('30 days')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'sets the value if it is 0' do
|
|
|
|
setting.update(default_artifacts_expire_in: '0')
|
|
|
|
|
|
|
|
expect(setting).to be_valid
|
|
|
|
expect(setting.default_artifacts_expire_in).to eq('0')
|
|
|
|
end
|
|
|
|
|
|
|
|
def expect_invalid
|
|
|
|
expect(setting).to be_invalid
|
|
|
|
expect(setting.errors.messages)
|
|
|
|
.to have_key(:default_artifacts_expire_in)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-04-02 18:10:28 +05:30
|
|
|
it { is_expected.to validate_presence_of(:max_attachment_size) }
|
|
|
|
|
|
|
|
it do
|
|
|
|
is_expected.to validate_numericality_of(:max_attachment_size)
|
|
|
|
.only_integer
|
|
|
|
.is_greater_than(0)
|
|
|
|
end
|
|
|
|
|
2019-03-02 22:35:43 +05:30
|
|
|
it do
|
|
|
|
is_expected.to validate_numericality_of(:local_markdown_version)
|
|
|
|
.only_integer
|
|
|
|
.is_greater_than_or_equal_to(0)
|
|
|
|
.is_less_than(65536)
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'key restrictions' do
|
|
|
|
it 'supports all key types' do
|
|
|
|
expect(described_class::SUPPORTED_KEY_TYPES).to contain_exactly(:rsa, :dsa, :ecdsa, :ed25519)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not allow all key types to be disabled' do
|
|
|
|
described_class::SUPPORTED_KEY_TYPES.each do |type|
|
|
|
|
setting["#{type}_key_restriction"] = described_class::FORBIDDEN_KEY_VALUE
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(setting).not_to be_valid
|
|
|
|
expect(setting.errors.messages).to have_key(:allowed_key_types)
|
|
|
|
end
|
|
|
|
|
|
|
|
where(:type) do
|
|
|
|
described_class::SUPPORTED_KEY_TYPES
|
|
|
|
end
|
|
|
|
|
|
|
|
with_them do
|
|
|
|
let(:field) { :"#{type}_key_restriction" }
|
|
|
|
|
|
|
|
it { is_expected.to validate_presence_of(field) }
|
|
|
|
it { is_expected.to allow_value(*KeyRestrictionValidator.supported_key_restrictions(type)).for(field) }
|
|
|
|
it { is_expected.not_to allow_value(128).for(field) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-04-02 18:10:28 +05:30
|
|
|
it_behaves_like 'an object with email-formated attributes', :admin_notification_email do
|
|
|
|
subject { setting }
|
|
|
|
end
|
2016-08-24 12:49:21 +05:30
|
|
|
|
2016-11-24 13:41:30 +05:30
|
|
|
# Upgraded databases will have this sort of content
|
|
|
|
context 'repository_storages is a String, not an Array' do
|
2017-09-10 17:25:29 +05:30
|
|
|
before do
|
2018-11-08 19:23:39 +05:30
|
|
|
described_class.where(id: setting.id).update_all(repository_storages: 'default')
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
2016-11-24 13:41:30 +05:30
|
|
|
|
|
|
|
it { expect(setting.repository_storages).to eq(['default']) }
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
context 'auto_devops_domain setting' do
|
|
|
|
context 'when auto_devops_enabled? is true' do
|
|
|
|
before do
|
|
|
|
setting.update(auto_devops_enabled: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'can be blank' do
|
|
|
|
setting.update(auto_devops_domain: '')
|
|
|
|
|
|
|
|
expect(setting).to be_valid
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with a valid value' do
|
|
|
|
before do
|
|
|
|
setting.update(auto_devops_domain: 'domain.com')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is valid' do
|
|
|
|
expect(setting).to be_valid
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with an invalid value' do
|
|
|
|
before do
|
|
|
|
setting.update(auto_devops_domain: 'definitelynotahostname')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is invalid' do
|
|
|
|
expect(setting).to be_invalid
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-11-24 13:41:30 +05:30
|
|
|
context 'repository storages' do
|
2016-08-24 12:49:21 +05:30
|
|
|
before do
|
2016-11-24 13:41:30 +05:30
|
|
|
storages = {
|
|
|
|
'custom1' => 'tmp/tests/custom_repositories_1',
|
|
|
|
'custom2' => 'tmp/tests/custom_repositories_2',
|
2017-09-10 17:25:29 +05:30
|
|
|
'custom3' => 'tmp/tests/custom_repositories_3'
|
2016-11-24 13:41:30 +05:30
|
|
|
|
|
|
|
}
|
2016-08-24 12:49:21 +05:30
|
|
|
allow(Gitlab.config.repositories).to receive(:storages).and_return(storages)
|
|
|
|
end
|
|
|
|
|
2016-11-24 13:41:30 +05:30
|
|
|
describe 'inclusion' do
|
|
|
|
it { is_expected.to allow_value('custom1').for(:repository_storages) }
|
2017-08-17 22:00:37 +05:30
|
|
|
it { is_expected.to allow_value(%w(custom2 custom3)).for(:repository_storages) }
|
2016-11-24 13:41:30 +05:30
|
|
|
it { is_expected.not_to allow_value('alternative').for(:repository_storages) }
|
2017-08-17 22:00:37 +05:30
|
|
|
it { is_expected.not_to allow_value(%w(alternative custom1)).for(:repository_storages) }
|
2016-11-24 13:41:30 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe 'presence' do
|
|
|
|
it { is_expected.not_to allow_value([]).for(:repository_storages) }
|
|
|
|
it { is_expected.not_to allow_value("").for(:repository_storages) }
|
|
|
|
it { is_expected.not_to allow_value(nil).for(:repository_storages) }
|
|
|
|
end
|
2016-08-24 12:49:21 +05:30
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
context 'housekeeping settings' do
|
|
|
|
it { is_expected.not_to allow_value(0).for(:housekeeping_incremental_repack_period) }
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'wants the full repack period to be at least the incremental repack period' do
|
2017-08-17 22:00:37 +05:30
|
|
|
subject.housekeeping_incremental_repack_period = 2
|
|
|
|
subject.housekeeping_full_repack_period = 1
|
|
|
|
|
|
|
|
expect(subject).not_to be_valid
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
it 'wants the gc period to be at least the full repack period' do
|
|
|
|
subject.housekeeping_full_repack_period = 100
|
|
|
|
subject.housekeeping_gc_period = 90
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
expect(subject).not_to be_valid
|
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
it 'allows the same period for incremental repack and full repack, effectively skipping incremental repack' do
|
|
|
|
subject.housekeeping_incremental_repack_period = 2
|
|
|
|
subject.housekeeping_full_repack_period = 2
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'allows the same period for full repack and gc, effectively skipping full repack' do
|
|
|
|
subject.housekeeping_full_repack_period = 100
|
|
|
|
subject.housekeeping_gc_period = 100
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'gitaly timeouts' do
|
|
|
|
[:gitaly_timeout_default, :gitaly_timeout_medium, :gitaly_timeout_fast].each do |timeout_name|
|
|
|
|
it do
|
|
|
|
is_expected.to validate_presence_of(timeout_name)
|
|
|
|
is_expected.to validate_numericality_of(timeout_name).only_integer
|
|
|
|
.is_greater_than_or_equal_to(0)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
[:gitaly_timeout_medium, :gitaly_timeout_fast].each do |timeout_name|
|
|
|
|
it "validates that #{timeout_name} is lower than timeout_default" do
|
|
|
|
subject[:gitaly_timeout_default] = 50
|
|
|
|
subject[timeout_name] = 100
|
|
|
|
|
|
|
|
expect(subject).to be_invalid
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts all timeouts equal' do
|
|
|
|
subject.gitaly_timeout_default = 0
|
|
|
|
subject.gitaly_timeout_medium = 0
|
|
|
|
subject.gitaly_timeout_fast = 0
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'accepts timeouts in descending order' do
|
|
|
|
subject.gitaly_timeout_default = 50
|
|
|
|
subject.gitaly_timeout_medium = 30
|
|
|
|
subject.gitaly_timeout_fast = 20
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'rejects timeouts in ascending order' do
|
|
|
|
subject.gitaly_timeout_default = 20
|
|
|
|
subject.gitaly_timeout_medium = 30
|
|
|
|
subject.gitaly_timeout_fast = 50
|
|
|
|
|
|
|
|
expect(subject).to be_invalid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'rejects medium timeout larger than default' do
|
|
|
|
subject.gitaly_timeout_default = 30
|
|
|
|
subject.gitaly_timeout_medium = 50
|
|
|
|
subject.gitaly_timeout_fast = 20
|
|
|
|
|
|
|
|
expect(subject).to be_invalid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'rejects medium timeout smaller than fast' do
|
|
|
|
subject.gitaly_timeout_default = 30
|
|
|
|
subject.gitaly_timeout_medium = 15
|
|
|
|
subject.gitaly_timeout_fast = 20
|
|
|
|
|
|
|
|
expect(subject).to be_invalid
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
2018-10-15 14:42:47 +05:30
|
|
|
|
|
|
|
describe 'enforcing terms' do
|
|
|
|
it 'requires the terms to present when enforcing users to accept' do
|
|
|
|
subject.enforce_terms = true
|
|
|
|
|
|
|
|
expect(subject).to be_invalid
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'is valid when terms are created' do
|
|
|
|
create(:term)
|
|
|
|
subject.enforce_terms = true
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
end
|
|
|
|
end
|
2019-05-18 00:54:41 +05:30
|
|
|
|
|
|
|
describe 'when external authorization service is enabled' do
|
|
|
|
before do
|
|
|
|
setting.external_authorization_service_enabled = true
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.not_to allow_value('not a URL').for(:external_authorization_service_url) }
|
|
|
|
it { is_expected.to allow_value('https://example.com').for(:external_authorization_service_url) }
|
|
|
|
it { is_expected.to allow_value('').for(:external_authorization_service_url) }
|
|
|
|
it { is_expected.not_to allow_value(nil).for(:external_authorization_service_default_label) }
|
|
|
|
it { is_expected.not_to allow_value(11).for(:external_authorization_service_timeout) }
|
|
|
|
it { is_expected.not_to allow_value(0).for(:external_authorization_service_timeout) }
|
|
|
|
it { is_expected.not_to allow_value('not a certificate').for(:external_auth_client_cert) }
|
|
|
|
it { is_expected.to allow_value('').for(:external_auth_client_cert) }
|
|
|
|
it { is_expected.to allow_value('').for(:external_auth_client_key) }
|
|
|
|
|
|
|
|
context 'when setting a valid client certificate for external authorization' do
|
|
|
|
let(:certificate_data) { File.read('spec/fixtures/passphrase_x509_certificate.crt') }
|
|
|
|
|
|
|
|
before do
|
|
|
|
setting.external_auth_client_cert = certificate_data
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'requires a valid client key when a certificate is set' do
|
|
|
|
expect(setting).not_to allow_value('fefefe').for(:external_auth_client_key)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'requires a matching certificate' do
|
|
|
|
other_private_key = File.read('spec/fixtures/x509_certificate_pk.key')
|
|
|
|
|
|
|
|
expect(setting).not_to allow_value(other_private_key).for(:external_auth_client_key)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'the credentials are valid when the private key can be read and matches the certificate' do
|
|
|
|
tls_attributes = [:external_auth_client_key_pass,
|
|
|
|
:external_auth_client_key,
|
|
|
|
:external_auth_client_cert]
|
|
|
|
setting.external_auth_client_key = File.read('spec/fixtures/passphrase_x509_certificate_pk.key')
|
|
|
|
setting.external_auth_client_key_pass = '5iveL!fe'
|
|
|
|
|
|
|
|
setting.validate
|
|
|
|
|
|
|
|
expect(setting.errors).not_to include(*tls_attributes)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
context 'restrict creating duplicates' do
|
2019-05-18 00:54:41 +05:30
|
|
|
let!(:current_settings) { described_class.create_from_defaults }
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2019-05-18 00:54:41 +05:30
|
|
|
it 'returns the current settings' do
|
|
|
|
expect(described_class.create_from_defaults).to eq(current_settings)
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
2018-11-08 19:23:39 +05:30
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
describe 'setting Sentry DSNs' do
|
|
|
|
context 'server DSN' do
|
|
|
|
it 'strips leading and trailing whitespace' do
|
|
|
|
subject.update(sentry_dsn: ' http://test ')
|
|
|
|
|
|
|
|
expect(subject.sentry_dsn).to eq('http://test')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'handles nil values' do
|
|
|
|
subject.update(sentry_dsn: nil)
|
|
|
|
|
|
|
|
expect(subject.sentry_dsn).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'client-side DSN' do
|
|
|
|
it 'strips leading and trailing whitespace' do
|
|
|
|
subject.update(clientside_sentry_dsn: ' http://test ')
|
|
|
|
|
|
|
|
expect(subject.clientside_sentry_dsn).to eq('http://test')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'handles nil values' do
|
|
|
|
subject.update(clientside_sentry_dsn: nil)
|
|
|
|
|
|
|
|
expect(subject.clientside_sentry_dsn).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
describe '#disabled_oauth_sign_in_sources=' do
|
|
|
|
before do
|
|
|
|
allow(Devise).to receive(:omniauth_providers).and_return([:github])
|
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it 'removes unknown sources (as strings) from the array' do
|
|
|
|
subject.disabled_oauth_sign_in_sources = %w[github test]
|
2018-03-17 18:26:18 +05:30
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
expect(subject).to be_valid
|
|
|
|
expect(subject.disabled_oauth_sign_in_sources).to eq ['github']
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it 'removes unknown sources (as symbols) from the array' do
|
|
|
|
subject.disabled_oauth_sign_in_sources = %i[github test]
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
expect(subject.disabled_oauth_sign_in_sources).to eq ['github']
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
|
|
|
|
2018-11-08 19:23:39 +05:30
|
|
|
it 'ignores nil' do
|
|
|
|
subject.disabled_oauth_sign_in_sources = nil
|
|
|
|
|
|
|
|
expect(subject).to be_valid
|
|
|
|
expect(subject.disabled_oauth_sign_in_sources).to be_empty
|
2018-03-17 18:26:18 +05:30
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe 'performance bar settings' do
|
|
|
|
describe 'performance_bar_allowed_group' do
|
|
|
|
context 'with no performance_bar_allowed_group_id saved' do
|
|
|
|
it 'returns nil' do
|
|
|
|
expect(setting.performance_bar_allowed_group).to be_nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with a performance_bar_allowed_group_id saved' do
|
|
|
|
let(:group) { create(:group) }
|
|
|
|
|
|
|
|
before do
|
2018-11-08 19:23:39 +05:30
|
|
|
setting.update!(performance_bar_allowed_group_id: group.id)
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns the group' do
|
2018-11-08 19:23:39 +05:30
|
|
|
expect(setting.reload.performance_bar_allowed_group).to eq(group)
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'performance_bar_enabled' do
|
|
|
|
context 'with the Performance Bar is enabled' do
|
|
|
|
let(:group) { create(:group) }
|
|
|
|
|
|
|
|
before do
|
2018-11-08 19:23:39 +05:30
|
|
|
setting.update!(performance_bar_allowed_group_id: group.id)
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns true' do
|
2018-11-08 19:23:39 +05:30
|
|
|
expect(setting.reload.performance_bar_enabled).to be_truthy
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
context 'diff limit settings' do
|
|
|
|
describe '#diff_max_patch_bytes' do
|
|
|
|
context 'validations' do
|
|
|
|
it { is_expected.to validate_presence_of(:diff_max_patch_bytes) }
|
|
|
|
|
|
|
|
it do
|
|
|
|
is_expected.to validate_numericality_of(:diff_max_patch_bytes)
|
|
|
|
.only_integer
|
|
|
|
.is_greater_than_or_equal_to(Gitlab::Git::Diff::DEFAULT_MAX_PATCH_BYTES)
|
|
|
|
.is_less_than_or_equal_to(Gitlab::Git::Diff::MAX_PATCH_BYTES_UPPER_BOUND)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2018-12-13 13:39:08 +05:30
|
|
|
|
2019-05-18 00:54:41 +05:30
|
|
|
it_behaves_like 'application settings examples'
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|