Skip to content
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

Oracle doesn't throw error when there is missing permission for the sequence while insertion #149

Closed
rathoreamrsingh opened this issue Jul 5, 2024 · 4 comments

Comments

@rathoreamrsingh
Copy link

Steps to reproduce:

  1. Create a table with sequence for ID in it.
  2. Create different user.
  3. Grant Select permission for the table to this newly created user but don't grant select permission on sequence.
  4. Insert data to the table using below method.
    DatabaseClient dbclient = getDbClient();
    Flux<? extends Result> flux = dbclient.inConnectionMany(connection -> {
    Statement statement = connection.createStatement(insertStatement);
    bindData(recordsToSave, statement);
    return Flux.from(statement.execute())
    .doOnNext(data -> {
    log.info(data);
    })
    .doOnError(data -> {
    log.error(data);
    });
    });

This results in OracleResultImpl$BatchUpdateErrorResult which comes under doOnNext rather than doOnError.

The expectation is error should be in error stream doOnError but it is coming as data in doOnNext stream.

Please suggest how to resolve this issue or can be caught in proper manner.

@Michael-A-McMahon
Copy link
Member

I can appreciate the confusion here: We might expect the Statement.execute() Publisher to emit onError when that statement fails. This is actually not what the Publisher is specified to do. The Publisher is specified to emit Result objects, and a Result may represent a database error.

For a batch update statement, you can add an operator which maps each Result to the count of rows updated by each set of bind values in your batch:

Flux.from(statement.execute())
  .flatMap(Result::getRowsUpdated)

The getRowsUpdated method will return a Publisher that emits the error to onError.

Hope this helps. Please let me know.

@rathoreamrsingh
Copy link
Author

rathoreamrsingh commented Jul 30, 2024

In this somehow the .flatMap(Result::getRowsUpdated) is getting called twice.
Second time since there is no bind with the statement it is throwing below mentioned error.

Caused by: oracle.r2dbc.impl.OracleR2dbcException$OracleR2dbcNonTransientException: One or more binds not set after calling add()

it is supposed to call this statement once.

@Michael-A-McMahon
Copy link
Member

I think you'll be able to resolve this error by not calling add() after calling bind methods for the last set of binds. It is an awkward requirement, but we do this be compliant with the R2DBC Specification (r2dbc/r2dbc-spi#259).

Receiving two results here is something I may want to fix. The first result should be an update count, and the second result should be the error. This behavior is an artifact of an early version of the SPI compliance test (r2dbc/r2dbc-spi#259). I noticed that in the latest version, the requirement to return an update count is gone (https://github.com/r2dbc/r2dbc-spi/blob/fc07457586183f32bf88ca65ada62c819bc164e4/r2dbc-spi-test/src/main/java/io/r2dbc/spi/test/TestKit.java#L672). So there's really no need to do this anymore.

@Michael-A-McMahon
Copy link
Member

Hope this was resolved. Otherwise, let me know if I can help.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants