When foreign key support for compressed chunks was added we moved the FK constraint from the uncompressed chunk to the compressed chunk as part of compress_chunk and moved it back as part of decompress_chunk. With the addition of partially compressed chunks in 2.10.x this approach was no longer sufficient and the FK constraint needs to be present on both the uncompressed and the compressed chunk. While this patch will fix future compressed chunks a migration has to be run after upgrading timescaledb to migrate existing chunks affected by this. The following code will fix any affected hypertables: ``` CREATE OR REPLACE FUNCTION pg_temp.constraint_columns(regclass, int2[]) RETURNS text[] AS $$ SELECT array_agg(attname) FROM unnest($2) un(attnum) LEFT JOIN pg_attribute att ON att.attrelid=$1 AND att.attnum = un.attnum; $$ LANGUAGE SQL SET search_path TO pg_catalog, pg_temp; DO $$ DECLARE ht_id int; ht regclass; chunk regclass; con_oid oid; con_frelid regclass; con_name text; con_columns text[]; chunk_id int; BEGIN -- iterate over all hypertables that have foreign key constraints FOR ht_id, ht in SELECT ht.id, format('%I.%I',ht.schema_name,ht.table_name)::regclass FROM _timescaledb_catalog.hypertable ht WHERE EXISTS ( SELECT FROM pg_constraint con WHERE con.contype='f' AND con.conrelid=format('%I.%I',ht.schema_name,ht.table_name)::regclass ) LOOP RAISE NOTICE 'Hypertable % has foreign key constraint', ht; -- iterate over all foreign key constraints on the hypertable -- and check that they are present on every chunk FOR con_oid, con_frelid, con_name, con_columns IN SELECT con.oid, con.confrelid, con.conname, pg_temp.constraint_columns(con.conrelid,con.conkey) FROM pg_constraint con WHERE con.contype='f' AND con.conrelid=ht LOOP RAISE NOTICE 'Checking constraint % %', con_name, con_columns; -- check that the foreign key constraint is present on the chunk FOR chunk_id, chunk IN SELECT ch.id, format('%I.%I',ch.schema_name,ch.table_name)::regclass FROM _timescaledb_catalog.chunk ch WHERE ch.hypertable_id=ht_id LOOP RAISE NOTICE 'Checking chunk %', chunk; IF NOT EXISTS ( SELECT FROM pg_constraint con WHERE con.contype='f' AND con.conrelid=chunk AND con.confrelid=con_frelid AND pg_temp.constraint_columns(con.conrelid,con.conkey) = con_columns ) THEN RAISE WARNING 'Restoring constraint % on chunk %', con_name, chunk; PERFORM _timescaledb_functions.constraint_clone(con_oid, chunk); INSERT INTO _timescaledb_catalog.chunk_constraint(chunk_id, dimension_slice_id, constraint_name, hypertable_constraint_name) VALUES (chunk_id, NULL, con_name, con_name); END IF; END LOOP; END LOOP; END LOOP; END $$; DROP FUNCTION pg_temp.constraint_columns(regclass, int2[]); ```
Linux/macOS | Linux i386 | Windows | Coverity | Code Coverage | OpenSSF |
---|---|---|---|---|---|
TimescaleDB
TimescaleDB is an open-source database designed to make SQL scalable for time-series data. It is engineered up from PostgreSQL and packaged as a PostgreSQL extension, providing automatic partitioning across time and space (partitioning key), as well as full SQL support.
If you prefer not to install or administer your instance of TimescaleDB, try Timescale, our fully managed cloud offering (pay-as-you-go, with a free trial to start).
To determine which option is best for you, see Timescale Products for more information about our Apache-2 version, TimescaleDB Community (self-hosted), and Timescale Cloud (hosted), including: feature comparisons, FAQ, documentation, and support.
Below is an introduction to TimescaleDB. For more information, please check out these other resources:
- Developer Documentation
- Slack Channel
- Timescale Community Forum
- Timescale Release Notes & Future Plans
For reference and clarity, all code files in this repository reference
licensing in their header (either the Apache-2-open-source license
or Timescale License (TSL)
). Apache-2 licensed binaries can be built by passing -DAPACHE_ONLY=1
to bootstrap
.
(To build TimescaleDB from source, see instructions in Building from source.)
Using TimescaleDB
TimescaleDB scales PostgreSQL for time-series data via automatic partitioning across time and space (partitioning key), yet retains the standard PostgreSQL interface.
In other words, TimescaleDB exposes what look like regular tables, but are actually only an abstraction (or a virtual view) of many individual tables comprising the actual data. This single-table view, which we call a hypertable, is comprised of many chunks, which are created by partitioning the hypertable's data in either one or two dimensions: by a time interval, and by an (optional) "partition key" such as device id, location, user id, etc. (Architecture discussion)
Virtually all user interactions with TimescaleDB are with hypertables. Creating tables and indexes, altering tables, inserting data, selecting data, etc., can (and should) all be executed on the hypertable.
From the perspective of both use and management, TimescaleDB just looks and feels like PostgreSQL, and can be managed and queried as such.
Before you start
PostgreSQL's out-of-the-box settings are typically too conservative for modern
servers and TimescaleDB. You should make sure your postgresql.conf
settings are tuned, either by using timescaledb-tune
or doing it manually.
Creating a hypertable
-- Do not forget to create timescaledb extension
CREATE EXTENSION timescaledb;
-- We start by creating a regular SQL table
CREATE TABLE conditions (
time TIMESTAMPTZ NOT NULL,
location TEXT NOT NULL,
temperature DOUBLE PRECISION NULL,
humidity DOUBLE PRECISION NULL
);
-- Then we convert it into a hypertable that is partitioned by time
SELECT create_hypertable('conditions', 'time');
Inserting and querying data
Inserting data into the hypertable is done via normal SQL commands:
INSERT INTO conditions(time, location, temperature, humidity)
VALUES (NOW(), 'office', 70.0, 50.0);
SELECT * FROM conditions ORDER BY time DESC LIMIT 100;
SELECT time_bucket('15 minutes', time) AS fifteen_min,
location, COUNT(*),
MAX(temperature) AS max_temp,
MAX(humidity) AS max_hum
FROM conditions
WHERE time > NOW() - interval '3 hours'
GROUP BY fifteen_min, location
ORDER BY fifteen_min DESC, max_temp DESC;
In addition, TimescaleDB includes additional functions for time-series
analysis that are not present in vanilla PostgreSQL. (For example, the time_bucket
function above.)
Installation
TimescaleDB is available pre-packaged for several platforms (Linux, Docker, MacOS, Windows). More information can be found in our documentation.
To build from source, see instructions here.
Timescale, a fully managed TimescaleDB in the cloud, is available via a free trial. Create a PostgreSQL database in the cloud with TimescaleDB pre-installed so you can power your application with TimescaleDB without the management overhead.
Resources
Architecture documents
Useful tools
- timescaledb-tune: Helps set your PostgreSQL configuration settings based on your system's resources.
- timescaledb-parallel-copy:
Parallelize your initial bulk loading by using PostgreSQL's
COPY
across multiple workers.
Additional documentation
- Why use TimescaleDB?
- Migrating from PostgreSQL
- Writing data
- Querying and data analytics
- Tutorials and sample data
Community & help
- Slack Channel
- Github Issues
- Timescale Support: see support options (community & subscription)
Releases & updates
- Timescale Release Notes & Future Plans: see planned and in-progress updates and detailed information about current and past releases. - Subscribe to Timescale Release Notes to get notified about new releases, fixes, and early access/beta programs.