logstash-filter-elasticsearch 0.1.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 +15 -0
 - data/.gitignore +4 -0
 - data/Gemfile +3 -0
 - data/LICENSE +13 -0
 - data/Rakefile +6 -0
 - data/lib/logstash/filters/elasticsearch.rb +74 -0
 - data/logstash-filter-elasticsearch.gemspec +27 -0
 - data/rakelib/publish.rake +9 -0
 - data/rakelib/vendor.rake +169 -0
 - data/spec/filters/elasticsearch_spec.rb +1 -0
 - metadata +91 -0
 
    
        checksums.yaml
    ADDED
    
    | 
         @@ -0,0 +1,15 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            ---
         
     | 
| 
      
 2 
     | 
    
         
            +
            !binary "U0hBMQ==":
         
     | 
| 
      
 3 
     | 
    
         
            +
              metadata.gz: !binary |-
         
     | 
| 
      
 4 
     | 
    
         
            +
                YWFiNDI0NzAxYzY1MGI0NWQxZDU2M2E2MjI3NjgyMDY0ZGQwZDk3Zg==
         
     | 
| 
      
 5 
     | 
    
         
            +
              data.tar.gz: !binary |-
         
     | 
| 
      
 6 
     | 
    
         
            +
                MGViMTAzMWFhOWQxNTFjYTU1NmIyNmNiYjg3OTI1MmZhYTkxOTM3Ng==
         
     | 
| 
      
 7 
     | 
    
         
            +
            SHA512:
         
     | 
| 
      
 8 
     | 
    
         
            +
              metadata.gz: !binary |-
         
     | 
| 
      
 9 
     | 
    
         
            +
                ZmZmODQxNTk5MzM3ZjE2ZTgxZjExMTI3NDQwODRmODlkZGFjZmYxMTFkNGQy
         
     | 
| 
      
 10 
     | 
    
         
            +
                NzMwYzU5YmQ2ZWIzMTFkM2VjMzg5MDA0OGQyM2UwZDIwOGUwYTNjZDI4ZTZl
         
     | 
| 
      
 11 
     | 
    
         
            +
                MThiNTI3MTg3Yzc3ZDYyZTc0MTRlNGViODgyZTkwYjY0NGQyM2E=
         
     | 
| 
      
 12 
     | 
    
         
            +
              data.tar.gz: !binary |-
         
     | 
| 
      
 13 
     | 
    
         
            +
                NzEzMjMxZWI5MjdhZDBiNTVhZGI1OGI0MTcyNjRkNzBiOGQyMTc5MWVjMzZj
         
     | 
| 
      
 14 
     | 
    
         
            +
                ZTI1NTc0OGRiNDg2YTExMjBlZjg5MTY2ZjEyNjA0YTA2ZTBkY2I5N2ZkZmNj
         
     | 
| 
      
 15 
     | 
    
         
            +
                MzdlNGNlMDNjMzA5MTVhODE1OGI5ODc3N2E4ODMzNjllYmI3ZjQ=
         
     | 
    
        data/.gitignore
    ADDED
    
    
    
        data/Gemfile
    ADDED
    
    
    
        data/LICENSE
    ADDED
    
    | 
         @@ -0,0 +1,13 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            Copyright (c) 2012-2014 Elasticsearch <http://www.elasticsearch.org>
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            Licensed under the Apache License, Version 2.0 (the "License");
         
     | 
| 
      
 4 
     | 
    
         
            +
            you may not use this file except in compliance with the License.
         
     | 
| 
      
 5 
     | 
    
         
            +
            You may obtain a copy of the License at
         
     | 
| 
      
 6 
     | 
    
         
            +
             
     | 
| 
      
 7 
     | 
    
         
            +
                http://www.apache.org/licenses/LICENSE-2.0
         
     | 
| 
      
 8 
     | 
    
         
            +
             
     | 
| 
      
 9 
     | 
    
         
            +
            Unless required by applicable law or agreed to in writing, software
         
     | 
| 
      
 10 
     | 
    
         
            +
            distributed under the License is distributed on an "AS IS" BASIS,
         
     | 
| 
      
 11 
     | 
    
         
            +
            WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
         
     | 
| 
      
 12 
     | 
    
         
            +
            See the License for the specific language governing permissions and
         
     | 
| 
      
 13 
     | 
    
         
            +
            limitations under the License.
         
     | 
    
        data/Rakefile
    ADDED
    
    
| 
         @@ -0,0 +1,74 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            require "logstash/filters/base"
         
     | 
| 
      
 2 
     | 
    
         
            +
            require "logstash/namespace"
         
     | 
| 
      
 3 
     | 
    
         
            +
            require "logstash/util/fieldreference"
         
     | 
| 
      
 4 
     | 
    
         
            +
             
     | 
| 
      
 5 
     | 
    
         
            +
             
     | 
| 
      
 6 
     | 
    
         
            +
            # Search elasticsearch for a previous log event and copy some fields from it
         
     | 
| 
      
 7 
     | 
    
         
            +
            # into the current event.  Below is a complete example of how this filter might
         
     | 
| 
      
 8 
     | 
    
         
            +
            # be used.  Whenever logstash receives an "end" event, it uses this elasticsearch
         
     | 
| 
      
 9 
     | 
    
         
            +
            # filter to find the matching "start" event based on some operation identifier.
         
     | 
| 
      
 10 
     | 
    
         
            +
            # Then it copies the @timestamp field from the "start" event into a new field on
         
     | 
| 
      
 11 
     | 
    
         
            +
            # the "end" event.  Finally, using a combination of the "date" filter and the
         
     | 
| 
      
 12 
     | 
    
         
            +
            # "ruby" filter, we calculate the time duration in hours between the two events.
         
     | 
| 
      
 13 
     | 
    
         
            +
            #
         
     | 
| 
      
 14 
     | 
    
         
            +
            #       if [type] == "end" {
         
     | 
| 
      
 15 
     | 
    
         
            +
            #          elasticsearch {
         
     | 
| 
      
 16 
     | 
    
         
            +
            #             hosts => ["es-server"]
         
     | 
| 
      
 17 
     | 
    
         
            +
            #             query => "type:start AND operation:%{[opid]}"
         
     | 
| 
      
 18 
     | 
    
         
            +
            #             fields => ["@timestamp", "started"]
         
     | 
| 
      
 19 
     | 
    
         
            +
            #          }
         
     | 
| 
      
 20 
     | 
    
         
            +
            #
         
     | 
| 
      
 21 
     | 
    
         
            +
            #          date {
         
     | 
| 
      
 22 
     | 
    
         
            +
            #             match => ["[started]", "ISO8601"]
         
     | 
| 
      
 23 
     | 
    
         
            +
            #             target => "[started]"
         
     | 
| 
      
 24 
     | 
    
         
            +
            #          }
         
     | 
| 
      
 25 
     | 
    
         
            +
            #
         
     | 
| 
      
 26 
     | 
    
         
            +
            #          ruby {
         
     | 
| 
      
 27 
     | 
    
         
            +
            #             code => "event['duration_hrs'] = (event['@timestamp'] - event['started']) / 3600 rescue nil"
         
     | 
| 
      
 28 
     | 
    
         
            +
            #          }
         
     | 
| 
      
 29 
     | 
    
         
            +
            #       }
         
     | 
| 
      
 30 
     | 
    
         
            +
            #
         
     | 
| 
      
 31 
     | 
    
         
            +
            class LogStash::Filters::Elasticsearch < LogStash::Filters::Base
         
     | 
| 
      
 32 
     | 
    
         
            +
              config_name "elasticsearch"
         
     | 
| 
      
 33 
     | 
    
         
            +
              milestone 1
         
     | 
| 
      
 34 
     | 
    
         
            +
             
     | 
| 
      
 35 
     | 
    
         
            +
              # List of elasticsearch hosts to use for querying.
         
     | 
| 
      
 36 
     | 
    
         
            +
              config :hosts, :validate => :array
         
     | 
| 
      
 37 
     | 
    
         
            +
             
     | 
| 
      
 38 
     | 
    
         
            +
              # Elasticsearch query string
         
     | 
| 
      
 39 
     | 
    
         
            +
              config :query, :validate => :string
         
     | 
| 
      
 40 
     | 
    
         
            +
             
     | 
| 
      
 41 
     | 
    
         
            +
              # Comma-delimited list of <field>:<direction> pairs that define the sort order
         
     | 
| 
      
 42 
     | 
    
         
            +
              config :sort, :validate => :string, :default => "@timestamp:desc"
         
     | 
| 
      
 43 
     | 
    
         
            +
             
     | 
| 
      
 44 
     | 
    
         
            +
              # Hash of fields to copy from old event (found via elasticsearch) into new event
         
     | 
| 
      
 45 
     | 
    
         
            +
              config :fields, :validate => :hash, :default => {}
         
     | 
| 
      
 46 
     | 
    
         
            +
             
     | 
| 
      
 47 
     | 
    
         
            +
              public
         
     | 
| 
      
 48 
     | 
    
         
            +
              def register
         
     | 
| 
      
 49 
     | 
    
         
            +
                require "elasticsearch"
         
     | 
| 
      
 50 
     | 
    
         
            +
             
     | 
| 
      
 51 
     | 
    
         
            +
                @logger.info("New ElasticSearch filter", :hosts => @hosts)
         
     | 
| 
      
 52 
     | 
    
         
            +
                @client = Elasticsearch::Client.new hosts: @hosts
         
     | 
| 
      
 53 
     | 
    
         
            +
              end # def register
         
     | 
| 
      
 54 
     | 
    
         
            +
             
     | 
| 
      
 55 
     | 
    
         
            +
              public
         
     | 
| 
      
 56 
     | 
    
         
            +
              def filter(event)
         
     | 
| 
      
 57 
     | 
    
         
            +
                return unless filter?(event)
         
     | 
| 
      
 58 
     | 
    
         
            +
             
     | 
| 
      
 59 
     | 
    
         
            +
                begin
         
     | 
| 
      
 60 
     | 
    
         
            +
                  query_str = event.sprintf(@query)
         
     | 
| 
      
 61 
     | 
    
         
            +
             
     | 
| 
      
 62 
     | 
    
         
            +
                  results = @client.search q: query_str, sort: @sort, size: 1
         
     | 
| 
      
 63 
     | 
    
         
            +
             
     | 
| 
      
 64 
     | 
    
         
            +
                  @fields.each do |old, new|
         
     | 
| 
      
 65 
     | 
    
         
            +
                    event[new] = results['hits']['hits'][0]['_source'][old]
         
     | 
| 
      
 66 
     | 
    
         
            +
                  end
         
     | 
| 
      
 67 
     | 
    
         
            +
             
     | 
| 
      
 68 
     | 
    
         
            +
                  filter_matched(event)
         
     | 
| 
      
 69 
     | 
    
         
            +
                rescue => e
         
     | 
| 
      
 70 
     | 
    
         
            +
                  @logger.warn("Failed to query elasticsearch for previous event",
         
     | 
| 
      
 71 
     | 
    
         
            +
                               :query => query_str, :event => event, :error => e)
         
     | 
| 
      
 72 
     | 
    
         
            +
                end
         
     | 
| 
      
 73 
     | 
    
         
            +
              end # def filter
         
     | 
| 
      
 74 
     | 
    
         
            +
            end # class LogStash::Filters::Elasticsearch
         
     | 
| 
         @@ -0,0 +1,27 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            Gem::Specification.new do |s|
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
              s.name            = 'logstash-filter-elasticsearch'
         
     | 
| 
      
 4 
     | 
    
         
            +
              s.version         = '0.1.0'
         
     | 
| 
      
 5 
     | 
    
         
            +
              s.licenses        = ['Apache License (2.0)']
         
     | 
| 
      
 6 
     | 
    
         
            +
              s.summary         = "Search elasticsearch for a previous log event and copy some fields from it into the current event"
         
     | 
| 
      
 7 
     | 
    
         
            +
              s.description     = "Search elasticsearch for a previous log event and copy some fields from it into the current event."
         
     | 
| 
      
 8 
     | 
    
         
            +
              s.authors         = ["Elasticsearch"]
         
     | 
| 
      
 9 
     | 
    
         
            +
              s.email           = 'richard.pijnenburg@elasticsearch.com'
         
     | 
| 
      
 10 
     | 
    
         
            +
              s.homepage        = "http://logstash.net/"
         
     | 
| 
      
 11 
     | 
    
         
            +
              s.require_paths = ["lib"]
         
     | 
| 
      
 12 
     | 
    
         
            +
             
     | 
| 
      
 13 
     | 
    
         
            +
              # Files
         
     | 
| 
      
 14 
     | 
    
         
            +
              s.files = `git ls-files`.split($\)+::Dir.glob('vendor/*')
         
     | 
| 
      
 15 
     | 
    
         
            +
             
     | 
| 
      
 16 
     | 
    
         
            +
              # Tests
         
     | 
| 
      
 17 
     | 
    
         
            +
              s.test_files = s.files.grep(%r{^(test|spec|features)/})
         
     | 
| 
      
 18 
     | 
    
         
            +
             
     | 
| 
      
 19 
     | 
    
         
            +
              # Special flag to let us know this is actually a logstash plugin
         
     | 
| 
      
 20 
     | 
    
         
            +
              s.metadata = { "logstash_plugin" => "true", "group" => "filter" }
         
     | 
| 
      
 21 
     | 
    
         
            +
             
     | 
| 
      
 22 
     | 
    
         
            +
              # Gem dependencies
         
     | 
| 
      
 23 
     | 
    
         
            +
              s.add_runtime_dependency 'logstash', '>= 1.4.0', '< 2.0.0'
         
     | 
| 
      
 24 
     | 
    
         
            +
              s.add_runtime_dependency 'elasticsearch'
         
     | 
| 
      
 25 
     | 
    
         
            +
             
     | 
| 
      
 26 
     | 
    
         
            +
            end
         
     | 
| 
      
 27 
     | 
    
         
            +
             
     | 
| 
         @@ -0,0 +1,9 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            require "gem_publisher"
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            desc "Publish gem to RubyGems.org"
         
     | 
| 
      
 4 
     | 
    
         
            +
            task :publish_gem do |t|
         
     | 
| 
      
 5 
     | 
    
         
            +
              gem_file = Dir.glob(File.expand_path('../*.gemspec',File.dirname(__FILE__))).first
         
     | 
| 
      
 6 
     | 
    
         
            +
              gem = GemPublisher.publish_if_updated(gem_file, :rubygems)
         
     | 
| 
      
 7 
     | 
    
         
            +
              puts "Published #{gem}" if gem
         
     | 
| 
      
 8 
     | 
    
         
            +
            end
         
     | 
| 
      
 9 
     | 
    
         
            +
             
     | 
    
        data/rakelib/vendor.rake
    ADDED
    
    | 
         @@ -0,0 +1,169 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            require "net/http"
         
     | 
| 
      
 2 
     | 
    
         
            +
            require "uri"
         
     | 
| 
      
 3 
     | 
    
         
            +
            require "digest/sha1"
         
     | 
| 
      
 4 
     | 
    
         
            +
             
     | 
| 
      
 5 
     | 
    
         
            +
            def vendor(*args)
         
     | 
| 
      
 6 
     | 
    
         
            +
              return File.join("vendor", *args)
         
     | 
| 
      
 7 
     | 
    
         
            +
            end
         
     | 
| 
      
 8 
     | 
    
         
            +
             
     | 
| 
      
 9 
     | 
    
         
            +
            directory "vendor/" => ["vendor"] do |task, args|
         
     | 
| 
      
 10 
     | 
    
         
            +
              mkdir task.name
         
     | 
| 
      
 11 
     | 
    
         
            +
            end
         
     | 
| 
      
 12 
     | 
    
         
            +
             
     | 
| 
      
 13 
     | 
    
         
            +
            def fetch(url, sha1, output)
         
     | 
| 
      
 14 
     | 
    
         
            +
             
     | 
| 
      
 15 
     | 
    
         
            +
              puts "Downloading #{url}"
         
     | 
| 
      
 16 
     | 
    
         
            +
              actual_sha1 = download(url, output)
         
     | 
| 
      
 17 
     | 
    
         
            +
             
     | 
| 
      
 18 
     | 
    
         
            +
              if actual_sha1 != sha1
         
     | 
| 
      
 19 
     | 
    
         
            +
                fail "SHA1 does not match (expected '#{sha1}' but got '#{actual_sha1}')"
         
     | 
| 
      
 20 
     | 
    
         
            +
              end
         
     | 
| 
      
 21 
     | 
    
         
            +
            end # def fetch
         
     | 
| 
      
 22 
     | 
    
         
            +
             
     | 
| 
      
 23 
     | 
    
         
            +
            def file_fetch(url, sha1)
         
     | 
| 
      
 24 
     | 
    
         
            +
              filename = File.basename( URI(url).path )
         
     | 
| 
      
 25 
     | 
    
         
            +
              output = "vendor/#{filename}"
         
     | 
| 
      
 26 
     | 
    
         
            +
              task output => [ "vendor/" ] do
         
     | 
| 
      
 27 
     | 
    
         
            +
                begin
         
     | 
| 
      
 28 
     | 
    
         
            +
                  actual_sha1 = file_sha1(output)
         
     | 
| 
      
 29 
     | 
    
         
            +
                  if actual_sha1 != sha1
         
     | 
| 
      
 30 
     | 
    
         
            +
                    fetch(url, sha1, output)
         
     | 
| 
      
 31 
     | 
    
         
            +
                  end
         
     | 
| 
      
 32 
     | 
    
         
            +
                rescue Errno::ENOENT
         
     | 
| 
      
 33 
     | 
    
         
            +
                  fetch(url, sha1, output)
         
     | 
| 
      
 34 
     | 
    
         
            +
                end
         
     | 
| 
      
 35 
     | 
    
         
            +
              end.invoke
         
     | 
| 
      
 36 
     | 
    
         
            +
             
     | 
| 
      
 37 
     | 
    
         
            +
              return output
         
     | 
| 
      
 38 
     | 
    
         
            +
            end
         
     | 
| 
      
 39 
     | 
    
         
            +
             
     | 
| 
      
 40 
     | 
    
         
            +
            def file_sha1(path)
         
     | 
| 
      
 41 
     | 
    
         
            +
              digest = Digest::SHA1.new
         
     | 
| 
      
 42 
     | 
    
         
            +
              fd = File.new(path, "r")
         
     | 
| 
      
 43 
     | 
    
         
            +
              while true
         
     | 
| 
      
 44 
     | 
    
         
            +
                begin
         
     | 
| 
      
 45 
     | 
    
         
            +
                  digest << fd.sysread(16384)
         
     | 
| 
      
 46 
     | 
    
         
            +
                rescue EOFError
         
     | 
| 
      
 47 
     | 
    
         
            +
                  break
         
     | 
| 
      
 48 
     | 
    
         
            +
                end
         
     | 
| 
      
 49 
     | 
    
         
            +
              end
         
     | 
| 
      
 50 
     | 
    
         
            +
              return digest.hexdigest
         
     | 
| 
      
 51 
     | 
    
         
            +
            ensure
         
     | 
| 
      
 52 
     | 
    
         
            +
              fd.close if fd
         
     | 
| 
      
 53 
     | 
    
         
            +
            end
         
     | 
| 
      
 54 
     | 
    
         
            +
             
     | 
| 
      
 55 
     | 
    
         
            +
            def download(url, output)
         
     | 
| 
      
 56 
     | 
    
         
            +
              uri = URI(url)
         
     | 
| 
      
 57 
     | 
    
         
            +
              digest = Digest::SHA1.new
         
     | 
| 
      
 58 
     | 
    
         
            +
              tmp = "#{output}.tmp"
         
     | 
| 
      
 59 
     | 
    
         
            +
              Net::HTTP.start(uri.host, uri.port, :use_ssl => (uri.scheme == "https")) do |http|
         
     | 
| 
      
 60 
     | 
    
         
            +
                request = Net::HTTP::Get.new(uri.path)
         
     | 
| 
      
 61 
     | 
    
         
            +
                http.request(request) do |response|
         
     | 
| 
      
 62 
     | 
    
         
            +
                  fail "HTTP fetch failed for #{url}. #{response}" if [200, 301].include?(response.code)
         
     | 
| 
      
 63 
     | 
    
         
            +
                  size = (response["content-length"].to_i || -1).to_f
         
     | 
| 
      
 64 
     | 
    
         
            +
                  count = 0
         
     | 
| 
      
 65 
     | 
    
         
            +
                  File.open(tmp, "w") do |fd|
         
     | 
| 
      
 66 
     | 
    
         
            +
                    response.read_body do |chunk|
         
     | 
| 
      
 67 
     | 
    
         
            +
                      fd.write(chunk)
         
     | 
| 
      
 68 
     | 
    
         
            +
                      digest << chunk
         
     | 
| 
      
 69 
     | 
    
         
            +
                      if size > 0 && $stdout.tty?
         
     | 
| 
      
 70 
     | 
    
         
            +
                        count += chunk.bytesize
         
     | 
| 
      
 71 
     | 
    
         
            +
                        $stdout.write(sprintf("\r%0.2f%%", count/size * 100))
         
     | 
| 
      
 72 
     | 
    
         
            +
                      end
         
     | 
| 
      
 73 
     | 
    
         
            +
                    end
         
     | 
| 
      
 74 
     | 
    
         
            +
                  end
         
     | 
| 
      
 75 
     | 
    
         
            +
                  $stdout.write("\r      \r") if $stdout.tty?
         
     | 
| 
      
 76 
     | 
    
         
            +
                end
         
     | 
| 
      
 77 
     | 
    
         
            +
              end
         
     | 
| 
      
 78 
     | 
    
         
            +
             
     | 
| 
      
 79 
     | 
    
         
            +
              File.rename(tmp, output)
         
     | 
| 
      
 80 
     | 
    
         
            +
             
     | 
| 
      
 81 
     | 
    
         
            +
              return digest.hexdigest
         
     | 
| 
      
 82 
     | 
    
         
            +
            rescue SocketError => e
         
     | 
| 
      
 83 
     | 
    
         
            +
              puts "Failure while downloading #{url}: #{e}"
         
     | 
| 
      
 84 
     | 
    
         
            +
              raise
         
     | 
| 
      
 85 
     | 
    
         
            +
            ensure
         
     | 
| 
      
 86 
     | 
    
         
            +
              File.unlink(tmp) if File.exist?(tmp)
         
     | 
| 
      
 87 
     | 
    
         
            +
            end # def download
         
     | 
| 
      
 88 
     | 
    
         
            +
             
     | 
| 
      
 89 
     | 
    
         
            +
            def untar(tarball, &block)
         
     | 
| 
      
 90 
     | 
    
         
            +
              require "archive/tar/minitar"
         
     | 
| 
      
 91 
     | 
    
         
            +
              tgz = Zlib::GzipReader.new(File.open(tarball))
         
     | 
| 
      
 92 
     | 
    
         
            +
              # Pull out typesdb
         
     | 
| 
      
 93 
     | 
    
         
            +
              tar = Archive::Tar::Minitar::Input.open(tgz)
         
     | 
| 
      
 94 
     | 
    
         
            +
              tar.each do |entry|
         
     | 
| 
      
 95 
     | 
    
         
            +
                path = block.call(entry)
         
     | 
| 
      
 96 
     | 
    
         
            +
                next if path.nil?
         
     | 
| 
      
 97 
     | 
    
         
            +
                parent = File.dirname(path)
         
     | 
| 
      
 98 
     | 
    
         
            +
                
         
     | 
| 
      
 99 
     | 
    
         
            +
                mkdir_p parent unless File.directory?(parent)
         
     | 
| 
      
 100 
     | 
    
         
            +
             
     | 
| 
      
 101 
     | 
    
         
            +
                # Skip this file if the output file is the same size
         
     | 
| 
      
 102 
     | 
    
         
            +
                if entry.directory?
         
     | 
| 
      
 103 
     | 
    
         
            +
                  mkdir path unless File.directory?(path)
         
     | 
| 
      
 104 
     | 
    
         
            +
                else
         
     | 
| 
      
 105 
     | 
    
         
            +
                  entry_mode = entry.instance_eval { @mode } & 0777
         
     | 
| 
      
 106 
     | 
    
         
            +
                  if File.exists?(path)
         
     | 
| 
      
 107 
     | 
    
         
            +
                    stat = File.stat(path)
         
     | 
| 
      
 108 
     | 
    
         
            +
                    # TODO(sissel): Submit a patch to archive-tar-minitar upstream to
         
     | 
| 
      
 109 
     | 
    
         
            +
                    # expose headers in the entry.
         
     | 
| 
      
 110 
     | 
    
         
            +
                    entry_size = entry.instance_eval { @size }
         
     | 
| 
      
 111 
     | 
    
         
            +
                    # If file sizes are same, skip writing.
         
     | 
| 
      
 112 
     | 
    
         
            +
                    next if stat.size == entry_size && (stat.mode & 0777) == entry_mode
         
     | 
| 
      
 113 
     | 
    
         
            +
                  end
         
     | 
| 
      
 114 
     | 
    
         
            +
                  puts "Extracting #{entry.full_name} from #{tarball} #{entry_mode.to_s(8)}"
         
     | 
| 
      
 115 
     | 
    
         
            +
                  File.open(path, "w") do |fd|
         
     | 
| 
      
 116 
     | 
    
         
            +
                    # eof? check lets us skip empty files. Necessary because the API provided by
         
     | 
| 
      
 117 
     | 
    
         
            +
                    # Archive::Tar::Minitar::Reader::EntryStream only mostly acts like an
         
     | 
| 
      
 118 
     | 
    
         
            +
                    # IO object. Something about empty files in this EntryStream causes
         
     | 
| 
      
 119 
     | 
    
         
            +
                    # IO.copy_stream to throw "can't convert nil into String" on JRuby
         
     | 
| 
      
 120 
     | 
    
         
            +
                    # TODO(sissel): File a bug about this.
         
     | 
| 
      
 121 
     | 
    
         
            +
                    while !entry.eof?
         
     | 
| 
      
 122 
     | 
    
         
            +
                      chunk = entry.read(16384)
         
     | 
| 
      
 123 
     | 
    
         
            +
                      fd.write(chunk)
         
     | 
| 
      
 124 
     | 
    
         
            +
                    end
         
     | 
| 
      
 125 
     | 
    
         
            +
                      #IO.copy_stream(entry, fd)
         
     | 
| 
      
 126 
     | 
    
         
            +
                  end
         
     | 
| 
      
 127 
     | 
    
         
            +
                  File.chmod(entry_mode, path)
         
     | 
| 
      
 128 
     | 
    
         
            +
                end
         
     | 
| 
      
 129 
     | 
    
         
            +
              end
         
     | 
| 
      
 130 
     | 
    
         
            +
              tar.close
         
     | 
| 
      
 131 
     | 
    
         
            +
              File.unlink(tarball) if File.file?(tarball)
         
     | 
| 
      
 132 
     | 
    
         
            +
            end # def untar
         
     | 
| 
      
 133 
     | 
    
         
            +
             
     | 
| 
      
 134 
     | 
    
         
            +
            def ungz(file)
         
     | 
| 
      
 135 
     | 
    
         
            +
             
     | 
| 
      
 136 
     | 
    
         
            +
              outpath = file.gsub('.gz', '')
         
     | 
| 
      
 137 
     | 
    
         
            +
              tgz = Zlib::GzipReader.new(File.open(file))
         
     | 
| 
      
 138 
     | 
    
         
            +
              begin
         
     | 
| 
      
 139 
     | 
    
         
            +
                File.open(outpath, "w") do |out|
         
     | 
| 
      
 140 
     | 
    
         
            +
                  IO::copy_stream(tgz, out)
         
     | 
| 
      
 141 
     | 
    
         
            +
                end
         
     | 
| 
      
 142 
     | 
    
         
            +
                File.unlink(file)
         
     | 
| 
      
 143 
     | 
    
         
            +
              rescue
         
     | 
| 
      
 144 
     | 
    
         
            +
                File.unlink(outpath) if File.file?(outpath)
         
     | 
| 
      
 145 
     | 
    
         
            +
               raise
         
     | 
| 
      
 146 
     | 
    
         
            +
              end
         
     | 
| 
      
 147 
     | 
    
         
            +
              tgz.close
         
     | 
| 
      
 148 
     | 
    
         
            +
            end
         
     | 
| 
      
 149 
     | 
    
         
            +
             
     | 
| 
      
 150 
     | 
    
         
            +
            desc "Process any vendor files required for this plugin"
         
     | 
| 
      
 151 
     | 
    
         
            +
            task "vendor" do |task, args|
         
     | 
| 
      
 152 
     | 
    
         
            +
             
     | 
| 
      
 153 
     | 
    
         
            +
              @files.each do |file| 
         
     | 
| 
      
 154 
     | 
    
         
            +
                download = file_fetch(file['url'], file['sha1'])
         
     | 
| 
      
 155 
     | 
    
         
            +
                if download =~ /.tar.gz/
         
     | 
| 
      
 156 
     | 
    
         
            +
                  prefix = download.gsub('.tar.gz', '').gsub('vendor/', '')
         
     | 
| 
      
 157 
     | 
    
         
            +
                  untar(download) do |entry|
         
     | 
| 
      
 158 
     | 
    
         
            +
                    if !file['files'].nil?
         
     | 
| 
      
 159 
     | 
    
         
            +
                      next unless file['files'].include?(entry.full_name.gsub(prefix, ''))
         
     | 
| 
      
 160 
     | 
    
         
            +
                      out = entry.full_name.split("/").last
         
     | 
| 
      
 161 
     | 
    
         
            +
                    end
         
     | 
| 
      
 162 
     | 
    
         
            +
                    File.join('vendor', out)
         
     | 
| 
      
 163 
     | 
    
         
            +
                  end
         
     | 
| 
      
 164 
     | 
    
         
            +
                elsif download =~ /.gz/
         
     | 
| 
      
 165 
     | 
    
         
            +
                  ungz(download)
         
     | 
| 
      
 166 
     | 
    
         
            +
                end
         
     | 
| 
      
 167 
     | 
    
         
            +
              end
         
     | 
| 
      
 168 
     | 
    
         
            +
             
     | 
| 
      
 169 
     | 
    
         
            +
            end
         
     | 
| 
         @@ -0,0 +1 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            require 'spec_helper'
         
     | 
    
        metadata
    ADDED
    
    | 
         @@ -0,0 +1,91 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            --- !ruby/object:Gem::Specification
         
     | 
| 
      
 2 
     | 
    
         
            +
            name: logstash-filter-elasticsearch
         
     | 
| 
      
 3 
     | 
    
         
            +
            version: !ruby/object:Gem::Version
         
     | 
| 
      
 4 
     | 
    
         
            +
              version: 0.1.0
         
     | 
| 
      
 5 
     | 
    
         
            +
            platform: ruby
         
     | 
| 
      
 6 
     | 
    
         
            +
            authors:
         
     | 
| 
      
 7 
     | 
    
         
            +
            - Elasticsearch
         
     | 
| 
      
 8 
     | 
    
         
            +
            autorequire: 
         
     | 
| 
      
 9 
     | 
    
         
            +
            bindir: bin
         
     | 
| 
      
 10 
     | 
    
         
            +
            cert_chain: []
         
     | 
| 
      
 11 
     | 
    
         
            +
            date: 2014-11-05 00:00:00.000000000 Z
         
     | 
| 
      
 12 
     | 
    
         
            +
            dependencies:
         
     | 
| 
      
 13 
     | 
    
         
            +
            - !ruby/object:Gem::Dependency
         
     | 
| 
      
 14 
     | 
    
         
            +
              name: logstash
         
     | 
| 
      
 15 
     | 
    
         
            +
              requirement: !ruby/object:Gem::Requirement
         
     | 
| 
      
 16 
     | 
    
         
            +
                requirements:
         
     | 
| 
      
 17 
     | 
    
         
            +
                - - ! '>='
         
     | 
| 
      
 18 
     | 
    
         
            +
                  - !ruby/object:Gem::Version
         
     | 
| 
      
 19 
     | 
    
         
            +
                    version: 1.4.0
         
     | 
| 
      
 20 
     | 
    
         
            +
                - - <
         
     | 
| 
      
 21 
     | 
    
         
            +
                  - !ruby/object:Gem::Version
         
     | 
| 
      
 22 
     | 
    
         
            +
                    version: 2.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: 1.4.0
         
     | 
| 
      
 30 
     | 
    
         
            +
                - - <
         
     | 
| 
      
 31 
     | 
    
         
            +
                  - !ruby/object:Gem::Version
         
     | 
| 
      
 32 
     | 
    
         
            +
                    version: 2.0.0
         
     | 
| 
      
 33 
     | 
    
         
            +
            - !ruby/object:Gem::Dependency
         
     | 
| 
      
 34 
     | 
    
         
            +
              name: elasticsearch
         
     | 
| 
      
 35 
     | 
    
         
            +
              requirement: !ruby/object:Gem::Requirement
         
     | 
| 
      
 36 
     | 
    
         
            +
                requirements:
         
     | 
| 
      
 37 
     | 
    
         
            +
                - - ! '>='
         
     | 
| 
      
 38 
     | 
    
         
            +
                  - !ruby/object:Gem::Version
         
     | 
| 
      
 39 
     | 
    
         
            +
                    version: '0'
         
     | 
| 
      
 40 
     | 
    
         
            +
              type: :runtime
         
     | 
| 
      
 41 
     | 
    
         
            +
              prerelease: false
         
     | 
| 
      
 42 
     | 
    
         
            +
              version_requirements: !ruby/object:Gem::Requirement
         
     | 
| 
      
 43 
     | 
    
         
            +
                requirements:
         
     | 
| 
      
 44 
     | 
    
         
            +
                - - ! '>='
         
     | 
| 
      
 45 
     | 
    
         
            +
                  - !ruby/object:Gem::Version
         
     | 
| 
      
 46 
     | 
    
         
            +
                    version: '0'
         
     | 
| 
      
 47 
     | 
    
         
            +
            description: Search elasticsearch for a previous log event and copy some fields from
         
     | 
| 
      
 48 
     | 
    
         
            +
              it into the current event.
         
     | 
| 
      
 49 
     | 
    
         
            +
            email: richard.pijnenburg@elasticsearch.com
         
     | 
| 
      
 50 
     | 
    
         
            +
            executables: []
         
     | 
| 
      
 51 
     | 
    
         
            +
            extensions: []
         
     | 
| 
      
 52 
     | 
    
         
            +
            extra_rdoc_files: []
         
     | 
| 
      
 53 
     | 
    
         
            +
            files:
         
     | 
| 
      
 54 
     | 
    
         
            +
            - .gitignore
         
     | 
| 
      
 55 
     | 
    
         
            +
            - Gemfile
         
     | 
| 
      
 56 
     | 
    
         
            +
            - LICENSE
         
     | 
| 
      
 57 
     | 
    
         
            +
            - Rakefile
         
     | 
| 
      
 58 
     | 
    
         
            +
            - lib/logstash/filters/elasticsearch.rb
         
     | 
| 
      
 59 
     | 
    
         
            +
            - logstash-filter-elasticsearch.gemspec
         
     | 
| 
      
 60 
     | 
    
         
            +
            - rakelib/publish.rake
         
     | 
| 
      
 61 
     | 
    
         
            +
            - rakelib/vendor.rake
         
     | 
| 
      
 62 
     | 
    
         
            +
            - spec/filters/elasticsearch_spec.rb
         
     | 
| 
      
 63 
     | 
    
         
            +
            homepage: http://logstash.net/
         
     | 
| 
      
 64 
     | 
    
         
            +
            licenses:
         
     | 
| 
      
 65 
     | 
    
         
            +
            - Apache License (2.0)
         
     | 
| 
      
 66 
     | 
    
         
            +
            metadata:
         
     | 
| 
      
 67 
     | 
    
         
            +
              logstash_plugin: 'true'
         
     | 
| 
      
 68 
     | 
    
         
            +
              group: filter
         
     | 
| 
      
 69 
     | 
    
         
            +
            post_install_message: 
         
     | 
| 
      
 70 
     | 
    
         
            +
            rdoc_options: []
         
     | 
| 
      
 71 
     | 
    
         
            +
            require_paths:
         
     | 
| 
      
 72 
     | 
    
         
            +
            - lib
         
     | 
| 
      
 73 
     | 
    
         
            +
            required_ruby_version: !ruby/object:Gem::Requirement
         
     | 
| 
      
 74 
     | 
    
         
            +
              requirements:
         
     | 
| 
      
 75 
     | 
    
         
            +
              - - ! '>='
         
     | 
| 
      
 76 
     | 
    
         
            +
                - !ruby/object:Gem::Version
         
     | 
| 
      
 77 
     | 
    
         
            +
                  version: '0'
         
     | 
| 
      
 78 
     | 
    
         
            +
            required_rubygems_version: !ruby/object:Gem::Requirement
         
     | 
| 
      
 79 
     | 
    
         
            +
              requirements:
         
     | 
| 
      
 80 
     | 
    
         
            +
              - - ! '>='
         
     | 
| 
      
 81 
     | 
    
         
            +
                - !ruby/object:Gem::Version
         
     | 
| 
      
 82 
     | 
    
         
            +
                  version: '0'
         
     | 
| 
      
 83 
     | 
    
         
            +
            requirements: []
         
     | 
| 
      
 84 
     | 
    
         
            +
            rubyforge_project: 
         
     | 
| 
      
 85 
     | 
    
         
            +
            rubygems_version: 2.4.1
         
     | 
| 
      
 86 
     | 
    
         
            +
            signing_key: 
         
     | 
| 
      
 87 
     | 
    
         
            +
            specification_version: 4
         
     | 
| 
      
 88 
     | 
    
         
            +
            summary: Search elasticsearch for a previous log event and copy some fields from it
         
     | 
| 
      
 89 
     | 
    
         
            +
              into the current event
         
     | 
| 
      
 90 
     | 
    
         
            +
            test_files:
         
     | 
| 
      
 91 
     | 
    
         
            +
            - spec/filters/elasticsearch_spec.rb
         
     |