puma-status-forked 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: a084f07e47e934496b5b8fdd8a251166fbffdb9fdbeefc53a420cf49d9753bb5
4
+ data.tar.gz: a4cfb6c706590a8e25d021c5389d9d3199ca78224ad0167bc58ec85204dbe3c8
5
+ SHA512:
6
+ metadata.gz: 00e4d9eb55b42d3c943c0060433088fd6460d24d149200862b927a7089da5c99b23d7e530eac27797a2ac1fbedaa50a64d69dc9ba1fcef1b798301e9c2785037
7
+ data.tar.gz: 910eb9778e282a78b47c833ea761adc7ec3a82e08a7139288ced5a56d2ac193ddfd1595dfc11157ec175343a433034492b8bd94dbd19d16692a7a65f36396c39
data/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) [year] [fullname]
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
data/bin/puma-status ADDED
@@ -0,0 +1,5 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'puma-status'
4
+
5
+ run
data/lib/core.rb ADDED
@@ -0,0 +1,110 @@
1
+ require 'yaml'
2
+ require 'json'
3
+ require 'net_x/http_unix'
4
+ require 'openssl'
5
+ require 'time'
6
+ require 'open3'
7
+ require_relative 'stats'
8
+
9
+ def get_stats(state_file_path)
10
+ puma_state = YAML.load_file(state_file_path)
11
+
12
+ uri = URI.parse(puma_state["control_url"])
13
+
14
+ address = if uri.scheme =~ /unix/i
15
+ [uri.scheme, '://', uri.host, uri.path].join
16
+ else
17
+ [uri.host, uri.path].join
18
+ end
19
+
20
+ client = NetX::HTTPUnix.new(address, uri.port)
21
+
22
+ if uri.scheme =~ /ssl/i
23
+ client.use_ssl = true
24
+ client.verify_mode = OpenSSL::SSL::VERIFY_NONE if ENV['SSL_NO_VERIFY'] == '1'
25
+ end
26
+
27
+ req = Net::HTTP::Get.new("/stats?token=#{puma_state["control_auth_token"]}")
28
+ resp = client.request(req)
29
+ raw_stats = JSON.parse(resp.body)
30
+ debug raw_stats
31
+ stats = Stats.new(raw_stats)
32
+
33
+ hydrate_stats(stats, puma_state, state_file_path)
34
+ end
35
+
36
+ def get_memory_from_top(raw_memory)
37
+ case raw_memory[-1].downcase
38
+ when 'g'
39
+ (raw_memory[0...-1].to_f*1024).to_i
40
+ when 'm'
41
+ raw_memory[0...-1].to_i
42
+ else
43
+ raw_memory.to_i/1024
44
+ end
45
+ end
46
+
47
+ PID_COLUMN = 0
48
+ MEM_COLUMN = 5
49
+ CPU_COLUMN = 8
50
+ OPEN3_STDOUT = 1
51
+
52
+ def get_top_stats(pids)
53
+ pids.each_slice(19).inject({}) do |res, pids19|
54
+ top_result = Open3.popen3({ 'LC_ALL' => 'C' }, "top -b -n 1 -p #{pids19.map(&:to_i).join(',')}")[OPEN3_STDOUT].read
55
+ top_result.split("\n").last(pids19.length).map { |row| r = row.split(' '); [r[PID_COLUMN].to_i, get_memory_from_top(r[MEM_COLUMN]), r[CPU_COLUMN].to_f] }
56
+ .inject(res) { |hash, row| hash[row[0]] = { mem: row[1], pcpu: row[2] }; hash }
57
+ res
58
+ end
59
+ end
60
+
61
+ def hydrate_stats(stats, puma_state, state_file_path)
62
+ stats.pid = puma_state['pid']
63
+ stats.state_file_path = state_file_path
64
+
65
+ workers_pids = stats.workers.map(&:pid)
66
+
67
+ top_stats = get_top_stats(workers_pids)
68
+
69
+ stats.tap do |s|
70
+ stats.workers.map do |wstats|
71
+ wstats.mem = top_stats.dig(wstats.pid, :mem) || 0
72
+ wstats.pcpu = top_stats.dig(wstats.pid, :pcpu) || 0
73
+ wstats.killed = !top_stats.key?(wstats.pid) || (wstats.mem <=0 && wstats.pcpu <= 0)
74
+ end
75
+ end
76
+ end
77
+
78
+ def format_stats(stats)
79
+ master_line = "#{stats.pid} (#{stats.state_file_path})"
80
+ master_line += " Version: #{stats.version} |" if stats.version
81
+ master_line += " Uptime: #{seconds_to_human(stats.uptime)}"
82
+ master_line += " | Phase: #{stats.phase}" if stats.phase
83
+
84
+ if stats.booting?
85
+ master_line += " #{yellow("booting")}"
86
+ else
87
+ master_line += " | Load: #{color(75, 50, stats.load, asciiThreadLoad(stats.running_threads, stats.spawned_threads, stats.max_threads))}"
88
+ master_line += " | Req: #{stats.requests_count}" if stats.requests_count
89
+ end
90
+
91
+ output = [master_line] + stats.workers.map do |wstats|
92
+ worker_line = " └ #{wstats.pid.to_s.rjust(5, ' ')} CPU: #{color(75, 50, wstats.pcpu, wstats.pcpu.to_s.rjust(5, ' '))}% Mem: #{color(1000, 750, wstats.mem, wstats.mem.to_s.rjust(4, ' '))} MB Uptime: #{seconds_to_human(wstats.uptime)}"
93
+
94
+ if wstats.booting?
95
+ worker_line += " #{yellow("booting")}"
96
+ elsif wstats.killed?
97
+ worker_line += " #{red("killed")}"
98
+ else
99
+ worker_line += " | Load: #{color(75, 50, wstats.load, asciiThreadLoad(wstats.running_threads, wstats.spawned_threads, wstats.max_threads))}"
100
+ worker_line += " | Phase: #{red(wstats.phase)}" if wstats.phase != stats.phase
101
+ worker_line += " | Req: #{wstats.requests_count}" if wstats.requests_count
102
+ worker_line += " Queue: #{red(wstats.backlog.to_s)}" if wstats.backlog > 0
103
+ worker_line += " Last checkin: #{red(wstats.last_checkin)}" if wstats.last_checkin >= 10
104
+ end
105
+
106
+ worker_line
107
+ end
108
+
109
+ output.join("\n")
110
+ end
data/lib/helpers.rb ADDED
@@ -0,0 +1,64 @@
1
+ require 'rainbow'
2
+
3
+ def debug(str)
4
+ puts str if ENV.key?('DEBUG')
5
+ end
6
+
7
+ def yellow(str)
8
+ rainbow(str, :yellow)
9
+ end
10
+
11
+ def red(str)
12
+ rainbow(str, :red)
13
+ end
14
+
15
+ def rainbow(str, color_name)
16
+ return str if ENV.key?('NO_COLOR')
17
+ Rainbow(str.to_s).send(color_name)
18
+ end
19
+
20
+ def color(critical, warn, value, str = nil)
21
+ str = value unless str
22
+ color_level = if value >= critical
23
+ :red
24
+ elsif value < critical && value >= warn
25
+ :yellow
26
+ else
27
+ :green
28
+ end
29
+ rainbow(str, color_level)
30
+ end
31
+
32
+ def asciiThreadLoad(running, spawned, total)
33
+ full = "█"
34
+ half= "░"
35
+ empty = " "
36
+
37
+ full_count = running
38
+ half_count = [spawned - running, 0].max
39
+ empty_count = total - half_count - full_count
40
+
41
+ "#{running}[#{full*full_count}#{half*half_count}#{empty*empty_count}]#{total}"
42
+ end
43
+
44
+ def seconds_to_human(seconds)
45
+
46
+ #=> 0m 0s
47
+ #=> 59m59s
48
+ #=> 1h 0m
49
+ #=> 23h59m
50
+ #=> 1d 0h
51
+ #=> 24d
52
+
53
+ if seconds <= 0
54
+ "--m--s"
55
+ elsif seconds < 60*60
56
+ "#{(seconds/60).to_s.rjust(2, ' ')}m#{(seconds%60).to_s.rjust(2, ' ')}s"
57
+ elsif seconds >= 60*60*1 && seconds < 60*60*24
58
+ "#{(seconds/(60*60*1)).to_s.rjust(2, ' ')}h#{((seconds%(60*60*1))/60).to_s.rjust(2, ' ')}m"
59
+ elsif seconds > 60*60*24 && seconds < 60*60*24*10
60
+ "#{(seconds/(60*60*24)).to_s.rjust(2, ' ')}d#{((seconds%(60*60*24))/(60*60*1)).to_s.rjust(2, ' ')}h"
61
+ else
62
+ "#{seconds/(60*60*24)}d".rjust(6, ' ')
63
+ end
64
+ end
@@ -0,0 +1,48 @@
1
+ require_relative './helpers'
2
+ require_relative './core.rb'
3
+ require 'parallel'
4
+
5
+ def run
6
+ debug "puma-status"
7
+
8
+ if ARGV.count < 1
9
+ puts "Call with:"
10
+ puts "\tpuma-status path/to/puma.state"
11
+ exit -1
12
+ end
13
+
14
+ errors = []
15
+
16
+ outputs = Parallel.map(ARGV, in_threads: ARGV.count) do |state_file_path|
17
+ begin
18
+ debug "State file: #{state_file_path}"
19
+ format_stats(get_stats(state_file_path))
20
+ rescue Errno::ENOENT => e
21
+ if e.message =~ /#{state_file_path}/
22
+ errors << "#{yellow(state_file_path)} doesn't exist"
23
+ elsif e.message =~ /connect\(2\) for [^\/]/
24
+ errors << "#{yellow("Relative Unix socket")}: the Unix socket of the control app has a relative path. Please, ensure you are running from the same folder as puma."
25
+ else
26
+ errors << "#{red(state_file_path)} an unhandled error occured: #{e.inspect}"
27
+ end
28
+ nil
29
+ rescue Errno::EISDIR => e
30
+ if e.message =~ /#{state_file_path}/
31
+ errors << "#{yellow(state_file_path)} isn't a state file"
32
+ else
33
+ errors << "#{red(state_file_path)} an unhandled error occured: #{e.inspect}"
34
+ end
35
+ nil
36
+ rescue => e
37
+ errors << "#{red(state_file_path)} an unhandled error occured: #{e.inspect}"
38
+ nil
39
+ end
40
+ end
41
+
42
+ outputs.compact.each { |output| puts output }
43
+
44
+ if errors.any?
45
+ puts ""
46
+ errors.each { |error| puts error }
47
+ end
48
+ end
data/lib/stats.rb ADDED
@@ -0,0 +1,163 @@
1
+ class Stats
2
+
3
+ class Worker
4
+ def initialize(wstats)
5
+ @wstats = wstats
6
+ end
7
+
8
+ def pid
9
+ @wstats['pid']
10
+ end
11
+
12
+ def killed=(killed)
13
+ @wstats['killed'] = killed
14
+ end
15
+
16
+ def killed?
17
+ !!@wstats['killed']
18
+ end
19
+
20
+ def mem=(mem)
21
+ @wstats['mem'] = mem
22
+ end
23
+
24
+ def mem
25
+ @wstats['mem']
26
+ end
27
+
28
+ def pcpu=(pcpu)
29
+ @wstats['pcpu'] = pcpu
30
+ end
31
+
32
+ def pcpu
33
+ @wstats['pcpu']
34
+ end
35
+
36
+ def booting?
37
+ @wstats.key?('last_status') && @wstats['last_status'].empty?
38
+ end
39
+
40
+ def running
41
+ @wstats.dig('last_status', 'running') || @wstats['running'] || 0
42
+ end
43
+ alias :total_threads :running
44
+ alias :spawned_threads :running
45
+
46
+ def max_threads
47
+ @wstats.dig('last_status', 'max_threads') || @wstats['max_threads'] || 0
48
+ end
49
+ alias :total_threads :max_threads
50
+
51
+ def pool_capacity
52
+ @wstats.dig('last_status', 'pool_capacity') || @wstats['pool_capacity'] || 0
53
+ end
54
+
55
+ def running_threads
56
+ max_threads - pool_capacity
57
+ end
58
+
59
+ def phase
60
+ @wstats['phase']
61
+ end
62
+
63
+ def load
64
+ running_threads/total_threads.to_f*100
65
+ end
66
+
67
+ def uptime
68
+ return 0 unless @wstats.key?('started_at')
69
+ (Time.now - Time.parse(@wstats['started_at'])).to_i
70
+ end
71
+
72
+ def requests_count
73
+ @wstats.dig('last_status', 'requests_count') || @wstats['requests_count']
74
+ end
75
+
76
+ def backlog
77
+ @wstats.dig('last_status', 'backlog') || 0
78
+ end
79
+
80
+ def last_checkin
81
+ (Time.now - Time.parse(@wstats['last_checkin'])).round
82
+ rescue
83
+ 0
84
+ end
85
+ end
86
+
87
+ def initialize(stats)
88
+ @stats = stats
89
+ end
90
+
91
+ def workers
92
+ @workers ||= (@stats['worker_status'] || [@stats]).map { |wstats| Worker.new(wstats) }
93
+ end
94
+
95
+ def pid=(pid)
96
+ @stats['pid'] = pid
97
+ end
98
+
99
+ def pid
100
+ @stats['pid']
101
+ end
102
+
103
+ def state_file_path=(state_file_path)
104
+ @stats['state_file_path'] = state_file_path
105
+ end
106
+
107
+ def state_file_path
108
+ @stats['state_file_path']
109
+ end
110
+
111
+ def uptime
112
+ return 0 unless @stats.key?('started_at')
113
+ (Time.now - Time.parse(@stats['started_at'])).to_i
114
+ end
115
+
116
+ def booting?
117
+ workers.all?(&:booting?)
118
+ end
119
+
120
+ def total_threads
121
+ workers.reduce(0) { |total, wstats| total + wstats.max_threads }
122
+ end
123
+
124
+ def running_threads
125
+ workers.reduce(0) { |total, wstats| total + wstats.running_threads }
126
+ end
127
+
128
+ def spawned_threads
129
+ workers.reduce(0) { |total, wstats| total + wstats.spawned_threads }
130
+ end
131
+
132
+ def max_threads
133
+ workers.reduce(0) { |total, wstats| total + wstats.max_threads }
134
+ end
135
+
136
+ def requests_count
137
+ workers_with_requests_count = workers.select(&:requests_count)
138
+ return if workers_with_requests_count.none?
139
+ workers_with_requests_count.reduce(0) { |total, wstats| total + wstats.requests_count }
140
+ end
141
+
142
+ def running
143
+ @stats['running'] || 0
144
+ end
145
+
146
+ def pool_capacity
147
+ @stats['pool_capacity'] || 0
148
+ end
149
+
150
+ def phase
151
+ @stats['phase']
152
+ end
153
+
154
+ def load
155
+ running_threads/total_threads.to_f*100
156
+ end
157
+
158
+ def version
159
+ return nil unless @stats.key?('versions')
160
+
161
+ "#{@stats['versions']['puma']}/#{@stats['versions']['ruby']['engine']}#{@stats['versions']['ruby']['version']}p#{@stats['versions']['ruby']['patchlevel']}"
162
+ end
163
+ end
metadata ADDED
@@ -0,0 +1,134 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: puma-status-forked
3
+ version: !ruby/object:Gem::Version
4
+ version: '1.0'
5
+ platform: ruby
6
+ authors:
7
+ - Yoann Lecuyer
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2024-05-09 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rainbow
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: 3.1.1
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: 3.1.1
27
+ - !ruby/object:Gem::Dependency
28
+ name: net_http_unix
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '0.2'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '0.2'
41
+ - !ruby/object:Gem::Dependency
42
+ name: parallel
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '1'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '1'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rspec
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '3.8'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '3.8'
69
+ - !ruby/object:Gem::Dependency
70
+ name: climate_control
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '0.2'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '0.2'
83
+ - !ruby/object:Gem::Dependency
84
+ name: timecop
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - "~>"
88
+ - !ruby/object:Gem::Version
89
+ version: '0.9'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - "~>"
95
+ - !ruby/object:Gem::Version
96
+ version: '0.9'
97
+ description:
98
+ email:
99
+ executables:
100
+ - puma-status
101
+ extensions: []
102
+ extra_rdoc_files: []
103
+ files:
104
+ - LICENSE
105
+ - bin/puma-status
106
+ - lib/core.rb
107
+ - lib/helpers.rb
108
+ - lib/puma-status-forked.rb
109
+ - lib/stats.rb
110
+ homepage: https://github.com/robin249/puma-status
111
+ licenses:
112
+ - MIT
113
+ metadata: {}
114
+ post_install_message:
115
+ rdoc_options: []
116
+ require_paths:
117
+ - lib
118
+ required_ruby_version: !ruby/object:Gem::Requirement
119
+ requirements:
120
+ - - ">="
121
+ - !ruby/object:Gem::Version
122
+ version: 2.6.0
123
+ required_rubygems_version: !ruby/object:Gem::Requirement
124
+ requirements:
125
+ - - ">="
126
+ - !ruby/object:Gem::Version
127
+ version: '0'
128
+ requirements: []
129
+ rubygems_version: 3.5.6
130
+ signing_key:
131
+ specification_version: 4
132
+ summary: Command-line tool for puma to display information about running request/process.
133
+ We have replaced the colorize gem by rainbow gem due to GPL license issue.
134
+ test_files: []