2012-04-03 01:34:16 +00:00
|
|
|
# Copyright (c) 2011 The Chromium Embedded Framework Authors. All rights
|
|
|
|
# reserved. Use of this source code is governed by a BSD-style license that
|
|
|
|
# can be found in the LICENSE file.
|
|
|
|
|
2020-01-09 16:42:00 +02:00
|
|
|
from __future__ import absolute_import
|
2012-04-03 01:34:16 +00:00
|
|
|
from glob import iglob
|
2020-01-10 12:53:02 +02:00
|
|
|
from io import open
|
2012-04-03 01:34:16 +00:00
|
|
|
import os
|
2021-11-16 14:22:55 -05:00
|
|
|
import fnmatch
|
2012-04-03 01:34:16 +00:00
|
|
|
import shutil
|
|
|
|
import sys
|
|
|
|
import time
|
|
|
|
|
2017-05-28 15:04:18 +02:00
|
|
|
|
|
|
|
def read_file(name, normalize=True):
|
|
|
|
""" Read a file. """
|
|
|
|
try:
|
2020-01-10 12:53:02 +02:00
|
|
|
with open(name, 'r', encoding='utf-8') as f:
|
|
|
|
# read the data
|
|
|
|
data = f.read()
|
|
|
|
if normalize:
|
|
|
|
# normalize line endings
|
|
|
|
data = data.replace("\r\n", "\n")
|
|
|
|
return data
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to read file ' + name + ': ' + strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
2012-04-03 01:34:16 +00:00
|
|
|
def write_file(name, data):
|
2017-05-28 15:04:18 +02:00
|
|
|
""" Write a file. """
|
|
|
|
try:
|
2023-12-26 13:36:48 -07:00
|
|
|
with open(name, 'w', encoding='utf-8', newline='\n') as f:
|
2020-01-10 12:53:02 +02:00
|
|
|
# write the data
|
2020-01-13 11:53:36 +01:00
|
|
|
if sys.version_info.major == 2:
|
2020-01-12 16:17:19 +02:00
|
|
|
f.write(data.decode('utf-8'))
|
2020-01-13 11:53:36 +01:00
|
|
|
else:
|
2020-01-12 16:17:19 +02:00
|
|
|
f.write(data)
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to write file ' + name + ': ' + strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
2012-04-03 01:34:16 +00:00
|
|
|
def path_exists(name):
|
2017-05-28 15:04:18 +02:00
|
|
|
""" Returns true if the path currently exists. """
|
|
|
|
return os.path.exists(name)
|
|
|
|
|
2012-04-03 01:34:16 +00:00
|
|
|
|
2020-04-30 15:59:23 -04:00
|
|
|
def write_file_if_changed(name, data):
|
|
|
|
""" Write a file if the contents have changed. Returns True if the file was written. """
|
|
|
|
if path_exists(name):
|
|
|
|
old_contents = read_file(name)
|
|
|
|
else:
|
|
|
|
old_contents = ''
|
|
|
|
|
|
|
|
if (data != old_contents):
|
|
|
|
write_file(name, data)
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2012-04-03 01:34:16 +00:00
|
|
|
def backup_file(name):
|
2017-05-28 15:04:18 +02:00
|
|
|
""" Rename the file to a name that includes the current time stamp. """
|
|
|
|
move_file(name, name + '.' + time.strftime('%Y-%m-%d-%H-%M-%S'))
|
|
|
|
|
|
|
|
|
|
|
|
def copy_file(src, dst, quiet=True):
|
|
|
|
""" Copy a file. """
|
|
|
|
try:
|
2018-05-07 18:50:39 +02:00
|
|
|
shutil.copy2(src, dst)
|
2017-05-28 15:04:18 +02:00
|
|
|
if not quiet:
|
|
|
|
sys.stdout.write('Transferring ' + src + ' file.\n')
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to copy file from ' + src + ' to ' + dst + ': ' +
|
|
|
|
strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
def move_file(src, dst, quiet=True):
|
|
|
|
""" Move a file. """
|
|
|
|
try:
|
|
|
|
shutil.move(src, dst)
|
|
|
|
if not quiet:
|
|
|
|
sys.stdout.write('Moving ' + src + ' file.\n')
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to move file from ' + src + ' to ' + dst + ': ' +
|
|
|
|
strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
def copy_files(src_glob, dst_folder, quiet=True):
|
|
|
|
""" Copy multiple files. """
|
|
|
|
for fname in iglob(src_glob):
|
|
|
|
dst = os.path.join(dst_folder, os.path.basename(fname))
|
|
|
|
if os.path.isdir(fname):
|
|
|
|
copy_dir(fname, dst, quiet)
|
|
|
|
else:
|
|
|
|
copy_file(fname, dst, quiet)
|
|
|
|
|
|
|
|
|
|
|
|
def remove_file(name, quiet=True):
|
|
|
|
""" Remove the specified file. """
|
|
|
|
try:
|
|
|
|
if path_exists(name):
|
|
|
|
os.remove(name)
|
|
|
|
if not quiet:
|
|
|
|
sys.stdout.write('Removing ' + name + ' file.\n')
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to remove file ' + name + ': ' + strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
def copy_dir(src, dst, quiet=True):
|
|
|
|
""" Copy a directory tree. """
|
|
|
|
try:
|
|
|
|
remove_dir(dst, quiet)
|
|
|
|
shutil.copytree(src, dst)
|
|
|
|
if not quiet:
|
|
|
|
sys.stdout.write('Transferring ' + src + ' directory.\n')
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to copy directory from ' + src + ' to ' + dst +
|
|
|
|
': ' + strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
def remove_dir(name, quiet=True):
|
|
|
|
""" Remove the specified directory. """
|
|
|
|
try:
|
|
|
|
if path_exists(name):
|
|
|
|
shutil.rmtree(name)
|
|
|
|
if not quiet:
|
|
|
|
sys.stdout.write('Removing ' + name + ' directory.\n')
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to remove directory ' + name + ': ' + strerror)
|
|
|
|
raise
|
|
|
|
|
|
|
|
|
|
|
|
def make_dir(name, quiet=True):
|
|
|
|
""" Create the specified directory. """
|
|
|
|
try:
|
|
|
|
if not path_exists(name):
|
|
|
|
if not quiet:
|
|
|
|
sys.stdout.write('Creating ' + name + ' directory.\n')
|
|
|
|
os.makedirs(name)
|
2020-01-09 16:42:00 +02:00
|
|
|
except IOError as e:
|
|
|
|
(errno, strerror) = e.args
|
2017-05-28 15:04:18 +02:00
|
|
|
sys.stderr.write('Failed to create directory ' + name + ': ' + strerror)
|
|
|
|
raise
|
|
|
|
|
2012-04-03 01:34:16 +00:00
|
|
|
|
|
|
|
def get_files(search_glob):
|
2021-11-16 14:22:55 -05:00
|
|
|
""" Returns all files matching |search_glob|. """
|
|
|
|
recursive_glob = '**' + os.path.sep
|
|
|
|
if recursive_glob in search_glob:
|
|
|
|
if sys.version_info >= (3, 5):
|
|
|
|
result = iglob(search_glob, recursive=True)
|
|
|
|
else:
|
|
|
|
# Polyfill for recursive glob pattern matching added in Python 3.5.
|
|
|
|
result = get_files_recursive(*search_glob.split(recursive_glob))
|
|
|
|
else:
|
|
|
|
result = iglob(search_glob)
|
|
|
|
|
2017-05-28 15:04:18 +02:00
|
|
|
# Sort the result for consistency across platforms.
|
2021-11-16 14:22:55 -05:00
|
|
|
return sorted(result)
|
|
|
|
|
|
|
|
|
|
|
|
def get_files_recursive(directory, pattern):
|
|
|
|
""" Returns all files in |directory| matching |pattern| recursively. """
|
|
|
|
for root, dirs, files in os.walk(directory):
|
|
|
|
for basename in files:
|
|
|
|
if fnmatch.fnmatch(basename, pattern):
|
|
|
|
filename = os.path.join(root, basename)
|
|
|
|
yield filename
|
2017-05-28 15:04:18 +02:00
|
|
|
|
2012-10-08 17:47:37 +00:00
|
|
|
|
|
|
|
def read_version_file(file, args):
|
2017-05-28 15:04:18 +02:00
|
|
|
""" Read and parse a version file (key=value pairs, one per line). """
|
|
|
|
lines = read_file(file).split("\n")
|
|
|
|
for line in lines:
|
|
|
|
parts = line.split('=', 1)
|
|
|
|
if len(parts) == 2:
|
|
|
|
args[parts[0]] = parts[1]
|
|
|
|
|
2014-10-22 21:48:11 +00:00
|
|
|
|
|
|
|
def eval_file(src):
|
2017-05-28 15:04:18 +02:00
|
|
|
""" Loads and evaluates the contents of the specified file. """
|
|
|
|
return eval(read_file(src), {'__builtins__': None}, None)
|
|
|
|
|
2014-10-22 21:48:11 +00:00
|
|
|
|
|
|
|
def normalize_path(path):
|
2017-05-28 15:04:18 +02:00
|
|
|
""" Normalizes the path separator to match the Unix standard. """
|
|
|
|
if sys.platform == 'win32':
|
|
|
|
return path.replace('\\', '/')
|
|
|
|
return path
|