mirror of
https://github.com/googleapis/genai-toolbox.git
synced 2026-02-06 05:04:58 -05:00
Introduces a new --prebuilt <source_type> flag. This flag is mutually exclusive with the existing --tools-file flag. Added a new directory cmd/prebuiltconfigs/ to store the prebuilt tools.yaml files (e.g., alloydb.yaml, postgres.yaml, etc.). These YAML files are embedded into the Go binary using the //go:embed directive. --------- Co-authored-by: Averi Kitsch <akitsch@google.com>
100 lines
7.2 KiB
YAML
100 lines
7.2 KiB
YAML
sources:
|
|
cloudsql-pg-source:
|
|
kind: cloud-sql-postgres
|
|
project: ${CLOUD_SQL_POSTGRES_PROJECT}
|
|
region: ${CLOUD_SQL_POSTGRES_REGION}
|
|
instance: ${CLOUD_SQL_POSTGRES_INSTANCE}
|
|
database: ${CLOUD_SQL_POSTGRES_DATABASE}
|
|
user: ${CLOUD_SQL_POSTGRES_USER}
|
|
password: ${CLOUD_SQL_POSTGRES_PASSWORD}
|
|
|
|
tools:
|
|
execute_sql:
|
|
kind: postgres-execute-sql
|
|
source: cloudsql-pg-source
|
|
description: Use this tool to execute sql.
|
|
|
|
list_tables:
|
|
kind: postgres-sql
|
|
source: cloudsql-pg-source
|
|
description: "Lists detailed schema information (object type, columns, constraints, indexes, triggers, owner, comment) as JSON for user-created tables (ordinary or partitioned). Filters by a comma-separated list of names. If names are omitted, lists all tables in user schemas."
|
|
statement: |
|
|
WITH desired_relkinds AS (
|
|
SELECT ARRAY['r', 'p']::char[] AS kinds -- Always consider both 'TABLE' and 'PARTITIONED TABLE'
|
|
),
|
|
table_info AS (
|
|
SELECT
|
|
t.oid AS table_oid,
|
|
ns.nspname AS schema_name,
|
|
t.relname AS table_name,
|
|
pg_get_userbyid(t.relowner) AS table_owner,
|
|
obj_description(t.oid, 'pg_class') AS table_comment,
|
|
t.relkind AS object_kind
|
|
FROM
|
|
pg_class t
|
|
JOIN
|
|
pg_namespace ns ON ns.oid = t.relnamespace
|
|
CROSS JOIN desired_relkinds dk
|
|
WHERE
|
|
t.relkind = ANY(dk.kinds) -- Filter by selected table relkinds ('r', 'p')
|
|
AND (NULLIF(TRIM($1), '') IS NULL OR t.relname = ANY(string_to_array($1,','))) -- $1 is object_names
|
|
AND ns.nspname NOT IN ('pg_catalog', 'information_schema', 'pg_toast')
|
|
AND ns.nspname NOT LIKE 'pg_temp_%' AND ns.nspname NOT LIKE 'pg_toast_temp_%'
|
|
),
|
|
columns_info AS (
|
|
SELECT
|
|
att.attrelid AS table_oid, att.attname AS column_name, format_type(att.atttypid, att.atttypmod) AS data_type,
|
|
att.attnum AS column_ordinal_position, att.attnotnull AS is_not_nullable,
|
|
pg_get_expr(ad.adbin, ad.adrelid) AS column_default, col_description(att.attrelid, att.attnum) AS column_comment
|
|
FROM pg_attribute att LEFT JOIN pg_attrdef ad ON att.attrelid = ad.adrelid AND att.attnum = ad.adnum
|
|
JOIN table_info ti ON att.attrelid = ti.table_oid WHERE att.attnum > 0 AND NOT att.attisdropped
|
|
),
|
|
constraints_info AS (
|
|
SELECT
|
|
con.conrelid AS table_oid, con.conname AS constraint_name, pg_get_constraintdef(con.oid) AS constraint_definition,
|
|
CASE con.contype WHEN 'p' THEN 'PRIMARY KEY' WHEN 'f' THEN 'FOREIGN KEY' WHEN 'u' THEN 'UNIQUE' WHEN 'c' THEN 'CHECK' ELSE con.contype::text END AS constraint_type,
|
|
(SELECT array_agg(att.attname ORDER BY u.attposition) FROM unnest(con.conkey) WITH ORDINALITY AS u(attnum, attposition) JOIN pg_attribute att ON att.attrelid = con.conrelid AND att.attnum = u.attnum) AS constraint_columns,
|
|
NULLIF(con.confrelid, 0)::regclass AS foreign_key_referenced_table,
|
|
(SELECT array_agg(att.attname ORDER BY u.attposition) FROM unnest(con.confkey) WITH ORDINALITY AS u(attnum, attposition) JOIN pg_attribute att ON att.attrelid = con.confrelid AND att.attnum = u.attnum WHERE con.contype = 'f') AS foreign_key_referenced_columns
|
|
FROM pg_constraint con JOIN table_info ti ON con.conrelid = ti.table_oid
|
|
),
|
|
indexes_info AS (
|
|
SELECT
|
|
idx.indrelid AS table_oid, ic.relname AS index_name, pg_get_indexdef(idx.indexrelid) AS index_definition,
|
|
idx.indisunique AS is_unique, idx.indisprimary AS is_primary, am.amname AS index_method,
|
|
(SELECT array_agg(att.attname ORDER BY u.ord) FROM unnest(idx.indkey::int[]) WITH ORDINALITY AS u(colidx, ord) LEFT JOIN pg_attribute att ON att.attrelid = idx.indrelid AND att.attnum = u.colidx WHERE u.colidx <> 0) AS index_columns
|
|
FROM pg_index idx JOIN pg_class ic ON ic.oid = idx.indexrelid JOIN pg_am am ON am.oid = ic.relam JOIN table_info ti ON idx.indrelid = ti.table_oid
|
|
),
|
|
triggers_info AS (
|
|
SELECT tg.tgrelid AS table_oid, tg.tgname AS trigger_name, pg_get_triggerdef(tg.oid) AS trigger_definition, tg.tgenabled AS trigger_enabled_state
|
|
FROM pg_trigger tg JOIN table_info ti ON tg.tgrelid = ti.table_oid WHERE NOT tg.tgisinternal
|
|
)
|
|
SELECT
|
|
ti.schema_name,
|
|
ti.table_name AS object_name,
|
|
json_build_object(
|
|
'schema_name', ti.schema_name,
|
|
'object_name', ti.table_name,
|
|
'object_type', CASE ti.object_kind
|
|
WHEN 'r' THEN 'TABLE'
|
|
WHEN 'p' THEN 'PARTITIONED TABLE'
|
|
ELSE ti.object_kind::text -- Should not happen due to WHERE clause
|
|
END,
|
|
'owner', ti.table_owner,
|
|
'comment', ti.table_comment,
|
|
'columns', COALESCE((SELECT json_agg(json_build_object('column_name',ci.column_name,'data_type',ci.data_type,'ordinal_position',ci.column_ordinal_position,'is_not_nullable',ci.is_not_nullable,'column_default',ci.column_default,'column_comment',ci.column_comment) ORDER BY ci.column_ordinal_position) FROM columns_info ci WHERE ci.table_oid = ti.table_oid), '[]'::json),
|
|
'constraints', COALESCE((SELECT json_agg(json_build_object('constraint_name',cons.constraint_name,'constraint_type',cons.constraint_type,'constraint_definition',cons.constraint_definition,'constraint_columns',cons.constraint_columns,'foreign_key_referenced_table',cons.foreign_key_referenced_table,'foreign_key_referenced_columns',cons.foreign_key_referenced_columns)) FROM constraints_info cons WHERE cons.table_oid = ti.table_oid), '[]'::json),
|
|
'indexes', COALESCE((SELECT json_agg(json_build_object('index_name',ii.index_name,'index_definition',ii.index_definition,'is_unique',ii.is_unique,'is_primary',ii.is_primary,'index_method',ii.index_method,'index_columns',ii.index_columns)) FROM indexes_info ii WHERE ii.table_oid = ti.table_oid), '[]'::json),
|
|
'triggers', COALESCE((SELECT json_agg(json_build_object('trigger_name',tri.trigger_name,'trigger_definition',tri.trigger_definition,'trigger_enabled_state',tri.trigger_enabled_state)) FROM triggers_info tri WHERE tri.table_oid = ti.table_oid), '[]'::json)
|
|
) AS object_details
|
|
FROM table_info ti ORDER BY ti.schema_name, ti.table_name;
|
|
parameters:
|
|
- name: table_names
|
|
type: string
|
|
description: "Optional: A comma-separated list of table names. If empty, details for all tables in user-accessible schemas will be listed."
|
|
|
|
toolsets:
|
|
cloud-sql-postgres-database-tools:
|
|
- execute_sql
|
|
- list_tables
|