2012-11-22 13:34:16 -05:00
|
|
|
class Namespace < ActiveRecord::Base
|
2016-05-28 22:54:17 -04:00
|
|
|
acts_as_paranoid
|
|
|
|
|
2016-10-06 17:17:11 -04:00
|
|
|
include CacheMarkdownField
|
2015-02-05 17:20:55 -05:00
|
|
|
include Sortable
|
2013-03-21 16:11:08 -04:00
|
|
|
include Gitlab::ShellAdapter
|
2016-10-31 07:00:53 -04:00
|
|
|
include Routable
|
2013-03-21 16:11:08 -04:00
|
|
|
|
2016-10-06 17:17:11 -04:00
|
|
|
cache_markdown_field :description, pipeline: :description
|
|
|
|
|
2012-11-23 13:11:09 -05:00
|
|
|
has_many :projects, dependent: :destroy
|
2012-11-22 13:34:16 -05:00
|
|
|
belongs_to :owner, class_name: "User"
|
|
|
|
|
2016-10-31 07:00:53 -04:00
|
|
|
belongs_to :parent, class_name: "Namespace"
|
|
|
|
has_many :children, class_name: "Namespace", foreign_key: :parent_id
|
|
|
|
|
2013-09-26 07:49:22 -04:00
|
|
|
validates :owner, presence: true, unless: ->(n) { n.type == "Group" }
|
2015-02-03 00:15:44 -05:00
|
|
|
validates :name,
|
2015-12-07 16:17:24 -05:00
|
|
|
presence: true,
|
2016-12-07 12:15:26 -05:00
|
|
|
uniqueness: { scope: :parent_id },
|
2016-12-02 07:54:57 -05:00
|
|
|
length: { maximum: 255 },
|
|
|
|
namespace_name: true
|
2015-02-03 00:15:44 -05:00
|
|
|
|
2016-12-02 07:54:57 -05:00
|
|
|
validates :description, length: { maximum: 255 }
|
2015-02-03 00:15:44 -05:00
|
|
|
validates :path,
|
2015-12-07 16:17:12 -05:00
|
|
|
presence: true,
|
2016-12-02 07:54:57 -05:00
|
|
|
length: { maximum: 255 },
|
|
|
|
namespace: true
|
2012-11-22 13:34:16 -05:00
|
|
|
|
|
|
|
delegate :name, to: :owner, allow_nil: true, prefix: true
|
|
|
|
|
2013-03-21 16:11:08 -04:00
|
|
|
after_update :move_dir, if: :path_changed?
|
2016-11-21 08:33:58 -05:00
|
|
|
after_commit :refresh_access_of_projects_invited_groups, on: :update, if: -> { previous_changes.key?('share_with_group_lock') }
|
2016-06-22 17:04:51 -04:00
|
|
|
|
|
|
|
# Save the storage paths before the projects are destroyed to use them on after destroy
|
|
|
|
before_destroy(prepend: true) { @old_repository_storage_paths = repository_storage_paths }
|
2012-11-21 00:54:05 -05:00
|
|
|
after_destroy :rm_dir
|
2012-11-23 01:11:09 -05:00
|
|
|
|
2013-02-12 02:16:45 -05:00
|
|
|
scope :root, -> { where('type IS NULL') }
|
2012-11-22 23:24:09 -05:00
|
|
|
|
2015-03-24 09:53:30 -04:00
|
|
|
class << self
|
|
|
|
def by_path(path)
|
2015-12-14 21:53:52 -05:00
|
|
|
find_by('lower(path) = :value', value: path.downcase)
|
2015-03-24 09:53:30 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Case insensetive search for namespace by path or name
|
|
|
|
def find_by_path_or_name(path)
|
|
|
|
find_by("lower(path) = :path OR lower(name) = :path", path: path.downcase)
|
|
|
|
end
|
|
|
|
|
2016-03-04 06:06:25 -05:00
|
|
|
# Searches for namespaces matching the given query.
|
|
|
|
#
|
|
|
|
# This method uses ILIKE on PostgreSQL and LIKE on MySQL.
|
|
|
|
#
|
|
|
|
# query - The search query as a String
|
|
|
|
#
|
|
|
|
# Returns an ActiveRecord::Relation
|
2015-03-24 09:53:30 -04:00
|
|
|
def search(query)
|
2016-03-04 06:06:25 -05:00
|
|
|
t = arel_table
|
|
|
|
pattern = "%#{query}%"
|
|
|
|
|
|
|
|
where(t[:name].matches(pattern).or(t[:path].matches(pattern)))
|
2015-03-24 09:53:30 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def clean_path(path)
|
2015-04-17 06:03:54 -04:00
|
|
|
path = path.dup
|
2015-04-21 09:49:16 -04:00
|
|
|
# Get the email username by removing everything after an `@` sign.
|
2016-08-25 12:48:08 -04:00
|
|
|
path.gsub!(/@.*\z/, "")
|
2015-04-21 09:49:16 -04:00
|
|
|
# Remove everything that's not in the list of allowed characters.
|
2016-08-25 12:48:08 -04:00
|
|
|
path.gsub!(/[^a-zA-Z0-9_\-\.]/, "")
|
|
|
|
# Remove trailing violations ('.atom', '.git', or '.')
|
|
|
|
path.gsub!(/(\.atom|\.git|\.)*\z/, "")
|
|
|
|
# Remove leading violations ('-')
|
|
|
|
path.gsub!(/\A\-+/, "")
|
2015-03-24 09:53:30 -04:00
|
|
|
|
2015-04-21 06:00:08 -04:00
|
|
|
# Users with the great usernames of "." or ".." would end up with a blank username.
|
2015-06-03 08:57:12 -04:00
|
|
|
# Work around that by setting their username to "blank", followed by a counter.
|
2015-04-21 06:00:08 -04:00
|
|
|
path = "blank" if path.blank?
|
|
|
|
|
2015-03-24 09:53:30 -04:00
|
|
|
counter = 0
|
|
|
|
base = path
|
2015-04-21 06:00:08 -04:00
|
|
|
while Namespace.find_by_path_or_name(path)
|
2015-03-24 09:53:30 -04:00
|
|
|
counter += 1
|
|
|
|
path = "#{base}#{counter}"
|
|
|
|
end
|
|
|
|
|
|
|
|
path
|
|
|
|
end
|
2012-11-27 01:31:15 -05:00
|
|
|
end
|
|
|
|
|
2012-11-22 13:34:16 -05:00
|
|
|
def to_param
|
2016-10-31 07:00:53 -04:00
|
|
|
full_path
|
2012-11-22 13:34:16 -05:00
|
|
|
end
|
2012-11-22 23:11:09 -05:00
|
|
|
|
|
|
|
def human_name
|
|
|
|
owner_name
|
|
|
|
end
|
2012-11-23 01:11:09 -05:00
|
|
|
|
2012-11-24 15:11:46 -05:00
|
|
|
def move_dir
|
2016-05-09 15:41:48 -04:00
|
|
|
if any_project_has_container_registry_tags?
|
2016-05-12 14:03:04 -04:00
|
|
|
raise Exception.new('Namespace cannot be moved, because at least one project has tags in container registry')
|
2016-05-09 15:41:48 -04:00
|
|
|
end
|
|
|
|
|
2016-06-22 17:04:51 -04:00
|
|
|
# Move the namespace directory in all storages paths used by member projects
|
|
|
|
repository_storage_paths.each do |repository_storage_path|
|
|
|
|
# Ensure old directory exists before moving it
|
|
|
|
gitlab_shell.add_namespace(repository_storage_path, path_was)
|
|
|
|
|
|
|
|
unless gitlab_shell.mv_namespace(repository_storage_path, path_was, path)
|
2016-11-24 00:36:15 -05:00
|
|
|
Rails.logger.error "Exception moving path #{repository_storage_path} from #{path_was} to #{path}"
|
|
|
|
|
2016-06-22 17:04:51 -04:00
|
|
|
# if we cannot move namespace directory we should rollback
|
|
|
|
# db changes in order to prevent out of sync between db and fs
|
|
|
|
raise Exception.new('namespace directory cannot be moved')
|
2012-12-20 15:16:51 -05:00
|
|
|
end
|
2016-06-22 17:04:51 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
Gitlab::UploadsTransfer.new.rename_namespace(path_was, path)
|
|
|
|
|
|
|
|
# If repositories moved successfully we need to
|
|
|
|
# send update instructions to users.
|
|
|
|
# However we cannot allow rollback since we moved namespace dir
|
|
|
|
# So we basically we mute exceptions in next actions
|
|
|
|
begin
|
|
|
|
send_update_instructions
|
|
|
|
rescue
|
|
|
|
# Returning false does not rollback after_* transaction but gives
|
|
|
|
# us information about failing some of tasks
|
|
|
|
false
|
2012-11-27 01:31:15 -05:00
|
|
|
end
|
2012-11-24 15:11:46 -05:00
|
|
|
end
|
2012-11-21 00:54:05 -05:00
|
|
|
|
2016-05-09 15:41:48 -04:00
|
|
|
def any_project_has_container_registry_tags?
|
2016-05-13 09:45:57 -04:00
|
|
|
projects.any?(&:has_container_registry_tags?)
|
2016-05-09 15:41:48 -04:00
|
|
|
end
|
|
|
|
|
2012-12-20 15:16:51 -05:00
|
|
|
def send_update_instructions
|
2015-09-29 09:37:50 -04:00
|
|
|
projects.each do |project|
|
|
|
|
project.send_move_instructions("#{path_was}/#{project.path}")
|
|
|
|
end
|
2012-12-20 15:16:51 -05:00
|
|
|
end
|
2013-11-15 08:25:37 -05:00
|
|
|
|
|
|
|
def kind
|
|
|
|
type == 'Group' ? 'group' : 'user'
|
|
|
|
end
|
2014-11-14 09:06:39 -05:00
|
|
|
|
|
|
|
def find_fork_of(project)
|
2015-12-14 21:53:52 -05:00
|
|
|
projects.joins(:forked_project_link).find_by('forked_project_links.forked_from_project_id = ?', project.id)
|
2014-11-14 09:06:39 -05:00
|
|
|
end
|
2016-06-22 17:04:51 -04:00
|
|
|
|
2016-09-01 19:49:48 -04:00
|
|
|
def lfs_enabled?
|
|
|
|
# User namespace will always default to the global setting
|
|
|
|
Gitlab.config.lfs.enabled
|
|
|
|
end
|
|
|
|
|
2016-10-31 07:00:53 -04:00
|
|
|
def full_path
|
|
|
|
if parent
|
|
|
|
parent.full_path + '/' + path
|
|
|
|
else
|
|
|
|
path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-06-22 17:04:51 -04:00
|
|
|
private
|
|
|
|
|
|
|
|
def repository_storage_paths
|
|
|
|
# We need to get the storage paths for all the projects, even the ones that are
|
|
|
|
# pending delete. Unscoping also get rids of the default order, which causes
|
|
|
|
# problems with SELECT DISTINCT.
|
|
|
|
Project.unscoped do
|
|
|
|
projects.select('distinct(repository_storage)').to_a.map(&:repository_storage_path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def rm_dir
|
|
|
|
# Remove the namespace directory in all storages paths used by member projects
|
|
|
|
@old_repository_storage_paths.each do |repository_storage_path|
|
|
|
|
# Move namespace directory into trash.
|
|
|
|
# We will remove it later async
|
|
|
|
new_path = "#{path}+#{id}+deleted"
|
|
|
|
|
|
|
|
if gitlab_shell.mv_namespace(repository_storage_path, path, new_path)
|
|
|
|
message = "Namespace directory \"#{path}\" moved to \"#{new_path}\""
|
|
|
|
Gitlab::AppLogger.info message
|
|
|
|
|
|
|
|
# Remove namespace directroy async with delay so
|
|
|
|
# GitLab has time to remove all projects first
|
|
|
|
GitlabShellWorker.perform_in(5.minutes, :rm_namespace, repository_storage_path, new_path)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-11-21 08:33:58 -05:00
|
|
|
|
|
|
|
def refresh_access_of_projects_invited_groups
|
|
|
|
Group.
|
|
|
|
joins(project_group_links: :project).
|
|
|
|
where(projects: { namespace_id: id }).
|
|
|
|
find_each(&:refresh_members_authorized_projects)
|
|
|
|
end
|
2016-10-31 07:00:53 -04:00
|
|
|
|
|
|
|
def full_path_changed?
|
|
|
|
path_changed? || parent_id_changed?
|
|
|
|
end
|
2012-11-22 13:34:16 -05:00
|
|
|
end
|