I’m building an analytics system for a mobile application and have had some difficulty deciding how to store and process large amounts of data.
Each row will represent a ‘view’ (like a web page) and store some fixed attributes, like user agent and date. Additionally, each view may have a varying number of extra attributes, which relate to actions performed or content identifiers.
I’ve looked at Amazon SimpleDb which handles the varying number of attributes well, but has no support for GROUP BY and doesn’t seem to perform well when COUNTing rows either. Generating a monthly graph with 30 data points would require a query for each day per dataset.
MySQL handles the COUNT and GROUP modifiers much better but additional attributes require storage in a link table and a JOIN to retrieve views where attributes match a given value, which isn’t very fast. 5.1’s partitioning feature may help speed things up a bit.
What I have gathered from a lot of reading and profiling queries on the aforementioned systems is that ultimately all of the data needs to be aggregated and stored in tables for quick report generation.
Have I missed anything obvious in my research and is there a better way to do this than use MySQL? It doesn’t feel like the right task for the job, but I can’t find anything capable of both GROUP/COUNT queries and a flexible table structure.
Keeping it in MySQL: If the amount of writes are limited / reads are more common, and the data is relatively simple (i.e: you can predict possible characters), you could try to use a text/blob column in the main table, which is updated with comma separated values or key/value pairs with an
AFTER INSERT / UPDATEtrigger on the join table. You keep the actual data in a separate table, so searching for MAX’s / specific ‘extra’ attributes can still be done relatively fast, but retrieving the complete dataset for one of your ‘views’ would be a single row in the main table, which you can split into the separate values with the script / application you’re using, relieving much of the stress on the database itself.The downside of this is a tremendous increase in cost of updates / inserts in the join table: every alteration of data would require a query on all related data for a record, and a second insert into the ‘normal’ table, something like
However, as analytics data goes it usually trails somewhat, so possibly not every update has to trigger an update in cache, just a daily cronscript filling the cache with yesterdays data could do.