rotoscope 0.2.1
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/.clang-format +2 -0
- data/.github/PULL_REQUEST_TEMPLATE +3 -0
- data/.gitignore +7 -0
- data/.rubocop.yml +15 -0
- data/.ruby-version +1 -0
- data/Gemfile +3 -0
- data/LICENSE +22 -0
- data/README.md +215 -0
- data/Rakefile +36 -0
- data/bin/fmt +9 -0
- data/dev.yml +12 -0
- data/ext/rotoscope/callsite.c +78 -0
- data/ext/rotoscope/callsite.h +17 -0
- data/ext/rotoscope/extconf.rb +10 -0
- data/ext/rotoscope/rotoscope.c +391 -0
- data/ext/rotoscope/rotoscope.h +65 -0
- data/ext/rotoscope/stack.c +92 -0
- data/ext/rotoscope/stack.h +29 -0
- data/ext/rotoscope/strmemo.c +33 -0
- data/ext/rotoscope/strmemo.h +14 -0
- data/ext/rotoscope/tracepoint.c +8 -0
- data/ext/rotoscope/tracepoint.h +17 -0
- data/lib/rotoscope.rb +71 -0
- data/lib/uthash/uthash.h +1107 -0
- data/rotoscope.gemspec +23 -0
- data/test/fixture_inner.rb +10 -0
- data/test/fixture_outer.rb +10 -0
- data/test/monadify.rb +16 -0
- data/test/rotoscope_test.rb +482 -0
- metadata +129 -0
data/rotoscope.gemspec
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
Gem::Specification.new do |s|
|
3
|
+
s.name = 'rotoscope'
|
4
|
+
s.version = '0.2.1'
|
5
|
+
s.date = '2017-06-19'
|
6
|
+
|
7
|
+
s.authors = ["Jahfer Husain"]
|
8
|
+
s.email = 'jahfer.husain@shopify.com'
|
9
|
+
s.homepage = 'https://github.com/shopify/rotoscope'
|
10
|
+
s.license = 'MIT'
|
11
|
+
|
12
|
+
s.summary = "Tracing Ruby"
|
13
|
+
s.description = "Rotoscope performs introspection of method calls in Ruby programs."
|
14
|
+
|
15
|
+
s.files = `git ls-files`.split("\n")
|
16
|
+
s.required_ruby_version = ">= 2.2.0"
|
17
|
+
s.extensions = %w(ext/rotoscope/extconf.rb)
|
18
|
+
|
19
|
+
s.add_development_dependency 'rake-compiler', '~> 0.9'
|
20
|
+
s.add_development_dependency 'mocha', '~> 0.14'
|
21
|
+
s.add_development_dependency 'minitest', '~> 5.0'
|
22
|
+
s.add_development_dependency 'rubocop', '~> 0.49'
|
23
|
+
end
|
data/test/monadify.rb
ADDED
@@ -0,0 +1,482 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
$LOAD_PATH.unshift File.expand_path('../../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift File.expand_path('../', __FILE__)
|
4
|
+
require 'rotoscope'
|
5
|
+
require 'minitest'
|
6
|
+
require 'zlib'
|
7
|
+
require 'fileutils'
|
8
|
+
require 'csv'
|
9
|
+
|
10
|
+
require 'fixture_inner'
|
11
|
+
require 'fixture_outer'
|
12
|
+
require 'monadify'
|
13
|
+
|
14
|
+
module MyModule
|
15
|
+
def module_method; end
|
16
|
+
end
|
17
|
+
|
18
|
+
module PrependedModule
|
19
|
+
def prepended_method; end
|
20
|
+
end
|
21
|
+
|
22
|
+
class Example
|
23
|
+
prepend PrependedModule
|
24
|
+
include MyModule
|
25
|
+
extend MyModule
|
26
|
+
extend Monadify
|
27
|
+
|
28
|
+
class << self
|
29
|
+
def singleton_method
|
30
|
+
true
|
31
|
+
end
|
32
|
+
|
33
|
+
def apply(val)
|
34
|
+
monad val
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
def normal_method
|
39
|
+
true
|
40
|
+
end
|
41
|
+
|
42
|
+
def exception_method
|
43
|
+
oops
|
44
|
+
rescue
|
45
|
+
nil
|
46
|
+
end
|
47
|
+
|
48
|
+
def yielding_method
|
49
|
+
yield
|
50
|
+
end
|
51
|
+
|
52
|
+
private
|
53
|
+
|
54
|
+
def oops
|
55
|
+
raise "I've made a terrible mistake"
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
ROOT_FIXTURE_PATH = File.expand_path('../', __FILE__)
|
60
|
+
INNER_FIXTURE_PATH = File.expand_path('../fixture_inner.rb', __FILE__)
|
61
|
+
OUTER_FIXTURE_PATH = File.expand_path('../fixture_outer.rb', __FILE__)
|
62
|
+
MONADIFY_PATH = File.expand_path('monadify.rb', ROOT_FIXTURE_PATH)
|
63
|
+
|
64
|
+
class RotoscopeTest < MiniTest::Test
|
65
|
+
def setup
|
66
|
+
@logfile = File.expand_path('tmp/test.csv')
|
67
|
+
end
|
68
|
+
|
69
|
+
def teardown
|
70
|
+
FileUtils.remove_file(@logfile) if File.file?(@logfile)
|
71
|
+
end
|
72
|
+
|
73
|
+
def test_new
|
74
|
+
rs = Rotoscope.new(@logfile, blacklist: ['tmp'], flatten: true)
|
75
|
+
assert rs.is_a?(Rotoscope)
|
76
|
+
end
|
77
|
+
|
78
|
+
def test_close
|
79
|
+
rs = Rotoscope.new(@logfile)
|
80
|
+
assert rs.close
|
81
|
+
end
|
82
|
+
|
83
|
+
def test_closed?
|
84
|
+
rs = Rotoscope.new(@logfile)
|
85
|
+
refute_predicate rs, :closed?
|
86
|
+
rs.close
|
87
|
+
assert_predicate rs, :closed?
|
88
|
+
end
|
89
|
+
|
90
|
+
def test_state
|
91
|
+
rs = Rotoscope.new(@logfile)
|
92
|
+
assert_equal :open, rs.state
|
93
|
+
rs.trace do
|
94
|
+
assert_equal :tracing, rs.state
|
95
|
+
end
|
96
|
+
assert_equal :open, rs.state
|
97
|
+
rs.close
|
98
|
+
assert_equal :closed, rs.state
|
99
|
+
end
|
100
|
+
|
101
|
+
def test_mark
|
102
|
+
contents = rotoscope_trace do |rs|
|
103
|
+
Example.new.normal_method
|
104
|
+
rs.mark
|
105
|
+
end
|
106
|
+
|
107
|
+
assert_includes contents.split("\n"), '--- '
|
108
|
+
end
|
109
|
+
|
110
|
+
def test_mark_with_custom_strings
|
111
|
+
mark_strings = ["Hello", "ÅÉÎØÜ åéîøü"]
|
112
|
+
contents = rotoscope_trace do |rs|
|
113
|
+
e = Example.new
|
114
|
+
e.normal_method
|
115
|
+
mark_strings.each { |str| rs.mark(str) }
|
116
|
+
end
|
117
|
+
|
118
|
+
content_lines = contents.split("\n")
|
119
|
+
mark_strings.each do |str|
|
120
|
+
assert_includes content_lines, "--- #{str}"
|
121
|
+
end
|
122
|
+
end
|
123
|
+
|
124
|
+
def test_flatten
|
125
|
+
contents = rotoscope_trace(flatten: true) do
|
126
|
+
Example.new.normal_method
|
127
|
+
end
|
128
|
+
|
129
|
+
assert_equal [
|
130
|
+
{ entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
131
|
+
{ entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "Example", caller_method_name: "new", caller_method_level: "class" },
|
132
|
+
{ entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
133
|
+
], parse_and_normalize(contents)
|
134
|
+
end
|
135
|
+
|
136
|
+
def test_flatten_removes_duplicates
|
137
|
+
contents = rotoscope_trace(flatten: true) do
|
138
|
+
e = Example.new
|
139
|
+
10.times { e.normal_method }
|
140
|
+
end
|
141
|
+
|
142
|
+
assert_equal [
|
143
|
+
{ entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
144
|
+
{ entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "Example", caller_method_name: "new", caller_method_level: "class" },
|
145
|
+
{ entity: "Integer", method_name: "times", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
146
|
+
{ entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "Integer", caller_method_name: "times", caller_method_level: "instance" },
|
147
|
+
], parse_and_normalize(contents)
|
148
|
+
end
|
149
|
+
|
150
|
+
def test_start_trace_and_stop_trace
|
151
|
+
rs = Rotoscope.new(@logfile)
|
152
|
+
rs.start_trace
|
153
|
+
Example.new.normal_method
|
154
|
+
rs.stop_trace
|
155
|
+
rs.close
|
156
|
+
contents = File.read(@logfile)
|
157
|
+
|
158
|
+
assert_equal [
|
159
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
160
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
161
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
162
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
163
|
+
{ event: "call", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
164
|
+
{ event: "return", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
165
|
+
], parse_and_normalize(contents)
|
166
|
+
|
167
|
+
assert_frames_consistent contents
|
168
|
+
end
|
169
|
+
|
170
|
+
def test_traces_instance_method
|
171
|
+
contents = rotoscope_trace { Example.new.normal_method }
|
172
|
+
assert_equal [
|
173
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
174
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
175
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
176
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
177
|
+
{ event: "call", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
178
|
+
{ event: "return", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 }
|
179
|
+
], parse_and_normalize(contents)
|
180
|
+
|
181
|
+
assert_frames_consistent contents
|
182
|
+
end
|
183
|
+
|
184
|
+
def test_traces_yielding_method
|
185
|
+
contents = rotoscope_trace do
|
186
|
+
e = Example.new
|
187
|
+
e.yielding_method { e.normal_method }
|
188
|
+
end
|
189
|
+
|
190
|
+
assert_equal [
|
191
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
192
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
193
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
194
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
195
|
+
{ event: "call", entity: "Example", method_name: "yielding_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
196
|
+
{ event: "call", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
197
|
+
{ event: "return", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
198
|
+
{ event: "return", entity: "Example", method_name: "yielding_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 }
|
199
|
+
], parse_and_normalize(contents)
|
200
|
+
|
201
|
+
assert_frames_consistent contents
|
202
|
+
end
|
203
|
+
|
204
|
+
def test_calls_are_consistent_after_exception
|
205
|
+
contents = rotoscope_trace { Example.new.exception_method }
|
206
|
+
assert_frames_consistent contents
|
207
|
+
end
|
208
|
+
|
209
|
+
def test_traces_and_formats_singletons_of_a_class
|
210
|
+
contents = rotoscope_trace { Example.singleton_method }
|
211
|
+
assert_equal [
|
212
|
+
{ event: "call", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
213
|
+
{ event: "return", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 }
|
214
|
+
], parse_and_normalize(contents)
|
215
|
+
|
216
|
+
assert_frames_consistent contents
|
217
|
+
end
|
218
|
+
|
219
|
+
def test_traces_and_formats_singletons_of_an_instance
|
220
|
+
contents = rotoscope_trace { Example.new.singleton_class.singleton_method }
|
221
|
+
assert_equal [
|
222
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
223
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
224
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
225
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
226
|
+
{ event: "call", entity: "Example", method_name: "singleton_class", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
227
|
+
{ event: "return", entity: "Example", method_name: "singleton_class", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
228
|
+
{ event: "call", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
229
|
+
{ event: "return", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
230
|
+
], parse_and_normalize(contents)
|
231
|
+
|
232
|
+
assert_frames_consistent contents
|
233
|
+
end
|
234
|
+
|
235
|
+
def test_traces_included_module_method
|
236
|
+
contents = rotoscope_trace { Example.new.module_method }
|
237
|
+
assert_equal [
|
238
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
239
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
240
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
241
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
242
|
+
{ event: "call", entity: "Example", method_name: "module_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
243
|
+
{ event: "return", entity: "Example", method_name: "module_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 }
|
244
|
+
], parse_and_normalize(contents)
|
245
|
+
|
246
|
+
assert_frames_consistent contents
|
247
|
+
end
|
248
|
+
|
249
|
+
def test_traces_extended_module_method
|
250
|
+
contents = rotoscope_trace { Example.module_method }
|
251
|
+
assert_equal [
|
252
|
+
{ event: "call", entity: "Example", method_name: "module_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
253
|
+
{ event: "return", entity: "Example", method_name: "module_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 }
|
254
|
+
], parse_and_normalize(contents)
|
255
|
+
|
256
|
+
assert_frames_consistent contents
|
257
|
+
end
|
258
|
+
|
259
|
+
def test_traces_prepended_module_method
|
260
|
+
contents = rotoscope_trace { Example.new.prepended_method }
|
261
|
+
assert_equal [
|
262
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
263
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
264
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
265
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
266
|
+
{ event: "call", entity: "Example", method_name: "prepended_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
267
|
+
{ event: "return", entity: "Example", method_name: "prepended_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 }
|
268
|
+
], parse_and_normalize(contents)
|
269
|
+
|
270
|
+
assert_frames_consistent contents
|
271
|
+
end
|
272
|
+
|
273
|
+
def test_trace_ignores_calls_if_blacklisted
|
274
|
+
contents = rotoscope_trace(blacklist: [INNER_FIXTURE_PATH, OUTER_FIXTURE_PATH]) do
|
275
|
+
foo = FixtureOuter.new
|
276
|
+
foo.do_work
|
277
|
+
end
|
278
|
+
|
279
|
+
assert_equal [
|
280
|
+
{ event: "call", entity: "FixtureOuter", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
281
|
+
{ event: "call", entity: "FixtureOuter", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
282
|
+
{ event: "return", entity: "FixtureOuter", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
283
|
+
{ event: "return", entity: "FixtureOuter", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
284
|
+
{ event: "call", entity: "FixtureOuter", method_name: "do_work", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
285
|
+
{ event: "return", entity: "FixtureOuter", method_name: "do_work", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
286
|
+
], parse_and_normalize(contents)
|
287
|
+
|
288
|
+
assert_frames_consistent contents
|
289
|
+
end
|
290
|
+
|
291
|
+
def test_trace_ignores_writes_in_fork
|
292
|
+
contents = rotoscope_trace do |rotoscope|
|
293
|
+
fork do
|
294
|
+
Example.singleton_method
|
295
|
+
rotoscope.mark
|
296
|
+
rotoscope.close
|
297
|
+
end
|
298
|
+
Example.singleton_method
|
299
|
+
Process.wait
|
300
|
+
end
|
301
|
+
assert_equal [
|
302
|
+
{ event: "call", entity: "RotoscopeTest", method_name: "fork", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
303
|
+
{ event: "return", entity: "RotoscopeTest", method_name: "fork", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
304
|
+
{ event: "call", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
305
|
+
{ event: "return", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
306
|
+
{ event: "call", entity: "Process", method_name: "wait", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
307
|
+
{ event: "return", entity: "Process", method_name: "wait", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
308
|
+
], parse_and_normalize(contents)
|
309
|
+
end
|
310
|
+
|
311
|
+
def test_trace_disabled_on_close
|
312
|
+
contents = rotoscope_trace do |rotoscope|
|
313
|
+
Example.singleton_method
|
314
|
+
rotoscope.close
|
315
|
+
rotoscope.mark
|
316
|
+
Example.singleton_method
|
317
|
+
end
|
318
|
+
assert_equal [
|
319
|
+
{ event: "call", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
320
|
+
{ event: "return", entity: "Example", method_name: "singleton_method", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
321
|
+
], parse_and_normalize(contents)
|
322
|
+
end
|
323
|
+
|
324
|
+
def test_trace_flatten
|
325
|
+
contents = rotoscope_trace(flatten: true) { Example.new.normal_method }
|
326
|
+
assert_equal [
|
327
|
+
{ entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
328
|
+
{ entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "Example", caller_method_name: "new", caller_method_level: "class" },
|
329
|
+
{ entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
330
|
+
], parse_and_normalize(contents)
|
331
|
+
end
|
332
|
+
|
333
|
+
def test_trace_flatten_across_files
|
334
|
+
contents = rotoscope_trace(flatten: true) do
|
335
|
+
foo = FixtureOuter.new
|
336
|
+
foo.do_work
|
337
|
+
end
|
338
|
+
|
339
|
+
assert_equal [
|
340
|
+
{ entity: "FixtureOuter", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
341
|
+
{ entity: "FixtureOuter", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "FixtureOuter", caller_method_name: "new", caller_method_level: "class" },
|
342
|
+
{ entity: "FixtureInner", method_name: "new", method_level: "class", filepath: "/fixture_outer.rb", lineno: -1, caller_entity: "FixtureOuter", caller_method_name: "initialize", caller_method_level: "instance" },
|
343
|
+
{ entity: "FixtureInner", method_name: "initialize", method_level: "instance", filepath: "/fixture_outer.rb", lineno: -1, caller_entity: "FixtureInner", caller_method_name: "new", caller_method_level: "class" },
|
344
|
+
{ entity: "FixtureOuter", method_name: "do_work", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
345
|
+
{ entity: "FixtureInner", method_name: "do_work", method_level: "instance", filepath: "/fixture_outer.rb", lineno: -1, caller_entity: "FixtureOuter", caller_method_name: "do_work", caller_method_level: "instance" },
|
346
|
+
{ entity: "FixtureInner", method_name: "sum", method_level: "instance", filepath: "/fixture_inner.rb", lineno: -1, caller_entity: "FixtureInner", caller_method_name: "do_work", caller_method_level: "instance" }
|
347
|
+
], parse_and_normalize(contents)
|
348
|
+
end
|
349
|
+
|
350
|
+
def test_trace_uses_io_objects
|
351
|
+
string_io = StringIO.new
|
352
|
+
Rotoscope.trace(string_io) do
|
353
|
+
Example.new.normal_method
|
354
|
+
end
|
355
|
+
refute_predicate string_io, :closed?
|
356
|
+
assert_predicate string_io, :eof?
|
357
|
+
contents = string_io.string
|
358
|
+
|
359
|
+
assert_equal [
|
360
|
+
{ event: "call", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
361
|
+
{ event: "call", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
362
|
+
{ event: "return", entity: "Example", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
363
|
+
{ event: "return", entity: "Example", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
364
|
+
{ event: "call", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
365
|
+
{ event: "return", entity: "Example", method_name: "normal_method", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
366
|
+
], parse_and_normalize(contents)
|
367
|
+
|
368
|
+
assert_frames_consistent contents
|
369
|
+
end
|
370
|
+
|
371
|
+
def test_stop_trace_before_start_does_not_raise
|
372
|
+
rs = Rotoscope.new(@logfile)
|
373
|
+
rs.stop_trace
|
374
|
+
end
|
375
|
+
|
376
|
+
def test_gc_rotoscope_without_stop_trace_does_not_crash
|
377
|
+
proc {
|
378
|
+
rs = Rotoscope.new(@logfile)
|
379
|
+
rs.start_trace
|
380
|
+
}.call
|
381
|
+
GC.start
|
382
|
+
end
|
383
|
+
|
384
|
+
def test_gc_rotoscope_without_stop_trace_does_not_break_process_cleanup
|
385
|
+
child_pid = fork do
|
386
|
+
rs = Rotoscope.new(@logfile)
|
387
|
+
rs.start_trace
|
388
|
+
end
|
389
|
+
Process.waitpid(child_pid)
|
390
|
+
assert_equal true, $CHILD_STATUS.success?
|
391
|
+
end
|
392
|
+
|
393
|
+
def test_log_path
|
394
|
+
rs = Rotoscope.new(File.expand_path('tmp/test.csv.gz'))
|
395
|
+
GC.start
|
396
|
+
assert_equal File.expand_path('tmp/test.csv.gz'), rs.log_path
|
397
|
+
end
|
398
|
+
|
399
|
+
def test_ignores_calls_inside_of_threads
|
400
|
+
thread = nil
|
401
|
+
contents = rotoscope_trace do
|
402
|
+
thread = Thread.new { Example.new }
|
403
|
+
end
|
404
|
+
thread.join
|
405
|
+
|
406
|
+
assert_equal [
|
407
|
+
{ event: "call", entity: "Thread", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
408
|
+
{ event: "call", entity: "Thread", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
409
|
+
{ event: "return", entity: "Thread", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
410
|
+
{ event: "return", entity: "Thread", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
411
|
+
], parse_and_normalize(contents)
|
412
|
+
end
|
413
|
+
|
414
|
+
def test_dynamic_class_creation
|
415
|
+
contents = rotoscope_trace { Class.new }
|
416
|
+
|
417
|
+
assert_equal [
|
418
|
+
{ event: "call", entity: "Class", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
419
|
+
{ event: "call", entity: "#<Class:0xXXXXXX>", method_name: "initialize", method_level: "instance", filepath: "/rotoscope_test.rb", lineno: -1 },
|
420
|
+
{ event: "call", entity: "Object", method_name: "inherited", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
421
|
+
{ event: "return", entity: "Object", method_name: "inherited", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
422
|
+
{ event: "return", entity: "#<Class:0xXXXXXX>", method_name: "initialize", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
423
|
+
{ event: "return", entity: "Class", method_name: "new", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 }
|
424
|
+
], parse_and_normalize(contents)
|
425
|
+
end
|
426
|
+
|
427
|
+
def test_dynamic_methods_in_blacklist
|
428
|
+
skip <<-FAILING_TEST_CASE
|
429
|
+
Return events for dynamically created methods (define_method, define_singleton_method)
|
430
|
+
do not have the correct stack frame information (the call of a dynamically defined method
|
431
|
+
is correctly treated as a Ruby :call, but its return must be treated as a :c_return)
|
432
|
+
FAILING_TEST_CASE
|
433
|
+
|
434
|
+
contents = rotoscope_trace(blacklist: [MONADIFY_PATH]) { Example.apply("my value!") }
|
435
|
+
|
436
|
+
assert_equal [
|
437
|
+
{ event: "call", entity: "Example", method_name: "apply", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
438
|
+
{ event: "call", entity: "Example", method_name: "monad", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
439
|
+
{ event: "return", entity: "Example", method_name: "monad", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
440
|
+
{ event: "return", entity: "Example", method_name: "apply", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1 },
|
441
|
+
], parse_and_normalize(contents)
|
442
|
+
end
|
443
|
+
|
444
|
+
def test_flatten_with_dynamic_methods_in_blacklist
|
445
|
+
# the failing test above passes when using `flatten: true` since unmatched stack returns are ignored
|
446
|
+
contents = rotoscope_trace(blacklist: [MONADIFY_PATH], flatten: true) { Example.apply("my value!") }
|
447
|
+
|
448
|
+
assert_equal [
|
449
|
+
{ entity: "Example", method_name: "apply", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "<ROOT>", caller_method_name: "<UNKNOWN>", caller_method_level: "<UNKNOWN>" },
|
450
|
+
{ entity: "Example", method_name: "monad", method_level: "class", filepath: "/rotoscope_test.rb", lineno: -1, caller_entity: "Example", caller_method_name: "apply", caller_method_level: "class" },
|
451
|
+
], parse_and_normalize(contents)
|
452
|
+
end
|
453
|
+
|
454
|
+
private
|
455
|
+
|
456
|
+
def parse_and_normalize(csv_string)
|
457
|
+
CSV.parse(csv_string, headers: true, header_converters: :symbol).map do |row|
|
458
|
+
row = row.to_h
|
459
|
+
row[:lineno] = -1
|
460
|
+
row[:filepath] = row[:filepath].gsub(ROOT_FIXTURE_PATH, '')
|
461
|
+
row[:entity] = row[:entity].gsub(/:0x[a-fA-F0-9]{4,}/m, ":0xXXXXXX")
|
462
|
+
row
|
463
|
+
end
|
464
|
+
end
|
465
|
+
|
466
|
+
def assert_frames_consistent(csv_string)
|
467
|
+
assert_equal csv_string.scan(/\Acall/).size, csv_string.scan(/\Areturn/).size
|
468
|
+
end
|
469
|
+
|
470
|
+
def rotoscope_trace(blacklist: [], flatten: false)
|
471
|
+
Rotoscope.trace(@logfile, blacklist: blacklist, flatten: flatten) { |rotoscope| yield rotoscope }
|
472
|
+
File.read(@logfile)
|
473
|
+
end
|
474
|
+
|
475
|
+
def unzip(path)
|
476
|
+
File.open(path) { |f| Zlib::GzipReader.new(f).read }
|
477
|
+
end
|
478
|
+
end
|
479
|
+
|
480
|
+
# https://github.com/seattlerb/minitest/pull/683 needed to use
|
481
|
+
# autorun without affecting the exit status of forked processes
|
482
|
+
Minitest.run(ARGV)
|