티스토리 툴바


2012/05/11 00:54 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

WHY?

ear or war 방식의 배포환경에서 이미지 파일 하나 또는 js파일이나 jar 하나 바뀌었을때 빨리 재배포하고 싶었음.

또는 제약사항으로 인하여 로컬에서 테스트가 어렵고 서버에서만 확인이 가능한 경우의 프로그램 디버깅, 수정

또다른 케이스로는 CI환경이 구성되어 있지만 각 모듈의 오너가 달라 임의로 전체 ear을 서버에 배포시키기 부담될때.


HOW

사전제약사항 : jeus 에 특화된 명령을 사용.

1. 갱신할 파일을 준비한다. ex) lepffm-face.png 

2. ear 파일에서 war를 추출한다.

jar xvf test.ear test-web.war

3. 새로운 이미지를 war에 갱신한다.

jar uvf test-web.war images/lepffm-face.png

4. 정상적 으로 갱신되었는지 확인한다.

jat tvf test-web.war | grep lepffm-face.png

5. ear을 update 한다.


jar uvf test.ear test-web.war

cf) application.xml 관련 오류가 난다면 대신 zip을 쓴다

zip -u test.ear test-web.war

6. redeploy

jeusadmin ..... redeploy test


lessons Learned

jar 명령도 잘 알아두면 도움이 된다.

deploy mode에 너무 겁먹지 마라.




저작자 표시 비영리 변경 금지
posted by pffm
TAG ear, jar, redeploy, war
2012/04/27 11:00 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

WHY?

ROLLBACK 하려고 하는데 다른 곳에서 참조한 문서들로는 잘 안먹더라...


example


@TransactionConfiguration(transactionManager="txManager", defaultRollback=true) // transactionManager id나 name을 명시..

public class YourRollbackTest {

@Transactional

@Test public void testRollbackTest() throws Exception{

                // do test 
         }

}



@Test 어노테이션이 붙은 곳에서 @Transactional을 사용하면 항상 rollback 처리해주네요.. 

저작자 표시 비영리 변경 금지
posted by pffm
2012/04/27 09:23 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

WHY?

프로젝트에서 SpringMVC의 controller를 점점 복잡하게 작성하는 경향이 발생함.(서비스 로직을 웹에서 구현하는 것으로 보임.. )

테스트 코드가 있어야 안전하게 리팩토링이 가능할테니... 

springframework이 버전을 올리면서 복잡해지는 것도 있지만 이런 장점들이 계속 쌓여 나가는 것이 마음에 든다.  



example


import static org.junit.Assert.*;

import static org.springframework.test.web.ModelAndViewAssert.*;

import org.springframework.mock.web.*;

import org.springframework.web.servlet.*;

import org.springframework.web.servlet.mvc.annotation.*;

import org.springframework.context.*;

import org.codehaus.jettison.json.JSONObject;

@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations = {"file:src/.../xxx.xml" })

public class TistoryTest {

@Inject  ApplicationContext applicationContext;

@Inject AnnotationMethodHandlerAdapter handlerAdapter;

@Inject DefaultAnnotationHandlerMapping handlerMapping;

MockHttpServletRequest request;
MockHttpServletResponse response;
/// MockHttpSession session; // if you need session data...
ModelAndView mav;

        // 테스트할 spring MVC controller 
         YourTestController controller;

@Before
public void setUp() throws Exception {
response = new MockHttpServletResponse();
request = new MockHttpServletRequest();
/// session = new MockHttpSession(null);
}
@After
public void tearDown() throws Exception {
/// session.invalidate();
}

        // ViewName을 반환하는 일반적인 케이스를 위한 테스트 
@Test  public void testViewNameReturnMethod () throws Exception{
// [given]
                request.setRequestURI("/tistory.do");
                request.setMethod("GET");
request.addParameter("method"," viewNameReturnMethod "); // if you using method style
request.addParameter("yourParam","yourValue"); // set parameter
                /// saveSession("MYKEY","MYVALUE");

// [when]
handleInterceptor(controller);
                handlerAdapter.handle(request, response, controller);

// [then]
assertViewName(mav, "yourViewName"); // check viewname
String attr1 = assertAndReturnModelAttributeOfType(mav,
"yourAttribute1", String.class);
assertEquals(attr1,"expected result");
}

        // ajax 통신을 할 경우 직접 ResponseBody에 출력하기 때문에 response의 내용으로 테스트를 수행한다. 
@Test public void testJsonReturnMethod() throws Exception{
// [given]
                request.setRequestURI("/tistory.do");
                request.setMethod("GET");
request.addParameter("method","jsonReturnMethod"); // if you using method style
request.addParameter("yourParam","yourValue"); // set parameter
                /// saveSession("MYKEY","MYVALUE");

// [when]
handleInterceptor(controller);
                handlerAdapter.handle(request, response, controller);

// [then]
               JSONObject json = new JSONObject(response.getContentAsString());
               String result = json.getString("your_result_json_key");
assertEquals(result,"expected json value");
}
protected void handleInterceptor(Object controller) throws Exception {
for (final HandlerInterceptor interceptor : handlerMapping.getHandler(
request).getInterceptors()) {
interceptor.preHandle(request, response, controller);
}
}
/*  protected void saveSession(String key, Object value) {
session.putValue(key, value);
request.setSession(session);
}  */
}

[주의사항]
session scoped bean을 사용할 경우 다음 에러가 발생함 : 
java.lang.IllegalStateException: No Scope registered for scope 'session'
아래 참고자료를 이용하여 해결할 것.
ref : http://blog.solidcraft.eu/2011/04/how-to-test-spring-session-scoped-beans.html

저작자 표시 비영리 변경 금지
posted by pffm
2012/04/25 11:06 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

WHY?

윈도우에서 실행되는 데몬성 프로그램 중에서 복수개의 프로세스를 띄우면 안되는 프로그램이 있어서...


HOW?

1. tasklist 명령으로 해당 프로세스가 떠있는지 확인.

2. 유무에 따라 프로세스 실행


example

@echo off

set PROCEXIST=NO

set PCNAME="blahblah.exe"

tasklist /nh /fi "Imagename eq %PCNAME%" | find /C %PCNAME% && set PROCEXIST=YES


if /I "%PROCEXIST%"=="NO" goto :EXEC


echo %PCNAME% already exist.

tasklist /v /fi "Imagename eq %PCNAME%" /fo list

echo ******* press any key to exit *******

echo.

pause >NUL

goto :EXIT


:EXEC

blahblah.exe


:EXIT


References 

http://www.computing.net/answers/programming/dos-command-for-wait-5-seconds/11192.html

tasklist : http://technet.microsoft.com/en-us/library/bb491010.aspx

저작자 표시 비영리 변경 금지
posted by pffm
2012/04/16 08:54 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

- WHY?


gradle의 멀티 플로젝트 설정 가이드를 따라가면 늘 최상위 경로에 settings.gradle 을 구성하게된다.

당연히 전체 프로젝트에 대해 계층적인 구조를 가져가는 것이 상식이다.

하지만 eclipse 관점에서만 볼때, 모든 프로젝트는 항상 같은 레벨에서 관리가 된다. 

때문에 상위 경로에서 멀티 프로젝트를 관리하는 개념을 도입할때 어떻게 해야 할지 고민하게 된다.

eclipse project 단위로 형상을 관리할 경우, 상위 경로에 있는 멀티프로젝트 설정파일은 버전관리 대상에서 제외 되어 버리기 때문이다. 


물론... gradle user guide에서 설명을 안했을 뿐이지, 멀티프로젝트 관리를 반드시 계층적으로 해야 하는 것은 아니다. 


다음과 같이 설정할 수 있다.


- project-emma : multi-project관리용 root project

   settings.gradle

   build.gradle

- project-navi : child project 1. Type : java library

   build.gradle

- project-bovary : child project 2. Type: web application. it depends on project-navi

   build.gradle


[$project-emma/settings.gradle]

// define projects

include 'navi', 'bovary'


// define sub-projects location

project(':navi').projectDir = new File(rootDir, '../project-navi')

project(':bovary').projectDir = new File(rootDir, '../project-bovary')

[$project-emma/build.gradle]
// define subproject configuration
subprojects{
// define plugins
apply plugin: 'java'
repositories {
mavenCentral()
}
// set compile encoding
[compileJava, compileTestJava]*.options*.encoding = 'UTF-8'
}
[$project-navi/build.gradle]
jar {
destinationDir = file("dist"); // 별도 위치에 저장하고 싶을 경우에만 명시.
}
[$project-bovari/build.gradle]
apply plugin: 'war' // compile -> war -> assemble
dependencies {
compile project(':navi') // navi project에 대한 의존성을 명시.
 }

당연하지만, 멀티프로젝트 관리용으로 빈 프로젝트를 만들어야 할 필요는 없다. (navi의 하위 경로에 지정해도 가능하다는 말임) 앞으로도 유용하게 사용할 수 있을 듯 싶음. 


저작자 표시 비영리 변경 금지
posted by pffm
2012/04/12 14:07 분류없음
크리에이티브 커먼즈 라이선스
Creative Commons License

json 파일을 추가할 경우가 생겨서..


1. http.m 파일 수정

$webtob/config/http.m


json            MimeType = "application/octet-stream", SvrType=HTML


2. compile
wscfl -i http.m

3. webtob 재기동
wsdown
wsboot

저작자 표시 비영리 변경 금지
posted by pffm
2012/04/12 09:14 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

[WHY] 

eclipse에서 grails plugin을 설치하지 않고도  ant  view에서 편리하게 각각의 gradle 명령을 사용하고 싶었음.


참고한 reference를 보면 gradle이 설치되어 있고 GRADLE_HOME 환경변수가 설정된 상태를 전제로 한다.


앞글에서 설명했던 gradlew 를 이용하면 미리 gradle binary들 설치하고 환경변수를 설정하지 않고도 빌드실행이 가능해진다. 




[Prerequisite]

== gradle wrapper생성을 통해 만들어진 파일들 (3)

gradlew.bat or gradlew

gradle/wrapper/gradle-wrapper.jar

gradle/wrapper/gradle-wrapper.properties


== 실제 build를 수행할 build 파일

build.gradle

// this is sample...

apply plugin: 'java'

repositories {

mavenCentral()

}

[compileJava, compileTestJava]*.options*.encoding = 'UTF-8'

jar {

destinationDir = file("dist")

}


== gradle을 수행할 ant build wrapper 

build.xml

<project name="build-gradle"  default="help"  basedir=".">

<property environment="env" />

<property name="gradle.executable" location="./gradlew.bat" /><!-- for window -->

<target name="help">

<exec executable="${gradle.executable}" dir=".">

<arg value="tasks" />

</exec>

</target>

<target name="build"> <!-- 모든 포함된 project의 build task를 실행 -->

<exec executable="${gradle.executable}" dir=".">

<arg value="build" />

</exec>

</target>

<!--

<target name="subproject-build" description=" 특정 subproject의 build task만 실행시키고 싶을때 사용">

<exec executable="${gradle.executable}" dir=".">

<arg value=":common:build" />

</exec>

</target>

-->

</project>

이제, 필요할 때마다 ant target을 추가하면된다. 



reference : https://gist.github.com/735788 

저작자 표시 비영리 변경 금지
posted by pffm
2012/04/12 08:47 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

maven이든 ant든 다른 어떤 툴이든, 우선 많은 용량의 binary를 다운 받아서 작업을 해야한다.

plugin으로 제공되는 경우도 용량 자체는 크게 다르지 않거니와 행여나 버전이 낮아 쓰기 불편한 경우도 있어

새로 시작할 때마다 부담이 되는 경우가 있다. 

gradle은 eclipse 지원을 일찍 포기한 것으로 보이는데, 상위버전에서 제공되는 명령이 제대로 먹히지 않는다.

(물론 별도의 GRADLE_HOME 을 설정할 수 있도록 preference 에서 설정화면을 제공하고 있다.)


gradle wrapper의 핵심 idea는  gradle을 설치하지 않은 상태에서도 gradle을 사용할 수 있도록 제공하는 기능이다.


물론 아무것도 없이 실행되는 것은 아니다. 

단지 최소한의 쉘스크립트와 바이너리를 통해 이 작업을 수행할 수 있다. 



먼저, gradle wrapper 스크립트를 만들기 위해서는 당연하게도 gradle이 설치되어 있는 곳에서 시작하여야 한다.

build.gradle을 다음과 같이 작성한다.

// 현재 gradle 버전은'1.0-milestone-9' 이다. 

task wrapper(type: Wrapper) {

gradleVersion = '1.0-milestone-9'

}

defaultTasks 'wrapper'


실행이 되면

쉘파일과 (윈도우용인 gradlew.bat , *nix용인  gradlew)

실행을 위한 jar (gradle/wrapper/gradle-wrapper.jar )

설정정보를 포함한 property ((gradle/wrapper/gradle-wrapper.properties)

가 만들어진다. 

(cf. 새 버전이 나올 경우 wrapper task를 실행하여 새로 생성하거나 단순히 gradle-wrapper.properties 에서 binary를 다운 받을 경로를 변경해주면 된다.)


이제 준비완료.


gradlew가 해주는 역할은 단순하다.

gradle 배포본이 있는지 체크해서 없으면 gradle-wrapper.properties에 정의되어 있는 경로에서 다운로드를 받고,

있을 경우에는 모든 argument를 gradle 에게 전달한다. 


현재 gradle-1.0-milestone-9-bin.zip의 크기는 29M다. ( gradle-wrapper.jar 는 39K )


기본으로 설치되는 위치는 

$USER_HOME/.gradle/wrapper/dists


이제 신규 프로젝트에

저 파일을 같이 포함하면 준비완료.

gradlew 를 실행하면 된다. 



reference : http://www.gradle.org/docs/current/userguide/gradle_wrapper.html


저작자 표시 비영리 변경 금지
posted by pffm
2012/04/05 07:48 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

jqgrid의 이렇게 다양한 기능이 있다는걸 새롭게 알게됨.


formatter:'date',formatoptions: {newformat:'Y-m-d'} 을 추가


{name:'started',   index:'started',   sortable:false, align:'left', width:90, formatter:'date',formatoptions: {newformat:'Y-m-d'}},


referece) http://www.trirand.com/jqgridwiki/doku.php?id=wiki:colmodel_options

저작자 표시 비영리 변경 금지
posted by pffm
2012/04/05 07:40 IT
크리에이티브 커먼즈 라이선스
Creative Commons License

Rest Client example

- reference

http://blog.springsource.org/2009/03/27/rest-in-spring-3-resttemplate/
http://dlinsin.blogspot.com/2009/11/playing-with-spring-resttemplate.html http://static.springsource.org/spring/docs/3.0.x/javadoc-api/org/springframework/web/client/RestTemplate.html


import org.springframework.web.client.*;

...

@RequestMapping(value="/rest", method=RequestMethod.GET)

public String rest( Model model) throws Exception{

String url = "http://localhost/sample?email={mail}"; 

RestOperations restTemplate = new RestTemplate();

String result = restTemplate.getForObject(url, String.class, "tistory@tistory.com"); // url, responseType, urlVariable

//DELETE delete(String, String...)

//GET getForObject(String, Class, String...)

//POST postForLocation(String, Object, String...)

//PUT put(String, Object, String...)  

model.addAttribute("result",result);

// json방식으로 처리할 경우 바로 result를 return 하면 될듯..  

return "sample/RestResult";

}


저작자 표시 비영리 변경 금지
posted by pffm
 <PREV 1 2 3 4 5 ... 25    NEXT>