binnacle 0.4.8
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +3 -0
- data/.ruby-gemset +1 -0
- data/.ruby-version +1 -0
- data/Gemfile +8 -0
- data/Gemfile.lock +138 -0
- data/LICENSE.txt +22 -0
- data/README.asc +264 -0
- data/Rakefile +48 -0
- data/bin/binnacle +5 -0
- data/binnacle.gemspec +49 -0
- data/lib/binnacle.rb +112 -0
- data/lib/binnacle/client.rb +115 -0
- data/lib/binnacle/commands/commands.rb +20 -0
- data/lib/binnacle/commands/help.rb +21 -0
- data/lib/binnacle/commands/tail.rb +187 -0
- data/lib/binnacle/configuration.rb +278 -0
- data/lib/binnacle/connection.rb +66 -0
- data/lib/binnacle/errors.rb +4 -0
- data/lib/binnacle/http_logging/adapters/ethon.rb +44 -0
- data/lib/binnacle/http_logging/adapters/excon.rb +62 -0
- data/lib/binnacle/http_logging/adapters/http.rb +63 -0
- data/lib/binnacle/http_logging/adapters/httpclient.rb +93 -0
- data/lib/binnacle/http_logging/adapters/net_http.rb +54 -0
- data/lib/binnacle/http_logging/adapters/patron.rb +24 -0
- data/lib/binnacle/http_logging/adapters/typhoeus.rb +20 -0
- data/lib/binnacle/http_logging/http_logger.rb +83 -0
- data/lib/binnacle/logging/formatter.rb +62 -0
- data/lib/binnacle/logging/logging.rb +36 -0
- data/lib/binnacle/logging/request_log_subscriber.rb +120 -0
- data/lib/binnacle/resource.rb +54 -0
- data/lib/binnacle/resources/event.rb +100 -0
- data/lib/binnacle/trap/backtrace.rb +130 -0
- data/lib/binnacle/trap/exception_event.rb +176 -0
- data/lib/binnacle/trap/middleware.rb +45 -0
- data/lib/binnacle/trap/railtie.rb +28 -0
- data/lib/binnacle/version.rb +3 -0
- data/lib/generators/binnacle/binnacle_generator.rb +78 -0
- data/lib/generators/binnacle/templates/firebase-messaging-sw.js.erb +35 -0
- data/spec/adapters/ethon_adapter.rb +19 -0
- data/spec/adapters/excon_adapter.rb +10 -0
- data/spec/adapters/faraday_adapter.rb +46 -0
- data/spec/adapters/http_adapter.rb +14 -0
- data/spec/adapters/http_base_adapter.rb +37 -0
- data/spec/adapters/httparty_adapter.rb +10 -0
- data/spec/adapters/httpclient_adapter.rb +21 -0
- data/spec/adapters/net_http_adapter.rb +14 -0
- data/spec/adapters/open_uri_adapter.rb +17 -0
- data/spec/adapters/patron_adapter.rb +29 -0
- data/spec/adapters/typhoeus_adapter.rb +25 -0
- data/spec/backtrace_spec.rb +197 -0
- data/spec/binnacle_spec.rb +33 -0
- data/spec/client_spec.rb +128 -0
- data/spec/commands_spec.rb +91 -0
- data/spec/configuration_spec.rb +139 -0
- data/spec/connection_spec.rb +34 -0
- data/spec/exception_event_spec.rb +72 -0
- data/spec/http_logger_spec.rb +77 -0
- data/spec/logger_spec.rb +60 -0
- data/spec/spec_helper.rb +117 -0
- data/spec/support/index.html +8 -0
- data/spec/support/index.html.gz +0 -0
- data/spec/support/test.bin +0 -0
- data/spec/support/test_server.rb +31 -0
- data/spec/support/utf8-invalid.html +0 -0
- data/spec/support/utf8.html +8 -0
- data/spec/vcr/binnacle/configure_can_be_configured_via_a_hash_of_options.yml +42 -0
- data/spec/vcr/binnacle/configure_creates_a_module_level_instance_of_a_binnacle_client.yml +42 -0
- data/spec/vcr/binnacle_client/events_invokes_the_events_api.yml +124 -0
- data/spec/vcr/binnacle_client/events_returns_a_collection_of_event_objects.yml +124 -0
- data/spec/vcr/binnacle_client/recents_invokes_the_events_api_recents.yml +83 -0
- data/spec/vcr/binnacle_client/recents_returns_a_collection_of_event_objects.yml +83 -0
- data/spec/vcr/binnacle_client/report_exception_invokes_the_events_api_signal.yml +121 -0
- data/spec/vcr/binnacle_client/signal_asynch_invokes_the_events_api_signal.yml +74 -0
- data/spec/vcr/binnacle_client/signal_invokes_the_events_api_signal.yml +74 -0
- data/spec/vcr/binnacle_client_ready_/returns_true_if_a_connection_has_been_successfully_established.yml +42 -0
- data/spec/vcr/binnacle_command/tail_command_with_n_flag_returns_recent_events.yml +81 -0
- data/spec/vcr/binnacle_connection/initialize_fails_with_incorrect_credentials.yml +40 -0
- data/spec/vcr/binnacle_connection/initialize_retrieves_available_endpoints_upon_successful_connection.yml +42 -0
- data/spec/vcr/binnacle_http_logger/_htt_party_adapter_should_log__ge_t_requests.yml +122 -0
- data/spec/vcr/binnacle_http_logger/_http_adapter_should_log__ge_t_requests.yml +165 -0
- data/spec/vcr/binnacle_http_logger/_http_client_adapter_should_log__ge_t_requests.yml +274 -0
- data/spec/vcr/binnacle_http_logger/ethon_adapter_should_log__ge_t_requests.yml +87 -0
- data/spec/vcr/binnacle_http_logger/excon_adapter_should_log__ge_t_requests.yml +204 -0
- data/spec/vcr/binnacle_http_logger/faraday_adapter_should_log__ge_t_requests.yml +200 -0
- data/spec/vcr/binnacle_http_logger/net_http_adapter_should_log__ge_t_requests.yml +827 -0
- data/spec/vcr/binnacle_http_logger/open_uri_adapter_should_log__ge_t_requests.yml +751 -0
- data/spec/vcr/binnacle_http_logger/patron_adapter_should_log__ge_t_requests.yml +124 -0
- data/spec/vcr/binnacle_http_logger/typhoeus_adapter_should_log__ge_t_requests.yml +124 -0
- data/spec/vcr/binnacle_logging/logging_allows_passing_other_parameters_using_a_hash.yml +119 -0
- data/spec/vcr/binnacle_logging/logging_invokes_the_events_api_signal.yml +75 -0
- data/spec/vcr/binnacle_logging/logging_respects_the_logger_severity.yml +81 -0
- data/vendor/assets/javascripts/atmosphere/atmosphere.js +3487 -0
- data/vendor/assets/javascripts/base64/base64.js +61 -0
- data/vendor/assets/javascripts/binnacle.js +5 -0
- data/vendor/assets/javascripts/binnacle/binnacle.js +9090 -0
- data/vendor/assets/javascripts/firebase/firebase-app-externs.js +261 -0
- data/vendor/assets/javascripts/firebase/firebase-app.js +35 -0
- data/vendor/assets/javascripts/firebase/firebase-auth-externs.js +1300 -0
- data/vendor/assets/javascripts/firebase/firebase-auth.js +241 -0
- data/vendor/assets/javascripts/firebase/firebase-database-externs.js +1700 -0
- data/vendor/assets/javascripts/firebase/firebase-database.js +260 -0
- data/vendor/assets/javascripts/firebase/firebase-messaging-externs.js +164 -0
- data/vendor/assets/javascripts/firebase/firebase-messaging.js +35 -0
- data/vendor/assets/javascripts/firebase/firebase-storage-externs.js +663 -0
- data/vendor/assets/javascripts/firebase/firebase-storage.js +52 -0
- data/vendor/assets/javascripts/firebase/firebase.js +611 -0
- data/vendor/assets/javascripts/moment/locale/af.js +73 -0
- data/vendor/assets/javascripts/moment/locale/ar-dz.js +59 -0
- data/vendor/assets/javascripts/moment/locale/ar-kw.js +59 -0
- data/vendor/assets/javascripts/moment/locale/ar-ly.js +126 -0
- data/vendor/assets/javascripts/moment/locale/ar-ma.js +60 -0
- data/vendor/assets/javascripts/moment/locale/ar-sa.js +105 -0
- data/vendor/assets/javascripts/moment/locale/ar-tn.js +59 -0
- data/vendor/assets/javascripts/moment/locale/ar.js +142 -0
- data/vendor/assets/javascripts/moment/locale/az.js +105 -0
- data/vendor/assets/javascripts/moment/locale/be.js +134 -0
- data/vendor/assets/javascripts/moment/locale/bg.js +90 -0
- data/vendor/assets/javascripts/moment/locale/bn.js +119 -0
- data/vendor/assets/javascripts/moment/locale/bo.js +119 -0
- data/vendor/assets/javascripts/moment/locale/br.js +108 -0
- data/vendor/assets/javascripts/moment/locale/bs.js +143 -0
- data/vendor/assets/javascripts/moment/locale/ca.js +88 -0
- data/vendor/assets/javascripts/moment/locale/cs.js +172 -0
- data/vendor/assets/javascripts/moment/locale/cv.js +63 -0
- data/vendor/assets/javascripts/moment/locale/cy.js +81 -0
- data/vendor/assets/javascripts/moment/locale/da.js +60 -0
- data/vendor/assets/javascripts/moment/locale/de-at.js +79 -0
- data/vendor/assets/javascripts/moment/locale/de-ch.js +78 -0
- data/vendor/assets/javascripts/moment/locale/de.js +78 -0
- data/vendor/assets/javascripts/moment/locale/dv.js +100 -0
- data/vendor/assets/javascripts/moment/locale/el.js +100 -0
- data/vendor/assets/javascripts/moment/locale/en-au.js +67 -0
- data/vendor/assets/javascripts/moment/locale/en-ca.js +63 -0
- data/vendor/assets/javascripts/moment/locale/en-gb.js +67 -0
- data/vendor/assets/javascripts/moment/locale/en-ie.js +67 -0
- data/vendor/assets/javascripts/moment/locale/en-nz.js +67 -0
- data/vendor/assets/javascripts/moment/locale/eo.js +73 -0
- data/vendor/assets/javascripts/moment/locale/es-do.js +82 -0
- data/vendor/assets/javascripts/moment/locale/es.js +83 -0
- data/vendor/assets/javascripts/moment/locale/et.js +80 -0
- data/vendor/assets/javascripts/moment/locale/eu.js +66 -0
- data/vendor/assets/javascripts/moment/locale/fa.js +107 -0
- data/vendor/assets/javascripts/moment/locale/fi.js +107 -0
- data/vendor/assets/javascripts/moment/locale/fo.js +60 -0
- data/vendor/assets/javascripts/moment/locale/fr-ca.js +74 -0
- data/vendor/assets/javascripts/moment/locale/fr-ch.js +78 -0
- data/vendor/assets/javascripts/moment/locale/fr.js +83 -0
- data/vendor/assets/javascripts/moment/locale/fy.js +75 -0
- data/vendor/assets/javascripts/moment/locale/gd.js +76 -0
- data/vendor/assets/javascripts/moment/locale/gl.js +77 -0
- data/vendor/assets/javascripts/moment/locale/gom-latn.js +122 -0
- data/vendor/assets/javascripts/moment/locale/he.js +99 -0
- data/vendor/assets/javascripts/moment/locale/hi.js +124 -0
- data/vendor/assets/javascripts/moment/locale/hr.js +145 -0
- data/vendor/assets/javascripts/moment/locale/hu.js +109 -0
- data/vendor/assets/javascripts/moment/locale/hy-am.js +95 -0
- data/vendor/assets/javascripts/moment/locale/id.js +83 -0
- data/vendor/assets/javascripts/moment/locale/is.js +127 -0
- data/vendor/assets/javascripts/moment/locale/it.js +70 -0
- data/vendor/assets/javascripts/moment/locale/ja.js +80 -0
- data/vendor/assets/javascripts/moment/locale/jv.js +83 -0
- data/vendor/assets/javascripts/moment/locale/ka.js +89 -0
- data/vendor/assets/javascripts/moment/locale/kk.js +87 -0
- data/vendor/assets/javascripts/moment/locale/km.js +58 -0
- data/vendor/assets/javascripts/moment/locale/kn.js +126 -0
- data/vendor/assets/javascripts/moment/locale/ko.js +69 -0
- data/vendor/assets/javascripts/moment/locale/ky.js +88 -0
- data/vendor/assets/javascripts/moment/locale/lb.js +137 -0
- data/vendor/assets/javascripts/moment/locale/lo.js +70 -0
- data/vendor/assets/javascripts/moment/locale/lt.js +117 -0
- data/vendor/assets/javascripts/moment/locale/lv.js +97 -0
- data/vendor/assets/javascripts/moment/locale/me.js +111 -0
- data/vendor/assets/javascripts/moment/locale/mi.js +64 -0
- data/vendor/assets/javascripts/moment/locale/mk.js +90 -0
- data/vendor/assets/javascripts/moment/locale/ml.js +81 -0
- data/vendor/assets/javascripts/moment/locale/mr.js +159 -0
- data/vendor/assets/javascripts/moment/locale/ms-my.js +83 -0
- data/vendor/assets/javascripts/moment/locale/ms.js +82 -0
- data/vendor/assets/javascripts/moment/locale/my.js +96 -0
- data/vendor/assets/javascripts/moment/locale/nb.js +63 -0
- data/vendor/assets/javascripts/moment/locale/ne.js +123 -0
- data/vendor/assets/javascripts/moment/locale/nl-be.js +88 -0
- data/vendor/assets/javascripts/moment/locale/nl.js +88 -0
- data/vendor/assets/javascripts/moment/locale/nn.js +60 -0
- data/vendor/assets/javascripts/moment/locale/pa-in.js +124 -0
- data/vendor/assets/javascripts/moment/locale/pl.js +107 -0
- data/vendor/assets/javascripts/moment/locale/pt-br.js +61 -0
- data/vendor/assets/javascripts/moment/locale/pt.js +65 -0
- data/vendor/assets/javascripts/moment/locale/ro.js +75 -0
- data/vendor/assets/javascripts/moment/locale/ru.js +183 -0
- data/vendor/assets/javascripts/moment/locale/sd.js +98 -0
- data/vendor/assets/javascripts/moment/locale/se.js +61 -0
- data/vendor/assets/javascripts/moment/locale/si.js +71 -0
- data/vendor/assets/javascripts/moment/locale/sk.js +150 -0
- data/vendor/assets/javascripts/moment/locale/sl.js +162 -0
- data/vendor/assets/javascripts/moment/locale/sq.js +70 -0
- data/vendor/assets/javascripts/moment/locale/sr-cyrl.js +110 -0
- data/vendor/assets/javascripts/moment/locale/sr.js +110 -0
- data/vendor/assets/javascripts/moment/locale/ss.js +89 -0
- data/vendor/assets/javascripts/moment/locale/sv.js +69 -0
- data/vendor/assets/javascripts/moment/locale/sw.js +59 -0
- data/vendor/assets/javascripts/moment/locale/ta.js +130 -0
- data/vendor/assets/javascripts/moment/locale/te.js +89 -0
- data/vendor/assets/javascripts/moment/locale/tet.js +68 -0
- data/vendor/assets/javascripts/moment/locale/th.js +67 -0
- data/vendor/assets/javascripts/moment/locale/tl-ph.js +62 -0
- data/vendor/assets/javascripts/moment/locale/tlh.js +120 -0
- data/vendor/assets/javascripts/moment/locale/tr.js +90 -0
- data/vendor/assets/javascripts/moment/locale/tzl.js +91 -0
- data/vendor/assets/javascripts/moment/locale/tzm-latn.js +58 -0
- data/vendor/assets/javascripts/moment/locale/tzm.js +58 -0
- data/vendor/assets/javascripts/moment/locale/uk.js +151 -0
- data/vendor/assets/javascripts/moment/locale/ur.js +99 -0
- data/vendor/assets/javascripts/moment/locale/uz-latn.js +58 -0
- data/vendor/assets/javascripts/moment/locale/uz.js +58 -0
- data/vendor/assets/javascripts/moment/locale/vi.js +79 -0
- data/vendor/assets/javascripts/moment/locale/x-pseudo.js +68 -0
- data/vendor/assets/javascripts/moment/locale/yo.js +60 -0
- data/vendor/assets/javascripts/moment/locale/zh-cn.js +111 -0
- data/vendor/assets/javascripts/moment/locale/zh-hk.js +105 -0
- data/vendor/assets/javascripts/moment/locale/zh-tw.js +104 -0
- data/vendor/assets/javascripts/moment/min/locales.js +9252 -0
- data/vendor/assets/javascripts/moment/min/moment-with-locales.js +13700 -0
- data/vendor/assets/javascripts/moment/min/moment.min.js +7 -0
- data/vendor/assets/javascripts/moment/min/tests.js +78265 -0
- data/vendor/assets/javascripts/moment/moment.js +4463 -0
- data/vendor/assets/javascripts/moment/src/lib/create/check-overflow.js +34 -0
- data/vendor/assets/javascripts/moment/src/lib/create/date-from-array.js +21 -0
- data/vendor/assets/javascripts/moment/src/lib/create/from-anything.js +110 -0
- data/vendor/assets/javascripts/moment/src/lib/create/from-array.js +140 -0
- data/vendor/assets/javascripts/moment/src/lib/create/from-object.js +16 -0
- data/vendor/assets/javascripts/moment/src/lib/create/from-string-and-array.js +50 -0
- data/vendor/assets/javascripts/moment/src/lib/create/from-string-and-format.js +113 -0
- data/vendor/assets/javascripts/moment/src/lib/create/from-string.js +202 -0
- data/vendor/assets/javascripts/moment/src/lib/create/local.js +5 -0
- data/vendor/assets/javascripts/moment/src/lib/create/parsing-flags.js +26 -0
- data/vendor/assets/javascripts/moment/src/lib/create/utc.js +5 -0
- data/vendor/assets/javascripts/moment/src/lib/create/valid.js +49 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/abs.js +18 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/add-subtract.js +21 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/as.js +61 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/bubble.js +61 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/constructor.js +44 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/create.js +122 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/duration.js +16 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/get.js +25 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/humanize.js +85 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/iso-string.js +56 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/prototype.js +50 -0
- data/vendor/assets/javascripts/moment/src/lib/duration/valid.js +35 -0
- data/vendor/assets/javascripts/moment/src/lib/format/format.js +92 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/base-config.js +44 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/calendar.js +15 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/constructor.js +5 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/en.js +15 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/formats.js +23 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/invalid.js +5 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/lists.js +93 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/locale.js +39 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/locales.js +186 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/ordinal.js +7 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/pre-post-format.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/prototype.js +69 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/relative.js +30 -0
- data/vendor/assets/javascripts/moment/src/lib/locale/set.js +49 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/add-subtract.js +55 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/calendar.js +26 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/clone.js +5 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/compare.js +59 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/constructor.js +77 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/creation-data.js +9 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/diff.js +62 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/format.js +57 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/from.js +17 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/get-set.js +55 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/locale.js +34 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/min-max.js +63 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/moment.js +28 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/now.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/prototype.js +150 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/start-end-of.js +59 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/to-type.js +34 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/to.js +17 -0
- data/vendor/assets/javascripts/moment/src/lib/moment/valid.js +15 -0
- data/vendor/assets/javascripts/moment/src/lib/parse/regex.js +54 -0
- data/vendor/assets/javascripts/moment/src/lib/parse/token.js +33 -0
- data/vendor/assets/javascripts/moment/src/lib/units/aliases.js +30 -0
- data/vendor/assets/javascripts/moment/src/lib/units/constants.js +9 -0
- data/vendor/assets/javascripts/moment/src/lib/units/day-of-month.js +39 -0
- data/vendor/assets/javascripts/moment/src/lib/units/day-of-week.js +364 -0
- data/vendor/assets/javascripts/moment/src/lib/units/day-of-year.js +36 -0
- data/vendor/assets/javascripts/moment/src/lib/units/hour.js +144 -0
- data/vendor/assets/javascripts/moment/src/lib/units/millisecond.js +69 -0
- data/vendor/assets/javascripts/moment/src/lib/units/minute.js +29 -0
- data/vendor/assets/javascripts/moment/src/lib/units/month.js +283 -0
- data/vendor/assets/javascripts/moment/src/lib/units/offset.js +235 -0
- data/vendor/assets/javascripts/moment/src/lib/units/priorities.js +16 -0
- data/vendor/assets/javascripts/moment/src/lib/units/quarter.js +32 -0
- data/vendor/assets/javascripts/moment/src/lib/units/second.js +29 -0
- data/vendor/assets/javascripts/moment/src/lib/units/timestamp.js +20 -0
- data/vendor/assets/javascripts/moment/src/lib/units/timezone.js +16 -0
- data/vendor/assets/javascripts/moment/src/lib/units/units.js +20 -0
- data/vendor/assets/javascripts/moment/src/lib/units/week-calendar-utils.js +65 -0
- data/vendor/assets/javascripts/moment/src/lib/units/week-year.js +107 -0
- data/vendor/assets/javascripts/moment/src/lib/units/week.js +67 -0
- data/vendor/assets/javascripts/moment/src/lib/units/year.js +75 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/abs-ceil.js +7 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/abs-floor.js +8 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/abs-round.js +7 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/compare-arrays.js +16 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/defaults.js +10 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/deprecate.js +55 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/extend.js +19 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/has-own-prop.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/hooks.js +13 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/index-of.js +18 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-array.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-date.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-function.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-number.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-object-empty.js +8 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-object.js +5 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/is-undefined.js +3 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/keys.js +19 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/map.js +7 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/some.js +19 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/to-int.js +12 -0
- data/vendor/assets/javascripts/moment/src/lib/utils/zero-fill.js +7 -0
- data/vendor/assets/javascripts/moment/src/locale/af.js +63 -0
- data/vendor/assets/javascripts/moment/src/locale/ar-dz.js +50 -0
- data/vendor/assets/javascripts/moment/src/locale/ar-kw.js +49 -0
- data/vendor/assets/javascripts/moment/src/locale/ar-ly.js +112 -0
- data/vendor/assets/javascripts/moment/src/locale/ar-ma.js +51 -0
- data/vendor/assets/javascripts/moment/src/locale/ar-sa.js +95 -0
- data/vendor/assets/javascripts/moment/src/locale/ar-tn.js +50 -0
- data/vendor/assets/javascripts/moment/src/locale/ar.js +128 -0
- data/vendor/assets/javascripts/moment/src/locale/az.js +96 -0
- data/vendor/assets/javascripts/moment/src/locale/be.js +125 -0
- data/vendor/assets/javascripts/moment/src/locale/bg.js +81 -0
- data/vendor/assets/javascripts/moment/src/locale/bn.js +109 -0
- data/vendor/assets/javascripts/moment/src/locale/bo.js +110 -0
- data/vendor/assets/javascripts/moment/src/locale/br.js +99 -0
- data/vendor/assets/javascripts/moment/src/locale/bs.js +133 -0
- data/vendor/assets/javascripts/moment/src/locale/ca.js +79 -0
- data/vendor/assets/javascripts/moment/src/locale/cs.js +163 -0
- data/vendor/assets/javascripts/moment/src/locale/cv.js +53 -0
- data/vendor/assets/javascripts/moment/src/locale/cy.js +72 -0
- data/vendor/assets/javascripts/moment/src/locale/da.js +50 -0
- data/vendor/assets/javascripts/moment/src/locale/de-at.js +69 -0
- data/vendor/assets/javascripts/moment/src/locale/de-ch.js +68 -0
- data/vendor/assets/javascripts/moment/src/locale/de.js +68 -0
- data/vendor/assets/javascripts/moment/src/locale/dv.js +89 -0
- data/vendor/assets/javascripts/moment/src/locale/el.js +88 -0
- data/vendor/assets/javascripts/moment/src/locale/en-au.js +58 -0
- data/vendor/assets/javascripts/moment/src/locale/en-ca.js +53 -0
- data/vendor/assets/javascripts/moment/src/locale/en-gb.js +58 -0
- data/vendor/assets/javascripts/moment/src/locale/en-ie.js +58 -0
- data/vendor/assets/javascripts/moment/src/locale/en-nz.js +57 -0
- data/vendor/assets/javascripts/moment/src/locale/eo.js +64 -0
- data/vendor/assets/javascripts/moment/src/locale/es-do.js +73 -0
- data/vendor/assets/javascripts/moment/src/locale/es.js +74 -0
- data/vendor/assets/javascripts/moment/src/locale/et.js +71 -0
- data/vendor/assets/javascripts/moment/src/locale/eu.js +57 -0
- data/vendor/assets/javascripts/moment/src/locale/fa.js +97 -0
- data/vendor/assets/javascripts/moment/src/locale/fi.js +98 -0
- data/vendor/assets/javascripts/moment/src/locale/fo.js +51 -0
- data/vendor/assets/javascripts/moment/src/locale/fr-ca.js +65 -0
- data/vendor/assets/javascripts/moment/src/locale/fr-ch.js +69 -0
- data/vendor/assets/javascripts/moment/src/locale/fr.js +74 -0
- data/vendor/assets/javascripts/moment/src/locale/fy.js +66 -0
- data/vendor/assets/javascripts/moment/src/locale/gd.js +67 -0
- data/vendor/assets/javascripts/moment/src/locale/gl.js +68 -0
- data/vendor/assets/javascripts/moment/src/locale/gom-latn.js +112 -0
- data/vendor/assets/javascripts/moment/src/locale/he.js +90 -0
- data/vendor/assets/javascripts/moment/src/locale/hi.js +115 -0
- data/vendor/assets/javascripts/moment/src/locale/hr.js +135 -0
- data/vendor/assets/javascripts/moment/src/locale/hu.js +100 -0
- data/vendor/assets/javascripts/moment/src/locale/hy-am.js +86 -0
- data/vendor/assets/javascripts/moment/src/locale/id.js +74 -0
- data/vendor/assets/javascripts/moment/src/locale/is.js +118 -0
- data/vendor/assets/javascripts/moment/src/locale/it.js +61 -0
- data/vendor/assets/javascripts/moment/src/locale/ja.js +71 -0
- data/vendor/assets/javascripts/moment/src/locale/jv.js +73 -0
- data/vendor/assets/javascripts/moment/src/locale/ka.js +80 -0
- data/vendor/assets/javascripts/moment/src/locale/kk.js +77 -0
- data/vendor/assets/javascripts/moment/src/locale/km.js +49 -0
- data/vendor/assets/javascripts/moment/src/locale/kn.js +116 -0
- data/vendor/assets/javascripts/moment/src/locale/ko.js +60 -0
- data/vendor/assets/javascripts/moment/src/locale/ky.js +78 -0
- data/vendor/assets/javascripts/moment/src/locale/lb.js +128 -0
- data/vendor/assets/javascripts/moment/src/locale/lo.js +61 -0
- data/vendor/assets/javascripts/moment/src/locale/lt.js +108 -0
- data/vendor/assets/javascripts/moment/src/locale/lv.js +88 -0
- data/vendor/assets/javascripts/moment/src/locale/me.js +101 -0
- data/vendor/assets/javascripts/moment/src/locale/mi.js +54 -0
- data/vendor/assets/javascripts/moment/src/locale/mk.js +81 -0
- data/vendor/assets/javascripts/moment/src/locale/ml.js +72 -0
- data/vendor/assets/javascripts/moment/src/locale/mr.js +150 -0
- data/vendor/assets/javascripts/moment/src/locale/ms-my.js +74 -0
- data/vendor/assets/javascripts/moment/src/locale/ms.js +73 -0
- data/vendor/assets/javascripts/moment/src/locale/my.js +86 -0
- data/vendor/assets/javascripts/moment/src/locale/nb.js +54 -0
- data/vendor/assets/javascripts/moment/src/locale/ne.js +114 -0
- data/vendor/assets/javascripts/moment/src/locale/nl-be.js +79 -0
- data/vendor/assets/javascripts/moment/src/locale/nl.js +79 -0
- data/vendor/assets/javascripts/moment/src/locale/nn.js +51 -0
- data/vendor/assets/javascripts/moment/src/locale/pa-in.js +115 -0
- data/vendor/assets/javascripts/moment/src/locale/pl.js +97 -0
- data/vendor/assets/javascripts/moment/src/locale/pt-br.js +52 -0
- data/vendor/assets/javascripts/moment/src/locale/pt.js +56 -0
- data/vendor/assets/javascripts/moment/src/locale/ro.js +66 -0
- data/vendor/assets/javascripts/moment/src/locale/ru.js +173 -0
- data/vendor/assets/javascripts/moment/src/locale/sd.js +88 -0
- data/vendor/assets/javascripts/moment/src/locale/se.js +51 -0
- data/vendor/assets/javascripts/moment/src/locale/si.js +61 -0
- data/vendor/assets/javascripts/moment/src/locale/sk.js +141 -0
- data/vendor/assets/javascripts/moment/src/locale/sl.js +152 -0
- data/vendor/assets/javascripts/moment/src/locale/sq.js +61 -0
- data/vendor/assets/javascripts/moment/src/locale/sr-cyrl.js +100 -0
- data/vendor/assets/javascripts/moment/src/locale/sr.js +100 -0
- data/vendor/assets/javascripts/moment/src/locale/ss.js +80 -0
- data/vendor/assets/javascripts/moment/src/locale/sv.js +60 -0
- data/vendor/assets/javascripts/moment/src/locale/sw.js +50 -0
- data/vendor/assets/javascripts/moment/src/locale/ta.js +120 -0
- data/vendor/assets/javascripts/moment/src/locale/te.js +79 -0
- data/vendor/assets/javascripts/moment/src/locale/tet.js +58 -0
- data/vendor/assets/javascripts/moment/src/locale/th.js +57 -0
- data/vendor/assets/javascripts/moment/src/locale/tl-ph.js +53 -0
- data/vendor/assets/javascripts/moment/src/locale/tlh.js +110 -0
- data/vendor/assets/javascripts/moment/src/locale/tr.js +81 -0
- data/vendor/assets/javascripts/moment/src/locale/tzl.js +82 -0
- data/vendor/assets/javascripts/moment/src/locale/tzm-latn.js +49 -0
- data/vendor/assets/javascripts/moment/src/locale/tzm.js +49 -0
- data/vendor/assets/javascripts/moment/src/locale/uk.js +142 -0
- data/vendor/assets/javascripts/moment/src/locale/ur.js +89 -0
- data/vendor/assets/javascripts/moment/src/locale/uz-latn.js +49 -0
- data/vendor/assets/javascripts/moment/src/locale/uz.js +49 -0
- data/vendor/assets/javascripts/moment/src/locale/vi.js +70 -0
- data/vendor/assets/javascripts/moment/src/locale/x-pseudo.js +58 -0
- data/vendor/assets/javascripts/moment/src/locale/yo.js +50 -0
- data/vendor/assets/javascripts/moment/src/locale/zh-cn.js +102 -0
- data/vendor/assets/javascripts/moment/src/locale/zh-hk.js +95 -0
- data/vendor/assets/javascripts/moment/src/locale/zh-tw.js +94 -0
- data/vendor/assets/javascripts/moment/src/moment.js +82 -0
- data/vendor/assets/javascripts/moment/templates/default.js +5 -0
- data/vendor/assets/javascripts/moment/templates/locale-header.js +6 -0
- data/vendor/assets/javascripts/moment/templates/test-header.js +6 -0
- metadata +943 -0
@@ -0,0 +1,120 @@
|
|
1
|
+
require 'json'
|
2
|
+
require 'action_pack'
|
3
|
+
require 'active_support/core_ext/class/attribute'
|
4
|
+
require 'active_support/log_subscriber'
|
5
|
+
|
6
|
+
module Binnacle
|
7
|
+
module Logging
|
8
|
+
class RequestLogSubscriber < ActiveSupport::LogSubscriber
|
9
|
+
def process_action(event)
|
10
|
+
return if Binnacle.configuration.ignore?(event)
|
11
|
+
payload = event.payload
|
12
|
+
data = extract_request(event, payload)
|
13
|
+
Binnacle.client.log_rails_event(data)
|
14
|
+
end
|
15
|
+
|
16
|
+
def redirect_to(event)
|
17
|
+
Thread.current[:binnacle_location] = event.payload[:location]
|
18
|
+
end
|
19
|
+
|
20
|
+
# TODO: Implement send_file and send_data
|
21
|
+
# def send_file(event)
|
22
|
+
# info { "Sent data #{event.payload[:filename]} (#{event.duration.round(1)}ms)" }
|
23
|
+
# end
|
24
|
+
#
|
25
|
+
# def send_data(event)
|
26
|
+
# info { "Sent data #{event.payload[:filename]} (#{event.duration.round(1)}ms)" }
|
27
|
+
# end
|
28
|
+
|
29
|
+
def unpermitted_parameters(event)
|
30
|
+
Thread.current[:binnacle_unpermitted_params] ||= []
|
31
|
+
Thread.current[:binnacle_unpermitted_params].concat(event.payload[:keys])
|
32
|
+
end
|
33
|
+
|
34
|
+
private
|
35
|
+
|
36
|
+
def extract_request(event, payload)
|
37
|
+
payload = event.payload
|
38
|
+
data = initial_data(payload)
|
39
|
+
data.merge!(extract_status(payload))
|
40
|
+
data.merge!(extract_runtimes(event, payload))
|
41
|
+
data.merge!(extract_location)
|
42
|
+
data.merge!(extract_unpermitted_params)
|
43
|
+
data.merge!(extract_event_details(event))
|
44
|
+
data.merge!({message: "#{data[:method]} #{data[:path]} AS #{data[:format]} (view: #{data[:view]}ms, db: #{data[:db]}ms)"})
|
45
|
+
end
|
46
|
+
|
47
|
+
def initial_data(payload)
|
48
|
+
path = extract_path(payload)
|
49
|
+
method = payload[:method]
|
50
|
+
format = extract_format(payload)
|
51
|
+
{
|
52
|
+
direction: :in,
|
53
|
+
method: method,
|
54
|
+
path: path,
|
55
|
+
format: format,
|
56
|
+
controller: payload[:params]['controller'],
|
57
|
+
action: payload[:params]['action'],
|
58
|
+
params: payload[:params],
|
59
|
+
}
|
60
|
+
end
|
61
|
+
|
62
|
+
def extract_path(payload)
|
63
|
+
path = payload[:path]
|
64
|
+
index = path.index('?')
|
65
|
+
index ? path[0, index] : path
|
66
|
+
end
|
67
|
+
|
68
|
+
def extract_event_details(event)
|
69
|
+
{
|
70
|
+
time: event.time,
|
71
|
+
transaction_id: event.transaction_id
|
72
|
+
}
|
73
|
+
end
|
74
|
+
|
75
|
+
def extract_format(payload)
|
76
|
+
(::ActionPack::VERSION::MAJOR == 3 && ::ActionPack::VERSION::MINOR == 0) ? payload[:formats].first : payload[:format]
|
77
|
+
end
|
78
|
+
|
79
|
+
def extract_status(payload)
|
80
|
+
if (status = payload[:status])
|
81
|
+
{ status: status.to_i }
|
82
|
+
elsif (error = payload[:exception])
|
83
|
+
exception, message = error
|
84
|
+
{ status: get_error_status_code(exception), error: "#{exception}: #{message}" }
|
85
|
+
else
|
86
|
+
{ status: 0 }
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
def get_error_status_code(exception)
|
91
|
+
status = ActionDispatch::ExceptionWrapper.rescue_responses[exception]
|
92
|
+
Rack::Utils.status_code(status)
|
93
|
+
end
|
94
|
+
|
95
|
+
def extract_runtimes(event, payload)
|
96
|
+
data = { duration: event.duration.to_f.round(2) }
|
97
|
+
data[:view] = payload[:view_runtime].to_f.round(2) if payload.key?(:view_runtime)
|
98
|
+
data[:db] = payload[:db_runtime].to_f.round(2) if payload.key?(:db_runtime)
|
99
|
+
data
|
100
|
+
end
|
101
|
+
|
102
|
+
def extract_location
|
103
|
+
location = Thread.current[:binnacle_location]
|
104
|
+
return {} unless location
|
105
|
+
|
106
|
+
Thread.current[:binnacle_location] = nil
|
107
|
+
{ location: location }
|
108
|
+
end
|
109
|
+
|
110
|
+
def extract_unpermitted_params
|
111
|
+
unpermitted_params = Thread.current[:binnacle_unpermitted_params]
|
112
|
+
return {} unless unpermitted_params
|
113
|
+
|
114
|
+
Thread.current[:binnacle_unpermitted_params] = nil
|
115
|
+
{ unpermitted_params: unpermitted_params }
|
116
|
+
end
|
117
|
+
|
118
|
+
end
|
119
|
+
end
|
120
|
+
end
|
@@ -0,0 +1,54 @@
|
|
1
|
+
require 'json'
|
2
|
+
|
3
|
+
module Binnacle
|
4
|
+
class Resource
|
5
|
+
attr_reader :route
|
6
|
+
attr_writer :connection
|
7
|
+
|
8
|
+
def post_asynch
|
9
|
+
Thread.new do
|
10
|
+
response = response_from_post(@connection, self.route, self.to_json)
|
11
|
+
|
12
|
+
if response.status == 401
|
13
|
+
Binnacle.binnacle_logger.error("Error communicating with Binnacle: #{response.body}")
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
def post
|
19
|
+
response = response_from_post(@connection, self.route, self.to_json)
|
20
|
+
|
21
|
+
if response.status == 401
|
22
|
+
Binnacle.binnacle_logger.error("Error communicating with Binnacle: #{response.body}")
|
23
|
+
else
|
24
|
+
JSON.parse(response.body)
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
def self.get(connection, path, query_params)
|
29
|
+
response = connection.get do |req|
|
30
|
+
req.url path
|
31
|
+
req.headers['Content-Type'] = 'application/json'
|
32
|
+
query_params.each do |n,v|
|
33
|
+
req.params[n] = v if v
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
if response.status == 401
|
38
|
+
Binnacle.binnacle_logger.error("Error communicating with Binnacle: #{response.body}")
|
39
|
+
else
|
40
|
+
JSON.parse(response.body).map { |r| self.from_hash(r) }
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
protected
|
45
|
+
|
46
|
+
def response_from_post(connection, route, body)
|
47
|
+
connection.post do |req|
|
48
|
+
req.url route
|
49
|
+
req.headers['Content-Type'] = 'application/json'
|
50
|
+
req.body = body
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
@@ -0,0 +1,100 @@
|
|
1
|
+
module Binnacle
|
2
|
+
class Event < Resource
|
3
|
+
|
4
|
+
attr_accessor :channel_id
|
5
|
+
attr_accessor :event_name
|
6
|
+
attr_accessor :client_id
|
7
|
+
attr_accessor :session_id
|
8
|
+
attr_accessor :client_event_time
|
9
|
+
attr_accessor :ip_address
|
10
|
+
attr_accessor :log_level
|
11
|
+
attr_accessor :tags
|
12
|
+
attr_accessor :json
|
13
|
+
attr_accessor :event_time
|
14
|
+
|
15
|
+
def configure(channel_id, event_name, client_id, session_id, log_level, ts = Time.now, tags = [], json = {})
|
16
|
+
self.channel_id = channel_id
|
17
|
+
self.event_name = event_name
|
18
|
+
self.client_id = client_id
|
19
|
+
self.session_id = session_id
|
20
|
+
self.timestamp = ts ? ts : Time.now
|
21
|
+
self.log_level = log_level
|
22
|
+
self.tags = tags
|
23
|
+
self.json = json
|
24
|
+
end
|
25
|
+
|
26
|
+
def configure_from_logging_progname(progname, channel_id, client_id, session_id, log_level, ts = Time.now, tags = [], json = {})
|
27
|
+
if progname.is_a?(Hash)
|
28
|
+
self.client_id = progname[:client_id] || client_id
|
29
|
+
self.session_id = progname[:session_id] || session_id
|
30
|
+
self.channel_id = progname[:channel_id] || channel_id
|
31
|
+
self.event_name = progname[:event_name]
|
32
|
+
self.tags = progname[:tags] || tags
|
33
|
+
self.json = json
|
34
|
+
self.json.merge!(progname[:json]) if progname[:json]
|
35
|
+
elsif progname.is_a?(String)
|
36
|
+
self.client_id = client_id
|
37
|
+
self.session_id = session_id
|
38
|
+
self.channel_id = channel_id
|
39
|
+
self.event_name = progname
|
40
|
+
self.tags = tags
|
41
|
+
self.json = json
|
42
|
+
end
|
43
|
+
|
44
|
+
self.timestamp = ts ? ts : Time.now
|
45
|
+
self.log_level = log_level
|
46
|
+
end
|
47
|
+
|
48
|
+
def timestamp=(ts)
|
49
|
+
self.client_event_time = ts.strftime("%Y-%m-%dT%H:%M:%S%z")
|
50
|
+
end
|
51
|
+
|
52
|
+
def self.from_hash(h)
|
53
|
+
event = self.new()
|
54
|
+
event.channel_id = h['channelId']
|
55
|
+
event.event_name = h['eventName']
|
56
|
+
event.client_id = h['clientId']
|
57
|
+
event.session_id = h['sessionId']
|
58
|
+
event.ip_address = h['ipAddress']
|
59
|
+
event.log_level = h['logLevel']
|
60
|
+
event.event_time = Time.at(h['eventTime']/1000)
|
61
|
+
event.tags = h['tags']
|
62
|
+
event.json = h['json']
|
63
|
+
|
64
|
+
event
|
65
|
+
end
|
66
|
+
|
67
|
+
def to_json
|
68
|
+
{
|
69
|
+
"channelId" => channel_id,
|
70
|
+
"sessionId" => session_id,
|
71
|
+
"clientEventTime" => client_event_time,
|
72
|
+
"eventName" => event_name,
|
73
|
+
"clientId" => client_id,
|
74
|
+
"logLevel" => log_level,
|
75
|
+
"tags" => tags,
|
76
|
+
"json" => json
|
77
|
+
}.to_json
|
78
|
+
end
|
79
|
+
|
80
|
+
def route
|
81
|
+
"/api/events/#{channel_id}"
|
82
|
+
end
|
83
|
+
|
84
|
+
def self.route(channel)
|
85
|
+
"/api/events/#{channel}"
|
86
|
+
end
|
87
|
+
|
88
|
+
def self.recents(connection, lines, since, channel)
|
89
|
+
path = [route(channel), 'recents'].compact.join('/')
|
90
|
+
|
91
|
+
get(connection, path, {'limit' => lines, 'since' => since})
|
92
|
+
end
|
93
|
+
|
94
|
+
def self.events(connection, channel, date, start_hour, end_hour, lines)
|
95
|
+
path = [route(channel), date].compact.join('/')
|
96
|
+
|
97
|
+
get(connection, path, {'start_hour' => start_hour, 'end_hour' => end_hour, 'limit' => lines})
|
98
|
+
end
|
99
|
+
end
|
100
|
+
end
|
@@ -0,0 +1,130 @@
|
|
1
|
+
module Binnacle
|
2
|
+
module Trap
|
3
|
+
# Front end to parsing the backtrace for each notice
|
4
|
+
class Backtrace
|
5
|
+
# --------------------------------------------------------------
|
6
|
+
# Attribution:
|
7
|
+
# Copied from Airbrake Gem. https://github.com/airbrake/airbrake
|
8
|
+
# --------------------------------------------------------------
|
9
|
+
|
10
|
+
# Handles backtrace parsing line by line
|
11
|
+
class Line
|
12
|
+
|
13
|
+
GEM_PATHS = (Gem.path | [Gem.default_dir]).map { |p| Regexp.escape(p) }
|
14
|
+
GEMS_REGEXP = %r{(#{GEM_PATHS.join('|')})/gems/([^/]+)-([\w.]+)/(.*)}
|
15
|
+
GEMS_RESULT = '\2 (\3) \4'.freeze
|
16
|
+
|
17
|
+
# regexp (optionnally allowing leading X: for windows support)
|
18
|
+
INPUT_FORMAT = %r{^((?:[a-zA-Z]:)?[^:]+):(\d+)(?::in `([^']+)')?$}.freeze
|
19
|
+
|
20
|
+
# The file portion of the line (such as app/models/user.rb)
|
21
|
+
attr_reader :file
|
22
|
+
|
23
|
+
# The line number portion of the line
|
24
|
+
attr_reader :number
|
25
|
+
|
26
|
+
# The method_name of the line (such as index)
|
27
|
+
attr_reader :method_name
|
28
|
+
|
29
|
+
# Parses a single line of a given backtrace
|
30
|
+
# @param [String] unparsed_line The raw line from +caller+ or some backtrace
|
31
|
+
# @return [Line] The parsed backtrace line
|
32
|
+
def self.parse(unparsed_line)
|
33
|
+
_, file, number, method_name = unparsed_line.match(INPUT_FORMAT).to_a
|
34
|
+
|
35
|
+
# Remove Rails root from log lines
|
36
|
+
file = defined?(Rails) ? file.gsub(Rails.root.to_s, '') : file
|
37
|
+
|
38
|
+
if file
|
39
|
+
# Clean those ERB lines, we don't need the internal autogenerated
|
40
|
+
# ERB method, what we do need (line number in ERB file) is already there
|
41
|
+
file = file.sub /(\.erb:\d+)\:in `__.*$/, "\\1"
|
42
|
+
|
43
|
+
# Remove RubyGems root directories
|
44
|
+
file = file.sub(GEMS_REGEXP, GEMS_RESULT)
|
45
|
+
end
|
46
|
+
|
47
|
+
new(file, number, method_name)
|
48
|
+
end
|
49
|
+
|
50
|
+
def initialize(file, number, method_name)
|
51
|
+
@file = file
|
52
|
+
@number = number
|
53
|
+
@method_name = method_name
|
54
|
+
end
|
55
|
+
|
56
|
+
# Reconstructs the line in a readable fashion
|
57
|
+
def to_s
|
58
|
+
"#{file}:#{number}:in `#{method_name}'"
|
59
|
+
end
|
60
|
+
|
61
|
+
def ==(other)
|
62
|
+
to_s == other.to_s
|
63
|
+
end
|
64
|
+
|
65
|
+
def inspect
|
66
|
+
"<Line:#{to_s}>"
|
67
|
+
end
|
68
|
+
|
69
|
+
def empty?
|
70
|
+
file.nil? && number.nil? && method_name.nil?
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
# holder for an Array of Backtrace::Line instances
|
75
|
+
attr_reader :lines
|
76
|
+
|
77
|
+
def self.parse(ruby_backtrace, opts = {})
|
78
|
+
ruby_lines = split_multiline_backtrace(ruby_backtrace)
|
79
|
+
|
80
|
+
filters = opts[:filters] || []
|
81
|
+
filtered_lines = ruby_lines.to_a.map do |line|
|
82
|
+
filters.inject(line) do |l, proc|
|
83
|
+
proc.call(l)
|
84
|
+
end
|
85
|
+
end.compact
|
86
|
+
|
87
|
+
lines = filtered_lines.collect do |unparsed_line|
|
88
|
+
Line.parse(unparsed_line)
|
89
|
+
end
|
90
|
+
|
91
|
+
new(lines.delete_if(&:empty?))
|
92
|
+
end
|
93
|
+
|
94
|
+
def initialize(lines)
|
95
|
+
@lines = lines
|
96
|
+
end
|
97
|
+
|
98
|
+
def inspect
|
99
|
+
"<Backtrace: " + lines.collect { |line| line.inspect }.join(", ") + ">"
|
100
|
+
end
|
101
|
+
|
102
|
+
def to_s
|
103
|
+
content = []
|
104
|
+
lines.each do |line|
|
105
|
+
content << line
|
106
|
+
end
|
107
|
+
content.join("\n")
|
108
|
+
end
|
109
|
+
|
110
|
+
def ==(other)
|
111
|
+
if other.respond_to?(:lines)
|
112
|
+
lines == other.lines
|
113
|
+
else
|
114
|
+
false
|
115
|
+
end
|
116
|
+
end
|
117
|
+
|
118
|
+
private
|
119
|
+
|
120
|
+
def self.split_multiline_backtrace(backtrace)
|
121
|
+
backtrace = [backtrace] unless backtrace.respond_to?(:to_a)
|
122
|
+
if backtrace.to_a.size == 1
|
123
|
+
backtrace.to_a.first.split(/\n\s*/)
|
124
|
+
else
|
125
|
+
backtrace
|
126
|
+
end
|
127
|
+
end
|
128
|
+
end
|
129
|
+
end
|
130
|
+
end
|
@@ -0,0 +1,176 @@
|
|
1
|
+
require_relative 'backtrace'
|
2
|
+
require 'json'
|
3
|
+
|
4
|
+
module Binnacle
|
5
|
+
module Trap
|
6
|
+
HTTP_HEADER_PREFIXES = [
|
7
|
+
'HTTP_'.freeze,
|
8
|
+
'CONTENT_TYPE'.freeze,
|
9
|
+
'CONTENT_LENGTH'.freeze
|
10
|
+
].freeze
|
11
|
+
|
12
|
+
HTTP_HEADER_SKIPS = [
|
13
|
+
'HTTP_COOKIE'.freeze,
|
14
|
+
'HTTP_X_CSRF_TOKEN'.freeze
|
15
|
+
]
|
16
|
+
|
17
|
+
class ExceptionEvent < ::Binnacle::Event
|
18
|
+
attr_reader :exception
|
19
|
+
attr_reader :env
|
20
|
+
attr_reader :request
|
21
|
+
attr_reader :component
|
22
|
+
attr_reader :method
|
23
|
+
attr_reader :module
|
24
|
+
attr_reader :backtrace
|
25
|
+
|
26
|
+
def initialize(exception, env)
|
27
|
+
@exception = unwrap_exception(exception)
|
28
|
+
@env = env
|
29
|
+
|
30
|
+
if env["action_dispatch.request.parameters"] != nil
|
31
|
+
@component = env['action_dispatch.request.parameters'][:controller] || nil
|
32
|
+
@method = env['action_dispatch.request.parameters'][:action] || nil
|
33
|
+
@module = env['action_dispatch.request.parameters'][:module] || nil
|
34
|
+
end
|
35
|
+
|
36
|
+
@request = ::Rack::Request.new(env)
|
37
|
+
|
38
|
+
extract_event_name
|
39
|
+
extract_session_id
|
40
|
+
extract_client_id
|
41
|
+
extract_backtrace
|
42
|
+
|
43
|
+
self.log_level = "EXCEPTION"
|
44
|
+
self.tags = []
|
45
|
+
build_json_payload
|
46
|
+
|
47
|
+
configure(
|
48
|
+
Binnacle.configuration.error_channel,
|
49
|
+
self.event_name,
|
50
|
+
self.client_id,
|
51
|
+
self.session_id,
|
52
|
+
self.log_level,
|
53
|
+
nil,
|
54
|
+
self.tags,
|
55
|
+
self.json
|
56
|
+
)
|
57
|
+
end
|
58
|
+
|
59
|
+
private
|
60
|
+
|
61
|
+
def unwrap_exception(exception)
|
62
|
+
if exception.respond_to?(:original_exception)
|
63
|
+
exception.original_exception
|
64
|
+
elsif exception.respond_to?(:continued_exception)
|
65
|
+
exception.continued_exception
|
66
|
+
end || exception
|
67
|
+
end
|
68
|
+
|
69
|
+
# The root Exception class name
|
70
|
+
def extract_event_name
|
71
|
+
self.event_name = @exception.class.to_s
|
72
|
+
end
|
73
|
+
|
74
|
+
# The affected User or some identifier that can be used to determine
|
75
|
+
# who was affected by the exception (Warden, Devise, etc. should be used
|
76
|
+
# if available to get this information)
|
77
|
+
def extract_session_id
|
78
|
+
self.session_id = (@env["rack.session"] ? @env["rack.session"]["session_id"] : nil) || @request.ip
|
79
|
+
end
|
80
|
+
|
81
|
+
def extract_backtrace
|
82
|
+
backtrace = Backtrace.parse(@exception.backtrace)
|
83
|
+
@backtrace = backtrace.lines.map do |line|
|
84
|
+
{ number: line.number, file: line.file, method: line.method_name }
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
def extract_headers
|
89
|
+
http_headers = {}
|
90
|
+
|
91
|
+
http_headers = @env.map.with_object({}) do |(key, value), headers|
|
92
|
+
if Binnacle::Trap::HTTP_HEADER_PREFIXES.any? { |prefix| key.to_s.start_with?(prefix) } && !HTTP_HEADER_SKIPS.include?(key.to_s)
|
93
|
+
headers[key] = value
|
94
|
+
end
|
95
|
+
|
96
|
+
headers
|
97
|
+
end
|
98
|
+
|
99
|
+
http_headers
|
100
|
+
end
|
101
|
+
|
102
|
+
def extract_framework
|
103
|
+
defined?(Rails) ? "Rails" : "unknown"
|
104
|
+
end
|
105
|
+
|
106
|
+
def extract_framework_version
|
107
|
+
defined?(Rails) ? Rails::VERSION::STRING : "unknown"
|
108
|
+
end
|
109
|
+
|
110
|
+
def extract_framework_params
|
111
|
+
{}
|
112
|
+
end
|
113
|
+
|
114
|
+
def extract_ruby_version
|
115
|
+
"#{RUBY_VERSION rescue '?.?.?'} p#{RUBY_PATCHLEVEL rescue '???'} #{RUBY_RELEASE_DATE rescue '????-??-??'} #{RUBY_PLATFORM rescue '????'}"
|
116
|
+
end
|
117
|
+
|
118
|
+
def extract_hostname
|
119
|
+
require 'socket' unless defined?(Socket)
|
120
|
+
Socket.gethostname
|
121
|
+
rescue
|
122
|
+
'UNKNOWN'
|
123
|
+
end
|
124
|
+
|
125
|
+
def extract_rails_environment_level
|
126
|
+
defined?(Rails) ? Rails.env : "UNKNOWN"
|
127
|
+
end
|
128
|
+
|
129
|
+
def extract_libraries_loaded
|
130
|
+
libraries = {}
|
131
|
+
begin
|
132
|
+
libraries = Hash[*Gem.loaded_specs.map{|name, gem_specification| [name, gem_specification.version.to_s]}.flatten]
|
133
|
+
rescue
|
134
|
+
end
|
135
|
+
|
136
|
+
libraries
|
137
|
+
end
|
138
|
+
|
139
|
+
def extract_http_params
|
140
|
+
@request.params rescue {}
|
141
|
+
end
|
142
|
+
|
143
|
+
def extract_client_id
|
144
|
+
session = @env["rack.session"] ? @env["rack.session"].to_hash : {}
|
145
|
+
warden_info = session.find { |k,v| k.start_with?('warden.') }
|
146
|
+
if warden_info
|
147
|
+
self.client_id = warden_info.last.first.first
|
148
|
+
else
|
149
|
+
self.client_id = ""
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
153
|
+
def build_json_payload
|
154
|
+
self.json = {
|
155
|
+
path: @request.path,
|
156
|
+
exception: event_name,
|
157
|
+
message: @exception.message,
|
158
|
+
component: @component,
|
159
|
+
method: @method,
|
160
|
+
environment_level: extract_rails_environment_level,
|
161
|
+
hostname: extract_hostname,
|
162
|
+
user_agent: @request.user_agent,
|
163
|
+
ruby_version: extract_ruby_version,
|
164
|
+
framework: extract_framework,
|
165
|
+
framework_version: extract_framework_version,
|
166
|
+
framework_params: extract_framework_params,
|
167
|
+
http_params: extract_http_params,
|
168
|
+
headers: extract_headers,
|
169
|
+
dependencies: extract_libraries_loaded,
|
170
|
+
backtrace: @backtrace
|
171
|
+
}
|
172
|
+
end
|
173
|
+
|
174
|
+
end
|
175
|
+
end
|
176
|
+
end
|