rack-healthz 0.1.0 → 0.2.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/Gemfile.lock +6 -0
- data/lib/rack/healthz.rb +38 -63
- data/lib/rack/healthz/accumulator.rb +108 -0
- data/lib/rack/healthz/request_accumulator.rb +4 -2
- data/lib/rack/healthz/response.rb +39 -0
- data/lib/rack/healthz/version.rb +1 -1
- data/rack-healthz.gemspec +1 -0
- metadata +18 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a57636e6570937379f557ffe6ae23130896cd754
|
4
|
+
data.tar.gz: 4d7f5099aad1b246100297615f972a1c7aa2a223
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 44d8de20b2f9b21afbe0985b87c76cdeabd7b125a9efda9f0b5a3870c55f5326897fbdc2e0117ef467dc5dea36a1a0577d7fbb6bfe7c2da3fee4cf67a71cfe0a
|
7
|
+
data.tar.gz: 24f300760fbe30759db3151f8be56604d1800288adc6ddccbe2279c97429e6a69fe14b2f35a181be70c07be0f650a010834abe308ebde6969559182b224be33d
|
data/Gemfile.lock
CHANGED
@@ -8,6 +8,11 @@ GEM
|
|
8
8
|
remote: https://rubygems.org/
|
9
9
|
specs:
|
10
10
|
autoloaded (2.2.1)
|
11
|
+
coderay (1.1.2)
|
12
|
+
method_source (0.9.2)
|
13
|
+
pry (0.12.2)
|
14
|
+
coderay (~> 1.1.0)
|
15
|
+
method_source (~> 0.9.0)
|
11
16
|
rack (2.0.7)
|
12
17
|
rake (10.5.0)
|
13
18
|
|
@@ -16,6 +21,7 @@ PLATFORMS
|
|
16
21
|
|
17
22
|
DEPENDENCIES
|
18
23
|
bundler (~> 2.0)
|
24
|
+
pry
|
19
25
|
rack
|
20
26
|
rack-healthz!
|
21
27
|
rake (~> 10.0)
|
data/lib/rack/healthz.rb
CHANGED
@@ -1,86 +1,61 @@
|
|
1
|
-
require
|
1
|
+
require 'rack/healthz/version'
|
2
|
+
|
2
3
|
require 'autoloaded'
|
3
4
|
require 'json'
|
5
|
+
require 'pry'
|
4
6
|
|
5
7
|
module Rack
|
6
8
|
class Healthz
|
7
9
|
Autoloaded.module {}
|
8
|
-
class Error < StandardError;
|
9
|
-
end
|
10
10
|
|
11
|
-
|
12
|
-
@app = app
|
13
|
-
@count = 0
|
14
|
-
@my_stats = {}
|
15
|
-
@up_at = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
16
|
-
@last_success_time = nil
|
17
|
-
@stats_path = stats_path
|
18
|
-
@max_time_between_requests = max_time_between_requests
|
19
|
-
end
|
11
|
+
DefaultPath = '/healthz'
|
20
12
|
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
else
|
26
|
-
@count += 1
|
27
|
-
start_time = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
28
|
-
result = @app.call env
|
29
|
-
end_time = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
30
|
-
elapsed_time = end_time - start_time
|
31
|
-
response_code = result.first.to_i
|
32
|
-
request_category = response_code / 100
|
33
|
-
if !(@my_stats.key? request_category)
|
34
|
-
@my_stats[request_category] = RequestAccumulator.new
|
35
|
-
end
|
36
|
-
@my_stats[request_category] << elapsed_time
|
37
|
-
if response_code < 400
|
38
|
-
@last_success_time = end_time
|
39
|
-
end
|
40
|
-
result
|
41
|
-
end
|
42
|
-
end
|
13
|
+
DefaultTimeBetweenRequests = 3600
|
14
|
+
|
15
|
+
HealthyStatus = 200
|
16
|
+
UnhealthyStatus = 503
|
43
17
|
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
18
|
+
ContentType = {'Content-Type' => 'application/json'}
|
19
|
+
|
20
|
+
class << self
|
21
|
+
def current_time
|
22
|
+
Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
49
23
|
end
|
50
|
-
end
|
51
24
|
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
25
|
+
def measure(&block)
|
26
|
+
start_time = current_time
|
27
|
+
|
28
|
+
block.call
|
29
|
+
|
30
|
+
current_time - start_time
|
57
31
|
end
|
58
32
|
end
|
59
33
|
|
60
|
-
|
61
|
-
!time_since_last_success(now).nil? and (time_since_last_success(now) < @max_time_between_requests)
|
34
|
+
class Error < StandardError;
|
62
35
|
end
|
63
36
|
|
64
|
-
|
65
|
-
@last_success_time ? now - @last_success_time : nil
|
66
|
-
end
|
37
|
+
attr_reader :accumulator
|
67
38
|
|
68
|
-
def
|
69
|
-
|
39
|
+
def initialize(app, **args)
|
40
|
+
@accumulator = Accumulator.new(app, **args)
|
70
41
|
end
|
71
42
|
|
72
|
-
def
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
}
|
80
|
-
@my_stats.each_pair do |status_category, reqest_accumulator|
|
81
|
-
result['stats'][status_category.to_s + 'xx'] = reqest_accumulator.to_h
|
43
|
+
def call(env)
|
44
|
+
with_now do
|
45
|
+
if accumulator.status_request?(env)
|
46
|
+
accumulator.response.to_a
|
47
|
+
else
|
48
|
+
accumulator.call(env).to_a
|
49
|
+
end
|
82
50
|
end
|
83
|
-
|
51
|
+
end
|
52
|
+
|
53
|
+
def with_now
|
54
|
+
old_now = Thread.current[:now]
|
55
|
+
Thread.current[:now] = Healthz.current_time
|
56
|
+
result = yield
|
57
|
+
Thread.current[:now] = old_now
|
58
|
+
result
|
84
59
|
end
|
85
60
|
end
|
86
61
|
end
|
@@ -0,0 +1,108 @@
|
|
1
|
+
module Rack
|
2
|
+
class Healthz
|
3
|
+
class Accumulator
|
4
|
+
attr_reader :count, :path, :app, :stats, :max_time_between_requests, :up_at, :last_success_time
|
5
|
+
|
6
|
+
def initialize(app, max_time_between_requests: DefaultTimeBetweenRequests, path: DefaultPath)
|
7
|
+
@app = app
|
8
|
+
@count = 0
|
9
|
+
@stats = {}
|
10
|
+
@up_at = Healthz.current_time
|
11
|
+
@last_success_time = nil
|
12
|
+
@max_time_between_requests = max_time_between_requests
|
13
|
+
@path = path
|
14
|
+
end
|
15
|
+
|
16
|
+
def call(env)
|
17
|
+
count!
|
18
|
+
|
19
|
+
response = nil
|
20
|
+
|
21
|
+
elapsed_time = Healthz.measure do
|
22
|
+
response = app.call(env)
|
23
|
+
end
|
24
|
+
|
25
|
+
handle_response! elapsed_time, Rack::Response.new(response)
|
26
|
+
|
27
|
+
response
|
28
|
+
end
|
29
|
+
|
30
|
+
def status_request?(env)
|
31
|
+
env.fetch('PATH_INFO') == path
|
32
|
+
end
|
33
|
+
|
34
|
+
def healthy?
|
35
|
+
!time_since_last_success.nil? and (time_since_last_success < max_time_between_requests)
|
36
|
+
end
|
37
|
+
|
38
|
+
def response
|
39
|
+
Response.new self
|
40
|
+
end
|
41
|
+
|
42
|
+
def time_since_last_success
|
43
|
+
last_success_time ? current_time - last_success_time : nil
|
44
|
+
end
|
45
|
+
|
46
|
+
def uptime
|
47
|
+
current_time - up_at
|
48
|
+
end
|
49
|
+
|
50
|
+
def status
|
51
|
+
if healthy?
|
52
|
+
"healthy"
|
53
|
+
else
|
54
|
+
"unhealthy"
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
def to_h
|
59
|
+
{}.tap do |result|
|
60
|
+
result[:count] = count
|
61
|
+
result[:uptime] = uptime
|
62
|
+
result[:time_since_last_success] = time_since_last_success
|
63
|
+
result[:status] = status
|
64
|
+
result[:stats] = stats_hash
|
65
|
+
end
|
66
|
+
end
|
67
|
+
|
68
|
+
private
|
69
|
+
|
70
|
+
def stats_hash
|
71
|
+
{}.tap do |result|
|
72
|
+
stats.values.each do |stat|
|
73
|
+
result[stat.category] = stat.to_h
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
#
|
79
|
+
# Internal Commands
|
80
|
+
#
|
81
|
+
def handle_response!(elapsed_time, response)
|
82
|
+
response_category = response_category_of response
|
83
|
+
|
84
|
+
stats[response_category] ||= RequestAccumulator.new(response_category)
|
85
|
+
stats[response_category] << elapsed_time
|
86
|
+
|
87
|
+
successful! if response.successful?
|
88
|
+
end
|
89
|
+
|
90
|
+
def successful!
|
91
|
+
@last_success_time = current_time
|
92
|
+
end
|
93
|
+
|
94
|
+
def count!
|
95
|
+
@count += 1
|
96
|
+
end
|
97
|
+
|
98
|
+
#Internal Queries
|
99
|
+
def current_time
|
100
|
+
Thread.current[:now]
|
101
|
+
end
|
102
|
+
|
103
|
+
def response_category_of response
|
104
|
+
response.status / 100
|
105
|
+
end
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
@@ -2,9 +2,10 @@ module Rack
|
|
2
2
|
class Healthz
|
3
3
|
class RequestAccumulator
|
4
4
|
|
5
|
-
attr_reader :min_time, :max_time, :count, :std, :mean
|
5
|
+
attr_reader :min_time, :max_time, :count, :std, :mean, :category
|
6
6
|
|
7
|
-
def initialize
|
7
|
+
def initialize(category)
|
8
|
+
@category = category
|
8
9
|
@count = 0
|
9
10
|
@sum_of_times = 0.0
|
10
11
|
@sum_of_square_times = 0.0
|
@@ -19,6 +20,7 @@ module Rack
|
|
19
20
|
@min_time = [@min_time, elapsed_time].min
|
20
21
|
@max_time = [@max_time, elapsed_time].max
|
21
22
|
@mean = @sum_of_times / @count
|
23
|
+
|
22
24
|
@std = if @count > 1
|
23
25
|
Math.sqrt((@sum_of_square_times / @count) - (@sum_of_times / @count) ** 2)
|
24
26
|
elsif @count == 1
|
@@ -0,0 +1,39 @@
|
|
1
|
+
module Rack
|
2
|
+
class Healthz
|
3
|
+
class Response
|
4
|
+
attr_reader :accumulator
|
5
|
+
|
6
|
+
def initialize(accumulator)
|
7
|
+
@accumulator = accumulator
|
8
|
+
end
|
9
|
+
|
10
|
+
def to_a
|
11
|
+
[return_status, content_type, [to_json]]
|
12
|
+
end
|
13
|
+
|
14
|
+
def to_json
|
15
|
+
to_h.to_json
|
16
|
+
end
|
17
|
+
|
18
|
+
def to_h
|
19
|
+
accumulator.to_h
|
20
|
+
end
|
21
|
+
|
22
|
+
def content_type
|
23
|
+
ContentType
|
24
|
+
end
|
25
|
+
|
26
|
+
def healthy?
|
27
|
+
accumulator.healthy?
|
28
|
+
end
|
29
|
+
|
30
|
+
def return_status
|
31
|
+
if healthy?
|
32
|
+
HealthyStatus
|
33
|
+
else
|
34
|
+
UnhealthyStatus
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
data/lib/rack/healthz/version.rb
CHANGED
data/rack-healthz.gemspec
CHANGED
@@ -38,6 +38,7 @@ Gem::Specification.new do |spec|
|
|
38
38
|
spec.add_development_dependency "bundler", "~> 2.0"
|
39
39
|
spec.add_development_dependency "rake", "~> 10.0"
|
40
40
|
spec.add_development_dependency "rack"
|
41
|
+
spec.add_development_dependency "pry"
|
41
42
|
|
42
43
|
spec.add_runtime_dependency "autoloaded"
|
43
44
|
end
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: rack-healthz
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.2.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Andrew Eberbach
|
@@ -9,7 +9,7 @@ authors:
|
|
9
9
|
autorequire:
|
10
10
|
bindir: exe
|
11
11
|
cert_chain: []
|
12
|
-
date: 2019-07-
|
12
|
+
date: 2019-07-13 00:00:00.000000000 Z
|
13
13
|
dependencies:
|
14
14
|
- !ruby/object:Gem::Dependency
|
15
15
|
name: bundler
|
@@ -53,6 +53,20 @@ dependencies:
|
|
53
53
|
- - ">="
|
54
54
|
- !ruby/object:Gem::Version
|
55
55
|
version: '0'
|
56
|
+
- !ruby/object:Gem::Dependency
|
57
|
+
name: pry
|
58
|
+
requirement: !ruby/object:Gem::Requirement
|
59
|
+
requirements:
|
60
|
+
- - ">="
|
61
|
+
- !ruby/object:Gem::Version
|
62
|
+
version: '0'
|
63
|
+
type: :development
|
64
|
+
prerelease: false
|
65
|
+
version_requirements: !ruby/object:Gem::Requirement
|
66
|
+
requirements:
|
67
|
+
- - ">="
|
68
|
+
- !ruby/object:Gem::Version
|
69
|
+
version: '0'
|
56
70
|
- !ruby/object:Gem::Dependency
|
57
71
|
name: autoloaded
|
58
72
|
requirement: !ruby/object:Gem::Requirement
|
@@ -84,7 +98,9 @@ files:
|
|
84
98
|
- bin/setup
|
85
99
|
- config.ru
|
86
100
|
- lib/rack/healthz.rb
|
101
|
+
- lib/rack/healthz/accumulator.rb
|
87
102
|
- lib/rack/healthz/request_accumulator.rb
|
103
|
+
- lib/rack/healthz/response.rb
|
88
104
|
- lib/rack/healthz/version.rb
|
89
105
|
- rack-healthz.gemspec
|
90
106
|
homepage:
|