백엔드 개발자(node.js)가 되는 과정 76

[WIL] 23.06.26 ~ 06. 30

프로젝트 회고 (공통) EJS 파일 사용 관련 어려움 백엔드에서 응답 받은 결과를 EJS 파일에 출력하는 과정에서 문법이 익숙하지 어려움이 있었다. 기존 프론트 구현 방식처럼 HTML에 javascript를 연동시켜서 페이지에 출력하는 방식과 EJS에 직접 백엔드 결과를 전달하는 방식을 모두 사용하게 되어 해당 부분에서 기능 구현이 다소 비효율적이게 된 것 같다. 추가 기능 구현에서의 어려움 업로드 된 이미지를 프론트에 출력하는 과정에서 이미지 사이즈가 일관성을 유지하도록 하기 위한 과정에서 어려움이 있었음 일관된 이미지 사이즈 출력 구현을 위해 리사이징을 시도하는 과정에서 sharp 패키지 사용을 시도했으나 input 관련 에러가 발생해서 구현에 한계에 부딪힘 이미지 리사이징 관련하여 AWS Lambd..

express에서 ejs에 css, js 파일 불러오기

백엔드를 공부하는 입장에서 프로젝트에서 프론트 작업을 하는 비중이 낮은데, 그렇다 보니 기초적인 부분에 대한 기억을 다시 끌어올려야 할 때가 많다. 오늘과 같은 예시도 그러한 부분 중 하나이다. ejs 파일에서 우리가 일반적으로 생각하는 경로 설정 방법으로 css나 js파일을 불러오도록 아래와 같이 작성할 경우 오류를 만날 수 있다. 나의 경우 static 폴더 내 css와 js폴더를 생성 후 각각 파일을 넣어뒀는데 아래와 같은 오류를 만날 수 있었다. 'Refused to execute script from 'http://localhost:3000/static/js/index.js' because its MIME type ('text/html') is not executable, and strict M..

Github Pull Request에 등록된 코드 가져오기

Pull Request에 등록된 코드를 눈으로만 리뷰하는 것과 직접 테스트 해보는 것은 아무래도 차이가 있는 것 같다. 그렇기에 아무리 코드 리뷰를 통해 문제가 없음을 확인했다 하더라도 직접 테스트 해보는 것 만큼 확신이 서지는 않기에 미리 pull 받아서 테스트 해보게 되는 것 같다. 방법은 간단하다. 테스트용 브랜치를 하나 생성해서 현재 최신으로 등록된 main 코드를 가져온다. git checkout -b test/pr origin/main merge 대상인 pr번호를 확인해서 가져오는 명령을 아래와 같이 작성한다. git pull origin pull/pr번호/head:test/pr 이와 같이 특정 브랜치에 테스트를 마친 후 merge를 하는 것이 좀 더 안전한 방법인 것 같다.

MySQL Sequelize에서 테이블에 필드(컬럼) 추가하기

웹사이트 프로젝트에서 서비스가 추가되다 보니 기존에 작성했던 ERD를 수정해야 하는 상황이 발생했다. 처음부터 ERD를 잘 짰으면 좋았겠지만 그러기는 쉽지 않은 것 같다. 어쨌든 이렇게 되면 특정 테이블 또는 테이블의 컬럼을 추가해야하는 상황이 발생한다. 이럴 때는 마이그레이션을 추가해주면 된다. npx sequelizse db:migrate 명령을 수행하면 migrations 폴더 내 js 파일을 순차적으로 실행하므로 기존 파일들 뒤에 신규 마이그레이션 파일을 추가해줌으로서 컬럼 또는 테이블을 추가해주는 명령을 더해주면 되는 것이다. 먼저 아래 명령을 실행하면 새로운 마이그레이션 파일이 생성된다. --name 이후의 이름은 자유롭게 지으면 되나 가급적 의미가 담겨있으면 좋을 것 같다. 아래 예시의 경우..

Docker로 MySQL 설치 및 실행하기 (Mac OS)

도커를 왜 쓰는 걸까? 도커에 대한 설명은 쉽게 찾아볼 수 있으므로 매우 간단하게 설명하자면, 하나의 컴퓨터를 여러 대의 컴퓨터인 마냥 써야할 경우에 사용한다. 예를 들어서 웹 서비스를 총 10개 운영한다고 했을 때 하나의 컴퓨터, 하나의 OS에서 10개 사이트에 대한 관리를 하는 것보다(가능한지 잘 모르겠다.) 독립된 10개의 공간에서 각각 관리하는 것이 유지 보수에 유리하기 때문이다. 왜냐하면 컴퓨팅 자원을 분리하여 독립적으로 사용할 수 있다는 장점이 있기 때문이다. 3번 웹 서비스 운영에 이슈가 생겼다고 해서 5번 서비스 운영에 지장을 주면 안되기 때문이다. 1. 도커 데스크탑 설치 Mac OS의 터미널을 통해서 도커를 설치하고, 켜고 끄거나 할 수 있겠지만 개인적으로 가장 간단하게는 그냥 도커 데..

[WIL] 23. 06. 19 ~ 06. 23

지난주 개인과제를 진행하면서 배운 내용을 정리했습니다. 개요 기존 과제에서 mongoose로 운영한 database를 MySQL과 sequelize로 전환한다. API 별 기능 Path API Method Verify Description /user POST 회원가입 /auth/login POST 로그인 /user GET 전체유저조회 /user:userId GET 개별유저조회 /user:userId/posts GET 유저가 작성한 게시글 조회 /posts GET 전체 게시글 조회 /posts POST ✔ 게시글 작성 /posts/:postId PUT ✔ 게시글 수정(유저 검증) /posts/:postId DELETE ✔ 게시글 삭제(유저 검증) /posts/:postId/comments GET 게시글 내..

nodejs 비밀번호 해싱 처리하기

비밀번호 해싱 처리 회원가입 시 유저의 비밀번호가 데이터베이스에 그대로 노출되지 않도록 해싱 처리 후 저장합니다. 해싱 처리를 위한 패키지를 설치합니다. npm install bcrypt user모델에서 User.init 아래 아래 메서드를 추가해줍니다. 이로서 유저 생성 전 비밀번호의 해싱 처리가 이루어 집니다. bcrypt 패키지를 사용하고, saltRounds를 10으로 입력해서 해싱 암호를 좀 더 강화합니다. // user.js const bcrypt = require('bcrypt'); User.beforeCreate(async (user) => { const saltRounds = 10; const hashedPassword = await bcrypt.hash(user.password, sal..

프로세스와 스레드에 대한 간단 정리

공부한 내용을 이해한대로 정리하자면! 프로세스를 하나의 실행 프로그램 단위로 볼 수 있을 것 같다. 가령 MS-WORD를 켰다고 했을 때 워드 프로그램 실행을 위한 단위를 프로세서로 보고, 내부적으로는 우리에게 보여줄 메뉴, 툴바 같은 UI출력, 키보드 입력 시 글자 입력, 첫 문장에서는 대문자로 자동 변경, 틀린 문자 빨간줄 긋기 등 각종 세부적인 기능 작동을 위한 명령 단위 하나하나를 스레드라고 볼 수 있다. CPU가 멀티 코어로 구성되어 있다면 프로세스 여러 개를 병렬로 동시에 처리할 수 있게 되고, 그만큼 수많은 프로세서를 다 처리하는데 걸리는 시간이 단축된다고 볼 수 있다. 더 나아가 프로세서 안에 스레드 또한 단일 스레드와 멀티 스레드로 나뉜다. 단일 스레드는 하나의 프로세서에서 스레드를 하나..

Sequelize로 MySQL JOIN하기

MySQL의 JOIN 적용을 위한 준비 게시글 작성 사이트를 예시로 들어보자. 데이터베이스에는 유저 테이블과 게시글 테이블이 구분되어 있을 것이고, 게시글 테이블을 통해 해당 게시글의 작성자가 누구인지, 좀 더 깊게 들어가서 작성자의 이메일은 무엇인지 알기 위해서는 두 테이블 간 JOIN 작업이 필요할 것이다. sequelize를 통해 두 테이블을 JOIN 해서 정보를 획득하는 방법에 대해 알아보자 두 테이블의 관계 설정 // 유저 모델과 포스트 모델 간 1대 다 관계 설정 // models/user.js static associate(models) { this.hasMany(models.Post, { sourceKey: 'id', foreignKey: 'userId', }); } // models/po..

Sequelize로 모델 생성 및 테이블 연동하기

모델생성 및 테이블 연동하기 마이그레이션을 통해 테이블을 생성하면 아래와 같이 models 폴더 내 members.js파일이 함께 생성된다. 코드를 보면 마이그레이션과 비슷한 형태로 생성된 것을 확인할 수 있다. const { Model } = require('sequelize'); module.exports = (sequelize, DataTypes) => { class Member extends Model {} Member.init( { name: DataTypes.STRING, team: DataTypes.STRING, position: DataTypes.STRING, emailAddress: DataTypes.STRING, phoneNumber: DataTypes.STRING, admissionD..