diff --git a/blog/_posts/2023-10-05-github-action-artifacts.md b/blog/_posts/2023-10-05-github-action-artifacts.md new file mode 100644 index 000000000..de9529b76 --- /dev/null +++ b/blog/_posts/2023-10-05-github-action-artifacts.md @@ -0,0 +1,276 @@ +--- +title: "Uploading and Downloading Artifacts with GitHub Actions" +categories: + - Tutorials +toc: true +author: Kumar Harsh + +internal-links: + - downloading artifacts + - uploading artifacts + - downloading artifacts with github actions + - uploading artifacts with github actions +--- + +If you've worked with projects hosted on GitHub, you're probably aware of [GitHub Actions](https://github.com/features/actions). GitHub Actions is GitHub's recommended way of setting up build automation and workflows from your GitHub-hosted repo. It supports multiple triggers and operations, including uploading and downloading artifacts. + +Artifacts represent the final or intermediate products of these build pipelines. In a build pipeline, the artifacts typically consist of executable binaries created at its end. Conversely, a test pipeline might generate artifacts in the form of test logs and results as files. These artifacts can then be used to update statuses across pull requests or commits. Some more use cases where uploading/downloading artifacts might come in handy include the following: + +* **Test results and coverage reports:** to review test outcomes, track code coverage trends, and share test reports with your team. +* **Deployment packages:** to store the deployable assets and facilitate subsequent deployment steps or manual releases. +* **Logs and diagnostics:** for troubleshooting and investigating issues that occur during the workflow runs. +* **Generated documentation or reports:** to make these accessible for review, dissemination, or archiving. +* **Storage of intermediate results:** to allow subsequent steps or workflows to access and use the data. +* **Sharing of artifacts across workflows or jobs.** + +In this article, you'll learn how to use artifacts with GitHub Actions to capture data from workflows and how to upload and download them as necessary using the [`actions/upload-artifact@v3`](https://github.com/actions/upload-artifact) and [`actions/download-artifact@v3`](https://github.com/actions/download-artifact) actions. + +## How to Use Artifacts with GitHub Actions + +In this tutorial, you'll learn how to use the upload and download artifacts actions to upload build results from one job and download them in another. This creates a release for every tag created in a GitHub repo. + +To follow along, all you need is a GitHub account. Because you'll write the workflow in the GitHub web app, you don't need any local development setup. However, you do need [git](https://git-scm.com/) installed on your local system so that you can clone the demo repo you will create in the tutorial and create tags in it. + +### How to Use Artifacts to Capture Data from Workflows + +Start by uploading artifacts from GitHub Actions workflows. You need to [fork this repo](https://github.com/krharsh17/gh-actions-upload-download/) to your GitHub account. This repo contains a [Vite](https://vitejs.dev/)-based [Vue](https://vuejs.org/) app that you can use. You'll generate a production build of this app using the workflow you create in this section. The static build output files are packaged and uploaded to GitHub for every workflow run using the `actions/upload-artifact@v3` action. + +#### Set Up the Workflow + +![Workflow]({{site.images}}{{page.slug}}/workflow.png)\ + +To get started, click on **Add file > Create new file** on the home page of the forked repo: + +
+![Adding a new file to the repo]({{site.images}}{{page.slug}}/UsmtQbk.png) +
+ +On the new file page that opens, set the file's location and name as `.github/workflows/node.js.yaml` and add the following code to it: + +~~~{.yaml caption="node.js.yaml"} +name: Node.js CI + +on: + push: + branches: [ "main" ] + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + - name: Use Node.js + uses: actions/setup-node@v3 + - run: npm ci + - run: npm run build --if-present +~~~ + +This sets up a basic workflow that uses the [`actions/checkout@v3`](https://github.com/actions/checkout) action to check out code from your repo and the [`actions/setup-node@v3`](https://github.com/actions/setup-node) action to set up a [Node.js](https://nodejs.org/en/) build environment. It then runs the `npm ci` command to clean and install the project's dependencies and the `npm run build` command to run the build script from the project (which is essentially just `vite build` behind the scenes). + +#### Identify the Data To Capture + +Before you can upload artifacts, you need to identify and locate them in your workflow's runtime. In this project, once the build command finishes executing, the resulting static web app is saved in a directory called `dist`. This is because the `vite build` command inherently directs its output to the dist folder as its default behavior. + +As part of the build artifacts, you need to upload everything stored in the `dist` folder after the build runs to your workflow runs page. + +#### Define the Artifacts in Your Workflow + +To upload the output generated in the `dist` folder, add the following step to your workflow file: + +~~~{.yaml caption="node.js.yaml"} +- uses: actions/upload-artifact@v3 + with: + name: Build + path: dist +~~~ + +The `actions/upload-artifact@v3` action takes in the name of the artifact to be uploaded and the path where you need to upload it as input to the action. If you don't provide a name via the input argument, the uploaded artifact is named `artifact` by default. The path argument is mandatory and can use wildcards to filter files inside directories (such as `dist/**/*.html` to upload all HTML files from the dist directory). + +Other input arguments that you can pass to this action are `if-no-files-found` and `retention-days`. The `if-no-files-found` argument lets you define the behavior in the case of the `path` argument not pointing to any files or directories that can be uploaded as artifacts. In other words, if the `path` you've defined is empty or does not exist, the action follows the behavior you set in the `if-no-files-found` argument. You can set it as `warn` for printing a warning as output (which is also the default behavior), `error` for marking the workflow run as failed and showing the error description in the run details, or `ignore` for ignoring and not showing any warnings or errors. `retention-days` lets you control how many days the artifacts remain available (between one to ninety days). + +The upload action can be included multiple times in the same job or workflow, and it accumulates all uploaded files in the same run's details page. If you choose to upload the same file twice, it retains the latest copy of the file. + +Here's what your workflow file should look like at this point: + +~~~{.yaml caption="node.js.yaml"} +name: Node.js CI + +on: + push: + branches: [ "main" ] + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + - name: Use Node.js + uses: actions/setup-node@v3 + - run: npm ci + - run: npm run build --if-present + - uses: actions/upload-artifact@v3 + with: + name: Build + path: dist +~~~ + +Commit this file to your repo and wait for the workflow run to be completed. + +#### How to Access the Uploaded Artifacts After the Workflow Runs + +This is what the GitHub Actions runs page will look like when it completes: + +
+![Workflow runs page]({{site.images}}{{page.slug}}/Df5GvyR.png) +
+ +Click on the run to view its details: + +
+![Workflow run details]({{site.images}}{{page.slug}}/526Vt3o.png) +
+ +You can view the uploaded artifact at the bottom of the page, under the **Artifacts** section. + +### How to Upload and Download Artifacts to Transfer Data Between Runs + +![Upload and Download]({{site.images}}{{page.slug}}/up & down.png)\ + +Now that you know how to upload artifacts from workflows, the next step is to learn how to download them in the same workflow when needed. In this section, you'll create a new job that waits for the build job to complete, downloads the build artifacts, and uploads them to a new GitHub release. You'll also update the workflow to be triggered only when pushing new tags so that releases can be created successfully. + +#### Update the Workflow + +To make the workflow run only on pushing new tags, update it to the following: + +~~~{.yaml caption="node.js.yaml"} +name: Node.js CI + +on: + push: + tags: # Remove the `branches` node and add this one + - '*' + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + - name: Use Node.js + uses: actions/setup-node@v3 + - run: npm ci + - run: npm run build --if-present + - uses: actions/upload-artifact@v3 + with: + name: Build + path: dist +~~~ + +Now, you're ready to add the next job that downloads the artifact and creates a release with it. + +#### Download Artifacts Using the actions/download-artifact Action + +Create a new job in your workflow with the name `release`. Here's what the config for the release job looks like: + +~~~{.yaml caption="node.js.yaml"} + release: + runs-on: ubuntu-latest + permissions: + contents: write + + needs: build + + steps: + - uses: actions/download-artifact@v3 + with: + name: Build + path: build + - run: zip -r build.zip build/ + - uses: ncipollo/release-action@v1 + with: + artifacts: "build.zip" +~~~ + +This job makes use of the `needs` node to define a dependency on the `build` job, which means that it will not start running until the `build` job completes running. This lets the artifacts upload successfully before the job tries to download them. + +It also defines a `permissions` node and requires the `contents: write` permission. This allows the `ncipollo/release-action@v1` action to create a release on your GitHub repository successfully. + +The job starts by using the `actions/download-artifact@v3` action to download the artifact named `Build` (specified by the `name` input argument) at the location `build` (specified by the `path` input argument). + +> Please note that the `path` here does not refer to the path from which the files need to be downloaded. Rather, the `path` stores the destination location, where the downloaded files need to be saved. + +Neither of those arguments is mandatory. Omitting the `name` argument means that the action downloads all artifacts that are available for download. Omitting the `path` argument means that the action downloads the artifacts to the current working directory of the actions runtime. + +Once the artifact named `Build` is downloaded at the location `build/`, the next step packages it into a ZIP file using the `zip` command. Finally, the [`ncipollo/release-action@v1`](https://github.com/ncipollo/release-action) action is used to upload the `build.zip` file to a new release created and associated with the tag that triggered this workflow run. + +#### Run the Workflow + +Commit this config file to your repo. Since you changed the trigger for the workflow, it will not run on this push. To trigger this workflow, you need to manually create a tag locally and push it to GitHub. + +To do that, clone your repo by running the following command (after replacing `` with your GitHub username): + +~~~{.bash caption=">_"} +git clone https://github.com//gh-actions-upload-download.git +~~~ + +Inside the cloned repo, run the following command to create a tag: + +~~~{.bash caption=">_"} +git tag -a v0.0.1 -m "Version 0.0.1" +~~~ + +Finally, run this command to push the tag: + +~~~{.bash caption=">_"} +git push origin v0.0.1 +~~~ + +This creates a new tag on your forked repo. Go to the **Actions** tab to see the newly triggered workflow run: + +
+![Newly triggered workflow run]({{site.images}}{{page.slug}}/6hYF97b.png) +
+ +#### How to Access the Artifacts + +You can look into the logs of the two jobs to understand how the upload and download worked. Here are the upload logs from the build job: + +
+![Upload logs]({{site.images}}{{page.slug}}/THlckxH.png) +
+ +Similar to the previous run, the `upload-artifacts` job uploads the artifacts to the `build/` folder. + +Here are the download logs from the release job: + +
+![Download logs]({{site.images}}{{page.slug}}/fDoAlVh.png) +
+ +The `download-artifacts` job first looks for the artifact to download using its name (*ie* `Build`) and sets up the directory structure for it (*ie* creates the destination path `build/` where the downloaded files are stored), counts the number of files to download, and then downloads them. You can further explore the logs of the other steps to see how the files were packaged and added to the new release. + +You'll also notice that the uploaded files from the build job are available as artifacts on the workflow run page: + +
+![Workflow run artifacts]({{site.images}}{{page.slug}}/gALcTcP.png) +
+ +However, the `build.zip` file created in the release job is not available because it was not uploaded using the `upload-artifacts` action. + +Navigate to the **Releases** page to see the newly created release with the `build.zip` file listed in its assets: + +
+![Release details]({{site.images}}{{page.slug}}/6VKJXqO.png) +
+ +This demonstrates that the artifacts were successfully downloaded from the GitHub runtime and uploaded to GitHub releases. The workflow created in this tutorial can be used as a simple workflow for creating releases on your Node.js-based projects. You can learn more about the upload and download actions from the GitHub documentation ([upload-artifacts](https://github.com/actions/upload-artifact#readme) and [download-artifacts](https://github.com/actions/download-artifact#readme)). + +## Conclusion + +GitHub Actions is a popular solution for build pipelines. Actions such as `upload-artifacts` and `download-artifacts` allow you to easily upload and download build artifacts and other files in between workflows. Additionally, they allow you to collect artifacts as workflow run results as well as use artifacts across multiple jobs in a workflow. + +With these new tricks up your sleeve, you're officially an expert at using GitHub Actions to share and grab those artifacts from your workflows! + +{% include_html cta/bottom-cta.html %} diff --git a/blog/assets/images/github-action-artifacts/526Vt3o.png b/blog/assets/images/github-action-artifacts/526Vt3o.png new file mode 100644 index 000000000..bc4eaf599 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/526Vt3o.png differ diff --git a/blog/assets/images/github-action-artifacts/6VKJXqO.png b/blog/assets/images/github-action-artifacts/6VKJXqO.png new file mode 100644 index 000000000..a1a6e1eed Binary files /dev/null and b/blog/assets/images/github-action-artifacts/6VKJXqO.png differ diff --git a/blog/assets/images/github-action-artifacts/6hYF97b.png b/blog/assets/images/github-action-artifacts/6hYF97b.png new file mode 100644 index 000000000..980b1ca8f Binary files /dev/null and b/blog/assets/images/github-action-artifacts/6hYF97b.png differ diff --git a/blog/assets/images/github-action-artifacts/Df5GvyR.png b/blog/assets/images/github-action-artifacts/Df5GvyR.png new file mode 100644 index 000000000..ff6198ae9 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/Df5GvyR.png differ diff --git a/blog/assets/images/github-action-artifacts/THlckxH.png b/blog/assets/images/github-action-artifacts/THlckxH.png new file mode 100644 index 000000000..c7e24071e Binary files /dev/null and b/blog/assets/images/github-action-artifacts/THlckxH.png differ diff --git a/blog/assets/images/github-action-artifacts/UsmtQbk.png b/blog/assets/images/github-action-artifacts/UsmtQbk.png new file mode 100644 index 000000000..ddd2a7fa3 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/UsmtQbk.png differ diff --git a/blog/assets/images/github-action-artifacts/fDoAlVh.png b/blog/assets/images/github-action-artifacts/fDoAlVh.png new file mode 100644 index 000000000..3e7d4178f Binary files /dev/null and b/blog/assets/images/github-action-artifacts/fDoAlVh.png differ diff --git a/blog/assets/images/github-action-artifacts/gALcTcP.png b/blog/assets/images/github-action-artifacts/gALcTcP.png new file mode 100644 index 000000000..7b6fb6c15 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/gALcTcP.png differ diff --git a/blog/assets/images/github-action-artifacts/header.jpg b/blog/assets/images/github-action-artifacts/header.jpg new file mode 100644 index 000000000..1eeabb455 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/header.jpg differ diff --git a/blog/assets/images/github-action-artifacts/up & down.png b/blog/assets/images/github-action-artifacts/up & down.png new file mode 100644 index 000000000..d86001899 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/up & down.png differ diff --git a/blog/assets/images/github-action-artifacts/workflow.png b/blog/assets/images/github-action-artifacts/workflow.png new file mode 100644 index 000000000..623f71c97 Binary files /dev/null and b/blog/assets/images/github-action-artifacts/workflow.png differ