morphological_metrics-space 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: f6b4ccee2a0992b0630c278981be1f22d28957d3
4
+ data.tar.gz: 9d327612b6dfc66f2475a07d339a721da5943180
5
+ SHA512:
6
+ metadata.gz: 9a6cdde119aeccb5e9db51c697175069a0b81b7da048bfa68d67937e8a252570f0838a3964fc5464746ed926e3330502523e7fba18e5dc464dcbe002e16122b5
7
+ data.tar.gz: 3d49b156895b5f2faa6a6a3e0187c526ea2d91e390a2ac13ca309467795f63b0d728ebed17d15da66e7edc14c27717f51e88a9f67dc76138eba75655bfa5d409
data/.autotest ADDED
@@ -0,0 +1,25 @@
1
+ # -*- ruby -*-
2
+
3
+ require "autotest/restart"
4
+
5
+ # Autotest.add_hook :initialize do |at|
6
+ # at.testlib = "minitest/unit"
7
+ #
8
+ # at.extra_files << "../some/external/dependency.rb"
9
+ #
10
+ # at.libs << ":../some/external"
11
+ #
12
+ # at.add_exception "vendor"
13
+ #
14
+ # at.add_mapping(/dependency.rb/) do |f, _|
15
+ # at.files_matching(/test_.*rb$/)
16
+ # end
17
+ #
18
+ # %w(TestA TestB).each do |klass|
19
+ # at.extra_class_map[klass] = "test/test_misc.rb"
20
+ # end
21
+ # end
22
+
23
+ # Autotest.add_hook :run_command do |at|
24
+ # system "rake build"
25
+ # end
data/History.txt ADDED
@@ -0,0 +1,6 @@
1
+ === 1.0.0 / 2014-06-15
2
+
3
+ * 1 major enhancement
4
+
5
+ * Birthday!
6
+
data/Manifest.txt ADDED
@@ -0,0 +1,8 @@
1
+ .autotest
2
+ History.txt
3
+ Manifest.txt
4
+ README.rdoc
5
+ Rakefile
6
+ bin/mm_space
7
+ lib/mm/space.rb
8
+ test/mm/test_space.rb
data/README.rdoc ADDED
@@ -0,0 +1,76 @@
1
+ = mm-space
2
+
3
+ * http://www.github.com/andrewcsmith/mm-space
4
+
5
+ == DESCRIPTION:
6
+
7
+ MM::Space is a framework for working with Morphological Metrics and
8
+ Morphological Mutations in Ruby. A core component of MM::Space is that it has a
9
+ notion of global distance throughout a series of measurements or
10
+ transformations. This is what "Space" implies. It uses coworker libraries
11
+ MM::Metric and MM::Mutation to drive these measurements and transformations.
12
+
13
+ == FEATURES/PROBLEMS:
14
+
15
+ * Nothing is implemented. (Does this qualify as a "FEATURE" or a "PROBLEM"? We
16
+ will never know.)
17
+
18
+ == SYNOPSIS:
19
+
20
+ See bin/mm_space for a full example implementation (with comments)
21
+
22
+ x = MM::Metric.olm intra_delta: :tenney, inter_delta: :abs
23
+ y = MM::Metric.olm intra_delta: :ratio, inter_delta: :abs
24
+ space = Space.new [x, y]
25
+ distances = [[0.1, -0.1], [0.2, -0.2], [0.3, -0.3]]
26
+ start = %w(1/1 5/4 3/2 8/7 9/8).map {|x| MM::Ratio.from_s(x)}
27
+ space.enter do |s|
28
+ morph start, to: distances.each
29
+ morph start, to: distances.each, threads: 4
30
+ # Change some parameter of the space
31
+ s.lowest = start.map {|x| MM::Ratio.from_s("8/1")}
32
+ # Etc. etc...
33
+ end
34
+
35
+ == REQUIREMENTS:
36
+
37
+ * MM
38
+ * MM::Metric
39
+ * MM::Ratio
40
+
41
+ == INSTALL:
42
+
43
+ * Will fix when I get this all implemented
44
+
45
+ == DEVELOPERS:
46
+
47
+ After checking out the source, run:
48
+
49
+ $ rake newb
50
+
51
+ This task will install any missing dependencies, run the tests/specs,
52
+ and generate the RDoc.
53
+
54
+ == LICENSE:
55
+
56
+ (The MIT License)
57
+
58
+ Copyright (c) 2014 FIX
59
+
60
+ Permission is hereby granted, free of charge, to any person obtaining
61
+ a copy of this software and associated documentation files (the
62
+ 'Software'), to deal in the Software without restriction, including
63
+ without limitation the rights to use, copy, modify, merge, publish,
64
+ distribute, sublicense, and/or sell copies of the Software, and to
65
+ permit persons to whom the Software is furnished to do so, subject to
66
+ the following conditions:
67
+
68
+ The above copyright notice and this permission notice shall be
69
+ included in all copies or substantial portions of the Software.
70
+
71
+ THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
72
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
73
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
74
+ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
75
+ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
76
+ TORT OR OTHERWISE, ARISING FROM, OUT OF OR
data/Rakefile ADDED
@@ -0,0 +1,24 @@
1
+ # -*- ruby -*-
2
+
3
+ require "rubygems"
4
+ require "hoe"
5
+
6
+ # Hoe.plugin :compiler
7
+ # Hoe.plugin :gem_prelude_sucks
8
+ # Hoe.plugin :inline
9
+ Hoe.plugin :minitest
10
+ # Hoe.plugin :racc
11
+ # Hoe.plugin :rcov
12
+ # Hoe.plugin :rdoc
13
+ # Hoe.plugin :travis
14
+
15
+ Hoe.spec "morphological_metrics-space" do
16
+ developer("Andrew Smith", "andrewchristophersmith@gmail.com")
17
+
18
+ # self.group_name = "mm-space" # if part of an organization/group
19
+ self.readme_file = "README.rdoc"
20
+ require_rubygems_version '>= 1.4'
21
+ license "MIT" # this should match the license in the README
22
+ end
23
+
24
+ # vim: syntax=ruby
data/bin/mm_space ADDED
@@ -0,0 +1,92 @@
1
+ #!/usr/bin/env ruby
2
+ #
3
+ # # MM::Space
4
+ #
5
+ # ## What is MM::Space?
6
+ #
7
+ # MM::Space is a framework for working with Morphological Metrics and
8
+ # Morphological Mutations in Ruby. A core component of MM::Space is that it has
9
+ # a notion of global distance throughout a series of measurements or
10
+ # transformations. This is what "Space" implies. It uses coworker libraries
11
+ # MM::Metric and MM::Mutation to drive these measurements and transformations.
12
+ #
13
+ # ## What does MM::Space do?
14
+ #
15
+ # MM::Space measures and transforms morphologies. You define *n* dimensions
16
+ # (usually in metric space) and MM::Space will help you take measurements that
17
+ # take all these dimensions into account. In this sense, it's like a
18
+ # multi-metric.
19
+ #
20
+ # Another (perhaps more important) facet of MM::Space is that it allows you to
21
+ # specify a global scaling for each dimension. Given a constrained set of
22
+ # possible morphologies, it's possible to have a "maximum metric distance" in a
23
+ # single dimension of metric space, i.e., the distance between the two morphs
24
+ # that are the furthest apart. MM::Space allows you to specify this for a given
25
+ # space, thereby scaling all distances to the global maximum. A side-note is
26
+ # that it's possible to specify one of these distances as an "edge" in one
27
+ # dimension, allowing for directionality in Metric measurements.
28
+ #
29
+ # ## How do I generate new morphologies?
30
+ #
31
+ # MM::Space contains a library of search functions! So, the Space is treated as
32
+ # a total search space, and the search functions search within that search space
33
+ # (using depth-first, best-first, stochastic search, etc.) to try and find a
34
+ # morphology that satisfies all your needs.
35
+ #
36
+ # ## How do I use MM::Space?
37
+ #
38
+ # Well I'm glad you asked. First, create some Metrics for the two dimensions:
39
+ #
40
+ x = MM::Metric.olm intra_delta: :tenney, inter_delta: :abs
41
+ y = MM::Metric.olm intra_delta: :ratio, inter_delta: :abs
42
+ #
43
+ # Next, decide upon a central "start" morph for our search
44
+ #
45
+ start = %w(1/1 5/4 3/2 8/7 9/8).map {|x| MM::Ratio.from_s(x)}
46
+ #
47
+ # Initialize a Space with two dimensions <x, y>
48
+ #
49
+ space = Space.new [x, y]
50
+ #
51
+ # Now let's impose some global scaling limits on either dimension of the space.
52
+ #
53
+ space.max_distance = [8.414, 3.0]
54
+ #
55
+ # Now, let's try and find a morph 0.4 away in both directions at the same time
56
+ #
57
+ space.morph start, to: [0.4, 0.4] # => Returns a morph
58
+ #
59
+ # Let's add some notion of "up" and "down" to our space. Let the "down" be the
60
+ # vector of the same length as start, but all unison ratios.
61
+ #
62
+ space.lowest = start.map {|x| MM::Ratio.from_s("1/1")}
63
+ #
64
+ # Now we can morph using distances that are signed in our global space!
65
+ # Basically, this means "find a morph 0.4 away, where the new morph is closer to
66
+ # the lowest point than the starting morph."
67
+ #
68
+ space.morph start, to: [-0.4, 0.4] # => Returns a morph
69
+ #
70
+ # Generate a bunch of distances
71
+ #
72
+ distances = [[0.1, -0.1], [0.2, -0.2], [0.3, -0.3]]
73
+ #
74
+ # Pass the Enumerator to the object. This keeps state and various other
75
+ # information for each iteration, making it faster for successive lookups.
76
+ #
77
+ space.morph start, to: distances.each
78
+ #
79
+ # Parallel threads!
80
+ #
81
+ space.morph start, to: distances.each, threads: 4
82
+ #
83
+ # Of course this wouldn't be Ruby if we didn't use a block!
84
+ #
85
+ space.enter do |s|
86
+ morph start, to: distances.each
87
+ morph start, to: distances.each, threads: 4
88
+ # Change some parameter of the space
89
+ s.lowest = start.map {|x| MM::Ratio.from_s("8/1")}
90
+ # Etc. etc...
91
+ end
92
+
data/lib/mm/space.rb ADDED
@@ -0,0 +1,167 @@
1
+ module MM; end
2
+
3
+ class MM::Space
4
+ VERSION = "1.0.0"
5
+
6
+ attr_accessor :delta, :search_klass
7
+ attr_reader :max_distance, :metric, :boundaries
8
+ attr_writer :adjacent_points_function, :cost_function
9
+
10
+ # Initialization method for MM::Space
11
+ #
12
+ # metric - Array of MM::Metrics, where each metric corresponds to a dimension
13
+ # in the MM::Space.
14
+ # opts - Hash with additional parameters. (default: {})
15
+ # :delta - The delta of the MM::Search function used in #morph.
16
+ # (default: 0.001)
17
+ # :boundaries - Array of same size as metric containing pairs [low,
18
+ # high], which should be the bounding vectors of a given MM::Space.
19
+ # :adjacent_points_function - Proc to use as the
20
+ # adjacent_points_function for MM::Search in #morph.
21
+ # :cost_function - Proc to use for cost_function for MM::Search in
22
+ # #morph.
23
+ #
24
+ # Returns an MM::Space object
25
+ def initialize metric, opts = {}
26
+ @metric = metric
27
+ @search_klass = opts[:search_klass] || MM::Search
28
+ @delta = opts[:delta] || 0.001
29
+ @boundaries = opts[:boundaries]
30
+ @adjacent_points_function = opts[:adjacent_points_function]
31
+ @cost_function = opts[:cost_function]
32
+ end
33
+
34
+ # Morphs to a given point within the space
35
+ #
36
+ # start_morph - Enumerable object of things to morph from
37
+ # to - Array to morph to, with one element for each dimension
38
+ #
39
+ # Returns Array of resulting MM::Ratio objects
40
+ def morph start_morph, to: nil, current_point: nil
41
+ if current_point
42
+ # puts "Finding from #{current_point.map {|r| r.join ' '}}"
43
+ searcher(start_morph, to).find_from_point current_point
44
+ else
45
+ searcher(start_morph, to).find
46
+ end
47
+ end
48
+
49
+ def max_distance= d
50
+ if d.respond_to? :each
51
+ # Assign global maxes to each dimension
52
+ d.zip(@metric).each do |distance_and_metric|
53
+ distance_and_metric[1].scale = MM::Scaling.get_global(distance_and_metric[0])
54
+ end
55
+
56
+ @max_distance = d
57
+ elsif d.is_a? Numeric
58
+ # Wrap it in an Array so it can be zipped
59
+ self.max_distance = [d]
60
+ else
61
+ raise ArgumentError, "arg to max_distance= must respond_to? #zip or be Numeric"
62
+ end
63
+ end
64
+
65
+ def metric= m
66
+ if m.respond_to? :each
67
+ @metric = m
68
+ elsif m.respond_to? :call
69
+ # Wrap it in an Array so it can be zipped
70
+ self.metric = [m]
71
+ else
72
+ raise ArgumentError, "arg to metric= must respond_to? #each or #call"
73
+ end
74
+ end
75
+
76
+ # Default cost_function to use if no other one is specified. Takes the root of
77
+ # the sum of the squares, or the generalized Euclidean distance.
78
+ #
79
+ # start_morph - morph to begin the morph from. This should be a valid morph in
80
+ # the space (i.e., not out of bounds), and should also work with MM::Metric.
81
+ # to - Destination vector. There should be one dimension in the Array for each
82
+ # element in @metric
83
+ #
84
+ # Returns a Proc that calculates how much the current difference vector
85
+ # differs from the requested difference vector.
86
+ def cost_function start_morph, to
87
+ @cost_function ||
88
+ ->(current_point) {
89
+ @metric.zip(to).inject(0) {|memo, x|
90
+ distance = x[0].call(start_morph, current_point)
91
+ unless @boundaries.nil?
92
+ start_to_lowest = x[0].call(start_morph, @boundaries[0][0])
93
+ current_to_lowest = x[0].call(current_point, @boundaries[0][0])
94
+ if start_to_lowest > current_to_lowest
95
+ distance = distance * -1.0
96
+ end
97
+ end
98
+ memo = memo + (distance - x[1]).abs ** 2
99
+ } ** 0.5
100
+ }
101
+ end
102
+
103
+ # Default adjacent_points_function. It takes all repeated permutations of a
104
+ # given morph.
105
+ #
106
+ # Returns the adjacent_points_function Proc.
107
+ def adjacent_points_function
108
+ @adjacent_points_function ||
109
+ ->(current_point) {
110
+ current_point.repeated_permutation(current_point.size)
111
+ }
112
+ end
113
+
114
+ def boundaries= boundaries
115
+ @boundaries = boundaries
116
+ self.max_distance = boundaries.zip(@metric).map {|boundary_metric|
117
+ boundary_metric[1].call(*boundary_metric[0])
118
+ }
119
+ end
120
+
121
+ # Allows for morphing within a given block.
122
+ #
123
+ # locals - Hash of key-value pairs where the key is the name of the local
124
+ # variable to be created and the value is the value of that variable. Note
125
+ # that it actually creates *methods*, rather than *variables*, and that these
126
+ # methods refer to instance variables that are then removed. It could get
127
+ # buggy if you were to try to create a variable that was the same name as an
128
+ # existing method or class variable. (default: {})
129
+ # block - Block to evaluate within the context of the instance.
130
+ #
131
+ # Returns the last element returned by the block.
132
+ def enter locals = {}, &block
133
+ create_local_variables locals
134
+ output = instance_eval &block
135
+ remove_local_variables locals
136
+ output
137
+ end
138
+
139
+ protected
140
+
141
+ def searcher start_morph, to
142
+ search = @search_klass.new(start_morph)
143
+ search.cost_function = cost_function start_morph, to
144
+ search.adjacent_points_function = adjacent_points_function
145
+ search.delta = @delta
146
+ search
147
+ end
148
+
149
+ private
150
+
151
+ def create_local_variables locals
152
+ locals.each do |name, value|
153
+ define_singleton_method name do
154
+ value
155
+ end
156
+ end
157
+ end
158
+
159
+ def remove_local_variables locals
160
+ locals.each do |name, value|
161
+ self.singleton_class.class_eval do
162
+ remove_method name
163
+ end
164
+ end
165
+ end
166
+ end
167
+
@@ -0,0 +1,161 @@
1
+ require "minitest/autorun"
2
+ require "mm/space"
3
+ require "mm"
4
+
5
+ class TestMM < Minitest::Test; end
6
+
7
+ class TestMM::TestSpace < Minitest::Test
8
+ def setup
9
+ @x = MM::Metric.olm intra_delta: :tenney, inter_delta: :abs
10
+ @y = MM::Metric.olm intra_delta: :log_ratio, inter_delta: :abs
11
+ @space = MM::Space.new [@x, @y], delta: 0.5
12
+ end
13
+
14
+ # Testing the attribute methods
15
+ def test_max_distance_multi_dimensional
16
+ @space.max_distance = [8.414, 3.0]
17
+ assert_equal [8.414, 3.0], @space.max_distance
18
+ end
19
+
20
+ def test_max_distance_single_dimensional_sets_properly
21
+ @space.max_distance = 8.414
22
+ assert_equal [8.414], @space.max_distance
23
+ end
24
+
25
+ def test_set_and_get_metric_multi_dimensional
26
+ @space.metric = [@y, @x]
27
+ assert_equal MM::Deltas.singleton_method(:tenney),
28
+ @space.metric[1].instance_variable_get(:@intra_delta)
29
+ end
30
+
31
+ def test_metric_single_dimensional_sets_properly
32
+ @space.metric = @y
33
+ assert_equal MM::Deltas.singleton_method(:log_ratio),
34
+ @space.metric[0].instance_variable_get(:@intra_delta)
35
+ end
36
+
37
+ def test_metric_boundaries_sets_max_distance
38
+ metric = MM::Metric.olm intra_delta: :abs, inter_delta: :abs
39
+ @space.metric = [metric]
40
+ @space.boundaries = [[[0.0, 0.0], [0.0, 1.0]]]
41
+ assert_equal [1.0], @space.max_distance
42
+ end
43
+
44
+ # Testing that the cost function is the root of sum of squares
45
+ def test_cost_function
46
+ cost = @space.cost_function(start_morph, [0.4, 0.4])
47
+ assert_in_delta 0.5657, cost.call(start_morph)
48
+ end
49
+
50
+ # Testing the morphing methods
51
+ def test_morph_to_not_nil
52
+ refute_nil new_morph
53
+ end
54
+
55
+ def test_morph_same_length
56
+ assert_equal start_morph.length, new_morph.length
57
+ end
58
+
59
+ def test_morph_proper_distance_away
60
+ @space.max_distance = [4.907, 0.263]
61
+ @space.delta = 0.15
62
+ res = @space.morph start_morph, to: [0.4, 0.4]
63
+ assert_in_delta 0.5657, root_of_sum_of_squares([@x, @y], start_morph, res), 0.15
64
+ end
65
+
66
+ def test_morph_works_with_single_dimensions
67
+ @space.metric = @x
68
+ @space.max_distance = [4.907]
69
+ @space.delta = 0.25
70
+ @new_morph = @space.morph start_morph, to: [0.4]
71
+ x_distance = @x.call(start_morph, new_morph)
72
+ assert_in_delta 0.4, x_distance, 0.25
73
+ end
74
+
75
+ # TODO: Perhaps redesign this so that you would pass a "direction" vector as
76
+ # well? 0 could be "closer to lowest" and 1 could be "closer to highest".
77
+ def test_morph_gets_negative_distances
78
+ metric = MM::Metric.olm intra_delta: :abs, inter_delta: :abs
79
+ @space = MM::Space.new [metric], delta: 0.001
80
+ @space.boundaries = [[[0.0, 0.0], [0.0, 1.0]]]
81
+ @space.adjacent_points_function = one_tenth_adjacent_point
82
+ start_morph = [0.0, 0.5]
83
+
84
+ res = @space.morph start_morph, to: [-0.1]
85
+ lowest = @space.boundaries[0][0]
86
+
87
+ refute_nil res
88
+ assert metric.call(lowest, start_morph) > metric.call(lowest, res)
89
+ assert_in_delta 0.1, metric.call(start_morph, res), 0.001
90
+ end
91
+
92
+ # Testing that the whole block situation works
93
+ def test_morph_enter_finds_in_block
94
+ @space.max_distance = [4.907, 0.263]
95
+ @space.delta = 0.15
96
+ res = @space.enter :start => start_morph do
97
+ morph start, to: [0.4, 0.4]
98
+ end
99
+ assert_in_delta 0.5657, root_of_sum_of_squares([@x, @y], start_morph, res), 0.15
100
+ end
101
+
102
+ def test_morph_enter_passes_self_to_block
103
+ res = @space.enter :start => start_morph do |s|
104
+ s.max_distance = [4.907, 0.263]
105
+ s.delta = 0.15
106
+ morph start, to: [0.4, 0.4]
107
+ end
108
+ assert_in_delta 0.5657, root_of_sum_of_squares([@x, @y], start_morph, res), 0.15
109
+ end
110
+
111
+ def test_morph_enter_receives_local_variables
112
+ res = @space.enter :start => start_morph do
113
+ start
114
+ end
115
+ assert_equal start_morph, res
116
+ end
117
+
118
+ def test_morph_enter_cleans_up_local_variables
119
+ @space.enter :start => start_morph do
120
+ start
121
+ end
122
+ refute_respond_to @space, :start
123
+ end
124
+
125
+ # Some helper buddies
126
+ def start_morph
127
+ @start_morph ||= %w(1/1 5/4 3/2).map {|x| MM::Ratio.from_s(x)}
128
+ end
129
+
130
+ def start_morph= start_morph
131
+ @start_morph = start_morph
132
+ end
133
+
134
+ def new_morph
135
+ @new_morph ||= @space.morph start_morph, to: [0.4, 0.4]
136
+ end
137
+
138
+ def call_metrics metrics, start, result
139
+ metrics.map do |m|
140
+ m.call(start, result)
141
+ end
142
+ end
143
+
144
+ def root_of_sum_of_squares metrics, v1, v2
145
+ (call_metrics metrics, v1, v2).inject(0) {|m, d| m = m + d**2} ** 0.5
146
+ end
147
+
148
+ def one_tenth_adjacent_point
149
+ ->(current_point) {
150
+ [-0.1, 0.0, 0.1].repeated_permutation(current_point.size).map {|v|
151
+ vz = v.zip(current_point).map {|vp|
152
+ vp = vp.inject(0.0, :+)
153
+ }
154
+ vz.any? {|p| p < 0.0 || p > 1.0 } ? nil : vz
155
+ }.compact
156
+ }
157
+ end
158
+
159
+ attr_writer :new_morph
160
+ end
161
+
metadata ADDED
@@ -0,0 +1,106 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: morphological_metrics-space
3
+ version: !ruby/object:Gem::Version
4
+ version: 1.0.0
5
+ platform: ruby
6
+ authors:
7
+ - Andrew Smith
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2017-10-29 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: minitest
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '5.10'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '5.10'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rdoc
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '4.0'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '4.0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: hoe
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '3.16'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '3.16'
55
+ description: |-
56
+ MM::Space is a framework for working with Morphological Metrics and
57
+ Morphological Mutations in Ruby. A core component of MM::Space is that it has a
58
+ notion of global distance throughout a series of measurements or
59
+ transformations. This is what "Space" implies. It uses coworker libraries
60
+ MM::Metric and MM::Mutation to drive these measurements and transformations.
61
+ email:
62
+ - andrewchristophersmith@gmail.com
63
+ executables:
64
+ - mm_space
65
+ extensions: []
66
+ extra_rdoc_files:
67
+ - History.txt
68
+ - Manifest.txt
69
+ - README.rdoc
70
+ files:
71
+ - ".autotest"
72
+ - History.txt
73
+ - Manifest.txt
74
+ - README.rdoc
75
+ - Rakefile
76
+ - bin/mm_space
77
+ - lib/mm/space.rb
78
+ - test/mm/test_space.rb
79
+ homepage: https://github.com/andrewcsmith/mm-space
80
+ licenses:
81
+ - MIT
82
+ metadata: {}
83
+ post_install_message:
84
+ rdoc_options:
85
+ - "--main"
86
+ - README.rdoc
87
+ require_paths:
88
+ - lib
89
+ required_ruby_version: !ruby/object:Gem::Requirement
90
+ requirements:
91
+ - - ">="
92
+ - !ruby/object:Gem::Version
93
+ version: '0'
94
+ required_rubygems_version: !ruby/object:Gem::Requirement
95
+ requirements:
96
+ - - ">="
97
+ - !ruby/object:Gem::Version
98
+ version: '1.4'
99
+ requirements: []
100
+ rubyforge_project:
101
+ rubygems_version: 2.6.13
102
+ signing_key:
103
+ specification_version: 4
104
+ summary: MM::Space is a framework for working with Morphological Metrics and Morphological
105
+ Mutations in Ruby
106
+ test_files: []