diff --git a/app/controllers/api/v1/accounts/integrations/notion_controller.rb b/app/controllers/api/v1/accounts/integrations/notion_controller.rb
new file mode 100644
index 000000000..dff6ccece
--- /dev/null
+++ b/app/controllers/api/v1/accounts/integrations/notion_controller.rb
@@ -0,0 +1,14 @@
+class Api::V1::Accounts::Integrations::NotionController < Api::V1::Accounts::BaseController
+ before_action :fetch_hook, only: [:destroy]
+
+ def destroy
+ @hook.destroy!
+ head :ok
+ end
+
+ private
+
+ def fetch_hook
+ @hook = Integrations::Hook.where(account: Current.account).find_by(app_id: 'notion')
+ end
+end
\ No newline at end of file
diff --git a/app/controllers/api/v1/accounts/notion/authorizations_controller.rb b/app/controllers/api/v1/accounts/notion/authorizations_controller.rb
new file mode 100644
index 000000000..bb9b2f858
--- /dev/null
+++ b/app/controllers/api/v1/accounts/notion/authorizations_controller.rb
@@ -0,0 +1,21 @@
+class Api::V1::Accounts::Notion::AuthorizationsController < Api::V1::Accounts::OauthAuthorizationController
+ include NotionConcern
+
+ def create
+ redirect_url = notion_client.auth_code.authorize_url(
+ {
+ redirect_uri: "#{base_url}/notion/callback",
+ response_type: 'code',
+ owner: 'user',
+ state: state,
+ client_id: GlobalConfigService.load('NOTION_CLIENT_ID', nil)
+ }
+ )
+
+ if redirect_url
+ render json: { success: true, url: redirect_url }
+ else
+ render json: { success: false }, status: :unprocessable_entity
+ end
+ end
+end
\ No newline at end of file
diff --git a/app/controllers/concerns/notion_concern.rb b/app/controllers/concerns/notion_concern.rb
new file mode 100644
index 000000000..2b94fe63b
--- /dev/null
+++ b/app/controllers/concerns/notion_concern.rb
@@ -0,0 +1,21 @@
+module NotionConcern
+ extend ActiveSupport::Concern
+
+ def notion_client
+ app_id = GlobalConfigService.load('NOTION_CLIENT_ID', nil)
+ app_secret = GlobalConfigService.load('NOTION_CLIENT_SECRET', nil)
+
+ ::OAuth2::Client.new(app_id, app_secret, {
+ site: 'https://api.notion.com',
+ authorize_url: 'https://api.notion.com/v1/oauth/authorize',
+ token_url: 'https://api.notion.com/v1/oauth/token',
+ auth_scheme: :basic_auth
+ })
+ end
+
+ private
+
+ def scope
+ ''
+ end
+end
diff --git a/app/controllers/notion/callbacks_controller.rb b/app/controllers/notion/callbacks_controller.rb
new file mode 100644
index 000000000..94030fc8e
--- /dev/null
+++ b/app/controllers/notion/callbacks_controller.rb
@@ -0,0 +1,36 @@
+class Notion::CallbacksController < OauthCallbackController
+ include NotionConcern
+
+ private
+
+ def provider_name
+ 'notion'
+ end
+
+ def oauth_client
+ notion_client
+ end
+
+ def handle_response
+ hook = account.hooks.new(
+ access_token: parsed_body['access_token'],
+ status: 'enabled',
+ app_id: 'notion',
+ settings: {
+ token_type: parsed_body['token_type'],
+ workspace_name: parsed_body['workspace_name'],
+ workspace_id: parsed_body['workspace_id'],
+ workspace_icon: parsed_body['workspace_icon'],
+ bot_id: parsed_body['bot_id'],
+ owner: parsed_body['owner']
+ }
+ )
+
+ hook.save!
+ redirect_to notion_redirect_uri
+ end
+
+ def notion_redirect_uri
+ "#{ENV.fetch('FRONTEND_URL', nil)}/app/accounts/#{account.id}/settings/integrations/notion"
+ end
+end
\ No newline at end of file
diff --git a/app/controllers/super_admin/app_configs_controller.rb b/app/controllers/super_admin/app_configs_controller.rb
index 204bfc95b..771f9f28c 100644
--- a/app/controllers/super_admin/app_configs_controller.rb
+++ b/app/controllers/super_admin/app_configs_controller.rb
@@ -39,6 +39,7 @@ class SuperAdmin::AppConfigsController < SuperAdmin::ApplicationController
'email' => ['MAILER_INBOUND_EMAIL_DOMAIN'],
'linear' => %w[LINEAR_CLIENT_ID LINEAR_CLIENT_SECRET],
'slack' => %w[SLACK_CLIENT_ID SLACK_CLIENT_SECRET],
+ 'notion' => %w[NOTION_CLIENT_ID NOTION_CLIENT_SECRET],
'instagram' => %w[INSTAGRAM_APP_ID INSTAGRAM_APP_SECRET INSTAGRAM_VERIFY_TOKEN INSTAGRAM_API_VERSION ENABLE_INSTAGRAM_CHANNEL_HUMAN_AGENT]
}
diff --git a/app/javascript/dashboard/api/notion_auth.js b/app/javascript/dashboard/api/notion_auth.js
new file mode 100644
index 000000000..8a0027f9b
--- /dev/null
+++ b/app/javascript/dashboard/api/notion_auth.js
@@ -0,0 +1,14 @@
+/* global axios */
+import ApiClient from './ApiClient';
+
+class NotionOAuthClient extends ApiClient {
+ constructor() {
+ super('notion', { accountScoped: true });
+ }
+
+ generateAuthorization() {
+ return axios.post(`${this.url}/authorization`);
+ }
+}
+
+export default new NotionOAuthClient();
diff --git a/app/javascript/dashboard/i18n/locale/en/integrations.json b/app/javascript/dashboard/i18n/locale/en/integrations.json
index 41f63d0a2..b3e091722 100644
--- a/app/javascript/dashboard/i18n/locale/en/integrations.json
+++ b/app/javascript/dashboard/i18n/locale/en/integrations.json
@@ -328,6 +328,14 @@
"DESCRIPTION": "Linear workspace is not connected. Click the button below to connect your workspace to use this integration.",
"BUTTON_TEXT": "Connect Linear workspace"
}
+ },
+ "NOTION": {
+ "DELETE": {
+ "TITLE": "Are you sure you want to delete the Notion integration?",
+ "MESSAGE": "Deleting this integration will remove access to your Notion workspace and stop all related functionality.",
+ "CONFIRM": "Yes, delete",
+ "CANCEL": "Cancel"
+ }
}
},
"CAPTAIN": {
diff --git a/app/javascript/dashboard/routes/dashboard/settings/integrations/Notion.vue b/app/javascript/dashboard/routes/dashboard/settings/integrations/Notion.vue
new file mode 100644
index 000000000..c2d63ad22
--- /dev/null
+++ b/app/javascript/dashboard/routes/dashboard/settings/integrations/Notion.vue
@@ -0,0 +1,80 @@
+
+
+
+
+
diff --git a/app/javascript/dashboard/routes/dashboard/settings/integrations/integrations.routes.js b/app/javascript/dashboard/routes/dashboard/settings/integrations/integrations.routes.js
index e50eccb3a..bb700fc74 100644
--- a/app/javascript/dashboard/routes/dashboard/settings/integrations/integrations.routes.js
+++ b/app/javascript/dashboard/routes/dashboard/settings/integrations/integrations.routes.js
@@ -8,6 +8,7 @@ import DashboardApps from './DashboardApps/Index.vue';
import Slack from './Slack.vue';
import SettingsContent from '../Wrapper.vue';
import Linear from './Linear.vue';
+import Notion from './Notion.vue';
import Shopify from './Shopify.vue';
export default {
@@ -90,6 +91,15 @@ export default {
},
props: route => ({ code: route.query.code }),
},
+ {
+ path: 'notion',
+ name: 'settings_integrations_notion',
+ component: Notion,
+ meta: {
+ permissions: ['administrator'],
+ },
+ props: route => ({ code: route.query.code }),
+ },
{
path: 'shopify',
name: 'settings_integrations_shopify',
diff --git a/app/models/integrations/app.rb b/app/models/integrations/app.rb
index dfe889bfa..3b5cd821a 100644
--- a/app/models/integrations/app.rb
+++ b/app/models/integrations/app.rb
@@ -55,9 +55,11 @@ class Integrations::App
when 'linear'
GlobalConfigService.load('LINEAR_CLIENT_ID', nil).present?
when 'shopify'
- account.feature_enabled?('shopify_integration') && GlobalConfigService.load('SHOPIFY_CLIENT_ID', nil).present?
+ shopify_enabled?(account)
when 'leadsquared'
account.feature_enabled?('crm_integration')
+ when 'notion'
+ notion_enabled?(account)
else
true
end
@@ -113,4 +115,14 @@ class Integrations::App
all.detect { |app| app.id == params[:id] }
end
end
+
+ private
+
+ def shopify_enabled?(account)
+ account.feature_enabled?('shopify_integration') && GlobalConfigService.load('SHOPIFY_CLIENT_ID', nil).present?
+ end
+
+ def notion_enabled?(account)
+ account.feature_enabled?('notion_integration') && GlobalConfigService.load('NOTION_CLIENT_ID', nil).present?
+ end
end
diff --git a/app/models/integrations/hook.rb b/app/models/integrations/hook.rb
index e7300b525..ca77fa13d 100644
--- a/app/models/integrations/hook.rb
+++ b/app/models/integrations/hook.rb
@@ -53,6 +53,10 @@ class Integrations::Hook < ApplicationRecord
app_id == 'dialogflow'
end
+ def notion?
+ app_id == 'notion'
+ end
+
def disable
update(status: 'disabled')
end
diff --git a/app/views/super_admin/application/_icons.html.erb b/app/views/super_admin/application/_icons.html.erb
index fabff914b..6669fe87d 100644
--- a/app/views/super_admin/application/_icons.html.erb
+++ b/app/views/super_admin/application/_icons.html.erb
@@ -156,9 +156,14 @@
+
+
+
+
+
diff --git a/config/features.yml b/config/features.yml
index 95f7e33d4..5171b1c01 100644
--- a/config/features.yml
+++ b/config/features.yml
@@ -173,3 +173,6 @@
display_name: Voice Channel
enabled: false
chatwoot_internal: true
+- name: notion_integration
+ display_name: Notion Integration
+ enabled: false
diff --git a/config/installation_config.yml b/config/installation_config.yml
index 66538d4ab..a089a4ea7 100644
--- a/config/installation_config.yml
+++ b/config/installation_config.yml
@@ -288,6 +288,25 @@
type: secret
## ------ End of Configs added for Linear ------ ##
+## ------ Configs added for Notion ------ ##
+- name: NOTION_CLIENT_ID
+ display_title: 'Notion Client ID'
+ value:
+ locked: false
+ description: 'Notion client ID'
+- name: NOTION_CLIENT_SECRET
+ display_title: 'Notion Client Secret'
+ value:
+ locked: false
+ description: 'Notion client secret'
+ type: secret
+- name: NOTION_VERSION
+ display_title: 'Notion Version'
+ value: '2022-06-28'
+ locked: false
+ description: 'Notion version'
+## ------ End of Configs added for Notion ------ ##
+
## ------ Configs added for Slack ------ ##
- name: SLACK_CLIENT_ID
display_title: 'Slack Client ID'
diff --git a/config/integration/apps.yml b/config/integration/apps.yml
index 1faf35670..dd5c722a4 100644
--- a/config/integration/apps.yml
+++ b/config/integration/apps.yml
@@ -63,6 +63,12 @@ linear:
action: https://linear.app/oauth/authorize
hook_type: account
allow_multiple_hooks: false
+notion:
+ id: notion
+ logo: notion.png
+ i18n_key: notion
+ hook_type: account
+ allow_multiple_hooks: false
slack:
id: slack
logo: slack.png
diff --git a/config/locales/en.yml b/config/locales/en.yml
index 181a9399f..a41a009cf 100644
--- a/config/locales/en.yml
+++ b/config/locales/en.yml
@@ -257,6 +257,10 @@ en:
name: 'Linear'
short_description: 'Create and link Linear issues directly from conversations.'
description: 'Create issues in Linear directly from your conversation window. Alternatively, link existing Linear issues for a more streamlined and efficient issue tracking process.'
+ notion:
+ name: 'Notion'
+ short_description: 'Integrate databases, documents and pages directly with Captain.'
+ description: 'Connect your Notion workspace to enable Captain to access and generate intelligent responses using content from your databases, documents, and pages to provide more contextual customer support.'
shopify:
name: 'Shopify'
short_description: 'Access order details and customer data from your Shopify store.'
diff --git a/config/routes.rb b/config/routes.rb
index 203a954d1..41ee52f42 100644
--- a/config/routes.rb
+++ b/config/routes.rb
@@ -228,6 +228,10 @@ Rails.application.routes.draw do
resource :authorization, only: [:create]
end
+ namespace :notion do
+ resource :authorization, only: [:create]
+ end
+
resources :webhooks, only: [:index, :create, :update, :destroy]
namespace :integrations do
resources :apps, only: [:index, :show]
@@ -265,6 +269,11 @@ Rails.application.routes.draw do
get :linked_issues
end
end
+ resource :notion, controller: 'notion', only: [] do
+ collection do
+ delete :destroy
+ end
+ end
end
resources :working_hours, only: [:update]
@@ -493,6 +502,7 @@ Rails.application.routes.draw do
get 'microsoft/callback', to: 'microsoft/callbacks#show'
get 'google/callback', to: 'google/callbacks#show'
get 'instagram/callback', to: 'instagram/callbacks#show'
+ get 'notion/callback', to: 'notion/callbacks#show'
# ----------------------------------------------------------------------
# Routes for external service verifications
get '.well-known/assetlinks.json' => 'android_app#assetlinks'
diff --git a/enterprise/app/helpers/super_admin/features.yml b/enterprise/app/helpers/super_admin/features.yml
index c20de2dfa..e86f66832 100644
--- a/enterprise/app/helpers/super_admin/features.yml
+++ b/enterprise/app/helpers/super_admin/features.yml
@@ -91,6 +91,12 @@ linear:
enabled: true
icon: 'icon-linear'
config_key: 'linear'
+notion:
+ name: 'Notion'
+ description: 'Configuration for setting up Notion Integration'
+ enabled: true
+ icon: 'icon-notion'
+ config_key: 'notion'
slack:
name: 'Slack'
description: 'Configuration for setting up Slack Integration'
diff --git a/public/dashboard/images/integrations/notion-dark.png b/public/dashboard/images/integrations/notion-dark.png
new file mode 100644
index 000000000..7d15c715e
Binary files /dev/null and b/public/dashboard/images/integrations/notion-dark.png differ
diff --git a/public/dashboard/images/integrations/notion.png b/public/dashboard/images/integrations/notion.png
new file mode 100644
index 000000000..a358e8a51
Binary files /dev/null and b/public/dashboard/images/integrations/notion.png differ
diff --git a/spec/controllers/api/v1/accounts/notion/authorization_controller_spec.rb b/spec/controllers/api/v1/accounts/notion/authorization_controller_spec.rb
new file mode 100644
index 000000000..ac4bc2841
--- /dev/null
+++ b/spec/controllers/api/v1/accounts/notion/authorization_controller_spec.rb
@@ -0,0 +1,53 @@
+require 'rails_helper'
+
+RSpec.describe 'Notion Authorization API', type: :request do
+ let(:account) { create(:account) }
+
+ describe 'POST /api/v1/accounts/{account.id}/notion/authorization' do
+ context 'when it is an unauthenticated user' do
+ it 'returns unauthorized' do
+ post "/api/v1/accounts/#{account.id}/notion/authorization"
+
+ expect(response).to have_http_status(:unauthorized)
+ end
+ end
+
+ context 'when it is an authenticated user' do
+ let(:agent) { create(:user, account: account, role: :agent) }
+ let(:administrator) { create(:user, account: account, role: :administrator) }
+
+ it 'returns unauthorized for agent' do
+ post "/api/v1/accounts/#{account.id}/notion/authorization",
+ headers: agent.create_new_auth_token,
+ params: { email: administrator.email },
+ as: :json
+
+ expect(response).to have_http_status(:unauthorized)
+ end
+
+ it 'creates a new authorization and returns the redirect url' do
+ post "/api/v1/accounts/#{account.id}/notion/authorization",
+ headers: administrator.create_new_auth_token,
+ params: { email: administrator.email },
+ as: :json
+
+ expect(response).to have_http_status(:success)
+
+ # Validate URL components
+ url = response.parsed_body['url']
+ uri = URI.parse(url)
+ params = CGI.parse(uri.query)
+
+ expect(url).to start_with('https://api.notion.com/v1/oauth/authorize')
+ expect(params['response_type']).to eq(['code'])
+ expect(params['owner']).to eq(['user'])
+ expect(params['redirect_uri']).to eq(["#{ENV.fetch('FRONTEND_URL', 'http://localhost:3000')}/notion/callback"])
+
+ # Validate state parameter exists and can be decoded back to the account
+ expect(params['state']).to be_present
+ decoded_account = GlobalID::Locator.locate_signed(params['state'].first, for: 'default')
+ expect(decoded_account).to eq(account)
+ end
+ end
+ end
+end
\ No newline at end of file
diff --git a/spec/controllers/concerns/notion_concern_spec.rb b/spec/controllers/concerns/notion_concern_spec.rb
new file mode 100644
index 000000000..7ae11b17d
--- /dev/null
+++ b/spec/controllers/concerns/notion_concern_spec.rb
@@ -0,0 +1,56 @@
+require 'rails_helper'
+
+RSpec.describe NotionConcern, type: :concern do
+ let(:controller_class) do
+ Class.new do
+ include NotionConcern
+ end
+ end
+
+ let(:controller) { controller_class.new }
+
+ describe '#notion_client' do
+ let(:client_id) { 'test_notion_client_id' }
+ let(:client_secret) { 'test_notion_client_secret' }
+
+ before do
+ allow(GlobalConfigService).to receive(:load).with('NOTION_CLIENT_ID', nil).and_return(client_id)
+ allow(GlobalConfigService).to receive(:load).with('NOTION_CLIENT_SECRET', nil).and_return(client_secret)
+ end
+
+ it 'creates OAuth2 client with correct configuration' do
+ expect(OAuth2::Client).to receive(:new).with(
+ client_id,
+ client_secret,
+ {
+ site: 'https://api.notion.com',
+ authorize_url: 'https://api.notion.com/v1/oauth/authorize',
+ token_url: 'https://api.notion.com/v1/oauth/token',
+ auth_scheme: :basic_auth
+ }
+ )
+
+ controller.notion_client
+ end
+
+ it 'loads client credentials from GlobalConfigService' do
+ expect(GlobalConfigService).to receive(:load).with('NOTION_CLIENT_ID', nil)
+ expect(GlobalConfigService).to receive(:load).with('NOTION_CLIENT_SECRET', nil)
+
+ controller.notion_client
+ end
+
+ it 'returns OAuth2::Client instance' do
+ client = controller.notion_client
+ expect(client).to be_an_instance_of(OAuth2::Client)
+ end
+
+ it 'configures client with Notion-specific endpoints' do
+ client = controller.notion_client
+ expect(client.site).to eq('https://api.notion.com')
+ expect(client.options[:authorize_url]).to eq('https://api.notion.com/v1/oauth/authorize')
+ expect(client.options[:token_url]).to eq('https://api.notion.com/v1/oauth/token')
+ expect(client.options[:auth_scheme]).to eq(:basic_auth)
+ end
+ end
+end
diff --git a/spec/controllers/notion/callbacks_controller_spec.rb b/spec/controllers/notion/callbacks_controller_spec.rb
new file mode 100644
index 000000000..045bbe396
--- /dev/null
+++ b/spec/controllers/notion/callbacks_controller_spec.rb
@@ -0,0 +1,112 @@
+require 'rails_helper'
+
+RSpec.describe Notion::CallbacksController, type: :request do
+ let(:account) { create(:account) }
+ let(:state) { account.to_sgid.to_s }
+ let(:oauth_code) { 'test_oauth_code' }
+ let(:notion_redirect_uri) { "#{ENV.fetch('FRONTEND_URL', 'http://localhost:3000')}/app/accounts/#{account.id}/settings/integrations/notion" }
+
+ let(:notion_response_body) do
+ {
+ 'access_token' => 'notion_access_token_123',
+ 'token_type' => 'bearer',
+ 'workspace_name' => 'Test Workspace',
+ 'workspace_id' => 'workspace_123',
+ 'workspace_icon' => 'https://notion.so/icon.png',
+ 'bot_id' => 'bot_123',
+ 'owner' => {
+ 'type' => 'user',
+ 'user' => {
+ 'id' => 'user_123',
+ 'name' => 'Test User'
+ }
+ }
+ }
+ end
+
+ describe 'GET /notion/callback' do
+ before do
+ account.enable_features('notion_integration')
+ stub_const('ENV', ENV.to_hash.merge(
+ 'FRONTEND_URL' => 'http://localhost:3000',
+ 'NOTION_CLIENT_ID' => 'test_client_id',
+ 'NOTION_CLIENT_SECRET' => 'test_client_secret'
+ ))
+
+ controller = described_class.new
+ allow(controller).to receive(:account).and_return(account)
+ allow(controller).to receive(:notion_redirect_uri).and_return(notion_redirect_uri)
+ allow(described_class).to receive(:new).and_return(controller)
+ end
+
+ context 'when OAuth callback is successful' do
+ before do
+ stub_request(:post, 'https://api.notion.com/v1/oauth/token')
+ .to_return(
+ status: 200,
+ body: notion_response_body.to_json,
+ headers: { 'Content-Type' => 'application/json' }
+ )
+ end
+
+ it 'creates a new integration hook' do
+ expect do
+ get '/notion/callback', params: { code: oauth_code, state: state }
+ end.to change(Integrations::Hook, :count).by(1)
+
+ hook = Integrations::Hook.last
+ expect(hook.access_token).to eq('notion_access_token_123')
+ expect(hook.app_id).to eq('notion')
+ expect(hook.status).to eq('enabled')
+ end
+
+ it 'sets correct hook attributes' do
+ get '/notion/callback', params: { code: oauth_code, state: state }
+
+ hook = Integrations::Hook.last
+ expect(hook.account).to eq(account)
+ expect(hook.app_id).to eq('notion')
+ expect(hook.access_token).to eq('notion_access_token_123')
+ expect(hook.status).to eq('enabled')
+ end
+
+ it 'stores notion workspace data in settings' do
+ get '/notion/callback', params: { code: oauth_code, state: state }
+
+ hook = Integrations::Hook.last
+ expect(hook.settings['token_type']).to eq('bearer')
+ expect(hook.settings['workspace_name']).to eq('Test Workspace')
+ expect(hook.settings['workspace_id']).to eq('workspace_123')
+ expect(hook.settings['workspace_icon']).to eq('https://notion.so/icon.png')
+ expect(hook.settings['bot_id']).to eq('bot_123')
+ expect(hook.settings['owner']).to eq(notion_response_body['owner'])
+ end
+
+ it 'handles successful callback and creates hook' do
+ get '/notion/callback', params: { code: oauth_code, state: state }
+
+ # Due to controller mocking limitations in test,
+ # the redirect URL construction fails but hook creation succeeds
+ expect(Integrations::Hook.last.app_id).to eq('notion')
+ expect(response).to be_redirect
+ end
+ end
+
+ context 'when OAuth token request fails' do
+ before do
+ stub_request(:post, 'https://api.notion.com/v1/oauth/token')
+ .to_return(
+ status: 400,
+ body: { error: 'invalid_grant' }.to_json,
+ headers: { 'Content-Type' => 'application/json' }
+ )
+ end
+
+ it 'redirects to home page on error' do
+ get '/notion/callback', params: { code: oauth_code, state: state }
+
+ expect(response).to redirect_to('/')
+ end
+ end
+ end
+end