특히 이것은 클라이언트 세션 쿠키를 사용하여 서버에서 세션을 식별 할 때 관련됩니다.
전체 웹 사이트에 SSL / HTTPS 암호화를 사용하는 가장 좋은 답변이며 중간 공격이 기존 클라이언트 세션 쿠키를 스니핑 할 수 없다는 것을 가장 잘 보장 할 수 있습니까?
그리고 세션 쿠키에 저장된 세션 값 자체에 일종의 암호화를 사용하는 것이 두 번째로 좋은 방법일까요?
악의적 인 사용자가 시스템에 물리적으로 액세스 할 수있는 경우에도 파일 시스템을 조사하여 유효한 세션 쿠키를 검색하고이를 사용하여 세션을 탈취 할 수 있습니까?
답변
세션 값을 암호화하면 효과가 없습니다. 세션 쿠키는 이미 임의의 값이므로 암호화하면 스니핑 될 수있는 다른 임의 값이 생성됩니다.
유일한 실제 솔루션은 HTTPS입니다. 전체 사이트에서 SSL을 수행하고 싶지 않다면 (성능 문제가있을 수 있음) 민감한 영역을 보호하는 SSL만으로 벗어날 수 있습니다. 그렇게하려면 먼저 로그인 페이지가 HTTPS인지 확인하십시오. 사용자가 로그인 할 때 일반 세션 쿠키 외에 보안 쿠키 (브라우저가 SSL 링크를 통해서만 전송 함)를 설정합니다. 그런 다음 사용자가 “민감한”영역 중 하나를 방문하면 HTTPS로 리디렉션하고 보안 쿠키가 있는지 확인합니다. 실제 사용자는이를 갖고 세션 하이재커는 그렇지 않습니다.
편집 :이 답변은 원래 2008 년에 작성되었습니다. 현재 2016 년이며 전체 사이트에 SSL을 사용하지 않을 이유가 없습니다. 더 이상 일반 텍스트 HTTP가 아닙니다!
답변
SSL은 스니핑 공격에만 도움이됩니다. 공격자가 귀하의 컴퓨터에 액세스 할 수있는 경우 귀하의 보안 쿠키도 복사 할 수 있다고 가정합니다.
최소한 오래된 쿠키가 잠시 후에 가치를 잃도록하십시오. 쿠키가 작동을 멈 추면 성공적인 hijaking 공격도 차단됩니다. 사용자가 한 달 이상 전에 로그인 한 세션의 쿠키가있는 경우 비밀번호를 다시 입력하도록합니다. 사용자가 사이트의 “로그 아웃”링크를 클릭 할 때마다 이전 세션 UUID를 다시 사용할 수 없도록하십시오.
이 아이디어가 효과가 있는지 확실하지 않지만 여기에 있습니다. 세션 쿠키에 일련 번호를 추가합니다. 아마도 다음과 같은 문자열 일 수 있습니다.
SessionUUID, 일련 번호, 현재 날짜 / 시간
이 문자열을 암호화하고 세션 쿠키로 사용하십시오. 일련 번호를 정기적으로 변경하십시오. 쿠키가 5 분 정도 지난 후 쿠키를 재발행하십시오. 원하는 경우 모든 페이지보기에서 재발행 할 수도 있습니다. 서버 측에서 해당 세션에 대해 발행 한 마지막 일련 번호를 기록해 둡니다. 누군가가 잘못된 일련 번호로 쿠키를 보낸 경우 공격자가 이전에 가로 챈 쿠키를 사용하고있을 수 있으므로 세션 UUID를 무효화하고 사용자에게 암호를 다시 입력 한 다음 새 쿠키를 다시 발급하도록 요청합니다.
사용자는 둘 이상의 컴퓨터를 가질 수 있으므로 둘 이상의 활성 세션을 가질 수 있습니다. 컴퓨터 사이를 전환 할 때마다 다시 로그인하도록 강요하는 작업을하지 마십시오.
답변
PHP 보안에 관한 책을 읽어 보셨나요? 추천.
비 SSL 인증 사이트에 대해 다음 방법으로 많은 성공을 거두었습니다.
-
동일한 계정에서 여러 세션을 허용하지 않고 IP 주소로만 확인하지 않도록합니다. 오히려 데이터베이스에 사용자 세션과 함께 저장되는 로그인시 생성 된 토큰과 IP 주소, HTTP_USER_AGENT 등으로 확인합니다.
-
관계 기반 하이퍼 링크 사용 링크 생성 (예 : http://example.com/secure.php?token=2349df98sdf98a9asdf8fas98df8 ) 링크는 페이지 리디렉션시 임의로 생성 된 x-BYTE (기본 크기) 임의 솔트 MD5 문자열과 함께 추가됩니다. 토큰은 요청 된 페이지에 해당합니다.
- 다시로드하면 몇 가지 검사가 수행됩니다.
- 원래 IP 주소
- HTTP_USER_AGENT
- 세션 토큰
- 당신은 요점을 얻습니다.
-
짧은 수명 세션 인증 쿠키. 위에 게시 된 것처럼 세션 유효성에 대한 직접적인 참조 중 하나 인 보안 문자열을 포함하는 쿠키는 좋은 생각입니다. x 분마다 만료되도록 만들고 해당 토큰을 다시 발급하고 세션을 새 데이터와 다시 동기화합니다. 데이터가 일치하지 않는 경우 사용자를 로그 아웃하거나 세션을 다시 인증하도록합니다.
저는이 주제에 대한 전문가가 아닙니다. 저는이 특정 주제에 대해 약간의 경험을 가지고 있습니다.이 중 일부가 누구에게나 도움이되기를 바랍니다.
답변
// Collect this information on every request
$aip = $_SERVER['REMOTE_ADDR'];
$bip = $_SERVER['HTTP_X_FORWARDED_FOR'];
$agent = $_SERVER['HTTP_USER_AGENT'];
session_start();
// Do this each time the user successfully logs in.
$_SESSION['ident'] = hash("sha256", $aip . $bip . $agent);
// Do this every time the client makes a request to the server, after authenticating
$ident = hash("sha256", $aip . $bip . $agent);
if ($ident != $_SESSION['ident'])
{
end_session();
header("Location: login.php");
// add some fancy pants GET/POST var headers for login.php, that lets you
// know in the login page to notify the user of why they're being challenged
// for login again, etc.
}
이것이하는 일은 사용자 세션에 대한 ‘문맥 적’정보, 단일 세션의 수명 동안 변경되지 않아야하는 정보를 캡처하는 것입니다. 사용자는 미국과 중국의 컴퓨터에 동시에 있지 않을 것입니다. 그렇죠? 따라서 세션 하이재킹 시도를 강하게 암시하는 동일한 세션 내에서 IP 주소가 갑자기 변경되면 세션을 종료하고 사용자가 다시 인증하도록하여 세션을 보호합니다. 이것은 해킹 시도를 방해하고 공격자는 세션에 대한 액세스 권한을 얻는 대신 강제로 로그인해야합니다. 사용자에게 시도를 알리고 (약간 ajax), vola, 약간 짜증이 나고 정보를 알고있는 사용자 및 세션 / 정보가 보호됩니다.
우리는 프록시 / 네트워크 뒤에있는 시스템에 대한 세션의 고유성을 캡처하기 위해 최선을 다하기 위해 User Agent 및 X-FORWARDED-FOR를 사용합니다. 그런 다음 더 많은 정보를 사용할 수있을 것입니다. 자유롭게 창의력을 발휘하십시오.
100 %는 아니지만 꽤 효과적입니다.
세션을 보호하고, 세션을 만료시키기 위해 할 수있는 일이 더 많습니다. 빈 HTTP_REFERER (도메인이 URL 표시 줄에 입력 됨)를 캡처하여 이탈하고 돌아 오는 사용자를 감지하거나 HTTP_REFERER의 값이 귀하의 도메인과 일치하는지 (사용자가 외부 / 제작 된 링크를 클릭하여 귀하의 대지).
세션을 만료하고 무기한 유효하게 유지하지 마십시오.
쿠키에 의존하지 마십시오. 쿠키는 도난 당할 수 있으며 세션 하이재킹을위한 공격 벡터 중 하나입니다.
답변
설명 안전한 쿠키 프로토콜보십시오 이Liu, Kovacs, Huang 및 Gouda 백서 .
문서에 명시된대로 :
클라이언트와 서버간에 실행되는 보안 쿠키 프로토콜은 인증, 기밀성, 무결성 및 재생 방지라는 네 가지 서비스를 제공해야합니다.
배포의 용이성 :
효율성 측면에서 우리의 프로토콜은 데이터베이스 조회 또는 공개 키 암호화를 포함하지 않습니다. 배포 가능성 측면에서 당사의 프로토콜은 기존 웹 서버에 쉽게 배포 할 수 있으며 인터넷 쿠키 사양을 변경할 필요가 없습니다.
간단히 말해서 안전하고 가벼우 며 저에게 아주 좋습니다.
답변
세션 hijaking을 100 % 방지 할 수있는 방법은 없지만 몇 가지 접근 방식을 사용하면 공격자가 세션을 hijaking하는 시간을 줄일 수 있습니다.
세션 hijaking 방지 방법 :
1-항상 SSL 인증서로 세션을 사용합니다.
2-httponly가 true로 설정된 경우에만 세션 쿠키 보내기 (JavaScript가 세션 쿠키에 액세스하지 못하도록 방지)
2-로그인 및 로그 아웃시 세션 재생성 ID 사용 (참고 : 연속적인 ajax 요청이있는 경우 여러 세션을 생성 할 수 있기 때문에 각 요청에서 세션 재생성을 사용하지 마십시오.)
3-세션 시간 제한 설정
4-브라우저 사용자 에이전트를 $ _SESSION 변수에 저장하고 각 요청에서 $ _SERVER [ ‘HTTP_USER_AGENT’]와 비교합니다.
5-토큰 쿠키를 설정하고 해당 쿠키의 만료 시간을 0으로 설정합니다 (브라우저가 닫힐 때까지). 각 요청에 대해 쿠키 값을 다시 생성합니다 (Ajax 요청의 경우 토큰 쿠키를 다시 생성하지 마십시오). 전의:
//set a token cookie if one not exist
if(!isset($_COOKIE['user_token'])){
//generate a random string for cookie value
$cookie_token = bin2hex(mcrypt_create_iv('16' , MCRYPT_DEV_URANDOM));
//set a session variable with that random string
$_SESSION['user_token'] = $cookie_token;
//set cookie with rand value
setcookie('user_token', $cookie_token , 0 , '/' , 'donategame.com' , true , true);
}
//set a sesison variable with request of www.example.com
if(!isset($_SESSION['request'])){
$_SESSION['request'] = -1;
}
//increment $_SESSION['request'] with 1 for each request at www.example.com
$_SESSION['request']++;
//verify if $_SESSION['user_token'] it's equal with $_COOKIE['user_token'] only for $_SESSION['request'] > 0
if($_SESSION['request'] > 0){
// if it's equal then regenerete value of token cookie if not then destroy_session
if($_SESSION['user_token'] === $_COOKIE['user_token']){
$cookie_token = bin2hex(mcrypt_create_iv('16' , MCRYPT_DEV_URANDOM));
$_SESSION['user_token'] = $cookie_token;
setcookie('user_token', $cookie_token , 0 , '/' , 'donategame.com' , true , true);
}else{
//code for session_destroy
}
}
//prevent session hijaking with browser user agent
if(!isset($_SESSION['user_agent'])){
$_SESSION['user_agent'] = $_SERVER['HTTP_USER_AGENT'];
}
if($_SESSION['user_agent'] != $_SERVER['HTTP_USER_AGENT']){
die('session hijaking - user agent');
}
참고 : ajax 요청으로 토큰 쿠키를 재생성하지 마십시오. 참고 : 위 코드는 예입니다. 참고 : 사용자가 로그 아웃하면 쿠키 토큰과 세션이 삭제되어야합니다.
6-일부 사용자의 IP는 각 요청에 따라 변경되기 때문에 세션 히잡을 방지하기 위해 사용자 IP를 사용하는 것은 좋은 방법이 아닙니다. 유효한 사용자에게 영향을 미치는
7-개인적으로 세션 데이터를 데이터베이스에 저장합니다. 어떤 방법을 채택할지는 귀하에게 달려 있습니다.
내 접근 방식에서 실수를 발견하면 나를 수정하십시오. 세션 hyjaking을 방지하는 더 많은 방법이 있으면 알려주세요.
답변
세션 ID에 증분 정수를 사용하지 마십시오. GUID 또는 임의의 긴 문자열을 사용하는 것이 훨씬 좋습니다.