fluent-plugin-event-tail 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +17 -0
- data/Gemfile +4 -0
- data/LICENSE.txt +22 -0
- data/README.md +123 -0
- data/Rakefile +7 -0
- data/fluent-plugin-event-tail.gemspec +20 -0
- data/lib/fluent/plugin/in_event_tail.rb +102 -0
- data/test/fluent/plugin/in_event_tail.rb +131 -0
- metadata +87 -0
data/.gitignore
ADDED
data/Gemfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
Copyright (c) 2013 Mario Freitas (imkira@gmail.com)
|
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
@@ -0,0 +1,123 @@
|
|
1
|
+
# fluent-plugin-event-tail
|
2
|
+
|
3
|
+
event-tail is an input plugin for [fluentd](http://fluentd.org) based on
|
4
|
+
[in_tail](http://docs.fluentd.org/articles/in_tail) but for reading
|
5
|
+
[tag, time, record] JSON messages from a file.
|
6
|
+
|
7
|
+
If you use ```fluent-logger-ruby```, ```fluent-logger-node```, and so on, you
|
8
|
+
are probably connecting to 'localhost' (or worse, to a remote host) and sending
|
9
|
+
messages via TCP sockets.
|
10
|
+
|
11
|
+
That's all fine but what happens if fluentd is down for some reason?
|
12
|
+
Well, most (if not all) logger implementations keep a list of pending failed
|
13
|
+
messages and try sending them periodically. But what if your application dies
|
14
|
+
before having the chance to flush everything to fluentd? Well, you will
|
15
|
+
probably lose those pending messages.
|
16
|
+
|
17
|
+
The reason I made this plugin is to allow me (and hopefully you too) to rather
|
18
|
+
send those messages, not via TCP sockets, but directly to a file and have
|
19
|
+
fluentd read them. The obvious advantage is that you always have backups of
|
20
|
+
your logs in your hard disks in case fluentd is not running or the DB where you
|
21
|
+
are aggregating them died for some reason. Another reason is, your logs are
|
22
|
+
still human readable and easily transformable (they are just newline separated
|
23
|
+
JSON strings).
|
24
|
+
|
25
|
+
The format of the messages is based on
|
26
|
+
[in_forward](http://docs.fluentd.org/articles/in_forward) plugin:
|
27
|
+
|
28
|
+
```
|
29
|
+
stream:
|
30
|
+
message...
|
31
|
+
|
32
|
+
message:
|
33
|
+
[tag, time, record]
|
34
|
+
or
|
35
|
+
[tag, [[time,record], [time,record], ...]]
|
36
|
+
|
37
|
+
example:
|
38
|
+
["myapp.access", [1308466941, {"a"=>1}], [1308466942, {"b"=>2}]]
|
39
|
+
```
|
40
|
+
|
41
|
+
This plugin is mostly based on
|
42
|
+
[in_tail](http://docs.fluentd.org/articles/in_tail),
|
43
|
+
and therefore you are expected to append newline separated JSON strings of
|
44
|
+
messages in the above format. Also note that this plugin supports formatted
|
45
|
+
time strings via the ```time_format``` config parameter, not just numeric UNIX
|
46
|
+
timestamps.
|
47
|
+
|
48
|
+
## Installation
|
49
|
+
|
50
|
+
Add this line to your application's Gemfile:
|
51
|
+
|
52
|
+
gem 'fluent-plugin-event-tail'
|
53
|
+
|
54
|
+
And then execute:
|
55
|
+
|
56
|
+
$ bundle
|
57
|
+
|
58
|
+
Or install it yourself as:
|
59
|
+
|
60
|
+
$ gem install fluent-plugin-event-tail
|
61
|
+
|
62
|
+
## Configuration
|
63
|
+
|
64
|
+
This plugin has the same configuration as
|
65
|
+
[in_tail](http://docs.fluentd.org/articles/in_tail),
|
66
|
+
with the exception of ```tag``` and ```format``` that were removed.
|
67
|
+
|
68
|
+
```
|
69
|
+
# fluent.conf
|
70
|
+
<match prefix.**>
|
71
|
+
type event_tail
|
72
|
+
path /var/log/your_app.log
|
73
|
+
pos_file /var/log/your_app.log.pos
|
74
|
+
# disable next line to enable custom time formatting
|
75
|
+
# time_format %d %b %Y %H:%M:%S
|
76
|
+
</match>
|
77
|
+
```
|
78
|
+
|
79
|
+
## Usage
|
80
|
+
|
81
|
+
After running fluentd, you can emit events by appending to the file:
|
82
|
+
|
83
|
+
```
|
84
|
+
# time 0 means "current time on server"
|
85
|
+
echo '["prefix.debug",0,{"foo":"bar"}]' >> /var/log/your_app.log
|
86
|
+
# time passed as UNIX timestamp (2013-02-20 01:18:31 +0900)
|
87
|
+
echo '["prefix.debug1",1361290711,{"foo1":"bar1"}]' >> /var/log/your_app.log
|
88
|
+
# same time but passed as string
|
89
|
+
echo '["prefix.debug2","2009-03-26 22:33:12 +0900",{"foo2":"bar2"}]' >> /var/log/your_app.log
|
90
|
+
# group multiple events in one line by tag
|
91
|
+
echo '["prefix.debug3",[[1361290714,{"foo3":"bar3"}],[1361290716,{"foo4":"bar4"}]]]' >> /var/log/your_app.log
|
92
|
+
```
|
93
|
+
|
94
|
+
fluentd will report something like:
|
95
|
+
```
|
96
|
+
prefix.debug: {"foo":"bar"}
|
97
|
+
prefix.debug1: {"foo1":"bar1"}
|
98
|
+
prefix.debug2: {"foo2":"bar2"}
|
99
|
+
prefix.debug3: {"foo3":"bar3"}
|
100
|
+
prefix.debug3: {"foo4":"bar4"}
|
101
|
+
````
|
102
|
+
|
103
|
+
Please note that contrarily to ```in_forward``` that only accepts UNIX
|
104
|
+
timestamps, fluent-plugin-event-tail supports the original in_tail
|
105
|
+
```time_format``` parameter, so you can also pass strings as event times.
|
106
|
+
The default value of ```time_format``` is ```'%Y-%m-%d %H:%M:%S %z```.
|
107
|
+
If you pass a numeric UNIX timestamp then ```time_format``` will be ignored.
|
108
|
+
|
109
|
+
## Contributing
|
110
|
+
|
111
|
+
You are very welcome to submit patches or improve this plugin.
|
112
|
+
Just make sure you send me a pull request.
|
113
|
+
|
114
|
+
## License
|
115
|
+
|
116
|
+
fluent-plugin-event-tail is licensed under the MIT license:
|
117
|
+
|
118
|
+
www.opensource.org/licenses/MIT
|
119
|
+
|
120
|
+
## Copyright
|
121
|
+
|
122
|
+
Copyright (c) 2013 Mario Freitas. See
|
123
|
+
[LICENSE.txt](http://github.com/imkira/fluent-plugin-event-tail/blob/master/LICENSE.txt) for further details.
|
data/Rakefile
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
|
5
|
+
Gem::Specification.new do |gem|
|
6
|
+
gem.name = "fluent-plugin-event-tail"
|
7
|
+
gem.version = "0.0.1"
|
8
|
+
gem.authors = ["Mario Freitas"]
|
9
|
+
gem.email = ["imkira@gmail.com"]
|
10
|
+
gem.description = %q{fluentd input plugin derived from in_tail and inspired by in_forward for reading [tag, time, record] messages from a file}
|
11
|
+
gem.summary = %q{fluentd input plugin for reading [tag, time, record] messages from a file}
|
12
|
+
gem.homepage = "https://github.com/imkira/fluent-plugin-event-tail"
|
13
|
+
|
14
|
+
gem.files = `git ls-files`.split($/)
|
15
|
+
gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
|
16
|
+
gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
|
17
|
+
gem.require_paths = ["lib"]
|
18
|
+
gem.add_development_dependency "fluentd"
|
19
|
+
gem.add_runtime_dependency "fluentd"
|
20
|
+
end
|
@@ -0,0 +1,102 @@
|
|
1
|
+
|
2
|
+
# -*- encoding: utf-8 -*-
|
3
|
+
|
4
|
+
# Copyright (c) 2013 Mario Freitas (imkira@gmail.com)
|
5
|
+
#
|
6
|
+
# MIT License
|
7
|
+
#
|
8
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
9
|
+
# a copy of this software and associated documentation files (the
|
10
|
+
# "Software"), to deal in the Software without restriction, including
|
11
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
12
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
13
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
14
|
+
# the following conditions:
|
15
|
+
#
|
16
|
+
# The above copyright notice and this permission notice shall be
|
17
|
+
# included in all copies or substantial portions of the Software.
|
18
|
+
#
|
19
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
20
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
21
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
22
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
23
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
24
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
25
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
26
|
+
|
27
|
+
require 'fluent/plugin/in_tail'
|
28
|
+
|
29
|
+
module Fluent
|
30
|
+
class EventTailInput < Fluent::TailInput
|
31
|
+
Fluent::Plugin.register_input('event_tail', self)
|
32
|
+
|
33
|
+
config_param :time_format, :string, :default => '%Y-%m-%d %H:%M:%S %z'
|
34
|
+
|
35
|
+
# don't need tag parameter
|
36
|
+
config_set_default(:tag, '')
|
37
|
+
|
38
|
+
# don't need format parameter
|
39
|
+
config_set_default(:format, '')
|
40
|
+
|
41
|
+
def configure_parser(conf)
|
42
|
+
# just disable the default parser
|
43
|
+
end
|
44
|
+
|
45
|
+
def receive_lines(lines)
|
46
|
+
es = MultiEventStream.new
|
47
|
+
tag = nil
|
48
|
+
lines.each do |line|
|
49
|
+
begin
|
50
|
+
line.chomp!
|
51
|
+
tag = parse_line(line) do |time, record|
|
52
|
+
es.add(time, record)
|
53
|
+
end
|
54
|
+
rescue
|
55
|
+
$log.warn line.dump, :error=>$!.to_s
|
56
|
+
$log.debug_backtrace
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
unless tag.nil? || es.empty?
|
61
|
+
begin
|
62
|
+
Engine.emit_stream(tag, es)
|
63
|
+
rescue => e
|
64
|
+
# ignore errors. Engine shows logs and backtraces.
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|
68
|
+
|
69
|
+
def parse_line(line, &block)
|
70
|
+
msg = Yajl.load(line)
|
71
|
+
tag = msg[0].to_s
|
72
|
+
entries = msg[1]
|
73
|
+
|
74
|
+
# [tag, [[time,record], [time,record], ...]]
|
75
|
+
if entries.is_a? Array
|
76
|
+
entries.each do |e|
|
77
|
+
time = parse_time(e[0])
|
78
|
+
record = e[1]
|
79
|
+
block.call(time, record)
|
80
|
+
end
|
81
|
+
|
82
|
+
# [tag, time, record]
|
83
|
+
else
|
84
|
+
time = parse_time(msg[1])
|
85
|
+
record = msg[2]
|
86
|
+
block.call(time, record)
|
87
|
+
end
|
88
|
+
|
89
|
+
tag
|
90
|
+
end
|
91
|
+
|
92
|
+
def parse_time(time)
|
93
|
+
if !@time_format.nil? and time.is_a? String
|
94
|
+
Time.strptime(time, @time_format).to_i
|
95
|
+
else
|
96
|
+
time = time.to_i
|
97
|
+
time = Engine.now if time == 0
|
98
|
+
time
|
99
|
+
end
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
@@ -0,0 +1,131 @@
|
|
1
|
+
require 'fluent/test'
|
2
|
+
require 'fluent/plugin/in_event_tail'
|
3
|
+
|
4
|
+
class EventTailInputTest < Test::Unit::TestCase
|
5
|
+
def setup
|
6
|
+
Fluent::Test.setup
|
7
|
+
FileUtils.rm_rf(TMP_DIR)
|
8
|
+
FileUtils.mkdir_p(TMP_DIR)
|
9
|
+
end
|
10
|
+
|
11
|
+
TMP_DIR = File.dirname(__FILE__) + "/../tmp"
|
12
|
+
|
13
|
+
CONFIG = %[
|
14
|
+
path #{TMP_DIR}/tail.log
|
15
|
+
tag t1
|
16
|
+
rotate_wait 2s
|
17
|
+
pos_file #{TMP_DIR}/tail.pos
|
18
|
+
]
|
19
|
+
|
20
|
+
def create_driver(conf = CONFIG)
|
21
|
+
Fluent::Test::InputTestDriver.new(Fluent::EventTailInput).configure(conf)
|
22
|
+
end
|
23
|
+
|
24
|
+
def test_configure
|
25
|
+
d = create_driver
|
26
|
+
assert_equal ["#{TMP_DIR}/tail.log"], d.instance.paths
|
27
|
+
assert_equal "t1", d.instance.tag
|
28
|
+
assert_equal 2, d.instance.rotate_wait
|
29
|
+
assert_equal "#{TMP_DIR}/tail.pos", d.instance.pos_file
|
30
|
+
end
|
31
|
+
|
32
|
+
def test_simple_emit
|
33
|
+
File.open("#{TMP_DIR}/tail.log", "w") {|f|
|
34
|
+
f.puts '["foo",123,{"bar":"hoge"}]'
|
35
|
+
}
|
36
|
+
|
37
|
+
d = create_driver
|
38
|
+
|
39
|
+
d.run do
|
40
|
+
sleep 1
|
41
|
+
|
42
|
+
File.open("#{TMP_DIR}/tail.log", "a") {|f|
|
43
|
+
f.puts '["foo3",789,{"bar3":"hoge3"}]'
|
44
|
+
}
|
45
|
+
sleep 1
|
46
|
+
end
|
47
|
+
|
48
|
+
emits = d.emits
|
49
|
+
assert_equal(emits.length, 1)
|
50
|
+
assert_equal("foo3", emits[0][0])
|
51
|
+
assert_equal(789, emits[0][1])
|
52
|
+
assert_equal({"bar3"=>"hoge3"}, emits[0][2])
|
53
|
+
end
|
54
|
+
|
55
|
+
def test_default_time_format
|
56
|
+
File.open("#{TMP_DIR}/tail.log", "w") {|f|
|
57
|
+
f.puts '["foo","10/Oct/2010:20:57:59 -0700",{"bar":"hoge"}]'
|
58
|
+
}
|
59
|
+
|
60
|
+
d = create_driver
|
61
|
+
|
62
|
+
d.run do
|
63
|
+
sleep 1
|
64
|
+
|
65
|
+
File.open("#{TMP_DIR}/tail.log", "a") {|f|
|
66
|
+
f.puts '["foo4","2012-10-22 11:57:59 -0100",{"bar4":"hoge4"}]'
|
67
|
+
}
|
68
|
+
sleep 1
|
69
|
+
end
|
70
|
+
|
71
|
+
emits = d.emits
|
72
|
+
assert_equal(emits.length, 1)
|
73
|
+
assert_equal("foo4", emits[0][0])
|
74
|
+
assert_equal(1350910679, emits[0][1])
|
75
|
+
assert_equal({"bar4"=>"hoge4"}, emits[0][2])
|
76
|
+
end
|
77
|
+
|
78
|
+
def test_composed_emit
|
79
|
+
File.open("#{TMP_DIR}/tail.log", "w") {|f|
|
80
|
+
f.puts '["foo",123,{"bar":"hoge"}]'
|
81
|
+
}
|
82
|
+
|
83
|
+
d = create_driver
|
84
|
+
|
85
|
+
d.run do
|
86
|
+
sleep 1
|
87
|
+
|
88
|
+
File.open("#{TMP_DIR}/tail.log", "a") {|f|
|
89
|
+
f.puts '["foo5",[[91011,{"bar5":"hoge5"}],' +
|
90
|
+
'["2011-10-24 12:30:20 -0400",{"bar6":"hoge6"}]]]'
|
91
|
+
}
|
92
|
+
sleep 1
|
93
|
+
end
|
94
|
+
|
95
|
+
emits = d.emits
|
96
|
+
assert_equal(emits.length, 2)
|
97
|
+
assert_equal("foo5", emits[0][0])
|
98
|
+
assert_equal(91011, emits[0][1])
|
99
|
+
assert_equal({"bar5"=>"hoge5"}, emits[0][2])
|
100
|
+
assert_equal("foo5", emits[1][0])
|
101
|
+
assert_equal(1319473820, emits[1][1])
|
102
|
+
assert_equal({"bar6"=>"hoge6"}, emits[1][2])
|
103
|
+
end
|
104
|
+
|
105
|
+
def test_custom_time_format
|
106
|
+
File.open("#{TMP_DIR}/tail.log", "w") {|f|
|
107
|
+
f.puts '["foo",123,{"bar":"hoge"}]'
|
108
|
+
}
|
109
|
+
|
110
|
+
conf = CONFIG + %[
|
111
|
+
time_format %d %b %Y %H:%M:%S
|
112
|
+
]
|
113
|
+
|
114
|
+
d = create_driver(conf)
|
115
|
+
|
116
|
+
d.run do
|
117
|
+
sleep 1
|
118
|
+
|
119
|
+
File.open("#{TMP_DIR}/tail.log", "a") {|f|
|
120
|
+
f.puts '["foo7","6 Dec 2001 12:33:45",{"bar7":"hoge7"}]'
|
121
|
+
}
|
122
|
+
sleep 1
|
123
|
+
end
|
124
|
+
|
125
|
+
emits = d.emits
|
126
|
+
assert_equal(emits.length, 1)
|
127
|
+
assert_equal("foo7", emits[0][0])
|
128
|
+
assert_equal(1007609625, emits[0][1])
|
129
|
+
assert_equal({"bar7"=>"hoge7"}, emits[0][2])
|
130
|
+
end
|
131
|
+
end
|
metadata
ADDED
@@ -0,0 +1,87 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: fluent-plugin-event-tail
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
prerelease:
|
6
|
+
platform: ruby
|
7
|
+
authors:
|
8
|
+
- Mario Freitas
|
9
|
+
autorequire:
|
10
|
+
bindir: bin
|
11
|
+
cert_chain: []
|
12
|
+
date: 2013-02-19 00:00:00.000000000 Z
|
13
|
+
dependencies:
|
14
|
+
- !ruby/object:Gem::Dependency
|
15
|
+
name: fluentd
|
16
|
+
requirement: !ruby/object:Gem::Requirement
|
17
|
+
none: false
|
18
|
+
requirements:
|
19
|
+
- - ! '>='
|
20
|
+
- !ruby/object:Gem::Version
|
21
|
+
version: '0'
|
22
|
+
type: :development
|
23
|
+
prerelease: false
|
24
|
+
version_requirements: !ruby/object:Gem::Requirement
|
25
|
+
none: false
|
26
|
+
requirements:
|
27
|
+
- - ! '>='
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: '0'
|
30
|
+
- !ruby/object:Gem::Dependency
|
31
|
+
name: fluentd
|
32
|
+
requirement: !ruby/object:Gem::Requirement
|
33
|
+
none: false
|
34
|
+
requirements:
|
35
|
+
- - ! '>='
|
36
|
+
- !ruby/object:Gem::Version
|
37
|
+
version: '0'
|
38
|
+
type: :runtime
|
39
|
+
prerelease: false
|
40
|
+
version_requirements: !ruby/object:Gem::Requirement
|
41
|
+
none: false
|
42
|
+
requirements:
|
43
|
+
- - ! '>='
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: '0'
|
46
|
+
description: fluentd input plugin derived from in_tail and inspired by in_forward
|
47
|
+
for reading [tag, time, record] messages from a file
|
48
|
+
email:
|
49
|
+
- imkira@gmail.com
|
50
|
+
executables: []
|
51
|
+
extensions: []
|
52
|
+
extra_rdoc_files: []
|
53
|
+
files:
|
54
|
+
- .gitignore
|
55
|
+
- Gemfile
|
56
|
+
- LICENSE.txt
|
57
|
+
- README.md
|
58
|
+
- Rakefile
|
59
|
+
- fluent-plugin-event-tail.gemspec
|
60
|
+
- lib/fluent/plugin/in_event_tail.rb
|
61
|
+
- test/fluent/plugin/in_event_tail.rb
|
62
|
+
homepage: https://github.com/imkira/fluent-plugin-event-tail
|
63
|
+
licenses: []
|
64
|
+
post_install_message:
|
65
|
+
rdoc_options: []
|
66
|
+
require_paths:
|
67
|
+
- lib
|
68
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
69
|
+
none: false
|
70
|
+
requirements:
|
71
|
+
- - ! '>='
|
72
|
+
- !ruby/object:Gem::Version
|
73
|
+
version: '0'
|
74
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
75
|
+
none: false
|
76
|
+
requirements:
|
77
|
+
- - ! '>='
|
78
|
+
- !ruby/object:Gem::Version
|
79
|
+
version: '0'
|
80
|
+
requirements: []
|
81
|
+
rubyforge_project:
|
82
|
+
rubygems_version: 1.8.24
|
83
|
+
signing_key:
|
84
|
+
specification_version: 3
|
85
|
+
summary: fluentd input plugin for reading [tag, time, record] messages from a file
|
86
|
+
test_files:
|
87
|
+
- test/fluent/plugin/in_event_tail.rb
|