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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: db83be1f5d3be9a990e8990b6f10b3fabd634939
4
- data.tar.gz: 2dce79d2d96a4bf49c7bcdd1c5605f79ba31df97
3
+ metadata.gz: 7a1b7a652d2e75a9e5a22d05a9627f1b5f63b94d
4
+ data.tar.gz: 5e7aa5063ddf5f986b341a9bb4ac6f9f165d3358
5
5
  SHA512:
6
- metadata.gz: 29423bd9409d40de8a4fe31e2e1893e3ffe7742b82f309cbc21df6b80f74082996d8ec62d5a69c67de6a07c50db8ed5dff4467187ca71ffedc74d2cb89ca534e
7
- data.tar.gz: 3741072a3a11271babcd8168db8cf49592f24cb3e415870f0bdf75e190132301fe7956caa3b1c3fcfec47fc36610cb566cf756ac6ee55c3129c294d42ab0e012
6
+ metadata.gz: 0711a3f6ddb2ef92913409ffb2e2a66a55b6dbec57f0fcf9057ab598803175254b26a40785996aab3d6ece9eae279f03fa55417167f057a0531d9d465e932116
7
+ data.tar.gz: b816f63c6fadea1d33066b65f8ca6067c2af1409e01e323353f57a58f99c3d28775dfc6d6d174c14cca138ab72fc63727b29709c1a93a578f62184d4ab1e0fa3
data/.travis.yml ADDED
@@ -0,0 +1,4 @@
1
+ language: ruby
2
+ rvm:
3
+ - 2.0
4
+ - 2.4
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 "handlers",
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. Here's what a handler looks like that responds to a query
32
- beginning with "hello...", "hi...", or "hey..." with its own random greeting:
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::Handler.register do |handler|
36
- handler.description = 'greet you'
36
+ Ego.plugin do |robot|
37
+ robot.can 'greet you'
37
38
 
38
- handler.listen /^(hello|hi|hey)/i, priority: 3
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
- handler.description = 'greet you'
55
+ robot.can 'greet you'
57
56
  ```
58
57
 
59
- The description is for your own reference, and answers the question "What can
60
- this handler do?"
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
- handler.listen /^(hello|hi|hey)/i, priority: 3
62
+ robot.on /^(hello|hi|hey)/i, 3 ...
64
63
  ```
65
64
 
66
- This is the listener, which specifies what queries should invoke the handler.
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 listeners match
70
- your query, you can optionally specify a `:priority` (higher number = higher
71
- priority). One handler can have as many listeners as you want, even ones with
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
- handler.run do |robot|
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 your handler matches the query. From here
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` is made available to you to respond to the user. Usually, you'll want
83
- to make use of part of the query in your handler. You can access named match
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::Handler.register do |handler|
88
- handler.description = 'repeat what you say'
89
-
90
- handler.listen /^say (?<input>.+)/i
89
+ Ego.plugin do |robot|
90
+ robot.can 'repeat what you say'
91
91
 
92
- handler.run do |robot, params|
93
- robot.respond params[:input]
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 handlers in `$XDG_CONFIG_HOME/ego/handlers/`
104
- (that's `~/.config/ego/handlers/` by default), and registers them
105
- automatically at runtime. Each handler goes in it's own file with an `.rb`
106
- extension (e.g., `~/.config/ego/handlers/my_handler.rb`). Be careful—ego will
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
@@ -31,4 +31,5 @@ Gem::Specification.new do |spec|
31
31
  spec.add_development_dependency "guard-rspec"
32
32
 
33
33
  spec.add_runtime_dependency "colorize", "~> 0.7"
34
+ spec.add_runtime_dependency "hooks", "~> 0.4"
34
35
  end
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/handler'
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
@@ -1,31 +1,60 @@
1
- module Ego::Filesystem
2
- HANDLER_GLOB = 'handler/*.rb'
3
-
4
- BASENAME = 'ego'
5
-
6
- XDG_CACHE_HOME = ENV['XDG_CACHE_HOME'] || File.expand_path('~/.cache')
7
- XDG_CONFIG_HOME = ENV['XDG_CONFIG_HOME'] || File.expand_path('~/.config')
8
- XDG_DATA_HOME = ENV['XDG_DATA_HOME'] || File.expand_path('~/.local/share')
9
-
10
- module_function
11
-
12
- def cache(path = '')
13
- File.join(XDG_CACHE_HOME, BASENAME, path)
14
- end
15
-
16
- def config(path = '')
17
- File.join(XDG_CONFIG_HOME, BASENAME, path)
18
- end
19
-
20
- def data(path = '')
21
- File.join(XDG_DATA_HOME, BASENAME, path)
22
- end
23
-
24
- def builtin_handlers
25
- Dir[File.expand_path(HANDLER_GLOB, __dir__)]
26
- end
27
-
28
- def user_handlers
29
- Dir[File.expand_path(HANDLER_GLOB, config)]
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 'listener'
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
- @@handlers = {}
7
- @@listeners = []
35
+ include Comparable
8
36
 
9
- attr_reader :name
10
- attr_accessor :description
37
+ attr_reader :condition, :action, :priority
11
38
 
12
- def initialize(name)
13
- @name = name
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
- def to_s
17
- "#{@description}"
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
- def listen(pattern, priority: 5, &parser)
21
- unless block_given?
22
- parser = Proc.new { |matches| matches }
23
- end
24
- @@listeners << Ego::Listener.new(pattern, priority, parser, @name)
25
- end
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
- def self.load(handler_names)
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
- def self.dispatch(robot, query)
65
- @@listeners.sort.reverse_each do |listener|
66
- if params = listener.match(query)
67
- return @@handlers[listener.handler].run(robot, params)
68
- end
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
- def self.handlers
73
- @@handlers
76
+ condition
74
77
  end
75
78
  end
76
79
  end