Data Model Fluidity in Analytics

The world of data, intelligence and analytics perpetually evolves, so too will the data structures we work with. With the rising popularity of data lakes, there’s a steadily increasing need for developers, consultants, and vendors to work with all kinds of data, be it structured, unstructured, or semistructured.

If you haven’t already categorized your work in these terms, you should probably start. It can be quite helpful in avoiding using the wrong tool for the job.

While this piece focuses in part on the Google Analytics ecosystem, the major principles should be applied in many other domains as well. The same notions of stricture, abstraction, and semantics apply in all analytics collection environments. One of my current projects acquires such data from back-end .NET application logs, and feeds them into Amazon Redshift; the same problems and ideas apply there too.

Ode to Structured Data

The problems of structured data models arise from its legacy: many vendors’ systems are built around performance, from an age when the most obvious way to achieve performance was to constrain the fields that your application could populate.

Who of you, consultants in the Google Analytics field, remember its genesis as Urchin? You may recall the days when we had only Pageviews, and then Google’s extension of the data model to incorporate Events (interactions). Many consultants found the Category/Action/Label model too limiting, so they tucked multiple other values (concatenated) into the Label field. This often proved unwieldy for analysis, so we asked Google for Custom Variables, and they gave us 5 “slots” to combine any key/values we wanted, initially.

See a trend here? They’ve long operated on fixed structures, which imposes limits on the kinds of things we can express about a particular interaction. With the addition of Custom Dimensions and Metrics, our capabilities in GA have improved, however we’re still bound by a structure, even if it’s so “wide” that we may rarely find its limits.

These trends persist, in any analytics platform which attempts to extract a fixed set of properties from its data stream; they have to assume some kind of schema to the data, to populate a consistent representation, which eventually becomes the reports you rely on.

The Rise of Semistructured Data

Vendors such as and Mixpanel took a fundamentally different approach, because they built their products on newer technology and under newer paradigms of computing. They turned the legacy problem on its head, focusing on a dynamic query capability at reporting time, and rather than modeling the data to a fixed structure at collection time.

They operate on a different assumption: you, the developer or consultant, need to collect everything you know in the app, rather than just what the analytics platform planned for (years prior).

To support this, these and other vendors adopted a simple convention: let a developer push anything as JSON.

Two important points to that assumption:

JSON is just the encoding, while the semantics of the data model are left to the developer, removed from the analytics platform’s own logic.


Google’s Tag Manager provides a similarly flexible dataLayer, but it fundamentally acts as a middle-ware to the same (old) Analytics platform: your GTM configuration just remodels the dataLayer events into something that fits Google Analytics. GTM’s function is good, provided our business semantics can either be sufficiently represented within the Analytics model, or we use GTM to augment Analytics with other platforms too.

In mobile applications, Firebase now accommodates some semi-structured analytics model as well, but this is built on BigQuery, a much newer technology than Google Analytics. The legacy mobile integration in Google Analytics remains fixed in structure, just like its web counterpart.

Model Mutation

I love the flexibility that semi-structured models provide. Instead of warping our business context to fit the abstract, generic things in a prescribed structure (e.g. Event’s Category, Action, Label), our data can be semantically modeled.

That’s great, until things change… which is, of course, inevitable.

The chaos monkey takes many forms. The business case evolves when new services plans are added, the UI changes when new payment options are added, and occasionally a developer will alter the attributes of an analytics (or dataLayer) event without notifying the analytics stakeholders. Something you couldn’t have planned for will throw a wrench in your beautifully architected data model.

The Hazards Of Weak Structure

Models with great flexibility increase the propensity of errors, when developers are given general rules instead of explicit templates. These errors include communication factors, such as differences in naming convention.

When errors occur, the development cost multiplies, especially when relying on third-party software teams. The communication overhead of scheduling calls or meetings, and the hours invested in review and testing cycles become expensive and unsustainable. This often results in software teams deprioritizing analytics integration from the rest of their backlog, because they separate it from the rest of the feature development.

In some cases, I’ve seen teams defer analytics integration until after launch. As a result, the launch is basically unmeasured, so there’s limited means (if any) of reporting the success of the development team in business terms. This should be seen as disastrous.

To the other extreme, providing only strict templates also inhibits the broader systems thinking necessary for developers to fully engage the analytics process. They’re effectively discouraged from including meaningful business attributes of their own volition, because they have not been offered an easy way to envision it, so many won’t even try. (They’re busy, after all.)

Sustainable Flow in Analytics Development

For a thriving analytics ecosystem within an organization, all stakeholders must be engaged to add value to the analytics model wherever possible. Analytics must be treated as an intrinsic part of the development workflow, by emphasizing the measurement of results especially at launch, and with every software release.

The goals of a sustainable flow are simple:

The realized benefits:

Provide some stricture for the mundane

What are the most common tracking cases you can think of, for all of your organization’s applications? Probably screenviews and button clicks, maybe some kind of form entry. Can you think of others that aren’t intrinsically linked to your business case?

For these common cases, some simple, standard models are quite sufficient:

Sharing these functions across all your applications enables developers to do the right thing when they’re building the app: just always register a screen view, the same way every time.

Standardization is the key: it must be easy, obvious, and consistent.

Please note that I’m referring to functions very specifically: these should be part of the platform code, a standard library that your organization applies in all software. These functions will produce a standardized dataLayer event and handle analytics accordingly, but their purpose is to enforce the standard for the benefit of the apps’ developers.

These functions should roughly approximate:

trackScreenView("home screen");
trackFormSubmit("newsletter signup");
trackButtonClick("newsletter unsubscribe");

It should be assumed that names will change, and understanding the business impact will require reconciliation of those changes over time. Therefore these methods should be further augmented:

Why would these be useful?

This reliability is profoundly valuable. For example, when you see an event screenView, you expect there to be a (textual) screenName attached to it. The approach above prevents a developer from overlooking that attribute, or accidentally providing the wrong kind of data for it.

When features are added to the application, developers have a consistent reference for correctly adding the analytics they need, often without even asking their analytics stakeholders first. While that asking might still be good or useful, the need is reduced by supporting augmentation or indirection through lookup tables.

The net effect is that time is saved for all participants and stakeholders, by avoiding the cost of errors extending the review cycle.

Prioritize semantic value

No application, in a business context, can be understood simply by screen views and UI events; these pieces are valuable, but they reflect the life of software, not the life of the business.

To derive the business context in analytics, there must be a projection of the business attributes into the data model (e.g. service plan, pricing, account expiration). This is where semi-structured data really thrives.

Here again though, there’s some benefit in standardizing the functions used by developers within your organization; it reduces the exposure to unexpected errors, and provides a semantic within the code that developers will recognize.

In Javascript, such a function might be expressed like so:

trackBusinessState("userLogin", { servicePlan: "premium" });

You may notice this bears some resemblance to the Firebase model for custom events in iOS; quite intentional.

A critical component of this semantic process is communication, though, and documentation can fulfill part of that requirement. The essential goal is to ensure that naming conventions, and any applicable constraints on associated attribute values are respected.

In widely dispersed teams, or multiple teams across large organizations, this may justify rigourous documentation of the naming conventions in use, in conjunction with a workflow that requires developers to validate the naming convention. In smaller, closely-collaborating teams conversation may be sufficient where the developers and analytics integrators convene regularly, however keeping a central record of adopted naming conventions still provides future value.

Why is this useful?

The value of semantic recognition should not be overlooked: when developers are maintaining code, they will encounter such a function trackBusinessState, and distinguish it as more sensitive than the rest of the application code.

Two Major Caveats

1. Avoid Over-Abstraction

The intent is not to provide a generic template for direct dataLayer interaction. This should never exist in application code:

dataLayer.push({ event: "trackEvent", eventCategory: "engagement", eventAction: "login" });

I’ve seen many cases where the only fixed element of the data model is the trackEvent string.

Approaches like this expose the prescribed data model of Google Analytics to the developer. That’s dangerous, because you’ve probably overlooked constraining it to ensure that the attributes the developer provided were valid. This fosters pollution of the GA reports which effectively unmanaged junk, when the developers neglect to coordinate with analytics stakeholders. In GA this can seriously compromise data quality, and at scale, can make legitimate data inaccessible.

2. Avoid Over-Specification

On the flip-side, it’s possible to become excessively strictured. Taken to an extreme, a team of developers might assume that all tracking should be expressed as separate functions for each event model. While this supports all the benefits of standardization, it becomes a terrible burden to maintain.

As the business case evolves, the availability of certain business attributes will likely change. When it does, the standard tracking functions that require those attributes may no longer be suitable, and will require replication, replacement, or other adaptation.

The Zen of Flow

With these principles in mind, and enacted by developers and other analytics stakeholders alike, you should find your software projects accelerated, with better data flowing sooner, and adapting faster with the business needs. There will be more time, less confusion, easier successes and a more flexible workflow for all stakeholders.

In a future post I’ll also address an agile model of analytics integration for new projects.

Coming Soon: an iOS Reference Implementation

In the near future, I’ll be releasing an example iOS application, which contains a module embodying these principles. Please watch me on Twitter for updates.

This sample application provides, among other things: