게시글

etc - 2년 전 등록 / 7달 전 수정

Node.js 교과서 질의응답용 포스트

조회수:
0

FAQ

  • 스스로 해보기 풀이는 없나요? - 유튜브나 인프런 강좌에서 제공합니다(인프런에서도 무료로 보실 수 있습니다.)
  • 오타&오류 정오표는 어디에 있나요? - 이 링크를 클릭하세요. 
  • 이미지 말고 다른 파일을 업로드하려면 어떡하나요? - multer로 다른 파일 형식도 업로드 가능합니다(동영상, 문서 등등).
  • pug 스타일이 너무 싫어요 ㅠㅠ - ejs나 다른 걸로 바꾸시면 됩니다. pug는 제 책에서 중요한 부분이 아닙니다. 그저 지면을 아끼기 위해 선택했을 뿐이에요.
  • bcrypt가 윈도에서 안 깔려요. - bcrypt 문서에 윈도에서 bcrypt 설치하는 방법에 대해 나와 있습니다. 3버전부터는 그렇게 적용하셔야 합니다.
  • user && twit && twit.Liker.map(l => l.id).includes(user.id) 이 부분에서 에러가 나신다면, user && twit && twit.Liker && twit.Liker.map으로 바꿔주세요.
  • 시퀄라이즈 5버전에서 WHERE parameter "id" has invalid "undefined" value 문제는 https://github.com/ZeroCho/nodejs-book/issues/42 참고하세요
  • TypeError [ERR_INVALID_CHAR]: Invalid character in header content ["Set-Cookie"] 에러 발생하시는 분은 Expires 앞에 줄바꿈을 제거해주세요. 줄바꿈 문자가 들어가면 안 됩니다.

다른 질문은 댓글로 입력해주세요. 또는 길벗 공식 홈페이지에 문의를 남겨주시면 됩니다.

투표로 게시글에 관해 피드백을 해주시면 게시글 수정 시 반영됩니다. 오류가 있다면 어떤 부분에 오류가 있는지도 알려주세요! 잘못된 정보가 퍼져나가지 않도록 도와주세요.
Copyright 2016- . 무단 전재 및 재배포 금지. 출처 표기 시 인용 가능.

댓글

59개의 댓글이 있습니다.
한 달 전
13장 실습중이네요 서버를 실행하면 nodemon이 계속 자기 혼자 재시작-시작-재시작-시작 무한 반복하네요. 뭐가 문제인건가요?? 그리고 저는 현재 몽고디비 서버를 실행할 때
sudo brew services start mongodb-community@4.2
를 사용 중인데 이 코드를 사용한 상태면 다음엔 몽고디비 사용 시 매번 저 코드를 입력할 필요가 없는거죠?
한 달 전
뭔가 시작 과정에서 자꾸 어떤 파일이 바뀌어서 재시작되는 것 같습니다. 일단 제 깃헙 코드를 사용해보세요. brew services start를 하셨으면 다음부터는 입력하실 필요 없습니다.
2달 전
안녕하세요! 항상 블로그 잘 보고있습니다. 한가지 궁금한게 제가 Node.js 교과서를 구입하였는데 늦게나마 공부하려고 합니다... 근데 문득 생각난게 집필당시의 버전이랑 요즘 버전에서 차이가 많이 난다면 개정판을 기다렸다가 사서 보는게 더 나을까 해서 질문드려요, 얼핏 알기로는 React 같은경우는 예전 책에없는 v2나 hook 등등 바뀐게 좀 있다고 들어서.. (잘몰라요;;) 혹시 Node도 그런가해서 질문드려요!
2달 전
노드는 상대적으로 변화가 적습니다. 개정판은 적어도 6개월 이상 소요됩니다. 개정판의 주요 목표는 설명을 쉽게 하는 거지, 새로운 내용은 별로 없습니다.
2달 전
밑 글 질문자입니다. socket.request를 찍어본 결과 session정보가 MemoryStore라는 곳에 있고 socekt.request.session에 존재하지 않습니다. MemoryStore에 있어 접근이 되지 않는데 어떻게 해결하나요
2달 전
socket.request.session이 undefined가 되는 경우이기 때문에 session을 찍어보실 필요는 없고 url이나 socket.id같은 부가 정보를 찍어보셔야 합니다.
2달 전
안녕하세요.

socket.io를 이용한 실시간 채팅 구현을 하던 중 문제가 생겨 질문드립니다.

socket.io와 express-session 연동을 위해 다음과 같이 코드를 작성하였습니다.
io.use((socket, next) => {
sessionMiddleware(socket.request, socket.request.res, next);
});

그런데 req.session.user에 저장한 별도의 정보가 제대로 전달이 될 때가 있지만, 가끔씩 undefined가 뜨는 경우가 있습니다.
해결법을 찾아봐도 모르겠어 질문 올립니다.
2달 전
그 상황을 찾아보려면 undefined가 뜰 때 socket.id나 socket.reuqest같은 것, socket.request.url 이런것도 함께 찍어보셔야할 것 같습니다.
4달 전
안녕하세요.
책을 읽으면서 궁금한점이 있어 질문합니다.

교제 chapter 12 실시간 경매 시스템(p.487) 만들기를 구현하면서 model간의 관계를 정의할 때,
```
db.Good.belongsTo(db.User, { as: "owner" });
db.Good.belongsTo(db.User, { as: "sold" });
db.User.hasMany(db.Auction);
db.Good.hasMany(db.Auction);
db.Auction.belongsTo(db.User);
db.Auction.belongsTo(db.Good);
```
아래와 같이 작성했는데 user 모델과 good 모델의 역관계를 설정하지 않은 이유가 있나요?
`Ex. db.User.hasMany(db.Good, { as: "soldItems" });`

주변에 친구가 orm에 작성할 때, 관계를 많이 잡아 놓으면 안좋다는 이야기를 들었는데 맞는 이야기일까요? 제 생각에는 확실히 맞는 관계라고 하면 관계를 잡아놓아도 큰 문제가 없다고 생각하는데 어떻게 생각하시나요??

감사합니다.
4달 전
저 관계는 include를 할 때에만 필요합니다. include를 하지 않으면 관계 설정은 필요 없습니다. 관계를 많이 잡아 놓아도 큰 문제는 없어보입니다.
4달 전
답변 감사합니다.
4달 전
안녕하세요. 제로님.

궁금한 점이 있어 글 남깁니다.

제가 궁금한것은 다음과 같습니다.


1. 특정 시간에 DB 작업을 하는 경우(CRUD)

만약 위와 같은 사항일 경우 일반적으로 책의 내용 중 하나인 노드 스케줄러를 사용하나요?

더불어 매일 정각 24시에 특정 작업을 해야 한다며 노드 서버 외에 별도의 노드 프로그램을 실행시켜주는 것이 안전할까요.

궁금합니다!
4달 전
node-sheduler보다는 운영체제 상에서 돌아가는 crontab을 사용하는 게 좋습니다.
5달 전
교재를 이용해서 웹사이트를 만들고 있는 중입니다. 그런데 multer 모듈을 이용해서 이미지 업로드를 구현하고 있는데 이미지가 웹페이지에 나타날 때 브로큰 이미지 형태로 나타납니다...src=twits.img 이부분이 잘못된 걸까요?? 왜 img/${req.file.filename}형태로 해야되는지 이해가 되지않습니다..ㅜ
6달 전
교재의 mysql부분 공부중인데 sequelize 이용해서 mysql이랑 express 랑 연결하는 부분 실습중인데요, 교재상으로는 276페이지에서 npm start를 하면

(node:9628) [SEQUELIZE0004] DeprecationWarning: A boolean value was passed to options.operatorsAliases. This is a no-op with v5 and should be removed.

이러한 메세지가 계속 뜨네요
테이블도 삭제해서 없는데 생성되는 SQL문도 안나오고 뭐가 원인인지 모르겠네요
구글링해도 뭔지 모르겠고 답답합니다 ㅜ 도와주세요

model에 user index comment 모두 그냥 복붙하고, config도 그대로 복붙해서 실행하는데도 안되요 ㅜㅜ
6달 전
저 메시지는 에러 메시지가 아닙니다. 단순한 경고메시지라서 무시해도 됩니다. 다른 에러메시지가 있을 거에요.
7달 전
144p 실습중에 TypeError [ERR_INVALID_CHAR]: Invalid character in header content ["Set-Cookie"] 에러가 발생합니다.




8080번 포트에서 서버 대기중입니다
_http_outgoing.js:464
throw err;
^

TypeError [ERR_INVALID_CHAR]: Invalid character in header content ["Set-Cookie"]
at storeHeader (_http_outgoing.js:411:5)
at processHeader (_http_outgoing.js:406:3)
at ServerResponse._storeHeader (_http_outgoing.js:312:9)
at ServerResponse.writeHead (_http_server.js:271:8)
at Server.http.createServer (c:\Users\illke\Desktop\Nodejs\server2.js:25:17)
at Server.emit (events.js:198:13)
at parserOnIncoming (_http_server.js:677:12)
at HTTPParser.parserOnHeadersComplete (_http_common.js:109:17)


어떻게해야할까요...
7달 전
아래 댓글 참조해주세요.
7달 전
말씀하신대로 Console.log(encodeURIComponent(name))을 수행했고, 로그인시 입력한 Name대로 잘 찍힙니다.


https://thebook.io/006982/ch04/02-08/

혹시 오타가 있나 싶어서 더북에있는 원문을 긁어서 붙여넣기하고 돌려봤습니다만 여전히 에러가 나네요.




c:\Users\illke\Desktop\Nodejs>node server4
8083번 포트에서 서버 대기 중입니다!
_http_outgoing.js:464
throw err;
^

TypeError [ERR_INVALID_CHAR]: Invalid character in header content ["Set-Cookie"]
at storeHeader (_http_outgoing.js:411:5)
at processHeader (_http_outgoing.js:406:3)
at ServerResponse._storeHeader (_http_outgoing.js:312:9)
at ServerResponse.writeHead (_http_server.js:271:8)
at Server.http.createServer (c:\Users\illke\Desktop\Nodejs\server4.js:25:13)
at Server.emit (events.js:198:13)
at parserOnIncoming (_http_server.js:677:12)
at HTTPParser.parserOnHeadersComplete (_http_common.js:109:17)

c:\Users\illke\Desktop\Nodejs>
7달 전
잘 찍힌다는 것은 한글 닉네임은 아니란 말씀이시죠? 혹시 다른 쿠키데이터중에도 한글같은 게 있나요? 브라우저에서 쿠키 지우기를 하고 다시 시도하는 것도 방법입니다. 분명 쿠키 문자중에 ASCII가 아닌 문자가 있는것 같습니다.
7달 전
닉네임은 na123을 사용했고, 쿠키 지우고 다시 시도해봤습니다만 아직도 에러가 나네요..

조금만 더 해보고 안되면 그냥 넘어가려고 합니다.
7달 전
혹시 전 원문에 Expires 앞에 줄바꿈까지 그대로 하셨나요? 줄바꿈이 있으면 안 됩니다. 저 부분은 책에 공간이 부족해서 다음 줄로 밀려난 거에요.
7달 전
줄바꿈때문이네요...감사합니다 해결했습니다
8달 전
162페이지의 실습 중, TypeError [ERR_INVALID_CHAR]: Invalid character in header content ["Set-Cookie"] 라는 에러가 나왔습니다. 해당 코드를 작성하다가
'Set-Cookie': `name=${encodeURIComponent(name)};
Expires=${expires.toGMTString()}; HttpOnly; Path=/`
문법 오류가 나서 set-cookie를 백틱을 사용했는데 실행해보니 저런 오류가 나옵니다....ㅠ 어떻게 해야 할까요
8달 전
encodeURIComponent(name) 이 부분을 console.log로 찍어보세요. 여기 글자가 오류인 것 같습니다.
8달 전
제로님. 궁금한점이 있습니다.

1. 28p 질문

해당 페이지에서 블로킹방식의 코드를 작성하는데 왜이렇게 작동하는지 궁금합니다.

console.log 는 콘솔 객체의 로그라는 메서드를 사용하는걸로 알고 있습니다.

그렇다면 log도 함수라는 이야기 아닌가요?

만약 맞다면 호출스택은 후입선출이기때문에 다음의 결과가 먼저 나와야 되는것 아닌가해서요.

console.log('다음 작업')

호출 순서대로 쌓이고 실행은 역순으로 실행되는게 아닐까요?

궁금하고 많이 헷갈리네요.

2. 112p 질문

해당 페이지의 동기와 비동기, 블로킹과 논블로킹 설명 내용에서 비동기-논블로킹 방식에서는 호출한 함수가 바로 return 처리 된다는 말이 어떤말일까요. 정확히 어떤걸 return(반환) 한다는건지 모르겠습니다.
8달 전
함수가 실행이 종료되지 않은채 다음 함수가 호출되는 경우에만 호출스택에 쌓입니다.

논블로킹에서 함수가 바로 리턴된다는 것은 콜백이 실행되지 않아도 현재 함수가 종료된다는 뜻입니다.
8달 전
안녕하세요. 지금 chapter 9 공부 중인데 p353 에서 profile.pug의 user.Following이라던지 p.352에서 main.pug의 twit.user.nick 등 온점(.) 뒤에 오는 Following이나 user.nick은 어디서 정의가 되어있는건가요? 저걸 뭐라고 부르나요? 저게 존재하는지 어떻게 알 수 있는건가요?
감사합니다.
8달 전
저도 그 부분이 헷갈렸는데 routes의 page.js 에서
res.render('main', {
title: 'NodeBird',
twits: posts,
user: req.user,
에서 posts가 main.pug의 twits로 넘어가는 것 같더라고요. posts를 console.log(`posts= ${JSON.stringify(posts)}`); 해서 찍어보시면 보낸 쿼리 결과가 json형태로 전달받고 있는걸 볼 수 있습니다. 마찬가지로 Following 이나 user.nick도 찍어보면 db 테이블의 컬럼 이름입니다.
9달 전
안녕하세요. 책 너무 잘 읽고 있습니다. sns만들어보는 예제를 따라해보니 팔로잉하는 유저의 트윗만 보이는게 아니라 모든 계정에서 모든 트윗이 보이는 식으로 구현되어 있는 거 같습니다. 팔로잉하는 유저의 트윗만 볼 수 있도록 수정을 해보려고 하는데 막혀서 질문드리고 싶습니다.

SELECT *
FROM nodebird.posts, nodebird.follow
WHERE nodebird.follow.followingid = nodebird.posts.userid
&& nodebird.follow.followerId = (req.user.id);

같은 느낌의 쿼리를 쓰려고 하는데 시퀄라이즈로 어떻게 표현해야할지 막막하네요. users나 posts 테이블 같은 경우에는 모듈로 받아와서 User.findAll() Post.findAll() 할 수 있는데 follow테이블 같이 직접 생성한게 아니라 n:m 관계를 통해서 생성된 테이블들은 어떻게 접근해야 할까요.
9달 전
쿼리를 어떻게 써야할 지 모르실 때는 db.sequelize.query(QUERY)로 직접 SQL 쓰시면 됩니다.
9달 전
chapter6에서 npm start 부분에 에러가 나고 있습니다.
구글링을 해서 다른 사람들이 말한대롤 설치도 다시 해봣는데도 같은 문제로 몇시간째 끙끙대고 있습니다.
뭐가 문제일까요?ㅠㅠㅠ

> learn-sequelize@0.0.0 start /Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize
> node ./bin/www

internal/modules/cjs/loader.js:798
throw err;
^

SyntaxError: /Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/config/config.json: Unexpected token } in JSON at position 196
at JSON.parse (<anonymous>)
at Object.Module._extensions..json (internal/modules/cjs/loader.js:795:27)
at Module.load (internal/modules/cjs/loader.js:653:32)
at tryModuleLoad (internal/modules/cjs/loader.js:593:12)
at Function.Module._load (internal/modules/cjs/loader.js:585:3)
at Module.require (internal/modules/cjs/loader.js:690:17)
at require (internal/modules/cjs/helpers.js:25:18)
at Object.<anonymous> (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/models/index.js:5:16)
at Module._compile (internal/modules/cjs/loader.js:776:30)
at Object.Module._extensions..js (internal/modules/cjs/loader.js:787:10)
at Module.load (internal/modules/cjs/loader.js:653:32)
at tryModuleLoad (internal/modules/cjs/loader.js:593:12)
at Function.Module._load (internal/modules/cjs/loader.js:585:3)
at Module.require (internal/modules/cjs/loader.js:690:17)
at require (internal/modules/cjs/helpers.js:25:18)
at Object.<anonymous> (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/app.js:9:17)
at Module._compile (internal/modules/cjs/loader.js:776:30)
at Object.Module._extensions..js (internal/modules/cjs/loader.js:787:10)
at Module.load (internal/modules/cjs/loader.js:653:32)
at tryModuleLoad (internal/modules/cjs/loader.js:593:12)
at Function.Module._load (internal/modules/cjs/loader.js:585:3)
at Module.require (internal/modules/cjs/loader.js:690:17)
npm ERR! code ELIFECYCLE
npm ERR! errno 1
npm ERR! learn-sequelize@0.0.0 start: `node ./bin/www`
npm ERR! Exit status 1
npm ERR!
npm ERR! Failed at the learn-sequelize@0.0.0 start script.
npm ERR! This is probably not a problem with npm. There is likely additional logging output above.

npm ERR! A complete log of this run can be found in:
npm ERR! /Users/jiyoonkim/.npm/_logs/2019-07-16T01_13_15_847Z-debug.log
9달 전
config/config.json에 오타가 있다고 나오네요. 에러메시지를 잘 보시면 에러의 원인을 알 수 있습니다
9달 전
config에 } syntax error가 있다고 떠서 봤었는데 {} 다 수가 맞아요ㅠ 그리고 깃헙에 올려놓으신 자료 그대로 카피해서 붙여넣어도 봤는데 그때는 이런 에러가 뜨더라구요.
> learn-sequelize@0.0.0 start /Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize
> node ./bin/www

(node:84998) [SEQUELIZE0004] DeprecationWarning: A boolean value was passed to options.operatorsAliases. This is a no-op with v5 and should be removed.
Unhandled rejection SequelizeConnectionError: Client does not support authentication protocol requested by server; consider upgrading MySQL client
at Promise.tap.then.catch.err (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/sequelize/lib/dialects/mysql/connection-manager.js:133:19)
at tryCatcher (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/util.js:16:23)
at Promise._settlePromiseFromHandler (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/promise.js:517:31)
at Promise._settlePromise (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/promise.js:574:18)
at Promise._settlePromise0 (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/promise.js:619:10)
at Promise._settlePromises (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/promise.js:695:18)
at _drainQueueStep (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/async.js:138:12)
at _drainQueue (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/async.js:131:9)
at Async._drainQueues (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/async.js:147:5)
at Immediate.Async.drainQueues [as _onImmediate] (/Users/jiyoonkim/Desktop/javascript/nodejs/zerocho/chapter6/learn-sequelize/node_modules/bluebird/js/release/async.js:17:14)
at runCallback (timers.js:705:18)
at tryOnImmediate (timers.js:676:5)
at processImmediate (timers.js:658:5)




그래서 operatorsAlias 라인만 지웠을때도 처음에 나왔던 syntax }가 뜨구요ㅠㅠ
9달 전
이건 다른 에러입니다. MySQL 5.7을 설치하세요. 8버전을 설치한다면 책 내용을 그대로 따라하셔야 합니다.
9달 전
자꾸 질문해서 죄송한데요ㅠ 5.7 설치되어 있어요. 안되서 8도 해봤는데 안되구요. 진도를 나가야하는데 여기서 막혀버리니 답답하네요ㅠ 에러가 저 두개 중 하나씩 번갈아 나네요.
9달 전
이 경우에는 https://github.com/ZeroCho/nodejs-book/issues/45 참조하세요. 만약 안 된다면 MySQL 삭제 후 재설치 하시고 다시 저 링크대로 해보세요.
9달 전
헐헐헐 돼요 이제 돼요!!!ㅠㅠㅠㅠ 정말 감사합니다
9달 전
그리고 혹시 operatorAliases 지울 때 위에 dialect: "mysql", 여기 콤마도 지우셨나요? 콤마 있으면 안 됩니다.
9달 전
multer모듈로 이미지와 게시글을 업로드하는 부분에서 이미지가 이미지 아이콘형태로 출력됩니다 원본 이미지 형태로 안나오고요 이 부분은 에러 메시지도 안나오네요 ㅜㅜ
9달 전
이미지 아이콘이 어떤 거죠?
9달 전
https://www.google.com/imgres?imgurl=https%3A%2F%2Fctl.s6img.com%2Fsociety6%2Fimg%2FXG0OlNwi-sp-8Dsh8pYRR1HJtGY%2Fw_700%2Fprints%2F~artwork%2Fs6-original-art-uploads%2Fsociety6%2Fuploads%2Fmisc%2F58b052f474184211b7fcb694e0888c2a%2F~~%2Fgoogle-chrome-broken-image-prints.jpg%3Fwait%3D0%26attempt%3D0&imgrefurl=https%3A%2F%2Fsociety6.com%2Fproduct%2Fgoogle-chrome-broken-image_print&docid=k5N2UEG0i7QuwM&tbnid=M_-c-LKpTHBQ3M%3A&vet=10ahUKEwie2ovO1LPjAhVXxIsBHUBQB_sQMwhDKAMwAw..i&w=700&h=700&hl=ko&bih=917&biw=1295&q=broken%20img&ved=0ahUKEwie2ovO1LPjAhVXxIsBHUBQB_sQMwhDKAMwAw&iact=mrc&uact=8
링크 주소를 적어놨습니다. 찾아보니 브로큰 이미지라 부르네요. 업로드 시 브로큰 이미지로 출력이 됩니다.
5달 전
혹시 어떻게 해결하셨나요??ㅠㅠ 저도 같은 문제인데..
5달 전
static 미들웨어 설정이 제대로 되어있는지 확인해보세요. upload 폴더를 가리켜야 합니다.
9달 전
문제 해결했는데 댓글 삭제 기능이 안되네요 질문이 유익하지 않아서 지울려했는데....
9달 전
로컬 로그인 부분에서 계정을 생성하고 로컬 로그인을 시도했지만 db에 계정 등록은 되고 웹상에서는 로그인이 되지 않은채 User.find has a function 메시지가 출력되면서 다른 페이지 이동도 안되는 현상이 일어납니다. 시퀄라이즈 5버전에서 find를 findOne으로 바꿔줘서 등록 유무 확인까지 해봤습니다. 이 이상은 어떻게 접근해야 해결할지 모르겠어서 질문드립니다.
9달 전
정확한 에러 메시지를 알려주셔야 합니다.에러 메시지 상 find를 계속 쓰신 것 같습니다. 모두 findOne으로 바꿔주세요.
9달 전
해결해서 댓글을 삭제하려했는데 본인이 삭제를 할수가없네요
10달 전
11.4 과정 질문입니다.
다른 부분은 다 의도한대로 동작하는데,
입장과 퇴장의 system 메시지가 user를 undefined로 표시합니다.
따라서 undefined님이 입장하셨습니다.와 같은 system 메시지가 표시됩니다.
req.session.color 값이 없어서 그런 것 같은데요.
app.js나 routes/index.js에서 console.log(req.sessionID)를 찍은 결과와
socket.js의 chat.on에서 console.log(req.sessionID)를 찍은 결과가 같지 않습니다.
소켓 연결 시 세션을 새로 만들어버려 기존에 입력한 session.color 값을 읽지 않는 것 같은데
제가 어디를 잘못 생각하고 있는 걸까요?

교재 깃헙에서 11.4 코드를 그대로 다운받고 .env를 수정하여 실행해도 같은 결과입니다.
10달 전
io.use sessionMiddleware 부분에서 sessionMiddleware가 재사용된 것이 맞죠? 새로 생성한 경우는 다른 세션이 생겨버릴 수 있습니다.
10달 전
네 app.js에서 webSocket에 인자로 넘겨주어 사용했습니다.

또한 교재 깃헙의 코드를 그대로 받아 사용해도 같은 결과가 발생합니다.
10달 전
희한하네요 제가 11.4를 방금 해봤는데 정상적으로 작동합니다. .env의 COOKIE_SECRET 값도 주신 것 맞죠? 프로젝트를 완전히 삭제 후 다시 해보세요. 만약 그래도 안 된다면 코드를 zerohch0@gmail.com으로 보내주세요.
10달 전
.env의 mongodb 비밀번호만 바꾸고 나머지는 교재 깃헙에서 다운받은 코드 그대로이기 때문에 보내드리는 건 의미가 없을 것 같습니다.

어쩌다보니 해결이 되었는데 pug 파일들의 localhost 부분을 공유기에서 할당해준 내부 ip인 192.168.0.x로 변경하고, 브라우저에서도 localhost:8005나 127.0.0.1:8005가 아닌 192.168.0.x:8005로 접속하니 정상적으로 동작합니다. 루프백으로 접속하는 것과 어떤 차이가 있는지 잘 모르겠습니다.
10달 전
최대한 간소화 한
다른 주소로 이동하는 리스너 버튼 사용법은 게시 해 주실 수 있나요?
PUG로 구현하려는데 문법을 몰라서 헤매고있습니다.. ㅠㅠ
PUG에서는 script 안에서 함수 구현을 못 하나요?
10달 전
script. 으로 script 뒤에 점을 붙이면 다음 줄부터 js를 사용할 수 있습니다. 다른 주소로 이동하는 것은 location.href = 주소; 로 가능합니다.
일 년 전
안녕하세요. 9장에서 로그인을 하려고하면 로그인이 안되는데 혹시 어디가 문제인지 봐주실 수 있을까요?
1. bcrypt대신 bcyrptjs 를 사용했습니다. <- bcrypt가 저의 컴퓨터에서는 안 깔리더군요
2. find() 대신 findOne() Function을 사용했습니다.
3. 소스 코드를 이용해서 작동했을 때는 잘 넘어갔습니다만, 제가 직접 코드를 쓰고싶었습니다.


/*****************auth.js************************/
const express = require('express');
const passport = require('passport');
const bcrypt = require("bcryptjs");
const { isLoggedIn, isNotLoggedIn } = require('./middlewares');
const { User } = require('../models');

const router = express.Router();
//auth/join
router.post('/join', isNotLoggedIn, async(req, res, next) => {
const { email, nick, password } = req.body;
try {
const exUser = await User.findOne({ where: { email } });
if(exUser) {
req.flash('joinError','This email is already registered');
return res.redirect('/join'); // Back to Register page
}
const hash = await bcrypt.hash(password, 12);
await User.create({
email,
nick,
password: hash
})
return res.redirect('/');
}
catch(error) {
console.error(error);
return next(error);
}
});
// auth/login
router.post('/login', isNotLoggedIn, (req, res, next) => {
passport.authenticate('local', (authError, user, info) => {
if (authError) {
console.error(authError);
return next(authError);
}
if (!user) {
req.flash('loginError', info.message);
return res.redirect('/');
}
return req.login(user, (loginError) => {
if (loginError) {
console.error(loginError);
return next(loginError);
}
return res.redirect('/');
});
})(req, res, next);
});

// auth/logout
router.get('/logout', isLoggedIn, (req, res) => {
req.logout();
req.session.destory(); //Session destory
res.redirect('/');
});

router.get('/kakao', passport.authenticate('kakao'));

router.get('/kakao/callback', passport.authenticate('kakao', {
failureRedirect: '/',
}), (req, res) => {
res.redirect('/');
});

module.exports = router;


/****************localStrategy.js************************/

const LocalStrategy = require('passport-local').Strategy;
const bcrypt = require("bcryptjs");

const { User } = require('../models');

module.exports = (passport) => {
passport.use(new LocalStrategy({
usernameField: 'email',
passwordField: 'password',
}, async (email, password, done) => {
try {
const exUser = await User.findOne({ where: { email } });
if (exUser) {
const result = await bcrypt.compare(password, exUser.password);
if (result) {
done(null, exUser);
} else {
done(null, false, { message: 'Id or password is wrong' });
}
} else {
done(null, false, { message: 'User is not registered' });
}
} catch (error) {
console.error(error);
done(error);
}
}));
};
일 년 전
소스 코드가 아니라 에러 메시지를 올려주셔야 합니다.
일 년 전
12장 auction.pug에서
block good
h2= good.name
div= '등록자: ' + good.owner.nick
div= '시작가: ' + good.price + '원'
strong#time(data-start=good.createdAt data-end=good.end)
img#good-img(src='/img/' + good.img)

script.
...
var es = new EventSource("/sse");
var time = document.querySelector('#time');
es.onmessage = function (e) {
var end = new Date(time.dataset.start);
var server = new Date(parseInt(e.data, 10));
end.setMinutes(end.getMinutes() + parseInt(time.dataset.end));
if (server >= end) {
return time.textContent = '00:00:00';
} else {
var t = end - server;
var seconds = ('0' + Math.floor((t / 1000) % 60)).slice(-2);
var minutes = ('0' + Math.floor((t / 1000 / 60) % 60)).slice(-2);
var hours = ('0' + Math.floor((t / (1000 * 60 * 60)) % 24)).slice(-2);
return time.textContent = hours + ':' + minutes + ':' + seconds;
}
};
위 코드 중에서 var end = new Date(time.dataset.start);에서 계속
Uncaught TypeError: Cannot read property 'dataset' of null at EventSource.es.onmessage
이 에러가 계속 뜨는데 어떻게 해야할까요 ㅠㅠ
일 년 전
var time = document.querySelector('#time');을 es.onmessage 안으로 넣어보세요.
일 년 전
9장과 10장에서 module 정의할때 module.export = (,) => () 로 되어있더라구요

저는 9장에서 user는 (,)=>()로 하고 오류가 없었는데 10장에서 domain을 (,)=>{}로 하니까 db.user.hasmany(db.domain) 이 오류가 나서 한참 찾았습니다.
근데 둘다 (,)=>()로 하니까 오류가 안나네요
람다함수(화살표함수)를 (...params) => () 로 하는 경우도 있나요? 영역은 {}로만 지정하는줄 알았는데 확인좀 부탁드려요
일 년 전
네 바로 리턴하는 함수입니다. => ()로 하면 그 내용물이 바로 리턴됩니다.
일 년 전
sequelize를 mariadb와 연동하기 위해서는 어떻게 해야하나요?
sequelize db:create하면
ERROR: Dialect mariadb does not support db:create / db:drop commands
이런 오류가 발생합니다.

npm i mariadb 했는데도 안되네요
일 년 전
mariadb는 sequelize db:create를 지원하지 않습니다. 워크벤치나 SQL문으로 데이터베이스를 생성하셔야겠네요.
일 년 전
400p를 공부하고 있습니다.
user.find 에서 id를 찾는데 db에 id값이 없어서 그런지
WHERE parameter "id" has invalid "undefined" value
라고 계속 뜹니다.
현재 sequelize 5버전이고 findOne으로 수정해서 하고있습니다.
제가 오타가 있나 싶어서 전부다 복붙하고 find -> findOne으로 변경후 진행중인데 계속 이렇네요 ㅠㅠㅠㅠ
일 년 전
이 부분은 https://github.com/ZeroCho/nodejs-book/issues/42 참고하세요. 시퀄라이즈 5버전에서 많은 게 바뀌었습니다 ㅠㅠ
일 년 전
안녕하세요. 제로님.

질문이 더 있어 코멘트 남깁니다.

1. 200p ~ 201p

cookie-parser 내용입니다.

첫번째 인자로 문자열을 넣을 경우 서명 된 쿠키가 된다고 하였는데, 실제로 다음과 같이 체크 해보니.


console.log(req.signedCookies);


서명 된 쿠키는 connect.id 하나만 존재하였습니다.


실제로는 다른 쿠키들도 설정하여 사용하고 있습니다.


정확히 어떻게 작용하는 것인지 궁금합니다.


예로 서명 된 쿠키를 전달 해주기 위해서는 다음과 같이 해야 한다는데.


다음과 같이 설정 할 경우 서명 된 사인이 일치하여 정상적으로 작동하는것이 아닌가해서요.


단순히 인자값을 넣는 것으로 서명 된 쿠키로 작동하는게 아닌 것 같아서요 ^^;


res.cookie('name', 'sign value', {signed: true});
일 년 전
서명된 쿠키가 오면 그 값들을 자동으로 파싱하는 겁니다. 인자로 주어진 키로 복호화해서요.
일 년 전
제로님. 답변 감사합니다.

말씀하신대로 서명 된 쿠키가 오면 그 값들을 자동으로 파싱한다면,

다음의 코드에는 왜 connect.sid 하나만 들어있는 걸까요.

서명 된 다른 쿠키들도 보여야 하는게 정상아닐까요.

많이 헷갈리네요 궁금한것들도 많구요 ^^;

console.log(req.signedCookies);
일 년 전
서명된 다른 쿠키가 어떤 것이 있나요?
일 년 전
제로님. 이제 이해하였습니다.


cookie-parser 미들웨어의 역할은 전달된 쿠키들을 첫번째 인자의 문자열로 파싱하는 것이였군요.


만약 서명 된 쿠키들을 세팅 하려면 {signed: true} 로 설정하면 되는 군요.


그래서 클라이언트에서 조작을 방지 할 수 있다고 하셨군요.


그런데 책에는 '이제 쿠키들은 제공한 문자열로 서명된 쿠키가 됩니다.' 라고 되어 있어.


쿠키를 세팅하기만 하면 첫번째 인자의 문자열로 서명되는 것처럼 마치 설명 되어 있어서요.


이 부분은 헷갈릴 수 있는 것 같습니다. ^^;;
일 년 전
감사합니다. 그 부분은 명확하게 수정하도록 하겠습니다.
일 년 전
안녕하세요. 제로초님 챕터9 SNS 만들기 부분에서 (예제 파일 9.4) TypeError: User.find is not a function 오류가 발생합니다. 코드를 github에서 다운로드해서 해봤는데도 오류가 납니다. 코드 비교도 해봤고 에러 나는 부분 들어가서 봤습니다. User.find({ where: { email } }); 문제인 거 같은데.. 잘 모르겠습니다. MySQL에서 users로 테이블 만들어져있습니다.
일 년 전
정요표에 적어둔대로 시퀄라이즈 4버전을 설치하거나 find를 findOne으로 수정하셔야 합니다. 현재 시퀄라이즈 5버전 설치하신 것 같네요.
일 년 전
감사합니다!
일 년 전
3.6 파일시스템 접근에서 console.log(data.toString()); 과 불러올때 utf8 을 붙이면 같은 결과가 나오는데 무슨 차이인가요?
일 년 전
utf8같은 것은 버퍼를 어떤 인코딩으로 해석할지를 나타내는데요. 기본적으로 utf8이라서 붙이나 안 붙이나 같은 결과가 나오는 것 같습니다.
일 년 전
흠..그럼 utf8 안붙이고 toString만 붙여도 되는거죠?
일 년 전
아마도 그럴겁니다.
일 년 전
안녕하세요 제로님!
이번엔 다름이 아니라 9.5.1에서 팔로잉 끊기를 실행해보려고하는데요.
Follow라는 테이블에 있는 값을 삭제해야하는데 어떻게 접근을 해야할지 감이안오네요.
User의 값을 삭제할때는 User.destroy({where: {id}}); 이런식으로 접근을 하면되는데
delete/:id/follow 이러한 api로 작성하려고하는데
await Follow.destroy({ where: { followerId: req.user.id, followingId: req.params.id}});
이러자니 Follow를 알지못해서 어떻게 진행하면될지 알려주시면 감사하겠습니다!!
일 년 전
먼저 현재 사용자를
const user = await User.findOne({ where: { id: req.user.id } });
로 찾은 뒤
await user.removeFollowing(req.params.id); 이런 식으로 하시면 됩니다. 다대다 관계의 가운데 테이블은 직접 접근할 수가 없어서요.
일 년 전
아아!! 책에는 destroy사용하라고 하셔서!! 계속 이걸로해보고있었는데!! 감사합니다1!!!!
일 년 전
안녕하세요..
9장의 find() 를 findOne()으로 변경하고, 9.3장 까지는 잘 수행되었는데,
9.4 ~ 9.5 까지의 소스를 Key-in하로 실행 시키면 아래 오류가 나은데 원인이 무엇일까요?

<화면상에 나타난 오류>

user is associated to user multiple times. To identify the correct association, you must use the 'as' keyword to specify the alias of the association you want to include.

SequelizeEagerLoadingError: user is associated to user multiple times. To identify the correct association, you must use the 'as' keyword to specify the alias of the association you want to include.
at Function._getIncludedAssociation (C:\DevNode\cafe24_nodejs_mysql\pmsadmin02_0402\node_modules\sequelize\lib\model.js:728:13)
일 년 전
key in하로 실행한다는 게 어떤 뜻인가요? as 키워드가 include할 때 빠지지 않았는지 살펴보세요.
일 년 전
363P에서 User간의 self join을 할때
db.User.belongsToMany(db.User, {
foreignKey: 'followingId',
as: 'Followers',
through: 'Follow'
});

db.User.belongsToMany(db.User, {
foreignKey: 'followerId',
as: 'Followings',
through: 'Follow'
});

as를 아래처럼하는 것이 아닌 위에처럼 하는건지가 궁금해서 문의남겼습니다!

db.User.belongsToMany(db.User, {
foreignKey: 'followingId',
as: 'Followings',
through: 'Follow'
});

db.User.belongsToMany(db.User, {
foreignKey: 'followerId',
as: 'Followers',
through: 'Follow'
});
일 년 전
foreignKey와 as는 반대 관계입니다. 팔로워는 팔로잉아이디를 받고, 팔로잉은 팔로워아이디를 받습니다.
일 년 전
그렇다면 Follow라는 테이블에 col명은 어떻게 정해지는건가요?
일 년 전
as에 적힌 문자열에 primaryKey를 붙여서 생성됩니다.
일 년 전
아!!! 너무나도감사합니다1!!
일 년 전
안녕하세요. 제로님. 책 잘보고 있습니다. 다름이 아니라 한 가지 궁금한점이 있어 질문 남깁니다.


1. 456p ~ 457p

- 해당 페이지의 1번 설명을 살펴보면 세션을 공유하기 위해 웹 소켓에서도 미들웨어를 사용하여 세션을 공유한다고 되어 있습니다.

그런데 궁금한점은 sessionMiddleware(socket.request, socket.request.res, next); 이렇게 세가지의 인자를 보내주고 있는데, 세션 미들웨어 변수를 살펴보면 해당 인자를 받는 부분이 없는데 어떻게 보내 줄 수 있는지 궁금합니다.

그리고 추가적으로 궁금한점은 세션을 웹소켓에서 미들웨어로 사용하면 그냥 공유를 할 수 있는 것인가요? 해당 페이지의 설명이 조금 부족한 것 같습니다 ㅜ.
일 년 전
모든 미들웨어는 인자로 (req, res, next)를 받습니다. 웹소켓에서 미들웨어로 사용하면 socket.request.session이 생깁니다. 세션미들웨어의 역할이 req 객체에 session 객체를 넣어주는 겁니다.
일 년 전
답변 감사합니다. 제로님.

app.use() 에서는 인자들을 받는 예제는 많이 보았는데,

미들웨어 자체에서 사용하는 모습은 처음보았습니다.

보통 미들웨어 세팅 시 다음처럼 설정 후 사용하는 것 같아서요.

app.use(sessionMiddleware);

또 하나 배워갑니다.
일 년 전
아 그 내용은 426쪽에 나와 있습니다.
일 년 전
제로님. 감사합니다.

372페이지의 2번 설명을 살펴보면 되는군요 미들웨어에서 if 문 처럼 작동방식을 커스터마이징 하기 위해서 예제와 같이 사용하는군요. 감사합니다!