appmap 0.34.2 → 0.35.2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +29 -0
- data/README.md +9 -0
- data/appmap.gemspec +1 -0
- data/ext/appmap/appmap.c +74 -5
- data/lib/appmap.rb +1 -1
- data/lib/appmap/class_map.rb +5 -6
- data/lib/appmap/config.rb +2 -4
- data/lib/appmap/event.rb +28 -10
- data/lib/appmap/hook.rb +12 -8
- data/lib/appmap/hook/method.rb +35 -28
- data/lib/appmap/rails/request_handler.rb +88 -0
- data/lib/appmap/rails/sql_handler.rb +13 -10
- data/lib/appmap/railtie.rb +3 -5
- data/lib/appmap/rspec.rb +10 -0
- data/lib/appmap/trace.rb +9 -7
- data/lib/appmap/util.rb +18 -1
- data/lib/appmap/version.rb +1 -1
- data/spec/abstract_controller4_base_spec.rb +1 -1
- data/spec/abstract_controller_base_spec.rb +6 -0
- data/spec/fixtures/hook/exception_method.rb +44 -0
- data/spec/hook_spec.rb +138 -5
- data/test/cli_test.rb +0 -10
- data/test/openssl_test.rb +0 -48
- metadata +17 -3
- data/lib/appmap/rails/action_handler.rb +0 -91
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 9e7d5c58bd3addf395c32591fe468c4a0b0e886d715cb711ffb6c898049455e5
|
4
|
+
data.tar.gz: 271ac50ee1ebe139ed8e52262a98a13e6b3d471ddcab49316669e66d0b7be57a
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: b84b1d76d8890a72fb376e95c7ab3811814b75e99304d2f1c15bb091cba1aef1e81f7688fd69c35b8c7735fd72167f89c67eeb25f76e54253503be610fff52cc
|
7
|
+
data.tar.gz: 12b27887ae25d8f91fa6321ea59c222c7ec7d1e2d6a198f6672fa166fcd3c32456fe0d3969dda729d4c4817129a3eb874f8bae0afb5dca6b74253ccf457bd51b
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,32 @@
|
|
1
|
+
# v0.35.2
|
2
|
+
* Make sure `MethodEvent#display_string` works when the value's `#to_s` and/or `#inspect`
|
3
|
+
methods have problems.
|
4
|
+
|
5
|
+
# v0.35.1
|
6
|
+
* Take out hooking of `IO` and `Logger` methods.
|
7
|
+
* Enable logging if either `APPMAP_DEBUG` or `DEBUG` is `true`.
|
8
|
+
|
9
|
+
# v0.35.0
|
10
|
+
* Provide a custom display string for files and HTTP requests.
|
11
|
+
* Report `mime_type` on HTTP response.
|
12
|
+
|
13
|
+
# v0.34.6
|
14
|
+
* Only warn once about problems determining database version for an ActiveRecord
|
15
|
+
connection.
|
16
|
+
|
17
|
+
# v0.34.5
|
18
|
+
* Ensure that hooking a method doesn't change its arity.
|
19
|
+
|
20
|
+
# v0.34.4
|
21
|
+
* Make sure `AppMap:Rails::SQLExaminer::ActiveRecordExaminer.server_version` only calls
|
22
|
+
`ActiveRecord::Base.connection.database_version` if it's available.
|
23
|
+
* Fix `AppMap:Rails::SQLExaminer::ActiveRecordExaminer.database_type` returns `:postgres`
|
24
|
+
in all supported versions of Rails.
|
25
|
+
|
26
|
+
# v0.34.3
|
27
|
+
* Fix a crash in `singleton_method_owner_name` that occurred if `__attached__.class` returned
|
28
|
+
something other than a `Module` or a `Class`.
|
29
|
+
|
1
30
|
# v0.34.2
|
2
31
|
* Add an extension that gets the name of the owner of a singleton method without calling
|
3
32
|
any methods that may have been redefined (e.g. `#to_s` or `.name`).
|
data/README.md
CHANGED
@@ -57,6 +57,15 @@ end
|
|
57
57
|
|
58
58
|
Then install with `bundle`.
|
59
59
|
|
60
|
+
**Railtie**
|
61
|
+
|
62
|
+
If you are using Ruby on Rails, require the railtie after Rails is loaded.
|
63
|
+
|
64
|
+
```
|
65
|
+
# application.rb is a good place to do this, along with all the other railties.
|
66
|
+
require 'appmap/railtie'
|
67
|
+
```
|
68
|
+
|
60
69
|
# Configuration
|
61
70
|
|
62
71
|
When you run your program, the `appmap` gem reads configuration settings from `appmap.yml`. Here's a sample configuration
|
data/appmap.gemspec
CHANGED
data/ext/appmap/appmap.c
CHANGED
@@ -8,19 +8,88 @@
|
|
8
8
|
(!SPECIAL_CONST_P(obj) && \
|
9
9
|
(BUILTIN_TYPE(obj) == T_CLASS || BUILTIN_TYPE(obj) == T_MODULE))
|
10
10
|
|
11
|
-
|
11
|
+
#define ARITIES_KEY "__arities__"
|
12
|
+
|
13
|
+
VALUE am_AppMapHook;
|
14
|
+
|
15
|
+
static VALUE
|
16
|
+
singleton_method_owner_name(VALUE klass, VALUE method)
|
12
17
|
{
|
13
18
|
VALUE owner = rb_funcall(method, rb_intern("owner"), 0);
|
14
19
|
VALUE attached = rb_ivar_get(owner, rb_intern("__attached__"));
|
15
20
|
if (!CLASS_OR_MODULE_P(attached)) {
|
16
21
|
attached = rb_funcall(attached, rb_intern("class"), 0);
|
17
22
|
}
|
18
|
-
|
23
|
+
|
24
|
+
// Did __attached__.class return an object that's a Module or a
|
25
|
+
// Class?
|
26
|
+
if (CLASS_OR_MODULE_P(attached)) {
|
27
|
+
// Yup, get it's name
|
28
|
+
return rb_mod_name(attached);
|
29
|
+
}
|
30
|
+
|
31
|
+
// Nope (which seems weird, but whatever). Fall back to calling
|
32
|
+
// #to_s on the method's owner and hope for the best.
|
33
|
+
return rb_funcall(owner, rb_intern("to_s"), 0);
|
34
|
+
}
|
35
|
+
|
36
|
+
|
37
|
+
static VALUE
|
38
|
+
am_define_method_with_arity(VALUE mod, VALUE name, VALUE arity, VALUE proc)
|
39
|
+
{
|
40
|
+
VALUE arities_key = rb_intern(ARITIES_KEY);
|
41
|
+
VALUE arities = rb_ivar_get(mod, arities_key);
|
42
|
+
|
43
|
+
if (arities == Qundef || NIL_P(arities)) {
|
44
|
+
arities = rb_hash_new();
|
45
|
+
rb_ivar_set(mod, arities_key, arities);
|
46
|
+
}
|
47
|
+
rb_hash_aset(arities, name, arity);
|
48
|
+
|
49
|
+
return rb_funcall(mod, rb_intern("define_method"), 2, name, proc);
|
50
|
+
}
|
51
|
+
|
52
|
+
static VALUE
|
53
|
+
am_get_method_arity(VALUE method, VALUE orig_arity_method)
|
54
|
+
{
|
55
|
+
VALUE owner = rb_funcall(method, rb_intern("owner"), 0);
|
56
|
+
VALUE arities = rb_ivar_get(owner, rb_intern(ARITIES_KEY));
|
57
|
+
VALUE name = rb_funcall(method, rb_intern("name"), 0);
|
58
|
+
VALUE arity = Qnil;
|
59
|
+
// See if we saved an arity for the method.
|
60
|
+
if (!NIL_P(arities)) {
|
61
|
+
arity = rb_hash_aref(arities, name);
|
62
|
+
}
|
63
|
+
// Didn't find one, call the original method.
|
64
|
+
if (NIL_P(arity)) {
|
65
|
+
VALUE bound_method = rb_funcall(orig_arity_method, rb_intern("bind"), 1, method);
|
66
|
+
arity = rb_funcall(bound_method, rb_intern("call"), 0);
|
67
|
+
}
|
68
|
+
|
69
|
+
return arity;
|
19
70
|
}
|
20
|
-
|
71
|
+
|
72
|
+
static VALUE
|
73
|
+
am_unbound_method_arity(VALUE method)
|
74
|
+
{
|
75
|
+
VALUE orig_unbound_method_arity = rb_ivar_get(am_AppMapHook, rb_intern("@unbound_method_arity"));
|
76
|
+
return am_get_method_arity(method, orig_unbound_method_arity);
|
77
|
+
}
|
78
|
+
|
79
|
+
static VALUE
|
80
|
+
am_method_arity(VALUE method)
|
81
|
+
{
|
82
|
+
VALUE orig_method_arity = rb_ivar_get(am_AppMapHook, rb_intern("@method_arity"));
|
83
|
+
return am_get_method_arity(method, orig_method_arity);
|
84
|
+
}
|
85
|
+
|
21
86
|
void Init_appmap() {
|
22
87
|
VALUE appmap = rb_define_module("AppMap");
|
23
|
-
|
88
|
+
am_AppMapHook = rb_define_class_under(appmap, "Hook", rb_cObject);
|
89
|
+
|
90
|
+
rb_define_singleton_method(am_AppMapHook, "singleton_method_owner_name", singleton_method_owner_name, 1);
|
24
91
|
|
25
|
-
|
92
|
+
rb_define_method(rb_cModule, "define_method_with_arity", am_define_method_with_arity, 3);
|
93
|
+
rb_define_method(rb_cUnboundMethod, "arity", am_unbound_method_arity, 0);
|
94
|
+
rb_define_method(rb_cMethod, "arity", am_method_arity, 0);
|
26
95
|
}
|
data/lib/appmap.rb
CHANGED
@@ -84,7 +84,7 @@ module AppMap
|
|
84
84
|
|
85
85
|
# Builds a class map from a config and a list of Ruby methods.
|
86
86
|
def class_map(methods)
|
87
|
-
ClassMap.build_from_methods(
|
87
|
+
ClassMap.build_from_methods(methods)
|
88
88
|
end
|
89
89
|
|
90
90
|
# Returns default metadata detected from the Ruby system and from the
|
data/lib/appmap/class_map.rb
CHANGED
@@ -61,25 +61,24 @@ module AppMap
|
|
61
61
|
location: location,
|
62
62
|
static: static,
|
63
63
|
labels: labels
|
64
|
-
}.delete_if { |
|
64
|
+
}.delete_if { |_, v| v.nil? || v == [] }
|
65
65
|
end
|
66
66
|
end
|
67
67
|
end
|
68
68
|
|
69
69
|
class << self
|
70
|
-
def build_from_methods(
|
70
|
+
def build_from_methods(methods)
|
71
71
|
root = Types::Root.new
|
72
72
|
methods.each do |method|
|
73
|
-
|
74
|
-
or raise "No package found for method #{method}"
|
75
|
-
add_function root, package, method
|
73
|
+
add_function root, method
|
76
74
|
end
|
77
75
|
root.children.map(&:to_h)
|
78
76
|
end
|
79
77
|
|
80
78
|
protected
|
81
79
|
|
82
|
-
def add_function(root,
|
80
|
+
def add_function(root, method)
|
81
|
+
package = method.package
|
83
82
|
static = method.static
|
84
83
|
|
85
84
|
object_infos = [
|
data/lib/appmap/config.rb
CHANGED
@@ -6,7 +6,7 @@ module AppMap
|
|
6
6
|
def initialize(path, package_name: nil, exclude: [], labels: [])
|
7
7
|
super path, package_name, exclude, labels
|
8
8
|
end
|
9
|
-
|
9
|
+
|
10
10
|
def to_h
|
11
11
|
{
|
12
12
|
path: path,
|
@@ -35,12 +35,10 @@ module AppMap
|
|
35
35
|
'OpenSSL::PKCS5' => Hook.new(%i[pbkdf2_hmac_sha1 pbkdf2_hmac], OPENSSL_PACKAGE),
|
36
36
|
'OpenSSL::Cipher' => Hook.new(%i[encrypt decrypt final], OPENSSL_PACKAGE),
|
37
37
|
'OpenSSL::X509::Certificate' => Hook.new(:sign, OPENSSL_PACKAGE),
|
38
|
-
'Logger' => Hook.new(:add, Package.new('logger', labels: %w[log io])),
|
39
38
|
'Net::HTTP' => Hook.new(:request, Package.new('net/http', package_name: 'net/http', labels: %w[http io])),
|
40
39
|
'Net::SMTP' => Hook.new(:send, Package.new('net/smtp', package_name: 'net/smtp', labels: %w[smtp email io])),
|
41
40
|
'Net::POP3' => Hook.new(:mails, Package.new('net/pop3', package_name: 'net/pop', labels: %w[pop pop3 email io])),
|
42
41
|
'Net::IMAP' => Hook.new(:send_command, Package.new('net/imap', package_name: 'net/imap', labels: %w[imap email io])),
|
43
|
-
'IO' => Hook.new(%i[read write open close], Package.new('io', labels: %w[io])),
|
44
42
|
'Marshal' => Hook.new(%i[dump load], Package.new('marshal', labels: %w[serialization marshal])),
|
45
43
|
'Psych' => Hook.new(%i[dump dump_stream load load_stream parse parse_stream], Package.new('yaml', package_name: 'psych', labels: %w[serialization yaml])),
|
46
44
|
'JSON::Ext::Parser' => Hook.new(:parse, Package.new('json', package_name: 'json', labels: %w[serialization json])),
|
@@ -105,7 +103,7 @@ module AppMap
|
|
105
103
|
hook = find_hook(defined_class)
|
106
104
|
return nil unless hook
|
107
105
|
|
108
|
-
Array(hook.method_names).include?(method_name) ? hook.package : nil
|
106
|
+
Array(hook.method_names).include?(method_name) ? hook.package : nil
|
109
107
|
end
|
110
108
|
|
111
109
|
def find_hook(defined_class)
|
data/lib/appmap/event.rb
CHANGED
@@ -31,25 +31,43 @@ module AppMap
|
|
31
31
|
def display_string(value)
|
32
32
|
return nil unless value
|
33
33
|
|
34
|
+
value_string = custom_display_string(value) || default_display_string(value)
|
35
|
+
|
36
|
+
(value_string||'')[0...LIMIT].encode('utf-8', invalid: :replace, undef: :replace, replace: '_')
|
37
|
+
end
|
38
|
+
|
39
|
+
protected
|
40
|
+
|
41
|
+
def custom_display_string(value)
|
42
|
+
case value
|
43
|
+
when File
|
44
|
+
"#{value.class}[path=#{value.path}]"
|
45
|
+
when Net::HTTP
|
46
|
+
"#{value.class}[#{value.address}:#{value.port}]"
|
47
|
+
when Net::HTTPGenericRequest
|
48
|
+
"#{value.class}[#{value.method} #{value.path}]"
|
49
|
+
end
|
50
|
+
rescue StandardError
|
51
|
+
nil
|
52
|
+
end
|
53
|
+
|
54
|
+
def default_display_string(value)
|
34
55
|
last_resort_string = lambda do
|
35
56
|
warn "AppMap encountered an error inspecting a #{value.class.name}: #{$!.message}"
|
36
57
|
'*Error inspecting variable*'
|
37
58
|
end
|
38
59
|
|
39
|
-
|
60
|
+
begin
|
61
|
+
value.to_s
|
62
|
+
rescue NoMethodError
|
40
63
|
begin
|
41
|
-
value.
|
42
|
-
rescue NoMethodError
|
43
|
-
begin
|
44
|
-
value.inspect
|
45
|
-
rescue StandardError
|
46
|
-
last_resort_string.call
|
47
|
-
end
|
64
|
+
value.inspect
|
48
65
|
rescue StandardError
|
49
66
|
last_resort_string.call
|
50
67
|
end
|
51
|
-
|
52
|
-
|
68
|
+
rescue StandardError
|
69
|
+
last_resort_string.call
|
70
|
+
end
|
53
71
|
end
|
54
72
|
end
|
55
73
|
end
|
data/lib/appmap/hook.rb
CHANGED
@@ -4,7 +4,10 @@ require 'English'
|
|
4
4
|
|
5
5
|
module AppMap
|
6
6
|
class Hook
|
7
|
-
LOG = (ENV['DEBUG'] == 'true')
|
7
|
+
LOG = (ENV['APPMAP_DEBUG'] == 'true' || ENV['DEBUG'] == 'true')
|
8
|
+
|
9
|
+
@unbound_method_arity = ::UnboundMethod.instance_method(:arity)
|
10
|
+
@method_arity = ::Method.instance_method(:arity)
|
8
11
|
|
9
12
|
class << self
|
10
13
|
def lock_builtins
|
@@ -45,7 +48,6 @@ module AppMap
|
|
45
48
|
hook = lambda do |hook_cls|
|
46
49
|
lambda do |method_id|
|
47
50
|
method = hook_cls.public_instance_method(method_id)
|
48
|
-
hook_method = Hook::Method.new(hook_cls, method)
|
49
51
|
|
50
52
|
warn "AppMap: Examining #{hook_cls} #{method.name}" if LOG
|
51
53
|
|
@@ -53,14 +55,16 @@ module AppMap
|
|
53
55
|
# Skip methods that have no instruction sequence, as they are obviously trivial.
|
54
56
|
next unless disasm
|
55
57
|
|
56
|
-
# Don't try and trace the AppMap methods or there will be
|
57
|
-
# a stack overflow in the defined hook method.
|
58
|
-
next if /\AAppMap[:\.]/.match?(hook_method.method_display_name)
|
59
|
-
|
60
58
|
next unless \
|
61
59
|
config.always_hook?(hook_cls, method.name) ||
|
62
60
|
config.included_by_location?(method)
|
63
61
|
|
62
|
+
hook_method = Hook::Method.new(config.package_for_method(method), hook_cls, method)
|
63
|
+
|
64
|
+
# Don't try and trace the AppMap methods or there will be
|
65
|
+
# a stack overflow in the defined hook method.
|
66
|
+
next if /\AAppMap[:\.]/.match?(hook_method.method_display_name)
|
67
|
+
|
64
68
|
hook_method.activate
|
65
69
|
end
|
66
70
|
end
|
@@ -94,9 +98,9 @@ module AppMap
|
|
94
98
|
end
|
95
99
|
|
96
100
|
if method
|
97
|
-
Hook::Method.new(cls, method).activate
|
101
|
+
Hook::Method.new(hook.package, cls, method).activate
|
98
102
|
else
|
99
|
-
warn "Method #{method_name} not found on #{cls.name}"
|
103
|
+
warn "Method #{method_name} not found on #{cls.name}"
|
100
104
|
end
|
101
105
|
end
|
102
106
|
end
|
data/lib/appmap/hook/method.rb
CHANGED
@@ -1,7 +1,9 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module AppMap
|
2
4
|
class Hook
|
3
5
|
class Method
|
4
|
-
attr_reader :hook_class, :hook_method
|
6
|
+
attr_reader :hook_package, :hook_class, :hook_method
|
5
7
|
|
6
8
|
# +method_display_name+ may be nil if name resolution gets
|
7
9
|
# deferred until runtime (e.g. for a singleton method on an
|
@@ -15,8 +17,9 @@ module AppMap
|
|
15
17
|
# with the method we're hooking.
|
16
18
|
TIME_NOW = Time.method(:now)
|
17
19
|
private_constant :TIME_NOW
|
18
|
-
|
19
|
-
def initialize(hook_class, hook_method)
|
20
|
+
|
21
|
+
def initialize(hook_package, hook_class, hook_method)
|
22
|
+
@hook_package = hook_package
|
20
23
|
@hook_class = hook_class
|
21
24
|
@hook_method = hook_method
|
22
25
|
|
@@ -30,7 +33,7 @@ module AppMap
|
|
30
33
|
msg = if method_display_name
|
31
34
|
"#{method_display_name}"
|
32
35
|
else
|
33
|
-
"#{hook_method.name} (class resolution
|
36
|
+
"#{hook_method.name} (class resolution deferred)"
|
34
37
|
end
|
35
38
|
warn "AppMap: Hooking " + msg
|
36
39
|
end
|
@@ -41,34 +44,38 @@ module AppMap
|
|
41
44
|
after_hook = self.method(:after_hook)
|
42
45
|
with_disabled_hook = self.method(:with_disabled_hook)
|
43
46
|
|
44
|
-
|
45
|
-
|
47
|
+
hook_method_def = nil
|
48
|
+
hook_class.instance_eval do
|
49
|
+
hook_method_def = Proc.new do |*args, &block|
|
50
|
+
instance_method = hook_method.bind(self).to_proc
|
46
51
|
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
52
|
+
# We may not have gotten the class for the method during
|
53
|
+
# initialization (e.g. for a singleton method on an embedded
|
54
|
+
# struct), so make sure we have it now.
|
55
|
+
defined_class,_ = Hook.qualify_method_name(hook_method) unless defined_class
|
51
56
|
|
52
|
-
|
53
|
-
|
54
|
-
|
57
|
+
hook_disabled = Thread.current[HOOK_DISABLE_KEY]
|
58
|
+
enabled = true if !hook_disabled && AppMap.tracing.enabled?
|
59
|
+
return instance_method.call(*args, &block) unless enabled
|
55
60
|
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
61
|
+
call_event, start_time = with_disabled_hook.() do
|
62
|
+
before_hook.(self, defined_class, args)
|
63
|
+
end
|
64
|
+
return_value = nil
|
65
|
+
exception = nil
|
66
|
+
begin
|
67
|
+
return_value = instance_method.(*args, &block)
|
68
|
+
rescue
|
69
|
+
exception = $ERROR_INFO
|
70
|
+
raise
|
71
|
+
ensure
|
72
|
+
with_disabled_hook.() do
|
73
|
+
after_hook.(self, call_event, start_time, return_value, exception)
|
74
|
+
end
|
69
75
|
end
|
70
76
|
end
|
71
77
|
end
|
78
|
+
hook_class.define_method_with_arity(hook_method.name, hook_method.arity, hook_method_def)
|
72
79
|
end
|
73
80
|
|
74
81
|
protected
|
@@ -76,11 +83,11 @@ module AppMap
|
|
76
83
|
def before_hook(receiver, defined_class, args)
|
77
84
|
require 'appmap/event'
|
78
85
|
call_event = AppMap::Event::MethodCall.build_from_invocation(defined_class, hook_method, receiver, args)
|
79
|
-
AppMap.tracing.record_event call_event, defined_class: defined_class, method: hook_method
|
86
|
+
AppMap.tracing.record_event call_event, package: hook_package, defined_class: defined_class, method: hook_method
|
80
87
|
[ call_event, TIME_NOW.call ]
|
81
88
|
end
|
82
89
|
|
83
|
-
def after_hook(call_event, start_time, return_value, exception)
|
90
|
+
def after_hook(receiver, call_event, start_time, return_value, exception)
|
84
91
|
require 'appmap/event'
|
85
92
|
elapsed = TIME_NOW.call - start_time
|
86
93
|
return_event = \
|
@@ -0,0 +1,88 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'appmap/event'
|
4
|
+
require 'appmap/hook'
|
5
|
+
|
6
|
+
module AppMap
|
7
|
+
module Rails
|
8
|
+
module RequestHandler
|
9
|
+
class HTTPServerRequest < AppMap::Event::MethodEvent
|
10
|
+
attr_accessor :request_method, :path_info, :params
|
11
|
+
|
12
|
+
def initialize(request)
|
13
|
+
super AppMap::Event.next_id_counter, :call, Thread.current.object_id
|
14
|
+
|
15
|
+
@request_method = request.request_method
|
16
|
+
@path_info = request.path_info.split('?')[0]
|
17
|
+
@params = ActionDispatch::Http::ParameterFilter.new(::Rails.application.config.filter_parameters).filter(request.params)
|
18
|
+
end
|
19
|
+
|
20
|
+
def to_h
|
21
|
+
super.tap do |h|
|
22
|
+
h[:http_server_request] = {
|
23
|
+
request_method: request_method,
|
24
|
+
path_info: path_info
|
25
|
+
}
|
26
|
+
|
27
|
+
h[:message] = params.keys.map do |key|
|
28
|
+
val = params[key]
|
29
|
+
{
|
30
|
+
name: key,
|
31
|
+
class: val.class.name,
|
32
|
+
value: self.class.display_string(val),
|
33
|
+
object_id: val.__id__
|
34
|
+
}
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
class HTTPServerResponse < AppMap::Event::MethodReturnIgnoreValue
|
41
|
+
attr_accessor :status, :mime_type
|
42
|
+
|
43
|
+
def initialize(response, parent_id, elapsed)
|
44
|
+
super AppMap::Event.next_id_counter, :return, Thread.current.object_id
|
45
|
+
|
46
|
+
self.status = response.status
|
47
|
+
self.mime_type = response.headers['Content-Type']
|
48
|
+
self.parent_id = parent_id
|
49
|
+
self.elapsed = elapsed
|
50
|
+
end
|
51
|
+
|
52
|
+
def to_h
|
53
|
+
super.tap do |h|
|
54
|
+
h[:http_server_response] = {
|
55
|
+
status: status,
|
56
|
+
mime_type: mime_type
|
57
|
+
}.compact
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
class HookMethod < AppMap::Hook::Method
|
63
|
+
def initialize
|
64
|
+
# ActionController::Instrumentation has issued start_processing.action_controller and
|
65
|
+
# process_action.action_controller since Rails 3. Therefore it's a stable place to hook
|
66
|
+
# the request. Rails controller notifications can't be used directly because they don't
|
67
|
+
# provide response headers, and we want the Content-Type.
|
68
|
+
super(nil, ActionController::Instrumentation, ActionController::Instrumentation.instance_method(:process_action))
|
69
|
+
end
|
70
|
+
|
71
|
+
protected
|
72
|
+
|
73
|
+
def before_hook(receiver, defined_class, _) # args
|
74
|
+
call_event = HTTPServerRequest.new(receiver.request)
|
75
|
+
# http_server_request events are i/o and do not require a package name.
|
76
|
+
AppMap.tracing.record_event call_event, defined_class: defined_class, method: hook_method
|
77
|
+
[ call_event, TIME_NOW.call ]
|
78
|
+
end
|
79
|
+
|
80
|
+
def after_hook(receiver, call_event, start_time, _, _) # return_value, exception
|
81
|
+
elapsed = TIME_NOW.call - start_time
|
82
|
+
return_event = HTTPServerResponse.new receiver.response, call_event.id, elapsed
|
83
|
+
AppMap.tracing.record_event return_event
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
@@ -72,21 +72,24 @@ module AppMap
|
|
72
72
|
end
|
73
73
|
|
74
74
|
class ActiveRecordExaminer
|
75
|
+
@@db_version_warning_issued = {}
|
76
|
+
|
77
|
+
def issue_warning
|
78
|
+
db_type = database_type
|
79
|
+
return if @@db_version_warning_issued[db_type]
|
80
|
+
warn("AppMap: Unable to determine database version for #{db_type.inspect}")
|
81
|
+
@@db_version_warning_issued[db_type] = true
|
82
|
+
end
|
83
|
+
|
75
84
|
def server_version
|
76
|
-
|
77
|
-
when :postgres
|
78
|
-
ActiveRecord::Base.connection.postgresql_version
|
79
|
-
when :sqlite
|
80
|
-
ActiveRecord::Base.connection.database_version.to_s
|
81
|
-
else
|
82
|
-
warn "Unable to determine database version for #{database_type.inspect}"
|
83
|
-
end
|
85
|
+
ActiveRecord::Base.connection.try(:database_version) || issue_warning
|
84
86
|
end
|
85
87
|
|
86
88
|
def database_type
|
87
|
-
|
89
|
+
type = ActiveRecord::Base.connection.adapter_name.downcase.to_sym
|
90
|
+
type = :postgres if type == :postgresql
|
88
91
|
|
89
|
-
|
92
|
+
type
|
90
93
|
end
|
91
94
|
|
92
95
|
def execute_query(sql)
|
data/lib/appmap/railtie.rb
CHANGED
@@ -13,13 +13,11 @@ module AppMap
|
|
13
13
|
# AppMap events.
|
14
14
|
initializer 'appmap.subscribe', after: 'appmap.init' do |_| # params: app
|
15
15
|
require 'appmap/rails/sql_handler'
|
16
|
-
require 'appmap/rails/
|
16
|
+
require 'appmap/rails/request_handler'
|
17
17
|
ActiveSupport::Notifications.subscribe 'sql.sequel', AppMap::Rails::SQLHandler.new
|
18
18
|
ActiveSupport::Notifications.subscribe 'sql.active_record', AppMap::Rails::SQLHandler.new
|
19
|
-
|
20
|
-
|
21
|
-
ActiveSupport::Notifications.subscribe \
|
22
|
-
'process_action.action_controller', AppMap::Rails::ActionHandler::HTTPServerResponse.new
|
19
|
+
|
20
|
+
AppMap::Rails::RequestHandler::HookMethod.new.activate
|
23
21
|
end
|
24
22
|
|
25
23
|
# appmap.trace begins recording an AppMap trace and writes it to appmap.json.
|
data/lib/appmap/rspec.rb
CHANGED
@@ -124,8 +124,18 @@ module AppMap
|
|
124
124
|
def initialize(example)
|
125
125
|
super
|
126
126
|
|
127
|
+
webdriver_port = lambda do
|
128
|
+
return unless defined?(page) && page&.driver
|
129
|
+
|
130
|
+
# This is the ugliest thing ever but I don't want to lose it.
|
131
|
+
# All the WebDriver calls are getting app-mapped and it's really unclear
|
132
|
+
# what they are.
|
133
|
+
page.driver.options[:http_client].instance_variable_get('@server_url').port
|
134
|
+
end
|
135
|
+
|
127
136
|
warn "Starting recording of example #{example}" if AppMap::RSpec::LOG
|
128
137
|
@trace = AppMap.tracing.trace
|
138
|
+
@webdriver_port = webdriver_port.()
|
129
139
|
end
|
130
140
|
|
131
141
|
def finish
|
data/lib/appmap/trace.rb
CHANGED
@@ -3,9 +3,10 @@
|
|
3
3
|
module AppMap
|
4
4
|
module Trace
|
5
5
|
class ScopedMethod < SimpleDelegator
|
6
|
-
attr_reader :defined_class, :static
|
7
|
-
|
8
|
-
def initialize(defined_class, method, static)
|
6
|
+
attr_reader :package, :defined_class, :static
|
7
|
+
|
8
|
+
def initialize(package, defined_class, method, static)
|
9
|
+
@package = package
|
9
10
|
@defined_class = defined_class
|
10
11
|
@static = static
|
11
12
|
super(method)
|
@@ -32,9 +33,9 @@ module AppMap
|
|
32
33
|
@tracing.any?(&:enabled?)
|
33
34
|
end
|
34
35
|
|
35
|
-
def record_event(event, defined_class: nil, method: nil)
|
36
|
+
def record_event(event, package: nil, defined_class: nil, method: nil)
|
36
37
|
@tracing.each do |tracer|
|
37
|
-
tracer.record_event(event, defined_class: defined_class, method: method)
|
38
|
+
tracer.record_event(event, package: package, defined_class: defined_class, method: method)
|
38
39
|
end
|
39
40
|
end
|
40
41
|
|
@@ -71,11 +72,12 @@ module AppMap
|
|
71
72
|
# Record a program execution event.
|
72
73
|
#
|
73
74
|
# The event should be one of the MethodEvent subclasses.
|
74
|
-
def record_event(event, defined_class: nil, method: nil)
|
75
|
+
def record_event(event, package: nil, defined_class: nil, method: nil)
|
75
76
|
return unless @enabled
|
76
77
|
|
77
78
|
@events << event
|
78
|
-
@methods << Trace::ScopedMethod.new(defined_class, method, event.static)
|
79
|
+
@methods << Trace::ScopedMethod.new(package, defined_class, method, event.static) \
|
80
|
+
if package && defined_class && method && (event.event == :call)
|
79
81
|
end
|
80
82
|
|
81
83
|
# Gets a unique list of the methods that were invoked by the program.
|
data/lib/appmap/util.rb
CHANGED
@@ -36,6 +36,23 @@ module AppMap
|
|
36
36
|
[ fname, extension ].join
|
37
37
|
end
|
38
38
|
|
39
|
+
# sanitize_paths removes ephemeral values from objects with
|
40
|
+
# embedded paths (e.g. an event or a classmap), making events
|
41
|
+
# easier to compare across runs.
|
42
|
+
def sanitize_paths(h)
|
43
|
+
require 'hashie'
|
44
|
+
h.extend(Hashie::Extensions::DeepLocate)
|
45
|
+
keys = %i(path location)
|
46
|
+
h.deep_locate ->(k,v,o) {
|
47
|
+
next unless keys.include?(k)
|
48
|
+
|
49
|
+
fix = ->(v) {v.gsub(%r{#{Gem.dir}/gems/.*(?=lib)}, '')}
|
50
|
+
keys.each {|k| o[k] = fix.(o[k]) if o[k] }
|
51
|
+
}
|
52
|
+
|
53
|
+
h
|
54
|
+
end
|
55
|
+
|
39
56
|
# sanitize_event removes ephemeral values from an event, making
|
40
57
|
# events easier to compare across runs.
|
41
58
|
def sanitize_event(event, &block)
|
@@ -49,7 +66,7 @@ module AppMap
|
|
49
66
|
|
50
67
|
case event[:event]
|
51
68
|
when :call
|
52
|
-
event
|
69
|
+
sanitize_paths(event)
|
53
70
|
end
|
54
71
|
|
55
72
|
event
|
data/lib/appmap/version.rb
CHANGED
@@ -39,7 +39,7 @@ describe 'AbstractControllerBase' do
|
|
39
39
|
expect(appmap).to include(<<-SERVER_REQUEST.strip)
|
40
40
|
http_server_request:
|
41
41
|
request_method: POST
|
42
|
-
path_info: "/api/users
|
42
|
+
path_info: "/api/users"
|
43
43
|
SERVER_REQUEST
|
44
44
|
end
|
45
45
|
it 'Properly captures method parameters in the appmap' do
|
@@ -45,6 +45,12 @@ describe 'AbstractControllerBase' do
|
|
45
45
|
request_method: POST
|
46
46
|
path_info: "/api/users"
|
47
47
|
SERVER_REQUEST
|
48
|
+
|
49
|
+
expect(appmap).to include(<<-SERVER_RESPONSE.strip)
|
50
|
+
http_server_response:
|
51
|
+
status: 201
|
52
|
+
mime_type: application/json; charset=utf-8
|
53
|
+
SERVER_RESPONSE
|
48
54
|
end
|
49
55
|
|
50
56
|
it 'properly captures method parameters in the appmap' do
|
@@ -9,3 +9,47 @@ class ExceptionMethod
|
|
9
9
|
raise 'Exception occurred in raise_exception'
|
10
10
|
end
|
11
11
|
end
|
12
|
+
|
13
|
+
# subclass from BasicObject so we don't get #to_s. Requires some
|
14
|
+
# hackery to implement the other methods normally provided by Object.
|
15
|
+
class NoToSMethod < BasicObject
|
16
|
+
def is_a?(*args)
|
17
|
+
return false
|
18
|
+
end
|
19
|
+
|
20
|
+
def class
|
21
|
+
return ::Class
|
22
|
+
end
|
23
|
+
|
24
|
+
def respond_to?(*args)
|
25
|
+
return false
|
26
|
+
end
|
27
|
+
|
28
|
+
def inspect
|
29
|
+
"NoToSMethod"
|
30
|
+
end
|
31
|
+
|
32
|
+
def say_hello
|
33
|
+
"hello"
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
class InspectRaises < NoToSMethod
|
38
|
+
def inspect
|
39
|
+
::Kernel.raise "#to_s missing, #inspect raises"
|
40
|
+
end
|
41
|
+
|
42
|
+
def say_hello
|
43
|
+
"hello"
|
44
|
+
end
|
45
|
+
end
|
46
|
+
|
47
|
+
class ToSRaises
|
48
|
+
def to_s
|
49
|
+
raise "#to_s raises"
|
50
|
+
end
|
51
|
+
|
52
|
+
def say_hello
|
53
|
+
"hello"
|
54
|
+
end
|
55
|
+
end
|
data/spec/hook_spec.rb
CHANGED
@@ -22,7 +22,7 @@ describe 'AppMap class Hooking', docker: false do
|
|
22
22
|
while tracer.event?
|
23
23
|
events << tracer.next_event.to_h
|
24
24
|
end
|
25
|
-
end.map(&AppMap::Util.method(:sanitize_event))
|
25
|
+
end.map(&AppMap::Util.method(:sanitize_event))
|
26
26
|
end
|
27
27
|
|
28
28
|
def invoke_test_file(file, setup: nil, &block)
|
@@ -50,7 +50,7 @@ describe 'AppMap class Hooking', docker: false do
|
|
50
50
|
def test_hook_behavior(file, events_yaml, setup: nil, &block)
|
51
51
|
config, tracer = invoke_test_file(file, setup: setup, &block)
|
52
52
|
|
53
|
-
events = collect_events(tracer)
|
53
|
+
events = collect_events(tracer).to_yaml
|
54
54
|
|
55
55
|
expect(Diffy::Diff.new(events_yaml, events).to_s).to eq('')
|
56
56
|
|
@@ -465,6 +465,132 @@ describe 'AppMap class Hooking', docker: false do
|
|
465
465
|
end
|
466
466
|
end
|
467
467
|
|
468
|
+
context 'string conversions works for the receiver when' do
|
469
|
+
|
470
|
+
it 'is missing #to_s' do
|
471
|
+
events_yaml = <<~YAML
|
472
|
+
---
|
473
|
+
- :id: 1
|
474
|
+
:event: :call
|
475
|
+
:defined_class: NoToSMethod
|
476
|
+
:method_id: respond_to?
|
477
|
+
:path: spec/fixtures/hook/exception_method.rb
|
478
|
+
:lineno: 24
|
479
|
+
:static: false
|
480
|
+
:parameters:
|
481
|
+
- :name: :args
|
482
|
+
:class: Symbol
|
483
|
+
:value: to_s
|
484
|
+
:kind: :rest
|
485
|
+
:receiver:
|
486
|
+
:class: Class
|
487
|
+
:value: NoToSMethod
|
488
|
+
- :id: 2
|
489
|
+
:event: :return
|
490
|
+
:parent_id: 1
|
491
|
+
- :id: 3
|
492
|
+
:event: :call
|
493
|
+
:defined_class: NoToSMethod
|
494
|
+
:method_id: say_hello
|
495
|
+
:path: spec/fixtures/hook/exception_method.rb
|
496
|
+
:lineno: 32
|
497
|
+
:static: false
|
498
|
+
:parameters: []
|
499
|
+
:receiver:
|
500
|
+
:class: Class
|
501
|
+
:value: NoToSMethod
|
502
|
+
- :id: 4
|
503
|
+
:event: :return
|
504
|
+
:parent_id: 3
|
505
|
+
:return_value:
|
506
|
+
:class: String
|
507
|
+
:value: hello
|
508
|
+
YAML
|
509
|
+
|
510
|
+
test_hook_behavior 'spec/fixtures/hook/exception_method.rb', events_yaml do
|
511
|
+
inst = NoToSMethod.new
|
512
|
+
# sanity check
|
513
|
+
expect(inst).not_to respond_to(:to_s)
|
514
|
+
inst.say_hello
|
515
|
+
end
|
516
|
+
end
|
517
|
+
|
518
|
+
it 'it is missing #to_s and it raises an exception in #inspect' do
|
519
|
+
events_yaml = <<~YAML
|
520
|
+
---
|
521
|
+
- :id: 1
|
522
|
+
:event: :call
|
523
|
+
:defined_class: NoToSMethod
|
524
|
+
:method_id: respond_to?
|
525
|
+
:path: spec/fixtures/hook/exception_method.rb
|
526
|
+
:lineno: 24
|
527
|
+
:static: false
|
528
|
+
:parameters:
|
529
|
+
- :name: :args
|
530
|
+
:class: Symbol
|
531
|
+
:value: to_s
|
532
|
+
:kind: :rest
|
533
|
+
:receiver:
|
534
|
+
:class: Class
|
535
|
+
:value: "*Error inspecting variable*"
|
536
|
+
- :id: 2
|
537
|
+
:event: :return
|
538
|
+
:parent_id: 1
|
539
|
+
- :id: 3
|
540
|
+
:event: :call
|
541
|
+
:defined_class: InspectRaises
|
542
|
+
:method_id: say_hello
|
543
|
+
:path: spec/fixtures/hook/exception_method.rb
|
544
|
+
:lineno: 42
|
545
|
+
:static: false
|
546
|
+
:parameters: []
|
547
|
+
:receiver:
|
548
|
+
:class: Class
|
549
|
+
:value: "*Error inspecting variable*"
|
550
|
+
- :id: 4
|
551
|
+
:event: :return
|
552
|
+
:parent_id: 3
|
553
|
+
:return_value:
|
554
|
+
:class: String
|
555
|
+
:value: hello
|
556
|
+
YAML
|
557
|
+
|
558
|
+
test_hook_behavior 'spec/fixtures/hook/exception_method.rb', events_yaml do
|
559
|
+
inst = InspectRaises.new
|
560
|
+
# sanity check
|
561
|
+
expect(inst).not_to respond_to(:to_s)
|
562
|
+
inst.say_hello
|
563
|
+
end
|
564
|
+
end
|
565
|
+
|
566
|
+
it 'it raises an exception in #to_s' do
|
567
|
+
events_yaml = <<~YAML
|
568
|
+
---
|
569
|
+
- :id: 1
|
570
|
+
:event: :call
|
571
|
+
:defined_class: ToSRaises
|
572
|
+
:method_id: say_hello
|
573
|
+
:path: spec/fixtures/hook/exception_method.rb
|
574
|
+
:lineno: 52
|
575
|
+
:static: false
|
576
|
+
:parameters: []
|
577
|
+
:receiver:
|
578
|
+
:class: ToSRaises
|
579
|
+
:value: "*Error inspecting variable*"
|
580
|
+
- :id: 2
|
581
|
+
:event: :return
|
582
|
+
:parent_id: 1
|
583
|
+
:return_value:
|
584
|
+
:class: String
|
585
|
+
:value: hello
|
586
|
+
YAML
|
587
|
+
|
588
|
+
test_hook_behavior 'spec/fixtures/hook/exception_method.rb', events_yaml do
|
589
|
+
ToSRaises.new.say_hello
|
590
|
+
end
|
591
|
+
end
|
592
|
+
end
|
593
|
+
|
468
594
|
it 're-raises exceptions' do
|
469
595
|
RSpec::Expectations.configuration.on_potential_false_positives = :nothing
|
470
596
|
|
@@ -500,7 +626,7 @@ describe 'AppMap class Hooking', docker: false do
|
|
500
626
|
:event: :call
|
501
627
|
:defined_class: ActiveSupport::SecurityUtils
|
502
628
|
:method_id: secure_compare
|
503
|
-
:path:
|
629
|
+
:path: lib/active_support/security_utils.rb
|
504
630
|
:lineno: 26
|
505
631
|
:static: true
|
506
632
|
:parameters:
|
@@ -598,7 +724,7 @@ describe 'AppMap class Hooking', docker: false do
|
|
598
724
|
:children:
|
599
725
|
- :name: secure_compare
|
600
726
|
:type: function
|
601
|
-
:location:
|
727
|
+
:location: lib/active_support/security_utils.rb:26
|
602
728
|
:static: true
|
603
729
|
:labels:
|
604
730
|
- security
|
@@ -624,7 +750,7 @@ describe 'AppMap class Hooking', docker: false do
|
|
624
750
|
config, tracer = invoke_test_file 'spec/fixtures/hook/compare.rb' do
|
625
751
|
expect(Compare.compare('string', 'string')).to be_truthy
|
626
752
|
end
|
627
|
-
cm = AppMap::ClassMap.build_from_methods(
|
753
|
+
cm = AppMap::Util.sanitize_paths(AppMap::ClassMap.build_from_methods(tracer.event_methods))
|
628
754
|
entry = cm[1][:children][0][:children][0][:children][0]
|
629
755
|
# Sanity check, make sure we got the right one
|
630
756
|
expect(entry[:name]).to eq('secure_compare')
|
@@ -669,4 +795,11 @@ describe 'AppMap class Hooking', docker: false do
|
|
669
795
|
end
|
670
796
|
end
|
671
797
|
end
|
798
|
+
|
799
|
+
it "preserves the arity of hooked methods" do
|
800
|
+
invoke_test_file 'spec/fixtures/hook/instance_method.rb' do
|
801
|
+
expect(InstanceMethod.instance_method(:say_echo).arity).to be(1)
|
802
|
+
expect(InstanceMethod.new.method(:say_echo).arity).to be(1)
|
803
|
+
end
|
804
|
+
end
|
672
805
|
end
|
data/test/cli_test.rb
CHANGED
@@ -56,12 +56,10 @@ class CLITest < Minitest::Test
|
|
56
56
|
Class frequency:
|
57
57
|
----------------
|
58
58
|
1 Main
|
59
|
-
1 IO
|
60
59
|
|
61
60
|
Method frequency:
|
62
61
|
----------------
|
63
62
|
1 Main.say_hello
|
64
|
-
1 IO#write
|
65
63
|
OUTPUT
|
66
64
|
end
|
67
65
|
|
@@ -82,20 +80,12 @@ class CLITest < Minitest::Test
|
|
82
80
|
{
|
83
81
|
"name": "Main",
|
84
82
|
"count": 1
|
85
|
-
},
|
86
|
-
{
|
87
|
-
"name": "IO",
|
88
|
-
"count": 1
|
89
83
|
}
|
90
84
|
],
|
91
85
|
"method_frequency": [
|
92
86
|
{
|
93
87
|
"name": "Main.say_hello",
|
94
88
|
"count": 1
|
95
|
-
},
|
96
|
-
{
|
97
|
-
"name": "IO#write",
|
98
|
-
"count": 1
|
99
89
|
}
|
100
90
|
]
|
101
91
|
}
|
data/test/openssl_test.rb
CHANGED
@@ -79,27 +79,6 @@ class OpenSSLTest < Minitest::Test
|
|
79
79
|
]
|
80
80
|
}
|
81
81
|
]
|
82
|
-
},
|
83
|
-
{
|
84
|
-
"name": "io",
|
85
|
-
"type": "package",
|
86
|
-
"children": [
|
87
|
-
{
|
88
|
-
"name": "IO",
|
89
|
-
"type": "class",
|
90
|
-
"children": [
|
91
|
-
{
|
92
|
-
"name": "write",
|
93
|
-
"type": "function",
|
94
|
-
"location": "IO#write",
|
95
|
-
"static": false,
|
96
|
-
"labels": [
|
97
|
-
"io"
|
98
|
-
]
|
99
|
-
}
|
100
|
-
]
|
101
|
-
}
|
102
|
-
]
|
103
82
|
}
|
104
83
|
]
|
105
84
|
JSON
|
@@ -167,33 +146,6 @@ class OpenSSLTest < Minitest::Test
|
|
167
146
|
"return_value": {
|
168
147
|
"class": "String"
|
169
148
|
}
|
170
|
-
},
|
171
|
-
{
|
172
|
-
"id": 5,
|
173
|
-
"event": "call",
|
174
|
-
"defined_class": "IO",
|
175
|
-
"method_id": "write",
|
176
|
-
"path": "IO#write",
|
177
|
-
"static": false,
|
178
|
-
"parameters": [
|
179
|
-
{
|
180
|
-
"name": "arg",
|
181
|
-
"class": "String",
|
182
|
-
"value": "Computed signature",
|
183
|
-
"kind": "rest"
|
184
|
-
}
|
185
|
-
],
|
186
|
-
"receiver": {
|
187
|
-
"class": "IO"
|
188
|
-
}
|
189
|
-
},
|
190
|
-
{
|
191
|
-
"id": 6,
|
192
|
-
"event": "return",
|
193
|
-
"parent_id": 5,
|
194
|
-
"return_value": {
|
195
|
-
"class": "Integer"
|
196
|
-
}
|
197
149
|
}
|
198
150
|
]
|
199
151
|
JSON
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: appmap
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.35.2
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Kevin Gilpin
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date: 2020-
|
11
|
+
date: 2020-10-01 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: activesupport
|
@@ -276,6 +276,20 @@ dependencies:
|
|
276
276
|
- - ">="
|
277
277
|
- !ruby/object:Gem::Version
|
278
278
|
version: '0'
|
279
|
+
- !ruby/object:Gem::Dependency
|
280
|
+
name: hashie
|
281
|
+
requirement: !ruby/object:Gem::Requirement
|
282
|
+
requirements:
|
283
|
+
- - ">="
|
284
|
+
- !ruby/object:Gem::Version
|
285
|
+
version: '0'
|
286
|
+
type: :development
|
287
|
+
prerelease: false
|
288
|
+
version_requirements: !ruby/object:Gem::Requirement
|
289
|
+
requirements:
|
290
|
+
- - ">="
|
291
|
+
- !ruby/object:Gem::Version
|
292
|
+
version: '0'
|
279
293
|
description:
|
280
294
|
email:
|
281
295
|
- kgilpin@gmail.com
|
@@ -324,7 +338,7 @@ files:
|
|
324
338
|
- lib/appmap/middleware/remote_recording.rb
|
325
339
|
- lib/appmap/minitest.rb
|
326
340
|
- lib/appmap/open.rb
|
327
|
-
- lib/appmap/rails/
|
341
|
+
- lib/appmap/rails/request_handler.rb
|
328
342
|
- lib/appmap/rails/sql_handler.rb
|
329
343
|
- lib/appmap/railtie.rb
|
330
344
|
- lib/appmap/record.rb
|
@@ -1,91 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
require 'appmap/event'
|
4
|
-
|
5
|
-
module AppMap
|
6
|
-
module Rails
|
7
|
-
module ActionHandler
|
8
|
-
Context = Struct.new(:id, :start_time)
|
9
|
-
|
10
|
-
module ContextKey
|
11
|
-
def context_key
|
12
|
-
"#{HTTPServerRequest.name}#call"
|
13
|
-
end
|
14
|
-
end
|
15
|
-
|
16
|
-
class HTTPServerRequest
|
17
|
-
include ContextKey
|
18
|
-
|
19
|
-
class Call < AppMap::Event::MethodCall
|
20
|
-
attr_accessor :payload
|
21
|
-
|
22
|
-
def initialize(payload)
|
23
|
-
super AppMap::Event.next_id_counter, :call, Thread.current.object_id
|
24
|
-
|
25
|
-
self.payload = payload
|
26
|
-
end
|
27
|
-
|
28
|
-
def to_h
|
29
|
-
super.tap do |h|
|
30
|
-
h[:http_server_request] = {
|
31
|
-
request_method: payload[:method],
|
32
|
-
path_info: payload[:path]
|
33
|
-
}
|
34
|
-
|
35
|
-
params = payload[:params]
|
36
|
-
h[:message] = params.keys.map do |key|
|
37
|
-
val = params[key]
|
38
|
-
{
|
39
|
-
name: key,
|
40
|
-
class: val.class.name,
|
41
|
-
value: self.class.display_string(val),
|
42
|
-
object_id: val.__id__
|
43
|
-
}
|
44
|
-
end
|
45
|
-
end
|
46
|
-
end
|
47
|
-
end
|
48
|
-
|
49
|
-
def call(_, started, finished, _, payload) # (name, started, finished, unique_id, payload)
|
50
|
-
event = Call.new(payload)
|
51
|
-
Thread.current[context_key] = Context.new(event.id, Time.now)
|
52
|
-
AppMap.tracing.record_event(event)
|
53
|
-
end
|
54
|
-
end
|
55
|
-
|
56
|
-
class HTTPServerResponse
|
57
|
-
include ContextKey
|
58
|
-
|
59
|
-
class Call < AppMap::Event::MethodReturnIgnoreValue
|
60
|
-
attr_accessor :payload
|
61
|
-
|
62
|
-
def initialize(payload, parent_id, elapsed)
|
63
|
-
super AppMap::Event.next_id_counter, :return, Thread.current.object_id
|
64
|
-
|
65
|
-
self.payload = payload
|
66
|
-
self.parent_id = parent_id
|
67
|
-
self.elapsed = elapsed
|
68
|
-
end
|
69
|
-
|
70
|
-
def to_h
|
71
|
-
super.tap do |h|
|
72
|
-
h[:http_server_response] = {
|
73
|
-
status: payload[:status]
|
74
|
-
}
|
75
|
-
end
|
76
|
-
end
|
77
|
-
end
|
78
|
-
|
79
|
-
def call(_, started, finished, _, payload) # (name, started, finished, unique_id, payload)
|
80
|
-
return unless Thread.current[context_key]
|
81
|
-
|
82
|
-
context = Thread.current[context_key]
|
83
|
-
Thread.current[context_key] = nil
|
84
|
-
|
85
|
-
event = Call.new(payload, context.id, Time.now - context.start_time)
|
86
|
-
AppMap.tracing.record_event(event)
|
87
|
-
end
|
88
|
-
end
|
89
|
-
end
|
90
|
-
end
|
91
|
-
end
|