Skip to content
This repository has been archived by the owner on Dec 17, 2024. It is now read-only.

postgres backend implemented #483

Merged
merged 5 commits into from
Jun 1, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 7 additions & 1 deletion helm-chart/.helmignore
Original file line number Diff line number Diff line change
Expand Up @@ -22,4 +22,10 @@
*.tmproj
.vscode/

grafana_dashboards/postgres
grafana_dashboards/prometheus
grafana_dashboards/influxdb/v5
grafana_dashboards/influxdb/v6
grafana_dashboards/influxdb/v7
grafana_dashboards/postgres/v5
grafana_dashboards/postgres/v6
grafana_dashboards/postgres/v7
pashagolub marked this conversation as resolved.
Show resolved Hide resolved
2 changes: 1 addition & 1 deletion helm-chart/Chart.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ dependencies:
- name: influxdb
repository: https://helm.influxdata.com/
version: "4.11.0"
condition: storage == influxdb
condition: influxdb.enabled
- name: metallb
repository: https://metallb.github.io/metallb
version: "0.12.1"
Expand Down
9 changes: 9 additions & 0 deletions helm-chart/templates/_helpers.tpl
Original file line number Diff line number Diff line change
Expand Up @@ -104,3 +104,12 @@ Return if ingress supports pathType.
{{- define "pgwatch2.ingress.supportsPathType" -}}
{{- or (eq (include "pgwatch2.ingress.isStable" .) "true") (and (eq (include "pgwatch2.ingress.apiVersion" .) "networking.k8s.io/v1beta1") (semverCompare ">= 1.18-0" .Capabilities.KubeVersion.Version)) -}}
{{- end -}}

{{- define "pgwatch2-storage" -}}
{{- if eq .Values.storage "influx" -}}
influxdb
{{- else -}}
{{- .Values.storage -}}
{{- end -}}
{{- end }}

213 changes: 211 additions & 2 deletions helm-chart/templates/configmaps.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,219 @@ kind: ConfigMap
metadata:
name: {{ include "pgwatch2.fullname" . }}-initdb
data:
initdb.sql: |
00_initdb.sql: |
CREATE USER {{ .Values.postgresql.user | default "pgwatch2" }} WITH PASSWORD '{{ .Values.postgresql.password | default "pgwatch2" }}';
CREATE DATABASE {{ .Values.postgresql.database | default "pgwatch2" }};
CREATE DATABASE {{ .Values.postgresql.database | default "pgwatch2" }} OWNER {{ .Values.postgresql.user | default "pgwatch2" }};
GRANT ALL PRIVILEGES ON DATABASE {{ .Values.postgresql.database | default "pgwatch2" }} TO {{ .Values.postgresql.user | default "pgwatch2" }};
{{ if eq .Values.storage "postgres" }}
01_create_metrics_db.sql: |
CREATE DATABASE {{ .Values.postgres_storage.database | default "pgwatch2_metrics" }} OWNER {{ .Values.postgresql.user | default "pgwatch2" }};
\c {{ .Values.postgres_storage.database | default "pgwatch2_metrics" }}

CREATE SCHEMA IF NOT EXISTS admin AUTHORIZATION {{ .Values.postgresql.user | default "pgwatch2" }};

GRANT ALL ON SCHEMA public TO {{ .Values.postgresql.user | default "pgwatch2" }};

DO $SQL$
BEGIN
EXECUTE format($$ALTER ROLE {{ .Values.postgresql.user | default "pgwatch2" }} IN DATABASE %s SET statement_timeout TO '5min'$$, current_database());
RAISE WARNING 'NB! Enabling asynchronous commit for pgwatch2 role - revert if possible data loss on crash is not acceptable!';
EXECUTE format($$ALTER ROLE {{ .Values.postgresql.user | default "pgwatch2" }} IN DATABASE %s SET synchronous_commit TO off$$, current_database());
END
$SQL$;

CREATE EXTENSION IF NOT EXISTS btree_gin;

SET ROLE TO {{ .Values.postgresql.user | default "pgwatch2" }};

create table admin.storage_schema_type (
schema_type text not null,
initialized_on timestamptz not null default now(),
check (schema_type in ('metric', 'metric-time', 'metric-dbname-time', 'custom', 'timescale'))
);

comment on table admin.storage_schema_type is 'identifies storage schema for other pgwatch2 components';

create unique index max_one_row on admin.storage_schema_type ((1));

/* for the Grafana drop-down. managed by the gatherer */
create table admin.all_distinct_dbname_metrics (
dbname text not null,
metric text not null,
created_on timestamptz not null default now(),
primary key (dbname, metric)
);

/* currently only used to store TimescaleDB chunk interval */
create table admin.config
(
key text not null primary key,
value text not null,
created_on timestamptz not null default now(),
last_modified_on timestamptz
);

-- to later change the value call the admin.change_timescale_chunk_interval(interval) function!
-- as changing the row directly will only be effective for completely new tables (metrics).
insert into admin.config select 'timescale_chunk_interval', '2 days';
insert into admin.config select 'timescale_compress_interval', '1 day';

create or replace function trg_config_modified() returns trigger
as $$
begin
new.last_modified_on = now();
return new;
end;
$$
language plpgsql;

create trigger config_modified before update on admin.config
for each row execute function trg_config_modified();

-- DROP FUNCTION IF EXISTS admin.ensure_dummy_metrics_table(text);
-- select * from admin.ensure_dummy_metrics_table('wal');
CREATE OR REPLACE FUNCTION admin.ensure_dummy_metrics_table(
metric text
)
RETURNS boolean AS
/*
creates a top level metric table if not already existing (non-existing tables show ugly warnings in Grafana).
expects the "metrics_template" table to exist.
*/
$SQL$
DECLARE
l_schema_type text;
l_template_table text := 'admin.metrics_template';
l_unlogged text := '';
BEGIN
SELECT schema_type INTO l_schema_type FROM admin.storage_schema_type;

IF NOT EXISTS (SELECT 1
FROM pg_tables
WHERE tablename = metric
AND schemaname = 'public')
THEN
IF metric ~ 'realtime' THEN
l_template_table := 'admin.metrics_template_realtime';
l_unlogged := 'UNLOGGED';
END IF;

IF l_schema_type = 'metric' THEN
EXECUTE format($$CREATE %s TABLE public."%s" (LIKE %s INCLUDING INDEXES)$$, l_unlogged, metric, l_template_table);
ELSIF l_schema_type = 'metric-time' THEN
EXECUTE format($$CREATE %s TABLE public."%s" (LIKE %s INCLUDING INDEXES) PARTITION BY RANGE (time)$$, l_unlogged, metric, l_template_table);
ELSIF l_schema_type = 'metric-dbname-time' THEN
EXECUTE format($$CREATE %s TABLE public."%s" (LIKE %s INCLUDING INDEXES) PARTITION BY LIST (dbname)$$, l_unlogged, metric, l_template_table);
ELSIF l_schema_type = 'timescale' THEN
IF metric ~ 'realtime' THEN
EXECUTE format($$CREATE TABLE public."%s" (LIKE %s INCLUDING INDEXES) PARTITION BY RANGE (time)$$, metric, l_template_table);
ELSE
PERFORM admin.ensure_partition_timescale(metric);
END IF;
END IF;

EXECUTE format($$COMMENT ON TABLE public."%s" IS 'pgwatch2-generated-metric-lvl'$$, metric);

RETURN true;

END IF;

RETURN false;
END;
$SQL$ LANGUAGE plpgsql;
GRANT EXECUTE ON FUNCTION admin.ensure_dummy_metrics_table(text) TO pgwatch2;

/*
NB! When possible the partitioned versions ("metric_store_part_time.sql"
or "metric_store_part_dbname_time.sql") (assuming PG11+) should be used
as much less IO would be then performed when removing old data.
NB! A fresh separate DB, only for pgwatch2 metrics storage purposes, is assumed.
*/


create table admin.metrics_template (
time timestamptz not null default now(),
dbname text not null,
data jsonb not null,
tag_data jsonb,
check (false)
);

comment on table admin.metrics_template is 'used as a template for all new metric definitions';

-- create index on admin.metrics_template using brin (dbname, time); /* consider BRIN instead for large data amounts */
create index on admin.metrics_template (dbname, time);
create index on admin.metrics_template using gin (dbname, tag_data, time) where tag_data notnull;

/*
something like below will be done by the gatherer AUTOMATICALLY:

create table public."some-metric"
(LIKE admin.metrics_template INCLUDING INDEXES);
COMMENT ON TABLE public."some-metric" IS 'pgwatch2-generated-metric-lvl';

*/


/* "realtime" metrics are non-persistent and have 1d retention */

-- drop table if exists metrics_template_realtime;
create unlogged table admin.metrics_template_realtime (
time timestamptz not null default now(),
dbname text not null,
data jsonb not null,
tag_data jsonb, -- no index!
check (false)
);

comment on table admin.metrics_template_realtime is 'used as a template for all new realtime metric definitions';

-- create index on admin.metrics_template using brin (dbname, time) with (pages_per_range=32); /* consider BRIN instead for large data amounts */
create index on admin.metrics_template_realtime (dbname, time);


RESET ROLE;

insert into admin.storage_schema_type select 'metric';

-- DROP FUNCTION IF EXISTS public.ensure_partition_metric(text);
-- select * from public.ensure_partition_metric('wal');

CREATE OR REPLACE FUNCTION admin.ensure_partition_metric(
metric text
)
RETURNS void AS
/*
creates a top level metric table if not already existing.
expects the "metrics_template" table to exist.
*/
$SQL$
DECLARE
l_template_table text := 'admin.metrics_template';
l_unlogged text := '';
BEGIN

PERFORM pg_advisory_xact_lock(regexp_replace( md5(metric) , E'\\D', '', 'g')::varchar(10)::int8);

IF NOT EXISTS (SELECT 1
FROM pg_tables
WHERE tablename = metric
AND schemaname = 'public')
THEN
--RAISE NOTICE 'creating partition % ...', metric;
IF metric ~ 'realtime' THEN
l_template_table := 'admin.metrics_template_realtime';
l_unlogged := 'UNLOGGED';
END IF;
EXECUTE format($$CREATE %s TABLE IF NOT EXISTS public.%s (LIKE %s INCLUDING INDEXES)$$, l_unlogged, quote_ident(metric), l_template_table);
EXECUTE format($$COMMENT ON TABLE public.%s IS 'pgwatch2-generated-metric-lvl'$$, quote_ident(metric));
END IF;

END;
$SQL$ LANGUAGE plpgsql;

GRANT EXECUTE ON FUNCTION admin.ensure_partition_metric(text) TO {{ .Values.postgresql.user | default "pgwatch2" }};
{{ end }}
---
apiVersion: v1
kind: ConfigMap
Expand Down
26 changes: 22 additions & 4 deletions helm-chart/templates/deployment.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ spec:
imagePullPolicy: {{ .Values.daemon.image.pullPolicy }}
env:
- name: PGHOST
value: {{ include "pgwatch2.fullname" . }}-postgresql
value: {{ .Values.postgresql.host | default (printf "%s-postgresql" (include "pgwatch2.fullname" .)) }}
- name: PGPORT
value: {{ .Values.postgresql.port | default "5432" | quote }}
- name: PGDATABASE
Expand All @@ -55,7 +55,7 @@ spec:
name: config-volume
env:
- name: PGHOST
value: {{ include "pgwatch2.fullname" . }}-postgresql
value: {{ .Values.postgresql.host | default (printf "%s-postgresql" (include "pgwatch2.fullname" .)) }}
- name: PGPORT
value: {{ .Values.postgresql.port | default "5432" | quote }}
- name: PGDATABASE
Expand All @@ -82,7 +82,7 @@ spec:
imagePullPolicy: {{ .Values.daemon.image.pullPolicy }}
env:
- name: PW2_PGHOST
value: {{ include "pgwatch2.fullname" . }}-postgresql
value: {{ .Values.postgresql.host | default (printf "%s-postgresql" (include "pgwatch2.fullname" .)) }}
- name: PW2_PGPORT
value: {{ .Values.postgresql.port | default "5432" | quote }}
- name: PW2_PGDATABASE
Expand All @@ -93,6 +93,7 @@ spec:
value: {{ .Values.postgresql.password | default "pgwatch2" }}
- name: PW2_PGSSL
value: {{ .Values.postgresql.ssl | default "False" | quote }}
{{ if eq .Values.storage "influx" }}
- name: PW2_IHOST
value: {{ include "pgwatch2.fullname" . }}-influxdb
- name: PW2_IPORT
Expand All @@ -105,6 +106,14 @@ spec:
value: {{ .Values.influxdb.password | default "pgwatch2" }}
- name: PW2_ISSL
value: {{ .Values.influxdb.ssl | default "False" | quote }}
{{ else if eq .Values.storage "postgres" }}
- name: PW2_DATASTORE
value: {{ .Values.storage | quote }}
- name: PW2_PG_METRIC_STORE_CONN_STR
value: {{ printf "postgresql://%s:%s@%s:%s/%s" (.Values.postgresql.user | default "pgwatch2") (.Values.postgresql.password | default "pgwatch2" ) ( .Values.postgresql.host | default (printf "%s-postgresql" (include "pgwatch2.fullname" .))) (.Values.postgresql.port | default "5432") (.Values.postgres_storage.database | default "pgwatch2_metrics" ) }}
- name: PW2_PG_RETENTION_DAYS
value: {{ .Values.postgres_storage.retention_days | default "14" | quote }}
{{ end }}
- name: PW2_INTERNAL_STATS_PORT
value: {{ .Values.daemon.port | default "8081" | quote }}
- name: PW2_WEBNOANONYMOUS
Expand Down Expand Up @@ -144,7 +153,7 @@ spec:
imagePullPolicy: {{ .Values.webui.image.pullPolicy }}
env:
- name: PW2_PGHOST
value: {{ include "pgwatch2.fullname" . }}-postgresql
value: {{ .Values.postgresql.host | default (printf "%s-postgresql" (include "pgwatch2.fullname" .)) }}
- name: PW2_PGPORT
value: {{ .Values.postgresql.port | default "5432" | quote }}
- name: PW2_PGDATABASE
Expand All @@ -155,6 +164,7 @@ spec:
value: {{ .Values.postgresql.password | default "pgwatch2" }}
- name: PW2_PGSSL
value: {{ .Values.postgresql.ssl | default "False" | quote}}
{{ if eq .Values.storage "influx" }}
- name: PW2_IHOST
value: {{ include "pgwatch2.fullname" . }}-influxdb
- name: PW2_IPORT
Expand All @@ -167,6 +177,14 @@ spec:
value: {{ .Values.influxdb.password | default "pgwatch2" }}
- name: PW2_ISSL
value: {{ .Values.influxdb.ssl | default "False" | quote }}
{{ else if eq .Values.storage "postgres" }}
- name: PW2_DATASTORE
value: {{ .Values.storage | quote }}
- name: PW2_PG_METRIC_STORE_CONN_STR
value: {{ printf "postgresql://%s:%s@%s:%s/%s" (.Values.postgresql.user | default "pgwatch2") (.Values.postgresql.password | default "pgwatch2" ) ( .Values.postgresql.host | default (printf "%s-postgresql" (include "pgwatch2.fullname" .))) (.Values.postgresql.port | default "5432") (.Values.postgres_storage.database | default "pgwatch2_metrics" ) }}
- name: PW2_PG_RETENTION_DAYS
value: {{ .Values.postgres_storage.retention_days | default "14" | quote }}
{{ end }}
- name: PW2_INTERNAL_STATS_PORT
value: {{ .Values.daemon.port | default "8081" | quote }}
- name: PW2_WEBNOANONYMOUS
Expand Down
6 changes: 3 additions & 3 deletions helm-chart/templates/grafana-dashboards.yaml
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
{{ range $path, $_ := .Files.Glob (printf "grafana_dashboards/%s/v%s/**.json" (include "pgwatch2-storage" .) (substr 0 1 .Subcharts.grafana.Chart.AppVersion)) }}
---
apiVersion: v1
kind: ConfigMap
metadata:
name: grafana-dashboards
name: grafana-dashboards-{{ regexReplaceAll ".*/" ($path | replace "/dashboard.json" "") "" }}
labels:
grafana_dashboard: "1"
data:
{{ range $path, $_ := .Files.Glob (printf "grafana_dashboards/influxdb/v%s/**.json" (substr 0 1 .Subcharts.grafana.Chart.AppVersion)) }}
{{ regexReplaceAll ".*/" ($path | replace "/dashboard.json" ".json") "" }}: |{{ printf "\n" }}
{{- $.Files.Get $path | indent 4}}
{{ end }}
{{ end }}
2 changes: 1 addition & 1 deletion helm-chart/templates/ingress.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
{{- $ingressPath := .Values.webui.ingress.path -}}
{{- $ingressPathType := .Values.webui.ingress.pathType -}}
{{- $extraPaths := .Values.webui.ingress.extraPaths -}}
apiVersion: {{ include "webui.ingress.apiVersion" . }}
apiVersion: {{ include "pgwatch2.ingress.apiVersion" . }}
kind: Ingress
metadata:
name: {{ $fullName }}
Expand Down
Loading