-
Notifications
You must be signed in to change notification settings - Fork 734
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
Split Caching review and thoughts #3377
Split Caching review and thoughts #3377
Conversation
@jaikumar-b: Thank you for submitting a pull request! Before we can merge it, you'll need to sign the Apollo Contributor License Agreement here: https://contribute.apollographql.com/ |
Closed and created an issue instead #3378 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I'm still struggling to understand what value you're getting out of this new identifier
field in the cache read/writes. I'm concerned about how the cache will know how to handle manual cache writes that are not associated with a fetch request, since those won't have an identifier. Will your chained caching mechanisms break if data is mutated in the cache without an identifier
?
I'm also feeling that we might not need to actually use the identifier when reading data, but only when writing it. When writing data, you use your query's caching policy to determine if you want to write it to only an in-memory store, vs in-memory & disk persisted. But reading should go through the chain and look for data in each cache. Unless you are trying to use knowledge based on the identifier to skip looking in one cache if you know the data won't be there? Which to me seems like it could be done within your cache chain implementation without needing to ever pass the identifier through the entire cache reading process. Does that sound right to you? I may be missing some things because I'm not 100% clear on what you're trying to achieve here yet.
In 1.0, our fetch functions have a requestContext
parameter that allows you to pass in an arbitrary type here. Rather than just using the UUID
, that might be a better approach, as it allows you to expose any additional stored information you might need in that object.
@@ -7,14 +7,14 @@ public protocol NormalizedCache { | |||
/// - Parameters: | |||
/// - key: The cache keys to load data for | |||
/// - Returns: A dictionary of cache keys to records containing the records that have been found. | |||
func loadRecords(forKeys keys: Set<CacheKey>) throws -> [CacheKey: Record] | |||
|
|||
func loadRecords(forKeys keys: Set<CacheKey>, identifier: UUID?) throws -> [CacheKey: Record] |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm wondering why you need the identifier added to the loadRecords
method. Neither of the implementations in this PR seem to be even using that parameter.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The goal was to use it in the custom implementation, but as pointed out, will mostly not need this and will remove.
if let cachedResult = cache[key] { | ||
return try cachedResult.get() | ||
} | ||
|
||
assert(pendingLoads.contains(key)) | ||
|
||
let values = try batchLoad(pendingLoads) | ||
|
||
let values = try batchLoad(pendingLoads, identifier) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There is a problem here. You could theoretically use the subscript
to add pendingLoads
with one identifier, and then call load
with a different identifier
, which would then run the batch load the the second identifier, even for the pendingLoads
that should use the other identifier.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ahh good catch, although I was able to fix this and get all the tests to pass by separating identifier loads vs non identifier loads by hashing the identifier with the cache key internal to the loader, kinda started to question whether this was all really needed when reading the cache as you pointed out.
|
Howdy folks!
The topic of hybrid caching has been bought up multiple times and seems like you guys don't have it on your roadmap anytime soon. Gonna take a stab at this and wanted get some feedback on an idea that can help folks interested in this to get a workable solution until you guys come up with a first class support for caching
#3275
#1467
Background
I would like to create an API at the query level that developers can use to define the storage policy of the query when initializing an instance like so
This is different than the Kotlin version of Apollo which uses cache control headers but I somehow prefer this over the Kotlin api if the storage policy is exclusively defined on the client side.
One hurdle in implementation of a custom normalized cache based on the policy defined by query is that there is no way to associate normalized cache methods with the query's storage policy so that the custom normalized cache implementation can make decisions according to the policies.
Proposal
Noticed that we have a
contextIdentifier
which can be instantiated for each fetch operation, propagating that identifier all the way to the caching layer allows for extending the functionality of existing InMemory and SQLite implementations Apollo has by creating some wrapper classes. These wrapper classes can have aStoragePolicyResolver
which maintains the policy vs identifier in memory till the request is complete from network chain.I was able to create a
NormalizedCacheChain
object that accepts implementations of NormalizedCache.Probably can extend more cache implementations as needed by the application.
Questions
NOTE -
Have this built out for older versions due to familiarity, seems like can be done for 1.x versions as well