mirror of
https://github.com/supabase/supabase.git
synced 2026-06-18 05:33:50 +08:00
Extends the SQL editor's destructive query detection to catch dangerous operations (DROP, DELETE, TRUNCATE) hidden within dynamic SQL execution patterns like `EXECUTE`, `EXECUTE format()`, `OPEN cursor FOR EXECUTE`, and `RETURN QUERY EXECUTE`. Previously, only direct statements like `DROP TABLE users` triggered warnings. Dynamic SQL inside PL/pgSQL blocks bypassed detection entirely. Closes FE-3603 and https://github.com/supabase/supabase/issues/46876 <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **Bug Fixes** * Improved detection of potentially destructive SQL across many execution styles, including DROP/DELETE/TRUNCATE, ALTER ... DROP COLUMN, and dynamic/executed or string-constructed queries to reduce missed destructive cases. * **Tests** * Expanded test coverage for SQL validation with extensive positive and negative scenarios covering dynamic execution patterns (EXECUTE variants, concatenation, dollar-quoted and escaped strings) to reduce false positives and negatives. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
947 lines
32 KiB
TypeScript
947 lines
32 KiB
TypeScript
import { safeSql } from '@supabase/pg-meta'
|
|
import { stripIndent } from 'common-tags'
|
|
import { describe, expect, it, test } from 'vitest'
|
|
|
|
import {
|
|
appendEnableRLSStatements,
|
|
checkAlterDatabaseConnection,
|
|
checkDestructiveQuery,
|
|
checkIfAppendLimitRequired,
|
|
filterTablesCoveredByEnsureRLSTrigger,
|
|
getCreateTablesMissingRLS,
|
|
hasActiveEnsureRLSTrigger,
|
|
isUpdateWithoutWhere,
|
|
suffixWithLimit,
|
|
} from './SQLEditor.utils'
|
|
import type { DatabaseEventTrigger } from '@/data/database-event-triggers/database-event-triggers-query'
|
|
|
|
const buildTrigger = (overrides: Partial<DatabaseEventTrigger> = {}): DatabaseEventTrigger => ({
|
|
oid: 1,
|
|
name: 'ensure_rls',
|
|
event: 'ddl_command_end',
|
|
enabled_mode: 'ORIGIN',
|
|
tags: ['CREATE TABLE'],
|
|
function_name: 'rls_auto_enable',
|
|
function_schema: 'public',
|
|
owner: 'postgres',
|
|
function_definition: null,
|
|
...overrides,
|
|
})
|
|
|
|
describe('SQLEditor.utils.ts:checkIfAppendLimitRequired', () => {
|
|
test('Should return false if limit passed is <= 0', () => {
|
|
const sql = 'select * from countries;'
|
|
const limit = -1
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return true if limit passed is > 0', () => {
|
|
const sql = 'select * from countries;'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(true)
|
|
})
|
|
test('Should return false if query already has a limit', () => {
|
|
const sql = 'select * from countries limit 10;'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query already has a limit (check for case-insensitiveness)', () => {
|
|
const sql = 'SELECT * FROM countries LIMIT 10;'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query already has a limit and offset', () => {
|
|
const sql = 'select * from countries limit 10 offset 0;'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query already has a limit and offset (flip order of limit and offset)', () => {
|
|
const sql = 'select * from countries offset 0 limit 1;'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query already has a limit, even if no value provided for limit', () => {
|
|
const sql = 'select * from countries limit'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query uses `FETCH FIRST` instead of limit ', () => {
|
|
const sql = 'select * from countries FETCH FIRST 5 rows only'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query uses `fetch first` instead of limit ', () => {
|
|
const sql = 'select * from countries fetch first 5 rows only'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query uses `fetch first` (with random spaces) instead of limit ', () => {
|
|
const sql = 'select * from countries FETCH FIRST 5 rows only'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query is not a select statement', () => {
|
|
const sql = 'create table test (id int8 primary key, name varchar);'
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if there are multiple queries I', () => {
|
|
const sql1 = `
|
|
select * from countries;
|
|
select * from cities;
|
|
`.trim()
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql1, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if there are multiple queries II', () => {
|
|
const sql1 = `
|
|
select * from countries;
|
|
select * from cities
|
|
`.trim()
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql1, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
// [Joshen] Opting to just avoid appending in this case to prevent making the logic overly complex atm
|
|
test('Should return false if query has with a comment I', () => {
|
|
const sql = `
|
|
-- This is a comment
|
|
select * from cities
|
|
`.trim()
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
test('Should return false if query has with a comment II', () => {
|
|
const sql = `
|
|
select * from cities
|
|
-- This is a comment
|
|
`.trim()
|
|
const limit = 100
|
|
const { appendAutoLimit } = checkIfAppendLimitRequired(sql, limit)
|
|
expect(appendAutoLimit).toBe(false)
|
|
})
|
|
})
|
|
|
|
// [Joshen] These will just need to test the cases when appendAutoLimit returns true then
|
|
describe('SQLEditor.utils.ts:suffixWithLimit', () => {
|
|
test('Should add the limit param properly if query ends without a semi colon', () => {
|
|
const sql = safeSql`select * from countries`
|
|
const limit = 100
|
|
const formattedSql = suffixWithLimit(sql, limit)
|
|
expect(formattedSql).toBe('select * from countries limit 100;')
|
|
})
|
|
test('Should add the limit param properly if query ends with a semi colon', () => {
|
|
const sql = safeSql`select * from countries;`
|
|
const limit = 100
|
|
const formattedSql = suffixWithLimit(sql, limit)
|
|
expect(formattedSql).toBe('select * from countries limit 100;')
|
|
})
|
|
test('Should add the limit param properly if query ends with multiple semi colon', () => {
|
|
const sql = safeSql`select * from countries;;;;;;;`
|
|
const limit = 100
|
|
const formattedSql = suffixWithLimit(sql, limit)
|
|
expect(formattedSql).toBe('select * from countries limit 100;')
|
|
})
|
|
})
|
|
|
|
describe(`SQLEditor.utils.ts:checkDestructiveQuery`, () => {
|
|
it('drop statement matches', () => {
|
|
const match = checkDestructiveQuery('drop table films, distributors;')
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('truncate statement matches', () => {
|
|
const match = checkDestructiveQuery('truncate films;')
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('delete statement matches', () => {
|
|
const match = checkDestructiveQuery("delete from films where kind <> 'Musical';")
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('delete statement after another statement matches', () => {
|
|
const match = checkDestructiveQuery(stripIndent`
|
|
select * from films;
|
|
|
|
delete from films where kind <> 'Musical';
|
|
`)
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it("rls policy containing delete doesn't match", () => {
|
|
const match = checkDestructiveQuery(stripIndent`
|
|
create policy "Users can delete their own files"
|
|
on storage.objects for delete to authenticated using (
|
|
bucket id = 'files' and (select auth.uid()) = owner
|
|
);
|
|
`)
|
|
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('capitalized statement matches', () => {
|
|
const match = checkDestructiveQuery("DELETE FROM films WHERE kind <> 'Musical';")
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it("comment containing keyword doesn't match", () => {
|
|
const match = checkDestructiveQuery(stripIndent`
|
|
-- Going to drop this in here, might delete later
|
|
select * from films;
|
|
`)
|
|
|
|
expect(match).toBe(false)
|
|
})
|
|
})
|
|
|
|
describe('SQLEditor.utils:updateWithoutWhere', () => {
|
|
it('contains an update query with a where clause', () => {
|
|
const match = isUpdateWithoutWhere(stripIndent`
|
|
UPDATE public.countries SET name = 'New Name' WHERE id = 1;
|
|
`)
|
|
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('contains an update query without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(stripIndent`
|
|
UPDATE public.countries SET name = 'New Name';
|
|
`)
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('contains an update query, with quoted identifiers with a where clause', () => {
|
|
const match = isUpdateWithoutWhere(stripIndent`
|
|
UPDATE "public"."countries" SET name = 'New Name' WHERE id = 1;
|
|
`)
|
|
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('contains an update query, with quoted identifiers without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(stripIndent`
|
|
UPDATE "public"."countries" SET name = 'New Name';
|
|
`)
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('catches update on a single quoted table name without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE "messages" SET id = 1;`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('does not flag update on a single quoted table name with a where clause', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE "messages" SET id = 1 WHERE id = 2;`)
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('catches update on a quoted schema with a bareword table without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE "public".messages SET id = 1;`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('catches update on a bareword schema with a quoted table without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE public."messages" SET id = 1;`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('catches update on a quoted table name containing a space without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE "my table" SET id = 1;`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('catches update on a quoted table name containing escaped quotes without a where clause', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE "weird""name" SET id = 1;`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('catches update where a quoted identifier contains the word where', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE "where table" SET id = 1;`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('catches update where a string literal contains the word where', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE messages SET name = 'where x';`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('does not flag update where the only "where" sits inside a string literal but a real where clause exists', () => {
|
|
const match = isUpdateWithoutWhere(`UPDATE messages SET name = 'where x' WHERE id = 1;`)
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('contains both an update query and a delete query, triggers destructive', () => {
|
|
const match = checkDestructiveQuery(stripIndent`
|
|
delete from countries; update countries set name = 'hello';
|
|
`)
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('contains both an update query and a delete query, triggers no where', () => {
|
|
const match = isUpdateWithoutWhere(stripIndent`
|
|
delete from countries; update countries set name = 'hello';
|
|
`)
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
it('contains both an update query and a delete query, triggers no where', () => {
|
|
const match = isUpdateWithoutWhere(stripIndent`
|
|
delete from countries; update countries set name = 'hello';
|
|
`)
|
|
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('should catch potential destructive queries', () => {
|
|
const DESTRUCTIVE_QUERIES = [
|
|
`ALTER TABLE test DROP COLUMN test;`,
|
|
`DELETE FROM test;`,
|
|
`DROP TABLE test;`,
|
|
`TRUNCATE TABLE test;`,
|
|
]
|
|
|
|
DESTRUCTIVE_QUERIES.forEach((query) => {
|
|
expect(checkDestructiveQuery(query), `Query ${query} should be destructive`).toBe(true)
|
|
})
|
|
})
|
|
|
|
it('should catch EXECUTE with DROP in string literal', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'DROP TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with DELETE in string literal', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'DELETE FROM users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with TRUNCATE in string literal', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'TRUNCATE TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE format with DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE format('DROP TABLE %I', table_name);`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE format with DELETE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE format('DELETE FROM %I', table_name);`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE format with TRUNCATE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE format('TRUNCATE TABLE %I', table_name);`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with string concatenation containing DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'DROP TABLE ' || table_name;`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with string concatenation containing DELETE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'DELETE FROM ' || table_name;`)).toBe(true)
|
|
})
|
|
|
|
it('should catch DO block with EXECUTE DROP', () => {
|
|
expect(
|
|
checkDestructiveQuery(stripIndent`
|
|
DO $$
|
|
BEGIN
|
|
EXECUTE 'DROP TABLE users';
|
|
END
|
|
$$;
|
|
`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch DO block with EXECUTE format DROP', () => {
|
|
expect(
|
|
checkDestructiveQuery(stripIndent`
|
|
DO $$
|
|
BEGIN
|
|
FOR rec IN SELECT tablename FROM pg_tables WHERE schemaname = 'public' LOOP
|
|
EXECUTE format('DROP TABLE %I', rec.tablename);
|
|
END LOOP;
|
|
END
|
|
$$;
|
|
`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch DO block with multiple EXECUTE format DROPs', () => {
|
|
expect(
|
|
checkDestructiveQuery(stripIndent`
|
|
DO $$
|
|
DECLARE
|
|
func_name text;
|
|
BEGIN
|
|
FOR func_name IN
|
|
SELECT routine_name FROM information_schema.routines WHERE routine_schema = 'public'
|
|
LOOP
|
|
EXECUTE format('DROP FUNCTION %I', func_name);
|
|
END LOOP;
|
|
END
|
|
$$;
|
|
`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch lowercase execute with drop', () => {
|
|
expect(checkDestructiveQuery(`execute 'drop table users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch mixed case EXECUTE Format with DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE Format('DROP TABLE users');`)).toBe(true)
|
|
})
|
|
|
|
it('should not flag EXECUTE with safe SQL', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'SELECT * FROM users';`)).toBe(false)
|
|
})
|
|
|
|
it('should not flag EXECUTE with INSERT', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'INSERT INTO users (name) VALUES (''test'')';`)).toBe(
|
|
false
|
|
)
|
|
})
|
|
|
|
it('should not flag EXECUTE with UPDATE and WHERE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'UPDATE users SET name = ''test'' WHERE id = 1';`)).toBe(
|
|
false
|
|
)
|
|
})
|
|
|
|
it('should catch EXECUTE IMMEDIATE with DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE IMMEDIATE 'DROP TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE IMMEDIATE with DELETE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE IMMEDIATE 'DELETE FROM users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE IMMEDIATE with TRUNCATE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE IMMEDIATE 'TRUNCATE TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch OPEN cursor FOR EXECUTE with DROP', () => {
|
|
expect(checkDestructiveQuery(`OPEN ref FOR EXECUTE 'DROP TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch OPEN cursor FOR EXECUTE format with DELETE', () => {
|
|
expect(checkDestructiveQuery(`OPEN ref FOR EXECUTE format('DELETE FROM %I', tbl);`)).toBe(true)
|
|
})
|
|
|
|
it('should catch OPEN cursor FOR EXECUTE with string concat', () => {
|
|
expect(checkDestructiveQuery(`OPEN ref FOR EXECUTE 'TRUNCATE TABLE ' || tbl;`)).toBe(true)
|
|
})
|
|
|
|
it('should catch RETURN QUERY EXECUTE with DROP', () => {
|
|
expect(checkDestructiveQuery(`RETURN QUERY EXECUTE 'DROP TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch RETURN QUERY EXECUTE format with DELETE', () => {
|
|
expect(checkDestructiveQuery(`RETURN QUERY EXECUTE format('DELETE FROM %I', tbl);`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with dollar-quoted string containing DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE $sql$DROP TABLE users$sql$;`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with dollar-quoted string containing TRUNCATE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE $body$TRUNCATE TABLE logs$body$;`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE concat with DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE concat('DROP TABLE ', 'users');`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE concat_ws with DELETE', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE concat_ws(' ', 'DELETE FROM', 'users');`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with E-string escape containing DROP', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE E'DROP TABLE users';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE with ALTER TABLE DROP COLUMN', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'ALTER TABLE users DROP COLUMN email';`)).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE format with ALTER TABLE DROP COLUMN', () => {
|
|
expect(
|
|
checkDestructiveQuery(`EXECUTE format('ALTER TABLE %I DROP COLUMN %I', 'users', 'email');`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE IMMEDIATE with ALTER TABLE DROP COLUMN', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE IMMEDIATE 'ALTER TABLE users DROP COLUMN email';`)).toBe(
|
|
true
|
|
)
|
|
})
|
|
|
|
it('should catch OPEN cursor FOR EXECUTE with ALTER TABLE DROP COLUMN', () => {
|
|
expect(
|
|
checkDestructiveQuery(`OPEN ref FOR EXECUTE 'ALTER TABLE users DROP COLUMN email';`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch RETURN QUERY EXECUTE with ALTER TABLE DROP COLUMN', () => {
|
|
expect(
|
|
checkDestructiveQuery(`RETURN QUERY EXECUTE 'ALTER TABLE users DROP COLUMN email';`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch EXECUTE dollar-quoted with ALTER TABLE DROP COLUMN', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE $sql$ALTER TABLE users DROP COLUMN email$sql$;`)).toBe(
|
|
true
|
|
)
|
|
})
|
|
|
|
it('should catch EXECUTE concat with ALTER TABLE DROP COLUMN', () => {
|
|
expect(
|
|
checkDestructiveQuery(`EXECUTE concat('ALTER TABLE ', 'users DROP COLUMN email');`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should catch DO block with EXECUTE ALTER TABLE DROP COLUMN', () => {
|
|
expect(
|
|
checkDestructiveQuery(stripIndent`
|
|
DO $$
|
|
BEGIN
|
|
EXECUTE 'ALTER TABLE users DROP COLUMN email';
|
|
END
|
|
$$;
|
|
`)
|
|
).toBe(true)
|
|
})
|
|
|
|
it('should not flag ALTER TABLE without DROP COLUMN', () => {
|
|
expect(checkDestructiveQuery(`EXECUTE 'ALTER TABLE users ADD COLUMN email text';`)).toBe(false)
|
|
})
|
|
|
|
it('should not flag variable assignment with safe SQL', () => {
|
|
expect(checkDestructiveQuery(`sql := 'SELECT * FROM users';`)).toBe(false)
|
|
})
|
|
|
|
it('should not flag OPEN cursor FOR EXECUTE with safe SQL', () => {
|
|
expect(checkDestructiveQuery(`OPEN ref FOR EXECUTE 'SELECT * FROM users';`)).toBe(false)
|
|
})
|
|
|
|
it('should not flag RETURN QUERY EXECUTE with safe SQL', () => {
|
|
expect(checkDestructiveQuery(`RETURN QUERY EXECUTE 'SELECT * FROM users';`)).toBe(false)
|
|
})
|
|
})
|
|
|
|
describe('SQLEditor.utils:getCreateTablesMissingRLS', () => {
|
|
it('flags a basic CREATE TABLE without RLS', () => {
|
|
const result = getCreateTablesMissingRLS('create table foo (id int8 primary key);')
|
|
expect(result).toEqual([{ schema: undefined, tableName: 'foo' }])
|
|
})
|
|
|
|
it('flags CREATE TABLE IF NOT EXISTS', () => {
|
|
const result = getCreateTablesMissingRLS(
|
|
'create table if not exists foo (id int8 primary key);'
|
|
)
|
|
expect(result).toHaveLength(1)
|
|
expect(result[0].tableName).toBe('foo')
|
|
})
|
|
|
|
it('flags schema-qualified CREATE TABLE', () => {
|
|
const result = getCreateTablesMissingRLS('create table public.foo (id int8 primary key);')
|
|
expect(result).toEqual([{ schema: 'public', tableName: 'foo' }])
|
|
})
|
|
|
|
it('flags quoted identifiers', () => {
|
|
const result = getCreateTablesMissingRLS(
|
|
'create table "public"."user_table" (id int8 primary key);'
|
|
)
|
|
expect(result).toEqual([{ schema: 'public', tableName: 'user_table' }])
|
|
})
|
|
|
|
it('flags quoted identifiers containing spaces', () => {
|
|
const result = getCreateTablesMissingRLS(
|
|
'create table "public"."My Table" (id int8 primary key);'
|
|
)
|
|
expect(result).toEqual([{ schema: 'public', tableName: 'My Table' }])
|
|
})
|
|
|
|
it('matches RLS to a table whose name contains spaces', () => {
|
|
const sql = stripIndent`
|
|
create table "My Table" (id int8 primary key);
|
|
alter table "My Table" enable row level security;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not flag when ENABLE ROW LEVEL SECURITY is in the same SQL', () => {
|
|
const sql = stripIndent`
|
|
create table foo (id int8 primary key);
|
|
alter table foo enable row level security;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not flag when ENABLE RLS shorthand is in the same SQL', () => {
|
|
const sql = stripIndent`
|
|
create table foo (id int8 primary key);
|
|
alter table foo enable rls;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('matches RLS to the right table when multiple tables created', () => {
|
|
const sql = stripIndent`
|
|
create table foo (id int8 primary key);
|
|
create table bar (id int8 primary key);
|
|
alter table foo enable row level security;
|
|
`
|
|
const result = getCreateTablesMissingRLS(sql)
|
|
expect(result).toHaveLength(1)
|
|
expect(result[0].tableName).toBe('bar')
|
|
})
|
|
|
|
it('does not flag when CREATE TABLE is inside a comment', () => {
|
|
const sql = stripIndent`
|
|
-- create table foo (id int8 primary key);
|
|
select 1;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not flag when there is no CREATE TABLE at all', () => {
|
|
expect(getCreateTablesMissingRLS('select * from foo;')).toEqual([])
|
|
})
|
|
|
|
it('schema-qualified RLS matches schema-qualified CREATE', () => {
|
|
const sql = stripIndent`
|
|
create table public.foo (id int8 primary key);
|
|
alter table public.foo enable row level security;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not flag when ALTER TABLE IF EXISTS enables RLS', () => {
|
|
const sql = stripIndent`
|
|
CREATE TABLE IF NOT EXISTS public."Conversations" (id int8 primary key);
|
|
ALTER TABLE IF EXISTS public."Conversations" ENABLE ROW LEVEL SECURITY;
|
|
GRANT ALL ON TABLE public."Conversations" TO postgres, anon, authenticated, service_role;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('flags CREATE TEMP TABLE', () => {
|
|
const result = getCreateTablesMissingRLS('create temp table foo (id int8 primary key);')
|
|
expect(result).toHaveLength(1)
|
|
expect(result[0].tableName).toBe('foo')
|
|
})
|
|
|
|
it('does not flag `select ... into var` inside a plpgsql function body', () => {
|
|
// Regression: the SELECT..INTO detector used to match variable assignments
|
|
// inside $$...$$ function bodies and surface them as \"new tables\".
|
|
const sql = stripIndent`
|
|
create or replace function schema_checks()
|
|
returns jsonb
|
|
language plpgsql
|
|
as $$
|
|
declare
|
|
ret jsonb;
|
|
begin
|
|
select jsonb_build_object('value', 'ok')
|
|
into ret;
|
|
return ret;
|
|
end;
|
|
$$;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not flag `select ... into var` inside a DO block', () => {
|
|
const sql = stripIndent`
|
|
do $$
|
|
declare
|
|
result int;
|
|
begin
|
|
select count(*) into result from information_schema.tables;
|
|
end
|
|
$$;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not flag CREATE TABLE text that appears inside a function body', () => {
|
|
const sql = stripIndent`
|
|
create or replace function noop()
|
|
returns void
|
|
language plpgsql
|
|
as $$
|
|
begin
|
|
-- create table foo (id int);
|
|
perform 1;
|
|
end;
|
|
$$;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('flags top-level CREATE TABLE alongside a function with INTO assignments', () => {
|
|
const sql = stripIndent`
|
|
create table public.foo (id int8 primary key);
|
|
create or replace function bar()
|
|
returns int
|
|
language plpgsql
|
|
as $$
|
|
declare
|
|
v int;
|
|
begin
|
|
select 1 into v;
|
|
return v;
|
|
end;
|
|
$$;
|
|
`
|
|
const result = getCreateTablesMissingRLS(sql)
|
|
expect(result).toEqual([{ schema: 'public', tableName: 'foo' }])
|
|
})
|
|
|
|
it('does not flag CREATE TABLE inside nested dollar-quoted dynamic SQL', () => {
|
|
// Regression: the `$sql$...$sql$` block inside the outer `$fn$...$fn$`
|
|
// body was previously pairing with the outer tag, letting the inner
|
|
// semicolon split the statement and exposing `create table fake` to the
|
|
// RLS warning.
|
|
const sql = stripIndent`
|
|
create function f()
|
|
returns void
|
|
language plpgsql
|
|
as $fn$
|
|
begin
|
|
execute $sql$create table fake(id int);$sql$;
|
|
end;
|
|
$fn$;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('handles custom dollar-quote tags (e.g. $body$...$body$)', () => {
|
|
const sql = stripIndent`
|
|
create or replace function f()
|
|
returns int
|
|
language plpgsql
|
|
as $body$
|
|
declare
|
|
v int;
|
|
begin
|
|
select 1 into v;
|
|
return v;
|
|
end;
|
|
$body$;
|
|
`
|
|
expect(getCreateTablesMissingRLS(sql)).toEqual([])
|
|
})
|
|
|
|
it('does not collide quoted identifiers that differ only by case', () => {
|
|
// "MyTable" and "mytable" are distinct tables in Postgres, so the ALTER
|
|
// here targets a different table than the CREATE — the warning must fire.
|
|
const sql = stripIndent`
|
|
create table "MyTable" (id int8 primary key);
|
|
alter table "mytable" enable row level security;
|
|
`
|
|
const result = getCreateTablesMissingRLS(sql)
|
|
expect(result).toHaveLength(1)
|
|
expect(result[0].tableName).toBe('MyTable')
|
|
})
|
|
})
|
|
|
|
describe('SQLEditor.utils:appendEnableRLSStatements', () => {
|
|
it('appends a single ALTER TABLE ENABLE RLS statement', () => {
|
|
const result = appendEnableRLSStatements('create table foo (id int8 primary key);', [
|
|
{ tableName: 'foo' },
|
|
])
|
|
expect(result).toContain('ALTER TABLE foo ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
|
|
it('appends one ALTER per table', () => {
|
|
const result = appendEnableRLSStatements(
|
|
'create table foo (id int8); create table bar (id int8);',
|
|
[{ tableName: 'foo' }, { tableName: 'bar' }]
|
|
)
|
|
expect(result).toContain('ALTER TABLE foo ENABLE ROW LEVEL SECURITY;')
|
|
expect(result).toContain('ALTER TABLE bar ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
|
|
it('schema-qualifies the table when schema is provided', () => {
|
|
const result = appendEnableRLSStatements('create table public.foo (id int8);', [
|
|
{ schema: 'public', tableName: 'foo' },
|
|
])
|
|
expect(result).toContain('ALTER TABLE public.foo ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
|
|
it('quotes identifiers that are not simple', () => {
|
|
const result = appendEnableRLSStatements('create table "My Table" (id int8);', [
|
|
{ tableName: 'My Table' },
|
|
])
|
|
expect(result).toContain('ALTER TABLE "My Table" ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
|
|
it('quotes mixed-case identifiers so Postgres does not fold them to lowercase', () => {
|
|
const result = appendEnableRLSStatements('create table "MyTable" (id int8);', [
|
|
{ tableName: 'MyTable' },
|
|
])
|
|
expect(result).toContain('ALTER TABLE "MyTable" ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
|
|
it('quotes mixed-case schema and table identifiers', () => {
|
|
const result = appendEnableRLSStatements('create table "MySchema"."MyTable" (id int8);', [
|
|
{ schema: 'MySchema', tableName: 'MyTable' },
|
|
])
|
|
expect(result).toContain('ALTER TABLE "MySchema"."MyTable" ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
|
|
it('returns the original SQL unchanged when there are no tables', () => {
|
|
const sql = 'select 1;'
|
|
expect(appendEnableRLSStatements(sql, [])).toBe(sql)
|
|
})
|
|
|
|
it('puts the terminator on its own line when SQL ends with a line comment', () => {
|
|
// Without this, the appended ';' would be swallowed by the line comment and
|
|
// the following ALTER TABLE would be parsed as part of the CREATE TABLE.
|
|
const sql = stripIndent`
|
|
create table foo (id int)
|
|
-- forgot the semicolon
|
|
`
|
|
const result = appendEnableRLSStatements(sql, [{ tableName: 'foo' }])
|
|
expect(result).toMatch(/-- forgot the semicolon\n;\n/)
|
|
expect(result).toContain('ALTER TABLE foo ENABLE ROW LEVEL SECURITY;')
|
|
})
|
|
})
|
|
|
|
describe('SQLEditor.utils:hasActiveEnsureRLSTrigger', () => {
|
|
it('returns false for undefined triggers', () => {
|
|
expect(hasActiveEnsureRLSTrigger(undefined)).toBe(false)
|
|
})
|
|
|
|
it('returns false for an empty list', () => {
|
|
expect(hasActiveEnsureRLSTrigger([])).toBe(false)
|
|
})
|
|
|
|
it('returns true when a trigger named "ensure_rls" is active', () => {
|
|
expect(hasActiveEnsureRLSTrigger([buildTrigger()])).toBe(true)
|
|
})
|
|
|
|
it('returns true when a trigger uses the rls_auto_enable function (renamed trigger)', () => {
|
|
expect(
|
|
hasActiveEnsureRLSTrigger([
|
|
buildTrigger({ name: 'something_else', function_name: 'rls_auto_enable' }),
|
|
])
|
|
).toBe(true)
|
|
})
|
|
|
|
it('returns false when the matching trigger is DISABLED', () => {
|
|
expect(hasActiveEnsureRLSTrigger([buildTrigger({ enabled_mode: 'DISABLED' })])).toBe(false)
|
|
})
|
|
|
|
it('ignores unrelated triggers', () => {
|
|
expect(
|
|
hasActiveEnsureRLSTrigger([buildTrigger({ name: 'audit_log', function_name: 'log_changes' })])
|
|
).toBe(false)
|
|
})
|
|
})
|
|
|
|
describe('SQLEditor.utils:filterTablesCoveredByEnsureRLSTrigger', () => {
|
|
it('returns the input unchanged when the trigger is not present', () => {
|
|
const tables = [{ tableName: 'foo' }, { schema: 'private', tableName: 'bar' }]
|
|
expect(filterTablesCoveredByEnsureRLSTrigger(tables, false)).toEqual(tables)
|
|
})
|
|
|
|
it('drops public-schema tables when the trigger is present', () => {
|
|
const tables = [
|
|
{ schema: 'public', tableName: 'foo' },
|
|
{ tableName: 'bar' }, // no schema → defaults to public
|
|
]
|
|
expect(filterTablesCoveredByEnsureRLSTrigger(tables, true)).toEqual([])
|
|
})
|
|
|
|
it('keeps tables in non-public schemas when the trigger is present', () => {
|
|
const tables = [
|
|
{ schema: 'public', tableName: 'foo' },
|
|
{ schema: 'private', tableName: 'bar' },
|
|
{ schema: 'app', tableName: 'baz' },
|
|
]
|
|
expect(filterTablesCoveredByEnsureRLSTrigger(tables, true)).toEqual([
|
|
{ schema: 'private', tableName: 'bar' },
|
|
{ schema: 'app', tableName: 'baz' },
|
|
])
|
|
})
|
|
|
|
it('matches the public schema case-insensitively', () => {
|
|
const tables = [{ schema: 'PUBLIC', tableName: 'foo' }]
|
|
expect(filterTablesCoveredByEnsureRLSTrigger(tables, true)).toEqual([])
|
|
})
|
|
})
|
|
|
|
describe('SQLEditor.utils:checkAlterDatabaseConnection', () => {
|
|
it('detects connection limit 0', () => {
|
|
const match = checkAlterDatabaseConnection('alter database postgres connection limit 0;')
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('detects allow_connections false', () => {
|
|
const match = checkAlterDatabaseConnection('alter database postgres allow_connections false;')
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('detects case-insensitive match', () => {
|
|
const match = checkAlterDatabaseConnection('ALTER DATABASE postgres CONNECTION LIMIT 0;')
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('detects statement among multiple statements', () => {
|
|
const match = checkAlterDatabaseConnection(stripIndent`
|
|
select * from countries;
|
|
alter database postgres connection limit 0;
|
|
`)
|
|
expect(match).toBe(true)
|
|
})
|
|
|
|
it('does not flag unrelated alter database statement', () => {
|
|
const match = checkAlterDatabaseConnection(
|
|
'alter database postgres set statement_timeout = 60000;'
|
|
)
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('does not flag non-alter statements', () => {
|
|
const match = checkAlterDatabaseConnection('select * from countries;')
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('ignores statements inside comments', () => {
|
|
const match = checkAlterDatabaseConnection(stripIndent`
|
|
-- alter database postgres connection limit 0;
|
|
select 1;
|
|
`)
|
|
expect(match).toBe(false)
|
|
})
|
|
|
|
it('detects both dangerous statements in same query', () => {
|
|
const match = checkAlterDatabaseConnection(stripIndent`
|
|
alter database postgres connection limit 0;
|
|
alter database postgres allow_connections false;
|
|
`)
|
|
expect(match).toBe(true)
|
|
})
|
|
})
|