Register now: Builder for Marketers & Designers

Announcing Visual Copilot - Figma to production in half the time

Builder.io logo
Contact Sales
Platform
Developers
Contact Sales
    Copy icon
    Twitter "X" icon
    LinkedIn icon
    Facebook icon

    A Guide to Server-Side Rendering

    Last updated: December 1, 2024

    Server-side rendering (SSR) has been around for a while, but it's worth exploring further. This technique can make your web apps faster and more SEO-friendly.

    In this guide, we'll explain SSR, why you might want to use it, and how to implement it without pulling your hair out. We'll cover the basics, compare it to client-side rendering, and discuss some practical examples.

    Builder.io SSR and SSG

    What is server-side rendering?

    Fundamentally, SSR is about rendering your web pages on the server instead of in the browser. When a user requests a page, the server does all the heavy lifting and sends a fully rendered page to the client. Then, the client-side JavaScript takes over to make it interactive.

    The server is doing the prep work in the kitchen, and the browser just has to plate and serve.

    Here's a minimal Express.js example:

    const express = require('express');
    const React = require('react');
    const ReactDOMServer = require('react-dom/server');
    const App = require('./App');
    
    const app = express();
    
    app.get('/', (req, res) => {
      const html = ReactDOMServer.renderToString(<App />);
      res.send(`
        <!DOCTYPE html>
        <html>
          <body>
            <div id="root">${html}</div>
            <script src="client.js"></script>
          </body>
        </html>
      `);
    });
    
    app.listen(3000, () => console.log('Server running on port 3000'));
    

    From server to browser with fully rendered pages

    When we talk about SSR delivering "fully rendered pages," it's important to understand what that actually means. Let's break it down:

    What is a fully rendered page?

    A fully rendered page is an HTML document containing all the content users would get when they first load the page. This includes:

    1. The complete DOM structure
    2. All text content
    3. Image placeholders and other media elements
    4. Initial styles

    Here's a basic example:

    <!DOCTYPE html>
    <html>
      <head>
        <title>My SSR Page</title>
        <style>
          /* Initial styles */
        </style>
      </head>
      <body>
        <header>
          <h1>Welcome to My Site</h1>
          <nav><!-- Fully populated navigation --></nav>
        </header>
        <main>
          <article>
            <h2>Article Title</h2>
            <p>This is the full content of the article...</p>
          </article>
        </main>
        <footer><!-- Fully populated footer --></footer>
        <script src="hydration.js"></script>
      </body>
    </html>
    

    The difference between CSR

    In contrast, a client-side rendered (CSR) initial HTML might be like this:

    <!DOCTYPE html>
    <html>
      <head>
        <title>My CSR Page</title>
      </head>
      <body>
        <div id="root"></div>
        <script src="bundle.js"></script>
      </body>
    </html>
    

    The CSR page relies entirely on JavaScript to populate the content.

    Benefits of fully rendered HTML

    1. Faster Initial Paint: The browser can start rendering content immediately.
    2. Better SEO: Search engines read all your content without executing JavaScript.
    3. Improved Accessibility: Screen readers and other assistive technologies can access content immediately.
    4. Resilience: Basic content is available even if JavaScript fails to load.

    The hydration process

    After sending the fully rendered HTML, SSR applications typically go through a process called hydration:

    1. The server sends the fully rendered HTML.
    2. The browser displays this HTML immediately.
    3. JavaScript loads and hydrates the page, adding interactivity.
    // Simplified React hydration example
    import { hydrateRoot } from 'react-dom/client';
    import App from './App';
    
    const domNode = document.getElementById('root');
    hydrateRoot(domNode, <App />);
    

    This process allows for fast initial loads while still providing the rich interactivity of modern web apps.

    Remember, while SSR provides these fully rendered pages, it's not without trade-offs. The server does more work, and you'll need to handle the state carefully between the server and the client. However, for many applications, the benefits of fully rendered pages make SSR a compelling choice.

    What's the difference between CSR and SSR?

    Client Side Rendering (CSR) and Server Side Rendering (SSR) are two different approaches to rendering web pages. Here's a breakdown of their main differences:

    Client-side rendering (CSR)

    1. The server sends a minimal HTML file with a JavaScript bundle.
    2. The browser downloads and runs the JavaScript.
    3. JavaScript creates the page content and makes it interactive.

    Pros:

    • Smooth interactions after the initial load
    • Fewer server resources are needed

    Cons:

    • Slower initial page load
    • Potential SEO challenges

    Server-side rendering (SSR)

    1. The server creates the full HTML content.
    2. The browser receives and displays the pre-rendered HTML quickly.
    3. JavaScript then loads to make the page fully interactive.

    Pros:

    • Faster initial page load
    • Better for SEO
    • Works well on slower devices

    Cons:

    • It can be more complex to set up
    • May use more server resources

    Here's a simple visual comparison:

    Client-side rendering vs Server-side rendering

    In essence, CSR works more in the browser, while SSR does more on the server. The choice between them depends on your project's specific needs, balancing factors like initial load time, SEO requirements, and server resources.

    SSR and search engines: a match made in HTTP

    Server-side rendering can have a big impact on how search engines see your site. Let's break it down:

    1. Faster Indexing

    Search engine crawlers need immediate access to your page content. Server-side rendering serves pre-rendered HTML right away, unlike client-side rendering where crawlers must wait for JavaScript to execute before seeing the content.

    Search engins for CSR and SSR
    1. Content consistency

    SSR ensures that search engines see the same content that users do. With client-side rendering, there's always a risk that the bot might miss some dynamically loaded content.

    1. Improved Load Times

    Search engines love fast sites. SSR can significantly cut down initial load times, which could give you a slight edge in rankings.

    // Pseudo-code for search engine ranking
    function calculateRanking(site) {
      let score = site.relevance;
      if (site.loadTime < FAST_THRESHOLD) {
        score += SPEED_BONUS;
      }
      return score;
    }
    
    1. Mobile-First Indexing

    With Google's mobile-first indexing, SSR's performance benefits on slower mobile connections become even more important.

    1. Social Media Previews

    While not strictly a search engine feature, SSR makes it easier to generate accurate previews when your content is shared on social platforms. This can indirectly boost your SEO by increasing engagement and backlinks.

    <!-- SSR makes it easier to include accurate meta tags -->
    <meta property="og:title" content="Your Dynamic Title Here">
    <meta property="og:description" content="Your Dynamic Description Here">
    

    SSR is a powerful tool for SEO, but it's not the only factor. Content quality, relevance, and overall user experience are crucial in search engine rankings. SSR simply ensures that search engines can efficiently crawl and index your content, potentially giving you an edge in visibility and performance metrics.

    How to actually do SSR

    Implementing SSR doesn't have to be complicated. Let's cover at how to do it using Next.js, a popular React framework that makes SSR straightforward:

    1. Set up a Next.js project.
    2. Create server-side rendered pages.
    3. Let Next.js handle serving the fully rendered HTML and client-side hydration.

    Here's a simple Next.js example using the App Router:

    // app/page.js
    async function getData() {
      const res = await fetch('<https://api.example.com/data>')
      if (!res.ok) {
        throw new Error('Failed to fetch data')
      }
      return res.json()
    }
    
    export default async function Home() {
      const data = await getData()
    
      return <h1>Hello {data.name}</h1>
    }
    

    In this example:

    • The Home component is an async function, allowing for server-side data fetching.
    • getData() fetches the data we need.
    • The component renders the data directly.

    Next.js automatically handles the SSR process:

    1. When a request comes in, Next.js runs this component on the server.
    2. It waits for the data to be fetched.
    3. It renders the component with the fetched data.
    4. The fully rendered HTML is sent to the client.
    5. Once the JavaScript loads in the browser, the page becomes interactive.

    This approach gives you the benefits of SSR without having to manually set up a server or manage the rendering process yourself.

    Higher-level SSR solutions

    If you don't want to reinvent the wheel, there are several frameworks that handle SSR complexities for you. Here's a rundown of popular options across different ecosystems:

    collage.png

    React

    • Next.js: The most popular React framework with built-in SSR support.
    • Remix: A full stack web framework that leverages React Router.
    • Gatsby: Primarily a static site generator, but also supports SSR.

    Vue

    • Nuxt.js: The go-to framework for Vue applications with SSR capabilities.

    Angular

    • Angular Universal: The official SSR solution for Angular applications.

    Svelte

    • SvelteKit: The official application framework for Svelte with SSR support.

    JavaScript (Framework-agnostic)

    • Astro: Allows you to use multiple frameworks and supports SSR.
    • Qwik: A new framework designed for optimal performance with built-in SSR support.

    PHP

    Ruby

    Python

    • Django: Can implement SSR using libraries like Django-Unicorn or HTMX.
    • Flask: Can be configured for SSR, often used with extensions like Flask-SSE.

    Each of these frameworks offers its own approach to SSR, often with additional features like static site generation, API routes, and more. The choice depends on your preferred language, ecosystem, and specific project requirements.

    Deployment and caching

    When deploying an SSR app:

    1. Build both client-side and server-side bundles.
    2. Run the SSR server as a background process.
    3. Use a process monitor like PM2 or Supervisor to keep your server running.

    Here's a basic deployment flow:

    SSR flowchart.png

    Don't forget about caching! Caching server-rendered pages can significantly reduce server load.

    SSR with Builder.io

    Builder.io provides support for server-side rendering (SSR) and static site generation (SSG) across all components and frameworks. This out-of-the-box functionality allows you to leverage the benefits of SSR and SSG without additional setup.

    Builder.io SSR and SSG

    Key features

    1. Framework Agnostic: Builder.io works with various frameworks that support SSR and SSG.
    2. Automatic Optimization: Builder optimizes your content for performance, including code splitting and lazy loading of off-screen components.
    3. Dynamic Rendering: You can render different content based on user attributes or A/B tests while maintaining SEO benefits.
    4. Easy Integration: Builder provides SDKs and documentation to seamlessly integrate your existing projects.

    Implementation example

    Here's a basic example of how you might fetch and render content server-side with Builder and Next.js:

    import { builder, BuilderComponent } from '@builder.io/react'
    
    builder.init('YOUR_API_KEY')
    
    export async function getStaticProps({ params }) {
      const page = await builder
        .get('page', {
          userAttributes: {
            urlPath: '/' + (params?.page?.join('/') || '')
          }
        })
        .toPromise()
    
      return {
        props: {
          page: page || null,
        },
        revalidate: 5
      }
    }
    
    export default function Page({ page }) {
      return (
        <BuilderComponent
          model="page"
          content={page}
        />
      )
    }
    

    Best practices

    1. Ensure you're using a framework that supports SSR or SSG.
    2. Follow your framework's guidelines for fetching data server-side when integrating Builder Pages or Sections.
    3. Refer to the getAsyncProps README for more information on handling server-side data.

    By leveraging Builder for SSR, you can combine the flexibility of a headless CMS with the performance benefits of server-side rendering, all while maintaining an easy-to-use visual editing experience.

    Wrapping up

    Server-side rendering (SSR) is a powerful approach in web development that can significantly enhance your application's performance, SEO, and user experience. Throughout this article, we've explored what SSR is, how it differs from client-side rendering, its impact on search engines, and practical implementation strategies using popular frameworks like Next.js.

    We've also discussed the concept of fully rendered pages and examined various SSR solutions across different ecosystems. While SSR offers many benefits, it's important to consider your project's specific needs when deciding whether to implement it.

    FAQ

    Q: How does SSR affect my development workflow?

    A: SSR can make development more complex, as you need to consider both server and client environments. You might need to adjust your build process and be cautious with browser-specific APIs.

    Q: How does SSR impact my site's Time to Interactive (TTI)?

    A: While SSR can improve initial content visibility, it might slightly delay TTI as the browser needs to load and hydrate the JavaScript after receiving the initial HTML.

    Q: Are there any security considerations specific to SSR?

    A: Yes, with SSR, you need to be more careful about exposing sensitive data or APIs on the server side. Always sanitize user inputs and be cautious about what data you include in the initial render.

    Q: How does SSR work with authentication and personalized content?

    A: SSR can work with authentication, but it requires careful handling. You might need to implement techniques like JWT tokens or server-side sessions to manage authenticated SSR requests.

    Newsletter

    Get the latest from Builder.io

    By submitting, you agree to our Privacy Policy

    Product

    Platform Overview

    Integrations

    What's New

    Open Source

    Builder

    Builder

    Mitosis

    Mitosis

    Mitosis

    AI Shell

    Mitosis

    Micro Agent

    Mitosis

    GPT Crawler

    Qwik

    Qwik

    Partytown

    Partytown

    Popular Guides

    From Design to Code Guide

    Composable Commerce Guide

    Headless CMS Guide

    Headless Commerce Guide

    Composable DXP Guide

    Design to Code

    Resources

    Blog

    Knowledge Base

    Community Forum

    Partners

    Templates

    Success Stories

    Showcase

    Resource Center

    Frameworks

    React

    React

    Next

    Next.js

    Qwik

    Qwik

    Gatsby

    Gatsby

    Angular

    Angular

    Vue

    Vue

    Svelte

    Svelte

    Remix logo

    Remix

    Nuxt

    Nuxt

    Hydrogen

    Hydrogen

    See All

    © 2024 Builder.io, Inc.

    Security

    Privacy Policy

    SaaS Terms

    Security & Compliance

    Cookie Preferences

    Gartner Cool Vendor 2024