# collects stats from currently running process module PrometheusExporter::Instrumentation class Process def self.start(client: nil, type: "ruby", frequency: 30) process_collector = new(type) client ||= PrometheusExporter::Client.default Thread.new do while true begin metric = process_collector.collect client.send_json metric rescue => e STDERR.puts("Prometheus Discoruse Failed To Collect Process Stats #{e}") ensure sleep frequency end end end end def initialize(type) @type = type end def collect metric = {} metric[:type] = "process" metric[:process_type] = @type collect_gc_stats(metric) collect_v8_stats(metric) collect_process_stats(metric) metric end def pid @pid = ::Process.pid end def rss @pagesize ||= `getconf PAGESIZE`.to_i rescue 4096 File.read("/proc/#{pid}/statm").split(' ')[1].to_i * @pagesize rescue 0 end def collect_process_stats(metric) metric[:pid] = pid metric[:rss] = rss end def collect_gc_stats(metric) stat = GC.stat metric[:heap_live_slots] = stat[:heap_live_slots] metric[:heap_free_slots] = stat[:heap_free_slots] metric[:major_gc_count] = stat[:major_gc_count] metric[:minor_gc_count] = stat[:minor_gc_count] metric[:total_allocated_objects] = stat[:total_allocated_objects] end def collect_v8_stats(metric) return if !defined? MiniRacer metric[:v8_heap_count] = metric[:v8_heap_size] = 0 metric[:v8_heap_size] = metric[:v8_physical_size] = 0 metric[:v8_used_heap_size] = 0 ObjectSpace.each_object(MiniRacer::Context) do |context| stats = context.heap_stats if stats metric[:v8_heap_count] += 1 metric[:v8_heap_size] += stats[:total_heap_size].to_i metric[:v8_used_heap_size] += stats[:used_heap_size].to_i metric[:v8_physical_size] += stats[:total_physical_size].to_i end end end end end