123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- require 'time_tracker'
- module Agents
- class HttpStatusAgent < Agent
- include WebRequestConcern
- include FormConfigurable
- can_dry_run!
- can_order_created_events!
- default_schedule "every_12h"
- form_configurable :url
- form_configurable :disable_redirect_follow, type: :boolean
- form_configurable :changes_only, type: :boolean
- form_configurable :headers_to_save
- description <<~MD
- The HttpStatusAgent will check a url and emit the resulting HTTP status code with the time that it waited for a reply. Additionally, it will optionally emit the value of one or more specified headers.
- Specify a `Url` and the Http Status Agent will produce an event with the HTTP status code. If you specify one or more `Headers to save` (comma-delimited) as well, that header or headers' value(s) will be included in the event.
- The `disable redirect follow` option causes the Agent to not follow HTTP redirects. For example, setting this to `true` will cause an agent that receives a 301 redirect to `http://yahoo.com` to return a status of 301 instead of following the redirect and returning 200.
- The `changes only` option causes the Agent to report an event only when the status changes. If set to false, an event will be created for every check. If set to true, an event will only be created when the status changes (like if your site goes from 200 to 500).
- MD
- event_description <<~MD
- Events will have the following fields:
- {
- "url": "...",
- "status": "...",
- "elapsed_time": "...",
- "headers": {
- "...": "..."
- }
- }
- MD
- def working?
- memory['last_status'].to_i > 0
- end
- def default_options
- {
- 'url' => "http://google.com",
- 'disable_redirect_follow' => "true",
- }
- end
- def validate_options
- errors.add(:base, "a url must be specified") unless options['url'].present?
- end
- def header_array(str)
- (str || '').split(',').map(&:strip)
- end
- def check
- check_this_url interpolated[:url], header_array(interpolated[:headers_to_save])
- end
- def receive(incoming_events)
- incoming_events.each do |event|
- interpolate_with(event) do
- check_this_url interpolated[:url],
- header_array(interpolated[:headers_to_save])
- end
- end
- end
- private
- def check_this_url(url, local_headers)
- # Track time
- measured_result = TimeTracker.track { ping(url) }
- current_status = measured_result.result ? measured_result.status.to_s : ''
- return if options['changes_only'] == 'true' && current_status == memory['last_status'].to_s
- payload = { 'url' => url, 'response_received' => false, 'elapsed_time' => measured_result.elapsed_time }
- # Deal with failures
- if measured_result.result
- final_url = boolify(interpolated['disable_redirect_follow']) ? url : measured_result.result.env.url.to_s
- payload.merge!({ 'final_url' => final_url, 'redirected' => (url != final_url), 'response_received' => true,
- 'status' => current_status })
- # Deal with headers
- if local_headers.present?
- header_results = local_headers.each_with_object({}) { |header, hash|
- hash[header] = measured_result.result.headers[header]
- }
- payload.merge!({ 'headers' => header_results })
- end
- create_event(payload:)
- memory['last_status'] = measured_result.status.to_s
- else
- create_event(payload:)
- memory['last_status'] = nil
- end
- end
- def ping(url)
- result = faraday.get url
- result.status > 0 ? result : nil
- rescue StandardError
- nil
- end
- end
- end
|