noticed 1.2.7 → 1.2.12
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +71 -11
- data/lib/generators/noticed/delivery_method_generator.rb +19 -0
- data/lib/generators/noticed/model_generator.rb +12 -1
- data/lib/generators/noticed/notification_generator.rb +1 -1
- data/lib/generators/noticed/templates/delivery_method.rb.tt +5 -0
- data/lib/noticed.rb +9 -0
- data/lib/noticed/base.rb +3 -0
- data/lib/noticed/coder.rb +1 -7
- data/lib/noticed/delivery_methods/action_cable.rb +10 -10
- data/lib/noticed/delivery_methods/base.rb +37 -7
- data/lib/noticed/delivery_methods/slack.rb +1 -1
- data/lib/noticed/delivery_methods/twilio.rb +1 -1
- data/lib/noticed/delivery_methods/vonage.rb +5 -1
- data/lib/noticed/model.rb +12 -1
- data/lib/noticed/text_coder.rb +16 -0
- data/lib/noticed/translation.rb +17 -15
- data/lib/noticed/version.rb +1 -1
- metadata +5 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 6fca882449d4696c1551badc82383f789610f19898dec345a64f33a225ca9f81
|
4
|
+
data.tar.gz: 8751afe5063970449573719d7bedb957d7c88e52f1efef96357d8754d48cec21
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 8ab4f29e4611ee0e5c3f548562e0265dc991ca45cb4789fbf7eb780430b21eddaef3407fbc7946f21bbff411149dd8e796197d9d0b334d85d8085aaea7beb54a
|
7
|
+
data.tar.gz: 7c6ce9b53092a182e405fc26274daae30e8ec6e2149c5f5794d3fa5c3728210f013f632c844dd00f9ad5f1d64cba12b09de6c311b9d367b580d91f899affb4c4
|
data/README.md
CHANGED
@@ -256,7 +256,7 @@ Sends an SMS notification via Twilio.
|
|
256
256
|
|
257
257
|
* `credentials: :get_twilio_credentials` - *Optional*
|
258
258
|
|
259
|
-
Use a custom method to retrieve the credentials for Twilio. Method should return a Hash with `:account_sid`, `:auth_token` and `:
|
259
|
+
Use a custom method to retrieve the credentials for Twilio. Method should return a Hash with `:account_sid`, `:auth_token` and `:phone_number` keys.
|
260
260
|
|
261
261
|
Defaults to `Rails.application.credentials.twilio[:account_sid]` and `Rails.application.credentials.twilio[:auth_token]`
|
262
262
|
|
@@ -311,22 +311,28 @@ Sends an SMS notification via Vonage / Nexmo.
|
|
311
311
|
|
312
312
|
### 🚚 Custom Delivery Methods
|
313
313
|
|
314
|
-
|
314
|
+
To generate a custom delivery method, simply run
|
315
315
|
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
end
|
320
|
-
```
|
316
|
+
`rails generate noticed:delivery_method Discord`
|
317
|
+
|
318
|
+
This will generate a new `DeliveryMethods::Discord` class inside the `app/notifications/delivery_methods` folder, which can be used to deliver notifications to Discord.
|
321
319
|
|
322
320
|
```ruby
|
323
|
-
class
|
321
|
+
class DeliveryMethods::Discord < Noticed::DeliveryMethods::Base
|
324
322
|
def deliver
|
325
323
|
# Logic for sending a Discord notification
|
326
324
|
end
|
327
325
|
end
|
328
326
|
```
|
329
327
|
|
328
|
+
You can use the custom delivery method thus created by adding a `deliver_by` line with a unique name and `class` option in your notification class.
|
329
|
+
|
330
|
+
```ruby
|
331
|
+
class MyNotification < Noticed::Base
|
332
|
+
deliver_by :discord, class: "DeliveryMethods::Discord"
|
333
|
+
end
|
334
|
+
```
|
335
|
+
|
330
336
|
Delivery methods have access to the following methods and attributes:
|
331
337
|
|
332
338
|
* `notification` - The instance of the Notification. You can call methods on the notification to let the user easily override formatting and other functionality of the delivery method.
|
@@ -339,7 +345,7 @@ Delivery methods have access to the following methods and attributes:
|
|
339
345
|
Callbacks for delivery methods wrap the *actual* delivery of the notification. You can use `before_deliver`, `around_deliver` and `after_deliver` in your custom delivery methods.
|
340
346
|
|
341
347
|
```ruby
|
342
|
-
class
|
348
|
+
class DeliveryMethods::Discord < Noticed::DeliveryMethods::Base
|
343
349
|
after_deliver do
|
344
350
|
# Do whatever you want
|
345
351
|
end
|
@@ -351,13 +357,13 @@ end
|
|
351
357
|
Rails 6.1+ can serialize Class and Module objects as arguments to ActiveJob. The following syntax should work for Rails 6.1+:
|
352
358
|
|
353
359
|
```ruby
|
354
|
-
deliver_by
|
360
|
+
deliver_by DeliveryMethods::Discord
|
355
361
|
```
|
356
362
|
|
357
363
|
For Rails 6.0 and earlier, you must pass strings of the class names in the `deliver_by` options.
|
358
364
|
|
359
365
|
```ruby
|
360
|
-
deliver_by :discord, class: "
|
366
|
+
deliver_by :discord, class: "DeliveryMethods::Discord"
|
361
367
|
```
|
362
368
|
|
363
369
|
We recommend the Rails 6.0 compatible options to prevent confusion.
|
@@ -401,6 +407,60 @@ Check if read / unread:
|
|
401
407
|
@notification.unread?
|
402
408
|
```
|
403
409
|
|
410
|
+
#### Associating Notifications
|
411
|
+
|
412
|
+
Adding notification associations to your models makes querying and deleting notifications easy and is a pretty critical feature of most applications.
|
413
|
+
|
414
|
+
For example, in most cases, you'll want to delete notifications for records that are destroyed.
|
415
|
+
|
416
|
+
##### JSON Columns
|
417
|
+
|
418
|
+
If you're using MySQL or Postgresql, the `params` column on the notifications table is in `json` or `jsonb` format and can be queried against directly.
|
419
|
+
|
420
|
+
For example, we can query the notifications and delete them on destroy like so:
|
421
|
+
|
422
|
+
```ruby
|
423
|
+
class Post < ApplicationRecord
|
424
|
+
def notifications
|
425
|
+
# Exact match
|
426
|
+
@notifications ||= Notification.where(params: { post: self })
|
427
|
+
|
428
|
+
# Or Postgres syntax to query the post key in the JSON column
|
429
|
+
# @notifications ||= Notification.where("params->'post' = ?", Noticed::Coder.dump(self).to_json)
|
430
|
+
end
|
431
|
+
|
432
|
+
before_destroy :destroy_notifications
|
433
|
+
|
434
|
+
def destroy_notifications
|
435
|
+
notifications.destroy_all
|
436
|
+
end
|
437
|
+
end
|
438
|
+
```
|
439
|
+
|
440
|
+
##### Polymorphic Assocation
|
441
|
+
|
442
|
+
If your notification is only associated with one model or you're using a `text` column for your params column , then a polymorphic association is what you'll want to use.
|
443
|
+
|
444
|
+
1. Add a polymorphic association to the Notification model. `rails g migration AddNotifiableToNotifications notifiable:belongs_to{polymorphic}`
|
445
|
+
|
446
|
+
2. Add `has_many :notifications, as: :notifiable, dependent: :destroy` to each model
|
447
|
+
|
448
|
+
3. Customize database `format: ` option to write the `notifiable` attribute(s) when saving the notification
|
449
|
+
|
450
|
+
```ruby
|
451
|
+
class ExampleNotification < Noticed::Base
|
452
|
+
deliver_by :database, format: :format_for_database
|
453
|
+
|
454
|
+
def format_for_database
|
455
|
+
{
|
456
|
+
notifiable: params.delete(:post),
|
457
|
+
type: self.class.name,
|
458
|
+
params: params
|
459
|
+
}
|
460
|
+
end
|
461
|
+
end
|
462
|
+
```
|
463
|
+
|
404
464
|
## 🙏 Contributing
|
405
465
|
|
406
466
|
This project uses [Standard](https://github.com/testdouble/standard) for formatting Ruby code. Please make sure to run `standardrb` before submitting pull requests.
|
@@ -0,0 +1,19 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "rails/generators/named_base"
|
4
|
+
|
5
|
+
module Noticed
|
6
|
+
module Generators
|
7
|
+
class DeliveryMethodGenerator < Rails::Generators::NamedBase
|
8
|
+
include Rails::Generators::ResourceHelpers
|
9
|
+
|
10
|
+
source_root File.expand_path("../templates", __FILE__)
|
11
|
+
|
12
|
+
desc "Generates a class for a custom delivery method with the given NAME."
|
13
|
+
|
14
|
+
def generate_notification
|
15
|
+
template "delivery_method.rb", "app/notifications/delivery_methods/#{singular_name}.rb"
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
@@ -15,7 +15,7 @@ module Noticed
|
|
15
15
|
argument :attributes, type: :array, default: [], banner: "field:type field:type"
|
16
16
|
|
17
17
|
def generate_notification
|
18
|
-
generate :model, name, "recipient:references{polymorphic}", "type",
|
18
|
+
generate :model, name, "recipient:references{polymorphic}", "type", params_column, "read_at:datetime", *attributes
|
19
19
|
end
|
20
20
|
|
21
21
|
def add_noticed_model
|
@@ -31,6 +31,17 @@ module Noticed
|
|
31
31
|
def model_path
|
32
32
|
@model_path ||= File.join("app", "models", "#{file_path}.rb")
|
33
33
|
end
|
34
|
+
|
35
|
+
def params_column
|
36
|
+
case ActiveRecord::Base.configurations.configs_for(spec_name: "primary").config["adapter"]
|
37
|
+
when "mysql"
|
38
|
+
"params:json"
|
39
|
+
when "postgresql"
|
40
|
+
"params:jsonb"
|
41
|
+
else
|
42
|
+
"params:text"
|
43
|
+
end
|
44
|
+
end
|
34
45
|
end
|
35
46
|
end
|
36
47
|
end
|
@@ -12,7 +12,7 @@ module Noticed
|
|
12
12
|
desc "Generates a notification with the given NAME."
|
13
13
|
|
14
14
|
def generate_notification
|
15
|
-
template "notification.rb", "app/notifications/#{
|
15
|
+
template "notification.rb", "app/notifications/#{file_path}.rb"
|
16
16
|
end
|
17
17
|
end
|
18
18
|
end
|
data/lib/noticed.rb
CHANGED
@@ -6,6 +6,7 @@ module Noticed
|
|
6
6
|
autoload :Base, "noticed/base"
|
7
7
|
autoload :Coder, "noticed/coder"
|
8
8
|
autoload :Model, "noticed/model"
|
9
|
+
autoload :TextCoder, "noticed/text_coder"
|
9
10
|
autoload :Translation, "noticed/translation"
|
10
11
|
|
11
12
|
module DeliveryMethods
|
@@ -33,4 +34,12 @@ module Noticed
|
|
33
34
|
|
34
35
|
class ValidationError < StandardError
|
35
36
|
end
|
37
|
+
|
38
|
+
class ResponseUnsuccessful < StandardError
|
39
|
+
attr_reader :response
|
40
|
+
|
41
|
+
def initialize(response)
|
42
|
+
@response = response
|
43
|
+
end
|
44
|
+
end
|
36
45
|
end
|
data/lib/noticed/base.rb
CHANGED
@@ -69,6 +69,9 @@ module Noticed
|
|
69
69
|
def run_delivery(recipient, enqueue: true)
|
70
70
|
delivery_methods = self.class.delivery_methods.dup
|
71
71
|
|
72
|
+
# Set recipient to instance var so it is available to Notification class
|
73
|
+
@recipient = recipient
|
74
|
+
|
72
75
|
# Run database delivery inline first if it exists so other methods have access to the record
|
73
76
|
if (index = delivery_methods.find_index { |m| m[:name] == :database })
|
74
77
|
delivery_method = delivery_methods.delete_at(index)
|
data/lib/noticed/coder.rb
CHANGED
@@ -2,18 +2,12 @@ module Noticed
|
|
2
2
|
class Coder
|
3
3
|
def self.load(data)
|
4
4
|
return if data.nil?
|
5
|
-
|
6
|
-
# Text columns need JSON parsing
|
7
|
-
if data.is_a?(String)
|
8
|
-
data = JSON.parse(data)
|
9
|
-
end
|
10
|
-
|
11
5
|
ActiveJob::Arguments.send(:deserialize_argument, data)
|
12
6
|
end
|
13
7
|
|
14
8
|
def self.dump(data)
|
15
9
|
return if data.nil?
|
16
|
-
ActiveJob::Arguments.send(:serialize_argument, data)
|
10
|
+
ActiveJob::Arguments.send(:serialize_argument, data)
|
17
11
|
end
|
18
12
|
end
|
19
13
|
end
|
@@ -17,16 +17,16 @@ module Noticed
|
|
17
17
|
|
18
18
|
def channel
|
19
19
|
@channel ||= begin
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
20
|
+
value = options[:channel]
|
21
|
+
case value
|
22
|
+
when String
|
23
|
+
value.constantize
|
24
|
+
when Symbol
|
25
|
+
notification.send(value)
|
26
|
+
when Class
|
27
|
+
value
|
28
|
+
else
|
29
|
+
Noticed::NotificationChannel
|
30
30
|
end
|
31
31
|
end
|
32
32
|
end
|
@@ -6,15 +6,15 @@ module Noticed
|
|
6
6
|
|
7
7
|
attr_reader :notification, :options, :recipient, :record
|
8
8
|
|
9
|
-
def perform(
|
10
|
-
@notification = notification_class.constantize.new(params)
|
11
|
-
@options = options
|
12
|
-
@recipient = recipient
|
13
|
-
@record = record
|
9
|
+
def perform(args)
|
10
|
+
@notification = args[:notification_class].constantize.new(args[:params])
|
11
|
+
@options = args[:options]
|
12
|
+
@recipient = args[:recipient]
|
13
|
+
@record = args[:record]
|
14
14
|
|
15
15
|
# Make notification aware of database record and recipient during delivery
|
16
|
-
@notification.record = record
|
17
|
-
@notification.recipient = recipient
|
16
|
+
@notification.record = args[:record]
|
17
|
+
@notification.recipient = args[:recipient]
|
18
18
|
|
19
19
|
run_callbacks :deliver do
|
20
20
|
deliver
|
@@ -24,6 +24,36 @@ module Noticed
|
|
24
24
|
def deliver
|
25
25
|
raise NotImplementedError, "Delivery methods must implement a deliver method"
|
26
26
|
end
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
# Helper method for making POST requests from delivery methods
|
31
|
+
#
|
32
|
+
# Usage:
|
33
|
+
# post("http://example.com", basic_auth: {user:, pass:}, json: {}, form: {})
|
34
|
+
#
|
35
|
+
def post(url, args = {})
|
36
|
+
basic_auth = args.delete(:basic_auth)
|
37
|
+
|
38
|
+
request = if basic_auth
|
39
|
+
HTTP.basic_auth(user: basic_auth[:user], pass: basic_auth[:pass])
|
40
|
+
else
|
41
|
+
HTTP
|
42
|
+
end
|
43
|
+
|
44
|
+
response = request.post(url, args)
|
45
|
+
|
46
|
+
if options[:debug]
|
47
|
+
Rails.logger.debug("POST #{url}")
|
48
|
+
Rails.logger.debug("Response: #{response.code}: #{response}")
|
49
|
+
end
|
50
|
+
|
51
|
+
if !options[:ignore_failure] && !response.status.success?
|
52
|
+
raise ResponseUnsuccessful.new(response)
|
53
|
+
end
|
54
|
+
|
55
|
+
response
|
56
|
+
end
|
27
57
|
end
|
28
58
|
end
|
29
59
|
end
|
@@ -2,7 +2,11 @@ module Noticed
|
|
2
2
|
module DeliveryMethods
|
3
3
|
class Vonage < Base
|
4
4
|
def deliver
|
5
|
-
|
5
|
+
response = post("https://rest.nexmo.com/sms/json", json: format)
|
6
|
+
status = response.parse.dig("messages", 0, "status")
|
7
|
+
if !options[:ignore_failure] && status != "0"
|
8
|
+
raise ResponseUnsuccessful.new(response)
|
9
|
+
end
|
6
10
|
end
|
7
11
|
|
8
12
|
private
|
data/lib/noticed/model.rb
CHANGED
@@ -5,17 +5,28 @@ module Noticed
|
|
5
5
|
included do
|
6
6
|
self.inheritance_column = nil
|
7
7
|
|
8
|
-
serialize :params,
|
8
|
+
serialize :params, noticed_coder
|
9
9
|
|
10
10
|
belongs_to :recipient, polymorphic: true
|
11
11
|
|
12
12
|
scope :newest_first, -> { order(created_at: :desc) }
|
13
|
+
scope :unread, -> { where(read_at: nil) }
|
14
|
+
scope :read, -> { where.not(read_at: nil) }
|
13
15
|
end
|
14
16
|
|
15
17
|
module ClassMethods
|
16
18
|
def mark_as_read!
|
17
19
|
update_all(read_at: Time.current, updated_at: Time.current)
|
18
20
|
end
|
21
|
+
|
22
|
+
def noticed_coder
|
23
|
+
case attribute_types["params"].type
|
24
|
+
when :json, :jsonb
|
25
|
+
Noticed::Coder
|
26
|
+
else
|
27
|
+
Noticed::TextCoder
|
28
|
+
end
|
29
|
+
end
|
19
30
|
end
|
20
31
|
|
21
32
|
# Rehydrate the database notification into the Notification object for rendering
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module Noticed
|
2
|
+
class TextCoder
|
3
|
+
def self.load(data)
|
4
|
+
return if data.nil?
|
5
|
+
|
6
|
+
# Text columns need JSON parsing
|
7
|
+
data = JSON.parse(data)
|
8
|
+
ActiveJob::Arguments.send(:deserialize_argument, data)
|
9
|
+
end
|
10
|
+
|
11
|
+
def self.dump(data)
|
12
|
+
return if data.nil?
|
13
|
+
ActiveJob::Arguments.send(:serialize_argument, data).to_json
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
data/lib/noticed/translation.rb
CHANGED
@@ -1,21 +1,23 @@
|
|
1
|
-
module
|
2
|
-
|
1
|
+
module Noticed
|
2
|
+
module Translation
|
3
|
+
extend ActiveSupport::Concern
|
3
4
|
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
5
|
+
# Returns the +i18n_scope+ for the class. Overwrite if you want custom lookup.
|
6
|
+
def i18n_scope
|
7
|
+
:notifications
|
8
|
+
end
|
8
9
|
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
10
|
+
def translate(key, **options)
|
11
|
+
I18n.translate(scope_translation_key(key), **options)
|
12
|
+
end
|
13
|
+
alias t translate
|
13
14
|
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
15
|
+
def scope_translation_key(key)
|
16
|
+
if key.to_s.start_with?(".")
|
17
|
+
"#{i18n_scope}.#{self.class.name.underscore}#{key}"
|
18
|
+
else
|
19
|
+
key
|
20
|
+
end
|
19
21
|
end
|
20
22
|
end
|
21
23
|
end
|
data/lib/noticed/version.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: noticed
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.2.
|
4
|
+
version: 1.2.12
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Chris Oliver
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2020-08-
|
11
|
+
date: 2020-08-24 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rails
|
@@ -92,9 +92,11 @@ files:
|
|
92
92
|
- README.md
|
93
93
|
- Rakefile
|
94
94
|
- app/channels/noticed/notification_channel.rb
|
95
|
+
- lib/generators/noticed/delivery_method_generator.rb
|
95
96
|
- lib/generators/noticed/model_generator.rb
|
96
97
|
- lib/generators/noticed/notification_generator.rb
|
97
98
|
- lib/generators/noticed/templates/README
|
99
|
+
- lib/generators/noticed/templates/delivery_method.rb.tt
|
98
100
|
- lib/generators/noticed/templates/notification.rb.tt
|
99
101
|
- lib/noticed.rb
|
100
102
|
- lib/noticed/base.rb
|
@@ -109,6 +111,7 @@ files:
|
|
109
111
|
- lib/noticed/delivery_methods/vonage.rb
|
110
112
|
- lib/noticed/engine.rb
|
111
113
|
- lib/noticed/model.rb
|
114
|
+
- lib/noticed/text_coder.rb
|
112
115
|
- lib/noticed/translation.rb
|
113
116
|
- lib/noticed/version.rb
|
114
117
|
- lib/tasks/noticed_tasks.rake
|