#!/usr/bin/env python3 import subprocess, urllib.request, re, os, tarfile from html.parser import HTMLParser URL = 'http://icedtea.classpath.org/download/drops/icedtea{}/{}' DOWNLOAD_URL = 'http://icedtea.wildebeest.org/download/source/' DOWNLOAD_HTML = DOWNLOAD_URL + '?C=M;O=D' ICEDTEA_JDKS = [7] BUNDLES = ['openjdk', 'corba', 'jaxp', 'jaxws', 'jdk', 'langtools', 'hotspot'] SRC_PATH = './sources.nix' def get_output(cmd, env = None): try: proc = subprocess.Popen(cmd, env = env, stdout = subprocess.PIPE) out = proc.communicate()[0] except subprocess.CalledProcessError as e: return None return out.decode('utf-8').strip() def nix_prefetch_url(url): env = os.environ.copy() env['PRINT_PATH'] = '1' out = get_output(['nix-prefetch-url', url], env = env) return out.split('\n') def get_nix_attr(path, attr): out = get_output(['nix-instantiate', '--eval-only', '-A', attr, path]) if len(out) < 2 or out[0] != '"' or out[-1] != '"': raise Exception('Cannot find Nix attribute "{}" (parsing failure?)'.format(attr)) # Strip quotes return out[1:-1] def get_jdk_attr(jdk, attr): return get_nix_attr(SRC_PATH, 'icedtea{}.{}'.format(jdk, attr)) class Parser(HTMLParser): def __init__(self, link_regex): HTMLParser.__init__(self) self.regex = link_regex self.href = None self.version = None def handle_starttag(self, tag, attrs): if self.href != None or tag != 'a': return href = None for attr in attrs: if attr[0] == 'href': href = attr[1] if href == None: return m = re.match(self.regex, href) if m != None: self.href = href self.version = m.group(1) def get_latest_version_url(major): f = urllib.request.urlopen(DOWNLOAD_HTML) html = f.read().decode('utf-8') f.close() parser = Parser(r'^icedtea\d?-({}\.\d[\d.]*)\.tar\.xz$'.format(major)) parser.feed(html) parser.close() if parser.href == None: raise Exception('Error: could not find download url for major version "{}"'.format(major)) return parser.version, DOWNLOAD_URL + parser.href def get_old_bundle_attrs(jdk, bundle): attrs = {} for attr in ('url', 'sha256'): attrs[attr] = get_jdk_attr(jdk, 'bundles.{}.{}'.format(bundle, attr)) return attrs def get_old_attrs(jdk): attrs = {} for attr in ('version', 'url', 'sha256'): attrs[attr] = get_jdk_attr(jdk, attr) attrs['bundles'] = {} for bundle in BUNDLES: attrs['bundles'][bundle] = get_old_bundle_attrs(jdk, bundle) return attrs def get_member_filename(tarball, name): for fname in tarball.getnames(): m = re.match(r'^icedtea\d?-\d[\d.]*/{}$'.format(name), fname) if m != None: return m.group(0) return None def get_member_file(tarball, name): path = get_member_filename(tarball, name) if path == None: raise Exception('Could not find "{}" inside tarball'.format(name)) f = tarball.extractfile(path) data = f.read().decode('utf-8') f.close() return data def get_new_bundle_attr(makefile, bundle, attr): var = '{}_{}'.format(bundle.upper(), attr.upper()) regex = r'^{} = (.*?)$'.format(var) m = re.search(regex, makefile, re.MULTILINE) if m == None: raise Exception('Could not find variable "{}" in Makefile.am'.format(var)) return m.group(1) def get_new_bundle_attrs(jdk, version, path): url = URL.format(jdk, version) attrs = {} print('Opening file: "{}"'.format(path)) tar = tarfile.open(name = path, mode = 'r:xz') makefile = get_member_file(tar, 'Makefile.am') hotspot_map = get_member_file(tar, 'hotspot.map.in') hotspot_map = hotspot_map.replace('@ICEDTEA_RELEASE@', version) for bundle in BUNDLES: battrs = {} battrs['url'] = '{}/{}.tar.bz2'.format(url, bundle) if bundle == 'hotspot': m = re.search(r'^default (.*?) (.*?) (.*?) (.*?)$', hotspot_map, re.MULTILINE) if m == None: raise Exception('Could not find info for hotspot bundle in hotspot.map.in') battrs['sha256'] = m.group(4) else: battrs['sha256'] = get_new_bundle_attr(makefile, bundle, 'sha256sum') attrs[bundle] = battrs tar.close() return attrs def get_new_attrs(jdk): print('Getting old attributes for JDK {}...'.format(jdk)) old_attrs = get_old_attrs(jdk) attrs = {} # The major version corresponds to a specific JDK (1 = OpenJDK6, 2 = OpenJDK7, 3 = OpenJDK8) major = jdk - 5 print('Getting latest version for JDK {}...'.format(jdk)) version, url = get_latest_version_url(major) print() print('Old version: {}'.format(old_attrs['version'])) print('New version: {}'.format(version)) print() if version == old_attrs['version']: print('No update available, skipping...') print() return old_attrs print('Update available, generating new attributes for JDK {}...'.format(jdk)) attrs['version'] = version attrs['url'] = url print('Downloading tarball from url "{}"...'.format(url)) print() attrs['sha256'], path = nix_prefetch_url(url) print() print('Inspecting tarball for bundle information...') attrs['bundles'] = get_new_bundle_attrs(jdk, attrs['version'], path) print('Done!') return attrs def generate_jdk(jdk): attrs = get_new_attrs(jdk) version = attrs['version'] src_url = attrs['url'].replace(version, '${version}') common_url = URL.format(jdk, version) src_common_url = URL.format(jdk, '${version}') src = ' icedtea{} = rec {{\n'.format(jdk) src += ' version = "{}";\n'.format(version) src += '\n' src += ' url = "{}";\n'.format(src_url) src += ' sha256 = "{}";\n'.format(attrs['sha256']) src += '\n' src += ' common_url = "{}";\n'.format(src_common_url) src += '\n' src += ' bundles = {\n' for bundle in BUNDLES: battrs = attrs['bundles'][bundle] b_url = battrs['url'] b_url = b_url.replace(common_url, '${common_url}') src += ' {} = rec {{\n'.format(bundle) src += ' url = "{}";\n'.format(b_url) src += ' sha256 = "{}";\n'.format(battrs['sha256']) src += ' };\n' if bundle != BUNDLES[-1]: src += '\n' src += ' };\n' src += ' };\n' return src def generate_sources(jdks): src = '# This file is autogenerated from update.py in the same directory.\n' src += '{\n' for jdk in jdks: print() print('Generating sources for JDK {}...'.format(jdk)) src += generate_jdk(jdk) src += '}\n' return src if __name__ == '__main__': print('Generating {}...'.format(SRC_PATH)) src = generate_sources(ICEDTEA_JDKS) f = open(SRC_PATH, 'w', encoding = 'utf-8') f.write(src) f.close() print() print('Update complete!')