-
-
Notifications
You must be signed in to change notification settings - Fork 727
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
Result codes #171
Comments
@hartbit There are two subjects in this issue:
Enabling extended error codes is a valid feature request: thanks for the suggestion! I'm balanced about exposing error codes through a specific GRDB API. Those codes are already available through the C SQLite module: import GRDB
import SQLiteMacOSX
// Without extended error codes
do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where error.code == SQLITE_CONSTRAINT {
// do something about it
}
// With extended error codes
do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where (error.code & SQLITE_CONSTRAINT) != 0 {
// do something about it
} (However, for some reason I don't understand yet, extended error codes are not usable this way (I get an "unresolved identifier" compilation error). I'd like this oddity to be resolved.) The reason why I'm not keen defining an error hierarchy similar to the one you provide above (enum DatabaseError, enum ConstraintError, etc.) is that I'd prefer error codes to keep their original names (SQLITE_CONSTRAINT, etc.). For me, it's part of the "don't surprise experts, don't confuse beginners" rule. Plus it fosters googleability, and supports SQLite documentation quotes: catch let error as DatabaseError where error.code == SQLITE_CONSTRAINT {
// https://www.sqlite.org/rescode.html#constraint says:
// > The SQLITE_CONSTRAINT error code means that an SQL constraint
// > violation occurred while trying to process an SQL statement.
...
} Now, I can hear that importing the C module is too much work for a use case that should be obvious. We could then modify the DatabaseError type so that its code has a These are my initial reactions to your suggestion. Feel free to discuss! |
👍 I think it might even be worth enabling by default. It's disabled by default in SQLite for historic reasons, but more detailed error codes can't hurt!
Very good argument. But I'd like to provide a counter-argument. If/when we provide extended result codes through sqlite3_extended_result_codes , matching against the non-extended codes like you show in your example requires a bitwise operation, which is much less obvious than what can be achieved through a nested enum:
Sounds like a good compromise. I just hope that we can find a solution which avoids the bitwise operation when comparing against the more general errors. |
Yep, I can see how non-obvious it is (and indeed my bitwise sample code above has a bug). What would you think of: do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where error.code == .SQLITE_CONSTRAINT_FOREIGNKEY {
// foreign key constraint error
} catch let error as DatabaseError where error.primaryResultCode == .SQLITE_CONSTRAINT {
// any other constraint error
} catch let error as DatabaseError {
// any other database error
error.code.rawValue // 5
error.code.description // "5 (database is locked)"
} catch {
// any other error
} I know that not using Swift enums may look like a missed opportunity. Yet I find it very valuable to preserve both the original names and raw values for googleability (sorry to repeat myself), and to allow users to introduce their own error codes (I find non extensible error enums painful to code with - Alamofire being my latest disappointment). And I don't much mind exposing the notions of "extended result codes" and "primary result codes" (see SQLite doc) since most non-trivial uses of GRDB eventually require one to strengthen one's SQLite skills anyway. See the Issue171 branch for the code that supports the example above. Thoughts, suggestions, improvements, utter disapproval? |
I'm quite happy with this direction :) Could I propose one modification? If we keep the same constant names as SQLite (totally agree with the reasoning), it might be confusing to have the property named |
@groue closed by mistake :p |
Good idea :-) |
Or is it? First, if DatabaseError has two properties do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where error.extendedResultCode == .SQLITE_CONSTRAINT_FOREIGNKEY {
// foreign key constraint error
} catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT {
// any other constraint error
} Yet is it obvious that Second, application-specific custom code wouldn't be so well supported. In the sample code below, the raw custom code is accessed with the longest property name, extension ResultCode {
static let appLogicError = ResultCode(rawValue: 1000)
}
do {
throw DatabaseError(code: .appLogicError)
} catch let error as DatabaseError {
error.extendedResultCode // 1000
error.resultCode // 232
} Conversely, the |
Latest version: do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT_FOREIGNKEY {
// foreign key constraint error
} catch let error as DatabaseError where error.primaryResultCode == .SQLITE_CONSTRAINT {
// any other constraint error
} |
The problem with those property names is that they defy the first part of the "don't surprise experts, don't confuse beginners" rule. I would have been surprised about them.
It is obvious to me who as I've already encountered the
I agree with this. But I would strive for a different solution. The problem here that I see is that we are mixing the concepts of application error codes and SQLite error codes. Couldn't we solve the problem by having a |
Hello again @hartbit You have good points for knowledgable users, and I have good points for novice users. And no we can't have a specific path for custom errors because they sometimes have to go through SQLite as Int32 (think errors returned by custom functions, for example). I propose that we forget property pairs, and move on. Here is a new proposal: There is a single This allows checking for both extended result codes and custom codes: do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT_FOREIGNKEY {
// foreign key constraint error
} catch let error as DatabaseError where error.resultCode == .appLogicError {
// app logic error
} Now we also need to check for categories of extended result codes (such as all extended result codes that match SQLITE_CONSTRAINT). For that, I propose that we introduce actual Swift pattern matching, with the do {
try db.execute("SOMETHING")
} catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT_FOREIGNKEY {
// foreign key constraint error
} catch let error as DatabaseError where error.resultCode ~= .SQLITE_CONSTRAINT {
// other constraint error
} The do {
try db.execute("SOMETHING")
} catch let error as DatabaseError {
switch error.resultCode {
case .SQLITE_CONSTRAINT_FOREIGNKEY: // foreign key constraint error
case .SQLITE_CONSTRAINT: // other constraint error
default: // other database error
}
} The ~= operator could be implemented this way: func ~= (pattern: ResultCode, value code: ResultCode) -> Bool {
if pattern.rawValue & 0xFF == 0 {
// pattern is not an extended result code: check if code extends it
return (value.rawValue & 0xFF) == pattern.rawValue
} else {
// pattern is an extended result code
return value.rawValue == pattern.rawValue
}
} A drawback of this approach is clearly the snippet below, because one has to carefully read the documentation in order to choose between // correct, but not ideal
catch let error as DatabaseError where error.resultCode ~= .SQLITE_CONSTRAINT
// incorrect, unfortunately
catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT Maybe we should drop Equatable from ResultCode in order to prevent this easy mistake: We could also drop // both correct with a fancy == operator:
catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT_FOREIGNKEY
catch let error as DatabaseError where error.resultCode == .SQLITE_CONSTRAINT What do you think? |
Looks like a promising direction. But I think that the dilemma we are now having with comparison is due to the fact that we are coalescing both extended result codes and result codes in the same type. For example, this might help (even though I still don't like the
But before we continue in this direction, could you explain to me why it's important to allow application-specific custom codes? It seems like this requirement is at the root of all the problems we've had in this thread and all solutions look less than ideal to me. If |
GRDB is a productivity-oriented library: elegancy is welcome, but only as long as it helps users :-) You invoke YAGNI against custom error codes. And indeed I haven't used any custom error code in any project using GRDB yet. Still, SQLite does not define any extended result code for "invalid argument" of custom functions, for example. I can also imagine transaction observers that want to communicate the reason why they invalidate a transaction when they throw an error from their
I see the point, and also had considered two different result code types. That's too much a burden in one developer's mind, I think. And which type is the extension point? What we need is:
The most difficult point here is forgiveness: errors don't happen that much. Many programmers will write code that handle errors, hoping that they are correctly handled, but won't actually test those code paths. That's because some errors are difficult to trigger. Considering our previous attempts:
Sorry if I nitpick a lot, @hartbit: there is a reason for this issue to exist :-) |
I'm sold now to your DatabaseError property pair resultCode/extendedResultCode. Thanks for your suggestion, @hartbit! |
@groue glad to hear it :) it's always worth discussing all design possibilities beforehand! |
Closing in favor of #180 |
One of the pieces which I would love to see in GRDB is better support for result/error codes. One important piece of that would be to start supporting extended error codes, which can be done easily enough (I have a PR ready in case). The second would be improved errors.
Quick and dirty ideas:
I this has a few issues: we've lost the error message, etc.. but perhaps it can help us find a good design.
The text was updated successfully, but these errors were encountered: