2015-12-07 16:17:12 -05:00
|
|
|
# NamespaceValidator
|
|
|
|
#
|
|
|
|
# Custom validator for GitLab namespace values.
|
|
|
|
#
|
2015-12-07 16:29:39 -05:00
|
|
|
# Values are checked for formatting and exclusion from a list of reserved path
|
|
|
|
# names.
|
2015-12-07 16:17:12 -05:00
|
|
|
class NamespaceValidator < ActiveModel::EachValidator
|
2017-04-05 09:41:00 -04:00
|
|
|
# All routes that appear on the top level must be listed here.
|
|
|
|
# This will make sure that groups cannot be created with these names
|
|
|
|
# as these routes would be masked by the paths already in place.
|
|
|
|
#
|
|
|
|
# Example:
|
|
|
|
# /api/api-project
|
|
|
|
#
|
|
|
|
# the path `api` shouldn't be allowed because it would be masked by `api/*`
|
|
|
|
#
|
|
|
|
TOP_LEVEL_ROUTES = Set.new(%w[
|
2016-09-29 04:36:38 -04:00
|
|
|
.well-known
|
2015-12-07 16:29:39 -05:00
|
|
|
admin
|
|
|
|
all
|
|
|
|
assets
|
|
|
|
ci
|
|
|
|
dashboard
|
|
|
|
files
|
|
|
|
groups
|
|
|
|
help
|
|
|
|
hooks
|
|
|
|
issues
|
|
|
|
merge_requests
|
2016-01-08 04:19:22 -05:00
|
|
|
new
|
2015-12-07 16:29:39 -05:00
|
|
|
notes
|
|
|
|
profile
|
|
|
|
projects
|
|
|
|
public
|
|
|
|
repository
|
2016-10-10 21:58:26 -04:00
|
|
|
robots.txt
|
2015-12-07 16:29:39 -05:00
|
|
|
s
|
|
|
|
search
|
|
|
|
services
|
|
|
|
snippets
|
|
|
|
teams
|
|
|
|
u
|
|
|
|
unsubscribes
|
|
|
|
users
|
2017-04-03 08:07:15 -04:00
|
|
|
api
|
|
|
|
autocomplete
|
|
|
|
search
|
|
|
|
member
|
|
|
|
explore
|
|
|
|
uploads
|
|
|
|
import
|
|
|
|
notification_settings
|
|
|
|
abuse_reports
|
|
|
|
invites
|
|
|
|
help
|
|
|
|
koding
|
|
|
|
health_check
|
|
|
|
jwt
|
|
|
|
oauth
|
|
|
|
sent_notifications
|
2017-04-05 09:41:00 -04:00
|
|
|
]).freeze
|
2015-12-07 16:29:39 -05:00
|
|
|
|
2017-04-05 09:41:00 -04:00
|
|
|
# All project routes with wildcard argument must be listed here.
|
|
|
|
# Otherwise it can lead to routing issues when route considered as project name.
|
|
|
|
#
|
|
|
|
# Example:
|
|
|
|
# /group/project/tree/deploy_keys
|
|
|
|
#
|
|
|
|
# without tree as reserved name routing can match 'group/project' as group name,
|
|
|
|
# 'tree' as project name and 'deploy_keys' as route.
|
|
|
|
#
|
2017-04-10 13:27:19 -04:00
|
|
|
|
2017-04-05 09:41:00 -04:00
|
|
|
WILDCARD_ROUTES = Set.new(%w[tree commits wikis new edit create update logs_tree
|
|
|
|
preview blob blame raw files create_dir find_file
|
2017-04-10 13:27:19 -04:00
|
|
|
artifacts graphs refs badges objects folders file])
|
2017-03-06 13:26:58 -05:00
|
|
|
|
2017-04-05 09:41:00 -04:00
|
|
|
STRICT_RESERVED = (TOP_LEVEL_ROUTES | WILDCARD_ROUTES)
|
2017-03-06 13:26:58 -05:00
|
|
|
|
2017-04-05 09:41:00 -04:00
|
|
|
def self.valid_full_path?(full_path)
|
2017-04-11 09:51:33 -04:00
|
|
|
path_segments = full_path.split('/')
|
|
|
|
root_segment = path_segments.shift
|
|
|
|
|
|
|
|
valid?(root_segment, type: :top_level) && valid_wildcard_segments?(path_segments)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.valid_wildcard_segments?(segments)
|
|
|
|
segments.all? { |segment| valid?(segment, type: :wildcard) }
|
2016-11-14 09:55:31 -05:00
|
|
|
end
|
|
|
|
|
2017-04-05 09:41:00 -04:00
|
|
|
def self.valid?(value, type: :strict)
|
|
|
|
!reserved?(value, type: type) && follow_format?(value)
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.reserved?(value, type: :strict)
|
|
|
|
case type
|
|
|
|
when :wildcard
|
|
|
|
WILDCARD_ROUTES.include?(value)
|
|
|
|
when :top_level
|
|
|
|
TOP_LEVEL_ROUTES.include?(value)
|
2017-03-06 13:26:58 -05:00
|
|
|
else
|
2017-04-05 09:41:00 -04:00
|
|
|
STRICT_RESERVED.include?(value)
|
2017-03-06 13:26:58 -05:00
|
|
|
end
|
2016-11-14 09:55:31 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.follow_format?(value)
|
|
|
|
value =~ Gitlab::Regex.namespace_regex
|
|
|
|
end
|
|
|
|
|
|
|
|
delegate :reserved?, :follow_format?, to: :class
|
|
|
|
|
2015-12-07 16:17:12 -05:00
|
|
|
def validate_each(record, attribute, value)
|
2016-11-14 09:55:31 -05:00
|
|
|
unless follow_format?(value)
|
2015-12-07 16:17:12 -05:00
|
|
|
record.errors.add(attribute, Gitlab::Regex.namespace_regex_message)
|
|
|
|
end
|
|
|
|
|
2017-04-05 09:41:00 -04:00
|
|
|
if reserved?(value, type: validation_type(record))
|
2015-12-07 16:17:12 -05:00
|
|
|
record.errors.add(attribute, "#{value} is a reserved name")
|
|
|
|
end
|
|
|
|
end
|
2017-04-05 09:41:00 -04:00
|
|
|
|
|
|
|
def validation_type(record)
|
|
|
|
case record
|
|
|
|
when Group
|
|
|
|
record.parent_id ? :wildcard : :top_level
|
|
|
|
when Project
|
|
|
|
:wildcard
|
|
|
|
else
|
|
|
|
:strict
|
|
|
|
end
|
|
|
|
end
|
2015-12-07 16:17:12 -05:00
|
|
|
end
|