-
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
1 changed file
with
146 additions
and
93 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -25,112 +25,161 @@ processes user signup's and call the `log_event` method to log details about the | |
|
||
```ruby | ||
class UsersController < ApplicationController | ||
include EventLoggerRails::Loggable | ||
def create | ||
user = User.new(user_params) | ||
if user.save | ||
log_event 'user.signup.success' | ||
redirect_to dashboard_path | ||
else | ||
log_event 'user.signup.failure', errors: user.errors | ||
render :new | ||
end | ||
end | ||
include EventLoggerRails::Loggable | ||
def create | ||
user = User.new(user_params) | ||
if user.save | ||
log_event 'user.signup.success' | ||
redirect_to dashboard_path | ||
else | ||
log_event 'user.signup.failure', data: { errors: user.errors.full_messages } | ||
render :new | ||
end | ||
end | ||
end | ||
``` | ||
|
||
In this example, a possible successful signup could be structured like this: | ||
|
||
```json | ||
{ | ||
"message": { | ||
"event_description": "Indicates a user signup was successful.", | ||
"event_identifier": "user.signup.success", | ||
"controller": "Users", | ||
"action": "create", | ||
"method": "POST", | ||
"path": "/users", | ||
"remote_ip": "::1", | ||
"parameters": { | ||
"user": { | ||
"first_name": "Test", | ||
"last_name": "User" | ||
} | ||
} | ||
}, | ||
"severity": "WARN", | ||
"timestamp": "2023-09-23 22:27:33 -0500" | ||
"host": "d6aeb6b0516c", | ||
"environment": "development", | ||
"service_name": "Kutima", | ||
"level": "WARN", | ||
"timestamp": "2023-09-29T23:23:16.633+00:00", | ||
"event_identifier": "user.signup.success", | ||
"event_description": "Indicates a user signup was successful.", | ||
"email": "[email protected]", | ||
"action": "create", | ||
"controller": "Registrations", | ||
"format": "application/x-www-form-urlencoded;charset=UTF-8", | ||
"method": "POST", | ||
"parameters": { | ||
"authenticity_token": "[FILTERED]", | ||
"user": { | ||
"email": "[email protected]", | ||
"password": "[FILTERED]" | ||
} | ||
}, | ||
"path": "/users", | ||
"remote_ip": "172.20.0.1" | ||
} | ||
``` | ||
|
||
...while a failed signup might look like this: | ||
|
||
```json | ||
{ | ||
"message": { | ||
"event_description": "Indicates a user signup was not successful.", | ||
"event_identifier": "user.signup.failure", | ||
"controller": "Users", | ||
"action": "create", | ||
"method": "POST", | ||
"path": "/users", | ||
"remote_ip": "::1", | ||
"parameters": { | ||
"user": { | ||
"first_name": "Test", | ||
"last_name": "User" | ||
} | ||
}, | ||
"errors": { | ||
"email": "is missing" | ||
} | ||
}, | ||
"severity": "WARN", | ||
"timestamp": "2023-09-23 22:27:33 -0500" | ||
"host": "d6aeb6b0516c", | ||
"environment": "development", | ||
"service_name": "DummyApp", | ||
"level": "WARN", | ||
"timestamp": "2023-09-29T23:01:17.554+00:00", | ||
"event_identifier": "user.signup.failure", | ||
"event_description": "Indicates a user signup was not successful.", | ||
"errors": [ | ||
"Email can't be blank", | ||
"Password can't be blank" | ||
], | ||
"action": "create", | ||
"controller": "Registrations", | ||
"format": "application/x-www-form-urlencoded;charset=UTF-8", | ||
"method": "POST", | ||
"parameters": { | ||
"authenticity_token": "[FILTERED]", | ||
"user": { | ||
"email": "", | ||
"password": "[FILTERED]" | ||
}, | ||
}, | ||
"path": "/users", | ||
"remote_ip": "172.20.0.1" | ||
} | ||
``` | ||
|
||
Note how the log entry from the previous example contains the data passed in via the optional `data` argument. | ||
|
||
You can also provide a logger level as an optional argument if you need to specify a logger level other than the default: | ||
|
||
```ruby | ||
log_event 'user.signup.failure', :error, errors: user.errors | ||
log_event 'user.signup.failure', level: :error, data: { errors: user.errors } | ||
``` | ||
|
||
This will output an event with the corresponding severity level. You must provide a valid logger level (`:debug, :info, :warn, :error, or :unknown`). | ||
|
||
```json | ||
{ | ||
"message": { | ||
"event_description": "Indicates a user signup was not successful.", | ||
"event_identifier": "user.signup.failure", | ||
"controller": "Users", | ||
"action": "create", | ||
"method": "POST", | ||
"path": "/users", | ||
"remote_ip": "::1", | ||
"parameters": { | ||
"user": { | ||
"first_name": "Test", | ||
"last_name": "User" | ||
} | ||
}, | ||
"errors": { | ||
"email": "is missing" | ||
} | ||
}, | ||
"severity": "ERROR", | ||
"timestamp": "2023-09-23 22:27:33 -0500" | ||
"host": "d6aeb6b0516c", | ||
"environment": "development", | ||
"service_name": "DummyApp", | ||
"level": "ERROR", | ||
"timestamp": "2023-09-29T23:01:17.554+00:00", | ||
"event_identifier": "user.signup.failure", | ||
"event_description": "Indicates a user signup was not successful.", | ||
"errors": [ | ||
"Email can't be blank", | ||
"Password can't be blank" | ||
], | ||
"action": "create", | ||
"controller": "Registrations", | ||
"format": "application/x-www-form-urlencoded;charset=UTF-8", | ||
"method": "POST", | ||
"parameters": { | ||
"authenticity_token": "[FILTERED]", | ||
"user": { | ||
"email": "", | ||
"password": "[FILTERED]" | ||
}, | ||
}, | ||
"path": "/users", | ||
"remote_ip": "172.20.0.1" | ||
} | ||
``` | ||
|
||
### Logging in Models | ||
|
||
You can also log events from within models by including the `EventLoggerRails::LoggableModel` concern and calling `log_event`. | ||
|
||
```ruby | ||
class User < ApplicationRecord | ||
include EventLoggerRails::LoggableModel | ||
after_create :log_signup | ||
private | ||
def log_signup | ||
log_event 'user.signup.success', data: { email: } | ||
end | ||
end | ||
``` | ||
|
||
By default, `event_logger_rails` will include the model name and instance ID, along with whatever data is passed. | ||
|
||
``` | ||
{ | ||
"host": "d6aeb6b0516c", | ||
"environment": "development", | ||
"service_name": "DummyApp", | ||
"level": "WARN", | ||
"timestamp": "2023-09-30T00:51:51.315+00:00", | ||
"event_identifier": "user.signup.success", | ||
"event_description": "Indicates a user signup was successful.", | ||
"email": "[email protected]", | ||
"model": "User", | ||
"instance_id": 38 | ||
} | ||
``` | ||
### Logging Outside of Controllers | ||
### Logging Everywhere Else | ||
You can log events from anywhere inside of your application by calling `EventLoggerRails.log` directly, though you won't get the additional context | ||
from the request. | ||
```ruby | ||
EventLoggerRails.log 'user.signup.success', :info, user_id: @user.id | ||
EventLoggerRails.log 'user.signup.success', :info, { user_id: @user.id } | ||
``` | ||
|
||
### Errors | ||
|
@@ -142,25 +191,29 @@ If you fail to register an event, the logger will emit an `event_logger_rails.ev | |
|
||
```json | ||
{ | ||
"message": { | ||
"event_description": "event_logger_rails.event.unregistered", | ||
"event_identifier": "Indicates provided event was unregistered." | ||
}, | ||
"severity": "ERROR", | ||
"timestamp": "2023-09-23 22:27:33 -0500" | ||
"host": "d6aeb6b0516c", | ||
"environment": "development", | ||
"service_name": "DummyApp", | ||
"level": "ERROR", | ||
"timestamp": "2023-09-29T23:27:53.714+00:00", | ||
"event_identifier": "event_logger_rails.event.unregistered", | ||
"event_description": "Indicates provided event was unregistered.", | ||
"message": "Event provided not registered: foo.bar" | ||
} | ||
``` | ||
|
||
If you provide an invalid log level, the logger will emit an `event_logger_rails.logger_level.invalid` event: | ||
|
||
```json | ||
{ | ||
"message": { | ||
"event_description": "event_logger_rails.logger_level.invalid", | ||
"event_identifier": "Indicates provided level was invalid." | ||
}, | ||
"severity": "ERROR", | ||
"timestamp": "2023-09-23 22:27:33 -0500" | ||
"host": "d6aeb6b0516c", | ||
"environment": "development", | ||
"service_name": "DummyApp", | ||
"level": "ERROR", | ||
"timestamp": "2023-09-29T23:30:29.761+00:00", | ||
"event_identifier": "event_logger_rails.logger_level.invalid", | ||
"event_description": "Indicates provided level was invalid.", | ||
"message": "Invalid logger level provided: 'foobar'. Valid levels: :debug, :info, :warn, :error, :unknown." | ||
} | ||
``` | ||
|
||
|
@@ -175,37 +228,37 @@ gem 'event_logger_rails' | |
And then execute: | ||
|
||
```bash | ||
$ bundle | ||
bundle | ||
``` | ||
|
||
Or install it yourself as: | ||
|
||
```bash | ||
$ gem install event_logger_rails | ||
gem install event_logger_rails | ||
``` | ||
|
||
Run the install generator to create a config file (`config/event_logger_rails.yml`) and example initializer: | ||
Run the install generator to create a config file (`config/event_logger_rails.yml`): | ||
|
||
```bash | ||
$ bin/rails generate event_logger_rails:install | ||
bin/rails generate event_logger_rails:install | ||
``` | ||
|
||
Add your events to the generated config file following the structure of the examples. | ||
|
||
You may opt to load in registered events in `config/application.rb` using the `config_for` helper provided by Rails. | ||
By default, `event_logger_rails` outputs to a separate log file (`log/event_logger_rails.#{Rails.env}.log`) from normal Rails log output, allowing | ||
you to ingest these logs independently. If you wish to set an alternative log device to capture output , you can configure it in `config/application.rb`: | ||
|
||
```ruby | ||
EventLoggerRails.setup do |config| | ||
config.registered_events = config_for[:registered_events] | ||
Rails.application.configure do |config| | ||
config.event_logger_rails.logdev = 'path/to/log.file' | ||
end | ||
``` | ||
|
||
Doing so eliminates the need for the generated initializer, so you should delete it if you choose to go this route. | ||
|
||
## Contributing | ||
|
||
Contributions are welcome. Feel free to open a PR. | ||
|
||
## License | ||
|
||
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT). | ||
|