loopback-next
loopback-next copied to clipboard
Support `inq` splitting in `findByForeignKeys`
Under the hood, inclusion resolvers are implemented using inq operator:
- Gather PK/FK values from source models.
- Query target models using
inqand PK/FK values from step 1. - Assign target models to navigational property in source models.
This can be problematic when the number of source instances is large, we don't
know if all databases support inq with arbitrary number of items.
To address this issue, LB3 is implementing "inq splitting", where a single query
with arbitrary-sized inq condition is split into multiple queries where each
query has a reasonably-sized inq condition.
Connectors are allowed to specify the maximum inq size supported by the
database via dataSource.settings.inqLimit option. By default, inqLimit is
set to 256.
In this task, we need to improve findByForeignKeys (see https://github.com/strongloop/loopback-next/issues/3443) to handle the maximum size of inq parameter supported by the target database (data-source). When the list of provided FK values is too long, then we should split it into smaller chunks and execute multiple queries.
However, because our Repository interface is generic and does not assume that
a repository has to be backed by a data-source, I am proposing to expose
inqLimit via a new property of the Repository interface instead of accessing
the parameter via DataSource settings.
/**
* Description of capabilities offered by the connector backing the given
* datasource.
*/
export interface RepositoryCapabilities {
/**
* Maximum number of items allowed for `inq` operators.
* This value is used to split queries when resolving related models
* for a large number of source instances.
*/
inqLimit?: number;
}
To preserve backwards compatibility with existing repository implementation, we
cannot add RepositoryCapabilities directly to the Repository class. We need
to introduce a new interface instead that Repositories can (or may not)
implement.
export interface RepositoryWithCapabilities {
capabilities: RepositoryCapabilities;
}
See #3387 for more details & a prototype implementation.
Acceptance criteria
To allow the helper to detect inqLimit, we need to extend Repository interfaces.
- [ ] Introduce
RepositoryCapabilitiesinterface (calledConnectorCapabilitiesin the spike), this interface will have a single propertyinqLimit(for now). - [ ] Introduce
RepositoryWithCapabilitiesinterface (calledWithCapabilitiesin the spike),= this interface should definecapabilitiesproperty. - [ ] Implement
isRepositoryWithCapabilitiestype guard - [ ] Implement
getRepositoryCapabilitieshelper
The rest should be straightforward:
- [ ] Modify
findByForeignKeysto obtaininqLimitfrom repository capabilities and implement query splitting (see the spike implementation). - [ ] Write unit-level tests where we verify what (and how many) queries are called by
findByForeignKeys. - [ ] Write integration-level tests (in
repository-tests) to verify that connectors can handleinqLimitthey are advertising. For example, create a test that runs a query returning 1000 records.
Discussion: From @raymondfeng: do we want to split the queries for users, or just have a limit amount of queries(i.e throws err msg if they exceed the limit)?
Any thoughts? @strongloop/loopback-maintainers
I was considering this use case:
Lots of shopping websites use Load More button(or scroll down) to load more items.
This kind of button/function only sends a certain amount of queries as the more queries it sends, the longer it takes.
From my understanding and my experience, it's developer's responsibility to decide how many queries to send.
So it makes more sense to me that we just set a limit for the amount of inq for findByForeignKeys instead of splitting it for them.
@agnes512, are we ready to estimate this task?
In LoopBack 3, we are splitting the queries for the user automatically - see https://github.com/strongloop/loopback-datasource-juggler/blob/814c55c7cd7ac49f3a52729949a5d0aaeb91853d/lib/include.js#L237-L264.
I am fine to put this feature on hold for now and wait if there are any users asking for it.
If we decide to do so, then I'd like to ensure that the enforced inq limit is large enough to support most users and that a helpful error message is reported when the limit is reached. Since #3443 has been already finished, I am proposing to open a new story where we will investigate what errors are reported now, add acceptance-level tests to trigger the scenario & verify the reported error, and ensure consistent behavior for all supported databases.
According to https://github.com/strongloop/loopback-next/issues/1352, this is out of scope for Q4.
In LoopBack 3, we are splitting the queries for the user automatically - see https://github.com/strongloop/loopback-datasource-juggler/blob/814c55c7cd7ac49f3a52729949a5d0aaeb91853d/lib/include.js#L237-L264.
I am fine to put this feature on hold for now and wait if there are any users asking for it.
If we decide to do so, then I'd like to ensure that the enforced inq limit is large enough to support most users and that a helpful error message is reported when the limit is reached. Since #3443 has been already finished, I am proposing to open a new story where we will investigate what errors are reported now, add acceptance-level tests to trigger the scenario & verify the reported error, and ensure consistent behavior for all supported databases.
I'm asking for this, if possible. I have some queries that I am not interested in paging and just want the api to return all of the rows even if it's more than 1000 in oracle.
Re-opening the issue for further discussion.
For further info on why we need this, please see the bug report I submitted #8773