写在前面
在这篇文章中,Deni Bertovic将向我们展示如何使用Docker来快速构建Haskell应用程序并生成Docker镜像。
备注: Haskell 是一种标准化的,通用的纯函数编程语言,有非限定性语义和强静态类型。作为一门函数编程语言,主要控制结构是函数。Haskell具有“证明即程序、命题为类型”的特征。(摘自维基百科)
接下来,我们将从两个案例入手,通过对比分析来帮助您理解。第一个案例,使用相同的Linux发行版(ubuntu:16.04)进行开发。第二个案例,使用不同的操作系统和发行版来进行开发。
关于Docker多阶段构建功能和“Stack Images Container”指令,会在该文章的最后有更多介绍。
案例一:在相同的操作系统或发行版上构建和部署
如果我们在相同的Linux发行版上构建Haskell应用程序(在本例中是一个Docker镜像),那么构建它的过程就会十分精简。
在这种情况下,我们就能够在本地构建我们的Haskell应用程序了。使用Stack,然后将编译后的二进制文件嵌入到Docker镜像中。
构建二进制文件和docker 镜像
build:
@stack build
@BINARY_PATH=${BINARY_PATH_RELATIVE} docker-compose build
正如我们所看到的,我们先在本地使用Stack来构建二进制文件,然后通过调用docker-compose和提供的Dockerfile构建出最终的docker镜像。
version: '2'
services:
myapp:
build:
context: .
args:
- BINARY_PATH
image: fpco/myapp
command: /opt/myapp/myapp复制代码
FROM ubuntu:16.04
RUN mkdir -p /opt/myapp/
ARG BINARY_PATH
WORKDIR /opt/myapp
RUN apt-get update && apt-get install -y \
ca-certificates \
libgmp-dev
COPY "$BINARY_PATH" /opt/myapp
COPY static /opt/myapp/static
COPY config /opt/myapp/config
CMD ["/opt/myapp/myapp"]复制代码
在Dockerfile中,我们只需要简单的将.stack-work/path/to/myapp复制到Dockerfile中的预定位置即可。
案例二:在不同的操作系统或发行版中构建
- 将所有构建时相关的文件打包到最终的Docker镜像中
- 将整个构建过程分解为Dockerfile.build(包含所有构建时相关的文件)和Dockerfile(包含运行时相关的文件和最终的二进制文件)两部分。这是一个相当繁琐的过程,需要一个shell脚本来辅助。首先构建第一个Docker镜像,然后利用这个docker镜像启动一个容器,获取已编译的二进制文件,之后再启动第二个容器,将这个已编译二进制文件嵌入其中,最后将其生成为新的docker镜像。您可以在Docker文档中了解到更多相关信息。
现在,Docker已经采用了一个名为多阶段构建的功能,它使整个构建过程变得简单化、自动化。它所带来的好处就是,我们不再使用构建工具和构建需求来扩大生产镜像,从而保证我们原本的镜像大小。
Docker多阶段构建
FROM fpco/stack-build:lts-9.9 as build
RUN mkdir /opt/build
COPY . /opt/build
RUN cd /opt/build && stack build --system-ghc
FROM ubuntu:16.04
RUN mkdir -p /opt/myapp
ARG BINARY_PATH
WORKDIR /opt/myapp
RUN apt-get update && apt-get install -y \
ca-certificates \
libgmp-dev
# NOTICE THIS LINE
COPY --from=build /opt/build/.stack-work/install/x86_64-linux/lts-9.9/8.0.2/bin .
COPY static /opt/myapp/static
COPY config /opt/myapp/config
CMD ["/opt/myapp/myapp"]
复制代码
使用“Stack Images Container”
首先让我们看看要将stack.yaml复制到stack-native.yaml所需的更改:
docker:Docker Carrying Haskell.jpg
enable: true
image:
container:
base: "fpco/myapp-base"
name: "fpco/myapp"
add:
static/: /opt/app/static
config/: /opt/app/config复制代码
我们需要将生成镜像的名称和本地目录添加到镜像中。Stack将自动为我们添加可执行文件。
让我们看一下我们的Dockerfile.base,我们将用它来构建我们的基础镜像。
FROM ubuntu:16:04
RUN mkdir -p /opt/app
WORKDIR /opt/app
RUN apt-get update && apt-get install -y \
ca-certificates \
libgmp-dev复制代码
正如我们所看到的,除了少了复制生成的二进制文件的部分,它与我们之前的Dockerfile没有什么不同,因为Stack将会为我们完成这一点。
我们可以使用make build-base来构建镜像。然后建立新的镜像来运行make build-stack-native。
构建用于stack image container
的基础镜像
build-base:
@docker build -t fpco/myapp-base -f Dockerfile.base .
复制代码
build-stack-native: build-base
@stack --stack-yaml stack-native.yaml build
@stack --stack-yaml stack-native.yaml image container
复制代码
现在来测试我们构建的镜像吧,让我们运行make run-stack-native试试看吧,如下所示:
**运行由stack image container
构建的容器
run-stack-native:
@docker run -p 3000:3000 -it -w /opt/app \
${IMAGE_NAME} myapp
复制代码
可以在Stack文档中阅读关于stack image container的更多信息。
总结
注:上面的示例代码可以在Github上找到,并且包含关于进程管理和权限处理的一些内容,为简洁起见,这些内容在本文中被省略了。