Build Your Own Analytics Tool on Velo by Wix
Disclaimer: I’m a Frontend developer at Wix, that’s right. But this post written as a result of my experience as a “regular” user. I even used a non Wix account to build it and honestly, even if 1000 of you will build an app on Wix, I won’t profit from it directly in any way 😀
So why I wrote this post? I really enjoyed building it. It’s simple (well, most of the parts), zero configuration, and of course, it’s free. Ah, and you don’t need to leave the browser — 100% cloud.
It all started when I published a package on npm and wanted to track the page views. Long time ago, I used “hits” for the same purpose but now I thought that it can be interesting to build something like that by myself. Also, hits just gives the number of page views. I though it can be nicer to have more segmented data, by time, refers, geo location etc.
What’s needed for an Analytics app?
Every analytics tool has at least the basic components:
- Embedded resource in the host site — script, image (pixel) etc. which send a request with relevant information about the user, the browser, ip, geo etc.
- Server to handle the requests.
- Database to store the records.
- Client app to display the data usually in a graphic way — tables, graphs, maps etc.
Another nice to have components:
- Monitoring
- Environments — staging / production for both the server and the database
- Authentication — only allowed user can have access to the data
All those component is part of Velo (former “Corvid”) and Wix echo-system out of the box with zero configurations.
Let’s Start
I won’t explain how to create a website on Wix (I recommend to start with the “blank template”), it’s straightforward, well documented and take few minutes. It takes another second to enable Velo on the website.
Also, I won’t explain how to do things which already well documented. Instead, I added links so you can get all the information you need from them. (If I missed some, please let me know).
Now let’s implement the components we talked about few lines ago.
Embedded Resource
There are several ways to do it. I choose to expose an endpoint which returns an SVG and just before that, add a record in the database. This way, I show a nice SVG badge in the package page and at the same time, it sends a request back to the server. I know I didn’t invent the wheel, “hits” does exactly the same, but I’m just about to show you how much it’s easy in Velo.
Velo hosts the code of both of the sides — the client and the server, just like a nodejs based website which uses JavaScript both on the client and the server side.
In order to expose an endpoint for external sources, we need to write a special server code (located under the “Backend” folder)? Why it’s special? Because usually the server code is accessible only internally to the website’s pages.
In order to create this special module, the file name should be called
“http-functions.js”.
Following the methods naming convention, I created the “view” endpoint which accepts GET
requests and respond with SVG [1].
import {ok} from 'wix-http-functions';export async function get_view(request) {
const svg = `<svg>....</svg>`;
const response = {
headers: {
'Content-Type': 'image/svg+xml',
'Content-Length': svg.length,
},
body: svg,
};
return ok(response);
}
Saving the Data
Velo offers a built-in database (“Content Manager”) with a strong yet easy API, called wixData
.
First thing is creating a table (“collection”) — views
and add the following columns: headers
(as Text), refrer
(as Text). This will be enough for the basic app, you can add more of course (I added project
so I could embed as many resources as I want).
It’s a good time to mention that, when you create a collection, pay attention to the permission section (“What’s the collection for?”). Start with set “anyone” to all the methods and customize it as you go.
Now we can modify the endpoint code to extract the relevant information from the request and save it in views
.
try {
await wixData.insert('views', {
referer,
headers: JSON.stringify(headers),
})
} catch (error) {
console.log(error);
}
(You can see the logs in the “Site Monitoring”)
The nice thing about the headers
is that they contain geo location information out of the box, so later, we can show a geo location visualization.
Now, when we have all the data, we want to show it visually.
Client App
This step should be straightforward, right? After all, Wix is a UI builder before anything. The good news is that for some components, it’s true. For example, it’s very easy to create a table and connect it to the database. It can be done even without coding. But when it comes to more sophisticated visualization elements such as graphs, charts, maps etc. there are no built-in options nor external apps to connect the data to the elements.
So there is no easy way (As fat as I know) but there is a way shown in this tutorial and that’s how I did it, but I have to admit that it’s not that convenient. I hope it will be changed in the future.
In a nutshell, the challenge is that even though we can “interact” with components through Velo code, we are not interacting with the DOM itself but with a special isolated DOM like. As a result, we can’t use JavaScript libraries which depends on interacting with the actual DOM.
The only (free) way to have access to a DOM is by adding an “HTML iframe” component. But here is the catch, we can’t inject data into the iframe directly because the communication between the page and the iframe has to be through post messages.
So here is the flow starts on page ready.
- Get the data from the
views
table.
1.1 In parallel, render the iframe (handle by Wix).
1.2. Establish an event listener to get messages from the iframe. - The iframe initiating a chart library, establish an event listener to get messages from the page, and posts a message to the parent saying “I’m ready”.
- The page (waits for the data if needed and) posts a message back with the data.
- The iframe gets the data and draw the chart accordingly.
If I missed you, maybe a diagram will help:
In this project, I’m using Google’s “Line Chart” so we need to fit the data to the DataTable
format.
The iframe’s code
Here is the code (notice the “loader” 😉)
The page code
And the result
If you don’t want this page to be visible to everyone, you can limit it.
This is it basically. From this point you can take it to wherever your imagination can take you — hit maps, different categorization, comparing timestamp etc. If you need my help with that, leave a comment or reach out.
If you want to get a copy of this website, let me know and I’ll send a copy to your Wix account.
And usually, if you have questions or ideas about how to improve this post (or my other posts), leave a comment or say hi 👋
[1] The endpoint can also respond with another image types such as png, jpg etc. encoded by base64
const img = Buffer.from('iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVQYV2PgjOr/DwADVAHy2H3MYgAAAABJRU5ErkJggg==', 'base64')
const response = {
headers: {
'Content-Type': 'image/png',
'Content-Length': img.length
},
body: img
};return ok(response);
[2] Yah, it’s a bit annoying to edit the code in the Editor, copy / paste it in your favorite code editor and then copy / paste it back.