If you have any interest in RxJS, this article is for you. If you have no idea what RxJS is, I advise you not to worry about it and check out rubic...
For further actions, you may consider blocking this person and/or reporting abuse
Have you tried reading RxJSβ source code? I bet youβll learn a lot more than youβd do with diagrams. Even more, it wonβt seem that difficult!
is there a particular spot in RxJS' source code you found to be helpful for learning RxJS?
Iβd say everything. Itβs all about linked lists and OOP concepts. It may seem a lot, but itβs the same pattern used almost across the entire library.
Okay, I'll do that, thanks.
Awesome! Iβm sure it will be a fun journey.
Iβd recommend creating an rxjs project in StackBlitz and try to debug the sources right there. With stackblitz you can debug the TS files.
Iβd be glad to help if youβll have questions! Good luck
I have a couple questions, Andrei. How did you get into RxJS? How has it helped you in your work?
About one year ago I started learning Angular, thatβs when I got started with rxjs as well. A couple of months ago I started digging through its source code out of curiosity and out of a little frustration that the diagrams werenβt enough for me, and I always felt like something was hidden from me.
I couldnβt say it helped in my job, because I stopped working due to some incoming exams, but Iβve been training myself on Stack Overflow. Now that Iβm more familiar with how it really works, I stopped thinking about βdata over timeβ and other analogies. For me, thinking about how the entities are connected with each other its enough. Of course, I cannot remember every detail, but my go-to documentation is the source code itself.
I feel like it helped me a lot to solve problems(e.g those from SO), which I couldnβt have been able to solve without knowing a bit of how the library really works.
I agree with your approach to go to the source code directly. And I really like the mental model for Observables in the context of Angular, it's like a perfect counterpart for the event-driven architecture of Node. I can't agree more, the source code should speak for itself.
I've been training on stack overflow too, actually I put the code for rubico up a couple times and got it roasted by the internet giants.
I guess in the Observable sense, there's no competition actually. In rxjs v7 Observables will be async iterable, and that actually ties in really nicely to rubico's
transform
. I made an issue here, but it looks like I won't have to do any extra work on it, unless I want to have backsupport for v6.I do have a problem with the operators though. rubico and rxjs can coexist, but I do not like the API with the operators. I am honestly competing rubico there because I think rubico can do it more concisely, more functionally, and more readably. I would like to have your thoughts on this.
Could you elaborate on
rubico can do it more concisely, more functionally, and more readably.
?I personally have nothing to complain about rxjs, it's my favorite library to be honest.
concisely
- what's the rxjs' flaw in this case?functionally
- I doubt it. rxjs has a lot of functionalities and some of them are extremely well-thought; it also offers enough tools to create marble testsreadably
- sometimes this is subjective, but I don't find rxjs code unreadable at allIn addition to this, rxjs is written in TS, which is a very big plus.
I personally think that rubico is a nice side-project and if you can develop something like this on your own(700+ commits), then reading rxjs' source code and understanding it will be a fun journey.
IMHO, there are also other parts that factor in whether people should use rubico instead of rxjs. rxjs is also used by other important libraries, such as nest.js, ngrx.
I believe it's pretty hard to convince people to learn rubico, as there is already a tool like rxjs which solves a lot problems and has a lot of functionalities that have been built over time. My opinion is that you should jump on the rxjs train!
concisely
- rxjs's fatal flaw in this regard is the Observable's influence over therxjs/operators
api. For example, rubico has only one functionmap
while rxjs hasconcatMap
,exhaustMap
,flatMap
,mergeMap
, andswitchMap
to name a few. The difference between these map functions is in the behavior on the underlying Observable. rubico does not have the issue of having to deal with an underlying Observable; it just projects each element of whatever collection is passed to it onto another collection of the same type. In this regard, rubico is more concise because it defines predictable behavior for a multitude of types (Array, Set, Map, ...) including the observable (v7 with @@asyncIterator).functionally
- I meant functionally as in the functional style, for exampleis more object oriented via
.pipe
and.subscribe
than the pure functional approach with rubicoNotice how with rubico, there are only functions except for the last input array. The power behind rubico is that the input is last instead of first. This is powerful because it doesn't lock you in to one type of collection; you have the freedom to transform any collection with rubico (including the v7 observable).
readably
- I know I sorta dug myself into the subjective hole with this one, but please hear me out. If we considerreadable
from the perspective of how many more people can read your code, rubico code will lead to less questions because there's less magic happening. rubico's functional purity and no special data types means you know exactly what is going to happen to your array if youmap
over it. In fact, please take a look at the implementation for map; there's really not a lot to it. It's just map that works on a lot of the built-in types. rubico also did not invent it's own paradigm, it just follows the functional one. In that regard, much of rubico's API naming is borrowed from the JavaScript language, popular JavaScript libraries, and even other languages. For example, RxJS, ramda, rubico, bash, this proposal, and scala all have the concept of apipe
whereby you chain functions together. Every single one of rubico's method names are derived in this way. Why? Because I wanted to make the code you write with rubico look as familiar and intuitive as possible. I cannot say the same for RxJS when they export functions in their 'rxjs/operators' API with names likebufferWhen
,mapTo
,switchMap
,switchMapTo
, etc. This API is questionable, to say the least, especially when there are whole articles dedicated to distinguishing the differences between some of these names.It's been an uphill battle to push out rubico for exactly the reasons you mention: RxJS has an entire ecosystem written around it and is way older and more mature. I think the difference here is rubico avoids many of the issues of RxJS by design. For example, you can consume async iterables with rubico's
transform
, but async iterables right now are causing problems for RxJS because they undermine so much of the functionality of Observables. From an API consumer's perspective (aka you and me), async iterables and observables are effectively the same; just attach a function to a stream of eventsThis problem with async iterables has motivated ben to publish the library in his tweet that provides strategies for converting Observables into async iterables. More details in this pr and this issue. If you ask me, RxJS is a sinking ship. Please come over to rubico.
An operator is a function which returns a function whose single argument is an
Observable
and whose return type is also anObservable
. So, theObservable
is just a fundamental unit of the library. If you'd spend some time reading the sources, you should see that this akin to thatNode
class which you use to build a linked list.These operators surely exist with a purpose. I find it very nice that rxjs has these options, because they are frequently met when solving problems.
rjxs'
map
might be the same as rubico's map, butmergeMap
is very useful when you're dealing with another observable(inner observable) and you need that value to be passed along in the stream.This makes me think that you didn't not carefully read on what observables are and what problems do they solve.
I find this more intuitiveL
than this:
because in the first snippet you can clearly understand which is the source and which is the consumer.
IMHO,
less magic
is not an advantage. As long as the magic is abstracted away from the dev, and it magically works, I see no problem in this.I'd that these operators exist with a purpose. Yes, at first
bufferWhen
,buffer
andbufferToggle
might seem not intuitive, but after understanding their purpose, everything should make sense. Not to mention how many benefits you'd get out of reading their source code.Here's an article which describes a possible use case for them(testing). I don't see a breaking change, it's just another feature.
I would disagree since it's used by important technologies, such as angular, ngrx etc...
I think you would benefit a lot if you started contributing to rxjs(by writing articles, SO, helping other people). I just see no point in going against this with another library, because rxjs is really great. It also has a lot more features: testing your own observables using marble tests, multicasting, adding teardown logic.
Andrei, I oppose rxjs because the ideology is weak. At no point in either of the release branches or the main site do they talk about why RxJS. It always just starts with what. It's like reading a research paper that starts with "hey guys, here's our research! it's important because we say so!". There needs to be a reason why the research is being done, otherwise no one will care. Just take a look at any TC39 proposal, it always starts with why. For RxJS, on the other hand, the current first time experience on the site is
They never answered my first question: why is this here in the first place? On rubico it's
Why should always precede what. Otherwise, your product will run in circles. I experienced this first hand at my previous company.
RxJS is giving me PTSD because all I see is what. RxJS is a sinking ship because the spec is moving on without them: async iterables are here to stay, while Observables are a stage 1 proposal. Observables are not async iterables, but they compete for the same streaming data model. Async Iterables were created partly because of the pains of NodeJS streams, which are analogs to RxJS Observables. Part of these pains are backpressure problems, from RxJS creator:
^ that turned into this library.
You can see excitement for async iterable streams in this issue in the streams spec. Read it for the good vibes. Here's a nice summary from domenic
Really, I'm just against spaghetti code. You'll get spaghetti if you model streams as a push datatype like Observable. You'll get worry free code if you model streams as async iterables. Finally, you'll get /comfy/ code if you feed an async iterable to rubico's
transform
. Check out this functional style async iterable webserver with rubico and denoAndrei, I don't think you actually like things that magically work because you are someone who reads source code. Isn't it core to you to understand the way something works by studying its source line by line? You've done that quite a bit for RxJS. I'm not really sure what they did to deserve that. I see your passion, that's why I want you to use and contribute to rubico. Stop wasting your talent on a sinking ship. The project is just me right now, but there's a lot of interesting stuff I've spec'd out. Really, I would love to have your help on this.
Because you don't actually know rxjs in depth.
Very interesting work. Thank you for your effort and sharing.
Curious how to create event streams in rubico as RxJS
fromEvent()
does. Tried to find it from the documentation and tutorials but couldn't. Sample code snippets or mention in documentation will be helpful as it was the most exciting feature to me. If event streams could be handled as in RxJS, then rubico will be the winning FRP framework for my next big project.A starting point perhaps :
DEMO
Hey @richard Tong,
thanks for this interesting perspective.
I will look into rubico and, if i have time / find something interesting, give you some feedback on this.
I found your post while searching for a state management solution for deno.
I also do have some custom rxjs operators published on npm, so I think, I can nicely compare the two experiences.
Rubico sounds interesting for me, since the idea of 'using the platform' is also my main paradigm right now.
But I also wonder whether rubico will be a real complete (or neraly complete, maybe without edge cases) replacement for the vast functionalities build into rxjs.
Let's see! Anyway, thanks for the post!
Hey Benjamin, thanks for the comment. I think you're right - rubico has a long way to go. Actually I was just on my way to implementing some stuff I have on its roadmap, since today my day is light. Cheers
Hey @richytong I like the design of your API, I do find it more intuitive, but I also like RxJS's design as well. You keep mentioning the "Why" and I suspect it's probably due to a very famous TED talk. If you absolutely need the why, just have in mind that RxJS is an implementation of reactivex in JS:
reactivex.io/intro.html
You can find the "why" of reactivex in the website above. For the why of RxJS, just add "in Javascript" at the very end.
On the topic of
map
,concatMap
,exhaustMap
,mergeMap
andswitchMap
all have their place and are useful in their own way. I'm not sure about how I would use them in the backend but I constantly use them on the frontend in different scenarios. EspeciallyswitchMap
.BehaviorSubject
in RxJS)multicast
in RxJS)map.pool
are bad for treeshaking.compose
calledpipe
:v