<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>Forem: Nguyễn Thế Minh</title>
    <description>The latest articles on Forem by Nguyễn Thế Minh (@minhpro).</description>
    <link>https://forem.com/minhpro</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F1042514%2F48e6ba0f-5e23-4d45-81ae-1dbee03af61a.jpeg</url>
      <title>Forem: Nguyễn Thế Minh</title>
      <link>https://forem.com/minhpro</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://forem.com/feed/minhpro"/>
    <language>en</language>
    <item>
      <title>Docker overview</title>
      <dc:creator>Nguyễn Thế Minh</dc:creator>
      <pubDate>Fri, 10 Mar 2023 10:14:21 +0000</pubDate>
      <link>https://forem.com/minhpro/docker-overview-4o2d</link>
      <guid>https://forem.com/minhpro/docker-overview-4o2d</guid>
      <description>&lt;h2&gt;
  
  
  Overview
&lt;/h2&gt;

&lt;p&gt;Docker là một open platform dùng cho việc phát triển, chuyển giao và run application.&lt;/p&gt;

&lt;p&gt;Docker cung cấp khả năng đóng gói và chạy app trong một môi trường isolated gọi là &lt;strong&gt;container&lt;/strong&gt;. Do đó chúng ta có thể chạy nhiều containers&lt;br&gt;
đồng thời trên một máy mà không lo ngại chúng sẽ xung đột với nhau hay với chính host machine.&lt;/p&gt;
&lt;h2&gt;
  
  
  Docker Architecture
&lt;/h2&gt;

&lt;p&gt;&lt;a href="https://res.cloudinary.com/practicaldev/image/fetch/s--n18BSz4g--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/uploads/articles/i9e1p96gd1gcbz17il2c.png" class="article-body-image-wrapper"&gt;&lt;img src="https://res.cloudinary.com/practicaldev/image/fetch/s--n18BSz4g--/c_limit%2Cf_auto%2Cfl_progressive%2Cq_auto%2Cw_880/https://dev-to-uploads.s3.amazonaws.com/uploads/articles/i9e1p96gd1gcbz17il2c.png" alt="Image description" width="880" height="460"&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Docker daemon&lt;/strong&gt; (dockerd): là một process chạy ngầm. dockerd quản lý &lt;em&gt;docker objects&lt;/em&gt;: images, containers, networks và volumes.&lt;br&gt;
Một dockerd có thể giao tiếp với dockerd khác (trên host khác) để quản lý docker &lt;em&gt;services&lt;/em&gt;.&lt;br&gt;
Dockerd cung cấp API để &lt;strong&gt;docker client&lt;/strong&gt; có thể tương tác (e.g. quản lý objects) với nó.&lt;/p&gt;

&lt;p&gt;Docker objects:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Images&lt;/strong&gt;: một image là template chứa các instructions (lệnh) dùng để tạo Docker containers. Thông thường, một image sẽ được based từ một
image khác và thêm vào đó phần customization. Ví dụ build một image based từ image &lt;em&gt;ubuntu&lt;/em&gt; nhưng có cài thêm Apache server và application.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Containers&lt;/strong&gt;: một container là một runnable instance của image. Container là &lt;em&gt;isolated&lt;/em&gt; với container khác và với host machine. Bạn có thể
kiểm soát tính &lt;em&gt;isolated&lt;/em&gt; của container network và storage với container khác và với host machine. &lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Khi một container bị removed thì mọi thay đổi và trạng thái (state, e.g. file, data, ...) không được lưu lại vào persistent storage, sẽ biến mất.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Docker client&lt;/strong&gt; hay &lt;em&gt;docker&lt;/em&gt; là một command line interface (CLI) dùng để tương tác với &lt;em&gt;dockerd&lt;/em&gt; thông qua các command (sử dụng API)&lt;br&gt;
ví dụ như &lt;em&gt;docker run&lt;/em&gt; (tạo container). Docker client có thể tương tác với nhiều &lt;em&gt;dockerd&lt;/em&gt; (cấu hình DOCKER_HOST).&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Docker registries&lt;/strong&gt; là nơi lưu trữ &lt;em&gt;docker images&lt;/em&gt;. &lt;a href="https://hub.docker.com/"&gt;Docker hub&lt;/a&gt; là một public registries, ai cũng dùng được.&lt;br&gt;
Bạn cũng có thể tạo private registry của mình. Docker được cấu hình tìm kiếm docker images từ Docker Hub. Bạn có thể pull image từ registry&lt;br&gt;
về máy local hay push image của mình lên registry.&lt;/p&gt;
&lt;h2&gt;
  
  
  Get started
&lt;/h2&gt;

&lt;p&gt;Để bắt đầu với Docker, thực hiện theo bài hướng dẫn ở link dưới đây:&lt;/p&gt;

&lt;p&gt;&lt;a href="https://docs.docker.com/get-started"&gt;https://docs.docker.com/get-started&lt;/a&gt;&lt;/p&gt;
&lt;h2&gt;
  
  
  Build docker images
&lt;/h2&gt;

&lt;p&gt;Docker builds images bằng cách đọc các lệnh từ &lt;strong&gt;Dockerfile&lt;/strong&gt; - một file text chứa tất cả các lệnh để build image.&lt;/p&gt;

&lt;p&gt;Format của Dockerfile:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="c"&gt;# Comment&lt;/span&gt;
INSTRUCTION arguments
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Các lệnh không phân biệt hoa thường. Nhưng convention thì nên viết lệnh UPPERCASE để phân biệt với tham số.&lt;/p&gt;

&lt;p&gt;rMột Dockerfile &lt;strong&gt;phải bắt đầu&lt;/strong&gt; bằng một lệnh &lt;strong&gt;FROM&lt;/strong&gt; để chỉ định &lt;em&gt;Base image&lt;/em&gt; và từ đó dùng cho các lệnh tiếp theo.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;FROM&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;FROM [--platform=&amp;lt;platform&amp;gt;] &amp;lt;image&amp;gt; [AS &amp;lt;name&amp;gt;]&lt;/code&gt;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;FROM&lt;/code&gt; có thể xuất hiện nhiều lần trong Dockerfile trong trường hợp sử dụng &lt;em&gt;build stage&lt;/em&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;AS name&lt;/code&gt; gán tên cho build stage và có thể được dùng cho lệnh &lt;code&gt;FROM&lt;/code&gt; sau đó, ví dụ &lt;code&gt;COPY --from&amp;lt;name&amp;gt;&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;--platform&lt;/code&gt; chỉ định platform của image, ví dụ: linux/amd64, linux/arm64, hay windows/amd64. Mặc định là platform của máy.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;RUN&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Lệnh RUN dùng để chạy bất kỳ commands nào trong một layer mới của image hiện tại và commit kết quả.&lt;/p&gt;

&lt;p&gt;Lệnh RUN có 2 forms:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;RUN &amp;lt;command&amp;gt;&lt;/code&gt; (&lt;em&gt;shell&lt;/em&gt; form, the command được chạy trong một shell, mặc định trong Linux là &lt;code&gt;/bin/sh -c&lt;/code&gt; và &lt;code&gt;cmd /S /C&lt;/code&gt; trong Windows)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;RUN ["executable", "param1", "param2", ..]&lt;/code&gt; (&lt;em&gt;exec&lt;/em&gt; form)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;CMD&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;CMD&lt;/code&gt; có 3 forms:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;CMD ["executable", 'param1", "param2", ..]&lt;/code&gt; (&lt;em&gt;exec&lt;/em&gt; form)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;CMD ["param1", "param2", ..]&lt;/code&gt;(như là &lt;em&gt;default parameters&lt;/em&gt; cho &lt;code&gt;ENTRYPOINT&lt;/code&gt;)&lt;/li&gt;
&lt;li&gt;
&lt;code&gt;CMD command param1 param2 ..&lt;/code&gt; (&lt;em&gt;shell&lt;/em&gt; form)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Có duy nhất một lệnh &lt;code&gt;CMD&lt;/code&gt; trong một Dockerfile. Nếu có nhiều thì chỉ lệnh &lt;code&gt;CMD&lt;/code&gt; cuối cùng có tác dụng.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Mục đích chính&lt;/strong&gt; của &lt;code&gt;CMD&lt;/code&gt; là &lt;strong&gt;cung cấp chương trình mặc định khi chạy container&lt;/strong&gt;. Nếu không có thì bạn phải chỉ định&lt;br&gt;
&lt;code&gt;ENTRYPOINT&lt;/code&gt; instruction.&lt;/p&gt;

&lt;p&gt;Nếu &lt;code&gt;CMD&lt;/code&gt; được sử dụng để cung cấp default cho &lt;code&gt;ENTRYPOINT&lt;/code&gt;, thì cả hai lệnh cần được khai báo dưới dạng ARRAY (&lt;code&gt;["abc", "xyz",..]&lt;/code&gt;)&lt;/p&gt;

&lt;p&gt;Khác với &lt;em&gt;shell&lt;/em&gt; form, thì &lt;em&gt;exec&lt;/em&gt; form sẽ không gọi command shell, đồng nghĩa với xử lý shell thông thường không xảy ra. Ví dụ&lt;br&gt;
&lt;code&gt;CMD ["echo", "$HOME"]&lt;/code&gt; sẽ không thực hiện thay thế biến môi trường &lt;code&gt;$HOME&lt;/code&gt;. Nếu muốn xử lý shell thì hoặc sử dụng &lt;em&gt;shell&lt;/em&gt; form hay &lt;br&gt;
chạy shell trực tiếp, ví dụ như &lt;code&gt;CMD [ "sh", "-c", "echo $HOME"]&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Khi sử dụng &lt;em&gt;exec&lt;/em&gt; form và chạy trực tiếp shell thì shell sẽ sử lý thay thế biến môi trường, chứ không phải docker.&lt;/p&gt;

&lt;p&gt;Khi được sử dụng ở dạng &lt;em&gt;shell&lt;/em&gt; hay &lt;em&gt;exec&lt;/em&gt;, &lt;code&gt;CMD&lt;/code&gt; được command được chạy khi running image (chạy container).&lt;/p&gt;

&lt;p&gt;Nếu sử dụng &lt;em&gt;shell&lt;/em&gt; form của &lt;code&gt;CMD&lt;/code&gt;, thì &lt;code&gt;&amp;lt;command&amp;gt;&lt;/code&gt; sẽ được chạy với &lt;code&gt;/bin/sh -c&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;FROM ubuntu
CMD &lt;span class="nb"&gt;echo&lt;/span&gt; &lt;span class="s2"&gt;"this is the shell form cmd."&lt;/span&gt; | &lt;span class="nb"&gt;wc&lt;/span&gt; -
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Nếu muốn chạy &lt;code&gt;&amp;lt;command&amp;gt;&lt;/code&gt; without shell, bạn phải sử dụng array form và chỉ định path tới executable.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;FROM ubuntu
CMD &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"/ur/bin/wc"&lt;/span&gt;, &lt;span class="s2"&gt;"--help"&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Nếu muốn container chạy same executable every time, bạn cần sử dụng lệnh &lt;code&gt;ENTRYPOINT&lt;/code&gt; (có thể combination với &lt;code&gt;CMD&lt;/code&gt;).&lt;/p&gt;

&lt;p&gt;Nếu bạn chỉ định command khi run container từ command &lt;code&gt;docker run &amp;lt;IMAGE&amp;gt; [COMMAND]&lt;/code&gt; thì command này sẽ overwrite default&lt;br&gt;
được chỉ định trong &lt;code&gt;CMD&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Không nhầm lẫn &lt;code&gt;RUN&lt;/code&gt; với &lt;code&gt;CMD&lt;/code&gt;. &lt;code&gt;RUN&lt;/code&gt; thực sự chạy command và commit kết quả; &lt;code&gt;CMD&lt;/code&gt; không chạy gì cả tại thời điểm build.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;EXPOSE&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;EXPOSE &amp;lt;port&amp;gt; [&amp;lt;port&amp;gt;/&amp;lt;protocol&amp;gt;...]&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;EXPOSE&lt;/code&gt; instruction báo rằng container sẽ listen nhưng ports nào lúc chạy. Bạn có thể chỉ định protocol là TCP hay UDP, &lt;br&gt;
mặc định là TCP. &lt;code&gt;EXPOSE&lt;/code&gt; thực tế không publish port, muốn publish port, sử dụng &lt;code&gt;-p&lt;/code&gt; khi chạy container &lt;code&gt;docker run&lt;/code&gt;.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;EXPOSE 80/tcp
EXPOSE 80/udp
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Publish port lúc chạy container:&lt;/p&gt;

&lt;p&gt;&lt;code&gt;$ docker run -p 80:80/tcp -p 80:80/udp&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;ENV&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ENV &amp;lt;key&amp;gt;=&amp;lt;value&amp;gt; ...&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ENV&lt;/code&gt; instruction set biến môi trường, giá trị của các biến này sẽ ở môi trường của các instruction phía sau.&lt;br&gt;
Các biến này được persist khi container chạy. Bạn có thể xem giá trị của chúng bằng &lt;code&gt;docker inspect&lt;/code&gt;, &lt;br&gt;
và thay đổi giá trị &lt;code&gt;docker run --env &amp;lt;key&amp;gt;=&amp;lt;value&amp;gt;&lt;/code&gt; (-e is short hand).&lt;/p&gt;

&lt;p&gt;Nếu một biến môi trường chỉ được sử dụng khi build mà không phải cho the final image, hãy cân nhắc set giá trị&lt;br&gt;
trong command:&lt;/p&gt;

&lt;p&gt;&lt;code&gt;RUN DEBIAN_FRONTEND=noninteractive apt-get update &amp;amp;&amp;amp; apt-get install -y ...&lt;/code&gt; &lt;/p&gt;

&lt;p&gt;Hay là sử dụng &lt;code&gt;ARG&lt;/code&gt;, sẽ không peristent trong image cuối cùng.&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;ARG &lt;span class="nv"&gt;DEBIAN_FRONTEND&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;noninteractive
RUN apt-get update &lt;span class="o"&gt;&amp;amp;&amp;amp;&lt;/span&gt; apt-get &lt;span class="nb"&gt;install&lt;/span&gt; &lt;span class="nt"&gt;-y&lt;/span&gt; ...
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;strong&gt;ADD&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;ADD có 2 forms:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;ADD &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="nt"&gt;--chown&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&amp;lt;user&amp;gt;:&amp;lt;group&amp;gt;] &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="nt"&gt;--checksum&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&amp;lt;checksum&amp;gt;] &amp;lt;src&amp;gt;... &amp;lt;dest&amp;gt;
ADD &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="nt"&gt;--chown&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&amp;lt;user&amp;gt;:&amp;lt;group&amp;gt;] &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"&amp;lt;src&amp;gt;"&lt;/span&gt;,... &lt;span class="s2"&gt;"&amp;lt;dest&amp;gt;"&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt; 
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Form thứ hai cần khi path có chứa white spaces.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ADD&lt;/code&gt; instruction thực hiện copy files, directories hay remote file URLs từ &lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; và thêm chúng vào filesystem của image&lt;br&gt;
tại path &lt;code&gt;&amp;lt;dest&amp;gt;&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Nhiều &lt;code&gt;&amp;lt;src&lt;/code&gt; có thể được chỉ định. Nếu là files hay directories thì path của chúng được quy ước là tương đối với thư mục khi build (build context).&lt;/p&gt;

&lt;p&gt;&lt;code&gt;&amp;lt;dest&amp;gt;&lt;/code&gt; là đường dẫn tuyệt đối hay tương đối với &lt;code&gt;WORKDIR&lt;/code&gt;, chỉ định source được copy vào chỗ nào của container.&lt;/p&gt;

&lt;p&gt;Add "test.txt" to &lt;code&gt;&amp;lt;WORKDIR&amp;gt;/relativeDir&lt;/code&gt;:&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ADD test.txt relativeDir&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;Hay sử dụng absolute path&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ADD test.txt /absoluteDir&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;Tất cả file và thư mục mới được tạo với a UID và GID 0, trừ khi sử dụng option &lt;code&gt;--chown&lt;/code&gt;, user and group can be name string or integer id.&lt;br&gt;
Name sẽ được lookup từ &lt;code&gt;/etc/passwd&lt;/code&gt; và &lt;code&gt;/etc/group&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Notes: nếu build bằng cách pass &lt;code&gt;Dockerfile&lt;/code&gt; through STDIN (&lt;code&gt;docker build - &amp;lt; somefile&lt;/code&gt;), khi đó không có build context, &lt;code&gt;Dockerfile&lt;/code&gt; chỉ&lt;br&gt;
có thể chứa URL based của &lt;code&gt;ADD&lt;/code&gt;. Có thể pass một compressed archive through STDIN (&lt;code&gt;docker build - &amp;lt; archive.tar.gz&lt;/code&gt;), &lt;code&gt;Dockerfile&lt;/code&gt; nằm ở&lt;br&gt;
root của archive and phần còn lại của archive sẽ được sử dụng như là context của build.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ADD&lt;/code&gt; tuân theo các rules sau:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; path phải nằm trong &lt;code&gt;context&lt;/code&gt; của build, bạn không thể &lt;code&gt;ADD ../something /somthing&lt;/code&gt; bởi vì bước đầu tiên khi build
&lt;code&gt;docker build&lt;/code&gt; là send context directory (và subdirectories) cho &lt;code&gt;docker daemon&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;Nếu &lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; là a URL và &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt; không kết thúc với &lt;code&gt;/&lt;/code&gt;, thì file download được từ URL sẽ được copy tới &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;Nêu &lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; là a URL và &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt; kết thúc với &lt;code&gt;/&lt;/code&gt;, thì file download được sẽ lưu tại &lt;code&gt;&amp;lt;dect&amp;gt;/&amp;lt;filename&amp;gt;&lt;/code&gt;. Ví dụ
&lt;code&gt;ADD http://example.com./foobar /&lt;/code&gt; sẽ lưu file tại &lt;code&gt;/foobar&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;Nếu &lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; là một directory, toàn bộ contents của directory được copy, bao gồm của filesystem metadata&lt;/li&gt;
&lt;li&gt;Nếu nhiều &lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; được dùng, sử dụng trực tiếp hay wildcard, thì &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt; phải là một directory, và phải kết thúc với &lt;code&gt;/&lt;/code&gt;.&lt;/li&gt;
&lt;li&gt;Nếu &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt; không kết thúc với &lt;code&gt;/&lt;/code&gt;, nó sẽ được coi là một file thông thường và nội dung của &lt;code&gt;&amp;lt;src&amp;gt;&lt;/code&gt; sẽ được lưu tại &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt;
&lt;/li&gt;
&lt;li&gt;Nếu không có &lt;code&gt;&amp;lt;dect&amp;gt;&lt;/code&gt; thì nó sẽ được tạo cùng với tất cả các thư mục còn thiếu trong path của nó.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;strong&gt;COPY&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;COPY&lt;/code&gt; có 2 forms:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;COPY &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="nt"&gt;--chown&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&amp;lt;user&amp;gt;:&amp;lt;group&amp;gt;] &amp;lt;src&amp;gt;... &amp;lt;dest&amp;gt;
COPY &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="nt"&gt;--chown&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&amp;lt;user&amp;gt;:&amp;lt;group&amp;gt;] &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"&amp;lt;src&amp;gt;"&lt;/span&gt;,... &lt;span class="s2"&gt;"&amp;lt;dest&amp;gt;"&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Form thứ hai được dùng khi path có chứa whitespace.&lt;/p&gt;

&lt;p&gt;Đặc biệt là &lt;code&gt;COPY&lt;/code&gt; nhận một flag &lt;code&gt;--from=&amp;lt;name&amp;gt;&lt;/code&gt; được sử dụng để chỉ định source location trỏ tới &lt;code&gt;build stage&lt;/code&gt; phía trước&lt;br&gt;
(tạo ra bời &lt;code&gt;FROM.. AS &amp;lt;name&amp;gt;&lt;/code&gt;).&lt;/p&gt;

&lt;p&gt;&lt;code&gt;COPY&lt;/code&gt; hoạt động giống &lt;code&gt;ADD&lt;/code&gt;, nhưng khác là &lt;code&gt;COPY&lt;/code&gt; không tự động extraction và không dùng với URL.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;ENTRYPOINT&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;The &lt;em&gt;exec&lt;/em&gt; form (preferred):&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ENTRYPOINT ["executable", "param1", "param2"]&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;The &lt;em&gt;shell&lt;/em&gt; form:&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ENTRYPOINT command param1 param2&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;ENTRYPOINT&lt;/code&gt; dùng để cấu hình executable cho container.&lt;/p&gt;

&lt;p&gt;Command line arguments của &lt;code&gt;docker run &amp;lt;image&amp;gt;&lt;/code&gt; sẽ được appended sau các phần trong &lt;em&gt;exec&lt;/em&gt; form của &lt;code&gt;ENTRYPOINT&lt;/code&gt;, và cũng &lt;br&gt;
override tất cả các phần của &lt;code&gt;CMD&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;&lt;em&gt;shell&lt;/em&gt; form hạn chế (prevent) mọi &lt;code&gt;CMD&lt;/code&gt; và &lt;code&gt;run&lt;/code&gt; command arguments. &lt;code&gt;ENTRYPOINT&lt;/code&gt; dạng shell sẽ chạy command như là subcommand&lt;br&gt;
của &lt;code&gt;/bin/sh -c&lt;/code&gt;, và không pass signals. Điều này có nghĩa là &lt;em&gt;executable&lt;/em&gt; process không phải là process &lt;code&gt;PID 1&lt;/code&gt; của containter&lt;br&gt;
và sẽ &lt;em&gt;không nhận&lt;/em&gt; Unix signals - process này sẽ không nhận &lt;code&gt;SIGNTERM&lt;/code&gt; từ &lt;code&gt;docker stop &amp;lt;container&amp;gt;&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Chỉ có lệnh &lt;code&gt;ENTRYPOINT&lt;/code&gt; cuối cùng trong &lt;code&gt;Dockerfile&lt;/code&gt; là có tác dụng. &lt;/p&gt;

&lt;p&gt;Ví dụ &lt;em&gt;exec&lt;/em&gt; form của &lt;code&gt;ENTRYPOINT&lt;/code&gt;&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;FROM ubuntu
ENTRYPOINT &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"top"&lt;/span&gt;, &lt;span class="s2"&gt;"-b"&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt;
CMD &lt;span class="o"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;"-c"&lt;/span&gt;&lt;span class="o"&gt;]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Khi chạy container, &lt;code&gt;top&lt;/code&gt; là process duy nhất.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;docker run -it --rm --name test &amp;lt;image&amp;gt;&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;Liệt kê process&lt;/p&gt;

&lt;p&gt;&lt;code&gt;docker exec test ps aux&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;Process chạy sẽ là &lt;code&gt;top -b -c&lt;/code&gt;, nếu chạy với arguments&lt;/p&gt;

&lt;p&gt;&lt;code&gt;docker run -it --rm --name test &amp;lt;image&amp;gt; -H&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;Process chạy sẽ là &lt;code&gt;top -b -H&lt;/code&gt;&lt;/p&gt;

</description>
      <category>docker</category>
      <category>containers</category>
    </item>
  </channel>
</rss>
