Skip to content

Conversation

@heonq
Copy link
Collaborator

@heonq heonq commented Feb 23, 2025

요구사항

기본 요구사항

공통

  • AWS 루트 유저 계정을 생성하세요. 이미 생성된 계정이 있다면 해당 계정을 활용해 주세요.
  • AWS Free Tier 제공 범위를 파악한 후, 해당 서비스를 적극 활용하세요.
  • 인스턴스의 중지 및 종료 과정을 꼭 숙지해야 합니다. EC2 과금 정책에 주의하여 프리 티어 한도 내에서 사용해 주세요.
  • 리전은 "아시아 태평양(서울)"으로 설정하세요.

백엔드 배포

프로젝트 구조 및 환경 설정

  • 배포에 적합한 프로젝트 구조를 설정합니다.
  • 개발(development) 및 배포(production) 환경 설정을 구분하고, 환경 변수를 사용해 관리합니다.

AWS S3를 이용한 파일 업로드 시스템 구축

  • AWS S3 버킷을 생성하고, 파일 업로드를 위한 설정을 완료합니다.
  • multer-s3 라이브러리를 사용하여 이미지 업로드 미들웨어를 S3로 변경합니다. (Presigned-URL을 통해 업로드하도록 구현했습니다.)
  • S3에 이미지 업로드가 정상적으로 작동하는지 확인합니다.

AWS RDS를 사용한 데이터베이스 관리

  • AWS RDS 인스턴스를 설정하고, 프로젝트 데이터베이스와 연결합니다.
  • RDS에서 데이터베이스의 초기화 및 CRUD 작업을 테스트합니다.

AWS EC2에서의 애플리케이션 운영

  • AWS EC2 인스턴스를 생성합니다. 프리티어에 해당하는 인스턴스 타입과 운영 체제(OS)를 선택하세요.
  • EC2 인스턴스에 대한 보안 그룹을 설정합니다.
    • HTTP(포트 80), HTTPS(포트 443), SSH(포트 22) 등 필요한 포트를 열어 네트워크 연결을 구성하세요.
  • pm2를 사용하여 애플리케이션을 백그라운드에서 실행시킵니다.
  • Nginx를 이용한 리버스 프록시 설정을 구축하고, 외부 접속을 관리합니다.

백엔드 테스트 구현

  • Jest 설정 파일(jest.config.js)을 만들고 기본 설정을 하세요.
  • 상품 CRUD 연산에 대한 유닛 테스트를 작성합니다.
    • 각 CRUD 연산에 대해 적절한 입력과 예상 출력을 정의하여 테스트 코드를 구현하세요.
  • 사용자의 접근 권한 검증을 고려하여 상품 CRUD 연산에 대한 시나리오를 테스트해 주세요.
  • 회원가입, 로그인에 대한 유닛 테스트를 작성합니다.
  • API 요청이나 데이터베이스 작업 등 비동기 코드에 대한 테스트를 작성하세요.
    • async/await와 done 콜백을 사용하여 비동기 코드의 완료를 테스트하세요.
  • Mock, Spy와 같은 테스트 더블을 사용하여 외부 서비스와의 상호 작용을 테스트하세요.
  • describe와 test 블록을 사용하여 테스트 케이스를 그룹화하고 정리하세요.

심화 요구사항

테스트 구현

  • Jest의 테스트 커버리지 도구를 사용해 코드 커버리지를 분석하고 결과를 확인해 보세요.
  • 커버리지 결과를 바탕으로 누락된 테스트 케이스를 추가합니다.
    • 커버리지 보고서를 검토하여 테스트되지 않은 코드 영역을 찾아내고 적절한 테스트를 추가해 보세요.

상품 이미지 업로드

  • AWS S3의 Presigned URL 기능을 활용하여 상품 이미지 업로드 기능을 구현합니다.

AWS Route 53을 활용한 도메인 관리

  • AWS Route 53을 사용하여 도메인을 구매하거나 기존 도메인을 연결합니다.
  • Route 53에서 DNS 설정을 관리하고, EC2 인스턴스와 연결합니다.
  • 도메인을 통한 애플리케이션 접속 및 운영을 테스트합니다.

SSL 인증서를 통한 HTTPS 연결 구현

  • SSL 인증서를 설정하여 EC2 인스턴스에서 HTTPS 연결을 구현합니다.
  • SSL 인증서는 AWS Certificate Manager(ACM)를 사용하여 무료로 생성하거나, 외부 인증 기관에서 구매할 수 있습니다.
  • certbot을 사용해 SSL 인증서를 설정했습니다.

주요 변경사항

  • S3와 CloudFront를 연동해서 리소스를 캐싱하고 S3에 대한 직접 접근을 차단하여 보안을 강화했습니다.

스크린샷

Nginx 설정

image

PM2 목록

image

EC2 보안 그룹 설정

스크린샷 2025-02-23 오후 7 25 43

멘토에게

  • 멘토링 시간 때 질문드렸던 것처럼 repository에 대한 테스트가 테스트를 실행할 때마다 결과가 달라집니다! describe.skip을 통해 다른 repository 테스트를 모두 스킵하고 하나의 repository에 대한 테스트를 실행했을 땐 항상 통과하는 것으로 봐선 테스트가 병렬적으로 실행되는 게 아닐까 싶습니다!

@@ -1,5 +1,5 @@
import { Request, Response, NextFunction } from 'express';
import { AsyncRequestHandler } from '../../types/asyncRequestHandler.types';
import type { AsyncRequestHandler } from '@/types/asyncRequestHandler.types';
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

Comment on lines +15 to +18
"paths": {
"@/*": ["./src/*"],
"@prismaDir/*": ["./prisma/*"]
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

@@ -1,5 +1,5 @@
import { Request, Response } from 'express';
import { UserService } from './service';
import { UserService } from '../service/service';
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

저는 경로에 동일한 단어가 두 번 연속으로 들어가면 위화감이 느껴져서(버그가 아닌데 버그처럼 보이는 기분) 저라면 index.ts를 통해 경로를 보기 좋게 처리해볼 것 같아요.

Comment on lines 4 to +5
export default class UserRepository {
constructor(private prismaClient: PrismaClient) {}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이 Repository 클래스는 prisma client를 주입받는 repository 클래스이니 이름을 더욱 구체화해본다면 UserPrismaRepository로 지어볼 수도 있을 것 같네요.
사실 지금처럼 Repository 구현체가 하나인 상황에서는 큰 실효성도 느끼고 힘들고, 그렇게 할 필요까지도 없지만, 추후 데이터를 가져올 수 있는 경로가 분산된다면(e.g. local, cache, file, ...) 이때는 명확하게 문제가 생기는 지점일 거에요.
그때는 아마 UserRepositry는 클래스가 아닌 인터페이스로 정의가 되곘죠?

class UserPrismaRepositry implements UserRepository
class UserCacheRepositry implements UserRepository
class UserLocalRepositry implements UserRepository

Comment on lines +1 to +8
import UserRepository from '../user/repository/userRepository';
import { AuthController } from './controller/controller';
import { AuthService } from './service/service';
import express from 'express';
import asyncRequestHandler from '../../core/handlers/asyncRequestHandler';
import { validateBody } from '../../core/middleware/validate';
import { signInRequestStruct, SignUpRequestStruct } from '../../structs/authStruct';
import asyncRequestHandler from '@/core/handlers/asyncRequestHandler';
import { validateBody } from '@/core/middleware/validate';
import { signInRequestStruct, SignUpRequestStruct } from '@/structs/authStruct';
import { prismaClient } from '@/prismaClient';
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

지금 eslint 적용하고 있는 중인데 import라는 룰을 적용하면 eslint에서 자동으로 import 구문 순서를 맞춰주는데 적용해보면 적은 비용으로 큰 효용을 만들어낼 수 있을 것 같아요.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants