Skip to content

Latest commit

 

History

History
163 lines (119 loc) · 5.22 KB

README.md

File metadata and controls

163 lines (119 loc) · 5.22 KB

README

This example application is a "Scoring" service that stores "games" and "plays" of the games in question.

There are two tables:

  • games that holds a name and description
  • plays is a hypertable that references a game, storing score and total_time.

Interesting things to observe here:

Walkthrough

Use bin/console to preload the environment and follow the next steps.

Let's start by creating a game and a single play.

lol = Game.create(name: "LoL", description: "League of Legends")
Play.create(game: lol,
  score: (rand * 100).to_i,
  total_time: (rand * 1000).to_i)

You can also insert a few hundreds/thousands/millions of records to test it properly.

100.times do
   Play.create(game: lol,
   score: (rand * 100).to_i,
   total_time: (rand * 1000).to_i)
end

You can play with multiple games and millions of play records to make it an impressive playground if you want 😉

Then you can experiment with the time_bucket funciton.

Play.group("time_bucket('1 min',created_at)").count
# => {2022-02-09 12:34:00 UTC=>1, 2022-02-09 12:39:00 UTC=>10100}

Hypertables in TimescaleDB are designed to be easy to manage and to behave predictably to users familiar with standard PostgreSQL tables. Along those lines, SQL commands to create, alter, or delete hypertables in TimescaleDB are identical to those in PostgreSQL. Even though hypertables are comprised of many interlinked chunks, commands made to the hypertable automatically propagate changes down to all of the chunks belonging to that hypertable.

When the model contains the acts_as_hypertable macro, it's possible to navigate into the hypertable internals:

 Play.hypertable
 # => #<Timescaledb::Hypertable:0x00007faa97df7e30
 # hypertable_schema: "public",
 # hypertable_name: "plays",
 # owner: "jonatasdp",
 # num_dimensions: 1,
 # num_chunks: 2,
 # compression_enabled: true,
 # tablespaces: nil>

Each hypertable has many chunks. Chunks are the subtables spread accross the time. You can check chunks metadata from the hypertable relation here:

Play.hypertable.chunks.pluck(:chunk_name)
# => ["_hyper_1_1_chunk", "_hyper_1_2_chunk"]

Chunks can also be compressed/decompressed and you can check the state using scopes:

Play.hypertable.chunks.compressed.count # => 2

Get a resume from chunks status:

Play.hypertable.chunks.resume
# => {:total=>2, :compressed=>0, :uncompressed=>2}

To get a full stats from all hypertables, you can see Timescaledb.stats:

Timescaledb.stats
 # => {:hypertables=>
 #  {:count=>1,
 #   :uncompressed=>0,
 #   :approximate_row_count=>{"plays"=>10100},
 #   :chunks=>{:total=>2, :compressed=>0, :uncompressed=>2},
 #   :size=>{:uncompressed=>"1.3 MB", :compressed=>"0 Bytes"}},
 # :continuous_aggregates=>{:total=>0},
 # :jobs_stats=>[{:success=>100, :runs=>100, :failures=>0}]}

Note that we haven't used compression yet. So, we can force compression directly from the chunk relation:

Play.hypertable.chunks.each(&:compress!)

Calling stats to check the compressed size:

Timescaledb.stats
 #  {:count=>1,  ...
 #   :size=>
 #    {:uncompressed=>"1.2 MB",
 #     :compressed=>"180 KB"}},
 # :continuous_aggregates=>{:total=>0},
 # :jobs_stats=>
 #  [{:success=>107,
 #    :runs=>107,
 #    :failures=>0}]}

⚠️ Note that the chunks are not very effective compressing here because of the example is incomplete and with a little amount of records.

You can decompress as the system will make the compression in the background as it already have a policy.

Play.hypertable.chunks.each(&:decompress!)

Dump schema

The lib also contains a schema_dumper that allows you to dump the schema and reload with the same hypertable options.

rails db:schema:dump

Confirm that the hypertable is on db/schema.rb file:

 grep hypertable db/schema.rb
  create_hypertable "plays", time_column: "created_at", chunk_time_interval: "1
  minute", compress_segmentby: "game_id", compress_orderby: "created_at ASC",
  compression_interval: "P7D"

And you can also reload the configuration manually in the test environment:

RAILS_ENV=test rails db:schema:load

Scenic views

If you use scenic views, it will automatically recognize and dump the views.

Check the view migration and the correspondent view defined in the sql file.

Note that scenic is not fully compatible with timescaledb continous aggregates and we created a helper to make it easy to use along with scenic.