Node

(Node.js) password 암호화 모듈 Crypto(crypto.js에 대해)

곽빵 2019. 11. 1. 16:55

Cipher : 일반적인 암호화로 평문 -> 암호문 -> 복호문(평문)의 방식으로 암호화하고 복호화 합니다.

             즉, 암호화를 하기 위한 Key 를 가지고 암호화하며, 복호화합니다.

 

Hmac: 암호화를 하면 다시 되돌릴 수 없기 때문에, 암호화 된 암호문 그 자체를 이용합니다. 흔히

      웹사이트 가입하면 운영자도 가입정보를 알 수 없다고 하는건 이렇게 Hmac으로 암호화되어서 입니다.

 

Module 가져 온뒤, 사용해보겠습니다.

 

1.Cipher 방식을 이용한 암호화

var crypto = require('crypto');

// 암호화
var cipher = crypto.createCipher('aes256','password'); 
// aes256 방식으로 password(key)로 암호화하겠다.
cipher.update('TestCipher','ascii','hex');
// update('평문(암호화이전의글)','input type','output type');
var cipherd = cipher.final('hex');
// 암호문

//복호화
var decipher = crypto.createDecipher('aes256','password');
decipher.update(cipherd,'hex','ascii');
var decipherd = decipher.final('ascii');
// 암호화 방법의 parameter 방식은 동일함




 

 

2.Hmac 방식을 이용한 암호화

 

UserSchema
        .virtual('password')
        .set(function (password) {
            this._password = password;
            this.salt = this.makeSalt();
            this.hashed_password = this.encryptPassword(password);
            console.log('virtual password 호출됨 : ' + this.hashed_password);
        })
        .get(function () {
            return this._password
        });

    // 스키마에 모델 인스턴스에서 사용할 수 있는 메소드 추가
    // 비밀번호 암호화 메소드
    UserSchema.method('encryptPassword', function (plainText, inSalt) {
        if (inSalt) {
            return crypto.createHmac('sha1', inSalt).update(plainText).digest('hex');
        } else {
            return crypto.createHmac('sha1', this.salt).update(plainText).digest('hex');
        }
    });

    // salt 값 만들기 메소드
    UserSchema.method('makeSalt', function () {
        return Math.round((new Date().valueOf() * Math.random())) + '';
    });

    // 인증 메소드 - 입력된 비밀번호와 비교 (true/false 리턴)
    UserSchema.method('authenticate', function (plainText, inSalt, hashed_password) {
        // plainText가 사용자가 입력한 password이고 hashed_password가 암호화되서 저장된 hashed_password 이다.
        if (inSalt) {
            console.log('authenticate 호출됨 : %s -> %s : %s', plainText, this.encryptPassword(plainText, inSalt), hashed_password);
            return this.encryptPassword(plainText, inSalt) === hashed_password;
        } else {
            console.log('authenticate 호출됨 : %s -> %s : %s', plainText, this.encryptPassword(plainText), this.hashed_password);
            return this.encryptPassword(plainText) === this.hashed_password;
        }
    });

virtual 을 이용해서 Hmac 암호화 방식을 사용해 보았습니다.

virtual 은 MongoDB에 저장되지 않는 속성이므로 편하게 set get을 설정 편의를 위해서 사용하는 것 같습니다.

(사실 왜 쓰는지는 모르겠지만, 개인적인 생각으로는 좀더 유연한 Schema가 되기위한 기능이라고 생각합니다..

정해진 컬렉션 속성들로 여러가지 정보를 추출 비교 하기 위해 만들어진..?)

 

실제로 passport 인증 모듈에서 password 인증할 때 사용해 보았습니다.

 

module.exports = new LocalStrategy({
    usernameField: 'email',
    passwordField: 'password',
    passReqToCallback: true // 이 옵션을 설정하면 아래 콜백 함수의 첫번째로 req 전달
}, function (req, email, password, done) {
    console.log('passport의 local-login 호출됨 : ' + email + ', ' + password);

    var database = req.app.get('database');
    database.UserModel.findOne({
        'email': email
    }, function (err, user) {
        console.log('-----------------로그인----------------- ');
        console.dir(user);// 해당 model 정보 출력
        if (err) {
            return done(err);
        }

        // 등록된 사용자가 없는경우
        if (!user) {
            console.log('계정이 일치하지 않음.');
            return done(null, false, req.flash('loginMessage', '등록된 계정이 없습니다.'));
            // 인증 실패 처리
        }
        // 비밀번호 비교하여 맞지 않는 경우
        var authenticated =
            user.authenticate(password, user._doc.salt, user._doc.hashed_password);
        if (!authenticated) {
            console.log('비밀번호 일치하지 않음');
            return done(null, false, req.flash('loginMessage', '비밀번호가 일치하지 않음'));
        }
        console.log('계정과 비밀번호가 일치함.');
        return done(null, user);
        // 인증 성공 처리 serializeUser에 user 정보 전달
    })
});