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 +7 -0
- data/.autotest +25 -0
- data/History.txt +6 -0
- data/Manifest.txt +8 -0
- data/README.rdoc +76 -0
- data/Rakefile +24 -0
- data/bin/mm_space +92 -0
- data/lib/mm/space.rb +167 -0
- data/test/mm/test_space.rb +161 -0
- metadata +106 -0
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
data/Manifest.txt
ADDED
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: []
|