blob: baa18e36c2bfadfa63ba1fa417fd5ea4eb03b172 [file] [log] [blame]
Anthony Kingb8607632015-05-01 22:06:37 +03001#!/usr/bin/env python
Marco Brohetcb5cdb42014-07-11 22:41:53 +02002# -*- coding: utf-8 -*-
Michael Bestas1ab959b2014-07-26 16:01:01 +03003# crowdin_sync.py
Marco Brohetcb5cdb42014-07-11 22:41:53 +02004#
5# Updates Crowdin source translations and pushes translations
6# directly to CyanogenMod's Gerrit.
7#
Michael Bestas97677e12015-02-08 13:11:59 +02008# Copyright (C) 2014-2015 The CyanogenMod Project
Marco Brohetcb5cdb42014-07-11 22:41:53 +02009#
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 Kingb8607632015-05-01 22:06:37 +030022# ################################# IMPORTS ################################## #
23
24from __future__ import print_function
Marco Brohet6b6b4e52014-07-20 00:05:16 +020025
26import argparse
Marco Brohetcb5cdb42014-07-11 22:41:53 +020027import git
28import os
Marco Brohetcb5cdb42014-07-11 22:41:53 +020029import subprocess
30import sys
Anthony Kingb8607632015-05-01 22:06:37 +030031
Marco Brohetcb5cdb42014-07-11 22:41:53 +020032from xml.dom import minidom
33
Anthony Kingb8607632015-05-01 22:06:37 +030034# ################################ FUNCTIONS ################################# #
35
36
37def 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 Brohet6b6b4e52014-07-20 00:05:16 +020051
Marco Brohet6b6b4e52014-07-20 00:05:16 +020052def push_as_commit(path, name, branch, username):
Anthony Kingb8607632015-05-01 22:06:37 +030053 print('Committing %s on branch %s' % (name, branch))
Marco Brohetcb5cdb42014-07-11 22:41:53 +020054
55 # Get path
Anthony Kingb8607632015-05-01 22:06:37 +030056 path = os.path.join(os.getcwd(), path)
57 if not path.endswith('.git'):
58 path = os.path.join(path, '.git')
Marco Brohetcb5cdb42014-07-11 22:41:53 +020059
Marco Brohet6b6b4e52014-07-20 00:05:16 +020060 # Create repo object
Marco Brohetcb5cdb42014-07-11 22:41:53 +020061 repo = git.Repo(path)
Marco Brohet6b6b4e52014-07-20 00:05:16 +020062
63 # Remove previously deleted files from Git
Anthony Kingb8607632015-05-01 22:06:37 +030064 files = repo.git.ls_files(d=True).split('\n')
65 if files and files[0]:
66 repo.git.rm(files)
Marco Brohet6b6b4e52014-07-20 00:05:16 +020067
68 # Add all files to commit
Marco Brohetcb5cdb42014-07-11 22:41:53 +020069 repo.git.add('-A')
Marco Brohet6b6b4e52014-07-20 00:05:16 +020070
71 # Create commit; if it fails, probably empty so skipping
Marco Brohetcb5cdb42014-07-11 22:41:53 +020072 try:
73 repo.git.commit(m='Automatic translation import')
74 except:
Anthony Kingb8607632015-05-01 22:06:37 +030075 print('Failed to create commit for %s, probably empty: skipping'
76 % name, file=sys.stderr)
Marco Brohetcb5cdb42014-07-11 22:41:53 +020077 return
Marco Brohet6b6b4e52014-07-20 00:05:16 +020078
79 # Push commit
Michael Bestasf96f67b2014-10-21 00:43:37 +030080 try:
Anthony Kingb8607632015-05-01 22:06:37 +030081 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 Bestasf96f67b2014-10-21 00:43:37 +030084 except:
Anthony Kingb8607632015-05-01 22:06:37 +030085 print('Failed to push commit for %s' % name, file=sys.stderr)
Marco Brohetcb5cdb42014-07-11 22:41:53 +020086
Anthony Kingb8607632015-05-01 22:06:37 +030087
88def check_run(cmd):
Michael Bestas97677e12015-02-08 13:11:59 +020089 p = subprocess.Popen(cmd, stdout=sys.stdout, stderr=sys.stderr)
90 ret = p.wait()
91 if ret != 0:
Anthony Kingb8607632015-05-01 22:06:37 +030092 print('Failed to run cmd: %s' % ' '.join(cmd), file=sys.stderr)
Michael Bestas97677e12015-02-08 13:11:59 +020093 sys.exit(ret)
94
Marco Brohet6b6b4e52014-07-20 00:05:16 +020095
Anthony Kingb8607632015-05-01 22:06:37 +030096def 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 Brohet6b6b4e52014-07-20 00:05:16 +0200101
Anthony Kingb8607632015-05-01 22:06:37 +0300102# ############################################################################ #
Marco Brohet6b6b4e52014-07-20 00:05:16 +0200103
Michael Bestas6b6db122015-02-08 13:22:22 +0200104
Anthony Kingb8607632015-05-01 22:06:37 +0300105def 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 Bestas6b6db122015-02-08 13:22:22 +0200118
Anthony Kingb8607632015-05-01 22:06:37 +0300119# ################################# PREPARE ################################## #
Marco Brohet6b6b4e52014-07-20 00:05:16 +0200120
Anthony Kingb8607632015-05-01 22:06:37 +0300121
122def 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 Brohetcb5cdb42014-07-11 22:41:53 +0200130 print('Found: crowdin-cli')
131
Anthony Kingb8607632015-05-01 22:06:37 +0300132 return True
Marco Brohetcb5cdb42014-07-11 22:41:53 +0200133
Marco Brohetcb5cdb42014-07-11 22:41:53 +0200134
Anthony Kingb8607632015-05-01 22:06:37 +0300135def 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 Brohet6b6b4e52014-07-20 00:05:16 +0200147
Michael Bestas4b26c4e2014-10-23 23:21:59 +0300148
Anthony Kingb8607632015-05-01 22:06:37 +0300149def check_files(branch):
Michael Bestas6c327e62015-05-02 01:58:01 +0300150 files = ['crowdin/extra_packages_%s.xml' % branch,
Anthony Kingb8607632015-05-01 22:06:37 +0300151 'crowdin/crowdin_%s.yaml' % branch,
152 'crowdin/crowdin_%s_aosp.yaml' % branch
153 ]
154 for f in files:
155 if not os.path.isfile(f):
156 print('You have no %s.' % f, file=sys.stderr)
157 return False
158 print('Found: %s' % f)
159 return True
Michael Bestas4b26c4e2014-10-23 23:21:59 +0300160
Anthony Kingb8607632015-05-01 22:06:37 +0300161# ################################### MAIN ################################### #
Michael Bestas4b26c4e2014-10-23 23:21:59 +0300162
Michael Bestas4b26c4e2014-10-23 23:21:59 +0300163
Anthony Kingb8607632015-05-01 22:06:37 +0300164def upload_crowdin(branch, no_upload=False):
Michael Bestas919053f2014-10-20 23:30:54 +0300165 print('\nSTEP 1: Upload Crowdin source translations')
Anthony Kingb8607632015-05-01 22:06:37 +0300166 if no_upload:
167 print('Skipping source translations upload')
168 return
169 print('\nUploading Crowdin source translations (AOSP supported languages)')
Marco Brohetcb5cdb42014-07-11 22:41:53 +0200170
Anthony King69a95382015-02-08 18:44:10 +0000171 # Execute 'crowdin-cli upload sources' and show output
Michael Bestas6c327e62015-05-02 01:58:01 +0300172 check_run(['crowdin-cli', '--config=crowdin/crowdin_%s.yaml' % branch,
Anthony Kingb8607632015-05-01 22:06:37 +0300173 'upload', 'sources'])
Anthony King69a95382015-02-08 18:44:10 +0000174
Anthony Kingb8607632015-05-01 22:06:37 +0300175 print('\nUploading Crowdin source translations '
176 '(non-AOSP supported languages)')
177 # Execute 'crowdin-cli upload sources' and show output
Michael Bestas6c327e62015-05-02 01:58:01 +0300178 check_run(['crowdin-cli', '--config=crowdin/crowdin_%s_aosp.yaml' % branch,
Anthony Kingb8607632015-05-01 22:06:37 +0300179 'upload', 'sources'])
180
181
182def download_crowdin(branch, xml, username, no_download=False):
Michael Bestas919053f2014-10-20 23:30:54 +0300183 print('\nSTEP 2: Download Crowdin translations')
Anthony Kingb8607632015-05-01 22:06:37 +0300184 if no_download:
185 print('Skipping translations download')
186 return
187
188 print('\nDownloading Crowdin translations (AOSP supported languages)')
Michael Bestas919053f2014-10-20 23:30:54 +0300189 # Execute 'crowdin-cli download' and show output
Michael Bestas6c327e62015-05-02 01:58:01 +0300190 check_run(['crowdin-cli', '--config=crowdin/crowdin_%s.yaml' % branch,
Anthony Kingb8607632015-05-01 22:06:37 +0300191 'download', '--ignore-match'])
Michael Bestas50579d22014-08-09 17:49:14 +0300192
Michael Bestasa02eb4b2015-02-08 15:47:01 +0200193 print('\nDownloading Crowdin translations (non-AOSP supported languages)')
Michael Bestas919053f2014-10-20 23:30:54 +0300194 # Execute 'crowdin-cli download' and show output
Michael Bestas6c327e62015-05-02 01:58:01 +0300195 check_run(['crowdin-cli', '--config=crowdin/crowdin_%s_aosp.yaml' % branch,
Anthony Kingb8607632015-05-01 22:06:37 +0300196 'download', '--ignore-match'])
Marco Brohetcb5cdb42014-07-11 22:41:53 +0200197
Michael Bestas919053f2014-10-20 23:30:54 +0300198 print('\nSTEP 3: Remove useless empty translations')
Anthony Kingb8607632015-05-01 22:06:37 +0300199 empty_contents = {
200 '<resources/>',
201 '<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"/>',
202 ('<resources xmlns:android='
203 '"http://schemas.android.com/apk/res/android"/>'),
204 ('<resources xmlns:android="http://schemas.android.com/apk/res/android"'
205 ' xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"/>'),
206 ('<resources xmlns:tools="http://schemas.android.com/tools"'
207 ' xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"/>')
208 }
209 xf = None
210 for xml_file in find_xml():
211 xf = open(xml_file).read()
Michael Bestas919053f2014-10-20 23:30:54 +0300212 for line in empty_contents:
Anthony Kingb8607632015-05-01 22:06:37 +0300213 if line in xf:
Michael Bestas919053f2014-10-20 23:30:54 +0300214 print('Removing ' + xml_file)
215 os.remove(xml_file)
216 break
Anthony Kingb8607632015-05-01 22:06:37 +0300217 del xf
Marco Brohetcb5cdb42014-07-11 22:41:53 +0200218
Michael Bestas919053f2014-10-20 23:30:54 +0300219 print('\nSTEP 4: Create a list of pushable translations')
220 # Get all files that Crowdin pushed
Anthony Kingb8607632015-05-01 22:06:37 +0300221 paths = []
222 files = [
Michael Bestas6c327e62015-05-02 01:58:01 +0300223 ('crowdin/crowdin_%s.yaml' % branch),
224 ('crowdin/crowdin_%s_aosp.yaml' % branch)
Anthony Kingb8607632015-05-01 22:06:37 +0300225 ]
Michael Bestas6c327e62015-05-02 01:58:01 +0300226 for c in files:
227 cmd = ['crowdin-cli', '--config=%s' % c, 'list', 'sources']
Anthony Kingb8607632015-05-01 22:06:37 +0300228 comm, ret = run_subprocess(cmd)
229 if ret != 0:
230 sys.exit(ret)
231 for p in str(comm[0]).split("\n"):
232 paths.append(p.replace('/%s' % branch, ''))
Michael Bestas50579d22014-08-09 17:49:14 +0300233
Michael Bestas919053f2014-10-20 23:30:54 +0300234 print('\nSTEP 5: Upload to Gerrit')
Anthony Kingb8607632015-05-01 22:06:37 +0300235 items = [x for sub in xml for x in sub.getElementsByTagName('project')]
Michael Bestas919053f2014-10-20 23:30:54 +0300236 all_projects = []
237
Anthony Kingb8607632015-05-01 22:06:37 +0300238 for path in paths:
239 path = path.strip()
Michael Bestas919053f2014-10-20 23:30:54 +0300240 if not path:
241 continue
242
Anthony Kingb8607632015-05-01 22:06:37 +0300243 if "/res" not in path:
244 print('WARNING: Cannot determine project root dir of '
245 '[%s], skipping.' % path)
Anthony King69a95382015-02-08 18:44:10 +0000246 continue
Anthony Kingb8607632015-05-01 22:06:37 +0300247 result = path.split('/res')[0].strip('/')
248 if result == path.strip('/'):
249 print('WARNING: Cannot determine project root dir of '
250 '[%s], skipping.' % path)
251 continue
Marco Brohet6b6b4e52014-07-20 00:05:16 +0200252
Michael Bestasc899b8c2015-03-03 00:53:19 +0200253 if result in all_projects:
Michael Bestasc899b8c2015-03-03 00:53:19 +0200254 continue
Michael Bestas50579d22014-08-09 17:49:14 +0300255
Anthony Kingb8607632015-05-01 22:06:37 +0300256 # When a project has multiple translatable files, Crowdin will
257 # give duplicates.
258 # We don't want that (useless empty commits), so we save each
259 # project in all_projects and check if it's already in there.
Michael Bestasc899b8c2015-03-03 00:53:19 +0200260 all_projects.append(result)
Anthony King69a95382015-02-08 18:44:10 +0000261
Anthony Kingb8607632015-05-01 22:06:37 +0300262 # Search android/default.xml or crowdin/extra_packages_%(branch).xml
263 # for the project's name
264 for project in items:
265 path = project.attributes['path'].value
266 if not (result + '/').startswith(path +'/'):
Michael Bestasc899b8c2015-03-03 00:53:19 +0200267 continue
Anthony Kingb8607632015-05-01 22:06:37 +0300268 if result != path:
269 if path in all_projects:
270 break
271 result = path
272 all_projects.append(result)
Anthony King69a95382015-02-08 18:44:10 +0000273
Anthony Kingb8607632015-05-01 22:06:37 +0300274 br = project.getAttribute('revision') or branch
Anthony King69a95382015-02-08 18:44:10 +0000275
Anthony Kingb8607632015-05-01 22:06:37 +0300276 push_as_commit(result, project.getAttribute('name'), br, username)
277 break
Anthony King69a95382015-02-08 18:44:10 +0000278
Anthony King69a95382015-02-08 18:44:10 +0000279
Anthony Kingb8607632015-05-01 22:06:37 +0300280def main():
281 if not check_dependencies():
282 sys.exit(1)
283
284 args = parse_args()
285 default_branch = args.branch
286
287 print('Welcome to the CM Crowdin sync script!')
288
289 xml_android = load_xml()
290 if xml_android is None:
291 sys.exit(1)
292
293 xml_extra = load_xml(x='crowdin/extra_packages_%s.xml' % default_branch)
294 if xml_extra is None:
295 sys.exit(1)
296
297 if not check_files(default_branch):
298 sys.exit(1)
299
300 upload_crowdin(default_branch, args.no_upload)
301 download_crowdin(default_branch, (xml_android, xml_extra),
302 args.username, args.no_download)
303 print('\nDone!')
304
305if __name__ == '__main__':
306 main()