From 436e7fd620a3b54a93b532924f82bb45dde72eef Mon Sep 17 00:00:00 2001 From: Matt Date: Wed, 8 Jul 2020 15:25:23 -0400 Subject: [PATCH] add docs on BT schema/rowkey (#669) * add docs on BT schema/rowkey * color code! --- docs/_layouts/sidebar.html | 4 +- docs/content/_docs/bigtable.md | 95 ++++++++++++++++++++++++++++++++++ 2 files changed, 98 insertions(+), 1 deletion(-) create mode 100644 docs/content/_docs/bigtable.md diff --git a/docs/_layouts/sidebar.html b/docs/_layouts/sidebar.html index 25ebc922b..ba0ee0ae7 100644 --- a/docs/_layouts/sidebar.html +++ b/docs/_layouts/sidebar.html @@ -55,7 +55,9 @@
  • Aggregations
  • - +
  • + BigTable +
  • Shell
  • diff --git a/docs/content/_docs/bigtable.md b/docs/content/_docs/bigtable.md new file mode 100644 index 000000000..69a1dc871 --- /dev/null +++ b/docs/content/_docs/bigtable.md @@ -0,0 +1,95 @@ +--- +title: BigTable +--- +# BigTable + +Some good to know points about how Heroic uses Bigtable + +## Row Key composition/schema + +Heroic stores time series in BigTable in a schema optimized for retrieving data for the purpose of dashboarding, for example retrieving and aggregating the CPU load on all machines of a backend service in a particular region. This is achieved by storing tags as part of the row keys in BigTable. + +Heroic’s BigTable schema stores all metrics with the same key, tags, and resource identifiers within roughly 50 days (4294967296 milliseconds) in the same row. All metrics within those 50 days are stored as separate cells of data, with the time period within the 50 days as column key (or column qualifier), and the measurement as data. + +This way, Heroic avoids having a single row for *all* data points in a time series from the beginning of time until the heat death of the universe. + + +Given the two datapoints for an example timeseries: + +```json +{ + "series": { + "key": "system", + "tags": { + "site": "gew", + "what": "cpu-idle-percentage", + "system-component": "cpu", + "cpu-type": "idle", + "unit": "%" + }, + "resource": { + "podname": "pod-example-123-abc", + "host": "database.example.com" + } + }, + "data": { + "type": "points", + "data": [[1300000000000, 42.0]] + } +} +``` + +and + +```json +{ + "series": { + "key": "system", + "tags": { + "site": "gew", + "what": "cpu-idle-percentage", + "system-component": "cpu", + "cpu-type": "idle", + "unit": "%" + }, + "resource": { + "podname": "pod-example-123-abc", + "host": "database.example.com" + } + }, + "data": { + "type": "points", + "data": [[1300001000000, 84.0]] + } +} + +``` + +In order to retain the exact timestamp, Heroic splits the original timestamp into the base-timestamp (to be used for the row key), and the delta, or offset timestamp (to be used in the column key, or qualifier). When queried, the sum of the base-timestamp in the row key and the delta timestamp in the column qualifier becomes the exact timestamp. + +To illustrate this, using the two metrics in the example above, Heroic calculates the base-timestamp and the delta timestamp as follows: + + + +Base-timestamp first metric: 1300000000000 - (1300000000000 % 4294967296 ) = 1297080123392 +Timestamp delta first metric: 1300000000000 - 1297080123392 = 2919876608 + +Base-timestamp second metric: 1300001000000 - ( 1300001000000 % 4294967296 ) = 1297080123392 +Timestamp delta second metric: 1300001000000 - 1297080123392 = 2920876608 + + + +Together with all tags and resources, sorted lexicographically, this ends up creating this row-key: + +system,cpu-type=idle,site=gew,system-component=cpu,unit=%,what=cpu-idle-percentage,1297080123392,database.example.com,pod-example-123-abc + + +And the two metrics sent 16 minutes apart are stored in the BigTable table like so: + + +| Row key | 2919876608 | 2920876608 | +|------------------------------------------------------------------------------------------------------------------------------------------- |------------ |------------ | +| system,cpu-type=idle,site=gew,system-component=cpu,unit=%,what=cpu-idle-percentage,1297080123392,database.example.com,pod-example-123-abc | 42.0 | 84.0 | + + +*** \ No newline at end of file