Skip to content

Commit

Permalink
feat: Add Public APIs for API Channel (chatwoot#2375)
Browse files Browse the repository at this point in the history
  • Loading branch information
sojan-official committed Jun 15, 2021
1 parent 0f377da commit 853db60
Show file tree
Hide file tree
Showing 29 changed files with 404 additions and 11 deletions.
1 change: 1 addition & 0 deletions .rubocop.yml
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ Rails/ApplicationController:
- 'app/controllers/widget_tests_controller.rb'
- 'app/controllers/widgets_controller.rb'
- 'app/controllers/platform_controller.rb'
- 'app/controllers/public_controller.rb'
Style/ClassAndModuleChildren:
EnforcedStyle: compact
Exclude:
Expand Down
7 changes: 4 additions & 3 deletions app/builders/contact_builder.rb
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
class ContactBuilder
pattr_initialize [:source_id!, :inbox!, :contact_attributes!]
pattr_initialize [:source_id!, :inbox!, :contact_attributes!, :hmac_verified]

def perform
contact_inbox = inbox.contact_inboxes.find_by(source_id: source_id)
Expand All @@ -18,7 +18,8 @@ def create_contact_inbox(contact)
::ContactInbox.create!(
contact_id: contact.id,
inbox_id: inbox.id,
source_id: source_id
source_id: source_id,
hmac_verified: hmac_verified || false
)
end

Expand All @@ -28,7 +29,7 @@ def update_contact_avatar(contact)

def create_contact
account.contacts.create!(
name: contact_attributes[:name],
name: contact_attributes[:name] || ::Haikunator.haikunate(1000),
phone_number: contact_attributes[:phone_number],
email: contact_attributes[:email],
identifier: contact_attributes[:identifier],
Expand Down
1 change: 1 addition & 0 deletions app/controllers/api/v1/widget/messages_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,7 @@ def message_update_params
end

def permitted_params
# timestamp parameter is used in create conversation method
params.permit(:id, :before, :website_token, contact: [:name, :email], message: [:content, :referer_url, :timestamp, :echo_id])
end

Expand Down
48 changes: 48 additions & 0 deletions app/controllers/public/api/v1/inboxes/contacts_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
class Public::Api::V1::Inboxes::ContactsController < Public::Api::V1::InboxesController
before_action :contact_inbox, except: [:create]
before_action :process_hmac

def create
source_id = params[:source_id] || SecureRandom.uuid
@contact_inbox = ::ContactBuilder.new(
source_id: source_id,
inbox: @inbox_channel.inbox,
contact_attributes: permitted_params.except(:identifier, :identifier_hash)
).perform
end

def show; end

def update
contact_identify_action = ContactIdentifyAction.new(
contact: @contact_inbox.contact,
params: permitted_params.to_h.deep_symbolize_keys.except(:identifier)
)
render json: contact_identify_action.perform
end

private

def contact_inbox
@contact_inbox = @inbox_channel.inbox.contact_inboxes.find_by!(source_id: params[:id])
end

def process_hmac
return if params[:identifier_hash].blank? && !@inbox_channel.hmac_mandatory
raise StandardError, 'HMAC failed: Invalid Identifier Hash Provided' unless valid_hmac?

@contact_inbox.update(hmac_verified: true) if @contact_inbox.present?
end

def valid_hmac?
params[:identifier_hash] == OpenSSL::HMAC.hexdigest(
'sha256',
@inbox_channel.hmac_token,
params[:identifier].to_s
)
end

def permitted_params
params.permit(:identifier, :identifier_hash, :email, :name, :avatar_url, custom_attributes: {})
end
end
24 changes: 24 additions & 0 deletions app/controllers/public/api/v1/inboxes/conversations_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
class Public::Api::V1::Inboxes::ConversationsController < Public::Api::V1::InboxesController
def index
@conversations = @contact_inbox.hmac_verified? ? @contact.conversations : @contact_inbox.conversations
end

def create
@conversation = create_conversation
end

private

def create_conversation
::Conversation.create!(conversation_params)
end

def conversation_params
{
account_id: @contact_inbox.contact.account_id,
inbox_id: @contact_inbox.inbox_id,
contact_id: @contact_inbox.contact_id,
contact_inbox_id: @contact_inbox.id
}
end
end
68 changes: 68 additions & 0 deletions app/controllers/public/api/v1/inboxes/messages_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
class Public::Api::V1::Inboxes::MessagesController < Public::Api::V1::InboxesController
before_action :set_message, only: [:update]

def index
@messages = @conversation.nil? ? [] : message_finder.perform
end

def create
@message = @conversation.messages.new(message_params)
@message.save
build_attachment
end

def update
@message.update!(message_update_params)
rescue StandardError => e
render json: { error: @contact.errors, message: e.message }.to_json, status: 500
end

private

def build_attachment
return if params[:attachments].blank?

params[:attachments].each do |uploaded_attachment|
attachment = @message.attachments.new(
account_id: @message.account_id,
file_type: helpers.file_type(uploaded_attachment&.content_type)
)
attachment.file.attach(uploaded_attachment)
end
@message.save!
end

def message_finder_params
{
filter_internal_messages: true,
before: params[:before]
}
end

def message_finder
@message_finder ||= MessageFinder.new(@conversation, message_finder_params)
end

def message_update_params
params.permit(submitted_values: [:name, :title, :value])
end

def permitted_params
params.permit(:content, :echo_id)
end

def set_message
@message = @conversation.messages.find(params[:id])
end

def message_params
{
account_id: @conversation.account_id,
sender: @contact_inbox.contact,
content: permitted_params[:content],
inbox_id: @conversation.inbox_id,
echo_id: permitted_params[:echo_id],
message_type: :incoming
}
end
end
23 changes: 23 additions & 0 deletions app/controllers/public/api/v1/inboxes_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
class Public::Api::V1::InboxesController < PublicController
before_action :set_inbox_channel
before_action :set_contact_inbox
before_action :set_conversation

private

def set_inbox_channel
@inbox_channel = ::Channel::Api.find_by!(identifier: params[:inbox_id])
end

def set_contact_inbox
return if params[:contact_id].blank?

@contact_inbox = @inbox_channel.inbox.contact_inboxes.find_by!(source_id: params[:contact_id])
end

def set_conversation
return if params[:conversation_id].blank?

@conversation = @contact_inbox.contact.conversations.find_by!(display_id: params[:conversation_id])
end
end
3 changes: 3 additions & 0 deletions app/controllers/public_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
class PublicController < ActionController::Base
skip_before_action :verify_authenticity_token
end
21 changes: 16 additions & 5 deletions app/models/channel/api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,19 @@
#
# Table name: channel_api
#
# id :bigint not null, primary key
# webhook_url :string
# created_at :datetime not null
# updated_at :datetime not null
# account_id :integer not null
# id :bigint not null, primary key
# hmac_mandatory :boolean default(FALSE)
# hmac_token :string
# identifier :string
# webhook_url :string
# created_at :datetime not null
# updated_at :datetime not null
# account_id :integer not null
#
# Indexes
#
# index_channel_api_on_hmac_token (hmac_token) UNIQUE
# index_channel_api_on_identifier (identifier) UNIQUE
#

class Channel::Api < ApplicationRecord
Expand All @@ -15,6 +23,9 @@ class Channel::Api < ApplicationRecord
validates :account_id, presence: true
belongs_to :account

has_secure_token :identifier
has_secure_token :hmac_token

has_one :inbox, as: :channel, dependent: :destroy

def name
Expand Down
1 change: 1 addition & 0 deletions app/models/channel/web_widget.rb
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
#
# id :integer not null, primary key
# feature_flags :integer default(3), not null
# hmac_mandatory :boolean default(FALSE)
# hmac_token :string
# pre_chat_form_enabled :boolean default(FALSE)
# pre_chat_form_options :jsonb
Expand Down
2 changes: 2 additions & 0 deletions app/views/public/api/v1/inboxes/contacts/create.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
json.source_id @contact_inbox.source_id
json.partial! 'public/api/v1/models/contact.json.jbuilder', resource: @contact_inbox.contact
2 changes: 2 additions & 0 deletions app/views/public/api/v1/inboxes/contacts/show.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
json.source_id @contact_inbox.source_id
json.partial! 'public/api/v1/models/contact.json.jbuilder', resource: @contact_inbox.contact
2 changes: 2 additions & 0 deletions app/views/public/api/v1/inboxes/contacts/update.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
json.source_id @contact_inbox.source_id
json.partial! 'public/api/v1/models/contact.json.jbuilder', resource: @contact_inbox.contact
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
json.partial! 'public/api/v1/models/conversation.json.jbuilder', resource: @conversation
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
json.array! @conversations do |conversation|
json.partial! 'public/api/v1/models/conversation.json.jbuilder', resource: conversation
end
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
json.partial! 'public/api/v1/models/message.json.jbuilder', resource: @message
3 changes: 3 additions & 0 deletions app/views/public/api/v1/inboxes/messages/index.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
json.array! @messages do |message|
json.partial! 'public/api/v1/models/message.json.jbuilder', resource: message
end
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
json.partial! 'public/api/v1/models/message.json.jbuilder', resource: @message
4 changes: 4 additions & 0 deletions app/views/public/api/v1/models/_contact.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
json.id resource.id
json.name resource.name
json.email resource.email
json.pubsub_token resource.pubsub_token
10 changes: 10 additions & 0 deletions app/views/public/api/v1/models/_conversation.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
json.id resource.display_id
json.inbox_id resource.inbox_id
json.contact_last_seen_at resource.contact_last_seen_at.to_i
json.status resource.status
json.messages do
json.array! resource.messages do |message|
json.partial! 'public/api/v1/models/message.json.jbuilder', resource: message
end
end
json.contact resource.contact
9 changes: 9 additions & 0 deletions app/views/public/api/v1/models/_message.json.jbuilder
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
json.id resource.id
json.content resource.content
json.message_type resource.message_type_before_type_cast
json.content_type resource.content_type
json.content_attributes resource.content_attributes
json.created_at resource.created_at.to_i
json.conversation_id resource.conversation.display_id
json.attachments resource.attachments.map(&:push_event_data) if resource.attachments.present?
json.sender resource.sender.push_event_data if resource.sender
3 changes: 3 additions & 0 deletions config/environments/development.rb
Original file line number Diff line number Diff line change
Expand Up @@ -85,4 +85,7 @@
resource '*', headers: :any, methods: :any, expose: ['access-token', 'client', 'uid', 'expiry']
end
end

# ref : https://medium.com/@emikaijuin/connecting-to-action-cable-without-rails-d39a8aaa52d5
config.action_cable.disable_request_forgery_protection = true
end
9 changes: 6 additions & 3 deletions config/environments/production.rb
Original file line number Diff line number Diff line change
Expand Up @@ -42,9 +42,12 @@
# Mount Action Cable outside main process or domain
# config.action_cable.mount_path = nil
# config.action_cable.url = 'wss://example.com/cable'
if ENV['FRONTEND_URL'].present?
config.action_cable.allowed_request_origins = [ENV['FRONTEND_URL'], %r{https?://#{URI.parse(ENV['FRONTEND_URL']).host}(:[0-9]+)?}]
end

# to enable connecting to the API channel public APIs
config.action_cable.disable_request_forgery_protection = true
# if ENV['FRONTEND_URL'].present?
# config.action_cable.allowed_request_origins = [ENV['FRONTEND_URL'], %r{https?://#{URI.parse(ENV['FRONTEND_URL']).host}(:[0-9]+)?}]
# end

# Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
config.force_ssl = ActiveModel::Type::Boolean.new.cast(ENV.fetch('FORCE_SSL', false))
Expand Down
18 changes: 18 additions & 0 deletions config/routes.rb
Original file line number Diff line number Diff line change
Expand Up @@ -199,6 +199,24 @@
end
end

# ----------------------------------------------------------------------
# Routes for inbox APIs Exposed to contacts
namespace :public, defaults: { format: 'json' } do
namespace :api do
namespace :v1 do
resources :inboxes do
scope module: :inboxes do
resources :contacts, only: [:create, :show, :update] do
resources :conversations, only: [:index, :create] do
resources :messages, only: [:index, :create, :update]
end
end
end
end
end
end
end

# ----------------------------------------------------------------------
# Used in mailer templates
resource :app, only: [:index] do
Expand Down
19 changes: 19 additions & 0 deletions db/migrate/20210602182058_add_hmac_to_api_channel.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
class AddHmacToApiChannel < ActiveRecord::Migration[6.0]
def change
add_column :channel_api, :identifier, :string
add_index :channel_api, :identifier, unique: true
add_column :channel_api, :hmac_token, :string
add_index :channel_api, :hmac_token, unique: true
add_column :channel_api, :hmac_mandatory, :boolean, default: false
add_column :channel_web_widgets, :hmac_mandatory, :boolean, default: false
set_up_existing_api_channels
end

def set_up_existing_api_channels
::Channel::Api.find_in_batches do |api_channels_batch|
Rails.logger.info "migrated till #{api_channels_batch.first.id}\n"
api_channels_batch.map(&:regenerate_hmac_token)
api_channels_batch.map(&:regenerate_identifier)
end
end
end
6 changes: 6 additions & 0 deletions db/schema.rb
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,11 @@
t.string "webhook_url"
t.datetime "created_at", precision: 6, null: false
t.datetime "updated_at", precision: 6, null: false
t.string "identifier"
t.string "hmac_token"
t.boolean "hmac_mandatory", default: false
t.index ["hmac_token"], name: "index_channel_api_on_hmac_token", unique: true
t.index ["identifier"], name: "index_channel_api_on_identifier", unique: true
end

create_table "channel_email", force: :cascade do |t|
Expand Down Expand Up @@ -201,6 +206,7 @@
t.string "hmac_token"
t.boolean "pre_chat_form_enabled", default: false
t.jsonb "pre_chat_form_options", default: {}
t.boolean "hmac_mandatory", default: false
t.index ["hmac_token"], name: "index_channel_web_widgets_on_hmac_token", unique: true
t.index ["website_token"], name: "index_channel_web_widgets_on_website_token", unique: true
end
Expand Down
Loading

0 comments on commit 853db60

Please sign in to comment.