debian-mirror-gitlab/lib/gitlab/conflict/parser.rb

77 lines
2 KiB
Ruby
Raw Normal View History

2016-09-13 17:45:13 +05:30
module Gitlab
module Conflict
class Parser
2016-11-03 12:29:30 +05:30
class UnresolvableError < StandardError
2016-09-13 17:45:13 +05:30
end
2016-11-03 12:29:30 +05:30
class UnmergeableFile < UnresolvableError
2016-09-13 17:45:13 +05:30
end
2016-11-03 12:29:30 +05:30
class UnsupportedEncoding < UnresolvableError
end
# Recoverable errors - the conflict can be resolved in an editor, but not with
# sections.
class ParserError < StandardError
2016-09-13 17:45:13 +05:30
end
2016-11-03 12:29:30 +05:30
class UnexpectedDelimiter < ParserError
2016-09-13 17:45:13 +05:30
end
2016-11-03 12:29:30 +05:30
class MissingEndDelimiter < ParserError
2016-09-13 17:45:13 +05:30
end
def parse(text, our_path:, their_path:, parent_file: nil)
raise UnmergeableFile if text.blank? # Typically a binary file
2016-09-29 09:46:39 +05:30
raise UnmergeableFile if text.length > 200.kilobytes
2016-09-13 17:45:13 +05:30
begin
text.to_json
rescue Encoding::UndefinedConversionError
raise UnsupportedEncoding
end
line_obj_index = 0
line_old = 1
line_new = 1
type = nil
lines = []
conflict_start = "<<<<<<< #{our_path}"
conflict_middle = '======='
conflict_end = ">>>>>>> #{their_path}"
text.each_line.map do |line|
full_line = line.delete("\n")
if full_line == conflict_start
raise UnexpectedDelimiter unless type.nil?
type = 'new'
elsif full_line == conflict_middle
raise UnexpectedDelimiter unless type == 'new'
type = 'old'
elsif full_line == conflict_end
raise UnexpectedDelimiter unless type == 'old'
type = nil
elsif line[0] == '\\'
type = 'nonewline'
lines << Gitlab::Diff::Line.new(full_line, type, line_obj_index, line_old, line_new, parent_file: parent_file)
else
lines << Gitlab::Diff::Line.new(full_line, type, line_obj_index, line_old, line_new, parent_file: parent_file)
line_old += 1 if type != 'new'
line_new += 1 if type != 'old'
line_obj_index += 1
end
end
raise MissingEndDelimiter unless type.nil?
lines
end
end
end
end