Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/s0/git-publish-subdir-action

GitHub Action to push a subdirectory as a branch to any git repo (e.g. for GitHub Pages)
https://github.com/s0/git-publish-subdir-action

actions git github github-actions github-pages ssh

Last synced: about 3 hours ago
JSON representation

GitHub Action to push a subdirectory as a branch to any git repo (e.g. for GitHub Pages)

Awesome Lists containing this project

README

        

# GitHub Action: Push git subdirectory as branch

[![](https://github.com/s0/git-publish-subdir-action/workflows/Scheduled%20tests/badge.svg)](https://github.com/s0/git-publish-subdir-action/actions?workflow=Scheduled+tests) [![codecov](https://codecov.io/gh/s0/git-publish-subdir-action/branch/master/graph/badge.svg)](https://codecov.io/gh/s0/git-publish-subdir-action) [![](https://raw.githubusercontent.com/s0/git-publish-subdir-action/gh-badges/drift.svg)](https://github.com/s0/libyear-node-action) [![](https://raw.githubusercontent.com/s0/git-publish-subdir-action/gh-badges/releases.svg)](https://github.com/s0/libyear-node-action)

This GitHub Action will take any subdirectory in your repository, and push it as the contents of a git branch to a repository and branch of your choosing, either over SSH or to the current repo.

You could use this for example to:

* Publishing a subdirectory to a repo's `gh-pages` branch, after optionally running a build step.
* Publishing build artifacts / binaries to another repository

The target repository can be anywhere accessible by a [Git SSH URL](https://git-scm.com/book/en/v2/Git-on-the-Server-The-Protocols#_the_ssh_protocol) (or the current repository).
If the target branch doesn't exist yet, it will be created automatically.

## Usage

Simply include the action `s0/git-publish-subdir-action@develop` in the appropriate point in your workflow, and pass in the required configuration options:

```yml
jobs:
deploy:
name: Deploy
runs-on: ubuntu-latest
steps:

# Any prerequisite steps
- uses: actions/checkout@master

# Deploy to local repo
- name: Deploy
uses: s0/git-publish-subdir-action@develop
env:
REPO: self
BRANCH: gh-pages
FOLDER: build
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

# Deploy to another repo
- name: Deploy
uses: s0/git-publish-subdir-action@develop
env:
REPO: [email protected]:owner/repo.git
BRANCH: gh-pages
FOLDER: build
SSH_PRIVATE_KEY: ${{ secrets.DEPLOY_PRIVATE_KEY }}
KNOWN_HOSTS_FILE: resources/known_hosts # Needed if target repo is not on github.com
```

## Examples

### When pushed to master, push `/public/site` to the `gh-pages` branch on the same repo

```yml
name: Deploy to GitHub Pages
on:
push:
branches:
- master

jobs:
deploy:
name: Deploy to GitHub Pages
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@master

- name: Deploy
uses: s0/git-publish-subdir-action@develop
env:
REPO: self
BRANCH: gh-pages
FOLDER: public/site
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
```

### When pushed to master, push the contents of `/public/site` to the `www` folder on the `gh-pages` branch on the same repo

```yml
name: Deploy to GitHub Pages
on:
push:
branches:
- master

jobs:
deploy:
name: Deploy to GitHub Pages
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@master

- name: Deploy
uses: s0/git-publish-subdir-action@develop
env:
REPO: self
BRANCH: gh-pages
FOLDER: public/site
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
TARGET_DIR: www
```

### When pushed to master, run a build step, then push `/build` to the `gh-pages` branch on another repo on GitHub

```yml
name: Deploy to GitHub Pages
on:
push:
branches:
- master

jobs:
deploy:
name: Deploy to GitHub Pages
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@master
- name: Use Node.js
uses: actions/setup-node@main
with:
node-version: 20.x
- name: npm install and build
run: |
npm install
npm run build

- name: Deploy
uses: s0/git-publish-subdir-action@develop
env:
REPO: [email protected]:owner/repo.git
BRANCH: gh-pages
FOLDER: build
SSH_PRIVATE_KEY: ${{ secrets.DEPLOY_PRIVATE_KEY }}
```

Note: the SSH Key needs to have write access to the given repo. It's recommended you use [Deploy Keys](https://developer.github.com/v3/guides/managing-deploy-keys/#deploy-keys) for this (see below),
and store the SSH private key as as Secret in the repository settings.

### When pushed to master, run a build step, then push `/dist` to the `artifacts` branch on a repo hosted at mydomain.com

```yml
name: Deploy to GitHub Pages
on:
push:
branches:
- master

jobs:
deploy:
name: Deploy to GitHub Pages
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@master
- name: Use Node.js
uses: actions/setup-node@main
with:
node-version: 20.x
- name: npm install and build
run: |
npm install
npm run build

- name: Deploy
uses: s0/git-publish-subdir-action@develop
env:
REPO: [email protected]:path/to/repo.git
BRANCH: artifacts
FOLDER: dist
SSH_PRIVATE_KEY: ${{ secrets.DEPLOY_PRIVATE_KEY }}
KNOWN_HOSTS_FILE: resources/known_hosts # Path relative to the root of the repository
```

You can generate a `known_hosts` file for a given domain by using `ssh-keyscan`, e.g:

```bash
> ssh-keyscan github.com
# github.com:22 SSH-2.0-babeld-f345ed5d
github.com ssh-rsa AAAAB3NzaC1yc2EAAAABIwAAAQEAq2A7hRGmdnm9tUDbO9IDSwBK6TbQa+PXYPCPy6rbTrTtw7PHkccKrpp0yVhp5HdEIcKr6pLlVDBfOLX9QUsyCOV0wzfjIJNlGEYsdlLJizHhbn2mUjvSAHQqZETYP81eFzLQNnPHt4EVVUh7VfDESU84KezmD5QlWpXLmvU31/yMf+Se8xhHTvKSCZIFImWwoG6mbUoWf9nzpIoaSjB+weqqUUmpaaasXVal72J+UX2B+2RPW3RcT0eOzQgqlJL3RKrTJvdsjE3JEAvGq3lGHSZXy28G3skua2SmVi/w4yCE6gbODqnTWlg7+wC604ydGXA8VJiS5ap43JXiUFFAaQ==
# github.com:22 SSH-2.0-babeld-f345ed5d
# github.com:22 SSH-2.0-babeld-f345ed5d
```

## Configuration

All configuration options are passed in via `env`, as environment variables.

| Env Variable | Description | Required? |
| ------------------ | ------------------------------------------------------ | ------------- |
| `REPO` | Either `self`, or an SSH url to the target repository. | Yes |
| `BRANCH` | The target branch to publish to. | Yes |
| `FOLDER` | The target subfolder you would like to publish | Yes |
| `SSH_PRIVATE_KEY` | The private key that should be used to authenticate on SSH. Don't include this directly in the workflow file, instead you must use [Secrets](https://help.github.com/en/articles/virtual-environments-for-github-actions#creating-and-using-secrets-encrypted-variables) | When `REPO` is an SSH URL |
| `KNOWN_HOSTS_FILE` | Path to a file in the repository that contains the known SSH fingerprints for the target host. | When the target host is not github.com |
| `GITHUB_TOKEN` | Should always be equal to `${{ secrets.GITHUB_TOKEN }}` | When `REPO = self` |
| `SQUASH_HISTORY` | If set to `true`, all previous commits on the target branch will be discarded. For example, if you are deploying a static site with lots of binary artifacts, this can help the repository becoming overly bloated. | No |
| `SKIP_EMPTY_COMMITS` | If set to `true`, commits will only be pushed if the contents of the target branch will be changed as a result. This is useful if, for example, you'd like to easily track which upstream changes result in changes to your target branch. | No |
| `MESSAGE` | A custom template to use as the commit message pushed to the target branch. See [custom commit messages](#custom-commit-messages). | No |
| `TAG` | A string following the [git-check-ref-format](https://git-scm.com/docs/git-check-ref-format) that tags the commit with a lightweight git-tag. | No |
| `CLEAR_GLOBS_FILE` | An optional path to a file to use as a list of globs defining which files to delete when clearing the target branch. | No |
| `COMMIT_NAME` | The username the autogenerated commit will use. If unset, uses the commit pusher's username. | No |
| `COMMIT_EMAIL` | The email the autogenerated commit will use. If unset, uses the commit pusher's email. | No |
| `TARGET_DIR` | An optional string to change the directory where the files are copied to. | No |

### Custom commit messages

You can specify a custom string to use in the commit message
when pushing to your target repository.
These strings support a number of placeholders that will be replaces with
relevant values:

| Placeholder | Description |
| ------------------ | ----------------------------------------------------- |
| `{target-branch}` | The name of the target branch being updated |
| `{sha}` | The 7-character sha of the HEAD of the current branch |
| `{long-sha}` | The full sha of the HEAD of the current branch |
| `{msg}` | The commit message for the HEAD of the current branch |

Example Usage:

```yml
jobs:
deploy:
- uses: s0/git-publish-subdir-action@develop
env:
# ...
MESSAGE: "This updates the content to the commit {sha} that had the message:\n{msg}"
```

### Custom clear operations

By default, this action will clear the target branch of any pre-existing files,
and only keep those that are defined in the target `FOLDER` when the action was
run.

This can now be overwritten by specifying a file with a custom list of globs to
define which files should be deleted from the target branch before copying the
new files over.

The environment variable `CLEAR_GLOBS_FILE` should point to the path of the
glob file (which can have any name) relative to root of the target repository.

**Note: using this feature will disable the default functionality of deleting
everything, and you will need to specify exactly what needs to be deleted.**

#### Examples

1. Default behaviour:

```
**/*
!.git
```

1. Default behaviour with a custom target directory:

```
target_dir/**/*
!.git
```

1. Delete everything except the `.git` and `foobar` folder:

```
**/*
!.git
!foobar/**/*
```

1. Only delete the folder `folder` (except `folder/a`), and also delete anything
matching `ini*al2`:

```
folder/*
!folder/a
ini*al2
```

For clarity, if we have the file `.clear-target-files`:

```
folder/*
!folder/a
ini*al2
```

And the workflow file `.github/workflows/ci.yml`:

```yml
jobs:
deploy:
- uses: s0/git-publish-subdir-action@develop
env:
# ...
CLEAR_GLOBS_FILE: ".clear-target-files"
```

And the target branch already had the files:

```
initial1
initial2
folder/a
folder/b
```

Then the files that would remain would be:

```
folder/a
initial1
```

An empty file can be used to indicate that the branch should not be cleared at
all.

## Usage with [Deploy Keys](https://developer.github.com/v3/guides/managing-deploy-keys/#deploy-keys)

When pushing to other repositories on GitHub or GitHub Enterprise,
the recommended mechanism is to use [Deploy Keys](https://developer.github.com/v3/guides/managing-deploy-keys/#deploy-keys) rather than your own private SSH key.
Deploy keys are SSH keys that can be added to specific repositories to be given write access to only
those repositories.
This is preferable to adding your own personal ssh private key to a repository's secrets store,
as it means that any actions that have access to this repository's secrets can only push
to repositories that have explicitly had the SSH key added as a deploy key,
and not *all* repositories that your user account have access to.

Use `ssh-keygen` to create a new ssh key, add these to GitHub in the relevant repo's deploy keys and Secrets, then delete them off your computer.

```
> cd /tmp
> ssh-keygen -t ed25519
Generating public/private ed25519 key pair.
Enter file in which to save the key (/home/sam/.ssh/id_ed25519): temp-deploy-key
Enter passphrase (empty for no passphrase):
Enter same passphrase again:
Your identification has been saved in temp-deploy-key.
Your public key has been saved in temp-deploy-key.pub.
The key fingerprint is:
SHA256:tQBSeWjZ4Er4YYjK4XQ4npfiK2xJPJLbGjTsYJq/9JI sam@optimus
The key's randomart image is:
+--[ED25519 256]--+
| ..+* |
| ..o o=.o |
|.=o.+.... . |
|B =+.o o . |
|+% oo S . |
|X=+ |
|B=+. |
|oBE. |
|+ooo. |
+----[SHA256]-----+
> cat temp-deploy-key.pub
ssh-ed25519 XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX user@localhost
> cat temp-deploy-key
-----BEGIN OPENSSH PRIVATE KEY-----
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX=
-----END OPENSSH PRIVATE KEY-----
```