Skip to content

Commit hooks based on experience, needs in teams I have worked, and acquired taste.

License

Notifications You must be signed in to change notification settings

ShellMagick/commit-hooks

Repository files navigation

ShellMagick / commit-hooks

Commit hooks based on experience, needs in teams I have worked, and acquired taste.

This repository uses the pre-commit framework and is heavily influenced by their out-of-the-box hooks.

Important

Notably, hooks in this repository do not support Python 3.8, only 3.9 and above. This is, because str.removeprefix is used, which has been introduced with Python 3.9.

Listing of hooks

In order of proposed configuration:

Using ShellMagick / commit-hooks with pre-commit

Add this to your .pre-commit-config.yaml:

-   repo: https://github.com/ShellMagick/commit-hooks
    rev: v24.06  # Use the ref you want to point at
    hooks:
    -   id: no-boms
    # -   id: ...

Suggested initial configuration for generic projects

In case you are working on a non-Python project (e.g., Java), you may find the following config useful:

default_install_hook_types: [ commit-msg, pre-commit, prepare-commit-msg ]
default_stages: [ pre-commit ]
repos:
-   repo: https://github.com/ShellMagick/commit-hooks
    rev: v24.06
    hooks:
    -   id: no-boms
    -   id: no-todos
    -   id: commiticketing
    -   id: cojira
        verbose: true
        args: [ '-l', '-u=$JIRA_URI', '-p=$JIRA_PAT', '-v=$ALLOWED_VERSIONS' ]
    -   id: lint-commit-message
...

We propose that cojira should be configured after commiticketing. This way commiticketing can prefix your commit message, if needed, and thus cojira can check without problems.

Available hooks

cojira

Warning

Consider this hook as EXPERIMENTAL in the sense of a) expect clumsy UX b) do not be surprised by minor bugs

In case you are working with a ticketing system, and you want to "bind" your commits to tickets (cf. commiticketing), you may also want to make sure that the referenced ticket is in a "desired" state.

This pre-commit hook can look up basic status of a JIRA-ticket based on the arguments given.

Important

Consider enabling verbose output for this hook, so that you see inline feedback.

Arguments

  • -l/--lenient: The hook is lenient regarding configuration. Enables a "soft onboarding" of this hook in projects.
    • In case specified, but no JIRA URI-root given (either parameter missing, or it is resolved as an empty String), then the hook early-exist with "success".
  • -u/--jira-uri: the URI-root for your JIRA instance.
    • In case it starts with $, it will be interpreted as an environment variable.
  • -p/--jira-pat: the PAT (personal access token) to be used for queries against the JIRA REST API (<-u>/rest/api/latest/issue/<ticket>).
    • In case it starts with $, it will be interpreted as an environment variable.
  • -i/--allow-status-category: Defines an "allowed" ("included") status category. This argument is repeatable.
    • These take precendence over "disallowed" ("excluded") categories.
    • By default this is an empty list (i.e., a category is implicitly "allowed" if and only if it is not "disallowed").
    • JSONPath of status category is: $.fields.status.statusCategory.key.
  • -e/--disallow-status-category: Defines an "disallowed" ("excluded") status category. This argument is repeatable.
    • In case none are specified, by default this contains the status category "done".
    • In case at least one are specified, only the specified values are considered.
    • JSONPath of status category is: $.fields.status.statusCategory.key.
  • -v/--allowed-fix-version: Defines an "allowed" fix version. This argument is repeatable.
    • In case it starts with $, it will be interpreted as an environment variable.
    • In case none are given, no check for fix versions is performed.
    • JSONPath of fix version is: $.fields.fixVersions[0].name, i.e., only the first fix version of the ticket is checked. In case multiple fix versions are defined on the ticket, that is considered an error (i.e., as if no fix versions were specified).

Possible outputs

Presuming correct configuration of JIRA URI and PAT, some examples of possible results are:

  • "Could not reify ticket from commit message" with return code 4 in case the commit message does not start with a ticketing reference (cf. commiticketing).
  • "Ticket has no fix version, but it is expected" with return code 3 in case the fix version in JIRA is empty (or multiple fix versions are defined), but -v is at least once defined for the hook.
  • "Fix version of ticket ("{ticket_version}") is not allowed" with return code 2 in case the fix version in JIRA is not empty, but does not correspond to any value given via -v.
  • "Ticket status category ("{category}") is not allowed" with return code 1 in case the status category is not on the "allowed" list (-i) and is on the "disallowed" list (-e).
  • "Ticket is OK according to COJIRA rules" with return code 0 if everything is according to expectations.

Side effect

In case:

  • the JIRA REST API could not be queried (incorrect URI/PAT)
  • or the "allowed" fix version list is not empty and the ticket's fix version is not in the "allowed" fix version list
  • or the ticket's status category is not in the "allowed" list and the ticket's status category is on the "disallowed" list

then the pre-commit hook results in a failure. No changes will be done in your repository (files and commits are not touched by this hook).

commiticketing

Many projects working with Git have a pre-defined workflow; let it be Git-Flow, GitHub-Flow, a derivation of them, or any other kind of flow.

What also is paramount that you have some kind of ticketing systems, and you want to correlate your work (i.e., commits) to a ticket—exactly one ticket; it is usually considered an anti-pattern to work on multiple tickets simultaneously or have commits, which do not correlate to any tickets.

In case you want to tie your commits to your tickets tightly, one of the easiest things you could do is prefixing your tickets with a corresponding ticketing reference (e.g., ISSUE-42: Add new gizmo).

This hook is aimed to help in this correlation work and remove the tedious chore of typing the prefix manually at each commit. It works by parsing the current branch name and extracting the ticketing information from it with additional, optional context. I.e.,

  • commits on feature/ISSUE-42 and feature/ISSUE-42-human-readable-description will be prefixed with ISSUE-42:␣
    • commits on branches bugfix/ISSUE-42, bugfix/ISSUE-42-desc, hotfix/ISSUE-42, hotfix/ISSUE-42-desc will be handled the same and receive the prefix ISSUE-42:␣ by default.
    • commits on user/username/ISSUE-42, user/username/ISSUE-42-desc, backup/username/ISSUE-42, and backup/username/ISSUE-42 are handled the same and receive the prefix ISSUE-42:␣ by default.

Note that the extraction happens from either the second or the third level of the branch name and the prefix is anything before a(n optional) second dash in the branch name, as well as :␣ (colon and whitespace). Additionally, it will make the very first character of your commit message upper case.

Please note that even though feature, bugfix, and hotfix are hinting towards using Git-Flow, the generic idea is not tied to Git-Flow at all and can be used by many other workflows.

Example

You are working on the ticket Fix the ODN relay in the project DSN and it has the ticket number 47. Thus, you check out the branch bugfix/DSN-47-fix-odn. You do a commit with the commit message open hatch, and this hook will automatically adjust your commit message to DSN-47: Open hatch.

Arguments

This hook has four optional arguments, three of them being repeatable:

  • -b/--branch: branch name prefixes to be processed by this hook and extracting ticketing information from the second level; this argument is repeatable
    • If none given, the following are implicitly processed: feature, bugfix, hotfix.
  • -t/--two-level: branch name prefixes to be processed by this hook and extracting ticketing information from the third level; this argument is repeatable
    • If none given, the following are implicitly processed: user, backup.
  • -l/--long-prefix: use long-prefixing, unless on the exclusion list
    • By default, this is turned off (i.e., long-prefixing is not used by default)
    • Long prefixing means infixing the first level of the branch name into the commit message, examples:
      • For the branch named bugfix/DSN-47 the short prefix DNS-47:␣ becomes the long prefix DNS-47:␣(bugfix)␣
      • For the branch named hotfix/DSN-47 the short prefix DNS-47:␣ becomes the long prefix DNS-47:␣(hotfix)␣
      • For the branch named backup/miles/DSN-47 the short prefix DNS-47:␣ becomes the long prefix DNS-47:␣(backup)␣
  • -e/--exclude-long-prefix: in case long-prefixing is turned on, exclude branches starting with any of these; this argument is repeatable
    • In case long-prefixing is not used (and thus by default), this argument does nothing
    • If none given, the following are implicitly excluded from long-prefixing: feature, user, backup

Side effects

This hook changes your commit message, in case it does not already conform the guidelines of this hook.

Please note that basic precautions are taken against doubly-prefixing, but mixing short/long-prefix configurations may cause surprises.

lint-commit-message

A basic linter for commit messages, based on How to Write a Git Commit Message by Chris Beams.

The linter checks the following (corresponding the rules from the linked article):

  1. ✅ There must be a commit message.
  2. ✅ The subject line and the body are separated by an empty line.
  3. ✅ The subject line is limited to 72 characters.
    • Note: This is more relaxed than the proposed 50 characters by the article. The reasoning behind this is that together with commiticketing, it is sometimes hard to adhere to the limit of 50.
    • Note: The limit is adjustable, see the arguments of this hook.
  4. ❌ Is not checked (would be "Capitilize the subject line"). Use commiticketing instead.
  5. ✅ The subject line does not end with a punctuation character (i.e., matching the regex [.,;?!\\-]$).
    • Note: This is stricter than the proposed rule "do not end the subject line with a period" by the article.
  6. ❌ Is not checked (would be "Use imperative mood in the subject line"), because it would need NLP.
  7. ✅ The lines of the body are limited to 120 characters.
    • Note: This is more relaxed than the proposed 72 characters by the article.
    • Note: The limit is adjustable, see the arguments of this hook.
  8. ❌ Is not checked (would be "Use the body to explain what and why vs. how"), because it would need advanced NLP.

Arguments

This hook has two optional arguments:

  • -sl/--subject-line-length: adjust the parameter of rule 2 (default value: 72).
  • -bl/--body-line-length: adjust the parameter of rule 6 (default value: 120).
  • -e/--ending: adjust the parameter of rule 4 (default: [.,;?!\\-])
    • Example: In case you want to forbid any non-alphanumeric-character, you could use \\W as parameter (note that this forbids parentheses too).

Side effects

This hook does not change your commit message, just reports violations and prevents the commit.

no-boms

Prevent commits containing any files with UTF BOMs of any kind.

Arguments

This hook does not have any arguments.

Side effects

This hook does not change any files, just reports violations and prevents the commit.

no-todos

Prevent commits containing (pre-)defined "comment tags".

Arguments

This hook has two optional, repeatable arguments:

  • -t/--todo-tag: defines a "comment tag" as forbidden. In case any of these strings is matched anywhere in any file being committed, the commit is prevented.
    • By default, the following strings are considered as tags: TODO, FIXME, XXX.
  • -e/--except-in: defines filenames, which are excluded from the check.
    • The same could be achieved by an exclude in the hooks configuration, but this argument is useful together with verbose, as in this case it will log the excluded files containing any of the "comment tags" (as a quasi-warning).

Side effects

This hook does not change any files, just reports violations and prevents the commit.

About

Commit hooks based on experience, needs in teams I have worked, and acquired taste.

Resources

License

Security policy

Stars

Watchers

Forks

Languages