#!/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:: # 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