mirror of
https://github.com/certbot/certbot.git
synced 2025-08-05 05:41:11 +03:00
* add packages to git commit * rename deploy stage * rename deploy jobs * set up github releases * remove v * tweak release script * remove publishing windows installer * update changelog
190 lines
6.8 KiB
Python
Executable File
190 lines
6.8 KiB
Python
Executable File
#!/usr/bin/env python
|
|
"""
|
|
Post-release script to publish artifacts created from Azure Pipelines.
|
|
|
|
This currently includes:
|
|
|
|
* Moving snaps from the beta channel to the stable channel
|
|
|
|
Setup:
|
|
- Install the snapcraft command line tool and log in to a privileged account.
|
|
- https://snapcraft.io/docs/installing-snapcraft
|
|
- Use the command `snapcraft login` to log in.
|
|
|
|
Run:
|
|
|
|
python tools/finish_release.py
|
|
|
|
Testing:
|
|
|
|
This script can be safely run between releases. When this is done, the script
|
|
should execute successfully.
|
|
|
|
"""
|
|
|
|
import argparse
|
|
import getpass
|
|
import glob
|
|
import os.path
|
|
import re
|
|
import subprocess
|
|
import sys
|
|
import tempfile
|
|
from zipfile import ZipFile
|
|
|
|
from azure.devops.connection import Connection
|
|
import requests
|
|
|
|
# Path to the root directory of the Certbot repository containing this script
|
|
REPO_ROOT = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
|
|
# This list contains the names of all Certbot DNS plugins. We used to have a
|
|
# CloudXNS plugin and since it's possible devs still have that directory
|
|
# locally, we filter it out here. If it's included in this list, this script
|
|
# will crash later when it fails to find a CloudXNS snap on the snap store with
|
|
# the current version since we no longer build it.
|
|
PLUGIN_SNAPS = [os.path.basename(path)
|
|
for path in glob.glob(os.path.join(REPO_ROOT, 'certbot-dns-*'))
|
|
if not path.endswith('certbot-dns-cloudxns')]
|
|
# This list contains the name of all Certbot snaps that should be published to
|
|
# the stable channel.
|
|
ALL_SNAPS = ['certbot'] + PLUGIN_SNAPS
|
|
# This is the count of the architectures currently supported by our snaps used
|
|
# for sanity checking.
|
|
SNAP_ARCH_COUNT = 3
|
|
|
|
|
|
def parse_args(args):
|
|
"""Parse command line arguments.
|
|
|
|
:param args: command line arguments with the program name removed. This is
|
|
usually taken from sys.argv[1:].
|
|
:type args: `list` of `str`
|
|
|
|
:returns: parsed arguments
|
|
:rtype: argparse.Namespace
|
|
|
|
"""
|
|
# Use the file's docstring for the help text and don't let argparse reformat it.
|
|
parser = argparse.ArgumentParser(description=__doc__,
|
|
formatter_class=argparse.RawDescriptionHelpFormatter)
|
|
return parser.parse_args(args)
|
|
|
|
|
|
def assert_logged_into_snapcraft():
|
|
"""Confirms that snapcraft is logged in to an account.
|
|
|
|
:raises SystemExit: if the command snapcraft is unavailable or it
|
|
isn't logged into an account
|
|
|
|
"""
|
|
cmd = 'snapcraft whoami'.split()
|
|
try:
|
|
subprocess.run(cmd, check=True, stdout=subprocess.DEVNULL,
|
|
stderr=subprocess.DEVNULL, universal_newlines=True)
|
|
except (subprocess.CalledProcessError, OSError):
|
|
print("Please make sure that the command line tool snapcraft is")
|
|
print("installed and that you have logged in to an account by running")
|
|
print("'snapcraft login'. If that fails, your credentials may have expired")
|
|
print("and you should run `snapcraft logout` followed by 'snapcraft login'.")
|
|
sys.exit(1)
|
|
|
|
|
|
def get_snap_revisions(snap, channel, version):
|
|
"""Finds the revisions for the snap and version in the given channel.
|
|
|
|
If you call this function without being logged in with snapcraft, it
|
|
will hang with no output.
|
|
|
|
:param str snap: the name of the snap on the snap store
|
|
:param str channel: snap channel to pull revisions from
|
|
:param str version: snap version number, e.g. 1.7.0
|
|
|
|
:returns: list of revision numbers
|
|
:rtype: `list` of `str`
|
|
|
|
:raises subprocess.CalledProcessError: if the snapcraft command
|
|
fails
|
|
|
|
:raises AssertionError: if the expected snaps are not found
|
|
|
|
"""
|
|
print('Getting revision numbers for', snap, version)
|
|
cmd = ['snapcraft', 'status', snap]
|
|
process = subprocess.run(cmd, check=True, stdout=subprocess.PIPE, universal_newlines=True)
|
|
pattern = f'^\s+{channel}\s+{version}\s+(\d+)\s*'
|
|
revisions = re.findall(pattern, process.stdout, re.MULTILINE)
|
|
assert len(revisions) == SNAP_ARCH_COUNT, f'Unexpected number of snaps found for {channel} {snap} {version} (expected {SNAP_ARCH_COUNT}, found {len(revisions)})'
|
|
return revisions
|
|
|
|
|
|
def promote_snaps(snaps, source_channel, version, progressive_percentage=None):
|
|
"""Promotes the given snaps from source_channel to the stable channel.
|
|
|
|
If the snaps have already been released to the stable channel, this
|
|
function will try to release them again which has no effect.
|
|
|
|
:param snaps: snap package names to be promoted
|
|
:type snaps: `list` of `str`
|
|
:param str source_channel: snap channel to promote from
|
|
:param str version: the version number that should be found in the
|
|
candidate channel, e.g. 1.7.0
|
|
:param progressive_percentage: specifies the percentage of a progressive
|
|
deployment
|
|
:type progressive_percentage: int or None
|
|
|
|
:raises SystemExit: if the command snapcraft is unavailable or it
|
|
isn't logged into an account
|
|
|
|
:raises subprocess.CalledProcessError: if a snapcraft command fails
|
|
for another reason
|
|
|
|
"""
|
|
assert_logged_into_snapcraft()
|
|
for snap in snaps:
|
|
revisions = get_snap_revisions(snap, source_channel, version)
|
|
# The loop below is kind of slow, so let's print some output about what
|
|
# it is doing.
|
|
print('Releasing', snap, 'snaps to the stable channel')
|
|
for revision in revisions:
|
|
cmd = ['snapcraft', 'release', snap, revision, 'stable']
|
|
if progressive_percentage:
|
|
cmd.extend(f'--progressive {progressive_percentage}'.split())
|
|
try:
|
|
subprocess.run(cmd, check=True, stdout=subprocess.PIPE, universal_newlines=True)
|
|
except subprocess.CalledProcessError as e:
|
|
print("The command", f"'{' '.join(cmd)}'", "failed.")
|
|
print("The output printed to stdout was:")
|
|
print(e.stdout)
|
|
raise
|
|
|
|
def fetch_version_number(major_version=None):
|
|
"""Retrieve version number for release from Azure Pipelines
|
|
|
|
:param major_version: only consider releases for the specified major
|
|
version
|
|
:type major_version: str or None
|
|
|
|
:returns: version number
|
|
|
|
"""
|
|
# Create a connection to the azure org
|
|
organization_url = 'https://dev.azure.com/certbot'
|
|
connection = Connection(base_url=organization_url)
|
|
|
|
# Find the build artifacts
|
|
build_client = connection.clients.get_build_client()
|
|
builds = build_client.get_builds('certbot', definitions='3')
|
|
for build in builds:
|
|
version = build_client.get_build('certbot', build.id).source_branch.split('v')[1]
|
|
if major_version is None or version.split('.')[0] == major_version:
|
|
return version
|
|
raise ValueError('Release not found on Azure Pipelines!')
|
|
|
|
def main(args):
|
|
parsed_args = parse_args(args)
|
|
version = fetch_version_number()
|
|
promote_snaps(ALL_SNAPS, 'beta', version)
|
|
|
|
if __name__ == "__main__":
|
|
main(sys.argv[1:])
|