2018-08-03 13:22:24 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-10-31 07:00:53 -04:00
|
|
|
# Store object full path in separate table for easy lookup and uniq validation
|
2017-02-04 13:26:11 -05:00
|
|
|
# Object must have name and path db fields and respond to parent and parent_changed? methods.
|
2016-10-31 07:00:53 -04:00
|
|
|
module Routable
|
|
|
|
extend ActiveSupport::Concern
|
|
|
|
|
|
|
|
included do
|
2018-05-10 20:19:59 -04:00
|
|
|
# Remove `inverse_of: source` when upgraded to rails 5.2
|
|
|
|
# See https://github.com/rails/rails/pull/28808
|
|
|
|
has_one :route, as: :source, autosave: true, dependent: :destroy, inverse_of: :source # rubocop:disable Cop/ActiveRecordDependent
|
2017-06-08 11:16:27 -04:00
|
|
|
has_many :redirect_routes, as: :source, autosave: true, dependent: :destroy # rubocop:disable Cop/ActiveRecordDependent
|
2016-10-31 07:00:53 -04:00
|
|
|
|
2016-12-07 12:16:02 -05:00
|
|
|
validates :route, presence: true
|
2016-10-31 07:00:53 -04:00
|
|
|
|
2017-02-04 13:26:11 -05:00
|
|
|
scope :with_route, -> { includes(:route) }
|
|
|
|
|
2018-01-31 11:52:09 -05:00
|
|
|
after_validation :set_path_errors
|
|
|
|
|
2017-02-04 13:26:11 -05:00
|
|
|
before_validation do
|
|
|
|
if full_path_changed? || full_name_changed?
|
|
|
|
prepare_route
|
|
|
|
end
|
|
|
|
end
|
2016-10-31 07:00:53 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
class_methods do
|
|
|
|
# Finds a single object by full path match in routes table.
|
|
|
|
#
|
|
|
|
# Usage:
|
|
|
|
#
|
|
|
|
# Klass.find_by_full_path('gitlab-org/gitlab-ce')
|
|
|
|
#
|
|
|
|
# Returns a single object, or nil.
|
2017-05-01 16:46:30 -04:00
|
|
|
def find_by_full_path(path, follow_redirects: false)
|
2019-06-13 09:12:28 -04:00
|
|
|
order_sql = Arel.sql("(CASE WHEN routes.path = #{connection.quote(path)} THEN 0 ELSE 1 END)")
|
2017-05-01 16:46:30 -04:00
|
|
|
found = where_full_path_in([path]).reorder(order_sql).take
|
|
|
|
return found if found
|
2016-10-31 07:00:53 -04:00
|
|
|
|
2017-05-01 16:46:30 -04:00
|
|
|
if follow_redirects
|
2019-06-13 09:12:28 -04:00
|
|
|
joins(:redirect_routes).find_by("LOWER(redirect_routes.path) = LOWER(?)", path)
|
2017-05-01 16:46:30 -04:00
|
|
|
end
|
2016-10-31 07:00:53 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Builds a relation to find multiple objects by their full paths.
|
|
|
|
#
|
|
|
|
# Usage:
|
|
|
|
#
|
2016-12-09 10:27:11 -05:00
|
|
|
# Klass.where_full_path_in(%w{gitlab-org/gitlab-ce gitlab-org/gitlab-ee})
|
2016-10-31 07:00:53 -04:00
|
|
|
#
|
|
|
|
# Returns an ActiveRecord::Relation.
|
2016-12-09 10:27:11 -05:00
|
|
|
def where_full_path_in(paths)
|
2019-06-13 09:12:28 -04:00
|
|
|
return none if paths.empty?
|
2016-10-31 07:00:53 -04:00
|
|
|
|
2019-06-13 09:12:28 -04:00
|
|
|
wheres = paths.map do |path|
|
|
|
|
"(LOWER(routes.path) = LOWER(#{connection.quote(path)}))"
|
2016-10-31 07:00:53 -04:00
|
|
|
end
|
|
|
|
|
2019-06-13 09:12:28 -04:00
|
|
|
joins(:route).where(wheres.join(' OR '))
|
2016-10-31 07:00:53 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-02-04 13:26:11 -05:00
|
|
|
def full_name
|
2018-07-02 11:09:49 -04:00
|
|
|
route&.name || build_full_name
|
2017-02-04 13:26:11 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def full_path
|
2018-07-02 11:09:49 -04:00
|
|
|
route&.path || build_full_path
|
2017-06-29 17:53:32 -04:00
|
|
|
end
|
|
|
|
|
2017-10-03 06:34:24 -04:00
|
|
|
def full_path_components
|
|
|
|
full_path.split('/')
|
|
|
|
end
|
|
|
|
|
2017-06-16 06:11:33 -04:00
|
|
|
def build_full_path
|
|
|
|
if parent && path
|
|
|
|
parent.full_path + '/' + path
|
|
|
|
else
|
|
|
|
path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-04-23 11:48:26 -04:00
|
|
|
# Group would override this to check from association
|
|
|
|
def owned_by?(user)
|
|
|
|
owner == user
|
|
|
|
end
|
|
|
|
|
2017-04-24 01:32:50 -04:00
|
|
|
private
|
|
|
|
|
2018-01-31 11:52:09 -05:00
|
|
|
def set_path_errors
|
|
|
|
route_path_errors = self.errors.delete(:"route.path")
|
|
|
|
self.errors[:path].concat(route_path_errors) if route_path_errors
|
|
|
|
end
|
|
|
|
|
2017-02-04 13:26:11 -05:00
|
|
|
def full_name_changed?
|
|
|
|
name_changed? || parent_changed?
|
|
|
|
end
|
|
|
|
|
|
|
|
def full_path_changed?
|
|
|
|
path_changed? || parent_changed?
|
|
|
|
end
|
|
|
|
|
|
|
|
def build_full_name
|
|
|
|
if parent && name
|
|
|
|
parent.human_name + ' / ' + name
|
|
|
|
else
|
|
|
|
name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def prepare_route
|
2016-10-31 07:00:53 -04:00
|
|
|
route || build_route(source: self)
|
2017-02-04 13:26:11 -05:00
|
|
|
route.path = build_full_path
|
|
|
|
route.name = build_full_name
|
2016-10-31 07:00:53 -04:00
|
|
|
end
|
|
|
|
end
|