musa-dsl 0.14.16
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +10 -0
- data/Gemfile +20 -0
- data/LICENSE.md +157 -0
- data/README.md +8 -0
- data/lib/musa-dsl/core-ext/array-apply-get.rb +18 -0
- data/lib/musa-dsl/core-ext/array-explode-ranges.rb +29 -0
- data/lib/musa-dsl/core-ext/array-to-neumas.rb +28 -0
- data/lib/musa-dsl/core-ext/array-to-serie.rb +20 -0
- data/lib/musa-dsl/core-ext/arrayfy.rb +15 -0
- data/lib/musa-dsl/core-ext/as-context-run.rb +44 -0
- data/lib/musa-dsl/core-ext/duplicate.rb +134 -0
- data/lib/musa-dsl/core-ext/dynamic-proxy.rb +55 -0
- data/lib/musa-dsl/core-ext/inspect-nice.rb +28 -0
- data/lib/musa-dsl/core-ext/key-parameters-procedure-binder.rb +85 -0
- data/lib/musa-dsl/core-ext/proc-nice.rb +13 -0
- data/lib/musa-dsl/core-ext/send-nice.rb +21 -0
- data/lib/musa-dsl/core-ext/string-to-neumas.rb +27 -0
- data/lib/musa-dsl/core-ext.rb +13 -0
- data/lib/musa-dsl/datasets/gdv-decorators.rb +221 -0
- data/lib/musa-dsl/datasets/gdv.rb +499 -0
- data/lib/musa-dsl/datasets/pdv.rb +44 -0
- data/lib/musa-dsl/datasets.rb +5 -0
- data/lib/musa-dsl/generative/darwin.rb +145 -0
- data/lib/musa-dsl/generative/generative-grammar.rb +294 -0
- data/lib/musa-dsl/generative/markov.rb +78 -0
- data/lib/musa-dsl/generative/rules.rb +282 -0
- data/lib/musa-dsl/generative/variatio.rb +331 -0
- data/lib/musa-dsl/generative.rb +5 -0
- data/lib/musa-dsl/midi/midi-recorder.rb +83 -0
- data/lib/musa-dsl/midi/midi-voices.rb +274 -0
- data/lib/musa-dsl/midi.rb +2 -0
- data/lib/musa-dsl/music/chord-definition.rb +99 -0
- data/lib/musa-dsl/music/chord-definitions.rb +13 -0
- data/lib/musa-dsl/music/chords.rb +326 -0
- data/lib/musa-dsl/music/equally-tempered-12-tone-scale-system.rb +204 -0
- data/lib/musa-dsl/music/scales.rb +584 -0
- data/lib/musa-dsl/music.rb +6 -0
- data/lib/musa-dsl/neuma/neuma.rb +181 -0
- data/lib/musa-dsl/neuma.rb +1 -0
- data/lib/musa-dsl/neumalang/neumalang.citrus +294 -0
- data/lib/musa-dsl/neumalang/neumalang.rb +179 -0
- data/lib/musa-dsl/neumalang.rb +3 -0
- data/lib/musa-dsl/repl/repl.rb +143 -0
- data/lib/musa-dsl/repl.rb +1 -0
- data/lib/musa-dsl/sequencer/base-sequencer-implementation-control.rb +189 -0
- data/lib/musa-dsl/sequencer/base-sequencer-implementation-play-helper.rb +354 -0
- data/lib/musa-dsl/sequencer/base-sequencer-implementation.rb +382 -0
- data/lib/musa-dsl/sequencer/base-sequencer-public.rb +261 -0
- data/lib/musa-dsl/sequencer/sequencer-dsl.rb +94 -0
- data/lib/musa-dsl/sequencer/sequencer.rb +3 -0
- data/lib/musa-dsl/sequencer.rb +1 -0
- data/lib/musa-dsl/series/base-series.rb +245 -0
- data/lib/musa-dsl/series/hash-serie-splitter.rb +194 -0
- data/lib/musa-dsl/series/holder-serie.rb +87 -0
- data/lib/musa-dsl/series/main-serie-constructors.rb +726 -0
- data/lib/musa-dsl/series/main-serie-operations.rb +1151 -0
- data/lib/musa-dsl/series/proxy-serie.rb +69 -0
- data/lib/musa-dsl/series/queue-serie.rb +94 -0
- data/lib/musa-dsl/series/series.rb +8 -0
- data/lib/musa-dsl/series.rb +1 -0
- data/lib/musa-dsl/transport/clock.rb +36 -0
- data/lib/musa-dsl/transport/dummy-clock.rb +47 -0
- data/lib/musa-dsl/transport/external-tick-clock.rb +31 -0
- data/lib/musa-dsl/transport/input-midi-clock.rb +124 -0
- data/lib/musa-dsl/transport/timer-clock.rb +102 -0
- data/lib/musa-dsl/transport/timer.rb +40 -0
- data/lib/musa-dsl/transport/transport.rb +137 -0
- data/lib/musa-dsl/transport.rb +9 -0
- data/lib/musa-dsl.rb +17 -0
- data/musa-dsl.gemspec +17 -0
- metadata +174 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 233adb825ed72badf41723781c9ca578b1ba7a38f01191376207fa44d0bfbe84
|
4
|
+
data.tar.gz: cb6e9d1557645c9cef7bc1e963438c54c7d9c0ffb3b8dac8cc6f45619fdd1616
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: dc451b441bfdf0a9814f2a4cdd88fb3a991ec1d5edd35eeed09fbe72735d8615da33645995d35a842c392d0810db8b9d166d1c8eae43c0c000687bff0824a989
|
7
|
+
data.tar.gz: d094f68c240f775035cbe4dc2aeed3086368826e99af91cbe9e68a3ef9f19c3dd95f328f71d63e97b457ff7f647b1c02d750793dc8bbcf3c1dfef842cb59cc01
|
data/.gitignore
ADDED
data/Gemfile
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
source 'https://rubygems.org'
|
2
|
+
|
3
|
+
group :neuma do
|
4
|
+
gem 'citrus', '~> 3.0.0'
|
5
|
+
|
6
|
+
end
|
7
|
+
|
8
|
+
group :transport do
|
9
|
+
gem 'midi-message', '~> 0.4', '>= 0.4.9'
|
10
|
+
gem 'midi-nibbler', '~> 0.2', '>= 0.2.4'
|
11
|
+
end
|
12
|
+
|
13
|
+
group :test do
|
14
|
+
gem 'rspec', '~> 3.0'
|
15
|
+
gem 'descriptive-statistics'
|
16
|
+
end
|
17
|
+
|
18
|
+
group :documentation do
|
19
|
+
gem 'rdoc'
|
20
|
+
end
|
data/LICENSE.md
ADDED
@@ -0,0 +1,157 @@
|
|
1
|
+
### GNU LESSER GENERAL PUBLIC LICENSE
|
2
|
+
|
3
|
+
Version 3, 29 June 2007
|
4
|
+
|
5
|
+
Copyright (C) 2007 Free Software Foundation, Inc.
|
6
|
+
<https://fsf.org/>
|
7
|
+
|
8
|
+
Everyone is permitted to copy and distribute verbatim copies of this
|
9
|
+
license document, but changing it is not allowed.
|
10
|
+
|
11
|
+
This version of the GNU Lesser General Public License incorporates the
|
12
|
+
terms and conditions of version 3 of the GNU General Public License,
|
13
|
+
supplemented by the additional permissions listed below.
|
14
|
+
|
15
|
+
#### 0. Additional Definitions.
|
16
|
+
|
17
|
+
As used herein, "this License" refers to version 3 of the GNU Lesser
|
18
|
+
General Public License, and the "GNU GPL" refers to version 3 of the
|
19
|
+
GNU General Public License.
|
20
|
+
|
21
|
+
"The Library" refers to a covered work governed by this License, other
|
22
|
+
than an Application or a Combined Work as defined below.
|
23
|
+
|
24
|
+
An "Application" is any work that makes use of an interface provided
|
25
|
+
by the Library, but which is not otherwise based on the Library.
|
26
|
+
Defining a subclass of a class defined by the Library is deemed a mode
|
27
|
+
of using an interface provided by the Library.
|
28
|
+
|
29
|
+
A "Combined Work" is a work produced by combining or linking an
|
30
|
+
Application with the Library. The particular version of the Library
|
31
|
+
with which the Combined Work was made is also called the "Linked
|
32
|
+
Version".
|
33
|
+
|
34
|
+
The "Minimal Corresponding Source" for a Combined Work means the
|
35
|
+
Corresponding Source for the Combined Work, excluding any source code
|
36
|
+
for portions of the Combined Work that, considered in isolation, are
|
37
|
+
based on the Application, and not on the Linked Version.
|
38
|
+
|
39
|
+
The "Corresponding Application Code" for a Combined Work means the
|
40
|
+
object code and/or source code for the Application, including any data
|
41
|
+
and utility programs needed for reproducing the Combined Work from the
|
42
|
+
Application, but excluding the System Libraries of the Combined Work.
|
43
|
+
|
44
|
+
#### 1. Exception to Section 3 of the GNU GPL.
|
45
|
+
|
46
|
+
You may convey a covered work under sections 3 and 4 of this License
|
47
|
+
without being bound by section 3 of the GNU GPL.
|
48
|
+
|
49
|
+
#### 2. Conveying Modified Versions.
|
50
|
+
|
51
|
+
If you modify a copy of the Library, and, in your modifications, a
|
52
|
+
facility refers to a function or data to be supplied by an Application
|
53
|
+
that uses the facility (other than as an argument passed when the
|
54
|
+
facility is invoked), then you may convey a copy of the modified
|
55
|
+
version:
|
56
|
+
|
57
|
+
- a) under this License, provided that you make a good faith effort
|
58
|
+
to ensure that, in the event an Application does not supply the
|
59
|
+
function or data, the facility still operates, and performs
|
60
|
+
whatever part of its purpose remains meaningful, or
|
61
|
+
- b) under the GNU GPL, with none of the additional permissions of
|
62
|
+
this License applicable to that copy.
|
63
|
+
|
64
|
+
#### 3. Object Code Incorporating Material from Library Header Files.
|
65
|
+
|
66
|
+
The object code form of an Application may incorporate material from a
|
67
|
+
header file that is part of the Library. You may convey such object
|
68
|
+
code under terms of your choice, provided that, if the incorporated
|
69
|
+
material is not limited to numerical parameters, data structure
|
70
|
+
layouts and accessors, or small macros, inline functions and templates
|
71
|
+
(ten or fewer lines in length), you do both of the following:
|
72
|
+
|
73
|
+
- a) Give prominent notice with each copy of the object code that
|
74
|
+
the Library is used in it and that the Library and its use are
|
75
|
+
covered by this License.
|
76
|
+
- b) Accompany the object code with a copy of the GNU GPL and this
|
77
|
+
license document.
|
78
|
+
|
79
|
+
#### 4. Combined Works.
|
80
|
+
|
81
|
+
You may convey a Combined Work under terms of your choice that, taken
|
82
|
+
together, effectively do not restrict modification of the portions of
|
83
|
+
the Library contained in the Combined Work and reverse engineering for
|
84
|
+
debugging such modifications, if you also do each of the following:
|
85
|
+
|
86
|
+
- a) Give prominent notice with each copy of the Combined Work that
|
87
|
+
the Library is used in it and that the Library and its use are
|
88
|
+
covered by this License.
|
89
|
+
- b) Accompany the Combined Work with a copy of the GNU GPL and this
|
90
|
+
license document.
|
91
|
+
- c) For a Combined Work that displays copyright notices during
|
92
|
+
execution, include the copyright notice for the Library among
|
93
|
+
these notices, as well as a reference directing the user to the
|
94
|
+
copies of the GNU GPL and this license document.
|
95
|
+
- d) Do one of the following:
|
96
|
+
- 0) Convey the Minimal Corresponding Source under the terms of
|
97
|
+
this License, and the Corresponding Application Code in a form
|
98
|
+
suitable for, and under terms that permit, the user to
|
99
|
+
recombine or relink the Application with a modified version of
|
100
|
+
the Linked Version to produce a modified Combined Work, in the
|
101
|
+
manner specified by section 6 of the GNU GPL for conveying
|
102
|
+
Corresponding Source.
|
103
|
+
- 1) Use a suitable shared library mechanism for linking with
|
104
|
+
the Library. A suitable mechanism is one that (a) uses at run
|
105
|
+
time a copy of the Library already present on the user's
|
106
|
+
computer system, and (b) will operate properly with a modified
|
107
|
+
version of the Library that is interface-compatible with the
|
108
|
+
Linked Version.
|
109
|
+
- e) Provide Installation Information, but only if you would
|
110
|
+
otherwise be required to provide such information under section 6
|
111
|
+
of the GNU GPL, and only to the extent that such information is
|
112
|
+
necessary to install and execute a modified version of the
|
113
|
+
Combined Work produced by recombining or relinking the Application
|
114
|
+
with a modified version of the Linked Version. (If you use option
|
115
|
+
4d0, the Installation Information must accompany the Minimal
|
116
|
+
Corresponding Source and Corresponding Application Code. If you
|
117
|
+
use option 4d1, you must provide the Installation Information in
|
118
|
+
the manner specified by section 6 of the GNU GPL for conveying
|
119
|
+
Corresponding Source.)
|
120
|
+
|
121
|
+
#### 5. Combined Libraries.
|
122
|
+
|
123
|
+
You may place library facilities that are a work based on the Library
|
124
|
+
side by side in a single library together with other library
|
125
|
+
facilities that are not Applications and are not covered by this
|
126
|
+
License, and convey such a combined library under terms of your
|
127
|
+
choice, if you do both of the following:
|
128
|
+
|
129
|
+
- a) Accompany the combined library with a copy of the same work
|
130
|
+
based on the Library, uncombined with any other library
|
131
|
+
facilities, conveyed under the terms of this License.
|
132
|
+
- b) Give prominent notice with the combined library that part of it
|
133
|
+
is a work based on the Library, and explaining where to find the
|
134
|
+
accompanying uncombined form of the same work.
|
135
|
+
|
136
|
+
#### 6. Revised Versions of the GNU Lesser General Public License.
|
137
|
+
|
138
|
+
The Free Software Foundation may publish revised and/or new versions
|
139
|
+
of the GNU Lesser General Public License from time to time. Such new
|
140
|
+
versions will be similar in spirit to the present version, but may
|
141
|
+
differ in detail to address new problems or concerns.
|
142
|
+
|
143
|
+
Each version is given a distinguishing version number. If the Library
|
144
|
+
as you received it specifies that a certain numbered version of the
|
145
|
+
GNU Lesser General Public License "or any later version" applies to
|
146
|
+
it, you have the option of following the terms and conditions either
|
147
|
+
of that published version or of any later version published by the
|
148
|
+
Free Software Foundation. If the Library as you received it does not
|
149
|
+
specify a version number of the GNU Lesser General Public License, you
|
150
|
+
may choose any version of the GNU Lesser General Public License ever
|
151
|
+
published by the Free Software Foundation.
|
152
|
+
|
153
|
+
If the Library as you received it specifies that a proxy can decide
|
154
|
+
whether future versions of the GNU Lesser General Public License shall
|
155
|
+
apply, that proxy's public statement of acceptance of any version is
|
156
|
+
permanent authorization for you to choose that version for the
|
157
|
+
Library.
|
data/README.md
ADDED
@@ -0,0 +1,8 @@
|
|
1
|
+
# Musa-DSL
|
2
|
+
|
3
|
+
Work in progress.
|
4
|
+
|
5
|
+
A programming language DSL based on Ruby for musical composition.
|
6
|
+
Emphasizes the creation of complex temporal structures independently of the audio rendering engine.
|
7
|
+
|
8
|
+
Some works can be listened on [yeste.studio](https://soundcloud.com/yeste-studio) Soundcloud.
|
@@ -0,0 +1,18 @@
|
|
1
|
+
# TODO hacer que *_nice permitar recibir atributos para indicar cómo se quieren procesar los parámetros (haciendo *, **, o sin hacer nada)
|
2
|
+
|
3
|
+
class Array
|
4
|
+
def apply method_name, source
|
5
|
+
|
6
|
+
source = [source] unless source.is_a? Array
|
7
|
+
|
8
|
+
self.each_with_index do |o, i|
|
9
|
+
o.send method_name, source[i % source.length]
|
10
|
+
end
|
11
|
+
end
|
12
|
+
|
13
|
+
def get method_name
|
14
|
+
self.collect { |o| o.send method_name }
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
|
@@ -0,0 +1,29 @@
|
|
1
|
+
class Array
|
2
|
+
def arrayfy
|
3
|
+
self
|
4
|
+
end
|
5
|
+
|
6
|
+
def repeat_to_size(new_size)
|
7
|
+
pos = -1
|
8
|
+
new_size -= 1
|
9
|
+
|
10
|
+
new_array = clone
|
11
|
+
new_array << self[(pos += 1) % size] while (pos + size) < new_size
|
12
|
+
|
13
|
+
new_array
|
14
|
+
end
|
15
|
+
|
16
|
+
def explode_ranges
|
17
|
+
array = []
|
18
|
+
|
19
|
+
each do |element|
|
20
|
+
if element.is_a? Range
|
21
|
+
element.to_a.each { |element| array << element }
|
22
|
+
else
|
23
|
+
array << element
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
array
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
require 'musa-dsl/series'
|
2
|
+
require 'musa-dsl/neumalang'
|
3
|
+
|
4
|
+
class Array
|
5
|
+
def to_neumas
|
6
|
+
if length > 1
|
7
|
+
MERGE(*collect { |e| convert_to_neumas(e) })
|
8
|
+
else
|
9
|
+
convert_to_neumas(first)
|
10
|
+
end
|
11
|
+
end
|
12
|
+
|
13
|
+
alias_method :neumas, :to_neumas
|
14
|
+
alias_method :n, :to_neumas
|
15
|
+
|
16
|
+
private
|
17
|
+
|
18
|
+
def convert_to_neumas(e)
|
19
|
+
case e
|
20
|
+
when Musa::Neumalang::Neumas then e
|
21
|
+
when Musa::Neumalang::Neuma::Parallel then _SE([e], extends: Musa::Neumalang::Neumas)
|
22
|
+
when String then e.to_neumas
|
23
|
+
else
|
24
|
+
raise ArgumentError, "Don't know how to convert to neumas #{e}"
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
@@ -0,0 +1,20 @@
|
|
1
|
+
require 'musa-dsl/series'
|
2
|
+
|
3
|
+
class Array
|
4
|
+
def to_serie(of_series: nil, recursive: nil)
|
5
|
+
of_series ||= false
|
6
|
+
recursive ||= false
|
7
|
+
|
8
|
+
raise ArgumentError, 'Cannot convert to serie of_series and recursive simultaneously' if recursive && of_series
|
9
|
+
|
10
|
+
if recursive
|
11
|
+
S(*(collect { |_| _.is_a?(Array) ? _.to_serie(recursive: true) : _ }))
|
12
|
+
elsif of_series
|
13
|
+
S(*(collect { |_| S(*_) }))
|
14
|
+
else
|
15
|
+
S(*self)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
alias_method :s, :to_serie
|
20
|
+
end
|
@@ -0,0 +1,44 @@
|
|
1
|
+
# TODO: hacer que *_nice permitar recibir atributos para indicar cómo se quieren procesar los parámetros (haciendo *, **, o sin hacer nada)
|
2
|
+
|
3
|
+
class Object
|
4
|
+
def as_context_run(procedure, *list_or_key_args, **key_args)
|
5
|
+
_as_context_run procedure, list_or_key_args, key_args
|
6
|
+
end
|
7
|
+
|
8
|
+
def _as_context_run(procedure, list_or_key_args = nil, key_args = nil)
|
9
|
+
if !list_or_key_args.nil? && list_or_key_args.is_a?(Hash)
|
10
|
+
key_args = list_or_key_args
|
11
|
+
list_or_key_args = nil
|
12
|
+
end
|
13
|
+
|
14
|
+
if procedure.lambda?
|
15
|
+
if !list_or_key_args.nil? && !list_or_key_args.empty?
|
16
|
+
if !key_args.nil? && !key_args.empty?
|
17
|
+
procedure.call *list_or_key_args, **key_args
|
18
|
+
else
|
19
|
+
procedure.call *list_or_key_args
|
20
|
+
end
|
21
|
+
else
|
22
|
+
if !key_args.nil? && !key_args.empty?
|
23
|
+
procedure.call **key_args
|
24
|
+
else
|
25
|
+
procedure.call
|
26
|
+
end
|
27
|
+
end
|
28
|
+
else
|
29
|
+
if !list_or_key_args.nil? && !list_or_key_args.empty?
|
30
|
+
if !key_args.nil? && !key_args.empty?
|
31
|
+
instance_exec *list_or_key_args, **key_args, &procedure
|
32
|
+
else
|
33
|
+
instance_exec *list_or_key_args, &procedure
|
34
|
+
end
|
35
|
+
else
|
36
|
+
if !key_args.nil? && !key_args.empty?
|
37
|
+
instance_exec **key_args, &procedure
|
38
|
+
else
|
39
|
+
instance_eval &procedure
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -0,0 +1,134 @@
|
|
1
|
+
# Based on https://github.com/adamluzsi/duplicate.rb/blob/master/lib/duplicate.rb
|
2
|
+
# Modifications by JSY
|
3
|
+
|
4
|
+
module Duplicate
|
5
|
+
extend self
|
6
|
+
|
7
|
+
def duplicate(object)
|
8
|
+
register = {}
|
9
|
+
|
10
|
+
_dup(register, object)
|
11
|
+
end
|
12
|
+
|
13
|
+
protected
|
14
|
+
|
15
|
+
def registered(object, register)
|
16
|
+
register[object.__id__]
|
17
|
+
end
|
18
|
+
|
19
|
+
def register_duplication(register, object, duplicate)
|
20
|
+
register[object.__id__] = duplicate
|
21
|
+
duplicate
|
22
|
+
end
|
23
|
+
|
24
|
+
def _dup(register, object)
|
25
|
+
return registered(object, register) if registered(object, register)
|
26
|
+
return register_duplication(register, object, object) unless identifiable?(object)
|
27
|
+
|
28
|
+
case object
|
29
|
+
|
30
|
+
when Array
|
31
|
+
dup_array(register, object)
|
32
|
+
|
33
|
+
when Hash
|
34
|
+
dup_hash(register, object)
|
35
|
+
|
36
|
+
when Range
|
37
|
+
dup_range(register, object)
|
38
|
+
|
39
|
+
when Struct
|
40
|
+
dup_struct(register, object)
|
41
|
+
|
42
|
+
when NilClass, Symbol, Numeric, TrueClass, FalseClass, Method
|
43
|
+
register_duplication(register, object, object)
|
44
|
+
|
45
|
+
else
|
46
|
+
dup_object(register, object)
|
47
|
+
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
def identifiable?(object)
|
52
|
+
object.class && object.respond_to?(:is_a?)
|
53
|
+
rescue NoMethodError
|
54
|
+
false
|
55
|
+
end
|
56
|
+
|
57
|
+
def dup_array(register, object)
|
58
|
+
duplication = dup_object(register, object)
|
59
|
+
duplication.map! { |e| _dup(register, e) }
|
60
|
+
end
|
61
|
+
|
62
|
+
def dup_hash(register, object)
|
63
|
+
duplication = dup_object(register, object)
|
64
|
+
object.reduce(duplication) { |hash, (k, v)| hash.merge!(_dup(register, k) => _dup(register, v)) }
|
65
|
+
end
|
66
|
+
|
67
|
+
def dup_range(register, range)
|
68
|
+
register_duplication(register, range, range.class.new(_dup(register, range.first), _dup(register, range.last)))
|
69
|
+
rescue StandardError
|
70
|
+
register_duplication(register, range, range.dup)
|
71
|
+
end
|
72
|
+
|
73
|
+
def dup_struct(register, struct)
|
74
|
+
duplication = register_duplication(register, struct, struct.dup)
|
75
|
+
|
76
|
+
struct.each_pair do |attr, value|
|
77
|
+
duplication.__send__("#{attr}=", _dup(register, value))
|
78
|
+
end
|
79
|
+
|
80
|
+
duplication
|
81
|
+
end
|
82
|
+
|
83
|
+
def dup_object(register, object)
|
84
|
+
dup_instance_variables(register, object, register_duplication(register, object, try_dup(object)))
|
85
|
+
end
|
86
|
+
|
87
|
+
def dup_instance_variables(register, object, duplication)
|
88
|
+
return duplication unless respond_to_instance_variables?(object)
|
89
|
+
|
90
|
+
object.instance_variables.each do |instance_variable|
|
91
|
+
value = get_instance_variable(object, instance_variable)
|
92
|
+
|
93
|
+
set_instance_variable(duplication, instance_variable, _dup(register, value))
|
94
|
+
end
|
95
|
+
|
96
|
+
duplication
|
97
|
+
end
|
98
|
+
|
99
|
+
def get_instance_variable(object, instance_variable_name)
|
100
|
+
object.instance_variable_get(instance_variable_name)
|
101
|
+
rescue NoMethodError
|
102
|
+
object.instance_eval(instance_variable_name.to_s)
|
103
|
+
end
|
104
|
+
|
105
|
+
def set_instance_variable(duplicate, instance_variable_name, value_to_set)
|
106
|
+
duplicate.instance_variable_set(instance_variable_name, value_to_set)
|
107
|
+
rescue NoMethodError
|
108
|
+
duplicate.instance_eval("#{instance_variable_name} = Marshal.load(#{Marshal.dump(value_to_set).inspect})")
|
109
|
+
end
|
110
|
+
|
111
|
+
def try_dup(object)
|
112
|
+
o = object.dup
|
113
|
+
|
114
|
+
o.tap do
|
115
|
+
object.singleton_class.included_modules.each do |m|
|
116
|
+
o.extend m unless o.is_a? m
|
117
|
+
end
|
118
|
+
end
|
119
|
+
rescue NoMethodError, TypeError
|
120
|
+
object
|
121
|
+
end
|
122
|
+
|
123
|
+
def respond_to_instance_variables?(object)
|
124
|
+
object.respond_to?(:instance_variables) && object.instance_variables.is_a?(Array)
|
125
|
+
rescue NoMethodError
|
126
|
+
false
|
127
|
+
end
|
128
|
+
end
|
129
|
+
|
130
|
+
class Object
|
131
|
+
def duplicate
|
132
|
+
Duplicate.duplicate(self)
|
133
|
+
end
|
134
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
require_relative 'send-nice'
|
2
|
+
|
3
|
+
module DynamicProxyModule
|
4
|
+
def method_missing(method_name, *args, **key_args, &block)
|
5
|
+
if @receiver.respond_to? method_name
|
6
|
+
@receiver._send_nice method_name, args, key_args, &block
|
7
|
+
else
|
8
|
+
super
|
9
|
+
end
|
10
|
+
end
|
11
|
+
|
12
|
+
def respond_to_missing?(method_name, include_private)
|
13
|
+
@receiver.respond_to?(method_name, include_private) || super
|
14
|
+
end
|
15
|
+
|
16
|
+
alias _is_a? is_a?
|
17
|
+
|
18
|
+
def is_a?(klass)
|
19
|
+
_is_a?(klass) || @receiver.is_a?(klass)
|
20
|
+
end
|
21
|
+
|
22
|
+
alias _kind_of? kind_of?
|
23
|
+
|
24
|
+
def kind_of?(klass)
|
25
|
+
_kind_of?(klass) || @receiver.is_a?(klass)
|
26
|
+
end
|
27
|
+
|
28
|
+
alias _instance_of? instance_of?
|
29
|
+
|
30
|
+
def instance_of?(klass)
|
31
|
+
_instance_of?(klass) || @receiver.instance_of?(klass)
|
32
|
+
end
|
33
|
+
|
34
|
+
alias _equalequal ==
|
35
|
+
|
36
|
+
def ==(object)
|
37
|
+
_equalequal(object) || @receiver.==(object)
|
38
|
+
end
|
39
|
+
|
40
|
+
alias _eql? eql?
|
41
|
+
|
42
|
+
def eql?(object)
|
43
|
+
_eql?(object) || @receiver.eql?(object)
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
class DynamicProxy
|
48
|
+
include DynamicProxyModule
|
49
|
+
|
50
|
+
def initialize(receiver = nil)
|
51
|
+
@receiver = receiver
|
52
|
+
end
|
53
|
+
|
54
|
+
attr_accessor :receiver
|
55
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
class Hash
|
2
|
+
def inspect
|
3
|
+
all = collect { |key, value| [', ', key.is_a?(Symbol) ? key.to_s + ': ' : key.to_s + ' => ', value.inspect] }.flatten
|
4
|
+
all.shift
|
5
|
+
'{ ' + all.join + ' }'
|
6
|
+
end
|
7
|
+
end
|
8
|
+
|
9
|
+
class Rational
|
10
|
+
def inspect
|
11
|
+
d = self - to_i
|
12
|
+
if d != 0
|
13
|
+
"#{to_i}(#{d.numerator}/#{d.denominator})"
|
14
|
+
else
|
15
|
+
to_i.to_s
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
alias _to_s to_s
|
20
|
+
|
21
|
+
def to_s
|
22
|
+
if to_i == self
|
23
|
+
to_i.to_s
|
24
|
+
else
|
25
|
+
_to_s
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,85 @@
|
|
1
|
+
class KeyParametersProcedureBinder
|
2
|
+
attr_reader :procedure
|
3
|
+
|
4
|
+
def initialize(procedure, on_rescue: nil)
|
5
|
+
@procedure = procedure
|
6
|
+
@on_rescue = on_rescue
|
7
|
+
|
8
|
+
@parameters = {}
|
9
|
+
@has_rest = false
|
10
|
+
@value_parameters_count = 0
|
11
|
+
|
12
|
+
procedure.parameters.each do |parameter|
|
13
|
+
@parameters[parameter[1]] = nil if parameter[0] == :key || parameter[0] == :keyreq
|
14
|
+
@has_rest = true if parameter[0] == :keyrest
|
15
|
+
|
16
|
+
@value_parameters_count += 1 if parameter[0] == :req || parameter[0] == :opt
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
def call(*value_parameters, **key_parameters)
|
21
|
+
_call value_parameters, key_parameters
|
22
|
+
end
|
23
|
+
|
24
|
+
def _call(value_parameters, key_parameters)
|
25
|
+
if @on_rescue
|
26
|
+
begin
|
27
|
+
__call value_parameters, key_parameters
|
28
|
+
rescue StandardError, ScriptError => e
|
29
|
+
@on_rescue.call e
|
30
|
+
end
|
31
|
+
else
|
32
|
+
__call value_parameters, key_parameters
|
33
|
+
end
|
34
|
+
end
|
35
|
+
|
36
|
+
def __call(value_parameters, key_parameters)
|
37
|
+
effective_key_parameters = apply(key_parameters)
|
38
|
+
|
39
|
+
if effective_key_parameters.empty?
|
40
|
+
if value_parameters.nil? || value_parameters.empty? || @value_parameters_count == 0
|
41
|
+
@procedure.call
|
42
|
+
else
|
43
|
+
@procedure.call *value_parameters.first(@value_parameters_count)
|
44
|
+
end
|
45
|
+
else
|
46
|
+
if value_parameters.nil? || value_parameters.empty?
|
47
|
+
@procedure.call **effective_key_parameters
|
48
|
+
else
|
49
|
+
@procedure.call *value_parameters, **effective_key_parameters
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
53
|
+
|
54
|
+
private :__call
|
55
|
+
|
56
|
+
def key?(key)
|
57
|
+
@has_rest || @parameters.include?(key)
|
58
|
+
end
|
59
|
+
|
60
|
+
alias_method :has_key?, :key?
|
61
|
+
|
62
|
+
def apply(hsh)
|
63
|
+
hsh ||= {}
|
64
|
+
|
65
|
+
result = @parameters.clone
|
66
|
+
|
67
|
+
@parameters.each_key do |parameter_name|
|
68
|
+
result[parameter_name] = hsh[parameter_name]
|
69
|
+
end
|
70
|
+
|
71
|
+
if @has_rest
|
72
|
+
hsh.each do |key, value|
|
73
|
+
result[key] = value unless result.key?(key)
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
result
|
78
|
+
end
|
79
|
+
|
80
|
+
def inspect
|
81
|
+
"KeyParametersProcedureBinder: parameters = #{@parameters} has_rest = #{@has_rest}"
|
82
|
+
end
|
83
|
+
|
84
|
+
alias to_s inspect
|
85
|
+
end
|