기본 콘텐츠로 건너뛰기

Ubuntu 14.04 + Oracle Java 8 설치용 도커 파일

개발 장비에는 이런저런 테스트용 라이브러리들이 깔리고 지워지기를 반복한다.
특히나 개인 개발 장비의 경우는 더욱 그렇다고 볼 수 있다.
몇 년 전부터 회사와 집에서 개발용 장비로 Ubuntu 또는 CentOS를 사용하고 있는데, 사용하다 보면 라이브러리나 서로 다른 의존성을 가지는 솔루션을 무분별하게 설치하다가 결국 장비를 다시 설치해야 하는 불상사를 겪곤 한다.

불상사를 겪게 되는 이유 중 첫째는 나 자신의 부주의함과 무식함이 그 원인이겠지만, 어쨌거나 언젠가부터는 뭔가 설치를 하고자 하면 겁부터 나곤 한다.

그러다 알게 된 것이 Docker였다.
Docker에 대한 설명을 이 포스트에서 진행할 생각은 없다.
단지, 여러 가지로 내 조건에 많은 부분 부합하는 것이 Docker 라는 Container 였다는 점을 이곳에 적어놓고 싶었다.

Docker에 대한 자세한 사항은 https://docs.docker.com/ 을 참고하면 되겠지만, 우선은 https://training.docker.com/self-paced-training 에서 먼저 튜토리얼을 진행해 보면 좀 더 빠르리라 생각된다.
내가 또 하나 처음 공부할 때 봤던 것은 [가장 빨리 만나는 Docker: 클라우드 플랫폼 어디서나 빠르게 배포하고 실행할 수 있는 리눅스 기반 경량화 컨테이너] (이재홍, 도서출판길벗, 2015년) 이라는 책을 구글플레이에서 구매해서 봤었는데, 이 책은 빠르게 볼 수 있어서 괜찮았다. 또한, 맨 뒤쪽에 Docker 명령어를 정리해놔서 레퍼런스로도 볼 만 하다고 할까. 참고로 저 책 저자와 전혀 관계도 없으며 책을 사라고 하는 이야기는 아니다.

최근까지 아무 생각 없이 Docker Hub 에 있는 다양한 Official 또는 개인이 올린 이미지를 이용해서 사용하고 있었는데, 자바프로젝트 역시 이미지의 기본은 Official로 올라온 java8을 주로 이용했었다. 그런데 이 이미지는 OpenJDK 로 만들어진 이미지였다. OpenJDK가 좋은가 Oracle JDK 가 좋은가를 논하고 싶은 생각은 없다. 단지, 진행하고 있는 프로젝트들이 기본적으로 Oracle JDK 를 이용하고 있다 보니 Oracle JDK로 만들어진 이미지를 사용하고 싶었을 뿐이다.

Oracle JDK의 Docker 이미지도 스프링 가이드(https://spring.io/guides/gs/spring-boot-docker/)에 있는 frolvlad/alpine-oraclejdk8:slim를 이용하면 매우 단순하게 해결할 수 있는 부분이고 지금까지 주로 사용을 해 왔었으나, Docker Image에 추가로 다른 솔루션을 설치하거나 할 경우나 Docker에 Bash로 접근해서 뭔가를 하려고 할 때 명령어들이 없어서 불편하다는 생각을 하게 되었다.

그래서 Ubuntu + Oracle JDK 로 이미지를 만들어서 사용하도록 Dockerfile 을 만들어 놓으면 편하겠다는 생각이 들었다. 아래 내용은 그 기본적인 Dockerfile 구성을 필요할 때 카피해서 사용하기 위해서 정리해 놓은 것이다.

시작하기 전에 아래에 예제로 만들어 놓은 Dockerfile 은 실제 구동할 수 있는 Dockerfile은 아님을 미리 밝혀놓는다. 또한, 아래 Dockerfile을 이용할 경우 네트워크 상태에 따라서는 매우 긴 시간이 걸릴 수 있음도 미리 밝혀두고자 한다.

기본 이미지는 Official ubuntu trusty(14.04) 버전을 이용했으며, apt-get 으로 자바를 설치하도록 설정된 Dockerfile 이다.

FROM ubuntu:trusty
MAINTAINER roadkh@gmail.com

RUN apt-get update && \
    apt-get upgrade -y && \
    apt-get install -y software-properties-common && \
    add-apt-repository ppa:webupd8team/java -y && \
    apt-get update && \
    echo oracle-java8-installer shared/accepted-oracle-license-v1-1 select true | /usr/bin/debconf-set-selections && \
    apt-get install -y oracle-java8-installer && \
    apt-get clean


ENV JAVA_HOME /usr/lib/jvm/java-8-oracle

참 간단하다.
위 내용을 적당한 디렉터리에 Dockerfile로 저장하고, 아래 명령을 실행해서 이미지를 생성해보면 상당한 시간과 처리를 진행한 후에 완료가 될 것이다.

docker build --tag java8 .

설치가 완료되면, 아래 명령어로 이미지를 확인해보자. 이미지 이름은 java8로 만들어진다.

docker images


마지막 부분에 보면 이미지 사이즈가 있는데, 보통의 Docker 이미지들이 500메가를 넘지 않는 것에 비하면 꽤 큰 편이다. 물론 1기가도 되지 않는 용량이 뭐 큰 것인가 생각할 수도 있지만...

앞서도 이야기했지만, 위의 Dockerfile 은 실제 Container가 생성돼서 실행될 때 할 일(Entrypoint 등)을 아무것도 주지 않은 파일이다. 따라서 위의 이미지를 이용해서 `docker run --name java8-test -d java8` 명령어로 Container를 실행해봐야 아무것도 실행되지 않고 Container만 생성됨을 알 수 있다. `docker ps` 로는 아무것도 나오지 않을 것이고 `docker ps -a` 를 해봐야 생성된 container를 볼 수 있다. 다시 한 번 강조하면 위 Dockerfile은 기본적인 Dockerfile에 대한 레이아웃일 뿐이다.

만약, 확인을 하고 싶다면 아래와 같이 해보자.

docker run -i -t --name java8-test java8

위와 같이 하면, 바로 Container 안의 쉘이 나올 것이다. (물론 exit 를 하고 나면 container는 종료될 것이다. 이 경우는 `docker start java8-test` 를 한 후에 `docker exec -i -t java8-test /bin/bash` 를 하면 된다)
`java -version` 명령을 입력해보면 아래와 같은 결과를 얻을 것이다.


나중에 내가 이 포스트를 보고 또 잊어버려서 넋 놓고 있을 경우를 위해 https://spring.io/guides/gs/spring-boot-docker/ 에 있는 Docker 파일을 위의 내용을 이용해서 다시 만들어 놓은 예제를 마지막으로 이 포스트를 마치려고 한다. 참 간단한 내용을 길게 적은 포스트였다.

FROM ubuntu:trusty
MAINTAINER roadkh@gmail.com

RUN apt-get update && \
    apt-get upgrade -y && \
    apt-get install -y software-properties-common && \
    add-apt-repository ppa:webupd8team/java -y && \
    apt-get update && \
    echo oracle-java8-installer shared/accepted-oracle-license-v1-1 select true | /usr/bin/debconf-set-selections && \
    apt-get install -y oracle-java8-installer && \
    apt-get clean


ENV JAVA_HOME /usr/lib/jvm/java-8-oracle

VOLUME /tmp
ADD gs-spring-boot-docker-0.1.0.jar app.jar
RUN sh -c 'touch /app.jar'
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/app.jar"]


추가하는 내용

앞서도 말했듯이 이 이미지는 개발용으로는 괜찮으나 일반 서비스용으로는 적합하지 않다고 생각한다. 개발 중에는 컨테이너에 접근해서 다양한 명령어를 이용해본다거나 추가로 몇 가지 애플리케이션을 더 깔아보고 Dockerfile을 수정한다거나 하는 여러 가지 일을 하므로 위와 같은 이미지가 필요해보이지 싶어서 정리해 본 것이다. 컴팩트한 이미지들 중 어떤 것은 vi 명령어 설치가 되어있지 않은 경우도 많으니까 말이다. 그렇더라도 실제 서비스에서는 필요한 부분만 들어가 있는 컴팩트한 이미지를 이용하는 것이 더 좋지 않을까 생각한다.


댓글

이 블로그의 인기 게시물

경력 개발자의 자기소개서에 대해서...

갑자기 뜬금없이 이런 글을 쓰다니 무슨 생각이야? 라고 생각하시는 분들이 있을지도 모르겠네요. 뜬금없음에 대한 변명은 잠시 접어두고 일단 오늘 쓰려고 하는 글을 시작해볼까 합니다. 개발자로 대충 16년을 그럭저럭 보내왔습니다. 시대적 상황으로 5년 차쯤에 대리로 처음 팀장을 시작했으니, 일반 개발자로 산 시간보다는 어쨌건 프로젝트 또는 팀의 리더로 산 시간이 더 많았던 것 같습니다. 그 기간 동안 남들보다 좀 심하게 회사를 많이 옮겨 다니다 보니 꽤 많은 면접을 볼 수 있는 경험이 있었고, 또 옮긴 회사가 대부분 팀을 리빌딩하는 곳이었다 보니 꽤 많은 채용절차에 관여할 기회가 있어서 어린 나이부터 비교적 많은 이력서를 검토했고 면접관으로도 여러 사람을 만날 수 있었습니다. 처음 면접을 보러 다니던 시절의 제 이력서의 자기소개서는 항상 "19XX년 봄 XX업계에 종사하시던 아버님과 집안일에 헌신적인 어머니의 유복한 가정에 1남 1녀의 막내로..." 로 시작되었습니다 (이 문장에 향수를 느끼시는 분들 많으실 거예요. ^^). 경력이 5년이 넘은 어느 날 도대체 이 문장을 왜 써야 하느냐는 의문이 생겨서 조금 바꾸긴 했습니다만, 그 뒤로도 꽤 오랜 세월을 이런 자기소개서가 항상 제 이력서에 붙어있었죠. 요즘 누가 저런 식으로 자기소개서를 써? 라고 생각하시는 분들 많으실 거로 생각해요. (대신 요즘은 대학 시절의 봉사활동이나 해외연수 이력이... 뭐 어차피 그놈이 그놈입니다.) 저런 자기소개서를 써야 한다는 것이 어디서 어떻게 시작된 것인지는 몰라도 회사를 그만두기 전인 2년 전까지도 약간의 표현은 다를지 모르지만 비슷한 문장으로 시작하는 자기소개서를 이력서에 첨부해서 보내는 지원자들을 볼 수 있었습니다. 이제 제가 뜬금없는 이런 글을 쓰게 된 이유를 밝히고 계속 진행해야겠네요. 블로그에 올릴 글을 준비하는 일이 생각보다 힘들어요. 블로그에 올리려고 준비한 주제에 맞는 소스를 작업하고 거기에 글을 입히다 보면 가끔 ...

Springframework 5에서 바뀌는 것들에 대한 간단 정리 및 생각

Spring framework 5 에 대해 많은 분이 기대와 두려움을 가지고 계시지 않을까 생각합니다. 특히 기대를 하고 계신 분들은 Reactive Programming 지원을 기대하고 계시지 않은가 생각이 드는데요. 7월 초에 John Thompson 이란 분이 D-Zone에 아주 깔끔하고 멋지게 정리를 잘해서 글을 쓰셨더라구요. 해당 글은  https://dzone.com/articles/whats-new-in-spring-framework-5 에서 확인을 하실 수 있습니다. 혹시 Spring framework 5에서 달라지는 내용의 좀 더 자세한 내용이 필요하신 분들은 Spring framework github의 wiki 를 참고하시면 됩니다. 본 포스트는 언제나 그렇듯이 윗글에 대한 번역이 아닙니다. 그저 윗글을 다시 정리하면서 제 생각을 한번 정리해 놓은 포스트입니다. Spring framework 5는 현재 5.0.0.RC2(2017.07.23일 기준)까지 릴리즈된 상황입니다. Spring framework 5에서 크게 변화하는 내용을 John Thompson은 8가지로 깔끔하게 정리해주고 있습니다. 1. JDK 지원 버전의 업데이트 5버전은 원래 JDK 9 버전의 지원을 위해서 시작됐던 프로젝트로 알고 있는데 맞는지는 모르겠네요. JDK 9의 Release가 늦어져서 Spring framework 5가 먼저 Release 될 것으로 보이지만, JDK 9가 Release가 되면 언제건 적용할 수 있다고 합니다. 좀 아쉬운 부분은 JDK의 최소 버전은 JDK 8이라는 부분이 아닐까 싶네요. 이 때문에 Spring framework 5에 무관심한 분들도 많으실 거라고 생각합니다. 지금 진행하는 프로젝트는 JDK 8을 기반으로 합니다만, 최근까지 다니던 회사의 경우는 JDK 7까지가 업그레이드 한계였던 회사였습니다. 아마도 JDK 업그레이드를 쉽게 못 하시는 회사들이 많으니 "나랑은 관계없는 얘기군...

자바에서 파일 타입을 확인하는 방법들... 그리고 Apache Tika...

뭐 역시나 내가 직접 조사한 것은 아니고 Java Code Geeks 에서 본 내용에 대한 정리이다. 자세한 사항은 [원문보기]  를 클릭해서 확인하면 된다. 원문에서는 코드와 스크린샷을 포함한 예제를 확인할 수 있으니 내용을 보시고자 하시는 분들은 꼭 원문을 참고하시길... 내가 쓰는 포스트가 대부분 그렇지만, 이 글도 그저 내용을 정리해서 내가 나중에 확인하기 위해 쓰는 글일 뿐... 1. Files.probeContentType(Path)를 이용하는 방법 [JDK 7] 이 방법은 파일 확장자를 이용해서 판단하는 것으로 보인다. 즉, gif 파일을 png라고 확장자를 주면 png 로 인식한다는 것. 또 확장자가 없는 파일의 경우는 null 을 리턴한다. 2. MimetypesFileTypeMap.getContentType(String) 을 이용하는 방법 [JDK 6] 이 방법은 클래스 명에서 유추가 가능한 것 처럼 Mime type 을 이용한다. 앞서의 경우와 같이 gif 파일을 png 라고 하거나 pdf 파일을 txt 로 하거나 하면 확장자로 판단을 한다. 1번 경우와 다른 점을 이야기하자면 기본적으로 텍스트나 이미지 확장자가 아닌 경우에는 xml 파일을 포함해서 application/octet-stream을 반환한다. 또한 확장자가 없는 파일의 경우도 application/octet-stream을 반환한다. 3. URLConnection.getContentType() 이 방법 역시 확장자에 의한 판단. 대신 null 대신 content/unknown을 반환 4. URLConnection.guessContentTypeFromName(String) 역시 확장자. 3번과 다른 것은 null을 반환. 5. URLConnection.guessContentTypeFromStream(InputStream) 예제에서는 전부 null 을 반환. 저자의 설명에 따르면 여기서 매개변수로 주어진 InputStream은 ma...