๐ณ Docker study 4th [ENG]
Running RStudio on Docker: Leveraging GPU for Data Analysis
by Arielle
In the previous <Docker Study 3rd> post, I set up a Python environment in a container, or virtual environment, using a Kaggle Image. For this 4th week, I conducted a hands-on session to build R, a widely used data analysis environment, using a Kaggle Image. This time, I delved deeper into Dockerfile, but even though it was covered in week 3, applying it in a new context made my mind go completely blank again. (Maybe I was a goldfish in my past life...๐คฆโโ๏ธ)
Since I'm constantly encountering new concepts, I feel like I'm learning at a slower pace, but at least I have the determination to solve the challenges ahead, which is a relief! ๐ช If you canโt summarize what a โDockerfileโ is in a single sentence before diving into this post, I highly recommend quickly reviewing the ๐ Dockerfile? section from the previous post! ๐
1. Setting up a R analysis environment
1.1 Installing a Kaggle GPU image for R
If you search for โKaggle gpu R imageโ, youโll find a link to download an image that allows you to use R inside a container. Since the process is similar to building an image for the Kaggle Python environment, it was relatively straightforward. As of the installation date (October 14, 2024), the latest version of R is v59.
I installed it docker pull gcr.io/kaggle-gpu-images/rstats:v59
on the virtual machine.
1.2 Running an R container in the RStudio Server IDE environment
So far, I have been using Docker by downloading the necessary images and running them in the container, essentially in a virtual environment terminal. This time, I attempted to run the container in an IDE environment (Web) using RStudio Server.
The image below shows the result of accessing the IDE through the R container, successfully running within a web interface.
This can be achieved with just a single command. Itโs a bit long, but if you break it down step by step, itโs not too difficult to understand.
โ
Rstudio Web Browser connect code ย docker run -d -p 8787:8787 --gpus all -v "/home/Arielle/rproject:/home/rstudio/rproject" --name rstudio-container gcr.io/kaggle-gpu-images/rstats:v59 /bin/bash -c "rstudio-server start && tail -f /dev/null"
The command docker run -d
is used to start a new container based on the specified image. The -d option enables โDetached modeโ, which allows the container to run in the background. This means that the terminal remains free for other tasks and is not locked to the container process.
If the -d option is not used, the terminal will remain attached to the containerโs process, preventing you from executing any other commands while the container is running. Therefore, using -d is important when you need to run a container in the background and continue working on other tasks in the terminal!
The option -p 8787:8787
is used to set up port forwarding between the host system and the container.
The first 8787 (before :) refers to the host systemโs port. The second 8787 (after :) refers to the containerโs port, where the RStudio Server runs by default. This configuration allows the RStudio Server inside the container to be accessible from your local system. As a result, you can open a web browser and access the RStudio Server by navigating to:
๐ http://localhost:8787
This way, the host system forwards requests from its port 8787 to the same port inside the container, enabling smooth interaction between the containerized R environment and your local browser.
However, while explaining this in a simple sentence, I realized that I didnโt have a clear understanding of ports. As I followed the lecture, I kept questioning whether I truly grasped the concept. (The only port numbers I was somewhat familiar with were 1004 and 22?? ๐คจ)
So, to never get confused about ports again, I decided to take a step back and properly organize my understanding of them! Letโs dive into it!
๐ชย Port?
First of all, I didnโt fully understand what ports were, but I had unknowingly been using them in various places. One representative example was when I set up a Mac virtual environment using RDP (check out the post here).
To make it clearer, I created a simple diagram that visually organizes my understanding of ports. See below!๐
From the client-server perspective, as shown on the left side of the diagram, each service that the server provides is assigned a specific port number, which allows clients to connect to it.
For example:
- RDP (Remote Desktop Protocol) uses port 3389.
- SSH (Secure Shell) uses port 22.
- VNC (Virtual Network Computing) uses port 5900.
What about the Docker container perspective? Now, letโs shift our understanding to Docker containers. In this perspective:
The traditional client (a user or device) is replaced by the local host. The traditional server is replaced by the container. A Docker container can be thought of as an isolated, self-contained environment designed to run a specific application or serviceโsimilar to an independent computer.
One important takeaway from the diagram is that containers operate within a โUser Spaceโ on top of the host operating system. Inside this space, each containerized service (application image) runs separately, maintaining its own environment.
Even though containers share the same host machine, each container runs its application in isolation, with dedicated resources and ports to communicate with the outside world. This is why port forwarding (e.g., -p 8787:8787) is crucial in Docker, as it allows external access to the services running inside containers.
You might wonder why I suddenly started talking about container structures while discussing ports. Earlier, I mentioned that each service is assigned a unique port number. This means that the RStudio service I just downloaded must also have a unique port number. That number is 8787, as seen in the command -p 8787:8787
!
However, youโll notice that 8787 appears twice, separated by a colon (:). The first 8787 represents the port assigned to the localhost, while the second 8787 is the port assigned to the container. This is different from traditional client-server models, where only the server side is assigned a port number. Here, both sides are assigned a port numberโthis is the key difference!
Another interesting point is that the port number assigned to the localhost does not have to be 8787; it can be set to any number the user chooses.
This raises another question: When would you manually specify the local host port number? The answer is when you run multiple RStudio Server containers on the same machine. In such cases, assigning different local ports to each container ensures flexibility (pliability).
However, the reason for setting the same port number for both the localhost and the container is not just for convenience but also for isolation. When multiple containers are running, having the same port number makes it easier to locate each service. This concept naturally leads to the discussion of isolation.
By default, a container operates in complete isolation from the external environment. However, when we specify a port pair, we establish communication between the localhost and the container. This allows the isolated container to become accessible from the outside, which is where Dockerโs true power comes into play!
Now, letโs return to the command for running a container on the web! After specifying the port, youโll notice the option --gpus all
. As the name suggests, this allows the container to access the local GPU resources. If your local host is equipped with a GPU, the container can utilize it for computation. However, if GPU access is not needed or your system doesnโt have a GPU, you can omit this option without any issue.
The part -v โ/home/Arielle/rproject:/home/rstudio/rprojectโ represents the concept of Volume Mounting in Docker. In simple terms, volume mounting is a feature that links a file or directory from the host system to a specific location inside the container.
Here, we are pairing:
- Host directory:
/home/Arielle/rproject
- Container directory:
/home/rstudio/rproject
This setup allows for seamless file sharing between the host and the container. You can:
- Use files created on your local machine inside the container
- Access files generated inside the container directly from the host
For example, if you run the command mkdir testfolder on your local machine, you can immediately see the newly created โtestfolderโ inside the containerโas shown in the image below (Container with WEB)
The option --name rstudio-container
is used to assign a name to the created container, and here, it is specified as rstudio-container
.
The last part,`gcr.io/kaggle-gpu-images/rstats:v59, represents the Docker image (RStudio), indicating its repository path and version.
2. How to work Python package in R?
If you have successfully connected to your local host through the container, you will see the R program interface as shown below. Since I had never worked with R before, I wasnโt familiar with its syntax. The command reticulate::py_config()
is a function in R that sets up and displays information about the Python environment.
Interestingly, R can integrate with Python using the reticulate package
. When I ran the command, I noticed the following message appeared in the console:
Would you like to create a default Python environment for the reticulate package? (Yes/no/cancel)
This message indicated that there was no existing Python environment linked to R, and the reticulate package was prompting me to install a new one. So, I simply selected โYESโ to install Python and proceed with the setup.
For reference, while I was able to visually confirm RStudio using the RStudio IDE, this time, I checked it through the containerโs terminal mode. (After all, itโs good to learn both GUI and terminal approaches!) First, I ran docker ps
to check the name of the running container (rstudio-container).
Then, I used the following command to directly access RStudio within the container. docker exec -it rstudio-container R
As shown in the image below, R ran successfully within the containerโs terminal, confirming that it works properly even without a graphical interface.
โ
ย docker exec -it rstudio-container R
์ฌ๊ธฐ์ -it
๋ -i
(interactive), -t
(tty)๊ฐ ํฉ์ณ์ง ๋ช
๋ น์ด๋ก ์ปจํ
์ด๋ ๋ด๋ถ์์ ๋ช
๋ น์ด๋ฅผ ์คํํ ์ ์๊ฒ ํด์ฃผ๋ฉด์ ๋์์ ํฐ๋ฏธ๋์์ ์
๋ ฅํ ๋ช
๋ น์ด์ ์ถ๋ ฅ์ ๋ณด๊ธฐ ์ํด ํ๋ฉด์ ํฐ๋ฏธ๋์ฒ๋ผ ๋ง๋ค์ด์ฃผ๋ ๊ธฐ๋ฅ์ ํ๋ค! (์ฐธ๊ณ ๋ก ํด๋น ํฐ๋ฏธ๋์์ exitํ๊ณ ์ถ์ผ๋ฉด q()
๋ก ๋์ค๋ฉด ๋๋ค!)
์๊น RStudio(WEB)์์ Python์ ์ค์นํด ์ฃผ์๋ค. ์ค์น ์ฌ๋ถ๋ฅผ ๋ฌผ์ด๋ณด๊ธธ๋ ์ผ๋๋ฒ๋ ์ค์น๋ฅผ ํ๊ธฐ๋ ํ๋๋ฐ ์ด๋์ ์ค์น ๋์๋์ง ํ๋ฒ ์ฏค์ ํ์ธํ ํ์๊ฐ ์์ ๊ฒ ๊ฐ๋ค. reticulate::py_config()
๋ก ํฐ๋ฏธ๋์์ ํ์ธํด๋ณด๋ /root/.local/share/r-minicconda/env/r-reticulate/bin/python
๊ฒฝ๋ก์ ์์นํด ์์์ ํ์ธํ๋ค. ๊ทธ๋ฐ๋ฐ ์ง๊ธ ๋ณด๋ฉด r-miniconda
๋ผ๋ ํด๋์ ํ์ ๊ฒฝ๋ก์ python์ด ์ค์น๋์ด ์๋ ๊ฑธ๋ก ๋ณด์ด๋๋ฐ, ๋๋ ํด๋น ํด๋๋ฅผ ์์ฑํ ์ ์ด ๋ถ๋ช
์๋ค.
์ด๋ฅผ ์ ๋๋ก ์ดํดํ๊ธฐ ์ํด์๋ Docker ์ปจํ ์ด๋ ๋ด๋ถ์์ RStudio์ Python ํ๊ฒฝ์ ์ด๋ป๊ฒ ์ฐ๊ฒฐํ๊ณ ์ํธ์์ฉํ๋์ง ์ดํดํ ํ์๊ฐ ์๋ค.
๋ฐ๋ก ์ ๊ทธ๋ฆผ์ผ๋ก ์ดํดํด๋ณด๋๋ก ํ์! ํ์ด์ฌ ํจํค์ง๋ฅผ ์ด์ฉํด Keras, Tensorflow๊ฐ R์์ ์ด๋ป๊ฒ ์๋ํ๋์ง ํ๋ก์ฐ๋ฅผ ๊ทธ๋ฆฐ ๊ทธ๋ฆผ์ผ๋ก ์ด 5๋จ๊ณ๋ก ์งํ ๋๋ค.
1๏ธโฃ R์์ ๋ฅ๋ฌ๋์ ์ํํ๋ ค๋ฉด R Keras ํจํค์ง๋ฅผ ์ฌ์ฉํด์ผ ํ๋ค. ํ์ง๋ง, R ์์ฒด์๋ Keras๋ TensorFlow๊ฐ ๋ด์ฅ๋์ด ์์ง ์๊ธฐ ๋๋ฌธ์ Python ํจํค์ง๋ฅผ ๋ถ๋ฌ์ ์ฌ์ฉํด์ผ ํ๋ค.
2๏ธโฃย ์ ์ด์ Python ํจํค์ง๊ฐ ํ์ํจ์ ์์์ผ๋ R๋ก ๊ฐ์ ธ์์ผ ํ๋๋ฐ ํน์ดํ๊ฒ R์์๋ reticulate
๋ผ๋ ํจํค์ง๊ฐ ํ์ํ๋ค. ๊ทธ๋์ /root/.local/share/r-minicconda/env/r-reticulate/bin/python
์ด ๊ฒฝ๋ก์ฒ๋ผ r-reticulate
๊ฐ python์ ์์ ๊ฒฝ๋ก์ ์์๋ ๊ฒ์ด๋ค!
3๏ธโฃย ํ์ฌ Python ํจํค์ง์ ๊ฒฝ์ฐ reticulate๊ฐ ์ฌ์ฉํ๋ Python ํ๊ฒฝ์ผ๋ก ๋ค์ด์ค๊ฒ ๋๋ค. ์ด๊ฒ ๋ฌด์จ ๋ง์ด๋ผ๋ฉด ๋ง์ฝ reticulate
๊ฐ miniconda๋ผ๋ ๊ฐ์ ํ๊ฒฝ์ ์ฌ์ฉํ๊ณ ์๋ค๋ฉด, Python ํจํค์ง๋ miniconda ํ๊ฒฝ ์์ ์ค์น๋์ด miniconda
๊ฐ Python ํจํค์ง๋ฅผ ์ ์ฅํ๊ณ ๊ด๋ฆฌํ๋ ๊ณต๊ฐ์ด ๋๋ ๊ฒ์ด๋ค. ๋ฐ๋ฉด์ ๋ง์ฝ reticulate
๊ฐ ๋ค๋ฅธ Python ํ๊ฒฝ์ ์ฌ์ฉ ์ค์ด๋ผ๋ฉด,๊ทธ ํ๊ฒฝ์ Python ํจํค์ง๊ฐ ์ค์น๋๋ค๋ ๋ป์ด๋ค! ์๋ฅผ ๋ค์ด ์์คํ
์ ๋ฏธ๋ฆฌ ์ค์น๋ Python์ด ์๋ค๋ฉด ๊ทธ ํ๊ฒฝ์ผ๋ก Python ํจํค์ง๊ฐ ์ค์น๋ ์๋ ์๋ค.
์๋ฌดํผ ํฌ์ธํธ๋ reticulate
๊ฐ ์ฌ์ฉํ๋ Python ํ๊ฒฝ์ ๋ฐ๋ผ Python ํจํค์ง๊ฐ ์ค์น๋๋ ์์น๊ฐ ๋ฌ๋ผ์ง๋ค๋ ๊ฒ์ด๋ค!
4๏ธโฃย ์์ ๋จ๊ณ๊ฐ ์ ์์ ์ผ๋ก ์งํ๋๋ฉด Python์ ์ํด keras, tensorflow๋ฅผ ์ฌ์ฉํ ์ ์๊ฒ ๋๋ค. keras๋ RStudio์์ ์๊ณ ๋ฆฌ์ฆ์ ์์ฑํ ์ ์๊ฒ ๋๋ค.(UI)
5๏ธโฃย ์ดํ Tensorflow๋ Keras์ ๋ฐฑ์๋๋ก ์๋ํ๋ฉฐ ์ค์ง์ ์ธ ์ฐ์ฐ์ ํ๊ฒ ๋๋ค.
3. Custom Dockerfile
์์ Docker study 3rd์์ Dockerfile์ ๋ํด์ ์์๋ดค์๋ค. ๊ทธ๋๋ ๋ถ์ด๋นต์ ์๋ก ๋ค๋ฉด์ Dockerfile์ ๋ถ์ด๋นต ํ์ ๋ง๋๋ ๋ ์ํผ ์ ๋๋ก ์ค๋ช ํ์๋๋ฐ, ์ด๋ฒ์๋ ์กฐ๊ธ ๋ ์ํํธ์จ์ด ๊ด์ ์์ ์๊ฐํด๋ณด๊ณ ์ ํ๋ค.
๐ย Dockerfile? (โ๐๐ปDocker study 3rdโ)
#๋ถ์ด๋นต ํ์ ๋ง๋๋ ๋ ์ํผ
Dockerfile์ ์ปจํ ์ด๋์์ ์คํํ ํ๊ฒฝ์ ๋ง๋๋ ๋ ์ํผ๋ก ์๊ฐํด ๋ดค๋ค(๋ถ์ด๋นต๐๐ ๋ค์ ๋ฑ์ฅ). ์ฐ๋ฆฌ๊ฐ ์ด๋ค ํ๋ก๊ทธ๋จ์ ์คํํ๋ ค๋ฉด ๊ทธ ํ๋ก๊ทธ๋จ์ด ๋์๊ฐ๊ฒ ํ ํ๊ฒฝ์ด ํ์ํ๋ค. ์๋ฅผ ๋ค์ด ํ๋ก๊ทธ๋จ์ด Python์ผ๋ก ์ง์ฌ ์์ผ๋ฉด Python์ ์ค์นํด์ผ ํ๋ ๊ฒ์ฒ๋ผ, Dockerfile์ ๊ทธ๋ฐ ๊ฒ๋ค์ ์๋์ผ๋ก ์ค์ ํด์ฃผ๋ ํ์ผ์ด๋ผ๊ณ ์ดํดํ๋ฉด ๋๋ค.
A. ๋ฒ ์ด์ค ์ด๋ฏธ์ง ์ค์ : ๋ถ์ด๋นต์ ๋ง๋ค ๋ ์ด๋ค ๋ง์ ๋ฐ์ฃฝ์ ์ธ์ง ์ ํ๋ ๊ฐ๋ ์ผ๋ก Python์ด๋ Ubuntu ๊ฐ์ ๊ธฐ๋ณธ ์ด๋ฏธ์ง๋ฅผ ์ค์ ํ๋ค!
B. ํ์ํ ํ๋ก๊ทธ๋จ ์ค์น: ๋ถ์ด๋นต ๋ฐ์ฃฝ์ ํ์ํ ์ฌ๋ฃ๋ฅผ ๋ฃ๋ ๊ฒ๊ณผ ๊ฐ๋ค! ํ๋ก๊ทธ๋จ์ด ์คํ๋๊ธฐ ์ํด ํ์ํย ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ ํจํค์ง๋ฅผ ์ค์นํ ์ ์๋ค๋ ๋ป์ผ๋ก ์๋ฅผ ๋ค์ด, ๋ถ์ด๋นต ๋ฐ์ฃฝ์ ์ด์ฝ๋ฆฟ ์นฉ์ ๋ฃ๋ฏ, ํ๋ก๊ทธ๋จ์ ํ์ํ ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ถ๊ฐํ๋ค!
C. ์คํ ๋ช ๋ น์ด: ๋ถ์ด๋นต์ด ์์ฑ๋ ํ์ ์ด๋ป๊ฒ ๊ตฌ์ธ์ง ์ ํ๋ ๊ฒ์ฒ๋ผ ์ปจํ ์ด๋๊ฐ ์ผ์ก์ ๋ ์ด๋ค ํ๋ก๊ทธ๋จ์ ์คํํ ์ง ์ค์ ํ ์ ์๋ค! ์๋ฅผ ๋ค์ด ๋ถ์ด๋นต์ ๊ตฝ๋ ์จ๋์ ์๊ฐ์ ์ ํ ์ ์๋ ๊ฒ์ฒ๋ผ ํ๋ก๊ทธ๋จ ์ญ์ ์๋์ผ๋ก ์คํ๋ ๋ช ๋ น์ด๋ฅผ ์ค์ ํ ์ ์๋ค!
Dockrfile์ ์ฐ๋ฆฌ๊ฐ ํด๋น ํ๋ก๊ทธ๋จ์ ๋์๊ฐ๊ฒ ํ ์ ์๋ ๊ธฐ๋ณธ ํ๊ฒฝ์ ๋ง๋ค์ด์ฃผ๋ ํ์ผ๋ก ๋ฒ ์ด์ค ์ด๋ฏธ์ง๋ฅผ ์ค์ ํ๊ณ , ํ์ํ ํจํค์ง๋ ์ค์ ํ๊ณ , ์คํ ๋ช
๋ น์ด๋ ์ง์ ์ค์ ํ ์ ์๋ค๊ณ ๋ฐฐ์ ๋ค. ์๋ ์ฌ์ง์ ์์์ ์ฐ๋ฆฌ๊ฐ ์ปจํ
์ด๋ ์ด๋ฏธ์ง๋ก ์ฌ์ฉํ Kaggle R ์ด๋ฏธ์ง์ Dockerfile์ด๋ค. RUN, ADD, ENV, ARG, LABEL, CMD
๊ฐ์ ๋ช
๋ น์ด๋ค๋ก ํ์ผ์ด ๊ตฌ์ฑ๋์ด ์๋ ๊ฑธ ํ์ธํ ์ ์๋๋ฐ ์ด๋ค์ ๊ธฐ๋ฅ์ ํ๋์ฉ ์ ๋ฆฌํด๋ณด๋๋ก ํ์!
๐ป Related Commands
โ
ย ADD
: ๋ก์ปฌ ํธ์คํธ ๋ด ํ์ผ์ด๋ ํด๋๋ฅผ Docker ์ด๋ฏธ์ง๋ก ๋ณต์ฌ
- ๋ก์ปฌ ํ์ผ์ ์ปจํ ์ด๋๋ก ๋ณต์ฌํ๊ฑฐ๋ URL์์ ํ์ผ์ ๊ฐ์ ธ์ฌ ์๋ ์๊ณ ๋ง์ฝ ํด๋น ํ์ผ์ด ์์ถํ์ผ์ผ ๊ฒฝ์ฐ ์๋์ผ๋ก ํ๋ฆฌ๊ธฐ๋ ํจ
ADD kaggle/ /kaggle/
- ๋ก์ปฌ ํธ์คํธ์
kaggle/
ํด๋๋ฅผ ์ปจํ ์ด๋์/kaggle/
ํด๋๋ก ๋ณต์ฌ
- ๋ก์ปฌ ํธ์คํธ์
โ
ย RUN
: ์ด๋ฏธ์ง๋ฅผ ๋น๋ํ ๋ ์คํํ ๋ช
๋ น์ด๋ฅผ ์ง์
- ์ฃผ๋ก ํ๋ก๊ทธ๋จ์ ์ค์นํ๊ฑฐ๋ ์คํฌ๋ฆฝํธ๋ฅผ ์คํํ ๋ ์ฌ์ฉ๋๊ณ , ์ด ๋ช ๋ น์ด๋ฅผ ์ธ ๋๋ง๋ค ์๋ก์ด ๋ ์ด์ด๊ฐ ๋ง๋ค์ด์ง!
- ๐คย โ์๋ก์ด ๋ ์ด์ด๊ฐ ๋ง๋ค์ด์ง๋คโ๋ ๋ป์ ๋์ปค์ ์ด๋ฏธ์ง ๊ด๋ฆฌ ์ธก๋ฉด์์ ์ดํดํด์ผ ํ๋๋ฐ, Docker ์ด๋ฏธ์ง๋ ์ฌ๋ฌ ๊ฐ์ ์ฝ๊ธฐ ์ ์ฉ ๋ ์ด์ด๋ก ์ด๋ฃจ์ด์ ธ ์์ด ๊ฐ ๋ ์ด์ด๋ ์ด์ ๋ ์ด์ด์ ๋ณ๊ฒฝ ์ฌํญ์ ๊ธฐ์ตํ๋ค. ์๋ฅผ ๋ค์ด ํ๋์ ์ด๋ฏธ์ง๋ฅผ ๋น๋ํ ๋ ์ฐ๋ถํฌ ์ด์์ฒด์ โ ํจํค์ง ์ค์น โ ํ์ผ ๋ณต์ฌ ๋จ๊ณ๋ฅผ ๊ฑฐ์น๊ฒ ๋๋๋ฐ ์ด ํ ์คํ ์ ์ด ๋ ์ด์ด(layer)๊ฐ ๊ด๋ฆฌํ๋ค๋ ์๋ฏธ๋ค! ํจ์จ์ฑ, ์บ์ฑ์ ๊ณ ๋ คํ์ ๋ ๋งค์ฐ ํฉ๋ฆฌ์ ์ธ ๋ฐฉ๋ฒ์์ด ๋ถ๋ช ํ๋ฐ ์๋ก์ด RUN ๋ช ๋ น์ ์คํํ ๋๋ง๋ค ์๋ก์ด ๋ ์ด์ด๊ฐ ์์ฑ๋๊ธฐ ๋๋ฌธ์, ๋ถํ์ํ RUN ๋ช ๋ น์ด๋ฅผ ๋ง์ด ์ฌ์ฉํ๋ฉด ์ด๋ฏธ์ง๊ฐ ๋ถํ์ํ๊ฒ ์ปค์ง ์ ์๋ค. ๋ฐ๋ผ์ ์ฌ๋ฌ ๋ช ๋ น์ ํ RUN ๋ช ๋ น์ด์ ๋ฌถ์ด์ ์คํํ๋ ๊ฒ์ ์ถ์ฒํจ!๐
RUN R -e "keras::install_keras(tensorflow = \"${TENSORFLOW_VERSION}\", extra_packages = c(\"pandas\"))"
- R์ ์คํํด์
keras
,tensorflow
,pandas
๊ฐ์ ํจํค์ง๋ฅผ ์ค์นํจ
- R์ ์คํํด์
โ
ย ENV
: ์ปจํ
์ด๋ ์์์ ์ฌ์ฉํ ํ๊ฒฝ ๋ณ์๋ฅผ ์ค์ (docker run)
- ์ด ๋ช ๋ น์ผ๋ก ์ค์ ๋ ํ๊ฒฝ ๋ณ์๋ Docker ์ด๋ฏธ์ง ๋น๋ ํ ์ปจํ ์ด๋ ์คํ ์์๋ ์ ์ง๊ฐ ๊ฐ๋ฅํจ
ENV R_HOME=/usr/local/lib/R
R_HOME
์ด๋ผ๋ ํ๊ฒฝ ๋ณ์๋ฅผ/usr/local/lib/R
๋ก ์ค์
โ
ย ARG
: ์ด๋ฏธ์ง ๋น๋(docker build) ์์ ์ฌ์ฉํ ๋ณ์ ์ ์
ARG
๋ก ์ ์๋ ๋ณ์๋ ๋น๋ ํ์(์ด๋ฏธ์ง๋ฅผ ๋ง๋ค ๋)๊น์ง๋ง ์ฌ์ฉ๋๊ณ ์คํ๋ ๋๋ ์ฌ์ฉํ ์ ์์ARG GIT_COMMIT=unknown ARG BUILD_DATE_RSTATS=unknown
GIT_COMMIT
๊ณผBUILD_DATE_RSTATS
๋ผ๋ ์ด๋ฆ์ ๊ฐ์ง๊ณ ์๊ณ , ๊ธฐ๋ณธ๊ฐ์ โunknownโ
โ
ย LABEL
: ์ด๋ฏธ์ง์ ๋ฉํ๋ฐ์ดํฐ(์ด๋ฏธ์ง๋ฅผ ์ค๋ช
ํ๋ ์ ๋ณด)๋ฅผ ์ถ๊ฐํจ
- ์ด๋ฏธ์ง๋ฅผ ๋ง๋ ๋ ์ง๋ ์ปค๋ฐ ์ ๋ณด ๋ฑ์ ๋ฃ์ด์ ๋ฒ์ ๊ด๋ฆฌ์ ๋์์ด ๋ผ์. key-value ํ์์ผ๋ก ๋ฐ์ดํฐ ์ ์ฅ
LABEL git-commit=$GIT_COMMIT LABEL build-date=$BUILD_DATE_RSTATS
git-commit
๊ณผbuild-date
๋ผ๋ ์ด๋ฆ์ผ๋กGIT_COMMIT
๊ณผBUILD_DATE_RSTATS
๊ฐ์ ๋ผ๋ฒจ๋ก ์ง์ ํจ
โ
ย CMD
: ์ปจํ
์ด๋๊ฐ ์์๋ ๋ ๊ธฐ๋ณธ์ ์ผ๋ก ์คํํ ๋ช
๋ น์ ์ง์
docker run
๋ช ๋ น์ด๋ก ์ปจํ ์ด๋๋ฅผ ์คํํ ๋, ์ถ๊ฐ ๋ช ๋ น์ ์ง์ ํ์ง ์์ผ๋ฉด CMD์์ ์ ํ ๋ช ๋ น์ด๊ฐ ์คํ๋จCMD ["R"]
- ์ด ๋ช ๋ น์ด๋ ์ปจํ ์ด๋๊ฐ ์์๋ ๋ ๊ธฐ๋ณธ์ ์ผ๋ก R ํ๋ก๊ทธ๋จ์ ์คํ
์ด๋ ๊ฒ ์ ๋ฆฌํด๋ดค๋๋ฐ, ์ง๊ธ ์ด ๊ธ์ ์ฝ๊ณ ์๋ ๋ถ์ด ์๋ค๋ฉด ์ดํด๊ฐ ๋๋์ง ๊ฑฑ์ ์ด๋ค. ์ฌ์ค ๋ช ๋ น์ด ๊ฐ๋ ๊ฐ์ ๋ถ๋ถ์ ์ค์ ๋ก ๋ด๊ฐ ์ง์ ์ค์ต์ ํด๋ณด๋๊ฒ ๊ฐ์ฅ ๋น ๋ฅธ ์ดํด ๋ฃจํธ๋ผ๊ณ ์๊ฐํ๋ค. ๊ทธ๋์ ๋ ์ญ์ ์์์ ์ค์นํ Kaggle RStudio์ ๋์ปคํ์ผ์ ๋ถ์ํด๋ณด๊ธฐ๋ก ํ๋ค.
์๋ ์ฝ๋๋ RStudio ์๋ฒ ํ๊ฒฝ์ ๊ตฌ์ถํ๊ธฐ ์ํด ์์ฑ๋ Dockerfile์ด๋ค. ๊ฐ๋จํ ์ดํด๋ณด๋ฉด FROM gcr.io/kaggle-gpu-images/rstats:${rstats_version}
์ ํตํด ๋ฒ ์ด์ค ์ด๋ฏธ์ง(gcr.io/kaggle-gpu-images/rstats)๋ฅผ ์ค์ ํ๊ณ ์๋ค. ์ด๋ Kaggle GPU ์ด๋ฏธ์ง๋ฅผ ๊ธฐ๋ฐ์ผ๋ก ํ์ฌ R๊ณผ RStudio ์๋ฒ๋ฅผ ์ฌ์ฉํ ์ ์๊ฒ ํด์ค๋ค. ๋ฐ๋ก ๋ฐ์ RUN
๋ถ๋ถ๋ ์ค์ํ๋ฐ, ๋ค์ ๋ช
๋ น์ด๋ค์ ํตํด ์ค์ง์ ์ผ๋ก RStudio ์๋ฒ๋ฅผ ๋ค์ด๋ก๋ ํ ์ค์นํ๊ณ ์๋ค. ๊ด๋ จ ๋ช
๋ น์ด๋ค์ ๊ตฌ์ฒด์ ์ผ๋ก ์ค๋ช
ํ์ง๋ ์์๊ฑฐ์ง๋ง(๊ฒ์์ ์ถ์ฒํฉ๋๋ค!) ์๊น ์์์ ๋ ์ด์ด ๊ฐ๋
์ ์ค๋ช
ํ๋๋ฐ ์ฐ์ฅ์ ์ผ๋ก &&
๋ฅผ ์ฌ์ฉํด ๋ ์ด์ด๋ฅผ ์ค์ด๊ณ ์๋ค๋ ์ ๋ ํ์ธ ๊ฐ๋ฅํ๋ค!
WORKDIR /home/rstudio
์์์ ๋ค๋ฃฌ ๋ช
๋ น์ด๋ ์๋์ง๋ง ๋จ์ด์์ ์ ์ถํ ์ ์๋ฏ์ด ๊ตฌ์ฒด์ ์ผ๋ก ์์
๋๋ ํ ๋ฆฌ๋ฅผ ์ค์ ํด ์ค ์๋ ์๋ค.
USER rstudio
๋ฅผ ํตํด ํด๋น ์ปจํ
์ด๋์ ์ฌ์ฉ์๋ฅผ โrstudioโ๋ก ์ง์ ํด ์ฃผ์๋ค. ๊ธฐ๋ณธ์ ์ผ๋ก Docker๋ root
์ฌ์ฉ์๋ก ์คํ๋๋ค. ๋ค๋ง ๋ณด์์ด๋ ํ์ผ ๊ถํ ๋ฌธ์ ๋ก ์ธํด ํน์ ์์
์ ์ผ๋ฐ ์ฌ์ฉ์๋ก ์ง์ ํ๋ ๊ฒฝ์ฐ๊ฐ ๋ง์๋ฐ ์ด ๋ USER
๋ช
๋ น์ด๋ฅผ ์ด์ฉํด ๊ถํ์ ๋ฐ๊ฟ์ค ์ ์๋ค.
๊ทธ ์ธ์๋ ์์์ ๋ค๋ค๋ ํฌํธ๊ฐ ๋์ ๋๋๋ฐ EXPOSE 8787
์ง์ ํด์ฃผ๋ฉด์ RStudio Server ์ชฝ์ 8787๋ฒ ํฌํธ๋ฅผ ์ด์ด์ฃผ์๋ค.
๐ย RStudio server .Dockerfile
RG rstats_version=v56
#base image(๋์ปคํ์ผ์ ๊ธฐ๋ณธ ๋ฒ ์ด์ค)๋ from์ผ๋ก ๋ถ๋ฌ์ค๊ธฐ
FROM gcr.io/kaggle-gpu-images/rstats:${rstats_version}
ARG rstudio_version=2021.09.0-351
RUN apt-get update && \
apt-get install -y gdebi-core && \
wget https://download2.rstudio.org/server/focal/amd64/rstudio-server-${rstudio_version}-amd64.deb && \
gdebi -n rstudio-server-${rstudio_version}-amd64.deb && \
apt-get clean && \
rm rstudio-server-${rstudio_version}-amd64.deb
WORKDIR /home/rstudio
COPY setup.R ./setup.R
RUN chown rstudio:rstudio ./setup.R
RUN chmod +x ./setup.R
USER rstudio
RUN Rscript ./setup.R && \
rm ./setup.R
USER root
EXPOSE 8787
# LABEL revised_by="Daniel Youk" \
# revised_date="2023-12-24"
ENTRYPOINT [ "/bin/bash"]
CMD ["-c", "rstudio-server start && tail -f /dev/null"]
๊ทธ๋ฆฌ๊ณ ํ๊ฐ์ง ์ค์ํ ์ ์ CMD ["-c", "rstudio-server start && tail -f /dev/null"]
์ด ๋ช
๋ น์ด๋ค. ์์์ CMD
๋ฅผ โ์ปจํ
์ด๋๊ฐ ์์๋ ๋ ๊ธฐ๋ณธ์ ์ผ๋ก ์คํํ ๋ช
๋ น์ ์ง์ โํ๋ค๊ณ ์ค๋ช
ํ๋ค. ์ด ๋ถ๋ถ์ ๋ด ๊ฒฝํ ์ ์์ ํ๋์ ๋น๊ต๋ฅผ ํตํด ๋ฐ๋ก ์ดํด๊ฐ ๊ฐ๋ฅํ๋ค.
์๋ ์ฌ์ง์ ์์์ ๋ค๋ฃฌ Kaggle R ์ด๋ฏธ์ง์ Dockerfile์ ๊ฐ์ฅ ๋ง์ง๋ง ์ค์ ์บก์ณ ๋ถ๋ถ์ด๋ค. ๋ณด๋ฉด CMD [โRโ]
์ด๋ผ๊ณ ์ ํ์๋๋ฐ, CMD
๋ก ์ปจํ
์ด๋ ์์ ๋ช
๋ น์ด๋ฅผ ์ง์ ํ ์ ์๊ธฐ ๋๋ฌธ์ ํด๋น ์ปจํ
์ด๋๋ฅผ ์คํํ๊ธฐ ์ํ ๊ธฐ๋ณธ ๋ช
๋ น์ด(docker exec)๋ฅผ ์์ฉํด docker exec -it rstudio-container R
์ด๋ ๊ฒ ์ฌ์ฉํ ์ ์๋ค. ์ด ๋ช
๋ น์ด๋ก ๋ฐ๋ก rstudio-container์ R ํ๋ก๊ทธ๋จ์ ์ ๊ทผํ ์ ์๋ ๊ฒ์ด๋ค.
๊ทธ๋ฐ๋ฐ RStudio Server๋ฅผ ๊ตฌ์ถํ๋ ๋์ปคํ์ผ์๋ CMD
๋ช
๋ น์ด๊ฐ ์๋ค. ๊ทธ๋ฐ๋ฐ ๋ถ๋ช
์๊น์๊ฐ ๋ค๋ฅธ ๊ฑธ๋ก ๋ณด์ ์ฐจ์ด์ ์ด ์๋ ๊ฒ์ด ํ์คํ๋ค.(๋ญ์ง..) ๊ฒฐ๋ก ๋ถํฐ ๋งํ์๋ฉด CMD ["-c", "rstudio-server start && tail -f /dev/null"]
๋ช
๋ น์ด๋ RStudio ์๋ฒ๋ฅผ ์์ํ๊ณ ์ปจํ
์ด๋๊ฐ ์ข
๋ฃ๋์ง ์๊ฒ ์ ์งํ๋ ๊ธฐ๋ฅ์ ํ๋ค. /dev/null
์ ๋ฆฌ๋
์ค ์์คํ
์์ ํน๋ณํ ํ์ผ๋ก ์
๋ ฅ๋ ๋ฐ์ดํฐ๋ฅผ ๋ชจ๋ ๋ฒ๋ฆฌ๋ ์ญํ ์ ํ๋๋ฐ tail -f
๋ช
๋ น์ด๋ก /dev/null
ํ์ผ์ ๊ณ์ ์ฐพ์ผ๋ ค๊ณ ํ๋ค. ํ์ง๋ง ํด๋น ํ์ผ์ ์์ ๋งํ๋ฏ์ด ๋ฌด์ธ๊ฐ ์
๋ ฅ๋๋ฉด ๋ฐ์ดํฐ๋ฅผ ๋ชจ๋ ๋ฒ๋ฆฌ๋ ์ญํ ์ ํ๊ธฐ ๋๋ฌธ์ ๋ฌดํ๋ฃจํ์ ๋น ์ง๊ฒ ๋๊ณ ์ปจํ
์ด๋๊ฐ ์ข
๋ฃ๋์ง ์๊ฒ ์ ์งํ๋ ๊ธฐ๋ฅ์ ํ ์ ์๋ ๊ฒ์ด๋ค!
3.1 Difference Between Docker CMD and ENTRYPOINT: When and How to Use Them?
Dockerfile์ ์์ฑํ ๋ CMD์ ENTRYPOINT ๋ ๋ช ๋ น์ด๊ฐ ๊ณ์ ๋์๋ค. ๋๋ค ์ปจํ ์ด๋๊ฐ ์์๋ ๋ ์คํ๋ ๋ช ๋ น์ ์ง์ ํ๋ ๋ฐ ์ฌ์ฉ๋๋ ๊ฑด ๋๊ฐ์ง๋ง ๊ทธ ์ญํ ๊ณผ ๋์ ๋ฐฉ์์ ์ฐจ์ด๊ฐ ์๋ ๊ฒ ๊ฐ๋ค. ์ด ๋ถ๋ถ๋ ๋ค์๋ ํท๊ฐ๋ฆฌ๊ธฐ ์ซ์ผ๋ ์ฌ๊ธฐ์ ์ ๋ฆฌํด ๋ณด๋ ค๊ณ ํ๋ค!
๐ชย CMD
๊ณ์ ์ธ๊ธ๋์๋ CMD๋ ์ปจํ
์ด๋๊ฐ ์คํ๋ ๋๋ง๋ค ์ํ๋์ง๋ง ๋ฎ์ด์ฐ๊ธฐ๊ฐ ๊ฐ๋ฅํ๋ค. ๋ง์ฝ ์ฌ์ฉ์๊ฐ docker run
๋ช
๋ น์ด๋ฅผ ์คํํ๋ฉด์ ๋ค๋ฅธ ๋ช
๋ น์ ์
๋ ฅํ๋ฉด CMD์์ ์ง์ ํ ๋ช
๋ น์ ์คํ๋์ง ์๊ณ ์๋ก ์
๋ ฅํ ๋ช
๋ น์ด ๋์ ์คํ์ด ๊ฐ๋ฅํ๋ค. ์์๋ฅผ ํตํด ์ดํดํด๋ณด๋๋ก ํ์!
์ฐ๋ฆฌ๋ CMD ["R"]
์ด ์ฝ๋๋ฅผ ์ด์ฉํด ์ปจํ
์ด๋๊ฐ ์์๋๋ฉด R ์ฝ์์ด ๊ธฐ๋ณธ์ผ๋ก ์คํ๋จ์ ์์์ ํ์ธํ๋ค. ๊ทธ๋ฐ๋ฐ ๋ง์ฝ ์ฌ์ฉ์๊ฐ ์ปจํ
์ด๋๋ฅผ ์คํ ํ๋ฉด์ ๋ค๋ฅธ ๋ช
๋ น์ ์
๋ ฅํ๋ค๋ฉด CMD์์ ์ง์ ํ R
์ ๋ฌด์๋๊ณ ์๋ก์ด ๋ช
๋ น์ด ์คํ๋๋ค. ์ด๊ฒ ๋ฌด์จ ๋ง์ธ๊ฐ ํ๋ฉด docker run <image> /bin/bash
๋ก ๋ช
๋ น์ด๋ฅผ ์คํํ๋ฉด R
๋์ /bin/bash
๊ฐ ์คํ ๋๋ค๋ ๋ง์ด๋ค.
๐ชย Entry Point
ENTRYPOINT๋ ํญ์ ์คํ๋ ๋ช
๋ น์ ์ง์ ํ๋ ๋ฐ ์ฌ์ฉ๋๋ค. CMD์ ๋ฌ๋ฆฌ docker run
์์ ๋ค๋ฅธ ๋ช
๋ น์ ์
๋ ฅํด๋ ENTRYPOINT์์ ์ง์ ํ ๋ช
๋ น์ ๋ฎ์ด์ฐ์ด์ง ์์ผ๋ฉฐ ๊ทธ๋๋ก ์คํ๋๋ ๊ฒ ๋์ ๊ฐ์ฅ ํฐ ํน์ง์ด๋ค. ์ด๊ฒ๋ ์์๋ฅผ ํตํด ์ดํดํด๋ณด๋ก ํ์!
ENTRYPOINT ["/bin/bash", "-c"]
์ด ์ฝ๋๋ ์ปจํ
์ด๋๊ฐ ์คํ๋ ๋ ํญ์ /bin/bash -c
๋ฅผ ์คํํ๋๋ก ํ๋ค. ๊ทธ๋ฐ๋ฐ ๋ง์ฝ ์ฌ์ฉ์๊ฐ docker run
์ ์ด์ฉํด docker run <image> "echo Hello"
๋ฅผ ์คํํ๋ฉด ๊ทธ ๋ช
๋ น์ /bin/bash -c
์ ์ธ์๋ก ์ ๋ฌ๋์ด /bin/bash -c "echo Hello"
๋ก ์คํ๋๋ค!
๐ย summary
ํญ๋ชฉ | CMD | ENTRYPOINT |
---|---|---|
๋ชฉ์ | ๊ธฐ๋ณธ ๋ช ๋ น์ ์ค์ (๋ฎ์ด์ฐ๊ธฐ ๊ฐ๋ฅ) | ํ์ ๋ช ๋ น์ ์ค์ (ํญ์ ์คํ๋จ) |
๋ฎ์ด์ฐ๊ธฐ | `docker run` ๋ช ๋ น์ด๋ก ๋ฎ์ด์์ | ๋ฎ์ด์ฐ์ง ์๊ณ , ์ธ์๋ฅผ ์ ๋ฌ ๊ฐ๋ฅ |
์ ์ฐ์ฑ | ๋ค๋ฅธ ๋ช ๋ น์ผ๋ก ๋์ฒดํ ์ ์์ | ์ธ์๋ฅผ ํตํด ๋ช ๋ น ์ถ๊ฐ ๊ฐ๋ฅ |
์์ | `CMD ["R"]` | `ENTRYPOINT ["/bin/bash", "-c"]` |
๊ฒฐ๋ก ์ ์ผ๋ก ๋ ๋ช ๋ น์ด ๋ชจ๋ ์ปจํ ์ด๋๊ฐ ์์๋ ๋ ์คํ๋ ๋ช ๋ น์ ์ง์ ํ์ง๋ง CMD๋ ๊ธฐ๋ณธ ์คํ ๋ช ๋ น์, ENTRYPOINT๋ ํญ์ ์คํ๋์ด์ผ ํ๋ ํ์ ๋ช ๋ น์ ์ค์ ํ๋ ๋ฐ ์ฌ์ฉ๋๋ค๋ ์ ์ ๊ผญ๊ธฐ์ตํ๊ธธ ๋ฐ๋๋ค! ์๋ ์ฝ๋๋ ์ค์ ํ๋ก์ ํธ์์ ๋ง์ด ์ฌ์ฉํ๋ ์กฐํฉ์ด๋ ์ฐธ๊ณ ํ๋ฉด ์ข์ ๋ฏ ํ๋ค!
ENTRYPOINT ["/bin/bash", "-c"]
CMD ["rstudio-server start && tail -f /dev/null"]
4. Docker Pipeline
์ด๋ฒ ํฌ์คํ ์ ๋ง๋ฌด๋ฆฌ ํ๊ธฐ ์ ์ ์ง๊ธ๊น์ง์ ํ์ต ๊ณผ์ ์ ์๊ฐ์ ์ผ๋ก ์ ๋ฆฌํด ๋ณด๋ ค๊ณ ํ๋ค. ์ง๊ธ๊น์ง์ ํฌ์คํ ์ Dockerfile์ ํตํด ์ด๋ฏธ์ง ์์ฑํ๊ณ ๊ทธ ์ด๋ฏธ์ง๋ฅผ ํตํด ์ปจํ ์ด๋ ์คํํ๋ ๋ถ๋ถ๊น์ง ์งํ๋์๋ค. (Dockerfile โ ์ด๋ฏธ์ง ์์ฑ โ Registry ์ ์ฅ โ ์ด๋ฏธ์ง ๊ฐ์ ธ์ค๊ธฐ โ ์ปจํ ์ด๋ ์คํ)
์ง๊ธ์ด์ผ ์ด๋ฏธ์ง๋ฅผ ๋ง๋ค๊ณ ๋ก์ปฌ์์ ์คํํ๋ ๋ฐ ํฐ ๋ฌธ์ ๊ฐ ์์๋ค. ํ์ง๋ง ์ปจํ ์ด๋์์ ์ฌ์ฉํ๋ ์ด๋ฏธ์ง๋ค์ด ๋ง์์ง๋ค๋ฉด ๋ก์ปฌ์ด๋ผ๋ ํ์ ๋ ๊ณต๊ฐ์์ ํจ์จ์ฑ์ด ๋จ์ด์ง๊ฒ ๋๋ค. ๊ทธ๋์ ๋ณดํต์ Docker Registry๋ผ๋ ๊ณณ์ ์ด๋ฏธ์ง๋ฅผ ์ ์ฅ(push)ํ๊ณ ํ์ํ ๋๋ง๋ค ๋น๊ฒจ์(pull) ๋ก์ปฌ ํน์ ๊ฐ์๋จธ์ ์์ ์ฌ์ฉํ๋ ๊ฒ ์ผ๋ฐ์ ์ด๋ค. ๋ค์ ํฌ์คํ ๋ถํฐ๋ ์ด ์ด์ผ๊ธฐ๋ฅผ ์ค์ฌ์ผ๋ก ์งํ๋๋ ๊ธฐ๋ํด ์ฃผ๊ธธ ๋ฐ๋๋ค! ๐
์ด๋ฒ ์ฃผ์ฐจ๋ ์ง๊ธ๊น์ง ๋์ปค๋ฅผ ๊ณต๋ถํ๋ฉฐ ๋น์ ์ ์ผ๋ก ์ดํดํ๋ ๋ถ๋ถ๋ค์ ๋ํด ์กฐ๊ธ ๋ ์ํํธ์จ์ด ์ธก๋ฉด์์ ์ด๋ก ์ ๋ค๋ฃจ์๋ค. ๊ทธ๋ฌ๋ค๋ณด๋ ๋ญ๋๊น.. ๋ถ๋ช ๋ด๊ฐ ์๊ณ ์๋ ๋ด์ฉ์ด์๋ ๊ฒ ๊ฐ์๋ฐ ๋๋ค์ ์๋ก์ด ๋ด์ฉ์ด ๋์ด๋ฒ๋ฆฐ ๊ฒ ๊ฐ์๋ค. ํนํ Dockerfile์ ENV, ARG ๋ช ๋ น์ด์ ๋ํด์๋ ์ด๋ฏธ์ง๋ฅผ ๋น๋ํ ๋, ์ปจํ ์ด๋๋ฅผ ์คํํ ๋๋ก ๊ตฌ๋ถ๋๋๋ฐ, ๊ตฌ์ฒด์ ์ผ๋ก ๋์ปคํ์ผ์ ์ง์ ์์ฑํด๋ณด๋ฉด์ ๊ฐ๋ ์ ๋ค์ง ํ์๊ฐ ์๋ค๊ณ ๋๊ผ๋ค.
์๋ฌดํผ ์ด๋ฒ์ฃผ์ฐจ๋ฅผ ๊ธฐ์ ์ผ๋ก ๋์ปค ํฌ์คํ ์๋ฆฌ์ฆ๋ ๋ฐํ์ ์ ๋์๋ค. ์ฌ์ ํ ๋ง์ด ๋ถ์กฑํ ๊ธ์ด์ง๋ง ๋ถ๋ช ๊ธ๋ก ๋จ๊ธฐ๊ณ ์ ๋ฆฌํ๋ ๊ฑด ์ดํด์ ๋๋๊ฐ ๋ค๋ฅผ ๊ฑฐ๋ผ๊ณ ์๊ฐํ๋ค. ์ด๋ฒ์ฃผ๋ ์ ๋ํ ๋ญ ํด๋ ์ง์ค์ด ์๋๋ ๋ ์ด์๋๋ฐ ์ผ๋จ ๋ฒํ จ๋ณด๋ ๊ฑธ๋ก ํ์!๐
Reference
Subscribe via RSS