2014-09-02 18:07:02 +05:30
|
|
|
class Snippet < ActiveRecord::Base
|
2015-04-26 12:48:37 +05:30
|
|
|
include Gitlab::VisibilityLevel
|
2015-09-11 14:41:01 +05:30
|
|
|
include Linguist::BlobHelper
|
|
|
|
include Participable
|
|
|
|
include Referable
|
|
|
|
include Sortable
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
default_value_for :visibility_level, Snippet::PRIVATE
|
2014-09-02 18:07:02 +05:30
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
belongs_to :author, class_name: 'User'
|
|
|
|
belongs_to :project
|
2014-09-02 18:07:02 +05:30
|
|
|
|
|
|
|
has_many :notes, as: :noteable, dependent: :destroy
|
|
|
|
|
|
|
|
delegate :name, :email, to: :author, prefix: true, allow_nil: true
|
|
|
|
|
|
|
|
validates :author, presence: true
|
|
|
|
validates :title, presence: true, length: { within: 0..255 }
|
2015-04-26 12:48:37 +05:30
|
|
|
validates :file_name,
|
|
|
|
length: { within: 0..255 },
|
|
|
|
format: { with: Gitlab::Regex.file_name_regex,
|
|
|
|
message: Gitlab::Regex.file_name_regex_message }
|
2014-09-02 18:07:02 +05:30
|
|
|
validates :content, presence: true
|
2015-04-26 12:48:37 +05:30
|
|
|
validates :visibility_level, inclusion: { in: Gitlab::VisibilityLevel.values }
|
2014-09-02 18:07:02 +05:30
|
|
|
|
|
|
|
# Scopes
|
2015-04-26 12:48:37 +05:30
|
|
|
scope :are_internal, -> { where(visibility_level: Snippet::INTERNAL) }
|
|
|
|
scope :are_private, -> { where(visibility_level: Snippet::PRIVATE) }
|
|
|
|
scope :are_public, -> { where(visibility_level: Snippet::PUBLIC) }
|
|
|
|
scope :public_and_internal, -> { where(visibility_level: [Snippet::PUBLIC, Snippet::INTERNAL]) }
|
2014-09-02 18:07:02 +05:30
|
|
|
scope :fresh, -> { order("created_at DESC") }
|
|
|
|
|
2016-06-16 23:09:34 +05:30
|
|
|
participant :author
|
|
|
|
participant :notes_with_associations
|
2015-09-11 14:41:01 +05:30
|
|
|
|
|
|
|
def self.reference_prefix
|
|
|
|
'$'
|
|
|
|
end
|
|
|
|
|
|
|
|
# Pattern used to extract `$123` snippet references from text
|
|
|
|
#
|
|
|
|
# This pattern supports cross-project references.
|
|
|
|
def self.reference_pattern
|
2016-06-02 11:05:42 +05:30
|
|
|
@reference_pattern ||= %r{
|
2015-09-11 14:41:01 +05:30
|
|
|
(#{Project.reference_pattern})?
|
|
|
|
#{Regexp.escape(reference_prefix)}(?<snippet>\d+)
|
|
|
|
}x
|
|
|
|
end
|
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
def self.link_reference_pattern
|
2016-06-02 11:05:42 +05:30
|
|
|
@link_reference_pattern ||= super("snippets", /(?<snippet>\d+)/)
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
def to_reference(from_project = nil)
|
|
|
|
reference = "#{self.class.reference_prefix}#{id}"
|
|
|
|
|
|
|
|
if cross_project_reference?(from_project)
|
|
|
|
reference = project.to_reference + reference
|
|
|
|
end
|
|
|
|
|
|
|
|
reference
|
|
|
|
end
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
def self.content_types
|
|
|
|
[
|
|
|
|
".rb", ".py", ".pl", ".scala", ".c", ".cpp", ".java",
|
|
|
|
".haml", ".html", ".sass", ".scss", ".xml", ".php", ".erb",
|
|
|
|
".js", ".sh", ".coffee", ".yml", ".md"
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
def data
|
|
|
|
content
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
def hook_attrs
|
|
|
|
attributes
|
|
|
|
end
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
def size
|
|
|
|
0
|
|
|
|
end
|
|
|
|
|
|
|
|
def name
|
|
|
|
file_name
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
def sanitized_file_name
|
|
|
|
file_name.gsub(/[^a-zA-Z0-9_\-\.]+/, '')
|
|
|
|
end
|
|
|
|
|
2014-09-02 18:07:02 +05:30
|
|
|
def mode
|
|
|
|
nil
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
def visibility_level_field
|
|
|
|
visibility_level
|
|
|
|
end
|
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
def no_highlighting?
|
|
|
|
content.lines.count > 1000
|
|
|
|
end
|
|
|
|
|
2016-06-16 23:09:34 +05:30
|
|
|
def notes_with_associations
|
|
|
|
notes.includes(:author)
|
|
|
|
end
|
|
|
|
|
2015-04-26 12:48:37 +05:30
|
|
|
class << self
|
2016-06-02 11:05:42 +05:30
|
|
|
# Searches for snippets with a matching title or file name.
|
|
|
|
#
|
|
|
|
# This method uses ILIKE on PostgreSQL and LIKE on MySQL.
|
|
|
|
#
|
|
|
|
# query - The search query as a String.
|
|
|
|
#
|
|
|
|
# Returns an ActiveRecord::Relation.
|
2015-04-26 12:48:37 +05:30
|
|
|
def search(query)
|
2016-06-02 11:05:42 +05:30
|
|
|
t = arel_table
|
|
|
|
pattern = "%#{query}%"
|
|
|
|
|
|
|
|
where(t[:title].matches(pattern).or(t[:file_name].matches(pattern)))
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
# Searches for snippets with matching content.
|
|
|
|
#
|
|
|
|
# This method uses ILIKE on PostgreSQL and LIKE on MySQL.
|
|
|
|
#
|
|
|
|
# query - The search query as a String.
|
|
|
|
#
|
|
|
|
# Returns an ActiveRecord::Relation.
|
2015-04-26 12:48:37 +05:30
|
|
|
def search_code(query)
|
2016-06-02 11:05:42 +05:30
|
|
|
table = Snippet.arel_table
|
|
|
|
pattern = "%#{query}%"
|
|
|
|
|
|
|
|
where(table[:content].matches(pattern))
|
2015-04-26 12:48:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def accessible_to(user)
|
|
|
|
where('visibility_level IN (?) OR author_id = ?', [Snippet::INTERNAL, Snippet::PUBLIC], user)
|
|
|
|
end
|
|
|
|
end
|
2014-09-02 18:07:02 +05:30
|
|
|
end
|