|  | #! /usr/bin/env python | 
|  |  | 
|  | # Copyright 2016 Hewlett Packard Enterprise Development Company, L.P. | 
|  | # | 
|  | # 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. | 
|  |  | 
|  | # This script is intended to be run as part of a periodic proposal bot | 
|  | # job in OpenStack infrastructure. | 
|  | # | 
|  | # In order to function correctly, the environment in which the | 
|  | # script runs must have | 
|  | #   * network access to the review.openstack.org Gerrit API | 
|  | #     working directory | 
|  | #   * network access to https://git.openstack.org/cgit | 
|  |  | 
|  | import json | 
|  | import re | 
|  |  | 
|  | try: | 
|  | # For Python 3.0 and later | 
|  | from urllib.error import HTTPError | 
|  | import urllib.request as urllib | 
|  | except ImportError: | 
|  | # Fall back to Python 2's urllib2 | 
|  | import urllib2 as urllib | 
|  | from urllib2 import HTTPError | 
|  |  | 
|  |  | 
|  | url = 'https://review.openstack.org/projects/' | 
|  |  | 
|  | # This is what a project looks like | 
|  | ''' | 
|  | "openstack-attic/akanda": { | 
|  | "id": "openstack-attic%2Fakanda", | 
|  | "state": "READ_ONLY" | 
|  | }, | 
|  | ''' | 
|  |  | 
|  |  | 
|  | def is_in_openstack_namespace(proj): | 
|  | return proj.startswith('openstack/') | 
|  |  | 
|  | # Rather than returning a 404 for a nonexistent file, cgit delivers a | 
|  | # 0-byte response to a GET request.  It also does not provide a | 
|  | # Content-Length in a HEAD response, so the way we tell if a file exists | 
|  | # is to check the length of the entire GET response body. | 
|  |  | 
|  |  | 
|  | def has_tempest_plugin(proj): | 
|  | try: | 
|  | r = urllib.urlopen( | 
|  | "https://git.openstack.org/cgit/%s/plain/setup.cfg" % proj) | 
|  | except HTTPError as err: | 
|  | if err.code == 404: | 
|  | return False | 
|  | p = re.compile(r'^tempest\.test_plugins', re.M) | 
|  | if p.findall(r.read().decode('utf-8')): | 
|  | return True | 
|  | else: | 
|  | False | 
|  |  | 
|  |  | 
|  | r = urllib.urlopen(url) | 
|  | # Gerrit prepends 4 garbage octets to the JSON, in order to counter | 
|  | # cross-site scripting attacks.  Therefore we must discard it so the | 
|  | # json library won't choke. | 
|  | content = r.read().decode('utf-8')[4:] | 
|  | projects = sorted(filter(is_in_openstack_namespace, json.loads(content))) | 
|  |  | 
|  | # Retrieve projects having no deb, puppet, ui or spec namespace as those | 
|  | # namespaces do not contains tempest plugins. | 
|  | projects_list = [i for i in projects if not ( | 
|  | i.startswith('openstack/deb-') or | 
|  | i.startswith('openstack/puppet-') or | 
|  | i.endswith('-ui') or | 
|  | i.endswith('-specs'))] | 
|  |  | 
|  | found_plugins = list(filter(has_tempest_plugin, projects_list)) | 
|  |  | 
|  | # Every element of the found_plugins list begins with "openstack/". | 
|  | # We drop those initial 10 octets when printing the list. | 
|  | for project in found_plugins: | 
|  | print(project[10:]) |