From 78fd5a2dbde93d8f5951e422c00e357813320250 Mon Sep 17 00:00:00 2001 From: Chris Gwilliams <517923+encima@users.noreply.github.com> Date: Thu, 11 Apr 2024 09:03:42 +0300 Subject: [PATCH] modify queries to be parameterized instead of formatted strings --- internal/inspect/bloat/bloat.go | 2 +- internal/inspect/index_sizes/index_sizes.go | 2 +- internal/inspect/queries.go | 88 +++++++------------ .../table_index_sizes/table_index_sizes.go | 2 +- .../total_index_size/total_index_size.go | 2 +- .../total_table_sizes/total_table_sizes.go | 2 +- .../inspect/unused_indexes/unused_indexes.go | 2 +- internal/inspect/vacuum_stats/vacuum_stats.go | 2 +- 8 files changed, 38 insertions(+), 64 deletions(-) diff --git a/internal/inspect/bloat/bloat.go b/internal/inspect/bloat/bloat.go index 4cd78d2fb..b9f63707a 100644 --- a/internal/inspect/bloat/bloat.go +++ b/internal/inspect/bloat/bloat.go @@ -27,7 +27,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.BloatQuery()) + rows, err := conn.Query(ctx, inspect.BLOAT_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) } diff --git a/internal/inspect/index_sizes/index_sizes.go b/internal/inspect/index_sizes/index_sizes.go index 57cca59f6..d05e72c42 100644 --- a/internal/inspect/index_sizes/index_sizes.go +++ b/internal/inspect/index_sizes/index_sizes.go @@ -24,7 +24,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.IndexSizesQuery()) + rows, err := conn.Query(ctx, inspect.INDEX_SIZE_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) } diff --git a/internal/inspect/queries.go b/internal/inspect/queries.go index 88d566293..f097d41b4 100644 --- a/internal/inspect/queries.go +++ b/internal/inspect/queries.go @@ -1,31 +1,20 @@ package inspect import ( - "fmt" - "strings" - "github.com/supabase/cli/internal/utils" ) -var excludedString string - -func init() { - var excluded = make([]string, len(utils.InternalSchemas)) - for i, schema := range utils.InternalSchemas { - excluded[i] = fmt.Sprintf("'%s'", schema) - } - excluded = append(excluded, "'pg_catalog'") - excludedString = strings.Join(excluded, ", ") +func GetExcludedSchemas() []string { + return append(utils.InternalSchemas, "pg_catalog") } -func BloatQuery() string { - return fmt.Sprintf(`WITH constants AS ( +const BLOAT_QUERY = `WITH constants AS ( SELECT current_setting('block_size')::numeric AS bs, 23 AS hdr, 4 AS ma ), bloat_info AS ( SELECT ma,bs,schemaname,tablename, - (datawidth+(hdr+ma-(case when hdr%%ma=0 THEN ma ELSE hdr%%ma END)))::numeric AS datahdr, - (maxfracsum*(nullhdr+ma-(case when nullhdr%%ma=0 THEN ma ELSE nullhdr%%ma END))) AS nullhdr2 + (datawidth+(hdr+ma-(case when hdr%ma=0 THEN ma ELSE hdr%ma END)))::numeric AS datahdr, + (maxfracsum*(nullhdr+ma-(case when nullhdr%ma=0 THEN ma ELSE nullhdr%ma END))) AS nullhdr2 FROM ( SELECT schemaname, tablename, hdr, ma, bs, @@ -43,7 +32,7 @@ func BloatQuery() string { SELECT schemaname, tablename, cc.relpages, bs, CEIL((cc.reltuples*((datahdr+ma- - (CASE WHEN datahdr%%ma=0 THEN ma ELSE datahdr%%ma END))+nullhdr2+4))/(bs-20::float)) AS otta + (CASE WHEN datahdr%ma=0 THEN ma ELSE datahdr%ma END))+nullhdr2+4))/(bs-20::float)) AS otta FROM bloat_info JOIN pg_class cc ON cc.relname = bloat_info.tablename JOIN pg_namespace nn ON cc.relnamespace = nn.oid AND nn.nspname = bloat_info.schemaname AND nn.nspname <> 'information_schema' @@ -79,9 +68,8 @@ SELECT FROM index_bloat) bloat_summary WHERE - schemaname NOT IN (%s) -ORDER BY raw_waste DESC, bloat DESC`, excludedString) -} + schemaname NOT IN (SELECT unnest($1::text[])) +ORDER BY raw_waste DESC, bloat DESC` // Ref: https://github.com/heroku/heroku-pg-extras/blob/main/commands/blocking.js#L7 const BLOCKING_QUERY = ` @@ -122,17 +110,15 @@ SELECT sum(heap_blks_hit) / nullif(sum(heap_blks_hit) + sum(heap_blks_read),0) AS ratio FROM pg_statio_user_tables` -func IndexSizesQuery() string { - return fmt.Sprintf(`SELECT c.relname AS name, +const INDEX_SIZE_QUERY = `SELECT c.relname AS name, pg_size_pretty(sum(c.relpages::bigint*8192)::bigint) AS size FROM pg_class c LEFT JOIN pg_namespace n ON (n.oid = c.relnamespace) -WHERE n.nspname NOT IN (%s) +WHERE n.nspname NOT IN (%1) AND n.nspname !~ '^pg_toast' AND c.relkind='i' GROUP BY c.relname -ORDER BY sum(c.relpages) DESC`, excludedString) -} +ORDER BY sum(c.relpages) DESC` const INDEX_USAGE_QUERY = `SELECT relname, CASE @@ -222,16 +208,14 @@ FROM pg_stat_user_tables ORDER BY seq_scan DESC` -func TableIndexSizesQuery() string { - return fmt.Sprintf(`SELECT c.relname AS table, +const TABLE_INDEX_SIZE_QUERY = `SELECT c.relname AS table, pg_size_pretty(pg_indexes_size(c.oid)) AS index_size FROM pg_class c LEFT JOIN pg_namespace n ON (n.oid = c.relnamespace) -WHERE n.nspname NOT IN (%s) +WHERE n.nspname NOT IN (SELECT unnest($1::text[])) AND n.nspname !~ '^pg_toast' AND c.relkind='r' -ORDER BY pg_indexes_size(c.oid) DESC`, excludedString) -} +ORDER BY pg_indexes_size(c.oid) DESC` const TABLE_RECORD_COUNTS_QUERY = `SELECT relname AS name, @@ -241,52 +225,43 @@ pg_stat_user_tables ORDER BY n_live_tup DESC` -func TableSizesQuery() string { - return fmt.Sprintf(`SELECT c.relname AS name, +const TABLE_SIZE_QUERY = `SELECT c.relname AS name, pg_size_pretty(pg_table_size(c.oid)) AS size FROM pg_class c LEFT JOIN pg_namespace n ON (n.oid = c.relnamespace) -WHERE n.nspname NOT IN (%s) +WHERE n.nspname NOT IN (SELECT unnest($1::text[])) AND n.nspname !~ '^pg_toast' AND c.relkind='r' -ORDER BY pg_table_size(c.oid) DESC`, excludedString) -} +ORDER BY pg_table_size(c.oid) DESC` -func TotalIndexSizesQuery() string { - return fmt.Sprintf(`SELECT pg_size_pretty(sum(c.relpages::bigint*8192)::bigint) AS size +const TOTAL_INDEX_SIZE_QUERY = `SELECT pg_size_pretty(sum(c.relpages::bigint*8192)::bigint) AS size FROM pg_class c LEFT JOIN pg_namespace n ON (n.oid = c.relnamespace) -WHERE n.nspname NOT IN (%s) +WHERE n.nspname NOT IN (SELECT unnest($1::text[])) AND n.nspname !~ '^pg_toast' -AND c.relkind='i'`, excludedString) -} +AND c.relkind='i'` -func TotalTableSizesQuery() string { - return fmt.Sprintf(`SELECT c.relname AS name, +const TOTAL_TABLE_SIZE_QUERY = `SELECT c.relname AS name, pg_size_pretty(pg_total_relation_size(c.oid)) AS size FROM pg_class c LEFT JOIN pg_namespace n ON (n.oid = c.relnamespace) -WHERE n.nspname NOT IN (%s) +WHERE n.nspname NOT IN (SELECT unnest($1::text[])) AND n.nspname !~ '^pg_toast' AND c.relkind='r' -ORDER BY pg_total_relation_size(c.oid) DESC`, excludedString) -} +ORDER BY pg_total_relation_size(c.oid) DESC` -func UnusedIndexesQuery() string { - return fmt.Sprintf(`SELECT +const UNUSED_INDEXES_QUERY = `SELECT schemaname || '.' || relname AS table, indexrelname AS index, pg_size_pretty(pg_relation_size(i.indexrelid)) AS index_size, idx_scan as index_scans FROM pg_stat_user_indexes ui JOIN pg_index i ON ui.indexrelid = i.indexrelid -WHERE NOT indisunique AND idx_scan < 50 AND pg_relation_size(relid) > 5 * 8192 AND schemaname NOT IN (%s) +WHERE NOT indisunique AND idx_scan < 50 AND pg_relation_size(relid) > 5 * 8192 AND schemaname NOT IN (SELECT unnest($1::text[])) ORDER BY pg_relation_size(i.indexrelid) / nullif(idx_scan, 0) DESC NULLS FIRST, -pg_relation_size(i.indexrelid) DESC`, excludedString) -} +pg_relation_size(i.indexrelid) DESC` -func VacuumStatsQuery() string { - return fmt.Sprintf(`WITH table_opts AS ( +const VACUUM_STATS_QUERY = `WITH table_opts AS ( SELECT pg_class.oid, relname, nspname, array_to_string(reloptions, '') AS relopts FROM @@ -295,12 +270,12 @@ func VacuumStatsQuery() string { SELECT oid, relname, nspname, CASE - WHEN relopts LIKE '%%autovacuum_vacuum_threshold%%' + WHEN relopts LIKE '%autovacuum_vacuum_threshold%' THEN substring(relopts, '.*autovacuum_vacuum_threshold=([0-9.]+).*')::integer ELSE current_setting('autovacuum_vacuum_threshold')::integer END AS autovacuum_vacuum_threshold, CASE - WHEN relopts LIKE '%%autovacuum_vacuum_scale_factor%%' + WHEN relopts LIKE '%autovacuum_vacuum_scale_factor%' THEN substring(relopts, '.*autovacuum_vacuum_scale_factor=([0-9.]+).*')::real ELSE current_setting('autovacuum_vacuum_scale_factor')::real END AS autovacuum_vacuum_scale_factor @@ -324,11 +299,10 @@ SELECT FROM pg_stat_user_tables psut INNER JOIN pg_class ON psut.relid = pg_class.oid INNER JOIN vacuum_settings ON pg_class.oid = vacuum_settings.oid -WHERE schema NOT IN (%s) +WHERE schema NOT IN (SELECT unnest($1::text[])) ORDER BY case when pg_class.reltuples = -1 then 1 else 0 end, - 1`, excludedString) -} + 1` diff --git a/internal/inspect/table_index_sizes/table_index_sizes.go b/internal/inspect/table_index_sizes/table_index_sizes.go index f4f8addb3..0b7f7b6e7 100644 --- a/internal/inspect/table_index_sizes/table_index_sizes.go +++ b/internal/inspect/table_index_sizes/table_index_sizes.go @@ -24,7 +24,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.TableIndexSizesQuery()) + rows, err := conn.Query(ctx, inspect.TABLE_INDEX_SIZE_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) } diff --git a/internal/inspect/total_index_size/total_index_size.go b/internal/inspect/total_index_size/total_index_size.go index b59e8590f..111f4a846 100644 --- a/internal/inspect/total_index_size/total_index_size.go +++ b/internal/inspect/total_index_size/total_index_size.go @@ -23,7 +23,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.TotalIndexSizesQuery()) + rows, err := conn.Query(ctx, inspect.TOTAL_INDEX_SIZE_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) } diff --git a/internal/inspect/total_table_sizes/total_table_sizes.go b/internal/inspect/total_table_sizes/total_table_sizes.go index 2016af2e3..436d873a8 100644 --- a/internal/inspect/total_table_sizes/total_table_sizes.go +++ b/internal/inspect/total_table_sizes/total_table_sizes.go @@ -24,7 +24,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.TotalTableSizesQuery()) + rows, err := conn.Query(ctx, inspect.TOTAL_TABLE_SIZE_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) } diff --git a/internal/inspect/unused_indexes/unused_indexes.go b/internal/inspect/unused_indexes/unused_indexes.go index 58cd3ce92..f1c17ffcf 100644 --- a/internal/inspect/unused_indexes/unused_indexes.go +++ b/internal/inspect/unused_indexes/unused_indexes.go @@ -26,7 +26,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.UnusedIndexesQuery()) + rows, err := conn.Query(ctx, inspect.UNUSED_INDEXES_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) } diff --git a/internal/inspect/vacuum_stats/vacuum_stats.go b/internal/inspect/vacuum_stats/vacuum_stats.go index 332305d3b..e3d7896ac 100644 --- a/internal/inspect/vacuum_stats/vacuum_stats.go +++ b/internal/inspect/vacuum_stats/vacuum_stats.go @@ -31,7 +31,7 @@ func Run(ctx context.Context, config pgconn.Config, fsys afero.Fs, options ...fu if err != nil { return err } - rows, err := conn.Query(ctx, inspect.VacuumStatsQuery()) + rows, err := conn.Query(ctx, inspect.VACUUM_STATS_QUERY, inspect.GetExcludedSchemas()) if err != nil { return errors.Errorf("failed to query rows: %w", err) }