Typesafe analytic actions
This commit is contained in:
@@ -19,6 +19,7 @@ import { gasPriceEstimator } from '../util/gas_price_estimator';
|
||||
import { Heartbeater } from '../util/heartbeater';
|
||||
import { generateAccountHeartbeater, generateBuyQuoteHeartbeater } from '../util/heartbeater_factory';
|
||||
import { providerStateFactory } from '../util/provider_state_factory';
|
||||
import { analyticsMiddleware } from '../redux/analytics_middleware';
|
||||
|
||||
fonts.include();
|
||||
|
||||
@@ -129,7 +130,7 @@ export class ZeroExInstantProvider extends React.Component<ZeroExInstantProvider
|
||||
networkId: state.network,
|
||||
providerName: state.providerState.name,
|
||||
});
|
||||
analytics.track('Widget - Opened');
|
||||
analytics.widgetOpened();
|
||||
}
|
||||
public componentWillUnmount(): void {
|
||||
if (this._accountUpdateHeartbeat) {
|
||||
|
||||
@@ -21,10 +21,12 @@ export const analyticsMiddleware: Middleware = store => next => middlewareAction
|
||||
|
||||
switch (nextAction.type) {
|
||||
case ActionTypes.SET_ACCOUNT_STATE_READY:
|
||||
if (curAccount.state === AccountState.Ready && !_.isEqual(curAccount, prevAccount)) {
|
||||
if (curAccount.state === AccountState.Ready && prevAccount.state !== AccountState.Ready) {
|
||||
const ethAddress = curAccount.address;
|
||||
analytics.addUserProperties({ ethAddress });
|
||||
analytics.track('Wallet - Ready');
|
||||
analytics.walletReady({
|
||||
numAssetsAvailable: curState.availableAssets ? curState.availableAssets.length : 0,
|
||||
});
|
||||
}
|
||||
break;
|
||||
case ActionTypes.UPDATE_ACCOUNT_ETH_BALANCE:
|
||||
|
||||
@@ -1,35 +1,37 @@
|
||||
import { ObjectMap } from '@0x/types';
|
||||
import { logUtils } from '@0x/utils';
|
||||
|
||||
import { ANALYTICS_ENABLED } from '../constants';
|
||||
import { heapUtil } from './heap';
|
||||
|
||||
import { HeapAnalytics, heapUtil } from './heap';
|
||||
|
||||
const evaluteHeapCall = (heapFunctionCall: (heap: HeapAnalytics) => void): void => {
|
||||
if (!ANALYTICS_ENABLED) {
|
||||
return;
|
||||
}
|
||||
|
||||
const curHeap = heapUtil.getHeap();
|
||||
if (curHeap) {
|
||||
try {
|
||||
heapFunctionCall(curHeap);
|
||||
} catch (e) {
|
||||
// We never want analytics to crash our React component
|
||||
// TODO: error reporter here
|
||||
logUtils.log('Analytics error', e);
|
||||
}
|
||||
}
|
||||
enum EventNames {
|
||||
WALLET_OPENED = 'Wallet - Opened',
|
||||
WALLET_READY = 'Wallet - Ready',
|
||||
WIDGET_OPENED = 'Widget - Opened',
|
||||
}
|
||||
const track = (eventName: EventNames, eventData: ObjectMap<string | number> = {}): void => {
|
||||
heapUtil.evaluateHeapCall(heap => heap.track(eventName, eventData));
|
||||
};
|
||||
function trackingEventFnWithoutPayload(eventName: EventNames): () => void {
|
||||
return () => {
|
||||
track(eventName);
|
||||
};
|
||||
}
|
||||
function trackingEventFnWithPayload<T extends ObjectMap<string | number>>(
|
||||
eventName: EventNames,
|
||||
): (eventDataProperties: T) => void {
|
||||
return (eventDataProperties: T) => {
|
||||
track(eventName, eventDataProperties);
|
||||
};
|
||||
}
|
||||
|
||||
export const analytics = {
|
||||
// TODO(sk): make these more specific
|
||||
addUserProperties: (properties: ObjectMap<string | number>): void => {
|
||||
evaluteHeapCall(heap => heap.addUserProperties(properties));
|
||||
heapUtil.evaluateHeapCall(heap => heap.addUserProperties(properties));
|
||||
},
|
||||
addEventProperties: (properties: ObjectMap<string | number>): void => {
|
||||
evaluteHeapCall(heap => heap.addEventProperties(properties));
|
||||
},
|
||||
track: (eventName: string, eventProperties?: ObjectMap<string | number>): void => {
|
||||
evaluteHeapCall(heap => heap.track(eventName, eventProperties));
|
||||
heapUtil.evaluateHeapCall(heap => heap.addEventProperties(properties));
|
||||
},
|
||||
walletOpened: trackingEventFnWithoutPayload(EventNames.WALLET_OPENED),
|
||||
walletReady: trackingEventFnWithPayload<{ numAssetsAvailable: number }>(EventNames.WALLET_READY),
|
||||
widgetOpened: trackingEventFnWithoutPayload(EventNames.WIDGET_OPENED),
|
||||
};
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
import { ObjectMap } from '@0x/types';
|
||||
import { logUtils } from '@0x/utils';
|
||||
|
||||
import { HEAP_ANALYTICS_DEVELOPMENT_APP_ID } from '../constants';
|
||||
import { ANALYTICS_ENABLED, HEAP_ANALYTICS_DEVELOPMENT_APP_ID } from '../constants';
|
||||
|
||||
export interface HeapAnalytics {
|
||||
loaded: boolean;
|
||||
@@ -82,4 +83,20 @@ export const heapUtil = {
|
||||
|
||||
return setupZeroExInstantHeap();
|
||||
},
|
||||
evaluateHeapCall: (heapFunctionCall: (heap: HeapAnalytics) => void): void => {
|
||||
if (!ANALYTICS_ENABLED) {
|
||||
return;
|
||||
}
|
||||
|
||||
const curHeap = heapUtil.getHeap();
|
||||
if (curHeap) {
|
||||
try {
|
||||
heapFunctionCall(curHeap);
|
||||
} catch (e) {
|
||||
// We never want analytics to crash our React component
|
||||
// TODO: error reporter here
|
||||
logUtils.log('Analytics error', e);
|
||||
}
|
||||
}
|
||||
},
|
||||
};
|
||||
|
||||
Reference in New Issue
Block a user