-
-
Notifications
You must be signed in to change notification settings - Fork 18.2k
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: Implement option 'delete_rows' of argument 'if_exists' in 'DataFrame.to_sql' API. #60376
base: main
Are you sure you want to change the base?
feat: Implement option 'delete_rows' of argument 'if_exists' in 'DataFrame.to_sql' API. #60376
Conversation
@WillAyd I chose the name @erfannariman tagging you due to your help/interest during the lifecycle of this issue. |
e443d8d
to
33cd0d6
Compare
3c33249
to
1ef5a87
Compare
b71c0d9
to
1843040
Compare
15bda94
to
2eb19e7
Compare
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.
I'm not sure that the test failures are related. Restarted so let's see...
My remaining feedback is rather minor; overall I think the implementation looks good.
@mroeschke care to take a look?
5f6ab41
to
d1b01d2
Compare
d1b01d2
to
3e8813f
Compare
77dc01c
to
4c8fcda
Compare
NVM. CI is fine now. |
pandas/tests/io/test_sql.py
Outdated
with pandasSQL.run_transaction() as cur: | ||
cur.execute(table_stmt) | ||
|
||
if conn_name != "sqlite_buildin" and "adbc" not in conn_name: |
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.
I think to make this test more manageable we can create a generic pd.SQLError
and use that whenever these are raised instead. I don't think there's a lot of value to exposing the error messages from the underlying libraries to end users, and it would help simplify this test
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.
Yes. I totally agree on that 💯 .
pandas should definitely provide a common interface for SQL-related errors. But I believe this is out of the current scope ?
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.
I actually think we already have one in place - check pandas.errors.DatabaseError
. If you catch where these problems are in the implementation and just do raise DatabaseError from exc
then you can also just catch the DatabaseError
in the test
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.
yes, sure.
So as of now the only place raising pd.errors.DatabaseError
is the method execute
.
The new implementation delete_rows
is using cursor.execute
instead - which is at the driver level.
So the options we have to comply with what you suggested is:
- raise
pd.errors.DatabaseError
fromdelete_rows
. The issue being other methods (append
,replace
) would not raise it. - We start using
self.execute
but I tried in the past and it only works for SQLite becauserun_transaction
closes the connection while ADBC and sqlalchemy don't.
Do you see another option or have a preference ?
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.
Where within the class hierarchies does that execute get called? It seems like there is where you can catch the exception from the library and re-raise a more generic one
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.
I appreciate wanting to get this in but I don't think the issue is out of scope - we would just be choosing to ignore it, which delays the problem for a later date.
Do you see a reasonable path to a pre-cursor that can fix some of the implementation issues with execute and get us back to a place where we consistently use the internal class methods, rather than cherry-picking calls to a third party execute
?
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.
I appreciate wanting to get this in but I don't think the issue is out of scope - we would just be choosing to ignore it, which delays the problem for a later date.
And just to clarify, the issue we're discussing is the lack of a common interface for database related errors ?
I have sent a patch. It touches parts of code that are out of my league but that are required for the new test test_delete_rows_is_atomic
to work - using self.execute
.
It is not meant to be a bullet proof and solve previously existing problems (but it definitely changes existing APIs) but might be a middle-ground to what is to come and solves it for the newly introduced delete_rows
.
Please, let me know if that is feasible.
BTW, thanks a ton for the hard work reviewing it multiple times.
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.
@WillAyd LMK as soon you have the time to review :). TY
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.
The main goal would be to unify our internal implementation so that we can call self.execute
consistently instead of mixing in calls to cursor.execute
and managing one-off behaviors scattered throughout the code.
Sorry for not being clear, but when I said "precursor" i was hoping for a separate PR. Can you split off your changes to unify that internal usage into a new PR so we can review that and get it through? We can jump back to this afterwards.
W.r.t. to the current resolution catching a bare Exception
isn't the path we want to go down - that is generally way to broad of a stroke, and would even catching something like a KeyboardInterrupt and repurpose it as a DatabaseError
.
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.
Sorry for not being clear, but when I said "precursor" i was hoping for a separate PR. Can you split off your changes to unify that internal usage into a new PR so we can review that and get it through? We can jump back to this afterwards.
No problem ! I'll cherry pick the changes.
W.r.t. to the current resolution catching a bare Exception isn't the path we want to go down - that is generally way to broad of a stroke, and would even catching something like a KeyboardInterrupt and repurpose it as a DatabaseError.
That's how it's done today in the codebase
I tried to reduce changes to a minimum as this is clearly not related to the original issue but a requirement for it. Now we have a new one discussion can be moved there. I'll mark you as reviewer as soon as I open it.
0fccb84
to
49e0bcc
Compare
…Frame.to_sql' API.
Co-authored-by: Matthew Roeschke <[email protected]>
…das-dev#60532) * first * second * Update object_array.py * third * ascii * ascii2 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * ascii3 * style * style * style * style * docs * reset * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * Update doc/source/whatsnew/v3.0.0.rst --------- Co-authored-by: Abby VeCasey <[email protected]> Co-authored-by: Matthew Roeschke <[email protected]> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
8cadb78
to
b0c2eff
Compare
blocked by #60748. Changing to draft. |
doc/source/whatsnew/v3.0.0.rst
file if fixing a bug or adding a new feature.