diff --git a/README.md b/README.md index 5b50808e..116750a6 100644 --- a/README.md +++ b/README.md @@ -275,3 +275,41 @@ Example data: ``` For convenience, data prefixed with `0x` or containing an odd number of characters is supported although it's not part of the standard. + +## String enum support + +This target can store columns with the json-schema "enum" type as a pg DOMAIN type instead of a TEXT column. This can save space and improve performance. + +It is an opt-in feature because it might result in data loss if the actual data does not match the schema's advertised encoding. + +Please consider these several downsides to take into consideration before activating this feature: +- it changes the sort behavior of the resulting column +- string operations will not be available +- portability of the data is reduced +- it is not possible to add remove or modify the enum values +- enums are not shared accross tables, each column get his own custom type + +To enable it, set the `storage_optimized_enum` option to `True`. + +Example schema: +```json +{ + "type": "object", + "properties": { + "my_enum": { + "type": "string", + "enum": ["foo", "bar", "baz"] + } + } +} +``` + +Data will be stored as a custom domain type in the database. The domain name will be `enum__`. The domain will be created in the same schema as the table. + +Example generated SQL: +```sql +CREATE TYPE enum_123456_my_enum AS ENUM ('foo', 'bar', 'baz'); +CREATE TABLE my_table ( + my_enum enum_123456_my_enum +); +``` diff --git a/target_postgres/connector.py b/target_postgres/connector.py index c9d250f6..b9ec1639 100644 --- a/target_postgres/connector.py +++ b/target_postgres/connector.py @@ -3,6 +3,7 @@ from __future__ import annotations import atexit +import hashlib import io import signal import typing as t @@ -16,7 +17,7 @@ import sqlalchemy as sa from singer_sdk import SQLConnector from singer_sdk import typing as th -from sqlalchemy.dialects.postgresql import ARRAY, BIGINT, BYTEA, JSONB +from sqlalchemy.dialects.postgresql import ARRAY, BIGINT, BYTEA, ENUM, JSONB from sqlalchemy.engine import URL from sqlalchemy.engine.url import make_url from sqlalchemy.types import ( @@ -92,6 +93,29 @@ def interpret_content_encoding(self) -> bool: """ return self.config.get("interpret_content_encoding", False) + @cached_property + def storage_optimized_enum(self) -> bool: + """Whether to use storage optimized enum. + + Stores columns with the json-schema "enum" type as a pg DOMAIN type + instead of a TEXT column. This can save space and improve performance. + + It is an opt-in feature because it might result in data loss if the + actual data does not match the schema's advertised encoding. + + Please consider these several downsides to take into consideration + before activating: + - it changes the sort behavior of the resulting column + - string operations will not be available + - portability of the data is reduced + - it is not possible to add remove or modify the enum values + - enums are not shared accross tables, each column get his own custom type + + Returns: + True if the feature is enabled, False otherwise. + """ + return self.config.get("storage_optimized_enum", False) + def prepare_table( # type: ignore[override] self, full_table_name: str, @@ -146,7 +170,12 @@ def prepare_table( # type: ignore[override] self.prepare_column( full_table_name=table.fullname, column_name=property_name, - sql_type=self.to_sql_type(property_def), + sql_type=self.to_sql_type( + schema_name=schema_name, + table_name=table_name, + property_name=property_name, + jsonschema_type=property_def, + ), connection=connection, column_object=column_object, ) @@ -171,22 +200,38 @@ def copy_table_structure( Returns: The new table object. """ - _, schema_name, table_name = self.parse_full_table_name(full_table_name) - meta = sa.MetaData(schema=schema_name) - new_table: sa.Table - columns = [] + _, to_table_schema_name, to_table_name = self.parse_full_table_name( + full_table_name + ) if self.table_exists(full_table_name=full_table_name): raise RuntimeError("Table already exists") - for column in from_table.columns: - columns.append(column._copy()) - if as_temp_table: - new_table = sa.Table(table_name, meta, *columns, prefixes=["TEMPORARY"]) - new_table.create(bind=connection) - return new_table - else: - new_table = sa.Table(table_name, meta, *columns) - new_table.create(bind=connection) - return new_table + + from_table_meta = sa.MetaData(schema=from_table.schema) + to_table_meta = sa.MetaData(schema=to_table_schema_name) + columns = [ + # Make sure this temporary table do not issue a CREATE TYPE + # or DROP TYPE statement for custom enums + sa.Column( + c.name, + ENUM( + *c.type.enums, + name=c.type.name, + metadata=from_table_meta, + create_type=False, + ), + ) + if isinstance(c.type, ENUM) and hasattr(c.type, "enums") + else c.copy() + for c in from_table.columns + ] + new_table = sa.Table( + to_table_name, + to_table_meta, + *columns, + prefixes=["TEMPORARY"] if as_temp_table else [], + ) + new_table.create(bind=connection, checkfirst=True) + return new_table @contextmanager def _connect(self) -> t.Iterator[sa.engine.Connection]: @@ -195,7 +240,7 @@ def _connect(self) -> t.Iterator[sa.engine.Connection]: def drop_table(self, table: sa.Table, connection: sa.engine.Connection): """Drop table data.""" - table.drop(bind=connection) + table.drop(bind=connection, checkfirst=True) def clone_table( self, new_table_name, table, metadata, connection, temp_table @@ -218,7 +263,13 @@ def clone_table( new_table.create(bind=connection) return new_table - def to_sql_type(self, jsonschema_type: dict) -> sa.types.TypeEngine: # type: ignore[override] + def to_sql_type( # type: ignore[override] + self, + schema_name: str | None, + table_name: str, + property_name: str, + jsonschema_type: dict, + ) -> sa.types.TypeEngine: """Return a JSON Schema representation of the provided type. By default will call `typing.to_sql_type()`. @@ -229,6 +280,9 @@ def to_sql_type(self, jsonschema_type: dict) -> sa.types.TypeEngine: # type: ig from the base class for all unhandled cases. Args: + schema_name: The name of the target table schema. + table_name: The name of the target table. + property_name: The name of the property. jsonschema_type: The JSON Schema representation of the source type. Returns: @@ -242,10 +296,12 @@ def to_sql_type(self, jsonschema_type: dict) -> sa.types.TypeEngine: # type: ig elif isinstance(jsonschema_type["type"], list): for entry in jsonschema_type["type"]: json_type_dict = {"type": entry} - if jsonschema_type.get("format", False): - json_type_dict["format"] = jsonschema_type["format"] + if fmt := jsonschema_type.get("format", False): + json_type_dict["format"] = fmt if encoding := jsonschema_type.get("contentEncoding", False): json_type_dict["contentEncoding"] = encoding + if enum := jsonschema_type.get("enum", False): + json_type_dict["enum"] = enum json_type_array.append(json_type_dict) else: msg = "Invalid format for jsonschema type: not str or list." @@ -260,16 +316,30 @@ def to_sql_type(self, jsonschema_type: dict) -> sa.types.TypeEngine: # type: ig return NOTYPE() sql_type_array = [] for json_type in json_type_array: - picked_type = self.pick_individual_type(jsonschema_type=json_type) + picked_type = self.pick_individual_type( + schema_name=schema_name, + table_name=table_name, + property_name=property_name, + jsonschema_type=json_type, + ) if picked_type is not None: sql_type_array.append(picked_type) return PostgresConnector.pick_best_sql_type(sql_type_array=sql_type_array) - def pick_individual_type(self, jsonschema_type: dict): + def pick_individual_type( + self, + schema_name: str | None, + table_name: str, + property_name: str, + jsonschema_type: dict, + ): """Select the correct sql type assuming jsonschema_type has only a single type. Args: + schema_name: The name of the target table schema. + table_name: The name of the target table. + property_name: The name of the property. jsonschema_type: A jsonschema_type array containing only a single type. Returns: @@ -292,6 +362,20 @@ def pick_individual_type(self, jsonschema_type: dict): and jsonschema_type.get("contentEncoding") == "base16" ): return HexByteString() + if self.storage_optimized_enum and jsonschema_type.get("enum"): + # make sure that the enum name is unique and that the uniqueness part + # can be determined using the first 71 characters of the enum name + # this is a limitation of postgres type names + hasher = hashlib.md5(usedforsecurity=False) + hasher.update(f"{schema_name}__{table_name}__{property_name}".encode()) + hash_str = hasher.hexdigest() + name_hash = hash_str[0:15] + unique_enum_name = f"enum_{name_hash}_{property_name}" + + # ensure our enum is created in the correct schema + meta = sa.MetaData(schema=schema_name) + return ENUM(*jsonschema_type["enum"], name=unique_enum_name, metadata=meta) + individual_type = th.to_sql_type(jsonschema_type) if isinstance(individual_type, VARCHAR): return TEXT() @@ -308,6 +392,7 @@ def pick_best_sql_type(sql_type_array: list): An instance of the best SQL type class based on defined precedence order. """ precedence_order = [ + ENUM, HexByteString, ARRAY, JSONB, @@ -372,18 +457,21 @@ def create_empty_table( # type: ignore[override] columns.append( sa.Column( property_name, - self.to_sql_type(property_jsonschema), + self.to_sql_type( + schema_name=meta.schema, + table_name=table_name, + property_name=property_name, + jsonschema_type=property_jsonschema, + ), primary_key=is_primary_key, autoincrement=False, # See: https://github.com/MeltanoLabs/target-postgres/issues/193 # noqa: E501 ) ) - if as_temp_table: - new_table = sa.Table(table_name, meta, *columns, prefixes=["TEMPORARY"]) - new_table.create(bind=connection) - return new_table - new_table = sa.Table(table_name, meta, *columns) - new_table.create(bind=connection) + new_table = sa.Table( + table_name, meta, *columns, prefixes=["TEMPORARY"] if as_temp_table else [] + ) + new_table.create(bind=connection, checkfirst=True) return new_table def prepare_column( @@ -536,7 +624,13 @@ def _adapt_column_type( # type: ignore[override] current_type_collation = self.remove_collation(current_type) # Check if the existing column type and the sql type are the same - if str(sql_type) == str(current_type): + cmp_a = str(sql_type) + cmp_b = str(current_type) + if isinstance(sql_type, ENUM) and hasattr(sql_type, "enums"): + cmp_a = "ENUM " + str(sql_type.enums) + if isinstance(current_type, ENUM) and hasattr(current_type, "enums"): + cmp_b = "ENUM " + str(current_type.enums) + if cmp_a == cmp_b: # The current column and sql type are the same # Nothing to do return @@ -544,7 +638,6 @@ def _adapt_column_type( # type: ignore[override] # Not the same type, generic type or compatible types # calling merge_sql_types for assistnace compatible_sql_type = self.merge_sql_types([current_type, sql_type]) - if str(compatible_sql_type) == str(current_type): # Nothing to do return @@ -794,7 +887,6 @@ def get_table_columns( # type: ignore[override] """ inspector = sa.inspect(connection) columns = inspector.get_columns(table_name, schema_name) - return { col_meta["name"]: sa.Column( col_meta["name"], diff --git a/target_postgres/sinks.py b/target_postgres/sinks.py index ea8b8df3..57f500ef 100644 --- a/target_postgres/sinks.py +++ b/target_postgres/sinks.py @@ -226,7 +226,6 @@ def upsert( insert_stmt = sa.insert(to_table).from_select( names=from_table.columns, select=select_stmt ) - connection.execute(insert_stmt) # Update @@ -254,7 +253,12 @@ def column_representation( columns.append( sa.Column( property_name, - self.connector.to_sql_type(property_jsonschema), + self.connector.to_sql_type( + schema_name=self.schema_name, + table_name=self.table_name, + property_name=property_name, + jsonschema_type=property_jsonschema, + ), ) ) return columns diff --git a/target_postgres/target.py b/target_postgres/target.py index 8587102d..f497c941 100644 --- a/target_postgres/target.py +++ b/target_postgres/target.py @@ -200,6 +200,23 @@ def __init__( "in an error if the data is not encoded as expected." ), ), + th.Property( + "storage_optimized_enum", + th.BooleanType, + default=False, + description=( + "If set to true, the target will store enum values as a custom pg type " + "instead of a text column. This can save space and improve performance," + " but may make the data harder to query and analyze." + "Please consider these several downsides to take into consideration " + "before activating this feature:" + " it changes the sort behavior of the resulting column," + " string operations will not be available," + " portability of the data is reduced," + " it is not possible to add remove or modify the enum values," + " enums are not shared accross tables." + ), + ), th.Property( "ssl_enable", th.BooleanType, diff --git a/target_postgres/tests/data_files/storage_optimized_enum_off.singer b/target_postgres/tests/data_files/storage_optimized_enum_off.singer new file mode 100644 index 00000000..06a0ccf9 --- /dev/null +++ b/target_postgres/tests/data_files/storage_optimized_enum_off.singer @@ -0,0 +1,8 @@ +{"type":"SCHEMA","stream":"test_storage_optimized_enum_off","schema":{"type":"object","properties":{"id":{"type":"string"},"chain":{"type":"string","enum":["avalanche","ethereum","optimism","polygon"]},"chain_edge_cases":{"type":"string","enum":["AvAlAnChE"," EtH ","a\u005Cb","PŒLŸflôÑ","_éèàÀÇÉ","_-.*$@#%","ù€/\\\n\r\t","'"]}},"required":["id","chain","chain_edge_cases"]},"key_properties":["id"]} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_1","chain":"avalanche","chain_edge_cases":"AvAlAnChE"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_2","chain":"ethereum","chain_edge_cases":" EtH "},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_3","chain":"optimism","chain_edge_cases":"a\\b"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_4","chain":"polygon","chain_edge_cases":"PŒLŸflôÑ"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_5","chain":"avalanche","chain_edge_cases":"_éèàÀÇÉ"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_7","chain":"optimism","chain_edge_cases":"ù€/\\\n\r\t"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_valid_8","chain":"polygon","chain_edge_cases":"'"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} diff --git a/target_postgres/tests/data_files/storage_optimized_enum_off_schema_update.singer b/target_postgres/tests/data_files/storage_optimized_enum_off_schema_update.singer new file mode 100644 index 00000000..d4bd0b7c --- /dev/null +++ b/target_postgres/tests/data_files/storage_optimized_enum_off_schema_update.singer @@ -0,0 +1,4 @@ +{"type":"SCHEMA","stream":"test_storage_optimized_enum_off","schema":{"type":"object","properties":{"id":{"type":"string"},"chain":{"type":"string","enum":["avalanche","base","ethereum","optimism","polygon"]},"chain_edge_cases":{"type":"string","enum":["AvAlAnChE","New Base Chain", " EtH ","a\u005Cb","PŒLŸflôÑ","_éèàÀÇÉ","_-.*$@#%","ù€/\\\n\r\t","'"]}},"required":["id","chain","chain_edge_cases"]},"key_properties":["id"]} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_not_in_original_schema","chain":"base","chain_edge_cases":"New Base Chain"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"SCHEMA","stream":"test_storage_optimized_enum_off","schema":{"type":"object","properties":{"id":{"type":"string"},"chain":{"type":"string","enum":["avalanche"]},"chain_edge_cases":{"type":"string","enum":["AvAlAnChE"]}},"required":["id","chain","chain_edge_cases"]},"key_properties":["id"]} +{"type":"RECORD","stream":"test_storage_optimized_enum_off","record":{"id":"test_still_in_current_schema","chain":"avalanche","chain_edge_cases":"AvAlAnChE"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} diff --git a/target_postgres/tests/data_files/storage_optimized_enum_on.singer b/target_postgres/tests/data_files/storage_optimized_enum_on.singer new file mode 100644 index 00000000..6a654480 --- /dev/null +++ b/target_postgres/tests/data_files/storage_optimized_enum_on.singer @@ -0,0 +1,8 @@ +{"type":"SCHEMA","stream":"test_storage_optimized_enum_on","schema":{"type":"object","properties":{"id":{"type":"string"},"chain":{"type":"string","enum":["avalanche","ethereum","optimism","polygon"]},"chain_edge_cases":{"type":"string","enum":["AvAlAnChE"," EtH ","a\u005Cb","PŒLŸflôÑ","_éèàÀÇÉ","_-.*$@#%","ù€/\\\n\r\t","'"]}},"required":["id","chain","chain_edge_cases"]},"key_properties":["id"]} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_1","chain":"avalanche","chain_edge_cases":"AvAlAnChE"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_2","chain":"ethereum","chain_edge_cases":" EtH "},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_3","chain":"optimism","chain_edge_cases":"a\\b"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_4","chain":"polygon","chain_edge_cases":"PŒLŸflôÑ"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_5","chain":"avalanche","chain_edge_cases":"_éèàÀÇÉ"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_7","chain":"optimism","chain_edge_cases":"ù€/\\\n\r\t"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_valid_8","chain":"polygon","chain_edge_cases":"'"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} diff --git a/target_postgres/tests/data_files/storage_optimized_enum_on_schema_update.singer b/target_postgres/tests/data_files/storage_optimized_enum_on_schema_update.singer new file mode 100644 index 00000000..14a0454d --- /dev/null +++ b/target_postgres/tests/data_files/storage_optimized_enum_on_schema_update.singer @@ -0,0 +1,4 @@ +{"type":"SCHEMA","stream":"test_storage_optimized_enum_on","schema":{"type":"object","properties":{"id":{"type":"string"},"chain":{"type":"string","enum":["avalanche","base","ethereum","optimism","polygon"]},"chain_edge_cases":{"type":"string","enum":["AvAlAnChE","New Base Chain", " EtH ","a\u005Cb","PŒLŸflôÑ","_éèàÀÇÉ","_-.*$@#%","ù€/\\\n\r\t","'"]}},"required":["id","chain","chain_edge_cases"]},"key_properties":["id"]} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_not_in_original_schema","chain":"base","chain_edge_cases":"New Base Chain"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} +{"type":"SCHEMA","stream":"test_storage_optimized_enum_on","schema":{"type":"object","properties":{"id":{"type":"string"},"chain":{"type":"string","enum":["avalanche"]},"chain_edge_cases":{"type":"string","enum":["AvAlAnChE"]}},"required":["id","chain","chain_edge_cases"]},"key_properties":["id"]} +{"type":"RECORD","stream":"test_storage_optimized_enum_on","record":{"id":"test_still_in_current_schema","chain":"avalanche","chain_edge_cases":"AvAlAnChE"},"time_extracted":"2023-09-15T19:33:01.841018+00:00"} diff --git a/target_postgres/tests/test_target_postgres.py b/target_postgres/tests/test_target_postgres.py index 9a0ba6d5..c64ab0e6 100644 --- a/target_postgres/tests/test_target_postgres.py +++ b/target_postgres/tests/test_target_postgres.py @@ -133,6 +133,9 @@ def verify_data( if isinstance(row[col], memoryview): row[col] = bytes(row[col]) + # sort by primary key to compare with the actual db data + check_data.sort(key=lambda x: x[primary_key]) + assert result_dict == check_data else: raise ValueError("Invalid check_data - not dict or list of dicts") @@ -580,6 +583,66 @@ def test_base16_content_encoding_interpreted(postgres_config_no_ssl): verify_data(target, "test_base_16_encoding_interpreted", 7, "id", rows) +def test_data_is_always_inserted_when_storage_optimized_enum_is_off( + postgres_config_no_ssl, +): + """Make sure we can insert enum data into the database when storage_optimized_enum is off""" + postgres_config_modified = copy.deepcopy(postgres_config_no_ssl) + postgres_config_modified["storage_optimized_enum"] = False + target = TargetPostgres(config=postgres_config_modified) + target.target_connector.allow_column_alter = True + + # we can insert valid enum data + singer_file_to_target("storage_optimized_enum_off.singer", target) + rows = [ + {"id": "test_valid_1", "chain": "avalanche", "chain_edge_cases": "AvAlAnChE"}, + {"id": "test_valid_2", "chain": "ethereum", "chain_edge_cases": " EtH "}, + {"id": "test_valid_3", "chain": "optimism", "chain_edge_cases": "a\\b"}, + {"id": "test_valid_4", "chain": "polygon", "chain_edge_cases": "PŒLŸflôÑ"}, + {"id": "test_valid_5", "chain": "avalanche", "chain_edge_cases": "_éèàÀÇÉ"}, + {"id": "test_valid_7", "chain": "optimism", "chain_edge_cases": "ù€/\\\n\r\t"}, + {"id": "test_valid_8", "chain": "polygon", "chain_edge_cases": "'"}, + ] + verify_data(target, "test_storage_optimized_enum_off", 7, "id", rows) + + # we allow shema updates when storage_optimized_enum is off + singer_file_to_target("storage_optimized_enum_off_schema_update.singer", target) + rows = rows + [ + { + "id": "test_not_in_original_schema", + "chain": "base", + "chain_edge_cases": "New Base Chain", + }, + { + "id": "test_still_in_current_schema", + "chain": "avalanche", + "chain_edge_cases": "AvAlAnChE", + }, + ] + verify_data(target, "test_storage_optimized_enum_off", 9, "id", rows) + + +def test_storage_optimized_enum_on(postgres_config_no_ssl): + """Make sure we can insert valid enum data into the database when storage_optimized_enum is on""" + postgres_config_modified = copy.deepcopy(postgres_config_no_ssl) + postgres_config_modified["storage_optimized_enum"] = True + target = TargetPostgres(config=postgres_config_modified) + target.target_connector.allow_column_alter = False + + singer_file_to_target("storage_optimized_enum_on.singer", target) + + rows = [ + {"id": "test_valid_1", "chain": "avalanche", "chain_edge_cases": "AvAlAnChE"}, + {"id": "test_valid_2", "chain": "ethereum", "chain_edge_cases": " EtH "}, + {"id": "test_valid_3", "chain": "optimism", "chain_edge_cases": "a\\b"}, + {"id": "test_valid_4", "chain": "polygon", "chain_edge_cases": "PŒLŸflôÑ"}, + {"id": "test_valid_5", "chain": "avalanche", "chain_edge_cases": "_éèàÀÇÉ"}, + {"id": "test_valid_7", "chain": "optimism", "chain_edge_cases": "ù€/\\\n\r\t"}, + {"id": "test_valid_8", "chain": "polygon", "chain_edge_cases": "'"}, + ] + verify_data(target, "test_storage_optimized_enum_on", 7, "id", rows) + + def test_activate_version_hard_delete(postgres_config_no_ssl): """Activate Version Hard Delete Test""" table_name = "test_activate_version_hard"