authsignal-ruby 4.1.0 → 5.0.1
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 +4 -4
- data/CHANGELOG.md +1 -5
- data/Gemfile +4 -0
- data/Gemfile.lock +5 -3
- data/README.md +27 -93
- data/lib/authsignal/api_error.rb +18 -6
- data/lib/authsignal/client.rb +26 -31
- data/lib/authsignal/configuration.rb +2 -2
- data/lib/authsignal/version.rb +1 -1
- data/lib/authsignal.rb +33 -27
- metadata +5 -8
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 60927c210d3c9ca7c01e5fc6a3b2cddfbd069693cc221cd585f24154e07fd515
|
4
|
+
data.tar.gz: 9ca1be169ea1f4e24510e42de4a8b6cc79628b7e88a35bc46e279a2a4d254440
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 2ea4f562bea332f0db4f3b5fbf24a4cf6993bdb17d2879e232219ad7814ab5b59ca2c3a9e52087d295f285bd15f9c890abe8c43162875b6af1af22171049fbc7
|
7
|
+
data.tar.gz: b75556ccafcf243fd4ff33c7d8c25731ae9dfc6bc51637b64a667ab8faa99ae8b2fdf1bb5c7039dff2701e0f338c39c7be64ec09add77273c490353e2eec3f2c
|
data/CHANGELOG.md
CHANGED
data/Gemfile
CHANGED
data/Gemfile.lock
CHANGED
@@ -1,8 +1,8 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
authsignal-ruby (
|
5
|
-
faraday (>= 2)
|
4
|
+
authsignal-ruby (5.0.1)
|
5
|
+
faraday (>= 2.0.1)
|
6
6
|
faraday-retry (~> 2.2)
|
7
7
|
|
8
8
|
GEM
|
@@ -15,6 +15,7 @@ GEM
|
|
15
15
|
bigdecimal
|
16
16
|
rexml
|
17
17
|
diff-lcs (1.5.1)
|
18
|
+
dotenv (3.1.7)
|
18
19
|
faraday (2.12.0)
|
19
20
|
faraday-net_http (>= 2.0, < 3.4)
|
20
21
|
json
|
@@ -56,9 +57,10 @@ PLATFORMS
|
|
56
57
|
|
57
58
|
DEPENDENCIES
|
58
59
|
authsignal-ruby!
|
60
|
+
dotenv
|
59
61
|
rake (~> 13.0)
|
60
62
|
rspec (~> 3.2)
|
61
63
|
webmock (~> 3.14)
|
62
64
|
|
63
65
|
BUNDLED WITH
|
64
|
-
2.5.
|
66
|
+
2.5.14
|
data/README.md
CHANGED
@@ -1,6 +1,8 @@
|
|
1
|
-
|
1
|
+
<img width="1070" alt="Authsignal" src="https://raw.githubusercontent.com/authsignal/authsignal-node/main/.github/images/authsignal.png">
|
2
2
|
|
3
|
-
|
3
|
+
# Authsignal Ruby SDK
|
4
|
+
|
5
|
+
The Authsignal Ruby library for server-side applications.
|
4
6
|
|
5
7
|
## Installation
|
6
8
|
|
@@ -10,72 +12,11 @@ Add this line to your application's Gemfile:
|
|
10
12
|
gem "authsignal-ruby"
|
11
13
|
```
|
12
14
|
|
13
|
-
|
14
|
-
|
15
|
-
$ bundle install
|
16
|
-
|
17
|
-
Or install it yourself as:
|
18
|
-
|
19
|
-
$ gem install authsignal-ruby
|
20
|
-
|
21
|
-
## Initialization
|
22
|
-
|
23
|
-
Initialize the Authsignal Ruby SDK, ensuring you do not hard code the Authsignal Secret Key, always keep this safe.
|
24
|
-
|
25
|
-
In Ruby on Rails, you would typically place this code block in a file like `config/initializers/authsignal.rb`
|
26
|
-
|
27
|
-
```ruby
|
28
|
-
Authsignal.setup do |config|
|
29
|
-
config.api_secret_key = ENV["AUTHSIGNAL_SECRET_KEY"]
|
30
|
-
end
|
31
|
-
```
|
32
|
-
|
33
|
-
You can find your `api_secret_key` in the [Authsignal Portal](https://portal.authsignal.com/organisations/tenants/api).
|
34
|
-
|
35
|
-
You must specify the correct `baseUrl` for your tenant's region.
|
36
|
-
|
37
|
-
| Region | Base URL |
|
38
|
-
| ----------- | ----------------------------------- |
|
39
|
-
| US (Oregon) | https://signal.authsignal.com/v1 |
|
40
|
-
| AU (Sydney) | https://au.signal.authsignal.com/v1 |
|
41
|
-
| EU (Dublin) | https://eu.signal.authsignal.com/v1 |
|
42
|
-
|
43
|
-
For example, to set the base URL to use our AU region:
|
44
|
-
|
45
|
-
```
|
46
|
-
require 'authsignal'
|
47
|
-
|
48
|
-
Authsignal.setup do |config|
|
49
|
-
config.api_secret_key = ENV["AUTHSIGNAL_SECRET_KEY"]
|
50
|
-
config.base_uri = "https://au.signal.authsignal.com/v1"
|
15
|
+
## Documentation
|
51
16
|
|
52
|
-
|
53
|
-
config.retry = true # default value: false
|
17
|
+
Refer to our [SDK documentation](https://docs.authsignal.com/sdks/server/overview) for information on how to use this SDK.
|
54
18
|
|
55
|
-
|
56
|
-
config.debug = true # default value: false
|
57
|
-
end
|
58
|
-
```
|
59
|
-
|
60
|
-
## Usage
|
61
|
-
|
62
|
-
Authsignal's server side signal API has four main api calls `track`, `get_action`, `get_user`, `enroll_verified_authenticator`.
|
63
|
-
|
64
|
-
For more details on these api calls, refer to our [official Ruby SDK docs](https://docs.authsignal.com/sdks/server/ruby#track).
|
65
|
-
|
66
|
-
Example:
|
67
|
-
|
68
|
-
```ruby
|
69
|
-
Authsignal.track user_id: 'AS_001', action: 'withdraw', idempotency_key: 'a_random_hash'
|
70
|
-
|
71
|
-
# returns:
|
72
|
-
# {
|
73
|
-
# success?: true,
|
74
|
-
# state: 'ALLOW',
|
75
|
-
# idempotency_key: 'a_random_hash',
|
76
|
-
# ... rest of payload ...
|
77
|
-
# }
|
78
|
-
```
|
19
|
+
Or check out our [Ruby on Rails Quickstart Guide](https://docs.authsignal.com/integrations/ruby-on-rails).
|
79
20
|
|
80
21
|
### Response & Error handling
|
81
22
|
|
@@ -84,18 +25,20 @@ The Authsignal SDK offers two response formats. By default, its methods return t
|
|
84
25
|
Example:
|
85
26
|
|
86
27
|
```ruby
|
87
|
-
Authsignal.enroll_verified_authenticator
|
88
|
-
|
89
|
-
|
90
|
-
|
28
|
+
Authsignal.enroll_verified_authenticator(
|
29
|
+
user_id: 'AS_001',
|
30
|
+
attributes: {
|
31
|
+
verification_method: 'INVALID',
|
32
|
+
email: 'hamish@authsignal.com'
|
33
|
+
}
|
34
|
+
)
|
91
35
|
|
92
36
|
# returns:
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
# }
|
37
|
+
{
|
38
|
+
"error": "invalid_request",
|
39
|
+
"errorCode": "invalid_request",
|
40
|
+
"errorDescription": "body.verificationMethod must be equal to one of the allowed values - allowedValues: AUTHENTICATOR_APP,EMAIL_MAGIC_LINK,EMAIL_OTP,SMS"
|
41
|
+
}
|
99
42
|
```
|
100
43
|
|
101
44
|
All methods have a bang (!) counterpart that raises an Authsignal::ApiError if the request fails.
|
@@ -103,23 +46,14 @@ All methods have a bang (!) counterpart that raises an Authsignal::ApiError if t
|
|
103
46
|
Example:
|
104
47
|
|
105
48
|
```ruby
|
106
|
-
Authsignal.enroll_verified_authenticator!
|
107
|
-
|
108
|
-
|
109
|
-
|
49
|
+
Authsignal.enroll_verified_authenticator!(
|
50
|
+
user_id: 'AS_001',
|
51
|
+
attributes: {
|
52
|
+
verification_method: 'INVALID',
|
53
|
+
email: 'hamish@authsignal.com'
|
54
|
+
}
|
55
|
+
)
|
110
56
|
|
111
57
|
# raise:
|
112
|
-
|
58
|
+
<Authsignal::ApiError: AuthsignalError: 400 - body.verificationMethod must be equal to one of the allowed values - allowedValues: AUTHENTICATOR_APP,EMAIL_MAGIC_LINK,EMAIL_OTP,SMS.
|
113
59
|
```
|
114
|
-
|
115
|
-
## Development
|
116
|
-
|
117
|
-
After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` or `bundle exec rspec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
|
118
|
-
|
119
|
-
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
120
|
-
|
121
|
-
Log request/response against test server: `Authsignal.configuration.debug = true`
|
122
|
-
|
123
|
-
## License
|
124
|
-
|
125
|
-
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
|
data/lib/authsignal/api_error.rb
CHANGED
@@ -2,18 +2,30 @@
|
|
2
2
|
|
3
3
|
module Authsignal
|
4
4
|
class ApiError < StandardError
|
5
|
-
attr_reader :
|
5
|
+
attr_reader :status_code, :error_code, :error_description
|
6
6
|
|
7
|
-
def initialize(
|
8
|
-
|
9
|
-
@error = error
|
10
|
-
@description = description
|
7
|
+
def initialize(status_code, error_code, error_description = nil)
|
8
|
+
message = format_message(status_code, error_code, error_description)
|
11
9
|
|
12
10
|
super(message)
|
11
|
+
|
12
|
+
@status_code = status_code
|
13
|
+
@error_code = error_code
|
14
|
+
@error_description = error_description
|
13
15
|
end
|
14
16
|
|
15
17
|
def to_s
|
16
|
-
"#{super}
|
18
|
+
"#{super} status_code: #{status_code}, error_code: #{error_code}, error_description: #{error_description}"
|
19
|
+
end
|
20
|
+
|
21
|
+
private
|
22
|
+
|
23
|
+
def format_message(status_code, error_code, error_description)
|
24
|
+
"AuthsignalError: #{status_code} - #{format_description(error_code, error_description)}"
|
25
|
+
end
|
26
|
+
|
27
|
+
def format_description(error_code, error_description)
|
28
|
+
error_description && error_description.length > 0 ? error_description : error_code
|
17
29
|
end
|
18
30
|
end
|
19
31
|
end
|
data/lib/authsignal/client.rb
CHANGED
@@ -17,7 +17,7 @@ module Authsignal
|
|
17
17
|
@api_key = require_api_key
|
18
18
|
|
19
19
|
@client = Faraday.new do |builder|
|
20
|
-
builder.url_prefix = Authsignal.configuration.
|
20
|
+
builder.url_prefix = Authsignal.configuration.api_url
|
21
21
|
builder.adapter :net_http
|
22
22
|
builder.request :authorization, :basic, @api_key, nil
|
23
23
|
|
@@ -36,47 +36,50 @@ module Authsignal
|
|
36
36
|
end
|
37
37
|
end
|
38
38
|
|
39
|
-
def
|
40
|
-
|
41
|
-
|
39
|
+
def get_user(user_id:)
|
40
|
+
path = "users/#{url_encode(user_id)}"
|
41
|
+
make_request(:get, path)
|
42
|
+
end
|
42
43
|
|
43
|
-
|
44
|
-
|
44
|
+
def update_user(user_id:, attributes:)
|
45
|
+
make_request(:post, "users/#{url_encode(user_id)}", body: attributes)
|
46
|
+
end
|
45
47
|
|
46
|
-
|
48
|
+
def delete_user(user_id:)
|
49
|
+
make_request(:delete, "users/#{url_encode(user_id)}")
|
47
50
|
end
|
48
51
|
|
49
|
-
def
|
50
|
-
|
51
|
-
path = "users/#{url_encode(user_id)}?redirectUrl=#{redirect_url}"
|
52
|
-
else
|
53
|
-
path = "users/#{url_encode(user_id)}"
|
54
|
-
end
|
55
|
-
make_request(:get, path)
|
52
|
+
def get_authenticators(user_id:)
|
53
|
+
make_request(:get, "users/#{url_encode(user_id)}/authenticators")
|
56
54
|
end
|
57
55
|
|
58
|
-
def
|
59
|
-
make_request(:post, "users/#{url_encode(user_id)}", body:
|
56
|
+
def enroll_verified_authenticator(user_id:, attributes:)
|
57
|
+
make_request(:post, "users/#{url_encode(user_id)}/authenticators", body: attributes)
|
60
58
|
end
|
61
59
|
|
62
|
-
def
|
63
|
-
make_request(:delete, "users/#{url_encode(user_id)}")
|
60
|
+
def delete_authenticator(user_id:, user_authenticator_id:)
|
61
|
+
make_request(:delete, "users/#{url_encode(user_id)}/authenticators/#{url_encode(user_authenticator_id)}")
|
62
|
+
end
|
63
|
+
|
64
|
+
def track(user_id:, action:, attributes:)
|
65
|
+
path = "users/#{user_id}/actions/#{action}"
|
66
|
+
|
67
|
+
make_request(:post, path, body: attributes)
|
64
68
|
end
|
65
69
|
|
66
|
-
def validate_challenge(user_id: nil,
|
70
|
+
def validate_challenge(token:, user_id: nil, action: nil)
|
67
71
|
path = "validate"
|
68
72
|
body = { user_id: user_id, token: token, action: action }
|
69
73
|
|
70
74
|
make_request(:post, path, body: body)
|
71
75
|
end
|
72
76
|
|
73
|
-
def get_action(user_id
|
77
|
+
def get_action(user_id:, action:, idempotency_key:)
|
74
78
|
make_request(:get, "users/#{url_encode(user_id)}/actions/#{action}/#{url_encode(idempotency_key)}")
|
75
79
|
end
|
76
80
|
|
77
|
-
def
|
78
|
-
|
79
|
-
make_request(:patch, "users/#{url_encode(user_id)}/actions/#{action}/#{url_encode(idempotency_key)}", body: body)
|
81
|
+
def update_action(user_id:, action:, idempotency_key:, attributes:)
|
82
|
+
make_request(:patch, "users/#{url_encode(user_id)}/actions/#{action}/#{url_encode(idempotency_key)}", body: attributes)
|
80
83
|
end
|
81
84
|
|
82
85
|
##
|
@@ -85,14 +88,6 @@ module Authsignal
|
|
85
88
|
make_request(:post , "users/#{url_encode(user_id)}", body: user_payload)
|
86
89
|
end
|
87
90
|
|
88
|
-
def enroll_verified_authenticator(user_id, authenticator)
|
89
|
-
make_request(:post, "users/#{url_encode(user_id)}/authenticators", body: authenticator)
|
90
|
-
end
|
91
|
-
|
92
|
-
def delete_authenticator(user_id:, user_authenticator_id:)
|
93
|
-
make_request(:delete, "users/#{url_encode(user_id)}/authenticators/#{url_encode(user_authenticator_id)}")
|
94
|
-
end
|
95
|
-
|
96
91
|
private
|
97
92
|
|
98
93
|
def url_encode(s)
|
@@ -13,7 +13,7 @@ module Authsignal
|
|
13
13
|
end
|
14
14
|
|
15
15
|
config_option :api_secret_key
|
16
|
-
config_option :
|
16
|
+
config_option :api_url
|
17
17
|
config_option :debug
|
18
18
|
config_option :retry
|
19
19
|
|
@@ -22,7 +22,7 @@ module Authsignal
|
|
22
22
|
|
23
23
|
# set default attribute values
|
24
24
|
@defaults = OpenStruct.new({
|
25
|
-
|
25
|
+
api_url: 'https://signal.authsignal.com/v1/',
|
26
26
|
retry: false,
|
27
27
|
debug: false
|
28
28
|
})
|
data/lib/authsignal/version.rb
CHANGED
data/lib/authsignal.rb
CHANGED
@@ -25,14 +25,14 @@ module Authsignal
|
|
25
25
|
configuration.defaults
|
26
26
|
end
|
27
27
|
|
28
|
-
def get_user(user_id
|
29
|
-
response = Client.new.get_user(user_id: user_id
|
28
|
+
def get_user(user_id:)
|
29
|
+
response = Client.new.get_user(user_id: user_id)
|
30
30
|
|
31
31
|
handle_response(response)
|
32
32
|
end
|
33
33
|
|
34
|
-
def update_user(user_id:,
|
35
|
-
response = Client.new.update_user(user_id: user_id,
|
34
|
+
def update_user(user_id:, attributes:)
|
35
|
+
response = Client.new.update_user(user_id: user_id, attributes: attributes)
|
36
36
|
|
37
37
|
handle_response(response)
|
38
38
|
end
|
@@ -43,42 +43,44 @@ module Authsignal
|
|
43
43
|
handle_response(response)
|
44
44
|
end
|
45
45
|
|
46
|
-
def
|
47
|
-
response = Client.new.
|
46
|
+
def get_authenticators(user_id:)
|
47
|
+
response = Client.new.get_authenticators(user_id: user_id)
|
48
48
|
|
49
49
|
handle_response(response)
|
50
50
|
end
|
51
51
|
|
52
|
-
def
|
53
|
-
|
54
|
-
response = Client.new.update_action_state(user_id: user_id, action: action, idempotency_key: idempotency_key, state: state)
|
52
|
+
def enroll_verified_authenticator(user_id:, attributes:)
|
53
|
+
response = Client.new.enroll_verified_authenticator(user_id: user_id, attributes: attributes)
|
55
54
|
|
56
55
|
handle_response(response)
|
57
56
|
end
|
58
57
|
|
59
|
-
def
|
60
|
-
response = Client.new.
|
58
|
+
def delete_authenticator(user_id:, user_authenticator_id:)
|
59
|
+
response = Client.new.delete_authenticator(user_id: user_id, user_authenticator_id: user_authenticator_id)
|
61
60
|
|
62
61
|
handle_response(response)
|
63
62
|
end
|
64
63
|
|
65
|
-
def
|
66
|
-
response = Client.new.
|
64
|
+
def track(user_id:, action:, attributes:)
|
65
|
+
response = Client.new.track(user_id: user_id, action: action, attributes: attributes)
|
66
|
+
handle_response(response)
|
67
|
+
end
|
67
68
|
|
69
|
+
def validate_challenge(token:, user_id: nil, action: nil)
|
70
|
+
response = Client.new.validate_challenge(token: token,user_id: user_id, action: action)
|
71
|
+
|
68
72
|
handle_response(response)
|
69
73
|
end
|
70
74
|
|
71
|
-
def
|
72
|
-
|
73
|
-
raise ArgumentError, "User ID value" unless event[:user_id].to_s.length > 0
|
75
|
+
def get_action(user_id:, action:, idempotency_key:)
|
76
|
+
response = Client.new.get_action(user_id: user_id, action: action, idempotency_key: idempotency_key)
|
74
77
|
|
75
|
-
response = Client.new.track(event)
|
76
78
|
handle_response(response)
|
77
79
|
end
|
78
80
|
|
79
|
-
def
|
80
|
-
response = Client.new.
|
81
|
-
|
81
|
+
def update_action(user_id:, action:, idempotency_key:, attributes:)
|
82
|
+
response = Client.new.update_action(user_id: user_id, action: action, idempotency_key: idempotency_key, attributes: attributes)
|
83
|
+
|
82
84
|
handle_response(response)
|
83
85
|
end
|
84
86
|
|
@@ -93,15 +95,19 @@ module Authsignal
|
|
93
95
|
end
|
94
96
|
|
95
97
|
def handle_success_response(response)
|
96
|
-
response.body.
|
98
|
+
if response.body.is_a?(Array)
|
99
|
+
{ success?: true, data: response.body }
|
100
|
+
else
|
101
|
+
response.body.merge(success?: true)
|
102
|
+
end
|
97
103
|
end
|
98
104
|
|
99
105
|
def handle_error_response(response)
|
100
106
|
case response.body
|
101
107
|
when Hash
|
102
|
-
|
108
|
+
{ status_code: response.status, success?: false, error_code: response.body[:error], error_description: response.body[:error_description] }
|
103
109
|
else
|
104
|
-
{
|
110
|
+
{ status_code: response&.status || 500, success?: false }
|
105
111
|
end
|
106
112
|
end
|
107
113
|
end
|
@@ -110,11 +116,11 @@ module Authsignal
|
|
110
116
|
(methods - NON_API_METHODS).each do |method|
|
111
117
|
define_singleton_method("#{method}!") do |*args, **kwargs|
|
112
118
|
send(method, *args, **kwargs).tap do |response|
|
113
|
-
|
114
|
-
|
115
|
-
|
119
|
+
status_code = response[:status_code]
|
120
|
+
error_code = response[:error_code]
|
121
|
+
error_description = response[:error_description]
|
116
122
|
|
117
|
-
raise ApiError.new(
|
123
|
+
raise ApiError.new(status_code, error_code, error_description) unless response[:success?]
|
118
124
|
end
|
119
125
|
end
|
120
126
|
end
|
metadata
CHANGED
@@ -1,14 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: authsignal-ruby
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version:
|
4
|
+
version: 5.0.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- justinsoong
|
8
|
-
autorequire:
|
9
8
|
bindir: exe
|
10
9
|
cert_chain: []
|
11
|
-
date:
|
10
|
+
date: 2025-03-25 00:00:00.000000000 Z
|
12
11
|
dependencies:
|
13
12
|
- !ruby/object:Gem::Dependency
|
14
13
|
name: faraday
|
@@ -16,14 +15,14 @@ dependencies:
|
|
16
15
|
requirements:
|
17
16
|
- - ">="
|
18
17
|
- !ruby/object:Gem::Version
|
19
|
-
version:
|
18
|
+
version: 2.0.1
|
20
19
|
type: :runtime
|
21
20
|
prerelease: false
|
22
21
|
version_requirements: !ruby/object:Gem::Requirement
|
23
22
|
requirements:
|
24
23
|
- - ">="
|
25
24
|
- !ruby/object:Gem::Version
|
26
|
-
version:
|
25
|
+
version: 2.0.1
|
27
26
|
- !ruby/object:Gem::Dependency
|
28
27
|
name: faraday-retry
|
29
28
|
requirement: !ruby/object:Gem::Requirement
|
@@ -112,7 +111,6 @@ licenses:
|
|
112
111
|
metadata:
|
113
112
|
homepage_uri: https://www.authsignal.com
|
114
113
|
source_code_uri: https://github.com/authsignal/authsignal-ruby
|
115
|
-
post_install_message:
|
116
114
|
rdoc_options: []
|
117
115
|
require_paths:
|
118
116
|
- lib
|
@@ -127,8 +125,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
127
125
|
- !ruby/object:Gem::Version
|
128
126
|
version: '0'
|
129
127
|
requirements: []
|
130
|
-
rubygems_version: 3.
|
131
|
-
signing_key:
|
128
|
+
rubygems_version: 3.6.2
|
132
129
|
specification_version: 4
|
133
130
|
summary: The Authsignal ruby server side signal API.
|
134
131
|
test_files: []
|