From ad28e91698c5d0affade8fcca339e61f5303ca51 Mon Sep 17 00:00:00 2001 From: "James E. Blair" Date: Wed, 27 Nov 2013 10:43:22 -0800 Subject: [PATCH] Add a zuul client Add a command line client called 'zuul' that supports one command to start with: 'enqueue'. It allows an operator (one with access to the gearman server) to enqueue an arbitrary change in a specified pipeline. It uses gearman to communicate with the Zuul server, which now has an added RPC listener component to answer such requests via gearman. Add tests for the client RPC interface. Raise an exception if a Gerrit query does not produce a change. Unlike events from Gerrit, user (or admin) submitted events over the RPC bus are more likely to reference invalid changes. To validate those, the Gerrit trigger will raise an exception (and remove from its cache) changes which prove to be invalid. Change-Id: Ife07683a736c15f4db44a0f9881f3f71b78716b2 --- setup.cfg | 1 + tests/test_scheduler.py | 88 ++++++++++++++++++++++++++++- zuul/cmd/client.py | 119 ++++++++++++++++++++++++++++++++++++++++ zuul/cmd/server.py | 3 + zuul/model.py | 3 + zuul/rpcclient.py | 61 ++++++++++++++++++++ zuul/rpclistener.py | 116 +++++++++++++++++++++++++++++++++++++++ zuul/scheduler.py | 5 ++ zuul/trigger/gerrit.py | 9 ++- 9 files changed, 401 insertions(+), 4 deletions(-) create mode 100644 zuul/cmd/client.py create mode 100644 zuul/rpcclient.py create mode 100644 zuul/rpclistener.py diff --git a/setup.cfg b/setup.cfg index 45f8e42d98..9ff62d67c9 100644 --- a/setup.cfg +++ b/setup.cfg @@ -22,6 +22,7 @@ warnerrors = True [entry_points] console_scripts = zuul-server = zuul.cmd.server:main + zuul = zuul.cmd.client:main [build_sphinx] source-dir = doc/source diff --git a/tests/test_scheduler.py b/tests/test_scheduler.py index 4832af9424..63ab94f2af 100755 --- a/tests/test_scheduler.py +++ b/tests/test_scheduler.py @@ -44,6 +44,8 @@ import testtools import zuul.scheduler import zuul.webapp +import zuul.rpclistener +import zuul.rpcclient import zuul.launcher.gearman import zuul.reporter.gerrit import zuul.reporter.smtp @@ -351,8 +353,10 @@ class FakeGerrit(object): change.setReported() def query(self, number): - change = self.changes[int(number)] - return change.query() + change = self.changes.get(int(number)) + if change: + return change.query() + return {} def startWatching(self, *args, **kw): pass @@ -806,6 +810,7 @@ class TestScheduler(testtools.TestCase): self.fake_gerrit.upstream_root = self.upstream_root self.webapp = zuul.webapp.WebApp(self.sched, port=0) + self.rpc = zuul.rpclistener.RPCListener(self.config, self.sched) self.sched.setLauncher(self.launcher) self.sched.registerTrigger(self.gerrit) @@ -824,6 +829,7 @@ class TestScheduler(testtools.TestCase): self.sched.reconfigure(self.config) self.sched.resume() self.webapp.start() + self.rpc.start() self.launcher.gearman.waitForServer() self.registerJobs() self.builds = self.worker.running_builds @@ -857,6 +863,8 @@ class TestScheduler(testtools.TestCase): self.statsd.join() self.webapp.stop() self.webapp.join() + self.rpc.stop() + self.rpc.join() threads = threading.enumerate() if len(threads) > 1: self.log.error("More than one thread is running: %s" % threads) @@ -956,12 +964,14 @@ class TestScheduler(testtools.TestCase): while True: done = True for connection in self.gearman_server.active_connections: - if connection.functions: + if (connection.functions and + connection.client_id != 'Zuul RPC Listener'): done = False if done: break time.sleep(0) self.gearman_server.functions = set() + self.rpc.register() def haveAllBuildsReported(self): # See if Zuul is waiting on a meta job to complete @@ -2947,3 +2957,75 @@ class TestScheduler(testtools.TestCase): FakeSMTP.messages[1]['to_email']) self.assertEqual(A.messages[0], FakeSMTP.messages[1]['body']) + + def test_client_enqueue(self): + "Test that the RPC client can enqueue a change" + A = self.fake_gerrit.addFakeChange('org/project', 'master', 'A') + A.addApproval('CRVW', 2) + A.addApproval('APRV', 1) + + client = zuul.rpcclient.RPCClient('127.0.0.1', + self.gearman_server.port) + r = client.enqueue(pipeline='gate', + project='org/project', + trigger='gerrit', + change='1', + patchset='1') + self.waitUntilSettled() + self.assertEqual(self.getJobFromHistory('project-merge').result, + 'SUCCESS') + self.assertEqual(self.getJobFromHistory('project-test1').result, + 'SUCCESS') + self.assertEqual(self.getJobFromHistory('project-test2').result, + 'SUCCESS') + self.assertEqual(A.data['status'], 'MERGED') + self.assertEqual(A.reported, 2) + self.assertEqual(r, True) + + def test_client_enqueue_negative(self): + "Test that the RPC client returns errors" + client = zuul.rpcclient.RPCClient('127.0.0.1', + self.gearman_server.port) + with testtools.ExpectedException(zuul.rpcclient.RPCFailure, + "Invalid project"): + r = client.enqueue(pipeline='gate', + project='project-does-not-exist', + trigger='gerrit', + change='1', + patchset='1') + client.shutdown() + self.assertEqual(r, False) + + with testtools.ExpectedException(zuul.rpcclient.RPCFailure, + "Invalid pipeline"): + r = client.enqueue(pipeline='pipeline-does-not-exist', + project='org/project', + trigger='gerrit', + change='1', + patchset='1') + client.shutdown() + self.assertEqual(r, False) + + with testtools.ExpectedException(zuul.rpcclient.RPCFailure, + "Invalid trigger"): + r = client.enqueue(pipeline='gate', + project='org/project', + trigger='trigger-does-not-exist', + change='1', + patchset='1') + client.shutdown() + self.assertEqual(r, False) + + with testtools.ExpectedException(zuul.rpcclient.RPCFailure, + "Invalid change"): + r = client.enqueue(pipeline='gate', + project='org/project', + trigger='gerrit', + change='1', + patchset='1') + client.shutdown() + self.assertEqual(r, False) + + self.waitUntilSettled() + self.assertEqual(len(self.history), 0) + self.assertEqual(len(self.builds), 0) diff --git a/zuul/cmd/client.py b/zuul/cmd/client.py new file mode 100644 index 0000000000..a5327a24df --- /dev/null +++ b/zuul/cmd/client.py @@ -0,0 +1,119 @@ +#!/usr/bin/env python +# Copyright 2012 Hewlett-Packard Development Company, L.P. +# Copyright 2013 OpenStack Foundation +# +# Licensed under the Apache License, Version 2.0 (the "License"); you may +# not use this file except in compliance with the License. You may obtain +# a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +# License for the specific language governing permissions and limitations +# under the License. + +import argparse +import ConfigParser +import logging +import logging.config +import os +import sys + +import zuul.rpcclient + + +class Client(object): + log = logging.getLogger("zuul.Client") + + def __init__(self): + self.args = None + self.config = None + self.gear_server_pid = None + + def parse_arguments(self): + parser = argparse.ArgumentParser( + description='Zuul Project Gating System Client.') + parser.add_argument('-c', dest='config', + help='specify the config file') + parser.add_argument('-v', dest='verbose', action='store_true', + help='verbose output') + parser.add_argument('--version', dest='version', action='store_true', + help='show zuul version') + + subparsers = parser.add_subparsers(title='commands', + description='valid commands', + help='additional help') + + cmd_enqueue = subparsers.add_parser('enqueue', help='enqueue a change') + cmd_enqueue.add_argument('--trigger', help='trigger name', + required=True) + cmd_enqueue.add_argument('--pipeline', help='pipeline name', + required=True) + cmd_enqueue.add_argument('--project', help='project name', + required=True) + cmd_enqueue.add_argument('--change', help='change id', + required=True) + cmd_enqueue.add_argument('--patchset', help='patchset number', + required=True) + cmd_enqueue.set_defaults(func=self.enqueue) + + self.args = parser.parse_args() + + def read_config(self): + self.config = ConfigParser.ConfigParser() + if self.args.config: + locations = [self.args.config] + else: + locations = ['/etc/zuul/zuul.conf', + '~/zuul.conf'] + for fp in locations: + if os.path.exists(os.path.expanduser(fp)): + self.config.read(os.path.expanduser(fp)) + return + raise Exception("Unable to locate config file in %s" % locations) + + def setup_logging(self): + if self.args.verbose: + logging.basicConfig(level=logging.DEBUG) + + def main(self): + self.parse_arguments() + self.read_config() + self.setup_logging() + + if self.args.version: + from zuul.version import version_info as zuul_version_info + print "Zuul version: %s" % zuul_version_info.version_string() + sys.exit(0) + + self.server = self.config.get('gearman', 'server') + if self.config.has_option('gearman', 'port'): + self.port = self.config.get('gearman', 'port') + else: + self.port = 4730 + + if self.args.func(): + sys.exit(0) + else: + sys.exit(1) + + def enqueue(self): + client = zuul.rpcclient.RPCClient(self.server, self.port) + r = client.enqueue(pipeline=self.args.pipeline, + project=self.args.project, + trigger=self.args.trigger, + change=self.args.change, + patchset=self.args.patchset) + return r + + +def main(): + client = Client() + client.main() + + +if __name__ == "__main__": + sys.path.insert(0, '.') + main() diff --git a/zuul/cmd/server.py b/zuul/cmd/server.py index 710f35d879..3a51b1c44d 100755 --- a/zuul/cmd/server.py +++ b/zuul/cmd/server.py @@ -172,6 +172,7 @@ class Server(object): import zuul.trigger.gerrit import zuul.trigger.timer import zuul.webapp + import zuul.rpclistener if (self.config.has_option('gearman_server', 'start') and self.config.getboolean('gearman_server', 'start')): @@ -185,6 +186,7 @@ class Server(object): gerrit = zuul.trigger.gerrit.Gerrit(self.config, self.sched) timer = zuul.trigger.timer.Timer(self.config, self.sched) webapp = zuul.webapp.WebApp(self.sched) + rpc = zuul.rpclistener.RPCListener(self.config, self.sched) gerrit_reporter = zuul.reporter.gerrit.Reporter(gerrit) smtp_reporter = zuul.reporter.smtp.Reporter( self.config.get('smtp', 'default_from') @@ -207,6 +209,7 @@ class Server(object): self.sched.reconfigure(self.config) self.sched.resume() webapp.start() + rpc.start() signal.signal(signal.SIGHUP, self.reconfigure_handler) signal.signal(signal.SIGUSR1, self.exit_handler) diff --git a/zuul/model.py b/zuul/model.py index 0c694301c8..9a076d6efe 100644 --- a/zuul/model.py +++ b/zuul/model.py @@ -802,6 +802,9 @@ class TriggerEvent(object): self.newrev = None # timer self.timespec = None + # For events that arrive with a destination pipeline (eg, from + # an admin command, etc): + self.forced_pipeline = None def __repr__(self): ret = '