summaryrefslogtreecommitdiff
path: root/lib/gitlab/gfm/uploads_rewriter.rb
blob: b6eeb5d9a2b52d06d703194fc42ee05561f74ceb (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
65
66
67
68
require 'fileutils'

module Gitlab
  module Gfm
    ##
    # Class that rewrites markdown links for uploads
    #
    # Using a pattern defined in `FileUploader` it copies files to a new
    # project and rewrites all links to uploads in in a given text.
    #
    #
    class UploadsRewriter
      def initialize(text, source_project, _current_user)
        @text = text
        @source_project = source_project
        @pattern = FileUploader::MARKDOWN_PATTERN
      end

      def rewrite(target_project)
        return @text unless needs_rewrite?

        @text.gsub(@pattern) do |markdown|
          file = find_file(@source_project, $~[:secret], $~[:file])
          break markdown unless file.try(:exists?)

          new_uploader = FileUploader.new(target_project)
          with_link_in_tmp_dir(file.file) do |open_tmp_file|
            new_uploader.store!(open_tmp_file)
          end
          new_uploader.markdown_link
        end
      end

      def needs_rewrite?
        files.any?
      end

      def files
        referenced_files = @text.scan(@pattern).map do
          find_file(@source_project, $~[:secret], $~[:file])
        end

        referenced_files.compact.select(&:exists?)
      end

      private

      def find_file(project, secret, file)
        uploader = FileUploader.new(project, secret: secret)
        uploader.retrieve_from_store!(file)
        uploader.file
      end

      # Because the uploaders use 'move_to_store' we must have a temporary
      # file that is allowed to be (re)moved.
      def with_link_in_tmp_dir(file)
        dir = Dir.mktmpdir('UploadsRewriter', File.dirname(file))
        # The filename matters to Carrierwave so we make sure to preserve it
        tmp_file = File.join(dir, File.basename(file))
        File.link(file, tmp_file)
        # Open the file to placate Carrierwave
        File.open(tmp_file) { |open_file| yield open_file }
      ensure
        FileUtils.rm_rf(dir)
      end
    end
  end
end