-
Notifications
You must be signed in to change notification settings - Fork 114
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
Stores. Pagination. Frustration #63
Comments
For pagination we want to get backend + frontend hybrid solution which merges scalability of the first and usability of the second. I think the first fluxy lib which will document this will get popularity boost. I'm not really a flux architecture fan, but you've got the point. The only reasonable attempt to discuss it I have found in 30 minutes is: That's something.
"Content" stores keep app data as First things first. Most tutorial examples represent app state as Arrays of models. Array is extremely unsuitable for get / update / delete operations. You have to implement even most primitive access methods in store, have to mess with offsets, ohmy... This is extremely low-level and stupid. So, I agree here, Map is better choice if we limit ourselves for simplest types. It support one-line basic ops and can be iterated (e.g. used for INDEX) as well. I talk in general, but you can imagine ImmutableJS objects as a reference. They have enough methods to get rid of Lodash helpers for this stuff. But what about those 3 types of stores? We've got data duplication concerns as soon as we copied data relations to separate stores. What are stores after all? They are poor-man's database. Reactive database, and so what? There are already a lot of them. And if we agree to see stores as tables of database, we should apply same well-known, time-proven rules and criterias to them: normalization, rich query language, ability to join, import / export questions, history, etc. Referential integrity features would be highly desirable in this case of Indexed lists. But we have none. So the set of stores feels and behaves like the most uncomfortable database ever. I still can't believe that for 6+ years of frontend evolvement we have nothing reuseful here. |
Should we just wait for Relay to solve this for us in some magic way? It won't be REST on backend, so I'm personally not very excited. But it's interesting to inspect their direction, at least. They merge Stores in one single Store. That's even closer to DB concept. Joins should be easier as well as imports / exports. They add GraphQL language: queries over data. That's DB aspect too. There are other very interesting approaches. DataScript is a lightweight frontend DB. It's in ClojureScript with JS bindings. Sadly, no JS documentation, only test suite with rather obscure examples. Received very good feedback from top guys. RxJS gives another approach. Same immutable database we want can be viewed as stream. Anyone wants to classify and explain all this? 😃 |
So I don't have solutions for you, but I'm in the same boat if that helps. It seems so weird that it is so hard to do a basic CRUD app with all this "new and amazing technology", something that takes 0 time in Rails. I'm currently working on getting the simplicity of simple_form, kaminari etc. up the React + Flummox, but it's still quite a lot of work. |
Yeah I'm running into the same issues, still haven't come up with a good solution yet. Things are not so bad in simple apps but as soon as you have complex relationships it gets a bit more complicated. |
I'll keep this open because I agree that Flux doesn't have a well-established pattern for solving this. For complex stuff, you really need some sort of query language. |
@ivan-kleshnin Btw I appreciate you laying out your frustrations here. They are shared by many of us, including me. I haven't said anything on this thread because I also don't have a good solution. |
Yeah I wonder if its a query language or perhaps a base set of stores for avoiding all the boilerplate (similar to @gaearon has). |
Yeah, creating a base store that takes implements the ideas in @gaearon's example project could conceivably work pretty well. I've been using a base store for stuff like serializing Immutable data, but not really for pagination yet. |
@nambrot, @tappleby, @acdlite thank you guys. It's very helpful to know we share common feelings about this. There is always a chance one missed something, considering today's velocity of changes. I'm sure many of us are working on CRUD demo apps, so I propose to share their examples here (as soon as they are more or less "ready") and discuss them. I want to detail my complaints about them as well. Most of such apps are TODO-apps. They are too-specific and not really similar to what most of us build for business. Those CRUD apps should really account next pitfalls:
Relay direction implies component should declare it's external data dependencies. With data load actions we can emulate this to some degree in imperative way. Rather primitive solution is next. I think it's crucial to have responsive page without blank screens. Gradual page load is better as soon as page doesn't jerk and shows some sort of spinner. So I consider the way of data loading in this flummox demo as non-mainstream solution.
As soon as we load data in actions we:
If I understand correctly, Relay will bring some "magic" declarative solution which will undercover load this data for us. I see a lot of benefits and drawbacks with this approach so it will be at least controversial. Anyway that's enough arguments to mark data load as additional pain point. |
Think this might be slightly relevant here. I just listened to TheChangelog #131 with @wycats and @tomdale, and especially found their comments on Ember Data fascinating. http://emberjs.com/blog/2014/03/18/the-road-to-ember-data-1-0.html I think what we are really lacking is the best practice to managing our "models" and their relationships. I so heartedly agree with their assessment that basically everyone reinvents syncing data. Like Flux is very helpful in making the "how" work, but it's kind of a pain to be able to declaratively specify how we should sync the data in the stores with the various requests being made from the App. I think the Ember team showed pretty well that they can get the right abstraction with the Router and I'm excited to see whether they can achieve the same with Ember Data and what that possibly means for us Fluxxors. |
@nambrot 👍 I think that's pretty relevant. |
Look Swarm JS? |
@MrEfrem, that's M of MVC with a lot of magic. Not inspirational from the first sight. |
@ivan-kleshnin Thanks for the link, very good read. I actually found http://christianalfoni.github.io/javascript/2015/02/06/plant-a-baobab-tree-in-your-flux-application.html#comment-1902000761 to be most reflective of what I'm looking for. Not sure about Swarm JS either. Personally, I'm looking for a CRUD/REST compatible solution as 90% of all my apps fit greatly in within that abstraction |
Yes, I also keep wondering how often CRUD/REST is used and how few examples of it are published. |
I'm interested in this too. I've used Ember for the past few years, and wanted to check out React. The React layer itself is amazing, but it seems like there's huge missing pieces/patterns for implementing basic DB-backed CRUD model stuff. |
I've created a schematic view of most popular frontend archs https://github.com/Paqmind/reactive |
There's something in the making at Facebook : React + GraphQL + Flux + Relay: https://www.youtube.com/watch?v=9sc8Pyc51uU |
Hey Guys, sorry to revive this without a constructive solution, but I just finished my CRUD example for anyone interested https://github.com/nambrot/rails-webpack-react-flux/ |
This is the elephant in the room. I'm looking for solutions to the problems that modern SPAs must face but what I find is the nth way to increment a counter. Some thoughts:
Let's take modern frontend development seriously. ES6/7, cursors (other hyped terms follow...) are cute and I have fun playing with them in my spare time but in order to get my daily work done I must solve the big problem of state management in all my apps. If you can point me to resources that I'm missing, please (please!) share them. I'll be extraordinarily happy to be wrong and take it all back. |
The Case for Flux by @gaearon touches this topic briefly. Perhaps Dan might want to elaborate on usage scenarios like this, either in general, or in context of his new redux project and its recent update: reduxjs/redux#46. |
With Redux, I'd implement a generic store (or a higher-order store) for pagination and use store functional composition to reuse this functionality across different stores. |
I agree with @gcanti I think there are two issues at hand.
The whole Flux movement is very focus on 1), making sure that data state on the client itself is easy to reason about and one change in one part of the app is very predictable on its effect on the client side state. I 100% agree that the next step, and imo more difficult but also more important problem is that once you introduced stateful clients how to make sure they they are consistent. I think the reason why this is not a much-debated topic is because you can get away with "eventual consistency". Most people simply can afford to just load data once and assume it's relatively consistent due to session length and other factors. If you are interested in truly globally distributed and consistent state management, I imagine you don't have much choice than to look at Meteor or RethinkDB as far as I can see. Otherwise I think we are milking REST as far as we can already. |
Who doesn't? 💯 But he produces content faster than I'm able to consume, and I'm trying hard... So I thought it was useful to provide another reference, even though he was mentioned here a few times already. |
I too am attempting to figure out these issue for a Reflux app. Namely, the best approach to getting a single entity from a store w/o having to load the entire database clientside first (re: Pagination) yet also avoiding having duplicate data spanning separate Stores to keep in sync. Reflux seems to introduce the other 'feature' of Stores trigger() indifferentiable events. Reflux moved Flux's case statements of actions from the Store level to the consuming View. Now if I attempt to have a single store and load "all" vs. a single entity not already in the store, the View has to case out what is going on in the listener callback. |
FWIW Redux today has an example with pagination. |
@gcanti hits this one so hard on the head for me it hurts :) I'm struggling with the problems he outlines EXACTLY right now in a React + Reflux application, and looking around for examples almost everything is completely contrived; they either keep all their application state in the store as an object (i.e. Todos) that don't load anything from the server and just add to it over time or they load data in the most naive way possible at the beginning as a "starter" set of data and then manipulate thereafter, but I'm having a really hard time finding something realistic where they a) load data from the server that ideally is kept maximally consistent with the server and b) has multiple dependent requests. On top of that I'd love to see something that handles this once you add in routing but that's a little orthogonal to this discussion I suppose. |
Not exactly a flummox question, but would be great to discuss it somewhere.
It's up to @acdlite to close this as offtopic if he wants to keep issues clear (some people do).
Some people also prefer to discuss on Gitter. But those threads are temporal and non indexable.
Pagination is a good example to illustrate some thoughts. Everyone makes pagination so
we should have countless implementations, libraries and competition here, right?
Well... actually no.
When I've come from backend programming I just wanted to reimplement basic CRUD stuff. All that features I had before: filters, sorts, pagination. I expected to find multiple solutions, smart decisions for this. So naive! I started with Backbone. I read through a lot of Backbone books. I watched a couple of Backbone video courses. About zero enlightment. They didn't even approach the questions of state management. Load all your data, then do frontend-only filters, frontend-only sorts, frontend-only pagination. Backend versions were more scalable!
I switched to Ampersand. Same things there. Just improved Backbone, still bloody low-level.
Logicless templates. No, thanks. Then React. Better render, same widgets / todo apps.
Still no state management solutions.
The text was updated successfully, but these errors were encountered: