Handle failed fetches/decodes from BitmapFetcher.
[chromium-blink-merge.git] / mojo / services / upload_service.py
blob184e35fd031da0399ef689185fb729e930f7a0cd
1 #!/usr/bin/env python
2 # Copyright 2014 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file.
6 import argparse
7 import imp
8 import os
9 import subprocess
10 import sys
11 import tempfile
12 import time
13 import zipfile
15 SERVICES = ["html_viewer", "network_service", "network_service_apptests"]
17 # A service does not need to expose interfaces. Those that do expose interfaces
18 # have their mojoms located in the directories listed below, in paths relative
19 # to the directory of this script.
20 MOJOMS_IN_DIR = {
21 "network_service": os.path.join("network", "public", "interfaces")
24 # The network service is downloaded out-of-band rather than dynamically by the
25 # shell and thus can be stored zipped in the cloud. Other services are intended
26 # to be downloaded dynamically by the shell, which doesn't currently understand
27 # zipped binaries.
28 SERVICES_WITH_ZIPPED_BINARIES = ["network_service", "network_service_apptests"]
30 if not sys.platform.startswith("linux"):
31 print "Only support linux for now"
32 sys.exit(1)
34 root_path = os.path.realpath(
35 os.path.join(
36 os.path.dirname(
37 os.path.realpath(__file__)),
38 os.pardir,
39 os.pardir))
41 find_depot_tools_path = os.path.join(root_path, "tools", "find_depot_tools.py")
42 find_depot_tools = imp.load_source("find_depot_tools", find_depot_tools_path)
44 depot_tools_path = find_depot_tools.add_depot_tools_to_path()
45 gsutil_exe = os.path.join(depot_tools_path, "third_party", "gsutil", "gsutil")
47 def get_version_name(custom_build):
48 if custom_build:
49 branch = subprocess.check_output(
50 ["git", "rev-parse", "--abbrev-ref", "HEAD"], cwd=root_path).strip()
51 try:
52 base = subprocess.check_output(
53 ["git", "config", "--get", "branch." + branch + ".base"],
54 cwd=root_path).strip()
55 issue = subprocess.check_output(
56 ["git", "config", "--get", "branch." + branch + ".rietveldissue"],
57 cwd=root_path).strip()
58 patchset = subprocess.check_output(
59 ["git", "config", "--get", "branch." + branch + ".rietveldpatchset"],
60 cwd=root_path).strip()
61 except subprocess.CalledProcessError:
62 return None
64 if not base or not issue or not patchset:
65 return None
66 else:
67 return "custom_build_base_%s_issue_%s_patchset_%s" % (base, issue,
68 patchset)
69 else:
70 return subprocess.check_output(["git", "rev-parse", "HEAD"],
71 cwd=root_path).strip()
73 def gsutil_cp(source, dest, dry_run):
74 if dry_run:
75 print "gsutil cp %s %s" % (source, dest)
76 else:
77 subprocess.check_call([gsutil_exe, "cp", source, dest])
80 def upload_mojoms(version_name, service, absolute_mojom_directory_path,
81 dry_run):
82 dest = "gs://mojo/" + service + "/" + version_name + "/" + "mojoms.zip"
84 with tempfile.NamedTemporaryFile() as mojom_zip_file:
85 with zipfile.ZipFile(mojom_zip_file, 'w') as z:
86 for root, _, files in os.walk(absolute_mojom_directory_path):
87 for filename in files:
88 absolute_file_path = os.path.join(root, filename)
89 relative_file_path = os.path.relpath(absolute_file_path, root)
90 z.write(absolute_file_path, relative_file_path)
91 gsutil_cp(mojom_zip_file.name, dest, dry_run)
94 def upload_binary(version_name, service, binary_dir, platform, dry_run):
95 dest_dir = "gs://mojo/" + service + "/" + version_name + "/" + platform + "/"
96 should_zip = service in SERVICES_WITH_ZIPPED_BINARIES
97 binary_name = service + ".mojo"
98 absolute_binary_path = os.path.join(root_path, binary_dir, binary_name)
100 if not should_zip:
101 # Upload the binary.
102 dest = dest_dir + binary_name
103 gsutil_cp(absolute_binary_path, dest, dry_run)
105 # Update the pointer to the service's location to point to the
106 # newly-uploaded binary.
107 service_location = dest.replace("gs://", "https://storage.googleapis.com/")
108 location_file = ("gs://mojo/services/" + platform + "/" + service +
109 "_location")
110 with tempfile.NamedTemporaryFile() as tmp:
111 tmp.write(service_location)
112 tmp.flush()
113 gsutil_cp(tmp.name, location_file, dry_run)
114 return
116 # Zip the binary before uploading it to the cloud.
117 dest = dest_dir + binary_name + ".zip"
118 with tempfile.NamedTemporaryFile() as binary_zip_file:
119 with zipfile.ZipFile(binary_zip_file, 'w') as z:
120 with open(absolute_binary_path) as service_binary:
121 zipinfo = zipfile.ZipInfo(binary_name)
122 zipinfo.external_attr = 0o777 << 16
123 zipinfo.compress_type = zipfile.ZIP_DEFLATED
124 zipinfo.date_time = time.gmtime(os.path.getmtime(absolute_binary_path))
125 z.writestr(zipinfo, service_binary.read())
126 gsutil_cp(binary_zip_file.name, dest, dry_run)
129 def main():
130 parser = argparse.ArgumentParser(
131 description="Upload service mojoms and binaries to Google storage")
132 parser.add_argument("-n", "--dry-run", action="store_true", help="Dry run")
133 parser.add_argument(
134 "--linux-x64-binary-dir",
135 help="Path to the dir containing the linux-x64 service binary relative "
136 "to the repo root, e.g. out/Release")
137 parser.add_argument(
138 "--android-arm-binary-dir",
139 help="Path to the dir containing the android-arm service binary relative "
140 "to the repo root, e.g. out/android_Release")
141 parser.add_argument("service",
142 help="The service to be uploaded (one of %s)" % SERVICES)
143 parser.add_argument(
144 "--custom-build", action="store_true",
145 help="Indicates that this is a build with change that is not committed. "
146 "The change must be uploaded to Rietveld. The script needs to be "
147 "run from the branch associated with the change.")
148 args = parser.parse_args()
150 if args.service not in SERVICES:
151 print args.service + " is not one of the recognized services:"
152 print SERVICES
153 return 1
155 version_name = get_version_name(args.custom_build)
156 if args.custom_build and not version_name:
157 print ("When uploading a custom build, the corresponding change to source "
158 "code must be uploaded to Rietveld. Besides, this script needs to "
159 "be run from the branch associated with the change.")
160 return 1
162 if args.service in MOJOMS_IN_DIR:
163 script_dir = os.path.dirname(os.path.realpath(__file__))
164 absolute_mojom_directory_path = os.path.join(script_dir,
165 MOJOMS_IN_DIR[args.service])
166 upload_mojoms(version_name, args.service, absolute_mojom_directory_path,
167 args.dry_run)
169 if args.linux_x64_binary_dir:
170 upload_binary(version_name, args.service, args.linux_x64_binary_dir,
171 "linux-x64", args.dry_run)
173 if args.android_arm_binary_dir:
174 upload_binary(version_name, args.service, args.android_arm_binary_dir,
175 "android-arm", args.dry_run)
177 if not args.dry_run:
178 print "Uploaded artifacts for version %s" % (version_name, )
179 else:
180 print "No artifacts uploaded (dry run)"
181 return 0
183 if __name__ == '__main__':
184 sys.exit(main())