Targets have the ability to produce output outside of the build environment. You can save files and docker images to your local machine or push them to remote repositories. Targets can also run commands that affect the local environment outside of the build, such as running database migrations, but not all targets produce output. Let's take a look at which commands produce output and how to use them.
Saving Files
We've already seen how the command SAVE ARTIFACT copies a file or directory from the build environment into the target's artifact environment.
This gives us the ability to copy files between targets, but it does not allow us to save any files to our local machine.
build:COPY main.go .RUN go build -o output/example main.go SAVE ARTIFACT output/exampledocker:# COPY command copies files from the +build targetCOPY +build/example .ENTRYPOINT ["/go-workdir/example"] SAVE IMAGE go-example:latest
In order to save the file locally , we need to add AS LOCAL to the command.
build:COPY main.go .RUN go build -o output/example main.go SAVE ARTIFACT output/example AS LOCAL local-output/go-example
If we run this example with earthly +build, we'll see a local-output directory show up locally with a go-example file inside of it.
Saving Docker Images
Saving Docker images to your local machine is easy with the SAVE IMAGE command.
build:COPY main.go .RUN go build -o output/example main.go SAVE ARTIFACT output/exampledocker:COPY +build/example .ENTRYPOINT ["/go-workdir/example"] SAVE IMAGE go-example:latest
In this example, running earthly +docker will save an image named go-example with the tag latest.
If we run a target as a reference in FROM or COPY, outputs will not be produced. Take this Earthfile for example.
build:COPY main.go .RUN go build -o output/example main.go SAVE ARTIFACT output/example AS LOCAL local-output/go-exampledocker:COPY +build/example .ENTRYPOINT ["/go-workdir/example"] SAVE IMAGE go-example:latest
In this case, running earthly +docker will not produce any output. In other words, you will not have a local-output/go-example written locally, but running earthly +build will still produce output as expected.
The exception to this rule is the BUILD command. If you want to use COPY or FROM and still have Earthly create local-output/go-example locally, you'll need to use the BUILD command to do so.
build:COPY main.go .RUN go build -o output/example main.go SAVE ARTIFACT output/example AS LOCAL local-output/go-exampledocker: BUILD +buildCOPY +build/example .ENTRYPOINT ["/go-workdir/example"] SAVE IMAGE go-example:latest
Running earthly +docker in this case will now output local-output/go-example locally.
The Push Flag
Docker Images
In addition to saving files and images locally, we can also push them to remote repositories.
docker:COPY +build/example .ENTRYPOINT ["/go-workdir/example"] SAVE IMAGE --push go-example:latest
Note that adding the --push flag to SAVE IMAGE is not enough, we'll also need to invoke push when we run earthly. earthly --push +docker.
External Changes
You can also use --push as part of a RUN command to define commands that have an effect external to the build. These kinds of effects are only allowed to take place if the entire build succeeds.
Just like saving files, any command that uses --pushwill only produce output if called directly, earthly --push +target-with-pushor via aBUILD command. Calling a target via FROM or COPY will not invoke --push.
VERSION 0.8FROM node:13.10.1-alpine3.11WORKDIR /js-examplebuild:# In JS, there's nothing to build in this simple form.# The source is also the artifact used in production.COPY src/index.js . SAVE ARTIFACT index.js /dist/index.js AS LOCAL ./dist/index.jsdocker:COPY +build/dist distENTRYPOINT ["node", "./dist/index.js"] SAVE IMAGE js-example:latest
VERSION 0.8FROM openjdk:8-jdk-alpineRUN apk add --update --no-cache gradleWORKDIR /java-examplebuild:COPY build.gradle ./COPY src srcRUN gradle buildRUN gradle install SAVE ARTIFACT build/install/java-example/bin /bin AS LOCAL build/bin SAVE ARTIFACT build/install/java-example/lib /lib AS LOCAL build/libdocker:COPY +build/bin binCOPY +build/lib libENTRYPOINT ["/java-example/bin/java-example"] SAVE IMAGE java-example:latest
VERSION 0.8FROM python:3WORKDIR /codebuild:# In Python, there's nothing to build.COPY src src SAVE ARTIFACT src /srcdocker:COPY +build/src srcENTRYPOINT ["python3", "./src/hello.py"] SAVE IMAGE --push python-example:latest