summaryrefslogtreecommitdiff
path: root/lib/bundler/lockfile_generator.rb
blob: 7fb7255ad7f3410c096ac3f16033a4b6b4fae159 (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
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
# frozen_string_literal: true

module Bundler
  class LockfileGenerator
    attr_reader :definition
    attr_reader :out

    # @private
    def initialize(definition)
      @definition = definition
      @out = String.new
    end

    def self.generate(definition)
      new(definition).generate!
    end

    def generate!
      add_sources
      add_platforms
      add_dependencies
      add_optional_groups
      add_gemfiles
      add_locked_ruby_version
      add_bundled_with

      out
    end

  private

    def add_sources
      definition.send(:sources).lock_sources.each_with_index do |source, idx|
        out << "\n" unless idx.zero?

        # Add the source header
        out << source.to_lock

        # Find all specs for this source
        specs = definition.resolve.select {|s| source.can_lock?(s) }
        add_specs(specs)
      end
    end

    def add_specs(specs)
      # This needs to be sorted by full name so that
      # gems with the same name, but different platform
      # are ordered consistently
      specs.sort_by(&:full_name).each do |spec|
        next if spec.name == "bundler".freeze
        out << spec.to_lock
      end
    end

    def add_platforms
      add_section("PLATFORMS", definition.platforms)
    end

    def add_dependencies
      out << "\nDEPENDENCIES\n"

      handled = []
      definition.dependencies.sort_by(&:to_s).each do |dep|
        next if handled.include?(dep.name)
        out << "  #{dep.name}"
        unless dep.requirement.none?
          reqs = dep.requirement.requirements.map {|o, v| "#{o} #{v}" }.sort.reverse
          out << " (#{reqs.join(", ")})"
        end
        out << "!" if dep.source
        out << "\n"
        add_value(dep.options_to_lock, 4)
        handled << dep.name
      end
    end

    def add_optional_groups
      add_section("OPTIONAL GROUPS", definition.optional_groups)
    end

    def add_gemfiles
      return unless SharedHelpers.md5_available?
      gemfiles = {}
      definition.gemfiles.each do |file|
        md5 = Digest::MD5.file(file).hexdigest
        if file.to_s.start_with?(Bundler.root.to_s)
          file = file.relative_path_from(Bundler.root)
        end
        gemfiles[file] = "md5 #{md5}"
      end
      add_section("GEMFILE CHECKSUMS", gemfiles)
    end

    def add_locked_ruby_version
      return unless locked_ruby_version = definition.locked_ruby_version
      add_section("RUBY VERSION", locked_ruby_version.to_s)
    end

    def add_bundled_with
      add_section("BUNDLED WITH", definition.locked_bundler_version.to_s)
    end

    def add_section(name, value)
      out << "\n#{name}\n"
      add_value(value, 2)
    end

    def add_value(value, indentation)
      indent = " " * indentation
      case value
      when Array
        value.map(&:to_s).sort.each do |val|
          out << "#{indent}#{val}\n"
        end
      when Hash
        value.to_a.sort_by {|k, _| k.to_s }.each do |key, val|
          Array(val).sort.each do |v|
            out << "#{indent}#{key}: #{v}\n"
          end
        end
      when String
        out << "#{indent} #{value}\n"
      else
        raise ArgumentError, "#{value.inspect} can't be serialized in a lockfile"
      end
    end
  end
end