Skip to content

Commit

Permalink
Support create directory table with location.
Browse files Browse the repository at this point in the history
When we create directory table with location, the file directory path will concat
with location at beginning.

Authored-by: Zhang Wenchao [email protected]
  • Loading branch information
wenchaozhang-123 committed Jan 2, 2025
1 parent e431071 commit c6bc4a3
Show file tree
Hide file tree
Showing 10 changed files with 86 additions and 16 deletions.
47 changes: 35 additions & 12 deletions src/backend/commands/dirtablecmds.c
Original file line number Diff line number Diff line change
Expand Up @@ -122,22 +122,45 @@ CreateDirectoryTable(CreateDirectoryTableStmt *stmt, Oid relId)
bool nulls[Natts_pg_directory_table];
HeapTuple tuple;
char *dirTablePath;
Form_pg_class pg_class_tuple;
HeapTuple class_tuple;
Oid spcId = chooseTableSpace(stmt);
RelFileNode relFileNode = {0};

class_tuple = SearchSysCache1(RELOID, ObjectIdGetDatum(relId));
if (!HeapTupleIsValid(class_tuple))
elog(ERROR, "cache lookup failed for relation %u", relId);
pg_class_tuple = (Form_pg_class) GETSTRUCT(class_tuple);
if (stmt->location)
{
if (spcId == InvalidOid ||
spcId == DEFAULTTABLESPACE_OID)
dirTablePath = psprintf("base/%s", stmt->location);
else if (spcId == GLOBALTABLESPACE_OID)
dirTablePath = psprintf("global/%s", stmt->location);
else
dirTablePath = psprintf("pg_tblspc/%s", stmt->location);
}
else
{
Form_pg_class pg_class_tuple = NULL;
HeapTuple class_tuple = NULL;
RelFileNode relFileNode = {0};

class_tuple = SearchSysCache1(RELOID, ObjectIdGetDatum(relId));
if (!HeapTupleIsValid(class_tuple))
elog(ERROR, "cache lookup failed for relation %u", relId);
pg_class_tuple = (Form_pg_class) GETSTRUCT(class_tuple);

relFileNode.spcNode = spcId;
relFileNode.dbNode = MyDatabaseId;
relFileNode.relNode = pg_class_tuple->relfilenode;
dirTablePath = UFileFormatPathName(&relFileNode);
relFileNode.spcNode = spcId;
relFileNode.dbNode = MyDatabaseId;
relFileNode.relNode = pg_class_tuple->relfilenode;

ReleaseSysCache(class_tuple);
dirTablePath = UFileFormatPathName(&relFileNode);
ReleaseSysCache(class_tuple);
}

/*
* We will check whether the directory path has existed only when use the local file am.
*/
if (UFileExists(spcId, dirTablePath) && GetTablespaceFileHandler(spcId) == &localFileAm)
ereport(ERROR,
(errcode(ERRCODE_DUPLICATE_OBJECT),
errmsg("directory table path \"%s\" already exists",
dirTablePath)));

/*
* Acquire DirectoryTableLock to ensure that no DROP DIRECTORY TABLE
Expand Down
1 change: 1 addition & 0 deletions src/backend/nodes/copyfuncs.c
Original file line number Diff line number Diff line change
Expand Up @@ -6290,6 +6290,7 @@ _copyCreateDirectoryTableStmt(const CreateDirectoryTableStmt *from)
CopyCreateStmtFields((const CreateStmt *) from, (CreateStmt *) newnode);

COPY_STRING_FIELD(tablespacename);
COPY_STRING_FIELD(location);

return newnode;
}
Expand Down
1 change: 1 addition & 0 deletions src/backend/nodes/equalfuncs.c
Original file line number Diff line number Diff line change
Expand Up @@ -3474,6 +3474,7 @@ _equalCreateDirectoryTableStmt(const CreateDirectoryTableStmt *a, const CreateDi
return false;

COMPARE_STRING_FIELD(tablespacename);
COMPARE_STRING_FIELD(location);

return true;
}
Expand Down
1 change: 1 addition & 0 deletions src/backend/nodes/outfuncs.c
Original file line number Diff line number Diff line change
Expand Up @@ -4094,6 +4094,7 @@ _outCreateDirectoryTableStmt(StringInfo str, const CreateDirectoryTableStmt *nod

_outCreateStmtInfo(str, (const CreateStmt *) node);
WRITE_STRING_FIELD(tablespacename);
WRITE_STRING_FIELD(location);
}

static void
Expand Down
1 change: 1 addition & 0 deletions src/backend/nodes/readfast.c
Original file line number Diff line number Diff line change
Expand Up @@ -1853,6 +1853,7 @@ _readCreateDirectoryTableStmt(void)
_readCreateStmt_common(&local_node->base);

READ_STRING_FIELD(tablespacename);
READ_STRING_FIELD(location);

READ_DONE();
}
Expand Down
16 changes: 12 additions & 4 deletions src/backend/parser/gram.y
Original file line number Diff line number Diff line change
Expand Up @@ -416,6 +416,7 @@ static void check_expressions_in_partition_key(PartitionSpec *spec, core_yyscan_
access_method_clause attr_name
table_access_method_clause name cursor_name file_name
opt_index_name cluster_index_specification opt_file_name
%type <str> OptWithLocation

%type <list> func_name handler_name qual_Op qual_all_Op subquery_Op
opt_class opt_inline_handler opt_validator validator_clause
Expand Down Expand Up @@ -6013,6 +6014,11 @@ OptWith:
| /*EMPTY*/ { $$ = NIL; }
;

OptWithLocation:
WITH LOCATION Sconst { $$ = $3; }
| /*EMPTY*/ { $$ = NULL; }
;

OnCommitOption: ON COMMIT DROP { $$ = ONCOMMIT_DROP; }
| ON COMMIT DELETE_P ROWS { $$ = ONCOMMIT_DELETE_ROWS; }
| ON COMMIT PRESERVE ROWS { $$ = ONCOMMIT_PRESERVE_ROWS; }
Expand Down Expand Up @@ -8530,7 +8536,7 @@ AlterStorageUserMappingStmt:

CreateDirectoryTableStmt:
CREATE DIRECTORY TABLE qualified_name
table_access_method_clause OptTableSpace OptTagOptList
table_access_method_clause OptTableSpace OptWithLocation OptTagOptList
{
CreateDirectoryTableStmt *n = makeNode(CreateDirectoryTableStmt);
$4->relpersistence = RELPERSISTENCE_PERMANENT;
Expand All @@ -8545,13 +8551,14 @@ CreateDirectoryTableStmt:
n->base.if_not_exists = false;
n->base.distributedBy = GetDirectoryTableDistributedBy();
n->base.relKind = RELKIND_DIRECTORY_TABLE;
n->base.tags = $7;
n->tablespacename = $6;
n->location = $7;
n->base.tags = $8;

$$ = (Node *) n;
}
| CREATE DIRECTORY TABLE IF_P NOT EXISTS qualified_name
table_access_method_clause OptTableSpace OptTagOptList
table_access_method_clause OptTableSpace OptWithLocation OptTagOptList
{
CreateDirectoryTableStmt *n = makeNode(CreateDirectoryTableStmt);
$7->relpersistence = RELPERSISTENCE_PERMANENT;
Expand All @@ -8566,8 +8573,9 @@ CreateDirectoryTableStmt:
n->base.if_not_exists = true;
n->base.distributedBy = GetDirectoryTableDistributedBy();
n->base.relKind = RELKIND_DIRECTORY_TABLE;
n->base.tags = $10;
n->tablespacename = $9;
n->location = $10;
n->base.tags = $11;

$$ = (Node *) n;
}
Expand Down
1 change: 1 addition & 0 deletions src/include/nodes/parsenodes.h
Original file line number Diff line number Diff line change
Expand Up @@ -3412,6 +3412,7 @@ typedef struct CreateDirectoryTableStmt
{
CreateStmt base;
char *tablespacename;
char *location; /* dtlocation for pg_directory_table */
} CreateDirectoryTableStmt;

typedef struct AlterDirectoryTableStmt
Expand Down
8 changes: 8 additions & 0 deletions src/test/regress/input/directory_table.source
Original file line number Diff line number Diff line change
Expand Up @@ -648,6 +648,14 @@ SELECT relative_path, tag FROM dir_table4 ORDER BY 1;
SAVEPOINT s2;
ROLLBACK;

-- test create directory table with location
CREATE DIRECTORY TABLE dir_table_with_location TABLESPACE directory_tblspc WITH LOCATION '/test_dirtable';
CREATE DIRECTORY TABLE dir_table_with_location2 TABLESPACE directory_tblspc WITH LOCATION '/test_dirtable'; -- error
CREATE DIRECTORY TABLE dir_table_with_location3 WITH LOCATION '/test_dirtable';
SELECT count(*) FROM pg_directory_table;
DROP DIRECTORY TABLE dir_table_with_location WITH CONTENT;
DROP DIRECTORY TABLE dir_table_with_location3 WITH CONTENT;

-- clean up
DROP DIRECTORY TABLE IF EXISTS dir_table1;
DROP DIRECTORY TABLE IF EXISTS dir_table2;
Expand Down
13 changes: 13 additions & 0 deletions src/test/regress/output/directory_table.source
Original file line number Diff line number Diff line change
Expand Up @@ -2063,6 +2063,19 @@ SELECT relative_path, tag FROM dir_table4 ORDER BY 1;

SAVEPOINT s2;
ROLLBACK;
-- test create directory table with location
CREATE DIRECTORY TABLE dir_table_with_location TABLESPACE directory_tblspc WITH LOCATION '/test_dirtable';
CREATE DIRECTORY TABLE dir_table_with_location2 TABLESPACE directory_tblspc WITH LOCATION '/test_dirtable'; -- error
ERROR: directory table path "pg_tblspc//test_dirtable" already exists
CREATE DIRECTORY TABLE dir_table_with_location3 WITH LOCATION '/test_dirtable';
SELECT count(*) FROM pg_directory_table;
count
-------
8
(1 row)

DROP DIRECTORY TABLE dir_table_with_location WITH CONTENT;
DROP DIRECTORY TABLE dir_table_with_location3 WITH CONTENT;
-- clean up
DROP DIRECTORY TABLE IF EXISTS dir_table1;
DROP DIRECTORY TABLE IF EXISTS dir_table2;
Expand Down
13 changes: 13 additions & 0 deletions src/test/regress/output/directory_table_optimizer.source
Original file line number Diff line number Diff line change
Expand Up @@ -2063,6 +2063,19 @@ SELECT relative_path, tag FROM dir_table4 ORDER BY 1;

SAVEPOINT s2;
ROLLBACK;
-- test create directory table with location
CREATE DIRECTORY TABLE dir_table_with_location TABLESPACE directory_tblspc WITH LOCATION '/test_dirtable';
CREATE DIRECTORY TABLE dir_table_with_location2 TABLESPACE directory_tblspc WITH LOCATION '/test_dirtable'; -- error
ERROR: directory table path "pg_tblspc//test_dirtable" already exists
CREATE DIRECTORY TABLE dir_table_with_location3 WITH LOCATION '/test_dirtable';
SELECT count(*) FROM pg_directory_table;
count
-------
8
(1 row)

DROP DIRECTORY TABLE dir_table_with_location WITH CONTENT;
DROP DIRECTORY TABLE dir_table_with_location3 WITH CONTENT;
-- clean up
DROP DIRECTORY TABLE IF EXISTS dir_table1;
DROP DIRECTORY TABLE IF EXISTS dir_table2;
Expand Down

0 comments on commit c6bc4a3

Please sign in to comment.