forked from qt-creator/qt-creator
		
	GPL-3.0 is deprecated by SPDX.
Change done by
 find . -type f -exec perl -pi -e 's/LicenseRef-Qt-Commercial OR GPL-3.0(?!-)/LicenseRef-Qt-Commercial OR GPL-3.0-only/g' {} \;
Change-Id: If316a498e3f27d2030b86d4e7743b3237ce09939
Reviewed-by: Lucie Gerard <lucie.gerard@qt.io>
Reviewed-by: <github-actions-qt-creator@cristianadam.eu>
Reviewed-by: Eike Ziller <eike.ziller@qt.io>
		
	
		
			
				
	
	
		
			213 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #!/usr/bin/env python
 | |
| 
 | |
| # Copyright (C) 2016 The Qt Company Ltd.
 | |
| # SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
 | |
| 
 | |
| """
 | |
| A simple program that parses untranslated.ts files
 | |
| 
 | |
| current directory *must* be the top level qtcreator source directory
 | |
| 
 | |
| Usage:
 | |
|     scripts/uichanges.py old_untranslated.ts qtcreator_untranslated.ts
 | |
| 
 | |
|     IN TOP LEVEL QTC SOURCE DIRECTORY!
 | |
| """
 | |
| 
 | |
| import os, sys, string
 | |
| import platform
 | |
| import subprocess
 | |
| 
 | |
| from xml.sax import saxutils, handler, make_parser
 | |
| 
 | |
| baseDir = os.getcwd()
 | |
| transDir = os.path.join(baseDir, 'share/qtcreator/translations')
 | |
| unchangedContexts = 0
 | |
| 
 | |
| # --- The ContentHandler
 | |
| 
 | |
| # Generate a tree consisting of hash of context names.
 | |
| # Each context name value contains a hash of messages
 | |
| # Each message value contains a the file name (or '<unknown>')
 | |
| class Generator(handler.ContentHandler):
 | |
| 
 | |
|     def __init__(self):
 | |
|         handler.ContentHandler.__init__(self)
 | |
|         self._tree = {}
 | |
|         self._contextTree = {}
 | |
|         self._context = ''
 | |
|         self._file = ''
 | |
|         self._msg = ''
 | |
|         self._chars = ''
 | |
| 
 | |
|     # ContentHandler methods
 | |
| 
 | |
|     def startDocument(self):
 | |
|         self._tree = {}
 | |
|         self._contextTree = {}
 | |
|         self._context = ''
 | |
|         self._file = ''
 | |
|         self._chars = ''
 | |
| 
 | |
|     def startElement(self, name, attrs):
 | |
|         if name == 'location':
 | |
|             fn = attrs.get('filename')
 | |
|             if fn:
 | |
|                 fn = os.path.normpath(os.path.join(transDir, fn))
 | |
|                 fn = os.path.relpath(fn, baseDir)
 | |
|             else:
 | |
|                 fn = '<unknown>'
 | |
|             self._file = fn
 | |
|             return
 | |
| 
 | |
|     def endElement(self, name):
 | |
|         if name == 'name':
 | |
|             if self._context == '':
 | |
|                 self._context = self._chars.strip()
 | |
|                 self._chars = ''
 | |
|         elif name == 'source':
 | |
|             if self._chars:
 | |
|                 self._msg = self._chars.strip()
 | |
|                 self._chars = ''
 | |
|         elif name == 'message':
 | |
|             if self._msg:
 | |
|                 self._contextTree[self._msg] = self._file
 | |
| 
 | |
|             self._chars = ''
 | |
|             self._file = '<unknown>'
 | |
|             self._msg = ''
 | |
|         elif name == 'context':
 | |
|             if self._context != '':
 | |
|                  self._tree[self._context] = self._contextTree
 | |
|             self._contextTree = {}
 | |
|             self._context = ''
 | |
| 
 | |
|     def characters(self, content):
 | |
|         self._chars += content
 | |
| 
 | |
|     def tree(self):
 | |
|         return self._tree
 | |
| 
 | |
| def commitsForFile(file):
 | |
|     output = ''
 | |
|     if file == '<unknown>':
 | |
|         return output
 | |
| 
 | |
|     try:
 | |
|         output = subprocess.check_output(u'git log -1 -- "{0}"'.format(file),
 | |
|                                          shell=True, stderr=subprocess.STDOUT,
 | |
|                                          universal_newlines=True)
 | |
|     except:
 | |
|         output = ''
 | |
| 
 | |
|     return output
 | |
| 
 | |
| def examineMsg(ctx, msg, oldFile, newFile):
 | |
|     if oldFile == newFile:
 | |
|         # return ('', u'    EQL Message: "{0}" ({1})\n'.format(msg, oldFile))
 | |
|         return ('', '')
 | |
| 
 | |
|     if oldFile == '':
 | |
|         return (commitsForFile(newFile), u'    ADD: "{0}" ({1})\n'.format(msg, newFile))
 | |
| 
 | |
|     if newFile == '':
 | |
|         return (commitsForFile(oldFile), u'    DEL: "{0}" ({1})\n'.format(msg, oldFile))
 | |
| 
 | |
|     return (commitsForFile(newFile), u'    MOV: "{0}" ({1} -> {2})\n'.format(msg, oldFile, newFile))
 | |
| 
 | |
| def diffContext(ctx, old, new):
 | |
|     oldMsgSet = set(old.keys())
 | |
|     newMsgSet = set(new.keys())
 | |
| 
 | |
|     gitResults = set()
 | |
|     report = ''
 | |
|     unchanged = 0
 | |
| 
 | |
|     for m in sorted(oldMsgSet.difference(newMsgSet)):
 | |
|         res = examineMsg(ctx, m, old[m], '')
 | |
|         gitResults.add(res[0])
 | |
|         report = report + res[1]
 | |
|         if not res[1]:
 | |
|             unchanged += 1
 | |
| 
 | |
|     for m in sorted(newMsgSet.difference(oldMsgSet)):
 | |
|         res = examineMsg(ctx, m, '', new[m])
 | |
|         gitResults.add(res[0])
 | |
|         report = report + res[1]
 | |
|         if not res[1]:
 | |
|             unchanged += 1
 | |
| 
 | |
|     for m in sorted(oldMsgSet.intersection(newMsgSet)):
 | |
|         res = examineMsg(ctx, m, old[m], new[m])
 | |
|         gitResults.add(res[0])
 | |
|         report = report + res[1]
 | |
|         if not res[1]:
 | |
|             unchanged += 1
 | |
| 
 | |
|     gitResults.discard('')
 | |
| 
 | |
|     if not report:
 | |
|         return ''
 | |
| 
 | |
|     report = u'\nContext "{0}":\n{1}    {2} unchanged messages'.format(ctx, report, unchanged)
 | |
| 
 | |
|     if gitResults:
 | |
|         report += '\n\n    Git Commits:\n'
 | |
|         for g in gitResults:
 | |
|             if g:
 | |
|                 g = u'        {}'.format(g.replace('\n', '\n        '), errors='replace')
 | |
|                 report += g
 | |
| 
 | |
|     return report
 | |
| 
 | |
| 
 | |
| def stringify(obj):
 | |
|     stringTypes = (str, unicode) if sys.version_info.major == 2 else (str)
 | |
|     if isinstance(obj, stringTypes):
 | |
|         return obj
 | |
|     if isinstance(obj, bytes):
 | |
|         tmp = obj.decode('cp1252') if platform.system() in ('Microsoft','Windows') else obj.decode()
 | |
|         return tmp
 | |
| 
 | |
| # --- The main program
 | |
| 
 | |
| oldGenerator = Generator()
 | |
| oldParser = make_parser()
 | |
| oldParser.setContentHandler(oldGenerator)
 | |
| oldParser.parse(sys.argv[1])
 | |
| 
 | |
| oldTree = oldGenerator.tree()
 | |
| 
 | |
| newGenerator = Generator()
 | |
| newParser = make_parser()
 | |
| newParser.setContentHandler(newGenerator)
 | |
| newParser.parse(sys.argv[2])
 | |
| 
 | |
| newTree = newGenerator.tree()
 | |
| 
 | |
| oldContextSet = set(oldTree.keys())
 | |
| newContextSet = set(newTree.keys())
 | |
| 
 | |
| for c in sorted(oldContextSet.difference(newContextSet)):
 | |
|     report = diffContext(c, oldTree[c], {})
 | |
|     if report:
 | |
|         print(stringify(report.encode('utf-8')))
 | |
|     else:
 | |
|         unchangedContexts += 1
 | |
| 
 | |
| for c in sorted(newContextSet.difference(oldContextSet)):
 | |
|     report = diffContext(c, {}, newTree[c])
 | |
|     if report:
 | |
|         print(stringify(report.encode('utf-8')))
 | |
|     else:
 | |
|         unchangedContexts += 1
 | |
| 
 | |
| for c in sorted(newContextSet.intersection(oldContextSet)):
 | |
|     report = diffContext(c, oldTree[c], newTree[c])
 | |
|     if report:
 | |
|         print(stringify(report.encode('utf-8')))
 | |
|     else:
 | |
|         unchangedContexts += 1
 | |
| 
 | |
| print(u'{0} unchanged contexts'.format(unchangedContexts))
 |