summaryrefslogtreecommitdiff
path: root/.gitlab-ci/check-commit.py
blob: b533bc4200a30b1607b4a6d2bb32315bd8325e57 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#!/usr/bin/env python3
# vim: set expandtab shiftwidth=4:
# -*- Mode: python; coding: utf-8; indent-tabs-mode: nil -*- */
#
# This script tests a few things against the commit messages, search for
# `def test_` to see the actual tests run.

import git
import os
import pytest

if os.environ.get('CI'):
    # Environment variables set by gitlab
    CI_COMMIT_SHA = os.environ['CI_COMMIT_SHA']
    # This is intentionally hardcoded to master. CI_MERGE_REQUEST_TARGET_BRANCH_NAME
    # is only available when run with only: [merge_requests]
    # but that generates a detached pipeline with only this job in it.
    # Since merging into a non-master branch is not a thing in libevdev
    # anyway, we can hardcode this here.
    CI_MERGE_REQUEST_TARGET_BRANCH_NAME = 'master'
    CI_SERVER_HOST = os.environ['CI_SERVER_HOST']
    UPSTREAM = 'upstream'
else:
    # Local emulation mode when called directly
    import argparse

    parser = argparse.ArgumentParser(description='Commit message checker - local emulation mode')
    parser.add_argument('--sha', help='The commit message to start at (default: HEAD}',
                        default='HEAD')
    parser.add_argument('--branch', help='The branch name to merge to (default: master)',
                        default='master')
    parser.add_argument('--remote', help='The remote name (default: origin)',
                        default='origin')
    args = parser.parse_args()

    CI_COMMIT_SHA = args.sha
    CI_MERGE_REQUEST_TARGET_BRANCH_NAME = args.branch
    CI_SERVER_HOST = None
    UPSTREAM = 'origin'
    print(f'Running in local testing mode.')

print(f'Merging {CI_COMMIT_SHA} into {CI_MERGE_REQUEST_TARGET_BRANCH_NAME}')

# We need to add the real libevdev as remote, our origin here is the user's
# fork.
repo = git.Repo('.')
if UPSTREAM not in repo.remotes:
    upstream = repo.create_remote('upstream', f'https://{CI_SERVER_HOST}/libevdev/libevdev.git')
    upstream.fetch()

sha = CI_COMMIT_SHA
branch = CI_MERGE_REQUEST_TARGET_BRANCH_NAME

commits = list(repo.iter_commits(f'{UPSTREAM}/{branch}..{sha}'))


def error(commit, message, long_message=''):
    info = ('After correcting the above issue(s), force-push to the same branch.\n'
            'This will re-trigger the CI.\n\n'
            'A list of requirements for commit messages is available at\n'
            'https://gitlab.freedesktop.org/libevdev/libevdev/blob/master/CODING_STYLE.md')

    msg = (f'\n'
           f'Commit message check failed: {message}\n\n'
           f'  commit: {str(commit)}\n'
           f'  author: {commit.author.name} <{commit.author.email}>\n'
           f'\n'
           f'  {commit.summary}\n'
           f'\n'
           f'\n'
           f'{long_message}\n\n'
           f'{info}\n\n')
    return msg


@pytest.mark.parametrize('commit', commits)
class TestCommits:
    def test_author_email(self, commit):
        assert '@users.noreply.gitlab.freedesktop.org' not in commit.author.email, \
            error(commit, 'git author email invalid',
                  ('Please set your name and email with the commands\n',
                   '    git config --global user.name Your Name\n'
                   '    git config --global user.email your.email@provider.com\n'))

    def test_signed_off_by(self, commit):
        if not commit.message.startswith('Revert "'):
            assert 'Signed-off-by:' in commit.message, \
                error(commit, 'missing Signed-off-by tag',
                      'Please add the required "Signed-off-by: author information" line\n'
                      'to the commit message')

    def test_fixup(self, commit):
        assert not commit.message.startswith('fixup!'), \
            error(commit, 'Remove fixup! tag',
                  'Leftover "fixup!" commit message detected, please squash')
        assert not commit.message.startswith('squash!'), \
            error(commit, 'Remove squash! tag',
                  'Leftover "squash!" commit message detected, please squash')

    def test_line_length(self, commit):
        lines = commit.message.split('\n')
        first_line = lines[0]

        assert len(first_line) < 85, \
            error(commit, 'Commit message subject line too long')

        try:
            second_line = lines[1]
            assert second_line == '', \
                error(commit, 'Second line in commit message must be emtpy')
        except IndexError:
            pass


if __name__ == '__main__':
    pytest.main([__file__])