summaryrefslogtreecommitdiff
path: root/lib/gitlab/git/encoding_helper.rb
blob: f918074cb1405d45efd6c2327f5dc4163d94b6d4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
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