weather_agent.rb 7.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. require 'date'
  2. require 'cgi'
  3. module Agents
  4. class WeatherAgent < Agent
  5. cannot_receive_events!
  6. gem_dependency_check { defined?(Wunderground) && defined?(ForecastIO) }
  7. description <<-MD
  8. The Weather Agent creates an event for the day's weather at a given `location`.
  9. #{'## Include `forecast_io` and `wunderground` in your Gemfile to use this Agent!' if dependencies_missing?}
  10. You also must select `which_day` you would like to get the weather for where the number 0 is for today and 1 is for tomorrow and so on. Weather is only returned for 1 week at a time.
  11. The weather can be provided by either Wunderground or ForecastIO. To choose which `service` to use, enter either `forecastio` or `wunderground`.
  12. The `location` can be a US zipcode, or any location that Wunderground supports. To find one, search [wunderground.com](http://wunderground.com) and copy the location part of the URL. For example, a result for San Francisco gives `http://www.wunderground.com/US/CA/San_Francisco.html` and London, England gives `http://www.wunderground.com/q/zmw:00000.1.03772`. The locations in each are `US/CA/San_Francisco` and `zmw:00000.1.03772`, respectively.
  13. If you plan on using ForecastIO, the `location` must be a comma-separated string of co-ordinates (longitude, latitude). For example, San Francisco would be `37.7771,-122.4196`.
  14. You must setup an [API key for Wunderground](http://www.wunderground.com/weather/api/) in order to use this Agent with Wunderground.
  15. You must setup an [API key for Forecast](https://developer.forecast.io/) in order to use this Agent with ForecastIO.
  16. Set `expected_update_period_in_days` to the maximum amount of time that you'd expect to pass between Events being created by this Agent.
  17. If you want to see the returned texts in your language, then set the `language` parameter in ISO 639-1 format.
  18. MD
  19. event_description <<-MD
  20. Events look like this:
  21. {
  22. "location": "12345",
  23. "date": {
  24. "epoch": "1357959600",
  25. "pretty": "10:00 PM EST on January 11, 2013"
  26. },
  27. "high": {
  28. "fahrenheit": "64",
  29. "celsius": "18"
  30. },
  31. "low": {
  32. "fahrenheit": "52",
  33. "celsius": "11"
  34. },
  35. "conditions": "Rain Showers",
  36. "icon": "rain",
  37. "icon_url": "http://icons-ak.wxug.com/i/c/k/rain.gif",
  38. "skyicon": "mostlycloudy",
  39. ...
  40. }
  41. MD
  42. default_schedule "8pm"
  43. def working?
  44. event_created_within?((interpolated['expected_update_period_in_days'].presence || 2).to_i) && !recent_error_logs? && key_setup?
  45. end
  46. def key_setup?
  47. interpolated['api_key'].present? && interpolated['api_key'] != "your-key" && interpolated['api_key'] != "put-your-key-here"
  48. end
  49. def default_options
  50. {
  51. 'service' => 'wunderground',
  52. 'api_key' => 'your-key',
  53. 'location' => '94103',
  54. 'which_day' => '1',
  55. 'language' => 'EN',
  56. 'expected_update_period_in_days' => '2'
  57. }
  58. end
  59. def check
  60. if key_setup?
  61. create_event :payload => model(weather_provider, which_day).merge('location' => location)
  62. end
  63. end
  64. private
  65. def weather_provider
  66. interpolated["service"].presence || "wunderground"
  67. end
  68. def which_day
  69. (interpolated["which_day"].presence || 1).to_i
  70. end
  71. def location
  72. interpolated["location"].presence || interpolated["zipcode"]
  73. end
  74. def language
  75. interpolated['language'].presence || 'EN'
  76. end
  77. def validate_options
  78. errors.add(:base, "service must be set to 'forecastio' or 'wunderground'") unless ["forecastio", "wunderground"].include?(weather_provider)
  79. errors.add(:base, "location is required") unless location.present?
  80. errors.add(:base, "api_key is required") unless interpolated['api_key'].present?
  81. errors.add(:base, "which_day selection is required") unless which_day.present?
  82. end
  83. def wunderground
  84. if key_setup?
  85. forecast = Wunderground.new(interpolated['api_key'], language: language.upcase).forecast_for(location)
  86. merged = {}
  87. forecast['forecast']['simpleforecast']['forecastday'].each { |daily| merged[daily['period']] = daily }
  88. forecast['forecast']['txt_forecast']['forecastday'].each { |daily| (merged[daily['period']] || {}).merge!(daily) }
  89. merged
  90. end
  91. end
  92. def forecastio
  93. if key_setup?
  94. ForecastIO.api_key = interpolated['api_key']
  95. lat, lng = location.split(',')
  96. ForecastIO.forecast(lat, lng, params: {lang: language.downcase})['daily']['data']
  97. end
  98. end
  99. def model(weather_provider,which_day)
  100. if weather_provider == "wunderground"
  101. wunderground[which_day]
  102. elsif weather_provider == "forecastio"
  103. forecastio.each do |value|
  104. timestamp = Time.at(value.time)
  105. if (timestamp.to_date - Time.now.to_date).to_i == which_day
  106. day = {
  107. 'date' => {
  108. 'epoch' => value.time.to_s,
  109. 'pretty' => timestamp.strftime("%l:%M %p %Z on %B %d, %Y"),
  110. 'day' => timestamp.day,
  111. 'month' => timestamp.month,
  112. 'year' => timestamp.year,
  113. 'yday' => timestamp.yday,
  114. 'hour' => timestamp.hour,
  115. 'min' => timestamp.strftime("%M"),
  116. 'sec' => timestamp.sec,
  117. 'isdst' => timestamp.isdst ? 1 : 0 ,
  118. 'monthname' => timestamp.strftime("%B"),
  119. 'monthname_short' => timestamp.strftime("%b"),
  120. 'weekday_short' => timestamp.strftime("%a"),
  121. 'weekday' => timestamp.strftime("%A"),
  122. 'ampm' => timestamp.strftime("%p"),
  123. 'tz_short' => timestamp.zone
  124. },
  125. 'period' => which_day.to_i,
  126. 'high' => {
  127. 'fahrenheit' => value.temperatureMax.round().to_s,
  128. 'epoch' => value.temperatureMaxTime.to_s,
  129. 'fahrenheit_apparent' => value.apparentTemperatureMax.round().to_s,
  130. 'epoch_apparent' => value.apparentTemperatureMaxTime.to_s,
  131. 'celsius' => ((5*(Float(value.temperatureMax) - 32))/9).round().to_s
  132. },
  133. 'low' => {
  134. 'fahrenheit' => value.temperatureMin.round().to_s,
  135. 'epoch' => value.temperatureMinTime.to_s,
  136. 'fahrenheit_apparent' => value.apparentTemperatureMin.round().to_s,
  137. 'epoch_apparent' => value.apparentTemperatureMinTime.to_s,
  138. 'celsius' => ((5*(Float(value.temperatureMin) - 32))/9).round().to_s
  139. },
  140. 'conditions' => value.summary,
  141. 'icon' => value.icon,
  142. 'avehumidity' => (value.humidity * 100).to_i,
  143. 'sunriseTime' => value.sunriseTime.to_s,
  144. 'sunsetTime' => value.sunsetTime.to_s,
  145. 'moonPhase' => value.moonPhase.to_s,
  146. 'precip' => {
  147. 'intensity' => value.precipIntensity.to_s,
  148. 'intensity_max' => value.precipIntensityMax.to_s,
  149. 'intensity_max_epoch' => value.precipIntensityMaxTime.to_s,
  150. 'probability' => value.precipProbability.to_s,
  151. 'type' => value.precipType
  152. },
  153. 'dewPoint' => value.dewPoint.to_s,
  154. 'avewind' => {
  155. 'mph' => value.windSpeed.round().to_s,
  156. 'kph' => (Float(value.windSpeed) * 1.609344).round().to_s,
  157. 'degrees' => value.windBearing.to_s
  158. },
  159. 'visibility' => value.visibility.to_s,
  160. 'cloudCover' => value.cloudCover.to_s,
  161. 'pressure' => value.pressure.to_s,
  162. 'ozone' => value.ozone.to_s
  163. }
  164. return day
  165. end
  166. end
  167. end
  168. end
  169. end
  170. end