웹마스터 팁

Draggable Captcha System for XE1.7.5 Member Module

QapTcha for XpressEngine 1.7.5


스팸 회원가입을 차단하기 위한 드래그 버튼 방식의 캡챠를 XE 회원가입 폼에 적용하는 내용입니다.

코어 1.7.5 에서 테스트하였으며, 수정된 Member Module의 기본스킨(default) 파일을 첨부합니다. 

이전 버전, 또는 기본스킨 이외의 스킨사용으로 발생하는 오류 등에 대해서는 질문을 받지 않습니다.

오픈소스 플러그인을 포함하고 있으므로 비영리적 재배포만 무한자유로 가능합니다. 

Happy World Wide Web!


QapTcha-DEMO.jpg




1. QapTcha 언어 설정


<!--// Qaptcha Language items -->
	<item name="draggable_qaptcha">
		<value xml:lang="ko"><![CDATA[캡챠 드래그 버튼]]></value>
		<value xml:lang="en"><![CDATA[Draggable Captcha System]]></value>
	</item>
	<item name="msg_start_drag">
		<value xml:lang="ko"><![CDATA[잠김 : 등록 버튼을 활성화하려면 위 버튼을 우측 끝까지 드래그하십시오.]]></value>
		<value xml:lang="en"><![CDATA[Locked : Form cannot be submited.]]></value>
	</item>	
	<item name="msg_end_drag">
		<value xml:lang="ko"><![CDATA[해제 : 등록 버튼이 활성화 되었습니다.]]></value>
		<value xml:lang="en"><![CDATA[Unlocked : Form can be submited.]]></value>
	</item>
	<item name="msg_qaptcha_key_not_exist">
		<value xml:lang="ko"><![CDATA[캡챠키를 생성할 수 있는 OS 환경이 아닙니다.]]></value>
		<value xml:lang="en"><![CDATA[Capable of generating a Qaptcha key is not the OS environment.]]></value>
	</item>

lang.xml 파일을 열고 맨하단에 캡챠에서 사용할 언어 item을 추가합니다.



2. 회원가입 스킨 파일 수정


<load target="../../tpl/js/signup_check.js" />
<!--// datepicker javascript plugin load -->
<!--%load_js_plugin("ui")-->
<!--%load_js_plugin("ui.datepicker")-->
<!--// QapTcha Load -->
<load target="./css/qaptcha.css" />
<load target="./js/QapTcha.jquery.js" />
<load target="./js/jquery.ui.touch.js" />
<include target="./common_header.html" />

회원가입 폼을 출력하는 signup_form.html 을 열고 4번라인 아래부터 헤더 include문 위까지

캡챠가 사용할 css, js파일을 로드하는 구문을 추가합니다.  


<!--// QapTcha Drag Start -->
		<div class="control-group">
			<div class="control-label"><em style="color:red">*</em> {$lang->draggable_qaptcha}</div>
			<div class="controls" style="padding-top:5px">
				<div class="QapTcha"></div>
			</div>
		</div>
		<!--// QapTcha Drag End-->
		<div class="btnArea" style="border-top:1px solid #ccc;padding-top:10px">
			<input type="submit" value="{$lang->cmd_registration}" class="btn btn-inverse pull-right" />
			<a href="{getUrl('act','','member_srl','')}" class="btn pull-left">{$lang->cmd_cancel}</a>
		</div>
	</form>

폼 안에 캡챠 드래그 버튼을 추가하는 태그를 위와 같이 추가합니다.

드래그 버튼은 폼 전송 버튼 바로 위에 위치하도록 만듭니다.


<script>
jQuery(function($){
	// label for setup
	$('.control-label[for]').each(function(){
		var $this = $(this);
		if($this.attr('for') == ''){
			$this.attr('for', $this.next().children(':visible:first').attr('id'));
		}
	});
});
(function($){
	$(function(){
		var option = { changeMonth: true, changeYear: true, gotoCurrent: false,yearRange:'-100:+10', dateFormat:'yy-mm-dd', onSelect:function(){
			$(this).prev('input[type="hidden"]').val(this.value.replace(/-/g,""))}
		};
		$.extend(option,$.datepicker.regional['{$lang_type}']);
		$(".inputDate").datepicker(option);
		$(".dateRemover").click(function() {
			$(this).prevAll('input').val('');
			return false;});
	});
})(jQuery);
jQuery(function($){
	// QapTcha Language
	var start_drag = '{$lang->msg_start_drag}';
	var end_drag = '{$lang->msg_end_drag}';
	$('.QapTcha').QapTcha({ txtLock: start_drag, txtUnlock: end_drag });
	// jQuery.ui.touch Plugin
	$('.Slider').addTouch();
});
</script>

맨하단 스크립트에는 폼라벨과 생년월일 입력(팝업 캘린더)을 위한 jquery 스크립트가 이미 들어 있습니다.

캽차에서 사용할 언어와 터치 동작을 위한 함수 호출 구문을 위와 같이 스크립트 안에 추가합니다.



3. 플러그인, CSS, 버튼 이미지 업로드


./modules/member/skins/default/js/QapTcha.jquery.js

./modules/member/skins/default/js/jquery.ui.touch.js

./modules/member/skins/default/css/qaptcha.css

./modules/member/skins/default/css/btn_qaptcha.jpg


위 경로를 따라 jQuery 플러그인 파일, CSS 파일, 버튼 이미지를 업로드합니다.



4. 액션 추가


<action name="procMemberResetAuthMail" type="controller" ruleset="resetAuthMail" standalone="true" />
<action name="procMemberSpammerManage" type="controller" standalone="true" />
<action name="setQapTchaSession" type="controller" standalone="true" /> <!--// QapTcha Session creation -->
72번 라인 다음 줄, Admin 액션이 시작되는 바로 윗줄에 위와 같이 캡챠가 사용할 setQapTchaSession 액션을 추가합니다.



5. 컨트롤러 수정


/***** QapTcha 세션 세팅 ****************************************************/
	function setQapTchaSession() 
	{
		$qaptcha_response_error = true;

		$response_action = Context::get('response_action');
		$qaptcha_key = Context::get('qaptcha_key');

		// qaptcha_key가 없다면
		if(!$qaptcha_key) return new Object(-1, 'msg_qaptcha_key_not_exist');

		if(isset($response_action) && isset($qaptcha_key)) 
		{
			$_SESSION['qaptcha_key'] = false;
			if(htmlentities($response_action, ENT_QUOTES, 'UTF-8') == 'qaptcha') 
			{
				$_SESSION['qaptcha_key'] = $qaptcha_key;
				$qaptcha_response_error = false;
			}
		}

		$this->add('qaptcha_response_error', $qaptcha_response_error);
	}
	/***************************************************************************/

	/**
	 * Join Membership
	 *
	 * @return void|Object (void : success, Object : fail)
	 */
	function procMemberInsert()
	{

member.controller.php 파일을 열면 250번 라인에서 회원가입을 실행하는 procMemberInsert() 메소드가 있습니다.

그 윗부분에 위와 같이 캡챠 세션을 세팅하는 소스코드를 복사하여 붙여넣기 합니다.

회원가입 폼에서 캡챠 버튼을 드래그하면 AJAX 방식으로 위 액션이 실행되고 생성된 키값을 받아 $_SESSION에 세팅해 두는 기능을 수행합니다. 실제로 폼이 전송되면 전송된 폼 안에서 이 키값으로 속성(name)을 찾고 값(value)이 비어 있는지를 확인할 것입니다. 드래깅 했을때는 속성과 값이 모두 채워져 전송됩니다.



/**
	 * Join Membership
	 *
	 * @return void|Object (void : success, Object : fail)
	 */
	function procMemberInsert()
	{
		if (Context::getRequestMethod () == "GET") return new Object (-1, "msg_invalid_request");
		/***** QapTcha 세션 확인 ******************************************************/
		if(!isset($_SESSION['qaptcha_key'])) return new Object (-1, "msg_signup_disabled");

		if(isset($_SESSION['qaptcha_key']) && !empty($_SESSION['qaptcha_key'])) 
		{

			$QapTchaInput = $_SESSION['qaptcha_key'];
			$QapTchaInputValue = $_POST[''.$QapTchaInput.''];
			if(isset($QapTchaInputValue) && !empty($QapTchaInputValue))
			{
				unset($_SESSION['qaptcha_key']);
				return new Object (-1, "msg_signup_disabled");
			}
		}
		/*****************************************************************************/
		$oMemberModel = &getModel ('member');
		$config = $oMemberModel->getMemberConfig();

procMemberInsert() 메소드 안에서는 요청방법을 확인하는 코드 다음줄에 위 소스코드를 복사하여 붙여넣기 합니다.

캡챠 드래깅 없이 호출되었거나, 웹브라우저를 닫고 나간 경우 일정시간이 지난후에는 세션이 자동으로 삭제되기 때문에 세션값이 없다면 하위 로직은 진행되지 않을 것입니다. 만약 세션은 있지만 이전에 저장해둔 세션값으로 전송된 폼의 속성(name)값을 찾았을때 값이 여전히 채워져있다면 마찬가지로 로직은 멈추게 됩니다. 전송된 속성(name)값이 이전에 저장된 세션과 달라도 멈춥니다. qaptcha_key 값은 뷰에서 버튼이 드래깅 될때마다 매번 다른 키값으로 생성되고 서버에서는 그 값을 받아 세션에 다시 저장합니다.


[중요]

만약 자바스크립트가 지원되지 않은 OS 환경에서는

위와 같은 드래그 방식의 캡챠는 정상적으로 동작하지 않습니다.


XE 1.7.5에 포함되어 있는 Captcha Member 애드온을 켜면 바로 다음 로직에서 트리거가 작동하기 때문에 간섭받지 않습니다. QapTcha를 애드온 또는 트리거로 활용하지 않은 것은 컨트롤러 로직 안에 직접 포함시키기 위한 것이고 애드온이나 트리거 역시 네이티브 소스코드로 동작한다고는 하지만 보안에 따른 검증은 별개의 문제이기 때문에 직접 구현하고자 시도하였습니다. 옵션의 설정은 원문사이트를 참고하세요.


DEMO : http://xeschool.cafe24.com/

http://www.myjqueryplugins.com/jquery-plugin/qaptcha


※ 첨부파일에는 같은 내용으로 수정된 회원정보수정 폼(modify_info.html)과 모바일(m.skins)을 위한 수정파일도 포함되어 있습니다. (생년월일 입력/삭제 등 수정된 파일을 포함합니다.)


제목 글쓴이 날짜
메뉴가 많을때 레이아웃 쉽게 변경하기 웹빌드ver2 2014.05.23
XE 코어 1.7 버전에서 최근이미지 출력하기 [1] file 때린데 또때려 2013.12.14
스팸 회원 삭제시 동시에 관련 글 / 댓글 자동 삭제 + 스팸IP 등록기능 구현방법입니다 [20] sejin7940 2013.05.22
1.7.5에서 추가된 게시물당 최신댓글 불러오기 스킨에서 활용법 [2] file mAKEkr 2014.05.20
xe_documents DB 테이블이 깨져서 게시판이 엉망이 됄경우 file 지으니 2014.05.17
스케치북 추천+등록을 활용한 비추천+등록 버튼 만들기 [2] file oscarmike 2014.05.16
1.7.3.4 + sketchbook5 에서 대댓글 에디터 사용불가문제 [4] 뒤뒤 2013.08.07
XE CDN 서비스 중단에 대응하는 CDN 주소 변경법 (자신의 CDN서버로) [2] Stellar 2013.11.08
스케치북 게시판에 로그인/로그아웃 버튼 만들기 [1] Arp. 2013.09.22
최근문서 or 콘텐츠 위젯 공지글 뽑아오기 [8] Xiso 2010.11.29
제이쿼리 최신 버전 사용하기 [8] Lansi 2013.10.08
네이버 검색 API XE하고 연동하기 [3] mindpainter 2013.01.09
본문 링크 새창으로 띄우기.. [9] 멀티비타민 2013.11.11
푸시윙 댓글푸쉬서비스 소셜댓글 모드도 추가하기 [2] file XE힘들당휴 2014.05.11
이메일 주소로 계정 찾기시 잘못된 계정 인증요청이라고 뜰때 임시 대처법 file 루비스코 2014.05.10
출석부 모바일에서 출석이 혹시 안될경우. [6] BJ람보 2014.05.06
비회원 GPS 정보 처리 웹빌드ver2 2014.05.09
[주옥시리즈] [1.7.4] 로그인창 옆에 신규 쪽지 수 보여주기 [3] socialskyo 2013.09.09
XE를 사용중 백지현상이 있을경우 체크해볼 요소 [2] jambox 2014.02.08
무한 스크롤 스크립트 [4] 웹빌드 2014.04.16