Skip to content

Commit ac78556

Browse files
authored
Building Docker Images for .NET Core Applications (dotnet#984)
* init draft * spelling * minor edit * addressed comments * added to TOC * addressed feedback - thank you. * minor edit * adding @SteveLasker edits * minor edits * edits per Bill * spelling fix * addressing @glennc comments * Last edits * spelling * minor edits * three three edit * rebuild * folder restructure
1 parent 300a030 commit ac78556

File tree

4 files changed

+215
-0
lines changed

4 files changed

+215
-0
lines changed
Lines changed: 214 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,214 @@
1+
---
2+
title: Building .NET Core Docker Images
3+
description: Understanding Docker images and .NET Core
4+
keywords: .NET, .NET Core, Docker
5+
author: spboyer
6+
manager: wpickett
7+
ms.date: 08/29/2016
8+
ms.topic: article
9+
ms.prod: .net-core
10+
ms.technology: .net-core-technologies
11+
ms.devlang: dotnet
12+
ms.assetid: 03c28597-7e73-46d6-a9c3-f9cb55642739
13+
---
14+
15+
16+
#Building Docker Images for .NET Core Applications
17+
18+
In order to get an understanding of how to use .NET Core and Docker together, we must first get to know the different Docker images that are offered and when is the right use case for them. Here we will walk through the variations offered, build an ASP.NET Core Web API, use the Yeoman Docker tools to create a debuggable container as well as peek at how Visual Studio Code can assist in the process.
19+
20+
## Docker Image Optimizations
21+
22+
When building Docker images for developers, we focused on three main scenarios:
23+
24+
- Images used to develop .NET Core apps
25+
- Images used to build .NET Core apps
26+
- Images used to run .NET Core apps
27+
28+
Why three images?
29+
When developing, building and running containerized applications, we have different priorities.
30+
- **Development:** How fast can you iterate changes, and the ability to debug the changes. The size of the image isn't as important, rather can you make changes to your code and see them quickly. Some of our tools, like [yo docker](https://aka.ms/yodocker) for use in VS Code use this image during development time.
31+
- **Build:** What's needed to compile your app. This includes the compiler and any other dependencies to optimize the binaries. This image isn't the image you deploy, rather it's an image you use to build the content you place into a production image. This image would be used in your continuous integration, or build environment. For instance, rather than installing all the dependencies directly on a build agent, the build agent would instance a build image to compile the application with all the dependencies required to build the app contained within the image. Your build agent only needs to know how to run this Docker image.
32+
- **Production:** How fast you can deploy and start your image. This image is small so it can quickly travel across the network from your Docker Registry to your Docker hosts. The contents are ready to run enabling the fastest time from Docker run to processing results. In the immutable Docker model, there's no need for dynamic compilation of code. The content you place in this image would be limited to the binaries and content needed to run the application. For example, the published output using `dotnet publish` which contains the compiled binaries, images, .js and .css files. Over time, you'll see images that contain pre-jitted packages.
33+
34+
## Docker image variations
35+
36+
To achieve the goals above, we provide image variants under [microsoft/dotnet](https://hub.docker.com/r/microsoft/dotnet/).
37+
38+
- `microsoft/dotnet:<version>-sdk` : that is **microsoft/dotnet-preview2-sdk**, this image contains the .NET Core SDK which includes the .NET Core and Command Line Tools (CLI). This image maps to the **development scenario**. You would use this image for local development, debugging and unit testing. For example, all the development you do, before you check in your code. This image can also be used for your **build** scenarios.
39+
40+
- `microsoft/dotnet:<version>-core` : that is **microsoft/dotnet:1.0.0-core**, image which runs [portable .NET Core applications](https://docs.microsoft.com/en-us/dotnet/articles/core/app-types) and it is optimized for running your application in **production**. It does not contain the SDK, and is meant to take the optimized output of `dotnet publish`. The portable runtime is well suited for Docker container scenarios as running multiple containers benefit from shared image layers.
41+
42+
## Alternative images
43+
44+
In addition to the optimized scenarios of development, build and production, we provide additional images:
45+
46+
- `microsoft/dotnet:<version>-onbuild` : that is **microsoft/dotnet:onbuild**, contains [ONBUILD](https://docs.docker.com/engine/reference/builder/#/onbuild) triggers. The build will [COPY](https://docs.docker.com/engine/reference/builder/#/copy) your application, run `dotnet restore` and create an [ENTRYPOINT](https://docs.docker.com/engine/reference/builder/#/entrypoint) `dotnet run` instruction to run the application when the Docker image is run. While not an optimized image for production, some may find it useful to simply copy their source code into an image and run it.
47+
48+
- `microsoft/dotnet:<version>-core-deps` : that is **microsoft/dotnet:1.0.0-core-deps**, if you wish to run self-contained applications use this image. It contains the operating system with all of the native dependencies needed by .NET Core. This image can also be used as a base image for your own custom CoreFX or CoreCLR builds. While the **onbuild** variant is optimized to simply place your code in an image and run it, this image is optimized to have only the operating system dependencies required to run .NET Core apps that have the .NET Runtime packaged with the application. This image isn't generally optimized for running multiple .NET Core containers on the same host, as each image carries the .NET Core runtime within the application, and you will not benefit from image layering.
49+
50+
Latest versions of each variant:
51+
52+
- `microsoft/dotnet` or `microsoft/dotnet:latest` (includes SDK)
53+
- `microsoft/dotnet:onbuild`
54+
- `microsoft/dotnet:core`
55+
- `microsoft/dotnet:core-deps`
56+
57+
Here is a list of the images after a `docker pull <imagename>` on a development machine to show the various sizes. Notice, the development/build variant, `microsoft/dotnet:1.0.0-preview2-sdk` is larger as it contains the SDK to develop and build your application. The production variant, `microsoft/dotnet:core` is smaller, as it only contains the .NET Core runtime.
58+
The minimal image capable of being used on Linux, `core-deps`, is quite smaller, however your application will need to copy a private copy of the .NET Runtime with it. Since containers are already private isolation barriers, you will lose that optimization when running multiple dotnet based containers.
59+
60+
```
61+
REPOSITORY TAG IMAGE ID SIZE
62+
microsoft/dotnet onbuild 19b6a6c4b1db 540.4 MB
63+
microsoft/dotnet 1.0.0-preview2-sdk a92c3d9ad0e7 540.4 MB
64+
microsoft/dotnet core 5224a9f2a2aa 253.2 MB
65+
microsoft/dotnet 1.0.0-core-deps c981a2eebe0e 166.2 MB
66+
microsoft/dotnet core-deps c981a2eebe0e 166.2 MB
67+
microsoft/dotnet latest 03c10abbd08a 540.4 MB
68+
microsoft/dotnet 1.0.0-core b8da4a1fd280 253.2 MB
69+
```
70+
71+
## Prerequisites
72+
73+
To build and run, you'll need a few things installed:
74+
75+
- [.NET Core](http://dot.net)
76+
- [Docker for Windows or Docker for Mac](http://www.docker.com/products/docker) from Docker to run your Docker containers locally
77+
- [Yeoman generator for ASP.NET](https://github.com/omnisharp/generator-aspnet) for creating the Web API application
78+
- [Yeoman generator for Docker](http://aka.ms/yodocker) from Microsoft
79+
80+
Install the Yeoman generators for ASP.NET Core and Docker using npm
81+
82+
```
83+
npm install -g yo generator-aspnet generator-docker
84+
```
85+
86+
> [!NOTE]
87+
> This sample will be using [Visual Studio Code](http://code.visualstudio.com) for the editor.
88+
89+
## Creating the Web API application
90+
91+
For a reference point, before we containerize the application, first run the application locally.
92+
93+
Create a directory for your application.
94+
95+
Open a command or terminal session in that directory and use the ASP.NET Yeoman generator by typing the following:
96+
```
97+
yo aspnet
98+
```
99+
100+
Select **Web API Application** and type **api** for the name of the app and tap enter. Once the application is scaffolded, change to the `/api` directory and restore the NuGet dependencies using `dotnet restore`.
101+
102+
```
103+
cd api
104+
dotnet restore
105+
```
106+
107+
Test the application using `dotnet run` and browsing to **http://localhost:5000/api/values**
108+
109+
```javascript
110+
[
111+
"value1",
112+
"value2"
113+
]
114+
```
115+
116+
Use `Ctrl+C` to stop the application.
117+
118+
## Adding Docker support
119+
120+
Adding Docker support to the project is achieved using the Yeoman generator from Microsoft. It currently supports .NET Core, Node.js and Go projects by creating a Dockerfile and scripts that help build and run projects inside containers. Visual Studio Code specific files are also added (launch.json, tasks.json) for editor debugging and command palette support.
121+
122+
```console
123+
$ yo docker
124+
125+
_-----_ ╭──────────────────────────╮
126+
| | │ Welcome to the Docker │
127+
|--(o)--| │ generator! │
128+
`---------´ │ Let's add Docker │
129+
( _´U`_ ) │ container magic to your │
130+
/___A___\ /│ app! │
131+
| ~ | ╰──────────────────────────╯
132+
__'.___.'__
133+
´ ` |° ´ Y `
134+
135+
? What language is your project using? (Use arrow keys)
136+
.NET Core
137+
Golang
138+
Node.js
139+
140+
```
141+
142+
- Select `.NET Core` as the project type
143+
- `rtm` for the version of .NET Core
144+
- `Y` the project uses a web server
145+
- `5000` is the port the Web API application is listening on (http://localhost:5000)
146+
- `api` for the image name
147+
- `api` for the service name
148+
- `api` for the compose project
149+
- `Y` to overwrite the current Dockerfile
150+
151+
When the generator is complete, the following files are added to the project
152+
153+
- .vscode/launch.json
154+
- Dockerfile.debug
155+
- Dockerfile
156+
- docker-compose.debug.yml
157+
- docker-compose.yml
158+
- dockerTask.ps1
159+
- dockerTask.sh
160+
- .vscode/tasks.json
161+
162+
The generator creates two Dockerfiles.
163+
164+
**Dockerfile.debug** - this file is based on the **microsoft/dotnet:1.0.0-preview2-sdk** image which if you note from the list of image variants, includes the SDK, CLI and .NET Core and will be the image used for development and debugging (F5). Including all of these components produces a larger image with a size roughly of 540MB.
165+
166+
**Dockerfile** - this image is the release image based on **microsoft/dotnet:1.0.0-core** and should be used for production. This image when built is approximately 253 MB.
167+
168+
### Creating the Docker images
169+
Using the `dockerTask.sh` or `dockerTask.ps1` script, we can build or compose the image and container for the **api** application for a specific environment. Build the **debug** image by running the following command.
170+
171+
```bash
172+
./dockerTask.sh build debug
173+
```
174+
175+
The image will build the ASP.NET application, run `dotnet restore`, add the debugger to the image, set an `ENTRYPOINT` and finally copy the app to the image. The result is a Docker image named *api* with a `TAG` of *debug*. See the images on the machine using `docker images`.
176+
177+
```bash
178+
docker images
179+
180+
REPOSITORY TAG IMAGE ID CREATED SIZE
181+
api debug 70e89fbc5dbe a few seconds ago 779.8 MB
182+
```
183+
184+
Another way to generate the image and run the application within the Docker container is to open the application in Visual Studio Code and use the debugging tools.
185+
186+
Select the debugging icon in the View Bar on the left side of VS Code.
187+
188+
![vscode debugging icon](./media/building-net-docker-images/debugging_debugicon.png)
189+
190+
Then tap the play icon or F5 to generate the image and start the application within the container. The Web API will be launched using your default web browser at http://localhost:5000.
191+
192+
![VSCode Docker Tools Debug](./media/building-net-docker-images/docker-tools-vscode-f5.png)
193+
194+
You may set break points in your application, step through, etc. just as if the application was running locally on your development machine as opposed to inside the container. The benefit to debugging within the container is this is the same image that would be deployed to a production environment.
195+
196+
Creating the release or production image requires simply running the command from the terminal passing the `release` environment name.
197+
198+
```bash
199+
./dockerTask build release
200+
```
201+
202+
The command creates the image based on the smaller **microsoft/dotnet:core** base image, [EXPOSE](https://docs.docker.com/engine/reference/builder/#/expose) port 5000, sets the [ENTRYPOINT](https://docs.docker.com/engine/reference/builder/#/entrypoint) for `dotnet api.dll` and copies it to the `/app` directory. There is no debugger, SDK or `dotnet restore` resulting in a much smaller image. The image is named **api** with a `TAG` of **latest**.
203+
204+
```
205+
REPOSITORY TAG IMAGE ID CREATED SIZE
206+
api debug 70e89fbc5dbe 1 hour ago 779.8 MB
207+
api latest ef17184c8de6 1 hour ago 260.7 MB
208+
```
209+
210+
## Summary
211+
212+
Using the Docker generator to add the necessary files to our Web API application made the process simple to create the development and production versions of the images. The tooling is cross platform by also providing a PowerShell script to accomplish the same results on Windows and Visual Studio Code integration providing step through debugging of the application within the container. By understanding the image variants and the target scenarios, you can optimize your inner-loop development process, while achieving optimized images for production deployments.
213+
214+
Loading
Loading

docs/toc.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -274,6 +274,7 @@
274274
### [Developing ASP.NET Core applications](core/tutorials/aspnet-core.md)
275275
### [How to Manage Package Dependency Versions for .NET Core 1.0](core/tutorials/managing-package-dependency-versions.md)
276276
### [Using MSBuild to build .NET Core projects](core/tutorials/target-dotnetcore-with-msbuild.md)
277+
### [Building Docker Images for .NET Core Applications](core/docker/building-net-docker-images.md)
277278
## [App Types](core/app-types.md)
278279
## [Packages, Metapackages and Frameworks](core/packages.md)
279280
## [Deploying](core/deploying/index.md)

0 commit comments

Comments
 (0)