65 lines
2.2 KiB
Ruby
65 lines
2.2 KiB
Ruby
|
module Gitlab
|
||
|
module Git
|
||
|
module EncodingHelper
|
||
|
extend self
|
||
|
|
||
|
# This threshold is carefully tweaked to prevent usage of encodings detected
|
||
|
# by CharlockHolmes with low confidence. If CharlockHolmes confidence is low,
|
||
|
# we're better off sticking with utf8 encoding.
|
||
|
# Reason: git diff can return strings with invalid utf8 byte sequences if it
|
||
|
# truncates a diff in the middle of a multibyte character. In this case
|
||
|
# CharlockHolmes will try to guess the encoding and will likely suggest an
|
||
|
# obscure encoding with low confidence.
|
||
|
# There is a lot more info with this merge request:
|
||
|
# https://gitlab.com/gitlab-org/gitlab_git/merge_requests/77#note_4754193
|
||
|
ENCODING_CONFIDENCE_THRESHOLD = 40
|
||
|
|
||
|
def encode!(message)
|
||
|
return nil unless message.respond_to? :force_encoding
|
||
|
|
||
|
# if message is utf-8 encoding, just return it
|
||
|
message.force_encoding("UTF-8")
|
||
|
return message if message.valid_encoding?
|
||
|
|
||
|
# return message if message type is binary
|
||
|
detect = CharlockHolmes::EncodingDetector.detect(message)
|
||
|
return message.force_encoding("BINARY") if detect && detect[:type] == :binary
|
||
|
|
||
|
# force detected encoding if we have sufficient confidence.
|
||
|
if detect && detect[:encoding] && detect[:confidence] > ENCODING_CONFIDENCE_THRESHOLD
|
||
|
message.force_encoding(detect[:encoding])
|
||
|
end
|
||
|
|
||
|
# encode and clean the bad chars
|
||
|
message.replace clean(message)
|
||
|
rescue
|
||
|
encoding = detect ? detect[:encoding] : "unknown"
|
||
|
"--broken encoding: #{encoding}"
|
||
|
end
|
||
|
|
||
|
def encode_utf8(message)
|
||
|
detect = CharlockHolmes::EncodingDetector.detect(message)
|
||
|
if detect
|
||
|
begin
|
||
|
CharlockHolmes::Converter.convert(message, detect[:encoding], 'UTF-8')
|
||
|
rescue ArgumentError => e
|
||
|
Rails.logger.warn("Ignoring error converting #{detect[:encoding]} into UTF8: #{e.message}")
|
||
|
|
||
|
''
|
||
|
end
|
||
|
else
|
||
|
clean(message)
|
||
|
end
|
||
|
end
|
||
|
|
||
|
private
|
||
|
|
||
|
def clean(message)
|
||
|
message.encode("UTF-16BE", undef: :replace, invalid: :replace, replace: "")
|
||
|
.encode("UTF-8")
|
||
|
.gsub("\0".encode("UTF-8"), "")
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
end
|