-
-
Notifications
You must be signed in to change notification settings - Fork 15.2k
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
Redux and it's relation to CQRS (and other things) #351
Comments
So event sourcing, if you haven't come across it before, treats everything that happens in your domain as an "event", records every event, and regards the events as the source-of-truth. All other data is just viewed as derived state, and should be able to be regenerated by replaying events. There's a very strong analogy with Redux here, I think. Actions are events, and using pure reducer functions to change state guarantees that replaying the actions will get you back the same state. Event sourcing and CQRS go hand in hand, but it is possible to do either one without the other. (Not all the leading proponent of event sourcing would agree with that statement, but that's another discussion.) The long and the short of it is: I think there are interesting things to be learned from both CQRS and Event Sourcing, and they may help get clarity around some of the ideas in Redux. I do find the language from CQRS clearer than that from Flux and Redux, but that's probably a matter of taste. I certainly think the word "Store" has become very overloaded in the Flux world though. |
P.S. I might, when I get a moment, see if I can take my toy example and convert it to use serialisable actions. It shouldn't be hard, and will help get rid of some of the metaprogramming black magic under the hood. |
+1 For any non-familiar to CQRS + Event-Source, here's a great resource for learning: @notahat, I do agree with you:
IMHO, CQRS and Flux are very alike, diverging only in their terminology, while Event-Source is exactly what Redux DevTools implements. |
I added a diagram that'll hopefully make what I was doing with my toy CQRS example a bit clearer: https://github.com/envato/zero-flux/blob/master/docs/zero-flux.pdf |
While maybe not intentional, the "single store" in redux reminds me of a presentation by @gregoryyoung where he describes "Current State is a Left Fold of previous behaviors" @gaearon Do you think there is a direct correlation between the Redux store and the ideas behind EventStore? |
I do agree, in my experience using several Flux implementations, Usually doing something in flux happens in an async action. Those would be better called commands probably. |
Closing, as this doesn't seem actionable. |
Action creators are basically commands Flux & Redux are IMO implementations of the general CQRS pattern / principle. From previous discussions I've had, I think I recall the reason for fresh terminology is either
As you said, the CQRS principle is often conflated with DDD / Event Sourcing etc - which I think can detract from the simple core idea. |
@glenjamin for me action creators should not be command. We leave in the frontend so basically there is no command, when user does some action in the UI it should rather be an event. Backend concepts can't always strictly relate to frontend concepts, because backend receive user intent in an aynchronous fashion while front receives it synchronously, so indeed the things always happen in the past. I don't think action creators are needed at all, particularly if you start using redux-saga. See how I replace an action creator here: http://stackoverflow.com/a/34623840/82609 redux-thunk: <div onClick={e => dispatch(actions.loadUserProfile(123)}>Robert</div>
function loadUserProfile(userId) {
return dispatch => fetch(`http://data.com/${userId}`)
.then(res => res.json())
.then(
data => dispatch({ type: 'USER_PROFILE_LOADED', data }),
err => dispatch({ type: 'USER_PROFILE_LOAD_FAILED', err })
);
} redux-saga: <div onClick={e => dispatch({ type: 'USER_NAME_CLICKED', 123 })}>Robert</div>
function* loadUserProfileSaga() {
while(true) {
const action = yield take("USER_NAME_CLICKED")
const userId = action.payload;
try {
const userProfile = yield fetch('http://data.com/${userId}')
yield put({ type: 'USER_PROFILE_LOADED', userProfile })
}
catch(err) {
yield put({ type: 'USER_PROFILE_LOAD_FAILED', err })
}
}
} Also just wrote a little bit about how Domain-Driven-Design (often used in backend with eventsourcing) can also be used in frontend. See #1315 (comment) |
Hi, I think this is a very interesting topic! In my project I have used react-redux and thunks to create the front-end basics. I have long planned to use event-sourcing/CQRS on the server to handle, process and optimize data. In learning, and loving, Redux I realized that this will most likely be the perfect environments for processing the commands server-side and generate the checkpoints that the clients will see. The store will be mostly like a complex cache for the data store, loaded on demand. As commands come in, redux-sagas (or maybe thunks) will process them and send actions both to persistence (checkpoints) and the reducers. The clients will read the result of this after receiving RT-updates and refresh their redux-state correspondingly. Does anyone have any experience with this, or any recommendations against it? |
@tvedtorama I agree with you, Redux can work in the backend/server-side even like a web server or your entire backend. |
Hey, if you want to use Redux for the backend, remember Redux is a very tiny library of only 70loc conceptually, and Redux concepts are largely inspired from existing backend stream processing systems. Instead of porting back raw Redux to the backend, where most of the tooling (devtools/react-redux/reselect...) can't be used, and many required tooling is missing (log persistance, state indexation, snapshotting and persisted log replay), just to reuse these existing 70loc, why not leverage existing backend toolings that solved the problem even before Redux existed in the first place? I'm thinking of very simple tools, like EventStore , which permit for a long time to persist a distributed event-log, and to build log projections (equivalent of Redux action reducers). As the projection can emit new events it can also act like a saga. For more advanced usecases, many tools already exist as well, like Kafka, Samza, Storm... Don't forget to look as domain driven design, as modeling your app with events may not be enough in most cases, and you should probably understand what is a bounded context is and also an aggregate root. |
Just to add that actually devtools can be used on backend via sockets, here's an example. |
great @zalmoxisus :) But not sure it's very useful for backend, I'll have to wait to see a real usage of that to be convinced :p. People building event-sourced apps usually replay the whole log after a reducer/projection change, and it's less likely the user use time travel in my opinion. Eventually it could help to reset given state to reproduce the context of a bug, but it's already what snapshots provide anyway :) |
@slorber Thanks for the great feedback! I'll look into the server-targeted tools you mentioned. However, I think the ability to use the same frameworks on both front-end and server could be an advantage in itself, as this lessens the learning curve of the tool-stack. I even hope to reuse some of the reducers between the layers. DDD issues are definitely also valid concerns, and one should be careful not to just copy the structures of the front end. I was thinking to deal with bounded contexts either by simply defining parallel structures in the store - or parallel stores with different sagas / reducers altogether. These parallel structures would then partially respond to the same events, but calculate their state differently. @sebas5384 Very interesting projects, I'll try to look into them in more depth when time permits. |
@notahat Thanks very much! Do you have another copy of that diagram you created somewhere? That link appears to be dead. |
Opening this issue to continue an interesting Twitter discussion, copied below for posterity:
Edygar de Lima @edygarDeLima says:
Dan Abramov @dan_abramov says:
Bill Fisher @fisherwebdev says:
Pete Yandell @notahat says:
Dan Abramov @dan_abramov says:
Pete Yandell @notahat says:
Pete Yandell @notahat says:
The text was updated successfully, but these errors were encountered: