ego 0.3.0 → 0.4.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 +4 -4
- data/.travis.yml +4 -0
- data/.yardopts +1 -0
- data/README.md +40 -35
- data/ego.gemspec +1 -0
- data/lib/ego.rb +40 -3
- data/lib/ego/capability.rb +30 -0
- data/lib/ego/filesystem.rb +58 -29
- data/lib/ego/handler.rb +62 -59
- data/lib/ego/options.rb +14 -2
- data/lib/ego/plugin.rb +58 -0
- data/lib/ego/plugins/capabilities.rb +17 -0
- data/lib/ego/plugins/fallback.rb +36 -0
- data/lib/ego/plugins/robot_io.rb +16 -0
- data/lib/ego/plugins/social.rb +19 -0
- data/lib/ego/plugins/system.rb +21 -0
- data/lib/ego/printer.rb +95 -0
- data/lib/ego/robot.rb +157 -10
- data/lib/ego/robot_error.rb +8 -0
- data/lib/ego/runner.rb +51 -20
- data/lib/ego/version.rb +2 -1
- data/spec/ego/capability_spec.rb +23 -0
- data/spec/ego/handler_spec.rb +63 -0
- data/spec/ego/options_spec.rb +23 -13
- data/spec/ego/plugin_spec.rb +68 -0
- data/spec/ego/printer_spec.rb +120 -0
- data/spec/ego/robot_error_spec.rb +12 -0
- data/spec/ego/robot_spec.rb +283 -26
- metadata +36 -10
- data/lib/ego/formatter.rb +0 -36
- data/lib/ego/handler/default.rb +0 -31
- data/lib/ego/handler/echo.rb +0 -9
- data/lib/ego/handler/greet.rb +0 -17
- data/lib/ego/handler/handlers.rb +0 -15
- data/lib/ego/handler/self.rb +0 -9
- data/lib/ego/listener.rb +0 -25
- data/spec/ego/formatter_spec.rb +0 -53
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA1:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 7a1b7a652d2e75a9e5a22d05a9627f1b5f63b94d
|
4
|
+
data.tar.gz: 5e7aa5063ddf5f986b341a9bb4ac6f9f165d3358
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 0711a3f6ddb2ef92913409ffb2e2a66a55b6dbec57f0fcf9057ab598803175254b26a40785996aab3d6ece9eae279f03fa55417167f057a0531d9d465e932116
|
7
|
+
data.tar.gz: b816f63c6fadea1d33066b65f8ca6067c2af1409e01e323353f57a58f99c3d28775dfc6d6d174c14cca138ab72fc63727b29709c1a93a578f62184d4ab1e0fa3
|
data/.travis.yml
ADDED
data/.yardopts
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
--markup markdown --name-tag hook:"Runs Hooks" - LICENSE.txt
|
data/README.md
CHANGED
@@ -26,19 +26,18 @@ If you want to interact with ego as a REPL, try:
|
|
26
26
|
## Extending
|
27
27
|
|
28
28
|
Ego does very few things out of the box, but it's designed to be extensible!
|
29
|
-
You can personalize ego and teach it to do new things by defining
|
29
|
+
You can personalize ego and teach it to do new things by defining plug-ins,
|
30
30
|
which are small scripts that tell ego what queries to listen for and how to
|
31
|
-
respond to them
|
32
|
-
beginning with "hello...", "hi...", or
|
31
|
+
respond to them and can even add entirely new features. Here's what a plug-in
|
32
|
+
looks like that responds to a query beginning with "hello...", "hi...", or
|
33
|
+
"hey..." with its own random greeting:
|
33
34
|
|
34
35
|
```ruby
|
35
|
-
Ego
|
36
|
-
|
36
|
+
Ego.plugin do |robot|
|
37
|
+
robot.can 'greet you'
|
37
38
|
|
38
|
-
|
39
|
-
|
40
|
-
handler.run do |robot|
|
41
|
-
robot.respond [
|
39
|
+
robot.on /^(hello|hi|hey)/i, 3 do
|
40
|
+
say [
|
42
41
|
'Hello.',
|
43
42
|
'Hi.',
|
44
43
|
'Ciao.',
|
@@ -53,44 +52,45 @@ end
|
|
53
52
|
Let's break that down:
|
54
53
|
|
55
54
|
```ruby
|
56
|
-
|
55
|
+
robot.can 'greet you'
|
57
56
|
```
|
58
57
|
|
59
|
-
|
60
|
-
this
|
58
|
+
This adds a new "capability", which serves as documentation for the user and
|
59
|
+
answers the question "What can this plug-in do?"
|
61
60
|
|
62
61
|
```ruby
|
63
|
-
|
62
|
+
robot.on /^(hello|hi|hey)/i, 3 ...
|
64
63
|
```
|
65
64
|
|
66
|
-
This is the
|
67
|
-
The first argument is a regular expression to match the query against.
|
65
|
+
This is the condition that determines what queries should invoke the following
|
66
|
+
action. The first argument is a regular expression to match the query against.
|
68
67
|
Sometimes you may want to match very specific things and sometimes something
|
69
|
-
broader. To help ego respond the right way when two or more
|
70
|
-
your query, you can optionally specify a
|
71
|
-
priority)
|
72
|
-
different priorities.
|
68
|
+
broader. To help ego respond the right way when two or more patterns match
|
69
|
+
your query, you can optionally specify a priority (higher number = higher
|
70
|
+
priority) as the second argument.
|
73
71
|
|
74
72
|
```ruby
|
75
|
-
|
76
|
-
|
73
|
+
... do
|
74
|
+
say [
|
75
|
+
'Hello.',
|
76
|
+
'Hi.',
|
77
|
+
'Ciao.',
|
78
|
+
].sample
|
77
79
|
end
|
78
80
|
```
|
79
81
|
|
80
|
-
This is the part that gets run when
|
82
|
+
This is the part that gets run when the pattern matches the query. From here
|
81
83
|
you can do anything you want including deferring to external programs. The
|
82
|
-
`robot`
|
83
|
-
to make use of part of the query
|
84
|
-
groups through the optional `params` parameter:
|
84
|
+
`robot` provides various methods to you to respond to the user. Usually, you'll
|
85
|
+
want to make use of part of the query inside the action. You can access named
|
86
|
+
match groups through the optional `params` parameter:
|
85
87
|
|
86
88
|
```ruby
|
87
|
-
Ego
|
88
|
-
|
89
|
-
|
90
|
-
handler.listen /^say (?<input>.+)/i
|
89
|
+
Ego.plugin do |robot|
|
90
|
+
robot.can 'repeat what you say'
|
91
91
|
|
92
|
-
|
93
|
-
|
92
|
+
robot.on /^say (?<input>.+)/i do |params|
|
93
|
+
say params[:input]
|
94
94
|
end
|
95
95
|
end
|
96
96
|
```
|
@@ -100,12 +100,17 @@ Try it out (this one is already included):
|
|
100
100
|
$ ego say something
|
101
101
|
> something
|
102
102
|
|
103
|
-
Ego looks for user-defined
|
104
|
-
(that's `~/.config/ego/
|
105
|
-
automatically at runtime. Each
|
106
|
-
extension (e.g., `~/.config/ego/
|
103
|
+
Ego looks for user-defined plug-ins in `$XDG_CONFIG_HOME/ego/plugins/`
|
104
|
+
(that's `~/.config/ego/plugins/` by default), and registers them
|
105
|
+
automatically at runtime. Each plug-in goes in it's own file with an `.rb`
|
106
|
+
extension (e.g., `~/.config/ego/plugins/my_plugin.rb`). Be careful—ego will
|
107
107
|
execute any Ruby scripts in this directory indiscriminately.
|
108
108
|
|
109
|
+
### See Also
|
110
|
+
|
111
|
+
- [API documentation](http://www.rubydoc.info/gems/ego)
|
112
|
+
- [Examples in the wiki](https://github.com/noahfrederick/ego/wiki)
|
113
|
+
|
109
114
|
## License
|
110
115
|
|
111
116
|
Copyright (C) 2016-2017 Noah Frederick
|
data/ego.gemspec
CHANGED
data/lib/ego.rb
CHANGED
@@ -1,5 +1,42 @@
|
|
1
|
-
require_relative 'ego/version'
|
2
|
-
require_relative 'ego/options'
|
3
1
|
require_relative 'ego/filesystem'
|
4
2
|
require_relative 'ego/robot'
|
5
|
-
require_relative 'ego/
|
3
|
+
require_relative 'ego/plugin'
|
4
|
+
require_relative 'ego/printer'
|
5
|
+
require_relative 'ego/version'
|
6
|
+
|
7
|
+
module Ego
|
8
|
+
# Public interface for defining a plug-in.
|
9
|
+
#
|
10
|
+
# Ego looks for user-defined plug-ins in `$XDG_CONFIG_HOME/ego/plugins/`
|
11
|
+
# (that's `~/.config/ego/plugins/` by default), and registers them
|
12
|
+
# automatically at runtime. Each plug-in goes in it's own file with an `.rb`
|
13
|
+
# extension (e.g., `~/.config/ego/plugins/my_plugin.rb`).
|
14
|
+
#
|
15
|
+
# Be careful—ego will execute any Ruby scripts in this directory
|
16
|
+
# indiscriminately.
|
17
|
+
#
|
18
|
+
# @example Create and register a new plug-in
|
19
|
+
# # ~/.config/ego/plugins/echo.rb
|
20
|
+
# Ego.plugin do |robot|
|
21
|
+
# robot.can 'repeat what you say'
|
22
|
+
#
|
23
|
+
# robot.on /^say (?<input>.+)/i do |params|
|
24
|
+
# say params[:input]
|
25
|
+
# end
|
26
|
+
# end
|
27
|
+
#
|
28
|
+
# @param name [String, nil] the plug-in name (uses plug-in file's basename if given `nil`)
|
29
|
+
# @param builtin [Boolean] whether to register as a built-in plug-in
|
30
|
+
# @param body the plug-in body
|
31
|
+
# @return [Plugin] the instantiated plug-in
|
32
|
+
#
|
33
|
+
# @see Robot
|
34
|
+
def self.plugin(name = nil, builtin: false, &body)
|
35
|
+
if name.nil?
|
36
|
+
path = caller_locations(1, 1)[0].absolute_path
|
37
|
+
name = File.basename(path, '.*')
|
38
|
+
end
|
39
|
+
|
40
|
+
Plugin.register(name, body, builtin: builtin)
|
41
|
+
end
|
42
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
module Ego
|
2
|
+
# A capability defines functionality added to a `Robot` instance by a
|
3
|
+
# plug-in.
|
4
|
+
#
|
5
|
+
# @note New capabilities should be specified by plug-ins using the
|
6
|
+
# `robot#can` method.
|
7
|
+
#
|
8
|
+
# @example Add a capability to the robot instance
|
9
|
+
# Ego.plugin do |robot|
|
10
|
+
# robot.can 'repeat what you say'
|
11
|
+
# # ...
|
12
|
+
# end
|
13
|
+
#
|
14
|
+
# @see Robot#can
|
15
|
+
class Capability
|
16
|
+
attr_reader :desc, :plugin
|
17
|
+
|
18
|
+
# @param desc [String] the capability description answering "What can the robot do?"
|
19
|
+
# @param plugin [Plugin] the plug-in that provides the capability
|
20
|
+
def initialize(desc, plugin)
|
21
|
+
@desc = desc
|
22
|
+
@plugin = plugin
|
23
|
+
end
|
24
|
+
|
25
|
+
# @return [String] the capability description
|
26
|
+
def to_s
|
27
|
+
@desc
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
data/lib/ego/filesystem.rb
CHANGED
@@ -1,31 +1,60 @@
|
|
1
|
-
module Ego
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
File.
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
1
|
+
module Ego
|
2
|
+
# Provides utility methods for getting configuration, data, and cache paths.
|
3
|
+
module Filesystem
|
4
|
+
# Glob pattern for matching plug-in files
|
5
|
+
PLUGIN_GLOB = 'plugins/*.rb'
|
6
|
+
|
7
|
+
# XDG subdirectoy name
|
8
|
+
BASENAME = 'ego'
|
9
|
+
|
10
|
+
# Value of `$XDG_CACHE_HOME` or fallback if not set
|
11
|
+
XDG_CACHE_HOME = ENV['XDG_CACHE_HOME'] || File.expand_path('~/.cache')
|
12
|
+
# Value of `$XDG_CONFIG_HOME` or fallback if not set
|
13
|
+
XDG_CONFIG_HOME = ENV['XDG_CONFIG_HOME'] || File.expand_path('~/.config')
|
14
|
+
# Value of `$XDG_DATA_HOME` or fallback if not set
|
15
|
+
XDG_DATA_HOME = ENV['XDG_DATA_HOME'] || File.expand_path('~/.local/share')
|
16
|
+
|
17
|
+
module_function
|
18
|
+
|
19
|
+
# @param path [String] path to append
|
20
|
+
# @return [String] the path to cache directory with `path` appended
|
21
|
+
#
|
22
|
+
# @see config
|
23
|
+
# @see data
|
24
|
+
def cache(path = '')
|
25
|
+
File.join(XDG_CACHE_HOME, BASENAME, path)
|
26
|
+
end
|
27
|
+
|
28
|
+
# @param path [String] path to append
|
29
|
+
# @return [String] the path to config directory with `path` appended
|
30
|
+
#
|
31
|
+
# @see cache
|
32
|
+
# @see data
|
33
|
+
def config(path = '')
|
34
|
+
File.join(XDG_CONFIG_HOME, BASENAME, path)
|
35
|
+
end
|
36
|
+
|
37
|
+
# @param path [String] path to append
|
38
|
+
# @return [String] the path to data directory with `path` appended
|
39
|
+
#
|
40
|
+
# @see cache
|
41
|
+
# @see config
|
42
|
+
def data(path = '')
|
43
|
+
File.join(XDG_DATA_HOME, BASENAME, path)
|
44
|
+
end
|
45
|
+
|
46
|
+
# @return [Array] all built-in plug-in paths
|
47
|
+
#
|
48
|
+
# @see user_plugins
|
49
|
+
def builtin_plugins
|
50
|
+
Dir[File.expand_path(PLUGIN_GLOB, __dir__)]
|
51
|
+
end
|
52
|
+
|
53
|
+
# @return [Array] all user plug-in paths
|
54
|
+
#
|
55
|
+
# @see builtin_plugins
|
56
|
+
def user_plugins
|
57
|
+
Dir[File.expand_path(PLUGIN_GLOB, config)]
|
58
|
+
end
|
30
59
|
end
|
31
60
|
end
|
data/lib/ego/handler.rb
CHANGED
@@ -1,76 +1,79 @@
|
|
1
|
-
require_relative '
|
2
|
-
require_relative 'formatter'
|
1
|
+
require_relative 'robot_error'
|
3
2
|
|
4
3
|
module Ego
|
4
|
+
# Handlers map user queries to actions.
|
5
|
+
#
|
6
|
+
# @note Handlers should be registered by plug-ins using the `robot#on`
|
7
|
+
# method.
|
8
|
+
#
|
9
|
+
# @example Add a handler to the robot instance
|
10
|
+
# robot.on(/^pattern/) do
|
11
|
+
# # ...
|
12
|
+
# end
|
13
|
+
#
|
14
|
+
# @example Add a handler with priority to the robot instance
|
15
|
+
# robot.on(/^pattern/, 7) do
|
16
|
+
# # ...
|
17
|
+
# end
|
18
|
+
#
|
19
|
+
# @example Add multiple handlers for the same action
|
20
|
+
# robot.on(
|
21
|
+
# /pattern/ => 6,
|
22
|
+
# /other pattern/ => 7,
|
23
|
+
# /another/ => 3,
|
24
|
+
# ) do
|
25
|
+
# # ...
|
26
|
+
# end
|
27
|
+
#
|
28
|
+
# @example Passing a lambda as a condition
|
29
|
+
# robot.on(->(query) { query.length > 10 }) do
|
30
|
+
# # ...
|
31
|
+
# end
|
32
|
+
#
|
33
|
+
# @see Robot#on
|
5
34
|
class Handler
|
6
|
-
|
7
|
-
@@listeners = []
|
35
|
+
include Comparable
|
8
36
|
|
9
|
-
attr_reader :
|
10
|
-
attr_accessor :description
|
37
|
+
attr_reader :condition, :action, :priority
|
11
38
|
|
12
|
-
|
13
|
-
|
39
|
+
# @param condition [Proc, #match] the condition that triggers the supplied action
|
40
|
+
# @param action [Proc] the block to be executed when condition is met
|
41
|
+
# @param priority [Integer] the handler priority (higher number = higher priority)
|
42
|
+
def initialize(condition, action, priority = 5)
|
43
|
+
@condition = normalize(condition)
|
44
|
+
@action = action
|
45
|
+
@priority = priority
|
14
46
|
end
|
15
47
|
|
16
|
-
|
17
|
-
|
48
|
+
# Compare `priority` with `other.priority`.
|
49
|
+
#
|
50
|
+
# @param other [Handler] the handler to compare with
|
51
|
+
def <=>(other)
|
52
|
+
@priority <=> other.priority
|
18
53
|
end
|
19
54
|
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
def run(robot = nil, params = nil, &action)
|
28
|
-
if block_given?
|
29
|
-
@action = action
|
30
|
-
end
|
31
|
-
|
32
|
-
if robot.nil?
|
33
|
-
return
|
34
|
-
elsif @action.arity == 1
|
35
|
-
@action.call(robot)
|
36
|
-
else
|
37
|
-
@action.call(robot, params)
|
38
|
-
end
|
39
|
-
end
|
40
|
-
|
41
|
-
def self.register(name: nil)
|
42
|
-
if name.nil?
|
43
|
-
handler_path = caller_locations(1, 1)[0].absolute_path
|
44
|
-
name = File.basename(handler_path, '.*')
|
45
|
-
end
|
46
|
-
|
47
|
-
handler = Ego::Handler.new(name)
|
48
|
-
yield handler
|
49
|
-
|
50
|
-
@@handlers[handler.name] = handler
|
51
|
-
end
|
52
|
-
|
53
|
-
def self.has(handler_name)
|
54
|
-
@@handlers.has_key? handler_name
|
55
|
+
# Match the given query against the condition.
|
56
|
+
#
|
57
|
+
# @param query [String] the query to match the condition against
|
58
|
+
# @return [false] if condition doesn't match
|
59
|
+
# @return the return value of condition
|
60
|
+
def handle(query)
|
61
|
+
@condition.call(query) || false
|
55
62
|
end
|
56
63
|
|
57
|
-
|
58
|
-
handler_names.each do |path|
|
59
|
-
handler = File.basename(path, '.*')
|
60
|
-
require path unless has(handler)
|
61
|
-
end
|
62
|
-
end
|
64
|
+
protected
|
63
65
|
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
66
|
+
# Normalize the condition to a callable object.
|
67
|
+
#
|
68
|
+
# @param condition [Proc, #match] the condition that triggers the supplied action
|
69
|
+
def normalize(condition)
|
70
|
+
if condition.respond_to?(:match)
|
71
|
+
# Must assign regexp to avoid recursion in lambda
|
72
|
+
regexp = condition
|
73
|
+
condition = ->(query) { regexp.match(query) }
|
69
74
|
end
|
70
|
-
end
|
71
75
|
|
72
|
-
|
73
|
-
@@handlers
|
76
|
+
condition
|
74
77
|
end
|
75
78
|
end
|
76
79
|
end
|