eureka-bot 1.0.16
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +30 -0
- data/.rspec +2 -0
- data/Gemfile +2 -0
- data/LICENSE.txt +22 -0
- data/README.md +2 -0
- data/eureka-bot.gemspec +26 -0
- data/lib/eureka-bot.rb +37 -0
- data/lib/eureka_bot/controller.rb +69 -0
- data/lib/eureka_bot/controller/response.rb +42 -0
- data/lib/eureka_bot/exceptions.rb +28 -0
- data/lib/eureka_bot/instrumentation.rb +25 -0
- data/lib/eureka_bot/integration.rb +11 -0
- data/lib/eureka_bot/integration/new_relic.rb +10 -0
- data/lib/eureka_bot/job.rb +14 -0
- data/lib/eureka_bot/job/input.rb +18 -0
- data/lib/eureka_bot/job/output.rb +12 -0
- data/lib/eureka_bot/notifications.rb +25 -0
- data/lib/eureka_bot/resolver.rb +78 -0
- data/lib/eureka_bot/sender.rb +8 -0
- data/lib/eureka_bot/version.rb +3 -0
- data/log/.keep +0 -0
- data/spec/eureka-bot/controller/response_spec.rb +21 -0
- data/spec/eureka-bot/controller_spec.rb +59 -0
- data/spec/eureka-bot/instrumentation_spec.rb +18 -0
- data/spec/eureka-bot/job_spec.rb +76 -0
- data/spec/eureka-bot/resolver_spec.rb +94 -0
- data/spec/spec_helper.rb +30 -0
- data/spec/support/settings_helper.rb +5 -0
- data/tmp/.keep +0 -0
- metadata +160 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: 4ea62c75b6709c9fbbc8afdbe0c6083e10674edb
|
4
|
+
data.tar.gz: c5fe994ed3a20a1159ca1e4b6f1884158586ec75
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: a1a16fe9ea47346ac3f887f4c62067bd375776cc896468987f05a68d35e82c7f98aeb73bdecf810a3482cdfd8b21fd16c3e4e75166a54a716e288158c0844933
|
7
|
+
data.tar.gz: 9b2bb0afc377b35c07b7747ac8bed6a049388030c1e123694dc46619c8a6ba6fa42e30a1f3aea14ae4c2ba0231f29b0ebbec2320320f1912cede9be89cd88381
|
data/.gitignore
ADDED
@@ -0,0 +1,30 @@
|
|
1
|
+
/.bundle
|
2
|
+
|
3
|
+
/.yardoc
|
4
|
+
/_yardoc/
|
5
|
+
/coverage/
|
6
|
+
/pkg/
|
7
|
+
|
8
|
+
# Ignore all logfiles and tempfiles.
|
9
|
+
/log/*
|
10
|
+
/tmp/*
|
11
|
+
!/log/.keep
|
12
|
+
!/tmp/.keep
|
13
|
+
mkmf.log
|
14
|
+
|
15
|
+
nbproject
|
16
|
+
.idea
|
17
|
+
*~
|
18
|
+
|
19
|
+
*.swp
|
20
|
+
*.swap
|
21
|
+
*.bundle
|
22
|
+
*.so
|
23
|
+
*.o
|
24
|
+
*.a
|
25
|
+
|
26
|
+
.vagrant
|
27
|
+
.ruby-env
|
28
|
+
.DS_Store
|
29
|
+
Vagrantfile
|
30
|
+
Gemfile.lock
|
data/.rspec
ADDED
data/Gemfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
Copyright (c) 2016 dizer
|
2
|
+
|
3
|
+
MIT License
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
a copy of this software and associated documentation files (the
|
7
|
+
"Software"), to deal in the Software without restriction, including
|
8
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
the following conditions:
|
12
|
+
|
13
|
+
The above copyright notice and this permission notice shall be
|
14
|
+
included in all copies or substantial portions of the Software.
|
15
|
+
|
16
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.md
ADDED
data/eureka-bot.gemspec
ADDED
@@ -0,0 +1,26 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
require 'eureka_bot/version'
|
5
|
+
|
6
|
+
Gem::Specification.new do |spec|
|
7
|
+
spec.name = 'eureka-bot'
|
8
|
+
spec.version = EurekaBot::VERSION
|
9
|
+
spec.authors = ['Yury Alexandrov']
|
10
|
+
spec.email = ['dizeru@gmail.com']
|
11
|
+
spec.summary = %q{Run your messenger bots}
|
12
|
+
spec.homepage = 'https://github.com/dizer/eureka-bot'
|
13
|
+
spec.license = 'MIT'
|
14
|
+
|
15
|
+
spec.files = `git ls-files -z`.split("\x0")
|
16
|
+
spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
|
17
|
+
spec.test_files = spec.files.grep(%r{^(spec)/})
|
18
|
+
spec.require_paths = ["lib"]
|
19
|
+
spec.bindir = 'bin'
|
20
|
+
|
21
|
+
spec.add_runtime_dependency 'activesupport', '~> 5.0', '>= 5.0.0'
|
22
|
+
spec.add_runtime_dependency 'activejob', '~> 5.0', '>= 5.0.0'
|
23
|
+
|
24
|
+
spec.add_development_dependency 'rspec', '~> 3.5', '>= 3.5.0'
|
25
|
+
spec.add_development_dependency 'webmock', '~> 3.3.0', '>= 3.3.0'
|
26
|
+
end
|
data/lib/eureka-bot.rb
ADDED
@@ -0,0 +1,37 @@
|
|
1
|
+
$:.unshift File.dirname(__FILE__)
|
2
|
+
|
3
|
+
require 'active_support'
|
4
|
+
require 'active_support/core_ext'
|
5
|
+
require 'active_support/concern'
|
6
|
+
require 'active_support/notifications'
|
7
|
+
|
8
|
+
module EurekaBot
|
9
|
+
extend ActiveSupport::Autoload
|
10
|
+
cattr_reader :logger
|
11
|
+
|
12
|
+
autoload :Controller
|
13
|
+
autoload :Exceptions
|
14
|
+
autoload :Instrumentation
|
15
|
+
autoload :Job
|
16
|
+
autoload :Resolver
|
17
|
+
autoload :Sender
|
18
|
+
autoload :Version
|
19
|
+
|
20
|
+
eager_autoload do
|
21
|
+
autoload :Integration
|
22
|
+
autoload :Notifications
|
23
|
+
end
|
24
|
+
|
25
|
+
include EurekaBot::Exceptions
|
26
|
+
|
27
|
+
def self.logger=(logger)
|
28
|
+
@@logger = logger
|
29
|
+
EurekaBot::Job.logger = logger
|
30
|
+
logger
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
EurekaBot.logger = Logger.new(nil)
|
35
|
+
EurekaBot.logger.level = Logger::INFO
|
36
|
+
|
37
|
+
EurekaBot.eager_load!
|
@@ -0,0 +1,69 @@
|
|
1
|
+
class EurekaBot::Controller
|
2
|
+
extend ActiveSupport::Autoload
|
3
|
+
include ActiveSupport::Callbacks
|
4
|
+
include EurekaBot::Instrumentation
|
5
|
+
|
6
|
+
define_callbacks :action
|
7
|
+
|
8
|
+
autoload :Response
|
9
|
+
|
10
|
+
attr_reader :params, :message, :logger, :response
|
11
|
+
# cattr_accessor :exception_handler
|
12
|
+
|
13
|
+
def initialize(params: {}, message: nil, response: nil, logger: EurekaBot.logger)
|
14
|
+
@params = params
|
15
|
+
@message = message
|
16
|
+
@logger = logger
|
17
|
+
@response = response || response_class.new(logger: logger)
|
18
|
+
@response.controller = self
|
19
|
+
end
|
20
|
+
|
21
|
+
def execute(action)
|
22
|
+
if respond_to?(action, include_all: false)
|
23
|
+
instrument 'controller.execute' do
|
24
|
+
run_callbacks :action do
|
25
|
+
public_send(action)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
else
|
29
|
+
raise UnknownAction.new("Action #{action} is not defined in #{self.class}")
|
30
|
+
end
|
31
|
+
self
|
32
|
+
end
|
33
|
+
|
34
|
+
def answer(**params)
|
35
|
+
instrument 'controller.answer', params do
|
36
|
+
response << params
|
37
|
+
nil
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def redirect(controller, action)
|
42
|
+
instance = controller.new(
|
43
|
+
params: params,
|
44
|
+
message: message,
|
45
|
+
response: response,
|
46
|
+
logger: logger
|
47
|
+
)
|
48
|
+
instrument 'controller.redirect', {from: self.class.to_s, to: {controller: controller.to_s, action: action}} do
|
49
|
+
instance.public_send(action)
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
def response_class
|
54
|
+
EurekaBot::Controller::Response
|
55
|
+
end
|
56
|
+
|
57
|
+
def trace_error(e, params={})
|
58
|
+
logger.error(e.http_body) if e.respond_to?(:http_body)
|
59
|
+
EurekaBot.exception_handler(e, self.class, params)
|
60
|
+
end
|
61
|
+
|
62
|
+
def self.has_action?(action)
|
63
|
+
return false unless action.present?
|
64
|
+
public_instance_methods(false).include?(action.to_sym)
|
65
|
+
end
|
66
|
+
|
67
|
+
class UnknownAction < StandardError; end
|
68
|
+
|
69
|
+
end
|
@@ -0,0 +1,42 @@
|
|
1
|
+
class EurekaBot::Controller::Response
|
2
|
+
include ActiveSupport::Callbacks
|
3
|
+
define_callbacks :add
|
4
|
+
|
5
|
+
attr_reader :logger, :data
|
6
|
+
attr_accessor :controller
|
7
|
+
|
8
|
+
def initialize(logger: EurekaBot.logger)
|
9
|
+
@logger = logger
|
10
|
+
@data = []
|
11
|
+
@order_counters = {}
|
12
|
+
end
|
13
|
+
|
14
|
+
def add(sync: false, **params)
|
15
|
+
run_callbacks :add do
|
16
|
+
message = params
|
17
|
+
if message[:order_queue]
|
18
|
+
@order_counters[message[:order_queue]] = (@order_counters[message[:order_queue]] || 0) + 1
|
19
|
+
message = {order: @order_counters[message[:order_queue]]}.merge(message)
|
20
|
+
end
|
21
|
+
if sync
|
22
|
+
self.class.sender_class.new.deliver(message)
|
23
|
+
else
|
24
|
+
@data << message
|
25
|
+
nil
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
def <<(params={})
|
31
|
+
add(params)
|
32
|
+
end
|
33
|
+
|
34
|
+
def to_a
|
35
|
+
data
|
36
|
+
end
|
37
|
+
|
38
|
+
def self.sender_class
|
39
|
+
@@sender_class ||= EurekaBot::Sender
|
40
|
+
end
|
41
|
+
|
42
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
module EurekaBot::Exceptions
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
included do
|
5
|
+
cattr_accessor :exception_chain
|
6
|
+
|
7
|
+
def self.add_exception_handler(&block)
|
8
|
+
self.exception_chain = (self.exception_chain || []) + [block]
|
9
|
+
end
|
10
|
+
|
11
|
+
def self.exception_handler(e, klass, args)
|
12
|
+
self.exception_chain.each do |handler|
|
13
|
+
handler.call(e, klass, args) rescue nil
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
add_exception_handler do |e, klass, args|
|
18
|
+
EurekaBot.logger.error(
|
19
|
+
{
|
20
|
+
class: e.class.to_s,
|
21
|
+
message: e.message,
|
22
|
+
args: args,
|
23
|
+
backtrace: e.backtrace
|
24
|
+
}.to_json
|
25
|
+
)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
module EurekaBot::Instrumentation
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
def instrument(name, level: Logger::DEBUG, **payload, &block)
|
5
|
+
self.class.instrument(name, level: level, **payload, &block)
|
6
|
+
end
|
7
|
+
|
8
|
+
def self.prefix
|
9
|
+
'eureka-bot'
|
10
|
+
end
|
11
|
+
|
12
|
+
class_methods do
|
13
|
+
def instrument(name, level: Logger::DEBUG, **payload, &block)
|
14
|
+
ActiveSupport::Notifications.instrument(
|
15
|
+
[
|
16
|
+
EurekaBot::Instrumentation.prefix,
|
17
|
+
name
|
18
|
+
].compact.join('.'),
|
19
|
+
{level: level}.merge(payload),
|
20
|
+
&block
|
21
|
+
)
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
end
|
@@ -0,0 +1,14 @@
|
|
1
|
+
require 'active_job'
|
2
|
+
|
3
|
+
class EurekaBot::Job < ActiveJob::Base
|
4
|
+
extend ActiveSupport::Autoload
|
5
|
+
autoload :Input
|
6
|
+
autoload :Output
|
7
|
+
include EurekaBot::Instrumentation
|
8
|
+
|
9
|
+
queue_as do
|
10
|
+
self.class.name.underscore.gsub('/', '__')
|
11
|
+
end
|
12
|
+
|
13
|
+
self.logger = EurekaBot.logger
|
14
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
class EurekaBot::Job::Input < EurekaBot::Job
|
2
|
+
queue_as :input
|
3
|
+
|
4
|
+
def perform(resolver_class, message)
|
5
|
+
instrument 'job.input' do
|
6
|
+
resolver = resolver_class.constantize.new(
|
7
|
+
message: message,
|
8
|
+
logger: logger
|
9
|
+
)
|
10
|
+
resolver.execute
|
11
|
+
end
|
12
|
+
rescue StandardError => e
|
13
|
+
EurekaBot.exception_handler(e, self.class, resolver_class: resolver_class, message: message)
|
14
|
+
end
|
15
|
+
|
16
|
+
class UnknownResolverClass < StandardError;
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,12 @@
|
|
1
|
+
class EurekaBot::Job::Output < EurekaBot::Job
|
2
|
+
queue_as :output
|
3
|
+
|
4
|
+
def perform(sender_class, message, *args)
|
5
|
+
instrument 'job.output' do
|
6
|
+
sender_class.constantize.new.deliver(message, *args)
|
7
|
+
end
|
8
|
+
rescue StandardError => e
|
9
|
+
EurekaBot.exception_handler(e, self.class, sender_class: sender_class, message: message)
|
10
|
+
end
|
11
|
+
|
12
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
module EurekaBot::Notification
|
2
|
+
def self.log_notification(severity=::Logger::DEBUG, name, started, finished, id, payload)
|
3
|
+
EurekaBot.logger.add(
|
4
|
+
severity,
|
5
|
+
{
|
6
|
+
name: name,
|
7
|
+
execution_time: finished - started,
|
8
|
+
started: started,
|
9
|
+
finished: finished,
|
10
|
+
id: id,
|
11
|
+
payload: payload
|
12
|
+
}.to_json
|
13
|
+
)
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
ActiveSupport::Notifications.subscribe /eureka-bot\..*/ do |*args|
|
18
|
+
severity = if args[0] == 'eureka-bot.job.perform'
|
19
|
+
::Logger::INFO
|
20
|
+
else
|
21
|
+
::Logger::DEBUG
|
22
|
+
end
|
23
|
+
EurekaBot::Notification.log_notification(severity, *args)
|
24
|
+
end
|
25
|
+
|
@@ -0,0 +1,78 @@
|
|
1
|
+
class EurekaBot::Resolver
|
2
|
+
include EurekaBot::Instrumentation
|
3
|
+
|
4
|
+
class ActionNotFound < StandardError; end
|
5
|
+
class ControllerNotFound < StandardError; end
|
6
|
+
|
7
|
+
attr_reader :logger, :message, :response, :controller
|
8
|
+
|
9
|
+
def initialize(message:, response: nil, logger: EurekaBot.logger)
|
10
|
+
@logger = logger
|
11
|
+
@message = message
|
12
|
+
@response = response
|
13
|
+
end
|
14
|
+
|
15
|
+
def resolve
|
16
|
+
nil
|
17
|
+
end
|
18
|
+
|
19
|
+
def resolved
|
20
|
+
@resolved ||= begin
|
21
|
+
resolved = resolve
|
22
|
+
instrument 'resolver.resolved', level: Logger::INFO, result: resolved, message: message
|
23
|
+
resolved
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
def execute
|
28
|
+
raise ControllerNotFound.new("Can't resolve #{message}") unless resolved
|
29
|
+
@controller = resolve_controller(resolved[:controller]).new(
|
30
|
+
params: resolved[:params] || {},
|
31
|
+
message: message,
|
32
|
+
logger: logger,
|
33
|
+
response: response
|
34
|
+
)
|
35
|
+
instrument 'resolver.execute' do
|
36
|
+
controller.execute(resolved[:action])
|
37
|
+
end
|
38
|
+
self
|
39
|
+
end
|
40
|
+
|
41
|
+
def parameterize(str)
|
42
|
+
ActiveSupport::Inflector.parameterize(str, separator: '_').presence
|
43
|
+
end
|
44
|
+
|
45
|
+
def decode_params(str, delimiter: ':')
|
46
|
+
action_raw, version, *params = str.split(delimiter)
|
47
|
+
{
|
48
|
+
action: parameterize(action_raw),
|
49
|
+
version: version,
|
50
|
+
params: {raw: params, version: version} || {}
|
51
|
+
}
|
52
|
+
end
|
53
|
+
|
54
|
+
def encode_params(action:, version: 'v1', params: [])
|
55
|
+
([action, version] + params.map(&:to_json)).join(':')
|
56
|
+
end
|
57
|
+
|
58
|
+
def resolve_controller(klass)
|
59
|
+
instrument 'resolver.resolve_controller' do
|
60
|
+
return klass if klass.is_a?(Class) && klass <= EurekaBot::Controller
|
61
|
+
camelized = ActiveSupport::Inflector.camelize(klass)
|
62
|
+
camelized = [camelized, 'Controller'].join unless camelized.include?('Controller')
|
63
|
+
[
|
64
|
+
camelized,
|
65
|
+
[controller_namespace, camelized].join('::')
|
66
|
+
].each do |variant|
|
67
|
+
constant= ActiveSupport::Inflector.safe_constantize(variant)
|
68
|
+
return constant if constant
|
69
|
+
end
|
70
|
+
raise ControllerNotFound.new(klass)
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
def controller_namespace
|
75
|
+
EurekaBot
|
76
|
+
end
|
77
|
+
|
78
|
+
end
|
data/log/.keep
ADDED
File without changes
|
@@ -0,0 +1,21 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
RSpec.describe EurekaBot::Controller::Response do
|
4
|
+
|
5
|
+
context 'message order' do
|
6
|
+
let(:response) { EurekaBot::Controller::Response.new }
|
7
|
+
it do
|
8
|
+
response.add(order_queue: 'test-lock')
|
9
|
+
response.add(order_queue: 'test-lock')
|
10
|
+
response.add(order_queue: 'test-lock-2')
|
11
|
+
response.add
|
12
|
+
expect(response.to_a).to include(
|
13
|
+
hash_including(order_queue: 'test-lock', order: 1),
|
14
|
+
hash_including(order_queue: 'test-lock', order: 2),
|
15
|
+
hash_including(order_queue: 'test-lock-2', order: 1),
|
16
|
+
hash_not_including(:order_queue, :order)
|
17
|
+
)
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
end
|
@@ -0,0 +1,59 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
RSpec.describe EurekaBot::Controller do
|
4
|
+
|
5
|
+
context 'execute' do
|
6
|
+
let(:controller_klass) {
|
7
|
+
Class.new(EurekaBot::Controller) do
|
8
|
+
def sample_action; end
|
9
|
+
end
|
10
|
+
}
|
11
|
+
let(:controller) { controller_klass.new }
|
12
|
+
it do
|
13
|
+
expect(controller).to receive(:sample_action).once
|
14
|
+
controller.execute(:sample_action)
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
context 'response' do
|
19
|
+
let(:controller) { EurekaBot::Controller.new }
|
20
|
+
|
21
|
+
it 'is EurekaBot::Controller::Response subclass' do
|
22
|
+
expect(controller.response).to be_a_kind_of(EurekaBot::Controller::Response)
|
23
|
+
end
|
24
|
+
|
25
|
+
context '#answer' do
|
26
|
+
let(:controller) do
|
27
|
+
Class.new(EurekaBot::Controller) do
|
28
|
+
def some_action
|
29
|
+
answer(some_action: 'finished')
|
30
|
+
end
|
31
|
+
end.new
|
32
|
+
end
|
33
|
+
|
34
|
+
before do
|
35
|
+
controller.some_action
|
36
|
+
end
|
37
|
+
|
38
|
+
it do
|
39
|
+
expect(controller.response.to_a).to include(include(some_action: 'finished'))
|
40
|
+
end
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
context 'redirect' do
|
45
|
+
let(:controller_klass) {
|
46
|
+
Class.new(EurekaBot::Controller) do
|
47
|
+
def sample_action_1
|
48
|
+
redirect params[:klass], :sample_action_2
|
49
|
+
end
|
50
|
+
def sample_action_2; end
|
51
|
+
end
|
52
|
+
}
|
53
|
+
let(:controller) { controller_klass.new(params: {klass: controller_klass}) }
|
54
|
+
it do
|
55
|
+
expect_any_instance_of(controller_klass).to receive(:sample_action_2).once
|
56
|
+
controller.execute(:sample_action_1)
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
RSpec.describe EurekaBot::Instrumentation do
|
4
|
+
context 'level' do
|
5
|
+
let(:resolver) { EurekaBot::Resolver.new(message: {}) }
|
6
|
+
it do
|
7
|
+
payloads = []
|
8
|
+
subscriber = ActiveSupport::Notifications.subscribe /.*/ do |name, started, finished, id, payload|
|
9
|
+
expect(payload).to include(:level)
|
10
|
+
payloads << payload
|
11
|
+
end
|
12
|
+
|
13
|
+
expect{resolver.resolved}.to change{payloads.length}.from(0).to(1)
|
14
|
+
|
15
|
+
ActiveSupport::Notifications.unsubscribe(subscriber)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,76 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
RSpec.describe EurekaBot::Job do
|
4
|
+
context '#execute' do
|
5
|
+
let(:resolver_class) do
|
6
|
+
class TestResolver < EurekaBot::Resolver
|
7
|
+
end
|
8
|
+
TestResolver
|
9
|
+
end
|
10
|
+
|
11
|
+
let(:controller_class) do
|
12
|
+
class TestController < EurekaBot::Controller
|
13
|
+
def some_action
|
14
|
+
answer(some_action: 'finished')
|
15
|
+
end
|
16
|
+
end
|
17
|
+
TestController
|
18
|
+
end
|
19
|
+
|
20
|
+
let(:job_class) do
|
21
|
+
$job_result = nil
|
22
|
+
class TestJob < EurekaBot::Job::Input
|
23
|
+
def perform(resolver_class, message)
|
24
|
+
$job_result = super
|
25
|
+
end
|
26
|
+
end
|
27
|
+
TestJob
|
28
|
+
end
|
29
|
+
|
30
|
+
after do
|
31
|
+
Object.send(:remove_const, :TestController)
|
32
|
+
Object.send(:remove_const, :TestJob)
|
33
|
+
Object.send(:remove_const, :TestResolver)
|
34
|
+
end
|
35
|
+
|
36
|
+
before do
|
37
|
+
allow_any_instance_of(resolver_class).to receive(:resolve).and_return(
|
38
|
+
controller: controller_class,
|
39
|
+
action: :some_action
|
40
|
+
)
|
41
|
+
end
|
42
|
+
|
43
|
+
context 'results' do
|
44
|
+
before do
|
45
|
+
job_class.perform_later(resolver_class.to_s, {})
|
46
|
+
end
|
47
|
+
|
48
|
+
it do
|
49
|
+
expect($job_result).to be
|
50
|
+
expect(
|
51
|
+
$job_result.controller.response.to_a
|
52
|
+
).to include(include(some_action: 'finished'))
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
context 'exceptions' do
|
57
|
+
let(:controller_class) do
|
58
|
+
class TestController < EurekaBot::Controller
|
59
|
+
def some_action
|
60
|
+
raise StandardError.new('test action error')
|
61
|
+
end
|
62
|
+
end
|
63
|
+
TestController
|
64
|
+
end
|
65
|
+
it do
|
66
|
+
$result = nil
|
67
|
+
EurekaBot.add_exception_handler do |e|
|
68
|
+
$result = e.message
|
69
|
+
end
|
70
|
+
expect{
|
71
|
+
job_class.perform_later(resolver_class.to_s, {})
|
72
|
+
}.to change{$result}.from(nil).to('test action error')
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
@@ -0,0 +1,94 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
RSpec.describe EurekaBot::Resolver do
|
4
|
+
context '#execute' do
|
5
|
+
let(:controller_class) do
|
6
|
+
Class.new(EurekaBot::Controller) do
|
7
|
+
def some_action
|
8
|
+
answer(some_action: 'finished')
|
9
|
+
end
|
10
|
+
end
|
11
|
+
end
|
12
|
+
|
13
|
+
let(:resolver) do
|
14
|
+
Class.new(EurekaBot::Resolver).new(message: {})
|
15
|
+
end
|
16
|
+
|
17
|
+
before do
|
18
|
+
allow(resolver).to receive(:resolve).and_return(
|
19
|
+
controller: controller_class,
|
20
|
+
action: :some_action
|
21
|
+
)
|
22
|
+
resolver.execute
|
23
|
+
end
|
24
|
+
|
25
|
+
it do
|
26
|
+
expect(resolver.controller.response.to_a).to include(include(some_action: 'finished'))
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
context '#exception' do
|
31
|
+
let(:controller_class) do
|
32
|
+
Class.new(EurekaBot::Controller) do
|
33
|
+
def some_action
|
34
|
+
raise StandardError.new
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
|
39
|
+
let(:resolver) do
|
40
|
+
Class.new(EurekaBot::Resolver).new(message: {})
|
41
|
+
end
|
42
|
+
|
43
|
+
before do
|
44
|
+
allow(resolver).to receive(:resolve).and_return(
|
45
|
+
controller: controller_class,
|
46
|
+
action: :some_action
|
47
|
+
)
|
48
|
+
end
|
49
|
+
|
50
|
+
it do
|
51
|
+
expect{resolver.execute}.to raise_error(StandardError)
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
55
|
+
context 'namespace' do
|
56
|
+
let(:controller_class) do
|
57
|
+
Class.new(EurekaBot::Controller) do
|
58
|
+
def some_action
|
59
|
+
answer(some_action: 'finished')
|
60
|
+
end
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
let(:resolver_class) do
|
65
|
+
Class.new(EurekaBot::Resolver) do
|
66
|
+
def resolve
|
67
|
+
return {
|
68
|
+
controller: 'test',
|
69
|
+
action: :some_action
|
70
|
+
}
|
71
|
+
end
|
72
|
+
|
73
|
+
def controller_namespace
|
74
|
+
NamespaceTestResolver
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
let(:resolver) do
|
80
|
+
resolver_class.new(message: {})
|
81
|
+
end
|
82
|
+
|
83
|
+
before do
|
84
|
+
module NamespaceTestResolver; end
|
85
|
+
NamespaceTestResolver::TestController = controller_class
|
86
|
+
end
|
87
|
+
|
88
|
+
it do
|
89
|
+
expect(resolver.resolve).to eq(controller: 'test', action: :some_action)
|
90
|
+
resolver.execute
|
91
|
+
expect(resolver.controller.response.to_a).to include(include(some_action: 'finished'))
|
92
|
+
end
|
93
|
+
end
|
94
|
+
end
|
data/spec/spec_helper.rb
ADDED
@@ -0,0 +1,30 @@
|
|
1
|
+
ENV['NEW_RELIC_ENV'] = 'test'
|
2
|
+
|
3
|
+
require 'bundler'
|
4
|
+
Bundler.require
|
5
|
+
require './lib/eureka-bot'
|
6
|
+
require 'webmock/rspec'
|
7
|
+
EurekaBot::Job.queue_adapter = :inline
|
8
|
+
|
9
|
+
$:.unshift File.dirname(__FILE__) + '/..'
|
10
|
+
Dir['spec/support/**/*.rb'].each { |f| require f }
|
11
|
+
|
12
|
+
RSpec.configure do |config|
|
13
|
+
config.include SettingsHelper
|
14
|
+
|
15
|
+
config.expect_with :rspec do |expectations|
|
16
|
+
expectations.include_chain_clauses_in_custom_matcher_descriptions = true
|
17
|
+
end
|
18
|
+
|
19
|
+
config.mock_with :rspec do |mocks|
|
20
|
+
mocks.verify_partial_doubles = true
|
21
|
+
end
|
22
|
+
|
23
|
+
config.disable_monkey_patching!
|
24
|
+
# config.warnings = true
|
25
|
+
|
26
|
+
config.profile_examples = 10
|
27
|
+
config.order = :random
|
28
|
+
|
29
|
+
Kernel.srand config.seed
|
30
|
+
end
|
data/tmp/.keep
ADDED
File without changes
|
metadata
ADDED
@@ -0,0 +1,160 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: eureka-bot
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 1.0.16
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Yury Alexandrov
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
date: 2018-04-08 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: activesupport
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - "~>"
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '5.0'
|
20
|
+
- - ">="
|
21
|
+
- !ruby/object:Gem::Version
|
22
|
+
version: 5.0.0
|
23
|
+
type: :runtime
|
24
|
+
prerelease: false
|
25
|
+
version_requirements: !ruby/object:Gem::Requirement
|
26
|
+
requirements:
|
27
|
+
- - "~>"
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: '5.0'
|
30
|
+
- - ">="
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: 5.0.0
|
33
|
+
- !ruby/object:Gem::Dependency
|
34
|
+
name: activejob
|
35
|
+
requirement: !ruby/object:Gem::Requirement
|
36
|
+
requirements:
|
37
|
+
- - "~>"
|
38
|
+
- !ruby/object:Gem::Version
|
39
|
+
version: '5.0'
|
40
|
+
- - ">="
|
41
|
+
- !ruby/object:Gem::Version
|
42
|
+
version: 5.0.0
|
43
|
+
type: :runtime
|
44
|
+
prerelease: false
|
45
|
+
version_requirements: !ruby/object:Gem::Requirement
|
46
|
+
requirements:
|
47
|
+
- - "~>"
|
48
|
+
- !ruby/object:Gem::Version
|
49
|
+
version: '5.0'
|
50
|
+
- - ">="
|
51
|
+
- !ruby/object:Gem::Version
|
52
|
+
version: 5.0.0
|
53
|
+
- !ruby/object:Gem::Dependency
|
54
|
+
name: rspec
|
55
|
+
requirement: !ruby/object:Gem::Requirement
|
56
|
+
requirements:
|
57
|
+
- - "~>"
|
58
|
+
- !ruby/object:Gem::Version
|
59
|
+
version: '3.5'
|
60
|
+
- - ">="
|
61
|
+
- !ruby/object:Gem::Version
|
62
|
+
version: 3.5.0
|
63
|
+
type: :development
|
64
|
+
prerelease: false
|
65
|
+
version_requirements: !ruby/object:Gem::Requirement
|
66
|
+
requirements:
|
67
|
+
- - "~>"
|
68
|
+
- !ruby/object:Gem::Version
|
69
|
+
version: '3.5'
|
70
|
+
- - ">="
|
71
|
+
- !ruby/object:Gem::Version
|
72
|
+
version: 3.5.0
|
73
|
+
- !ruby/object:Gem::Dependency
|
74
|
+
name: webmock
|
75
|
+
requirement: !ruby/object:Gem::Requirement
|
76
|
+
requirements:
|
77
|
+
- - "~>"
|
78
|
+
- !ruby/object:Gem::Version
|
79
|
+
version: 3.3.0
|
80
|
+
- - ">="
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: 3.3.0
|
83
|
+
type: :development
|
84
|
+
prerelease: false
|
85
|
+
version_requirements: !ruby/object:Gem::Requirement
|
86
|
+
requirements:
|
87
|
+
- - "~>"
|
88
|
+
- !ruby/object:Gem::Version
|
89
|
+
version: 3.3.0
|
90
|
+
- - ">="
|
91
|
+
- !ruby/object:Gem::Version
|
92
|
+
version: 3.3.0
|
93
|
+
description:
|
94
|
+
email:
|
95
|
+
- dizeru@gmail.com
|
96
|
+
executables: []
|
97
|
+
extensions: []
|
98
|
+
extra_rdoc_files: []
|
99
|
+
files:
|
100
|
+
- ".gitignore"
|
101
|
+
- ".rspec"
|
102
|
+
- Gemfile
|
103
|
+
- LICENSE.txt
|
104
|
+
- README.md
|
105
|
+
- eureka-bot.gemspec
|
106
|
+
- lib/eureka-bot.rb
|
107
|
+
- lib/eureka_bot/controller.rb
|
108
|
+
- lib/eureka_bot/controller/response.rb
|
109
|
+
- lib/eureka_bot/exceptions.rb
|
110
|
+
- lib/eureka_bot/instrumentation.rb
|
111
|
+
- lib/eureka_bot/integration.rb
|
112
|
+
- lib/eureka_bot/integration/new_relic.rb
|
113
|
+
- lib/eureka_bot/job.rb
|
114
|
+
- lib/eureka_bot/job/input.rb
|
115
|
+
- lib/eureka_bot/job/output.rb
|
116
|
+
- lib/eureka_bot/notifications.rb
|
117
|
+
- lib/eureka_bot/resolver.rb
|
118
|
+
- lib/eureka_bot/sender.rb
|
119
|
+
- lib/eureka_bot/version.rb
|
120
|
+
- log/.keep
|
121
|
+
- spec/eureka-bot/controller/response_spec.rb
|
122
|
+
- spec/eureka-bot/controller_spec.rb
|
123
|
+
- spec/eureka-bot/instrumentation_spec.rb
|
124
|
+
- spec/eureka-bot/job_spec.rb
|
125
|
+
- spec/eureka-bot/resolver_spec.rb
|
126
|
+
- spec/spec_helper.rb
|
127
|
+
- spec/support/settings_helper.rb
|
128
|
+
- tmp/.keep
|
129
|
+
homepage: https://github.com/dizer/eureka-bot
|
130
|
+
licenses:
|
131
|
+
- MIT
|
132
|
+
metadata: {}
|
133
|
+
post_install_message:
|
134
|
+
rdoc_options: []
|
135
|
+
require_paths:
|
136
|
+
- lib
|
137
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
138
|
+
requirements:
|
139
|
+
- - ">="
|
140
|
+
- !ruby/object:Gem::Version
|
141
|
+
version: '0'
|
142
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
143
|
+
requirements:
|
144
|
+
- - ">="
|
145
|
+
- !ruby/object:Gem::Version
|
146
|
+
version: '0'
|
147
|
+
requirements: []
|
148
|
+
rubyforge_project:
|
149
|
+
rubygems_version: 2.6.14
|
150
|
+
signing_key:
|
151
|
+
specification_version: 4
|
152
|
+
summary: Run your messenger bots
|
153
|
+
test_files:
|
154
|
+
- spec/eureka-bot/controller/response_spec.rb
|
155
|
+
- spec/eureka-bot/controller_spec.rb
|
156
|
+
- spec/eureka-bot/instrumentation_spec.rb
|
157
|
+
- spec/eureka-bot/job_spec.rb
|
158
|
+
- spec/eureka-bot/resolver_spec.rb
|
159
|
+
- spec/spec_helper.rb
|
160
|
+
- spec/support/settings_helper.rb
|