Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/bvarnai/respository-installer
A command-line, configuration based tool to install git repositories
https://github.com/bvarnai/respository-installer
bash-script command-line curl devops git jq shell-script tooling
Last synced: 19 days ago
JSON representation
A command-line, configuration based tool to install git repositories
- Host: GitHub
- URL: https://github.com/bvarnai/respository-installer
- Owner: bvarnai
- License: mit
- Created: 2024-03-18T05:30:26.000Z (10 months ago)
- Default Branch: main
- Last Pushed: 2024-04-26T09:59:29.000Z (8 months ago)
- Last Synced: 2024-10-29T22:47:47.175Z (2 months ago)
- Topics: bash-script, command-line, curl, devops, git, jq, shell-script, tooling
- Language: Shell
- Homepage:
- Size: 6.33 MB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# installer - a configuration based repository installer
![CI badge](https://github.com/bvarnai/respository-installer/actions/workflows/ci.yml/badge.svg)
**installer** is a tool to help users to work with multiple *Git* repositories from the initial clone to getting updates.
**Where does it fit?**
I worked in a Java development team, we had about 15 repositories. I needed a simple tool which is
- Self-contained and updateable
- Configuration based
- Supports development `streams` (for example parallel tooling for java17, java21 etc.)
- *Git* only (minimal platform specific code)and **nothing** more.
:bulb: I use the word *project* interchangeably with *repository*
## Table of contents
- [installer - a configuration based repository installer](#installer---a-configuration-based-repository-installer)
- [Table of contents](#table-of-contents)
- [Demo](#demo)
- [Installation](#installation)
- [Supported SCM types](#supported-scm-types)
- [GitHub](#github)
- [Token configuration](#token-configuration)
- [Bitbucket Enterprise](#bitbucket-enterprise)
- [Token configuration](#token-configuration-1)
- [Plain HTTP](#plain-http)
- [Getting started for the first time](#getting-started-for-the-first-time)
- [Git credentials](#git-credentials)
- [Prerequisites](#prerequisites)
- [Configuration](#configuration)
- [Workspace](#workspace)
- [Configuration file](#configuration-file)
- [Usage](#usage)
- [Options](#options)
- [Options for development/testing](#options-for-developmenttesting)
- [Link mode](#link-mode)
- [Stream explained](#stream-explained)
- [Custom environments](#custom-environments)
- [Dependencies](#dependencies)
- [Link/unlink](#linkunlink)
- [Commands](#commands)
- [help](#help)
- [list](#list)
- [install](#install)
- [update](#update)
- [FAQ](#faq)
- [Development notes](#development-notes)---
## Demo
![installer demo](docs/demo.gif)
---
## Installation
For easier understanding of the initial steps, the following diagram provides an overview
![Overview](docs/overview.png)
To get started, you will need the following
- Configuration file aka `projects.json` (see [Configuration file](#configuration-file))
And these environment variables pointing to your configuration
- `INSTALLER_CONFIG_URL` - URL of the configuration `projects.json`
- `INSTALLER_CONFIG_SCM` - type of SCM (GitHub etc.) used for the configurationIf you are using GitHub, only `INSTALLER_CONFIG_URL` is needed.
:memo: For the best user experience, I recommend forking **installer** and set defaults according to your environment
### Supported SCM types
**installer** needs to know how to get the configuration file form the SCM without actually cloning it. This means assembling a URL used by `curl` to get the configuration. The following SCM types are supported:
- github - GitHub *[default]*
- bitbucket_server - Bitbucket Enterprise (server/data center)
- plain - Plain HTTP:bulb: This is only used for configuration file discovery, you can use any *Git* platform later for your projects. Authentication for *Git* commands are based on your *Git* configuration.
:warning: Bitbucket Cloud is not yet supported
#### GitHub
To get the configuration file the following URL format is used:
```
https://#token#@raw.githubusercontent.com///#branch#//
```
The following variables are used in the URL:- `#token#` is replaced with `INSTALLER_CONFIG_TOKEN` env. variable which holds your *Personal access token* or PAT
- `#branch#` is replaced with the current branch (this done automatically)##### Token configuration
To create token or PAT follow the official guide [Creating a personal access token (classic)](https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/managing-your-personal-access-tokens#creating-a-personal-access-token-classic)
Make sure you set `repo` scope (and nothing more) when creating the PAT.
![github-pat](docs/github-pat.png).
:bulb: Token is needed for private repositories only
For example, using your private repositories would need the following settings:
```bash
export INSTALLER_CONFIG_URL=https://#token#@raw.githubusercontent.com/user/repo/#branch#/projects.json
export INSTALLER_CONFIG_TOKEN=1bacnotmyrealtoken123beefbea
```#### Bitbucket Enterprise
Since Bitbucket uses the URL's query string to specify the branch, there is no need to use special URL variables. The format is the following:
```
https:///projects//repos//raw//?
```##### Token configuration
To create token or `HTTP access token` follow the official guide [HTTP access tokens](https://confluence.atlassian.com/bitbucketserver/http-access-tokens-939515499.html)
Make sure you set `Project read` and `Repository read` permissions (and nothing more) when creating the token.
![bitbucket-token](docs/bitbucket-token.png)
Token is inserted in the header using `curl`
```
-H Authorization: Bearer ${token}
```:bulb: Token is needed for private repositories only
For example, using your private repositories would need the following settings:
```bash
export INSTALLER_CONFIG_URL=https://contoso/projects/project/repos/repo/raw/projects.json
export INSTALLER_CONFIG_SCM=bitbucket_server
export INSTALLER_CONFIG_TOKEN=1bacnotmyrealtoken123beefbea
```#### Plain HTTP
This type is mainly used for testing and it's very similar to GitHub's format, only that `token` or any other authentication is not supported.
To get the configuration file the following URL format is used:
```
https:////#branch#//
```
The following variables are used in the URL:- `#branch#` is replaced with the current/working branch (this done by the script)
For example, using your `localhost` server for configuration:
```
export INSTALLER_CONFIG_URL=https://localhost:8080/folder/#branch#/projects.json
```Branches are simply folders like `main`, `master` etc.
:memo: You can set these variables in `~/.profile` or `~/.bashrc` to make them permanent
### Getting started for the first time
Next get the **installer** with `curl` for the first time:
```bash
curl -L https://raw.githubusercontent.com/bvarnai/respository-installer/main/src/installer.sh -o installer.sh && chmod +x installer.sh
```Finally run **installer** in the current working directory:
```bash
./installer.sh
```
:tada: Once downloaded **installer** will upgrade itself, no need to run `curl` again.### Git credentials
:warning: **installer** doesn't manage your *Git* credentials (ssh keys etc.) in any way. You need setup *Git* credentials to work with the repositories specified in your configuration
### Prerequisites
Following tools are required and must be installed:
- `git`
- `curl`
- `jq`
- `sed`
- `uname`
- `awk`
- `grep`
- `bash` >= 4.0.0## Configuration
### Workspace
Workspace is the directory where your repositories/projects are cloned. It's also the current working directory where **installer** runs. Projects in the configuration
are specified *relative* to this directory.Example layout with `installer.sh` present:
```bash
workspace-root
project1
project2
subfolder/project3
installer.sh
projects.json
```### Configuration file
The configuration file is called `projects.json` and it's downloaded using the `INSTALLER_CONFIG_URL` environment variable. It contains information about all your *Git* projects, including setup instructions.
```json
{
"bootstrap": "myproject",
"projects": [
{
"name": "myproject",
"category": "development",
"default": "true",
"urls": {
"fetch": "https://github.com/octocat/Hello-World.git",
"push": "[email protected]:octocat/Hello-World.git"
},
"options": {
"clone": "--depth 1"
},
"configuration": [
"core.autocrlf false",
"core.safecrlf false"
],
"branch": "master",
"update": "true",
"doLast": [
"./do_something.sh"
]
}
]
}
```| Elements | | | Description |
| -------------- | ------------------------ | ---- | ----------- |
| bootstrap | | | Bootstrap project is always added implicitly. Referenced by `name` in `projects` |
| projects | | | Array of projects |
| | name | | Project name |
| | path | | Project path. Relative to workspace root. If not specified `name` will be used as path *[optional]* |
| | category | | Project category. Informal tagging of projects. Displayed during project listing *[optional]* |
| | default | | Whether to install the project if no project set is specified |
| | urls | | *Git* repository URLs |
| | | fetch | URL used for `fetch` |
| | | push | URL used for `push`. If not specified `fetch` URL will be used *[optional]* |
| | options | | *Git* command options |
| | | clone | Options for `clone` command. For example `--depth 1`" would result in a shallow clone *[optional]* |
| | configuration | | Array of *Git* configuration `config` options, repository scope. Add `--global` for global scope *[optional]* |
| | branch | | Default branch |
| | update | | Whether to force the repository update and reset to latest on the default branch |
| | doLast | | Array of shell commands to execute after repository update *[optional]* |:memo: Additional notes
- A bootstrap project is simply a project that is always installed
- :warning: A bootstrap project must be set to default `default==true` as well
- Different `fetch` and `push` URLs can be used to reduce load in *Git* hosting server, for example use `https` for `fetch` and `ssh` for `push`
- Setting `update==false` means repositories are fetched but not updated. This is desirable for development projects, so working branches are felt unchanged
- :warning: Setting `update==true` means repositories are fetched, reset and updated. This also means the branch will be switched to the default branch:bulb: You can use a bootstrap project to host your DevOps scripts etc. for example doLast scripts
## Usage
Command syntax is the following:
```bash
./installer.sh [options] [] [arguments]
```Optional elements are shown in brackets []. For example, command may take a list of projects as an argument.
### Options
- `-y, --yes` - skip user prompts
- `--link` - use symlinks to target directory
- `--branch` - overrides `branch` setting in configuration
- `--stream` - specifies the `stream` of the configuration
- `--fetch-all` - fetches all remotes, branches
- `--prune` - prune during fetch
- `--git-quiet` - pass quite to git commands (not everything is suppressed)
- `--skip-dolast` - do not run doLast commends (useful in CI environments where some setup is not wanted)#### Options for development/testing
- `--skip-self-update` - skip the script update step
- `--use-local-config` - use a local configuration file### Link mode
In some cases, you don't want to have a fresh clone of a project to save some time. For example *Jenkins* multibranch pipeline would create a new workspace and make a fresh clone using **installer**. This is where `link` mode can help.
Let see an example *Jenkinsfile*
```groovy
pipeline {environment {
// installer configuration
INSTALLER_SELF_URL = 'https://raw.githubusercontent.com/bvarnai/respository-installer/#branch#/src/installer.sh'
INSTALLER_CONFIG_URL = 'https://raw.githubusercontent.com/bvarnai/respository-installer/#branch#/src/projects.json'// use a directory outside of job's workspace
SHARED_WORKSPACE = "${WORKSPACE}/../shared_workspace"
}stages {
stage('Prepare workspace') {
steps {
// install dependencies
sh '''
mkdir -p ${SHARED_WORKSPACE}curl -s -o installer.sh -L ${INSTALLER_SELF_URL} && chmod +x installer.sh
./installer.sh --yes --link ${SHARED_WORKSPACE} myproject1 myproject2
'''
}
}
stage ('Next') {
steps {
...
}
}
}
}
```This will create symlinks `myproject1` and `myproject2` in the job's workspace, pointing to `../shared_workspace/myproject1` and `../shared_workspace/myproject2` directories respectively.
:warning: If you have multiple executors, parallel jobs might be running on the same shared workspace directory. This can be prevented by using the `EXECUTOR_NUMBER` variable
```groovy
SHARED_WORKSPACE = "${WORKSPACE}/../shared_workspace/${EXECUTOR_NUMBER}"
```### Stream explained
For example the team is working on a "theoretical" Java update, migrating from Java 8 to Java 17. In the development project repository, they created a branch `java17` and started to work. However `main` development continues on Java 8 until everything is ready. `java17` branch needs the Java 17 JDK, tools etc. This means there are two parallel `stream`s of development. There will be two `projects.json` files on the corresponding branches with default branches set to `main` or `java17`.
If a developer works on `java17` branch, simply switches tooling to that stream
```bash
./installer.sh --stream java17 update
```Other developer who remains on `main` just continues as
```bash
./installer.sh --stream main update
```### Custom environments
#### Dependencies
In some cases you want/need to manage your dependencies independently from the environment. For example *Git for Windows* does not include `jq` by default.
You can add custom code to **installer** to bootstrap `jq` and download it on the fly.```bash
function user_get_dependencies()
{
# put your code here
:
}
```This function could override the following globals with the location of these executables:
- `INSTALLER_JQ`
- `INSTALLER_CURL`An example implementation for `jq` for *Linux/Git Bash*
```bash
function user_get_dependencies()
{
if jq --version > /dev/null 2>&1; then
# jq is available in PATH
INSTALLER_JQ='jq'
elif .installer/jq --version > /dev/null 2>&1; then
# jq is available in local temp
INSTALLER_JQ='.installer/jq'
else
# jq will be downloaded
local JQSourceURL
if [[ "${system}" == "Linux" ]]; then
JQSourceURL='https://github.com/jqlang/jq/releases/download/jq-1.7.1/jq-linux-amd64'
elif [[ "${system}" =~ ^(MINGW64_NT|MSYS_NT) ]]; then
JQSourceURL='https://github.com/jqlang/jq/releases/download/jq-1.7.1/jq-windows-amd64.exe'
else
err "Unsupported system ${system}"
exit 1
fi
INSTALLER_JQ='.installer/jq'
fiif [[ -n ${JQSourceURL} ]]; then
log "Getting jq..."
local httpCode
httpCode=$(curl_get "${JQSourceURL}" '' "${INSTALLER_JQ}")
if [[ ${httpCode} -ne 200 ]] ; then
err "Failed to download jq binary"
exit 1
fi# set executable permission
chmod +x "${INSTALLER_JQ}" > /dev/null 2>&1;
fi
}
```:memo: I recommend to use `.installer` directory as a "temp" directory used to store dependencies
#### Link/unlink
Creating symbolic links might be platform specific. It's definitely the case for *Windows* clients.
There are user functions `user_link` and `user_unlink` to handle symbolic links.
The default implementation is tested on the following platforms:
- Linux amd64 - Ubuntu
- Windows amd64 - Git for Windows 64 bit
- 2.41.0+### Commands
The following commands are available. For options see [Options](#options)
---
#### help```bash
./installer.sh help
```Displays the help.
---
#### list```bash
./installer.sh list
```Lists available projects.
---
#### install```bash
./installer.sh install [project...]
```Installs a project(s). This is the default command, if nothing else is specified.
Arguments:
- `project` - the list of projects to install separated by a whitespace
:memo: If you run `install` without any arguments, all projects marked `default==true` will be installed
---
#### update```bash
./installer.sh update
```Updates existing projects in the current directory.
---
## FAQ
To be added. If you have any questions, just create an issue and I will respond.
## Development notes
- I used Google's [Shell Style Guide](https://google.github.io/styleguide/shellguide.html) with the help of [ShellCheck](https://www.shellcheck.net/)
- Tests written in [Bats-core: Bash Automated Testing System](https://github.com/bats-core/bats-core)