leak_profiler 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.rubocop.yml +36 -0
- data/.rubocop_todo.yml +27 -0
- data/README.md +85 -0
- data/Rakefile +8 -0
- data/lib/leak_profiler/allocations.rb +97 -0
- data/lib/leak_profiler/leak_profiler.rb +26 -0
- data/lib/leak_profiler/memory_usage.rb +28 -0
- data/lib/leak_profiler.rb +3 -0
- metadata +50 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 858a9f5099616dfa89103dfdc39b9ea1d3ea1c57f2a0d9cd70f2567e7d67e977
|
4
|
+
data.tar.gz: a53f3aeedb0d9d64929dbfa8dde45768b164f3fbbb0adf4952ca1f75a39e5c4a
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: a8940ecc511d02760bb585e6c2bdb82740a97fd47264924e6ad233a0eebcd90b90724e5e11915543c24e1d60dbafd988e38de80d2b775898e74ac1c9e0c786a0
|
7
|
+
data.tar.gz: 778f50f3c181d01dfd90a067f47959feb31a0998df767fee645b60019535cbb0dc0663fc785f2b99adb3039eafd9fddf6d0a1d2291254f0c2f65202aaaf18c97
|
data/.rubocop.yml
ADDED
@@ -0,0 +1,36 @@
|
|
1
|
+
inherit_from: .rubocop_todo.yml
|
2
|
+
|
3
|
+
AllCops:
|
4
|
+
EnabledByDefault: true
|
5
|
+
TargetRubyVersion: 3.1
|
6
|
+
SuggestExtensions: false
|
7
|
+
|
8
|
+
Bundler/GemComment:
|
9
|
+
Enabled: false
|
10
|
+
|
11
|
+
Metrics:
|
12
|
+
Enabled: false
|
13
|
+
|
14
|
+
Style/Copyright:
|
15
|
+
Enabled: false
|
16
|
+
|
17
|
+
Style/MissingElse:
|
18
|
+
Enabled: false
|
19
|
+
|
20
|
+
Style/StringLiterals:
|
21
|
+
EnforcedStyle: single_quotes
|
22
|
+
|
23
|
+
Style/StringLiteralsInInterpolation:
|
24
|
+
EnforcedStyle: double_quotes
|
25
|
+
|
26
|
+
Layout/MultilineAssignmentLayout:
|
27
|
+
Enabled: false
|
28
|
+
|
29
|
+
Layout/LineLength:
|
30
|
+
Max: 180
|
31
|
+
|
32
|
+
Lint/ConstantResolution:
|
33
|
+
Enabled: false
|
34
|
+
|
35
|
+
Lint/SuppressedException:
|
36
|
+
Enabled: false
|
data/.rubocop_todo.yml
ADDED
@@ -0,0 +1,27 @@
|
|
1
|
+
# This configuration was generated by
|
2
|
+
# `rubocop --auto-gen-config`
|
3
|
+
# on 2025-03-31 02:21:42 UTC using RuboCop version 1.75.1.
|
4
|
+
# The point is for the user to remove these configuration records
|
5
|
+
# one by one as the offenses are removed from the code base.
|
6
|
+
# Note that changes in the inspected code, or installation of new
|
7
|
+
# versions of RuboCop, may require this file to be generated again.
|
8
|
+
|
9
|
+
# Offense count: 3
|
10
|
+
# Configuration parameters: AllowedConstants.
|
11
|
+
Style/Documentation:
|
12
|
+
Exclude:
|
13
|
+
- 'spec/**/*'
|
14
|
+
- 'test/**/*'
|
15
|
+
- 'lib/leak_profiler/allocations.rb'
|
16
|
+
- 'lib/leak_profiler/leak_profiler.rb'
|
17
|
+
- 'lib/leak_profiler/memory_usage.rb'
|
18
|
+
|
19
|
+
# Offense count: 4
|
20
|
+
# Configuration parameters: AllowedMethods, RequireForNonPublicMethods.
|
21
|
+
Style/DocumentationMethod:
|
22
|
+
Exclude:
|
23
|
+
- 'spec/**/*'
|
24
|
+
- 'test/**/*'
|
25
|
+
- 'lib/leak_profiler/allocations.rb'
|
26
|
+
- 'lib/leak_profiler/leak_profiler.rb'
|
27
|
+
- 'lib/leak_profiler/memory_usage.rb'
|
data/README.md
ADDED
@@ -0,0 +1,85 @@
|
|
1
|
+
# LeakProfiler
|
2
|
+
|
3
|
+
This is a Ruby gem for profiling memory leaks in Ruby applications.
|
4
|
+
It provides tools to help identify and analyze memory usage patterns, making it easier to find and fix memory leaks.
|
5
|
+
|
6
|
+
## Installation
|
7
|
+
|
8
|
+
Install the gem and add to the application's Gemfile by executing:
|
9
|
+
|
10
|
+
```bash
|
11
|
+
bundle add leak_profiler
|
12
|
+
```
|
13
|
+
|
14
|
+
If bundler is not being used to manage dependencies, install the gem by executing:
|
15
|
+
|
16
|
+
```bash
|
17
|
+
gem install leak_profiler
|
18
|
+
```
|
19
|
+
|
20
|
+
## Usage
|
21
|
+
|
22
|
+
```ruby
|
23
|
+
require 'leak_profiler'
|
24
|
+
|
25
|
+
LeakProfiler.new.report.report_rss
|
26
|
+
|
27
|
+
# ... your code that may have memory leaks ...
|
28
|
+
```
|
29
|
+
|
30
|
+
### `LeakProfiler#new`
|
31
|
+
* Arguments:
|
32
|
+
* `output_dir` (default `./leak_profiler`): Specify the output directory for report.
|
33
|
+
|
34
|
+
### `LeakProfiler#report`
|
35
|
+
This method outputs where the object was allocated and where it is referenced, like:
|
36
|
+
|
37
|
+
```
|
38
|
+
Allocations ================================================================================
|
39
|
+
/home/watson/src/fluentd/lib/fluent/plugin_helper/thread.rb:70 retains 1098464 bytes, allocations 165 objects
|
40
|
+
/home/watson/src/fluentd/lib/fluent/plugin/metrics_local.rb:58 retains 56400 bytes, allocations 50 objects
|
41
|
+
/home/watson/.rbenv/versions/3.4.2/lib/ruby/3.4.0/open3.rb:534 retains 50080 bytes, allocations 544 objects
|
42
|
+
/home/watson/src/fluentd/lib/fluent/msgpack_factory.rb:105 retains 44400 bytes, allocations 50 objects
|
43
|
+
/home/watson/.rbenv/versions/3.4.2/lib/ruby/site_ruby/3.4.0/rubygems/specification.rb:1093 retains 43846 bytes, allocations 409 objects
|
44
|
+
/home/watson/src/fluentd/lib/fluent/plugin.rb:181 retains 40960 bytes, allocations 23 objects
|
45
|
+
/home/watson/src/fluentd/lib/fluent/msgpack_factory.rb:99 retains 36000 bytes, allocations 200 objects
|
46
|
+
/home/watson/.rbenv/versions/3.4.2/lib/ruby/3.4.0/open3.rb:535 retains 29400 bytes, allocations 49 objects
|
47
|
+
/home/watson/.rbenv/versions/3.4.2/lib/ruby/gems/3.4.0/gems/csv-3.3.3/lib/csv/writer.rb:154 retains 26820 bytes, allocations 56 objects
|
48
|
+
/home/watson/.rbenv/versions/3.4.2/lib/ruby/gems/3.4.0/gems/csv-3.3.3/lib/csv.rb:2983 retains 26560 bytes, allocations 61 objects
|
49
|
+
Referrers --------------------------------------------------------------------------------
|
50
|
+
/home/watson/src/fluentd/lib/fluent/plugin_helper/thread.rb:70 object is referred at:
|
51
|
+
Fiber (allocated at /home/watson/src/fluentd/lib/fluent/plugin/metrics_local.rb:58)
|
52
|
+
NameError::message (allocated at /home/watson/.rbenv/versions/3.4.2/lib/ruby/3.4.0/open3.rb:534)
|
53
|
+
NoMethodError (allocated at /home/watson/.rbenv/versions/3.4.2/lib/ruby/3.4.0/open3.rb:534)
|
54
|
+
Fluent::PluginHelper::ChildProcess::ProcessInfo (allocated at /home/watson/src/fluentd/lib/fluent/plugin_helper/child_process.rb:355)
|
55
|
+
Hash (allocated at /home/watson/src/fluentd/lib/fluent/plugin/formatter_csv.rb:55)
|
56
|
+
...
|
57
|
+
```
|
58
|
+
|
59
|
+
**Allocations**: This section lists the locations in the code where object was allocated, along with the size of the allocation and the number of objects allocated.
|
60
|
+
|
61
|
+
**Referrers**: This section lists the locations in the code where the object is referenced.
|
62
|
+
|
63
|
+
* Arguments:
|
64
|
+
* `interval` (default `30`): The interval in seconds for report.
|
65
|
+
* `max_allocations` (default `10`): Outputs the specified number of objects that use a lot of memory.
|
66
|
+
* `max_referrers` (default `3`): Outputs the number of references in order of the amount of memory used.
|
67
|
+
* `report` (defalut `nil`): Specify the logger object if you want to use custom logger.
|
68
|
+
|
69
|
+
### `LeakProfiler#report_rss`
|
70
|
+
This method outputs the RSS (Resident Set Size) of the process with CSV format, like:
|
71
|
+
|
72
|
+
```
|
73
|
+
elapsed [sec],memory usage (rss) [MB]
|
74
|
+
0,47.20703125
|
75
|
+
1,53.40234375
|
76
|
+
2,55.02734375
|
77
|
+
3,55.90234375
|
78
|
+
```
|
79
|
+
|
80
|
+
* Arguments:
|
81
|
+
* `interval` (default `1`): The interval in seconds for report.
|
82
|
+
|
83
|
+
## Contributing
|
84
|
+
|
85
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/Watson1978/leak_profiler.
|
data/Rakefile
ADDED
@@ -0,0 +1,97 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'objspace'
|
4
|
+
|
5
|
+
class LeakProfiler
|
6
|
+
class Allocations
|
7
|
+
def initialize(logger:, interval:, max_allocations:, max_referrers:)
|
8
|
+
@logger = logger
|
9
|
+
@interval = interval
|
10
|
+
@max_allocations = max_allocations
|
11
|
+
@max_referrers = max_referrers
|
12
|
+
end
|
13
|
+
|
14
|
+
def report
|
15
|
+
Thread.start do
|
16
|
+
loop do
|
17
|
+
allocations = {}
|
18
|
+
|
19
|
+
ObjectSpace.trace_object_allocations_start
|
20
|
+
sleep(@interval)
|
21
|
+
ObjectSpace.trace_object_allocations_stop
|
22
|
+
|
23
|
+
ObjectSpace.each_object.each do |obj|
|
24
|
+
key = allocated_location(obj)
|
25
|
+
next unless key
|
26
|
+
|
27
|
+
allocations[key] ||= {}
|
28
|
+
allocations[key][:metrics] ||= Hash.new { |h, k| h[k] = 0 }
|
29
|
+
allocations[key][:metrics][:count] += 1
|
30
|
+
allocations[key][:metrics][:bytes] += ObjectSpace.memsize_of(obj)
|
31
|
+
|
32
|
+
allocations[key][:sample_object] = obj
|
33
|
+
end
|
34
|
+
|
35
|
+
report_allocations(allocations)
|
36
|
+
report_referrer_objects(allocations)
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
private
|
42
|
+
|
43
|
+
def report_allocations(allocations)
|
44
|
+
return if @max_allocations <= 0
|
45
|
+
|
46
|
+
@logger.add(Logger::Severity::INFO, "Allocations #{"=" * 80}")
|
47
|
+
sort(allocations).take(@max_allocations).each do |key, value|
|
48
|
+
@logger.add(Logger::Severity::INFO, "#{key} retains #{value[:metrics][:bytes]} bytes, allocations #{value[:metrics][:count]} objects")
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
def report_referrer_objects(allocations)
|
53
|
+
return if @max_referrers <= 0
|
54
|
+
|
55
|
+
@logger.add(Logger::Severity::INFO, "Referrers #{"-" * 80}")
|
56
|
+
sort(allocations).take(@max_referrers).each do |key, value|
|
57
|
+
referrer_objects = detect_referrer_objects(value[:sample_object])
|
58
|
+
|
59
|
+
logs = referrer_objects.map do |r|
|
60
|
+
" #{r[:referrer_object].class} (allocated at #{r[:referrer_object_allocated_line]})"
|
61
|
+
end
|
62
|
+
|
63
|
+
@logger.add(Logger::Severity::INFO, "#{key} object is referred at:")
|
64
|
+
logs.uniq.each do |log|
|
65
|
+
@logger.add(Logger::Severity::INFO, log)
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def detect_referrer_objects(object)
|
71
|
+
referrer_objects = []
|
72
|
+
ObjectSpace.each_object.each do |obj|
|
73
|
+
r = ObjectSpace.reachable_objects_from(obj)
|
74
|
+
begin
|
75
|
+
if r&.include?(object)
|
76
|
+
key = allocated_location(obj)
|
77
|
+
next unless key
|
78
|
+
|
79
|
+
referrer_objects << { referrer_object: obj, referrer_object_allocated_line: key }
|
80
|
+
end
|
81
|
+
rescue StandardError
|
82
|
+
end
|
83
|
+
end
|
84
|
+
referrer_objects
|
85
|
+
end
|
86
|
+
|
87
|
+
def allocated_location(obj)
|
88
|
+
return unless ObjectSpace.allocation_sourcefile(obj)
|
89
|
+
|
90
|
+
"#{ObjectSpace.allocation_sourcefile(obj)}:#{ObjectSpace.allocation_sourceline(obj)}"
|
91
|
+
end
|
92
|
+
|
93
|
+
def sort(allocations)
|
94
|
+
allocations.sort_by { |_, v| -v[:metrics][:bytes] }
|
95
|
+
end
|
96
|
+
end
|
97
|
+
end
|
@@ -0,0 +1,26 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative 'allocations'
|
4
|
+
require_relative 'memory_usage'
|
5
|
+
require 'fileutils'
|
6
|
+
|
7
|
+
class LeakProfiler
|
8
|
+
def initialize(output_dir: './leak_profiler')
|
9
|
+
@output_dir = output_dir
|
10
|
+
|
11
|
+
FileUtils.mkdir_p(@output_dir)
|
12
|
+
end
|
13
|
+
|
14
|
+
def report(interval: 30, max_allocations: 10, max_referrers: 3, logger: nil)
|
15
|
+
logger ||= Logger.new(File.join(@output_dir, "leak_profiler-#{Process.pid}.log"))
|
16
|
+
LeakProfiler::Allocations.new(logger: logger, interval: interval, max_allocations: max_allocations, max_referrers: max_referrers).report
|
17
|
+
|
18
|
+
self
|
19
|
+
end
|
20
|
+
|
21
|
+
def report_rss(interval: 1)
|
22
|
+
LeakProfiler::MemoryUsage.new(output_dir: @output_dir, interval: interval).report
|
23
|
+
|
24
|
+
self
|
25
|
+
end
|
26
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class LeakProfiler
|
4
|
+
class MemoryUsage
|
5
|
+
def initialize(output_dir:, interval:)
|
6
|
+
@output_dir = output_dir
|
7
|
+
@interval = interval
|
8
|
+
end
|
9
|
+
|
10
|
+
def report
|
11
|
+
pid = Process.pid
|
12
|
+
|
13
|
+
Thread.new do
|
14
|
+
i = 0
|
15
|
+
File.open(File.expand_path(File.join(@output_dir, "memory-usage-#{pid}.csv")), 'w') do |f|
|
16
|
+
f.puts('elapsed [sec],memory usage (rss) [MB]')
|
17
|
+
|
18
|
+
loop do
|
19
|
+
rss = Integer(`ps -o rss= -p #{pid}`) / 1024.0
|
20
|
+
f.puts("#{i},#{rss}")
|
21
|
+
i += @interval
|
22
|
+
sleep(@interval)
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
metadata
ADDED
@@ -0,0 +1,50 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: leak_profiler
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Watson
|
8
|
+
bindir: bin
|
9
|
+
cert_chain: []
|
10
|
+
date: 2025-03-31 00:00:00.000000000 Z
|
11
|
+
dependencies: []
|
12
|
+
description: A simple profiler for Ruby to detect memory leak.
|
13
|
+
email:
|
14
|
+
- watson1978@gmail.com
|
15
|
+
executables: []
|
16
|
+
extensions: []
|
17
|
+
extra_rdoc_files: []
|
18
|
+
files:
|
19
|
+
- ".rubocop.yml"
|
20
|
+
- ".rubocop_todo.yml"
|
21
|
+
- README.md
|
22
|
+
- Rakefile
|
23
|
+
- lib/leak_profiler.rb
|
24
|
+
- lib/leak_profiler/allocations.rb
|
25
|
+
- lib/leak_profiler/leak_profiler.rb
|
26
|
+
- lib/leak_profiler/memory_usage.rb
|
27
|
+
homepage: https://github.com/Watson1978/leak_profiler
|
28
|
+
licenses: []
|
29
|
+
metadata:
|
30
|
+
homepage_uri: https://github.com/Watson1978/leak_profiler
|
31
|
+
source_code_uri: https://github.com/Watson1978/leak_profiler
|
32
|
+
rubygems_mfa_required: 'true'
|
33
|
+
rdoc_options: []
|
34
|
+
require_paths:
|
35
|
+
- lib
|
36
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - ">="
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: 3.1.0
|
41
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
42
|
+
requirements:
|
43
|
+
- - ">="
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: '0'
|
46
|
+
requirements: []
|
47
|
+
rubygems_version: 3.6.6
|
48
|
+
specification_version: 4
|
49
|
+
summary: A simple profiler for Ruby to detect memory leak.
|
50
|
+
test_files: []
|