Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Provide framework for generic lazily evaluated operation results #1350

Merged
merged 152 commits into from
Aug 22, 2024
Merged
Show file tree
Hide file tree
Changes from 27 commits
Commits
Show all changes
152 commits
Select commit Hold shift + click to select a range
80667bd
Rename ResultTable -> Result
RobinTF Apr 19, 2024
31b2c11
Wrap idTable in variant
RobinTF Apr 19, 2024
4d0204c
Add ability to create `Result` from generator
RobinTF Apr 19, 2024
515ed0c
Start fixing caching issues
RobinTF Apr 22, 2024
ca1cbed
Avoid another class of exceptions
RobinTF Apr 22, 2024
9e7f3cb
Optimize imports
RobinTF Apr 23, 2024
4c75d42
Introduce ReusableGenerator class
RobinTF Apr 23, 2024
892e4a5
Try to make caching work
RobinTF Apr 23, 2024
586365c
Fiddle around with const a bit
RobinTF Apr 23, 2024
80e2dbd
Add more TODOs
RobinTF Apr 23, 2024
18ca5b1
Fix TextLimit code after rebase
RobinTF Apr 28, 2024
86a9f4b
Fix compilation issues for ReusableGenerator
RobinTF Apr 28, 2024
7f0a5e7
Remove offset calculations from exporter
RobinTF Apr 28, 2024
aee20dd
Fix typo
RobinTF Apr 28, 2024
7576b2e
Add comments
RobinTF Apr 28, 2024
7765a25
Make supportsLimit private to avoid misuse
RobinTF Apr 28, 2024
f815be8
Properly use minimum limit if present
RobinTF May 1, 2024
90cca50
Start adding code to manipulate code after cache extraction
RobinTF May 1, 2024
694c21f
Implement fallback mechanism for failed cache share
RobinTF May 5, 2024
ea8b81f
Fix accidental edit of Usage.md
RobinTF May 5, 2024
50e4529
Consume result as master
RobinTF May 5, 2024
16eedd8
Add proper condition variables
RobinTF May 10, 2024
bf8f085
Implement code that allows for proper recomputation of cache size
RobinTF May 10, 2024
771eb5b
Refactor a bit
RobinTF May 12, 2024
8aa9060
Aggregate tables at the end of lazy results
RobinTF May 12, 2024
b499c6e
Overload constructor of Result class
RobinTF May 17, 2024
6b3f05c
Try to properly calculate duration
RobinTF May 18, 2024
ff5a6ea
Apply formatting
RobinTF May 18, 2024
b974c7d
Fix compilation on gcc 11 and gcc 12
RobinTF May 18, 2024
8b99020
Add correct visibility modifiers
RobinTF May 18, 2024
5f8ab65
Try fixing the compilation issue for real this time
RobinTF May 18, 2024
3a95ef5
Try to fix compilation issue on macOS
RobinTF May 18, 2024
e2dc667
Implement PoC lazy operation for index scan and filter operations
RobinTF May 20, 2024
070494f
Fix double limit offset row
RobinTF May 20, 2024
f631c13
Fix wrong assertion
RobinTF May 20, 2024
c249628
Properly request lazy results when limit clause is present
RobinTF May 20, 2024
ea7fd79
Fix bugs and segfaults
RobinTF May 20, 2024
1e9d0d6
Formatting
RobinTF May 21, 2024
dfb7ba6
Apply small refactoring change
RobinTF May 21, 2024
7d01e59
Correct wrong order of ternary statement
RobinTF May 22, 2024
5b2335a
Add TODO
RobinTF May 22, 2024
93a5892
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF May 23, 2024
9c445ea
Change how maxSend works
RobinTF May 23, 2024
b9ca4aa
Correct call order
RobinTF May 23, 2024
43dddd0
Correct call order
RobinTF May 23, 2024
4ac7892
Rethink approach to apply limits and offset
RobinTF May 27, 2024
ef17e67
Add back headers
RobinTF May 27, 2024
aabb81b
Add back result limiter for subqueries
RobinTF Jun 1, 2024
66a38b4
Try to fix subtle bug with runtime information detail
RobinTF Jun 1, 2024
999baee
Merge branch 'max-send-changes' into refactor-result-table
RobinTF Jun 5, 2024
c291ff7
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jun 5, 2024
9f17e07
Add back comment
RobinTF Jun 5, 2024
389f3f1
Rename `resultTable` -> `result`
RobinTF Jun 5, 2024
000af28
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jun 6, 2024
ba142a0
Add correctness check to prevent double move due to race condition
RobinTF Jun 9, 2024
44562c7
Start implementing tests for new cache feature and fixing bugs along …
RobinTF Jun 13, 2024
0f3a59a
Some Test cleanup
RobinTF Jun 13, 2024
d226849
Mark variable as maybe_unused
RobinTF Jun 13, 2024
552a268
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jun 13, 2024
cde135a
Restructure recomputeSize a bit to avoid unwanted behaviour
RobinTF Jun 13, 2024
cf6b4c9
Add remaining cache tests
RobinTF Jun 14, 2024
b2138bf
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jun 14, 2024
0c589e3
Add tests for `IteratorWrapper`
RobinTF Jun 14, 2024
c465685
Fix line endings
RobinTF Jun 15, 2024
d17fc7d
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jun 28, 2024
93c2360
Add tests for CacheableGenerator
RobinTF Jun 29, 2024
15b435e
Add Filter tests
RobinTF Jun 29, 2024
633bf06
Clear Cache before running tests
RobinTF Jun 29, 2024
6d5a95e
Add test to fix coverage
RobinTF Jun 30, 2024
55b4fec
Address some sonarcloud issues
RobinTF Jun 30, 2024
e5ceacc
Add tests for ExportQueryExecutionTrees
RobinTF Jun 30, 2024
d172dc8
Divide Result class into 3 dedicated classes
RobinTF Jul 5, 2024
5b004b8
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jul 5, 2024
b95edfd
Remove parameter for supportsLimit
RobinTF Jul 5, 2024
acc99c3
Fix formatting
RobinTF Jul 5, 2024
7900619
Format again
RobinTF Jul 5, 2024
0d0133a
Also perform definedness check for lazy results
RobinTF Jul 5, 2024
27ab692
Drop definedness caching mechanism
RobinTF Jul 5, 2024
2da169f
Add comment
RobinTF Jul 5, 2024
0cbb47d
Split lambdas into dedicated functions
RobinTF Jul 6, 2024
5adda07
Make move/copy constructors explicit
RobinTF Jul 6, 2024
e0cdf18
Fix undefined behaviour
RobinTF Jul 6, 2024
5ad5b8a
Workaround segfault
RobinTF Jul 6, 2024
db187f0
Try different attempt to fix double locking
RobinTF Jul 7, 2024
4ba81bd
Avoid pseudo false-positive thread sanitizer warning
RobinTF Jul 7, 2024
373f009
Restructure code to avoid class of race conditions
RobinTF Jul 11, 2024
27e451e
Clarify currently buggy behaviour
RobinTF Jul 11, 2024
96982aa
Fix macOS build
RobinTF Jul 11, 2024
58a193d
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jul 23, 2024
93a71cc
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jul 24, 2024
9c07e4f
Fix wrong merge conflict resolution
RobinTF Jul 24, 2024
1da9b17
Use `#pragma once`
RobinTF Jul 24, 2024
c6aa641
Simplify caching structure to fix bugs
RobinTF Jul 25, 2024
838a97e
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Jul 31, 2024
d74b89d
Fix size calculation
RobinTF Jul 31, 2024
bfc9d8f
Remove complicated atomic mutex check mechanism
RobinTF Aug 1, 2024
592c670
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 1, 2024
564d54b
Avoid copy
RobinTF Aug 1, 2024
d566d5b
Use timer class to simplify calls
RobinTF Aug 1, 2024
a164ef4
Start adding some documentation
RobinTF Aug 1, 2024
f9f22f4
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 1, 2024
23713cc
Unlock before notifying
RobinTF Aug 1, 2024
c56dd82
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 2, 2024
603a48b
Adjust Cache to simplify code a bit
RobinTF Aug 2, 2024
389a9bf
Merge branch 'helper-master' into refactor-result-table
RobinTF Aug 7, 2024
934015c
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 7, 2024
c7d58a4
Rework code (again) so that generator does not get cached
RobinTF Aug 7, 2024
4b93109
Make diff smaller and add some more tests
RobinTF Aug 7, 2024
7294b10
Fix problems with `RuntimeInformation`
RobinTF Aug 13, 2024
3b9fee8
Simplify timing calculation
RobinTF Aug 13, 2024
8acb05a
Fix sonarqube issues
RobinTF Aug 13, 2024
3ee5510
Use const ref
RobinTF Aug 13, 2024
72d1033
Apply microoptimization with mutex
RobinTF Aug 13, 2024
767a6cb
Another correction for reported result sizes
RobinTF Aug 14, 2024
8c834da
Rename member function
RobinTF Aug 14, 2024
4555dab
compute value directly after waiting without success
RobinTF Aug 14, 2024
d8f7fc1
Check size before aggregating id tables
RobinTF Aug 14, 2024
bbe10ef
Merge Result and ProtoResult back together
RobinTF Aug 14, 2024
345d79c
Re-arrange functions to make diff smaller
RobinTF Aug 14, 2024
a101433
Fix bugs with limit and optimize iteration
RobinTF Aug 15, 2024
b390f4f
Use higher precision timing for `RuntimeInformation`
RobinTF Aug 15, 2024
ce62baa
Incorporate PR comments for `CacheableGenerator` and `ConcurrentCache`
RobinTF Aug 15, 2024
9755da8
Address more PR comments
RobinTF Aug 15, 2024
c0ef64b
Fix unused warnings
RobinTF Aug 15, 2024
44d103f
Fix flickering timing information
RobinTF Aug 15, 2024
4a3f09b
Actually fix the timing issue
RobinTF Aug 15, 2024
0164d9c
Small improvements to query updates
RobinTF Aug 15, 2024
f23849c
Add a lot of documentation
RobinTF Aug 15, 2024
3c69ebf
Rename some functions
RobinTF Aug 15, 2024
383696b
Fix spelling error
RobinTF Aug 15, 2024
d89e896
Fix build on macOS
RobinTF Aug 15, 2024
b5dd60c
Address even more PR comments
RobinTF Aug 16, 2024
4f71329
Add Unit tests for edge case
RobinTF Aug 16, 2024
21beae5
Fix sortedBy and is defined check for multiple `IdTable`s
RobinTF Aug 16, 2024
eadbc01
Consistent separator comment length
RobinTF Aug 16, 2024
cb86d14
Reorder function to fix build
RobinTF Aug 16, 2024
f0434b3
Add unit tests for newly added `ConcurrentCache` features
RobinTF Aug 17, 2024
d181a9f
Add unit tests for lazy index scans
RobinTF Aug 17, 2024
68dcb6f
Fix wrong filter calculations
RobinTF Aug 18, 2024
6c785b2
Add tests for new `Operation` functionality
RobinTF Aug 18, 2024
ef0f872
Rename variable and fix functionality
RobinTF Aug 19, 2024
77b6f58
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 19, 2024
dff4910
Add unit tests for `Result` class
RobinTF Aug 19, 2024
a431021
Add tests for more coverage and skip tests relying on expensive checks
RobinTF Aug 19, 2024
476a691
Add test for edge case and skip test on expensive checks disabled
RobinTF Aug 19, 2024
b8b03f9
Fix sonarcloud issues
RobinTF Aug 20, 2024
1d83e86
Adjust limit values for better coverage
RobinTF Aug 20, 2024
538814f
Address PR comments
RobinTF Aug 20, 2024
d7fcc98
Reduce code duplication in tests
RobinTF Aug 21, 2024
a471bcf
More PR comments
RobinTF Aug 21, 2024
226b577
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 21, 2024
0437773
Merge remote-tracking branch 'ad-freiburg/master' into refactor-resul…
RobinTF Aug 21, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
219 changes: 119 additions & 100 deletions src/engine/ExportQueryExecutionTrees.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

cppcoro::generator<const IdTable&> ExportQueryExecutionTrees::getIdTables(
const Result& result) {
if (result.isDataEvaluated()) {
if (result.isFullyMaterialized()) {
co_yield result.idTable();
} else {
for (const IdTable& idTable : result.idTables()) {
Expand All @@ -28,20 +28,26 @@ cppcoro::generator<const IdTable&> ExportQueryExecutionTrees::getIdTables(
// Return a range that contains the indices of the rows that have to be exported
// from the `idTable` given the `LimitOffsetClause`. It takes into account the
// LIMIT, the OFFSET, and the actual size of the `idTable`
cppcoro::generator<ExportQueryExecutionTrees::IndexWithTable>
cppcoro::generator<ExportQueryExecutionTrees::TableWithRange>
ExportQueryExecutionTrees::getRowIndices(LimitOffsetClause limitOffset,
const Result& result) {
if (limitOffset._limit.value_or(1) == 0) {
co_return;
}
for (const IdTable& idTable : getIdTables(result)) {
uint64_t currentOffset = limitOffset.actualOffset(idTable.numRows());
uint64_t upperBound = limitOffset.upperBound(idTable.numRows());
for (size_t index = currentOffset; index < upperBound; index++) {
co_yield {index, idTable};
if (currentOffset != upperBound) {
co_yield {idTable, std::views::iota(currentOffset, upperBound)};
}
limitOffset._offset -= currentOffset;
if (limitOffset._limit.has_value()) {
limitOffset._limit =
limitOffset._limit.value() - (upperBound - currentOffset);
}
if (limitOffset._limit.value_or(1) == 0) {
break;
}
}
}

Expand All @@ -52,22 +58,24 @@ ExportQueryExecutionTrees::constructQueryResultToTriples(
const ad_utility::sparql_types::Triples& constructTriples,
LimitOffsetClause limitAndOffset, std::shared_ptr<const Result> result,
CancellationHandle cancellationHandle) {
for (auto [i, idTable] : getRowIndices(limitAndOffset, *result)) {
ConstructQueryExportContext context{i, idTable, result->localVocab(),
qet.getVariableColumns(),
qet.getQec()->getIndex()};
using enum PositionInTriple;
for (const auto& triple : constructTriples) {
auto subject = triple[0].evaluate(context, SUBJECT);
auto predicate = triple[1].evaluate(context, PREDICATE);
auto object = triple[2].evaluate(context, OBJECT);
if (!subject.has_value() || !predicate.has_value() ||
!object.has_value()) {
continue;
for (const auto& [idTable, range] : getRowIndices(limitAndOffset, *result)) {
for (uint64_t i : range) {
ConstructQueryExportContext context{i, idTable, result->localVocab(),
qet.getVariableColumns(),
qet.getQec()->getIndex()};
using enum PositionInTriple;
for (const auto& triple : constructTriples) {
auto subject = triple[0].evaluate(context, SUBJECT);
auto predicate = triple[1].evaluate(context, PREDICATE);
auto object = triple[2].evaluate(context, OBJECT);
if (!subject.has_value() || !predicate.has_value() ||
!object.has_value()) {
continue;
}
co_yield {std::move(subject.value()), std::move(predicate.value()),
std::move(object.value())};
cancellationHandle->throwIfCancelled();
}
co_yield {std::move(subject.value()), std::move(predicate.value()),
std::move(object.value())};
cancellationHandle->throwIfCancelled();
}
}
}
Expand Down Expand Up @@ -135,31 +143,33 @@ nlohmann::json ExportQueryExecutionTrees::idTableToQLeverJSONArray(
AD_CORRECTNESS_CHECK(result != nullptr);
nlohmann::json json = nlohmann::json::array();

for (auto [rowIndex, idTable] : getRowIndices(limitAndOffset, *result)) {
// We need the explicit `array` constructor for the special case of zero
// variables.
json.push_back(nlohmann::json::array());
auto& row = json.back();
for (const auto& opt : columns) {
if (!opt) {
row.emplace_back(nullptr);
continue;
}
const auto& currentId = idTable(rowIndex, opt->columnIndex_);
const auto& optionalStringAndXsdType = idToStringAndType(
qet.getQec()->getIndex(), currentId, result->localVocab());
if (!optionalStringAndXsdType.has_value()) {
row.emplace_back(nullptr);
continue;
}
const auto& [stringValue, xsdType] = optionalStringAndXsdType.value();
if (xsdType) {
row.emplace_back('"' + stringValue + "\"^^<" + xsdType + '>');
} else {
row.emplace_back(stringValue);
for (const auto& [idTable, range] : getRowIndices(limitAndOffset, *result)) {
for (uint64_t rowIndex : range) {
// We need the explicit `array` constructor for the special case of zero
// variables.
json.push_back(nlohmann::json::array());
auto& row = json.back();
for (const auto& opt : columns) {
if (!opt) {
row.emplace_back(nullptr);
continue;
}
const auto& currentId = idTable(rowIndex, opt->columnIndex_);
const auto& optionalStringAndXsdType = idToStringAndType(
qet.getQec()->getIndex(), currentId, result->localVocab());
if (!optionalStringAndXsdType.has_value()) {
row.emplace_back(nullptr);
continue;
}
const auto& [stringValue, xsdType] = optionalStringAndXsdType.value();
if (xsdType) {
row.emplace_back('"' + stringValue + "\"^^<" + xsdType + '>');
} else {
row.emplace_back(stringValue);
}
}
cancellationHandle->throwIfCancelled();
}
cancellationHandle->throwIfCancelled();
}
return json;
}
Expand Down Expand Up @@ -375,32 +385,34 @@ nlohmann::json ExportQueryExecutionTrees::selectQueryResultToSparqlJSON(
return b;
};

for (auto [rowIndex, idTable] : getRowIndices(limitAndOffset, *result)) {
// TODO: ordered_json` entries are ordered alphabetically, but insertion
// order would be preferable.
nlohmann::ordered_json binding;
for (const auto& column : columns) {
const auto& currentId = idTable(rowIndex, column->columnIndex_);
const auto& optionalValue = idToStringAndType(
qet.getQec()->getIndex(), currentId, result->localVocab());
if (!optionalValue.has_value()) {
continue;
}
const auto& [stringValue, xsdType] = optionalValue.value();
nlohmann::ordered_json b;
if (!xsdType) {
// No xsdType, this means that `stringValue` is a plain string literal
// or entity.
b = stringToBinding(stringValue);
} else {
b["value"] = stringValue;
b["type"] = "literal";
b["datatype"] = xsdType;
for (const auto& [idTable, range] : getRowIndices(limitAndOffset, *result)) {
for (uint64_t rowIndex : range) {
// TODO: ordered_json` entries are ordered alphabetically, but insertion
// order would be preferable.
nlohmann::ordered_json binding;
for (const auto& column : columns) {
const auto& currentId = idTable(rowIndex, column->columnIndex_);
const auto& optionalValue = idToStringAndType(
qet.getQec()->getIndex(), currentId, result->localVocab());
if (!optionalValue.has_value()) {
continue;
}
const auto& [stringValue, xsdType] = optionalValue.value();
nlohmann::ordered_json b;
if (!xsdType) {
// No xsdType, this means that `stringValue` is a plain string literal
// or entity.
b = stringToBinding(stringValue);
} else {
b["value"] = stringValue;
b["type"] = "literal";
b["datatype"] = xsdType;
}
binding[column->variable_] = std::move(b);
}
binding[column->variable_] = std::move(b);
bindings.emplace_back(std::move(binding));
cancellationHandle->throwIfCancelled();
}
bindings.emplace_back(std::move(binding));
cancellationHandle->throwIfCancelled();
}
resultJson["results"]["bindings"] = std::move(bindings);
return resultJson;
Expand Down Expand Up @@ -450,15 +462,18 @@ ExportQueryExecutionTrees::selectQueryResultToStream(

// special case : binary export of IdTable
if constexpr (format == MediaType::octetStream) {
for (auto [i, idTable] : getRowIndices(limitAndOffset, *result)) {
for (const auto& columnIndex : selectedColumnIndices) {
if (columnIndex.has_value()) {
co_yield std::string_view{reinterpret_cast<const char*>(&idTable(
i, columnIndex.value().columnIndex_)),
sizeof(Id)};
for (const auto& [idTable, range] :
getRowIndices(limitAndOffset, *result)) {
for (uint64_t i : range) {
for (const auto& columnIndex : selectedColumnIndices) {
if (columnIndex.has_value()) {
co_yield std::string_view{reinterpret_cast<const char*>(&idTable(
i, columnIndex.value().columnIndex_)),
sizeof(Id)};
}
}
cancellationHandle->throwIfCancelled();
}
cancellationHandle->throwIfCancelled();
}
co_return;
}
Expand All @@ -478,22 +493,24 @@ ExportQueryExecutionTrees::selectQueryResultToStream(
constexpr auto& escapeFunction = format == MediaType::tsv
? RdfEscaping::escapeForTsv
: RdfEscaping::escapeForCsv;
for (auto [i, idTable] : getRowIndices(limitAndOffset, *result)) {
for (size_t j = 0; j < selectedColumnIndices.size(); ++j) {
if (selectedColumnIndices[j].has_value()) {
const auto& val = selectedColumnIndices[j].value();
Id id = idTable(i, val.columnIndex_);
auto optionalStringAndType =
idToStringAndType<format == MediaType::csv>(
qet.getQec()->getIndex(), id, result->localVocab(),
escapeFunction);
if (optionalStringAndType.has_value()) [[likely]] {
co_yield optionalStringAndType.value().first;
for (const auto& [idTable, range] : getRowIndices(limitAndOffset, *result)) {
for (uint64_t i : range) {
for (size_t j = 0; j < selectedColumnIndices.size(); ++j) {
if (selectedColumnIndices[j].has_value()) {
const auto& val = selectedColumnIndices[j].value();
Id id = idTable(i, val.columnIndex_);
auto optionalStringAndType =
idToStringAndType<format == MediaType::csv>(
qet.getQec()->getIndex(), id, result->localVocab(),
escapeFunction);
if (optionalStringAndType.has_value()) [[likely]] {
co_yield optionalStringAndType.value().first;
}
}
co_yield j + 1 < selectedColumnIndices.size() ? separator : '\n';
}
co_yield j + 1 < selectedColumnIndices.size() ? separator : '\n';
cancellationHandle->throwIfCancelled();
}
cancellationHandle->throwIfCancelled();
}
LOG(DEBUG) << "Done creating readable result.\n";
}
Expand Down Expand Up @@ -601,18 +618,20 @@ ad_utility::streams::stream_generator ExportQueryExecutionTrees::
auto selectedColumnIndices =
qet.selectedVariablesToColumnIndices(selectClause, false);
// TODO<joka921> we could prefilter for the nonexisting variables.
for (auto [i, idTable] : getRowIndices(limitAndOffset, *result)) {
co_yield "\n <result>";
for (size_t j = 0; j < selectedColumnIndices.size(); ++j) {
if (selectedColumnIndices[j].has_value()) {
const auto& val = selectedColumnIndices[j].value();
Id id = idTable(i, val.columnIndex_);
co_yield idToXMLBinding(val.variable_, id, qet.getQec()->getIndex(),
result->localVocab());
for (const auto& [idTable, range] : getRowIndices(limitAndOffset, *result)) {
for (uint64_t i : range) {
co_yield "\n <result>";
for (size_t j = 0; j < selectedColumnIndices.size(); ++j) {
if (selectedColumnIndices[j].has_value()) {
const auto& val = selectedColumnIndices[j].value();
Id id = idTable(i, val.columnIndex_);
co_yield idToXMLBinding(val.variable_, id, qet.getQec()->getIndex(),
result->localVocab());
}
}
co_yield "\n </result>";
cancellationHandle->throwIfCancelled();
}
co_yield "\n </result>";
cancellationHandle->throwIfCancelled();
}
co_yield "\n</results>";
co_yield "\n</sparql>";
Expand Down Expand Up @@ -659,8 +678,9 @@ nlohmann::json ExportQueryExecutionTrees::computeQueryResultAsQLeverJSON(
const ad_utility::Timer& requestTimer,
CancellationHandle cancellationHandle) {
auto timeUntilFunctionCall = requestTimer.msecs();
std::shared_ptr<const Result> result =
qet.getResult(query._limitOffset._limit.has_value());
// Always request lazy if possible, the lower memory footprint outvalues the
// potential overhead of generators.
std::shared_ptr<const Result> result = qet.getResult(true);
result->logResultSize();

nlohmann::json j;
Expand All @@ -685,8 +705,7 @@ nlohmann::json ExportQueryExecutionTrees::computeQueryResultAsQLeverJSON(
j["runtimeInformation"]["meta"] = nlohmann::ordered_json(
qet.getRootOperation()->getRuntimeInfoWholeQuery());
RuntimeInformation runtimeInformation = qet.getRootOperation()->runtimeInfo();
runtimeInformation.addLimitOffsetRow(
query._limitOffset, std::chrono::milliseconds::zero(), false);
runtimeInformation.addLimitOffsetRow(query._limitOffset, false);
j["runtimeInformation"]["query_execution_tree"] =
nlohmann::ordered_json(runtimeInformation);

Expand Down
12 changes: 9 additions & 3 deletions src/engine/ExportQueryExecutionTrees.h
Original file line number Diff line number Diff line change
Expand Up @@ -178,16 +178,20 @@ class ExportQueryExecutionTrees {
const parsedQuery::SelectClause& selectClause,
LimitOffsetClause limitAndOffset, CancellationHandle cancellationHandle);

struct IndexWithTable {
size_t index_;
// Helper type that contains an `IdTable` and a view with related indices to
// access the `IdTable` with.
struct TableWithRange {
const IdTable& idTable_;
std::ranges::iota_view<uint64_t, uint64_t> view_;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you need the original absolute indices into the IdTable?
Otherwise you could return a single
std::ranges::subrange<const IdTable&>
and all your loops become singular again, and all the sonarcloud stuff goes away?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I need the values as well as a reference to the original IdTable to fill in the ConstructQueryExportContext

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It would be possible however, to refactor ConstructQueryExportContext to get some row reference instead, but this might be better suited for a follow-up PR.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thinking about it it doesn't solve the original issue. yielding subranges still requires 2 loops

};

// Yield all `IdTables` provided by the given `result`.
static cppcoro::generator<const IdTable&> getIdTables(const Result& result);

// Return a range that contains the indices of the rows that have to be
// exported from the `idTable` given the `LimitOffsetClause`. It takes into
// account the LIMIT, the OFFSET, and the actual size of the `idTable`
static cppcoro::generator<IndexWithTable> getRowIndices(
static cppcoro::generator<TableWithRange> getRowIndices(
LimitOffsetClause limitOffset, const Result& result);

FRIEND_TEST(ExportQueryExecutionTrees, getIdTablesReturnsSingletonIterator);
Expand All @@ -201,4 +205,6 @@ class ExportQueryExecutionTrees {
ensureCorrectSlicingOfIdTablesWhenFirstAndSecondArePartial);
FRIEND_TEST(ExportQueryExecutionTrees,
ensureCorrectSlicingOfIdTablesWhenFirstAndLastArePartial);
FRIEND_TEST(ExportQueryExecutionTrees,
ensureGeneratorIsNotConsumedWhenNotRequired);
};
Loading
Loading