-
-
Notifications
You must be signed in to change notification settings - Fork 407
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
EmberData | Request Service #860
Conversation
A better option IMO would be to:
I don't think ember-data is perfect, but when it hasn't been a good fit i've had zero problems the general ember-data paradigm has worked well for me in practice building many applications over many years. maybe some of us will have to fork this repo and put everything in a new namespace to keep our apps working without the churn of rewriting everything to fit in with some new model we didn't want. $0.02 😭 😞 |
|
||
## Motivation | ||
|
||
- `Serializer` lacks the context necessary to serialize/normalize data on a per-request basis |
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 can't think of a time when I've needed to do this.
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.
Amending requests and responses in the adapter happens quite often. In fact it turns out to be so necessary that both EmberData's default adapters do this and EmberData's core logic does this post-serialization but with-request-context in order to ensure that the cache receives enough context to correctly be patched. These special things EmberData is doing come at a high cost to flexibility and performance, and can't be rectified by user-land changes. So even if you aren't doing this, you are actually doing this.
I'd say every app that I've worked on has had to customize behavior in the adapter to support normalization or serialization at least once. It could be that I've hit some weird bias where in a hundred apps I haven't encountered the opposite, but it's also simply the case that more often than not APIs include meta in headers, return error responses in different formats, utilize binary formats to save bytes, or return ""
or {}
on success when something more was expected or needed.
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.
Sounds awesome!
I took a quick look at our adapters and serializers. There are a few things that it would be good if this new paradigm supported (and maybe it does, and I've only missed it in the RFC).
-
Great if a handler can "pause a request" (not resolve/fulfill it), and initiate another request to the backend, fetching additional information, and only resolve the original request after the second request has completed.
We use something like this today, to dynamically fetch translations via a special endpoint, in response to metadata stored on the regular response payloads.
-
Changing the HTTP type before a request hits the wire, for example changing GET to POST (we use this for certain search queries, if the query body is larger than the URL max limit). But reading the RFC, it sounds like this would be doable?
-
Our serializers currently do mostly one of these tasks: (a) skip certain attributes on the model from being sent over the wire and (b) transform model attributes, e.g. between
string-date
andmoment-js-instance
.It would be good if these common use-cases was easy to implement under the new paradigm. Maybe by storing metadata on the Model or ModelAttribute, and then reading these out in the handler and acting on them.
Or, this logic could be moved closer to the model, e.g. by attaching serialization and de-serialization methods directly on models and model attributes somehow?
This seems to me like a move towards lower-level primitives, which I don't have anything against! But if more logic will be customized outside Ember Data, it helps to think about easy tools we can add that make doing so easier. For example, being able to set metadata on model properties, that can somehow be read and acted on during serialization.
Overall I'm very positive! 🏅
Handlers will be invoked in the order they are registered ("fifo", first-in | ||
first-out), and may only be registered up until the first request is made. It | ||
is recommended but not required to register all handlers at one time in order | ||
to ensure explicitly visible handler ordering. |
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.
In some frameworks with a list of handlers invoked in a particular order (e.g. Ruby Rack), there are often methods such as:
- insert
- insert_before
- move_after
They all help adding in additional "functions" at a specific index.
Could be useful here too. But something that can easily be saved for a later revision (not needed for v1). However, maybe insert
would be a better name than use
, because it's easier to add the sibling methods mentioned above (works better name-wise).
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.
Initially a DAG was considered (similar to how initializers and addons can be ordered); however, the trouble with these approaches is while the ordering seems to make sense when viewing the late-add in isolation, the lack of explicit upfront ordering still generates high levels of confusion. If it turns out there's a strong use-case for just-in-time additions of handlers we can consider that in a new RFC.
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.
Makes sense! At least for us, I don't see this as a problem.
Also, makes sense to keep the scope of initial RFC down a bit. As you mention, if this turns out to be a problem for a lot of people, one can always follow up later with additional API surface.
Yes, making this trivial is one of the primary motivations of this pattern. There are two things enabled here:
So for instance: {
async request(context, next) {
const { records, url } = context.request;
if (records && url.endWith('/users')) {
const responses = await Promise.all(records.map(record => next(buildQuery(record))));
// .. munge responses together
return data;
}
}
} The same goes for the inverse, you can join requests in a handler so long as the original requests resolve correctly. For instance to batch calls to {
requests = new Map();
nextTick = null;
async batchRequest(requests, next) {
// call next as few or as many times as needed
}
async request(context, next) {
if (context.request.op === 'findRecord') {
this.requests.set(context, defer());
this.nextTick = this.nextTick || setTimeout(() => {
this.batchRequest(this.requests, next);
this.requests = new Map();
this.nextTick = null;
}, 10);
return defer.promise;
}
}
}
Doable, and handler can continue the chain with a modified query. The power of this is that the cache will only ever be aware of the initial query. At first this sounds like a limitation, but it ensures that a broad set of abilities to fulfill data from sources of varying natures is neatly abstracted away. So from the cache's perspective, if what you initially queried was a
Serialization/Normalization is a handler concern for requests originating via the store. The reason we don't provide a direct replacement for serializers is because doing it inline like this brings enough power and simplicity that a few well crafted functions specific to your app will always beat any pattern we can design. That said: (b) is mostly a separate RFC – coming soon – in which we will replace |
Great, I assumed that would be possible (I think this architecture is likely a bit more powerful then what we currently need, but definitely a good approach!).
Makes a lot of sense. In this particular example, that's exactly what you'd want (cache it as if it was a GET).
Sounds great! Yeah, I think when this has landed, we'll probably also notice smaller 'pain points' when migrating from "the old way". And as we do, those pain points can be discussed and if a lot of others have similar issues, those can likely be remedied fairly easily. Either via community "plugin handlers", by sharing best practices (e.g. in docs; overall I think docs are underutilized, in that everything doesn't need to be supported in a framework via public API, sometimes 'receipts' [this is how you do it] is better, and requires no additional API surface) or by making small additions to remedy. Overall I'm very positive on these upcoming changes for Ember Data! 🏅 @runspired I've upped our sponsorship to $500. Do you know anyone else that will be working on this, where sponsorship may be appreciated? |
@les2 thanks for the feedback! Here are some thoughts on each bit.
We very much care for EmberData, these changes allow us to evolve into a better future. The design proposed here allows us to support dozens of new use-cases, solve innumerable bugs, simplify one of the most complex aspects beginners and advanced users alike express regular frustration about, remove one of the biggest footguns when using EmberData (lack of pagination out of the box) and remove several entanglements with classic-ember (mixins, EmberObject, resolver usage).
This is a new package, but it is also very much EmberData. What the
What we are proposing here is that the same infrastructure that makes it super simple to fetch and work with modeled data would allow you to just as easily and simply fetch and work with raw data, with the added benefit that this coordination and standardization of how you fetch data results in both an improved developer experience when working in an app and an easier time maintaining and building applications as key request concerns can be coordinated centrally.
I'm glad <3 It has worked well enough for lots of folks, and I've seen it used to great power and effect and also to great misery and sorrow. I chose to invest in the project entirely because I believed it's paradigms could be improved such that the "worked well" becomes "worked amazing" as the rule, not the exception.
I updated the RFC to describe a bit more about the transition path. I honestly doubt anyone will feel the need to fork, but if someone did want to maintain the legacy behaviors forever they could in-fact do so not by forking but by choosing to adopt and continue maintaining the legacy-handler, adapter, and serializer packages once we choose to end our own support. The power of the RequestManager approach is that these older legacy behaviors can still be described by it. The challenge would be around removing the Mixins, jQuery, RSVP, and EmberObject from the adapter and serializer package in order to maintain compatibility with Ember once those concepts are fully eliminated.
Your $0.02 is hugely appreciated, and I'm glad to know another person who has gotten great utility out of this library in the past. Give this new pattern some trust and time. As with I'd go further and say we'll choose to maintain support for Adapter and Serializer for as long as Ember classic exists (likely until Ember 6.0) even if we ship our own 6.0 prior to that. It is overall low-cost for us to maintain this part of the codebase as long as Classic is around (the same is not true of Model, the changes to which will be a separate RFC), though specifically maintain means very occasional bug-fixes. These classes have been frozen in time due to their complexity for many years and that is unlikely to change. |
No, but if anyone out there would like to help there is (as always) much to do and it would be hugely appreciated <3 |
Something this RFC doesn't explicitly layout but should: how to handle errors thrown by requests from the store generated due to a backgroundReload (see: emberjs/data#3809) My thinking is: Since the RFC gives control over catching and handling the error to the request-manager, end users can choose to swallow the error. This allows us to consider two options:
|
Probably obvious, but I guess there are a few common reasons for errors during background reload:
I'd say the response may vary depending on which one that triggered it.
|
9378e5c
to
2093cde
Compare
Moving to FCP, barring further discussion will merge at earliest Friday December 2nd. |
* feat: request-manager * more docs * more docs * Add installation instructions * fix link * remove bad ref * update docs * more docs * more niceness * finish README doc * full response docs * stash work * some updates * cleanup compile * delete file * stash work * update flow chart * more updates to align with v2.1 cache * update package name * fix missing file * more docs * spike impl * nice things * test updates * address feedback on readmes * add request validator * more tests * more tests * more tests * polish streams support * add abort controller support * add Response handling * fix lint * more docs * more docs * rfc updates * initial spike of new cache types * update lockfile * isolate changes * isolate more things * remove exp types * udpate imports * docs updates * update docs * finish overview * fix lint * fix test * update lockfile * skip fix for now * fix tests * fix unstable test * we dont have prod tests for request manager yet so dont run them in prod * actually fix url * nvm * fix content-length check
Advance RFC #860 "EmberData | Request Service" to Stage Ready for Release
Advance RFC #860 `"EmberData | Request Service"` to Stage Released
Rendered