Setup analytics destinations (e.g. Google Analytics, Facebook Pixel) in a really simple way on React/node. Then you can send events (e.g. Order Completed) using the Segment/RudderStack specs, and have them automatically formatted for each destination correctly.
- Tiny (2kb compressed) & fast core lib.
- Great developer experience - send events (e.g.
page
/track
/identify
) immediately and EventFan will replay them for each destination as soon as they finish loading. - Supports React
- TypeScript types included, based on the standard Segment/RudderStack specs.
- Easy to extend.
- High reliability - handles network errors with destinations (e.g. failing to load a third party script) gracefully.
yarn add eventfan
Initialise just once in your application:
import EventFan, { FacebookPixel } from "event-fan";
const eventFan = new EventFan({
destinations: [
new FacebookPixel({ pixelId: "YOUR-PIXEL-ID" }),
new GA4({ measurementId: "YOUR-GOOGLE-ANALYTICS-ID" }),
],
});
For React, instead wrap your app with the provider component:
export default function App() {
return (
<EventFanProvider
destinations={[
new FacebookPixel({ pixelId: "YOUR-PIXEL-ID" }),
new GA4({ measurementId: "YOUR-GOOGLE-ANALYTICS-ID" }),
]}
>
<h1>Your app</h1>
</EventFanProvider>
);
}
Note that for React, you can then access the methods (detailed below) with the useEventFan
hook:
const { track, page, identify } = useEventFan();
useEffect(() => {
track<Ecommerce.CheckoutStarted>("Checkout Started", {
value: 100.0,
});
}, []);
You must fire page calls on each page view. By default it will use the page <Title/>
and url, unless you specify these:
eventFan.page();
Identify a user when they log in:
eventFan.identify("userID", {
first_name: "Jane",
last_name: "Doe",
email: "jane@example.com",
// ...
});
Track events, using 50+ Segment/Rudderstack Specification types (included), or with your own types (created with
TEvent<name, properties>
). With standard events the properties are automatically converted to the correct format for
each destination (e.g. in this case Facebook Pixel's Purchase
event):
import { Ecommerce } from "event-fan";
eventFan.track<Ecommerce.OrderCompleted>("Order Completed", {
order_id: "order_UUID",
// ...
});
For pure JavaScript users, just omit the typing:
eventFan.track("Order Completed", {
order_id: "order_UUID",
// ...
});
To add custom events, extend the TEvent
type:
type CustomEvent = TEvent<
"Custom Event Name",
{
iceCream: string;
}
>;
eventFan.track<CustomEvent>("Custom Event Name", {
iceCream: "vanilla",
});
Either use the default mappings (similar to RudderStack/Segment), or write your own:
export function customOrderCompleted({
props,
}: Ecommerce.OrderCompleted): TEvent<"Purchase", Purchase> {
// E.g. start with the default mapping
const defaults = FacebookPixel.orderCompleted({ props });
// And change some properties (in this case `content_type`)
return {
...defaults,
content_type: FacebookPixel.ContentType.DESTINATION,
};
}
// Add to your client
const facebookPixel = new FacebookPixel({ pixelId: "123" });
facebookPixel.mapping["Order Completed"] = customOrderCompleted;
await eventFan.addDestination(facebookPixel);
Have a new destination you want to add? Simply implement the Destination
class:
class CustomDestination implements Destination {
initialise(): Promise<void> {
destinationNodeModule.initialise(this.eventKey);
}
// ...
}
Note that there is a helpful loadScript
util exported, that you may want to use if you are loading third party scripts
from a url.
To test in your application, we recommend using mocks & spys for unit tests, and then connecting to real staging destinations for end-to-end tests.
const mockTrack = jest.fn();
jest.spyOn(eventFanInstance, "track").mockImplementation(mockTrack);
// Do something, then...
expect(mockTrack).toHaveBeenCalledWith({
// Expected Call
});
Note that in react you'll need to spy on useEventFan as follows:
import * as EventFan from "eventfan";
// Within your test:
const mockTrack = jest.fn();
jest.spyOn(EventFan, "useEventFan").mockReturnValue({ track: mockTrack });
We recommend looking in /src/e2e/
for examples of end-to-end tests.
- Codegen the basics by running
npx hygen generator new [destinationName]
within/src/destinations/
. - Add the destination name to
/src/destinations/DestinationName.ts
- Create the initialise, identify, page and track methods in your destination. You can run
yarn dev
whilst doing this to see the impact in a real browser, with hot reloading (note you need to update/e2e/react
to add the destination staging credentials to do this). Make sure to add full unit testing and at least one e2e test. - Add the destination to the EventFan client as a dynamic import (in
/src/client/utils/loadDestinationsDynamically
).
These are added within /src/destinations/[destination]/mapping
. Make sure to add corresponding TypeScript types and a
unit test.