Git Setup and Usage Guide

Installation

Install Git using your package manager:

sudo pacman -S git

Configuration

Note: all of the Git configuration commands (i.e. git config) herein are for setting repository-specific configuration values. To set global values (which are used by default unless repository-specific ones are set), use the same commands but add the --global option. For example:

git config --global user.name "<user-name>"

Before executing any of the listed Git configuration commands, first navigate to the root directory of the Git repository for which one wants to set repository-specific configuration values:

cd <path/to/root-git-repo>

Global configuration values are stored in the ~/.gitconfig file; whereas, repository-specific values are stored in the .git/config file within the root directory of the repository.

To check the current Git configuration values:

git config --list

Global or repository-specific values can be updated either by editing their respective configuration files; or by re-running the appropriate git config commands with updated values. The latter method is recommended.

User-Name and Email Address

A user-name and email address is associated with each commit. Commits can also be signed using a GPG key as detailed in Setting Up a GPG Key.

To set the user-name and email address:

git config user.name "<user-name>"
git config user.email "<user-email@some-domain.com>"

Setting Up a GPG Key

If one has previously configured Git to use a different key format when signing with the --gpg-sign option, then unset this configuration so that the default format of OpenPGP will be used. The OpenPGP format is used by Codeberg and most other Git repository hosting service providers.

To unset:

git config --unset gpg.format

List all GPG keys in long form for which a public and private key is available (the private key is required to sign commits):

gpg --list-secret-keys --keyid-format=long

This command should output something similar to the following:

----------------------------
sec   ed25519/ABD5FEOZFBA3BJF9 2023-05-25 [SC]
      88VB7M1CY0A554QIOWE94U1BABD5FEOZFBA3BJF9
uid                 [ultimate] First Last <mail@domain.com>
ssb   cv25519/P78BHGJJ5213WRV1 2023-05-25 [E]

If no keys are present, see the GPG Key Generation guide.

The 16 alphanumeric characters (i.e. ABD5FEOZFBA3BJF9) after the "ed25519/" (or whatever algorithm was used to create the key-pair) is the referred to as the "GPG key ID". This value is used to identify the signing key used to sign Git commits, and therefore, must be set within the Git configuration:

git config user.signingkey "<gpg-key-id>"

Once set, signing commits can be enabled by default by setting the commit.gpgsign configuration option to true:

git config commit.gpgsign true

Alternatively, one can manually opt to sign each commit by setting the -S flag:

git commit -S -m "Some commit message

Default Git Editor

To set the default editor (e.g. to Neovim):

git config core.editor nvim

To set this for other programs as well, set the standardised VISUAL and EDITOR environment variables in one's terminal resource file (e.g. bashrc):

export VISUAL=nvim
export EDITOR="$VISUAL"

See the VISUAL vs. EDITOR question and answers on the Unix and Linux Stack Exchange website for more information on these environment variables.

Usage

Creating a Local Repository

Create a directory with the project's name:

mkdir <project-name>

Change into the directory:

cd <project-name>

Initialise the Git repository:

git init

Create any files and/or directories that are needed and then add them all to the next (i.e. first) commit:

git add .

Alternatively, one can add a specific file or directory manually:

git add <filename>

To check which files have been added ("staged") for the next commit:

git diff --staged --name-only

Once the desired files and directories have been added using the add command, make the first commit:

git commit -m "Initial commit"

Creating a Remote Repository

After initialising a local repository, one can specify a remote repository to tell Git where to push changes. The URL of the remote repository will depend on the hosting service and their conventions. Here, codeberg.org is used:

git remote add origin https://codeberg.org/user/project

Create or edit files, then stage the ones ready to be committed:

git add <filename>

Commit the changes, adding a short descriptive message pertaining to the changes:

git commit -m "Descriptive message"

Push the changes to the remote repository hosting service set-up prior:

git push -u origin master

The -u option sets the upstream remote to origin (which was the name of the remote repository set-up prior). The master is the name of the branch onto which changes shall be pushed upstream. This name should be the same as the one specified when creating the repository.

When connecting via HTTPS, Git will prompt for one's username and password. If one has set-up two-factor authentication, then a Personal Access Token (PAT) will have to be created which will be used as if its a password. Make sure to select the necessary scopes when creating the PAT otherwise one will not have the necessary privileges to perform account actions. Enter the username and password/PAT and then load the web-page for the remote repository and observe the changes.

Printing Commits

Print a list of all Git commits, including those that are on unnamed ("lost") branches (use q to quit):

git log --reflog

For example, the above command will display commits E and F in the following graph:

A---B---C---D (master, HEAD)
     \
      \-E---F

Squashing Commits

After a series of commits have been made, the history can be "squashed" before pushing the changes as a single change (commit). This makes tracking changes more manageable. Print the Git log with git log --oneline, then copy the hash that is immediately older than the oldest hash you want to squash. For example, given the following log output:

66fa363 (HEAD -> master) Update DOC.md structure
406535b Format code
1e92e69 Auto generate docs
q68yd2d Implement expansion using resolveExpandParams
af8g87b Auto generate docs
... more commits

Assuming one wants to merge the commits from 66fa363 through q68ud2d into a single commit, then run:

git rebase -i af8g87b

Git will then open your text editor to allow the interactive rebase. Change all pick actions to squash and then save and quit.

Git will then open the text editor again allowing you to edit the commit messages. This is where you can condense all previous commit messages into one. Save and quit when complete.

Re-run git log --oneline to check if commit history has been successfully updated. One should see something like:

5uv34rt (HEAD -> master) Condensed commit title
af8g87b Auto generate docs
... more commits

One can then push the changes if happy with the log.