| #!/usr/bin/env python |
| # Copyright 2015 The PDFium Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| import datetime |
| import glob |
| import os |
| import re |
| import subprocess |
| import sys |
| |
| def os_name(): |
| if sys.platform.startswith('linux'): |
| return 'linux' |
| if sys.platform.startswith('win'): |
| return 'win' |
| if sys.platform.startswith('darwin'): |
| return 'mac' |
| raise Exception('Confused, can not determine OS, aborting.') |
| |
| |
| def RunCommand(cmd): |
| try: |
| subprocess.check_call(cmd) |
| return None |
| except subprocess.CalledProcessError as e: |
| return e |
| |
| |
| def RunCommandPropagateErr(cmd, stdout_has_errors=False, |
| exit_status_on_error=None): |
| """Run a command as a subprocess. |
| |
| Errors in that subprocess are printed out if it returns an error exit code. |
| |
| Args: |
| cmd: Command to run as a list of strings. |
| stdout_has_errors: Whether to print stdout instead of stderr on an error |
| exit. |
| exit_status_on_error: If specified, upon an error in the subprocess the |
| caller script exits immediately with the given status. |
| """ |
| p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE) |
| output, err = p.communicate() |
| |
| if p.returncode: |
| PrintErr('\nError when invoking "%s"' % ' '.join(cmd)) |
| if stdout_has_errors: |
| PrintErr(output) |
| |
| PrintErr(err) |
| |
| if exit_status_on_error is not None: |
| sys.exit(exit_status_on_error) |
| |
| return None |
| |
| return output |
| |
| |
| # RunCommandExtractHashedFiles returns a tuple: (raised_exception, hashed_files) |
| # It runs the given command. If it fails it will return an exception and None. |
| # If it succeeds it will return None and the list of processed files extracted |
| # from the output of the command. It expects lines in this format: |
| # MD5:<path_to_image_file>:<md5_hash_in_hex> |
| # The returned hashed_files is a list of (file_path, MD5-hash) pairs. |
| def RunCommandExtractHashedFiles(cmd): |
| try: |
| output = subprocess.check_output(cmd, universal_newlines=True) |
| ret = [] |
| for line in output.split('\n'): |
| line = line.strip() |
| if line.startswith("MD5:"): |
| ret.append([x.strip() for x in line.lstrip("MD5:").rsplit(":", 1)]) |
| return None, ret |
| except subprocess.CalledProcessError as e: |
| return e, None |
| |
| |
| class DirectoryFinder: |
| '''A class for finding directories and paths under either a standalone |
| checkout or a chromium checkout of PDFium.''' |
| |
| def __init__(self, build_location): |
| # |build_location| is typically "out/Debug" or "out/Release". |
| # Expect |my_dir| to be .../pdfium/testing/tools. |
| self.my_dir = os.path.dirname(os.path.realpath(__file__)) |
| self.testing_dir = os.path.dirname(self.my_dir) |
| if (os.path.basename(self.my_dir) != 'tools' or |
| os.path.basename(self.testing_dir) != 'testing'): |
| raise Exception('Confused, can not find pdfium root directory, aborting.') |
| self.pdfium_dir = os.path.dirname(self.testing_dir) |
| # Find path to build directory. This depends on whether this is a |
| # standalone build vs. a build as part of a chromium checkout. For |
| # standalone, we expect a path like .../pdfium/out/Debug, but for |
| # chromium, we expect a path like .../src/out/Debug two levels |
| # higher (to skip over the third_party/pdfium path component under |
| # which chromium sticks pdfium). |
| self.base_dir = self.pdfium_dir |
| one_up_dir = os.path.dirname(self.base_dir) |
| two_up_dir = os.path.dirname(one_up_dir) |
| if (os.path.basename(two_up_dir) == 'src' and |
| os.path.basename(one_up_dir) == 'third_party'): |
| self.base_dir = two_up_dir |
| self.build_dir = os.path.join(self.base_dir, build_location) |
| self.os_name = os_name() |
| |
| def ExecutablePath(self, name): |
| '''Finds compiled binaries under the build path.''' |
| result = os.path.join(self.build_dir, name) |
| if self.os_name == 'win': |
| result = result + '.exe' |
| return result |
| |
| def ScriptPath(self, name): |
| '''Finds other scripts in the same directory as this one.''' |
| return os.path.join(self.my_dir, name) |
| |
| def WorkingDir(self, other_components=''): |
| '''Places generated files under the build directory, not source dir.''' |
| result = os.path.join(self.build_dir, 'gen', 'pdfium') |
| if other_components: |
| result = os.path.join(result, other_components) |
| return result |
| |
| def TestingDir(self, other_components=''): |
| '''Finds test files somewhere under the testing directory.''' |
| result = self.testing_dir |
| if other_components: |
| result = os.path.join(result, other_components) |
| return result |
| |
| |
| def GetBooleanGnArg(arg_name, build_dir, verbose=False): |
| '''Extract the value of a boolean flag in args.gn''' |
| cwd = os.getcwd() |
| os.chdir(build_dir) |
| gn_args_output = subprocess.check_output( |
| ['gn', 'args', '.', '--list=%s' % arg_name, '--short']) |
| os.chdir(cwd) |
| arg_match_output = re.search('%s = (.*)' % arg_name, gn_args_output).group(1) |
| if verbose: |
| print >> sys.stderr, "Found '%s' for value of %s" % (arg_match_output, arg) |
| return arg_match_output == 'true' |
| |
| |
| def PrintWithTime(s): |
| """Prints s prepended by a timestamp.""" |
| print '[%s] %s' % (datetime.datetime.now().strftime("%Y%m%d %H:%M:%S"), |
| s) |
| |
| |
| def PrintErr(s): |
| """Prints s to stderr.""" |
| print >> sys.stderr, s |