batch packer now works & reports useful output

This commit is contained in:
2014-12-17 21:55:29 +01:00
parent bcb5184b45
commit 05ebc867e0

View File

@@ -25,6 +25,45 @@ eg:
bam_pack_test.py /path/to/blend_files
"""
VERBOSE = 0
BLENDER_EXIT_CODE = 3
# --------------------
# Runs inside Blender!
#
# Cheap hack to avoid having 2x scripts!
import sys
bpy = sys.modules.get("bpy")
if bpy is not None:
import os
# ----
# write paths
argv = sys.argv
argv = argv[argv.index("--") + 1:]
FILE_OUT = argv[0]
data = bpy.utils.blend_paths(absolute=True)
data = [(f, os.path.exists(f)) for f in data]
import json
with open(FILE_OUT, 'w', encoding='utf-8') as f:
json.dump(
data, f, ensure_ascii=False,
check_circular=False,
# optional (pretty)
sort_keys=True, indent=4, separators=(',', ': '),
)
sys.exit(BLENDER_EXIT_CODE)
del bpy
# End Blender Code!
# -----------------
# ------------------
# Ensure module path
import os
@@ -32,6 +71,7 @@ import sys
path = os.path.normpath(os.path.join(os.path.dirname(os.path.abspath(__file__)), "..", "..", "client", "cli"))
if path not in sys.path:
sys.path.append(path)
del os, sys, path
# --------
@@ -40,6 +80,36 @@ import sys
def args_as_string(args):
""" Print args so we can paste them to run them again.
"""
import shlex
return " ".join([shlex.quote(c) for c in args])
def run(cmd, cwd=None):
if VERBOSE:
print(">>> ", args_as_string(cmd))
import subprocess
kwargs = {
"stderr": subprocess.PIPE,
"stdout": subprocess.PIPE,
}
if cwd is not None:
kwargs["cwd"] = cwd
proc = subprocess.Popen(cmd, **kwargs)
stdout, stderr = proc.communicate()
returncode = proc.returncode
if VERBOSE:
sys.stdout.write(" stdout: %s\n" % stdout.strip())
sys.stdout.write(" stderr: %s\n" % stderr.strip())
sys.stdout.write(" return: %d\n" % returncode)
return stdout, stderr, returncode
def iter_files(path, filename_check=None):
for dirpath, dirnames, filenames in sorted(os.walk(path)):
@@ -57,26 +127,129 @@ def iter_blends(path):
yield from iter_files(path, filename_check=lambda f: os.path.splitext(f)[1].lower() == ".blend")
def pack_blend_test(blendfile):
TEMP_ZIP = "temp.zip"
def pack_blend_test(blendfile_src, log, blender_bin):
def json_from_file(filepath):
with open(filepath, 'r', encoding='utf-8') as f:
import json
return json.load(f)
def args_blender_read_paths(filepath):
return (
blender_bin,
"--background",
filepath,
"-noaudio",
"--python", __file__,
"--",
FILE_OUT,
)
def unzip(filepath_src, path_dst):
import zipfile
with zipfile.ZipFile(filepath_src) as zf:
zf.extractall(path_dst)
import shutil
TEMP_ZIP = "/tmp/temp.zip"
TEMP_EXTRACT = "/tmp/temp_out"
FILE_OUT = "/tmp/files.json"
def cleanup():
_ = TEMP_EXTRACT
if os.path.exists(_):
shutil.rmtree(_)
_ = TEMP_ZIP
if os.path.exists(_):
os.remove(_)
_ = FILE_OUT
if os.path.exists(_):
os.remove(_)
argv = (
"pack", blendfile,
"pack", blendfile_src,
"--output", TEMP_ZIP,
"--quiet",
)
import bam
print("bam", " ".join(argv))
log.info("bam " + " ".join(argv))
bam.main(argv)
# extract zip
os.makedirs(TEMP_EXTRACT, exist_ok=True)
unzip(TEMP_ZIP, TEMP_EXTRACT)
os.remove(TEMP_ZIP)
blendfile_dst = os.path.join(TEMP_EXTRACT, os.path.basename(blendfile_src))
stdout_src, stderr_src, returncode = run(args_blender_read_paths(blendfile_src))
if returncode != BLENDER_EXIT_CODE:
log.error("Python exception running blender!")
cleanup()
return True
data_src = json_from_file(FILE_OUT)
os.remove(FILE_OUT)
stdout_dst, stderr_dst, returncode = run(args_blender_read_paths(blendfile_dst))
if returncode != BLENDER_EXIT_CODE:
log.error("Python exception running blender! (aborting this file!)")
cleanup()
return True
data_dst = json_from_file(FILE_OUT)
os.remove(FILE_OUT)
shutil.rmtree(TEMP_EXTRACT)
del returncode
is_error = False
# just extra check... not essential but means we know quickly if library state is different
if stdout_src.count(b'LIB ERROR') != stdout_dst.count(b'LIB ERROR'):
log.error("Library errors differ in packed library, with the following output")
log.error("*** SOURCE STDOUT ***\n" + stdout_src.decode('utf-8'))
log.error("*** PACKED STDOUT ***\n" + stdout_dst.decode('utf-8'))
is_error = True
data_src_basename = {os.path.basename(f_full): (f_full, f_ok) for f_full, f_ok in data_src}
data_dst_basename = {os.path.basename(f_full): (f_full, f_ok) for f_full, f_ok in data_dst}
# do magic!
for f_src_nameonly, (f_src_full, f_src_ok) in data_src_basename.items():
if f_src_ok:
f_dst_full, f_dst_ok = data_dst_basename[f_src_nameonly]
if not f_dst_ok:
log.error("%r (%r -> %r) failed!" % (blendfile_src, f_src_full, f_dst_full))
is_error = True
else:
# log.info("found %r -> %r" % (f_src_full, f_dst_full))
pass
return is_error
def pack_blend_recursive_test(
paths,
log,
blender_bin="blender",
):
num_blend = 0
num_blend_fail = 0
for path in paths:
for f in iter_blends(path):
pack_blend_test(f)
try:
is_error = pack_blend_test(f, log, blender_bin)
except Exception as e:
log.exception(e)
is_error = True
num_blend_fail += is_error
num_blend += 1
log.info("tally, %d blends, %d failed" % (num_blend, num_blend_fail))
def create_argparse():
@@ -104,6 +277,11 @@ def create_argparse():
def main(argv=None):
import logging
logging.basicConfig(level=logging.DEBUG)
log = logging.getLogger("PACK")
log.setLevel(logging.DEBUG)
if argv is None:
argv = sys.argv[1:]
@@ -113,10 +291,10 @@ def main(argv=None):
pack_blend_recursive_test(
args.paths,
log,
blender_bin=args.blender_bin or "blender",
)
if __name__ == "__main__":
main()