설문 조사 폼을 만들이 있어서 작업 중에 설문 조사의 질문과 선택 옵션의 수는 가변적이므로 Parameter 컬렉션을 사용했다. 작업을 마무리 한 후 정상적으로 동작하는 지 테스트를 한 후 서버에 배포했다.


그런데.


이른바 "내 컴퓨터에서는 잘 되는데요?" 문제가 발생했다.


그렇다. 테스트를 할 때는 아무리 많은 질문과 옵션을 추가해도 잘 동작하던 것이, 서버에스는 IndexOutOfBound 예외를 내면서 설문이 제대로 처리되지 않는 문제가 발생했다.


설문 조사를 올리시는 분께서는 "음.. 7개가 넘어가면 설문 내용이 없어져요."라고 했고, 나는 "그럴리가 없다! 그 부분에 있어서 내 로직이 틀릴 리가 없다!" 라고 했다.

무지하면 용감하다더니, 어찌도 그리 당당할 수 있었을까? 다시 생각하니 부끄럽다.


서버에 배포된 설문 조사 폼을 확인해 보니, 정말 7개 이상으로 질문을 추가하면 죽는 것이 아닌가? 똑같은 설문 내용으로 내 컴퓨터(개발 컴퓨터)에서 테스트를 하니 "내 컴퓨터에서는 잘 된다."..


이 때까지는 자바스크립트로 설문 내용을 추가하는 부분에 이상이 있을 것으로 예상했다. 그런데 역시 이때까지는 Integer의 범위를 벗어나는 것도 아니고, 왜 하필 7개일까? 라며 이상하게만 생각하고는 자바스크립트 부분을 확인해 봤지만, 역시 이상은 없었다.


그러다 문득 톰캣 이 자식이 방해 공작을 하는 것이 아닐까 하는 생각에

"tomcat post form input limit length"를 키워드로 검색하니 한 블로그 글이 검색됐다.

그리고 그 글에는 내 문제의 원인이 무엇인지 명확하게 나와있었다. 그렇다. 파라미터의 수를 제한하는 설정이 있었던 것이다.


테스트 서버의 톰캣 설정은 그 파라미터 갯수 제한이 무한대라서 파라미터 갯수의 제한이 있다는 걸 모르는 상태에서는 그런 문제가 발생할 지 전혀 예측할 수 없었고,

실서버에서는 maxParameterCount="50" 으로 50개의 제한이 있었다.


질문 하나를 구성하는 속성은 제목, 질문 타입, 옵션 5개로 7개였으니, 톰캣이 수용할 수 있는 최대 질문의 수는 7개가 되는 것이고, 그 수 이상이 되면 서버가 빵~ 예외를 터뜨린다. 여태까지 파일 업로드 크기의 제한이 있는 줄은 알았지만, 파라미터의 수에 제한이 있을 줄은 몰랐다. 폼 파라미터의 갯수 및 크기 제한을 하는 설정은 아래와 같다.


server.xml 파일


maxParamterCount 속성은 폼 파라미터의 최대 수이고, maxPostSize는 폼의 최대 전송 용량(byte)이다. 값을 -1로 하는 경우 제한을 두지 않도록 한다.



"내 컴퓨터에서는 잘 되는데요?"

"님 컴퓨터에서도 잘 돌아가요"가 될 수 있도록 내가 사용하는 것들을 잘 알고 써야겠다.

Ubuntu 위에서 톰캣을 이용해서 웹 애플리케이션을 돌리고 있는데, 이놈의 시간대가 바뀌질 않는다.

"리눅스 시간 설정"을 키워드로 해서 검색한 글들을 참고해서 시스템의 타임존 설정 및 시간 동기화를 해보고,

톰캣에 JAVA_OPTS를 추가해서 톰캣의 타임 존을 설정해보기도 했으나 효과가 없다.

(startup.sh에 JAVA_OPT 변수를 추가한 것은 적용이 전혀 되지 않는 듯 하다. 아마 잘못해서 그런 거겠지만..쉘에서 date 명령어를 실행하면 원하는 대로 한국 시간대에 맞춰 KST로 표현이 되지만, 서버에서 출력하는 타임존은 계속 UTC다.


여러 방법을 시도하던 중에 dpkg-reconfigure tzdata 명령어를 이용해서 아시아/서울로 지정하니, 웹 서버에서 출력하는 시간대가 KST가 되었다.  (이 전에 /usr/share/zoneinfo에 있던 Asia/Seoul 타임존을 /etc/localtime에 링크 했었으나 원하는 대로 동작하지 않았다.)


dpkg-reconfigure tzdata 를 이용해서 서버의 로컬 시간대 설정


date 명령어에서 사용하는 timezone 정보와 자바(혹은 톰캣)에서 사용하는 timezone의 참조 하는 설정 값이 서로 다른가 보다. 이래서 모르면 몸이 고생한다. 얼마나 헤매었는지 모르겠네.


자바(혹은 톰캣)에서 사용하는 타임존의 설정은 /etc/localtime을 참조한다.

$ more /etc/localtime

타임존을 서울로 수정한 후 위 명령을 수행하면 KST-9를 확인할 수 있다.


현재는 해외에서 접속할 일이 없는지라 서버의 시간대를 한국표준시로 설정했지만 나중에 국제화나 지역화를 고려한다면 시간대에 대해 정확하게 이해하고서 사용해야겠다.


References...

리눅스 시간 설정(ikinox님) - http://ikinox.tistory.com/70

http://hacksforge.com/How-to-change-time-zone-in-Ubuntu-Linux.html

http://www.christopherirish.com/2012/03/21/how-to-set-the-timezone-on-ubuntu-server/

https://mail-archives.apache.org/mod_mbox/tomcat-users/200501.mbox/%3C45b72a5705012008352dcc3a39@mail.gmail.com%3E


  Moneycomb 서비스를 개발하면서 메인 도메인인 nabakcompany.com에 서브 도메인으로 moneycomb.nabakcompany.com을 추가하고, 같은 메인 도메인과 같은 웹 서버에서 리스닝 포트 번호만 달리해서 서비스를 하려고 했는데, DNS에 대한 이해가 부족한 탓인지 그게 안된다는 걸 이제야 알았다. 그러니까 포트 번호를 숨길 수는 없다.


  톰캣 문서와 검색을 통해서 서버 아이피가 동일하고, 서브 도메인이 다른 경우에는 웹 서버에서 가상 호스트를 추가하여, 각각의 서브 도메인에 대해 각각의 웹 애플리케이션을 제공할 수 있다는 것을 알게 되었다.


  현재 서버는 Apache 웹 서버 없이 Apache Tomcat 7로만 구동되고 있다.


| server.xml 파일에 호스트 추가

  $CATALINA_HOME 에 위치한 conf 폴더 내에 서버 설정을 위한 server.xml 파일이 있다. 호스트를 추가하기 위해서는  server.xml 파일을 아래와 같이 수정한다.


  Engine 엘리먼트 내에 Host 엘리먼트를 추가한다.

  톰캣을 설치한 후 설정을 수정하지 않았다면 파일의 끝부분에 localhost Host가 있다. 서브 도메인을 위한 호스트는 그 아래에 추가한다.


  Host의 속성으로 name에 DNS 호스팅 서비스(혹은 네임 서버)에서 추가한 서브 도메인을 입력하고, appBase는 웹 애플리케이션이 위치할 디렉토리를 지정한다. 상대경로로 입력할 경우 $CATALINA_HOME 아래가 기준이 된다.

다른 속성에 대한 설명은 아파치 톰캣7 문서를 참고.



| 디렉터리 및 ROOT.xml 파일 추가

  톰캣 5 이하에서는 server.xml 내에 Host 엘리먼트를 추가하면서 그 안에 Context 엘리먼트를 함께 추가해서 호스트 컨텍스트 정보를 포함시켰는데, 톰캣 6이상 부터는 따로 분리할 것을 권고한다.(라는 것을 다른 블로그에서 보았다..)[각주:1]

  $CATALINA_HOME/conf/Catalina 에 서브 도메인 이름과 같은 디렉터리를 만든다.

server.xml에서 설정한 서브 도메인이 sub.domain.com 이므로 디렉터리 이름도 sub.domain.com 이다.

/var/lib/tomcat7/conf/Catalina$ mkdir sub.domain.com 

(위의 /var/lib/tomcat7은 톰캣이 설치된 경로, 일반적으로는 /usr/local/apache-tomcat-7.x.x)

  그리고 만든 디렉터리 안에 ROOT.xml 파일을 생성하고 아래와 같이 입력한다.

docBase 속성은 컨텍스트의 루트로 사용될 애플리케이션이 위치한 경로이고,

privileged 속성은 값이 true일 때. 매니저 서블릿 같은 컨테이너 서블릿을 사용할 수 있도록 허용한다. (Set to true to allow this context to use container servlets, like the manager servlet. - 라고 되어있는데 무슨 의미인지 정확하게는 모르겠다..)



| 웹 애플리케이션 배치

  docBase에 설정한 경로에 웹 애플리케이션을 배치하면 webapps에 있는 것 맨치롬 서브 도메인으로 접속할 수 있다.


| 문제

  Host 엘리먼트의 appBase와 Context의 docBase의 정확한 기능이나..  unpackWARs 속성 값을 true로 했을 때 어디에 위치한 war 파일의 압축이 풀리는 건지 모르겠다.

appBase나 docBase에 넣어봤지만 어디에서도 압축이 풀리지 않아서 그냥 tar 명령어로 풀어서 배치했다. -_- (왜 있는데 쓰질 못하니!)

  매번 파일럿 프로젝트만 개발하고 운영 단계까지 넘어가지 않다보니, 그냥 '돌아만'가는 서버 구축을 하다가, '제대로 돌아' 가는 서버를 구축하려니 공부해야할 것들이 점점 더 쌓이는 것 같다. 웹 서버 로드밸런싱이나, DB 서버 확장 등 어디서부터 시작해야할 지 막막하다.


잘못된 내용이나, 빠진, 부족한 내용에 대한 태클은 환영입니다. :)


References


  1. http://www.sjune.net/archives/807 [본문으로]

어떤 URL로 리다이렉트하거나, 링크를 클릭했을 때 이동하지 않고 다운로드 하게 하려면,

브라우저는 응답 헤더의 내용을 보고 URL로 이동할지 다운로드를 할지 결정하기 때문에

서버측에서 아래와 같은 헤더 정보를 담아 주어야 한다.


Content-Type: application/octet-stream

Content-Disposition: attachment; filename=<브라우저에서 저장할 파일 이름>


Netty를 이용해서 웹 소켓 서버를 만들고 클라이언트에서 웹 소켓을 열고 통신을 하다보니

웹 소켓에서는 아직 텍스트 통신만 지원한다.


따라서 양쪽에서 서로 쓰기 좋은 Json 포맷으로 통신하다보니 서버쪽은 자바로 구성되어 Gson 라이브러리를 쓰고,

클라이언트에서 자바스크립트로 Json <-> Object 가 필요해서 찾아보니 내장 객체로 JSON이 있다는 것을 알게 되었다.

(관련 Stack Overflow 글)


JSON Parsing 내장 객체를 지원하는 브라우저는 다음과 같다


(http://caniuse.com/json)


JSON 객체의 함수로는 Object를 Json 문자열로 변환하는 stringify 와 반대로 Json 문자열을 객체로 변환하는 parse 함수가 있다.


JSON에 관한 설명은 MSDN에 잘 나와있어 사용방법에 관한 내용은 생략하고 매개변수에 대한 내용만 스크랩했다.

http://msdn.microsoft.com/ko-kr/library/ie/cc836466(v=vs.94).aspx


1. JSON.stringify(value [, replacer] [,space])

value

필수 요소. JavaScript 값은 일반적으로 변환할 개체 또는 배열입니다.

replacer

선택 사항입니다. 결과를 변환하는 함수 또는 배열입니다.

replacer가 함수이면 JSON.stringify는 키와 각 멤버의 값을 전달하여 함수를 호출합니다. 반환 값은 원본 값 대신 사용됩니다. 함수가 undefined를 반환하면 멤버가 제외됩니다. 루트 개체의 키는 빈 문자열인 ""입니다.

replacer가 배열이면 배열에 키 값이 있는 멤버만 변환됩니다. 멤버가 변환되는 순서는 배열의 키 순서와 같습니다. replacer 배열은 value 인수도 배열인 경우 무시됩니다.

space

선택 사항입니다. 읽기 쉽도록 들여쓰기, 공백, 줄 바꿈 문자를 반환 값 JSON 텍스트에 추가합니다.

space가 생략되면 반환 값 텍스트가 추가 공백 없이 생성됩니다.

space가 숫자이면 반환 값 텍스트가 각 수준의 지정된 공백 수로 들여쓰기됩니다. space가 10보다 크면 텍스트가 10칸 들여쓰기됩니다.

space가 '\t'와 같이 빈 문자열이 아니면 반환 값 텍스트가 각 수준의 문자열의 문자로 들여쓰기됩니다.

space가 10자보다 긴 문자열이면 처음 10자가 사용됩니다.

변환할 객체에 toJSON(key) 함수가 있는 경우 해당 키를 가진 값을 변환하여 반환한다.



2. JSON.parse(value [,reviver])

value

필수 요소. JavaScript 값은 일반적으로 변환할 개체 또는 배열입니다.

replacer

선택 사항입니다. 결과를 변환하는 함수 또는 배열입니다.

replacer가 함수이면 JSON.stringify는 키와 각 멤버의 값을 전달하여 함수를 호출합니다. 반환 값은 원본 값 대신 사용됩니다. 함수가 undefined를 반환하면 멤버가 제외됩니다. 루트 개체의 키는 빈 문자열인 ""입니다.

replacer가 배열이면 배열에 키 값이 있는 멤버만 변환됩니다. 멤버가 변환되는 순서는 배열의 키 순서와 같습니다. replacer 배열은 value 인수도 배열인 경우 무시됩니다.

space

선택 사항입니다. 읽기 쉽도록 들여쓰기, 공백, 줄 바꿈 문자를 반환 값 JSON 텍스트에 추가합니다.

space가 생략되면 반환 값 텍스트가 추가 공백 없이 생성됩니다.

space가 숫자이면 반환 값 텍스트가 각 수준의 지정된 공백 수로 들여쓰기됩니다. space가 10보다 크면 텍스트가 10칸 들여쓰기됩니다.

space가 '\t'와 같이 빈 문자열이 아니면 반환 값 텍스트가 각 수준의 문자열의 문자로 들여쓰기됩니다.

space가 10자보다 긴 문자열이면 처음 10자가 사용됩니다.



+ Recent posts