tomlplusplus/python/utils.py
Mark Gillard 6f7f6c6554 moved the unicode function generator stuff to another repo
also:
- updated github templates
2020-08-01 22:37:39 +03:00

202 lines
4.4 KiB
Python

#!/usr/bin/env python3
# This file is a part of toml++ and is subject to the the terms of the MIT license.
# Copyright (c) 2019-2020 Mark Gillard <mark.gillard@outlook.com.au>
# See https://github.com/marzer/tomlplusplus/blob/master/LICENSE for the full license text.
# SPDX-License-Identifier: MIT
import sys
import os
import os.path as path
import subprocess
import traceback
import shutil
import fnmatch
import requests
import hashlib
def is_tool(name):
return shutil.which(name) is not None
__script_folder = None
def get_script_folder():
global __script_folder
if __script_folder is None:
__script_folder = path.dirname(path.realpath(sys.argv[0]))
return __script_folder
def read_all_text_from_file(path, fallback_url=None):
try:
print("Reading {}".format(path))
with open(path, 'r', encoding='utf-8') as f:
text = f.read()
return text
except:
if fallback_url is not None:
print("Couldn't read file locally, downloading from {}...".format(fallback_url))
response = requests.get(
fallback_url,
timeout=1
)
text = response.text
with open(path, 'w', encoding='utf-8', newline='\n') as f:
print(text, end='', file=f)
return text
else:
raise
def run_python_script(script_path, *args):
subprocess.check_call(
['py' if is_tool('py') else 'python3', script_path] + [arg for arg in args]
)
def repeat_pattern(pattern, count):
if len(pattern) == 1:
return pattern * count
text = ''
for i in range(0, count):
text = text + pattern[i % len(pattern)]
return text
def delete_directory(dir_path):
if (path.exists(dir_path)):
print('Deleting {}'.format(dir_path))
shutil.rmtree(dir_path)
def is_collection(val):
if isinstance(val, (list, tuple, dict, set, range)):
return True
return False
def is_pow2(v):
return v & (v-1) == 0
def next_power_of_2(n):
if n == 0:
return 1
if n & (n - 1) == 0:
return n
while n & (n - 1) > 0:
n &= (n - 1)
return n << 1
def get_all_files(dir, all=None, any=None):
files = [f for f in [path.join(dir, f) for f in os.listdir(dir)] if path.isfile(f)]
if (files and all is not None):
if (not is_collection(all)):
all = (all,)
all = [f for f in all if f is not None]
for fil in all:
files = fnmatch.filter(files, fil)
if (files and any is not None):
if (not is_collection(any)):
any = (any,)
any = [f for f in any if f is not None]
if any:
results = set()
for fil in any:
results.update(fnmatch.filter(files, fil))
files = [f for f in results]
return files
def make_divider(text = None, text_col = 40, pattern = '-', line_length = 120):
if (text is None):
return "//" + repeat_pattern(pattern, line_length-2)
else:
text = "//{} {} ".format(repeat_pattern(pattern, text_col - 2), text);
if (len(text) < line_length):
return text + repeat_pattern(pattern, line_length - len(text))
else:
return text
def __hash_combine_32(current_hash, next_hash):
current_hash = current_hash & 0xFFFFFFFF
next_hash = next_hash & 0xFFFFFFFF
current_hash = current_hash ^ (next_hash + 0x9E3779B9 + (current_hash << 6) + (current_hash >> 2))
return current_hash & 0xFFFFFFFF
def __hash_combine_64(current_hash, next_hash):
current_hash = current_hash & 0xFFFFFFFFFFFFFFFF
next_hash = next_hash & 0xFFFFFFFFFFFFFFFF
current_hash = current_hash ^ (next_hash + 0x9E3779B97F4A7C15 + (current_hash << 12) + (current_hash >> 4))
return current_hash & 0xFFFFFFFFFFFFFFFF
__hash_combine_func = None
if sys.hash_info.width == 64:
__hash_combine_func = __hash_combine_64
elif sys.hash_info.width == 32:
__hash_combine_func = __hash_combine_32
def hash_combine(current_hash, next_hash, *nexts):
global __hash_combine_func
current_hash = __hash_combine_func(current_hash, next_hash)
if nexts:
for n in nexts:
current_hash = __hash_combine_func(current_hash, n)
return current_hash
def multi_hash(obj, *objs):
assert obj is not None
h = hash(obj)
if objs:
for o in objs:
assert o is not None
h = hash_combine(h, hash(o))
return h
def multi_sha256(*objs):
assert objs
h = hashlib.sha256()
for o in objs:
assert o is not None
h.update(str(o).encode('utf-8'))
return h.hexdigest()
def run(main_func):
try:
result = main_func()
if result is None:
sys.exit(0)
else:
sys.exit(int(result))
except Exception as err:
print(
'Fatal error: [{}] {}'.format(
type(err).__name__,
str(err)
),
file=sys.stderr
)
traceback.print_exc(file=sys.stderr)
sys.exit(-1)