Skip to content

Latest commit

 

History

History
354 lines (256 loc) · 10.3 KB

UPGRADE.md

File metadata and controls

354 lines (256 loc) · 10.3 KB

Noticed Upgrade Guide

Follow this guide to upgrade your Noticed implementation to the next version

Noticed 2.1

We've added a counter cache to Noticed::Event to keep track of the associated notifications.

Run the following command to copy over the migrations:

rails noticed:install:migrations

Noticed 2.0

We've made some major changes to Noticed to simplify and support more delivery methods.

Models

Instead of having models live in your application, Noticed v2 adds models managed by the gem.

Delete the Notification model at app/models/notification.rb.

Then run the new migrations:

rails noticed:install:migrations
rails db:migrate

To migrate your data to the new tables, loop through your existing notifications and create new records for each one. You can do this in a Rake task or in the Rails console:

# Temporarily define the model to access the old table
class TempNotification < ActiveRecord::Base
  self.inheritance_column = nil
  self.table_name = "notifications"
end

# Migrate each record to the new tables
TempNotification.find_each do |notification|
  attributes = notification.attributes.slice("type", "created_at", "updated_at").with_indifferent_access
  attributes[:type] = attributes[:type].sub("Notification", "Notifier")
  attributes[:params] = Noticed::Coder.load(notification.params)
  attributes[:params] = {} if attributes[:params].try(:has_key?, "noticed_error") # Skip invalid records

  # Extract related record to `belongs_to :record` association
  # This allows ActiveRecord associations instead of querying the JSON data
  # attributes[:record] = attributes[:params].delete(:user) || attributes[:params].delete(:account)

  attributes[:notifications_attributes] = [{
    type: "#{attributes[:type]}::Notification",
    recipient_type: notification.recipient_type,
    recipient_id: notification.recipient_id,
    read_at: notification.read_at,
    seen_at: notification.read_at,
    created_at: notification.created_at,
    updated_at: notification.updated_at
  }]
  Noticed::Event.create!(attributes)
end

After migrating, you can drop the old notifications table.

Mark As Read

To be more consistent with Rails:

mark_as_read! has been renamed to mark_as_read mark_as_unread! has been renamed to mark_as_unread

Parent Class

Noticed::Base has been deprecated in favor of Noticed::Event. This is an STI model that tracks all Notifier deliveries and recipients.

class CommentNotifier < Noticed::Event
end

Database Delivery Method

The database delivery is now baked into notifications.

You will need to remove deliver_by :database from your notifiers.

Notifiers

For clarity, we've renamed app/notifications to app/notifiers.

Notifiers - the class that delivers notifications Notification - the database record of the notification

We recommend renaming your existing classes to match. You'll also need to update the type column on existing notifications when renaming.

Noticed::Notification.find_each do |notification|
  notification.update(type: notification.type.sub("Notification", "Notifier"))
end

Delivery Method Configuration

Configuration for each delivery method can be contained within a block now. This improves organization for delivery method options by defining them in the block. Procs/Lambdas will be evaluated when needed and symbols can be used to call a method.

If you are using a symbol to call a method, we pass the notification object as an argument to the method. This allows you to access the notification object within the method. Your method must accept a single argument. If you don't need to use the object you can just use (*).

class CommentNotifier < Noticed::Event
  deliver_by :action_cable do |config|
    config.channel = "NotificationChannel"
    config.stream = ->{ recipient }
    config.message = :to_websocket
  end

  def to_websocket(notification)
    { foo: :bar }
  end
end
class CommentNotifier < Noticed::Event
  include IosNotifier

  def data_only?
    false
  end

  def url
    comment_thread_path(record.thread)
  end
end

module IosNotifier
  extend ActiveSupport::Concern

  included do
    deliver_by :ios do |config|
      config.format = :ios_format
      config.apns_key = :ios_cert
      config.key_id = :ios_key_id
      config.team_id = :ios_team_id
      config.bundle_identifier = :ios_bundle_id
      config.device_tokens = :ios_device_tokens
      config.if = :send_ios_notification?
    end
  end

  def ios_format(apn)
    apn.alert = { title:, body: } unless data_only?
    apn.mutable_content = true
    apn.content_available = true
    apn.sound = "notification.m4r"
    apn.custom_payload = {
      url:,
      type: self.class.name,
      id: record.id,
      image_url: "" || image_url,
      params: params.to_json
    }
  end

  def ios_cert(*)
    Rails.application.credentials.dig(:ios, Rails.env.to_sym, :apns_token_cert)
  end

  def ios_key_id(*)
    Rails.application.credentials.dig(:ios, Rails.env.to_sym, :key_id)
  end

  def ios_team_id(*)
    Rails.application.credentials.dig(:ios, Rails.env.to_sym, :team_id)
  end

  def ios_bundle_id(*)
    Rails.application.credentials.dig(:ios, Rails.env.to_sym, :bundle_identifier)
  end

  def ios_device_tokens(notification)
    notification.recipient.ios_device_tokens
  end

  def send_ios_notification?(notification)
    recipient = notification.recipient
    return false unless recipient.is_a?(User)

    recipient.send_notifications?
  end
end

Deliver Later

Notifications are always delivered later now. deliver and deliver_later perform the same action.

Required Params

param and params have been renamed to required_param(s) to be more clear.

class CommentNotifier < Noticed::Event
  required_param :comment
  required_params :account, :comment
end

Helper methods

Helper methods defined in Notifiers have changed slightly. In order to access helper methods from Notification objects, for example:

<div>
  <% @user.notifications.each do |notification| %>
    <%= link_to notification.message, notification.url %>
  <% end %>
</div>

You’ll need to wrap helper methods in the new notification_methods block within your Notifier:

class NewCommentNotifier < Noticed::Event
  deliver_by :email do |config|
    # ...
  end

  notification_methods do
    # I18n helpers still available here
    def message
      t(".message")
    end

    # URL helpers are available here too
    def url
      user_post_path(recipient, params[:post])
    end
  end
end

Notification Model Methods

In order to extend the Noticed models you'll need to use a concern an a to_prepare block:

# config/initializers/noticed.rb
module NotificationExtensions
  extend ActiveSupport::Concern

  included do
    belongs_to :organisation

    scope :filter_by_type, ->(type) { where(type:) }
    scope :exclude_type, ->(type) { where.not(type:) }
  end

  # You can also add instance methods here
end

Rails.application.config.to_prepare do
  # You can extend Noticed::Event or Noticed::Notification here
  Noticed::Event.include NotificationExtensions
  Noticed::Notification.include NotificationExtensions
end

Has Noticed Notifications

has_noticed_notifications has been superceded with a record polymorphic relationship that can be directly associated and queried with ActiveRecord.

We recommend backfilling the record association if your notification params has a primary related record and switching to a has_many association instead.

class Comment < ApplicationRecord
  has_many :noticed_events, as: :record, dependent: :destroy, class_name: "Noticed::Event"
end

If you would like to keep the JSON querying, you can implement a method for querying your model depending on the database you use:

# Define the
param_name = "user"

# PostgreSQL
model.where("params @> ?", Noticed::Coder.dump(param_name.to_sym => self).to_json)

# MySQL
model.where("JSON_CONTAINS(params, ?)", Noticed::Coder.dump(param_name.to_sym => self).to_json)

# SQLite
model.where("json_extract(params, ?) = ?", "$.#{param_name}", Noticed::Coder.dump(self).to_json)

# Other
model.where(params: {param_name.to_sym => self})

Recipient Notifications Association

Recipients can be associated with notifications using the following. This is useful for displaying notifications in your UI.

class User < ApplicationRecord
  has_many :notifications, as: :recipient, dependent: :destroy, class_name: "Noticed::Notification"
end

Delivery Method Changes

Options for delivery methods have been renamed for clarity and consistency.

ActionCable

  • The format option has been renamed to message.
  • The Noticed::NotificationChannel has been removed and an example channel is provided in the ActionCable docs.

Email Delivery Method

  • method is now a required option. Previously, it was inferred from the notification name but we've decided it would be better to be explicit.

FCM

  • The format option has been renamed to json.
  • The device_tokens option is now required and should return an Array of device tokens.
  • The invalid_token option replaces the cleanup_device_tokens method for handling invalid/expired tokens.
  • We no longer wrap the json payload in the message{} key. This means we are more compatible with the FCM docs and any future changes that Google make.

iOS

  • The cert_path option has been renamed to apns_key and should be given the key and not a path.
  • The device_tokens option is now required and should return an Array of device tokens.
  • The invalid_token option replaces the cleanup_device_tokens method for handling invalid/expired tokens.

Microsoft Teams

  • The format option has been renamed to json.

Slack

  • The format option has been renamed to json.
  • The url option now defaults to "https://slack.com/api/chat.postMessage instead of Rails.application.credentials.dig(:slack, :notification_url)

Twilio Messaging

  • Twilio has been renamed to :twilio_messaging to make room for :twilio_voice and other services they may provide in the future.
  • The format option has been renamed to json.

Vonage SMS

  • Vonage has been renamed to :vonage_sms to make room for other Vonage services in the future.
  • The format option has been renamed to json and is now required.