Post

Choosing the Right Stack: MERN vs Next.js vs Rails + Next.js

As a developer embarking on a new web project, one of the most crucial decisions you’ll make is choosing the right technology stack. This decision will influence your development speed, application performance, maintainability, and even your team’s happiness. If you’re considering a JavaScript-based frontend, you have several excellent options for structuring your application.

In this post, I’ll explore three popular approaches for building modern web applications:

  1. The MERN Stack (MongoDB, Express.js, React, Node.js)
  2. Next.js as a full-stack solution
  3. Ruby on Rails backend with Next.js frontend

I’ll compare these approaches across several dimensions including development speed, performance, scalability, and developer experience to help you make an informed decision for your project.

Option 1: The MERN Stack

The MERN stack is a popular JavaScript-based tech stack that uses MongoDB as the database, Express.js as the server framework, React for the frontend, and Node.js as the runtime environment.

How It Works

In a typical MERN stack architecture:

  • MongoDB stores your data as JSON-like documents with flexible schemas
  • Express.js provides a framework for building your API endpoints
  • React handles the user interface and client-side logic
  • Node.js executes your server-side JavaScript code

For example, your Express server might define routes like:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
// routes/api/users.js
router.post('/', async (req, res) => {
  try {
    const { name, email, password } = req.body;
    
    // Check if user exists
    let user = await User.findOne({ email });
    if (user) {
      return res.status(400).json({ errors: [{ msg: 'User already exists' }] });
    }
    
    // Create new user
    user = new User({
      name,
      email,
      password
    });
    
    // Hash password
    const salt = await bcrypt.genSalt(10);
    user.password = await bcrypt.hash(password, salt);
    await user.save();
    
    // Generate JWT
    const payload = { user: { id: user.id } };
    jwt.sign(payload, config.get('jwtSecret'), { expiresIn: 360000 }, (err, token) => {
      if (err) throw err;
      res.json({ token });
    });
  } catch (err) {
    console.error(err.message);
    res.status(500).send('Server Error');
  }
});

Advantages

  1. JavaScript Everywhere: Using JavaScript for both frontend and backend means you don’t need to context-switch between languages.
  2. Flexible Data Structure: MongoDB’s schema-less nature makes it easy to evolve your data structure over time.
  3. Large Ecosystem: Each component of the MERN stack has extensive libraries and tools.
  4. RESTful API Architecture: Clear separation between frontend and backend forces good API design practices.
  5. Real-time Applications: Node.js excels at handling real-time, data-intensive applications.

Challenges

  1. Manual Setup for SEO: You’ll need to implement additional solutions for SEO, as React’s client-side rendering isn’t optimal for search engines by default.
  2. More Configuration: You’ll need to set up routing, state management, and server-side rendering yourself.
  3. DevOps Complexity: You’ll need to deploy and manage both a Node.js server and a React application.
  4. Learning Curve for NoSQL: If you’re coming from a relational database background, MongoDB might require some adjustment.

Option 2: Next.js as a Full-Stack Solution

Next.js is a React framework that provides structure, features, and optimizations for your React application. Recent versions of Next.js have evolved to provide full-stack capabilities.

How It Works

Next.js simplifies the development process by providing:

  • Server-Side Rendering (SSR): Renders pages on the server for better SEO and initial load times
  • Static Site Generation (SSG): Pre-renders pages at build time for optimal performance
  • API Routes: Create API endpoints directly within your Next.js application
  • File-based Routing: Define routes based on your file structure

For example, you might structure your API endpoints like this:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
// pages/api/users.js
import bcrypt from 'bcryptjs';
import jwt from 'jsonwebtoken';
import User from '../../models/User';

export default async function handler(req, res) {
  if (req.method === 'POST') {
    const { name, email, password } = req.body;

    try {
      // Check if user exists
      let user = await User.findOne({ email });
      if (user) {
        return res.status(400).json({ errors: [{ msg: 'User already exists' }] });
      }

      // Create new user
      user = new User({
        name,
        email,
        password
      });

      // Hash password
      const salt = await bcrypt.genSalt(10);
      user.password = await bcrypt.hash(password, salt);
      await user.save();

      // Generate JWT
      const payload = { user: { id: user.id } };
      const token = jwt.sign(payload, process.env.JWT_SECRET, { expiresIn: '100h' });

      res.status(200).json({ token });
    } catch (err) {
      console.error(err.message);
      res.status(500).json({ errors: [{ msg: 'Server error' }] });
    }
  } else {
    res.status(405).end(); // Method Not Allowed
  }
}

Advantages

  1. Built-in SSR and SSG: Excellent for SEO and performance
  2. Simplified Development: File-based routing and API routes reduce boilerplate
  3. Unified Deployment: Deploy both frontend and backend as a single application
  4. Image Optimization: Built-in image optimization for better performance
  5. Incremental Static Regeneration (ISR): Update static content without rebuilding the entire site

Challenges

  1. Learning Curve: Next.js has its own patterns and conventions to learn
  2. Less Flexibility: The unified approach can be constraining for complex backend logic
  3. Database Integration: You still need to set up and configure your database connection
  4. Complex State Management: For large applications, you may need additional state management solutions

Option 3: Ruby on Rails Backend with Next.js Frontend

This approach combines the mature backend capabilities of Ruby on Rails with the modern frontend features of Next.js.

How It Works

In this architecture:

  • Ruby on Rails serves as an API-only backend, handling database operations, complex business logic, and authentication
  • Next.js handles the frontend, consuming the Rails API

Your Rails controller might look like:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
# app/controllers/api/users_controller.rb
class Api::UsersController < ApplicationController
  def create
    user = User.new(user_params)
    if user.save
      token = JWT.encode({ user_id: user.id }, ENV['JWT_SECRET'], 'HS256')
      render json: { token: token }, status: :created
    else
      render json: { errors: user.errors.full_messages }, status: :unprocessable_entity
    end
  end
  
  private
  
  def user_params
    params.require(:user).permit(:name, :email, :password, :password_confirmation)
  end
end

And your Next.js page might fetch data like:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
// pages/dashboard.js
export async function getServerSideProps(context) {
  const token = context.req.cookies.token;
  
  try {
    const res = await fetch(`${process.env.RAILS_API_URL}/api/dashboard`, {
      headers: {
        'Authorization': `Bearer ${token}`
      }
    });
    
    if (res.ok) {
      const data = await res.json();
      return { props: { data } };
    } else {
      return {
        redirect: {
          destination: '/login',
          permanent: false,
        },
      };
    }
  } catch (error) {
    return {
      redirect: {
        destination: '/login',
        permanent: false,
      },
    };
  }
}

Advantages

  1. Best of Both Worlds: Leverage Rails’ mature backend capabilities with Next.js’ frontend optimizations
  2. Rails Ecosystem: Access to Rails’ robust gems for things like authentication, authorization, and admin panels
  3. Strong Conventions: Rails’ “convention over configuration” philosophy can speed up backend development
  4. Database Migrations: Rails’ migration system makes database changes safe and manageable
  5. Active Record: Rails’ ORM simplifies database interactions

Challenges

  1. Multiple Languages: Working with both Ruby and JavaScript requires context-switching
  2. Deployment Complexity: You’ll need to deploy and manage two separate applications
  3. Authentication Coordination: Ensuring secure authentication between the two systems requires careful planning
  4. API Design: You’ll need to thoughtfully design the API contract between your frontend and backend
  5. Team Expertise: Requires team members familiar with both ecosystems

Which Stack Should You Choose?

The best choice depends on your specific needs and constraints:

Choose MERN if:

  • You want to work exclusively with JavaScript
  • Your team has strong Node.js and MongoDB experience
  • You value flexibility in your database schema
  • You’re building data-intensive applications with real-time features
  • You need fine-grained control over your backend architecture

Choose Next.js if:

  • SEO is a critical concern for your application
  • You want faster development with less configuration
  • You prefer a unified deployment model
  • You’re comfortable with its conventions and constraints
  • Your backend logic is relatively straightforward

Choose Rails + Next.js if:

  • You have existing Rails expertise on your team
  • Your application requires complex backend business logic
  • You value Rails’ mature ecosystem for things like admin interfaces
  • Data integrity and relational data are important for your application
  • You want to leverage Rails’ battle-tested security features

Conclusion

There’s no universally “right” choice among these three options - each has its strengths and weaknesses. Consider your team’s skills, your project’s specific requirements, and your long-term maintenance plans when making your decision.

The JavaScript ecosystem continues to evolve rapidly, and each of these approaches represents a valid way to build modern web applications. By understanding the trade-offs involved, you can make a more informed decision that aligns with your project goals and team capabilities.

Have you built applications using one of these stacks? What were your experiences? I’d love to hear your thoughts in the comments!


This blog post was created to help developers understand the trade-offs between different tech stacks for building web applications. The code examples are simplified for illustration purposes.

This post is licensed under CC BY 4.0 by the author.