forked from kernelci/kernelci-pipeline
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrigger.py
executable file
·158 lines (137 loc) · 4.67 KB
/
trigger.py
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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
#!/usr/bin/env python3
#
# SPDX-License-Identifier: LGPL-2.1-or-later
#
# Copyright (C) 2021, 2022 Collabora Limited
# Author: Guillaume Tucker <[email protected]>
# Author: Jeny Sadadia <[email protected]>
from datetime import datetime, timedelta
import json
import logging
import sys
import time
import kernelci
import kernelci.build
import kernelci.config
import kernelci.db
from kernelci.cli import Args, Command, parse_opts
import urllib
import requests
from base import Service
class Trigger(Service):
def __init__(self, configs, args):
super().__init__(configs, args, 'trigger')
self._build_configs = configs['build_configs']
def _log_revision(self, message, build_config, head_commit):
self.log.info(f"{message:32s} {build_config.name:32s} {head_commit}")
def _run_trigger(self, build_config, force, timeout):
head_commit = kernelci.build.get_branch_head(build_config)
node_count = self._api.count_nodes({
"revision.commit": head_commit,
})
if node_count > 0:
if force:
self._log_revision(
"Resubmitting existing revision", build_config, head_commit
)
else:
self._log_revision(
"Existing revision", build_config, head_commit
)
return
else:
self._log_revision(
"New revision", build_config, head_commit
)
revision = {
'tree': build_config.tree.name,
'url': build_config.tree.url,
'branch': build_config.branch,
'commit': head_commit,
}
checkout_timeout = datetime.utcnow() + timedelta(minutes=timeout)
node = {
'name': 'checkout',
'path': ['checkout'],
'revision': revision,
'timeout': checkout_timeout.isoformat(),
}
try:
self._api.create_node(node)
except requests.exceptions.HTTPError as ex:
detail = ex.response.json().get('detail')
if detail:
self.log.error(detail)
except Exception as ex:
self.traceback(ex)
def _iterate_build_configs(self, force, build_configs_list,
timeout):
for name, config in self._build_configs.items():
if not build_configs_list or name in build_configs_list:
self._run_trigger(config, force, timeout)
def _setup(self, args):
return {
'poll_period': int(args.poll_period),
'force': args.force,
'build_configs_list': (args.build_configs or '').split(),
'startup_delay': int(args.startup_delay or 0),
'timeout': args.timeout,
}
def _run(self, ctx):
poll_period, force, build_configs_list, startup_delay, timeout = (
ctx[key] for key in (
'poll_period', 'force', 'build_configs_list', 'startup_delay',
'timeout'
)
)
if startup_delay:
self.log.info(f"Delay: {startup_delay}s")
time.sleep(startup_delay)
while True:
self._iterate_build_configs(force, build_configs_list,
timeout)
if poll_period:
self.log.info(f"Sleeping for {poll_period}s")
time.sleep(poll_period)
else:
self.log.info("Not polling.")
break
return True
class cmd_run(Command):
help = "Submit a new revision to the API based on local git repo"
args = [
Args.api_config,
]
opt_args = [
{
'name': '--poll-period',
'type': int,
'help': "Polling period in seconds, disabled when set to 0",
},
{
'name': '--force',
'action': 'store_true',
'help': "Always create a new checkout node",
},
{
'name': '--build-configs',
'help': "List of build configurations to monitor",
},
{
'name': '--startup-delay',
'type': int,
'help': "Delay loop at startup by a number of seconds",
},
{
'name': '--timeout',
'type': float,
'help': "Timeout minutes for checkout node",
},
]
def __call__(self, configs, args):
return Trigger(configs, args).run(args)
if __name__ == '__main__':
opts = parse_opts('trigger', globals())
configs = kernelci.config.load('config/pipeline.yaml')
status = opts.command(configs, opts)
sys.exit(0 if status is True else 1)