#! /usr/bin/env nix-shell #! nix-shell -i python3 -p python3Packages.pyyaml import shutil import json import urllib.request import tempfile from sys import exit import os import subprocess import re import json import argparse import yaml import json FAKE_HASH = 'sha256-AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA=' NIXPKGS_ROOT = subprocess.Popen(['git', 'rev-parse', '--show-toplevel'], stdout=subprocess.PIPE, text=True).communicate()[0].strip() def load_code(name, **kwargs): with open(f"{NIXPKGS_ROOT}/pkgs/development/compilers/flutter/update/{name}.in", 'r') as f: code = f.read() for (key, value) in kwargs.items(): code = code.replace(f"@{key}@", value) return code # Return out paths def nix_build(code): temp = tempfile.NamedTemporaryFile(mode='w') temp.write(code) temp.flush() os.fsync(temp.fileno()) process = subprocess.Popen( [ "nix-build", "--impure", "--no-out-link", "--expr", f"with import {NIXPKGS_ROOT} {{}}; callPackage {temp.name} {{}}"], stdout=subprocess.PIPE, text=True) process.wait() temp.close() return process.stdout.read().strip().splitlines()[0] # Return errors def nix_build_to_fail(code): temp = tempfile.NamedTemporaryFile(mode='w') temp.write(code) temp.flush() os.fsync(temp.fileno()) process = subprocess.Popen( [ "nix-build", "--impure", "--keep-going", "--no-link", "--expr", f"with import {NIXPKGS_ROOT} {{}}; callPackage {temp.name} {{}}"], stderr=subprocess.PIPE, text=True) stderr = "" while True: line = process.stderr.readline() if not line: break stderr += line print(line.strip()) process.wait() temp.close() return stderr def get_engine_hashes(engine_version, flutter_version): code = load_code("get-engine-hashes.nix", nixpkgs_root=NIXPKGS_ROOT, flutter_version=flutter_version, engine_version=engine_version) stderr = nix_build_to_fail(code) pattern = re.compile( rf"/nix/store/.*-flutter-engine-source-{engine_version}-(.+?-.+?)-(.+?-.+?).drv':\n\s+specified: .*\n\s+got:\s+(.+?)\n") matches = pattern.findall(stderr) result_dict = {} for match in matches: flutter_platform, architecture, got = match result_dict.setdefault(flutter_platform, {})[architecture] = got def sort_dict_recursive(d): return { k: sort_dict_recursive(v) if isinstance( v, dict) else v for k, v in sorted( d.items())} result_dict = sort_dict_recursive(result_dict) return result_dict def get_artifact_hashes(flutter_compact_version): code = load_code("get-artifact-hashes.nix", nixpkgs_root=NIXPKGS_ROOT, flutter_compact_version=flutter_compact_version) stderr = nix_build_to_fail(code) pattern = re.compile( r"/nix/store/.*-flutter-artifacts-(.+?)-(.+?).drv':\n\s+specified: .*\n\s+got:\s+(.+?)\n") matches = pattern.findall(stderr) result_dict = {} for match in matches: flutter_platform, architecture, got = match result_dict.setdefault(flutter_platform, {})[architecture] = got def sort_dict_recursive(d): return { k: sort_dict_recursive(v) if isinstance( v, dict) else v for k, v in sorted( d.items())} result_dict = sort_dict_recursive(result_dict) return result_dict def get_dart_hashes(dart_version, channel): platforms = [ "x86_64-linux", "aarch64-linux", "x86_64-darwin", "aarch64-darwin"] result_dict = {} for platform in platforms: code = load_code( "get-dart-hashes.nix", dart_version=dart_version, channel=channel, platform=platform) stderr = nix_build_to_fail(code) pattern = re.compile(r"got:\s+(.+?)\n") result_dict[platform] = pattern.findall(stderr)[0] return result_dict def get_flutter_hash_and_src(flutter_version): code = load_code( "get-flutter.nix", flutter_version=flutter_version, hash="") stderr = nix_build_to_fail(code) pattern = re.compile(r"got:\s+(.+?)\n") hash = pattern.findall(stderr)[0] code = load_code( "get-flutter.nix", flutter_version=flutter_version, hash=hash) return (hash, nix_build(code)) def get_pubspec_lock(flutter_compact_version, flutter_src): code = load_code( "get-pubspec-lock.nix", flutter_compact_version=flutter_compact_version, flutter_src=flutter_src, hash="") stderr = nix_build_to_fail(code) pattern = re.compile(r"got:\s+(.+?)\n") hash = pattern.findall(stderr)[0] code = load_code( "get-pubspec-lock.nix", flutter_compact_version=flutter_compact_version, flutter_src=flutter_src, hash=hash) pubspec_lock_file = nix_build(code) with open(pubspec_lock_file, 'r') as f: pubspec_lock_yaml = f.read() return yaml.safe_load(pubspec_lock_yaml) def get_engine_swiftshader_rev(engine_version): with urllib.request.urlopen(f"https://github.com/flutter/engine/raw/{engine_version}/DEPS") as f: deps = f.read().decode('utf-8') pattern = re.compile(r"Var\('swiftshader_git'\) \+ '\/SwiftShader\.git' \+ '@' \+ \'([0-9a-fA-F]{40})\'\,") rev = pattern.findall(deps)[0] return rev def get_engine_swiftshader_hash(engine_swiftshader_rev): code = load_code( "get-engine-swiftshader.nix", engine_swiftshader_rev=engine_swiftshader_rev, hash="") stderr = nix_build_to_fail(code) pattern = re.compile(r"got:\s+(.+?)\n") return pattern.findall(stderr)[0] def write_data( nixpkgs_flutter_version_directory, flutter_version, channel, engine_hash, engine_hashes, engine_swiftshader_hash, engine_swiftshader_rev, dart_version, dart_hash, flutter_hash, artifact_hashes, pubspec_lock): with open(f"{nixpkgs_flutter_version_directory}/data.json", "w") as f: f.write(json.dumps({ "version": flutter_version, "engineVersion": engine_hash, "engineSwiftShaderHash": engine_swiftshader_hash, "engineSwiftShaderRev": engine_swiftshader_rev, "channel": channel, "engineHashes": engine_hashes, "dartVersion": dart_version, "dartHash": dart_hash, "flutterHash": flutter_hash, "artifactHashes": artifact_hashes, "pubspecLock": pubspec_lock, }, indent=2).strip() + "\n") def update_all_packages(): versions_directory = f"{NIXPKGS_ROOT}/pkgs/development/compilers/flutter/versions" versions = [directory for directory in os.listdir(versions_directory)] versions = sorted(versions, key=lambda x: ( int(x.split('_')[0]), int(x.split('_')[1])), reverse=True) new_content = [ "flutterPackages-bin = recurseIntoAttrs (callPackage ../development/compilers/flutter { });", "flutterPackages-source = recurseIntoAttrs (callPackage ../development/compilers/flutter { useNixpkgsEngine = true; });", "flutterPackages = flutterPackages-bin;" "flutter = flutterPackages.stable;", ] + [f"flutter{version.replace('_', '')} = flutterPackages.v{version};" for version in versions] with open(f"{NIXPKGS_ROOT}/pkgs/top-level/all-packages.nix", 'r') as file: lines = file.read().splitlines(keepends=True) start = -1 end = -1 for i, line in enumerate(lines): if "flutterPackages-bin = recurseIntoAttrs (callPackage ../development/compilers/flutter { });" in line: start = i if start != -1 and len(line.strip()) == 0: end = i break if start != -1 and end != -1: del lines[start:end] lines[start:start] = [f" {l}\n" for l in new_content] with open(f"{NIXPKGS_ROOT}/pkgs/top-level/all-packages.nix", 'w') as file: file.write("".join(lines)) # Finds Flutter version, Dart version, and Engine hash. # If the Flutter version is given, it uses that. Otherwise finds the # latest stable Flutter version. def find_versions(flutter_version=None, channel=None): engine_hash = None dart_version = None releases = json.load(urllib.request.urlopen( "https://storage.googleapis.com/flutter_infra_release/releases/releases_linux.json")) if not channel: channel = 'stable' if not flutter_version: hash = releases['current_release'][channel] release = next( filter( lambda release: release['hash'] == hash, releases['releases'])) flutter_version = release['version'] tags = subprocess.Popen(['git', 'ls-remote', '--tags', 'https://github.com/flutter/engine.git'], stdout=subprocess.PIPE, text=True).communicate()[0].strip() try: engine_hash = next( filter( lambda line: line.endswith(f'refs/tags/{flutter_version}'), tags.splitlines())).split('refs')[0].strip() except StopIteration: exit( f"Couldn't find Engine hash for Flutter version: {flutter_version}") try: dart_version = next( filter( lambda release: release['version'] == flutter_version, releases['releases']))['dart_sdk_version'] if " " in dart_version: dart_version = dart_version.split(' ')[2][:-1] except StopIteration: exit( f"Couldn't find Dart version for Flutter version: {flutter_version}") return (flutter_version, engine_hash, dart_version, channel) def main(): parser = argparse.ArgumentParser(description='Update Flutter in Nixpkgs') parser.add_argument('--version', type=str, help='Specify Flutter version') parser.add_argument('--channel', type=str, help='Specify Flutter release channel') parser.add_argument('--artifact-hashes', action='store_true', help='Whether to get artifact hashes') args = parser.parse_args() (flutter_version, engine_hash, dart_version, channel) = find_versions(args.version, args.channel) flutter_compact_version = '_'.join(flutter_version.split('.')[:2]) if args.artifact_hashes: print( json.dumps( get_artifact_hashes(flutter_compact_version), indent=2).strip() + "\n") return print(f"Flutter version: {flutter_version} ({flutter_compact_version}) on ({channel})") print(f"Engine hash: {engine_hash}") print(f"Dart version: {dart_version}") dart_hash = get_dart_hashes(dart_version, channel) (flutter_hash, flutter_src) = get_flutter_hash_and_src(flutter_version) nixpkgs_flutter_version_directory = f"{NIXPKGS_ROOT}/pkgs/development/compilers/flutter/versions/{flutter_compact_version}" if os.path.exists(f"{nixpkgs_flutter_version_directory}/data.json"): os.remove(f"{nixpkgs_flutter_version_directory}/data.json") os.makedirs(nixpkgs_flutter_version_directory, exist_ok=True) update_all_packages() common_data_args = { "nixpkgs_flutter_version_directory": nixpkgs_flutter_version_directory, "flutter_version": flutter_version, "channel": channel, "dart_version": dart_version, "engine_hash": engine_hash, "flutter_hash": flutter_hash, "dart_hash": dart_hash, } write_data( pubspec_lock={}, artifact_hashes={}, engine_hashes={}, engine_swiftshader_hash=FAKE_HASH, engine_swiftshader_rev='0', **common_data_args) pubspec_lock = get_pubspec_lock(flutter_compact_version, flutter_src) write_data( pubspec_lock=pubspec_lock, artifact_hashes={}, engine_hashes={}, engine_swiftshader_hash=FAKE_HASH, engine_swiftshader_rev='0', **common_data_args) artifact_hashes = get_artifact_hashes(flutter_compact_version) write_data( pubspec_lock=pubspec_lock, artifact_hashes=artifact_hashes, engine_hashes={}, engine_swiftshader_hash=FAKE_HASH, engine_swiftshader_rev='0', **common_data_args) engine_hashes = get_engine_hashes(engine_hash, flutter_version) write_data( pubspec_lock=pubspec_lock, artifact_hashes=artifact_hashes, engine_hashes=engine_hashes, engine_swiftshader_hash=FAKE_HASH, engine_swiftshader_rev='0', **common_data_args) engine_swiftshader_rev = get_engine_swiftshader_rev(engine_hash) engine_swiftshader_hash = get_engine_swiftshader_hash(engine_swiftshader_rev) write_data( pubspec_lock=pubspec_lock, artifact_hashes=artifact_hashes, engine_hashes=engine_hashes, engine_swiftshader_hash=engine_swiftshader_hash, engine_swiftshader_rev=engine_swiftshader_rev, **common_data_args) if __name__ == "__main__": main()