-
Notifications
You must be signed in to change notification settings - Fork 92
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat!: Implementation of batch ddl in dbapi #1092
base: main
Are you sure you want to change the base?
Changes from 2 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
|
@@ -16,6 +16,8 @@ | |||||
|
||||||
from enum import Enum | ||||||
from typing import TYPE_CHECKING, List | ||||||
|
||||||
from google.cloud.spanner_dbapi import parse_utils | ||||||
from google.cloud.spanner_dbapi.parsed_statement import ( | ||||||
ParsedStatement, | ||||||
StatementType, | ||||||
|
@@ -28,6 +30,52 @@ | |||||
|
||||||
if TYPE_CHECKING: | ||||||
from google.cloud.spanner_dbapi.cursor import Cursor | ||||||
from google.cloud.spanner_dbapi.connection import Connection | ||||||
|
||||||
|
||||||
class BatchDdlExecutor: | ||||||
"""Executor that is used when a DDL batch is started. These batches only | ||||||
accept DDL statements. All DDL statements are buffered locally and sent to | ||||||
Spanner when runBatch() is called. | ||||||
|
||||||
:type "Connection": :class:`~google.cloud.spanner_dbapi.connection.Connection` | ||||||
:param connection: | ||||||
""" | ||||||
|
||||||
def __init__(self, connection: "Connection"): | ||||||
self._connection = connection | ||||||
self._statements: List[str] = [] | ||||||
|
||||||
def execute_statement(self, parsed_statement: ParsedStatement): | ||||||
"""Executes the statement when ddl batch is active by buffering the | ||||||
statement in-memory. | ||||||
|
||||||
This method is internal and not for public use as it can change anytime. | ||||||
|
||||||
:type parsed_statement: ParsedStatement | ||||||
:param parsed_statement: parsed statement containing sql query | ||||||
""" | ||||||
from google.cloud.spanner_dbapi import ProgrammingError | ||||||
|
||||||
if parsed_statement.statement_type != StatementType.DDL: | ||||||
raise ProgrammingError("Only DDL statements are allowed in batch DDL mode.") | ||||||
self._statements.extend( | ||||||
parse_utils.parse_and_get_ddl_statements(parsed_statement.statement.sql) | ||||||
) | ||||||
|
||||||
def run_batch(self): | ||||||
"""Executes all the buffered statements on the active ddl batch by | ||||||
making a call to Spanner. | ||||||
|
||||||
This method is internal and not for public use as it can change anytime. | ||||||
""" | ||||||
from google.cloud.spanner_dbapi import ProgrammingError | ||||||
|
||||||
if self._connection._client_transaction_started: | ||||||
raise ProgrammingError( | ||||||
"Cannot execute DDL statement when transaction is already active." | ||||||
) | ||||||
return self._connection.database.update_ddl(self._statements).result() | ||||||
|
||||||
|
||||||
class BatchDmlExecutor: | ||||||
|
@@ -48,10 +96,13 @@ def execute_statement(self, parsed_statement: ParsedStatement): | |||||
"""Executes the statement when dml batch is active by buffering the | ||||||
statement in-memory. | ||||||
|
||||||
This method is internal and not for public use as it can change anytime. | ||||||
|
||||||
:type parsed_statement: ParsedStatement | ||||||
:param parsed_statement: parsed statement containing sql query and query | ||||||
params | ||||||
""" | ||||||
|
||||||
from google.cloud.spanner_dbapi import ProgrammingError | ||||||
|
||||||
if ( | ||||||
|
@@ -61,16 +112,20 @@ def execute_statement(self, parsed_statement: ParsedStatement): | |||||
raise ProgrammingError("Only DML statements are allowed in batch DML mode.") | ||||||
self._statements.append(parsed_statement.statement) | ||||||
|
||||||
def run_batch_dml(self): | ||||||
def run_batch(self): | ||||||
"""Executes all the buffered statements on the active dml batch by | ||||||
making a call to Spanner. | ||||||
|
||||||
This method is internal and not for public use as it can change anytime. | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nit: here and elsewhere
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could it be that you have not pushed that change yet? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes I did not. Thanks done |
||||||
""" | ||||||
return run_batch_dml(self._cursor, self._statements) | ||||||
|
||||||
|
||||||
def run_batch_dml(cursor: "Cursor", statements: List[Statement]): | ||||||
"""Executes all the dml statements by making a batch call to Spanner. | ||||||
|
||||||
This method is internal and not for public use as it can change anytime. | ||||||
|
||||||
:type cursor: Cursor | ||||||
:param cursor: Database Cursor object | ||||||
|
||||||
|
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -19,7 +19,11 @@ | |||||||||||||||||||||||||||||||||||||||||||||||
from google.api_core.gapic_v1.client_info import ClientInfo | ||||||||||||||||||||||||||||||||||||||||||||||||
from google.cloud import spanner_v1 as spanner | ||||||||||||||||||||||||||||||||||||||||||||||||
from google.cloud.spanner_dbapi import partition_helper | ||||||||||||||||||||||||||||||||||||||||||||||||
from google.cloud.spanner_dbapi.batch_dml_executor import BatchMode, BatchDmlExecutor | ||||||||||||||||||||||||||||||||||||||||||||||||
from google.cloud.spanner_dbapi.batch_executor import ( | ||||||||||||||||||||||||||||||||||||||||||||||||
BatchMode, | ||||||||||||||||||||||||||||||||||||||||||||||||
BatchDmlExecutor, | ||||||||||||||||||||||||||||||||||||||||||||||||
BatchDdlExecutor, | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
from google.cloud.spanner_dbapi.parse_utils import _get_statement_type | ||||||||||||||||||||||||||||||||||||||||||||||||
from google.cloud.spanner_dbapi.parsed_statement import ( | ||||||||||||||||||||||||||||||||||||||||||||||||
StatementType, | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -91,7 +95,9 @@ class Connection: | |||||||||||||||||||||||||||||||||||||||||||||||
should end a that a new one should be started when the next statement is executed. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
def __init__(self, instance, database=None, read_only=False): | ||||||||||||||||||||||||||||||||||||||||||||||||
def __init__( | ||||||||||||||||||||||||||||||||||||||||||||||||
self, instance, database=None, read_only=False, buffer_ddl_statements=False | ||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Also, in order to force a major version bump, you can add a (Also note that the title of the PR will be included in the release notes, and as such is intended for public consumption. We should therefore try to keep that title as descriptive as possible for external users so they understand what the change is.) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Added description and changed title to force a major version bump |
||||||||||||||||||||||||||||||||||||||||||||||||
): | ||||||||||||||||||||||||||||||||||||||||||||||||
self._instance = instance | ||||||||||||||||||||||||||||||||||||||||||||||||
self._database = database | ||||||||||||||||||||||||||||||||||||||||||||||||
self._ddl_statements = [] | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -114,8 +120,10 @@ def __init__(self, instance, database=None, read_only=False): | |||||||||||||||||||||||||||||||||||||||||||||||
# made atleast one call to Spanner. | ||||||||||||||||||||||||||||||||||||||||||||||||
self._spanner_transaction_started = False | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_mode = BatchMode.NONE | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_ddl_executor: BatchDdlExecutor = None | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_dml_executor: BatchDmlExecutor = None | ||||||||||||||||||||||||||||||||||||||||||||||||
self._transaction_helper = TransactionRetryHelper(self) | ||||||||||||||||||||||||||||||||||||||||||||||||
self._buffer_ddl_statements = buffer_ddl_statements | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@property | ||||||||||||||||||||||||||||||||||||||||||||||||
def autocommit(self): | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -126,6 +134,25 @@ def autocommit(self): | |||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
return self._autocommit | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@property | ||||||||||||||||||||||||||||||||||||||||||||||||
def buffer_ddl_statements(self): | ||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is this a read-only property? As in; can it only be set when a connection is created, and never changed after that? If so, is that something that we can/should document? Or is it clear from the context? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes its a read only property and there are no setters for it but in python the private variable can also be overridden. |
||||||||||||||||||||||||||||||||||||||||||||||||
"""Whether to buffer ddl statements for this connection. | ||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This needs more documentation. I (think I) understand what it means. To someone who is not familiar with Cloud Spanner, it is not clear what this means. It is also not clear when they should enable/disable this. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done |
||||||||||||||||||||||||||||||||||||||||||||||||
If the connection is in auto commit mode then this flag doesn't have any | ||||||||||||||||||||||||||||||||||||||||||||||||
significance as ddl statements would be executed as they come. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
For connection not in auto commit mode: | ||||||||||||||||||||||||||||||||||||||||||||||||
If enabled ddl statements would be buffered at client and not executed | ||||||||||||||||||||||||||||||||||||||||||||||||
at cloud spanner. When a non ddl statement comes or a transaction is | ||||||||||||||||||||||||||||||||||||||||||||||||
committed then all the existing buffered ddl statements would be executed. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
If disabled then its an error to execute ddl statement in autocommit mode. | ||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Done |
||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
:rtype: bool | ||||||||||||||||||||||||||||||||||||||||||||||||
:returns: _buffer_ddl_statements flag value. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
return self._buffer_ddl_statements | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@autocommit.setter | ||||||||||||||||||||||||||||||||||||||||||||||||
def autocommit(self, value): | ||||||||||||||||||||||||||||||||||||||||||||||||
"""Change this connection autocommit mode. Setting this value to True | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -365,7 +392,8 @@ def commit(self): | |||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
return | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
self.run_prior_DDL_statements() | ||||||||||||||||||||||||||||||||||||||||||||||||
if self.buffer_ddl_statements: | ||||||||||||||||||||||||||||||||||||||||||||||||
self.run_prior_DDL_statements() | ||||||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._spanner_transaction_started and not self._read_only: | ||||||||||||||||||||||||||||||||||||||||||||||||
self._transaction.commit() | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -463,8 +491,42 @@ def validate(self): | |||||||||||||||||||||||||||||||||||||||||||||||
"Expected: [[1]]" % result | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def start_batch_ddl(self): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is not BatchMode.NONE: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError( | ||||||||||||||||||||||||||||||||||||||||||||||||
"Cannot start a DDL batch when a batch is already active" | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
if self.read_only: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError( | ||||||||||||||||||||||||||||||||||||||||||||||||
"Cannot start a DDL batch when the connection is in read-only mode" | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
if self.buffer_ddl_statements: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError( | ||||||||||||||||||||||||||||||||||||||||||||||||
"Cannot start a DDL batch when _buffer_ddl_statements flag is True" | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_mode = BatchMode.DDL | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_ddl_executor = BatchDdlExecutor(self) | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def execute_batch_ddl_statement(self, parsed_statement: ParsedStatement): | ||||||||||||||||||||||||||||||||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do we want this method to be public? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Added comment that its an internal method to this and other methods as well |
||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is not BatchMode.DDL: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError( | ||||||||||||||||||||||||||||||||||||||||||||||||
"Cannot execute statement when the BatchMode is not DDL" | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_ddl_executor.execute_statement(parsed_statement) | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def start_batch_dml(self, cursor): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is not BatchMode.NONE: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError( | ||||||||||||||||||||||||||||||||||||||||||||||||
"Cannot start a DML batch when a batch is already active" | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -478,6 +540,9 @@ def start_batch_dml(self, cursor): | |||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def execute_batch_dml_statement(self, parsed_statement: ParsedStatement): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is not BatchMode.DML: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError( | ||||||||||||||||||||||||||||||||||||||||||||||||
"Cannot execute statement when the BatchMode is not DML" | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -486,22 +551,34 @@ def execute_batch_dml_statement(self, parsed_statement: ParsedStatement): | |||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def run_batch(self): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
result_set = None | ||||||||||||||||||||||||||||||||||||||||||||||||
olavloite marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is BatchMode.NONE: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError("Cannot run a batch when the BatchMode is not set") | ||||||||||||||||||||||||||||||||||||||||||||||||
try: | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is BatchMode.DML: | ||||||||||||||||||||||||||||||||||||||||||||||||
many_result_set = self._batch_dml_executor.run_batch_dml() | ||||||||||||||||||||||||||||||||||||||||||||||||
result_set = self._batch_dml_executor.run_batch() | ||||||||||||||||||||||||||||||||||||||||||||||||
elif self._batch_mode is BatchMode.DDL: | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_ddl_executor.run_batch() | ||||||||||||||||||||||||||||||||||||||||||||||||
finally: | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_mode = BatchMode.NONE | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_dml_executor = None | ||||||||||||||||||||||||||||||||||||||||||||||||
return many_result_set | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_ddl_executor = None | ||||||||||||||||||||||||||||||||||||||||||||||||
return result_set | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def abort_batch(self): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is BatchMode.NONE: | ||||||||||||||||||||||||||||||||||||||||||||||||
raise ProgrammingError("Cannot abort a batch when the BatchMode is not set") | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is BatchMode.DML: | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_dml_executor = None | ||||||||||||||||||||||||||||||||||||||||||||||||
if self._batch_mode is BatchMode.DDL: | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_ddl_executor = None | ||||||||||||||||||||||||||||||||||||||||||||||||
self._batch_mode = BatchMode.NONE | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -510,6 +587,9 @@ def partition_query( | |||||||||||||||||||||||||||||||||||||||||||||||
parsed_statement: ParsedStatement, | ||||||||||||||||||||||||||||||||||||||||||||||||
query_options=None, | ||||||||||||||||||||||||||||||||||||||||||||||||
): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
statement = parsed_statement.statement | ||||||||||||||||||||||||||||||||||||||||||||||||
partitioned_query = parsed_statement.client_side_statement_params[0] | ||||||||||||||||||||||||||||||||||||||||||||||||
self._partitioned_query_validation(partitioned_query, statement) | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -534,6 +614,9 @@ def partition_query( | |||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
@check_not_closed | ||||||||||||||||||||||||||||||||||||||||||||||||
def run_partition(self, encoded_partition_id): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
partition_id: PartitionId = partition_helper.decode_from_string( | ||||||||||||||||||||||||||||||||||||||||||||||||
encoded_partition_id | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -550,6 +633,9 @@ def run_partitioned_query( | |||||||||||||||||||||||||||||||||||||||||||||||
self, | ||||||||||||||||||||||||||||||||||||||||||||||||
parsed_statement: ParsedStatement, | ||||||||||||||||||||||||||||||||||||||||||||||||
): | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
This method is internal and not for public use as it can change anytime. | ||||||||||||||||||||||||||||||||||||||||||||||||
""" | ||||||||||||||||||||||||||||||||||||||||||||||||
statement = parsed_statement.statement | ||||||||||||||||||||||||||||||||||||||||||||||||
partitioned_query = parsed_statement.client_side_statement_params[0] | ||||||||||||||||||||||||||||||||||||||||||||||||
self._partitioned_query_validation(partitioned_query, statement) | ||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -584,10 +670,23 @@ def connect( | |||||||||||||||||||||||||||||||||||||||||||||||
pool=None, | ||||||||||||||||||||||||||||||||||||||||||||||||
user_agent=None, | ||||||||||||||||||||||||||||||||||||||||||||||||
client=None, | ||||||||||||||||||||||||||||||||||||||||||||||||
buffer_ddl_statements=False, | ||||||||||||||||||||||||||||||||||||||||||||||||
route_to_leader_enabled=True, | ||||||||||||||||||||||||||||||||||||||||||||||||
): | ||||||||||||||||||||||||||||||||||||||||||||||||
"""Creates a connection to a Google Cloud Spanner database. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
:type buffer_ddl_statements: bool | ||||||||||||||||||||||||||||||||||||||||||||||||
:param buffer_ddl_statements: Whether to buffer ddl statements at client | ||||||||||||||||||||||||||||||||||||||||||||||||
side. If the connection is in auto commit mode then this flag doesn't | ||||||||||||||||||||||||||||||||||||||||||||||||
have any significance as ddl statements would be executed as they come. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
For connection not in auto commit mode: | ||||||||||||||||||||||||||||||||||||||||||||||||
If enabled ddl statements would be buffered at client and not executed | ||||||||||||||||||||||||||||||||||||||||||||||||
at cloud spanner. When a non ddl statement comes or a transaction is | ||||||||||||||||||||||||||||||||||||||||||||||||
committed then all the existing buffered ddl statements would be executed. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
If disabled then its an error to execute ddl statement in autocommit mode. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
:type instance_id: str | ||||||||||||||||||||||||||||||||||||||||||||||||
:param instance_id: The ID of the instance to connect to. | ||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -658,7 +757,9 @@ def connect( | |||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||
instance = client.instance(instance_id) | ||||||||||||||||||||||||||||||||||||||||||||||||
conn = Connection( | ||||||||||||||||||||||||||||||||||||||||||||||||
instance, instance.database(database_id, pool=pool) if database_id else None | ||||||||||||||||||||||||||||||||||||||||||||||||
instance, | ||||||||||||||||||||||||||||||||||||||||||||||||
instance.database(database_id, pool=pool) if database_id else None, | ||||||||||||||||||||||||||||||||||||||||||||||||
buffer_ddl_statements=buffer_ddl_statements, | ||||||||||||||||||||||||||||||||||||||||||||||||
) | ||||||||||||||||||||||||||||||||||||||||||||||||
if pool is not None: | ||||||||||||||||||||||||||||||||||||||||||||||||
conn._own_pool = False | ||||||||||||||||||||||||||||||||||||||||||||||||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Isn't this a breaking change?
We can probably get away with it now, but this is something that we need to be more careful with, and we should try to mark whatever is not for public consumption as such as much as possible.
Either by adding a
_
at the start of the method name, or by explicitly adding documentation that the method is not for public use, and can change at any moment.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ack. Added the following documentation on the method
This method is internal and not for public use as it can change anytime.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Instead of renaming the method, we could mark it as deprecated and create a new one with the
_
prefix which indicates that it is internal. Please avoid a major version bump if at all possible as it is disruptive for downstream users.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is not the reason because of which we are proposing a major version bump. Please check https://docs.google.com/document/d/1mSVC_AhPpdSA0xUoj4LUt2o2tVM_LDv6IdssiwWnbmw for details