I.lib()/I.lib(Java)

[암호화] 메시지 다이제스트(Message Digest)

.07274. 2012. 7. 6. 14:26

[펌] http://blog.naver.com/xxrcn11?Redirect=Log&logNo=20135507694

 

자세한 내용을 알고 싶은 분은 다음의 블로그를 참조하세요

http://blog.naver.com/dury00?Redirect=Log&logNo=130044978311

내용

가) 일종의 데이터 지문

나) 데이터가 바뀌지 않았다는 것을 증명

다) 데이터의 길이에 상관없이 보통 16 또는 20byte

라) FTP파일 검증에 많이 사용

마) 해쉬를 이용

바) 한 방향 변환이고 역으로는 불가능

자바에서의 메시지 다이제스트 사용

가) java.security.MessageDigest

getInstance()로 인자 생성

MD5, SHA-1이 주로 사용됨

나) update()

데이터를 해쉬 함

다) digest()

바이트 배열로 해쉬를 반환

적은 데이터일 경우 digest에 직접 입력 가능

라) 파일의 메시지 다이제스트 계산

import java.io.BufferedInputStream;

import java.io.FileInputStream;

import java.io.IOException;

import java.security.MessageDigest;

import java.security.NoSuchAlgorithmException;

import com.Ostermiller.util.Base64;

public class DigestFile {

public static void main(String[] args) throws NoSuchAlgorithmException, IOException {

MessageDigest md = MessageDigest.getInstance("MD5");

BufferedInputStream in = new BufferedInputStream(new FileInputStream("c:\\test.txt"));

int theByte = 0;

while((theByte = in.read()) != -1)

md.update((byte)theByte);

in.close();

byte[] theDigest = md.digest();

System.out.println(Base64.encode(theDigest));

}

}

마) 실행결과

test.txt 파일에는 ‘this is a test file’이 들어 있음 : [B@1a758

‘this is a test file.’로 수정한 결과 : [B@1b67f74

바) 스트림에서의 메시지 다이제스트

DigestInputStream & DigestOutputStream

MessageDigest messageDigest = MessageDigest.getInstance(“MD5”);

DigestOutputStream output = new

DigestOutputStream(existingOutputStream, messageDigest);

출력에 쓰는 모든 바이트가 자동으로 메시지 다이제스트를 계산할 때 사용됨

예제

import java.io.BufferedInputStream;

import java.io.FileInputStream;

import java.security.DigestInputStream;

import java.security.MessageDigest;

import com.Ostermiller.util.Base64;

public class DigestStreamExample {

public static void main(String[] args) throws Exception{

MessageDigest md = MessageDigest.getInstance("MD5");

BufferedInputStream in = new BufferedInputStream(new FileInputStream("c:\\test.txt"));

//DigestInputStream 생성한다.

DigestInputStream digestIn = new DigestInputStream(in, md);

// 모든 데이터를 읽어들인다.

while(digestIn.read() != -1);

byte[] theDigest = md.digest();

System.out.println(Base64.encode(theDigest));

}

}

패스워드 인증

가) 의미

패스워드를 해쉬하여 저장할 경우 복호화는 불가능하지만 모든 가능한 패스워드를 해쉬한 다음 비교해서 추즉하는게 가능함

이런 공격을 막기 위해 salt와 패스워드를 같이 저장해야 함

서버가 실제 패스워드를 알 필요 없이 사용자가 정한 패스워드를 검증하는게 목적

나) 패스워드 저장

랜덤 salt 생성하여 패스워드 앞에 붙임

이것을 해쉬함

패스워드 입력될 때마다 비교를 위해 salt와 해쉬 결과를 붙여서 저장



다) 패스워드 인증

저장된 패스워드의 salt와 사용자가 입력한 패스워드를 합친다.

salt와 패스워드를 해쉬한다.

결과 해쉬값을 파일에 저장된 것을 비교한다. 같다면 패스워드가 같은 것이다.

라) 예제

import java.io.ByteArrayOutputStream;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.security.MessageDigest;

import java.security.SecureRandom;

import java.util.Arrays;

public class Passwordauthenticator {

public static void main(String[] args) throws Exception{

createPassword("iloveyou");

authenticatePassword("iloveyou");

}

private static void createPassword(String password) throws Exception {

// salt 생성

SecureRandom random = new SecureRandom();

byte[] salt = new byte[12];

random.nextBytes(salt);

// salt 패스워드를 붙여서 해쉬한다.

MessageDigest md = MessageDigest.getInstance("MD5");

md.update(salt);

md.update(password.getBytes("UTF-8"));

byte[] digest = md.digest();

// salt 해쉬 결과를 파일에 저장한다.

FileOutputStream fos = new FileOutputStream("c:\\password.txt");

fos.write(salt);

fos.write(digest);

fos.close();

}

private static void authenticatePassword(String password) throws Exception {

// 파일을 읽어들인다.

ByteArrayOutputStream baos = new ByteArrayOutputStream();

FileInputStream fis = new FileInputStream("c:\\password.txt");

int theByte = 0;

while((theByte = fis.read()) != -1)

baos.write(theByte);

fis.close();

byte[] hashedPasswordWithSalt = baos.toByteArray();

baos.reset();

//salt 얻는다.

byte[] salt = new byte[12];

System.arraycopy(hashedPasswordWithSalt, 0, salt, 0, 12);

// salt 입력한 패스워드를 붙여서 해쉬한다.

MessageDigest md = MessageDigest.getInstance("MD5");

md.update(salt);

md.update(password.getBytes("UTF-8"));

byte[] digest = md.digest();

// 파일에 있는 내용과 비교한다.

byte[] digestInFile = new byte[hashedPasswordWithSalt.length-12];

System.arraycopy(hashedPasswordWithSalt, 12, digestInFile, 0, hashedPasswordWithSalt.length-12);

if(Arrays.equals(digest, digestInFile))

System.out.println("Password matches.");

else

System.out.println("Password does not matches.");

}

}