chore: Update analytics events (#6050)

This commit is contained in:
Pranav Raj S 2022-12-08 20:53:13 -08:00 committed by GitHub
parent 7dc790a7e0
commit 6200559123
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
17 changed files with 1108 additions and 55 deletions

View file

@ -24,7 +24,6 @@ class DashboardController < ActionController::Base
'API_CHANNEL_NAME', 'API_CHANNEL_NAME',
'API_CHANNEL_THUMBNAIL', 'API_CHANNEL_THUMBNAIL',
'ANALYTICS_TOKEN', 'ANALYTICS_TOKEN',
'ANALYTICS_HOST',
'DIRECT_UPLOADS_ENABLED', 'DIRECT_UPLOADS_ENABLED',
'HCAPTCHA_SITE_KEY', 'HCAPTCHA_SITE_KEY',
'LOGOUT_REDIRECT_LINK', 'LOGOUT_REDIRECT_LINK',

View file

@ -47,6 +47,9 @@ import {
import eventListenerMixins from 'shared/mixins/eventListenerMixins'; import eventListenerMixins from 'shared/mixins/eventListenerMixins';
import uiSettingsMixin from 'dashboard/mixins/uiSettings'; import uiSettingsMixin from 'dashboard/mixins/uiSettings';
import { isEditorHotKeyEnabled } from 'dashboard/mixins/uiSettings'; import { isEditorHotKeyEnabled } from 'dashboard/mixins/uiSettings';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../../helper/AnalyticsHelper';
const createState = (content, placeholder, plugins = []) => { const createState = (content, placeholder, plugins = []) => {
return EditorState.create({ return EditorState.create({
@ -268,6 +271,7 @@ export default {
); );
this.state = this.editorView.state.apply(tr); this.state = this.editorView.state.apply(tr);
this.emitOnChange(); this.emitOnChange();
AnalyticsHelper.track(ANALYTICS_EVENTS.USED_MENTIONS);
return false; return false;
}, },
@ -297,6 +301,7 @@ export default {
this.emitOnChange(); this.emitOnChange();
tr.scrollIntoView(); tr.scrollIntoView();
AnalyticsHelper.track(ANALYTICS_EVENTS.INSERTED_A_CANNED_RESPONSE);
return false; return false;
}, },

View file

@ -161,6 +161,9 @@ import { LocalStorage, LOCAL_STORAGE_KEYS } from '../../../helper/localStorage';
import { trimContent, debounce } from '@chatwoot/utils'; import { trimContent, debounce } from '@chatwoot/utils';
import wootConstants from 'dashboard/constants'; import wootConstants from 'dashboard/constants';
import { isEditorHotKeyEnabled } from 'dashboard/mixins/uiSettings'; import { isEditorHotKeyEnabled } from 'dashboard/mixins/uiSettings';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../../helper/AnalyticsHelper';
const EmojiInput = () => import('shared/components/emoji/EmojiInput'); const EmojiInput = () => import('shared/components/emoji/EmojiInput');
@ -698,6 +701,7 @@ export default {
}, },
replaceText(message) { replaceText(message) {
setTimeout(() => { setTimeout(() => {
AnalyticsHelper.track(ANALYTICS_EVENTS.INSERTED_A_CANNED_RESPONSE);
this.message = message; this.message = message;
}, 100); }, 100);
}, },

View file

@ -0,0 +1,9 @@
export const EXECUTED_A_MACRO = 'Executed a macro';
export const SENT_MESSAGE = 'Sent a message';
export const SENT_PRIVATE_NOTE = 'Sent a private note';
export const INSERTED_A_CANNED_RESPONSE = 'Inserted a canned response';
export const USED_MENTIONS = 'Used mentions';
export const MERGED_CONTACTS = 'Used merge contact option';
export const ADDED_TO_CANNED_RESPONSE = 'Used added to canned response option';
export const ADDED_A_CUSTOM_ATTRIBUTE = 'Added a custom attribute';
export const ADDED_AN_INBOX = 'Added an inbox';

View file

@ -0,0 +1,67 @@
import { AnalyticsBrowser } from '@june-so/analytics-next';
class AnalyticsHelper {
constructor({ token: analyticsToken } = {}) {
this.analyticsToken = analyticsToken;
this.analytics = null;
this.user = {};
}
async init() {
if (!this.analyticsToken) {
return;
}
let [analytics] = await AnalyticsBrowser.load({
writeKey: this.analyticsToken,
});
this.analytics = analytics;
}
identify(user) {
if (!this.analytics) {
return;
}
this.user = user;
this.analytics.identify(this.user.email, {
userId: this.user.id,
email: this.user.email,
name: this.user.name,
avatar: this.user.avatar_url,
});
const { accounts, account_id: accountId } = this.user;
const [currentAccount] = accounts.filter(
account => account.id === accountId
);
if (currentAccount) {
this.analytics.group(currentAccount.id, this.user.id, {
name: currentAccount.name,
});
}
}
track(eventName, properties = {}) {
if (!this.analytics) {
return;
}
this.analytics.track({
userId: this.user.id,
event: eventName,
properties,
});
}
page(params) {
if (!this.analytics) {
return;
}
this.analytics.page(params);
}
}
export * as ANALYTICS_EVENTS from './events';
export default new AnalyticsHelper(window.analyticsConfig);

View file

@ -1,4 +1,4 @@
import posthog from 'posthog-js'; import AnalyticsHelper from './AnalyticsHelper';
export const CHATWOOT_SET_USER = 'CHATWOOT_SET_USER'; export const CHATWOOT_SET_USER = 'CHATWOOT_SET_USER';
export const CHATWOOT_RESET = 'CHATWOOT_RESET'; export const CHATWOOT_RESET = 'CHATWOOT_RESET';
@ -8,16 +8,9 @@ export const ANALYTICS_RESET = 'ANALYTICS_RESET';
export const initializeAnalyticsEvents = () => { export const initializeAnalyticsEvents = () => {
window.bus.$on(ANALYTICS_IDENTITY, ({ user }) => { window.bus.$on(ANALYTICS_IDENTITY, ({ user }) => {
if (window.analyticsConfig) { AnalyticsHelper.identify(user);
posthog.identify(user.id, { name: user.name, email: user.email });
}
});
window.bus.$on(ANALYTICS_RESET, () => {
if (window.analyticsConfig) {
posthog.reset();
}
}); });
window.bus.$on(ANALYTICS_RESET, () => {});
}; };
export const initializeChatwootEvents = () => { export const initializeChatwootEvents = () => {

View file

@ -24,6 +24,9 @@ import MergeContact from 'dashboard/modules/contact/components/MergeContact';
import ContactAPI from 'dashboard/api/contacts'; import ContactAPI from 'dashboard/api/contacts';
import { mapGetters } from 'vuex'; import { mapGetters } from 'vuex';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../helper/AnalyticsHelper';
export default { export default {
components: { MergeContact }, components: { MergeContact },
@ -72,6 +75,7 @@ export default {
} }
}, },
async onMergeContacts(childContactId) { async onMergeContacts(childContactId) {
AnalyticsHelper.track(ANALYTICS_EVENTS.MERGED_CONTACTS);
try { try {
await this.$store.dispatch('contacts/merge', { await this.$store.dispatch('contacts/merge', {
childId: childContactId, childId: childContactId,

View file

@ -72,6 +72,9 @@ import AddCannedModal from 'dashboard/routes/dashboard/settings/canned/AddCanned
import WootDropdownItem from 'shared/components/ui/dropdown/DropdownItem'; import WootDropdownItem from 'shared/components/ui/dropdown/DropdownItem';
import WootDropdownMenu from 'shared/components/ui/dropdown/DropdownMenu'; import WootDropdownMenu from 'shared/components/ui/dropdown/DropdownMenu';
import { copyTextToClipboard } from 'shared/helpers/clipboard'; import { copyTextToClipboard } from 'shared/helpers/clipboard';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../../helper/AnalyticsHelper';
export default { export default {
components: { components: {
@ -127,6 +130,7 @@ export default {
this.$emit('toggle', false); this.$emit('toggle', false);
}, },
showCannedResponseModal() { showCannedResponseModal() {
AnalyticsHelper.track(ANALYTICS_EVENTS.ADDED_TO_CANNED_RESPONSE);
this.isCannedResponseModalOpen = true; this.isCannedResponseModalOpen = true;
}, },
}, },

View file

@ -35,6 +35,9 @@
import alertMixin from 'shared/mixins/alertMixin'; import alertMixin from 'shared/mixins/alertMixin';
import { mixin as clickaway } from 'vue-clickaway'; import { mixin as clickaway } from 'vue-clickaway';
import MacroPreview from './MacroPreview'; import MacroPreview from './MacroPreview';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../../../helper/AnalyticsHelper';
export default { export default {
components: { components: {
MacroPreview, MacroPreview,
@ -64,6 +67,7 @@ export default {
macroId: macro.id, macroId: macro.id,
conversationIds: [this.conversationId], conversationIds: [this.conversationId],
}); });
AnalyticsHelper.track(ANALYTICS_EVENTS.EXECUTED_A_MACRO);
this.showAlert(this.$t('MACROS.EXECUTE.EXECUTED_SUCCESSFULLY')); this.showAlert(this.$t('MACROS.EXECUTE.EXECUTED_SUCCESSFULLY'));
} catch (error) { } catch (error) {
this.showAlert(this.$t('MACROS.ERROR')); this.showAlert(this.$t('MACROS.ERROR'));

View file

@ -7,6 +7,7 @@ import dashboard from './dashboard/dashboard.routes';
import login from './login/login.routes'; import login from './login/login.routes';
import store from '../store'; import store from '../store';
import { validateLoggedInRoutes } from '../helper/routeHelpers'; import { validateLoggedInRoutes } from '../helper/routeHelpers';
import AnalyticsHelper from '../helper/AnalyticsHelper';
const routes = [...login.routes, ...dashboard.routes, ...authRoute.routes]; const routes = [...login.routes, ...dashboard.routes, ...authRoute.routes];
@ -117,6 +118,11 @@ export const validateRouteAccess = (to, from, next, { getters }) => {
export const initalizeRouter = () => { export const initalizeRouter = () => {
const userAuthentication = store.dispatch('setUser'); const userAuthentication = store.dispatch('setUser');
router.beforeEach((to, from, next) => { router.beforeEach((to, from, next) => {
AnalyticsHelper.page(to.name || '', {
path: to.path,
name: to.name,
});
if (validateSSOLoginParams(to)) { if (validateSSOLoginParams(to)) {
clearBrowserSessionCookies(); clearBrowserSessionCookies();
next(); next();

View file

@ -10,6 +10,9 @@ import {
isOnUnattendedView, isOnUnattendedView,
} from './helpers/actionHelpers'; } from './helpers/actionHelpers';
import messageReadActions from './actions/messageReadActions'; import messageReadActions from './actions/messageReadActions';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../../helper/AnalyticsHelper';
// actions // actions
const actions = { const actions = {
getConversation: async ({ commit }, conversationId) => { getConversation: async ({ commit }, conversationId) => {
@ -171,6 +174,11 @@ const actions = {
status: MESSAGE_STATUS.PROGRESS, status: MESSAGE_STATUS.PROGRESS,
}); });
const response = await MessageApi.create(pendingMessage); const response = await MessageApi.create(pendingMessage);
AnalyticsHelper.track(
pendingMessage.private
? ANALYTICS_EVENTS.SENT_PRIVATE_NOTE
: ANALYTICS_EVENTS.SENT_MESSAGE
);
commit(types.ADD_MESSAGE, { commit(types.ADD_MESSAGE, {
...response.data, ...response.data,
status: MESSAGE_STATUS.SENT, status: MESSAGE_STATUS.SENT,

View file

@ -6,6 +6,9 @@ import WebChannel from '../../api/channel/webChannel';
import FBChannel from '../../api/channel/fbChannel'; import FBChannel from '../../api/channel/fbChannel';
import TwilioChannel from '../../api/channel/twilioChannel'; import TwilioChannel from '../../api/channel/twilioChannel';
import { throwErrorMessage } from '../utils/api'; import { throwErrorMessage } from '../utils/api';
import AnalyticsHelper, {
ANALYTICS_EVENTS,
} from '../../helper/AnalyticsHelper';
const buildInboxData = inboxParams => { const buildInboxData = inboxParams => {
const formData = new FormData(); const formData = new FormData();
@ -117,6 +120,12 @@ export const getters = {
}, },
}; };
const sendAnalyticsEvent = channelType => {
AnalyticsHelper.track(ANALYTICS_EVENTS.ADDED_AN_INBOX, {
channelType,
});
};
export const actions = { export const actions = {
get: async ({ commit }) => { get: async ({ commit }) => {
commit(types.default.SET_INBOXES_UI_FLAG, { isFetching: true }); commit(types.default.SET_INBOXES_UI_FLAG, { isFetching: true });
@ -134,6 +143,8 @@ export const actions = {
const response = await WebChannel.create(params); const response = await WebChannel.create(params);
commit(types.default.ADD_INBOXES, response.data); commit(types.default.ADD_INBOXES, response.data);
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
const { channel = {} } = params;
sendAnalyticsEvent(channel.type);
return response.data; return response.data;
} catch (error) { } catch (error) {
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
@ -146,6 +157,7 @@ export const actions = {
const response = await WebChannel.create(buildInboxData(params)); const response = await WebChannel.create(buildInboxData(params));
commit(types.default.ADD_INBOXES, response.data); commit(types.default.ADD_INBOXES, response.data);
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
sendAnalyticsEvent('website');
return response.data; return response.data;
} catch (error) { } catch (error) {
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
@ -158,6 +170,7 @@ export const actions = {
const response = await TwilioChannel.create(params); const response = await TwilioChannel.create(params);
commit(types.default.ADD_INBOXES, response.data); commit(types.default.ADD_INBOXES, response.data);
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
sendAnalyticsEvent('twilio');
return response.data; return response.data;
} catch (error) { } catch (error) {
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
@ -170,6 +183,7 @@ export const actions = {
const response = await FBChannel.create(params); const response = await FBChannel.create(params);
commit(types.default.ADD_INBOXES, response.data); commit(types.default.ADD_INBOXES, response.data);
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });
sendAnalyticsEvent('facebook');
return response.data; return response.data;
} catch (error) { } catch (error) {
commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false }); commit(types.default.SET_INBOXES_UI_FLAG, { isCreating: false });

View file

@ -1,8 +1,3 @@
/* eslint no-console: 0 */
/* eslint-env browser */
/* eslint-disable no-new */
/* Vue Core */
import Vue from 'vue'; import Vue from 'vue';
import VueI18n from 'vue-i18n'; import VueI18n from 'vue-i18n';
import VueRouter from 'vue-router'; import VueRouter from 'vue-router';
@ -32,7 +27,6 @@ import constants from '../dashboard/constants';
import * as Sentry from '@sentry/vue'; import * as Sentry from '@sentry/vue';
import 'vue-easytable/libs/theme-default/index.css'; import 'vue-easytable/libs/theme-default/index.css';
import { Integrations } from '@sentry/tracing'; import { Integrations } from '@sentry/tracing';
import posthog from 'posthog-js';
import { import {
initializeAnalyticsEvents, initializeAnalyticsEvents,
initializeChatwootEvents, initializeChatwootEvents,
@ -40,6 +34,7 @@ import {
import FluentIcon from 'shared/components/FluentIcon/DashboardIcon'; import FluentIcon from 'shared/components/FluentIcon/DashboardIcon';
import VueDOMPurifyHTML from 'vue-dompurify-html'; import VueDOMPurifyHTML from 'vue-dompurify-html';
import { domPurifyConfig } from '../shared/helpers/HTMLSanitizer'; import { domPurifyConfig } from '../shared/helpers/HTMLSanitizer';
import AnalyticsHelper from '../dashboard/helper/AnalyticsHelper';
Vue.config.env = process.env; Vue.config.env = process.env;
@ -51,12 +46,6 @@ if (window.errorLoggingConfig) {
}); });
} }
if (window.analyticsConfig) {
posthog.init(window.analyticsConfig.token, {
api_host: window.analyticsConfig.host,
});
}
Vue.use(VueDOMPurifyHTML, domPurifyConfig); Vue.use(VueDOMPurifyHTML, domPurifyConfig);
Vue.use(VueRouter); Vue.use(VueRouter);
Vue.use(VueI18n); Vue.use(VueI18n);
@ -90,6 +79,7 @@ window.WootConstants = constants;
window.axios = createAxios(axios); window.axios = createAxios(axios);
window.bus = new Vue(); window.bus = new Vue();
initializeChatwootEvents(); initializeChatwootEvents();
AnalyticsHelper.init();
initializeAnalyticsEvents(); initializeAnalyticsEvents();
initalizeRouter(); initalizeRouter();

View file

@ -51,11 +51,10 @@
} }
window.errorLoggingConfig = '<%= ENV.fetch('SENTRY_DSN', '')%>' window.errorLoggingConfig = '<%= ENV.fetch('SENTRY_DSN', '')%>'
</script> </script>
<% if @global_config['ANALYTICS_TOKEN'].present? && @global_config['ANALYTICS_HOST'].present? %> <% if @global_config['ANALYTICS_TOKEN'].present? %>
<script> <script>
window.analyticsConfig = { window.analyticsConfig = {
token: '<%= @global_config['ANALYTICS_TOKEN'] %>', token: '<%= @global_config['ANALYTICS_TOKEN'] %>',
host: '<%= @global_config['ANALYTICS_HOST'] %>',
} }
</script> </script>
<% end %> <% end %>

View file

@ -42,8 +42,6 @@
value: value:
- name: ANALYTICS_TOKEN - name: ANALYTICS_TOKEN
value: value:
- name: ANALYTICS_HOST
value:
- name: DIRECT_UPLOADS_ENABLED - name: DIRECT_UPLOADS_ENABLED
value: false value: false
locked: false locked: false

View file

@ -22,6 +22,7 @@
"@chatwoot/prosemirror-schema": "https://github.com/chatwoot/prosemirror-schema.git#7e8acadd10d7b932c0dc0bd0a18f804434f83517", "@chatwoot/prosemirror-schema": "https://github.com/chatwoot/prosemirror-schema.git#7e8acadd10d7b932c0dc0bd0a18f804434f83517",
"@chatwoot/utils": "^0.0.10", "@chatwoot/utils": "^0.0.10",
"@hcaptcha/vue-hcaptcha": "^0.3.2", "@hcaptcha/vue-hcaptcha": "^0.3.2",
"@june-so/analytics-next": "^1.36.5",
"@rails/actioncable": "6.1.3", "@rails/actioncable": "6.1.3",
"@rails/ujs": "^7.0.3-1", "@rails/ujs": "^7.0.3-1",
"@rails/webpacker": "5.3.0", "@rails/webpacker": "5.3.0",
@ -46,7 +47,6 @@
"md5": "^2.3.0", "md5": "^2.3.0",
"ninja-keys": "^1.1.9", "ninja-keys": "^1.1.9",
"opus-recorder": "^8.0.5", "opus-recorder": "^8.0.5",
"posthog-js": "^1.13.7",
"prosemirror-markdown": "1.5.1", "prosemirror-markdown": "1.5.1",
"prosemirror-state": "1.3.4", "prosemirror-state": "1.3.4",
"prosemirror-view": "1.18.4", "prosemirror-view": "1.18.4",

1003
yarn.lock

File diff suppressed because it is too large Load diff