#!/usr/bin/env python3 ######################################################################## # Generates a new release. ######################################################################## import sys import re import subprocess import io import os import fileinput if sys.version_info < (3, 0): sys.stdout.write("Sorry, requires Python 3.x or better\n") sys.exit(1) def colored(r, g, b, text): return "\033[38;2;{};{};{}m{} \033[38;2;255;255;255m".format(r, g, b, text) def extractnumbers(s): return tuple(map(int,re.findall("(\d+)\.(\d+)\.(\d+)",str(s))[0])) def toversionstring(major, minor, rev): return str(major)+"."+str(minor)+"."+str(rev) def topaddedversionstring(major, minor, rev): return str(major)+str(minor).zfill(3)+str(rev).zfill(3) pipe = subprocess.Popen(["git", "rev-parse", "--abbrev-ref", "HEAD"], stdout=subprocess.PIPE, stderr=subprocess.STDOUT) branchresult = pipe.communicate()[0].decode().strip() if(branchresult != "master"): print(colored(255, 0, 0, "We recommend that you release on master, you are on '"+branchresult+"'")) ret = subprocess.call(["git", "remote", "update"]) if(ret != 0): sys.exit(ret) pipe = subprocess.Popen(["git", "log", "HEAD..", "--oneline"], stdout=subprocess.PIPE, stderr=subprocess.STDOUT) uptodateresult = pipe.communicate()[0].decode().strip() if(len(uptodateresult) != 0): print(uptodateresult) sys.exit(-1) pipe = subprocess.Popen(["git", "rev-parse", "--show-toplevel"], stdout=subprocess.PIPE, stderr=subprocess.STDOUT) maindir = pipe.communicate()[0].decode().strip() scriptlocation = os.path.dirname(os.path.abspath(__file__)) print("repository: "+maindir) pipe = subprocess.Popen(["git", "describe", "--abbrev=0", "--tags"], stdout=subprocess.PIPE, stderr=subprocess.STDOUT) versionresult = pipe.communicate()[0].decode().strip() print("last version: "+versionresult ) try: currentv = extractnumbers(versionresult) except: currentv = [0,0,0] if(len(sys.argv) != 2): nextv = (currentv[0],currentv[1], currentv[2]+1) print ("please specify version number, e.g. "+toversionstring(*nextv)) sys.exit(-1) try: newversion = extractnumbers(sys.argv[1]) except: print("can't parse version number "+sys.argv[1]) sys.exit(-1) print("checking that new version is valid") if(newversion[0] != currentv[0]): assert newversion[0] == currentv[0] + 1 assert newversion[1] == 0 assert newversion[2] == 0 elif (newversion[1] != currentv[1]): assert newversion[1] == currentv[1] + 1 assert newversion[2] == 0 else : assert newversion[2] == currentv[2] + 1 atleastminor= (currentv[0] != newversion[0]) or (currentv[1] != newversion[1]) if(atleastminor): print(colored(0, 255, 0, "This is more than a revision.")) releasefile = maindir + os.sep + "RELEASES.md" releasedata = open(releasefile).read() pattern = re.compile("#\s+\d+\.\d+") m = pattern.search(releasedata) if(m == None): print(colored(255, 0, 0, "You are preparing a new minor release and you have not yet updated RELEASES.md.")) sys.exit(-1) versionfilerel = os.sep + "include" + os.sep + "simdjson" + os.sep + "simdjson_version.h" versionfile = maindir + versionfilerel with open(versionfile, 'w') as file: file.write("// /include/simdjson/simdjson_version.h automatically generated by release.py,\n") file.write("// do not change by hand\n") file.write("#ifndef SIMDJSON_SIMDJSON_VERSION_H\n") file.write("#define SIMDJSON_SIMDJSON_VERSION_H\n") file.write("\n") file.write("/** The version of simdjson being used (major.minor.revision) */\n") file.write("#define SIMDJSON_VERSION "+toversionstring(*newversion)+"\n") file.write("\n") file.write("namespace simdjson {\n") file.write("enum {\n") file.write(" /**\n") file.write(" * The major version (MAJOR.minor.revision) of simdjson being used.\n") file.write(" */\n") file.write(" SIMDJSON_VERSION_MAJOR = "+str(newversion[0])+",\n") file.write(" /**\n") file.write(" * The minor version (major.MINOR.revision) of simdjson being used.\n") file.write(" */\n") file.write(" SIMDJSON_VERSION_MINOR = "+str(newversion[1])+",\n") file.write(" /**\n") file.write(" * The revision (major.minor.REVISION) of simdjson being used.\n") file.write(" */\n") file.write(" SIMDJSON_VERSION_REVISION = "+str(newversion[2])+"\n") file.write("};\n") file.write("} // namespace simdjson\n") file.write("\n") file.write("#endif // SIMDJSON_SIMDJSON_VERSION_H\n") print(versionfile + " modified") newmajorversionstring = str(newversion[0]) mewminorversionstring = str(newversion[1]) newrevversionstring = str(newversion[2]) newversionstring = str(newversion[0]) + "." + str(newversion[1]) + "." + str(newversion[2]) cmakefile = maindir + os.sep + "CMakeLists.txt" sonumber = None pattern = re.compile("set\(SIMDJSON_LIB_SOVERSION \"(\d+)\" CACHE STRING \"simdjson library soversion\"\)") with open (cmakefile, 'rt') as myfile: for line in myfile: m = pattern.search(line) if m != None: sonumber = int(m.group(1)) break print("so library number "+str(sonumber)) if(atleastminor): print("Given that we have a minor revision, it seems necessary to bump the so library number") print("See https://github.com/simdjson/simdjson/issues/661") sonumber += 1 for line in fileinput.input(cmakefile, inplace=1, backup='.bak'): line = re.sub('SIMDJSON_SEMANTIC_VERSION "\d+\.\d+\.\d+','SIMDJSON_SEMANTIC_VERSION "'+newversionstring, line.rstrip()) line = re.sub('SIMDJSON_LIB_VERSION "\d+','SIMDJSON_LIB_VERSION "'+str(sonumber), line) line = re.sub('set\(PROJECT_VERSION_MAJOR \d+','set(PROJECT_VERSION_MAJOR '+newmajorversionstring, line) line = re.sub('set\(PROJECT_VERSION_MINOR \d+','set(PROJECT_VERSION_MINOR '+mewminorversionstring, line) line = re.sub('set\(PROJECT_VERSION_PATCH \d+','set(PROJECT_VERSION_PATCH '+newrevversionstring, line) line = re.sub('set\(SIMDJSON_LIB_SOVERSION \"\d+\"','set(SIMDJSON_LIB_SOVERSION \"'+str(sonumber)+'\"', line) print(line) print("modified "+cmakefile+", a backup was made") doxyfile = maindir + os.sep + "Doxyfile" for line in fileinput.input(doxyfile, inplace=1, backup='.bak'): line = re.sub('PROJECT_NUMBER = "\d+\.\d+\.\d+','PROJECT_NUMBER = "'+newversionstring, line.rstrip()) print(line) print("modified "+doxyfile+", a backup was made") cp = subprocess.run(["bash", "amalgamate.sh"], stdout=subprocess.DEVNULL, cwd=maindir+ os.sep + "singleheader") # doesn't capture output if(cp.returncode != 0): print("Failed to run amalgamate") cp = subprocess.run(["doxygen"], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL, cwd=maindir) # doesn't capture output if(cp.returncode != 0): print("Failed to run doxygen") #ipe = subprocess.Popen(["doxygen"], stdout=subprocess.PIPE, stderr=subprocess.STDOUT, cwd=maindir) #doxygenresult = pipe.communicate()[0].decode().strip() pattern = re.compile("https://simdjson.org/api/(\d+\.\d+\.\d+)/index.html") readmefile = maindir + os.sep + "README.md" readmedata = open(readmefile).read() m = pattern.search(readmedata) if m == None: print(colored(255, 0, 0, 'I cannot find a link to the API documentation in your README?????')) else: detectedreadme = m.group(1) print("found a link to your API documentation in the README file: "+detectedreadme+" ("+toversionstring(*newversion)+")") if(atleastminor): if(detectedreadme != toversionstring(*newversion)): print(colored(255, 0, 0, "Consider updating the readme link to "+toversionstring(*newversion))) print("Please run the tests before issuing a release. \n") print("to issue release, enter \n git commit -a && git push && git tag -a v"+toversionstring(*newversion)+" -m \"version "+toversionstring(*newversion)+"\" && git push --tags \n")