shopify-bundler 1.10.7
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +16 -0
- data/.rspec +3 -0
- data/.rubocop.yml +96 -0
- data/.rubocop_todo.yml +177 -0
- data/.travis.yml +104 -0
- data/CHANGELOG.md +2131 -0
- data/CODE_OF_CONDUCT.md +42 -0
- data/CONTRIBUTING.md +32 -0
- data/DEVELOPMENT.md +118 -0
- data/ISSUES.md +96 -0
- data/LICENSE.md +23 -0
- data/README.md +40 -0
- data/Rakefile +309 -0
- data/bin/rake +14 -0
- data/bin/rspec +10 -0
- data/bin/rubocop +11 -0
- data/bundler.gemspec +34 -0
- data/exe/bundle +21 -0
- data/exe/bundle_ruby +60 -0
- data/exe/bundler +21 -0
- data/lib/bundler.rb +499 -0
- data/lib/bundler/capistrano.rb +16 -0
- data/lib/bundler/cli.rb +435 -0
- data/lib/bundler/cli/binstubs.rb +37 -0
- data/lib/bundler/cli/cache.rb +34 -0
- data/lib/bundler/cli/check.rb +37 -0
- data/lib/bundler/cli/clean.rb +25 -0
- data/lib/bundler/cli/common.rb +56 -0
- data/lib/bundler/cli/config.rb +88 -0
- data/lib/bundler/cli/console.rb +37 -0
- data/lib/bundler/cli/exec.rb +51 -0
- data/lib/bundler/cli/gem.rb +208 -0
- data/lib/bundler/cli/init.rb +32 -0
- data/lib/bundler/cli/inject.rb +32 -0
- data/lib/bundler/cli/install.rb +188 -0
- data/lib/bundler/cli/lock.rb +35 -0
- data/lib/bundler/cli/open.rb +22 -0
- data/lib/bundler/cli/outdated.rb +86 -0
- data/lib/bundler/cli/package.rb +45 -0
- data/lib/bundler/cli/platform.rb +42 -0
- data/lib/bundler/cli/show.rb +74 -0
- data/lib/bundler/cli/update.rb +72 -0
- data/lib/bundler/cli/viz.rb +26 -0
- data/lib/bundler/constants.rb +5 -0
- data/lib/bundler/current_ruby.rb +218 -0
- data/lib/bundler/definition.rb +675 -0
- data/lib/bundler/dep_proxy.rb +45 -0
- data/lib/bundler/dependency.rb +119 -0
- data/lib/bundler/deployment.rb +62 -0
- data/lib/bundler/deprecate.rb +17 -0
- data/lib/bundler/dsl.rb +478 -0
- data/lib/bundler/endpoint_specification.rb +100 -0
- data/lib/bundler/env.rb +82 -0
- data/lib/bundler/environment.rb +41 -0
- data/lib/bundler/fetcher.rb +288 -0
- data/lib/bundler/fetcher/base.rb +26 -0
- data/lib/bundler/fetcher/dependency.rb +88 -0
- data/lib/bundler/fetcher/downloader.rb +60 -0
- data/lib/bundler/fetcher/index.rb +32 -0
- data/lib/bundler/friendly_errors.rb +92 -0
- data/lib/bundler/gem_helper.rb +191 -0
- data/lib/bundler/gem_helpers.rb +26 -0
- data/lib/bundler/gem_installer.rb +9 -0
- data/lib/bundler/gem_path_manipulation.rb +8 -0
- data/lib/bundler/gem_tasks.rb +5 -0
- data/lib/bundler/graph.rb +173 -0
- data/lib/bundler/index.rb +199 -0
- data/lib/bundler/injector.rb +62 -0
- data/lib/bundler/inline.rb +58 -0
- data/lib/bundler/installer.rb +242 -0
- data/lib/bundler/installer/parallel_installer.rb +122 -0
- data/lib/bundler/installer/standalone.rb +48 -0
- data/lib/bundler/lazy_specification.rb +82 -0
- data/lib/bundler/lockfile_parser.rb +199 -0
- data/lib/bundler/match_platform.rb +13 -0
- data/lib/bundler/psyched_yaml.rb +26 -0
- data/lib/bundler/remote_specification.rb +82 -0
- data/lib/bundler/resolver.rb +350 -0
- data/lib/bundler/retry.rb +60 -0
- data/lib/bundler/ruby_dsl.rb +11 -0
- data/lib/bundler/ruby_version.rb +116 -0
- data/lib/bundler/rubygems_ext.rb +175 -0
- data/lib/bundler/rubygems_integration.rb +674 -0
- data/lib/bundler/runtime.rb +285 -0
- data/lib/bundler/settings.rb +263 -0
- data/lib/bundler/setup.rb +26 -0
- data/lib/bundler/shared_helpers.rb +176 -0
- data/lib/bundler/similarity_detector.rb +61 -0
- data/lib/bundler/source.rb +35 -0
- data/lib/bundler/source/git.rb +286 -0
- data/lib/bundler/source/git/git_proxy.rb +190 -0
- data/lib/bundler/source/path.rb +224 -0
- data/lib/bundler/source/path/installer.rb +43 -0
- data/lib/bundler/source/rubygems.rb +436 -0
- data/lib/bundler/source/rubygems/remote.rb +38 -0
- data/lib/bundler/source_list.rb +101 -0
- data/lib/bundler/spec_set.rb +156 -0
- data/lib/bundler/ssl_certs/.document +1 -0
- data/lib/bundler/ssl_certs/AddTrustExternalCARoot-2048.pem +25 -0
- data/lib/bundler/ssl_certs/AddTrustExternalCARoot.pem +32 -0
- data/lib/bundler/ssl_certs/Class3PublicPrimaryCertificationAuthority.pem +14 -0
- data/lib/bundler/ssl_certs/DigiCertHighAssuranceEVRootCA.pem +23 -0
- data/lib/bundler/ssl_certs/EntrustnetSecureServerCertificationAuthority.pem +28 -0
- data/lib/bundler/ssl_certs/GeoTrustGlobalCA.pem +20 -0
- data/lib/bundler/ssl_certs/certificate_manager.rb +64 -0
- data/lib/bundler/stub_specification.rb +23 -0
- data/lib/bundler/templates/Executable +16 -0
- data/lib/bundler/templates/Executable.standalone +12 -0
- data/lib/bundler/templates/Gemfile +4 -0
- data/lib/bundler/templates/newgem/.travis.yml.tt +4 -0
- data/lib/bundler/templates/newgem/CODE_OF_CONDUCT.md.tt +22 -0
- data/lib/bundler/templates/newgem/Gemfile.tt +4 -0
- data/lib/bundler/templates/newgem/LICENSE.txt.tt +21 -0
- data/lib/bundler/templates/newgem/README.md.tt +41 -0
- data/lib/bundler/templates/newgem/Rakefile.tt +29 -0
- data/lib/bundler/templates/newgem/bin/console.tt +14 -0
- data/lib/bundler/templates/newgem/bin/setup.tt +8 -0
- data/lib/bundler/templates/newgem/exe/newgem.tt +3 -0
- data/lib/bundler/templates/newgem/ext/newgem/extconf.rb.tt +3 -0
- data/lib/bundler/templates/newgem/ext/newgem/newgem.c.tt +9 -0
- data/lib/bundler/templates/newgem/ext/newgem/newgem.h.tt +6 -0
- data/lib/bundler/templates/newgem/gitignore.tt +16 -0
- data/lib/bundler/templates/newgem/lib/newgem.rb.tt +12 -0
- data/lib/bundler/templates/newgem/lib/newgem/version.rb.tt +7 -0
- data/lib/bundler/templates/newgem/newgem.gemspec.tt +43 -0
- data/lib/bundler/templates/newgem/rspec.tt +2 -0
- data/lib/bundler/templates/newgem/spec/newgem_spec.rb.tt +11 -0
- data/lib/bundler/templates/newgem/spec/spec_helper.rb.tt +2 -0
- data/lib/bundler/templates/newgem/test/newgem_test.rb.tt +11 -0
- data/lib/bundler/templates/newgem/test/test_helper.rb.tt +4 -0
- data/lib/bundler/ui.rb +7 -0
- data/lib/bundler/ui/rg_proxy.rb +17 -0
- data/lib/bundler/ui/shell.rb +108 -0
- data/lib/bundler/ui/silent.rb +44 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo.rb +5 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/dependency_graph.rb +277 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/errors.rb +69 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/gem_metadata.rb +3 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/modules/specification_provider.rb +99 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/modules/ui.rb +63 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/resolution.rb +434 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/resolver.rb +43 -0
- data/lib/bundler/vendor/molinillo/lib/molinillo/state.rb +51 -0
- data/lib/bundler/vendor/net/http/faster.rb +26 -0
- data/lib/bundler/vendor/net/http/persistent.rb +1230 -0
- data/lib/bundler/vendor/net/http/persistent/ssl_reuse.rb +128 -0
- data/lib/bundler/vendor/thor/lib/thor.rb +484 -0
- data/lib/bundler/vendor/thor/lib/thor/actions.rb +319 -0
- data/lib/bundler/vendor/thor/lib/thor/actions/create_file.rb +103 -0
- data/lib/bundler/vendor/thor/lib/thor/actions/create_link.rb +59 -0
- data/lib/bundler/vendor/thor/lib/thor/actions/directory.rb +118 -0
- data/lib/bundler/vendor/thor/lib/thor/actions/empty_directory.rb +135 -0
- data/lib/bundler/vendor/thor/lib/thor/actions/file_manipulation.rb +316 -0
- data/lib/bundler/vendor/thor/lib/thor/actions/inject_into_file.rb +107 -0
- data/lib/bundler/vendor/thor/lib/thor/base.rb +656 -0
- data/lib/bundler/vendor/thor/lib/thor/command.rb +133 -0
- data/lib/bundler/vendor/thor/lib/thor/core_ext/hash_with_indifferent_access.rb +77 -0
- data/lib/bundler/vendor/thor/lib/thor/core_ext/io_binary_read.rb +10 -0
- data/lib/bundler/vendor/thor/lib/thor/core_ext/ordered_hash.rb +98 -0
- data/lib/bundler/vendor/thor/lib/thor/error.rb +32 -0
- data/lib/bundler/vendor/thor/lib/thor/group.rb +281 -0
- data/lib/bundler/vendor/thor/lib/thor/invocation.rb +178 -0
- data/lib/bundler/vendor/thor/lib/thor/line_editor.rb +17 -0
- data/lib/bundler/vendor/thor/lib/thor/line_editor/basic.rb +35 -0
- data/lib/bundler/vendor/thor/lib/thor/line_editor/readline.rb +88 -0
- data/lib/bundler/vendor/thor/lib/thor/parser.rb +4 -0
- data/lib/bundler/vendor/thor/lib/thor/parser/argument.rb +73 -0
- data/lib/bundler/vendor/thor/lib/thor/parser/arguments.rb +175 -0
- data/lib/bundler/vendor/thor/lib/thor/parser/option.rb +125 -0
- data/lib/bundler/vendor/thor/lib/thor/parser/options.rb +218 -0
- data/lib/bundler/vendor/thor/lib/thor/rake_compat.rb +71 -0
- data/lib/bundler/vendor/thor/lib/thor/runner.rb +322 -0
- data/lib/bundler/vendor/thor/lib/thor/shell.rb +81 -0
- data/lib/bundler/vendor/thor/lib/thor/shell/basic.rb +421 -0
- data/lib/bundler/vendor/thor/lib/thor/shell/color.rb +149 -0
- data/lib/bundler/vendor/thor/lib/thor/shell/html.rb +126 -0
- data/lib/bundler/vendor/thor/lib/thor/util.rb +267 -0
- data/lib/bundler/vendor/thor/lib/thor/version.rb +3 -0
- data/lib/bundler/vendored_molinillo.rb +2 -0
- data/lib/bundler/vendored_persistent.rb +11 -0
- data/lib/bundler/vendored_thor.rb +3 -0
- data/lib/bundler/version.rb +6 -0
- data/lib/bundler/vlad.rb +11 -0
- data/lib/bundler/worker.rb +72 -0
- data/man/bundle-config.ronn +187 -0
- data/man/bundle-exec.ronn +136 -0
- data/man/bundle-gem.ronn +77 -0
- data/man/bundle-install.ronn +398 -0
- data/man/bundle-package.ronn +66 -0
- data/man/bundle-platform.ronn +42 -0
- data/man/bundle-update.ronn +188 -0
- data/man/bundle.ronn +98 -0
- data/man/gemfile.5.ronn +495 -0
- data/man/index.txt +8 -0
- metadata +346 -0
@@ -0,0 +1,44 @@
|
|
1
|
+
module Bundler
|
2
|
+
module UI
|
3
|
+
class Silent
|
4
|
+
def info(message, newline = nil)
|
5
|
+
end
|
6
|
+
|
7
|
+
def confirm(message, newline = nil)
|
8
|
+
end
|
9
|
+
|
10
|
+
def warn(message, newline = nil)
|
11
|
+
end
|
12
|
+
|
13
|
+
def error(message, newline = nil)
|
14
|
+
end
|
15
|
+
|
16
|
+
def debug(message, newline = nil)
|
17
|
+
end
|
18
|
+
|
19
|
+
def debug?
|
20
|
+
false
|
21
|
+
end
|
22
|
+
|
23
|
+
def quiet?
|
24
|
+
false
|
25
|
+
end
|
26
|
+
|
27
|
+
def ask(message)
|
28
|
+
end
|
29
|
+
|
30
|
+
def level=(name)
|
31
|
+
end
|
32
|
+
|
33
|
+
def level(name = nil)
|
34
|
+
end
|
35
|
+
|
36
|
+
def trace(message, newline = nil)
|
37
|
+
end
|
38
|
+
|
39
|
+
def silence
|
40
|
+
yield
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -0,0 +1,5 @@
|
|
1
|
+
require 'bundler/vendor/molinillo/lib/molinillo/gem_metadata'
|
2
|
+
require 'bundler/vendor/molinillo/lib/molinillo/errors'
|
3
|
+
require 'bundler/vendor/molinillo/lib/molinillo/resolver'
|
4
|
+
require 'bundler/vendor/molinillo/lib/molinillo/modules/ui'
|
5
|
+
require 'bundler/vendor/molinillo/lib/molinillo/modules/specification_provider'
|
@@ -0,0 +1,277 @@
|
|
1
|
+
require 'set'
|
2
|
+
require 'tsort'
|
3
|
+
|
4
|
+
module Bundler::Molinillo
|
5
|
+
# A directed acyclic graph that is tuned to hold named dependencies
|
6
|
+
class DependencyGraph
|
7
|
+
include Enumerable
|
8
|
+
|
9
|
+
# Enumerates through the vertices of the graph.
|
10
|
+
# @return [Array<Vertex>] The graph's vertices.
|
11
|
+
def each
|
12
|
+
vertices.values.each { |v| yield v }
|
13
|
+
end
|
14
|
+
|
15
|
+
include TSort
|
16
|
+
|
17
|
+
alias_method :tsort_each_node, :each
|
18
|
+
|
19
|
+
def tsort_each_child(vertex, &block)
|
20
|
+
vertex.successors.each(&block)
|
21
|
+
end
|
22
|
+
|
23
|
+
# Topologically sorts the given vertices.
|
24
|
+
# @param [Enumerable<Vertex>] vertices the vertices to be sorted, which must
|
25
|
+
# all belong to the same graph.
|
26
|
+
# @return [Array<Vertex>] The sorted vertices.
|
27
|
+
def self.tsort(vertices)
|
28
|
+
TSort.tsort(
|
29
|
+
lambda { |b| vertices.each(&b) },
|
30
|
+
lambda { |v, &b| (v.successors & vertices).each(&b) }
|
31
|
+
)
|
32
|
+
end
|
33
|
+
|
34
|
+
# A directed edge of a {DependencyGraph}
|
35
|
+
# @attr [Vertex] origin The origin of the directed edge
|
36
|
+
# @attr [Vertex] destination The destination of the directed edge
|
37
|
+
# @attr [Object] requirement The requirement the directed edge represents
|
38
|
+
Edge = Struct.new(:origin, :destination, :requirement)
|
39
|
+
|
40
|
+
# @return [{String => Vertex}] the vertices of the dependency graph, keyed
|
41
|
+
# by {Vertex#name}
|
42
|
+
attr_reader :vertices
|
43
|
+
|
44
|
+
def initialize
|
45
|
+
@vertices = {}
|
46
|
+
end
|
47
|
+
|
48
|
+
# Initializes a copy of a {DependencyGraph}, ensuring that all {#vertices}
|
49
|
+
# are properly copied.
|
50
|
+
def initialize_copy(other)
|
51
|
+
super
|
52
|
+
@vertices = {}
|
53
|
+
traverse = lambda do |new_v, old_v|
|
54
|
+
return if new_v.outgoing_edges.size == old_v.outgoing_edges.size
|
55
|
+
old_v.outgoing_edges.each do |edge|
|
56
|
+
destination = add_vertex(edge.destination.name, edge.destination.payload)
|
57
|
+
add_edge_no_circular(new_v, destination, edge.requirement)
|
58
|
+
traverse.call(destination, edge.destination)
|
59
|
+
end
|
60
|
+
end
|
61
|
+
other.vertices.each do |name, vertex|
|
62
|
+
new_vertex = add_vertex(name, vertex.payload, vertex.root?)
|
63
|
+
new_vertex.explicit_requirements.replace(vertex.explicit_requirements)
|
64
|
+
traverse.call(new_vertex, vertex)
|
65
|
+
end
|
66
|
+
end
|
67
|
+
|
68
|
+
# @return [String] a string suitable for debugging
|
69
|
+
def inspect
|
70
|
+
"#{self.class}:#{vertices.values.inspect}"
|
71
|
+
end
|
72
|
+
|
73
|
+
# @return [Boolean] whether the two dependency graphs are equal, determined
|
74
|
+
# by a recursive traversal of each {#root_vertices} and its
|
75
|
+
# {Vertex#successors}
|
76
|
+
def ==(other)
|
77
|
+
return false unless other
|
78
|
+
vertices.each do |name, vertex|
|
79
|
+
other_vertex = other.vertex_named(name)
|
80
|
+
return false unless other_vertex
|
81
|
+
return false unless other_vertex.successors.map(&:name).to_set == vertex.successors.map(&:name).to_set
|
82
|
+
end
|
83
|
+
end
|
84
|
+
|
85
|
+
# @param [String] name
|
86
|
+
# @param [Object] payload
|
87
|
+
# @param [Array<String>] parent_names
|
88
|
+
# @param [Object] requirement the requirement that is requiring the child
|
89
|
+
# @return [void]
|
90
|
+
def add_child_vertex(name, payload, parent_names, requirement)
|
91
|
+
vertex = add_vertex(name, payload)
|
92
|
+
parent_names.each do |parent_name|
|
93
|
+
unless parent_name
|
94
|
+
vertex.root = true
|
95
|
+
next
|
96
|
+
end
|
97
|
+
parent_node = vertex_named(parent_name)
|
98
|
+
add_edge(parent_node, vertex, requirement)
|
99
|
+
end
|
100
|
+
vertex
|
101
|
+
end
|
102
|
+
|
103
|
+
# @param [String] name
|
104
|
+
# @param [Object] payload
|
105
|
+
# @return [Vertex] the vertex that was added to `self`
|
106
|
+
def add_vertex(name, payload, root = false)
|
107
|
+
vertex = vertices[name] ||= Vertex.new(name, payload)
|
108
|
+
vertex.payload ||= payload
|
109
|
+
vertex.root ||= root
|
110
|
+
vertex
|
111
|
+
end
|
112
|
+
|
113
|
+
# Detaches the {#vertex_named} `name` {Vertex} from the graph, recursively
|
114
|
+
# removing any non-root vertices that were orphaned in the process
|
115
|
+
# @param [String] name
|
116
|
+
# @return [void]
|
117
|
+
def detach_vertex_named(name)
|
118
|
+
return unless vertex = vertices.delete(name)
|
119
|
+
vertex.outgoing_edges.each do |e|
|
120
|
+
v = e.destination
|
121
|
+
v.incoming_edges.delete(e)
|
122
|
+
detach_vertex_named(v.name) unless v.root? || v.predecessors.any?
|
123
|
+
end
|
124
|
+
end
|
125
|
+
|
126
|
+
# @param [String] name
|
127
|
+
# @return [Vertex,nil] the vertex with the given name
|
128
|
+
def vertex_named(name)
|
129
|
+
vertices[name]
|
130
|
+
end
|
131
|
+
|
132
|
+
# @param [String] name
|
133
|
+
# @return [Vertex,nil] the root vertex with the given name
|
134
|
+
def root_vertex_named(name)
|
135
|
+
vertex = vertex_named(name)
|
136
|
+
vertex if vertex && vertex.root?
|
137
|
+
end
|
138
|
+
|
139
|
+
# Adds a new {Edge} to the dependency graph
|
140
|
+
# @param [Vertex] origin
|
141
|
+
# @param [Vertex] destination
|
142
|
+
# @param [Object] requirement the requirement that this edge represents
|
143
|
+
# @return [Edge] the added edge
|
144
|
+
def add_edge(origin, destination, requirement)
|
145
|
+
if destination.path_to?(origin)
|
146
|
+
raise CircularDependencyError.new([origin, destination])
|
147
|
+
end
|
148
|
+
add_edge_no_circular(origin, destination, requirement)
|
149
|
+
end
|
150
|
+
|
151
|
+
private
|
152
|
+
|
153
|
+
def add_edge_no_circular(origin, destination, requirement)
|
154
|
+
edge = Edge.new(origin, destination, requirement)
|
155
|
+
origin.outgoing_edges << edge
|
156
|
+
destination.incoming_edges << edge
|
157
|
+
edge
|
158
|
+
end
|
159
|
+
|
160
|
+
# A vertex in a {DependencyGraph} that encapsulates a {#name} and a
|
161
|
+
# {#payload}
|
162
|
+
class Vertex
|
163
|
+
# @return [String] the name of the vertex
|
164
|
+
attr_accessor :name
|
165
|
+
|
166
|
+
# @return [Object] the payload the vertex holds
|
167
|
+
attr_accessor :payload
|
168
|
+
|
169
|
+
# @return [Arrary<Object>] the explicit requirements that required
|
170
|
+
# this vertex
|
171
|
+
attr_reader :explicit_requirements
|
172
|
+
|
173
|
+
# @return [Boolean] whether the vertex is considered a root vertex
|
174
|
+
attr_accessor :root
|
175
|
+
alias_method :root?, :root
|
176
|
+
|
177
|
+
# @param [String] name see {#name}
|
178
|
+
# @param [Object] payload see {#payload}
|
179
|
+
def initialize(name, payload)
|
180
|
+
@name = name
|
181
|
+
@payload = payload
|
182
|
+
@explicit_requirements = []
|
183
|
+
@outgoing_edges = []
|
184
|
+
@incoming_edges = []
|
185
|
+
end
|
186
|
+
|
187
|
+
# @return [Array<Object>] all of the requirements that required
|
188
|
+
# this vertex
|
189
|
+
def requirements
|
190
|
+
incoming_edges.map(&:requirement) + explicit_requirements
|
191
|
+
end
|
192
|
+
|
193
|
+
# @return [Array<Edge>] the edges of {#graph} that have `self` as their
|
194
|
+
# {Edge#origin}
|
195
|
+
attr_accessor :outgoing_edges
|
196
|
+
|
197
|
+
# @return [Array<Edge>] the edges of {#graph} that have `self` as their
|
198
|
+
# {Edge#destination}
|
199
|
+
attr_accessor :incoming_edges
|
200
|
+
|
201
|
+
# @return [Array<Vertex>] the vertices of {#graph} that have an edge with
|
202
|
+
# `self` as their {Edge#destination}
|
203
|
+
def predecessors
|
204
|
+
incoming_edges.map(&:origin)
|
205
|
+
end
|
206
|
+
|
207
|
+
# @return [Array<Vertex>] the vertices of {#graph} where `self` is a
|
208
|
+
# {#descendent?}
|
209
|
+
def recursive_predecessors
|
210
|
+
vertices = predecessors
|
211
|
+
vertices += vertices.map(&:recursive_predecessors).flatten(1)
|
212
|
+
vertices.uniq!
|
213
|
+
vertices
|
214
|
+
end
|
215
|
+
|
216
|
+
# @return [Array<Vertex>] the vertices of {#graph} that have an edge with
|
217
|
+
# `self` as their {Edge#origin}
|
218
|
+
def successors
|
219
|
+
outgoing_edges.map(&:destination)
|
220
|
+
end
|
221
|
+
|
222
|
+
# @return [Array<Vertex>] the vertices of {#graph} where `self` is an
|
223
|
+
# {#ancestor?}
|
224
|
+
def recursive_successors
|
225
|
+
vertices = successors
|
226
|
+
vertices += vertices.map(&:recursive_successors).flatten(1)
|
227
|
+
vertices.uniq!
|
228
|
+
vertices
|
229
|
+
end
|
230
|
+
|
231
|
+
# @return [String] a string suitable for debugging
|
232
|
+
def inspect
|
233
|
+
"#{self.class}:#{name}(#{payload.inspect})"
|
234
|
+
end
|
235
|
+
|
236
|
+
# @return [Boolean] whether the two vertices are equal, determined
|
237
|
+
# by a recursive traversal of each {Vertex#successors}
|
238
|
+
def ==(other)
|
239
|
+
shallow_eql?(other) &&
|
240
|
+
successors.to_set == other.successors.to_set
|
241
|
+
end
|
242
|
+
|
243
|
+
# @return [Boolean] whether the two vertices are equal, determined
|
244
|
+
# solely by {#name} and {#payload} equality
|
245
|
+
def shallow_eql?(other)
|
246
|
+
other &&
|
247
|
+
name == other.name &&
|
248
|
+
payload == other.payload
|
249
|
+
end
|
250
|
+
|
251
|
+
alias_method :eql?, :==
|
252
|
+
|
253
|
+
# @return [Fixnum] a hash for the vertex based upon its {#name}
|
254
|
+
def hash
|
255
|
+
name.hash
|
256
|
+
end
|
257
|
+
|
258
|
+
# Is there a path from `self` to `other` following edges in the
|
259
|
+
# dependency graph?
|
260
|
+
# @return true iff there is a path following edges within this {#graph}
|
261
|
+
def path_to?(other)
|
262
|
+
equal?(other) || successors.any? { |v| v.path_to?(other) }
|
263
|
+
end
|
264
|
+
|
265
|
+
alias_method :descendent?, :path_to?
|
266
|
+
|
267
|
+
# Is there a path from `other` to `self` following edges in the
|
268
|
+
# dependency graph?
|
269
|
+
# @return true iff there is a path following edges within this {#graph}
|
270
|
+
def ancestor?(other)
|
271
|
+
other.path_to?(self)
|
272
|
+
end
|
273
|
+
|
274
|
+
alias_method :is_reachable_from?, :ancestor?
|
275
|
+
end
|
276
|
+
end
|
277
|
+
end
|
@@ -0,0 +1,69 @@
|
|
1
|
+
module Bundler::Molinillo
|
2
|
+
# An error that occurred during the resolution process
|
3
|
+
class ResolverError < StandardError; end
|
4
|
+
|
5
|
+
# An error caused by searching for a dependency that is completely unknown,
|
6
|
+
# i.e. has no versions available whatsoever.
|
7
|
+
class NoSuchDependencyError < ResolverError
|
8
|
+
# @return [Object] the dependency that could not be found
|
9
|
+
attr_accessor :dependency
|
10
|
+
|
11
|
+
# @return [Array<Object>] the specifications that depended upon {#dependency}
|
12
|
+
attr_accessor :required_by
|
13
|
+
|
14
|
+
# @param [Object] dependency @see {#dependency}
|
15
|
+
# @param [Array<Object>] required_by @see {#required_by}
|
16
|
+
def initialize(dependency, required_by = [])
|
17
|
+
@dependency = dependency
|
18
|
+
@required_by = required_by
|
19
|
+
super()
|
20
|
+
end
|
21
|
+
|
22
|
+
def message
|
23
|
+
sources = required_by.map { |r| "`#{r}`" }.join(' and ')
|
24
|
+
message = "Unable to find a specification for `#{dependency}`"
|
25
|
+
message << " depended upon by #{sources}" unless sources.empty?
|
26
|
+
message
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
# An error caused by attempting to fulfil a dependency that was circular
|
31
|
+
#
|
32
|
+
# @note This exception will be thrown iff a {Vertex} is added to a
|
33
|
+
# {DependencyGraph} that has a {DependencyGraph::Vertex#path_to?} an
|
34
|
+
# existing {DependencyGraph::Vertex}
|
35
|
+
class CircularDependencyError < ResolverError
|
36
|
+
# [Set<Object>] the dependencies responsible for causing the error
|
37
|
+
attr_reader :dependencies
|
38
|
+
|
39
|
+
# @param [Array<DependencyGraph::Vertex>] nodes the nodes in the dependency
|
40
|
+
# that caused the error
|
41
|
+
def initialize(nodes)
|
42
|
+
super "There is a circular dependency between #{nodes.map(&:name).join(' and ')}"
|
43
|
+
@dependencies = nodes.map(&:payload).to_set
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
# An error caused by conflicts in version
|
48
|
+
class VersionConflict < ResolverError
|
49
|
+
# @return [{String => Resolution::Conflict}] the conflicts that caused
|
50
|
+
# resolution to fail
|
51
|
+
attr_reader :conflicts
|
52
|
+
|
53
|
+
# @param [{String => Resolution::Conflict}] conflicts see {#conflicts}
|
54
|
+
def initialize(conflicts)
|
55
|
+
pairs = []
|
56
|
+
conflicts.values.flatten.map(&:requirements).flatten.each do |conflicting|
|
57
|
+
conflicting.each do |source, conflict_requirements|
|
58
|
+
conflict_requirements.each do |c|
|
59
|
+
pairs << [c, source]
|
60
|
+
end
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
super "Unable to satisfy the following requirements:\n\n" \
|
65
|
+
"#{pairs.map { |r, d| "- `#{r}` required by `#{d}`" }.join("\n")}"
|
66
|
+
@conflicts = conflicts
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,99 @@
|
|
1
|
+
module Bundler::Molinillo
|
2
|
+
# Provides information about specifcations and dependencies to the resolver,
|
3
|
+
# allowing the {Resolver} class to remain generic while still providing power
|
4
|
+
# and flexibility.
|
5
|
+
#
|
6
|
+
# This module contains the methods that users of Bundler::Molinillo must to implement,
|
7
|
+
# using knowledge of their own model classes.
|
8
|
+
module SpecificationProvider
|
9
|
+
# Search for the specifications that match the given dependency.
|
10
|
+
# The specifications in the returned array will be considered in reverse
|
11
|
+
# order, so the latest version ought to be last.
|
12
|
+
# @note This method should be 'pure', i.e. the return value should depend
|
13
|
+
# only on the `dependency` parameter.
|
14
|
+
#
|
15
|
+
# @param [Object] dependency
|
16
|
+
# @return [Array<Object>] the specifications that satisfy the given
|
17
|
+
# `dependency`.
|
18
|
+
def search_for(dependency)
|
19
|
+
[]
|
20
|
+
end
|
21
|
+
|
22
|
+
# Returns the dependencies of `specification`.
|
23
|
+
# @note This method should be 'pure', i.e. the return value should depend
|
24
|
+
# only on the `specification` parameter.
|
25
|
+
#
|
26
|
+
# @param [Object] specification
|
27
|
+
# @return [Array<Object>] the dependencies that are required by the given
|
28
|
+
# `specification`.
|
29
|
+
def dependencies_for(specification)
|
30
|
+
[]
|
31
|
+
end
|
32
|
+
|
33
|
+
# Determines whether the given `requirement` is satisfied by the given
|
34
|
+
# `spec`, in the context of the current `activated` dependency graph.
|
35
|
+
#
|
36
|
+
# @param [Object] requirement
|
37
|
+
# @param [DependencyGraph] activated the current dependency graph in the
|
38
|
+
# resolution process.
|
39
|
+
# @param [Object] spec
|
40
|
+
# @return [Boolean] whether `requirement` is satisfied by `spec` in the
|
41
|
+
# context of the current `activated` dependency graph.
|
42
|
+
def requirement_satisfied_by?(requirement, activated, spec)
|
43
|
+
true
|
44
|
+
end
|
45
|
+
|
46
|
+
# Returns the name for the given `dependency`.
|
47
|
+
# @note This method should be 'pure', i.e. the return value should depend
|
48
|
+
# only on the `dependency` parameter.
|
49
|
+
#
|
50
|
+
# @param [Object] dependency
|
51
|
+
# @return [String] the name for the given `dependency`.
|
52
|
+
def name_for(dependency)
|
53
|
+
dependency.to_s
|
54
|
+
end
|
55
|
+
|
56
|
+
# @return [String] the name of the source of explicit dependencies, i.e.
|
57
|
+
# those passed to {Resolver#resolve} directly.
|
58
|
+
def name_for_explicit_dependency_source
|
59
|
+
'user-specified dependency'
|
60
|
+
end
|
61
|
+
|
62
|
+
# @return [String] the name of the source of 'locked' dependencies, i.e.
|
63
|
+
# those passed to {Resolver#resolve} directly as the `base`
|
64
|
+
def name_for_locking_dependency_source
|
65
|
+
'Lockfile'
|
66
|
+
end
|
67
|
+
|
68
|
+
# Sort dependencies so that the ones that are easiest to resolve are first.
|
69
|
+
# Easiest to resolve is (usually) defined by:
|
70
|
+
# 1) Is this dependency already activated?
|
71
|
+
# 2) How relaxed are the requirements?
|
72
|
+
# 3) Are there any conflicts for this dependency?
|
73
|
+
# 4) How many possibilities are there to satisfy this dependency?
|
74
|
+
#
|
75
|
+
# @param [Array<Object>] dependencies
|
76
|
+
# @param [DependencyGraph] activated the current dependency graph in the
|
77
|
+
# resolution process.
|
78
|
+
# @param [{String => Array<Conflict>}] conflicts
|
79
|
+
# @return [Array<Object>] a sorted copy of `dependencies`.
|
80
|
+
def sort_dependencies(dependencies, activated, conflicts)
|
81
|
+
dependencies.sort_by do |dependency|
|
82
|
+
name = name_for(dependency)
|
83
|
+
[
|
84
|
+
activated.vertex_named(name).payload ? 0 : 1,
|
85
|
+
conflicts[name] ? 0 : 1,
|
86
|
+
]
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
# Returns whether this dependency, which has no possible matching
|
91
|
+
# specifications, can safely be ignored.
|
92
|
+
#
|
93
|
+
# @param [Object] dependency
|
94
|
+
# @return [Boolean] whether this dependency can safely be skipped.
|
95
|
+
def allow_missing?(dependency)
|
96
|
+
false
|
97
|
+
end
|
98
|
+
end
|
99
|
+
end
|