935 lines
36 KiB
Python
Executable File
935 lines
36 KiB
Python
Executable File
# Licensed to Elasticsearch under one or more contributor
|
|
# license agreements. See the NOTICE file distributed with
|
|
# this work for additional information regarding copyright
|
|
# ownership. Elasticsearch licenses this file to you 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 re
|
|
import tempfile
|
|
import shutil
|
|
import os
|
|
import datetime
|
|
import argparse
|
|
import github3
|
|
import smtplib
|
|
import subprocess
|
|
import sys
|
|
|
|
from functools import partial
|
|
|
|
from email.mime.multipart import MIMEMultipart
|
|
from email.mime.text import MIMEText
|
|
|
|
from os.path import dirname, abspath
|
|
|
|
"""
|
|
This tool builds a release from the a given elasticsearch plugin branch.
|
|
In order to execute it go in the top level directory and run:
|
|
$ python3 dev_tools/build_release.py --branch master --publish --remote origin
|
|
|
|
By default this script runs in 'dry' mode which essentially simulates a release. If the
|
|
'--publish' option is set the actual release is done.
|
|
If not in 'dry' mode, a mail will be automatically sent to the mailing list.
|
|
You can disable it with the option '--disable_mail'
|
|
|
|
$ python3 dev_tools/build_release.py --publish --remote origin --disable_mail
|
|
|
|
The script takes over almost all
|
|
steps necessary for a release from a high level point of view it does the following things:
|
|
|
|
- run prerequisite checks ie. check for S3 credentials available as env variables
|
|
- detect the version to release from the specified branch (--branch) or the current branch
|
|
- check that github issues related to the version are closed
|
|
- creates a version release branch & updates pom.xml to point to a release version rather than a snapshot
|
|
- creates a master release branch & updates README.md to point to the latest release version for the given elasticsearch branch
|
|
- builds the artifacts
|
|
- commits the new version and merges the version release branch into the source branch
|
|
- merges the master release branch into the master branch
|
|
- creates a tag and pushes branch and master to the specified origin (--remote)
|
|
- publishes the releases to sonatype
|
|
- publishes the releases to S3 using s3cmd
|
|
- send a mail based on github issues fixed by this version
|
|
|
|
Once it's done it will print all the remaining steps.
|
|
|
|
Prerequisites:
|
|
- s3cmd needs to be installed (this should work: s3cmd ls s3://download.elasticsearch.org/elasticsearch)
|
|
- Python 3k for script execution
|
|
- github3 module (pip-3.3 install github3.py)
|
|
- GITHUB (login/password) or key exported via ENV Variables (GITHUB_LOGIN, GITHUB_PASSWORD or GITHUB_KEY)
|
|
(see https://github.com/settings/applications#personal-access-tokens) - Optional: default to no authentication
|
|
- SMTP_HOST - Optional: default to localhost
|
|
- MAIL_SENDER - Optional: default to 'david@pilato.fr': must be authorized to send emails to elasticsearch mailing list
|
|
- MAIL_TO - Optional: default to 'discuss+announcements@elastic.co'
|
|
"""
|
|
env = os.environ
|
|
|
|
LOG = env.get('ES_RELEASE_LOG', '/tmp/elasticsearch_release.log')
|
|
ROOT_DIR = abspath(os.path.join(abspath(dirname(__file__)), '../'))
|
|
README_FILE = ROOT_DIR + '/README.md'
|
|
POM_FILE = ROOT_DIR + '/pom.xml'
|
|
DEV_TOOLS_DIR = ROOT_DIR + '/dev-tools'
|
|
|
|
# console colors
|
|
OKGREEN = '\033[92m'
|
|
ENDC = '\033[0m'
|
|
FAIL = '\033[91m'
|
|
|
|
##########################################################
|
|
#
|
|
# Utility methods (log and run)
|
|
#
|
|
##########################################################
|
|
# Log a message
|
|
def log(msg):
|
|
log_plain('\n%s' % msg)
|
|
|
|
|
|
# Purge the log file
|
|
def purge_log():
|
|
try:
|
|
os.remove(LOG)
|
|
except FileNotFoundError:
|
|
pass
|
|
|
|
|
|
# Log a message to the LOG file
|
|
def log_plain(msg):
|
|
f = open(LOG, mode='ab')
|
|
f.write(msg.encode('utf-8'))
|
|
f.close()
|
|
|
|
|
|
# Run a command and log it
|
|
def run(command, quiet=False):
|
|
log('%s: RUN: %s\n' % (datetime.datetime.now(), command))
|
|
if os.system('%s >> %s 2>&1' % (command, LOG)):
|
|
msg = ' FAILED: %s [see log %s]' % (command, LOG)
|
|
if not quiet:
|
|
print(msg)
|
|
raise RuntimeError(msg)
|
|
|
|
##########################################################
|
|
#
|
|
# Clean logs and check JAVA and Maven
|
|
#
|
|
##########################################################
|
|
try:
|
|
purge_log()
|
|
JAVA_HOME = env['JAVA_HOME']
|
|
except KeyError:
|
|
raise RuntimeError("""
|
|
Please set JAVA_HOME in the env before running release tool
|
|
On OSX use: export JAVA_HOME=`/usr/libexec/java_home -v '1.7*'`""")
|
|
|
|
try:
|
|
MVN = 'mvn'
|
|
# make sure mvn3 is used if mvn3 is available
|
|
# some systems use maven 2 as default
|
|
run('mvn3 --version', quiet=True)
|
|
MVN = 'mvn3'
|
|
except RuntimeError:
|
|
pass
|
|
|
|
|
|
def java_exe():
|
|
path = JAVA_HOME
|
|
return 'export JAVA_HOME="%s" PATH="%s/bin:$PATH" JAVACMD="%s/bin/java"' % (path, path, path)
|
|
|
|
|
|
def verify_java_version(version):
|
|
s = os.popen('%s; java -version 2>&1' % java_exe()).read()
|
|
if ' version "%s.' % version not in s:
|
|
raise RuntimeError('got wrong version for java %s:\n%s' % (version, s))
|
|
|
|
|
|
def verify_mvn_java_version(version, mvn):
|
|
s = os.popen('%s; %s --version 2>&1' % (java_exe(), mvn)).read()
|
|
if 'Java version: %s' % version not in s:
|
|
raise RuntimeError('got wrong java version for %s %s:\n%s' % (mvn, version, s))
|
|
|
|
|
|
##########################################################
|
|
#
|
|
# String and file manipulation utils
|
|
#
|
|
##########################################################
|
|
# Utility that returns the name of the release branch for a given version
|
|
def release_branch(branchsource, version):
|
|
return 'release_branch_%s_%s' % (branchsource, version)
|
|
|
|
|
|
# Reads the given file and applies the
|
|
# callback to it. If the callback changed
|
|
# a line the given file is replaced with
|
|
# the modified input.
|
|
def process_file(file_path, line_callback):
|
|
fh, abs_path = tempfile.mkstemp()
|
|
modified = False
|
|
with open(abs_path, 'w', encoding='utf-8') as new_file:
|
|
with open(file_path, encoding='utf-8') as old_file:
|
|
for line in old_file:
|
|
new_line = line_callback(line)
|
|
modified = modified or (new_line != line)
|
|
new_file.write(new_line)
|
|
os.close(fh)
|
|
if modified:
|
|
#Remove original file
|
|
os.remove(file_path)
|
|
#Move new file
|
|
shutil.move(abs_path, file_path)
|
|
return True
|
|
else:
|
|
# nothing to do - just remove the tmp file
|
|
os.remove(abs_path)
|
|
return False
|
|
|
|
|
|
# Split a version x.y.z as an array of digits [x,y,z]
|
|
def split_version_to_digits(version):
|
|
return list(map(int, re.findall(r'\d+', version)))
|
|
|
|
|
|
# Guess the next snapshot version number (increment last digit)
|
|
def guess_snapshot(version):
|
|
digits = split_version_to_digits(version)
|
|
source = '%s.%s.%s' % (digits[0], digits[1], digits[2])
|
|
destination = '%s.%s.%s' % (digits[0], digits[1], digits[2] + 1)
|
|
return version.replace(source, destination)
|
|
|
|
|
|
# Guess the anchor in generated documentation
|
|
# Looks like this "#version-230-for-elasticsearch-13"
|
|
def get_doc_anchor(release, esversion):
|
|
plugin_digits = split_version_to_digits(release)
|
|
es_digits = split_version_to_digits(esversion)
|
|
return '#version-%s%s%s-for-elasticsearch-%s%s' % (
|
|
plugin_digits[0], plugin_digits[1], plugin_digits[2], es_digits[0], es_digits[1])
|
|
|
|
|
|
# Moves the pom.xml file from a snapshot to a release
|
|
def remove_maven_snapshot(pom, release):
|
|
pattern = '<version>%s-SNAPSHOT</version>' % release
|
|
replacement = '<version>%s</version>' % release
|
|
|
|
def callback(line):
|
|
return line.replace(pattern, replacement)
|
|
|
|
process_file(pom, callback)
|
|
|
|
|
|
# Moves the pom.xml file to the next snapshot
|
|
def add_maven_snapshot(pom, release, snapshot):
|
|
pattern = '<version>%s</version>' % release
|
|
replacement = '<version>%s-SNAPSHOT</version>' % snapshot
|
|
|
|
def callback(line):
|
|
return line.replace(pattern, replacement)
|
|
|
|
process_file(pom, callback)
|
|
|
|
|
|
# Moves the README.md file from a snapshot to a release version. Doc looks like:
|
|
# ## Version 2.5.0-SNAPSHOT for Elasticsearch: 1.x
|
|
# It needs to be updated to
|
|
# ## Version 2.5.0 for Elasticsearch: 1.x
|
|
def update_documentation_in_released_branch(readme_file, release, esversion):
|
|
pattern = '## Version (.)+ for Elasticsearch: (.)+'
|
|
es_digits = split_version_to_digits(esversion)
|
|
replacement = '## Version %s for Elasticsearch: %s.%s\n' % (
|
|
release, es_digits[0], es_digits[1])
|
|
|
|
def callback(line):
|
|
# If we find pattern, we replace its content
|
|
if re.search(pattern, line) is not None:
|
|
return replacement
|
|
else:
|
|
return line
|
|
|
|
process_file(readme_file, callback)
|
|
|
|
|
|
# Moves the README.md file from a snapshot to a release (documentation link)
|
|
# We need to find the right branch we are on and update the line
|
|
# | es-1.3 | Build from source | [2.4.0-SNAPSHOT](https://github.com/elasticsearch/elasticsearch-cloud-azure/tree/es-1.3/#version-240-snapshot-for-elasticsearch-13) |
|
|
# | es-1.2 | 2.3.0 | [2.3.0](https://github.com/elasticsearch/elasticsearch-cloud-azure/tree/v2.3.0/#version-230-snapshot-for-elasticsearch-13) |
|
|
def update_documentation_to_released_version(readme_file, repo_url, release, branch, esversion):
|
|
pattern = '%s' % branch
|
|
replacement = '| %s | %s | [%s](%stree/v%s/%s) |\n' % (
|
|
branch, release, release, repo_url, release, get_doc_anchor(release, esversion))
|
|
|
|
def callback(line):
|
|
# If we find pattern, we replace its content
|
|
if line.find(pattern) >= 0:
|
|
return replacement
|
|
else:
|
|
return line
|
|
|
|
process_file(readme_file, callback)
|
|
|
|
|
|
# Update installation instructions in README.md file
|
|
def set_install_instructions(readme_file, artifact_name, release):
|
|
pattern = 'bin/plugin -?install elasticsearch/%s/.+' % artifact_name
|
|
replacement = 'bin/plugin install elasticsearch/%s/%s' % (artifact_name, release)
|
|
|
|
def callback(line):
|
|
return re.sub(pattern, replacement, line)
|
|
|
|
process_file(readme_file, callback)
|
|
|
|
|
|
# Checks the pom.xml for the release version. <version>2.0.0-SNAPSHOT</version>
|
|
# This method fails if the pom file has no SNAPSHOT version set ie.
|
|
# if the version is already on a release version we fail.
|
|
# Returns the next version string ie. 0.90.7
|
|
def find_release_version(src_branch):
|
|
git_checkout(src_branch)
|
|
with open(POM_FILE, encoding='utf-8') as file:
|
|
for line in file:
|
|
match = re.search(r'<version>(.+)-SNAPSHOT</version>', line)
|
|
if match:
|
|
return match.group(1)
|
|
raise RuntimeError('Could not find release version in branch %s' % src_branch)
|
|
|
|
|
|
# extract a value from pom.xml after a given line
|
|
def find_from_pom(tag, first_line=None):
|
|
with open(POM_FILE, encoding='utf-8') as file:
|
|
previous_line_matched = False
|
|
if first_line is None:
|
|
previous_line_matched = True
|
|
for line in file:
|
|
if previous_line_matched:
|
|
match = re.search(r'<%s>(.+)</%s>' % (tag, tag), line)
|
|
if match:
|
|
return match.group(1)
|
|
|
|
if first_line is not None:
|
|
match = re.search(r'%s' % first_line, line)
|
|
if match:
|
|
previous_line_matched = True
|
|
|
|
if first_line is not None:
|
|
raise RuntimeError('Could not find %s in pom.xml file after %s' % (tag, first_line))
|
|
else:
|
|
raise RuntimeError('Could not find %s in pom.xml file' % tag)
|
|
|
|
|
|
# Get artifacts which have been generated in target/releases
|
|
def get_artifacts(artifact_id, release):
|
|
artifact_path = ROOT_DIR + '/target/releases/%s-%s.zip' % (artifact_id, release)
|
|
print(' Path %s' % artifact_path)
|
|
if not os.path.isfile(artifact_path):
|
|
raise RuntimeError('Could not find required artifact at %s' % artifact_path)
|
|
return artifact_path
|
|
|
|
|
|
# Read sha1 / md5 files from local maven repository
|
|
# and returns the checksum files as well
|
|
# as the given files in a list
|
|
def read_checksums(release_file, artifact_id, version):
|
|
res = []
|
|
dirname = os.path.normpath(os.path.expanduser('~/.m2/repository/org/elasticsearch/%s/%s/' % (artifact_id, version)))
|
|
directory = os.path.abspath(release_file)
|
|
file = os.path.basename(release_file)
|
|
checksum_file_sha1 = '%s/%s.sha1' % (dirname, file)
|
|
checksum_file_md5 = '%s/%s.md5' % (dirname, file)
|
|
|
|
if not os.path.exists(dirname):
|
|
raise RuntimeError('Failed to read checksums from %s' % dirname)
|
|
if not os.path.isfile(checksum_file_sha1):
|
|
raise RuntimeError('Failed to read checksum file %s' % checksum_file_sha1)
|
|
if not os.path.isfile(checksum_file_md5):
|
|
raise RuntimeError('Failed to read checksum file %s' % checksum_file_md5)
|
|
|
|
res += [os.path.join(directory, checksum_file_md5), os.path.join(directory, checksum_file_sha1), release_file]
|
|
return res
|
|
|
|
|
|
# Format a GitHub issue as plain text
|
|
def format_issues_plain(issues, title='Fix'):
|
|
response = ""
|
|
|
|
if len(issues) > 0:
|
|
response += '%s:\n\n' % title
|
|
for issue in issues:
|
|
response += ' * [%s] - %s ( %s )\n' % (issue.number, issue.title, issue.html_url)
|
|
|
|
return response
|
|
|
|
|
|
# Format a GitHub issue as html text
|
|
def format_issues_html(issues, title='Fix'):
|
|
response = ""
|
|
|
|
if len(issues) > 0:
|
|
response += '<h2>%s</h2>\n<ul>\n' % title
|
|
for issue in issues:
|
|
response += '<li>[<a href="%s">%s</a>] - %s\n' % (issue.html_url, issue.number, issue.title)
|
|
response += '</ul>\n'
|
|
|
|
return response
|
|
|
|
|
|
##########################################################
|
|
#
|
|
# GIT commands
|
|
#
|
|
##########################################################
|
|
# Returns the hash of the current git HEAD revision
|
|
def get_head_hash():
|
|
return os.popen('git rev-parse --verify HEAD 2>&1').read().strip()
|
|
|
|
|
|
# Returns the name of the current branch
|
|
def get_current_branch():
|
|
return os.popen('git rev-parse --abbrev-ref HEAD 2>&1').read().strip()
|
|
|
|
|
|
# runs get fetch on the given remote
|
|
def fetch(remote):
|
|
run('git fetch %s' % remote)
|
|
|
|
|
|
# Creates a new release branch from the given source branch
|
|
# and rebases the source branch from the remote before creating
|
|
# the release branch. Note: This fails if the source branch
|
|
# doesn't exist on the provided remote.
|
|
def create_release_branch(remote, src_branch, release):
|
|
git_checkout(src_branch)
|
|
run('git pull --rebase %s %s' % (remote, src_branch))
|
|
run('git checkout -b %s' % (release_branch(src_branch, release)))
|
|
|
|
|
|
# Stages the given files for the next git commit
|
|
def add_pending_files(*files):
|
|
for file in files:
|
|
run('git add %s' % file)
|
|
|
|
|
|
# Executes a git commit with 'release [version]' as the commit message
|
|
def commit_release(artifact_id, release):
|
|
run('git commit -m "prepare release %s-%s"' % (artifact_id, release))
|
|
|
|
|
|
# Commit documentation changes on the master branch
|
|
def commit_master(release):
|
|
run('git commit -m "update documentation with release %s"' % release)
|
|
|
|
|
|
# Commit next snapshot files
|
|
def commit_snapshot():
|
|
run('git commit -m "prepare for next development iteration"')
|
|
|
|
|
|
# Put the version tag on on the current commit
|
|
def tag_release(release):
|
|
run('git tag -a v%s -m "Tag release version %s"' % (release, release))
|
|
|
|
|
|
# Checkout a given branch
|
|
def git_checkout(branch):
|
|
run('git checkout %s' % branch)
|
|
|
|
|
|
# Merge the release branch with the actual branch
|
|
def git_merge(src_branch, release_version):
|
|
git_checkout(src_branch)
|
|
run('git merge %s' % release_branch(src_branch, release_version))
|
|
|
|
|
|
# Push the actual branch and master branch
|
|
def git_push(remote, src_branch, release_version, dry_run):
|
|
if not dry_run:
|
|
run('git push %s %s master' % (remote, src_branch)) # push the commit and the master
|
|
run('git push %s v%s' % (remote, release_version)) # push the tag
|
|
else:
|
|
print(' dryrun [True] -- skipping push to remote %s %s master' % (remote, src_branch))
|
|
|
|
|
|
##########################################################
|
|
#
|
|
# Maven commands
|
|
#
|
|
##########################################################
|
|
# Run a given maven command
|
|
def run_mvn(*cmd):
|
|
for c in cmd:
|
|
run('%s; %s -f %s %s' % (java_exe(), MVN, POM_FILE, c))
|
|
|
|
|
|
# Run deploy or package depending on dry_run
|
|
# Default to run mvn package
|
|
# When run_tests=True a first mvn clean test is run
|
|
def build_release(run_tests=False, dry_run=True):
|
|
target = 'deploy'
|
|
tests = '-DskipTests'
|
|
if run_tests:
|
|
tests = ''
|
|
if dry_run:
|
|
target = 'install'
|
|
run_mvn('clean %s %s' % (target, tests))
|
|
|
|
|
|
##########################################################
|
|
#
|
|
# Amazon S3 publish commands
|
|
#
|
|
##########################################################
|
|
# Upload files to S3
|
|
def publish_artifacts(artifacts, base='elasticsearch/elasticsearch', dry_run=True):
|
|
for artifact in artifacts:
|
|
if dry_run:
|
|
print('Skip Uploading %s to Amazon S3 in %s' % (artifact, base))
|
|
else:
|
|
print('Uploading %s to Amazon S3' % artifact)
|
|
run('s3cmd put %s s3://download.elasticsearch.org/%s/' % (os.path.abspath(artifact), base))
|
|
|
|
|
|
##########################################################
|
|
#
|
|
# Email and Github Management
|
|
#
|
|
##########################################################
|
|
# Create a Github repository instance to access issues
|
|
def get_github_repository(reponame,
|
|
login=env.get('GITHUB_LOGIN', None),
|
|
password=env.get('GITHUB_PASSWORD', None),
|
|
key=env.get('GITHUB_KEY', None)):
|
|
if login:
|
|
g = github3.login(login, password)
|
|
elif key:
|
|
g = github3.login(token=key)
|
|
else:
|
|
g = github3.GitHub()
|
|
|
|
return g.repository("elastic", reponame)
|
|
|
|
|
|
# Check if there are some remaining open issues and fails
|
|
def check_opened_issues(version, repository, reponame):
|
|
opened_issues = [i for i in repository.iter_issues(state='open', labels='%s' % version)]
|
|
if len(opened_issues) > 0:
|
|
raise NameError(
|
|
'Some issues [%s] are still opened. Check https://github.com/elasticsearch/%s/issues?labels=%s&state=open'
|
|
% (len(opened_issues), reponame, version))
|
|
|
|
|
|
# List issues from github: can be done anonymously if you don't
|
|
# exceed a given number of github API calls per day
|
|
def list_issues(version,
|
|
repository,
|
|
severity='bug'):
|
|
issues = [i for i in repository.iter_issues(state='closed', labels='%s,%s' % (severity, version))]
|
|
return issues
|
|
|
|
|
|
def read_email_template(format='html'):
|
|
file_name = '%s/email_template.%s' % (DEV_TOOLS_DIR, format)
|
|
log('open email template %s' % file_name)
|
|
with open(file_name, encoding='utf-8') as template_file:
|
|
data = template_file.read()
|
|
return data
|
|
|
|
|
|
# Read template messages
|
|
template_email_html = read_email_template('html')
|
|
template_email_txt = read_email_template('txt')
|
|
|
|
|
|
# Get issues from github and generates a Plain/HTML Multipart email
|
|
def prepare_email(artifact_id, release_version, repository,
|
|
artifact_name, artifact_description, project_url,
|
|
severity_labels_bug='bug',
|
|
severity_labels_update='update',
|
|
severity_labels_new='new',
|
|
severity_labels_doc='doc'):
|
|
## Get bugs from github
|
|
issues_bug = list_issues(release_version, repository, severity=severity_labels_bug)
|
|
issues_update = list_issues(release_version, repository, severity=severity_labels_update)
|
|
issues_new = list_issues(release_version, repository, severity=severity_labels_new)
|
|
issues_doc = list_issues(release_version, repository, severity=severity_labels_doc)
|
|
|
|
## Format content to plain text
|
|
plain_issues_bug = format_issues_plain(issues_bug, 'Fix')
|
|
plain_issues_update = format_issues_plain(issues_update, 'Update')
|
|
plain_issues_new = format_issues_plain(issues_new, 'New')
|
|
plain_issues_doc = format_issues_plain(issues_doc, 'Doc')
|
|
|
|
## Format content to html
|
|
html_issues_bug = format_issues_html(issues_bug, 'Fix')
|
|
html_issues_update = format_issues_html(issues_update, 'Update')
|
|
html_issues_new = format_issues_html(issues_new, 'New')
|
|
html_issues_doc = format_issues_html(issues_doc, 'Doc')
|
|
|
|
if len(issues_bug) + len(issues_update) + len(issues_new) + len(issues_doc) > 0:
|
|
plain_empty_message = ""
|
|
html_empty_message = ""
|
|
|
|
else:
|
|
plain_empty_message = "No issue listed for this release"
|
|
html_empty_message = "<p>No issue listed for this release</p>"
|
|
|
|
msg = MIMEMultipart('alternative')
|
|
msg['Subject'] = '[ANN] %s %s released' % (artifact_name, release_version)
|
|
text = template_email_txt % {'release_version': release_version,
|
|
'artifact_id': artifact_id,
|
|
'artifact_name': artifact_name,
|
|
'artifact_description': artifact_description,
|
|
'project_url': project_url,
|
|
'empty_message': plain_empty_message,
|
|
'issues_bug': plain_issues_bug,
|
|
'issues_update': plain_issues_update,
|
|
'issues_new': plain_issues_new,
|
|
'issues_doc': plain_issues_doc}
|
|
|
|
html = template_email_html % {'release_version': release_version,
|
|
'artifact_id': artifact_id,
|
|
'artifact_name': artifact_name,
|
|
'artifact_description': artifact_description,
|
|
'project_url': project_url,
|
|
'empty_message': html_empty_message,
|
|
'issues_bug': html_issues_bug,
|
|
'issues_update': html_issues_update,
|
|
'issues_new': html_issues_new,
|
|
'issues_doc': html_issues_doc}
|
|
|
|
# Record the MIME types of both parts - text/plain and text/html.
|
|
part1 = MIMEText(text, 'plain')
|
|
part2 = MIMEText(html, 'html')
|
|
|
|
# Attach parts into message container.
|
|
# According to RFC 2046, the last part of a multipart message, in this case
|
|
# the HTML message, is best and preferred.
|
|
msg.attach(part1)
|
|
msg.attach(part2)
|
|
|
|
return msg
|
|
|
|
|
|
def send_email(msg,
|
|
dry_run=True,
|
|
mail=True,
|
|
sender=env.get('MAIL_SENDER'),
|
|
to=env.get('MAIL_TO', 'discuss+announcements@elastic.co'),
|
|
smtp_server=env.get('SMTP_SERVER', 'localhost')):
|
|
msg['From'] = 'Elasticsearch Team <%s>' % sender
|
|
msg['To'] = 'Elasticsearch Announcement List <%s>' % to
|
|
# save mail on disk
|
|
with open(ROOT_DIR + '/target/email.txt', 'w') as email_file:
|
|
email_file.write(msg.as_string())
|
|
if mail and not dry_run:
|
|
s = smtplib.SMTP(smtp_server, 25)
|
|
s.sendmail(sender, to, msg.as_string())
|
|
s.quit()
|
|
else:
|
|
print('generated email: open %s/target/email.txt' % ROOT_DIR)
|
|
print(msg.as_string())
|
|
|
|
|
|
def print_sonatype_notice():
|
|
settings = os.path.join(os.path.expanduser('~'), '.m2/settings.xml')
|
|
if os.path.isfile(settings):
|
|
with open(settings, encoding='utf-8') as settings_file:
|
|
for line in settings_file:
|
|
if line.strip() == '<id>sonatype-nexus-snapshots</id>':
|
|
# moving out - we found the indicator no need to print the warning
|
|
return
|
|
print("""
|
|
NOTE: No sonatype settings detected, make sure you have configured
|
|
your sonatype credentials in '~/.m2/settings.xml':
|
|
|
|
<settings>
|
|
...
|
|
<servers>
|
|
<server>
|
|
<id>sonatype-nexus-snapshots</id>
|
|
<username>your-jira-id</username>
|
|
<password>your-jira-pwd</password>
|
|
</server>
|
|
<server>
|
|
<id>sonatype-nexus-staging</id>
|
|
<username>your-jira-id</username>
|
|
<password>your-jira-pwd</password>
|
|
</server>
|
|
</servers>
|
|
...
|
|
</settings>
|
|
""")
|
|
|
|
|
|
def check_s3_credentials():
|
|
if not env.get('AWS_ACCESS_KEY_ID', None) or not env.get('AWS_SECRET_ACCESS_KEY', None):
|
|
raise RuntimeError(
|
|
'Could not find "AWS_ACCESS_KEY_ID" / "AWS_SECRET_ACCESS_KEY" in the env variables please export in order to upload to S3')
|
|
|
|
|
|
def check_github_credentials():
|
|
if not env.get('GITHUB_KEY', None) and not env.get('GITHUB_LOGIN', None):
|
|
log(
|
|
'WARN: Could not find "GITHUB_LOGIN" / "GITHUB_PASSWORD" or "GITHUB_KEY" in the env variables. You could need it.')
|
|
|
|
|
|
def check_email_settings():
|
|
if not env.get('MAIL_SENDER', None):
|
|
raise RuntimeError('Could not find "MAIL_SENDER"')
|
|
|
|
|
|
def check_command_exists(name, cmd):
|
|
try:
|
|
print('%s' % cmd)
|
|
subprocess.check_output(cmd, shell=True, stderr=subprocess.STDOUT)
|
|
except subprocess.CalledProcessError:
|
|
raise RuntimeError('Could not run command %s - please make sure it is installed' % (name))
|
|
|
|
|
|
def run_and_print(text, run_function):
|
|
try:
|
|
print(text, end='')
|
|
run_function()
|
|
print(OKGREEN + 'OK' + ENDC)
|
|
except RuntimeError:
|
|
print(FAIL + 'NOT OK' + ENDC)
|
|
|
|
def check_env_var(text, env_var):
|
|
try:
|
|
print(text, end='')
|
|
env[env_var]
|
|
print(OKGREEN + 'OK' + ENDC)
|
|
except KeyError:
|
|
print(FAIL + 'NOT OK' + ENDC)
|
|
|
|
|
|
def check_environment_and_commandline_tools():
|
|
#check_env_var('Checking for AWS env configuration AWS_SECRET_ACCESS_KEY_ID... ', 'AWS_SECRET_ACCESS_KEY')
|
|
#check_env_var('Checking for AWS env configuration AWS_ACCESS_KEY_ID... ', 'AWS_ACCESS_KEY_ID')
|
|
# check_env_var('Checking for SONATYPE env configuration SONATYPE_USERNAME... ', 'SONATYPE_USERNAME')
|
|
# check_env_var('Checking for SONATYPE env configuration SONATYPE_PASSWORD... ', 'SONATYPE_PASSWORD')
|
|
# check_env_var('Checking for GPG env configuration GPG_KEY_ID... ', 'GPG_KEY_ID')
|
|
# check_env_var('Checking for GPG env configuration GPG_PASSPHRASE... ', 'GPG_PASSPHRASE')
|
|
# check_env_var('Checking for S3 repo upload env configuration S3_BUCKET_SYNC_TO... ', 'S3_BUCKET_SYNC_TO')
|
|
# check_env_var('Checking for git env configuration GIT_AUTHOR_NAME... ', 'GIT_AUTHOR_NAME')
|
|
# check_env_var('Checking for git env configuration GIT_AUTHOR_EMAIL... ', 'GIT_AUTHOR_EMAIL')
|
|
|
|
run_and_print('Checking command: gpg... ', partial(check_command_exists, 'gpg', 'gpg --version'))
|
|
run_and_print('Checking command: expect... ', partial(check_command_exists, 'expect', 'expect -v'))
|
|
run_and_print('Checking command: s3cmd... ', partial(check_command_exists, 's3cmd', 's3cmd --version'))
|
|
|
|
run_and_print('Checking java version... ', partial(verify_java_version, '1.7'))
|
|
run_and_print('Checking java mvn version... ', partial(verify_mvn_java_version, '1.7', MVN))
|
|
|
|
|
|
if __name__ == '__main__':
|
|
parser = argparse.ArgumentParser(description='Builds and publishes a Elasticsearch Plugin Release')
|
|
parser.add_argument('--branch', '-b', metavar='master', default=get_current_branch(),
|
|
help='The branch to release from. Defaults to the current branch.')
|
|
parser.add_argument('--skiptests', '-t', dest='tests', action='store_false',
|
|
help='Skips tests before release. Tests are run by default.')
|
|
parser.set_defaults(tests=True)
|
|
parser.add_argument('--remote', '-r', metavar='origin', default='origin',
|
|
help='The remote to push the release commit and tag to. Default is [origin]')
|
|
parser.add_argument('--publish', '-p', dest='dryrun', action='store_false',
|
|
help='Publishes the release. Disable by default.')
|
|
parser.add_argument('--disable_mail', '-dm', dest='mail', action='store_false',
|
|
help='Do not send a release email. Email is sent by default.')
|
|
parser.add_argument('--check', dest='check', action='store_true',
|
|
help='Checks and reports for all requirements and then exits')
|
|
|
|
parser.set_defaults(dryrun=True)
|
|
parser.set_defaults(mail=True)
|
|
parser.set_defaults(check=False)
|
|
args = parser.parse_args()
|
|
|
|
src_branch = args.branch
|
|
remote = args.remote
|
|
run_tests = args.tests
|
|
dry_run = args.dryrun
|
|
mail = args.mail
|
|
|
|
if args.check:
|
|
check_environment_and_commandline_tools()
|
|
sys.exit(0)
|
|
|
|
if src_branch == 'master':
|
|
raise RuntimeError('Can not release the master branch. You need to create another branch before a release')
|
|
|
|
# we print a notice if we can not find the relevant infos in the ~/.m2/settings.xml
|
|
print_sonatype_notice()
|
|
|
|
if not dry_run:
|
|
check_s3_credentials()
|
|
print('WARNING: dryrun is set to "false" - this will push and publish the release')
|
|
if mail:
|
|
check_email_settings()
|
|
print('An email to %s will be sent after the release'
|
|
% env.get('MAIL_TO', 'discuss+announcements@elastic.co'))
|
|
input('Press Enter to continue...')
|
|
|
|
check_github_credentials()
|
|
|
|
print(''.join(['-' for _ in range(80)]))
|
|
print('Preparing Release from branch [%s] running tests: [%s] dryrun: [%s]' % (src_branch, run_tests, dry_run))
|
|
print(' JAVA_HOME is [%s]' % JAVA_HOME)
|
|
print(' Running with maven command: [%s] ' % (MVN))
|
|
|
|
release_version = find_release_version(src_branch)
|
|
artifact_id = find_from_pom('artifactId')
|
|
artifact_name = find_from_pom('name')
|
|
artifact_description = find_from_pom('description')
|
|
project_url = find_from_pom('url')
|
|
|
|
try:
|
|
elasticsearch_version = find_from_pom('elasticsearch.version')
|
|
except RuntimeError:
|
|
# With projects using elasticsearch-parent project, we need to consider elasticsearch version
|
|
# to be after <artifactId>elasticsearch-parent</artifactId>
|
|
elasticsearch_version = find_from_pom('version', '<artifactId>elasticsearch-parent</artifactId>')
|
|
|
|
print(' Artifact Id: [%s]' % artifact_id)
|
|
print(' Release version: [%s]' % release_version)
|
|
print(' Elasticsearch: [%s]' % elasticsearch_version)
|
|
if elasticsearch_version.find('-SNAPSHOT') != -1:
|
|
raise RuntimeError('Can not release with a SNAPSHOT elasticsearch dependency: %s' % elasticsearch_version)
|
|
|
|
# extract snapshot
|
|
default_snapshot_version = guess_snapshot(release_version)
|
|
snapshot_version = input('Enter next snapshot version [%s]:' % default_snapshot_version)
|
|
snapshot_version = snapshot_version or default_snapshot_version
|
|
|
|
print(' Next version: [%s-SNAPSHOT]' % snapshot_version)
|
|
print(' Artifact Name: [%s]' % artifact_name)
|
|
print(' Artifact Description: [%s]' % artifact_description)
|
|
print(' Project URL: [%s]' % project_url)
|
|
|
|
if not dry_run:
|
|
smoke_test_version = release_version
|
|
|
|
try:
|
|
git_checkout('master')
|
|
master_hash = get_head_hash()
|
|
git_checkout(src_branch)
|
|
version_hash = get_head_hash()
|
|
run_mvn('clean') # clean the env!
|
|
create_release_branch(remote, 'master', release_version)
|
|
print(' Created release branch [%s]' % (release_branch('master', release_version)))
|
|
create_release_branch(remote, src_branch, release_version)
|
|
print(' Created release branch [%s]' % (release_branch(src_branch, release_version)))
|
|
except RuntimeError:
|
|
print('Logs:')
|
|
with open(LOG, 'r') as log_file:
|
|
print(log_file.read())
|
|
sys.exit(-1)
|
|
|
|
success = False
|
|
try:
|
|
########################################
|
|
# Start update process in version branch
|
|
########################################
|
|
pending_files = [POM_FILE, README_FILE]
|
|
remove_maven_snapshot(POM_FILE, release_version)
|
|
update_documentation_in_released_branch(README_FILE, release_version, elasticsearch_version)
|
|
print(' Done removing snapshot version')
|
|
add_pending_files(*pending_files) # expects var args use * to expand
|
|
commit_release(artifact_id, release_version)
|
|
print(' Committed release version [%s]' % release_version)
|
|
print(''.join(['-' for _ in range(80)]))
|
|
print('Building Release candidate')
|
|
input('Press Enter to continue...')
|
|
print(' Checking github issues')
|
|
repository = get_github_repository(artifact_id)
|
|
check_opened_issues(release_version, repository, artifact_id)
|
|
if not dry_run:
|
|
print(' Running maven builds now and publish to sonatype - run-tests [%s]' % run_tests)
|
|
else:
|
|
print(' Running maven builds now run-tests [%s]' % run_tests)
|
|
build_release(run_tests=run_tests, dry_run=dry_run)
|
|
artifact = get_artifacts(artifact_id, release_version)
|
|
artifact_and_checksums = read_checksums(artifact, artifact_id, release_version)
|
|
print(''.join(['-' for _ in range(80)]))
|
|
|
|
########################################
|
|
# Start update process in master branch
|
|
########################################
|
|
git_checkout(release_branch('master', release_version))
|
|
update_documentation_to_released_version(README_FILE, project_url, release_version, src_branch,
|
|
elasticsearch_version)
|
|
set_install_instructions(README_FILE, artifact_id, release_version)
|
|
add_pending_files(*pending_files) # expects var args use * to expand
|
|
commit_master(release_version)
|
|
|
|
print('Finish Release -- dry_run: %s' % dry_run)
|
|
input('Press Enter to continue...')
|
|
|
|
print(' merge release branch')
|
|
git_merge(src_branch, release_version)
|
|
print(' tag')
|
|
tag_release(release_version)
|
|
|
|
add_maven_snapshot(POM_FILE, release_version, snapshot_version)
|
|
update_documentation_in_released_branch(README_FILE, '%s-SNAPSHOT' % snapshot_version, elasticsearch_version)
|
|
add_pending_files(*pending_files)
|
|
commit_snapshot()
|
|
|
|
print(' merge master branch')
|
|
git_merge('master', release_version)
|
|
|
|
print(' push to %s %s -- dry_run: %s' % (remote, src_branch, dry_run))
|
|
git_push(remote, src_branch, release_version, dry_run)
|
|
print(' publish artifacts to S3 -- dry_run: %s' % dry_run)
|
|
publish_artifacts(artifact_and_checksums, base='elasticsearch/%s' % (artifact_id) , dry_run=dry_run)
|
|
print(' preparing email (from github issues)')
|
|
msg = prepare_email(artifact_id, release_version, repository, artifact_name, artifact_description, project_url)
|
|
input('Press Enter to send email...')
|
|
print(' sending email -- dry_run: %s, mail: %s' % (dry_run, mail))
|
|
send_email(msg, dry_run=dry_run, mail=mail)
|
|
|
|
pending_msg = """
|
|
Release successful pending steps:
|
|
* close and release sonatype repo: https://oss.sonatype.org/
|
|
* check if the release is there https://oss.sonatype.org/content/repositories/releases/org/elasticsearch/%(artifact_id)s/%(version)s
|
|
* tweet about the release
|
|
"""
|
|
print(pending_msg % {'version': release_version,
|
|
'artifact_id': artifact_id,
|
|
'project_url': project_url})
|
|
success = True
|
|
finally:
|
|
if not success:
|
|
print('Logs:')
|
|
with open(LOG, 'r') as log_file:
|
|
print(log_file.read())
|
|
git_checkout('master')
|
|
run('git reset --hard %s' % master_hash)
|
|
git_checkout(src_branch)
|
|
run('git reset --hard %s' % version_hash)
|
|
try:
|
|
run('git tag -d v%s' % release_version)
|
|
except RuntimeError:
|
|
pass
|
|
elif dry_run:
|
|
print('End of dry_run')
|
|
input('Press Enter to reset changes...')
|
|
git_checkout('master')
|
|
run('git reset --hard %s' % master_hash)
|
|
git_checkout(src_branch)
|
|
run('git reset --hard %s' % version_hash)
|
|
run('git tag -d v%s' % release_version)
|
|
|
|
# we delete this one anyways
|
|
run('git branch -D %s' % (release_branch('master', release_version)))
|
|
run('git branch -D %s' % (release_branch(src_branch, release_version)))
|
|
|
|
# Checkout the branch we started from
|
|
git_checkout(src_branch)
|