Commit 5a0c0e15 authored by Nick Thomas's avatar Nick Thomas

Address review comments

parent d40a7d41
module FormHelper module FormHelper
prepend ::EE::FormHelper prepend ::EE::FormHelper
def form_errors(model, headline = 'The form contains the following') def form_errors(model, type: 'form')
return unless model.errors.any? return unless model.errors.any?
pluralized = 'error'.pluralize(model.errors.count) pluralized = 'error'.pluralize(model.errors.count)
headline = "The #{type} contains the following #{pluralized}:"
headline = headline + ' ' + pluralized + ':'
content_tag(:div, class: 'alert alert-danger', id: 'error_explanation') do content_tag(:div, class: 'alert alert-danger', id: 'error_explanation') do
content_tag(:h4, headline) << content_tag(:h4, headline) <<
......
...@@ -16,13 +16,9 @@ class ApplicationSetting < ActiveRecord::Base ...@@ -16,13 +16,9 @@ class ApplicationSetting < ActiveRecord::Base
# Setting a key restriction to `-1` means that all keys of this type are # Setting a key restriction to `-1` means that all keys of this type are
# forbidden. # forbidden.
FORBIDDEN_KEY_VALUE = -1 FORBIDDEN_KEY_VALUE = KeyRestrictionValidator::FORBIDDEN
SUPPORTED_KEY_TYPES = %i[rsa dsa ecdsa ed25519].freeze SUPPORTED_KEY_TYPES = %i[rsa dsa ecdsa ed25519].freeze
def self.supported_key_restrictions(type)
[0, *Gitlab::SSHPublicKey.supported_sizes(type), FORBIDDEN_KEY_VALUE]
end
serialize :restricted_visibility_levels # rubocop:disable Cop/ActiveRecordSerialize serialize :restricted_visibility_levels # rubocop:disable Cop/ActiveRecordSerialize
serialize :import_sources # rubocop:disable Cop/ActiveRecordSerialize serialize :import_sources # rubocop:disable Cop/ActiveRecordSerialize
serialize :disabled_oauth_sign_in_sources, Array # rubocop:disable Cop/ActiveRecordSerialize serialize :disabled_oauth_sign_in_sources, Array # rubocop:disable Cop/ActiveRecordSerialize
...@@ -169,11 +165,11 @@ class ApplicationSetting < ActiveRecord::Base ...@@ -169,11 +165,11 @@ class ApplicationSetting < ActiveRecord::Base
numericality: { greater_than_or_equal_to: 0 } numericality: { greater_than_or_equal_to: 0 }
SUPPORTED_KEY_TYPES.each do |type| SUPPORTED_KEY_TYPES.each do |type|
validates :"#{type}_key_restriction", validates :"#{type}_key_restriction", presence: true, key_restriction: { type: type }
presence: true,
inclusion: { in: ApplicationSetting.supported_key_restrictions(type) }
end end
validates :allowed_key_types, presence: true
validates_each :restricted_visibility_levels do |record, attr, value| validates_each :restricted_visibility_levels do |record, attr, value|
value&.each do |level| value&.each do |level|
unless Gitlab::VisibilityLevel.options.value?(level) unless Gitlab::VisibilityLevel.options.value?(level)
...@@ -489,8 +485,7 @@ class ApplicationSetting < ActiveRecord::Base ...@@ -489,8 +485,7 @@ class ApplicationSetting < ActiveRecord::Base
def key_restriction_for(type) def key_restriction_for(type)
attr_name = "#{type}_key_restriction" attr_name = "#{type}_key_restriction"
# rubocop:disable GitlabSecurity/PublicSend has_attribute?(attr_name) ? public_send(attr_name) : FORBIDDEN_KEY_VALUE # rubocop:disable GitlabSecurity/PublicSend
has_attribute?(attr_name) ? public_send(attr_name) : FORBIDDEN_KEY_VALUE
end end
private private
......
require 'digest/md5' require 'digest/md5'
class Key < ActiveRecord::Base class Key < ActiveRecord::Base
include AfterCommitQueue
include Gitlab::CurrentSettings include Gitlab::CurrentSettings
include Sortable include Sortable
......
class KeyRestrictionValidator < ActiveModel::EachValidator
FORBIDDEN = -1
def self.supported_sizes(type)
Gitlab::SSHPublicKey.supported_sizes(type)
end
def self.supported_key_restrictions(type)
[0, *supported_sizes(type), FORBIDDEN]
end
def validate_each(record, attribute, value)
unless valid_restriction?(value)
record.errors.add(attribute, "must be forbidden, allowed, or one of these sizes: #{supported_sizes_message}")
end
end
private
def supported_sizes_message
sizes = self.class.supported_sizes(options[:type])
sizes.to_sentence(last_word_connector: ', or ', two_words_connector: ' or ')
end
def valid_restriction?(value)
choices = self.class.supported_key_restrictions(options[:type])
choices.include?(value)
end
end
...@@ -3,8 +3,8 @@ ...@@ -3,8 +3,8 @@
- if key.valid? - if key.valid?
= icon 'key', class: 'settings-list-icon hidden-xs' = icon 'key', class: 'settings-list-icon hidden-xs'
- else - else
= icon 'exclamation-triangle', class: 'settings-list-icon hidden-xs', = icon 'exclamation-triangle', class: 'settings-list-icon hidden-xs has-tooltip',
title: 'The key is disabled because it is invalid' title: key.errors.full_messages.join(', ')
.key-list-item-info .key-list-item-info
......
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
%strong= @key.last_used_at.try(:to_s, :medium) || 'N/A' %strong= @key.last_used_at.try(:to_s, :medium) || 'N/A'
.col-md-8 .col-md-8
= form_errors(@key, 'The key has the following') unless @key.valid? = form_errors(@key, type: 'key') unless @key.valid?
%p %p
%span.light Fingerprint: %span.light Fingerprint:
%code.key-fingerprint= @key.fingerprint %code.key-fingerprint= @key.fingerprint
......
--- ---
title: Add settings for minimum key strength and allowed key type title: Add settings for minimum SSH key strength and allowed key type
merge_request: 13712 merge_request: 13712
author: Cory Hinshaw author: Cory Hinshaw
type: added type: added
...@@ -7,12 +7,13 @@ class AddMinimumKeyLengthToApplicationSettings < ActiveRecord::Migration ...@@ -7,12 +7,13 @@ class AddMinimumKeyLengthToApplicationSettings < ActiveRecord::Migration
disable_ddl_transaction! disable_ddl_transaction!
def up def up
# A key restriction has two possible states: # A key restriction has these possible states:
# #
# * -1 means "this key type is completely disabled" # * -1 means "this key type is completely disabled"
# * >= 0 means "keys must have at least this many bits to be valid" # * 0 means "all keys of this type are valid"
# * > 0 means "keys must have at least this many bits to be valid"
# #
# A value of 0 is equivalent to "there are no restrictions on keys of this type" # The default is 0, for backward compatibility
add_column_with_default :application_settings, :rsa_key_restriction, :integer, default: 0 add_column_with_default :application_settings, :rsa_key_restriction, :integer, default: 0
add_column_with_default :application_settings, :dsa_key_restriction, :integer, default: 0 add_column_with_default :application_settings, :dsa_key_restriction, :integer, default: 0
add_column_with_default :application_settings, :ecdsa_key_restriction, :integer, default: 0 add_column_with_default :application_settings, :ecdsa_key_restriction, :integer, default: 0
......
# Security # Security
- [Password length limits](password_length_limits.md) - [Password length limits](password_length_limits.md)
- [Restrict allowed SSH key technologies and minimum length](ssh_keys_restrictions.md) - [Restrict SSH key technologies and minimum length](ssh_keys_restrictions.md)
- [Rack attack](rack_attack.md) - [Rack attack](rack_attack.md)
- [Webhooks and insecure internal web services](webhooks.md) - [Webhooks and insecure internal web services](webhooks.md)
- [Information exclusivity](information_exclusivity.md) - [Information exclusivity](information_exclusivity.md)
......
...@@ -2,12 +2,13 @@ ...@@ -2,12 +2,13 @@
`ssh-keygen` allows users to create RSA keys with as few as 768 bits, which `ssh-keygen` allows users to create RSA keys with as few as 768 bits, which
falls well below recommendations from certain standards groups (such as the US falls well below recommendations from certain standards groups (such as the US
NIST). Some organizations deploying Gitlab will need to enforce minimum key NIST). Some organizations deploying GitLab will need to enforce minimum key
strength, either to satisfy internal security policy or for regulatory strength, either to satisfy internal security policy or for regulatory
compliance. compliance.
Similarly, certain standards groups recommend using RSA or ECDSA over the older Similarly, certain standards groups recommend using RSA, ECDSA, or ED25519 over
DSA and administrators may need to limit the allowed SSH key algorithms. the older DSA, and administrators may need to limit the allowed SSH key
algorithms.
GitLab allows you to restrict the allowed SSH key technology as well as specify GitLab allows you to restrict the allowed SSH key technology as well as specify
the minimum key length for each technology. the minimum key length for each technology.
......
...@@ -125,7 +125,7 @@ module API ...@@ -125,7 +125,7 @@ module API
ApplicationSetting::SUPPORTED_KEY_TYPES.each do |type| ApplicationSetting::SUPPORTED_KEY_TYPES.each do |type|
optional :"#{type}_key_restriction", optional :"#{type}_key_restriction",
type: Integer, type: Integer,
values: ApplicationSetting.supported_key_restrictions(type), values: KeyRestrictionValidator.supported_key_restrictions(type),
desc: "Restrictions on the complexity of uploaded #{type.upcase} keys. A value of #{ApplicationSetting::FORBIDDEN_KEY_VALUE} disables all #{type.upcase} keys." desc: "Restrictions on the complexity of uploaded #{type.upcase} keys. A value of #{ApplicationSetting::FORBIDDEN_KEY_VALUE} disables all #{type.upcase} keys."
end end
......
...@@ -37,8 +37,8 @@ module Gitlab ...@@ -37,8 +37,8 @@ module Gitlab
end end
def check(cmd, changes) def check(cmd, changes)
check_valid_actor!
check_protocol! check_protocol!
check_valid_actor!
check_active_user! check_active_user!
check_project_accessibility! check_project_accessibility!
check_project_moved! check_project_moved!
......
...@@ -13,6 +13,10 @@ module Gitlab ...@@ -13,6 +13,10 @@ module Gitlab
Technologies.find { |tech| tech.name.to_s == name.to_s } Technologies.find { |tech| tech.name.to_s == name.to_s }
end end
def self.technology_for_key(key)
Technologies.find { |tech| key.is_a?(tech.key_class) }
end
def self.supported_sizes(name) def self.supported_sizes(name)
technology(name)&.supported_sizes technology(name)&.supported_sizes
end end
...@@ -37,9 +41,7 @@ module Gitlab ...@@ -37,9 +41,7 @@ module Gitlab
end end
def type def type
return unless valid? technology.name if valid?
technology.name
end end
def bits def bits
...@@ -63,12 +65,7 @@ module Gitlab ...@@ -63,12 +65,7 @@ module Gitlab
def technology def technology
@technology ||= @technology ||=
begin self.class.technology_for_key(key) || raise("Unsupported key type: #{key.class}")
tech = Technologies.find { |tech| key.is_a?(tech.key_class) }
raise "Unsupported key type: #{key.class}" unless tech
tech
end
end end
end end
end end
...@@ -165,12 +165,10 @@ describe Gitlab::GitAccess do ...@@ -165,12 +165,10 @@ describe Gitlab::GitAccess do
stub_application_setting(rsa_key_restriction: 4096) stub_application_setting(rsa_key_restriction: 4096)
end end
it 'does not allow keys which are too small' do it 'does not allow keys which are too small', aggregate_failures: true do
aggregate_failures do expect(actor).not_to be_valid
expect(actor).not_to be_valid expect { pull_access_check }.to raise_unauthorized('Your SSH key must be at least 4096 bits.')
expect { pull_access_check }.to raise_unauthorized('Your SSH key must be at least 4096 bits.') expect { push_access_check }.to raise_unauthorized('Your SSH key must be at least 4096 bits.')
expect { push_access_check }.to raise_unauthorized('Your SSH key must be at least 4096 bits.')
end
end end
end end
...@@ -179,12 +177,10 @@ describe Gitlab::GitAccess do ...@@ -179,12 +177,10 @@ describe Gitlab::GitAccess do
stub_application_setting(rsa_key_restriction: ApplicationSetting::FORBIDDEN_KEY_VALUE) stub_application_setting(rsa_key_restriction: ApplicationSetting::FORBIDDEN_KEY_VALUE)
end end
it 'does not allow keys which are too small' do it 'does not allow keys which are too small', aggregate_failures: true do
aggregate_failures do expect(actor).not_to be_valid
expect(actor).not_to be_valid expect { pull_access_check }.to raise_unauthorized(/Your SSH key type is forbidden/)
expect { pull_access_check }.to raise_unauthorized(/Your SSH key type is forbidden/) expect { push_access_check }.to raise_unauthorized(/Your SSH key type is forbidden/)
expect { push_access_check }.to raise_unauthorized(/Your SSH key type is forbidden/)
end
end end
end end
end end
......
...@@ -77,6 +77,15 @@ describe ApplicationSetting do ...@@ -77,6 +77,15 @@ describe ApplicationSetting do
expect(described_class::SUPPORTED_KEY_TYPES).to contain_exactly(:rsa, :dsa, :ecdsa, :ed25519) expect(described_class::SUPPORTED_KEY_TYPES).to contain_exactly(:rsa, :dsa, :ecdsa, :ed25519)
end 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 where(:type) do
described_class::SUPPORTED_KEY_TYPES described_class::SUPPORTED_KEY_TYPES
end end
...@@ -85,7 +94,7 @@ describe ApplicationSetting do ...@@ -85,7 +94,7 @@ describe ApplicationSetting do
let(:field) { :"#{type}_key_restriction" } let(:field) { :"#{type}_key_restriction" }
it { is_expected.to validate_presence_of(field) } it { is_expected.to validate_presence_of(field) }
it { is_expected.to allow_value(*described_class.supported_key_restrictions(type)).for(field) } it { is_expected.to allow_value(*KeyRestrictionValidator.supported_key_restrictions(type)).for(field) }
it { is_expected.not_to allow_value(128).for(field) } it { is_expected.not_to allow_value(128).for(field) }
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment