Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 1 | #!/usr/bin/env python |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 2 | # -*- coding: utf-8 -*- |
Michael Bestas | 1ab959b | 2014-07-26 16:01:01 +0300 | [diff] [blame] | 3 | # crowdin_sync.py |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 4 | # |
| 5 | # Updates Crowdin source translations and pushes translations |
| 6 | # directly to CyanogenMod's Gerrit. |
| 7 | # |
Michael Bestas | 97677e1 | 2015-02-08 13:11:59 +0200 | [diff] [blame] | 8 | # Copyright (C) 2014-2015 The CyanogenMod Project |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 9 | # |
| 10 | # Licensed under the Apache License, Version 2.0 (the "License"); |
| 11 | # you may not use this file except in compliance with the License. |
| 12 | # You may obtain a copy of the License at |
| 13 | # |
| 14 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 15 | # |
| 16 | # Unless required by applicable law or agreed to in writing, software |
| 17 | # distributed under the License is distributed on an "AS IS" BASIS, |
| 18 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 19 | # See the License for the specific language governing permissions and |
| 20 | # limitations under the License. |
| 21 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 22 | # ################################# IMPORTS ################################## # |
| 23 | |
| 24 | from __future__ import print_function |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 25 | |
| 26 | import argparse |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 27 | import git |
| 28 | import os |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 29 | import subprocess |
| 30 | import sys |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 31 | |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 32 | from xml.dom import minidom |
| 33 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 34 | # ################################ FUNCTIONS ################################# # |
| 35 | |
| 36 | |
| 37 | def run_subprocess(cmd, silent=False): |
| 38 | p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, |
| 39 | universal_newlines=True) |
| 40 | comm = p.communicate() |
| 41 | exit_code = p.returncode |
| 42 | if exit_code != 0 and not silent: |
| 43 | print("There was an error running the subprocess.\n" |
| 44 | "cmd: %s\n" |
| 45 | "exit code: %d\n" |
| 46 | "stdout: %s\n" |
| 47 | "stderr: %s" % (cmd, exit_code, comm[0], comm[1]), |
| 48 | file=sys.stderr) |
| 49 | return comm, exit_code |
| 50 | |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 51 | |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 52 | def push_as_commit(path, name, branch, username): |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 53 | print('Committing %s on branch %s' % (name, branch)) |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 54 | |
| 55 | # Get path |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 56 | path = os.path.join(os.getcwd(), path) |
| 57 | if not path.endswith('.git'): |
| 58 | path = os.path.join(path, '.git') |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 59 | |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 60 | # Create repo object |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 61 | repo = git.Repo(path) |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 62 | |
| 63 | # Remove previously deleted files from Git |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 64 | files = repo.git.ls_files(d=True).split('\n') |
| 65 | if files and files[0]: |
| 66 | repo.git.rm(files) |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 67 | |
| 68 | # Add all files to commit |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 69 | repo.git.add('-A') |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 70 | |
| 71 | # Create commit; if it fails, probably empty so skipping |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 72 | try: |
| 73 | repo.git.commit(m='Automatic translation import') |
| 74 | except: |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 75 | print('Failed to create commit for %s, probably empty: skipping' |
| 76 | % name, file=sys.stderr) |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 77 | return |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 78 | |
| 79 | # Push commit |
Michael Bestas | f96f67b | 2014-10-21 00:43:37 +0300 | [diff] [blame] | 80 | try: |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 81 | repo.git.push('ssh://%s@review.cyanogenmod.org:29418/%s' % (username, name), |
| 82 | 'HEAD:refs/for/%s%%topic=translation' % branch) |
| 83 | print('Successfully pushed commit for %s' % name) |
Michael Bestas | f96f67b | 2014-10-21 00:43:37 +0300 | [diff] [blame] | 84 | except: |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 85 | print('Failed to push commit for %s' % name, file=sys.stderr) |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 86 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 87 | |
| 88 | def check_run(cmd): |
Michael Bestas | 97677e1 | 2015-02-08 13:11:59 +0200 | [diff] [blame] | 89 | p = subprocess.Popen(cmd, stdout=sys.stdout, stderr=sys.stderr) |
| 90 | ret = p.wait() |
| 91 | if ret != 0: |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 92 | print('Failed to run cmd: %s' % ' '.join(cmd), file=sys.stderr) |
Michael Bestas | 97677e1 | 2015-02-08 13:11:59 +0200 | [diff] [blame] | 93 | sys.exit(ret) |
| 94 | |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 95 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 96 | def find_xml(): |
| 97 | for dp, dn, file_names in os.walk(os.getcwd()): |
| 98 | for f in file_names: |
| 99 | if os.path.splitext(f)[1] == '.xml': |
| 100 | yield os.path.join(dp, f) |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 101 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 102 | # ############################################################################ # |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 103 | |
Michael Bestas | 6b6db12 | 2015-02-08 13:22:22 +0200 | [diff] [blame] | 104 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 105 | def parse_args(): |
| 106 | parser = argparse.ArgumentParser( |
| 107 | description="Synchronising CyanogenMod's translations with Crowdin") |
| 108 | sync = parser.add_mutually_exclusive_group() |
| 109 | parser.add_argument('-u', '--username', help='Gerrit username', |
| 110 | required=True) |
| 111 | parser.add_argument('-b', '--branch', help='CyanogenMod branch', |
| 112 | required=True) |
| 113 | sync.add_argument('--no-upload', action='store_true', |
| 114 | help='Only download CM translations from Crowdin') |
| 115 | sync.add_argument('--no-download', action='store_true', |
| 116 | help='Only upload CM source translations to Crowdin') |
| 117 | return parser.parse_args() |
Michael Bestas | 6b6db12 | 2015-02-08 13:22:22 +0200 | [diff] [blame] | 118 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 119 | # ################################# PREPARE ################################## # |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 120 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 121 | |
| 122 | def check_dependencies(): |
| 123 | print('\nSTEP 0: Checking dependencies & define shared variables') |
| 124 | |
| 125 | # Check for Ruby version of crowdin-cli |
| 126 | cmd = ['gem', 'list', 'crowdin-cli', '-i'] |
| 127 | if run_subprocess(cmd, silent=True)[1] != 0: |
| 128 | print('You have not installed crowdin-cli.', file=sys.stderr) |
| 129 | return False |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 130 | print('Found: crowdin-cli') |
| 131 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 132 | return True |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 133 | |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 134 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 135 | def load_xml(x='android/default.xml'): |
| 136 | # Variables regarding android/default.xml |
| 137 | print('Loading: %s' % x) |
| 138 | try: |
| 139 | return minidom.parse(x) |
| 140 | except IOError: |
| 141 | print('You have no %s.' % x, file=sys.stderr) |
| 142 | return None |
| 143 | except Exception: |
| 144 | # TODO: minidom should not be used. |
| 145 | print('Malformed %s.' % x, file=sys.stderr) |
| 146 | return None |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 147 | |
Michael Bestas | 4b26c4e | 2014-10-23 23:21:59 +0300 | [diff] [blame] | 148 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 149 | def check_files(branch): |
| 150 | files = ['crowdin/config.yaml', |
| 151 | 'crowdin/extra_packages_%s.xml' % branch, |
| 152 | 'crowdin/config_aosp.yaml', |
| 153 | 'crowdin/crowdin_%s.yaml' % branch, |
| 154 | 'crowdin/crowdin_%s_aosp.yaml' % branch |
| 155 | ] |
| 156 | for f in files: |
| 157 | if not os.path.isfile(f): |
| 158 | print('You have no %s.' % f, file=sys.stderr) |
| 159 | return False |
| 160 | print('Found: %s' % f) |
| 161 | return True |
Michael Bestas | 4b26c4e | 2014-10-23 23:21:59 +0300 | [diff] [blame] | 162 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 163 | # ################################### MAIN ################################### # |
Michael Bestas | 4b26c4e | 2014-10-23 23:21:59 +0300 | [diff] [blame] | 164 | |
Michael Bestas | 4b26c4e | 2014-10-23 23:21:59 +0300 | [diff] [blame] | 165 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 166 | def upload_crowdin(branch, no_upload=False): |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 167 | print('\nSTEP 1: Upload Crowdin source translations') |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 168 | if no_upload: |
| 169 | print('Skipping source translations upload') |
| 170 | return |
| 171 | print('\nUploading Crowdin source translations (AOSP supported languages)') |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 172 | |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 173 | # Execute 'crowdin-cli upload sources' and show output |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 174 | check_run(['crowdin-cli', |
| 175 | '--config=crowdin/crowdin_%s.yaml' % branch, |
| 176 | '--identity=crowdin/config.yaml', |
| 177 | 'upload', 'sources']) |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 178 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 179 | print('\nUploading Crowdin source translations ' |
| 180 | '(non-AOSP supported languages)') |
| 181 | # Execute 'crowdin-cli upload sources' and show output |
| 182 | check_run(['crowdin-cli', |
| 183 | '--config=crowdin/crowdin_%s_aosp.yaml' % branch, |
| 184 | '--identity=crowdin/config_aosp.yaml', |
| 185 | 'upload', 'sources']) |
| 186 | |
| 187 | |
| 188 | def download_crowdin(branch, xml, username, no_download=False): |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 189 | print('\nSTEP 2: Download Crowdin translations') |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 190 | if no_download: |
| 191 | print('Skipping translations download') |
| 192 | return |
| 193 | |
| 194 | print('\nDownloading Crowdin translations (AOSP supported languages)') |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 195 | # Execute 'crowdin-cli download' and show output |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 196 | check_run(['crowdin-cli', |
| 197 | '--config=crowdin/crowdin_%s.yaml' % branch, |
| 198 | '--identity=crowdin/config.yaml', |
| 199 | 'download', '--ignore-match']) |
Michael Bestas | 50579d2 | 2014-08-09 17:49:14 +0300 | [diff] [blame] | 200 | |
Michael Bestas | a02eb4b | 2015-02-08 15:47:01 +0200 | [diff] [blame] | 201 | print('\nDownloading Crowdin translations (non-AOSP supported languages)') |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 202 | # Execute 'crowdin-cli download' and show output |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 203 | check_run(['crowdin-cli', |
| 204 | '--config=crowdin/crowdin_%s_aosp.yaml' % branch, |
| 205 | '--identity=crowdin/config_aosp.yaml', |
| 206 | 'download', '--ignore-match']) |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 207 | |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 208 | print('\nSTEP 3: Remove useless empty translations') |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 209 | empty_contents = { |
| 210 | '<resources/>', |
| 211 | '<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"/>', |
| 212 | ('<resources xmlns:android=' |
| 213 | '"http://schemas.android.com/apk/res/android"/>'), |
| 214 | ('<resources xmlns:android="http://schemas.android.com/apk/res/android"' |
| 215 | ' xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"/>'), |
| 216 | ('<resources xmlns:tools="http://schemas.android.com/tools"' |
| 217 | ' xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"/>') |
| 218 | } |
| 219 | xf = None |
| 220 | for xml_file in find_xml(): |
| 221 | xf = open(xml_file).read() |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 222 | for line in empty_contents: |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 223 | if line in xf: |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 224 | print('Removing ' + xml_file) |
| 225 | os.remove(xml_file) |
| 226 | break |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 227 | del xf |
Marco Brohet | cb5cdb4 | 2014-07-11 22:41:53 +0200 | [diff] [blame] | 228 | |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 229 | print('\nSTEP 4: Create a list of pushable translations') |
| 230 | # Get all files that Crowdin pushed |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 231 | paths = [] |
| 232 | files = [ |
| 233 | ('crowdin/crowdin_%s.yaml' % branch, 'crowdin/config.yaml'), |
| 234 | ('crowdin/crowdin_%s_aosp.yaml' % branch, 'crowdin/config_aosp.yaml') |
| 235 | ] |
| 236 | for c, i in files: |
| 237 | cmd = ['crowdin-cli', '--config=%s' % c, '--identity=%s' % i, |
| 238 | 'list', 'sources'] |
| 239 | comm, ret = run_subprocess(cmd) |
| 240 | if ret != 0: |
| 241 | sys.exit(ret) |
| 242 | for p in str(comm[0]).split("\n"): |
| 243 | paths.append(p.replace('/%s' % branch, '')) |
Michael Bestas | 50579d2 | 2014-08-09 17:49:14 +0300 | [diff] [blame] | 244 | |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 245 | print('\nSTEP 5: Upload to Gerrit') |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 246 | items = [x for sub in xml for x in sub.getElementsByTagName('project')] |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 247 | all_projects = [] |
| 248 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 249 | for path in paths: |
| 250 | path = path.strip() |
Michael Bestas | 919053f | 2014-10-20 23:30:54 +0300 | [diff] [blame] | 251 | if not path: |
| 252 | continue |
| 253 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 254 | if "/res" not in path: |
| 255 | print('WARNING: Cannot determine project root dir of ' |
| 256 | '[%s], skipping.' % path) |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 257 | continue |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 258 | result = path.split('/res')[0].strip('/') |
| 259 | if result == path.strip('/'): |
| 260 | print('WARNING: Cannot determine project root dir of ' |
| 261 | '[%s], skipping.' % path) |
| 262 | continue |
Marco Brohet | 6b6b4e5 | 2014-07-20 00:05:16 +0200 | [diff] [blame] | 263 | |
Michael Bestas | c899b8c | 2015-03-03 00:53:19 +0200 | [diff] [blame] | 264 | if result in all_projects: |
Michael Bestas | c899b8c | 2015-03-03 00:53:19 +0200 | [diff] [blame] | 265 | continue |
Michael Bestas | 50579d2 | 2014-08-09 17:49:14 +0300 | [diff] [blame] | 266 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 267 | # When a project has multiple translatable files, Crowdin will |
| 268 | # give duplicates. |
| 269 | # We don't want that (useless empty commits), so we save each |
| 270 | # project in all_projects and check if it's already in there. |
Michael Bestas | c899b8c | 2015-03-03 00:53:19 +0200 | [diff] [blame] | 271 | all_projects.append(result) |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 272 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 273 | # Search android/default.xml or crowdin/extra_packages_%(branch).xml |
| 274 | # for the project's name |
| 275 | for project in items: |
| 276 | path = project.attributes['path'].value |
| 277 | if not (result + '/').startswith(path +'/'): |
Michael Bestas | c899b8c | 2015-03-03 00:53:19 +0200 | [diff] [blame] | 278 | continue |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 279 | if result != path: |
| 280 | if path in all_projects: |
| 281 | break |
| 282 | result = path |
| 283 | all_projects.append(result) |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 284 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 285 | br = project.getAttribute('revision') or branch |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 286 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 287 | push_as_commit(result, project.getAttribute('name'), br, username) |
| 288 | break |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 289 | |
Anthony King | 69a9538 | 2015-02-08 18:44:10 +0000 | [diff] [blame] | 290 | |
Anthony King | b860763 | 2015-05-01 22:06:37 +0300 | [diff] [blame^] | 291 | def main(): |
| 292 | if not check_dependencies(): |
| 293 | sys.exit(1) |
| 294 | |
| 295 | args = parse_args() |
| 296 | default_branch = args.branch |
| 297 | |
| 298 | print('Welcome to the CM Crowdin sync script!') |
| 299 | |
| 300 | xml_android = load_xml() |
| 301 | if xml_android is None: |
| 302 | sys.exit(1) |
| 303 | |
| 304 | xml_extra = load_xml(x='crowdin/extra_packages_%s.xml' % default_branch) |
| 305 | if xml_extra is None: |
| 306 | sys.exit(1) |
| 307 | |
| 308 | if not check_files(default_branch): |
| 309 | sys.exit(1) |
| 310 | |
| 311 | upload_crowdin(default_branch, args.no_upload) |
| 312 | download_crowdin(default_branch, (xml_android, xml_extra), |
| 313 | args.username, args.no_download) |
| 314 | print('\nDone!') |
| 315 | |
| 316 | if __name__ == '__main__': |
| 317 | main() |