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 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,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/gem_tasks'
4
+ require 'rubocop/rake_task'
5
+
6
+ RuboCop::RakeTask.new
7
+
8
+ task default: :rubocop
@@ -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
@@ -0,0 +1,3 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'leak_profiler/leak_profiler'
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: []