Summarized using AI

React Server Components... in Rails?

Jérôme Parent-Lévesque • April 03, 2024 • Montréal, Canada • Talk

In this engaging talk, Jérôme Parent-Lévesque explores the integration of React Server Components within the Rails framework at the April 2024 Montreal.rb Meetup. He discusses the evolution of web development practices, particularly the potential of React Server Components to enhance user experience by reducing server-client round trips and enabling data streaming. Although React is widely used alongside Rails, Parent-Lévesque questions its current limitations and presents alternatives.

Key Points Discussed:
- Understanding React Server Components: These components are designed to simplify web development by allowing data to be streamed back to the client as soon as it's available. However, their usage is presently limited to the Next.js framework with Node backends.
- React vs. Hotwire: Parent-Lévesque contrasts the older approach of using React with the newer Hotwire framework favored by Rails, emphasizing React's advantages in managing complex state and providing a superior user experience.
- Issues with Current Implementations: He outlines some pain points in using React with Rails, including slower development processes due to the need for additional APIs, increased loading times, and cumbersome JavaScript bundles leading to data fetching challenges.
- Proposed Solutions: To mitigate these issues, he proposes delivering data with the initial HTML payload, reducing round trips, and illustrating how Data Structure changes can aid flexibility. He advocates for a streaming approach that would load components progressively, improving the overall performance and user experience.
- Rendering Challenges: The challenge of distributing props from server to front-end components efficiently is explored. He suggests ways to streamline data transfer and replication of the rendering process using JSON formats.
- Integration with React's Newest Features: Parent-Lévesque provides insight into utilizing features like the Suspense API to handle loading states while waiting for data, addressing potential reflows and user experience issues during rendering.
- Testing Considerations: He briefly discusses the implications of streaming on testing practices, suggesting that end-to-end testing could become more complex but unit tests could remain unaffected.

Conclusions: Parent-Lévesque concludes that React Server Components pave the way for a more responsive development cycle within Rails environments by allowing for optimized data handling and reduced rendering times. The talk encourages continued exploration and refinement of integrating React with Rails, underscoring the balance between backend and frontend responsibilities to enhance developer experience and application performance.

React Server Components... in Rails?
Jérôme Parent-Lévesque • Montréal, Canada • Talk

Date: April 03, 2024
Published: unknown
Announced: unknown

Montreal.rb Ruby Talk 2024/04 - React Server Components... in Rails? - Jérôme Parent-Lévesque - Engineering Lead at Potloc

React Server Components are the next milestone in the React core team's initiative to ease web development (and are the hot new thing!). They aim to greatly improve the user experience by lowering the number of roundtrips necessary between the client and server and by streaming data as soon as it becomes available. However, they are currently only usable in the Next.js framework with a Node backend... But couldn't we mimic what this backend does in Rails and leverage this to modernize React development in Rails?

In this talk we will explain some of the inner workings of React Server Components, their benefits and how they can be implemented in any backend.

Montreal.rb Meetup April 2024

00:00:02.960 all right recording is started perfect H hi everyone thanks so much for coming so as Matt said this is something that uh
00:00:09.360 it's been almost a year in the making uh was an idea that yeah that was sparked
00:00:15.360 uh almost a year ago we did the little demo at the hackathon and since then I've still been toying with it and
00:00:21.119 trying to make it uh more and more usable and and trying to understand better as well everything that's
00:00:26.480 happening under the hood so the presentation today is sort of the culmination of all that and trying to
00:00:31.880 share this knowledge with all of you because I think there's something in there that could be really interesting for the future of development in
00:00:37.760 rails uh so let's start first with a quick survey of the room trying to get an idea of who I'm talking to here so
00:00:45.160 what who of you uh are using rails on a regular basis all right big number cool what
00:00:53.399 about react also a huge number all right so there's a lot of companies that are
00:00:58.600 using uh rea I think that shows that's good react and rails work well today uh
00:01:05.119 but there's also this new kit on the Block hot wire the new uh oneperson framework approach that rails is pushing
00:01:10.680 forwards are any of you using hot wire also on a regular basis one two and a half all right H
00:01:19.000 good well if you're one of those people you might feel something like this and hopefully my clicker is
00:01:28.720 working no
00:01:37.680 uh I'll try to get it working but uh yeah you might feel something like this so this is a a quick meme I found um and
00:01:45.719 uh and yeah like reactor so 2022 right then 2023 rails released a rail 7 I
00:01:52.200 thought hot water came out to 202 oh my God maybe uh kind of like that means
00:01:58.039 though yeah it was set as the default ra 7 I think if I remember correctly and so it gained a lot of adoption at least in
00:02:04.640 last year and and a bit more and yeah clearly like uh there's a lot of uh interest in this right because you
00:02:11.680 there's a lot of benefits right you don't need to know react um you can develop faster you don't need um a new
00:02:19.280 data fetching layer but I'll get into that but I want to talk about I think my clicker isn't working so what I want to
00:02:26.599 talk about is um that I think there's still a place for react in the development world right yes you maybe
00:02:32.440 you can go faster with Hotwire but Hotwire isn't Hotwire and similar Solutions right like HDMX and things
00:02:38.519 like that there aren't they aren't particularly well suited to um like what I would call front-end applications
00:02:45.319 right not just a web page but an application that has a ton of state and UI management and that has a just a
00:02:51.840 complicated set of features that um uh that it needs to support and for those
00:02:57.959 purposes I still still think that react is King so some of the the pros that I think still makes react a winner in some
00:03:05.239 cases I think it's a sort of a decision you have to take on a pro project basis
00:03:11.159 is uh well first of all that it might just be already built into your company and your code base you might have a team
00:03:16.760 that's already proficient that react and is using it daily you don't want to refactor all your code refactor Your
00:03:23.519 Design system to something completely different that would be way too big of an undertaking and then what I mentioned is
00:03:30.360 that react is still king of ux uh you can do everything through Hotwire but you might need to put a little more
00:03:36.640 effort when you want to just get the experience just the way you want it and
00:03:41.799 finally arguably it's also an easier development mental model especially if you have a complicated UI H then uh this
00:03:49.599 notion of UI is a function of state where the whole UI is defined from the state is something that's very
00:03:54.680 convenient yes Andy I agree with the arguably because e is way simpler than we actually
00:04:00.239 yeah and like hot wire idea wire other thing I wanted to say king of ux is
00:04:05.519 becoming disputed now I presented last month GL DSL for web which lets you use Ruby in the front end and half the code
00:04:12.120 you to react in Ruby instead check it out guys yeah so interesting H and yeah
00:04:19.120 like I think you know uh every technology has a specific like there's going to be pros and cons to everything
00:04:25.360 and I don't think there's one king for all projects I think at the end of the day you need to take the right right decision for whatever you're building
00:04:31.680 and react I still believe is one of the the good options today um but it comes
00:04:36.840 with some downsides so you if you've been working with rails as many of sorry react on Rails as many of you have been
00:04:42.639 uh as said have said that you do then you might have noticed that you know yeah there's slower development you need
00:04:48.800 this extra data fetching API often graph sometimes rest and on top of that you
00:04:54.840 also have slower page loads right so you need um there's this extra round trip
00:05:00.280 request that needs to go from the client to the server to fetch the data and so you might get into this sort of uh
00:05:06.600 spinner hell that you might have noticed on some react Pages where you start loading the page and then there's a
00:05:12.320 bunch of spinners that are waiting for data to fetch and then once that's fetched maybe it loads components that
00:05:17.520 also themselves need to go fetch more data and you get into this U what they call data fetching waterfall problem and
00:05:24.960 finally of course it's larger JavaScript bundles which might lead to also a slower uh page load time so just to
00:05:33.800 exemplify the the waterfall problem I'm gonna use this type of diagram a lot today to to go over um the pros and cons
00:05:40.680 of different approaches so the server sends the client the initial web page
00:05:46.360 and JavaScript bundle the the then the JavaScript bundle once it's loaded that means react
00:05:52.080 is loaded and it's able to render the page and then there's that roundtrip request to go fetch the data and then
00:05:57.400 render that finally so that's the situation today and the this presentation what I want to go over with
00:06:03.720 you is like just try to figure out how we could make this better how could we make this closer to uh Hotwire HDMX
00:06:11.639 while keeping the developer experience also great and uh and Isa fuse
00:06:17.639 yeah so when I see this the first thing I think about is you know let's try to
00:06:23.599 remove the round trip right why not just fetch the data directly at the start and send that straight to the to the the
00:06:29.759 front end on the first request and we can do that oh I hope
00:06:34.840 you're sitting close enough for the small code I apologize um and we can do that and today this is already supported
00:06:41.199 in in most uh react Integrations in rails so what I'm showing at the top here is the Erb file the template where
00:06:47.840 there's a helper method in this case called react component where you can specify the react component you want to
00:06:53.599 render on the front end so from the back end you can already specify what you want to render and pass it the props
00:07:00.319 that it needs so you can already pass it all the data it needs technically and um just to give you an
00:07:06.720 idea of how this might be implemented under the hood usually you'd use a gem for this but under the hood it might
00:07:12.080 look something like this where the helper gets replaced by this sort of template div that contains you know data
00:07:20.000 attributes that specify what's the react component you want to render what are its props and then there's a little
00:07:26.280 JavaScript piece of code that can run and dynamically load the right react class and pass it to props and render
00:07:33.840 that all right so just quickly go over this pseudo code completely untested by the way but um you know you get the this
00:07:42.440 element you'd get the class that you need to load and import that dynamically
00:07:47.599 and then parse the props and create the component and render that in the root that you uh that you've
00:07:55.639 instantiated okay and now great we've already solved the problem there's no more
00:08:01.080 waterfall but there's a key cost here that's kind of hidden and not shown in this diagram uh the talked earlier that
00:08:08.080 we still want to maintain the developer experience and speed of development well there's something here that's going to
00:08:13.800 be very annoying and so if I show you for example what the entry point component might look like that kind of
00:08:20.199 that can give you an idea a better idea of where that problem lies so here's a an example page and the
00:08:27.400 entry point component I might use a card from your design system some HTML elements like the header and then have
00:08:34.560 some sub react components used as well in this case I just use the body and
00:08:40.039 footer um and keep in mind you know those body and footer components are themselves going to have many children
00:08:46.080 of other react components and you're going to quickly get into uh a very huge tree of components that need to be
00:08:52.720 rendered but the problem here comes from the fact that the props that you're passing from the server well they need
00:08:58.680 to be distributed around to all of the specific react components on the front end that need specific data points for
00:09:05.120 them to render the correct thing and so you run into this situation where from the back end you need to know not just
00:09:13.519 all the data you need to render but also format it in a way that the front end can distribute it to all the right
00:09:18.880 components and then on the front end you get stuck with this problem of Distributing all of that so you end up
00:09:24.760 with a prop drilling problem commonly called where you know you keep having having to redistribute the props and
00:09:30.920 then pass them down to their children with themselves who themselves have to also pass it down to their children yeah
00:09:37.680 curious are you a fan of using react context to solve this problem uh or part of the
00:09:44.959 problem I personally I don't know if I'm a fan it depends on the situation here
00:09:50.120 it's I don't think it would solve the problem right like You' still need to
00:09:55.519 come up with the common structure of the data that the front end and the back would agree on to as to where you go and
00:10:02.440 get that I the problem pass like having pull it from the
00:10:09.079 context yeah it would it would definitely solve the problem of like having to explicitly pass it but then
00:10:14.360 you still need to know where to go look for that data so the components aren't as reusable anymore because they need to
00:10:20.640 know exactly at what key they need to go yeah um but yeah I mean at least you can
00:10:28.320 clean up the code a little bit let's say yeah um yeah so so this is what this might
00:10:34.880 look like and so is there a way that we could make this better is there a way you know we want each component to be
00:10:41.279 able to know to to to have all of its data that it's need that it needs but the back end would like to maybe
00:10:49.200 directly pass the props to what to the comp sorry to the front end components that need them and so we can come up
00:10:56.320 with a sort of pseudo code that could look something like this maybe we could you know in our view instead of
00:11:02.079 rendering one entry point component start thinking about rendering a whole tree of react components and so here I'm
00:11:09.279 reproducing the entry point but in the view and the react component helper now
00:11:14.360 I'm I'm uh kind of imagining that it can support a block argument where you can
00:11:20.160 pass children that is not the case of any reacton rail gems that I know of today but theoretically that's possible
00:11:28.279 and you could so this way specify a whole structure of the the the page you
00:11:33.720 want to render like the whole react Tre that you want to render and now your props well you can specify directly the
00:11:40.519 props that each component needs right and you could even you know think about
00:11:46.320 making a backend notion of a component as a model that knows exactly the react component it's rendering the props it
00:11:52.600 needs and it could take care of data fetching for that you could come up with a whole framework around
00:11:57.680 that and so just to show you that this could work we could do uh let me show you what the the what this um template
00:12:06.200 might render down to so we have the similar template like divs used this template at the at the top here so we
00:12:13.639 have the react card with the same children and here notice we're using other like templates within that or
00:12:20.560 specifying the body and the photo components and then the only difference in the code is that now we can iterate
00:12:26.560 over the children of the component we're looking at and for any of those uh import the right react component pass
00:12:33.160 this props and basically we're rebuilding the jsx tree in this little bit of JavaScript so again skip some
00:12:40.680 Corners here not handling correctly the like uh the Dom node for H1 but you can imagine how this could
00:12:49.639 work all right and so we have this great situation now where the there's no more
00:12:57.120 um extra roundtrip the database queries are all done at the start all the data is sent to the client and uh you try it
00:13:04.800 in production and you well you get users that are a bit like this I'm going to take this moment to
00:13:11.240 take a sip of
00:13:17.720 water so what happened well we didn't really think
00:13:23.600 about this scenario right like what happens if your props like all the database queries that need to happen to
00:13:29.440 fetch all the data take a while and even if they don't take a while now your client your users still have to wait for
00:13:36.480 the entire time that all the data needs to be fetched before they see something on the page so the first content full
00:13:42.519 paint is happening much later than before right and just to show you let's compare that to what the situation was U
00:13:49.800 with the traditional react on Rails approach right so the first content full paint happened as soon as we had uh
00:13:57.519 react starting to render on the front end we didn't need the all the database queries because that happened in a
00:14:03.279 separate API column so at least the user had some initial feedback and knew that something was going on they didn't have
00:14:09.680 to you know wonder if the page was even working at
00:14:14.759 all so um let's think about how to solve that problem but first I want to take a
00:14:20.959 step back and uh revisit the format that we've been using to pass down this this
00:14:26.839 jsx template that we want to render on the front end so right now like the lat latest version
00:14:32.360 we had was something like this where we were using HTML to Nest a bunch of
00:14:38.079 placeholders to to to be used to identify which of those are react components and how they should be
00:14:43.759 rendered but we could just as well use Jason for this and it's going to make our data parsing or the parsing of this
00:14:49.920 structure much easier in the future and so let's switch over to Jason just for the the sake of this
00:14:55.920 demo so we can have the same uh component represent the same jsx tree as a Json where you have a root component
00:15:02.839 the card it has some children and each children can have you know is also a
00:15:08.199 component in itself and has some props optionally so we have the same data
00:15:14.240 definition we can do the same thing and send that you know through our Erb
00:15:19.399 template for example we could just send that as Json that we store in a little JavaScript variable we process this
00:15:26.040 thing the same way as before or the same way without the the Dom uh manipulation or the Dom
00:15:32.319 lookups and but basically it could be a just a different approach to passing the
00:15:37.560 data from the back end all right so let's Circle back to
00:15:42.600 the the slowness problem of the data fetching let's say that in our example it's the body component the props sorry
00:15:49.399 for the body component that are very slow to load what we could do is instead just
00:15:54.959 maybe not show the body component right maybe we wait we wait until we have the right data so we could put a placeholder
00:16:02.079 instead and just say look look front end wait for it I will get you what ever
00:16:07.399 needs to go there but for now just keep in mind there will be something that goes there and then the second key idea
00:16:13.959 is that we could change this format into something that's streamed from the
00:16:19.360 server so our server would initially send this but then as soon as the data
00:16:24.680 is ready as soon as the placeholder is ready to be replaced then we keep adding some more lines to the stream uh that
00:16:31.360 contain whatever needs to go and replace that placeholder so it might look something like this so first we get the first
00:16:38.240 block and then a few seconds later we send out another block containing what needs to be rendered with the ID that
00:16:44.160 references where that needs to
00:16:49.399 go and so now the only question is how do we actually render that on the front end and that's a a tricky one so let me
00:16:56.560 actually go back um the first part we've done right that that that works and the
00:17:01.800 placeholder we can just skip or put a a a empty div if we want to but how do we
00:17:09.079 after react has rendered something how do we go and hook into the render process to add something that was
00:17:15.319 initially missing that's actually very tricky to do in react right like that having this interaction with react from
00:17:21.559 outside react World is pretty tricky it's possible but not uh super
00:17:26.799 convenient to do but fortunately with react's next release of server components um they have an API to do
00:17:34.080 just that and it's called create from readable stream with create from readable stream
00:17:40.520 you can pass a data stream directly to that function it expects a specific format which we'll get into but it can
00:17:47.960 then render that content directly in the jsx and uh even do some nice things for
00:17:53.640 us like it will actually be integrated with the react renderer and we'll see uh in a few slides like some some of the
00:17:59.080 benefits of them um but the only thing we need to get this working is basically to use uh
00:18:05.960 so to pass the data stream in the right format so react expects a specific format as you can expect to know how it
00:18:12.039 needs to render the react components using this method and so we had our Json format
00:18:19.559 this is the actual react format well actually it's the sorry the one of the previous versions of what this wire
00:18:26.039 format looked like since I built those slides the the for the the format that slain slightly but is still very
00:18:32.760 similar um and so this is very close to our Json approach uh you'll notice that
00:18:38.120 our block here of line zero is the main content that we're that we're trying to send so uh with a few different a few
00:18:46.960 differences right so first of all the first two lines are indicating where to
00:18:52.400 load certain uh react components and what files are they and what chunks are they if you're using Webb and it's
00:18:58.880 chunking your JavaScript and at what import name are they this is something that I kind of skipped over in the
00:19:05.080 previous implementations but it would have also been required to be able to to implement any of the previous uh little
00:19:11.320 scripts that I showed uh and so this zero block being our main uh body or main uh sorry page
00:19:18.520 that we're trying to render is first referencing line one uh and it's so it's
00:19:24.400 saying okay this is a first of all a card component that has props here the props would
00:19:31.320 actually be expanded we can imagine that you might have you might be passing some data here and then it has children and
00:19:38.400 in the children same as before we have our H1 header with its title we have our placeholder same as
00:19:46.120 before we just put a an ID reference and finally we have our footer which itself is also a frontend
00:19:52.840 component so it needs to refer to the line where we defined that and it has its own props and then This is highly
00:19:59.240 streamable right the idea behind using this and not Json is basically just that it's a little more compact so you're
00:20:05.240 going to send a little less data over the wire um and you can stream this the
00:20:11.200 same way as we talked before where after a few seconds you you know keep the connection open from the server and once
00:20:16.880 your data is ready stream the final part so in this case the uh last body
00:20:22.120 component and its props and this will work directly with this create from readable stream react
00:20:30.240 API and now well now their picture is looking much better so now that means
00:20:36.039 you can have your front end which right from the start is getting
00:20:41.960 this initial payload with the placeholders so it renders really quickly and you get a first contentful pane that gets uh quickly into the the
00:20:49.520 eyes of the users while in the background your server has already started doing all the database all the data fetching from the
00:20:55.840 database all those slow queries are already running and only once they're ready they get streamed to the the front
00:21:01.679 end and you get to render um and and they render as soon as as soon as that's
00:21:08.640 ready oh yes sorry so you have two reg
00:21:15.159 Parts cont the sh aeton layout yeah in
00:21:20.240 this case the naming is a bit confusing but imagine the shell being like the the jsx tree we sent with the placeholders
00:21:27.400 and then content is once we have everything Final in practice though with this structure you can actually send data in
00:21:35.679 in pieces right you can have many placeholders and each of them can be streamed as soon as they're ready so you
00:21:40.919 might have like multiple different arrows that come in from the data from the from the back end as soon as that's
00:21:46.320 ready you keep rendering a bit more when you're rering the
00:21:52.559 sh um is there Reflow that's produced once uh the content loads in
00:22:00.120 uh yeah exactly yeah so that's one of the challenges we'll tackle in a minute but yes definitely like that uh
00:22:06.039 placeholder that we put right now is nothing so once you add the thing in is going to make everything jump in the
00:22:11.799 page and that might lead to a bad user experience I think that's actually my next slide yeah so what do we show while
00:22:18.440 the elements load right um because again we would have that jumping
00:22:25.320 problem well react has an aply named suspense API just for this uh you can
00:22:31.679 wrap your elements in suspense provide a fallback the fallback is what's going to
00:22:36.799 be rendered until every children of this has done loading and uh now you can specify for
00:22:43.480 example a skeleton so that your page doesn't jump right or just a big loading spinner
00:22:49.760 yeah yes correct but the nice thing about using Creed from readable stream
00:22:55.000 is that since it integrates directly in the react renderer while it supports suspense as well so all those placeholders are directly supported and
00:23:02.120 will be able to make use of suspense and show the fall back until the the placeholder has been replaced by the
00:23:08.080 actual content you want and uh yeah so just to show you an
00:23:14.320 example of how suspense might be used in jsx so in the our entry point component you would put it just to wrap uh
00:23:20.600 whatever you want to to show a fallback instead so in our case maybe just a body but note that we could also hide the
00:23:27.279 entire page if we wanted to until the the body is loaded and in our case we can show like a spinner from our design
00:23:34.000 system component let's say and uh that will be the placeholder until the data
00:23:39.360 is loaded and all we need to do now is specify the suspense as part of our our
00:23:46.520 tree so we just need to adjust so that this change is reflected as well in the data we sent from the back end and so
00:23:53.559 our wire format is changed slightly uh our children of the main block now have
00:23:59.120 a the the L3 reference to suspense here suspense being just a sort of constant
00:24:04.799 string I don't know exactly why they pick that in the format and uh it specifies the fallback as a prop right
00:24:11.120 here which is itself a reference to another uh front end component called spinner and it sells and the children is
00:24:18.440 where we put the placeholder for what we want to um to load
00:24:23.840 later and so basically the the whole conclusion of this is that you know to make react server components work in
00:24:30.200 rails all we need to do is replicate their Warrior format replicate that streaming format that I just showed so
00:24:36.120 we need a way to go from a definition of the components on the back end render that down to this format support
00:24:43.159 streaming so that we can add lines as soon as the data is all loaded and Bob's
00:24:48.640 your uncle you plug that in to create from a readable stream on the front end and you got a A working version of this
00:24:55.640 yeah yes Andy question format that just showed us for theable
00:25:01.320 data yeah so that comes from the react server component standard exactly yeah
00:25:06.760 unfortunately they have very very little documentation about this format which is um a bit sad because I it kind of feels
00:25:14.440 like they're they're locking this into just a nodejs backend World um but there
00:25:20.640 is starting to be more and more documentation more people digging in of course the source the source code is also open source um so it is possible to
00:25:28.320 replicate this and I'm hoping to see more efforts from the community as well in the future for that uh so I went ahead and tried it out
00:25:37.640 and that's like what we started at the hackathon last year over the last uh over the last year I've been refining the code a little bit just show you uh
00:25:45.159 something working in this so I'll jump over to the demo and uh yeah just show you an idea of like what this might look
00:25:51.440 like from a user perspective but also from a software developer perspective
00:26:23.720 all right one second your life
00:26:36.960 all right I think we're good to go all right so this is the um the react note demo actually this is a demo that the
00:26:43.440 react team themselves built for showcasing uh server components and what
00:26:48.520 I did is I took the demo and reimplemented all the backend and rails but the front end is basically the
00:26:55.799 same so if I refresh the page you'll see uh well okay that was a bit
00:27:02.360 too quick I think a lot of stuff was cached but it's also running on my Local Host uh so there's not too much to see
00:27:08.520 but just to show you some functionalities of the app it's a note taking app you can click on the notes to see what's happening but the contents
00:27:16.399 you can edit and uh change what you want you get a preview on the right that updates in
00:27:22.760 real time and when you're done you can save everything you can also search
00:27:29.159 and you can create new notes but I'll skip that for now so this is basically the the simple app and uh just to show
00:27:35.360 you the so this is all running with in with a a rails backend but just to show you the streaming aspect of this I'm
00:27:41.600 going to slow down artificially what happens when we load a a single note I'm
00:27:46.919 going to go back in the code and just put a sleep of two seconds in here and
00:27:52.880 uh me refresh the page just in case and now if I click this note you'll see the
00:27:58.360 than I was talking about so we're using the suspense to show a fallback and uh
00:28:03.799 and yeah and I can show you in the uh in
00:28:08.840 the inspect tab the network what's happening
00:28:13.960 here and so you'll
00:28:20.000 see just move this so it's a bit more
00:28:25.240 convenient well this is not working well at all on my laptop okay um sorry about that you're
00:28:32.799 not going to see much of the the stuff on the left Ah that's why I probably was
00:28:38.480 on a bigger screen right before okay um there we go so there's the initial
00:28:45.640 page load in this demo I'm cheating a little bit um the streaming format isn't
00:28:51.159 sent with the first page load it's actually a separate API call that the front end makes to get the data so we
00:28:57.600 are back to the the round trip problem I swear this is just temporary and this is
00:29:02.640 actually to match the server components demo that reacted they also do the same thing uh it makes this much easier if
00:29:09.200 you want to stream the data D directly with the first page load you have to embed that into the HTML um and so you
00:29:15.600 might have to use a like a an open-ended HTML that at the HTML page that at the
00:29:22.200 end you keep adding like script tags that keep pushing data into an array or something like that a little more
00:29:27.399 complicated this makes it just a bit easier but it is possible to also send it with the first page Lo and so this uh
00:29:34.720 first page uses the wire format so you can see here the the response I hope this is visible sorry for the small
00:29:41.840 screen yeah how about that yeah perfect so uh so our our API
00:29:51.000 that Returns the streaming format you can see it's working and if I click on a
00:29:56.080 note here
00:30:01.720 uh I do not see it there we go uh so it streamed the
00:30:08.720 data what I'll do is I'll go over and copy this to curl just so you can see the streaming
00:30:17.200 happening and so if I launch the same query you'll see uh we kind of stopped
00:30:23.440 here around this line this was the the main like block starting with z
00:30:31.600 and uh and then the final content loaded so after after this stuff here it took a
00:30:36.760 couple seconds and then we finally got the rest uh the rest being in this case the the content that would replace the
00:30:44.279 the Skeleton on the right and um yeah to show you a little
00:30:50.360 bit more about how this looks like from a developer perspective let me show you a bit of the code uh so to render this
00:30:59.080 wiio format we first need to define the the tree of react components and HTML
00:31:04.840 elements that we want to render um so I decided to go with an approach for this
00:31:10.760 based on Flex uh it's not the Erb like we've been doing since the started this
00:31:15.799 presentation just because I would have needed to parse HTML to then render it this wire format not as straightforward
00:31:22.720 using Flex is a bit easier it's all ruby and I can hook into the the methods to render it exactly how I want
00:31:29.240 to um so just to show you the main app so with flex you have a template method
00:31:34.919 in which you define uh like all the HTML elements you want to render and um and
00:31:41.080 in this case the only key difference is that you can render as well front end component so here you'll notice the
00:31:47.720 search field and the edit button are actually front end components yes Andy I just wanted to mention Flex is about two
00:31:53.679 years old but it's component structure and uh um API is almost exactly lifted
00:32:00.639 from glimmer originally glimmer it was created in 2007 the desktop version of
00:32:05.760 glimmer and glimmer copied something else it copied shoes and Mer copy uh so
00:32:11.440 Flex kind of copied almost exactly the same DSL glimmer so what you're showing
00:32:16.600 us here is very similar to glimmer DSL awesome great to know so yeah it could
00:32:21.720 make even an implementation of this with the glimmer DSL that renders down to this wi format um
00:32:28.480 and so the developer experience is pretty much the same as um as a regular Erb template but on top of that you now
00:32:35.000 get an easier way to specify front end components and um and the second new thing is this
00:32:43.120 suspense notion right so you can wrap some certain parts of your code in
00:32:48.240 suspense specify a fallback and then you can render um some
00:32:54.519 components that might take a while to do data loading in this case the noteless comp component I can show you which is which we specify as an
00:33:02.039 async render type of component this is just the little flag that we use to identify that and um and with this async
00:33:10.039 render what's what that's going to do is Once We Run The render method for this
00:33:15.120 component we're going to spin that off into a new fiber render the code asynchronously and only when it's ready
00:33:22.159 stream back the data to the through the uh the stream that's been open
00:33:28.000 but yeah um any questions about the code before I go back to the final points of the presentation is there anything specific you'd like to see uh or
00:33:35.320 something you think I forgot to demo also
00:33:43.679 possible yeah I'm confused though when should we use R like uh react components
00:33:51.360 pure of R foral and when should use for example sections and include like style
00:33:57.559 stuff because looking at first for me I'm confused of uh where we are separating
00:34:03.799 the layers of what would be like in the rugby what should be only react yeah
00:34:09.399 that's a really good question actually um this might be a new problem and a new
00:34:16.079 question that like needs some standards and conventions in our code bases but it's a good problem to have because now
00:34:21.560 you can decide how much of the rendering do you want to happen on the back end versus on the front end and you can do
00:34:27.960 do this on a per component level right I can turn this section into a a frontend
00:34:34.280 component and nothing's changed in my code right uh like I um H like you want
00:34:40.919 to add interactivity somewhere okay just move that component to the front end and you're good to go of course the react
00:34:47.159 implementation is a little nicer there in the sense that you don't need to rewrite your Ruby code into JavaScript
00:34:53.200 you it's still the JavaScript with react and if you move it to the front end you get access to
00:34:58.359 State and hooks right um but transferring from this format to uh to
00:35:04.079 the front end is also not that bad and you can still do a split of like having a a back end that is that takes care of
00:35:10.480 all the data fetching like a component for data fetching and then a component for rendering and user interactivity and
00:35:16.400 that would still work within this framework like you're on the on the on this definition here right you can have
00:35:22.359 your section that fetches the data and passes it down to the front end component to render it there is a super
00:35:28.640 flexibility side um all right so let me jump back to
00:35:34.320 the presentation real quick I'm GNA just share my screen once
00:35:39.760 more I think it's GNA be
00:35:54.079 easier okay all right uh so what is this react
00:35:59.800 server components feature then right so we've seen today that it's pretty much just this specification of the streaming
00:36:06.119 format that we saw and on top of that it's an API on the front end to take that definition to take that stream and
00:36:12.680 render that to the jsx tree it's also on the back end an API to
00:36:19.520 render um so backend react components or components that don't use state in
00:36:25.800 certain other hooks down to this wire format down to this streaming format um so in this sense it's
00:36:34.839 technically support for rendering JavaScript react components either on the server or on the client or both
00:36:41.119 right uh there's a few catches like so the the use States stuff like that and on this on the server you can use async
00:36:47.920 components and do data fetching directly from your database so there's maybe even an argument that like having them in
00:36:54.119 separate languages is a bit clearer you don't risk trying to do database fetching from the front end which
00:36:59.400 wouldn't work anyway but you know you don't risk uh injecting a secret key uh
00:37:04.520 in in a front end component for example Andy I disagree with that I think it developers would be a lot more
00:37:10.280 productive if it was one language on the server and the client because I used to program apps that were client server
00:37:16.680 before the internet started and that was a lot more Collective than the waste of time that
00:37:22.200 we have nowadays with four or five different languages but but what you're showing is very interested keep
00:37:28.359 sorry yeah know it's super interesting I think actually you know the like most of
00:37:33.640 the logic you're going to have is going to be around data fetching or around uh like frontend State and interactivity so
00:37:40.560 these are already things that have to be either on the front end or the back end right they can't like you can't do the
00:37:46.480 data fetching on the front end you can't do the uh user interactivity on the back end as much and so given that the stuff
00:37:54.079 that actually needs to transfer between front end and back end component for me is just this simple definition of like
00:38:00.520 hierarchy of components and that is easy to transfer there's a bit more logic obviously but I think it's a it might be
00:38:07.760 a worthwhile tradeoff and then my final Point here is that it's also pretty Bare Bones right
00:38:13.560 it leaves a lot of choice for framework creators and we had to dig pretty deep into the internals to figure out all
00:38:19.720 that stuff right and that's because actually react has decided to position themselves now as a what they call a
00:38:25.599 meta framework uh on official documentation for the next release they don't recommend you use react directly
00:38:32.720 they don't recommend you use the LI directly they recommend you use a framework that is set up to use react
00:38:38.240 and set up all the stuff around server components for you because there's a lot of stuff to worry about and this demo
00:38:44.160 like it was the simple part of that but then there's a bunch more about race conditions what happens if your stream
00:38:50.400 doesn't load in the right order and then we'll get into SSR in a second of server side rendering there's a lot of
00:38:55.960 complexities around that and it react is arguing that you're better off just leaving that to a full-fledged framework
00:39:02.480 that has already handled all all these edge cases for you the one thing it is not though uh so
00:39:10.480 server components are not server s side rendering and react makes this very clear distinction between the two for me
00:39:17.160 at first I was like what what do you mean they're both they both have like server in the name they for me server
00:39:22.319 components are rendered on the server like you're rendering down this this tree to the streaming format but the
00:39:29.200 distinction that reacts make that react makes is that the um rendering to the
00:39:34.440 streaming format is different than rendering to HTML and that's what they call service side rendering but they
00:39:40.599 also have an API to render the um all the jsx components including the front
00:39:46.960 end ones directly to HTML from the server so that way you're able to send a
00:39:52.960 a first payload to your users that already contains the built HTML so they don't even have have to wait for the
00:39:58.960 initial like uh react rendering on the front end they don't even have to wait for the JavaScript bundle to load and in
00:40:04.839 fact you can combine these two into something like this right now your
00:40:10.160 server can do the server side rendering that's my first step here so send the
00:40:15.599 HTML down to the client directly the client already has uh something visible
00:40:20.680 at that point and then can start downloading the JavaScript and do a process they call hydrating which is
00:40:26.800 just attaching your JavaScript code to the already existing HTML as opposed to rendering it from scratch and in
00:40:34.119 parallel you still have the same process of fetching all your data and streaming that to the front end through react
00:40:40.400 server components and now you get the best of everything where your first content full
00:40:45.720 paint happens really early the page becomes interactive as soon as the JavaScript is loaded and then your final
00:40:51.680 content paints uh it streams in and and keeps painting um as soon as it's ready
00:40:59.680 and so just to Circle back to to the first one of the first slides I had on reacting rails today well what would it
00:41:06.520 look like tomorrow now if we um take into consideration everything I've just presented so the pros stay the same but
00:41:12.800 the cons now well the slow development well now you don't need this data fetching anymore right this graphql API
00:41:20.240 was only used for the front end to have a way to go and query what it needs but now we're passing all that down directly
00:41:26.560 from the back end so we get rid of that and then the slower page load as we saw with the combination of SSR and react
00:41:33.839 server components well you don't need the round trip at all in fact you can even render stuff from the back end
00:41:39.119 directly down to HTML and it's going to be even faster than before and uh the final point the larger
00:41:46.560 about JavaScript bundles that still holds if you want to use react in the front end because you need interactivity
00:41:52.280 you need like all the the the nice methods behind this it is a little smaller though because now you can pick
00:41:58.359 and choose what you render on the front end and on the back end so you might decide to offload some of the stuff to
00:42:03.440 the back end which might also mean you don't need to use uh this specific npm Library anymore and you can just use a
00:42:09.960 ruby gem equivalent or something that's already built into rails yes um I don't
00:42:16.319 remember which Library did this but uh there was one um that I had
00:42:23.520 the being six months a year ago which
00:42:29.079 essentially didn't download any of the um event handler code oh I think I know
00:42:38.160 are you talking about quick is it possible you're talking about quick
00:42:45.200 lar yeah uh so this is a
00:42:50.559 like yeah I that's a good point so so what I've talked today right is about react specifically because react is the
00:42:56.839 first of the big front-end libraries to start implementing this kind of stuff
00:43:02.040 but if this picks up and people like it then I expect a lot more front end libraries to support this kind of thing
00:43:08.720 I think that the library you're talking about is quick I don't know it super well so I don't know if it's really compatible with this this sort of
00:43:15.119 development model that uh um definitely like other libraries and they're also
00:43:20.559 building Frameworks like a solid JS a solid start for example right they might
00:43:25.680 start looking at something like and I used react today but really you
00:43:30.760 could substitute that for anything else if uh if in the future they also add support for you yeah I had a question so
00:43:37.319 you mentioned that we're minimizing or eliminating data fetching API calls that's because now you're relying on
00:43:43.680 streaming for rendering the parts that are not rendered right away correct whereas in the past they got they ended
00:43:49.960 up fetching extra data later Y correct and I believe that was my last
00:44:03.440 everyone just a I guess final words like um I put a slide in there I'll share the
00:44:09.760 slides with everyone in the in the Meetup um I put a slide with references if you're curious about like where I
00:44:15.480 learned about this kind of stuff and um I also put a link to the demo code it's
00:44:21.440 all open source for like the version I built in rails uh if you're looking at interested at contributing go ahead and
00:44:27.599 and reach out I'm not sure how much of how much time I'll have to actually turn this into a framework it's something I'd
00:44:32.720 love to do but we'll see what happens but if you're also interested reach out and we can talk yeah so the problem that
00:44:40.000 you presented we we have it solved in our codebase at my company lexo by basically
00:44:48.400 um we usually pass any uh data that is available right away when we render a
00:44:54.680 page immediately to the react components and then we defer any data that is that
00:45:00.480 needs to be fetched later into the components themselves um and we don't have a problem where like basically the
00:45:08.800 patient will render right away and the components that need extra data will make extra API calls and we haven't had
00:45:14.440 any issues with that to be honest um I mean Vic what do you think do we have any problems with that believe so we
00:45:21.480 have it solved just by simply making the components responsible for whether they
00:45:26.680 want data to be received right away in the params or arguments or whether to make the call inside the component and
00:45:32.720 not render anything as soon as the data comes in it pops up uh so I'm curious is
00:45:38.280 that a solution is that a problem for you guys was that a problem is that is that part of the reason this was
00:45:44.480 created uh initially the app was built as a single page application with a a
00:45:49.520 single page application and uh relying on the design system uh heavily so
00:45:56.119 everything was buil with a basically a placeholder a div fruit and you're done so there wasn't any mixture of some Erv
00:46:04.280 code and some border BL that you could do in rails and then Plug and Play react
00:46:10.319 uh we needed that interface to be I don't know I guess an example would be
00:46:16.079 uh if you're building Gmail right pretty much there's no like individual comp
00:46:22.079 components that you would then later attach together it's the full thing and GMA will have a single spinner on the
00:46:29.200 entire interface our app is build the same way uh five six years ago it was
00:46:35.480 the train but also it fit the use case that we had uh so yeah there's a way
00:46:40.720 that you can build the app and you can have Dynamic components this way and you won't be facing this but if you need the
00:46:47.280 UI interaction to have this extra and you are in a
00:46:53.200 single page ification Paradigm then yeah you kind of SC because you don't have
00:46:58.800 that that possibility uh the the other thing is really the design system that
00:47:05.240 we had we're not CSS based we're like CSS and GS and material UI and other
00:47:11.200 things like this so are we
00:47:17.800 so um so basically anything that you're building needs to be JavaScript based
00:47:24.440 because otherwise there's no design in there's no color there's no nothing and so
00:47:30.280 that's CSS CSS can be apps something else yeah it's where you
00:47:38.920 you're basically the planing the CSS as you know Javas perance so it's two different paradigms
00:47:45.720 you join both of them together and yeah you're not in a world where you can devop a front page uh sorry a front V
00:47:54.559 appication that can with a mix scenario of some
00:48:01.319 backu so you're saying your has like one view
00:48:06.839 which everything there why you make it like
00:48:15.599 AP what you mean AP API it's pretty much it is yeah just
00:48:27.640 and other things
00:48:34.640 are for each entry point yeah okay that's a different I mean we do it
00:48:41.480 different obviously many different ways of doing this the paradig we had in the day we
00:48:49.040 still have today is a user experience that requires this sort of uh of
00:48:54.400 rendering and also that's how we build the design system meaning that the frontend developers had everything
00:49:01.000 available to them in JavaScript um without getting into coding but from a
00:49:06.640 uh hiring perspective it's quite hard to hire Ry devs of the certain seniority
00:49:12.599 that no full stack entirely uh so being able to hire JavaScript typescript
00:49:18.599 developers to concentrate on the front end was also kind of so you you have
00:49:23.799 like a mixture of many decision stats yeah end up costing you 200% the cost
00:49:28.920 you would have built it with Ruby you would have it would have cost you if few used Ruby that's my experience or even
00:49:35.480 400% is Javascript is a
00:49:41.599 disase yeah I mean I really like your back end part that was similar to the glimmer DSL it's a shame you didn't
00:49:47.720 attend my talk last month sorry I online on YouTube it's called frontend Ruby
00:49:53.319 glimmer DSL for web because it has the you have the back end piece of the puzzle solved uh glimmer glimmer has the
00:50:00.960 front end piece solved it would be great to you integrate the two together so I definitely discussed that with you after
00:50:07.520 the talk the front end part of in my opinion nothing intriguing or interesting comes out of react and I
00:50:13.839 didn't think server components were interesting at all in my opinion it was extreme over engineering from what I saw
00:50:21.359 and a lot of the front end code that you showed us is extremely ugly compared to what I showed last month so I mean to me
00:50:27.359 that's like you showed me PHP in 2005 I showed you be on Rails 2005 sorry like
00:50:33.200 that's what I me yeah yeah but what also it's kind of like somebody showing me the first car but I showed them the first rocket the car is nothing you know
00:50:40.119 like compared to it so definitely check out my talk with this is Thinking Inside
00:50:46.160 the Box react or what's common in JS uh what was demonstrated last month was
00:50:51.880 completely outside the box like we complete like a much much simpler solution that ends up shrinking the code
00:50:58.319 half so anyway sounds super cool I'll try to check it out uh just the the the
00:51:04.760 front end actually I didn't really show any front end today right what I showed was like what would be hidden behind a
00:51:10.599 framework or a library uh the front end stays the same old react so if you don't like that you're not going to like that
00:51:15.920 today but um it doesn't change your front end it means that your existing design system can still work and you
00:51:21.720 don't have to change anything about that um and you asked earlier about like uh what problem this solves I think
00:51:30.200 um I think it it's depending on your use case you
00:51:36.799 might actually be running into bottlenecks of performance or um uh like
00:51:42.160 scaling problems that make this kind of solution interesting um I think overall
00:51:48.680 though the main purpose of react server components is more around developer experience it's it is about building
00:51:54.559 this kind of optimized architecture optimize delivery system
00:51:59.960 for the users as easy as possible and in a way where developers can now very
00:52:05.520 easily choose what goes on the back end or the front end and one of my favorite Arguments for this is like uh the
00:52:11.599 ability to collocate everything that's related to one sort of thing or like
00:52:17.079 entity that you're rendering on the front end right now you can collocate the data fetching the JavaScript what
00:52:23.760 the the the you know the data fetching and the component all the rendering all in one place on your back and and um I
00:52:32.319 find that mental model of development very very convenient I'm still not sure
00:52:38.200 if this is the way to go for the future uh I think this is very very new and the communities also seems to be divided
00:52:44.400 there's also a server actions another feature from react that I'm much more on the edge there like as to whether that's
00:52:51.119 actually interesting or not or if that's dangerous um so I think next year is going to be interesting and uh I want to
00:52:58.760 build this because I want to jump in and see what could come out of it and if this is Ruby generates JavaScript code is
00:53:06.520 what you're saying or we have components on front it it actually it just renders to the streaming format that's all it
00:53:12.200 does yeah yeah any other questions yeah probably
00:53:17.720 from the Tes perspective now that you are um I say liberating over streaming
00:53:24.760 uh responses uh how did you deal with testing uh for example because let's say
00:53:32.319 tomorrow you want to make sure that you are rending 50 records from the database and since now you are using a lot of
00:53:38.720 streaming what happens if for one test it will probably get 10 or for the next one when I get 20 uh like have you
00:53:46.119 experienced flak test in the past because of that that's a really good question uh the reality is there are no
00:53:52.599 tests for that demo app uh this was just built to to see if it work Works um there I think I have a couple backend
00:53:59.200 tests just to make sure I can render to the the streaming format but that's about it um so uh this is untested
00:54:06.680 grounds I don't know um I think you could you know fake the streaming which
00:54:11.799 sounds very ugly and inconvenient but if maybe you have some helpers for that could work like you can test your front
00:54:18.680 end by faking the back end then you know at least you're you're starting to get somewhere otherwise uh have a way to dis
00:54:26.599 the async component maybe I'm I'm not sure yeah it's interesting saying from a
00:54:31.680 personal perspective I don't see like depends on the case sure but much like
00:54:38.599 value of testing only the return of the data because in the end we can either do like a system test to check it out use
00:54:45.520 your behavior or if you would like to check the return of the data doesn't need to be from the stream itself can be
00:54:51.839 for a p unit test from a part of the what you would like ref itself so you
00:54:57.920 either checking the data from a service or you can also check the behavior from a system test
00:55:04.920 like ATT tach from the response itself being the data uh yeah I don't know I don't know
00:55:14.319 yeah I mean the the the streaming format is parsable you could have utilities
00:55:19.839 helpers and your rspec test to you know in a request test for example to check that it contains the data you want um um
00:55:27.599 yeah I think that's probably the way i' go do the streaming breakes essentially
00:55:33.160 ja unit testing um or would you still be able to
00:55:38.839 load up component and the props that would be coming in you need to go through
00:55:44.799 streaming because you you mentioned you could fake the stream data that jeez that's like that's unreadable
00:55:53.119 so does it break it or there's still way to uh it depends right I guess like if I
00:55:59.720 was to do uh unit testing on the front end i' render individual components and
00:56:05.839 the streaming would happen outside of that right like the the create for mutable stream method which would be
00:56:12.680 what's problematic would be happening in some sort of uh entry point file right
00:56:17.839 in like the the sort of root JavaScript that's injected by the framework um and
00:56:23.520 so you don't uh for unit testing you just load the component that you need it doesn't
00:56:29.319 care about any streaming you still pass the props just says before right it doesn't change any of that
00:56:34.640 um it's only it's really only if you want to do like more of an endtoend test more of assistant test I think and in
00:56:39.880 that case you also have the back end to complete so
00:56:47.200 yeah all right I guess I'll stop the recording here but feel free to ask me
00:56:52.839 uh more questions uh after come come talk to me I'll stay stay for a bit before leaving and uh thanks very much
00:57:00.720 for coming
Explore all talks recorded at Montreal.rb Meetup
+6