molinillo 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,79 @@
1
+ require File.expand_path('../spec_helper', __FILE__)
2
+ require 'set'
3
+
4
+ module Molinillo
5
+ describe DependencyGraph do
6
+ describe 'in general' do
7
+ before do
8
+ @graph = DependencyGraph.new
9
+ @root = @graph.add_root_vertex('Root', 'Root')
10
+ @root2 = @graph.add_root_vertex('Root2', 'Root2')
11
+ @child = @graph.add_child_vertex('Child', 'Child', %w(Root), 'Child')
12
+ end
13
+
14
+ it 'returns root vertices by name' do
15
+ @graph.root_vertex_named('Root').
16
+ should.equal @root
17
+ end
18
+
19
+ it 'returns vertices by name' do
20
+ @graph.vertex_named('Root').
21
+ should.equal @root
22
+ @graph.vertex_named('Child').
23
+ should.equal @child
24
+ end
25
+
26
+ it 'returns nil for non-existant root vertices' do
27
+ @graph.root_vertex_named('missing').
28
+ should.equal nil
29
+ end
30
+
31
+ it 'returns nil for non-existant vertices' do
32
+ @graph.vertex_named('missing').
33
+ should.equal nil
34
+ end
35
+ end
36
+
37
+ describe 'detaching a node' do
38
+ before do
39
+ @graph = DependencyGraph.new
40
+ end
41
+
42
+ it 'detaches a root vertex without successors' do
43
+ root = @graph.add_root_vertex('root', 'root')
44
+ @graph.detach_vertex_named(root.name)
45
+ @graph.vertex_named(root.name).
46
+ should.equal nil
47
+ @graph.vertices.count.
48
+ should.equal 0
49
+ end
50
+
51
+ it 'detaches a root vertex with successors' do
52
+ root = @graph.add_root_vertex('root', 'root')
53
+ child = @graph.add_child_vertex('child', 'child', %w(root), 'child')
54
+ @graph.detach_vertex_named(root.name)
55
+ @graph.vertex_named(root.name).
56
+ should.equal nil
57
+ @graph.vertex_named(child.name).
58
+ should.equal nil
59
+ @graph.vertices.count.
60
+ should.equal 0
61
+ end
62
+
63
+ it 'detaches a root vertex with successors with other parents' do
64
+ root = @graph.add_root_vertex('root', 'root')
65
+ root2 = @graph.add_root_vertex('root2', 'root2')
66
+ child = @graph.add_child_vertex('child', 'child', %w(root root2), 'child')
67
+ @graph.detach_vertex_named(root.name)
68
+ @graph.vertex_named(root.name).
69
+ should.equal nil
70
+ @graph.vertex_named(child.name).
71
+ should.equal child
72
+ child.predecessors.
73
+ should.equal Set[root2]
74
+ @graph.vertices.count.
75
+ should.equal 2
76
+ end
77
+ end
78
+ end
79
+ end
@@ -0,0 +1,113 @@
1
+ require File.expand_path('../spec_helper', __FILE__)
2
+ require 'json'
3
+ require 'pathname'
4
+
5
+ module Molinillo
6
+ FIXTURE_DIR = Pathname.new('spec/resolver_integration_specs')
7
+ FIXTURE_INDEX_DIR = FIXTURE_DIR + 'index'
8
+ FIXTURE_CASE_DIR = FIXTURE_DIR + 'case'
9
+
10
+ class TestCase
11
+ require File.expand_path('../spec_helper/index', __FILE__)
12
+ require File.expand_path('../spec_helper/specification', __FILE__)
13
+ require File.expand_path('../spec_helper/ui', __FILE__)
14
+
15
+ attr_accessor :name, :requested, :base, :conflicts, :resolver, :result, :index
16
+
17
+ # rubocop:disable Metrics/MethodLength
18
+ def initialize(fixture_path)
19
+ File.open(fixture_path) do |fixture|
20
+ JSON.load(fixture).tap do |test_case|
21
+ self.name = test_case['name']
22
+ self.index = TestIndex.new(test_case['index'] || 'awesome')
23
+ self.requested = test_case['requested'].map do |(name, reqs)|
24
+ VersionKit::Dependency.new name, reqs.split(',').map(&:chomp)
25
+ end
26
+ add_dependencies_to_graph = lambda do |graph, parent, hash|
27
+ name = hash['name']
28
+ version = VersionKit::Version.new(hash['version'])
29
+ dependency = index.specs[name].find { |s| s.version == version }
30
+ node = if parent
31
+ graph.add_vertex(name, dependency).tap do |v|
32
+ graph.add_edge(parent, v, dependency)
33
+ end
34
+ else
35
+ graph.add_root_vertex(name, dependency)
36
+ end
37
+ hash['dependencies'].each do |dep|
38
+ add_dependencies_to_graph.call(graph, node, dep)
39
+ end
40
+ end
41
+ self.result = test_case['resolved'].reduce(DependencyGraph.new) do |graph, r|
42
+ graph.tap do |g|
43
+ add_dependencies_to_graph.call(g, nil, r)
44
+ end
45
+ end
46
+ self.base = test_case['base'].reduce(DependencyGraph.new) do |graph, r|
47
+ graph.tap do |g|
48
+ add_dependencies_to_graph.call(g, nil, r)
49
+ end
50
+ end
51
+ self.conflicts = test_case['conflicts'].to_set
52
+ end
53
+ end
54
+
55
+ self.resolver = Resolver.new(index, TestUI.new)
56
+ end
57
+ # rubocop:enable Metrics/MethodLength
58
+ end
59
+
60
+ describe Resolver do
61
+
62
+ describe 'dependency resolution' do
63
+ Dir.glob(FIXTURE_CASE_DIR + '**/*.json').map do |fixture|
64
+ test_case = TestCase.new(fixture)
65
+ it test_case.name do
66
+ resolve = lambda { test_case.resolver.resolve(test_case.requested, test_case.base) }
67
+
68
+ if test_case.conflicts.any?
69
+ error = should.raise ResolverError do
70
+ resolve.call
71
+ end
72
+
73
+ names = case error
74
+ when CircularDependencyError
75
+ error.dependencies.map(&:name)
76
+ when VersionConflict
77
+ error.conflicts.keys
78
+ end.to_set
79
+ names.should.equal test_case.conflicts
80
+ else
81
+ result = resolve.call
82
+
83
+ pretty_dependencies = lambda do |dg|
84
+ dg.vertices.values.map { |v| "#{v.payload.name} (#{v.payload.version})" }.sort
85
+ end
86
+ pretty_dependencies.call(result).should.
87
+ equal pretty_dependencies.call(test_case.result)
88
+
89
+ result.should.equal test_case.result
90
+ end
91
+ end
92
+ end
93
+ end
94
+
95
+ describe 'in general' do
96
+ before do
97
+ @resolver = Resolver.new(TestIndex.new('awesome'), TestUI.new)
98
+ end
99
+
100
+ it 'can resolve a list of 0 requirements' do
101
+ @resolver.resolve([], DependencyGraph.new).
102
+ should.equal DependencyGraph.new
103
+ end
104
+
105
+ it 'includes the source of a user-specified unsatisfied dependency' do
106
+ should.raise VersionConflict do
107
+ @resolver.resolve([VersionKit::Dependency.new('missing', '3.0')], DependencyGraph.new)
108
+ end.message.should.match /required by `user-specified dependency`/
109
+ end
110
+ end
111
+
112
+ end
113
+ end
@@ -0,0 +1,34 @@
1
+
2
+ # Set up coverage analysis
3
+ #-----------------------------------------------------------------------------#
4
+
5
+ if (ENV['CI'] || ENV['GENERATE_COVERAGE']) && RUBY_VERSION >= '2.0.0'
6
+ require 'simplecov'
7
+ require 'codeclimate-test-reporter'
8
+
9
+ if ENV['CI']
10
+ SimpleCov.formatter = CodeClimate::TestReporter::Formatter
11
+ elsif ENV['GENERATE_COVERAGE']
12
+ SimpleCov.formatter = SimpleCov::Formatter::HTMLFormatter
13
+ end
14
+ SimpleCov.start do
15
+ add_filter '/vendor/'
16
+ add_filter '/lib/molinillo/modules/'
17
+ end
18
+ CodeClimate::TestReporter.start
19
+ end
20
+
21
+ # Set up
22
+ #-----------------------------------------------------------------------------#
23
+
24
+ require 'pathname'
25
+ ROOT = Pathname.new(File.expand_path('../../', __FILE__))
26
+ $LOAD_PATH.unshift((ROOT + 'lib').to_s)
27
+ $LOAD_PATH.unshift((ROOT + 'spec').to_s)
28
+
29
+ require 'bundler/setup'
30
+ require 'bacon'
31
+ require 'mocha-on-bacon'
32
+ require 'pretty_bacon'
33
+ require 'version_kit'
34
+ require 'molinillo'
@@ -0,0 +1,59 @@
1
+ module Molinillo
2
+ class TestIndex
3
+ attr_accessor :specs
4
+ include SpecificationProvider
5
+
6
+ def initialize(fixture_name)
7
+ File.open(FIXTURE_INDEX_DIR + (fixture_name + '.json'), 'r') do |fixture|
8
+ self.specs = JSON.load(fixture).reduce(Hash.new([])) do |specs_by_name, (name, versions)|
9
+ specs_by_name.tap do |specs|
10
+ specs[name] = versions.map { |s| TestSpecification.new s }.sort { |x, y| x.version <=> y.version }
11
+ end
12
+ end
13
+ end
14
+ end
15
+
16
+ def requirement_satisfied_by?(requirement, _activated, spec)
17
+ case requirement
18
+ when TestSpecification
19
+ VersionKit::Dependency.new(requirement.name, requirement.version).satisfied_by?(spec.version)
20
+ when VersionKit::Dependency
21
+ requirement.satisfied_by?(spec.version)
22
+ end
23
+ end
24
+
25
+ def search_for(dependency)
26
+ pre_release = dependency_pre_release?(dependency)
27
+ specs[dependency.name].reject do |spec|
28
+ pre_release ? false : spec.version.pre_release?
29
+ end
30
+ end
31
+
32
+ def name_for(dependency)
33
+ dependency.name
34
+ end
35
+
36
+ def dependencies_for(dependency)
37
+ dependency.dependencies
38
+ end
39
+
40
+ def sort_dependencies(dependencies, activated, conflicts)
41
+ dependencies.sort_by do |d|
42
+ [
43
+ activated.vertex_named(d.name).payload ? 0 : 1,
44
+ dependency_pre_release?(d) ? 0 : 1,
45
+ conflicts[d.name] ? 0 : 1,
46
+ specs[d.name].count,
47
+ ]
48
+ end
49
+ end
50
+
51
+ private
52
+
53
+ def dependency_pre_release?(dependency)
54
+ dependency.requirement_list.requirements.any? do |r|
55
+ VersionKit::Version.new(r.reference_version).pre_release?
56
+ end
57
+ end
58
+ end
59
+ end
@@ -0,0 +1,22 @@
1
+ module Molinillo
2
+ class TestSpecification
3
+ attr_accessor :name, :version, :dependencies
4
+ def initialize(hash)
5
+ self.name = hash['name']
6
+ self.version = VersionKit::Version.new(hash['version'])
7
+ self.dependencies = hash['dependencies'].map do |(name, requirement)|
8
+ VersionKit::Dependency.new(name, requirement.split(',').map(&:chomp))
9
+ end
10
+ end
11
+
12
+ def ==(other)
13
+ name == other.name &&
14
+ version == other.version &&
15
+ dependencies == other.dependencies
16
+ end
17
+
18
+ def to_s
19
+ "#{name} (#{version})"
20
+ end
21
+ end
22
+ end
@@ -0,0 +1,9 @@
1
+ module Molinillo
2
+ class TestUI
3
+ include UI
4
+
5
+ def output
6
+ @output ||= File.open('/dev/null', 'w')
7
+ end
8
+ end
9
+ end
@@ -0,0 +1,33 @@
1
+ require File.expand_path('../spec_helper', __FILE__)
2
+
3
+ module Molinillo
4
+ describe ResolutionState do
5
+ describe DependencyState do
6
+ before do
7
+ @state = DependencyState.new(
8
+ 'name',
9
+ %w(requirement1 requirement2 requirement3),
10
+ DependencyGraph.new,
11
+ 'requirement',
12
+ %w(possibility1 possibility),
13
+ 0,
14
+ {}
15
+ )
16
+ end
17
+
18
+ it 'pops a possibility state' do
19
+ possibility_state = @state.pop_possibility_state
20
+ %w(name requirements activated requirement conflicts).each do |attr|
21
+ possibility_state.send(attr).
22
+ should.equal @state.send(attr)
23
+ end
24
+ possibility_state.is_a?(PossibilityState).
25
+ should.be.true?
26
+ possibility_state.depth.
27
+ should.equal @state.depth + 1
28
+ possibility_state.possibilities.
29
+ should.equal %w(possibility)
30
+ end
31
+ end
32
+ end
33
+ end
metadata ADDED
@@ -0,0 +1,98 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: molinillo
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Samuel E. Giddins
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-10-26 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: bundler
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ~>
18
+ - !ruby/object:Gem::Version
19
+ version: '1.5'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ~>
25
+ - !ruby/object:Gem::Version
26
+ version: '1.5'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - '>='
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - '>='
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ description:
42
+ email:
43
+ - segiddins@segiddins.me
44
+ executables: []
45
+ extensions: []
46
+ extra_rdoc_files: []
47
+ files:
48
+ - lib/molinillo/dependency_graph.rb
49
+ - lib/molinillo/errors.rb
50
+ - lib/molinillo/gem_metadata.rb
51
+ - lib/molinillo/modules/specification_provider.rb
52
+ - lib/molinillo/modules/ui.rb
53
+ - lib/molinillo/resolution.rb
54
+ - lib/molinillo/resolver.rb
55
+ - lib/molinillo/state.rb
56
+ - lib/molinillo.rb
57
+ - README.md
58
+ - LICENSE
59
+ - spec/dependency_graph_spec.rb
60
+ - spec/resolver_spec.rb
61
+ - spec/spec_helper/index.rb
62
+ - spec/spec_helper/specification.rb
63
+ - spec/spec_helper/ui.rb
64
+ - spec/spec_helper.rb
65
+ - spec/state_spec.rb
66
+ homepage: https://github.com/CocoaPods/Molinillo
67
+ licenses:
68
+ - MIT
69
+ metadata: {}
70
+ post_install_message:
71
+ rdoc_options: []
72
+ require_paths:
73
+ - lib
74
+ required_ruby_version: !ruby/object:Gem::Requirement
75
+ requirements:
76
+ - - '>='
77
+ - !ruby/object:Gem::Version
78
+ version: '0'
79
+ required_rubygems_version: !ruby/object:Gem::Requirement
80
+ requirements:
81
+ - - '>='
82
+ - !ruby/object:Gem::Version
83
+ version: '0'
84
+ requirements: []
85
+ rubyforge_project:
86
+ rubygems_version: 2.0.14
87
+ signing_key:
88
+ specification_version: 4
89
+ summary: Provides support for dependency resolution
90
+ test_files:
91
+ - spec/dependency_graph_spec.rb
92
+ - spec/resolver_spec.rb
93
+ - spec/spec_helper/index.rb
94
+ - spec/spec_helper/specification.rb
95
+ - spec/spec_helper/ui.rb
96
+ - spec/spec_helper.rb
97
+ - spec/state_spec.rb
98
+ has_rdoc: