metaphor 0.2.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.
- data/LICENSE +23 -0
- data/README.markdown +92 -0
- data/lib/metaphor.rb +33 -0
- data/lib/metaphor/input/stdin_input.rb +34 -0
- data/lib/metaphor/processor/detour.rb +34 -0
- data/lib/metaphor/processor/json_processor.rb +18 -0
- data/lib/metaphor/processor/print_message.rb +18 -0
- data/lib/metaphor/processor/wiretap.rb +28 -0
- metadata +78 -0
data/LICENSE
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
The MIT License
|
2
|
+
|
3
|
+
Copyright (c) 2009 Sean O'Halpin
|
4
|
+
Copyright (c) 2010 Craig R Webster <http://barkingiguana.com/>
|
5
|
+
|
6
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
7
|
+
a copy of this software and associated documentation files (the
|
8
|
+
'Software'), to deal in the Software without restriction, including
|
9
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
10
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
11
|
+
permit persons to whom the Software is furnished to do so, subject to
|
12
|
+
the following conditions:
|
13
|
+
|
14
|
+
The above copyright notice and this permission notice shall be
|
15
|
+
included in all copies or substantial portions of the Software.
|
16
|
+
|
17
|
+
THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
|
18
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
19
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
20
|
+
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
21
|
+
CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
22
|
+
TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
23
|
+
SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.markdown
ADDED
@@ -0,0 +1,92 @@
|
|
1
|
+
Metaphor
|
2
|
+
========
|
3
|
+
|
4
|
+
From the Greek: μεταφορά - metaphora, meaning "transfer"
|
5
|
+
|
6
|
+
|
7
|
+
Rationale
|
8
|
+
---------
|
9
|
+
|
10
|
+
As programmers an awful lot of what we do revolves around taking an
|
11
|
+
input, doing something to it, and sending the result somewhere. Quite a
|
12
|
+
bit of time is wasted writing the "glue" that transfers the results from
|
13
|
+
one place to the next. Metaphor aims to provide a standardised and easy
|
14
|
+
to use way of passing results, leaving you to concentrate on the bit
|
15
|
+
that's important - the business logic.
|
16
|
+
|
17
|
+
|
18
|
+
Programming with Metaphor
|
19
|
+
-------------------------
|
20
|
+
|
21
|
+
metaphor = Metaphor.new
|
22
|
+
metaphor.processors << Foo.new
|
23
|
+
metaphor.processors << Bar.new
|
24
|
+
metaphor.processors << Baz.new
|
25
|
+
metaphor.processors << Metaphor::Processor::PrintMessage.new
|
26
|
+
|
27
|
+
# Process one message
|
28
|
+
metaphor.call(headers, body) # => [ new_headers, new_body ]
|
29
|
+
# => false (if halted by processor)
|
30
|
+
|
31
|
+
# Process messages from this class until the Ruby VM is killed or the
|
32
|
+
# input returns nil
|
33
|
+
metaphor.call(StdinInput.new)
|
34
|
+
|
35
|
+
|
36
|
+
Classes used for input must respond to #get and return an array of headers
|
37
|
+
and the message body:
|
38
|
+
|
39
|
+
class StdinInput
|
40
|
+
def get
|
41
|
+
# return an array like this:
|
42
|
+
[
|
43
|
+
{ "header" => "value", ... }, # Message headers
|
44
|
+
"body" # Message body
|
45
|
+
]
|
46
|
+
end
|
47
|
+
end
|
48
|
+
|
49
|
+
Classes used as processors must respond to #call(headers, body):
|
50
|
+
|
51
|
+
class PrintMessage
|
52
|
+
def call(headers, body)
|
53
|
+
puts "Headers: #{headers.inspect}"
|
54
|
+
puts "Body : #{body.inspect}"
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
The return value of #call controls what happens to the message. If the
|
59
|
+
processor returns:
|
60
|
+
|
61
|
+
* An array of headers and the message body
|
62
|
+
- they are passed to the next processor
|
63
|
+
* Boolean false (or something that is === to it)
|
64
|
+
- Metaphor stops processing this message and discards it
|
65
|
+
* Any other value that is not false
|
66
|
+
- Metaphor passes the same headers and message that were passed into
|
67
|
+
this processor to the next processor in the chain
|
68
|
+
|
69
|
+
Contributing
|
70
|
+
------------
|
71
|
+
|
72
|
+
* Fork the project.
|
73
|
+
* Make your feature addition or bug fix.
|
74
|
+
* Add tests for it. This is important so I don't break it in a
|
75
|
+
future version unintentionally.
|
76
|
+
* Commit, do not mess with the Rakefile or Metaphor::VERSION. If you
|
77
|
+
want to have your own version, that is fine but bump version in a
|
78
|
+
commit by itself I can ignore when I pull.
|
79
|
+
* Send me a pull request. Bonus points for topic branches.
|
80
|
+
|
81
|
+
|
82
|
+
Authors
|
83
|
+
-------
|
84
|
+
|
85
|
+
* Sean O'Halpin
|
86
|
+
* Craig R Webster <http://barkingiguana.com/>
|
87
|
+
|
88
|
+
|
89
|
+
License
|
90
|
+
-------
|
91
|
+
|
92
|
+
Released under the MIT licence. See the LICENSE file for details.
|
data/lib/metaphor.rb
ADDED
@@ -0,0 +1,33 @@
|
|
1
|
+
class Metaphor
|
2
|
+
VERSION = '0.2.1'
|
3
|
+
attr_accessor :processors
|
4
|
+
|
5
|
+
def initialize
|
6
|
+
self.processors = []
|
7
|
+
end
|
8
|
+
|
9
|
+
def call(*args)
|
10
|
+
case args.size
|
11
|
+
when 1
|
12
|
+
while message = args.first.get
|
13
|
+
process_message(*message)
|
14
|
+
end
|
15
|
+
when 2
|
16
|
+
process_message(*args)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
private
|
21
|
+
def process_message(headers, body)
|
22
|
+
processors.each do |processor|
|
23
|
+
processor_output = processor.call(headers, body)
|
24
|
+
case
|
25
|
+
when processor_output === false
|
26
|
+
return false
|
27
|
+
when processor_output.respond_to?(:size) && processor_output.size == 2
|
28
|
+
headers, body = processor_output
|
29
|
+
end
|
30
|
+
end
|
31
|
+
[ headers, body ]
|
32
|
+
end
|
33
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
class Metaphor
|
2
|
+
module Input
|
3
|
+
class StdinInput
|
4
|
+
def initialize(source = STDIN, prompt = STDOUT)
|
5
|
+
@source = source
|
6
|
+
@prompt = prompt
|
7
|
+
end
|
8
|
+
|
9
|
+
def get
|
10
|
+
@prompt.puts "Enter headers in the format header:value."
|
11
|
+
@prompt.puts "Each header:value pair should be followed by a newline."
|
12
|
+
@prompt.puts "When you're done enter a blank line."
|
13
|
+
headers = {}
|
14
|
+
loop do
|
15
|
+
line = @source.readline.to_s.strip
|
16
|
+
break if line == ""
|
17
|
+
header, value = line.split(/:/, 2).map{|s|s.strip}
|
18
|
+
headers[header] = value
|
19
|
+
end
|
20
|
+
|
21
|
+
@prompt.puts "Enter the message body."
|
22
|
+
@prompt.puts "When you're done enter a blank line."
|
23
|
+
body = []
|
24
|
+
loop do
|
25
|
+
line = @source.readline.to_s.strip
|
26
|
+
break if line == ""
|
27
|
+
body << line
|
28
|
+
end
|
29
|
+
|
30
|
+
[ headers, body.join("\n") ]
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
class Metaphor
|
2
|
+
module Processor
|
3
|
+
class Detour
|
4
|
+
def initialize(default, detour)
|
5
|
+
@default = default
|
6
|
+
@detour = Metaphor.new
|
7
|
+
@detour.processors << detour
|
8
|
+
@detour.processors << default
|
9
|
+
@active = false
|
10
|
+
end
|
11
|
+
|
12
|
+
def activate
|
13
|
+
@active = true
|
14
|
+
end
|
15
|
+
|
16
|
+
def deactivate
|
17
|
+
@active = false
|
18
|
+
end
|
19
|
+
|
20
|
+
def active?
|
21
|
+
@active
|
22
|
+
end
|
23
|
+
|
24
|
+
def call(headers, body)
|
25
|
+
active_processor.call headers, body
|
26
|
+
end
|
27
|
+
|
28
|
+
private
|
29
|
+
def active_processor
|
30
|
+
(@active ? @detour : @default)
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
class Metaphor
|
2
|
+
module Processor
|
3
|
+
class JsonProcessor
|
4
|
+
def initialize
|
5
|
+
require 'json'
|
6
|
+
end
|
7
|
+
|
8
|
+
def call(headers, body)
|
9
|
+
result = JSON[body]
|
10
|
+
headers['content-type'] = case result
|
11
|
+
when Hash: 'application/x-ruby'
|
12
|
+
when String: 'application/json'
|
13
|
+
end
|
14
|
+
[ headers, result ]
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
class Metaphor
|
2
|
+
module Processor
|
3
|
+
class PrintMessage
|
4
|
+
def initialize(target = STDOUT)
|
5
|
+
@target = target
|
6
|
+
end
|
7
|
+
|
8
|
+
def call(headers, body)
|
9
|
+
headers.each_pair do |header, value|
|
10
|
+
@target.puts "#{header}:#{value}"
|
11
|
+
end
|
12
|
+
@target.puts
|
13
|
+
@target.puts body
|
14
|
+
@target.puts
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
class Metaphor
|
2
|
+
module Processor
|
3
|
+
class Wiretap
|
4
|
+
def initialize(default, wiretap)
|
5
|
+
@default = default
|
6
|
+
@wiretap = wiretap
|
7
|
+
@active = false
|
8
|
+
end
|
9
|
+
|
10
|
+
def active?
|
11
|
+
@active
|
12
|
+
end
|
13
|
+
|
14
|
+
def activate
|
15
|
+
@active = true
|
16
|
+
end
|
17
|
+
|
18
|
+
def deactivate
|
19
|
+
@active = false
|
20
|
+
end
|
21
|
+
|
22
|
+
def call(headers, body)
|
23
|
+
@wiretap.call(headers, body) if active?
|
24
|
+
@default.call(headers, body)
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
metadata
ADDED
@@ -0,0 +1,78 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: metaphor
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
hash: 21
|
5
|
+
prerelease: false
|
6
|
+
segments:
|
7
|
+
- 0
|
8
|
+
- 2
|
9
|
+
- 1
|
10
|
+
version: 0.2.1
|
11
|
+
platform: ruby
|
12
|
+
authors:
|
13
|
+
- Sean O'Halpin
|
14
|
+
- Craig R Webster
|
15
|
+
autorequire:
|
16
|
+
bindir: bin
|
17
|
+
cert_chain: []
|
18
|
+
|
19
|
+
date: 2010-08-20 00:00:00 +01:00
|
20
|
+
default_executable:
|
21
|
+
dependencies: []
|
22
|
+
|
23
|
+
description: Metaphor provides a standard interface for defining message processors and transformations and a simple framework for executing them.
|
24
|
+
email:
|
25
|
+
- craig@barkingiguana.com
|
26
|
+
executables: []
|
27
|
+
|
28
|
+
extensions: []
|
29
|
+
|
30
|
+
extra_rdoc_files: []
|
31
|
+
|
32
|
+
files:
|
33
|
+
- lib/metaphor/input/stdin_input.rb
|
34
|
+
- lib/metaphor/processor/detour.rb
|
35
|
+
- lib/metaphor/processor/json_processor.rb
|
36
|
+
- lib/metaphor/processor/print_message.rb
|
37
|
+
- lib/metaphor/processor/wiretap.rb
|
38
|
+
- lib/metaphor.rb
|
39
|
+
- LICENSE
|
40
|
+
- README.markdown
|
41
|
+
has_rdoc: true
|
42
|
+
homepage: http://github.com/seanohalpin/metaphor
|
43
|
+
licenses: []
|
44
|
+
|
45
|
+
post_install_message:
|
46
|
+
rdoc_options: []
|
47
|
+
|
48
|
+
require_paths:
|
49
|
+
- lib
|
50
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
51
|
+
none: false
|
52
|
+
requirements:
|
53
|
+
- - ">="
|
54
|
+
- !ruby/object:Gem::Version
|
55
|
+
hash: 3
|
56
|
+
segments:
|
57
|
+
- 0
|
58
|
+
version: "0"
|
59
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
60
|
+
none: false
|
61
|
+
requirements:
|
62
|
+
- - ">="
|
63
|
+
- !ruby/object:Gem::Version
|
64
|
+
hash: 23
|
65
|
+
segments:
|
66
|
+
- 1
|
67
|
+
- 3
|
68
|
+
- 6
|
69
|
+
version: 1.3.6
|
70
|
+
requirements: []
|
71
|
+
|
72
|
+
rubyforge_project: metaphor
|
73
|
+
rubygems_version: 1.3.7
|
74
|
+
signing_key:
|
75
|
+
specification_version: 3
|
76
|
+
summary: Generic pipeline processing
|
77
|
+
test_files: []
|
78
|
+
|