21.Nest.js와 Next.js에서 Middleware와 Guard 사용하기, 각 페이지별 접근 권한 설정

본 블로그 포스트에서는 Nest.js와 Next.js에서 Middleware와 Guard의 개념과 그 사용 방법에 대해 자세히 설명하겠습니다. 특히, 각 페이지별 접근 권한을 설정하는 방법에 대해서도 알아보겠습니다. 이 포스트를 통해 여러분은 Nest.js와 Next.js 환경에서 권한을 관리하고, 사용자 인증 및 인가를 효과적으로 구현할 수 있는 방법을 익히게 될 것입니다.

1. Nest.js에서 Middleware 이해하기

Nest.js는 Express.js 기반의 훌륭한 Node.js 프레임워크로, Middleware는 요청-응답 사이클에서 요청을 처리하는 중간 단계의 로직을 구성하는 데 사용됩니다. Middleware는 요청을 가로채어 처리하거나, 요청 객체를 변형하거나, 다음 Middleware로 요청을 전달할 수 있는 기능을 제공합니다.

1.1 Middleware의 용도

  • 로그인 여부 확인
  • 요청 기록 기록
  • 헤더 수정
  • 오류 처리

1.2 Nest.js에서 Middleware 구현하기

Nest.js에서는 Middleware를 클래스로 구현하고, 이를 모듈에서 등록하여 사용할 수 있습니다. 아래는 간단한 로그인 확인 Middleware의 예시입니다.


import { Injectable, NestMiddleware } from '@nestjs/common';
import { Request, Response, NextFunction } from 'express';

@Injectable()
export class AuthMiddleware implements NestMiddleware {
    use(req: Request, res: Response, next: NextFunction) {
        const token = req.headers.authorization;
        if (token) {
            // 인증 로직 (예: 디코딩 JWT)
            next();
        } else {
            res.status(403).send('Unauthorized');
        }
    }
}

위의 코드에서 `AuthMiddleware`는 요청에 포함된 토큰을 확인하고, 인증이 이루어지지 않은 경우 403 상태 코드를 반환합니다.

1.3 Middleware 등록하기

Middleware는 모듈에서 등록할 수 있습니다. 아래는 전체 모듈에 Middleware를 등록하는 방법입니다.


import { Module, MiddlewareConsumer, NestModule } from '@nestjs/common';
import { AuthMiddleware } from './auth.middleware';

@Module({
    // ...
})
export class AppModule implements NestModule {
    configure(consumer: MiddlewareConsumer) {
        consumer
            .apply(AuthMiddleware)
            .forRoutes('*'); // 모든 경로에 Middleware 적용
    }
}

2. Nest.js에서 Guard 사용하기

Guard는 요청 처리 전에 특정 조건(예: 권한)을 확인하는 데 사용됩니다. Guards는 다음 두 가지 주요 목적으로 사용됩니다.

  • 특정 경로 또는 핸들러에 대한 접근 권한 확인
  • 어떠한 요청을 처리할 수 있는지에 대한 조건을 설정

2.1 Guard의 정의 및 사용 방법

Nest.js의 Guards는 주로 `CanActivate` 인터페이스를 구현하여 생성합니다. 아래는 사용자의 역할 기반으로 접근을 제어하는 Guard의 예시입니다.


import { Injectable, CanActivate, ExecutionContext } from '@nestjs/common';

@Injectable()
export class RolesGuard implements CanActivate {
    canActivate(context: ExecutionContext): boolean {
        const request = context.switchToHttp().getRequest();
        const user = request.user; // 이전에 middleware에서 사용자 정보 설정
        const hasRole = () => user.roles.indexOf('admin') > -1;
        return user && user.roles && hasRole();
    }
}

2.2 Guard 등록 및 사용하기

Guard는 컨트롤러의 핸들러 또는 모듈에 더할 수 있습니다. 아래의 코드는 특정 컨트롤러에 Guard를 등록하는 예시입니다.


import { Controller, Get, UseGuards } from '@nestjs/common';
import { RolesGuard } from './roles.guard';

@Controller('admin')
export class AdminController {
    @Get()
    @UseGuards(RolesGuard)
    getAdminData() {
        return 'Admin Data';
    }
}

3. Next.js에서 Middleware 이해하기

Next.js에서 Middleware는 요청이 들어오는 단계에서 선처리하거나 특정 조건에 따라 사용자에게 응답할 수 있는 방법을 제공합니다. Next.js는 API Routes와 페이지에 Middleware를 적용할 수 있습니다.

3.1 Next.js Middleware의 용도

  • 사용자 인증
  • 404 처리
  • API 호출 간소화

3.2 Next.js에서 Middleware 활성화하기

Next.js 12 버전부터 제공되는 Middleware를 사용하여 각 라우트에 접근하기 전에 사용자 인증을 확인하는 예시입니다. Middleware는 `middleware.ts` 파일에서 정의합니다.


// middleware.ts
import { NextResponse } from 'next/server';

export function middleware(request) {
    const token = request.cookies.get('token');
    if (!token) {
        return NextResponse.redirect(new URL('/login', request.url));
    }
    return NextResponse.next();
}

4. Next.js에서 Guard 사용하기

Next.js에서는 Guard를 명시적으로 구현하는 방법은 없지만, 각 페이지에서 직접 사용자 인증을 처리할 수 있습니다. 예를 들어, 서버 사이드 렌더링(SSR)에서 페이지 접근 제어를 설정할 수 있습니다.

4.1 서버 사이드에서 접근 권한 검증하기

아래는 Next.js의 `getServerSideProps`를 사용하여 사용자의 인증 상태를 확인하는 예시입니다.


export async function getServerSideProps(context) {
    const { req } = context;
    const token = req.cookies.token;
    
    if (!token) {
        return {
            redirect: {
                destination: '/login',
                permanent: false,
            },
        };
    }

    // 사용자 데이터 fetch 등의 로직
    return {
        props: {}, // 성공적으로 인증된 경우 페이지에 전달할 데이터
    };
}

5. 결론

Nest.js와 Next.js에서 Middleware와 Guard를 활용함으로써 애플리케이션의 보안 및 유연성을 높일 수 있습니다. 각각의 기술 스택에서의 역할과 사용법을 깊이 있게 이해하고 있으면, 복잡한 인증 로직과 접근 권한 관리를 보다 효과적으로 구현할 수 있습니다. 이 포스트에서 다룬 내용을 바탕으로 여러분의 프로젝트에도 적절한 Middleware와 Guard를 적용하여 사용자 경험을 향상시키시길 바랍니다.

41.Nest.js와 Next.js에서 웹소켓 활용하기, Next.js 클라이언트에서 웹소켓 연결 및 데이터 수신

Lorem ipsum dolor sit amet, consectetur adipiscing elit.

웹소켓이란?

웹소켓(WebSocket)은 클라이언트와 서버 간의 전이중(양방향) 통신을 가능하게 하는 프로토콜입니다. 기존의 HTTP 프로토콜은 클라이언트가 요청을 하고 서버가 응답을 하는 구조인 반면, 웹소켓은 클라이언트와 서버가 서로 자유롭게 메시지를 주고받을 수 있는 연결을 유지합니다. 이러한 특성 덕분에 실시간 데이터 전송이 필요한 어플리케이션에서 웹소켓은 매우 유용합니다. 예를 들어, 채팅 애플리케이션, 실시간 게임, 주식 시세 전송 앱 등에서 널리 사용됩니다.

Nest.js 소개

Nest.js는 Node.js를 기반으로 하는 현대적인 웹 애플리케이션 프레임워크입니다. Angular에서 영감을 받아 모듈 기반 아키텍처를 채택하고 있으며, TypeScript를 기본으로 지원합니다. Nest.js를 사용하면 구조화된 코드를 작성할 수 있어 유지보수와 확장성이 뛰어납니다. Nest.js는 웹소켓을 포함한 다양한 통신 프로토콜을 지원하여 실시간 애플리케이션 개발에 적합합니다.

Next.js 소개

Next.js는 React를 기반으로 한 프레임워크로, 서버 사이드 렌더링(SSR)과 정적 사이트 생성(SSG) 기능을 제공합니다. 이를 통해 SEO에 유리하고 빠른 렌더링을 할 수 있는 장점이 있습니다. 또한, Next.js는 API 라우트 기능을 사용해 서버 측 로직도 간단히 작성할 수 있습니다. 이러한 특성 덕분에 Next.js는 실시간 애플리케이션을 개발하는 데 매우 유용합니다.

Nest.js에서 웹소켓 설정하기

Nest.js에서 웹소켓을 사용하는 방법은 다음과 같습니다. 먼저, 웹소켓 서버를 구현하기 위해 @nestjs/websockets 패키지를 설치합니다. 이를 통해 웹소켓의 기본적인 기능을 쉽게 사용할 수 있습니다.

설치

npm install @nestjs/websockets @nestjs/platform-socket.io socket.io

웹소켓 게이트웨이 만들기

다음으로, 웹소켓 게이트웨이를 생성합니다. 웹소켓 게이트웨이는 클라이언트의 요청을 받고 메시지를 송수신하는 역할을 합니다. 아래는 간단한 웹소켓 게이트웨이의 예입니다.


import { WebSocketGateway, WebSocketServer, OnGatewayConnection, OnGatewayDisconnect } from '@nestjs/websockets';
import { Server } from 'socket.io';

@WebSocketGateway()
export class AppGateway implements OnGatewayConnection, OnGatewayDisconnect {
    @WebSocketServer() server: Server;

    handleConnection(client: any) {
        console.log('클라이언트 연결됨');
    }

    handleDisconnect(client: any) {
        console.log('클라이언트 연결 끊김');
    }

    sendMessage(message: string) {
        this.server.emit('message', message);
    }
}

위 코드에서 handleConnectionhandleDisconnect 메서드는 각각 클라이언트가 연결되거나 연결이 끊어질 때 호출됩니다. sendMessage 메서드는 모든 클라이언트에게 메시지를 전송하는 기능을 수행합니다.

Next.js 클라이언트에서 웹소켓 연결하기

Next.js 클라이언트에서 웹소켓에 연결하기 위해 socket.io-client 라이브러리를 사용할 수 있습니다. 먼저 이 라이브러리를 설치해야 합니다.

설치

npm install socket.io-client

클라이언트 설정

먼저 Next.js 프로젝트의 페이지 컴포넌트에서 웹소켓 연결을 설정합니다. 아래는 기본적인 웹소켓 클라이언트 구현 예입니다.


import { useEffect } from 'react';
import { io } from 'socket.io-client';

const socket = io('http://localhost:3000');

const Home = () => {
    useEffect(() => {
        socket.on('message', (message) => {
            console.log('서버로부터 받은 메시지:', message);
        });

        return () => {
            socket.off('message');
        };
    }, []);

    const sendMessage = () => {
        socket.emit('message', '안녕하세요, 서버!');
    };

    return (
        

웹소켓 테스트

); }; export default Home;

위 코드에서는 socket.io-client를 사용하여 서버와의 웹소켓 연결을 설정하고 있습니다. 클라이언트가 연결되면 서버로부터 수신한 메시지를 콘솔에 출력합니다. 또한, 버튼을 클릭하면 서버에 메시지를 보낼 수 있습니다.

서버와 클라이언트 간의 데이터 송수신

이렇게 설정한 웹소켓 서버와 클라이언트 간에 실시간 데이터 송수신을 구현할 수 있습니다. 예를 들어, 클라이언트에서 메시지를 보내면 서버는 그 메시지를 처리하고 다시 클라이언트로 전달할 수 있습니다. 이 기능을 통해 다양한 실시간 애플리케이션을 개발할 수 있습니다.

서버에서 클라이언트로 데이터 전송

서버에서 클라이언트로 데이터 전송하는 예는 다음과 같습니다. 간단한 메시지를 전송하는 기능을 추가하여 클라이언트가 연결된 모든 클라이언트에게 매 5초마다 메시지를 전송해보겠습니다.


import { WebSocketGateway, WebSocketServer, OnGatewayConnection, OnGatewayDisconnect } from '@nestjs/websockets';
import { Server } from 'socket.io';

@WebSocketGateway()
export class AppGateway implements OnGatewayConnection, OnGatewayDisconnect {
    @WebSocketServer() server: Server;

    handleConnection(client: any) {
        console.log('클라이언트 연결됨');
    }

    handleDisconnect(client: any) {
        console.log('클라이언트 연결 끊김');
    }

    broadcastMessage() {
        setInterval(() => {
            this.server.emit('message', '서버에서 전송한 메시지입니다.');
        }, 5000);
    }
}

위 코드의 broadcastMessage 메서드는 매 5초마다 모든 연결된 클라이언트에게 메시지를 전송합니다. 이 메서드를 constructor 내에서 호출하여 서버가 실행되면 자동으로 클라이언트에 메시지를 보내도록 설정할 수 있습니다.

에러 처리와 최적화

실시간 웹소켓 애플리케이션에서 에러 처리는 중요한 부분입니다. 발생할 수 있는 다양한 에러를 예방하고 사용자가 이해할 수 있도록 적절한 피드백을 제공해야 합니다. 클라이언트에서는 웹소켓 연결의 상태를 관리하고 에러를 처리할 수 있는 방법을 구현하는 것이 좋습니다.

클라이언트 에러 처리


useEffect(() => {
    socket.on('connect_error', (err) => {
        console.error('서버에 연결할 수 없습니다:', err.message);
    });

    socket.on('disconnect', (reason) => {
        if (reason === 'io server disconnect') {
            console.log('서버가 클라이언트를 연결 해제했습니다.');
            // 자동 재연결 시도
            socket.connect();
        }
    });

    return () => {
        socket.off('connect_error');
        socket.off('disconnect');
    };
}, []);

위 코드에서는 연결 오류와 연결 해제 시의 이벤트를 처리하고 있습니다. 이러한 처리를 통해 사용자가 문제를 인지하고 적절한 조치를 취할 수 있게 됩니다.

결론

Nest.js와 Next.js를 활용한 웹소켓 애플리케이션 개발은 실시간 기능 구현을 위해 매우 효과적입니다. 간단한 설정만으로 양방향 통신을 할 수 있으며, 이를 통해 다양한 기능을 추가할 수 있습니다. 이번 글을 통해 기본적인 웹소켓 사용법과 클라이언트-서버 간의 데이터 송수신 방법을 살펴보았습니다. 다양한 웹소켓 기능을 실험하여 더 나아가고, 실시간 애플리케이션을 더욱 풍성하게 만드는 데 도움이 되길 바랍니다.

27.블로그 포스트 작성 및 에디터 기능 추가, 서버에서 마크다운 파일을 HTML로 변환하여 렌더링

현대 웹 애플리케이션 개발에서 블로그 시스템은 많은 인기를 끌고 있습니다. 특히나 개인이 자신을 표현할 수 있는 공간으로 블로그는 매우 중요한 역할을 합니다. 이번 포스트에서는 Nest.js와 Next.js를 활용하여 블로그 포스트 작성 및 에디터 기능을 추가하는 방법에 대해 논의하고자 합니다.

1. 프로젝트 구성

Nest.js는 서버 사이드 프레임워크로, TypeScript로 작성되어 있으며, Next.js는 리액트 기반의 서버 사이드 렌더링(SSR) 프레임워크입니다. 이 두 프레임워크를 조합하여 효율적인 블로그 애플리케이션을 구현할 수 있습니다. 먼저, Nest.js를 사용하여 API 서버를 구축하고, Next.js로 클라이언트 애플리케이션을 만듭니다.

2. Nest.js API 서버 구축

Nest.js에서 API를 구축하는 기본적인 과정은 다음과 같습니다. 먼저, Nest.js CLI를 사용하여 프로젝트를 생성합니다.

npm i -g @nestjs/cli
nest new blog-server

생성된 프로젝트 폴더로 이동한 후, 필요한 패키지를 설치합니다. Markdown 파일을 처리하기 위해 `marked` 패키지를 사용할 것입니다.

npm install marked

다음으로, 포스트를 관리할 수 있는 기본적인 CRUD API를 만듭니다. `posts`라는 모듈을 생성합니다.

nest generate module posts
nest generate service posts
nest generate controller posts

2.1. 포스트 DTO(Data Transfer Object) 정의

포스트 데이터를 다루기 위해 DTO를 만듭니다. `create-post.dto.ts` 파일을 생성하고 아래와 같이 작성합니다.

import { IsString, IsNotEmpty } from 'class-validator';

        export class CreatePostDto {
            @IsString()
            @IsNotEmpty()
            title: string;

            @IsString()
            @IsNotEmpty()
            content: string;
        }

2.2. 서비스 구현

`posts.service.ts` 파일을 열고 포스트 데이터를 관리하는 메서드를 추가합니다. 여기서는 간단히 메모리에 데이터를 저장합니다.

import { Injectable } from '@nestjs/common';
        import { CreatePostDto } from './dto/create-post.dto';

        interface Post {
            id: number;
            title: string;
            content: string;
        }

        @Injectable()
        export class PostsService {
            private posts: Post[] = [];
            private idCounter = 1;

            create(postDto: CreatePostDto): Post {
                const newPost = {
                    id: this.idCounter++,
                    ...postDto,
                };
                this.posts.push(newPost);
                return newPost;
            }

            findAll(): Post[] {
                return this.posts;
            }

            findOne(id: number): Post {
                return this.posts.find(post => post.id === id);
            }
        }

2.3. 컨트롤러 구현

`posts.controller.ts` 파일을 열고 API 엔드포인트를 추가합니다. 포스트를 추가하고, 목록을 가져오는 기능을 구현합니다.

import { Controller, Post, Body, Get } from '@nestjs/common';
        import { PostsService } from './posts.service';
        import { CreatePostDto } from './dto/create-post.dto';

        @Controller('posts')
        export class PostsController {
            constructor(private readonly postsService: PostsService) {}

            @Post()
            create(@Body() postDto: CreatePostDto) {
                return this.postsService.create(postDto);
            }

            @Get()
            findAll() {
                return this.postsService.findAll();
            }
        }

3. 클라이언트 애플리케이션 Next.js 구축

이제 Next.js를 사용하여 블로그 클라이언트를 구축하겠습니다. 새로운 Next.js 프로젝트를 생성합니다.

npx create-next-app blog-client

클라이언트 애플리케이션에서도 Markdown 파일을 사용하기 위해 `remark` 패키지를 설치합니다.

npm install remark remark-html

3.1. 포스트 작성 페이지

포스트를 작성할 수 있는 페이지를 만들어야 합니다. `pages/create-post.js` 파일을 생성하고 아래와 같이 작성합니다.

import { useState } from 'react';

        export default function CreatePost() {
            const [title, setTitle] = useState('');
            const [content, setContent] = useState('');

            const handleSubmit = async (e) => {
                e.preventDefault();
                const response = await fetch('http://localhost:3000/posts', {
                    method: 'POST',
                    headers: {
                        'Content-Type': 'application/json',
                    },
                    body: JSON.stringify({ title, content }),
                });
                if (response.ok) {
                    alert('포스트가 성공적으로 작성되었습니다.');
                }
            };

            return (
                
setTitle(e.target.value)} required />
); }

4. 마크다운 처리 및 HTML 렌더링

사용자가 작성한 마크다운 내용을 HTML로 변환하여 렌더링하는 기능을 추가해야 합니다. 이 과정에서 `remark`와 `remark-html`을 사용하여 마크다운을 HTML로 변환합니다.

4.1. Markdown to HTML 변환 구현

포스트를 조회하는 페이지를 만들어 HTML로 변환된 내용을 렌더링합니다. `pages/posts/[id].js` 파일을 생성하고 아래와 같이 작성합니다.

import { useEffect, useState } from 'react';
        import { remark } from 'remark';
        import html from 'remark-html';

        const Post = ({ post }) => {
            const [contentHtml, setContentHtml] = useState('');

            useEffect(() => {
                const convertMarkdownToHtml = async () => {
                    const processedContent = await remark()
                        .use(html)
                        .process(post.content);
                    setContentHtml(processedContent.toString());
                };
                convertMarkdownToHtml();
            }, [post.content]);

            return (
                

{post.title}

); }; export async function getServerSideProps(context) { const { id } = context.params; const res = await fetch(`http://localhost:3000/posts/${id}`); const post = await res.json(); return { props: { post } }; } export default Post;

5. 전체 흐름 정리

이제 사용자가 블로그 포스트를 작성하고, 이를 마크다운 형식으로 저장하고, 이후 이를 HTML로 변환하여 렌더링할 수 있는 전체적인 기능을 갖춘 시스템이 완성되었습니다. Nest.js는 백엔드에서 포스트를 저장하고 관리하며, Next.js는 클라이언트에서 포스트 작성과 내용을 표시하는 역할을 합니다. 이러한 시스템 구성은 개발의 유연성을 높여주며, 유지보수 또한 쉽게 만들어줍니다.

6. 결론

블로그 포스트 작성 및 에디터 기능을 추가하는 과정에서 Nest.js와 Next.js를 활용한 다양한 예제를 살펴보았습니다. 마크다운을 HTML로 변환하는 과정은 블로그 개발에서 매우 중요한 부분이며, 이를 통해 사용자에게 더 나은 콘텐츠를 제공할 수 있습니다. 앞으로 자신만의 블로그를 구축하는 데 있어 Nest.js와 Next.js를 적극 활용해보세요!

저자: 조광형

게시일: 2024년 11월 26일

6.Next.js와 Nest.js 기본 프로젝트 구조 만들기, Next.js의 페이지와 Nest.js의 모듈 구성하기

JavaScript 생태계에서 Next.js와 Nest.js는 각각 프론트엔드와 백엔드 개발에 혁신적인 접근 방식을 제공합니다. Next.js는 React 기반의 프레임워크로, 서버 사이드 렌더링(SSR)과 정적 웹 사이트 생성을 간편하게 처리할 수 있도록 도와줍니다. 반면, Nest.js는 Node.js의 기반 위에 구축된 프로그레시브한 백엔드 프레임워크로, 모듈화와 클래스 기반의 아키텍처를 강조합니다. 본 강좌에서는 Next.js와 Nest.js의 기본 프로젝트 구조를 만드는 방법과 각각의 페이지 및 모듈 구성하는 방법에 대해 자세히 알아보겠습니다.

1. Next.js 프로젝트 구조 이해하기

Next.js 프로젝트는 일반적으로 다음과 같은 구조를 가집니다:

    my-next-app/
    ├── public/
    ├── pages/
    ├── components/
    ├── styles/
    ├── package.json
    └── next.config.js
    

여기서 각 디렉토리와 파일의 역할은 다음과 같습니다:

  • public/: 정적 파일(이미지, 아이콘 등)을 저장하는 디렉토리입니다. 이곳에 있는 파일은 “/” 경로로 접근할 수 있습니다.
  • pages/: Next.js에서 가장 중요한 디렉토리로, 애플리케이션의 각 페이지를 정의합니다. 파일 이름이 URL 경로에 직접적으로 매핑됩니다. 예를 들어, pages/index.js는 루트 페이지(/)를 나타냅니다.
  • components/: 여러 페이지에서 재사용할 수 있는 React 컴포넌트들을 저장하는 디렉토리입니다.
  • styles/: CSS 파일이나 스타일 관련 모듈을 여기에 보관합니다. Next.js는 CSS와 Sass를 지원합니다.
  • package.json: 프로젝트의 메타데이터와 의존성을 정의합니다.
  • next.config.js: Next.js의 다양한 설정을 구성할 수 있는 파일입니다.

2. Nest.js 프로젝트 구조 이해하기

Nest.js 프로젝트는 다음과 같은 구조를 가집니다:

    my-nest-app/
    ├── src/
    │   ├── app.module.ts
    │   ├── main.ts
    │   └── /
    │       ├── .module.ts
    │       ├── .controller.ts
    │       └── .service.ts
    ├── test/
    ├── package.json
    └── tsconfig.json
    

각 디렉토리와 파일의 역할은 다음과 같습니다:

  • src/: 애플리케이션의 모든 소스 코드를 포함합니다. Nest.js의 핵심 로직이 여기에 위치합니다.
  • app.module.ts: 애플리케이션의 루트 모듈입니다. 모든 모듈을 조합하는 역할을 합니다.
  • main.ts: 애플리케이션의 진입점으로, Nest.js 서버를 시작하는 파일입니다.
  • /: 특정 모듈을 위한 디렉토리입니다. 각 모듈은 controller, service 및 다른 관련 파일들을 포함할 수 있습니다.
  • test/: 테스트 관련 파일을 저장하는 디렉토리입니다.
  • package.json: 프로젝트의 메타데이터와 의존성을 정의합니다.
  • tsconfig.json: TypeScript 컴파일러 설정 파일입니다.

3. Next.js 프로젝트 만들기

먼저 Next.js 프로젝트를 만듭니다. 터미널에서 다음 명령어를 실행하세요:

npx create-next-app my-next-app

이 명령어는 Next.js의 템플릿을 기반으로 새로운 프로젝트를 생성합니다. 생성된 my-next-app 디렉토리로 이동한 후 실행하면 기본 Next.js 애플리케이션을 확인할 수 있습니다:

cd my-next-app
npm run dev

브라우저를 열고 http://localhost:3000에 접속하면 기본 페이지를 확인할 수 있습니다.

4. Nest.js 프로젝트 만들기

Nest.js 프로젝트를 생성하기 위해 Nest CLI를 사용합니다. 먼저 Nest CLI를 설치합니다:

npm i -g @nestjs/cli

설치가 완료되면 새로운 Nest.js 프로젝트를 생성합니다:

nest new my-nest-app

생성된 my-nest-app 디렉토리로 이동하여 서버를 실행합니다:

cd my-nest-app
npm run start

브라우저를 열고 http://localhost:3000에 접속하면 기본 Nest.js 애플리케이션을 확인할 수 있습니다.

5. Next.js의 페이지 구성하기

Next.js에서 페이지를 구성하기 위해 pages/ 디렉토리에 새로운 JavaScript 파일을 추가합니다. 예를 들어, pages/about.js를 생성하고 다음 코드를 작성합니다:

import React from 'react';

    const About = () => {
        return (
            

About Us

이곳은 우리의 소개 페이지입니다.

); }; export default About;

웹 브라우저에서 http://localhost:3000/about를 열면 방금 작성한 소개 페이지를 확인할 수 있습니다. Next.js는 파일 기반 라우팅을 지원하기 때문에, 다른 페이지를 추가하는 것도 매우 쉬운 작업입니다.

6. Nest.js의 모듈 구성하기

Nest.js에서는 모듈을 통해 애플리케이션을 구성합니다. 예를 들어, ‘users’ 모듈을 생성하고 구성하는 방법을 보겠습니다. 터미널에서 다음 명령어를 실행하여 모듈을 생성합니다:

nest generate module users
nest generate controller users
nest generate service users

이 명령어는 src/users/ 디렉토리를 생성하고, 사용자 관리에 필요한 서비스 및 컨트롤러를 자동으로 생성합니다. src/users/users.module.ts 파일을 다음과 같이 수정합니다:

import { Module } from '@nestjs/common';
    import { UsersController } from './users.controller';
    import { UsersService } from './users.service';

    @Module({
        controllers: [UsersController],
        providers: [UsersService],
    })
    export class UsersModule {}

이제 src/users/users.controller.tssrc/users/users.service.ts 파일을 수정하여 컨트롤러와 서비스의 로직을 추가할 수 있습니다. 아래는 사용자 목록을 반환하는 간단한 서비스와 컨트롤러의 예입니다.

import { Controller, Get } from '@nestjs/common';
    import { UsersService } from './users.service';

    @Controller('users')
    export class UsersController {
        constructor(private readonly usersService: UsersService) {}

        @Get()
        getAllUsers() {
            return this.usersService.findAll();
        }
    }
    
import { Injectable } from '@nestjs/common';

    @Injectable()
    export class UsersService {
        private readonly users = ['Alice', 'Bob', 'Charlie'];

        findAll() {
            return this.users;
        }
    }
    

위의 코드를 바탕으로 Nest.js 애플리케이션에서 사용자 목록을 반환하는 API를 구성할 수 있습니다. 브라우저에서 http://localhost:3000/users에 접속하면 사용자 목록을 확인할 수 있습니다.

7. Next.js와 Nest.js를 함께 사용하기

Next.js와 Nest.js를 함께 사용하여 풀스택 애플리케이션을 구축할 수 있습니다. 이러한 구조에서는 Next.js가 프론트엔드 부분을 담당하고, Nest.js가 백엔드 API를 제공하는 역할을 수행합니다. 다음 단계에서는 두 애플리케이션이 서로 통신할 수 있도록 설정합니다.

7.1. Nest.js API 서버 설정

먼저 Nest.js 애플리케이션을 다음과 같이 설정하여 CORS를 활성화합니다:

import { NestFactory } from '@nestjs/core';
    import { AppModule } from './app.module';

    async function bootstrap() {
        const app = await NestFactory.create(AppModule);
        app.enableCors();
        await app.listen(3000);
    }
    bootstrap();

7.2. Next.js에서 API 호출하기

Next.js 애플리케이션에서 Nest.js API를 호출하려면 데이터 패칭을 위해 axios와 같은 라이브러리를 사용할 수 있습니다. pages/index.js 파일을 다음과 같이 수정합니다:

import React, { useEffect, useState } from 'react';
    import axios from 'axios';

    const Home = () => {
        const [users, setUsers] = useState([]);

        useEffect(() => {
            const fetchUsers = async () => {
                const response = await axios.get('http://localhost:3000/users');
                setUsers(response.data);
            };
            fetchUsers();
        }, []);

        return (
            

사용자 목록

    {users.map(user => (
  • {user}
  • ))}
); }; export default Home;

이제 Next.js 애플리케이션의 홈 페이지에서 Nest.js API를 통해 사용자 목록을 가져오는 기능이 추가되었습니다.

8. 결론

Next.js와 Nest.js를 함께 사용하여 풀스택 애플리케이션을 구축하는 방법에 대해 살펴보았습니다. 각 프레임워크의 프로젝트 구조와 구성 방법을 이해하고, 두 애플리케이션 간의 데이터 통신을 설정하는 과정을 통해, 여러분은 효율적이고 강력한 애플리케이션을 개발할 수 있습니다.

이번 강좌를 통해 Next.js와 Nest.js의 기본적인 사용법뿐만 아니라, 실제로 프로젝트를 설정하고 페이지 및 모듈을 구성하는 방법을 배웠습니다. 앞으로 이 두 기술을 바탕으로 더욱 발전된 애플리케이션을 만들어보시기를 바랍니다.

17.Next.js와 Nest.js로 사용자 인증 구현하기, 로그인, 로그아웃, 회원가입 API 만들기

웹 개발에서 사용자 인증은 필수적인 요소입니다. 사용자가 안전하게 시스템에 접근할 수 있도록 하고, 개인정보를 보호하는 것이 매우 중요하다. 이번 강좌에서는 Next.js와 Nest.js를 사용하여 사용자 인증을 구현하는 방법을 자세히 살펴보겠습니다.

1. 프로젝트 설정

사용자 인증 시스템을 구현하기 위해 두 가지 프레임워크인 Next.js와 Nest.js를 사용할 것입니다. Next.js는 프론트엔드 프레임워크로 서버 사이드 렌더링(SSR)과 정적 웹사이트 생성을 지원합니다. 반면 Nest.js는 Node.js 기반의 백엔드 프레임워크로, 현대적인 서버 애플리케이션을 구축하는 데 적합합니다.

1.1 Nest.js 초기 설정

npm i -g @nestjs/cli
nest new backend
cd backend
npm install @nestjs/passport passport passport-local @types/passport-local bcrypt @types/bcrypt
npm install --save @nestjs/jwt passport-jwt @types/passport-jwt

1.2 Next.js 초기 설정

npx create-next-app frontend
cd frontend
npm install axios

2. Nest.js에서 사용자 인증 API 구현

Nest.js에서는 사용자 인증을 구현하기 위해 Passport.js와 JWT(JSON Web Token)를 사용할 것입니다.

2.1 User 엔티티 생성

먼저 User 엔티티를 생성합니다. 다음은 user.entity.ts 파일의 코드입니다.

import { Entity, Column, PrimaryGeneratedColumn } from 'typeorm';

@Entity()
export class User {
    @PrimaryGeneratedColumn()
    id: number;

    @Column({ unique: true })
    email: string;

    @Column()
    password: string;
}

2.2 UserService 및 UserModule 생성

import { Injectable } from '@nestjs/common';
import { User } from './user.entity';
import { InjectRepository } from '@nestjs/typeorm';
import { Repository } from 'typeorm';
import * as bcrypt from 'bcrypt';

@Injectable()
export class UserService {
    constructor(
        @InjectRepository(User)
        private userRepository: Repository,
    ) {}

    async create(userDto: { email: string; password: string }): Promise {
        const hashedPassword = await bcrypt.hash(userDto.password, 10);
        const user = this.userRepository.create({ ...userDto, password: hashedPassword });
        return this.userRepository.save(user);
    }

    async findByEmail(email: string): Promise {
        return this.userRepository.findOne({ where: { email } });
    }
}

2.3 AuthModule 및 AuthService 생성

import { Module } from '@nestjs/common';
import { PassportModule } from '@nestjs/passport';
import { JwtModule } from '@nestjs/jwt';
import { AuthService } from './auth.service';
import { LocalStrategy } from './local.strategy';
import { UsersModule } from '../users/users.module';

@Module({
    imports: [
        UsersModule,
        PassportModule,
        JwtModule.register({
            secret: 'SECRET_KEY', // 비밀키 설정
            signOptions: { expiresIn: '60s' }, // 토큰 유효 시간 설정
        }),
    ],
    providers: [AuthService, LocalStrategy],
})
export class AuthModule {}

2.4 로그인 및 회원가입 API 엔드포인트 추가

import { Controller, Post, Body, Req } from '@nestjs/common';
import { AuthService } from './auth.service';

@Controller('auth')
export class AuthController {
    constructor(private readonly authService: AuthService) {}

    @Post('register')
    async register(@Body() userDto: { email: string; password: string }) {
        return this.authService.register(userDto);
    }

    @Post('login')
    async login(@Body() userDto: { email: string; password: string }) {
        return this.authService.login(userDto);
    }
}

3. Next.js의 사용자 인증 UI 구현하기

Next.js에서는 사용자 인증을 위한 로그인 및 회원가입 폼을 구현합니다. 사용자 입력을 처리하여 Nest.js 백엔드로 요청을 보내도록 합니다.

3.1 로그인 페이지 구현

import axios from 'axios';
import { useState } from 'react';

const Login = () => {
    const [email, setEmail] = useState('');
    const [password, setPassword] = useState('');

    const handleSubmit = async (e) => {
        e.preventDefault();
        const response = await axios.post('/api/auth/login', { email, password });
        // 로그인 성공 시 처리
    };

    return (
        
setEmail(e.target.value)} placeholder="Email" /> setPassword(e.target.value)} placeholder="Password" />
); }; export default Login;

3.2 회원가입 페이지 구현

import axios from 'axios';
import { useState } from 'react';

const Register = () => {
    const [email, setEmail] = useState('');
    const [password, setPassword] = useState('');

    const handleSubmit = async (e) => {
        e.preventDefault();
        const response = await axios.post('/api/auth/register', { email, password });
        // 회원가입 성공 시 처리
    };

    return (
        
setEmail(e.target.value)} placeholder="Email" /> setPassword(e.target.value)} placeholder="Password" />
); }; export default Register;

4. 사용자 인증 상태 관리

로그인 이후에는 사용자의 인증 상태를 관리해야 합니다. Next.js에서는 Context API 또는 Redux와 같은 상태 관리 라이브러리를 활용할 수 있습니다.

4.1 Context API를 사용한 상태 관리

import { createContext, useContext, useState } from 'react';

const AuthContext = createContext();

export const AuthProvider = ({ children }) => {
    const [user, setUser] = useState(null);
    const login = (userData) => setUser(userData);
    const logout = () => setUser(null);

    return (
        
            {children}
        
    );
};

export const useAuth = () => useContext(AuthContext);

5. 마무리

이번 강좌에서는 Next.js와 Nest.js를 사용하여 사용자 인증 시스템을 구현하는 방법을 살펴보았습니다. 우리는 로그인, 회원가입, 로그아웃 API를 생성하고 프론트엔드에서는 사용자 인터페이스를 구성했습니다.

사용자 인증은 웹 애플리케이션에서 매우 중요한 기능입니다. 이를 통해 사용자의 개인정보와 데이터 안전성을 높일 수 있습니다. 추가적으로 JWT를 사용하여 인증 상태를 유지하고, Refresh Token과 같은 기능을 구현하면 더욱 안전한 인증 시스템을 만들 수 있습니다.

앞으로 이 시스템에 추가할 수 있는 기능들로는 비밀번호 찾기, OAuth 인증(구글, 페이스북 등), 사용자 권한 관리 등이 있습니다. 이러한 기능들을 통해 더 강력하고 유용한 인증 시스템으로 발전시킬 수 있습니다.