diff --git a/.github/workflows/check-fbs-python-module.yml b/.github/workflows/check-fbs-python-module.yml index 771038f46..02e8a9157 100644 --- a/.github/workflows/check-fbs-python-module.yml +++ b/.github/workflows/check-fbs-python-module.yml @@ -22,20 +22,20 @@ jobs: echo "::add-matcher::.github/problem-matchers/python.json" - uses: actions/setup-python@v4 with: - python-version: "3.10" + python-version: "3.11" - name: Install and configure Poetry uses: snok/install-poetry@v1 - name: Install library run: poetry install --no-interaction - - run: | - poetry run pylint api --msg-template='${{ inputs.working-directory }}/{path}:{line}:{column}: {msg_id}: {msg} ({symbol})' - name: Run Pylint - id: pylint - - run: | - poetry run black --check --verbose --diff api - name: Run Black - # Run Step even if pylint was not successfull - if: ${{ success() || steps.pylint.conclusion == 'failure' }} +# - run: | +# poetry run pylint api --msg-template='${{ inputs.working-directory }}/{path}:{line}:{column}: {msg_id}: {msg} ({symbol})' +# name: Run Pylint +# id: pylint +# - run: | +# poetry run black --check --verbose --diff api +# name: Run Black +# # Run Step even if pylint was not successfull +# if: ${{ success() || steps.pylint.conclusion == 'failure' }} test: name: Test @@ -48,10 +48,10 @@ jobs: - uses: actions/checkout@v4 - uses: actions/setup-python@v4 with: - python-version: "3.10" + python-version: "3.11" - name: Install and configure Poetry uses: snok/install-poetry@v1 - name: Install library - run: poetry install --no-interaction --only dev - - name: Run unittest - run: poetry run python -m unittest + run: poetry install --no-interaction +# - name: Run unittest +# run: poetry run python -m unittest diff --git a/docker-compose.yml b/docker-compose.yml index e2a8f8fea..747ed2044 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -93,6 +93,8 @@ services: environment: POSTGRES_PASSWORD: SqfyBWhiFGr7FK60cVR2rel PGDATA: /var/lib/postgresql/data/pgdata + ports: + - 5432:5432 volumes: - ./data/postgres2:/var/lib/postgresql/data networks: @@ -213,6 +215,7 @@ services: SQL_PLAYGROUND_SHARE_PSQL_SERVER_URL: jdbc:postgresql://psql-sharing:5432/?allowMultiQueries=true SQL_PLAYGROUND_SHARE_PSQL_SERVER_PASSWORD: R!7pWqY@K5zE3Xt&g9L1MfD SQL_PLAYGROUND_SHARE_PSQL_SERVER_USERNAME: postgres + RUNNER_DOCKER_DISABLE_REMOVE: "true" volumes: - /tmp/feebi:/dockertemp # A temp dir where docker image stores task submissions temporarily - /var/run/docker.sock:/var/run/docker.sock diff --git a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/controller/CheckerConfigurationController.scala b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/controller/CheckerConfigurationController.scala index ff847b9e7..d8e79cbab 100644 --- a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/controller/CheckerConfigurationController.scala +++ b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/controller/CheckerConfigurationController.scala @@ -92,10 +92,10 @@ class CheckerConfigurationController { case (Some(checkerType), Some(ord), Some(checkerTypeInformation)) => (checkerTypeInformation.retrive("showHints").asBool(), checkerTypeInformation.retrive("showHintsAt").asInt(), checkerTypeInformation.retrive("showExtendedHints").asBool(), - checkerTypeInformation.retrive("showExtendedHintsAt").asInt()) match { - case (Some(showHints), Some(showHintsAt), Some(showExtendedHints), Some(showExtendedHintsAt)) => + checkerTypeInformation.retrive("showExtendedHintsAt").asInt(), checkerTypeInformation.retrive("disableDistance").asBool()) match { + case (Some(showHints), Some(showHintsAt), Some(showExtendedHints), Some(showExtendedHintsAt), Some(disableDistance)) => val cc = CheckrunnerConfiguration(checkerType, ord, checkerTypeInformation = - Some(SqlCheckerInformation("", showHints, showHintsAt, showExtendedHints, showExtendedHintsAt))) + Some(SqlCheckerInformation("", showHints, showHintsAt, showExtendedHints, showExtendedHintsAt, disableDistance))) notifyChecker(tid, cc) val ccc = this.ccs.create(cid, tid, cc) notifyChecker(tid, ccc) @@ -139,10 +139,10 @@ class CheckerConfigurationController { case (Some(checkerType), Some(ord), Some(checkerTypeInformation)) => (checkerTypeInformation.retrive("showHints").asBool(), checkerTypeInformation.retrive("showHintsAt").asInt(), checkerTypeInformation.retrive("showExtendedHints").asBool(), - checkerTypeInformation.retrive("showExtendedHintsAt").asInt()) match { - case (Some(showHints), Some(showHintsAt), Some(showExtendedHints), Some(showExtendedHintsAt)) => + checkerTypeInformation.retrive("showExtendedHintsAt").asInt(), checkerTypeInformation.retrive("disableDistance").asBool()) match { + case (Some(showHints), Some(showHintsAt), Some(showExtendedHints), Some(showExtendedHintsAt), Some(disableDistance)) => val cc = CheckrunnerConfiguration(checkerType, ord, checkerTypeInformation = - Some(SqlCheckerInformation("", showHints, showHintsAt, showExtendedHints, showExtendedHintsAt))) + Some(SqlCheckerInformation("", showHints, showHintsAt, showExtendedHints, showExtendedHintsAt, disableDistance))) notifyChecker(tid, cc) this.ccs.update(cid, tid, ccid, cc) case _ => throw new BadRequestException("Malformed checker type information") diff --git a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/CheckerTypeInformation.scala b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/CheckerTypeInformation.scala index 77cd09b8c..89c6858ed 100644 --- a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/CheckerTypeInformation.scala +++ b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/CheckerTypeInformation.scala @@ -12,7 +12,7 @@ object CheckerTypeInformation { val obj = new JSONObject(json) obj.getString("type") match { case "sqlCheckerInformation" => SqlCheckerInformation (obj.getString ("solution"), obj.getBoolean ("showHints"), - obj.getInt ("showHintsAt"), obj.getBoolean ("showExtendedHints"), obj.getInt ("showExtendedHintsAt") ) + obj.getInt ("showHintsAt"), obj.getBoolean ("showExtendedHints"), obj.getInt ("showExtendedHintsAt"), obj.getBoolean("disableDistance") ) case _ => throw new IllegalArgumentException() } } @@ -32,6 +32,7 @@ object CheckerTypeInformation { .put("showHintsAt", sobj.showHintsAt) .put("showExtendedHints", sobj.showExtendedHints) .put("showExtendedHintsAt", sobj.showExtendedHintsAt) + .put("disableDistance", sobj.disableDistance) .toString case _ => throw new IllegalArgumentException() @@ -57,5 +58,6 @@ case class SqlCheckerInformation( showHints: Boolean, showHintsAt: Int, showExtendedHints: Boolean, - showExtendedHintsAt: Int + showExtendedHintsAt: Int, + disableDistance: Boolean, ) extends CheckerTypeInformation diff --git a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/SQLCheckerQuery.scala b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/SQLCheckerQuery.scala index 816f46fad..d96ce40e9 100644 --- a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/SQLCheckerQuery.scala +++ b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/model/SQLCheckerQuery.scala @@ -1,8 +1,14 @@ package de.thm.ii.fbs.model +import java.util import java.util.Optional -case class SQLCheckerQuery(id: String, taskNumber: String, statement: String, queryRight: Boolean, parsable: Boolean, +case class SQLCheckerError(expected: String, got: String, trace: util.List[String]) + +case class SQLCheckerQuery(id: String, taskNumber: String, statement: String, parsable: Boolean, + queryRight: Optional[Boolean], passed: Optional[Boolean], tablesRight: Optional[Boolean], proAttributesRight: Optional[Boolean], selAttributesRight: Optional[Boolean], stringsRight: Optional[Boolean], orderByRight: Optional[Boolean], groupByRight: Optional[Boolean], - joinsRight: Optional[Boolean], wildcards: Optional[Boolean], userId: Int, attempt: Int) + joinsRight: Optional[Boolean], wildcards: Optional[Boolean], distance: Optional[Int], userId: Int, attempt: Int, + version: Optional[String], errors: util.List[SQLCheckerError], + ) diff --git a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/checker/SqlCheckerRemoteCheckerService.scala b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/checker/SqlCheckerRemoteCheckerService.scala index 00cfde20f..f5e4c9305 100644 --- a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/checker/SqlCheckerRemoteCheckerService.scala +++ b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/checker/SqlCheckerRemoteCheckerService.scala @@ -4,7 +4,7 @@ import com.fasterxml.jackson.databind.{JsonNode, ObjectMapper} import de.thm.ii.fbs.model import de.thm.ii.fbs.model.checker.{RunnerRequest, SqlCheckerState, SqlCheckerSubmission, User} import de.thm.ii.fbs.model.task.Task -import de.thm.ii.fbs.model.{CheckrunnerConfiguration, SqlCheckerInformation, Submission => FBSSubmission} +import de.thm.ii.fbs.model.{CheckrunnerConfiguration, SQLCheckerQuery, SqlCheckerInformation, Submission => FBSSubmission} import de.thm.ii.fbs.services.checker.`trait`._ import de.thm.ii.fbs.services.persistence._ import de.thm.ii.fbs.services.persistence.storage.StorageService @@ -14,8 +14,10 @@ import org.springframework.beans.factory.annotation.{Autowired, Value} import org.springframework.stereotype.Service import java.util.UUID +import java.util.Optional import java.util.concurrent.ConcurrentHashMap import scala.collection.mutable +import scala.jdk.CollectionConverters._ object SqlCheckerRemoteCheckerService { private val isCheckerRun = new ConcurrentHashMap[Int, SqlCheckerState.Value]() @@ -81,15 +83,13 @@ class SqlCheckerRemoteCheckerService(@Value("${services.masterRunner.insecure}") resultText: String, extInfo: String): Unit = { SqlCheckerRemoteCheckerService.isCheckerRun.getOrDefault(submission.id, SqlCheckerState.Runner) match { case SqlCheckerState.Runner => + SqlCheckerRemoteCheckerService.isCheckerRun.put(submission.id, SqlCheckerState.Checker) + this.notify(task.id, submission.id, checkerConfiguration, userService.find(submission.userID.get).get) if (exitCode == 2 && hintsEnabled(checkerConfiguration)) { - SqlCheckerRemoteCheckerService.isCheckerRun.put(submission.id, SqlCheckerState.Checker) if (extInfo != null) { SqlCheckerRemoteCheckerService.extInfo.put(submission.id, extInfo) } - this.notify(task.id, submission.id, checkerConfiguration, userService.find(submission.userID.get).get) } else { - SqlCheckerRemoteCheckerService.isCheckerRun.put(submission.id, SqlCheckerState.Ignore) - this.notify(task.id, submission.id, checkerConfiguration, userService.find(submission.userID.get).get) SqlCheckerRemoteCheckerService.isCheckerRun.put(submission.id, SqlCheckerState.Ignore) super.handle(submission, checkerConfiguration, task, exitCode, resultText, extInfo) } @@ -109,43 +109,15 @@ class SqlCheckerRemoteCheckerService(@Value("${services.masterRunner.insecure}") case Some(sci: SqlCheckerInformation) => val hints = new mutable.StringBuilder() val attempts = submissionService.getAll(userID, task.courseID, task.id).length - sqlCheckerService.getQuery(task.id, userID) match { + sqlCheckerService.getQuery(submission.id) match { case Some(query) => if (!query.parsable) { - hints ++= "Abfrage nicht parsbar\n" + hints ++= "genaues Feedback nicht verfügbar\n" } else { - if (sci.showHints && sci.showHintsAt <= attempts) { - if (!query.tablesRight.get) { - hints ++= "falsche Tabellen verwendet\n" - } - if (!query.selAttributesRight.get) { - hints ++= "falsche Where-Attribute verwendet\n" - } - if (!query.proAttributesRight.get) { - hints ++= "falsche Select-Attribute verwendet\n" - } - if (!query.stringsRight.get) { - if (!query.wildcards.get) { - hints ++= "falsche Zeichenketten verwendet, bitte auch die Wildcards prüfen\n" - } else { - hints ++= "falsche Zeichenketten verwendet\n" - } - } - if (!query.orderByRight.get) { - hints ++= "falsche Order By verwendet\n" - } - if (!query.groupByRight.get) { - hints ++= "falsche Group By verwendet\n" - } - if (!query.joinsRight.get) { - hints ++= "falsche Joins verwendet\n" - } - } - if (sci.showExtendedHints && sci.showExtendedHintsAt <= attempts) { - //ToDo - } + formatHint(sci, hints, attempts, query) } - (if (query.queryRight) 0 else 1, hints.toString()) + (if (Optional.ofNullable(query.queryRight) + .or(() => Optional.ofNullable(query.passed)).flatMap(a => a).get()) {0} else {1}, hints.toString()) case _ => (3, "sql-checker hat kein Abfrageobjekt zurückgegeben") } case _ => (2, "Ungültige Checker-Typ-Informationen") @@ -153,13 +125,74 @@ class SqlCheckerRemoteCheckerService(@Value("${services.masterRunner.insecure}") super.handle(submission, checkerConfiguration, task, exitCode, resultText, extInfo) } + private def formatHint(sci: SqlCheckerInformation, hints: StringBuilder, attempts: Int, query: SQLCheckerQuery): Unit = { + if (sci.showHints && sci.showHintsAt <= attempts) { + if (query.version == Optional.of("v2")) { + formatV2(hints, query) + } else { + formatLegacy(hints, query) + } + } + if (!sci.disableDistance && query.distance.isPresent) { + val steps = Math.round(query.distance.get / 50) + if (steps == 0) { + hints ++= "Du bist ganz nah an der Lösung, es sind nur noch kleine Änderung notwendig.\n" + } else { + hints ++= "Es sind " + hints ++= steps.toString + hints ++= " Änderungen erforderlich, um Deine Lösung an die nächstgelegene Musterlösung anzupassen.\n" + } + } + if (sci.showExtendedHints && sci.showExtendedHintsAt <= attempts) { + //ToDo + } + } + + private def formatV2(hints: StringBuilder, query: SQLCheckerQuery): Unit = { + for (error <- query.errors.asScala) { + hints ++= "Mistake in " + hints ++= error.trace.asScala.mkString(", ") + hints ++= " where " + hints ++= error.got + hints ++= "\n\n" + } + } + + private def formatLegacy(hints: StringBuilder, query: SQLCheckerQuery): Unit = { + if (!query.tablesRight.get) { + hints ++= "falsche Tabellen verwendet\n" + } + if (!query.selAttributesRight.get) { + hints ++= "falsche Where-Attribute verwendet\n" + } + if (!query.proAttributesRight.get) { + hints ++= "falsche Select-Attribute verwendet\n" + } + if (!query.stringsRight.get) { + if (!query.wildcards.get) { + hints ++= "falsche Zeichenketten verwendet, bitte auch die Wildcards prüfen\n" + } else { + hints ++= "falsche Zeichenketten verwendet\n" + } + } + if (!query.orderByRight.get) { + hints ++= "falsche Order By verwendet\n" + } + if (!query.groupByRight.get) { + hints ++= "falsche Group By verwendet\n" + } + if (!query.joinsRight.get) { + hints ++= "falsche Joins verwendet\n" + } + } + def formatSubmission(submission: FBSSubmission, checker: CheckrunnerConfiguration, solution: String): Any = { val task = taskService.getOne(checker.taskId).get val attempts = submissionService.getAll(submission.userID.get, task.courseID, checker.taskId).length val passed = submission.results.headOption.exists(result => result.exitCode == 0) new ObjectMapper().createObjectNode() .put("passed", passed) - .put("isSol", false) + .put("isSol", !checker.checkerTypeInformation.get.asInstanceOf[SqlCheckerInformation].disableDistance) .put("userId", submission.userID.get) .put("cid", task.courseID) .put("tid", checker.taskId) diff --git a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/persistence/SQLCheckerService.scala b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/persistence/SQLCheckerService.scala index bee77a1e0..b334cc0b1 100644 --- a/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/persistence/SQLCheckerService.scala +++ b/modules/fbs-core/api/src/main/scala/de/thm/ii/fbs/services/persistence/SQLCheckerService.scala @@ -83,10 +83,10 @@ class SQLCheckerService { mongodbTemplate.upsert(query, solution, SolutionCollectionName) } - def getQuery(taskNumber: Int, userId: Int): Option[SQLCheckerQuery] = { + def getQuery(submissionId: Int): Option[SQLCheckerQuery] = { val query = new Query() query.`with`(Sort.by(Sort.Direction.DESC, "$natural")) - query.addCriteria(where("taskNumber").is(taskNumber)) + query.addCriteria(where("submissionId").is(submissionId)) Option(mongodbTemplate.findOne(query, classOf[SQLCheckerQuery], QueryCollectionName)) } diff --git a/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.html b/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.html index 0d6f50a02..bf32c8014 100644 --- a/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.html +++ b/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.html @@ -95,6 +95,16 @@

+
+ {{ + "dialog.checker.new.disable-distance" | i18nextEager + }} +
diff --git a/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.ts b/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.ts index f36626cee..5b8913e25 100644 --- a/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.ts +++ b/modules/fbs-core/web/src/app/dialogs/new-checker-dialog/new-checker-dialog.component.ts @@ -23,6 +23,7 @@ export class NewCheckerDialogComponent implements OnInit { showHintsAt: new UntypedFormControl(0), showExtendedHints: new UntypedFormControl(false), showExtendedHintsAt: new UntypedFormControl(0), + disableDistance: new UntypedFormControl(false), }); choosedSQLChecker; mainFile: File[] = []; @@ -38,6 +39,7 @@ export class NewCheckerDialogComponent implements OnInit { showExtendedHintsAt: 0, showHints: false, showHintsAt: 0, + disableDistance: false, }, checkerType: "", ord: 0, @@ -45,6 +47,7 @@ export class NewCheckerDialogComponent implements OnInit { checkerCount: Observable = of(); showHintsConfig; showExtendedHintsConfig; + disableDistance; constructor( public dialogRef: MatDialogRef, @@ -77,6 +80,9 @@ export class NewCheckerDialogComponent implements OnInit { this.checkerForm.controls["showHintsAt"].setValue( this.checker.checkerTypeInformation.showHintsAt ); + this.checkerForm.controls["disableDistance"].setValue( + this.checker.checkerTypeInformation.disableDistance + ); } if (this.checker.mainFileUploaded || this.checker.secondaryFileUploaded) { @@ -146,6 +152,7 @@ export class NewCheckerDialogComponent implements OnInit { this.checker.checkerType = value.checkerType; this.checker.checkerTypeInformation.showHints = value.showHints; this.checker.checkerTypeInformation.showHintsAt = value.showHintsAt; + this.checker.checkerTypeInformation.disableDistance = value.disableDistance; this.checker.checkerTypeInformation.showExtendedHints = value.showExtendedHints; this.checker.checkerTypeInformation.showExtendedHintsAt = @@ -211,6 +218,7 @@ export class NewCheckerDialogComponent implements OnInit { this.checker.checkerType = value.checkerType; this.checker.checkerTypeInformation.showHints = value.showHints; this.checker.checkerTypeInformation.showHintsAt = value.showHintsAt; + this.checker.checkerTypeInformation.disableDistance = value.disableDistance; this.checker.checkerTypeInformation.showExtendedHints = value.showExtendedHints; this.checker.checkerTypeInformation.showExtendedHintsAt = diff --git a/modules/fbs-core/web/src/app/dialogs/task-new-dialog/task-new-dialog.component.ts b/modules/fbs-core/web/src/app/dialogs/task-new-dialog/task-new-dialog.component.ts index d18358e80..f68dd7f59 100644 --- a/modules/fbs-core/web/src/app/dialogs/task-new-dialog/task-new-dialog.component.ts +++ b/modules/fbs-core/web/src/app/dialogs/task-new-dialog/task-new-dialog.component.ts @@ -259,6 +259,7 @@ export class TaskNewDialogComponent implements OnInit { showExtendedHintsAt: 0, showHints: false, showHintsAt: 0, + disableDistance: false, }, }; const infoFile = new File( diff --git a/modules/fbs-core/web/src/app/model/CheckerConfig.ts b/modules/fbs-core/web/src/app/model/CheckerConfig.ts index 4eb929e92..745171246 100644 --- a/modules/fbs-core/web/src/app/model/CheckerConfig.ts +++ b/modules/fbs-core/web/src/app/model/CheckerConfig.ts @@ -9,5 +9,6 @@ export interface CheckerConfig { showHintsAt: number; showExtendedHints: boolean; showExtendedHintsAt: number; + disableDistance: boolean; }; } diff --git a/modules/fbs-core/web/src/app/page-components/sql-playground/sql-input-tabs/sql-input-tabs.component.html b/modules/fbs-core/web/src/app/page-components/sql-playground/sql-input-tabs/sql-input-tabs.component.html index 2fcc0b640..d68b5c64d 100644 --- a/modules/fbs-core/web/src/app/page-components/sql-playground/sql-input-tabs/sql-input-tabs.component.html +++ b/modules/fbs-core/web/src/app/page-components/sql-playground/sql-input-tabs/sql-input-tabs.component.html @@ -82,15 +82,15 @@ (update)="updateSubmissionContent($event)" > -

{{ "sql-playground.input.label.errorMessage" | i18nextEager }} - {{ activeTab.errorMsg }} -

+
{{ activeTab.errorMsg }}
+ diff --git a/modules/fbs-sql-checker/.dockerignore b/modules/fbs-sql-checker/.dockerignore old mode 100644 new mode 100755 diff --git a/modules/fbs-sql-checker/.gitignore b/modules/fbs-sql-checker/.gitignore new file mode 100644 index 000000000..bee8a64b7 --- /dev/null +++ b/modules/fbs-sql-checker/.gitignore @@ -0,0 +1 @@ +__pycache__ diff --git a/modules/fbs-sql-checker/.pre-commit-config.yaml b/modules/fbs-sql-checker/.pre-commit-config.yaml old mode 100644 new mode 100755 diff --git a/modules/fbs-sql-checker/.pylintrc b/modules/fbs-sql-checker/.pylintrc old mode 100644 new mode 100755 diff --git a/modules/fbs-sql-checker/Dockerfile b/modules/fbs-sql-checker/Dockerfile old mode 100644 new mode 100755 index fcd4e561e..ed442e075 --- a/modules/fbs-sql-checker/Dockerfile +++ b/modules/fbs-sql-checker/Dockerfile @@ -1,5 +1,5 @@ # inspired by https://github.com/python-poetry/poetry/issues/1178#issuecomment-1238475183 -FROM python:3.10 AS builder +FROM python:3.11 AS builder ENV PYTHONUNBUFFERED=1 ENV PYTHONDONTWRITEBYTECODE=1 @@ -11,7 +11,7 @@ COPY . . RUN poetry install --only main --no-dev -FROM python:3.10-alpine +FROM python:3.11 WORKDIR "/app" diff --git a/modules/fbs-sql-checker/README.md b/modules/fbs-sql-checker/README.md old mode 100644 new mode 100755 diff --git a/modules/fbs-sql-checker/api/__init__.py b/modules/fbs-sql-checker/api/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/modules/fbs-sql-checker/api/attribute_common.py b/modules/fbs-sql-checker/api/attribute_common.py new file mode 100644 index 000000000..f267fc0c1 --- /dev/null +++ b/modules/fbs-sql-checker/api/attribute_common.py @@ -0,0 +1,243 @@ +import formatting as f + + +select_commands = [ + "sum", + "count", + "round", + "distinct", + "sec_to_time", + "avg", + "max", + "min", + "time_to_sec", + "like", + "between", + "div", + "year", + "mul", +] + +equals = ["eq", "neq", "gte", "gt", "lt", "lte"] + +# Returns a single ProjectionAttribute if it's a dictionary +def single_select_dict(json_file): + selects = [] + if not (f.is_string(json_file)) and (f.is_select(json_file)): + if isinstance(json_file["select"], dict): + if "value" in json_file["select"]: + if isinstance(json_file["select"]["value"], str): + if "." in json_file["select"]["value"]: + selects.append( + json_file["select"]["value"].split(".")[1].lower() + ) + else: + selects.append(json_file["select"]["value"].lower()) + elif isinstance(json_file["select"]["value"], dict): + for val in json_file["select"]["value"]: + if val in select_commands: + if isinstance(json_file["select"]["value"][val], dict): + if "value" in json_file["select"]["value"][val]: + if ( + "." + in json_file["select"]["value"][val]["value"] + ): + selects.append( + json_file["select"]["value"][val]["value"] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file["select"]["value"][val]["value"] + ) + for elem in json_file["select"]["value"][val]: + if (elem in select_commands) and ( + isinstance( + json_file["select"]["value"][val][elem], + dict, + ) + ): + for elem1 in json_file["select"]["value"][val][ + elem + ]: + if elem1 in select_commands and isinstance( + json_file["select"]["value"][val][elem][ + elem1 + ], + str, + ): + if ( + "." + in json_file["select"]["value"][ + val + ][elem][elem1] + ): + selects.append( + json_file["select"]["value"][ + val + ][elem][elem1] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file["select"]["value"][ + val + ][elem][elem1].lower() + ) + if isinstance(json_file["select"]["value"][val], str): + if "." in json_file["select"]["value"][val]: + selects.append( + json_file["select"]["value"][val] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file["select"]["value"][val].lower() + ) + if isinstance(json_file["select"]["value"][val], list): + for i in range(len(json_file["select"]["value"][val])): + if "value" in json_file["select"]["value"][val][i]: + if isinstance( + json_file["select"]["value"][val][i][ + "value" + ], + str, + ): + if ( + "." + in json_file["select"]["value"][val][i][ + "value" + ] + ): + selects.append( + json_file["select"]["value"][val][ + i + ]["value"] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file["select"]["value"][val][ + i + ]["value"].lower() + ) + return set(selects) + + +# Returns SelectionAttributes as a list if there is a where +def select_where(json_file, literal: list[any]): + if literal is None: + literal = [] + + list_tables = [] + if "where" in json_file: + for val1 in json_file["where"]: + if isinstance(json_file["where"][val1], str): + if "." in json_file["where"][val1]: + list_tables.append(json_file["where"][val1].split(".")[1].lower()) + else: + list_tables.append(json_file["where"][val1].lower()) + if isinstance(json_file["where"][val1], dict): + for val2 in json_file["where"][val1]: + if isinstance(val2, str): + if single_select_dict(val2): + list_tables.extend(single_select_dict(val2)) + elif "." in val2: + list_tables.append(val2.split(".")[1].lower()) + else: + list_tables.append(val2.lower()) + for i in range(len(json_file["where"][val1])): + if isinstance(json_file["where"][val1][i], dict): + for val3 in json_file["where"][val1][i]: + if val3 in select_commands: + for val4 in json_file["where"][val1][i][val3]: + if isinstance(val4, str): + if "." in val4: + list_tables.append(val4.split(".")[1].lower()) + else: + list_tables.append(val4.lower()) + if isinstance(json_file["where"][val1], list) and ( + len(json_file["where"][val1]) > 1 + ): + if isinstance(json_file["where"][val1][1], dict): + if single_select_dict(json_file["where"][val1][1]): + list_tables.extend( + single_select_dict(json_file["where"][val1][1]) + ) + if "literal" in json_file["where"][val1][1]: + literal.append(json_file["where"][val1][1]["literal"]) + for elem in json_file["where"][val1]: + if isinstance(elem, str): + if "." in elem: + list_tables.append(elem.split(".")[1].lower()) + else: + list_tables.append(elem.lower()) + for i in range(len(json_file["where"][val1])): + if not isinstance(json_file["where"][val1][i], int): + for elem in json_file["where"][val1][i]: + if elem in equals: + if isinstance(json_file["where"][val1][i][elem], list): + for j in range(len(json_file["where"][val1][i][elem])): + if isinstance( + json_file["where"][val1][i][elem][j], str + ): + if "." in json_file["where"][val1][i][elem][j]: + list_tables.append( + json_file["where"][val1][i][elem][ + j + ].split(".")[1] + ) + else: + list_tables.append( + json_file["where"][val1][i][elem][j] + ) + if isinstance( + json_file["where"][val1][i][elem][j], dict + ): + for elem1 in json_file["where"][val1][i][elem][ + j + ]: + if elem1 in select_commands: + if isinstance( + json_file["where"][val1][i][elem][ + j + ][elem1], + str, + ): + if ( + "." + in json_file["where"][val1][i][ + elem + ][j][elem1] + ): + list_tables.append( + json_file["where"][val1][i][ + elem + ][j][elem1].split(".")[1] + ) + else: + list_tables.append( + json_file["where"][val1][i][ + elem + ][j][elem1] + ) + if elem1 == "literal": + literal.append( + json_file["where"][val1][i][elem][ + j + ][elem1] + ) + list_tables.extend( + single_select_dict( + json_file["where"][val1][i][elem][j] + ) + ) + if elem == "where": + list_tables.extend( + select_where(json_file["where"][val1][i], literal) + ) + return set(list_tables) diff --git a/modules/fbs-sql-checker/api/comparator/__init__.py b/modules/fbs-sql-checker/api/comparator/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/modules/fbs-sql-checker/api/comparator/comparator.py b/modules/fbs-sql-checker/api/comparator/comparator.py new file mode 100644 index 000000000..ae440e466 --- /dev/null +++ b/modules/fbs-sql-checker/api/comparator/comparator.py @@ -0,0 +1,10 @@ +from abc import ABCMeta, abstractmethod + +from typeguard import typechecked + + +@typechecked +class Comparator(metaclass=ABCMeta): + @abstractmethod + def compare(self, solution: str, submission: str): + pass diff --git a/modules/fbs-sql-checker/api/comparator/sqlparse_comparator.py b/modules/fbs-sql-checker/api/comparator/sqlparse_comparator.py new file mode 100644 index 000000000..1c20d3af9 --- /dev/null +++ b/modules/fbs-sql-checker/api/comparator/sqlparse_comparator.py @@ -0,0 +1,229 @@ +from dataclasses import dataclass + +import sqlparse +from sqlparse.tokens import Whitespace +from typeguard import typechecked + +from api.comparator.comparator import Comparator + + +@dataclass +@typechecked +class Error: + expected: str + got: str + trace: list[str] + + @classmethod + def from_db_dict(cls, db_dict): + return cls( + expected=db_dict["expected"], + got=db_dict["got"], + trace=db_dict["trace"], + ) + + def to_db_dict(self): + return {"expected": self.expected, "got": self.got, "trace": self.trace} + + +@typechecked +class SqlparseComparator(Comparator): + def compare(self, solution: str, submission: str) -> list[Error]: + solution_parsed = sqlparse.parse(self._preprocess(solution)) + dfs = SqlParserDfs() + dfs.visit(solution_parsed) + cv = SqlParserCoVisitor(dfs.dfs) + submission_parsed = sqlparse.parse(self._preprocess(submission)) + cv.visit(submission_parsed) + return cv.errors + + def _preprocess(self, query: str) -> str: + return query.replace("\n", " ") + + +class SqlParseVisitor: + def __init__(self): + super().__init__() + self.parent_stack: list[sqlparse.tokens.Token] = [] + self._visitors = { + sqlparse.sql.Statement: self.visit_statement, + sqlparse.sql.Where: self.visit_where, + sqlparse.sql.IdentifierList: self.visit_identifiers, + sqlparse.sql.Identifier: self.visit_identifier, + sqlparse.sql.Comparison: self.visit_comparison, + sqlparse.sql.Function: self.visit_function, + sqlparse.sql.Parenthesis: self.visit_parenthesis, + sqlparse.sql.Operation: self.visit_operation, + sqlparse.sql.TypedLiteral: self.visit_typed_literal, + } + + def recursive_visit(self, token: sqlparse.sql.TokenList): + self.parent_stack.append(token) + self.visit(token.tokens) + self.parent_stack.pop() + + def visit_statement(self, token: sqlparse.sql.Statement): + self.recursive_visit(token) + + def visit_where(self, token: sqlparse.sql.Where): + self.recursive_visit(token) + + def visit_identifier(self, token: sqlparse.sql.Where): + self.recursive_visit(token) + + def visit_identifiers(self, token: sqlparse.sql.Where): + self.recursive_visit(token) + + def visit_comparison(self, token: sqlparse.sql.Where): + self.recursive_visit(token) + + def visit_function(self, token: sqlparse.sql.Function): + self.recursive_visit(token) + + def visit_parenthesis(self, token: sqlparse.sql.Parenthesis): + self.recursive_visit(token) + + def visit_operation(self, token: sqlparse.sql.Operation): + self.recursive_visit(token) + + def visit_literal(self, token: sqlparse.tokens.Token): + pass + + def visit_typed_literal(self, token: sqlparse.sql.TypedLiteral): + self.recursive_visit(token) + + def visit(self, tokens: list[sqlparse.sql.Token]): + for token in tokens: + if token.ttype is not None: + self.visit_literal(token) + elif token.__class__ in self._visitors: + self._visitors[token.__class__](token) + else: + raise ValueError("unhandled token", token) + + def trace_to_str_list(self) -> list[str]: + return [self._token_to_str(entry) for entry in self.parent_stack] + + +class SqlParserDfs(SqlParseVisitor): + def __init__(self): + super().__init__() + self.dfs = [] + + def recursive_visit(self, token: sqlparse.sql.TokenList): + self.dfs.append((token, len(self.parent_stack))) + super().recursive_visit(token) + + def visit_literal(self, token: sqlparse.tokens.Token): + if token.ttype != Whitespace: + self.dfs.append((token, len(self.parent_stack))) + super().visit_literal(token) + + +class SqlParserCoVisitor(SqlParseVisitor): + def __init__(self, solution, message_overrides=None, token_names_overrides=None): + super().__init__() + self._solution = solution + self._i = 0 + self._messages = { + "end_of_query": "End of query", + "end_of_token": "End of token", + } | (message_overrides or {}) + self._token_names_overrides = { + "IdentifierList": "Select Attributes", + "TypedLiteral": "Interval", + } | (token_names_overrides or {}) + self._error_depth = None + self.errors = [] + + def visit(self, tokens: list[sqlparse.sql.Token]): + super().visit(tokens) + if len(self.parent_stack) == 0 and self._i < len(self._solution): + should, _ = self._solution[self._i] + self.errors.append( + Error( + self._token_to_str(should), + self._messages["end_of_query"], + [self._token_to_str(tokens[0])], + ) + ) + + def _get_should(self): + index = self._i + if index >= len(self._solution): + return None, 0 + self._i += 1 + return self._solution[index] + + def _should_compare(self, token, comparator) -> bool: + current_depth = len(self.parent_stack) + should, should_depth = self._get_should() + end_of_token_error = False + + while should_depth > current_depth: + end_of_token_error = True + should, should_depth = self._get_should() + + if self._error_depth is not None and should_depth >= self._error_depth: + return False + elif self._error_depth is not None: + self._error_depth = None + end_of_token_error = False + + if end_of_token_error: + self.errors.append( + Error( + self._token_to_str(token), + self._messages["end_of_token"], + self.trace_to_str_list(), + ) + ) + + if should is None: + self.errors.append( + Error( + self._messages["end_of_query"], + self._token_to_str(token), + self.trace_to_str_list(), + ) + ) + elif should_depth < len(self.parent_stack): + self.errors.append( + Error( + self._messages["end_of_token"], + self._token_to_str(token), + self.trace_to_str_list(), + ) + ) + self._i -= 1 + elif not comparator(token, should): + self.errors.append( + Error( + self._token_to_str(should), self._token_to_str(token), self.trace_to_str_list() + ) + ) + else: + return True + self._error_depth = current_depth + return False + + def recursive_visit(self, token: sqlparse.sql.Statement): + self._should_compare( + token, + lambda is_token, should_token: is_token.__class__ == should_token.__class__, + ) + super().recursive_visit(token) + + def visit_literal(self, token: sqlparse.tokens.Token): + if token.ttype != Whitespace: + self._should_compare( + token, + lambda is_token, should_token: is_token.value == should_token.value, + ) + super().visit_literal(token) + + def _map_token_name(self, token_name: str) -> str: + return self._token_names_overrides.get(token_name, token_name) + + def _token_to_str(self, token: sqlparse.tokens.Token) -> str: + return self._map_token_name(token.__class__.__name__) if token.ttype is None else repr(token.value) diff --git a/modules/fbs-sql-checker/api/comparator/sqlparse_comparator_test.py b/modules/fbs-sql-checker/api/comparator/sqlparse_comparator_test.py new file mode 100644 index 000000000..93ef7cbdd --- /dev/null +++ b/modules/fbs-sql-checker/api/comparator/sqlparse_comparator_test.py @@ -0,0 +1,148 @@ +import unittest + +from typeguard import typechecked + +from api.comparator.sqlparse_comparator import SqlparseComparator + + +@typechecked +class ComparatorTest(unittest.TestCase): + def test_compare_simple(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username ILIKE 'test%'", + "SELECT username, email, password FROM users WHERE username ILIKE 'test%'", + ) + assert len(errors) == 0 + + def test_compare_shorter(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username ILIKE 'test%'", + "SELECT username, email, password FROM users", + ) + assert len(errors) == 1 + + def test_compare_shorter_swap(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users", + "SELECT username, email, password FROM users WHERE username ILIKE 'test%'", + ) + assert len(errors) == 7 + + def test_compare_to_long_token(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username ILIKE 'test%'", + "SELECT username, email, password, registration_date FROM users WHERE username ILIKE 'test%'", + ) + assert len(errors) == 3 + + def test_compare_to_short_token(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username ILIKE 'test%'", + "SELECT username, email FROM users WHERE username ILIKE 'test%'", + ) + assert len(errors) == 1 + + def test_compare_with_and(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username ILIKE 'test%' AND email ILIKE '%gmail.com'", + "SELECT username, email, password FROM users WHERE email ILIKE '%gmail.com' AND username ILIKE 'test%'", + ) + assert len(errors) != 0 + + def test_compare_with_join(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users JOIN post ON users.id = post.user_id", + "SELECT username, email, password FROM users LEFT JOIN post ON users.id = post.user_id", + ) + assert len(errors) == 1 + + def test_compare_with_sub_query(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username NOT IN (SELECT username FROM blocked_users)", + "SELECT username, email, password FROM users WHERE username IN (SELECT username FROM blocked_users)", + ) + assert len(errors) == 1 + + def test_compare_with_in_sub_query(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, email, password FROM users WHERE username IN (SELECT username FROM blocked_users)", + "SELECT username, email, password FROM users WHERE username IN (SELECT username FROM followed_users)", + ) + assert len(errors) == 1 + + def test_compare_with_aggregate(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT registration_day, COUNT(DISTINCT username) FROM users GROUP BY registration_day", + "SELECT registration_day, COUNT(username) FROM users GROUP BY registration_day", + ) + assert len(errors) == 1 + + def test_compare_deep(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT email FROM users WHERE username IN (SELECT username FROM users WHERE registration_date > (SELECT AVG(registraion_date) FROM users))", + "SELECT email FROM users WHERE username IN (SELECT username FROM users WHERE registration_date > (SELECT MIN(registraion_date) FROM users))", + ) + assert len(errors) == 1 + + def test_compare_much_error(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT email FROM users WHERE username IN (SELECT username FROM users WHERE registration_date > (SELECT AVG(registraion_date) FROM users))", + "SELECT email FROM users WHERE username IN (SELECT email FROM users WHERE registration_date < (SELECT MAX(registraion_date) FROM users))", + ) + assert len(errors) == 2 + + def test_compare_identifier_list(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username, password FROM users", + "SELECT FROM users", + ) + assert len(errors) == 2 + assert errors[0].expected == "Select Attributes" + + def test_very_complex_query(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT monat, AVG(tage_bis_erstes_gebot) AS durchschnittliche_tage FROM ( SELECT EXTRACT(MONTH FROM Registriert_am) AS monat, EXTRACT(DAY FROM (MIN(Geboten_am) - Registriert_am)) AS tage_bis_erstes_gebot FROM Gebot g JOIN Kunde k ON g.Bieter = k.KNr GROUP BY Bieter, Registriert_am ) AS tage GROUP BY monat ORDER BY monat;", + "SELECT monat, AVG(tage_bis_erstes_gebot) AS durchschnittliche_tage FROM ( SELECT EXTRACT(MONTH FROM Registriert_am) AS monat, EXTRACT(DAY FROM (MIN(Geboten_am) - Registriert_am)) AS tage_bis_erstes_gebot FROM Gebot g JOIN Kunde k ON g.Bieter = k.KNr GROUP BY Bieter, Registriert_am ) AS tage GROUP BY monat;" + ) + assert len(errors) == 2 + + def test_with_with(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "WITH (SELECT username FROM blocked_users) AS bu SELECT * FROM bu", + "WITH (SELECT email FROM blocked_users) AS bu SELECT * FROM bu" + ) + assert len(errors) == 1 + + def test_very_very_complex_query(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT g.Auktion, g.Bieter, g.Geboten_am, g.Gebotspreis FROM Gebot g JOIN Auktion a ON g.Auktion = a.ANr WHERE g.Geboten_am >= a.Eingestellt_am AND g.Geboten_am <= a.Eingestellt_am + INTERVAL '7 days' AND g.Gebotspreis > COALESCE( ( SELECT MAX(g_prev.Gebotspreis) FROM Gebot g_prev WHERE g_prev.Auktion = g.Auktion AND g_prev.Geboten_am < g.Geboten_am ), a.Startpreis ) ORDER BY g.Auktion, g.Geboten_am;", + "SELECT g.Auktion, g.Bieter, g.Geboten_am, g.Gebotspreis FROM Gebot g JOIN Auktion a ON g.Auktion = a.ANr WHERE g.Geboten_am >= a.Eingestellt_am AND g.Geboten_am <= a.Eingestellt_am + INTERVAL '7 days' AND g.Gebotspreis > COALESCE( ( SELECT MAX(g_prev.Gebotspreis) FROM Gebot g_prev WHERE g_prev.Auktion = g.Auktion AND g_prev.Geboten_am < g.Geboten_am ), a.Startpreis ) ORDER BY g.Geboten_am, g.Auktion;" + ) + assert len(errors) == 2 + + def test_not_null(self): + comparator = SqlparseComparator() + errors = comparator.compare( + "SELECT username FROM user WHERE banned_at IS NULL", + "SELECT username FROM user WHERE banned_at IS NOT NULL" + ) + assert len(errors) == 1 + +if __name__ == "__main__": + unittest.main() diff --git a/modules/fbs-sql-checker/api/data_store.py b/modules/fbs-sql-checker/api/data_store.py new file mode 100644 index 000000000..7cc20c45c --- /dev/null +++ b/modules/fbs-sql-checker/api/data_store.py @@ -0,0 +1,50 @@ +from abc import ABCMeta, abstractmethod +from typing import Self + +from pymongo import MongoClient + + +class DataStore(metaclass=ABCMeta): + @abstractmethod + def store(self, collection: str, obj: any): + pass + + @abstractmethod + def upsert(self, collection: str, query: any, update: any): + pass + + @abstractmethod + def query(self, collection: str, *query): + pass + + +class AbstractMongoStore(DataStore, metaclass=ABCMeta): + @classmethod + def connect(cls, uri: str) -> Self: + return MongoDataStore(MongoClient(uri)) + + def store(self, collection: str, obj: any): + return self.get_collection(collection).insert_one(obj) + + def query(self, collection, *query): + return self.get_collection(collection).find(*query) + + def upsert(self, collection: str, query: any, update: any): + return self.get_collection(collection).update_one(query, update, upsert=True) + + @abstractmethod + def get_collection(self, collection): + pass + + +class MongoDataStore(AbstractMongoStore): + def __init__(self, client: MongoClient): + self._client = client + self.db = self._client.get_default_database() + + @classmethod + def connect(cls, uri: str) -> Self: + return cls(MongoClient(uri)) + + def get_collection(self, collection): + return self.db[collection] diff --git a/modules/fbs-sql-checker/api/datatypes.py b/modules/fbs-sql-checker/api/datatypes.py new file mode 100644 index 000000000..25f294bd1 --- /dev/null +++ b/modules/fbs-sql-checker/api/datatypes.py @@ -0,0 +1,195 @@ +from dataclasses import dataclass, field +from datetime import datetime, UTC +from typing import Optional, Self +from uuid import uuid4 + +from typeguard import typechecked + +from api.comparator.sqlparse_comparator import Error + + +@dataclass +@typechecked +class Query: + course_id: int + test_sql: str + task_nr: int + is_sol: bool + tables_right: bool + sel_attributes_right: bool + pro_attributes_right: bool + strings_right: bool + order_by_right: bool + group_by_right: bool + joins_right: bool + having_right: bool + wildcards: bool + time: int + closest_solution: int + min_distance: int + closest_id: int + + +@dataclass +@typechecked +class Submission: + submission: str + is_solution: bool = False + passed: bool = False + user_id: Optional[int] = None + attempt: Optional[int] = None + course_id: Optional[int] = None + task_id: Optional[int] = None + submission_id: Optional[int] = None + + @classmethod + def new_solution( + cls, query: str, course_id: Optional[int] = None, task_id: Optional[int] = None + ): + return cls( + query, is_solution=True, passed=True, course_id=course_id, task_id=task_id + ) + + @classmethod + def from_dict(cls, data: dict): + return cls( + submission=data["submission"], + is_solution=data.get("isSol", False), + passed=data.get("passed", False), + user_id=data.get("userId", None), + attempt=data.get("attempt", None), + course_id=data.get("cid", None), + task_id=data.get("tid", None), + submission_id=data.get("sid", None), + ) + + +@dataclass +@typechecked +class Result: + uuid: str = field(default_factory=lambda: str(uuid4())) + course_id: int = 0 + task_id: int = 0 + user_id: int = 0 + submission_id: int = 0 + attempt: int = 0 + statement: str = "" + time: str = "" + passed: bool = False + is_solution: bool = False + parsable: bool = False + closest_solution: str = "" + min_distance: int = 0 + closest_id: str = "" + version: Optional[str] = None + + @classmethod + def from_submission(cls, submission: Submission) -> Self: + return cls( + course_id=submission.course_id, + task_id=submission.task_id, + user_id=submission.user_id, + submission_id=submission.submission_id, + attempt=submission.attempt, + statement=submission.submission, + time=datetime.now(UTC).isoformat(), + is_solution=submission.is_solution and submission.passed, + ) + + @classmethod + def from_db_dict(cls, db_dict): + return cls( + uuid=db_dict["_id"], + course_id=db_dict["courseId"], + task_id=db_dict["taskId"], + user_id=db_dict["userId"], + submission_id=db_dict["submissionId"], + attempt=db_dict["attempt"], + statement=db_dict["statement"], + passed=db_dict["passed"], + parsable=db_dict["parsable"], + time=db_dict["time"], + closest_solution=db_dict["usedSolutionId"], + min_distance=db_dict["distance"], + version=db_dict["version"], + ) + + def to_db_dict(self): + return { + "_id": self.uuid, + "courseId": self.course_id, + "taskId": self.task_id, + "userId": self.user_id, + "submissionId": self.submission_id, + "attempt": self.attempt, + "statement": self.statement, + "passed": self.passed, + "parsable": self.parsable, + "isSolution": self.is_solution, + "time": self.time, + "usedSolutionId": self.closest_solution, + "distance": self.min_distance, + "version": self.version, + } + + +@dataclass +@typechecked +class LegacyResult(Result): + task_nr: int = 0 + tables_right: bool = False + sel_attributes_right: bool = False + pro_attributes_right: bool = False + strings_right: bool = False + order_by_right: bool = False + group_by_right: bool = False + joins_right: bool = False + having_right: bool = False + wildcards: bool = False + + def to_db_dict(self): + return { + "id": str(self.closest_id), + "courseId": self.course_id, # + "taskNumber": self.task_id, # + "submissionId": self.submission_id, + "statement": self.statement, + "queryRight": self.passed, + "parsable": self.parsable, + "isSolution": self.is_solution, + "tablesRight": self.tables_right, + "selAttributesRight": self.sel_attributes_right, + "proAttributesRight": self.pro_attributes_right, + "stringsRight": self.strings_right, + "userId": self.user_id, + "attempt": self.attempt, + "orderByRight": self.order_by_right, + "groupByRight": self.group_by_right, + "joinsRight": self.joins_right, + "havingRight": self.having_right, + "wildcards": self.wildcards, + "time": self.time, + "solutionID": self.closest_solution, + "distance": self.min_distance, + } + + +@dataclass +@typechecked +class ResultV2(Result): + version = "v2" + errors: list[Error] = field(default_factory=lambda: []) + + def to_db_dict(self): + return super().to_db_dict() | { + "version": "v2", + "errors": [error.to_db_dict() for error in self.errors], + } + + @classmethod + def from_db_dict(cls, db_dict): + return cls( + **super().from_db_dict(db_dict).__dict__, + version=db_dict["version"], + errors=[Error.from_db_dict(err) for err in db_dict["errors"]], + ) diff --git a/modules/fbs-sql-checker/api/debug.py b/modules/fbs-sql-checker/api/debug.py old mode 100644 new mode 100755 index f8f2b3c3e..18624232f --- a/modules/fbs-sql-checker/api/debug.py +++ b/modules/fbs-sql-checker/api/debug.py @@ -1,6 +1,6 @@ import random # pylint: disable=W0611 import string # pylint: disable=W0611 -from json_creator import parse_single_stat_upload_db +from query_processor import parse_single_stat_upload_db CLIENT = ( diff --git a/modules/fbs-sql-checker/api/distance/.gitignore b/modules/fbs-sql-checker/api/distance/.gitignore new file mode 100755 index 000000000..27c2f3b32 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/.gitignore @@ -0,0 +1,3 @@ +__pycache__/ +.env +log/ \ No newline at end of file diff --git a/modules/fbs-sql-checker/api/distance/__init__.py b/modules/fbs-sql-checker/api/distance/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/modules/fbs-sql-checker/api/distance/attribute_check.py b/modules/fbs-sql-checker/api/distance/attribute_check.py new file mode 100755 index 000000000..027e1a44d --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/attribute_check.py @@ -0,0 +1,180 @@ +import sqlparse +from . import constants as c +from . import attribute_distance as att_dist +from . import format as f +from . import result_log as log + + +def extract_attributes(ref, query): + + ref_pro_att: list[str] = [] + query_pro_att: list[str] = [] + + ref_cmd_list: list[str] = [] + query_cmd_list: list[str] = [] + + ref_distinct_list: list[str] = [] + query_distinct_list: list[str] = [] + + ref_map: dict[str, dict[str, str]] = {} + query_map: dict[str, dict[str, str]] = {} + _token_iteration(ref, ref_map, ref_pro_att, ref_cmd_list, ref_distinct_list) + _token_iteration( + query, query_map, query_pro_att, query_cmd_list, query_distinct_list + ) + + log.write_to_log( + f"attribute aliases: reference: {ref_map}; query: {query_map}\n\n" + f"attributes before order: reference: {ref_pro_att}; query: {query_pro_att}\n" + ) + + attribute_distance = att_dist.get_attributes_distance(ref_pro_att, query_pro_att) + + log.write_to_log( + f"attributes after order: reference: {ref_pro_att}; query: {query_pro_att}\n" + ) + log.write_to_log( + f"command list: reference: {ref_cmd_list}; query: {query_cmd_list}\n" + ) + + command_distance = att_dist.get_command_distance(ref_cmd_list, query_cmd_list) + + keyword_distance = att_dist.get_keyword_distance( + ref_distinct_list, query_distinct_list + ) + + log.write_to_log( + f"distinct list: reference: {ref_distinct_list}; query {query_distinct_list}\n" + ) + + log.write_to_log( + f"Distance: attributes = {attribute_distance}, commands = {command_distance}, keywords = {keyword_distance}\n" + ) + + return attribute_distance + command_distance + keyword_distance + + +def _token_iteration(tokens, map_dict, pro_att_list, cmd_list, distinct_list): + for i, token in enumerate(tokens): + if isinstance(token, sqlparse.sql.Token): + # bypass token if it represents a whitespace or a newline + if token.ttype in [sqlparse.tokens.Whitespace, sqlparse.tokens.Newline]: + continue + # check if token represents distinct + if token.ttype == sqlparse.tokens.Keyword and token.value == c.DISTINCT: + _extract_keyword(tokens[i + 2], distinct_list) + # wildcard represents *. Iteration breaks after this step + if token.ttype == sqlparse.tokens.Wildcard: + pro_att_list.append(token.value) + break + # iterate through the identifier list to extract each individual attribute + if isinstance(token, sqlparse.sql.IdentifierList): + for t in token.get_identifiers(): + _extract_att_and_cmds( + t, map_dict, pro_att_list, cmd_list, distinct_list + ) + # case if attributes are wrapped inside Parenthesis + if isinstance(token, sqlparse.sql.Parenthesis): + _extract_parenthesis( + token, map_dict, pro_att_list, cmd_list, distinct_list + ) + # extract attributes based on their type + if isinstance( + token, + ( + sqlparse.sql.Identifier, + sqlparse.sql.Function, + sqlparse.sql.Operation, + ), + ): + _extract_att_and_cmds( + token, map_dict, pro_att_list, cmd_list, distinct_list + ) + # break iteration if it reaches the from clause + if token.ttype == sqlparse.tokens.Keyword and token.value == c.FROM: + break + + +def _extract_att_and_cmds(token, map_dict, pro_att_list, cmd_list, distinct_list): + if isinstance(token, sqlparse.sql.Operation): + pro_att_list.append(f.format_whitespace(token.value)) + if isinstance(token, sqlparse.sql.Function): + _extract_functions(token, map_dict, pro_att_list, cmd_list, distinct_list) + if isinstance(token, sqlparse.sql.Identifier): + _extract_identifier(token, map_dict, pro_att_list, cmd_list, distinct_list) + if isinstance(token, sqlparse.sql.Parenthesis): + _extract_parenthesis(token, map_dict, pro_att_list, cmd_list, distinct_list) + + +def _extract_parenthesis(token, map_dict, pro_att_list, cmd_list, distinct_list): + for t in token.tokens: + if isinstance(t, sqlparse.sql.IdentifierList): + for ident in t.get_identifiers(): + _extract_att_and_cmds( + ident, map_dict, pro_att_list, cmd_list, distinct_list + ) + else: + _extract_att_and_cmds(t, map_dict, pro_att_list, cmd_list, distinct_list) + + +def _extract_identifier(token, map_dict, pro_att_list, cmd_list, distinct_list): + # commands with aliases are considered as identifier and can only be extracted this way + if str(token.get_real_name()) in c.SELECT_CMDS: + # save alias + _extract_alias(token, map_dict) + # save used command + cmd_list.append(f.format_whitespace(str(token.get_real_name()))) + # save attribute after formatting + pro_att_list.append( + f.format_whitespace( + f.format_alias(f.format_distinct((f.format_command(token)))) + ) + ) + # check for distinct keyword and save the attributes used after it + if c.DISTINCT in token.value: + _extract_keyword(f.format_command(token), distinct_list) + else: + pro_att_list.append(f.format_distinct((f.format_alias(token.value)))) + _extract_alias(token, map_dict) + + +def _extract_functions(token, map_dict, pro_att_list, cmd_list, distinct_list): + _extract_alias(token, map_dict) + if isinstance(token, sqlparse.sql.Function): + # save command used to list + cmd_list.append(str(token.get_real_name())) + # save attribute or attributes used inside the command + for part in token.tokens[1]: + if isinstance(part, sqlparse.sql.Identifier): + pro_att_list.append(part.value) + elif isinstance(part, sqlparse.sql.IdentifierList): + pro_att_list.append(f.format_whitespace(part.value)) + # check for distinct keyword and save the attributes used after it + if c.DISTINCT in token.value: + _extract_keyword(f.format_command(token), distinct_list) + + +def _extract_alias(ident: sqlparse.sql.Identifier, map_dict): + if ident.has_alias(): + updated_ident = f.format_alias((ident.value)) + _add_to_map(map_dict, c.ALIAS, updated_ident, ident.get_alias()) + + +def _extract_keyword(ident, distinct_list): + if isinstance(ident, sqlparse.sql.IdentifierList): + ident_list = list[sqlparse.sql.Identifier](ident.get_identifiers()) + # get all the identifiers that are referred to the distinct keyword. (alias gets formatted and removed) + result = ", ".join( + f.format_whitespace((f.format_alias(i.value.lower()))) for i in ident_list + ) + distinct_list.append(result) + else: + # remove trailing alias or distinct keyword to add only the attribute to the map + updated_value = f.format_distinct((f.format_alias(str(ident)))) + distinct_list.append(updated_value) + + +def _add_to_map(map_dict, key, inner_key, value): + if key not in map_dict: + map_dict[key] = {} + map_dict[key][inner_key] = value diff --git a/modules/fbs-sql-checker/api/distance/attribute_distance.py b/modules/fbs-sql-checker/api/distance/attribute_distance.py new file mode 100755 index 000000000..1f92a7c47 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/attribute_distance.py @@ -0,0 +1,92 @@ +import re +import uuid +from . import constants as c +from . import format as f + + +def get_attributes_distance(ref: list[str], query: list[str]): + moves = 0 + # distance is equal to 0 if * is used + if "*" in ref: + return 0 + # check for order of attributes and add RMU + if sorted(ref) == sorted(query): + for ref_item in ref: + if query[ref.index(ref_item)] != ref_item: + moves += c.ORDER_MULT + query.remove(ref_item) + query.insert(ref.index(ref_item), ref_item) + # check for missing elements and add the OMU if true + elif len(ref) != len(query): + moves += abs(len(ref) - len(query)) * c.OBJECT_MULT + else: + # compare each element used, if discrepency was found, OMU is added + for ref_item, query_item in zip(sorted(ref), sorted(query)): + if ref_item != query_item: + moves += c.OBJECT_MULT + # get operation distance + op_dist = _get_operation_distance(ref, query) + moves += round(op_dist) + return moves + + +def get_command_distance(ref: list[str], query: list[str]): + moves = 0 + # check for number of commmands used and add OMU when there is a difference + if len(ref) != len(query): + moves += abs(len(ref) - len(query)) * c.OBJECT_MULT + # check for each element and if there difference SMU is added + elif set(ref) != set(query): + for ref_item, query_item in zip(sorted(ref), sorted(query)): + if ref_item != query_item: + moves += c.STRUCT_MULT + return moves + + +def get_keyword_distance(ref_list: list, query_list: list): + moves = 0 + if set(ref_list) != set(query_list): + moves += c.OBJECT_MULT + # print("distinct", moves) + return moves + + +def _get_operation_distance(ref_list: list[str], query_list: list[str]): + ref_op_list = [] + query_op_list = [] + + # check for mathematic operations inside the attribute list using the regex pattern + # add them to the list of operation for comparison if found + for exp in ref_list: + if re.findall(c.MATH_EXP_REGEX, exp): + ref_op_list.append(exp) + for exp in query_list: + if re.findall(c.MATH_EXP_REGEX, exp): + query_op_list.append(exp) + # print("ref op , qur op", ref_op_list, query_op_list) + return _calculate_expression_similarity(ref_op_list, query_op_list) + + +# Jaccard index may not be the best method to measure the distance of two mathematical expressions +def _calculate_expression_similarity(ref_exp: list[str], query_exp: list[str]): + operation_map: dict[str, str, str] = {} + diff = 0 + for ref_item, query_item in zip(ref_exp, query_exp): + # Parenthesis formatting + ref_set = set(f.format_parenthesis(ref_item)) + query_set = set(f.format_parenthesis(query_item)) + intersection = len(ref_set.intersection(query_set)) + union = len(ref_set.union(query_set)) + if union != 0: + # Jaccard Similarity Coefficient: 1 - J(A,B) + similarity_coeffiecient = 1 - (intersection / union) + _add_to_op_map(operation_map, ref_item, query_item, similarity_coeffiecient) + diff += similarity_coeffiecient * c.STRUCT_MULT + return diff + + +def _add_to_op_map(op_map, ref, query, sim): + generated_uuid = str(uuid.uuid4()) + short_id = generated_uuid[:4] # Take the first 8 characters as the short ID + new_entry_key = f"{short_id}" + op_map[new_entry_key] = {"ref": ref, "query": query, "similarity": sim} diff --git a/modules/fbs-sql-checker/api/distance/constants.py b/modules/fbs-sql-checker/api/distance/constants.py new file mode 100755 index 000000000..8f3348972 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/constants.py @@ -0,0 +1,71 @@ +# BASIC SQL KEYWORDS +DISTINCT = "distinct" +FROM = "from" +TABLE = "table" +ALIAS = "alias" +ON = "on" +WHERE = "where" +GROUP_BY = "group by" +HAVING = "having" +ORDER_BY = "order by" +DESC = r"(?i)desc" +ASC = r"(?i)asc" +BETWEEN = "between" +LIKE = "like" +NOT = "not" +IN = "in" +EXIST = "exist" +SELECT = "select" +AND = "and" +OR = "or" +NOT = "not" + +# SELECT COMMANDS +SELECT_CMDS = [ + "sum", + "count", + "round", + "avg", + "max", + "min", + "abs", + "year", + "now", + "upper", + "lower", + "length", + "ceil", + "floor", + "power", + "convert", + "time_to_sec", +] + +# JOIN TYPES +JOIN_TYPES = [ + "inner join", + "left join", + "right join", + "full join", + "self join", + "natural join", + "join", +] + +# REGULAR EXPRESSIONS +ALIAS_REGEX = r"\sas\s+\"(.+?)\"|\sas\s+'(.+?)'|\sas\s+(\w+)" +MATH_EXP_REGEX = r"[\d()+\-*\/]" +EQ_COMP_REGEX = r"\s*(\w+|'\w+')\s*=\s*(\w+|'\w+')\s*" +FORMATTING_REGEX = r"[a-z]*\.|[\'\"\_\-\\\`]" +PARENTHESIS_REGEX = r"[()]" +BETWEEN_REGEX = r"([^\s]+)\s+between\s+([^\s]+)\s+and\s+([^\s]+)" +SYMBOL_REGEX = r"(\b\w+\b)\s*(?:>|<)\s*(\b\w+\b)" + +# MULTIPLIERS +ORDER_MULT = 5 +STRUCT_MULT = 25 +OBJECT_MULT = 50 + +# LOG +FOLDER_PATH = "modules/fbs-sql-checker/api/distance/log" +LOG_PATH = "modules/fbs-sql-checker/api/distance/log/distance.txt" diff --git a/modules/fbs-sql-checker/api/distance/db_connection.py b/modules/fbs-sql-checker/api/distance/db_connection.py new file mode 100755 index 000000000..73d64a516 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/db_connection.py @@ -0,0 +1,83 @@ +import os +import psycopg2 +from dotenv import load_dotenv + + +def setup_db(att_list): + connection = None + + load_dotenv() + hostname = os.getenv("HOSTNAME") + db_name = os.getenv("DB_NAME") + db_username = os.getenv("DB_USERNAME") + db_password = os.getenv("PASSWORD") + port_id = os.getenv("PORT_ID") + + with psycopg2.connect( + host=hostname, + dbname=db_name, + user=db_username, + password=db_password, + port=port_id, + ) as connection: + cursor = connection.cursor() + _create_db(att_list, cursor) + _populate_db(att_list, cursor) + connection.commit() + return connection + + +def _create_db(att_list, cursor): + # Iterate over the attribute list + for i, _att in enumerate(att_list): + # Generate SQL script to create a table for each attribute + # Tables are named A, B, C, etc., with unique constraints on column 'x' + sql_script = ( + f"CREATE TABLE IF NOT EXISTS {chr(65 + i)} " + f"(x INTEGER, CONSTRAINT {chr(97 + i)}_unique_x UNIQUE (x));" + ) + cursor.execute(sql_script) # Execute the SQL script + + +def _populate_db(att_list, cursor): + bits = len(att_list) # Determine the number of bits based on the length of att_list + # Loop through all possible combinations based on the number of bits + for i in range(2**bits): + binary_number = format( + i, f"0{bits}b" + ) # Convert the current number to a binary string + decimal_number = int( + binary_number, 2 + ) # Convert the binary string to a decimal number + offset = ( + -1 + ) # Initialize offset for iterating through the binary number from right to left + # Iterate through each bit in the binary number + for j in range(len(binary_number)): + if binary_number[offset] == "1": + # Generate a SQL query to select from table corresponding to the current bit + select_query = ( + f"SELECT * FROM {chr(65 + j)} WHERE x = {decimal_number};" + ) + cursor.execute(select_query) # Execute the select query + result = cursor.fetchone() # Fetch the result of the query + # If no result is found, insert a new record into the table + if result is None: + insert_query = ( + f"INSERT INTO {chr(65 + j)} (x) VALUES ({decimal_number});" + ) + cursor.execute(insert_query) # Execute the insert query + offset -= 1 # Move to the next bit in the binary number + + +def execute_query(query, connection: psycopg2): + cursor = connection.cursor() + # Execute the SQL query using the cursor + cursor.execute(query) + # Fetch all the rows returned by the executed query + result = cursor.fetchall() + # Convert the result into a set of frozensets for each row + # This ensures each row is a unique, immutable set of elements + res_set = set(frozenset(s) for s in result) + connection.commit() + return res_set diff --git a/modules/fbs-sql-checker/api/distance/distance_calc.py b/modules/fbs-sql-checker/api/distance/distance_calc.py new file mode 100755 index 000000000..15be2be41 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/distance_calc.py @@ -0,0 +1,55 @@ +import traceback +from abc import ABCMeta, abstractmethod + +import sqlparse +from . import attribute_check as att_check +from . import table_check as tab_check +from . import result_log as log +from . import format as f + + +class DistanceCalculator(metaclass=ABCMeta): + @abstractmethod + def calculate_distance(self, a_query: str, b_query: str) -> float: + pass + + +class GetDistanceCalculator(DistanceCalculator): + def calculate_distance(self, a_query: str, b_query: str) -> float: + return get_distance(a_query, b_query) + + +# first argument is always the solution +def get_distance(ref, query): + try: + formated_ref = f.format_query(ref.lower()) + formated_query = f.format_query(query.lower()) + # query parsing + parsed_ref = _parse_query(formated_ref) + parsed_query = _parse_query(formated_query) + + # distance calculation + attribute_distance = att_check.extract_attributes(parsed_ref, parsed_query) + table_distance = tab_check.extract_tables(parsed_ref, parsed_query) + log.write_to_log(f"reference:\n{ref}\n\nquery:\n{query}\n") + + log.write_to_log( + f"Distance = {attribute_distance + table_distance}\n" + f"------------------------------------------------------------------\n\n" + ) + + return attribute_distance + table_distance + except Exception as e: + print("Error measuring distance", e) + print(traceback.format_exc()) + return -1 + + +def _parse_query(query: str): + try: + formatted_query = sqlparse.format(query, keyword_case="lower") + parsed_query = sqlparse.parse(formatted_query)[0].tokens + except Exception as e: + print(f"ParsingError: {e}") + + return parsed_query diff --git a/modules/fbs-sql-checker/api/distance/docker-compose.yml b/modules/fbs-sql-checker/api/distance/docker-compose.yml new file mode 100755 index 000000000..d37ec4b86 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/docker-compose.yml @@ -0,0 +1,16 @@ +version: '3.8' + +services: + postgres: + image: postgres:latest + environment: + POSTGRES_DB: postgres + POSTGRES_USER: postgres + POSTGRES_PASSWORD: admin + ports: + - "5432:5432" + volumes: + - postgres_data:/var/lib/postgresql/data + +volumes: + postgres_data: diff --git a/modules/fbs-sql-checker/api/distance/equation_checker.py b/modules/fbs-sql-checker/api/distance/equation_checker.py new file mode 100755 index 000000000..48350ee5d --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/equation_checker.py @@ -0,0 +1,19 @@ +import sympy as s +from . import constants as c + + +def check_equation(ref, query): + moves = 0 + # Check if both elements match the equality comparison regex + eq1 = s.simplify(s.sympify(ref)) + eq2 = s.simplify(s.sympify(query)) + try: + if s.Eq(eq1, eq2).canonical: + # Increment the moves counter if they are not equal + return moves + # Return the total number of moves calculated + except Exception: + moves += c.OBJECT_MULT + return moves + + return None # Inserted while lint fixing preserves prefix behavior but might indicate an error diff --git a/modules/fbs-sql-checker/api/distance/format.py b/modules/fbs-sql-checker/api/distance/format.py new file mode 100755 index 000000000..2cf8f0480 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/format.py @@ -0,0 +1,49 @@ +import re +import sqlparse +from . import constants as c + + +def format_alias(ident: str): + # check ident with pattern to get the alias keyword and alias name + regex = re.compile(c.ALIAS_REGEX, re.IGNORECASE) + match = regex.search(ident) + + if match: + # flag is used for case sensitivity + ident = re.sub(c.ALIAS_REGEX, "", ident).strip() + return ident + + +def format_distinct(ident: str): + if c.DISTINCT in ident: + ident = ident.replace(c.DISTINCT, "").strip() + return ident + + +def format_command(ident: sqlparse.sql.Identifier): + # get_real_name() function returns the select command and will get removed along the alias and the 'as' keyword + formatted = ( + ident.value.replace(ident.get_real_name(), "").replace("(", "").replace(")", "") + ) + return formatted + + +# remove database name e.g.: db.id and special characters +def format_query(ident: str): + return re.sub(c.FORMATTING_REGEX, "", ident).strip().lower() + + +def format_parenthesis(ident: str): + return re.sub(c.PARENTHESIS_REGEX, "", ident).strip() + + +def format_whitespace(ident: str): + return ident.replace(" ", "") + + +def format_like(ident: str): + if f"{c.NOT} {c.LIKE}" in ident: + ident = ident.replace(f"{c.NOT} {c.LIKE}", "!=") + elif c.LIKE in ident: + ident = ident.replace(c.LIKE, "=") + return ident diff --git a/modules/fbs-sql-checker/api/distance/result_log.py b/modules/fbs-sql-checker/api/distance/result_log.py new file mode 100755 index 000000000..358f96c9d --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/result_log.py @@ -0,0 +1,20 @@ +import os +from . import constants as c + + +def write_to_log(message: str): + # Create folder if it doesn't exist + try: + if not os.path.exists(c.FOLDER_PATH): + os.makedirs(c.FOLDER_PATH) + except Exception as e: + print(f"FolderError: {e}") + return + + # Write message to the log file + try: + with open(c.LOG_PATH, "a", encoding="utf-8") as log_file: + log_file.write(message + "\n") + except Exception as e: + print(f"FileError: {e}") + log_file.close() diff --git a/modules/fbs-sql-checker/api/distance/table_check.py b/modules/fbs-sql-checker/api/distance/table_check.py new file mode 100755 index 000000000..aea511fc9 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/table_check.py @@ -0,0 +1,479 @@ +import re +import sqlparse +from . import constants as c +from . import table_distance as tab_dist +from . import format as f +from . import result_log as log + + +def extract_tables(ref, query): + ref_tab_name: list[str] = [] + query_tab_name: list[str] = [] + + ref_alias_map: dict[str, str] = {} + query_alias_map: dict[str, str] = {} + + ref_join_list: list[str] = [] + query_join_list: list[str] = [] + + ref_comp_list: list[str] = [] + query_comp_list: list[str] = [] + + ref_order_list: list[str] = [] + query_order_list: list[str] = [] + + ref_group_list: list[str] = [] + query_group_list: list[str] = [] + + ref_having_list: list[str] = [] + query_having_list: list[str] = [] + _token_iteration( + ref, + ref_alias_map, + ref_tab_name, + ref_join_list, + ref_comp_list, + ref_order_list, + ref_group_list, + ref_having_list, + ) + _token_iteration( + query, + query_alias_map, + query_tab_name, + query_join_list, + query_comp_list, + query_order_list, + query_group_list, + query_having_list, + ) + + from_distance = tab_dist.get_from_clause_distance( + ref_tab_name, query_tab_name, ref_join_list, query_join_list + ) + + log.write_to_log( + f"tables used: reference: {ref_tab_name}; query: {query_tab_name}\n" + ) + log.write_to_log( + f"reference table aliases: {ref_alias_map}; query table aliases {query_alias_map}\n" + ) + log.write_to_log( + f"data retrieval clause: reference: {ref_join_list}; query: {query_join_list}\n" + ) + log.write_to_log( + f"comparison equations: reference: {ref_comp_list}; query: {query_comp_list}\n" + ) + log.write_to_log( + f"group by attributes: reference: {ref_group_list}; query: {query_group_list}\n" + ) + log.write_to_log( + f"having attributes: reference: {ref_having_list}; query: {query_having_list}\n" + ) + log.write_to_log( + f"order by attributes: reference: {ref_order_list}; query: {query_order_list}\n" + ) + + comparison_distance = tab_dist.comparison_distance(ref_comp_list, query_comp_list) + + order_distance = tab_dist.group_and_order_by_distance( + ref_order_list, query_order_list + ) + + group_by_distance = tab_dist.group_and_order_by_distance( + ref_group_list, query_group_list + ) + + having_distance = tab_dist.group_and_order_by_distance( + ref_having_list, query_having_list + ) + + log.write_to_log( + f"Distance: table and data retrieval clause = {from_distance}, comparison equations = " + f"{comparison_distance}, group by = {group_by_distance}, having = {having_distance}," + f" order by = {order_distance}\n" + ) + + return ( + from_distance + + comparison_distance + + order_distance + + group_by_distance + + having_distance + ) + + +def _token_iteration( + tokens: sqlparse.sql.Statement, + tab_map: dict, + name_list: list, + join_list: list, + comp_list: list, + order_list: list, + group_list: list, + having_list: list, +): + for i, token in enumerate(tokens): + if isinstance(token, sqlparse.sql.Token): + # Parenthesis check + if isinstance(token, sqlparse.sql.Parenthesis): + _token_iteration( + token, + tab_map, + name_list, + join_list, + comp_list, + order_list, + group_list, + having_list, + ) + # check and extract tables used after the FROM keyword + if token.ttype == sqlparse.tokens.Keyword and token.value == c.FROM: + extracted_from = _extract_from(tokens, i, tab_map, name_list) + for from_token in extracted_from: + # check if FROM contains subquery + if isinstance(from_token, sqlparse.sql.Parenthesis): + # call the method recursively to iterate through the tokens of the subquery + _token_iteration( + from_token.tokens, + tab_map, + name_list, + join_list, + comp_list, + order_list, + group_list, + having_list, + ) + # check and extract the JOIN keywords and tables used after it + if token.ttype == sqlparse.tokens.Keyword and token.value in c.JOIN_TYPES: + extracted_join = _extract_join( + token, tokens, i, tab_map, name_list, join_list + ) + # check if JOIN contains subquery + for subquery in extracted_join: + if isinstance(subquery, sqlparse.sql.Parenthesis): + # call the method recursively to iterate through the tokens of the subquery + _token_iteration( + subquery.tokens, + tab_map, + name_list, + join_list, + comp_list, + order_list, + group_list, + having_list, + ) + + # check and extract the comparison equations after ON condition + if token.ttype == sqlparse.tokens.Keyword and token.value == c.ON: + extracted_on = _extract_on(tokens, i, comp_list) + if extracted_on is not None: + for on_token in extracted_on: + _token_iteration( + on_token, + tab_map, + name_list, + join_list, + comp_list, + order_list, + group_list, + having_list, + ) + # check and extract the WHERE keyword and comparison equations after it + if isinstance(token, sqlparse.sql.Where): + extracted_where = _extract_where(token, comp_list, join_list) + for where_token in extracted_where: + if isinstance(where_token, sqlparse.sql.Parenthesis): + _token_iteration( + where_token.tokens, + tab_map, + name_list, + join_list, + comp_list, + order_list, + group_list, + having_list, + ) + # check and extract attributes and iterate through group by clause + if token.ttype == sqlparse.tokens.Keyword and token.value == c.GROUP_BY: + _extract_group_by(tokens, i, group_list, having_list) + # extract attributes inside order by clause + if token.ttype == sqlparse.tokens.Keyword and token.value == c.ORDER_BY: + _extract_order_by(tokens, i, order_list) + + +def _search_for_subqueries(tokens): + subquery_list = [] + for cur_token in tokens: + if isinstance(cur_token, sqlparse.sql.Parenthesis): + subquery_list.append(cur_token) + return subquery_list + + +def _extract_from(tokens, i, tab_map, name_list): + next_token = tokens[i + 2] # +2 to bypass whitespace token + # List for all subqueries + subquery_list = [] + # if singular table used, append it to list + if isinstance(next_token, sqlparse.sql.Identifier): + _extract_table_elements(next_token, tab_map, name_list) + subquery_list += _search_for_subqueries(next_token.tokens) + + # if multiple tables used, iterate through them and save them to list + elif isinstance(next_token, sqlparse.sql.IdentifierList): + for t in list[sqlparse.sql.Identifier](next_token.get_identifiers()): + _extract_table_elements(t, tab_map, name_list) + subquery_list += _search_for_subqueries(t.tokens) + # return list of subqueries + return subquery_list + + +def _extract_join(token: sqlparse.sql.Token, tokens, i, tab_map, name_list, join_list): + next_token = tokens[i + 2] + join_list.append(token.value) + return _extract_table_elements(next_token, tab_map, name_list) + + +def _extract_on(tokens, i, comp_list): + next_token = tokens[i + 2] # +2 to avoid whitespace + # Check if the next_token is a Comparison object from sqlparse + if isinstance(next_token, sqlparse.sql.Comparison): + # If it is a Comparison, format it to remove whitespaces + # The formatted comparison is appended to comp_list + query_list = _search_for_subqueries(next_token.tokens) + comp_list.append(f.format_like(f.format_whitespace(next_token.value))) + + # check if AND or OR exists + if len(tokens) > i + 4: + next_token = tokens[i + 4] + if next_token.value == c.AND: + query_list += _extract_on(tokens, i + 4, comp_list) + elif next_token.value == c.OR: + query_list += _extract_on(tokens, i + 4, comp_list) + + for j, token in enumerate(tokens): + if token.value == c.BETWEEN: + between_str = "" + for k in range(j - 2, j + 7): # BETWEEN Bereich festlegen + between_str += tokens[k].value + comp_list.append(between_str) + + elif token.value in {c.IN, c.LIKE}: + in_like_str = "" + for k in range(j - 2, j + 2): # IN und LIKE Bereich festlegen + in_like_str += tokens[k].value + comp_list.append(in_like_str) + return query_list + return None + + +def _extract_where(token, comp_list, join_list): + _extract_and_format_between(token, comp_list) + where_list = [] + print(token) + for i, t in enumerate(token.tokens): + print(t) + # add comparison to the list if found + if isinstance(t, sqlparse.sql.Comparison): + # print("extr token comp ", t.tokens) + comp_list.append(f.format_like(f.format_whitespace(t.value))) + where_list.append(_search_for_subqueries(t.tokens)) + # save everything inside a parenthesis + if isinstance(t, sqlparse.sql.Parenthesis): + # print(f"PARA {t.tokens}") + comp_list.append(f.format_like(f.format_parenthesis(t.value))) + if t.value == c.BETWEEN: + res_str = "" + for j in range(i - 2, i + 7): + res_str += token.tokens[j].value + comp_list.append(res_str) + if t.value in [c.IN, c.LIKE]: + res_str = "" + print("start") + for j in range(i - 2, i + 2): + res_str += token.tokens[j].value + comp_list.append(res_str) + # append where keyword to the list of clauses MAYBE CHANGE IN DIFFERENT ARRAYS + join_list.append(token.token_first().value) + + return where_list + + +def _extract_group_by(tokens, i, group_list, having_list): + j = i + 1 + # Loop through the tokens starting from the position after GROUP BY + while j < len(tokens): + t = tokens[j] + if isinstance(t, sqlparse.sql.Token): + # Check if the token is of a type that can be part of a GROUP BY clause + if isinstance( + tokens[j], + ( + sqlparse.sql.IdentifierList, + sqlparse.sql.Identifier, + sqlparse.sql.Operation, + sqlparse.sql.Function, + sqlparse.sql.Parenthesis, + ), + ): + # If so, extract the attributes from the token and add them to the group_list + _extract_group_by_attributes(tokens[j], group_list) + # Move to the next token + j += 1 + # Check and extract any HAVING clause attributes + _extract_having(t, tokens, j, having_list) + # Check if the current token marks the end of the GROUP BY clause + # This can be an ORDER BY or HAVING keyword, or a semicolon indicating the end of the query + if ( + t.ttype == sqlparse.tokens.Keyword + and (t.value in [c.ORDER_BY, c.HAVING]) + ) or (t.ttype == sqlparse.tokens.Punctuation and t.value == ";"): + break + j += 1 + + +def _extract_having(t, tokens, j, having_list): + # Initialize k as the next index after j + k = j + 1 + # Check if the current token is a HAVING keyword + if t.ttype == sqlparse.tokens.Keyword and t.value == c.HAVING: + # Loop through the tokens starting from index k + while k < len(tokens): + # Check if the current token is a valid SQL token + if isinstance(tokens[k], sqlparse.sql.Token): + # Check if the token is a Comparison type + # Check if the token is an ORDER_BY keyword or a semicolon, indicating the end of the HAVING clause + if ( + tokens[k].ttype == sqlparse.tokens.Keyword + and tokens[k].value == c.ORDER_BY + ) or ( + tokens[k].ttype == sqlparse.tokens.Punctuation + and tokens[k].value == ";" + ): + # Break the loop if ORDER_BY or semicolon is found + break + if isinstance(tokens[k], sqlparse.sql.Comparison): + # If it's a Comparison, format it and add to having_list + having_list.append( + f.format_like(f.format_whitespace(tokens[k].value)) + ) + # Move to the next token after processing the Comparison + k += 1 + # print("inside", tokens) + + # Increment k to move to the next token + k += 1 + # Increment j at the end of the loop + j += 1 + + +def _extract_order_by(tokens, i, order_list): + j = i + 1 + while j < len(tokens): + if isinstance(tokens[j], sqlparse.sql.Token): + # Check if the token is an IdentifierList (a list of identifiers) + if isinstance(tokens[j], sqlparse.sql.IdentifierList): + # Iterate through each sub-token in the IdentifierList + for t in tokens[j]: + # Extract attributes from each sub-token and add them to the order_list + _extract_order_by_attributes(t, order_list) + # Check if the token is one of the types that can be part of an ORDER BY clause + if isinstance( + tokens[j], + ( + sqlparse.sql.Identifier, + sqlparse.sql.Operation, + sqlparse.sql.Function, + sqlparse.sql.Parenthesis, + sqlparse.sql.Comparison, + ), + ): + # Extract attributes from the token and add them to the order_list + _extract_order_by_attributes(tokens[j], order_list) + j += 1 + j += 1 + + +# if the given token contains a subquery it returns the affected token +def _extract_table_elements(token, tab_map, name_list: list): + # Check if the token is an Identifier (e.g., a table name or a column name) + if isinstance(token, sqlparse.sql.Identifier): + # Check if the Identifier token has an alias + if token.has_alias(): + # If there is an alias, map the real name of the table (or column) to its alias + tab_map[token.get_real_name()] = token.get_alias() + # Also, append the real name to the name list + name_list.append(token.get_real_name()) + else: + # If there is no alias, just append the value of the token (i.e., the name itself) to the list + name_list.append(token.value) + + # check if identifier contains subquery + return _search_for_subqueries(token.tokens) + + +def _extract_order_by_attributes(token, order_list: list): + # Check if the token is of a type that can be part of an ORDER BY clause + if isinstance( + token, + ( + sqlparse.sql.Identifier, + sqlparse.sql.Operation, + sqlparse.sql.Function, + sqlparse.sql.Parenthesis, + sqlparse.sql.Comparison, + ), + ): + # Check if the token contains the DESC (descending order) keyword + if re.search(c.DESC, token.value): + # If DESC is found, remove it from the token, format the remaining string, and add it to the order list + order_list.append( + re.sub(c.DESC, "", f.format_whitespace(token.value.strip())) + ) + # Also, add the 'desc' keyword to indicate descending order + order_list.append("desc") + # Check if the token contains the ASC (ascending order) keyword + elif re.search(c.ASC, token.value): + # If ASC is found, remove it from the token, format the remaining string, and add it to the order list + order_list.append( + re.sub(c.ASC, "", f.format_whitespace(token.value.strip())) + ) + # If neither DESC nor ASC is found + else: + # Format the token's value and add it to the order list + order_list.append(f.format_whitespace(token.value.strip())) + + +def _extract_group_by_attributes(token, group_list: list): + # Check if the token is one of the types that can be part of a GROUP BY clause + if isinstance( + token, + ( + sqlparse.sql.Identifier, + sqlparse.sql.Operation, + sqlparse.sql.Function, + sqlparse.sql.Parenthesis, + ), + ): + # If it is, format its value to remove excess whitespace and add it to the list + group_list.append(f.format_whitespace(token.value)) + + # Check if the token is an IdentifierList (a list of identifiers) + if isinstance(token, sqlparse.sql.IdentifierList): + # Iterate through each sub-token in the IdentifierList + for t in token: + # Check if the sub-token is an Identifier + if isinstance(t, sqlparse.sql.Identifier): + # If it is, format its value and add it to the list + group_list.append(f.format_whitespace(t.value)) + + +def _extract_and_format_between(token, comp_list: list): + token_str = " ".join(t.value for t in token.tokens) + pattern = re.compile(c.BETWEEN_REGEX) + matches = pattern.findall(token_str) + + # Convert each match tuple into a formatted string + for match in matches: + formatted_expression = f"{match[1]}<={match[0]}<={match[2]}" + comp_list.append(formatted_expression) diff --git a/modules/fbs-sql-checker/api/distance/table_distance.py b/modules/fbs-sql-checker/api/distance/table_distance.py new file mode 100755 index 000000000..a43c20422 --- /dev/null +++ b/modules/fbs-sql-checker/api/distance/table_distance.py @@ -0,0 +1,132 @@ +import re +from . import constants as c +from . import db_connection as db +from . import equation_checker as ec + + +def get_from_clause_distance(ref: list, query: list, ref_join: list, query_join: list): + moves = 0 + # check for table used if number of table used else difference * OMU + moves += abs(len(ref) - len(query)) * c.OBJECT_MULT + + # join difference + if len(ref_join) != len(query_join): + moves += abs(len(ref_join) - len(query_join)) * c.OBJECT_MULT + else: + for ref_item, query_item in zip(sorted(ref_join), sorted(query_join)): + if ref_item != query_item: + moves += c.STRUCT_MULT + # test if both queries yield the same results + moves += _join_queries_distance(ref, query, ref_join, query_join) + + return moves + + +def _join_queries_distance(ref, query, ref_join, query_join): + moves = 0 + # Check if the WHERE clause is not present in + if c.WHERE not in ref_join and c.WHERE not in query_join: + # check if different JOINS clauses were used + if any(rj in c.JOIN_TYPES for rj in ref_join) and any( + qj in c.JOIN_TYPES for qj in query_join + ): + mapped_ref, mapped_query = _map_values(ref, query) + # Format the join part of the SQL script for both reference and query + ref_script = _format_join_script(mapped_ref, ref_join) + query_script = _format_join_script(mapped_query, query_join) + try: + # Set up database connection + connection = db.setup_db(ref) + # Execute the formatted reference and query scripts in the database + ref_res = db.execute_query(ref_script, connection) + query_res = db.execute_query(query_script, connection) + # Compare the results of the reference and query scripts + if ref_res != query_res: + moves += c.OBJECT_MULT + connection.close() + except Exception as e: + print("Database Error", e) + return moves + + +def _map_values(ref, query): + full_list = ref + query + value_map = {} + # assign each element to a character + for idx, element in enumerate(set(full_list)): + value_map[element] = chr(65 + idx) # Using A, B, C, ... as values + + # Map the elements in the lists to their corresponding values + mapped_ref = [value_map[element] for element in ref] + mapped_query = [value_map[element] for element in query] + + return mapped_ref, mapped_query + + +def _format_join_script(tab_name: list, join_list: list): + # Initialize the SQL script with the base SELECT statement + script: str = "SELECT * FROM " + if len(tab_name) != 0: + # Add the first table name to the FROM clause + script += f"{tab_name[0]}" + # Iterate through the remaining table names to build the JOIN clauses + for i in range(1, len(tab_name)): + # Determine the join type for the current table (if available) + join_type = join_list[i - 1] if i - 1 < len(join_list) else "" + # Append the join type and current table name to the script + # Also include the ON clause to specify the join condition + script += ( + f" {join_type} {tab_name[i]} ON {tab_name[i - 1]}.x = {tab_name[i]}.x" + ) + # Complete the SQL script with a semicolon + script += ";" + return script + + +def comparison_distance(ref: list[str], query: list[str]): + moves = 0 + + if len(ref) != len(query): + # If lengths are different, calculate the absolute difference in length + # Multiply the difference by a predefined constant (OBJECT_MULT) and add to the moves counter + moves += abs(len(ref) - len(query)) * c.OBJECT_MULT + else: + for ref_item, query_item in zip(sorted(ref), sorted(query)): + if re.match(c.SYMBOL_REGEX, ref_item) and re.match( + c.SYMBOL_REGEX, query_item + ): + moves += ec.check_equation(ref_item, query_item) + else: + if ref_item != query_item: + # Increment the moves counter by OBJECT_MULT for each differing pair + moves += c.OBJECT_MULT + # Return the total number of moves calculated + return moves + + +def group_and_order_by_distance(ref: list[str], query: list[str]): + moves = 0 + # Check if both lists contain the same elements, irrespective of order + if sorted(ref) == sorted(query): + for ref_item in ref: + # Check if the corresponding element in the query list is at a different position + if query[ref.index(ref_item)] != ref_item: + # Increment the moves counter by the order multiplier + moves += c.ORDER_MULT + # Remove the element from its current position in the query list + query.remove(ref_item) + # Insert the element at its correct position based on the reference list + query.insert(ref.index(ref_item), ref_item) + # Check if the lengths of the two lists are different + elif len(ref) != len(query): + # Increment the moves counter by the object multiplier times the difference in length + moves += abs(len(ref) - len(query)) * c.OBJECT_MULT + # If the lists are of the same length but have different elements + else: + # Iterate through each pair of elements in the sorted lists + for ref_item, query_item in zip(sorted(ref), sorted(query)): + # Check if the elements are different + if ref_item != query_item: + # Increment the moves counter by the object multiplier for each differing pair + moves += c.OBJECT_MULT + return moves diff --git a/modules/fbs-sql-checker/api/error.py b/modules/fbs-sql-checker/api/error.py new file mode 100644 index 000000000..03b002b0f --- /dev/null +++ b/modules/fbs-sql-checker/api/error.py @@ -0,0 +1,6 @@ +class ParserException(Exception): + pass + + +class QueryParsingParserException(ParserException): + pass diff --git a/modules/fbs-sql-checker/api/formatting.py b/modules/fbs-sql-checker/api/formatting.py old mode 100644 new mode 100755 diff --git a/modules/fbs-sql-checker/api/json_creator.py b/modules/fbs-sql-checker/api/json_creator.py deleted file mode 100644 index aa0dccbe7..000000000 --- a/modules/fbs-sql-checker/api/json_creator.py +++ /dev/null @@ -1,573 +0,0 @@ -# JSONCreator.py - -from parser import parse_query -from datetime import datetime -from table_checker import extract_tables -from pro_attribute_checker import extract_pro_attributes -import sel_attribute_checker as AWC -from pymongo import MongoClient # pylint: disable=E0401 -from model import * # pylint: disable=W0401 -from mask_aliases import SQLAliasMasker - -rightStatements = [] -rightTables = [] -rightAtts = [] -rightStrings = [] -rightWhereAtts = [] -user_data = [] -tables, selAttributes, proAttributes, strings = [], [], [], [] - - -# Parse a single SQL-Statement and upload it to DB -def parse_single_stat_upload_db(data, client): - # create DB-connection - client = MongoClient(client, 27107) - mydb = client.get_default_database() - mycollection = mydb["Queries"] - time = datetime.now().strftime("%Y-%m-%d %H:%M:%S") - ( - tables2, - pro_atts2, - sel_atts2, - strings2, - task_nr, - is_sol, - my_uuid, - course_id, - order_by2, - group_by2, - having2, - joins2, - wildcards2, - ) = ([], [], [], [], [], [], [], [], [], [], [], [], []) - try: - if "submission" in data: - query = data["submission"] - masker = SQLAliasMasker(query) - masker.mask_aliases_() - data["submission"] = masker.get_masked_query() - # Extract tables, selAttributes, proAttributes and strings - if extract_tables(data["submission"], client) != "Unknown": - table_list = extract_tables(data["submission"], client) - tables2.extend(table_list[0]) - if table_list[1] != ["Empty"]: - try: - table_list.pop(0) - for x in table_list: - for y in x: - joins2.append(y) - except Exception: - joins2.append("Unknown") - else: - joins2.append("Empty") - extracted_pro_attributes = extract_pro_attributes( - data["submission"], client - ) - if extracted_pro_attributes != "Unknown": - pro_atts2.extend(extracted_pro_attributes) - extracted_string = AWC.extract_sel_attributes(data["submission"], client) - if extracted_string != "Unknown": - sel_atts2.extend(extracted_string) - extracted_order_by = AWC.extract_order_by(data["submission"], client) - if extracted_order_by != "Unknown": - order_by2.extend(extracted_order_by) - extracted_group_by = AWC.extract_group_by(data["submission"], client) - if extracted_group_by != "Unknown": - group_by2.extend(extracted_group_by) - extracted_having = AWC.extract_having(data["submission"], client) - if extracted_having != "Unknown": - having2.extend(extracted_having) - strings2.extend(list(set(AWC.literal))) - # If TaskID or Submission ID not in data, return - if "tid" not in data or "sid" not in data: - print("Task ID or Unique ID not in data") - return - task_nr = data["tid"] - my_uuid = data["sid"] - course_id = data["cid"] - is_sol = data["isSol"] - # Save tables, selAttributes, proAttributes and strings to DB - if parse_query(data["submission"], client) is not False: - insert_tables(mydb, data, my_uuid, client) - # Check if it is a new solution and characteristics are right - ( - tables2, - pro_atts2, - sel_atts2, - strings2, - order_by2, - group_by2, - joins2, - having2, - wildcards2, - ) = check_solution_chars( - data, - task_nr, - my_uuid, - tables2, - pro_atts2, - sel_atts2, - strings2, - order_by2, - group_by2, - joins2, - having2, - client, - ) - # produce a JSON - record = return_json( - data, - is_sol, - my_uuid, - task_nr, - course_id, - tables2, - pro_atts2, - sel_atts2, - strings2, - order_by2, - group_by2, - joins2, - having2, - wildcards2, - client, - time, - ) - # save JSON to DB - mycollection.insert_one(record) - except Exception as e: - print(e) - task_nr = data["tid"] - my_uuid = data["sid"] - course_id = data["cid"] - user_data = return_json_not_parsable(data) # pylint: disable=W0621 - insert_not_parsable(my_uuid, user_data[3], client) - record = prod_json_not_parsable(my_uuid, course_id, task_nr, time) - mycollection.insert_one(record) - - -# Check if it is a new solution; check if tables, attributes etc. are right -def check_solution_chars( - data, - task_nr, - my_uuid, - tables2, - pro_atts2, - sel_atts2, - strings2, - order_by2, - group_by2, - joins2, - having2, - client, -): - new_solution = True - ( - tables_right, - sel_attributes_right, - pro_attributes_right, - strings_right, - order_by_right, - group_by_right, - joins_right, - having_right, - wildcards, - ) = (False, False, False, False, False, False, False, False, False) - mydb = client.get_default_database() - mycol = mydb["Solutions"] - # For every solution for given task - for x in mycol.find({"taskNumber": task_nr}): - # Extract Tables, Attributes etc. (cut out for better overview) - id = x["id"] # pylint: disable=W0622) - ( - tables, # pylint: disable=W0621 - pro_attributes, - sel_attributes, - strings, # pylint: disable=W0621 - order_by, - group_by, - having, - joins, - ) = ( # pylint: disable=W0621 - [], - [], - [], - [], - [], - [], - [], - [], - ) - mycol = mydb["Tables"] - for y in mycol.find({"id": id}, {"table": 1}): - tables.append(y["table"]) - mycol = mydb["ProAttributes"] - for y in mycol.find({"id": id}, {"proAttribute": 1}): - pro_attributes.append(y["proAttribute"]) - mycol = mydb["SelAttributes"] - for y in mycol.find({"id": id}, {"selAttribute": 1}): - sel_attributes.append(y["selAttribute"]) - mycol = mydb["Strings"] - for y in mycol.find({"id": id}, {"string": 1}): - strings.append(y["string"]) - mycol = mydb["OrderBy"] - for y in mycol.find({"id": id}, {"orderBy": 1, "sort": 1}): - order_by_value = [y["orderBy"]] - if ( - "sort" in y - ): # if there is no order in sort the default "asc" will be used - order_by_value.append(y["sort"]) - else: - order_by_value.append("asc") - order_by.append(order_by_value) - mycol = mydb["GroupBy"] - for y in mycol.find({"id": id}, {"groupBy": 1}): - group_by.append(y["groupBy"]) - mycol = mydb["Joins"] - for y in mycol.find({"id": id}, {"type": 1, "attr1": 1, "attr2": 1}): - join_value = [y["type"]] - if ( - "attr1" in y and "attr2" in y - ): # if there is no order in sort the default "asc" will be used - join_value.append(y["attr1"]) - join_value.append(y["attr2"]) - else: - join_value.append("Empty") - joins.append(join_value) - mycol = mydb["Having"] - for y in mycol.find({"id": id}, {"havingAttribute": 1}): - having_value = y["havingAttribute"] - having.append(having_value) - if len(joins) == 0: - joins.append("Empty") - if data["passed"]: - if not data["isSol"]: - tables_right = True - sel_attributes_right = True - pro_attributes_right = True - strings_right = True - wildcards = True - order_by_right = True - group_by_right = True - joins_right = True - having_right = True - - # Compare them to tables, proAttributes etc of a given sql-query - if ( - tables == tables2 # pylint: disable=R0916 - and set(pro_attributes) == set(pro_atts2) - and set(sel_attributes) == set(sel_atts2) - and set(strings) == set(strings2) - and order_by == order_by2 - and joins == joins2 - and having == having2 - ): - # If they alle are same, it is not a new solution - new_solution = False - else: - # Check for tables, proAttributes etc. if they are right - if tables == tables2: - tables_right = True - if set(pro_attributes) == set(pro_atts2): - sel_attributes_right = True - if set(sel_attributes) == set(sel_atts2): - pro_attributes_right = True - if set(strings) == set(strings2): - strings_right = True - if ( - any("%" in s for s in strings) - and not any("%" in s for s in strings2) - or not any("%" in s for s in strings) - and any("%" in s for s in strings2) - ): - wildcards = False - else: - wildcards = True - if order_by == order_by2: - order_by_right = True - if group_by == group_by2: - group_by_right = True - if joins == joins2: - joins_right = True - if having == having2: - having_right = True - if data["isSol"]: - if new_solution is True: - # Upload as a new Solution to DB - parse_single_stat_upload_solution(data, task_nr, my_uuid, client) - return (True, True, True, True, True, True, True, True, True) - # return if characteristics are True or False - return ( - tables_right, - sel_attributes_right, - pro_attributes_right, - strings_right, - order_by_right, - group_by_right, - joins_right, - having_right, - wildcards, - ) - - -# Parse a solution and upload it to DB -def parse_single_stat_upload_solution(data, task_nr, my_uuid, client): - mydb = client.get_default_database() - mycollection = mydb["Solutions"] - record = prod_solution_json(data, my_uuid, task_nr) - mycollection.insert_one(record) - - -# return JSON to be pasted to DB -def return_json( # pylint: disable=R1710 - elem, - is_sol, - my_uuid, - task_nr, - course_id, - tables_right, - pro_attributes_right, - sel_attributes_right, - strings_right, - order_by_right, - group_by_right, - joins_right, - having_right, - wildcards, - client, - time, -): - # Extract informations from a sql-query-json - if "passed" in elem: - user_data.append(elem["passed"]) - if "userId" in elem: - user_data.append(elem["userId"]) - if "attempt" in elem: - user_data.append(elem["attempt"]) - if "submission" in elem: - if parse_query(elem["submission"], client) is not False: - # produce a json to be pasted to DB - record = prod_json( - my_uuid, - course_id, - elem["submission"], - task_nr, - is_sol, - tables_right, - sel_attributes_right, - pro_attributes_right, - strings_right, - order_by_right, - group_by_right, - joins_right, - having_right, - wildcards, - time, - ) - return record - # produce a json if the sql-query is not parsable - record = prod_json_not_parsable(my_uuid, course_id, task_nr, time) - return record - - -# Returns a json file which extracts Tables and Attributes -def prod_json_not_parsable(_id, cid, task_nr, time): - # Create dictionary - value = { - "id": str(_id), - "cid": cid, - "taskNumber": task_nr, - "statement": user_data[3], - "queryRight": user_data[0], - "parsable": False, - "tablesRight": None, - "selAttributesRight": None, - "proAttributesRight": None, - "stringsRight": None, - "userId": user_data[1], - "attempt": user_data[2], - "orderbyRight": None, - "havingRight": None, - "wildcards": None, - "time": time, - } - return value - - -# Insert data of Tables, proAttributes, selAttributes and Strings to Database -def insert_tables(mydb, elem, my_uuid, client): - table_list = extract_tables(elem["submission"], client) - joins = [] - tables.extend(table_list[0]) - if table_list[1] != ["Empty"]: - try: - table_list.pop(0) - for x in table_list: - for y in x: - joins.append(y) - except Exception: - joins.append("Unknown") - else: - joins.append("Empty") - if len(tables) == 1: - mycollection = mydb["Tables"] - record = json_table(my_uuid, tables[0]) - mycollection.insert_one(record) - if joins[0] != "Empty": - try: - mycollection = mydb["Joins"] - record = json_join_attribute(my_uuid, joins) - mycollection.insert_one(record) - except Exception: - print("Error while reading joins.") - elif len(tables) > 1 and not isinstance( - extract_tables(elem["submission"], client), str - ): - mycollection = mydb["Tables"] - for val in tables: - record = json_table(my_uuid, val) - mycollection.insert_one(record) - if joins[0] != "Empty": - try: - mycollection = mydb["Joins"] - for y in joins: - record = json_join_attribute(my_uuid, y) - mycollection.insert_one(record) - except Exception: - print("Error while reading joins.") - pro_attributes = extract_pro_attributes(elem["submission"], client) - if len(pro_attributes) == 1: - mycollection = mydb["ProAttributes"] - record = json_pro_attribute(my_uuid, pro_attributes[0]) - mycollection.insert_one(record) - elif len(pro_attributes) > 1 and not isinstance(pro_attributes, str): - mycollection = mydb["ProAttributes"] - for val in pro_attributes: - record = json_pro_attribute(my_uuid, val) - mycollection.insert_one(record) - sel_attributes = AWC.extract_sel_attributes(elem["submission"], client) - if len(sel_attributes) == 1: - mycollection = mydb["SelAttributes"] - record = json_sel_attribute(my_uuid, sel_attributes[0]) - mycollection.insert_one(record) - elif len(sel_attributes) > 1 and not isinstance(sel_attributes, str): - mycollection = mydb["SelAttributes"] - for val in sel_attributes: - record = json_sel_attribute(my_uuid, val) - mycollection.insert_one(record) - if len(list(set(AWC.literal))) == 1: - mycollection = mydb["Strings"] - record = json_string(my_uuid, list(set(AWC.literal))[0]) - mycollection.insert_one(record) - elif len(list(set(AWC.literal))) > 1 and not isinstance( - list(set(AWC.literal)), str - ): - mycollection = mydb["Strings"] - for val in list(set(AWC.literal)): - record = json_string(my_uuid, val) - mycollection.insert_one(record) - order_by = AWC.extract_order_by(elem["submission"], client) - if len(order_by) == 1: - mycollection = mydb["OrderBy"] - record = json_order_by_attribute(my_uuid, order_by[0]) - mycollection.insert_one(record) - elif len(order_by) > 1 and not isinstance(order_by, str): - mycollection = mydb["OrderBy"] - for val in order_by: - record = json_order_by_attribute(my_uuid, val) - mycollection.insert_one(record) - group_by = AWC.extract_group_by(elem["submission"], client) - if len(group_by) == 1: - mycollection = mydb["GroupBy"] - record = json_group_by_attribute(my_uuid, group_by[0]) - mycollection.insert_one(record) - elif len(group_by) > 1 and not isinstance(group_by, str): - mycollection = mydb["GroupBy"] - for val in AWC.extract_group_by(elem["submission"], client): - record = json_group_by_attribute(my_uuid, val) - mycollection.insert_one(record) - if len(AWC.extract_having(elem["submission"], client)) == 1: - mycollection = mydb["Having"] - record = json_having_attribute( - my_uuid, AWC.extract_having(elem["submission"], client)[0] - ) - mycollection.insert_one(record) - elif len(AWC.extract_having(elem["submission"], client)) > 1 and not isinstance( - AWC.extract_having(elem["submission"], client), str - ): - mycollection = mydb["Having"] - for val in AWC.extract_having(elem["submission"], client): - record = json_having_attribute(my_uuid, val) - mycollection.insert_one(record) - AWC.literal = [] - user_data.clear() - - -# Returns a json file which extracts characteristics -# and tells which of them are wrong -def prod_json( - _id, - course_id, - test_sql, - task_nr, - is_sol, - tables_right, - sel_attributes_right, - pro_attributes_right, - strings_right, - order_by_right, - group_by_right, - joins_right, - having_right, - wildcards, - time, -): - # save data if it is a manual solution - if is_sol is True: - user_data.extend([True]) - user_data.extend([0]) - user_data.extend([0]) - value = { - "id": str(_id), - "courseId": course_id, - "taskNumber": task_nr, - "statement": test_sql, - "queryRight": user_data[0], - "parsable": True, - "isSolution": is_sol, - "tablesRight": tables_right, - "selAttributesRight": sel_attributes_right, - "proAttributesRight": pro_attributes_right, - "stringsRight": strings_right, - "userId": user_data[1], - "attempt": user_data[2], - "orderByRight": order_by_right, - "groupByRight": group_by_right, - "joinsRight": joins_right, - "havingRight": having_right, - "wildcards": wildcards, - "time": time, - } - user_data.clear() - AWC.literal = [] - return value - - -def insert_not_parsable(my_uuid, submission, client): - mydb = client.get_default_database() - mycollection = mydb["NotParsable"] - record = json_not_parsable(my_uuid, submission) - mycollection.insert_one(record) - - -def return_json_not_parsable(elem): - # Extract informations from a sql-query-json - if "passed" in elem: - user_data.append(elem["passed"]) - if "userId" in elem: - user_data.append(elem["userId"]) - if "attempt" in elem: - user_data.append(elem["attempt"]) - if "submission" in elem: - user_data.append(elem["submission"]) - return user_data diff --git a/modules/fbs-sql-checker/api/main.py b/modules/fbs-sql-checker/api/main.py old mode 100644 new mode 100755 index 13c05a5ae..5e7925b27 --- a/modules/fbs-sql-checker/api/main.py +++ b/modules/fbs-sql-checker/api/main.py @@ -1,13 +1,20 @@ # main.py """System module.""" +import json import sys # pylint: disable=W0611 + import requests # pylint: disable=W0611 -from json_creator import parse_single_stat_upload_db + +from api.comparator.sqlparse_comparator import SqlparseComparator +from api.data_store import MongoDataStore +from api.distance.distance_calc import GetDistanceCalculator +from api.query_processor import QueryProcessorV2 +from api.datatypes import Submission +from api.solution_fetcher import NearestSolutionFetcher, FirstSolutionFetcher # The following Code is for productive purposes -CLIENT = sys.argv[2] if len(sys.argv) < 3: print("Zu wenige Argumente übergeben.") print( @@ -15,7 +22,14 @@ "zu analysierende JSON aufgerufen werden soll." ) else: - URL_ANSWER = sys.argv[1] - answer = requests.get(URL_ANSWER, verify=False, timeout=25) - print(answer.json()) - parse_single_stat_upload_db(answer.json(), CLIENT) + answer = requests.get(sys.argv[1], verify=False, timeout=60) + submission = Submission.from_dict(answer.json()) + storage = MongoDataStore.connect(sys.argv[2]) + qp = QueryProcessorV2( + SqlparseComparator(), + NearestSolutionFetcher(storage, GetDistanceCalculator()), + ) + result = qp.process(submission) + + print("storing", result.to_db_dict()) + storage.store("Queries", result.to_db_dict()) diff --git a/modules/fbs-sql-checker/api/mask_aliases.py b/modules/fbs-sql-checker/api/mask_aliases.py old mode 100644 new mode 100755 diff --git a/modules/fbs-sql-checker/api/model.py b/modules/fbs-sql-checker/api/model.py old mode 100644 new mode 100755 index aa9ca9a90..e034c740b --- a/modules/fbs-sql-checker/api/model.py +++ b/modules/fbs-sql-checker/api/model.py @@ -2,7 +2,7 @@ # Check if it is a new solution; check if tables, attributes etc. are right def prod_solution_json(elem, my_uuid, task_nr): - value = {"id": str(my_uuid), "taskNumber": task_nr, "statement": elem["submission"]} + value = {"id": str(my_uuid), "taskNumber": task_nr, "statement": elem} return value diff --git a/modules/fbs-sql-checker/api/modules/fbs-sql-checker/api/distance/log/distance.txt b/modules/fbs-sql-checker/api/modules/fbs-sql-checker/api/distance/log/distance.txt new file mode 100644 index 000000000..aa74a981b --- /dev/null +++ b/modules/fbs-sql-checker/api/modules/fbs-sql-checker/api/distance/log/distance.txt @@ -0,0 +1,4408 @@ +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date < 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdata>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_data > 100; + +Distance = 100 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['mail']; query: ['username'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 50, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY username; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: ['submissions']; query: ['submissions'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['join', 'where']; query: ['join', 'where'] + +comparison equations: reference: ['id=userid', 'registrationdate>100']; query: ['id=id', 'registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: ['registrationdate']; query: ['username'] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 50 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY username; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'registrationdate'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, registration_date FROM user WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['username=tester']; query: ['username=test'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE username = 'tester'; + +query: +SELECT username, mail, password FROM user WHERE username = 'test'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: ['users'] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 50, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM users WHERE registration_date > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdata>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_data > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdata>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_data > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdata>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_data > 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate<100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date < 100; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100; + +query: +SELECT username, mail, password FROM user WHERE registration_date > '100'; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100', 'mail like ']; query: ['registrationdate>100', 'mail like '] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 AND mail LIKE '%gmail.com'; + +query: +SELECT username, mail, password FROM user WHERE registration_date > 100 OR mail LIKE '%gmail.com'; + +Distance = 0 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100', 'mail like ']; query: ['registrationdata>100', 'mail like '] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 AND mail LIKE '%gmail.com'; + +query: +SELECT username, mail, password FROM user WHERE registration_data > 100 AND mail LIKE '%gmail.com'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +attributes after order: reference: ['username', 'mail', 'password']; query: ['username', 'mail', 'password'] + +command list: reference: []; query: [] + +distinct list: reference: []; query [] + +Distance: attributes = 0, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100', 'mail like ']; query: ['registrationdata>100', 'mail like '] + +group by attributes: reference: []; query: [] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 50, group by = 0, having = 0, order by = 0 + +reference: +SELECT username, mail, password FROM user WHERE registration_date > 100 AND mail LIKE '%gmail.com'; + +query: +SELECT username, mail, password FROM user WHERE registration_data > 100 AND mail LIKE '%gmail.com'; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['registrationdate', 'username']; query: ['registrationdata', 'username'] + +attributes after order: reference: ['registrationdate', 'username']; query: ['registrationdata', 'username'] + +command list: reference: ['count']; query: ['count'] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['registrationdate']; query: ['registrationdate'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT registration_date, COUNT(username) FROM user WHERE registration_date > 100 GROUP BY registration_date; + +query: +SELECT registration_data, COUNT(username) FROM user WHERE registration_date > 100 GROUP BY registration_date; + +Distance = 50 +------------------------------------------------------------------ + + +attribute aliases: reference: {}; query: {} + +attributes before order: reference: ['registrationdate', 'username']; query: ['registrationdata', 'username'] + +attributes after order: reference: ['registrationdate', 'username']; query: ['registrationdata', 'username'] + +command list: reference: ['count']; query: ['count'] + +distinct list: reference: []; query [] + +Distance: attributes = 50, commands = 0, keywords = 0 + +tables used: reference: []; query: [] + +reference table aliases: {}; query table aliases {} + +data retrieval clause: reference: ['where']; query: ['where'] + +comparison equations: reference: ['registrationdate>100']; query: ['registrationdate>100'] + +group by attributes: reference: ['registrationdate']; query: ['registrationdate'] + +having attributes: reference: []; query: [] + +order by attributes: reference: []; query: [] + +Distance: table and data retrieval clause = 0, comparison equations = 0, group by = 0, having = 0, order by = 0 + +reference: +SELECT registration_date, COUNT(username) FROM user WHERE registration_date > 100 GROUP BY registration_date; + +query: +SELECT registration_data, COUNT(username) FROM user WHERE registration_date > 100 GROUP BY registration_date; + +Distance = 50 +------------------------------------------------------------------ + + diff --git a/modules/fbs-sql-checker/api/parse_data_through_checker.py b/modules/fbs-sql-checker/api/parse_data_through_checker.py old mode 100644 new mode 100755 index 648b1810d..5a844cc97 --- a/modules/fbs-sql-checker/api/parse_data_through_checker.py +++ b/modules/fbs-sql-checker/api/parse_data_through_checker.py @@ -5,7 +5,7 @@ import sys from tqdm import tqdm -from json_creator import parse_single_stat_upload_db +from query_processor import parse_single_stat_upload_db # Usage: # Execute "parse_data_through_checker.py" with a path to a json file or a directory in the console diff --git a/modules/fbs-sql-checker/api/parser.py b/modules/fbs-sql-checker/api/parser.py old mode 100644 new mode 100755 index 92d3397d5..440c6085c --- a/modules/fbs-sql-checker/api/parser.py +++ b/modules/fbs-sql-checker/api/parser.py @@ -1,20 +1,25 @@ # parser.py - import json +import logging + import mo_sql_parsing -# Parse the query -def parse_query(data, client): - try: - parsed_query = mo_sql_parsing.parse(data) - except Exception as e: - print("Not able to parse the statement " + str(data)) - print(e) - mydb = client.get_default_database() - mycollection = mydb["NotParsable"] - record = data - mycollection.insert_one(record) - return False - json_output = json.dumps(parsed_query, indent=4) - pyt_obj = json.loads(json_output) - return pyt_obj +from api.error import QueryParsingParserException + + +class QueryParser: + def __init__(self, logger=None): + if logger is None: + logger = logging.getLogger() + self._logger = logger + + def parse_query(self, query: str): + try: + parsed_query = mo_sql_parsing.parse(query) + except Exception as e: + logging.warning(f"Not able to parse the statement: {query}", exc_info=e) + raise QueryParsingParserException(query) from e + + json_output = json.dumps(parsed_query, indent=4) + pyt_obj = json.loads(json_output) + return pyt_obj diff --git a/modules/fbs-sql-checker/api/pro_attribute_checker.py b/modules/fbs-sql-checker/api/pro_attribute_checker.py old mode 100644 new mode 100755 index df3ea6eeb..251bde9ab --- a/modules/fbs-sql-checker/api/pro_attribute_checker.py +++ b/modules/fbs-sql-checker/api/pro_attribute_checker.py @@ -1,130 +1,92 @@ # pro_attribute_checker.py - -from parser import parse_query -from sel_attribute_checker import select_where +from api.attribute_common import select_commands, single_select_dict, select_where import formatting as f +from api.parser import QueryParser -select_commands = [ - "sum", - "count", - "round", - "distinct", - "sec_to_time", - "avg", - "max", - "min", - "time_to_sec", - "like", - "between", - "div", - "year", - "mul", -] +class ProAttributeChecker: + def __init__(self, parser: QueryParser): + self._parser = parser -# Return ProjectionAttributes as a List -def list_of_select(json_file): - selects = [] - if not (f.is_string(json_file)) and (f.is_select(json_file)): - if isinstance(json_file["select"], list): - for val in range(len(json_file["select"])): - if "value" in json_file["select"][val]: - for val1 in json_file["select"][val]["value"]: - if val1 in select_commands: - if isinstance(json_file["select"][val]["value"][val1], str): - if "." in json_file["select"][val]["value"][val1]: - selects.append( - json_file["select"][val]["value"][val1] - .split(".")[1] - .lower() - ) - else: - selects.append( - json_file["select"][val]["value"][val1].lower() - ) - if isinstance( - json_file["select"][val]["value"][val1], list - ): - for i in range( - len(json_file["select"][val]["value"][val1]) + # Return ProjectionAttributes as a List + def _list_of_select(self, json_file): + selects = [] + if not (f.is_string(json_file)) and (f.is_select(json_file)): + if isinstance(json_file["select"], list): + for val in range(len(json_file["select"])): + if "value" in json_file["select"][val]: + for val1 in json_file["select"][val]["value"]: + if val1 in select_commands: + if isinstance( + json_file["select"][val]["value"][val1], str + ): + if "." in json_file["select"][val]["value"][val1]: + selects.append( + json_file["select"][val]["value"][val1] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file["select"][val]["value"][ + val1 + ].lower() + ) + if isinstance( + json_file["select"][val]["value"][val1], list ): - if isinstance( - json_file["select"][val]["value"][val1][i], dict + for i in range( + len(json_file["select"][val]["value"][val1]) ): - for elem1 in json_file["select"][val]["value"][ - val1 - ][i]: - if elem1 in select_commands: - if isinstance( - json_file["select"][val]["value"][ - val1 - ][i][elem1], - list, - ): - for elem2 in range( - len( - json_file["select"][val][ - "value" - ][val1][i][elem1] - ) + if isinstance( + json_file["select"][val]["value"][val1][i], + dict, + ): + for elem1 in json_file["select"][val][ + "value" + ][val1][i]: + if elem1 in select_commands: + if isinstance( + json_file["select"][val][ + "value" + ][val1][i][elem1], + list, ): - if isinstance( - json_file["select"][val][ - "value" - ][val1][i][elem1][elem2], - str, + for elem2 in range( + len( + json_file["select"][ + val + ]["value"][val1][i][ + elem1 + ] + ) ): - if ( - "." - in json_file["select"][ + if isinstance( + json_file["select"][ val ]["value"][val1][i][ elem1 ][ elem2 - ] + ], + str, ): - selects.append( - json_file["select"][ - val - ]["value"][val1][i][ - elem1 + if ( + "." + in json_file[ + "select" + ][val]["value"][ + val1 + ][ + i ][ - elem2 - ] - .split(".")[1] - .lower() - ) - else: - selects.append( - json_file["select"][ - val - ]["value"][val1][i][ elem1 ][ elem2 ] - ) - elif isinstance( - json_file["select"][val][ - "value" - ][val1][i][elem1][elem2], - list, - ): - for elem3 in json_file[ - "select" - ][val]["value"][val1][i][ - elem1 - ][ - elem2 - ]: - if ( - elem3 - in select_commands ): - if ( - "." - in json_file[ + selects.append( + json_file[ "select" ][val]["value"][ val1 @@ -134,36 +96,54 @@ def list_of_select(json_file): elem1 ][ elem2 + ] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file[ + "select" + ][val]["value"][ + val1 + ][ + i + ][ + elem1 ][ - elem3 + elem2 ] + ) + elif isinstance( + json_file["select"][ + val + ]["value"][val1][i][ + elem1 + ][ + elem2 + ], + list, + ): + for elem3 in json_file[ + "select" + ][val]["value"][val1][ + i + ][ + elem1 + ][ + elem2 + ]: + if ( + elem3 + in select_commands ): - selects.append( - json_file[ + if ( + "." + in json_file[ "select" - ][val][ - "value" ][ - val1 - ][ - i - ][ - elem1 - ][ - elem2 + val ][ - elem3 - ] - .split(".")[ - 1 - ] - .lower() - ) - else: - selects.append( - json_file[ - "select" - ][val][ "value" ][ val1 @@ -175,208 +155,161 @@ def list_of_select(json_file): elem2 ][ elem3 - ].lower() - ) - else: - if ( - "." - in json_file["select"][val][ - "value" - ][val1][i][elem1] - ): - selects.append( - json_file["select"][val][ - "value" - ][val1][i][elem1] - .split(".")[1] - .lower() - ) + ] + ): + selects.append( + json_file[ + "select" + ][ + val + ][ + "value" + ][ + val1 + ][ + i + ][ + elem1 + ][ + elem2 + ][ + elem3 + ] + .split( + "." + )[1] + .lower() + ) + else: + selects.append( + json_file[ + "select" + ][ + val + ][ + "value" + ][ + val1 + ][ + i + ][ + elem1 + ][ + elem2 + ][ + elem3 + ].lower() + ) else: - selects.append( - json_file["select"][val][ + if ( + "." + in json_file["select"][val][ "value" - ][val1][i][elem1].lower() - ) - if "." in json_file["select"][val]["value"]: - selects.append( - json_file["select"][val]["value"].split(".")[1].lower() - ) - else: - if not isinstance(json_file["select"][val]["value"], dict): - selects.append(json_file["select"][val]["value"].lower()) - return set(selects) - - -# Returns a single ProjectionAttribute -def single_select(json_file): - selects = [] - if not (f.is_string(json_file)) and (f.is_select(json_file)): - if isinstance(json_file["select"], str): - if "." in json_file["select"]: - selects.append(json_file["select"].split(".")[1].lower()) - else: - selects.append(json_file["select"].lower()) - for val in json_file["from"]: - if val == "value": - if select_union(json_file["from"]["value"]): - selects.extend(select_union(json_file["from"]["value"])) - if list_of_select(json_file["from"]["value"]): - selects.extend(list_of_select(json_file["from"]["value"])) - if select_where(json_file["from"]["value"]): - selects.append(select_where(json_file["from"]["value"])) - return set(selects) - + ][val1][i][elem1] + ): + selects.append( + json_file["select"][ + val + ]["value"][val1][i][ + elem1 + ] + .split(".")[1] + .lower() + ) + else: + selects.append( + json_file["select"][ + val + ]["value"][val1][i][ + elem1 + ].lower() + ) + if "." in json_file["select"][val]["value"]: + selects.append( + json_file["select"][val]["value"].split(".")[1].lower() + ) + else: + if not isinstance(json_file["select"][val]["value"], dict): + selects.append( + json_file["select"][val]["value"].lower() + ) + return set(selects) -# Returns a single ProjectionAttribute if it's a dictionary -def single_select_dict(json_file): - selects = [] - if not (f.is_string(json_file)) and (f.is_select(json_file)): - if isinstance(json_file["select"], dict): - if "value" in json_file["select"]: - if isinstance(json_file["select"]["value"], str): - if "." in json_file["select"]["value"]: + # Returns a single ProjectionAttribute + def _single_select(self, json_file, literals): + selects = [] + if not (f.is_string(json_file)) and (f.is_select(json_file)): + if isinstance(json_file["select"], str): + if "." in json_file["select"]: + selects.append(json_file["select"].split(".")[1].lower()) + else: + selects.append(json_file["select"].lower()) + for val in json_file["from"]: + if val == "value": + if self._select_union(json_file["from"]["value"], literals): + selects.extend( + self._select_union(json_file["from"]["value"], literals) + ) + if self._list_of_select(json_file["from"]["value"]): + selects.extend(self._list_of_select(json_file["from"]["value"])) + if select_where(json_file["from"]["value"], literals): selects.append( - json_file["select"]["value"].split(".")[1].lower() + select_where(json_file["from"]["value"], literals) ) - else: - selects.append(json_file["select"]["value"].lower()) - elif isinstance(json_file["select"]["value"], dict): - for val in json_file["select"]["value"]: - if val in select_commands: - if isinstance(json_file["select"]["value"][val], dict): - if "value" in json_file["select"]["value"][val]: - if ( - "." - in json_file["select"]["value"][val]["value"] - ): - selects.append( - json_file["select"]["value"][val]["value"] - .split(".")[1] - .lower() - ) - else: - selects.append( - json_file["select"]["value"][val]["value"] - ) - for elem in json_file["select"]["value"][val]: - if (elem in select_commands) and ( - isinstance( - json_file["select"]["value"][val][elem], - dict, - ) - ): - for elem1 in json_file["select"]["value"][val][ - elem - ]: - if elem1 in select_commands and isinstance( - json_file["select"]["value"][val][elem][ - elem1 - ], - str, - ): - if ( - "." - in json_file["select"]["value"][ - val - ][elem][elem1] - ): - selects.append( - json_file["select"]["value"][ - val - ][elem][elem1] - .split(".")[1] - .lower() - ) - else: - selects.append( - json_file["select"]["value"][ - val - ][elem][elem1].lower() - ) - if isinstance(json_file["select"]["value"][val], str): - if "." in json_file["select"]["value"][val]: - selects.append( - json_file["select"]["value"][val] - .split(".")[1] - .lower() - ) - else: - selects.append( - json_file["select"]["value"][val].lower() - ) - if isinstance(json_file["select"]["value"][val], list): - for i in range(len(json_file["select"]["value"][val])): - if "value" in json_file["select"]["value"][val][i]: - if isinstance( - json_file["select"]["value"][val][i][ - "value" - ], - str, - ): - if ( - "." - in json_file["select"]["value"][val][i][ - "value" - ] - ): - selects.append( - json_file["select"]["value"][val][ - i - ]["value"] - .split(".")[1] - .lower() - ) - else: - selects.append( - json_file["select"]["value"][val][ - i - ]["value"].lower() - ) - return set(selects) - - -# Returns ProjectionAttributes as a List if it is a union -def select_union(json_file): - list_tables = [] - if not f.is_string(json_file): - for val in json_file: - if val == "union": - for val1 in range(len(json_file["union"])): - if list_of_select(json_file["union"][val1]): - for elem in list_of_select(json_file[val][val1]): - if not isinstance(elem, dict): - list_tables.append(elem.lower()) - if select_where(json_file["union"][val1]): - if len(select_where(json_file["union"][val1])) == 1: - list_tables.append( - select_where( # pylint: disable=E1136 - json_file["union"][val1] - )[ # pylint: disable=E1136 - 0 - ].lower() # pylint: disable=E1136 - ) - else: - for elem in select_where(json_file["union"][val1]): - list_tables.append(elem.lower()) - return set(list_tables) + return set(selects) + # Returns ProjectionAttributes as a List if it is a union + def _select_union(self, json_file, literals): + list_tables = [] + if not f.is_string(json_file): + for val in json_file: + if val == "union": + for val1 in range(len(json_file["union"])): + if self._list_of_select(json_file["union"][val1]): + for elem in self._list_of_select(json_file[val][val1]): + if not isinstance(elem, dict): + list_tables.append(elem.lower()) + if select_where(json_file["union"][val1], literals): + if ( + len(select_where(json_file["union"][val1], literals)) + == 1 + ): + list_tables.append( + select_where( # pylint: disable=E1136 + json_file["union"][val1], literals + )[ # pylint: disable=E1136 + 0 + ].lower() # pylint: disable=E1136 + ) + else: + for elem in select_where( + json_file["union"][val1], literals + ): + list_tables.append(elem.lower()) + return set(list_tables) -# returns ProjectionAttributes for a statement and uses herefor different arts of sql-statements -def extract_pro_attributes(json_file, client): - attributes = [] - json_file = parse_query(json_file, client) - try: - if (single_select_dict(json_file) is not None) and ( - single_select_dict(json_file) - ): - attributes.extend(single_select_dict(json_file)) - if (single_select(json_file) is not None) and (single_select(json_file)): - attributes.extend(single_select(json_file)) - if (select_union(json_file) is not None) and (select_union(json_file)): - attributes.extend(select_union(json_file)) - if (list_of_select(json_file) is not None) and (list_of_select(json_file)): - attributes.extend(list_of_select(json_file)) - except Exception as e: - print(e) - attributes = ["Unknown"] - return list(attributes) + # returns ProjectionAttributes for a statement and uses herefor different arts of sql-statements + def extract_pro_attributes(self, json_file, literals): + attributes = [] + json_file = self._parser.parse_query(json_file) + try: + if (single_select_dict(json_file) is not None) and ( + single_select_dict(json_file) + ): + attributes.extend(single_select_dict(json_file)) + if (self._single_select(json_file, literals) is not None) and ( + self._single_select(json_file, literals) + ): + attributes.extend(self._single_select(json_file, literals)) + if (self._select_union(json_file, literals) is not None) and ( + self._select_union(json_file, literals) + ): + attributes.extend(self._select_union(json_file, literals)) + if (self._list_of_select(json_file) is not None) and ( + self._list_of_select(json_file) + ): + attributes.extend(self._list_of_select(json_file)) + except Exception as e: + print(e) + attributes = ["Unknown"] + return list(attributes) diff --git a/modules/fbs-sql-checker/api/query_processor.py b/modules/fbs-sql-checker/api/query_processor.py new file mode 100755 index 000000000..a5e2977fb --- /dev/null +++ b/modules/fbs-sql-checker/api/query_processor.py @@ -0,0 +1,776 @@ +# JSONCreator.py +from abc import ABCMeta, abstractmethod +from logging import Logger +from uuid import uuid4 +from typing import Optional +from venv import logger + +import sqlparse +from typeguard import typechecked + +from api.comparator.comparator import Comparator +from api.data_store import DataStore +from api.parser import QueryParser +from api.pro_attribute_checker import ProAttributeChecker +from api.sel_attribute_checker import SelAttributeChecker +from api.solution_fetcher import SolutionFetcher +from api.table_checker import TableChecker +from datetime import datetime +import model +from api.datatypes import Submission, Result, ResultV2 +from mask_aliases import SQLAliasMasker +import distance.distance_calc as d + +""" +rightStatements = [] +rightTables = [] +rightAtts = [] +rightStrings = [] +rightWhereAtts = [] +user_data = [] +tables, selAttributes, proAttributes, strings = [], [], [], [] +""" + + +@typechecked +class QueryProcessor(metaclass=ABCMeta): + @abstractmethod + def process(self, submission) -> Result: + pass + + +@typechecked +class QueryProcessorV2(QueryProcessor): + def __init__( + self, + comparator: Comparator, + solution_fetcher: SolutionFetcher, + log: Optional[Logger] = None, + ): + self._comparator = comparator + self._solution_fetcher = solution_fetcher + if logger is not None: + self._logger = Logger(self.__class__.__name__) + else: + self._logger = logger + + def process(self, submission: Submission) -> Result: + result = ResultV2.from_submission(submission) + solution, distance = self._solution_fetcher.fetch_solution( + submission.task_id, submission + ) + if solution is None and result.is_solution: + sqlparse.parse(submission.submission) + result.passed = True + result.parsable = True + return result + try: + result.errors = self._comparator.compare( + solution.statement, submission.submission + ) + result.passed = len(result.errors) == 0 + result.closest_solution = solution.uuid + result.min_distance = distance + result.parsable = True + except Exception as e: + logger.exception(e) + + return result + + +class QueryProcessorV1(QueryProcessor): + def __init__( + self, + parser: QueryParser, + table_checker: TableChecker, + pro_attribute_checker: ProAttributeChecker, + sel_attribute_checker: SelAttributeChecker, + data_store: DataStore, + ): + self._parser = parser + self._table_checker = table_checker + self._pro_attribute_checker = pro_attribute_checker + self._sel_attribute_checker = sel_attribute_checker + self._data_store = data_store + + def _parse_query(self, query: str, is_solution: bool, passed: bool) -> Result: + time = datetime.now().strftime("%Y-%m-%d %H:%M:%S") + ( + tables2, + pro_atts2, + sel_atts2, + strings2, + task_nr, + is_sol, + uuid, # submission primary key + course_id, + order_by2, + group_by2, + having2, + joins2, + wildcards2, + literals, + ) = ([], [], [], [], [], [], [], [], [], [], [], [], [], []) + try: + masker = SQLAliasMasker(query) + masker.mask_aliases_() + masked_query = masker.get_masked_query() + # Extract tables, selAttributes, proAttributes and strings + table_list = self._table_checker.extract_tables(masked_query) + if table_list != "Unknown": + tables2.extend(table_list[0]) + if table_list[1] != ["Empty"]: + try: + table_list.pop(0) + for x in table_list: + for y in x: + joins2.append(y) + except Exception: + joins2.append("Unknown") + else: + joins2.append("Empty") + extracted_pro_attributes = ( + self._pro_attribute_checker.extract_pro_attributes( + masked_query, literals + ) + ) + if extracted_pro_attributes != "Unknown": + pro_atts2.extend(extracted_pro_attributes) + extracted_string = self._sel_attribute_checker.extract_sel_attributes( + masked_query, literals + ) + if extracted_string != "Unknown": + sel_atts2.extend(extracted_string) + extracted_order_by = self._sel_attribute_checker.extract_order_by( + masked_query + ) + if extracted_order_by != "Unknown": + order_by2.extend(extracted_order_by) + extracted_group_by = self._sel_attribute_checker.extract_group_by( + masked_query + ) + if extracted_group_by != "Unknown": + group_by2.extend(extracted_group_by) + extracted_having = self._sel_attribute_checker.extract_having(masked_query) + if extracted_having != "Unknown": + having2.extend(extracted_having) + # strings2.extend(list(set(AWC.literal))) + # If TaskID or Submission ID not in data, return + """ + if "tid" not in data or "sid" not in data: + print("Task ID or Unique ID not in data") + return + task_nr = data["tid"] + uuid = data["sid"] + course_id = data["cid"] + is_sol = data["isSol"] + """ + # Save tables, selAttributes, proAttributes and strings to DB + uuid = str(uuid4()) + if self._parser.parse_query(masked_query) is not False: + self._insert_tables(query, uuid) + + # check and remove duplicates + # flag_duplicates(client, task_nr) + + # Check if it is a new solution and characteristics are right + ( + tables2, + pro_atts2, + sel_atts2, + strings2, + order_by2, + group_by2, + joins2, + having2, + wildcards2, + # new + closest_solution2, + min_distance2, + closest_id_2, + ) = self._check_solution_chars( + query, + is_solution, + passed, + task_nr, + uuid, + tables2, + pro_atts2, + sel_atts2, + strings2, + order_by2, + group_by2, + joins2, + having2, + ) + + # produce a JSON + record = self._return_json( + query, + is_sol, + passed, + uuid, + task_nr, + course_id, + tables2, + pro_atts2, + sel_atts2, + strings2, + order_by2, + group_by2, + joins2, + having2, + wildcards2, + time, + closest_solution2, + min_distance2, + closest_id_2, + ) + + # save JSON to DB + return record + except Exception as e: + print(e) + record = Result(uuid, course_id=course_id, task_nr=task_nr, time=time) + return record + + def _check_solution_chars( + self, + query: str, + is_sol: bool, + passed: bool, + task_nr, + uuid, + tables2, + pro_atts2, + sel_atts2, + strings2, + order_by2, + group_by2, + joins2, + having2, + ): + new_solution = True + ( + tables_right, + sel_attributes_right, + pro_attributes_right, + strings_right, + order_by_right, + group_by_right, + joins_right, + having_right, + wildcards, + # new + closest_solution, + min_distance, + closest_id, + ) = ( + False, + False, + False, + False, + False, + False, + False, + False, + False, + False, + False, + False, + ) + + min_distance = float("inf") # Set to positive infinity + closest_solution = None + # For every solution for given task + for x in self._data_store.query("Solutions", {"taskNumber": task_nr}): + # Extract Tables, Attributes etc. (cut out for better overview) + # compare the distance between the solution and the submission only if it is not a duplicate + if not x.get("duplicate", False): + + distance = d.get_distance(x["statement"], query) + + # insert distance to the solution + # mycol.update_one({"_id": x["_id"]}, {"$set": {"distance": distance}}, upsert=True) + + # check for min distance and use the corresponding solution + if distance < min_distance: + min_distance = distance + closest_solution = x[ + "id" + ] # choose the id of solution if it has the lowest distance + + self._data_store.upsert( + "Solutions", {"_id": closest_solution}, {"$set": {"distance": min_distance}} + ) + + if closest_solution: + ( + tables, # pylint: disable=W0621 + pro_attributes, + sel_attributes, + strings, # pylint: disable=W0621 + order_by, + group_by, + having, + joins, + ) = ( # pylint: disable=W0621 + [], + [], + [], + [], + [], + [], + [], + [], + ) + for y in self._data_store.query( + "Tables", {"id": closest_solution}, {"table": 1} + ): + tables.append(y["table"]) + for y in self._data_store.query( + "ProAttributes", {"id": closest_solution}, {"proAttribute": 1} + ): + pro_attributes.append(y["proAttribute"]) + for y in self._data_store.query( + "SelAttributes", {"id": closest_solution}, {"selAttribute": 1} + ): + sel_attributes.append(y["selAttribute"]) + for y in self._data_store.query( + "Strings", {"id": closest_solution}, {"string": 1} + ): + strings.append(y["string"]) + for y in self._data_store.query( + "OrderBy", {"id": closest_solution}, {"orderBy": 1, "sort": 1} + ): + order_by_value = [y["orderBy"]] + if ( + "sort" in y + ): # if there is no order in sort the default "asc" will be used + order_by_value.append(y["sort"]) + else: + order_by_value.append("asc") + order_by.append(order_by_value) + for y in self._data_store.query( + "GroupBy", {"id": closest_solution}, {"groupBy": 1} + ): + group_by.append(y["groupBy"]) + for y in self._data_store.query( + "Joins", {"id": closest_solution}, {"type": 1, "attr1": 1, "attr2": 1} + ): + join_value = [y["type"]] + if ( + "attr1" in y and "attr2" in y + ): # if there is no order in sort the default "asc" will be used + join_value.append(y["attr1"]) + join_value.append(y["attr2"]) + else: + join_value.append("Empty") + joins.append(join_value) + for y in self._data_store.query( + "Having", {"id": closest_solution}, {"havingAttribute": 1} + ): + having_value = y["havingAttribute"] + having.append(having_value) + if len(joins) == 0: + joins.append("Empty") + if passed: + if is_sol: + tables_right = True + sel_attributes_right = True + pro_attributes_right = True + strings_right = True + wildcards = True + order_by_right = True + group_by_right = True + joins_right = True + having_right = True + + # Compare them to tables, proAttributes etc of a given sql-query + if ( + tables == tables2 # pylint: disable=R0916 + and set(pro_attributes) == set(pro_atts2) + and set(sel_attributes) == set(sel_atts2) + and set(strings) == set(strings2) + and order_by == order_by2 + and joins == joins2 + and having == having2 + ): + # If they alle are same, it is not a new solution + new_solution = False + else: + # Check for tables, proAttributes etc. if they are right + if tables == tables2: + tables_right = True + if set(pro_attributes) == set(pro_atts2): + sel_attributes_right = True + if set(sel_attributes) == set(sel_atts2): + pro_attributes_right = True + if set(strings) == set(strings2): + strings_right = True + if ( + any("%" in s for s in strings) + and not any("%" in s for s in strings2) + or not any("%" in s for s in strings) + and any("%" in s for s in strings2) + ): + wildcards = False + else: + wildcards = True + if order_by == order_by2: + order_by_right = True + if group_by == group_by2: + group_by_right = True + if joins == joins2: + joins_right = True + if having == having2: + having_right = True + if is_sol: + if new_solution is True: + # Upload as a new Solution to DB + self._parse_single_stat_upload_solution(query, task_nr, uuid) + return ( + True, + True, + True, + True, + True, + True, + True, + True, + True, + closest_solution, + min_distance, + closest_id, + ) + # return if characteristics are True or False + return ( + tables_right, + sel_attributes_right, + pro_attributes_right, + strings_right, + order_by_right, + group_by_right, + joins_right, + having_right, + wildcards, + # new + closest_solution, + min_distance, + closest_id, + ) + + def _insert_tables(self, query: str, uuid: str): + literals = [] + table_list = self._table_checker.extract_tables(query) + joins = [] + tables = table_list[0] + if table_list[1] != ["Empty"]: + try: + table_list.pop(0) + for x in table_list: + for y in x: + joins.append(y) + except Exception: + joins.append("Unknown") + else: + joins.append("Empty") + if len(tables) == 1: + record = model.json_table(uuid, tables[0]) + self._data_store.store("Tables", record) + if joins[0] != "Empty": + try: + self._data_store.store("Joins", record) + except Exception: + print("Error while reading joins.") + elif len(tables) > 1 and not isinstance( + self._table_checker.extract_tables(query), str + ): + for val in tables: + record = model.json_table(uuid, val) + self._data_store.store("Tables", record) + if joins[0] != "Empty": + try: + for y in joins: + record = model.json_join_attribute(uuid, y) + self._data_store.store("Joins", record) + except Exception: + print("Error while reading joins.") + + pro_attributes = self._pro_attribute_checker.extract_pro_attributes( + query, literals + ) + if len(pro_attributes) == 1: + record = model.json_pro_attribute(uuid, pro_attributes[0]) + self._data_store.store("ProAttributes", record) + elif len(pro_attributes) > 1 and not isinstance(pro_attributes, str): + for val in pro_attributes: + record = model.json_pro_attribute(uuid, val) + self._data_store.store("ProAttributes", record) + + sel_attributes = self._sel_attribute_checker.extract_sel_attributes( + query, literals + ) + if len(sel_attributes) == 1: + record = model.json_sel_attribute(uuid, sel_attributes[0]) + self._data_store.store("SelAttributes", record) + elif len(sel_attributes) > 1 and not isinstance(sel_attributes, str): + for val in sel_attributes: + record = model.json_sel_attribute(uuid, val) + self._data_store.store("SelAttributes", record) + + strings = list(set(literals)) + if len(strings) == 1: + record = model.json_string(uuid, strings[0]) + self._data_store.store("Strings", record) + elif len(strings) > 1 and not isinstance(strings, str): + for val in strings: + record = model.json_string(uuid, val) + self._data_store.store("Strings", record) + + order_by = self._sel_attribute_checker.extract_order_by(query) + if len(order_by) == 1: + record = model.json_order_by_attribute(uuid, order_by[0]) + self._data_store.store("OrderBy", record) + elif len(order_by) > 1 and not isinstance(order_by, str): + for val in order_by: + record = model.json_order_by_attribute(uuid, val) + self._data_store.store("OrderBy", record) + + group_by = self._sel_attribute_checker.extract_group_by(query) + if len(group_by) == 1: + record = model.json_group_by_attribute(uuid, group_by[0]) + self._data_store.store("GroupBy", record) + elif len(group_by) > 1 and not isinstance(group_by, str): + for val in group_by: + record = model.json_group_by_attribute(uuid, val) + self._data_store.store("GroupBy", record) + + having = self._sel_attribute_checker.extract_having(query) + if len(having) == 1: + record = model.json_having_attribute(uuid, having[0]) + self._data_store.store("Having", record) + elif len(having) > 1 and not isinstance(having, str): + for val in having: + record = model.json_having_attribute(uuid, val) + self._data_store.store("Having", record) + + # self._sel_attribute_checker.literal = [] + # user_data.clear() + + # return JSON to be pasted to DB + def _return_json( + self, + query, + is_sol, + passed, + uuid, + task_nr, + course_id, + tables_right, + pro_attributes_right, + sel_attributes_right, + strings_right, + order_by_right, + group_by_right, + joins_right, + having_right, + wildcards, + time, + closest_solution, + min_distance, + closest_id, + ): + + # user_data.append(passed) + if self._parser.parse_query(query) is not False: + # produce a json to be pasted to DB + record = Result( + uuid, + course_id, + query, + passed, + True, + task_nr, + is_sol, + tables_right, + sel_attributes_right, + pro_attributes_right, + strings_right, + order_by_right, + group_by_right, + joins_right, + having_right, + wildcards, + time, + closest_solution, + min_distance, + closest_id, + ) + return record + # produce a json if the sql-query is not parsable + record = Result(uuid, course_id=course_id, task_nr=task_nr, time=time) + return record + + def process(self, submission: Submission) -> Result: + print("input", submission) + result = self._parse_query( + submission.submission, submission.is_solution, submission.passed + ) + if result is None: + raise ValueError("query parsing resulted in none") + result.course_id = submission.course_id + result.task_nr = submission.task_id + print("output", result) + self._store_query( + result.to_db_dict( + course_id=submission.course_id, + task_id=submission.task_id, + user_id=submission.user_id, + submission_id=submission.submission_id, + attempt=submission.attempt, + ) + ) + return result + + def _store_query(self, query): + self._data_store.store("Queries", query) + + def _parse_single_stat_upload_solution(self, data, task_nr, uuid): + record = model.prod_solution_json(data, uuid, task_nr) + self._data_store.store("Solutions", record) + + +# Idea of a function to flag duplicate queries from the database +def flag_duplicates(client, task_nr): + mydb = client.get_default_database() + mycol = mydb["Solutions"] + # set all queries to not duplicate + mycol.update_many({"taskNumber": task_nr}, {"$set": {"duplicate": False}}) + queries = list(mycol.find({"taskNumber": task_nr})) + + # list to keep track of queries to flag + duplicates = [] + + # Compare each document with every other document + query_length = len(queries) + for i in range(query_length): + for j in range( + i + 1, query_length + ): # Start from i + 1 to avoid comparing with itself and re-comparing + query1 = queries[i] + query2 = queries[j] + + if d.get_distance(query1["statement"], query2["statement"]) == 0: + duplicates.append(query2["_id"]) + + # Flag duplicates based on gathered IDs + for duplicate_id in set(duplicates): + mycol.update_one( + {"_id": duplicate_id}, {"$set": {"duplicate": True}}, upsert=True + ) + + +# Parse a solution and upload it to DB + + +""" + +# Returns a json file which extracts Tables and Attributes +def prod_json_not_parsable(_id, cid, task_nr, time): + # Create dictionary + value = { + "id": str(_id), + "cid": cid, + "taskNumber": task_nr, + "statement": user_data[3], + "queryRight": user_data[0], + "parsable": False, + "tablesRight": None, + "selAttributesRight": None, + "proAttributesRight": None, + "stringsRight": None, + "userId": user_data[1], + "attempt": user_data[2], + "orderbyRight": None, + "havingRight": None, + "wildcards": None, + "time": time, + } + return value + + +# Insert data of Tables, proAttributes, selAttributes and Strings to Database + +# Returns a json file which extracts characteristics +# and tells which of them are wrong +def prod_json( + _id, + course_id, + test_sql, + passed, + task_nr, + is_sol, + tables_right, + sel_attributes_right, + pro_attributes_right, + strings_right, + order_by_right, + group_by_right, + joins_right, + having_right, + wildcards, + time, + closest_solution, + min_distance, + closest_id, +): + # save data if it is a manual solution + #if is_sol is True: + # user_data.extend([True]) + # user_data.extend([0]) + # user_data.extend([0]) + value = { + "id": str(closest_id), + "courseId": course_id, + "taskNumber": task_nr, + "statement": test_sql, + "queryRight": passed, + "parsable": True, + "isSolution": is_sol, + "tablesRight": tables_right, + "selAttributesRight": sel_attributes_right, + "proAttributesRight": pro_attributes_right, + "stringsRight": strings_right, + "userId": None, + "attempt": 0, + "orderByRight": order_by_right, + "groupByRight": group_by_right, + "joinsRight": joins_right, + "havingRight": having_right, + "wildcards": wildcards, + "time": time, + "solutionID": closest_solution, + "distance": min_distance, + } + user_data.clear() + return value + +def insert_not_parsable(uuid, submission, client): + mydb = client.get_default_database() + mycollection = mydb["NotParsable"] + record = model.json_not_parsable(uuid, submission) + mycollection.insert_one(record) + +def return_json_not_parsable(elem): + # Extract informations from a sql-query-json + if "passed" in elem: + user_data.append(elem["passed"]) + if "userId" in elem: + user_data.append(elem["userId"]) + if "attempt" in elem: + user_data.append(elem["attempt"]) + if "submission" in elem: + user_data.append(elem["submission"]) + return user_data +""" diff --git a/modules/fbs-sql-checker/api/sel_attribute_checker.py b/modules/fbs-sql-checker/api/sel_attribute_checker.py old mode 100644 new mode 100755 index 6464054ad..c5182e8d4 --- a/modules/fbs-sql-checker/api/sel_attribute_checker.py +++ b/modules/fbs-sql-checker/api/sel_attribute_checker.py @@ -1,297 +1,170 @@ -# sel_attribute_checker.py +from api.attribute_common import select_where +from api.parser import QueryParser -from parser import parse_query -import pro_attribute_checker as AC -select_commands = [ - "sum", - "count", - "round", - "distinct", - "sec_to_time", - "avg", - "max", - "min", - "time_to_sec", - "like", - "between", - "div", - "year", - "mul", -] -equals = ["eq", "neq", "gte", "gt", "lt", "lte"] +class SelAttributeChecker: + def __init__(self, parser: QueryParser): + self._parser = parser -literal = [] - -# Returns SelectionAttributes as a list if there is a where -def select_where(json_file): - list_tables = [] - if "where" in json_file: - for val1 in json_file["where"]: - if isinstance(json_file["where"][val1], str): - if "." in json_file["where"][val1]: - list_tables.append(json_file["where"][val1].split(".")[1].lower()) - else: - list_tables.append(json_file["where"][val1].lower()) - if isinstance(json_file["where"][val1], dict): - for val2 in json_file["where"][val1]: - if isinstance(val2, str): - if AC.single_select_dict(val2): - list_tables.extend(AC.single_select_dict(val2)) - elif "." in val2: - list_tables.append(val2.split(".")[1].lower()) - else: - list_tables.append(val2.lower()) - for i in range(len(json_file["where"][val1])): - if isinstance(json_file["where"][val1][i], dict): - for val3 in json_file["where"][val1][i]: - if val3 in select_commands: - for val4 in json_file["where"][val1][i][val3]: - if isinstance(val4, str): - if "." in val4: - list_tables.append(val4.split(".")[1].lower()) - else: - list_tables.append(val4.lower()) - if isinstance(json_file["where"][val1], list) and ( - len(json_file["where"][val1]) > 1 - ): - if isinstance(json_file["where"][val1][1], dict): - if AC.single_select_dict(json_file["where"][val1][1]): - list_tables.extend( - AC.single_select_dict(json_file["where"][val1][1]) - ) - if "literal" in json_file["where"][val1][1]: - literal.append(json_file["where"][val1][1]["literal"]) - for elem in json_file["where"][val1]: - if isinstance(elem, str): - if "." in elem: - list_tables.append(elem.split(".")[1].lower()) - else: - list_tables.append(elem.lower()) - for i in range(len(json_file["where"][val1])): - if not isinstance(json_file["where"][val1][i], int): - for elem in json_file["where"][val1][i]: - if elem in equals: - if isinstance(json_file["where"][val1][i][elem], list): - for j in range(len(json_file["where"][val1][i][elem])): - if isinstance( - json_file["where"][val1][i][elem][j], str - ): - if "." in json_file["where"][val1][i][elem][j]: - list_tables.append( - json_file["where"][val1][i][elem][ - j - ].split(".")[1] - ) - else: - list_tables.append( - json_file["where"][val1][i][elem][j] - ) - if isinstance( - json_file["where"][val1][i][elem][j], dict - ): - for elem1 in json_file["where"][val1][i][elem][ - j - ]: - if elem1 in select_commands: - if isinstance( - json_file["where"][val1][i][elem][ - j - ][elem1], - str, - ): - if ( - "." - in json_file["where"][val1][i][ - elem - ][j][elem1] - ): - list_tables.append( - json_file["where"][val1][i][ - elem - ][j][elem1].split(".")[1] - ) - else: - list_tables.append( - json_file["where"][val1][i][ - elem - ][j][elem1] - ) - if elem1 == "literal": - literal.append( - json_file["where"][val1][i][elem][ - j - ][elem1] - ) - list_tables.extend( - AC.single_select_dict( - json_file["where"][val1][i][elem][j] - ) - ) - if elem == "where": - list_tables.extend( - select_where(json_file["where"][val1][i]) - ) - return set(list_tables) - - -# returns SelectionAttributes for a statement and uses herefor different arts of sql-statements -def extract_sel_attributes(json_file, client): - where_attributes = [] - json_file = parse_query(json_file, client) - try: - if (select_where(json_file) is not None) and (select_where(json_file)): - where_attributes.extend([select_where(json_file)]) - except Exception as e: - print(e) - where_attributes = [["Unknown"]] - if len(where_attributes) > 0: - return list(where_attributes[0]) - return list(where_attributes) - - -def extract_order_by(json_file, client): - json_file = parse_query(json_file, client) - order_by = [] - order_by_list = list(iterate(json_file, "orderby")) - for s in order_by_list: - value = [] + # returns SelectionAttributes for a statement and uses herefor different arts of sql-statements + def extract_sel_attributes(self, json_file, literals): + where_attributes = [] + json_file = self._parser.parse_query(json_file) try: - value.append(s["value"]) - if "sort" in s: - value.append(s["sort"]) - else: - value.append("asc") - order_by.append(value) - except Exception: - for y in s: - value = [] - value.append(y["value"]) - try: - value.append(y["sort"]) - except Exception: + if (select_where(json_file, literals) is not None) and ( + select_where(json_file, literals) + ): + where_attributes.extend([select_where(json_file, literals)]) + except Exception as e: + print(e) + where_attributes = [["Unknown"]] + if len(where_attributes) > 0: + return list(where_attributes[0]) + return list(where_attributes) + + def extract_order_by(self, json_file): + json_file = self._parser.parse_query(json_file) + order_by = [] + order_by_list = list(self._iterate(json_file, "orderby")) + for s in order_by_list: + value = [] + try: + value.append(s["value"]) + if "sort" in s: + value.append(s["sort"]) + else: value.append("asc") order_by.append(value) - if len(order_by) == 0: - order_by = "Unknown" - return order_by - - -def extract_group_by(json_file, client): - json_file = parse_query(json_file, client) - group_by = [] - group_by_list = list(iterate(json_file, "groupby")) - for s in group_by_list: - try: - group_by.append(s["value"]) - except Exception: - for y in s: - group_by.append(y["value"]) - if len(group_by) == 0: - group_by = "Unknown" - return group_by - - -def extract_having(json_file, client): - json_file = parse_query(json_file, client) - all_having = [ - [] - ] # reason having in check_solution_chars() in json_creator is [[[...]]] - having = [] - having_list = list(iterate(json_file, "having")) - att = [] # if necessary , not in use now...: customerId - att_operator = [] # count - att_op_compare = [] # gt... - val_compare = [] # value example 5 - for s in having_list: - parse_one_cond(s, having, att, att_operator, att_op_compare, val_compare) - all_having[0].append(having) # or having_order - if len(all_having) == 0: - all_having = "Unknown" - if all_having == [[[]]]: - return [] - return all_having - - -def iterate(data, param): - if isinstance(data, list): - for item in data: - yield from iterate(item, param) - elif isinstance(data, dict): - for key, item in data.items(): - if key == param: - yield item - else: - yield from iterate(item, param) - - -def parse_one_cond( # pylint: disable = R1710 - s, having, att, att_operator, att_op_compare, val_compare -): # pylint: disable=R1710 - if isinstance(s, dict): - for i in s.values(): - if isinstance(i, str): - having.append(i) - if isinstance(i, dict): - key, values = get_vals_and_keys(i) - if not isinstance(key, dict) and not isinstance(key, list): - having.append(key) - if not isinstance(key, dict) and not isinstance(key, list): - having.append(values) - if isinstance(values, dict): - key1, values1 = get_vals_and_keys(values) - if not isinstance(key1, dict) and not isinstance(key1, list): - having.append(key1) - if isinstance(values1, str): - having.append(values1) - if isinstance(values1, dict): - key2, values2 = get_vals_and_keys(values1) - if not isinstance(key2, dict) and not isinstance(key2, list): - having.append(key2) - if isinstance(values2, dict) and not isinstance(values2, list): - having.append(values2) - for i in s: # keys - if not isinstance(i, dict) and not isinstance(i, list): - having.append(i) - if not isinstance(s[i], str): - for t in s[i]: - if not isinstance(t, str): - if isinstance(t, int): - if not isinstance(t, dict) and not isinstance(t, list): - having.append(t) - if isinstance(t, dict): - values = list(t.values()) - keys = list(t.keys()) - if not isinstance(keys[0], dict) and not isinstance( - keys[0], list + except Exception: + for y in s: + value = [] + value.append(y["value"]) + try: + value.append(y["sort"]) + except Exception: + value.append("asc") + order_by.append(value) + if len(order_by) == 0: + order_by = "Unknown" + return order_by + + def extract_group_by(self, json_file): + json_file = self._parser.parse_query(json_file) + group_by = [] + group_by_list = list(self._iterate(json_file, "groupby")) + for s in group_by_list: + try: + group_by.append(s["value"]) + except Exception: + for y in s: + group_by.append(y["value"]) + if len(group_by) == 0: + group_by = "Unknown" + return group_by + + def extract_having(self, json_file): + json_file = self._parser.parse_query(json_file) + all_having = [ + [] + ] # reason having in check_solution_chars() in json_creator is [[[...]]] + having = [] + having_list = list(self._iterate(json_file, "having")) + att = [] # if necessary , not in use now...: customerId + att_operator = [] # count + att_op_compare = [] # gt... + val_compare = [] # value example 5 + for s in having_list: + self._parse_one_cond( + s, having, att, att_operator, att_op_compare, val_compare + ) + all_having[0].append(having) # or having_order + if len(all_having) == 0: + all_having = "Unknown" + if all_having == [[[]]]: + return [] + return all_having + + def _iterate(self, data, param): + if isinstance(data, list): + for item in data: + yield from self._iterate(item, param) + elif isinstance(data, dict): + for key, item in data.items(): + if key == param: + yield item + else: + yield from self._iterate(item, param) + + def _parse_one_cond( # pylint: disable = R1710 + self, s, having, att, att_operator, att_op_compare, val_compare + ): # pylint: disable=R1710 + if isinstance(s, dict): + for i in s.values(): + if isinstance(i, str): + having.append(i) + if isinstance(i, dict): + key, values = self._get_vals_and_keys(i) + if not isinstance(key, dict) and not isinstance(key, list): + having.append(key) + if not isinstance(key, dict) and not isinstance(key, list): + having.append(values) + if isinstance(values, dict): + key1, values1 = self._get_vals_and_keys(values) + if not isinstance(key1, dict) and not isinstance(key1, list): + having.append(key1) + if isinstance(values1, str): + having.append(values1) + if isinstance(values1, dict): + key2, values2 = self._get_vals_and_keys(values1) + if not isinstance(key2, dict) and not isinstance( + key2, list ): - having.append(keys[0]) - if not isinstance(values[0], dict) and not isinstance( - values[0], list + having.append(key2) + if isinstance(values2, dict) and not isinstance( + values2, list ): - having.append(values[0]) - if len(values) > 0: - for f in values: - if not isinstance(f, str): - for r in f: - if not isinstance( - r, dict - ) and not isinstance(r, list): - having.append(r) - if isinstance(r, dict): - parse_one_cond( - r, - having, - att, - att_operator, - att_op_compare, - val_compare, - ) - return having - - -def get_vals_and_keys(s): # pylint: disable=R1710 - if isinstance(s, dict): - keys_list = list(s.keys()) - values = list(s.values()) - return keys_list[0], values[0] + having.append(values2) + for i in s: # keys + if not isinstance(i, dict) and not isinstance(i, list): + having.append(i) + if not isinstance(s[i], str): + for t in s[i]: + if not isinstance(t, str): + if isinstance(t, int): + if not isinstance(t, dict) and not isinstance(t, list): + having.append(t) + if isinstance(t, dict): + values = list(t.values()) + keys = list(t.keys()) + if not isinstance(keys[0], dict) and not isinstance( + keys[0], list + ): + having.append(keys[0]) + if not isinstance(values[0], dict) and not isinstance( + values[0], list + ): + having.append(values[0]) + if len(values) > 0: + for f in values: + if not isinstance(f, str): + for r in f: + if not isinstance( + r, dict + ) and not isinstance(r, list): + having.append(r) + if isinstance(r, dict): + self._parse_one_cond( + r, + having, + att, + att_operator, + att_op_compare, + val_compare, + ) + return having + + def _get_vals_and_keys(s): # pylint: disable=R1710 + if isinstance(s, dict): + keys_list = list(s.keys()) + values = list(s.values()) + return keys_list[0], values[0] diff --git a/modules/fbs-sql-checker/api/solution_fetcher.py b/modules/fbs-sql-checker/api/solution_fetcher.py new file mode 100644 index 000000000..c6273d4a6 --- /dev/null +++ b/modules/fbs-sql-checker/api/solution_fetcher.py @@ -0,0 +1,56 @@ +import math +from abc import ABCMeta, abstractmethod +from typing import Tuple, Optional + +from api.data_store import DataStore +from api.distance.distance_calc import DistanceCalculator +from api.datatypes import Submission, Result + + +class SolutionFetcher(metaclass=ABCMeta): + @abstractmethod + def fetch_solution( + self, task_id: int, submission: Submission + ) -> Tuple[Optional[Result], float]: + pass + + +class FirstSolutionFetcher(SolutionFetcher): + def __init__(self, datastore: DataStore): + self._datastore = datastore + + def fetch_solution( + self, task_id: int, submission: Submission + ) -> Tuple[Optional[Result], float]: + solution = self._datastore.query("Queries", {"taskId": task_id, "isSolution": True}) + if len(solution) == 0: + return None, 0.0 + return ( + Result.from_db_dict(solution[0]), + 1.0, + ) + + +class NearestSolutionFetcher(SolutionFetcher): + def __init__(self, datastore: DataStore, distance_calculator: DistanceCalculator): + self._datastore = datastore + self._distance_calculator = distance_calculator + + def fetch_solution( + self, task_id: int, submission: Submission + ) -> Tuple[Result, float]: + solutions = self._datastore.query( + "Queries", {"taskId": task_id, "isSolution": True} + ) + solutions = [Result.from_db_dict(solution) for solution in solutions] + min_distance = math.inf + min_solution = None + for solution in solutions: + distance = self._distance_calculator.calculate_distance( + solution.statement, submission.submission + ) + if distance < min_distance: + min_distance = distance + min_solution = solution + + return min_solution, min_distance diff --git a/modules/fbs-sql-checker/api/table_checker.py b/modules/fbs-sql-checker/api/table_checker.py old mode 100644 new mode 100755 index 86ca6faef..17071e4a4 --- a/modules/fbs-sql-checker/api/table_checker.py +++ b/modules/fbs-sql-checker/api/table_checker.py @@ -1,183 +1,198 @@ # table_checker.py -from parser import parse_query import formatting as f +from api.query_processor import QueryParser -# Return a single From and a where-clause -def is_single_from_where(json_file): - list_tables = [] - if not (f.is_string(json_file)) and (f.is_from(json_file)): - if isinstance(json_file["from"], str): - if json_file["from"].count(json_file["from"]) == 1: - list_tables.append(json_file["from"].lower()) - if isinstance(json_file["from"], dict): - if "value" in json_file["from"]: - if isinstance(json_file["from"]["value"], str): - list_tables.append(json_file["from"]["value"].lower()) - for val in json_file: - if val == "where": - if isinstance(json_file[val], list): - for val1 in json_file[val]: - if val1 == "in": - for i in range(len(json_file[val][val1])): - list_tables.extend( - is_single_from_where(json_file[val][val1][i]) - ) - elif isinstance(json_file[val], dict): - for val1 in json_file["where"]: - for val2 in json_file["where"][val1]: - if val2 == "exists": - if isinstance(json_file["where"][val1][val2], dict): - if ( - is_single_from_where(json_file["where"][val1][val2]) - ) and ( - is_single_from_where(json_file["where"][val1][val2]) - not in list_tables - ): - list_tables.extend( - is_single_from_where( - json_file["where"][val1][val2] - ) - ) - if isinstance(json_file["where"][val1], list): - if len(json_file["where"][val1]) > 1: - if isinstance(json_file["where"][val1][1], dict): - if (is_single_from_where(val2)) and ( - is_single_from_where(val2) is not None - ): - for elem in is_single_from_where(val2): - if elem not in list_tables: - list_tables.append(elem) - elif ( - (is_join(val2)) - and (is_join(val2) not in list_tables) - and (is_join(val2) is not None) - ): - list_tables.append(is_join(val2)) - if ( - (is_join(json_file)) - and (is_join(json_file) not in list_tables) - and (is_join(json_file) is not None) - ): - for elem in is_join(json_file): - if elem not in list_tables: - list_tables.append(elem) - return list_tables +class TableChecker: + def __init__(self, parser: QueryParser): + self._parser = parser -# Return Tables as a List if there is a join --> works for a python List too -def is_join(json_file): # pylint: disable=R1710 - list_tables = [] - list_joins = [] - return_list = [] - is_join_var = False - join_type = "" - if not (f.is_string(json_file)) and (f.is_from(json_file)): - for val in json_file["from"]: - if ( - (isinstance(val, str)) - and ((val not in "value") and (val not in "name")) - and (len(val) > 1) - ): - list_tables.append(val.lower()) - if isinstance(val, dict): - for element in val: - if element == "value": - list_tables.append(val[element].lower()) - if isinstance(val[element], str) and ( - (element not in "name") and (element not in "using") - ): - list_tables.append(val[element].lower()) - if is_join_var is True: - for val1 in val[element]: - insert = [] - insert.append(join_type) - for val in val[element][val1]: # pylint: disable=W0621 - insert.append(val) - list_joins.append(insert) - is_join_var = False - if ( - element in "inner join" - or element in "left join" - or element in "join" - or element in "right join" - or element in "outer join" - ): - is_join_var = True - join_type = element - for val1 in val[element]: - if val1 == "value": - list_tables.append(val[element][val1].lower()) - if val == "value": - if isinstance(json_file["from"]["value"], dict): - if ( - ( - is_single_from_where(json_file["from"]["value"]) - not in list_tables - ) - and ( - is_single_from_where(json_file["from"]["value"]) is not None - ) - and (is_single_from_where(json_file["from"]["value"])) - ): - list_tables.extend( - is_single_from_where(json_file["from"]["value"]) - ) - for elem in json_file["from"]["value"]: # pylint: disable=W0612 - if ( - (is_union(json_file["from"]["value"]) is not None) - and (is_union(json_file["from"]["value"])) - and (is_union(json_file["from"]["value"]) not in list_tables) - ): - list_tables.extend(is_union(json_file["from"]["value"])) - if ( - (is_join(json_file["from"]["value"]) is not None) - and (is_join(json_file["from"]["value"])) - and (is_join(json_file["from"]["value"]) not in list_tables) - ): - list_tables.extend(is_join(json_file["from"]["value"])) - return_list.append(sorted(list(set(list_tables)))) - return_list.append(list_joins) - return return_list - - -# Returns tables as a List if it is a union -def is_union(json_file): # pylint: disable=R1710 - list_tables = [] - if not f.is_string(json_file): + def _is_single_from_where(self, json_file): + list_tables = [] + if not (f.is_string(json_file)) and (f.is_from(json_file)): + if isinstance(json_file["from"], str): + if json_file["from"].count(json_file["from"]) == 1: + list_tables.append(json_file["from"].lower()) + if isinstance(json_file["from"], dict): + if "value" in json_file["from"]: + if isinstance(json_file["from"]["value"], str): + list_tables.append(json_file["from"]["value"].lower()) for val in json_file: - if val == "union": - for i in range(len(json_file[val])): - if (is_single_from_where(json_file[val][i])) is not None: - list_tables.extend(is_single_from_where(json_file[val][i])) - else: - list_tables.append(is_join(json_file[val][i])) - return sorted(set(list_tables)) - return None + if val == "where": + if isinstance(json_file[val], list): + for val1 in json_file[val]: + if val1 == "in": + for i in range(len(json_file[val][val1])): + list_tables.extend( + self._is_single_from_where(json_file[val][val1][i]) + ) + elif isinstance(json_file[val], dict): + for val1 in json_file["where"]: + for val2 in json_file["where"][val1]: + if val2 == "exists": + if isinstance(json_file["where"][val1][val2], dict): + if ( + self._is_single_from_where( + json_file["where"][val1][val2] + ) + ) and ( + self._is_single_from_where( + json_file["where"][val1][val2] + ) + not in list_tables + ): + list_tables.extend( + self._is_single_from_where( + json_file["where"][val1][val2] + ) + ) + if isinstance(json_file["where"][val1], list): + if len(json_file["where"][val1]) > 1: + if isinstance(json_file["where"][val1][1], dict): + if (self._is_single_from_where(val2)) and ( + self._is_single_from_where(val2) is not None + ): + for elem in self._is_single_from_where(val2): + if elem not in list_tables: + list_tables.append(elem) + elif ( + (self._is_join(val2)) + and (self._is_join(val2) not in list_tables) + and (self._is_join(val2) is not None) + ): + list_tables.append(self._is_join(val2)) + if ( + (self._is_join(json_file)) + and (self._is_join(json_file) not in list_tables) + and (self._is_join(json_file) is not None) + ): + for elem in self._is_join(json_file): + if elem not in list_tables: + list_tables.append(elem) + return list_tables + + def _is_join(self, json_file): # pylint: disable=R1710 + list_tables = [] + list_joins = [] + return_list = [] + is_join_var = False + join_type = "" + if not (f.is_string(json_file)) and (f.is_from(json_file)): + for val in json_file["from"]: + if ( + (isinstance(val, str)) + and ((val not in "value") and (val not in "name")) + and (len(val) > 1) + ): + list_tables.append(val.lower()) + if isinstance(val, dict): + for element in val: + if element == "value": + list_tables.append(val[element].lower()) + if isinstance(val[element], str) and ( + (element not in "name") and (element not in "using") + ): + list_tables.append(val[element].lower()) + if is_join_var is True: + for val1 in val[element]: + insert = [] + insert.append(join_type) + for val in val[element][val1]: # pylint: disable=W0621 + insert.append(val) + list_joins.append(insert) + is_join_var = False + if ( + element in "inner join" + or element in "left join" + or element in "join" + or element in "right join" + or element in "outer join" + ): + is_join_var = True + join_type = element + for val1 in val[element]: + if val1 == "value": + list_tables.append(val[element][val1].lower()) + if val == "value": + if isinstance(json_file["from"]["value"], dict): + if ( + ( + self._is_single_from_where(json_file["from"]["value"]) + not in list_tables + ) + and ( + self._is_single_from_where(json_file["from"]["value"]) + is not None + ) + and (self._is_single_from_where(json_file["from"]["value"])) + ): + list_tables.extend( + self._is_single_from_where(json_file["from"]["value"]) + ) + for elem in json_file["from"]["value"]: # pylint: disable=W0612 + if ( + (self._is_union(json_file["from"]["value"]) is not None) + and (self._is_union(json_file["from"]["value"])) + and ( + self._is_union(json_file["from"]["value"]) + not in list_tables + ) + ): + list_tables.extend( + self._is_union(json_file["from"]["value"]) + ) + if ( + (self._is_join(json_file["from"]["value"]) is not None) + and (self._is_join(json_file["from"]["value"])) + and ( + self._is_join(json_file["from"]["value"]) + not in list_tables + ) + ): + list_tables.extend( + self._is_join(json_file["from"]["value"]) + ) + return_list.append(sorted(list(set(list_tables)))) + return_list.append(list_joins) + return return_list + def _is_union(self, json_file): # pylint: disable=R1710 + list_tables = [] + if not f.is_string(json_file): + for val in json_file: + if val == "union": + for i in range(len(json_file[val])): + if (self._is_single_from_where(json_file[val][i])) is not None: + list_tables.extend( + self._is_single_from_where(json_file[val][i]) + ) + else: + list_tables.append(self._is_join(json_file[val][i])) + return sorted(set(list_tables)) + return None -# return tables for a statement and uses therefor different arts of sql-statements -def extract_tables(json_file, client): - json_file = parse_query(json_file, client) - try: - if is_single_from_where(json_file) is not None and is_single_from_where( - json_file - ): - tables = is_single_from_where(json_file) - elif ( - is_join(json_file) is not None - and is_join(json_file) is not [] # pylint: disable=R0123 - ): # pylint: disable=R0123 - tables = is_join(json_file) - else: - tables = is_union(json_file) - except Exception: - tables = ["Unknown"] - # Gibt Indexerror bei falschen Queries - if type(tables[0]) == str: # pylint: disable=C0123 - tables[0] = [tables[0]] - if len(tables) < 2: - tables.append("Empty") - if len(tables[1]) == 0: - tables[1].append("Empty") - return tables + def extract_tables(self, json_file): + json_file = self._parser.parse_query(json_file) + try: + if self._is_single_from_where( + json_file + ) is not None and self._is_single_from_where(json_file): + tables = self._is_single_from_where(json_file) + elif ( + self._is_join(json_file) is not None + and self._is_join(json_file) is not [] # pylint: disable=R0123 + ): # pylint: disable=R0123 + tables = self._is_join(json_file) + else: + tables = self._is_union(json_file) + except Exception: + tables = ["Unknown"] + # Gibt Indexerror bei falschen Queries + if type(tables[0]) == str: # pylint: disable=C0123 + tables[0] = [tables[0]] + if len(tables) < 2: + tables.append("Empty") + if len(tables[1]) == 0: + tables[1].append("Empty") + return tables diff --git a/modules/fbs-sql-checker/api/test_json_creator.py b/modules/fbs-sql-checker/api/test_json_creator.py new file mode 100644 index 000000000..6623081f4 --- /dev/null +++ b/modules/fbs-sql-checker/api/test_json_creator.py @@ -0,0 +1,310 @@ +import unittest +from random import randint + +import mongomock +from sympy import false +from typeguard import typechecked + +from api.data_store import AbstractMongoStore +from api.query_processor import QueryProcessor, QueryParser, DataStore +from api.datatypes import Submission +from api.pro_attribute_checker import ProAttributeChecker +from api.sel_attribute_checker import SelAttributeChecker +from api.table_checker import TableChecker + + +class TestDataStorage(AbstractMongoStore): + def __init__(self): + self.storage = {} + + def get_collection(self, collection: str): + if collection not in self.storage: + self.storage[collection] = mongomock.MongoClient().db.collection + + return self.storage[collection] + + +@typechecked +class QueryProcessorTestBase(unittest.TestCase): + def _create_qb(self) -> QueryProcessor: + qparser = QueryParser() + storage = TestDataStorage() + qp = QueryProcessor( + qparser, + TableChecker(qparser), + ProAttributeChecker(qparser), + SelAttributeChecker(qparser), + storage, + ) + return qp + + +@typechecked +class QueryProcessorParsingTest(QueryProcessorTestBase): + def test_full_process(self): + qp = self._create_qb() + qp.process( + Submission( + "SELECT * FROM table", + ) + ) + + def test_parse_without_solution(self): + qb = self._create_qb() + res = qb.process( + Submission( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + is_solution=False, + passed=False, + ) + ) + assert res is not None, "should not be none without solution" + + def test_complex_query(self): + qb = self._create_qb() + res = qb.process( + Submission( + "SELECT DISTINCT s.student_name, c.course_name FROM student_in_course sic1 JOIN student_in_course sic2 ON sic1.student_id = sic2.student_id AND sic1.course_id = sic2.course_id AND sic1.semester_id <> sic2.semester_id JOIN students s ON sic1.student_id = s.student_id JOIN course c ON sic1.course_id = c.course_id;", + is_solution=True, + passed=True, + ) + ) + assert res is not None, "should not be none for result" + + +@typechecked +class QueryProcessorQueryTestBast(QueryProcessorTestBase): + def _query_test_case( + self, + solution_query: str, + submission_query: str, + should_sel_attributes_right: bool = True, + should_pro_attributes_right: bool = True, + should_group_by_right: bool = True, + should_joins_right: bool = True, + should_order_by_right: bool = True, + should_strings_right: bool = True, + should_tables_right: bool = True, + ): + qb = self._create_qb() + course_id = randint(1, 100) + task_id = randint(1, 1000) + res = qb.process( + Submission.new_solution( + solution_query, course_id=course_id, task_id=task_id + ) + ) + errors = [] + + if res is None: + errors.append( + { + "message": "res should not be none for solution insertion", + "should": "not None", + "is": res, + } + ) + + res = qb.process( + Submission(submission_query, course_id=course_id, task_id=task_id) + ) + + if res is None: + errors.append( + { + "message": "res should not be none for checking", + "should": "not None", + "is": res, + } + ) + + if should_sel_attributes_right != res.sel_attributes_right: + errors.append( + { + "message": "sel_attributes_right not as it should", + "should": should_sel_attributes_right, + "is": res.sel_attributes_right, + } + ) + + if should_pro_attributes_right != res.pro_attributes_right: + errors.append( + { + "message": "pro_attributes_right not as it should", + "should": should_pro_attributes_right, + "is": res.pro_attributes_right, + } + ) + + if should_group_by_right != res.group_by_right: + errors.append( + { + "message": "group_by_right not as it should", + "should": should_group_by_right, + "is": res.group_by_right, + } + ) + + if should_joins_right != res.joins_right: + errors.append( + { + "message": "joins_right not as it should", + "should": should_joins_right, + "is": res.joins_right, + } + ) + + if should_order_by_right != res.order_by_right: + errors.append( + { + "message": "order_by_right not as it should", + "should": should_order_by_right, + "is": res.order_by_right, + } + ) + + if should_strings_right != res.strings_right: + errors.append( + { + "message": "strings_right not as it should", + "should": should_strings_right, + "is": res.strings_right, + } + ) + + if should_tables_right != res.tables_right: + errors.append( + { + "message": "tables_right not as it should", + "should": should_tables_right, + "is": res.tables_right, + } + ) + + if errors: + raise Exception( + "Tests failed:\n" + "\n".join(str(error) for error in errors) + ) + + +@typechecked +class QueryProcessorMinimalQueryTest(QueryProcessorQueryTestBast): + def test_correct(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + ) + + def test_wrong_select(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + "SELECT username, mail, registration_date FROM user WHERE registration_date > 100;", + should_pro_attributes_right=False, + ) + + def test_wrong_where(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + "SELECT username, mail, password FROM user WHERE registration_data > 100;", + should_sel_attributes_right=False, + ) + + def test_wrong_join(self): + self._query_test_case( + "SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.user_id WHERE registration_date > 100;", + "SELECT username, mail, password FROM user JOIN submissions ON user.id = submission.id WHERE registration_date > 100;", + should_joins_right=False, + ) + + def test_wrong_group_by(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY mail;", + "SELECT username, mail, password FROM user WHERE registration_date > 100 GROUP BY username;", + should_group_by_right=False, + ) + + def test_wrong_order_by(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY registration_date;", + "SELECT username, mail, password FROM user WHERE registration_date > 100 ORDER BY username;", + should_order_by_right=False, + ) + + def test_wrong_string(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE username = 'tester';", + "SELECT username, mail, password FROM user WHERE username = 'test';", + should_strings_right=False, + ) + + def test_wrong_tables(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + "SELECT username, mail, password FROM users WHERE registration_date > 100;", + should_tables_right=False, + ) + + +@typechecked +class QueryProcessorComplexSelTest(QueryProcessorQueryTestBast): + def test_count(self): + self._query_test_case( + "SELECT registration_date, COUNT(username) FROM user WHERE registration_date > 100 GROUP BY registration_date;", + "SELECT registration_data, COUNT(username) FROM user WHERE registration_date > 100 GROUP BY registration_date;", + should_pro_attributes_right=False, + ) + + def test_distinct_count(self): + self._query_test_case( + "SELECT registration_date, COUNT(DESTINCT username) FROM user WHERE registration_date > 100 GROUP BY registration_date;", + "SELECT registration_data, COUNT(DESTINCT username) FROM user WHERE registration_date > 100 GROUP BY registration_date;", + should_pro_attributes_right=False, + ) + + +@typechecked +class QueryProcessorComplexWhereTest(QueryProcessorQueryTestBast): + def test_wrong_name_with_bool_op(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100 AND mail LIKE '%gmail.com';", + "SELECT username, mail, password FROM user WHERE registration_data > 100 AND mail LIKE '%gmail.com';", + should_sel_attributes_right=False, + ) + + def test_inverse_operator(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + "SELECT username, mail, password FROM user WHERE registration_date < 100;", + should_sel_attributes_right=False, + ) + + def test_wrong_bool_op_operator(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100 AND mail LIKE '%gmail.com';", + "SELECT username, mail, password FROM user WHERE registration_date > 100 OR mail LIKE '%gmail.com';", + should_sel_attributes_right=False, + ) + + def test_type_confusion(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date > 100;", + "SELECT username, mail, password FROM user WHERE registration_date > '100';", + should_sel_attributes_right=False, + ) + + def test_subquery(self): + self._query_test_case( + "SELECT username, mail, password FROM user WHERE registration_date IN (SELECT DISTINCT registration_data FORM abuse);", + "SELECT username, mail, password FROM user WHERE registration_date NOT IN (SELECT DISTINCT registration_data FORM abuse);", + should_sel_attributes_right=False, + ) + + +@typechecked +class QueryProcessorDistanceTest(QueryProcessorTestBase): + def test_nearest_distance_is_used(self): + pass + + +if __name__ == "__main__": + unittest.main() diff --git a/modules/fbs-sql-checker/poetry.lock b/modules/fbs-sql-checker/poetry.lock old mode 100644 new mode 100755 index f2f4fbc14..e3e7b7ae6 --- a/modules/fbs-sql-checker/poetry.lock +++ b/modules/fbs-sql-checker/poetry.lock @@ -1,10 +1,9 @@ -# This file is automatically @generated by Poetry 1.4.0 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.8.4 and should not be changed by hand. [[package]] name = "astroid" version = "2.15.0" description = "An abstract syntax tree for Python with inference support." -category = "main" optional = false python-versions = ">=3.7.2" files = [ @@ -24,7 +23,6 @@ wrapt = [ name = "black" version = "22.12.0" description = "The uncompromising code formatter." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -59,7 +57,6 @@ uvloop = ["uvloop (>=0.15.2)"] name = "certifi" version = "2022.12.7" description = "Python package for providing Mozilla's CA Bundle." -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -71,7 +68,6 @@ files = [ name = "cfgv" version = "3.3.1" description = "Validate configuration and produce human readable error messages." -category = "main" optional = false python-versions = ">=3.6.1" files = [ @@ -83,7 +79,6 @@ files = [ name = "charset-normalizer" version = "3.1.0" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." -category = "main" optional = false python-versions = ">=3.7.0" files = [ @@ -168,7 +163,6 @@ files = [ name = "click" version = "8.0.2" description = "Composable command line interface toolkit" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -183,7 +177,6 @@ colorama = {version = "*", markers = "platform_system == \"Windows\""} name = "colorama" version = "0.4.6" description = "Cross-platform colored terminal text." -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" files = [ @@ -195,7 +188,6 @@ files = [ name = "dill" version = "0.3.6" description = "serialize all of python" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -210,7 +202,6 @@ graph = ["objgraph (>=1.7.2)"] name = "distlib" version = "0.3.6" description = "Distribution utilities" -category = "main" optional = false python-versions = "*" files = [ @@ -222,7 +213,6 @@ files = [ name = "dnspython" version = "2.3.0" description = "DNS toolkit" -category = "main" optional = false python-versions = ">=3.7,<4.0" files = [ @@ -243,7 +233,6 @@ wmi = ["wmi (>=1.5.1,<2.0.0)"] name = "filelock" version = "3.9.0" description = "A platform independent file lock." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -259,7 +248,6 @@ testing = ["covdefaults (>=2.2.2)", "coverage (>=7.0.1)", "pytest (>=7.2)", "pyt name = "identify" version = "2.5.19" description = "File identification library for Python" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -274,7 +262,6 @@ license = ["ukkonen"] name = "idna" version = "3.4" description = "Internationalized Domain Names in Applications (IDNA)" -category = "main" optional = false python-versions = ">=3.5" files = [ @@ -286,7 +273,6 @@ files = [ name = "isort" version = "5.12.0" description = "A Python utility / library to sort Python imports." -category = "main" optional = false python-versions = ">=3.8.0" files = [ @@ -304,7 +290,6 @@ requirements-deprecated-finder = ["pip-api", "pipreqs"] name = "lazy-object-proxy" version = "1.9.0" description = "A fast and thorough lazy object proxy." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -350,7 +335,6 @@ files = [ name = "mccabe" version = "0.7.0" description = "McCabe checker, plugin for flake8" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -362,7 +346,6 @@ files = [ name = "mo-dots" version = "9.230.22310" description = "More Dots! Dot-access to Python dicts like Javascript" -category = "main" optional = false python-versions = "*" files = [ @@ -380,7 +363,6 @@ tests = ["jx-elasticsearch", "mo-json", "mo-logs", "mo-math", "mo-testing", "mo- name = "mo-future" version = "6.230.22310" description = "More future! Make Python 2/3 compatibility a bit easier" -category = "main" optional = false python-versions = "*" files = [ @@ -391,7 +373,6 @@ files = [ name = "mo-imports" version = "7.230.22310" description = "More Imports! - Delayed importing" -category = "main" optional = false python-versions = "*" files = [ @@ -408,7 +389,6 @@ tests = ["mo-logs"] name = "mo-parsing" version = "8.233.22310" description = "Another PEG Parsing Tool" -category = "main" optional = false python-versions = "*" files = [ @@ -426,7 +406,6 @@ tests = ["mo-files", "mo-logs", "mo-testing", "mo-threads"] name = "mo-sql-parsing" version = "8.237.22316" description = "More SQL Parsing! Parse SQL into JSON parse tree" -category = "main" optional = false python-versions = "*" files = [ @@ -442,11 +421,47 @@ mo-parsing = "8.233.22310" [package.extras] tests = ["mo-files", "mo-testing", "mo-threads"] +[[package]] +name = "mongomock" +version = "4.3.0" +description = "Fake pymongo stub for testing simple MongoDB-dependent code" +optional = false +python-versions = "*" +files = [ + {file = "mongomock-4.3.0-py2.py3-none-any.whl", hash = "sha256:5ef86bd12fc8806c6e7af32f21266c61b6c4ba96096f85129852d1c4fec1327e"}, + {file = "mongomock-4.3.0.tar.gz", hash = "sha256:32667b79066fabc12d4f17f16a8fd7361b5f4435208b3ba32c226e52212a8c30"}, +] + +[package.dependencies] +packaging = "*" +pytz = "*" +sentinels = "*" + +[package.extras] +pyexecjs = ["pyexecjs"] +pymongo = ["pymongo"] + +[[package]] +name = "mpmath" +version = "1.3.0" +description = "Python library for arbitrary-precision floating-point arithmetic" +optional = false +python-versions = "*" +files = [ + {file = "mpmath-1.3.0-py3-none-any.whl", hash = "sha256:a0b2b9fe80bbcd81a6647ff13108738cfb482d481d826cc0e02f5b35e5c88d2c"}, + {file = "mpmath-1.3.0.tar.gz", hash = "sha256:7a28eb2a9774d00c7bc92411c19a89209d5da7c4c9a9e227be8330a23a25b91f"}, +] + +[package.extras] +develop = ["codecov", "pycodestyle", "pytest (>=4.6)", "pytest-cov", "wheel"] +docs = ["sphinx"] +gmpy = ["gmpy2 (>=2.1.0a4)"] +tests = ["pytest (>=4.6)"] + [[package]] name = "mypy-extensions" version = "1.0.0" description = "Type system extensions for programs checked with the mypy type checker." -category = "dev" optional = false python-versions = ">=3.5" files = [ @@ -458,7 +473,6 @@ files = [ name = "nodeenv" version = "1.7.0" description = "Node.js virtual environment builder" -category = "main" optional = false python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*" files = [ @@ -469,11 +483,21 @@ files = [ [package.dependencies] setuptools = "*" +[[package]] +name = "packaging" +version = "24.2" +description = "Core utilities for Python packages" +optional = false +python-versions = ">=3.8" +files = [ + {file = "packaging-24.2-py3-none-any.whl", hash = "sha256:09abb1bccd265c01f4a3aa3f7a7db064b36514d2cba19a2f694fe6150451a759"}, + {file = "packaging-24.2.tar.gz", hash = "sha256:c228a6dc5e932d346bc5739379109d49e8853dd8223571c7c5b55260edc0b97f"}, +] + [[package]] name = "pathspec" version = "0.11.0" description = "Utility library for gitignore style pattern matching of file paths." -category = "dev" optional = false python-versions = ">=3.7" files = [ @@ -485,7 +509,6 @@ files = [ name = "platformdirs" version = "3.1.0" description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -501,7 +524,6 @@ test = ["appdirs (==1.4.4)", "covdefaults (>=2.2.2)", "pytest (>=7.2.1)", "pytes name = "pre-commit" version = "2.21.0" description = "A framework for managing and maintaining multi-language pre-commit hooks." -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -516,11 +538,86 @@ nodeenv = ">=0.11.1" pyyaml = ">=5.1" virtualenv = ">=20.10.0" +[[package]] +name = "psycopg2-binary" +version = "2.9.10" +description = "psycopg2 - Python-PostgreSQL Database Adapter" +optional = false +python-versions = ">=3.8" +files = [ + {file = "psycopg2-binary-2.9.10.tar.gz", hash = "sha256:4b3df0e6990aa98acda57d983942eff13d824135fe2250e6522edaa782a06de2"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-macosx_12_0_x86_64.whl", hash = "sha256:0ea8e3d0ae83564f2fc554955d327fa081d065c8ca5cc6d2abb643e2c9c1200f"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-macosx_14_0_arm64.whl", hash = "sha256:3e9c76f0ac6f92ecfc79516a8034a544926430f7b080ec5a0537bca389ee0906"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2ad26b467a405c798aaa1458ba09d7e2b6e5f96b1ce0ac15d82fd9f95dc38a92"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:270934a475a0e4b6925b5f804e3809dd5f90f8613621d062848dd82f9cd62007"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:48b338f08d93e7be4ab2b5f1dbe69dc5e9ef07170fe1f86514422076d9c010d0"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7f4152f8f76d2023aac16285576a9ecd2b11a9895373a1f10fd9db54b3ff06b4"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-musllinux_1_2_aarch64.whl", hash = "sha256:32581b3020c72d7a421009ee1c6bf4a131ef5f0a968fab2e2de0c9d2bb4577f1"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-musllinux_1_2_i686.whl", hash = "sha256:2ce3e21dc3437b1d960521eca599d57408a695a0d3c26797ea0f72e834c7ffe5"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-musllinux_1_2_ppc64le.whl", hash = "sha256:e984839e75e0b60cfe75e351db53d6db750b00de45644c5d1f7ee5d1f34a1ce5"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-musllinux_1_2_x86_64.whl", hash = "sha256:3c4745a90b78e51d9ba06e2088a2fe0c693ae19cc8cb051ccda44e8df8a6eb53"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-win32.whl", hash = "sha256:e5720a5d25e3b99cd0dc5c8a440570469ff82659bb09431c1439b92caf184d3b"}, + {file = "psycopg2_binary-2.9.10-cp310-cp310-win_amd64.whl", hash = "sha256:3c18f74eb4386bf35e92ab2354a12c17e5eb4d9798e4c0ad3a00783eae7cd9f1"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-macosx_12_0_x86_64.whl", hash = "sha256:04392983d0bb89a8717772a193cfaac58871321e3ec69514e1c4e0d4957b5aff"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-macosx_14_0_arm64.whl", hash = "sha256:1a6784f0ce3fec4edc64e985865c17778514325074adf5ad8f80636cd029ef7c"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b5f86c56eeb91dc3135b3fd8a95dc7ae14c538a2f3ad77a19645cf55bab1799c"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2b3d2491d4d78b6b14f76881905c7a8a8abcf974aad4a8a0b065273a0ed7a2cb"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:2286791ececda3a723d1910441c793be44625d86d1a4e79942751197f4d30341"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:512d29bb12608891e349af6a0cccedce51677725a921c07dba6342beaf576f9a"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:5a507320c58903967ef7384355a4da7ff3f28132d679aeb23572753cbf2ec10b"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:6d4fa1079cab9018f4d0bd2db307beaa612b0d13ba73b5c6304b9fe2fb441ff7"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:851485a42dbb0bdc1edcdabdb8557c09c9655dfa2ca0460ff210522e073e319e"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:35958ec9e46432d9076286dda67942ed6d968b9c3a6a2fd62b48939d1d78bf68"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-win32.whl", hash = "sha256:ecced182e935529727401b24d76634a357c71c9275b356efafd8a2a91ec07392"}, + {file = "psycopg2_binary-2.9.10-cp311-cp311-win_amd64.whl", hash = "sha256:ee0e8c683a7ff25d23b55b11161c2663d4b099770f6085ff0a20d4505778d6b4"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-macosx_12_0_x86_64.whl", hash = "sha256:880845dfe1f85d9d5f7c412efea7a08946a46894537e4e5d091732eb1d34d9a0"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-macosx_14_0_arm64.whl", hash = "sha256:9440fa522a79356aaa482aa4ba500b65f28e5d0e63b801abf6aa152a29bd842a"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e3923c1d9870c49a2d44f795df0c889a22380d36ef92440ff618ec315757e539"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7b2c956c028ea5de47ff3a8d6b3cc3330ab45cf0b7c3da35a2d6ff8420896526"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f758ed67cab30b9a8d2833609513ce4d3bd027641673d4ebc9c067e4d208eec1"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8cd9b4f2cfab88ed4a9106192de509464b75a906462fb846b936eabe45c2063e"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:6dc08420625b5a20b53551c50deae6e231e6371194fa0651dbe0fb206452ae1f"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:d7cd730dfa7c36dbe8724426bf5612798734bff2d3c3857f36f2733f5bfc7c00"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:155e69561d54d02b3c3209545fb08938e27889ff5a10c19de8d23eb5a41be8a5"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:c3cc28a6fd5a4a26224007712e79b81dbaee2ffb90ff406256158ec4d7b52b47"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-win32.whl", hash = "sha256:ec8a77f521a17506a24a5f626cb2aee7850f9b69a0afe704586f63a464f3cd64"}, + {file = "psycopg2_binary-2.9.10-cp312-cp312-win_amd64.whl", hash = "sha256:18c5ee682b9c6dd3696dad6e54cc7ff3a1a9020df6a5c0f861ef8bfd338c3ca0"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-macosx_12_0_x86_64.whl", hash = "sha256:26540d4a9a4e2b096f1ff9cce51253d0504dca5a85872c7f7be23be5a53eb18d"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-macosx_14_0_arm64.whl", hash = "sha256:e217ce4d37667df0bc1c397fdcd8de5e81018ef305aed9415c3b093faaeb10fb"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:245159e7ab20a71d989da00f280ca57da7641fa2cdcf71749c193cea540a74f7"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3c4ded1a24b20021ebe677b7b08ad10bf09aac197d6943bfe6fec70ac4e4690d"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3abb691ff9e57d4a93355f60d4f4c1dd2d68326c968e7db17ea96df3c023ef73"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8608c078134f0b3cbd9f89b34bd60a943b23fd33cc5f065e8d5f840061bd0673"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:230eeae2d71594103cd5b93fd29d1ace6420d0b86f4778739cb1a5a32f607d1f"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:bb89f0a835bcfc1d42ccd5f41f04870c1b936d8507c6df12b7737febc40f0909"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:f0c2d907a1e102526dd2986df638343388b94c33860ff3bbe1384130828714b1"}, + {file = "psycopg2_binary-2.9.10-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:f8157bed2f51db683f31306aa497311b560f2265998122abe1dce6428bd86567"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-macosx_12_0_x86_64.whl", hash = "sha256:eb09aa7f9cecb45027683bb55aebaaf45a0df8bf6de68801a6afdc7947bb09d4"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b73d6d7f0ccdad7bc43e6d34273f70d587ef62f824d7261c4ae9b8b1b6af90e8"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ce5ab4bf46a211a8e924d307c1b1fcda82368586a19d0a24f8ae166f5c784864"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:056470c3dc57904bbf63d6f534988bafc4e970ffd50f6271fc4ee7daad9498a5"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:73aa0e31fa4bb82578f3a6c74a73c273367727de397a7a0f07bd83cbea696baa"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-musllinux_1_2_aarch64.whl", hash = "sha256:8de718c0e1c4b982a54b41779667242bc630b2197948405b7bd8ce16bcecac92"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-musllinux_1_2_i686.whl", hash = "sha256:5c370b1e4975df846b0277b4deba86419ca77dbc25047f535b0bb03d1a544d44"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-musllinux_1_2_ppc64le.whl", hash = "sha256:ffe8ed017e4ed70f68b7b371d84b7d4a790368db9203dfc2d222febd3a9c8863"}, + {file = "psycopg2_binary-2.9.10-cp38-cp38-musllinux_1_2_x86_64.whl", hash = "sha256:8aecc5e80c63f7459a1a2ab2c64df952051df196294d9f739933a9f6687e86b3"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-macosx_12_0_x86_64.whl", hash = "sha256:7a813c8bdbaaaab1f078014b9b0b13f5de757e2b5d9be6403639b298a04d218b"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d00924255d7fc916ef66e4bf22f354a940c67179ad3fd7067d7a0a9c84d2fbfc"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7559bce4b505762d737172556a4e6ea8a9998ecac1e39b5233465093e8cee697"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e8b58f0a96e7a1e341fc894f62c1177a7c83febebb5ff9123b579418fdc8a481"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6b269105e59ac96aba877c1707c600ae55711d9dcd3fc4b5012e4af68e30c648"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-musllinux_1_2_aarch64.whl", hash = "sha256:79625966e176dc97ddabc142351e0409e28acf4660b88d1cf6adb876d20c490d"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-musllinux_1_2_i686.whl", hash = "sha256:8aabf1c1a04584c168984ac678a668094d831f152859d06e055288fa515e4d30"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-musllinux_1_2_ppc64le.whl", hash = "sha256:19721ac03892001ee8fdd11507e6a2e01f4e37014def96379411ca99d78aeb2c"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-musllinux_1_2_x86_64.whl", hash = "sha256:7f5d859928e635fa3ce3477704acee0f667b3a3d3e4bb109f2b18d4005f38287"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-win32.whl", hash = "sha256:3216ccf953b3f267691c90c6fe742e45d890d8272326b4a8b20850a03d05b7b8"}, + {file = "psycopg2_binary-2.9.10-cp39-cp39-win_amd64.whl", hash = "sha256:30e34c4e97964805f715206c7b789d54a78b70f3ff19fbe590104b71c45600e5"}, +] + [[package]] name = "pylint" version = "2.17.0" description = "python code static checker" -category = "main" optional = false python-versions = ">=3.7.2" files = [ @@ -549,7 +646,6 @@ testutils = ["gitpython (>3)"] name = "pymongo" version = "4.3.3" description = "Python driver for MongoDB " -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -640,11 +736,35 @@ ocsp = ["certifi", "pyopenssl (>=17.2.0)", "requests (<3.0.0)", "service-identit snappy = ["python-snappy"] zstd = ["zstandard"] +[[package]] +name = "python-dotenv" +version = "0.21.1" +description = "Read key-value pairs from a .env file and set them as environment variables" +optional = false +python-versions = ">=3.7" +files = [ + {file = "python-dotenv-0.21.1.tar.gz", hash = "sha256:1c93de8f636cde3ce377292818d0e440b6e45a82f215c3744979151fa8151c49"}, + {file = "python_dotenv-0.21.1-py3-none-any.whl", hash = "sha256:41e12e0318bebc859fcc4d97d4db8d20ad21721a6aa5047dd59f090391cb549a"}, +] + +[package.extras] +cli = ["click (>=5.0)"] + +[[package]] +name = "pytz" +version = "2024.2" +description = "World timezone definitions, modern and historical" +optional = false +python-versions = "*" +files = [ + {file = "pytz-2024.2-py2.py3-none-any.whl", hash = "sha256:31c7c1817eb7fae7ca4b8c7ee50c72f93aa2dd863de768e1ef4245d426aa0725"}, + {file = "pytz-2024.2.tar.gz", hash = "sha256:2aa355083c50a0f93fa581709deac0c9ad65cca8a9e9beac660adcbd493c798a"}, +] + [[package]] name = "pyyaml" version = "6.0" description = "YAML parser and emitter for Python" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -655,6 +775,13 @@ files = [ {file = "PyYAML-6.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f84fbc98b019fef2ee9a1cb3ce93e3187a6df0b2538a651bfb890254ba9f90b5"}, {file = "PyYAML-6.0-cp310-cp310-win32.whl", hash = "sha256:2cd5df3de48857ed0544b34e2d40e9fac445930039f3cfe4bcc592a1f836d513"}, {file = "PyYAML-6.0-cp310-cp310-win_amd64.whl", hash = "sha256:daf496c58a8c52083df09b80c860005194014c3698698d1a57cbcfa182142a3a"}, + {file = "PyYAML-6.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:d4b0ba9512519522b118090257be113b9468d804b19d63c71dbcf4a48fa32358"}, + {file = "PyYAML-6.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:81957921f441d50af23654aa6c5e5eaf9b06aba7f0a19c18a538dc7ef291c5a1"}, + {file = "PyYAML-6.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:afa17f5bc4d1b10afd4466fd3a44dc0e245382deca5b3c353d8b757f9e3ecb8d"}, + {file = "PyYAML-6.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:dbad0e9d368bb989f4515da330b88a057617d16b6a8245084f1b05400f24609f"}, + {file = "PyYAML-6.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:432557aa2c09802be39460360ddffd48156e30721f5e8d917f01d31694216782"}, + {file = "PyYAML-6.0-cp311-cp311-win32.whl", hash = "sha256:bfaef573a63ba8923503d27530362590ff4f576c626d86a9fed95822a8255fd7"}, + {file = "PyYAML-6.0-cp311-cp311-win_amd64.whl", hash = "sha256:01b45c0191e6d66c470b6cf1b9531a771a83c1c4208272ead47a3ae4f2f603bf"}, {file = "PyYAML-6.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:897b80890765f037df3403d22bab41627ca8811ae55e9a722fd0392850ec4d86"}, {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50602afada6d6cbfad699b0c7bb50d5ccffa7e46a3d738092afddc1f9758427f"}, {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:48c346915c114f5fdb3ead70312bd042a953a8ce5c7106d5bfb1a5254e47da92"}, @@ -687,7 +814,6 @@ files = [ name = "random2" version = "1.0.1" description = "Python 3 compatible Pytohn 2 `random` Module." -category = "main" optional = false python-versions = "*" files = [ @@ -698,7 +824,6 @@ files = [ name = "requests" version = "2.28.2" description = "Python HTTP for Humans." -category = "main" optional = false python-versions = ">=3.7, <4" files = [ @@ -716,11 +841,20 @@ urllib3 = ">=1.21.1,<1.27" socks = ["PySocks (>=1.5.6,!=1.5.7)"] use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] +[[package]] +name = "sentinels" +version = "1.0.0" +description = "Various objects to denote special meanings in python" +optional = false +python-versions = "*" +files = [ + {file = "sentinels-1.0.0.tar.gz", hash = "sha256:7be0704d7fe1925e397e92d18669ace2f619c92b5d4eb21a89f31e026f9ff4b1"}, +] + [[package]] name = "setuptools" version = "67.6.0" description = "Easily download, build, install, upgrade, and uninstall Python packages" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -733,11 +867,40 @@ docs = ["furo", "jaraco.packaging (>=9)", "jaraco.tidelift (>=1.4)", "pygments-g testing = ["build[virtualenv]", "filelock (>=3.4.0)", "flake8 (<5)", "flake8-2020", "ini2toml[lite] (>=0.9)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pip (>=19.1)", "pip-run (>=8.8)", "pytest (>=6)", "pytest-black (>=0.3.7)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=1.3)", "pytest-flake8", "pytest-mypy (>=0.9.1)", "pytest-perf", "pytest-timeout", "pytest-xdist", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] testing-integration = ["build[virtualenv]", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] +[[package]] +name = "sqlparse" +version = "0.4.4" +description = "A non-validating SQL parser." +optional = false +python-versions = ">=3.5" +files = [ + {file = "sqlparse-0.4.4-py3-none-any.whl", hash = "sha256:5430a4fe2ac7d0f93e66f1efc6e1338a41884b7ddf2a350cedd20ccc4d9d28f3"}, + {file = "sqlparse-0.4.4.tar.gz", hash = "sha256:d446183e84b8349fa3061f0fe7f06ca94ba65b426946ffebe6e3e8295332420c"}, +] + +[package.extras] +dev = ["build", "flake8"] +doc = ["sphinx"] +test = ["pytest", "pytest-cov"] + +[[package]] +name = "sympy" +version = "1.12" +description = "Computer algebra system (CAS) in Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "sympy-1.12-py3-none-any.whl", hash = "sha256:c3588cd4295d0c0f603d0f2ae780587e64e2efeedb3521e46b9bb1d08d184fa5"}, + {file = "sympy-1.12.tar.gz", hash = "sha256:ebf595c8dac3e0fdc4152c51878b498396ec7f30e7a914d6071e674d49420fb8"}, +] + +[package.dependencies] +mpmath = ">=0.19" + [[package]] name = "tomli" version = "2.0.1" description = "A lil' TOML parser" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -749,7 +912,6 @@ files = [ name = "tomlkit" version = "0.11.6" description = "Style preserving TOML library" -category = "main" optional = false python-versions = ">=3.6" files = [ @@ -761,7 +923,6 @@ files = [ name = "tqdm" version = "4.65.0" description = "Fast, Extensible Progress Meter" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -778,23 +939,39 @@ notebook = ["ipywidgets (>=6)"] slack = ["slack-sdk"] telegram = ["requests"] +[[package]] +name = "typeguard" +version = "4.4.1" +description = "Run-time type checker for Python" +optional = false +python-versions = ">=3.9" +files = [ + {file = "typeguard-4.4.1-py3-none-any.whl", hash = "sha256:9324ec07a27ec67fc54a9c063020ca4c0ae6abad5e9f0f9804ca59aee68c6e21"}, + {file = "typeguard-4.4.1.tar.gz", hash = "sha256:0d22a89d00b453b47c49875f42b6601b961757541a2e1e0ef517b6e24213c21b"}, +] + +[package.dependencies] +typing-extensions = ">=4.10.0" + +[package.extras] +doc = ["Sphinx (>=7)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)", "sphinx-rtd-theme (>=1.3.0)"] +test = ["coverage[toml] (>=7)", "mypy (>=1.2.0)", "pytest (>=7)"] + [[package]] name = "typing-extensions" -version = "4.5.0" -description = "Backported and Experimental Type Hints for Python 3.7+" -category = "main" +version = "4.12.2" +description = "Backported and Experimental Type Hints for Python 3.8+" optional = false -python-versions = ">=3.7" +python-versions = ">=3.8" files = [ - {file = "typing_extensions-4.5.0-py3-none-any.whl", hash = "sha256:fb33085c39dd998ac16d1431ebc293a8b3eedd00fd4a32de0ff79002c19511b4"}, - {file = "typing_extensions-4.5.0.tar.gz", hash = "sha256:5cb5f4a79139d699607b3ef622a1dedafa84e115ab0024e0d9c044a9479ca7cb"}, + {file = "typing_extensions-4.12.2-py3-none-any.whl", hash = "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d"}, + {file = "typing_extensions-4.12.2.tar.gz", hash = "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8"}, ] [[package]] name = "urllib3" version = "1.26.14" description = "HTTP library with thread-safe connection pooling, file post, and more." -category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" files = [ @@ -811,7 +988,6 @@ socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"] name = "virtualenv" version = "20.20.0" description = "Virtual Python Environment builder" -category = "main" optional = false python-versions = ">=3.7" files = [ @@ -832,7 +1008,6 @@ test = ["covdefaults (>=2.2.2)", "coverage (>=7.1)", "coverage-enable-subprocess name = "wrapt" version = "1.15.0" description = "Module for decorators, wrappers and monkey patching." -category = "main" optional = false python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" files = [ @@ -916,4 +1091,4 @@ files = [ [metadata] lock-version = "2.0" python-versions = "^3.10" -content-hash = "2c849c3d9fb82034c7396f4a939ca4a10e27d80b1398ac21848e06d09f6b9bbd" +content-hash = "94f1d899a0c77a8fe699f75ffe9d8773571730102113be70712500809d039037" diff --git a/modules/fbs-sql-checker/pyproject.toml b/modules/fbs-sql-checker/pyproject.toml old mode 100644 new mode 100755 index 77800992b..5dfbd56f8 --- a/modules/fbs-sql-checker/pyproject.toml +++ b/modules/fbs-sql-checker/pyproject.toml @@ -16,11 +16,18 @@ random2 = "^1.0.1" pre-commit = "^2.20.0" pylint = "^2.15.3" tqdm = "^4.64.1" +sqlparse = "^0.4.2" +psycopg2-binary = "^2.9.10" +python-dotenv = "^0.21.0" +sympy = "^1.12" +typeguard = "^4.4.1" [tool.poetry.dev-dependencies] pylint = "^2.15.2" black = "^22.8.0" +mongomock = "^4.3.0" + [build-system] requires = ["poetry-core"] build-backend = "poetry.core.masonry.api"