-
Notifications
You must be signed in to change notification settings - Fork 386
Feat: Integrating DQE Testing Approaches into SQLancer first in MySQL #1251
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
Conversation
|
Thanks, that's great! Some high-level comments first:
|
Regarding the
Given these differences, merging them into a single class could lead to confusion, increased coupling, and reduced clarity in terms of responsibility separation. I believe keeping them separate helps maintain a clean architecture and supports better extensibility in the future. That said, if you have any alternative suggestions or design ideas on how we can better organize this part of the code, I’d be more than happy to discuss them further. Thanks again for your thoughtful feedback! @mrigger @JensonSung |
|
I think current implementation is good but specialized for DQE method. |
First, revising ExepctedErrors like SQLQueryError to provide a structured information may help ease to identify errors compared to a string-formed information. While DQE does not distinguish which error is expected but compare whether errors are consistent. A clear name could be required. |
| boolean generateLimit = false; | ||
| int limit = 0; | ||
| if (operateOnSingleTable) { | ||
| if (Randomly.getBooleanWithSmallProbability()) { |
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 the expression generator already has methods for generating ORDER BYs, so we should try to re-use those. Could we perhaps also re-use or factor out components from the random query generator?
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 have add some tests for your mentioned method generateOrderBys() but I got some unexpected errors my repo commit. Sometimes test can't pass with many errors, sometimes this method generated statement like EXISTS (SELECT 1 WHERE FALSE) DESC, which never return rows in SQL statement. I don't know if the methed have some logic errors, or maybe my test have some wrongs, can you give me some suggestions? Thank you very much.
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.
Maybe only generating columns in order by expressions is enough for applying DQE, which can identify the correctness of complex expressions through WHERE clauses.
| String selectRowId = String.format("SELECT %s FROM %s", rowId, tableName); | ||
| SQLancerResultSet resultSet = new SQLQueryAdapter(selectRowId).executeAndGet(state); | ||
| HashSet<String> rows = new HashSet<>(); | ||
| if (resultSet != null) { |
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 we have some auxiliary methods for fetching rows (used by other test oracles). Can we perhaps re-use any of these here?
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.
@luliqwerty Try to find if we can re-use some auxiliary methods for fetching rows in other test oracles.
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.
Hello @mrigger @JensonSung
I can't find a suitable method to get rows you mentioned. The confusing method is CERTOracle.getRow(), which is used to get the count of rows. Can you give me some advice or maybe we can just use the way I used to do.
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 there could be one in https://github.com/sqlancer/sqlancer/blob/main/src/sqlancer/ComparatorHelper.java. If not, we could stick to the current implementation.
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.
Current implementation can reuse the method getResultSetFirstColumnAsString.
Okay, then let's leave them separate for now. I guess there is a conceptual commonality (both are about errors returned by the database system), but if that does not (yet) make sense at an implementation level, we could try to address this later. |
mrigger
left a comment
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 looks close to being merge ready. Could you have another look at my comments? @JensonSung, perhaps you could also do another review then?
| String selectRowId = String.format("SELECT %s FROM %s", rowId, tableName); | ||
| SQLancerResultSet resultSet = new SQLQueryAdapter(selectRowId).executeAndGet(state); | ||
| HashSet<String> rows = new HashSet<>(); | ||
| if (resultSet != null) { |
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 there could be one in https://github.com/sqlancer/sqlancer/blob/main/src/sqlancer/ComparatorHelper.java. If not, we could stick to the current implementation.
| String rowId = tableName + "." + COLUMN_ROWID; | ||
| String selectRowId = String.format("SELECT %s FROM %s", rowId, tableName); | ||
| SQLancerResultSet resultSet = new SQLQueryAdapter(selectRowId).executeAndGet(state, false); | ||
| HashSet<String> rows = new HashSet<>(); |
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.
Some of the code seems identical to the one above for the update case. Can we factor those out?
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, some methods could be extracted for fetching these helper columns, like fetching rowID. @luliqwerty
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.
@mrigger you are right, thanks a lot, I've reused the existing method.
|
Sorry for possible confusing. @sayJason is my another account. I try to use the same one. Any problem, @JensonSung |
mrigger
left a comment
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.
Just did another quick and noticed some final small requests. Could you please have a last look at them before we merge the PR? Other than that, @JensonSung, do the changes look good to you?
Excited to finally have DQE supported in SQLancer! Thanks a lot, @luliqwerty!
- DQEBase: remove unnecessary method - MySQLDQEOracle: simplify adding expectedErrors - MySQLErrors: add a expectedErrors
|
Good jobs @luliqwerty . LGTM. Happy to merge @mrigger . |
|
Agreed, lgtm, thanks a lot! |
|
I just noticed that when executing the tests, there seems to be an issue: Could you check whether you can reproduce this locally? This subsequently also results in a crash. |
This is an unexpected error, as the CI for the merge commit passed. The error indicates that the code is attempting to execute an SQL operation on a database connection that has already been closed. However, it seems that DQE did not perform this action. @mrigger @JensonSung |
It seems to be a crash bug. I run DQE locally in SQLancer, which executes without such errors. |
|
Could it be that the local version of MySQL and the one in the CI are different? If they are the same, perhaps it could be helpful to run for DQE a bit longer. It could be that it triggered a crash bug, as pointed out by @JensonSung. |
|
Btw, it would also be great to add DQE as part of the documentation of SQLancer in the table of the main |
Sorry for the late message. The local MySQL has a different version. I try a bit longer time to run DQE, which runs well and does not report errors. |
This PR is to Integrating DQE (Differential Query Execution) Testing Approaches into SQLancer. More information see DQE paper.
Take MySQL as the first step.
The code is divided into three main parts:
DQEBase.class: defines the core logic of DQE, including the addition and deletion of auxiliary columns, and the comparison of query results.SQLQueryError.java: provide standardized error representation and comparison functionality for SQL query operations in DQE.MySQLDQEOracle.class: Inherits fromDQEBaseand implements the specific logic of executing DQE in MySQL, including generating queries, executing queries, comparing results, etc.