안드로이드 aar 트러블슈팅 기록

안드로이드 aar 트러블슈팅 기록

너무 많은 시간을 허비했다.

오랜 삽질 끝에 해결은 했는데 도대체 뭐가 문제였는지도 잘 모르겠다.

일단 기록해놓지 않으면 나중에 기억나지 않을 것이기에 기록한다.

1. 1차 시도

일단 기존 쓰고 있는 앱 소스코드를 기반으로, 새로운 2개의 앱을 만들어내는게 핵심과제였다.

그런데 새로운 앱 2개에서는 만들어내자 기존에 잘 돌던 코드가 동작하지 않았다.

앱 위변조검증이라는 기능이었는데, 타 업체의 라이브러리에 의존하는 코드였다.

타업체가 만든 라이브러리는 2.1.7.jar 라는 파일을 쓰고 있었다.

알고보니 이 라이브러리는 안드로이드 앱의 패키지명이 변경되면 동작하지 않는 방식이었다.

정확히 표현하면 build.gradle 의 applicationId 값이 바뀌면 라이선스 어쩌구 하는 못보던 오류 메시지가 발생했다.

정식 라이선스 값이라는 String 변수가 있고, 해당 변수가 패키지명과 매칭되지 않으면 동작하지 못하도록 제어하는 것 같았다.

이것을 알아내는 데에도 엄청난 시간이 걸렸다(잘 동작하는 AS-IS 프로젝트를 갖다 놓고, TO-BE와 같아지도록 한 줄씩 수정하면서 언제 라이선스 어쩌구 오류 메시지가 발생하는지 테스트했다).

아무튼 패키지명에 알맞게 2개의 정식 라이선스 값을 받아내면 된다고 생각했는데, 새로운 버전의 라이브러리로 교체하라는 답변을 업체로부터 받았다.

새로운 버전의 라이브러리는 라이선스 값과 무관하게 잘 동작한다고 했다.

2. 2차 시도

새로운 버전의 라이브러리는 jar 파일이 아니고 aar 파일이었다.

검색을 동원해본 결과 aar 파일을 프로젝트에 적용하는 방법은 2가지가 있다.

첫번째는 libs 폴더에 aar 파일을 넣고, build.gradle 파일을 수정하는 것이다. 이것에 대해서는 https://blog.naver.com/bb_/221758376358 포스트에 정리해둔 바 있다.

결과적으로 이 방법으로 성공했다.

두번째 방법은 aar을 사용하고 있는 모듈을 추가하는 것이다.

예제로 받은 샘플 프로젝트가 이 방식을 사용하고 있어서 계속 시도해본 방법이다.

결과적으로 이 방법은 실패했지만 방법만 적어둔다.

(1) File – New – Import Module – 모듈 폴더를 선택.

개인적으로는 받은 샘플 프로젝트 안에 aar을 사용하는 모듈이 들어있어서, 해당 폴더를 그대로 복사해왔다. 폴더명은 library_name_v4.0.0 이었다.

(2) build.gradle 파일에 implementation project(path: ‘:library_name_v4.0.0’) 추가

(3) settings.gradle 파일에 include ‘:library_name_v4.0.0’ 추가

처음에는 두번째 방법(aar을 사용하고 있는 모듈을 추가)을 시도했는데, 계속 실패했다.

계속 실패했다는 것은 뭔가 잘못된 것 같으면 전체 롤백하고 처음부터 다시 하기를 반복했다는 뜻이다.

기존에 쓰던 2.1.7.jar 파일을 삭제하고, library_name_v4.0.0 모듈을 추가하고, 빨간줄이 발생한(오류가 발생한) 코드를 고치는 작업이었는데 고칠게 너무 많았다.

기존에 com.업체명~ 으로 시작했던 것들이 모두 net.업체명~ 식으로 패키지명이 바뀌었다.

또 사용하던 메서드들이 없어진 것들이 많아서 코드를 마구 삭제하다가, 너무 많이 건드린 것 같으면 롤백하기를 반복하였다.

결국 작업은 완료되었는데, 이제 앱이 죽는 현상이 생겼다.

한참 원인을 찾지 못하다가 로그캣을 보게 됐다.

——— beginning of crash 라고 해서 익셉션이 나와있었고, FATAL EXCEPTION 이 발생한 것이었다.

핵심이 되는 오류는 java.lang.VerifyError 였다.

Caused by: java.lang.VerifyError: Verifier rejected class com.타업체패키지명.타업체클래스명: void     com.타업체패키지명.타업체클래스명<init>(boolean) failed to verify: void     com.타업체패키지명.타업체클래스명<init>(boolean): [0x31] Expected initialization on uninitialized reference Precise Reference: java.lang.String (declaration of ‘com.타업체패키지명.타업체클래스명’ appears in /data/app/com.자사패키지명-1/split_lib_slice_9_apk.apk)

막막해서 지인들에게 도움을 요청해보니 검색하여 아래 답변을 주었다.

생각해보면 나도 검색해서 얻을 수 있는 정보였는데 찾지 못한 이유는, 하필 라이브러리에서 오류가 나는 해당 메서드 명이 verify여서 이게 자바 레벨의 오류가 아니라 특정 업체 라이브러리에서만 발생하는 특수한 오류라고 생각했다. 나는 VerifyError 라는 키워드만 빼고 나머지 키워드들로 검색을 하고 있었다…

VerifyError 발생 원인

1. 컴파일 시 사용한 라이브러리와 런타임 시 사용한 라이브러리 버전이 달라서 메서드 형태가 다른 경우

2. 사용한 라이브러리가 상위 버전의 JDK 에서 컴파일 된 경우

결국 라이브러리 버전을 맞춰야 한다.

저기 분명 라이브러리 버전을 맞춰야 한다고 했는데, 나는 어쩐 일인지 JDK버전 내지는 안드로이드 SDK 버전을 맞춰야 한다는 생각이 들었다.

뭐가 뭔지 모르겠는 상태에서 library_name_v4.0.0.aar 을 반디집으로 열어서 AndroidManifest.xml 을 열어보니 안드로이드 버전 29로 되어있었다.

내 프로젝트는 안드로이드 버전 28 이었는데, 어쩐 일인지 build.gradle을 29로 고치면 앱이 동작하지 않았다.

검색을 거듭한 결과 안드로이드 스튜디오 버전 4.0 이상에서 안드로이드 버전 29를 쓸 수 있다는 내용을 보았다.

내가 사용한 안드로이드 스튜디오는 버전 3.4 였다.

확실한지는 모르지만 지푸라기라도 잡는 심정으로 안드로이드 스튜디오 4.0을 깔고, 작업중인 프로젝트를 로드했다.

결과적으로 안드로이드 스튜디오 4.0 에서는 VerifyError 가 발생하지 않았다.

버전을 29로 올리지도 않았다.

그냥 내 프로젝트는 여전히 버전 28이고, aar 은 버전 29였다.

돌이켜보면 샘플 프로젝트도 3점대 안드로이드 스튜디오에서는 빨간줄이 그어져 빌드가 안됐고, 안드로이드 스튜디오 4.0 에서는 잘 열리고 빌드가 됐다.

뇌피셜이지만 해결된 이유는 aar 모듈이 안드로이드 스튜디오 4에서 만들어진 모듈이라서, 그걸 사용하는 프로젝트 역시 안드로이드 스튜디오 4에서 빌드해야 되는 것 같다.

문제는 이렇게 오류는 해결이 됐는데 앱위변조 검증이 통과가 됐다.

통과가 안되어야 정상인데 통과가 됐다.

이게 말이 안되는게 서버쪽에 앱의 고유 해시값이 있을건데 내가 이것저것 고쳤기 때문에, 서버에 올라가 있는 앱의 해시값과 내 앱의 해시값은 전혀 다를 것이란 말이다.

나중에 언급하겠지만 이 때는 verify 를 SimpleVerify 로 고쳐도 똑같이 위변조 검증이 통과됐다.

아무튼 오류는 발생하지 않으나 라이브러리가 똑바로 동작하지 않으니 모두 롤백했다.

=> 나중에 알게된 사실인데 이때 이미 성공을 한 것이었다.

앱위변조 검증은 서버에 따로 앱을 업로드해서 해시값을 만드는게 아니라, 첫번째로 앱위변조 요청(verify 메서드)을 했을 경우 해시값을 서버에 저장하고 그 값을 기준으로 검증하는 것이었다.

3. 3차 시도

2차 시도는 어제인 토요일 얘기고, 일요일인 오늘은 3차 시도였다.

처음부터 안드로이드 스튜디오 4.0 에서 프로젝트를 열어서 빌드를 하고 시작했다.

업체로부터 받은 aar 버전이 2가지가 있었는데, 하나가 어제 시도한 library_name_v4.0.0.aar 이었다.

사실 이것은 또 다른 업체로부터 받았으니 간접적으로 받은 파일에 해당한다.

다른 하나가 library_name_v4.0.4.aar 이었다.

이건 직접적으로 전달받은 거였다.

기존 버전의 버전 갭이 클수록 고쳐야할게 많다는 생각에 4.0.0을 먼저 시도한 것인데, 어제 잘 동작하지 않았으니 또 다시 지푸라기라도 잡는 심정으로 4.0.4 로 시작했다.

aar 을 추가할 때 모듈을 추가하는 방식을 사용하지 않고, 그냥 libs 폴더 안에 aar 파일을 넣는 방식을 사용했다.

그리고 기존 라이브러리(jar)를 지우고 진행을 하는데 아래 오류가 발생했다.

execution failed for task ‘:app:transformdexarchivewithexternallibsdexmergerfordebug’.

뭔가를 잘못했나 싶어서 롤백을 하고 다시 진행해도 똑같은 오류가 발생했다.

검색 결과 가장 도움이 된 포스트는 이것이다. https://miraclehwan.tistory.com/22

사용하는 라이브러리마다 참조하는 라이브러리 버전이 다 다를 경우, 메소드의 수가 계속 증가하여 64k개 이상을 가질 경우 위와 같은 에러가 발생한다.

(중략)

확실한 해결 방법은 버전이 다른 라이브러리를 사용할 경우 하나로 고정시키는 방법이다.

 

내가 이해한 바는, 결국 라이브러리 충돌이 난다는 얘기인 것이다.
분명히 기존 라이브러리를 지웠는데 말이다.

로그캣이었는지 디버그 콘솔이었는지 읽을 수 있는 것은 뭐든 꼼꼼히 읽어보니 못보던 패키지명이 있었다.

패키지명을 쫓아가보니 내가 모르는 jar에 속했다.

다시 말해 어떤 jar을 하나 더 의존하는 것처럼 보였다. 그 어떤 jar를 INI_4.1.2.jar 라고 하자.

아래 테스트를 통해 더 확실히 알 수 있었다.

기존 앱에서 2.1.7.jar 파일은 그대로 두고, INI_4.1.2.jar 를 삭제했다.

앱에 아무런 빨간줄도 나오지 않았지만, 실행해보면 앱은 오류가 발생됐다.

앱은 2.1.7.jar 만 의존하고 있지만, 2.1.7.jar 가 INI_4.1.2.jar 를 의존하고 있는 것이다(라고 추정했다).

그래서 프로젝트를 다시 전부 롤백하고 다시 시작했다.

(1) 안드로이드 스튜디오 4.0 으로 프로젝트를 열고, 빌드해서 실행했다.

(2) 기존 libs 폴더에서 2.1.7.jar 와 INI_4.1.2.jar 를 삭제했다.

(3) libs 폴더에 library_name_v4.0.4.aar 파일을 추가하고 build.gradle 파일을 알맞게 수정했다.

(4) 관련 코드를 수정했다.

이유는 모르겠지만 4.0.0 aar 모듈을 사용했을 때는 com.업체명~ 패키지를 전부 net.업체명~ 패키지명으로 바꿔야 했고 메서드도 많이 사라져있어서 난감했는데,

4.0.4.aar 을 libs에 추가해서 사용해보니 com.업체명~ 패키지가 전부 그대로였고 메서드도 거의 일치하는 모습을 보였다.

이게 모듈 사용 여부의 차이인지 버전의 차이(4.0.0, 4.0.4)인지는 불분명하다.

심증으로는 모듈을 사용한다고 패키지명이 바뀔리가 없다고 생각하는데

버전의 차이는 더 말이 안된다.

2.1.7 은 com 패키지를 쓰고, 4.0.0 은 net 패키지를 쓰고, 4.0.4 는 com  패키지를 쓴다는 말이기 때문이다.

아무튼 작업을 끝내고 실행해보니 이번에도 2차 시도 때처럼 앱 위변조검증에 통과하는 문제가 있었다.

또 롤백을 해야하나 고민하다가 API 문서를 읽어봤다.

4.0.0 과 4.0.4 API문서가 조금 달랐다.

4.0.0 은 verify 메서드를 쓰라고 했고, 4.0.4는 SimpleVerify 메서드를 쓰라고 했다.

2.1.7을 쓰던 기존 앱은 verify 메서드를 쓰고 있었다.

이걸 SimpleVerify 메서드로 고치니까 드디어 앱위변조 검증 결과 실패가 떴다.

(=> 나중에 알게된 사실인데 verify 를 써야한다. 그리고 앱위변조 검증은 서버에 따로 앱을 업로드해서 해시값을 만드는게 아니라, 첫번째로 앱위변조 요청(verify 메서드)을 했을 경우 해시값을 서버에 저장하고 그 값을 기준으로 검증하는 것이었다.)

이제 평일이 되면 서버에 앱 해시값을 등록해서 통과를 시킬 차례다…

이게 되어야 진짜 성공이고 아니면 또 롤백행일지도…

[Android] Could not connect to remote process. Aborting debug session.

[Android] Could not connect to remote process. Aborting debug session.

안드로이드 스튜디오에서 Run 또는 디버그를 수행했을 때 앱이 실행되지 않으면서 Debug 콘솔에 “Could not connect to remote process. Aborting debug session.” 메시지가 출력되는 문제.

안드로이드 스튜디오 우측상단 재생 버튼 왼쪽의 [app] 콤보박스를 클릭하고 [Edit Configurations…] 클릭 – Launch Options 의 Launch 콤보박스 값을 Nothing 에서 Default Activity 로 변경하여 다시 디버그 해본다.

참고사이트 : https://stackoverflow.com/questions/53962831/how-do-i-fix-could-not-connect-to-remote-process-while-trying-to-debug-the-app

[SpringBoot] 스프링부트 웹소켓(WebSocket) 예제

[SpringBoot] 스프링부트 웹소켓(WebSocket) 예제

1. 스프링부트 웹 프로젝트 생성

우선 스프링부트 웹 프로젝트가 있어야 한다.

새로 시작하려면 아래 포스트를 참고하면 된다.

https://blog.naver.com/bb_/222141978468

2. pom.xml 수정
아래 코드를 추가한다.

<!– websocket –>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-websocket</artifactId>
</dependency>

 

3. Websocket.java 파일 작성

원하는 패키지에 아래 파일을 작성한다.

package 원하는_패키지명;

import java.util.ArrayList;
import javax.websocket.OnClose;
import javax.websocket.OnError;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.server.ServerEndpoint;
import org.springframework.stereotype.Component;

@Component
@ServerEndpoint(“/websocket”)
public class Websocket {

    /**
     * 웹소켓 세션을 담는 ArrayList
     */

    private static ArrayList<Session> sessionList = new ArrayList<Session>();

    
    /**
     * 웹소켓 사용자 연결 성립하는 경우 호출
     */

    @OnOpen
    public void handleOpen(Session session) {
        if (session != null) {
            String sessionId = session.getId();
            
            System.out.println(“client is connected. sessionId == [“ + sessionId + “]”);
            sessionList.add(session);
            
            // 웹소켓 연결 성립되어 있는 모든 사용자에게 메시지 전송
            sendMessageToAll(“***** [USER-“ + sessionId + “] is connected. *****”);
        }
    }
    

    /**
     * 웹소켓 메시지(From Client) 수신하는 경우 호출
     */

    @OnMessage
    public String handleMessage(String message, Session session) {
        if (session != null) {
            String sessionId = session.getId();
            System.out.println(“message is arrived. sessionId == [“ + sessionId + “] / message == [“ + message + “]”);

            // 웹소켓 연결 성립되어 있는 모든 사용자에게 메시지 전송
            sendMessageToAll(“[USER-“ + sessionId + “] “ + message);
        }

        return null;
    }
    

    /**
     * 웹소켓 사용자 연결 해제하는 경우 호출
     */

    @OnClose
    public void handleClose(Session session) {
        if (session != null) {
            String sessionId = session.getId();
            System.out.println(“client is disconnected. sessionId == [“ + sessionId + “]”);
            
            // 웹소켓 연결 성립되어 있는 모든 사용자에게 메시지 전송
            sendMessageToAll(“***** [USER-“ + sessionId + “] is disconnected. *****”);
        }
    }

    
    /**
     * 웹소켓 에러 발생하는 경우 호출
     */

    @OnError
    public void handleError(Throwable t) {
        t.printStackTrace();
    }
    
    
    /**
     * 웹소켓 연결 성립되어 있는 모든 사용자에게 메시지 전송
     */

    private boolean sendMessageToAll(String message) {
        if (sessionList == null) {
            return false;
        }

        int sessionCount = sessionList.size();
        if (sessionCount < 1) {
            return false;
        }

        Session singleSession = null;

        for (int i = 0; i < sessionCount; i++) {
            singleSession = sessionList.get(i);
            if (singleSession == null) {
                continue;
            }

            if (!singleSession.isOpen()) {
                continue;
            }

            sessionList.get(i).getAsyncRemote().sendText(message);
        }

        return true;
    }
}

 

4. OOOApplication.java 파일 수정

@SpringBootApplication 가 붙어있는 자바 파일을 수정하면 된다. serverEndpointExporter 메서드를 만들고 @Bean 어노테이션을 붙여준다.

package 원하는_패키지명;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;

@SpringBootApplication
public class ProjectNameApplication {

    public static void main(String[] args) {
        SpringApplication.run(WebGameBaseApplication.class, args);
    }
    
    @Bean
    public ServerEndpointExporter serverEndpointExporter() {
        return new ServerEndpointExporter();
    }

}

 

5. chat.jsp 파일 작성
아래 파일을 적당한 위치에 작성한다.
여기서는 프로젝트/src/main/webapp/WEB-INF/views/chat.jsp 경로에 위치한다고 가정한다.

<%@ page contentType=“text/html; charset=utf-8” %>
<!DOCTYPE html>
<html>
<head>
<meta http-equiv=“Content-Type” content=“text/html; charset=UTF-8”>
<title>웹소켓 테스트 페이지</title>
<script type=“text/javascript”>
var g_webSocket = null;
window.onload = function() {
    g_webSocket = new WebSocket(“ws://localhost:8080/websocket”);
    
    
    /**
     * 웹소켓 사용자 연결 성립하는 경우 호출
     */

    g_webSocket.onopen = function(message) {
        addLineToChatBox(“Server is connected.”);
    };
    
    
    /**
     * 웹소켓 메시지(From Server) 수신하는 경우 호출
     */

    g_webSocket.onmessage = function(message) {
        addLineToChatBox(message.data);
    };

    /**
     * 웹소켓 사용자 연결 해제하는 경우 호출
     */

    g_webSocket.onclose = function(message) {
        addLineToChatBox(“Server is disconnected.”);
    };

    /**
     * 웹소켓 에러 발생하는 경우 호출
     */

    g_webSocket.onerror = function(message) {
        addLineToChatBox(“Error!”);
    };
}

/**
* 채팅 박스영역에 내용 한 줄 추가
*/

function addLineToChatBox(_line) {
    if (_line == null) {
        _line = “”;
    }
    
    var chatBoxArea = document.getElementById(“chatBoxArea”);
    chatBoxArea.value += _line + “\n”;
    chatBoxArea.scrollTop = chatBoxArea.scrollHeight;    
}

/**
* Send 버튼 클릭하는 경우 호출 (서버로 메시지 전송)
*/

function sendButton_onclick() {
    var inputMsgBox = document.getElementById(“inputMsgBox”);
    if (inputMsgBox == null || inputMsgBox.value == null || inputMsgBox.value.length == 0) {
        return false;
    }
    
    var chatBoxArea = document.getElementById(“chatBoxArea”);
    
    if (g_webSocket == null || g_webSocket.readyState == 3) {
        chatBoxArea.value += “Server is disconnected.\n”;
        return false;
    }
    
    // 서버로 메시지 전송
    g_webSocket.send(inputMsgBox.value);
    inputMsgBox.value = “”;
    inputMsgBox.focus();
    
    return true;
}

/**
* Disconnect 버튼 클릭하는 경우 호출
*/

function disconnectButton_onclick() {
    if (g_webSocket != null) {
        g_webSocket.close();    
    }
}

/**
* inputMsgBox 키입력하는 경우 호출
*/

function inputMsgBox_onkeypress() {
    if (event == null) {
        return false;
    }
    
    // 엔터키 누를 경우 서버로 메시지 전송
    var keyCode = event.keyCode || event.which;
    if (keyCode == 13) {
        sendButton_onclick();
    }
}
</script>
</head>
<body>
    <input id=“inputMsgBox” style=“width: 250px;” type=“text” onkeypress=“inputMsgBox_onkeypress()”>
    <input id=“sendButton” value=“Send” type=“button” onclick=“sendButton_onclick()”>
    <input id=“disconnectButton” value=“Disconnect” type=“button” onclick=“disconnectButton_onclick()”>
    <br/>
    <textarea id=“chatBoxArea” style=“width: 100%;” rows=“10” cols=“50” readonly=“readonly”></textarea>
</body>
</html>

 

6. OOOController.java 파일 수정
@Controller 가 붙어있는 아무 컨트롤러나 상관없다. @Controller 가 붙어있는 파일이 없다면 새로 생성하자.
도메인:포트/chat 을 주소창에 입력하면 chat.jsp 파일이 열리도록 컨트롤러를 내에 chat 메서드를 만들고 @RequestMapping 어노테이션을 붙여준다.

package 원하는_패키지명;

import java.util.Locale;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

@Controller
public class MainController {

    @RequestMapping(value=“/”)
    public String main() {
        return “index”;
    }
    
    @RequestMapping(value = “/chat”, method = RequestMethod.GET)
    public String chat(Locale locale, Model model) {
        return “chat”;
    }

}

 

7. 테스트

브라우저 여러 개를 띄워, 도메인:포트/chat (예를 들어 localhost:8080/chat) 에 접속한 후 채팅을 해보자.

[JAVA] 특정 URL 헤더의 쿠키 가져오기 (getCookiesOnly)

[JAVA] 특정 URL 헤더의 쿠키 가져오기 (getCookiesOnly)

특정 URL의 내용(responseBody)을 얻기 위해서는 일반적으로 자바에서 기본 제공하는 java.net.HttpURLConnection 클래스를 사용한다. 그런데 해당 클래스로는 응답코드가 301, 302일 경우(Location 으로 리다이렉트하는 경우)에는 Header 내의 Cookie 값들을 제대로 가져올 수 없었다.

대신 아래와 같이 GetMethod 클래스를 사용해보니 Cookie 값부터 Locaion 값까지 Header 내용을 잘 가져올 수 있었다.

 1. 라이브러리 임포트

/WEB-INF/lib/ 경로에 commons-httpclient-3.1.jar, commons-logging-1.0.4.jar, commons-codec-1.10.jar 파일을 위치시키기

또는 pom.xml 에 아래 내용 입력

<!– https://mvnrepository.com/artifact/commons-httpclient/commons-httpclient –>
<dependency>
    <groupId>commons-httpclient</groupId>
    <artifactId>commons-httpclient</artifactId>
    <version>3.1</version>
</dependency>

<dependency>
    <groupId>commons-codec</groupId>
    <artifactId>commons-codec</artifactId>
    <version>1.10</version>
</dependency>

2. 특정 URL 의 쿠키 가져오기 예제

import java.io.IOException;

import org.apache.commons.httpclient.Header;
import org.apache.commons.httpclient.HttpClient;
import org.apache.commons.httpclient.methods.GetMethod;

public class GetMethodTest {
   
   
    public static void main(String[] args) {
        try {
            GetMethodTest instance = new GetMethodTest();
            String cookies = instance.getCookiesOnly(“https://blog.naver.com/bb_”);
            System.out.println(“———-“);
            System.out.println(“cookies : ” + cookies);
           
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
   
   
    private String getCookiesOnly(String targetUrl) throws IOException, Exception {
        StringBuffer cookieBuff = new StringBuffer();
       
        HttpClient httpClient = null;
        GetMethod getMethod = null;
       
        try {
            httpClient = new HttpClient();
            getMethod = new GetMethod(targetUrl);
            getMethod.setFollowRedirects(false);

            httpClient.getHttpConnectionManager().getParams().setConnectionTimeout(10 * 1000);
            httpClient.getHttpConnectionManager().getParams().setSoTimeout(60 * 1000);

            int statusCode = httpClient.executeMethod(getMethod);

            Header[] headers = getMethod.getResponseHeaders();
            if (headers != null && headers.length > 0) {
                int headerCount = headers.length;
                for (int i=0; i<headerCount; i++) {
                    Header header = headers[i];
                    if (header == null) {
                        continue;
                    }
                   
                    if (header.getName() == null) {
                        continue;
                    }
                   
                    if (header.getValue() == null) {
                        continue;
                    }
                   
                    System.out.println(“header (” + i + “) : ” + header.getName() + ” : ” + header.getValue());
                   
                    if (header.getName().equalsIgnoreCase(“Set-Cookie”)) {
                        String oneCookie = header.getValue();
                        int idxSemicolon = oneCookie.indexOf(“;”);
                        if (idxSemicolon > -1) {
                            oneCookie = oneCookie.substring(0, idxSemicolon + 1);
                        }
                       
                        // 빈값 세팅은 skip
                        if (oneCookie.endsWith(“=\”\”;”) || oneCookie.endsWith(“=”;”)) {
                            continue;
                        }
                       
                        if (cookieBuff.length() > 0) {
                            cookieBuff.append(” “);
                        }
                       
                        cookieBuff.append(oneCookie);
                    }
                }
            }
           
            System.out.println(“statusCode : ” + statusCode);
            // System.out.println(“responseBody : ” + getMethod.getResponseBodyAsString());
           
        } catch (IOException e) {
            throw e;

        } catch (Exception e) {
            throw e;

        } finally {
            try {
                if (getMethod != null) {
                    getMethod.releaseConnection();
                }
            } catch (Exception e) {
            } finally {
                getMethod = null;
            }

            httpClient = null;
        }

        return cookieBuff.toString();
    }
}

3. 실행결과

header (0) : Date : Tue, 17 Nov 2020 03:21:02 GMT
header (1) : Content-Type : text/html;charset=UTF-8
header (2) : Content-Length : 2671
header (3) : Connection : close
header (4) : Vary : Accept-Encoding
header (5) : Vary : Accept-Encoding
header (6) : Cache-Control : no-cache
header (7) : Expires : Thu, 01 Jan 1970 00:00:00 GMT
header (8) : Set-Cookie : JSESSIONID=2E502B130F6A70B9A92D812644EFAAF7.jvm1; Path=/; Secure; HttpOnly
header (9) : Server : nxfps
header (10) : Referrer-policy : unsafe-url
statusCode : 200
———-
cookies : JSESSIONID=2E502B130F6A70B9A92D812644EFAAF7.jvm1;

4. 쿠키의 활용 (특정 페이지의 내용 가져오기)

cookies 변수는 아래와 같은 형태를 갖는다.

name1=value1;name2=value2;name3=value3;

이렇게 얻은 쿠키를 세팅해서 특정 URL의 내용(responseBody)를 가져오기 위해서는 HttpURLConnection 클래스의 setRequestProperty 메서드를 사용하면 된다.

예를 들면 아래와 같다.

String strUrl = “https://blog.naver.com/bb_”;
URL urlObj = new URL(strUrl);
HttpURLConnection conn = (HttpURLConnection) urlObj.openConnection();

(중략)

String cookies = “name1=value1;name2=value2;name3=value3;”;
conn.setRequestProperty(“Cookie”, cookies);

(후략)

참고삼아 HttpURLConnection 클래스로 특정 URL의 내용(responseBody)를 가져오는 코드를 남겨둔다. (GET방식)

public String getResponseBody(String strUrl, String encoding, String cookies) throws IOException, Exception {

        URL urlObj = null;
        HttpURLConnection conn = null;
        InputStreamReader isr = null;
        BufferedReader br = null;
        StringBuffer response = new StringBuffer();

        try {
            urlObj = new URL(strUrl);
            conn = (HttpURLConnection) urlObj.openConnection();
            conn.setDoOutput(true);
            conn.setUseCaches(false);
            conn.setRequestProperty(“access-type”, “AJAX”);
            conn.setConnectTimeout(30 * 1000);
            conn.setReadTimeout(30 * 1000);

            // 쿠키 세팅

            if (cookies != null && cookies.length() > 0) {
                conn.setRequestProperty(“Cookie”, cookies);
            }

            isr = new InputStreamReader(conn.getInputStream(), encoding);
            br = new BufferedReader(isr);

            String oneLine = null;
            while ((oneLine = br.readLine()) != null) {
                response.append(oneLine);
            }

        } catch (IOException e) {
            throw e;

        } catch (Exception e) {
            throw e;

        } finally {
            try {
                if (isr != null) {
                    isr.close();
                }
            } catch (IOException e) {
            } catch (Exception e) {
            } finally {
                isr = null;
            }
           
            try {
                if (br != null) {
                    br.close();
                }
            } catch (IOException e) {
            } catch (Exception e) {
            } finally {
                br = null;
            }
           
            try {
                if (conn != null) {
                    conn.disconnect();
                }
            } catch (NullPointerException e) {
            } catch (Exception e) {
            } finally {
                br = null;
            }
        }
       
        return response.toString();
    }

5. 쿠키의 활용 (특정 페이지로 페이지 리다이렉트)

cookies 변수는 아래와 같은 형태를 갖는다.

name1=value1;name2=value2;name3=value3;

이렇게 얻은 쿠키를 세팅해서 특정 페이지를 열 수 있다.

우선 백엔드 단에서는 해당 쿠키를 어트리뷰트에 담아서(ex : pageContext.setAttribute(“pageCookies”, cookies);) 프론트엔드로 내려주고, 이후 자바스크립트 단에서 document.cookie 값을 바꾸고 location.href 로 페이지를 이동시키면 된다.

아래는 jsp 페이지 예제다.

<%@page language=”java” %>
<%@page contentType=”text/html; charset=UTF-8″ %>

<html>

<script>

window.onload = function() {

    // document.cookie = “name1=value1;name2=value2;name3=value3;”;

    document.cookie = “${pageCookies}”;

    location.href = “http://이동하기_원하는_특정_페이지“;

}

</script>

</html>

[JAVA] json 파싱

[JAVA] json 파싱

1. 라이브러리 임포트

/WEB-INF/lib/ 경로에 json-20200518.jar 파일을 위치시키기

또는 pom.xml 에 아래 내용 입력

<!– https://mvnrepository.com/artifact/org.json/json –>
<dependency>
    <groupId>org.json</groupId>
    <artifactId>json</artifactId>
    <version>20200518</version>
</dependency>

2. 자바 json 파싱 예제

import org.json.JSONArray;
import org.json.JSONObject;

/**
 * org.json.JSONObject 라이브러리는 간편하게 json 파싱을 지원하나,
 * 해당하는 요소가 없을 경우 여지없이 Exception 을 발생시킨다.
 * 따라서 불필요한 오류가 발생하지 않도록 본 유틸 클래스로 감싸서 사용.
 */
public class JsonUtil {
   
   
    /**
     * 문자열을 json 객체로 파싱
     *
     * @param strJson
     * @return
     */
    public static JSONObject parseJsonObject(String strJson) {
        JSONObject result = null;
        try {
            result = new JSONObject(strJson);
           
        } catch (NullPointerException e) {
        } catch (Exception e) {}
       
        return result;
    }
   
   
    /**
     * json 객체에서 특정키로 json 객체 가져오기
     *
     * @param jsonObj
     * @param key
     * @return
     */
    public static JSONObject getJsonObject(JSONObject jsonObj, String key) {
        JSONObject result = null;
        try {
            result = jsonObj.getJSONObject(key);
           
        } catch (NullPointerException e) {
        } catch (Exception e) {}
       
        return result;
    }
   
   
    /**
     * json 객체에서 특정키로 json 배열 가져오기
     *
     * @param jsonObj
     * @param key
     * @return
     */
    public static JSONArray getJSONArray(JSONObject jsonObj, String key) {
        JSONArray result = null;
        try {
            result = jsonObj.getJSONArray(key);
           
        } catch (NullPointerException e) {
        } catch (Exception e) {}
       
        return result;
    }
   
   
    /**
     * json 객체에서 특정키로 문자열 가져오기
     *
     * @param jsonObj
     * @param key
     * @return
     */
    public static String getString(JSONObject jsonObj, String key) {
        String result = “”;
        try {
            result = jsonObj.getString(key);
            if (result == null) {
                result = “”;
            }
       
        } catch (NullPointerException e) {
        } catch (Exception e) {}
       
        return result;
    }
}

[JAVA] removeFirstSlice, removeLastSlice

[JAVA] removeFirstSlice, removeLastSlice

/**
     * 문자열(text) 안에 특정 문자열(slice)이 존재할 경우 앞에서부터 1개 제거
     *
     * @param text
     * @param slice
     * @return
     */
    public static String removeFirstSlice(String text, String slice) {
        if (text == null || text.length() == 0) {
            return “”;
        }
       
        if (slice == null || slice.length() == 0) {
            return text;
        }
       
        int idxSlice = text.indexOf(slice);
        if (idxSlice > -1) {
            text = text.substring(0, idxSlice) + text.substring(idxSlice + slice.length());
        }
       
        return text;
    }
   
   
    /**
     * 문자열(text) 안에 특정 문자열(slice)이 존재할 경우 뒤에서부터 1개 제거
     *
     * @param xml
     * @param slice
     * @return
     */
    public static String removeLastSlice(String text, String slice) {
        if (text == null || text.length() == 0) {
            return “”;
        }
       
        if (slice == null || slice.length() == 0) {
            return text;
        }
       
        int idxSlice = text.lastIndexOf(slice);
        if (idxSlice > -1) {
            text = text.substring(0, idxSlice) + text.substring(idxSlice + slice.length());
        }
       
        return text;
    }

[Tomcat] Apache 1개 Tomcat 2개 설정 (아파치 1개 톰캣 2개 설정)

[Tomcat] Apache 1개 Tomcat 2개 설정 (아파치 1개 톰캣 2개 설정)

Apache 1개에 Tomcat 2개 연결하는 방법. 기존에 톰캣과 연동되어 있는 아파치에, 포트를 다르게 하여 톰캣을 하나 추가하는 방법이다.

각각의 톰캣이 잘 동작하는 상태라고 전제하고, Apache 웹서버 설정 방법만 남겨둔다.

[AS-IS]

Apache 80 포트 => Tomcat(1)에 연결. 해당 톰캣의 포트는 8080, 리다이렉트 포트는 8443, AJP 포트는 8009, 셧다운 포트는 8005.

[TO-BE]

Apache 80 포트 => Tomcat(1)에 연결. 해당 톰캣의 포트는 8080, 리다이렉트 포트는 8443, AJP 포트는 8009, 셧다운 포트는 8005.

Apache 9090 포트 => Tomcat(2)에 연결. 해당 톰캣의 포트는 18080, 리다이렉트 포트는 18443, AJP 포트는 18009, 셧다운 포트는 18005.

1. conf/httpd.conf 파일 수정

[AS-IS]

Listen 80

[TO-BE]

Listen 80

Listen 9090

[AS-IS]

LoadModule jk_module modules/mod_jk.so
<IfModule mod_jk.c>
  JkWorkersFile conf/workers.properties
  JkLogFile logs/mod_jk.log
  JkLogLevel info

  JkMount /* worker1
</IfModule>

[TO-BE]

LoadModule jk_module modules/mod_jk.so
<IfModule mod_jk.c>
  JkWorkersFile conf/workers.properties
  JkLogFile logs/mod_jk.log
  JkLogLevel info

  JkMount /* worker1
</IfModule>

<VirtualHost *:9090>
  DocumentRoot “C:/tomcat2/webapps/ProjectName”
  JkLogFile logs/mod_jk.log
  JkLogLevel info


  JkMount /* worker2
</VirtualHost>

2. conf/workers.properties 파일 수정

[AS-IS]

worker.list=worker1
worker.worker1.port=8009
worker.worker1.host=localhost
worker.worker1.type=ajp13
worker.worker1.lbfactor=1


[TO-BE]

worker.list=worker1,worker2
worker.worker1.port=8009
worker.worker1.host=localhost
worker.worker1.type=ajp13
worker.worker1.lbfactor=1

worker.worker2.port=18009
worker.worker2.host=localhost
worker.worker2.type=ajp13
worker.worker2.lbfactor=1

 

[CSS] input border highlight

[CSS] input border highlight

HTML 인풋박스에 커서가 포커싱되어있을 때 테두리 하이라이트(border highlight) 수정하는 방법.

포커스된 인풋박스 테두리 없애기

<style>

input:focus {
    outline: 0px;
}

</style>

포커스된 인풋박스 테두리 수정 (ex : 하늘색 5px)

<style>

input:focus {
    outline: 5px solid #80dbff;
}

</style>

아이폰 14.2에서 인풋박스 테두리 하이라이트가 깨지는 현상이 있어서 위와 같이 css를 수정해봤는데 개선 효과가 있다.

[WebtoB/Tmax] WebtoB/Tmax 환경에서 자바스크립트 페이지 이동(location.href) 동작하지 않는 문제

[WebtoB/Tmax] WebtoB/Tmax 환경에서 자바스크립트 페이지 이동(location.href) 동작하지 않는 문제

WebtoB/Tmax 오류라고 하기는 어려운 문제이지만, 트러블 슈팅으로 며칠을 끌었던 문제이기에 여기에 기록해둠.

Apache/Tomcat 서버 구성에서 잘 동작하던 웹 어플리케이션을 WebtoB/Tmax 구성으로 변경했을 때 문제가 생겼음.

처음으로 인지했던 문제는, 첫화면에 해당하는 jsp 페이지가 하얗게 나오는 현상이었음.

여러가지 테스트 결과 웹 브라우저 주소창에 피지컬하게 주소를 입력하면 페이지가 보이지만, 자바스크립트(javascript, js) 단의 페이지 이동(location.href) 코드가 동작하지 않는 문제였음.

정확히 얘기하면 location.href = “http ://도메인주소:포트/test/page.jsp” 이러한 코드는 동작하는데, 도메인을 빼고 location.href = “/test/page.jsp” 라고 쓰여져 있는 경우 페이지 이동이 되지 않는 문제였음.

첫화면에 해당하는 jsp 페이지가 하얗게 나온 이유도, 해당 페이지가 로드되자마자(window.onload) location.href 코드를 사용하기 때문이었음.

WAS 엔지니어들도 자기들 문제가 아니라고 하면서 해결하지 못했음.

결국 며칠 동안 수정하지 못했던 이 문제는 다른 업체의 웹 어플리케이션에서도 동일한 현상이 재현되면서 해결됨.

WebtoB 설정을 고치면 되는 문제였음.

WebtoB 환경설정 파일인 http.m 파일에서 Headers 라는 부분을 주석처리하면 해결되었음.

아마도 도메인이 들어있지 않은 location.href 코드를 보안에 문제있는 코드로 인식한 것 같음.

http.m 파일

[AS-IS]

Headers = “security,X-XSS,X-Content,strict,X-Download”,

[TO-BE]

#Headers = “security,X-XSS,X-Content,strict,X-Download”,

아마 Headers 에 적혀있는 값 전부가 문제가 되진 않을 것 같음.

한 개 정도가 문제가 될테니 몇 개를 빼고 넣으면서 테스트해보면 알아냈을테지만, 운영 환경이고 그렇게까지 테스트할 상황이 안된다고 했기 때문에 Headers를 전체 주석처리 하였음.

이후 자바스크립트의 location.href 코드가 정상적으로 잘 동작함.

WAS 엔지니어도 모르는 현상이라고 하고, 고치고 나서도 모르겠다고 하는걸 보면 잘 알려지지 않은 속성들 같음.

끝.

[JAVA] AES256 암호화 예제

[JAVA] AES256 암호화 예제

자바에서 AES256 으로 평문을 암호화하고, 암호화된 값을 평문으로 복호화하는 예제이다.

Secret Key 가 “0123456789abcdefghij0123456789ab”일 때 encryptAES 메서드를 사용하면 평문 “Hello World!” 가 아래와 같이 AES256 암호화 된다.

Secret Key : 0123456789abcdefghij0123456789ab

입력 : Hello World!
결과 : YanblGzXpM13KWrqVqhMYA==

(만약 Base64로 만들 때 Base64.encodeBase64URLSafeString 을 사용하면, 결과는 YanblGzXpM13KWrqVqhMYA)

여기서 Secret Key는 필자가 임의로 정한 것이다. 원하는 임의의 32자리 문자열을 사용하면 된다.

암호화할 때의 Secret Key와 복호화할 때의 Secret Key가 일치하기만 하면 된다.

아래 메서드에서 키의 길이는 16, 24, 32 만 지원한다.

키의 길이가 32 이면 AES-256 이라 부르고, 키의 길이가 24 이면 AES-192, 키의 길이가 16 이면 AES-128 이라 부른다.

참고로 Base64 는 org.apache.commons.codec.binary.Base64 패키지를 import 해야 한다.

import java.util.Arrays;

import javax.crypto.Cipher;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;

import org.apache.commons.codec.binary.Base64;

public class AES256Util {

        public static void main(String[] args) {
        AES256Util aesUtil = new AES256Util();
       
        String originText = “Hello World!”;
        System.out.println(“originText : ” + originText);
       
        String encText = aesUtil.encryptAES(“0123456789abcdefghij0123456789ab”, originText, false);
        System.out.println(“encText (encodeBase64) : ” + encText);
       
        encText = aesUtil.encryptAES(“0123456789abcdefghij0123456789ab”, originText, true);
        System.out.println(“encText (encodeBase64URLSafeString) : ” + encText);
       
        String decText = aesUtil.decryptAES(“0123456789abcdefghij0123456789ab”, encText);
        System.out.println(“decText : ” + decText);
    }
    
    
    public String encryptAES(String keyString, String plainText, boolean bUrlSafe) {
        String cipherText = “”;
        if ((keyString == null) || keyString.length() == 0 || (plainText == null) || plainText.length() == 0) {
            return cipherText;
        }
       

        // 키의 길이는 16, 24, 32 만 지원
        if ((keyString.length() != 16) && (keyString.length() != 24) && (keyString.length() != 32)) {
            return cipherText;
        }
       
        try {
            byte[] keyBytes = keyString.getBytes(“UTF-8”);
            byte[] plainTextBytes = plainText.getBytes(“UTF-8”);

            Cipher cipher = Cipher.getInstance(“AES/CBC/PKCS5Padding”);
            int bsize = cipher.getBlockSize();
            IvParameterSpec ivspec = new IvParameterSpec(Arrays.copyOfRange(keyBytes, 0, bsize));
           
            SecretKeySpec secureKey = new SecretKeySpec(keyBytes, “AES”);
            cipher.init(Cipher.ENCRYPT_MODE, secureKey, ivspec);
            byte[] encrypted = cipher.doFinal(plainTextBytes);

            if (bUrlSafe) {
                cipherText = Base64.encodeBase64URLSafeString(encrypted);
            } else {
                cipherText = new String(Base64.encodeBase64(encrypted), “UTF-8”);
            }
           
        } catch (Exception e) {
            cipherText = “”;
            e.printStackTrace();
        }

        return cipherText;
    }
   
   
    public String decryptAES(String keyString, String cipherText) {
        String plainText = “”;
        if ((keyString == null) || keyString.length() == 0 || (cipherText == null) || cipherText.length() == 0) {
            return plainText;
        }
       
        if ((keyString.length() != 16) && (keyString.length() != 24) && (keyString.length() != 32)) {
            return plainText;
        }

        try {
            byte[] keyBytes = keyString.getBytes(“UTF-8”);
            byte[] cipherTextBytes = Base64.decodeBase64(cipherText.getBytes(“UTF-8”));

            Cipher cipher = Cipher.getInstance(“AES/CBC/PKCS5Padding”);
            int bsize = cipher.getBlockSize();
            IvParameterSpec ivspec = new IvParameterSpec(Arrays.copyOfRange(keyBytes, 0, bsize));
           
            SecretKeySpec secureKey = new SecretKeySpec(keyBytes, “AES”);
            cipher.init(Cipher.DECRYPT_MODE, secureKey, ivspec);
            byte[] decrypted = cipher.doFinal(cipherTextBytes);

            plainText = new String(decrypted, “UTF-8”);
           
        } catch (Exception e) {
            plainText = “”;
            e.printStackTrace();
        }

        return plainText;
    }
}

실행결과

originText : Hello World!
encText (encodeBase64) : YanblGzXpM13KWrqVqhMYA==
encText (encodeBase64URLSafeString) : YanblGzXpM13KWrqVqhMYA
decText : Hello World!

문제해결

만약 AES256 암호화 메서드 실행 시 java.security.InvalidKeyException: Illegal key size 오류가 발생하는 경우

이유는 자바가 기본적으로 키 길이를 128bit까지 제한하기 때문이다.
제한을 풀기 위해서는 번들로 제공하는(따로 제공하는) JCE Unlimited Strength 정책 파일을 다운받아서 적용해야 한다.

* JCE Unlimited Strength 정책 파일 다운로드
Java6 : jce_policy-6.zip
http://www.oracle.com/technetwork/java/javase/downloads/jce-6-download-429243.html

Java7 : UnlimitedJCEPolicyJDK7.zip
http://www.oracle.com/technetwork/java/javase/downloads/jce-7-download-432124.html

Java8 : jce_policy-8.zip
http://www.oracle.com/technetwork/java/javase/downloads/jce8-download-2133166.html

* JCE Unlimited Strength 정책 파일의 적용
Java8 을 예로 들겠다.

오라클 사이트에서 jce_policy-8.zip 파일을 다운받아서 압축을 풀면 UnlimitedJCEPolicyJDK8 폴더가 나온다.
UnlimitedJCEPolicyJDK8 폴더 안에는 local_policy.jar, US_export_policy.jar, README.txt 이렇게 파일 3개가 있다.

해당 jar 파일 2개를 복사해서 C:\[jdk1.8경로]\jre\lib\security 폴더 안에 붙여넣기(덮어쓰기) 하면 된다.
ex) C:\Java\jdk1.8.0_112\jre\lib\security\local_policy.jar 덮어쓰기

및 C:\Java\jdk1.8.0_112\jre\lib\security\US_export_policy.jar 덮어쓰기
(당연히 파일 덮어쓰기 전 기존 파일은 백업해둘 것)

AES256 검증

아래 사이트에서 AES256 암호화 결과 및 복호화 결과를 검증해볼 수 있다.

검증 사이트 : https://www.devglan.com/online-tools/aes-encryption-decryption 

Select Mode : CBC
Key Size in Bits : 256 bit
IV : 0123456789abcdef
Secret Key : 0123456789abcdefghij0123456789ab
Output Text Format : base64

참고로 IV 값은 Secret Key 값의 절반(16자리)을 입력하면 된다.

참고사이트 : https://hongsii.github.io/2018/04/05/java-aes256-error/

[SpringBoot] 이클립스(STS) 에서 스프링부트(Spring Boot) 시작하기

[SpringBoot] 이클립스(STS) 에서 스프링부트(Spring Boot) 시작하기

1. STS를 실행한다.

 

2. 원하는 워크스페이스 경로를 적고 [Launch] 버튼을 클릭한다.

 

3. STS 상단메뉴 [File] – [New] – [Spring Starter Project] 메뉴를 클릭한다.

 

4. Name 에 프로젝트 명을 적는다. (ex : TestProject)

Type 콤보박스는 Maven 을 선택한다.

Packaging 콤보박스는 Jar 를 선택한다.

Java Version 콤보박스는 원하는 버전을 선택한다.

Language 콤보박스는 Java 를 선택한다.

Group 은 패키지의 상위경로를 적는다.(ex : com.company)  보통 도메인 주소의 역순이다. 예를 들어 도메인이 company.com 일 경우 패키지 상위경로는 com.company 가 된다.

Atrifact 에는 프로젝트약칭을 적는다. (ex : test-project) 라이브러리화 됐을 때 jar에 들어가는 이름이라고 생각하면 된다.

Version 은 그대로 두거나 버전값을 적는다. (ex : 1.0)

Description 은 프로젝트 설명을 적는다. (ex : Spring Boot Project)

Package 에는 패키지 경로를 적는다. 패키지 상위경로 뒤에 프로젝트명을 붙인다. (ex : com.company.project)

하단의 [Next] 버튼을 클릭한다.

 

5. 검색창에 키워드를 입력해서 4가지를 추가한다. (1) Spring Boot DevTools (2) MyBatis Framework (3) MySQL Driver (4) Spring Web 을 추가하면 된다.

하단의 [Next] 버튼을 클릭한다.

 

6. 하단의 [Finish] 버튼을 클릭한다.

 

7. src/main/resources 하위의 application.properties 파일을 열고 아래와 같이 작성한다.

spring.mvc.view.prefix=/WEB-INF/views/
spring.mvc.view.suffix=.jsp
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost:3306/SchemaName
spring.datasource.username=username
spring.datasource.password=password
server.servlet.encoding.charset=utf-8
server.servlet.encoding.enabled=true
server.servlet.encoding.force=true

위의 MySQL 정보는 현재 프로젝트에 MySQL Driver 를 추가했기 때문에 오류가 나지 않도록 작성하는 것이다.

일단은 실제 MySQL 정보가 아니어도 좋고, 위와 똑같이 입력해도 프로젝트는 정상적으로 기동할 수 있다.

 

8. 프로젝트 폴더 하위의 pom.xml 파일을 열고 아래 내용을 추가한다.

    <!– TOMCAT –>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-tomcat</artifactId>
        <scope>provided</scope>
    </dependency>

    <!– JSP –>
    <dependency>
        <groupId>org.apache.tomcat.embed</groupId>
        <artifactId>tomcat-embed-jasper</artifactId>
    </dependency>

    <!– JSTL –>
    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>jstl</artifactId>
    </dependency>

닫는 태그인 </dependencies> 태그 바로 위에 붙여넣기 하면 된다.

 

9. 계속해서 pom.xml 파일을 수정한다. STS 하단의 Problems 탭을 봤을 때 “Maven Configuration Problem” 이 발생해서, 이를 해결하기 위해 아래 내용을 추가한다.

    <!– Maven Configuration Problem 해결 –>
    <maven-jar-plugin.version>3.1.1</maven-jar-plugin.version>

닫는 태그 </java.version> 밑에 추가하면 된다.

(참고사이트 : https://stackoverflow.com/questions/56212981/eclipse-showing-maven-configuration-problem-unknown)

 

10. 만약 STS 하단 Problems 탭에 “Project configuration is not up-to-date with pom.xml. Select Maven->Update Project… from the project context menu or use Quick Fix.” 오류 메시지가 표시되는 경우.

해당 항목을 마우스 우클릭하고 [Quick Fix] 메뉴를 클릭해서 관련 오류를 제거한다.

 

11. src/main/java 하위 com.company.project 패키지 안에 MainController.java 파일을 만들고 아래와 같이 내용 작성한다.

package com.company.project;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller
public class MainController {

    @RequestMapping(value=”/”)
    public String main() {
        return “index”;
    }
}

 

12. src/main 폴더 아래에 webapp 폴더 생성, 그 아래에 WEB-INF 폴더 생성, 그 아래에 views 폴더를 생성, 그 아래에 index.jsp 파일을 생성한다.

다시 말해 index.jsp 의 경로가 src/main/webapp/WEB-INF/views/index.jsp 가 되도록 관련 폴더 및 파일을 생성한다.

index.jsp 파일 내용은 다음과 같이 작성한다.

<html>
<head>
</head>
<body>
Hello World!
</body>
</html>

 

13. 프로젝트 폴더에서 마우스 우클릭 – [Run As] – [Spring Boot App] 으로 서버를 기동한다.

 

14. 크롬 등 웹 브라우저에서 localhost:8080 로 접속하면 Hello World! 가 표시된다.

 

스프링 부트 기본 프로젝트 구성이 정상적으로 완료되었다.

참고사이트 : https://badstorage.tistory.com/10


[HTML/CSS] CSS 점점점 말줄임표 효과 ellipsis

[HTML/CSS] CSS 점점점 말줄임표 효과 ellipsis

<html>
<head>
    <style>
    .ellipsis_test {
        width: 300px;
        white-space: nowrap;
        overflow: hidden;
        text-overflow: ellipsis;
    }
    </style>
</head>
<body>
    <div class=”ellipsis_test”>문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다 문자열을 길게 입력해본다</div>
</body>
</html>

[JAVA] heap dump 파일 생성하기 (jmap 명령어 사용방법)

[JAVA] heap dump 파일 생성하기 (jmap 명령어 사용방법)

리눅스 또는 윈도우 운영체제에 JDK 가 설치된 상태여야 한다. (JRE 만 설치한 경우 jmap 없음)

1. JVM 의 PID 알아내기

jps -v

2. 특정 PID에 대한 heap dump 파일 생성하기

jmap -dump:format=b,file=test_20201105.hprof [PID]

ex) jmap -dump:format=b,file=test_20201105.hprof 19368

[iOS] 아이폰 ipa 다운로드 링크 방법 (plist 파일 작성방법)

[iOS] 아이폰 ipa 다운로드 링크 방법 (plist 파일 작성방법)

아이폰으로 접속한 특정 웹페이지에서 ipa 파일을 다운로드 및 설치하게 하려면 어떻게 해야할까?

전체적인 이해를 돕기 위해 안드로이드 먼저 알아보자.

1. 안드로이드 apk 파일 다운로드 링크 방법


안드로이드의 앱 설치파일 확장자는 apk 다. apk 파일은 html 상에서 a 태그로 링크를 걸면 안드로이드 기기에서 해당 링크 클릭 시 apk 파일을 다운로드 받을 수 있고, 이어서 앱을 설치할 수 있다.

예를 들어 <a href=”https://특정도메인/test.apk“>클릭</a> 이라고 html 을 작성했다면 안드로이드 기기에서 “클릭” 버튼을 눌렀을 때 apk 파일 다운로드 및 설치가 잘 이루어진다는 뜻이다.

2. 아이폰 ipa 파일 다운로드 링크 방법

한편, 아이폰의 앱 설치파일 확장자는 ipa 다. ipa 파일은 안드로이드와 달리, html 상에서 a 태그로 링크를 걸어도 소용없다. 아이폰에서 해당 링크 클릭 시 ipa 파일을 다운로드 받을 수 없고, 설치를 진행할 수 없다.

예를 들어 <a href=”https://특정도메인/test.ipa“>클릭</a> 이라고 html 을 작성했다면 아이폰에서 “클릭” 버튼을 눌렀을 때 ipa 파일 다운로드 및 설치는 진행되지 않는다.

아이폰에서 특정 링크를 클릭 시 ipa 파일을 다운로드 및 설치하려면, 우선 plist 라는 확장자로 파일을 작성해야 한다. plist 파일을 만들고, 해당 plist 파일 안에 ipa 정보를 입력해야 한다. 그리고 a 태그에는 itms-services 라는 프로토콜로 plist 를 링크시켜야 한다.

예를 들어 plist 를 다운받을 수 있는 주소가 https://특정도메인/test.plist 라면, a 태그는 다음과 같이 작성한다.

<a href=”itms-services://?action=download-manifest&url=https://특정도메인/test.plist“>

그리고 plist 확장자 파일은 아래 예시와 같이 작성한다.

test.plist 예시

<?xml version=”1.0″ encoding=”UTF-8″?>
<!DOCTYPE plist PUBLIC “-//Apple//DTD PLIST 1.0//EN” http://www.apple.com/DTDs/PropertyList-1.0.dtd>
<plist version=”1.0″>
<dict>
    <key>items</key>
    <array>
        <dict>
            <key>assets</key>
            <array>
                <dict>
                    <key>kind</key>
                    <string>software-package</string>
                    <key>url</key>
                    <string>https:// 특정도메인/test.ipa</string>
                </dict>
            </array>
            <key>metadata</key>
            <dict>
                <key>bundle-identifier</key>
                <string>kr.co.ddoc.project</string>
                <key>kind</key>
                <string>software</string>
                <key>title</key>
                <string>테스트앱</string>
            </dict>
        </dict>
    </array>
</dict>
</plist>

1. <key>url</key> 다음에 위치한 <string> 태그 내용으로는 ipa 파일의 URL 경로를 입력한다.

2. <key>bundle-identifier</key> 다음에 위치한 <string> 태그 내용으로는 ipa 파일의 해당하는 앱 패키지명을 입력한다.

3. <key>title</key> 다음에 위치한 <string> 태그 내용으로는 앱의 제목을 입력한다.

여기서 주의할 점은 plist 파일을 바라보는 URL이 https 프로토콜을 사용해야 한다는 점이다.

참고로 웹서버(ex : 아파치 웹서버) 또는 WAS(ex : 톰캣) 에 SSL 인증서를 입히면 http 프로토콜로 동작하던 사이트가 https 프로토콜로 동작한다.

만약 업로드할만한 마땅한 서버가 없거나 https 인증서를 입힐 상황이 안된다면 드랍박스(dropbox)라는 웹사이트를 이용할 수 있다. 드랍박스에 파일 업로드 및 다운로드 링크를 만드는 방법은 아래 글을 참고하면 된다.

드랍박스(dropbox) 파일 다운로드 주소 만드는 방법 : https://blog.naver.com/bb_/221581346975

[Oracle] 오라클 Comment 설정(추가 또는 수정), 삭제, 조회

[Oracle] 오라클 Comment 설정(추가 또는 수정), 삭제, 조회

오라클에서는 COMMENT ON 명령어를 사용하여 테이블 또는 컬럼에 참고용으로 설명을 부여할 수 있다.

1. Comment 설정(추가 또는 수정)

테이블 : COMMENT ON TABLE 테이블명 IS ‘설명내용’;

컬럼 : COMMENT ON COLUMN 테이블명.컬럼명 IS ‘설명내용’;​

2. Comment 삭제

테이블 : COMMENT ON TABLE 테이블명 IS ”;

컬럼 : COMMENT ON COLUMN 테이블명.컬럼명 IS ”;

3. Comment 조회

테이블 : SELECT * FROM ALL_TAB_COMMENTS;

컬럼 : SELECT * FROM ALL_COL_COMMENTS;

참고사이트 : https://jhnyang.tistory.com/309

[Eclipse] 이클립스에서 _jsp.java 파일 위치

[Eclipse] 이클립스에서 _jsp.java 파일 위치

로컬환경 이클립스로 톰캣을 기동했을 때 .jsp 파일은 _jsp.java 파일로 변환되고, _jsp.java 파일은 .class 파일로 컴파일된다.

이 때 jsp 파일에서 자바 익셉션이 발생하면 콘솔 또는 로그에 찍힌 StackTrace 의 라인넘버와 jsp 파일의 라인넘버가 불일치하는 것을 볼 수 있다.

StackTrace 에 기록된 라인넘버는 jsp의 라인넘버가 아니라 _jsp.java 의 라인넘버와 매칭되기 때문이다.

따라서 _jsp.java 의 위치를 알아낼 필요가 있다.

작업 중인 프로젝트의 워크스페이스로 가면 .metadata 폴더가 있다.
.metadata 폴더 내의 .plugins\org.eclipse.wst.server.core 폴더 하위에서 검색해보면 _jsp.java 파일을 찾을 수 있다.

ex) c:\프로젝트_워크스페이스\.metadata\.plugins\org.eclipse.wst.server.core\tmp0\work\Catalina\localhost\_\org\apache\jsp\파일명_jsp.java

[JAVA] getOutputStream() has already been called for this response

[JAVA] getOutputStream() has already been called for this response

jsp 에서 java.lang.IllegalStateException: getOutputStream() has already been called for this response 오류 발생하는 경우.

아웃풋스트림(ex : ServletOutputStream)을 사용하기 전에 out.clear() 로 JspWriter 객체를 비워주면 오류가 사라진다.

참고사이트 : https://kooremo.tistory.com/entry/javalangIllegalStateException-getOutputStream-has-already-been-called-for-this-respons

[JAVA] 멀티파트 폼(multipart/form-data)으로 넘긴 파라미터 request.getParameter 로 받기

[JAVA] 멀티파트 폼(multipart/form-data)으로 넘긴 파라미터 request.getParameter 로 받기

멀티파트 폼(<form enctype=”multipart/form-data”>)으로 넘긴 파라미터는 단순 request.getParameter 메서드로 받으면 null 이 나오고, multipartRequest.getParameter 메서드로 받아야 한다.

예를 들어 아래와 같은 form 으로 데이터를 전송했다면,

<form id=”test_form” target=”_blank” action=”/upload_test.jsp” method=”post” enctype=”multipart/form-data”>
    <input type=”file” multiple=”multiple” id=”file_input” name=”fileobj[]”>
    <input type=”hidden” name=”param1″ value=”value1″>
    <input type=”hidden” name=”param2″ value=”value2″>
    <input type=”hidden” name=”param3″ value=”value3″>
</form>

아래와 같이 데이터를 받아야 한다.

// 예 : upload_test.jsp 파일의 내용

// 100MB
int maxSize = 1024*1024*100;

MultipartRequest multipartRequest = new MultipartRequest(request, “C:\\test\\”, maxSize, “UTF-8”, new DefaultFileRenamePolicy());

String param1 = multipartRequest.getParameter(“param1”);
String param2 = multipartRequest.getParameter(“param2”);
String param3 = multipartRequest.getParameter(“param3”);

그런데 단순 request.getParameter 메서드로 파라미터를 받고 싶은 경우가 있다.  DefaultFileRenamePolicy 클래스를 사용하지 않고 FileRenamePolicy 인터페이스를 직접 구현(Implement)하고 싶을 때, 그 안에서 직전 페이지로부터 넘겨받은 파라미터를 사용하고 싶으면 그럴 수 있다.

이 때는 action 뒤쪽에 파라미터를 붙여서 get 으로 보내면 된다.

예를 들어 action=”/upload_test.jsp” 이 아니라, action=”/upload_test.jsp?param1=value1″ 과 같은 식으로 수정하면 된다. (cf : document.getElementById(“test_form”).action = “/upload_test.jsp?param1=value1”;)

전달받는 쪽에서는 request.getParameter(“param1”); 로 받으면 된다.

[Tomcat] 윈도우 톰캣 System.out.println 이 기록되지 않는 경우

[Tomcat] 윈도우 톰캣 System.out.println 이 기록되지 않는 경우

보통 자바에서 System.out.println 또는 System.err.println 으로 찍은 내용은 톰캣 catalina.out 파일에 찍힌다.

그런데 윈도우에서 톰캣을 구동하는 경우(cmd 콘솔에서 .bat 파일로 톰캣을 실행하는 경우), System.out.println 또는 System.err.println 내용이 콘솔에 출력되지만 로그 파일로 남지 않는 경우가 있다.

${catalina.base}/conf/context.xml 파일의 아래 한 줄을 추가하고 재기동한다.

<Context swallowOutput=”true”>

이 때 해당 내용은 catalina.out에 찍히는게 아니라 localhost-날짜.log 파일에 찍힌다.

참고로 swallowOutput 는 true인 경우에 system.out과 system.err로 출력한 바이트가 웹애플리케이션의 logger로 리다이렉션 되는 옵션값이며, 기본값은 false다.

참고사이트 1 : https://crazythink.github.io/categories/Tomcat/

참고사이트 2 : https://esysop.tistory.com/11

[JAVA] 특정 폴더 하위의 모든 파일 경로 가져오기

[JAVA] 특정 폴더 하위의 모든 파일 경로 가져오기

    /**
     * 특정 폴더 하위의 모든 파일 경로 가져오기
     *
     * @param target
     * @return
     */
    public static StringList getFilePathList(File target) {
        StringList resultList = new StringList();
        getFilePathListCore(resultList, target);
        return resultList;
    }
   
   
    /**
     * 특정 폴더 하위의 모든 파일 경로를 가져오기 (재귀)
     *
     * @param resultList
     * @param target
     */
    private static void getFilePathListCore(StringList resultList, File target) {
        if (target == null || !target.exists()) {
            return;
        }
       
        if (target.isDirectory()) {
            File[] fileArr = target.listFiles();
            if (fileArr != null && fileArr.length > 0) {
                int fileCount = fileArr.length;
                for (int i=0; i<fileCount; i++) {
                    getFilePathListCore(resultList, fileArr[i]);
                }
            }
           
        } else if (target.isFile()) {
            resultList.add(target.getAbsolutePath());
        }
    }

[JAVA] 쓰레드 덤프보는 JSP

[JAVA] 쓰레드 덤프보는 JSP

https://greatkim91.tistory.com/167

[Windows] 윈도우 업데이트 끄기(Windows Update 끄기)

[Windows] 윈도우 업데이트 끄기(Windows Update 끄기)

1. 시작 – 실행 – services.msc 입력 후 엔터 (또는 제어판 – 서비스)


 

2. [서비스] 창이 뜨면 우측 목록에서 [Windows Update] 항목 찾아서 마우스 우클릭 – 속성(R)

3. [Windows Update 속성] 창이 뜨면 [일반] 탭에서 [시작 유형]을 [자동]에서 [수동] 또는 [사용 안 함]으로 변경

하단의 서비스 상태 항목에서 [중지] 버튼 클릭

 

참고사이트 : https://gbworld.tistory.com/1184

[JAVA] 특정폴더 하위의 모든 파일 삭제

[JAVA] 특정폴더 하위의 모든 파일 삭제

    public void deleteFilesInRootDir(File rootDir) {
        deleteFilesInRootDir(rootDir, rootDir);
    }
   
   
    private void deleteFilesInRootDir(File rootDir, File target) {
        if (rootDir == null) {
            return;
        }
       
        if (target == null || !target.exists()) {
            return;
        }
       
        if (target.isDirectory()) {
            File[] files = target.listFiles();
            for (File oneFile : files) {
                if (oneFile != null && oneFile.isFile() && oneFile.exists()) {
                    try {
                        oneFile.delete();
                        // System.out.println(“delete… [” + oneFile.getAbsolutePath() + “]”);
                    } catch (Exception e) {
                    }
                } else if (oneFile != null && oneFile.isDirectory()) {
                    deleteFilesInRootDir(rootDir, oneFile);
                }
            }
           
            // delete without root folder
            if (!rootDir.getAbsolutePath().equals(target.getAbsolutePath())) {
                try {
                    target.delete();
                    // System.out.println(“delete… [” + target.getAbsolutePath() + “]”);
                } catch (Exception e) {
                }
            }
        }
    }

[Android] 안드로이드 웹뷰(WebView) 파일선택, 여러개 파일선택

[Android] 안드로이드 웹뷰(WebView) 파일선택, 여러개 파일선택

아이폰(iOS) 하이브리드 앱에서는 <input type=”file”> 을 사용하면 해당 요소를 클릭 시 파일선택 창(FileChooser)이 잘 뜨지만, 안드로이드 웹뷰에서는 <input type=”file”> 요소를 클릭해도 아무 반응이 없다.

안드로이드 하이브리드앱에서 파일을 선택하는 기능은 아래와 같이 구현 가능하다.

1. mFilePathCallback 변수를 전역변수로 선언한다.

webView 를 사용할 Activity 클래스(또는 Activity 를 상속받은 클래스) 에 선언하면 된다.

    // 안드로이드 웹뷰에서 파일 첨부하기
    ValueCallback mFilePathCallback = null;

2. WebChromeClient 클래스의 onShowFileChooser 메서드를 오버라이딩해서 아래와 같이 작성한다.

파일 n개 선택(파일 여러개 선택, 파일 다중선택) 하려면 intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true); 코드를 포함시키자.

    webView.setWebChromeClient(new WebChromeClient() {
        // 안드로이드 웹뷰에서 파일 첨부하기
        @Override
        public boolean onShowFileChooser(WebView webView, ValueCallback filePathCallback, FileChooserParams fileChooserParams) {
            mFilePathCallback = filePathCallback;

            Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
            intent.addCategory(Intent.CATEGORY_OPENABLE);
            intent.setType(“image/*”);

            // 파일 n개 선택 가능하도록 처리
            intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true);

            startActivityForResult(intent, 0);
            return true;
        }
    });

만약 WebChromeClient 를 상속받아 별개의 자바파일로 구현했다면(ex : public class OOOWebChromeClient extends WebChromeClient {}), startActivityForResult 메서드를 곧바로 사용할 수 없을 것이다.

이 경우 WebChromeClient 생성자에서 activity 를 받아 thisActivity 라는 멤버변수에 저장해놓으면 thisActivity.startActivityForResult(intent, 0); 식으로 사용할 수 있다.

public class OOOWebChromeClient extends WebChromeClient {
    private final Activity thisActivity;

    public OOOWebChromeClient(Activity thisActivity) {
        super();
        this.thisActivity = thisActivity;
    }

    (중략)

}   

3. mFilePathCallback 변수가 선언된 클래스, 다시 말해서 Activify 클래스(또는 Activity를 상속한 클래스) 하단에 아래와 같이 작성한다.

참고로 파일을 n개 선택(여러개 선택, 다중선택)한 경우 data.getClipData() 값이 not null이며, 파일을 1개 선택한 경우 data.getData() 값이 not null이다.

    // 안드로이드 웹뷰에서 파일 첨부하기
    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        Log.e(“resultCode:: “, String.valueOf(resultCode));
        if (requestCode == 0 && resultCode == Activity.RESULT_OK) {

            // 파일 n개 선택한 경우
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN && data != null && data.getClipData() != null && data.getClipData().getItemCount() > 0) {
                int count = data.getClipData().getItemCount();

                Uri[] uriArr = new Uri[count];
                for (int i=0; i<count; i++) {
                    uriArr[i] = data.getClipData().getItemAt(i).getUri();
                }

                mFilePathCallback.onReceiveValue(uriArr);

            } else if (data != null && data.getData() != null) {
                // 파일 1개 선택한 경우
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
                    mFilePathCallback.onReceiveValue(WebChromeClient.FileChooserParams.parseResult(resultCode, data));
                } else {
                    mFilePathCallback.onReceiveValue(new Uri[]{data.getData()});
                }
            }

            mFilePathCallback = null;

        } else {
            mFilePathCallback.onReceiveValue(null);
        }
    }

만약 WebChromeClient 를 상속받아 별개의 자바파일로 구현했다면(ex : public class OOOWebChromeClient extends WebChromeClient {}), mFilePathCallback 변수를 사용할 수 없을 것이다. OOOWebChromeClien 클래스 내에 mFilePathCallback 의 getter, setter 메서드를 만들어서 접근하도록 하자.

참고사이트 1 : https://g-y-e-o-m.tistory.com/115

참고사이트 2 : https://medium.com/@kyle_seongwoo_jun/%EC%95%88%EB%93%9C%EB%A1%9C%EC%9D%B4%EB%93%9C-webview-filechooser-%EC%97%AC%EB%9F%AC-%ED%8C%8C%EC%9D%BC-%EC%84%A0%ED%83%9D%ED%95%98%EB%8A%94-%EB%B0%A9%EB%B2%95-8da48465ada3 

[Javscript] 자바스크립트 post 로 전송하는 함수

[Javscript] 자바스크립트 post 로 전송하는 함수

function postUrl(_url) {
    if (_url == null || _url == “”) {
        return false;
    }
   
    var strAction = “”;
    var idxQuestion = _url.indexOf(“?”);
    if (idxQuestion > -1) {
        strAction = _url.substring(0, idxQuestion);   
    }
   
    var targetForm = document.createElement(“FORM”);
    targetForm.setAttribute(“id”, “targetForm”);
    targetForm.setAttribute(“name”, “targetForm”);
    targetForm.setAttribute(“action”, strAction);
    targetForm.setAttribute(“target”, “_self”);
    targetForm.setAttribute(“method”, “POST”);
    targetForm.setAttribute(“accept-charset”, “UTF-8”);
    targetForm.style.display = “none”;
   
    if (idxQuestion > -1) {
        var params = _url.substring(idxQuestion + 1);
       
        while (params.length > 0) {
            var idxEqual = params.indexOf(“=”);
            if (idxEqual > -1) {
                var key = params.substring(0, idxEqual);
                var value = “”;
               
                var idxAmp = params.indexOf(“&”, idxEqual + 1);
                if (idxAmp > -1) {
                    value = params.substring(idxEqual + 1, idxAmp);
                    params = params.substring(idxAmp + 1);
                } else {
                    value = params.substring(idxEqual + 1);
                    params = “”;
                }
               
                var inputElem = document.createElement(“INPUT”);
                inputElem.name = key;
                inputElem.value = value;
               
                targetForm.appendChild(inputElem);
            }
        }
    }
   

    // 기존객체 삭제
    var oldObj = document.getElementById(“targetForm”);
    if (oldObj != null) {
        document.getElementById(“targetForm”).parentNode.removeChild(oldObj);
    }

    document.body.appendChild(targetForm);
    targetForm.submit();
}

[JAVA] 다음날(1일후, 내일), 다음달(내달), 다음연도(내년) 날짜시간 구하기

[JAVA] 다음날(1일후, 내일), 다음달(내달), 다음연도(내년) 날짜시간 구하기

1. 자바에서 현재 날짜시간 구하기

기준 : 현재 날짜시간

=> 20201008160222 (2020년 10월 08일 16시 02분 22초)

    try {
        SimpleDateFormat sdf = new SimpleDateFormat(“yyyyMMddHHmmss”);

        // 현재 날짜시간 구하기
        Calendar cal = Calendar.getInstance();

       

        // 출력
        String resultDate = sdf.format(cal.getTime());

        
        String year = resultDate.substring(0, 4);
        String month = resultDate.substring(4, 6);
        String day = resultDate.substring(6, 8);
        String hour = resultDate.substring(8, 10);
        String minute = resultDate.substring(10, 12);
        String second = resultDate.substring(12, 14);

​       

        System.out.println(resultDate + ” (” + year + “년 ” + month + “월 ” + day + “일 “
        + hour + “시 ” + minute + “분 ” + second + “초)”);

        
    } catch (Exception e) {
        e.printStackTrace();
    }

2. 자바에서 특정일의 날짜시간 구하기

기준 : 20200101092030 (2020년 01월 01일 09시 20분 30초)

=> 20200101092030 (2020년 01월 01일 09시 20분 30초)

     try {
        SimpleDateFormat sdf = new SimpleDateFormat(“yyyyMMddHHmmss”);
       
        // 특정일의 날짜시간 구하기
        // 2020년 01월 01일 09시 20분 30초
        String strDate = “20200101092030”;
        Date date = sdf.parse(strDate);

        Calendar cal = Calendar.getInstance();
        cal.setTime(date);
        

        // 출력
        String resultDate = sdf.format(cal.getTime());

        String year = resultDate.substring(0, 4);
        String month = resultDate.substring(4, 6);
        String day = resultDate.substring(6, 8);
        String hour = resultDate.substring(8, 10);
        String minute = resultDate.substring(10, 12);
        String second = resultDate.substring(12, 14);

        System.out.println(resultDate + ” (” + year + “년 ” + month + “월 ” + day + “일 “
        + hour + “시 ” + minute + “분 ” + second + “초)”);

    } catch (Exception e) {
        e.printStackTrace();
    }

3. 자바에서 특정일 다음날(1일 후) 날짜시간 구하기

기준 : 20200101092030 (2020년 01월 01일 09시 20분 30초)

=> 20200102092030 (2020년 01월 02일 09시 20분 30초)

    try {
        SimpleDateFormat sdf = new SimpleDateFormat(“yyyyMMddHHmmss”);
       
        // 특정일 다음날(1일 후) 날짜시간 구하기
        // 2020년 01월 01일 09시 20분 30초의 다음날(1일 후)
        String strDate = “20200101092030”;
        Date date = sdf.parse(strDate);

        Calendar cal = Calendar.getInstance();
        cal.setTime(date);
       
        cal.add(Calendar.DATE, 1); // 다음날(1일 후)

        // 출력
        String resultDate = sdf.format(cal.getTime());

        String year = resultDate.substring(0, 4);
        String month = resultDate.substring(4, 6);
        String day = resultDate.substring(6, 8);
        String hour = resultDate.substring(8, 10);
        String minute = resultDate.substring(10, 12);
        String second = resultDate.substring(12, 14);

        System.out.println(resultDate + ” (” + year + “년 ” + month + “월 ” + day + “일 “
        + hour + “시 ” + minute + “분 ” + second + “초)”);

    } catch (Exception e) {
        e.printStackTrace();
    }

위와 같이 cal.add(Calendar.DATE, 1); 명령어를 사용하면 다음날(1일 후)의 날짜를 가져온다.

해당 라인을 수정하면 이전날(1일 전, 어제)/다음날(1일 후, 내일), 이전달(지난달)/다음달(내달), 이전연도(작년)/다음연도(내년)의 날짜를 구할 수 있다.

기준 : 20200101092030 (2020년 01월 01일 09시 20분 30초)

이전날(1일 전, 어제) : cal.add(Calendar.DATE, -1);

=> 20191231092030 (2019년 12월 31일 09시 20분 30초)

다음날(1일 후, 내일) : cal.add(Calendar.DATE, 1);

=> 20200102092030 (2020년 01월 02일 09시 20분 30초)

이전달(지난달) : cal.add(Calendar.MONTH, -1);

=> 20191201092030 (2019년 12월 01일 09시 20분 30초)

다음달(내달) : cal.add(Calendar.MONTH, 1);

=> 20200201092030 (2020년 02월 01일 09시 20분 30초)

이전연도(작년) : cal.add(Calendar.YEAR, -1);

=> 20190101092030 (2019년 01월 01일 09시 20분 30초)

다음연도(내년) : cal.add(Calendar.YEAR, 1);

=> 20210101092030 (2021년 01월 01일 09시 20분 30초)

만약 시간(시), 분, 초를 조정하고 싶다면 cal.add  명령어의 인자를 바꾸면 된다. 시간은 Calendar.HOUR_OF_DAY, 분은 Calendar.MINUTE, 초는 Calendar.SECOND 를 사용하면 된다. (1시간전/1시간후, 1분전/1분후, 1초전/1초후 등)

참고로 Calendar.HOUR_OF_DAY 는 24시간제(00:00 ~ 23:59)를 뜻하고, Calendar.HOUR 는 12시간제(AM 00:00 ~ 11:59, PM 00:00 ~ 11:59)를 뜻한다.

[JAVA] 시작일과 종료일 사이의 날짜 가져오기

[JAVA] 시작일과 종료일 사이의 날짜 가져오기

    /**
     * beginDate 부터 endDate 사이의 날짜 가져오기
     *
     * ex) beginDate == 20200930, endDate == 20201004 일 때,
     *     [20200930, 20201001, 20201002, 20201003, 20201004] 리턴
     * @param beginDate
     * @param endDate
     * @return
     * @throws Exception
     */
    private ArrayList<String> getDaysBetweenDates(String beginDate, String endDate) throws Exception {
        if (beginDate != null && beginDate.length() >= 8) {
            beginDate = beginDate.substring(0, 8);
        } else {
            return null;
        }
       
        if (endDate != null && endDate.length() >= 8) {
            endDate = endDate.substring(0, 8);
        } else {
            return null;
        }
       
        int iBeginDate = Integer.parseInt(beginDate);
        int iEndDate = Integer.parseInt(endDate);
        if (iBeginDate > iEndDate) {
            return null;
        }
       
       
        // 시작날짜 세팅
        int beginYear = Integer.parseInt(beginDate.substring(0, 4));
        int beginMonth = Integer.parseInt(beginDate.substring(4, 6));
        int beginDay = Integer.parseInt(beginDate.substring(6, 8));
       
        Calendar cal = Calendar.getInstance();
        cal.set(beginYear, beginMonth – 1, beginDay);
       
       
        ArrayList<String> resultList = new ArrayList<String>();
       
        SimpleDateFormat sdf = new SimpleDateFormat(“yyyyMMdd”);
        String oneDay = “”;
       
        // beginDate 부터 endDate 사이의 날짜 담기
        while (Integer.parseInt(sdf.format(cal.getTime())) <= iEndDate) {
            oneDay = sdf.format(cal.getTime());
            resultList.add(oneDay);
           
            // 다음날로 날짜 조정
            cal.add(Calendar.DATE, 1);
        }
       
        return resultList;
    }

참고사이트 : ​https://bugnote.tistory.com/26

 

[Tibero] 티베로 테이블명 조회, 컬럼명 조회, 테이블 상세정보 조회

[Tibero] 티베로 테이블명 조회, 컬럼명 조회, 컬럼상세정보 조회

티베로에서 테이블명 조회, 컬럼명 조회, 컬럼상세정보 조회/보기/보는 방법.

1. 티베로 테이블명 조회

SELECT * FROM ALL_TAB_COMMENTS;


SELECT * FROM ALL_TAB_COMMENTS WHERE TABLE_NAME = ‘테이블명’;

테이블명, 테이블 종류, 테이블 주석 확인 가능

2. 티베로 컬럼명 조회

SELECT * FROM ALL_COL_COMMENTS WHERE TABLE_NAME = ‘테이블명’;

테이블명, 컬럼명, 컬럼 주석 확인 가능

3. 티베로 테이블 상세정보 조회

SELECT * FROM ALL_TAB_COLUMNS WHERE TABLE_NAME = ‘테이블명’; 

테이블명, 컬럼명, 데이터타입, 컬럼수정자, 컬럼의 데이터타입 오너, 데이터 길이, NULLABLE, 컬럼ID, 열의 기본값 길이 등 확인 가능

참고사이트 : https://domoyosi.tistory.com/6

[AWS] Travis CI ./gradlew: Permission denied

[AWS] Travis CI ./gradlew: Permission denied

Travis CI 에서 ./gradlew: Permission denied 오류 메시지가 발생하는 경우.

.travis.yml 파일에 아래 코드를 추가하고 다시 commit/push 해본다.

before_install:

    – chmod +x gradlew


참고사이트 : https://m.blog.naver.com/ggomjae/221778504421

[JAVA] PDFBox 암호(패스워드)가 걸려있는 파일인지 확인하는 방법

[JAVA] PDFBox 암호(패스워드)가 걸려있는 파일인지 확인하는 방법

PDFBox 라이브러리 사용 시 document.isEncrypted() == true 이면 암호가 걸려있는 파일로 판단했으나, 암호입력 없이 정상적으로 열리는 파일인데도 true 가 나오는 경우가 있었다.

구글링을 해보니 암호가 빈값이어도 isEncrypted() == true 가 나올 수 있다고 했다.

결론은 isEncrypted() 로 암호유무를 판단해낼 수 없다.

새로 찾은 해결책은 PDFBox 버전을 2.0.4 버전 이상으로 올리는 것이다. (pdfbox-app-2.0.4.jar)

1. AS-IS 코드 (PDFBox 1.8.9 버전)

public boolean isPDFReadable(String filePath) throws IOException {
    boolean result = false;
    PDDocument document = null;
   
    try {
        document = PDDocument.load(filePath);
        if (!document.isEncrypted()) {
            result = true;
        }
       
    } catch (Exception e) {
        result = false;

    } finally {
    }

    return result;
}

PDFBox 1점대 버전에서는 암호가 걸려있는 PDF도 PDDocument.load 가 가능하다.

그런데 패스워드가 걸려있지 않은 PDF 문서 임에도 불구하고 document.isEncrypted() == true 가 나오는 문제가 있었다.

이에 따라 아래와 같이 라이브러리를 버전업하고 코드를 수정했다.

2. TO-BE 코드 (PDFBox 2.0.4 버전 이상)

public boolean isPDFReadable(String filePath) throws IOException {
    boolean result = false;
    PDDocument document = null;
   
    try {
        document = PDDocument.load(new File(filePath));
        if (document != null) {
            result = true;
        }
       
    } catch (Exception e) {
        result = false;

    } finally {
    }

    return result;
}

PDFBox 2.0.4. 버전 이상에서는 패스워드가 걸려있는 PDF를 PDDocument.load 할 경우 load 가 되지 않고 Exception 이 발생한다. 이를 통해 암호가 걸려있는지 여부를 알아낼 수 있다.

참고사이트 : https://stackoverflow.com/questions/15896691/how-to-check-if-a-pdf-is-password-protected-or-not 

[AWS] FileZilla 에서 AWS EC2 접속하는 방법

[AWS] FileZilla 에서 AWS EC2 접속하는 방법 

root 계정과 비밀번호로 접속하는 방법은 아니고, ppk 파일(개인키)로 접속하는 방법이다.

1. 상단메뉴의 [파일(F)] –  [사이트 관리자(S)] 클릭

2. [사이트 관리자] 창이 뜨면 하단의 [새 사이트(N)] 버튼을 클릭하고 정보 입력

프로토콜 : [SFTP – SSH File Transfer Protocol] 선택

호스트 : AWS ES2 탄력적 IP 입력 ex) 3.34.119.***

포트 : 22 입력

로그온 유형 : [키 파일] 선택

사용자 : [ec2-user] 입력

키 파일 : 확장자 ppk 파일 선택

[JAVA] 자바 순열 조합

[JAVA] 자바 순열 조합

직접 짠 순열 조합.

public class TestGetCombination {

    public static void main(String[] args) {
        TestGetCombination testGetCombination = new TestGetCombination();
        testGetCombination.start();
    }
   
   
    public void start() {
        // 인덱스 4개 중 3개 선택하기 예제
        // ex) 0, 1, 2, 3, 4
        int totalCount = 4;
        int countToGet = 3;
       
        ArrayList<ArrayList<Integer>> resultList = null;
       
        // 조합 구하기
        System.out.println(totalCount +  “개 중 ” + countToGet + “개 조합 경우의 수 구하기”);
        resultList = getCombination(totalCount, countToGet);
        if (resultList != null && resultList.size() > 0) {
            int count = resultList.size();
            System.out.println(“count : ” + count);
            for (int i=0; i<count; i++) {
                System.out.println(i + ” : ” + resultList.get(i));
            }
        }
       
        System.out.println();
       
        // 순열 구하기
        System.out.println(totalCount +  “개 중 ” + countToGet + “개 순열 경우의 수 구하기”);
        resultList = getPermutation(totalCount, countToGet);
        if (resultList != null && resultList.size() > 0) {
            int count = resultList.size();
            System.out.println(“count : ” + count);
            for (int i=0; i<count; i++) {
                System.out.println(i + ” : ” + resultList.get(i));
            }
        }
    }
   
   
    /**
     * 조합 구하기
     *
     * ex 1) totalCount == 4
     *          countToGet == 1
     *       resultList == [[0],
     *                       [1],
     *                       [2],
     *                       [3]]
     *
     * ex 2) totalCount == 4
     *       countToGet == 2
     *       resultList == [[0, 1],
     *                       [0, 2],
     *                       [0, 3],
     *                       [1, 2],
     *                       [1, 3],
     *                       [2, 3]]
     *      
     * ex 3) totalCount == 4
     *       countToGet == 3
     *       resultList == [[0, 1, 2],
     *                       [0, 1, 3],
     *                       [0, 2, 3],
     *                       [1, 2, 3]]
     *            
     *      
     *      
     * @param totalCount
     * @param countToGet
     * @return
     */
    public ArrayList<ArrayList<Integer>> getCombination(int totalCount, int countToGet) {
        ArrayList<ArrayList<Integer>> resultList = new ArrayList<ArrayList<Integer>>();
        getCombinationCore(resultList, null, 0, totalCount, countToGet);
        return resultList;
    }
   
   
    /**
     * 조합 구하기 재귀메서드
     * 넘겨받은 resultList 객체에 경우의 수를 담는다. resultList 가 사실상의 리턴값이다.
     *
     * @param resultList
     * @param inputResult
     * @param axisIndex
     * @param totalCount
     * @param countToGet
     */
    private void getCombinationCore(ArrayList<ArrayList<Integer>> resultList, ArrayList<Integer> inputResult, int axisIndex, int totalCount, int countToGet) {
        for (int i=axisIndex; i<totalCount; i++) {
            ArrayList<Integer> oneResult = null;
            if (axisIndex == 0) {
                oneResult = new ArrayList<Integer>();
            } else {
                oneResult = (ArrayList<Integer>) inputResult.clone();
            }
           
            // 1개씩 인덱스를 담는다.
            oneResult.add(i);
           
            if (oneResult.size() == countToGet) {
                // countToGet 개 만큼 담았으면 resultList 객체에 추가한다.
                resultList.add(oneResult);
               
            } else if (oneResult.size() < countToGet) {
                // countToGet 개 만큼 담을 때까지 재귀한다.
                getCombinationCore(resultList, oneResult, i + 1, totalCount, countToGet);
               
            } else {
                break;
            }
        }
    }
   
   
    /**
     * 순열 구하기
     *
     * ex 1) totalCount == 4
     *          countToGet == 1
     *       resultList == [[0],
     *                       [1],
     *                       [2],
     *                       [3]]
     *      
     * ex 2) totalCount == 4
     *       countToGet == 2
     *       resultList == [[0, 1], [1, 0],
     *                       [0, 2], [2, 0],
     *                       [0, 3], [3, 0],
     *                       [1, 2], [2, 1],
     *                       [1, 3], [3, 1],
     *                       [2, 3], [3, 2]]
     *
     * ex 3) totalCount == 4
     *       countToGet == 3
     *       resultList == [[0, 1, 2], [0, 2, 1], [1, 0, 2], [1, 2, 0], [2, 0, 1], [2, 1, 0],
     *                       [0, 1, 3], [0, 3, 1], [1, 0, 3], [1, 3, 0], [3, 0, 1], [3, 1, 0],
     *                       [0, 2, 3], [0, 3, 2], [2, 0, 3], [2, 3, 0], [3, 0, 2], [3, 2, 0],
     *                       [1, 2, 3], [1, 3, 2], [2, 1, 3], [2, 3, 1], [3, 1, 2], [3, 2, 1]]
     *
     * @param totalCount
     * @param countToGet
     * @return
     */
    public ArrayList<ArrayList<Integer>> getPermutation(int totalCount, int countToGet) {
        // 1. 조합 구하기
        ArrayList<ArrayList<Integer>> combinationList = getCombination(totalCount, countToGet);
        if (combinationList == null || combinationList.size() == 0) {
            return null;
        }

        // 2. 모든 조합에 대해 순열 구하기. 다시 말해, 각 조합에 대해 순서 바꾸기.
        ArrayList<ArrayList<Integer>> resultList = new ArrayList<ArrayList<Integer>>();
        int count = combinationList.size();
        for (int i=0; i<count; i++) {
            // 특정 조합에 대해 순열 구하기
            getPermutationCore(resultList, combinationList.get(i));
        }
       
        return resultList;
       
    }
   
   
    /**
     * 특정 조합에 대해 순열 구하기. 다시 말해, 각 조합에 대해 순서 바꾸기.
     * 넘겨받은 resultList 객체에 경우의 수를 담는다. resultList 가 사실상의 리턴값이다.
     *
     * ex 1) inputList ==  [0]
     *       resultList 객체에 [[0]] 추가
     *      
     * ex 2) inputList ==  [0, 1]
     *       resultList 객체에 [[0, 1], [1, 0]] 추가
     *
     * ex 3) inputList == [0, 1, 2]
     *       resultList 객체에 [[0, 1, 2], [0, 2, 1], [1, 0, 2], [1, 2, 0], [2, 0, 1], [2, 1, 0]] 추가
     *      
     * ex 4) inputList == [1, 2, 3]
     *          resultList 객체에 [[1, 2, 3], [1, 3, 2], [2, 1, 3], [2, 3, 1], [3, 1, 2], [3, 2, 1]] 추가
     *
     * @param resultList
     * @param inputList
     */
    private void getPermutationCore(ArrayList<ArrayList<Integer>> resultList, ArrayList<Integer> inputList) {
       
        // 인덱스리스트 구하기
        ArrayList<ArrayList<Integer>> suffledIndexList = getSuffledIndexList(inputList.size());
        if (suffledIndexList == null || suffledIndexList.size() == 0) {
            return;
        }
       
        int count = suffledIndexList.size();
        for (int i=0; i<count; i++) {
            ArrayList<Integer> oneResult = new ArrayList<Integer>();
           
            ArrayList<Integer> indexList = suffledIndexList.get(i);
            for (int j=0; j<indexList.size(); j++) {
                oneResult.add(inputList.get(indexList.get(j)));
            }
           
            resultList.add(oneResult);
        }
    }
   
   
    /**
     * 인덱스리스트 구하기
     *
     * ex 1) totalCount == 1 일 경우, [[0]] 리턴
     * ex 2) totalCount == 2 일 경우, [[0, 1], [1, 0]] 리턴
     * ex 3) totalCount == 3 일 경우, [[0, 1, 2], [0, 2, 1], [1, 0, 2], [1, 2, 0], [2, 0, 1], [2, 1, 0]] 리턴
     *
     * @param count
     * @return
     */
    private ArrayList<ArrayList<Integer>> getSuffledIndexList(int count) {
        ArrayList<ArrayList<Integer>> suffledIndexList = new ArrayList<ArrayList<Integer>>();
        shuffleRecursively(suffledIndexList, null, count, null);
        return suffledIndexList;
    }
   
   
    /**
     * 인덱스리스트 구하기 재귀메서드
     * 넘겨받은 resultList 객체에 경우의 수를 담는다. resultList 가 사실상의 리턴값이다.
     *
     * @param resultList
     * @param inputResult
     * @param totalCount
     * @param exceptList
     */
    private void shuffleRecursively(ArrayList<ArrayList<Integer>> resultList, ArrayList<Integer> inputResult, int totalCount, ArrayList<Integer> exceptList) {
        if (exceptList == null) {
            exceptList = new ArrayList<Integer>();
        }
       
        for (int i=0; i<totalCount; i++) {
            if (exceptList.contains(i)) {
                continue;
            }
           
            ArrayList<Integer> oneResult = null;
            if (exceptList.size() == 0) {
                oneResult = new ArrayList<Integer>();
            } else {
                oneResult = (ArrayList<Integer>) inputResult.clone();
            }
           
            if (oneResult.size() < totalCount) {
                oneResult.add(i);
               
                if (oneResult.size() == totalCount) {
                    resultList.add(oneResult);
                   
                } else if (oneResult.size() < totalCount) {
                    ArrayList<Integer> newExceptList = (ArrayList<Integer>) exceptList.clone();
                    newExceptList.add(i);
                   
                    shuffleRecursively(resultList, oneResult, totalCount, newExceptList);
                }
            }
        }
    }
}

[JAVA] HashMap Key To Array (HashMap KeySet To Array)

[JAVA] HashMap Key To Array (HashMap KeySet To Array)

해시맵(HashMap)의 키(Key, KeySet)을 배열(Array)로 바꾸는 코드.

KeySet To Array

String[] array = hashMap.keySet().toArray(new String[hashMap.size()]);

예제

    public static void main(String[] args) {
        HashMap<String, Integer> hashMap = new HashMap<String, Integer>();
        hashMap.put(“a”, 1);
        hashMap.put(“b”, 2);
        hashMap.put(“c”, 3);
        hashMap.put(“d”, 4);
       
        hashMap.put(“키1”, 5);
        hashMap.put(“키2”, 6);
        hashMap.put(“키3”, 7);
       
        hashMap.put(“키3”, 10);
        hashMap.put(“키3”, 20);
       
        String[] array = hashMap.keySet().toArray(new String[hashMap.size()]);
       
        int len = array.length;

        System.out.println(“len : ” + len);

        System.out.println();
        for (int i=0; i<len; i++) {
            System.out.println(i + ” : “+ array[i]);
        }
    }

결과

len : 7

0 : a
1 : b
2 : c
3 : d
4 : 키1
5 : 키3
6 : 키2


[Excel] Excel 2007 xlsx에 읽을 수 없는 내용이 있습니다. 이 통합 문서의 내용을 복구하시겠습니까?

[Excel] Excel 2007 xlsx에 읽을 수 없는 내용이 있습니다. 이 통합 문서의 내용을 복구하시겠습니까?

Excel 2007 (Microsoft Office 2007)을 사용하면서, 일반적인 PC에서 잘 열리는 엑셀 문서인데, 특정 PC에서 열면 <‘~~~~~.xlsx’에 읽을 수 없는 내용이 있습니다. 이 통합 문서의 내용을 복구하시겠습니까? 이 통합 문서의 원본을 신뢰할 수 있는 경우 [예]를 클릭하십시오.> 메시지가 발생하고 [예] 버튼을 클릭해야만 문서가 열리는 경우.

(주의 : 문서가 열리지 않는 경우가 아님. [예] 버튼을 클릭하면 문서가 잘 열리는 상황임.)

해결방법

1. 엑셀 실행  – 좌측 상단 마이크로소프트 오피스 아이콘 클릭 – 하단의 [Excel 옵션] 버튼 클릭 – [Excel 옵션] 창이 뜨면 좌측 메뉴의 [리소스] 클릭 – 우측 [Microsoft Office 진단 실행] 항목의 [진단(D)] 버튼 클릭

=> 별 효과 없었음.

2. 2007 Microsoft Office 제품군 SP3(서비스 팩 3) 설치

=> 설치 후 엑셀이 메시지 없이 깨끗하게 열리게 되었음.

Office 2007 SP3 x86 Hotfix171011 Final 2017.10.11 최종버전 다운로드 :  https://m.blog.naver.com/ilovegaz/221338214664

원 출처는 https://support.microsoft.com/ko-kr/help/20171010/security-update-deployment-information-october-10-2017 이라고 함.

참고로 Office 2007은 32 비트까지만 존재함(64비트는 2010부터 지원함). 따라서 현재 Office 2007을 사용하고 있다면 컴퓨터 비트수와 상관없이 설치하면 됨.

참고사이트 1 : https://saysensibility.tistory.com/m/1001?category=740679

참고사이트 2 : https://m.blog.naver.com/ilovegaz/221338227086

[MySQL] MySQL root 패스워드 재설정 방법

[MySQL] MySQL root 패스워드 재설정 방법

MySQL 에서 root 계정의 패스워드/비밀번호를 잊어버렸을 때/분실했을 때/까먹었을 때/기억나지 않을 때 패스워드/비밀번호를 재설정/다시 설정하는 방법.

1. 서비스 중지

우분투 : service mysql stop

CentOS6 : service mysqld stop

2. 인증 생략 옵션 및 안전 모드로 데몬 실행

/usr/bin/mysqld_safe –skip-grant-tables &

3. 콘솔 접속

/usr/bin/mysql -u root mysql

4. 패스워드 변경

Mysql5.7 버전 미만

UPDATE mysql.user SET password=PASSWORD(‘패스워드’) WHERE user=’root’;

FLUSH PRIVILEGES;

quit

Mysql5.7 버전 이상

UPDATE mysql.user SET authentication_string=PASSWORD(‘패스워드’) WHERE user=’root’;

FLUSH PRIVILEGES;

quit

5. 서비스 재시작

우분투 : service mysql restart

CentOS6 : service mysqld restart

6. root 계정으로 로그인

변경한 패스워드를 입력하여 로그인 되면 성공.

mysql -u root -p

[AWS] 소유중인 도메인을 AWS EC2 서비스로 연결하기 (AWS 도메인 연결 방법)

[AWS] 갖고 있는 도메인을 AWS EC2 서비스로 연결하기 (AWS 도메인 연결 방법)

구입한 도메인이 있을 경우, 해당 도메인 주소로 접속했을 때 AWS EC2 서비스로 연결하는 방법이다.

조건 1) 소유하고 있는 도메인이 있음

조건 2) AWS EC2 에 띄워져있는 서비스가 있음

1. Route 53 (http://console.aws.amazon.com/route53) 에 접속한다.

좌측 메뉴의 [호스팅 영역] 클릭 – 화면의 [호스팅 영역 생성] 버튼을 클릭한다.

 

2. [도메인 이름] 항목에 갖고 있는 도메인 이름을 적는다.

참고로 필자는 ddoc.kr 이라는 도메인을 갖고 있다.

[유형]은 [퍼블릭 호스팅 영역]을 선택한다.

이어서 [호스팅 영역 생성] 버튼을 클릭한다.

 

3. 생성된 호스팅 영역(도메인 이름)을 클릭한다.

 

4. 레코드 목록을 볼 수 있다.

[값/트래픽 라우팅 대상] 이라고 해서 아래처럼 4개의 주소가 보일 것이다.

이 값들은 도메인 호스팅 사이트에 입력해야 하는 네임서버 값들이다.

 

예를 들어 가비아에 도메인을 등록했다면, [네임서버] 항목의 [설정] 버튼을 클릭하고 해당 네임서버 4개를 기입하면 된다.

아래 스크린샷을 참고할 것.

5. 다시 레코드 목록 화면으로 돌아와서, [레코드 생성] 버튼을 클릭한다.

6. [라우팅 정책] 중에서 [단순 라우팅] 을 선택하고 [다음] 버튼을 클릭한다.

 

7. [단순 레코드 정의] 버튼을 클릭한다.

 

8. [레코드 이름] 항목은 비워둔다. 서브 도메인을 사용할 경우만 기입하면 된다.

[값/트래픽 라우팅 대상] 콤보박스는 [레코드유형에 따라 IP 주소 또는 다른 값]을 선택하고, IP주소를 입력한다. (포트번호는 넣을 수 없음)

[레코드 유형은] [A – IPv4 주소 및 일부 AWS 리소스로 트래픽 라우팅]을 선택한다.

[TTL(초)]는 기본값 그대로 둔다.

이어서 [단순 레코드 정의] 버튼을 클릭한다.

 

9. [레코드 생성] 버튼을 클릭한다.




 참고사이트 : http://makebct.net/aws-%EC%99%B8%EB%B6%80-%EB%8F%84%EB%A9%94%EC%9D%B8-%EC%97%B0%EA%B2%B0-%EB%B0%A9%EB%B2%95-1/?cat=989/

[AWS] 아마존 AWS EC2 root 계정 접속하는 방법

[AWS] 아마존 AWS EC2 root 계정 접속하는 방법

EC2 (Elastic Compute Cloud) 의 기본 계정명은 ec2-user 이다.

초기에는 root 계정의 비밀번호가 설정되어있지 않으므로, ec2-user 계정으로 접속해서 root 계정의 패스워드를 세팅하면 된다.

1. root 계정의 새 패스워드 설정

sudo passwd root

2. root 계정으로 접속

su

참고사이트 : https://goddaehee.tistory.com/193

[SpringBoot] 스프링부트 포트 8080 에서 80 으로 변경하는 방법

[SpringBoot] 스프링부트 포트 8080 에서 80 으로 변경하는 방법

스프링부트는 특별한 설정을 하지 않았을 경우 기본 포트가 8080 이다.

application.properties 파일에 아래 내용을 추가한다.

server.port=80

또는 스프링부트 프로젝트를 기동할 때 -Dserver.port 옵션을 부여하면 된다고 한다.

java -Dserver.port=80 -jar spring-boot-application.jar

참고로 80 포트는 root 계정으로 기동해야 정상적으로 기동된다.

참고사이트 : https://stackoverrun.com/ko/q/10724065

[Android] 안드로이드에서 javascript(js) 함수 호출

[Android] 안드로이드에서 javascript(js) 함수 호출

안드로이드에서 자바스크립트 함수 호출 방법.

WebView webView = (WebView) findViewById(R.id.webView);
if (webView != null) {
    webView.loadUrl(“javascript:myFunctionName()”);
}


[Javascript] iframe innerHTML

[Javascript] iframe innerHTML

var x = document.getElementsByTagName(“iframe”)[0];

​var y = null;

if (x.contentWindow != null) {

    y = x.contentWindow;

} else if (x.contentDocument != null) {

    y = x.contentDocument;

}

if (y != null && y.document != null) {

    y = y.document;

}

y.getElementsByTagName(“html”)[0].innerHTML = “contents”;

위의 코드를 조금 수정해서, 아이프레임의 document 를 가져오는 코드 함수화

function getFrameDocument(_iframeId) {

    var frameObj = document.getElementById(_iframeId);

    var frameDoc = null;

    if (frameObj.contentWindow != null) {

        frameDoc = frameObj.contentWindow;

    } else if (frameObj.contentDocument != null) {

        frameDoc = frameObj.contentDocument;

    }

    

    if (frameDoc != null && frameDoc.document != null) {

        frameDoc = frameDoc.document;

    }

    

    return frameDoc;

}

이어서 html 태그 내의 내용을 innerHTML 함수로 가져오기

var frameDoc = getFrameDocument(“draftActionFrame”);

            

if (frameDoc != null && frameDoc.getElementsByTagName(“html”) != null) {

    alert(frameDoc.getElementsByTagName(“html”)[0].innerHTML);

}

[JAVA] reviseAmpersand (이미 “&”인 경우를 제외하고, “&” 기호를 “&”로 치환)

[JAVA] reviseAmpersand (이미 “&amp;”인 경우를 제외하고, “&” 기호를 “&amp;”로 치환)

ex 1) &abcde& => &amp;abcde&amp;

ex 2) &abcde&amp;&&& => &amp;abcde&amp;&amp;&amp;&amp;

    public static void main(String[] args) {
        String strTest1 = “&abcde&”;
        strTest1 = reviseAmpersand(strTest1);
        System.out.println(“strTest1 : ” + strTest1);

        // 결과 : strTest1 : &amp;abcde&amp;
        
        String strTest2 = “&abcde&amp;&&&”;
        strTest2 = reviseAmpersand(strTest2);
        System.out.println(“strTest2 : ” + strTest2);

        // 결과 strTest2 : &amp;abcde&amp;&amp;&amp;&amp;
    }
   
    /**
     * 이미 “&amp;”인 경우를 제외하고, “&” 기호를 “&amp;”로 치환한다.
     *
     * @param str
     * @return
     */
    public static String reviseAmpersand(String str) {
        if (str == null || str.length() == 0) {
            return “”;
        }
       
        String oneChar = “”;
        StringBuffer buff = new StringBuffer();
        int len = str.length();
        for (int i=0; i<len; i++) {
            oneChar = str.substring(i, i+1);
            if (oneChar.equals(“&”) && (i+5 > len || !str.substring(i, i+5).equals(“&amp;”))) {
                buff.append(“&amp;”);
                continue;
            }
           
            buff.append(oneChar);
        }
       
        return buff.toString();
    }

[JAVA] Image File to Base64 String (이미지 파일을 Base64 문자열로 변환)

[JAVA] Image File to Base64 String (이미지 파일을 Base64 문자열로 변환)

    String strBase64 = “”;

   

    File f = new File(filePath);
    if (f.exists() && f.isFile() && f.length() > 0) {
        byte[] bt = new byte[(int) f.length()];
        FileInputStream fis = null;

        try {
            fis = new FileInputStream(f);
            fis.read(bt);
            strBase64 = new String(Base64.encodeBase64(bt));

           

        } catch (Exception e) {
            throw e;
           
        } finally {
            try {
                if (fis != null) {
                    fis.close();
                }
            } catch (IOException e) {
            } catch (Exception e) {
        }
    }

[HTML] Base64 이미지를 HTML img 태그로 표시하기

[HTML] Base64 이미지를 HTML img 태그로 표시하기

html 파일 내용에 <img src=”data:image/png;base64,베이스64코드”> 형식으로 쓰면 그림이 된다.

예를 들어 아래와 같이 쓴다.

<img src=””>

 

참고사이트 : https://pilot376.tistory.com/3

[Linux/Ubuntu] 우분투 사용자 생성, 사용자 조회, 사용자 삭제

[Linux/Ubuntu] 우분투 사용자 생성, 사용자 조회, 사용자 삭제

우선 root로 로그인한다.

1. 사용자 만들기 (계정 생성)

sudo useradd -m [사용자계정]

ex) sudo useradd -m user01

useradd 명령의 -m 옵션을 사용하면 사용자의 홈 디렉토리도 함께 만든다.

2. 사용자 조회 (계정 조회)

모든 사용자 조회 : cat /etc/passwd

모든 사용자 조회 (아이디만) : cut -f1 -d: /etc/passwd

useradd 를 통해서 추가한 사용자만 조회 : grep /bin/bash /etc/passwd

useradd 를 통해서 추가한 사용자만 조회 (아이디만) : grep /bin/bash /etc/passwd | cut -f1 -d:

3. 특정사용자로 로그인 (해당 계정으로 로그인)

su – [사용자계정]

ex) su – user01

특정 사용자 아이디(여기서는 user01)로 로그인하는 명령어.

su 뒤에 대시(-)를 붙여줘야 환경변수를 물고 제대로 로그인된다.

참고로 su와 su – 는 다르다. su – 를 써야 환경변수와 워킹디렉토리가 변경된다. (참고 : https://storycompiler.tistory.com/44)

4. 특정사용자 삭제 (계정 삭제)

userdel [사용자계정]

ex) userdel user01

사용자를 잘못 생성했을 경우 위 명령어로 삭제 가능하다.

만약 “userdel: user [사용자계정] is currently logged in” 이라는 메시지가 나오고 삭제가 안되는 경우 아래 명령어를 사용하면 된다.

userdel -r -f [사용자계정]

ex) userdel -r -f user01

참고사이트 : https://withcoding.com/101

[MySQL/MariaDB] 문자열 결합, 문자열 합치기

[MySQL/MariaDB] 문자열 결합, 문자열 합치기

오라클에서는 파이프(“||”)를 이용해서 문자열을 결합하면 되지만, MySQL에서는 CONCAT이라는 함수를 사용해야 한다.

오라클 문자열 결합, 문자열 합치기

SELECT ‘a’ || ‘b’ || ‘c’ FROM DUAL;

=> abc

MySQL/MariaDB 문자열 결합, 문자열 합치기

SELECT CONCAT(‘a’, ‘b’, ‘c’);

=> abc

[MySQL/MariaDB] MySQL의 dual (오라클의 dual)

[MySQL/MariaDB] MySQL의 dual (오라클의 dual) 

MySQL은 dual이 없다.

만약 오라클에서 실행한 쿼리가 “SELECT 1+1 FROM DUAL;” 이라면,

MySQL에서는 “SELECT 1+1;” 라고 쓰면 충분하다.

확인해보니 MariaDB에서는 “SELECT 1+1 FROM DUAL;” 도 작동하고 “SELECT 1+1;” 도 작동한다.

[MySQL/MariaDB] 현재시간 가져오기 (오라클의 sysdate)

[MySQL] 현재시간 가져오기 (오라클의 sysdate)

오라클에서 현재일자 가져오기

SELECT sysdate FROM DUAL;

=> 20/09/09

오라클에서 현재시간(현재일시) 가져오기

SELECT to_char(sysdate, ‘yyyymmdd hh24miss’) FROM DUAL;

=>20200909 111105

MySQL/MariaDB에서 현재시간(현재일시) 가져오기

select NOW();

=> 2020-09-09 11:12:11

MySQL에서 현재시간(현재일시)을 가져올 때 SYSDATE()를 쓸 수도 있는데 NOW()를 쓰는 것을 권장한다.

NOW()는 하나의 쿼리 안에서 동일한 값을 리턴하지만 SYSDATE()는 쿼리 실행시점에 따라 값이 바뀐다.

예를 들어 매우 느린 쿼리가 있어서 레코드를 읽을 때 1시간이 걸린다면, SYSDATE() 값은 하나의 쿼리 안에서도 계속 바뀔 수 있다.

(참고사이트 : http://intomysql.blogspot.com/2010/12/sysdate-now.html)

[DBeaver] 오토커밋(Auto-commit) 해제 방법

[DBeaver] 오토커밋(Auto-commit) 해제 방법

DBeaver로 MsSQL을 붙어서 몇 가지 작업을 했는데 오토커밋(Auto-commit)이 기본적으로 켜져 있었다.

방법 : DBeaver 상단 메뉴의 [윈도우(w)] – [설정] – 환경 설정 창이 뜨면 좌측 트리의 [연결 유형] – Auto-commit by default 체크박스를 해제해주면 된다.

참고사이트 : https://pro-pennek.tistory.com/entry/DBeaver-Auto-commit-%EC%98%A4%ED%86%A0%EC%BB%A4%EB%B0%8B-%EC%84%A4%EC%A0%95-%EB%B0%8F-%ED%95%B4%EC%A0%9C-%EA%B8%80%EA%BC%B4-%EB%B3%80%EA%B2%BD

[DBeaver] 유용한 무료 DB 툴 dbeaver (MySQL, MsSQL, PostgreSQL 등)

[DBeaver] 유용한 무료 DB 툴 dbeaver (mysql, mssql 등)

엄청나게 다양한 SQL 벤더를 제공하고 있어서 MySQL, MsSQL, PostgreSQL 등 안되는 것 없이 다 된다.

JAVA jdk 1.8 버전에서 구동된다.

https://dbeaver.io/