Logo

dev-resources.site

for different kinds of informations.

Setting Up Dual Compilation (SSR + CSR) in ViteJS with vite-plugin-builder

Published at
1/8/2025
Categories
vite
express
ssr
csr
Author
yracnet
Categories
4 categories in total
vite
open
express
open
ssr
open
csr
open
Author
7 person written this
yracnet
open
Setting Up Dual Compilation (SSR + CSR) in ViteJS with vite-plugin-builder

Using vite-plugin-builder to Set Up an SSR and CSR Project in Vite

In this tutorial, I will show you how to create a project from scratch with Vite, configure it to build both Server-Side Rendering (SSR) and Client-Side Rendering (CSR) using the vite-plugin-builder plugin, which simplifies the whole process with a single command.

What is vite-plugin-builder?

vite-plugin-builder is a plugin for Vite that simplifies the setup and compilation of applications requiring both SSR and CSR in a single project (dual build). With it, you can easily manage both server and client entry points, which is ideal for modern applications that need to process content both on the server and on the client.


Steps to Create the Project and Set Up the Plugin

Below, I'll walk you through setting up Vite and vite-plugin-builder from scratch.

1. Create the Project with Vite

First, create a new Vite project using the following command. We will use React as an example, but you can choose any other framework if you prefer.

# Follow the steps to create a React project with TS or JS.
yarn create vite my-app
cd my-app
Enter fullscreen mode Exit fullscreen mode

2. Install the Required Dependencies

Now, install vite-plugin-builder as a development dependency:

yarn add -D vite-plugin-builder
Enter fullscreen mode Exit fullscreen mode

3. Configure the vite.config.ts File

Open the vite.config.ts file and configure the vite-plugin-builder plugin to handle both the server and client. Here’s a basic example:

import { defineConfig } from "vite";
import react from "@vitejs/plugin-react-swc";
import builder from "vite-plugin-builder";

// https://vite.dev/config/
export default defineConfig({
  base: "/",
  plugins: [
    react(),
    builder({
      // Server entry point
      serverEntry: "server/main.js",
      // Client entry point (optional as it inherits Vite's config)
      clientEntry: {
        main: "index.html",
      },
    }),
  ],
});
Enter fullscreen mode Exit fullscreen mode

4. Create the server/main.js File

The next step is to create the server/main.js file, which will serve as the entry point for your server (SSR). This file will represent your backend entry. Here’s a basic example of what it could contain:

console.log("Server is running...");
// You can create an Express server or configure another server depending on your app stack
// Server-side logic for SSR
Enter fullscreen mode Exit fullscreen mode

You can check out a full example at https://github.com/yracnet/vite-plugin-builder/tree/main/packages/examples

5. Build the Project for Production

Finally, to compile the project and prepare it for production, use:

yarn build
Enter fullscreen mode Exit fullscreen mode

This will generate two directories: one for the client and one for the server.

The expected structure will look like this:

# Server Output
/dist/app.js
/dist/bin/[name]-[hash].[ext]
/dist/assets/[name]-[hash].[ext]

# Client Output
/dist/public/index.html
/dist/public/assets/[name]-[hash].[ext]
/dist/public/chunks/[name]-[hash].[ext]
Enter fullscreen mode Exit fullscreen mode

6. Customizing the Project

According to the plugin documentation, you can customize various aspects, such as the output directories for the server and client, which can be different based on your needs.

Here’s an example of how to customize it:

import { defineConfig } from "vite";
import path from "path";
import react from "@vitejs/plugin-react-swc";
import builder from "vite-plugin-builder";

// https://vite.dev/config/
export default defineConfig({
  base: "/",
  plugins: [
    react(),
    builder({
      // Path to the server entry file
      serverEntry: "server/main.js",
      // (optional) Configuration for the server build
      serverConfig: {
        outDir: "app-sanbox/server", // Server output directory
        privateDir: "private-files", // Directory with private files to copy
        // Define variables for server-side compilation
        define: {
          "BUILD.BASE": '"/"',
          "BUILD.BASE_API": '"/api"',
          // Change to output client directory
          "BUILD.STATIC_DIR": '"../client"',
          "BUILD.SERVER_IP": '"0.0.0.0"',
          "BUILD.SERVER_PORT": "3001",
        },
      },
      // Client configuration
      clientConfig: {
        outDir: "app-sanbox/client",
      },
    }),
  ],
});
Enter fullscreen mode Exit fullscreen mode

This way, the server will use these variables during the final compilation:

import express from "express";
import { api } from "./api";
import { site } from "./site";

// ENV
const { SERVER_IP = BUILD.SERVER_IP, SERVER_PORT = BUILD.SERVER_PORT } =
  process.env;

// APP
const app = express();

// RUN
app.use(BUILD.BASE_API, api);
app.use(BUILD.BASE, site);
app.listen(SERVER_PORT, SERVER_IP, () =>
  console.log(`Server running on port ${SERVER_IP}:${SERVER_PORT}`)
);
Enter fullscreen mode Exit fullscreen mode

You can also configure the copying of static files inside the server directory using the serverConfig.privateDir variable, which will copy files from the private-files folder to /app-sanbox/server.

The private-files directory might include files such as:

# Private directory
/private-files/package.json
/private-files/.env
Enter fullscreen mode Exit fullscreen mode

The expected output for this configuration will look like this:

# Server Output
/app-sanbox/server/app.js
/app-sanbox/server/bin/[name]-[hash].[ext]
/app-sanbox/server/assets/[name]-[hash].[ext]
/app-sanbox/server/package.json
/app-sanbox/server/.env

# Client Output
/app-sanbox/client/index.html
/app-sanbox/client/assets/[name]-[hash].[ext]
/app-sanbox/client/chunks/[name]-[hash].[ext]
Enter fullscreen mode Exit fullscreen mode

Conclusion

With vite-plugin-builder, you can easily set up a project that supports both SSR and CSR in Vite. The plugin simplifies the configuration and build process, allowing you to focus on developing your application without additional complications.

For more details or more complex examples, you can check out the official documentation on GitHub.

That’s it! Now you have a hybrid project with both SSR and CSR running in Vite.

ssr Article's
30 articles in total
Favicon
Custom builder for Angular: My way
Favicon
Setting Up Dual Compilation (SSR + CSR) in ViteJS with vite-plugin-builder
Favicon
# Key New Features in React Router 7: Embracing the Remix Future
Favicon
Beginner SEO in React JS - React Helmet
Favicon
Setting up partial SSR for a React + TypeScript + webpack app from scratch
Favicon
Create an SSR Application with Vite, React, React Query and React Router
Favicon
Understanding Web Rendering: Performance Implications and Use Cases
Favicon
Make EditorJS work in Svelte(kit) SSR
Favicon
Client-side Rendering & Server-side Rendering
Favicon
A Practical Guide to CSR and SSR with React 19 and esbuild
Favicon
Fixing SSR Rendering Issues with Angular Resolver for Async Pipe Data
Favicon
Choosing Remix as a Server-Side Rendering (SSR) Framework
Favicon
Implementing Server-Side Rendering (SSR) with Next.js and Firebase for SEO-Friendly React Apps 🚀
Favicon
Do You Need to SSR Your Web Components?
Favicon
Web Components and SSR - 2024 Edition
Favicon
Dark side of Next.js - App Router
Favicon
How to achieve unified management of four types of global state data in Vue3?
Favicon
What do SSR, CSR, ISR and SSG mean? A complete guide for web developers
Favicon
Vue 3.5 “Tengen Toppa Gurren Lagann” Innovations: Advanced Features and Most Powerful Updates 🚀
Favicon
Inertiajs Server-side Rendering (SSR) For React (Vite Setup)
Favicon
Vaadin, the battery-included server-side AJAX framework
Favicon
How to add Supabase Auth to Astro
Favicon
Dive into Next.js Server-Side Rendering (SSR): From SPA to ISR
Favicon
Why do client components render as SSR in nextjs, marking components as "use client" still render its html as SSR why?
Favicon
Augmenting the client with Alpine.js
Favicon
Augmenting the client with Vue.js
Favicon
Server-Side Rendering (SSR): Uma Solução para SEO e Performance em Aplicações React
Favicon
SSR and CSR Explained
Favicon
A short history of AJAX and SSR
Favicon
How to Do Server-Side Rendering (SSR) in Next.js

Featured ones: