batch packer now works & reports useful output
This commit is contained in:
@@ -25,6 +25,45 @@ eg:
|
|||||||
bam_pack_test.py /path/to/blend_files
|
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
|
# Ensure module path
|
||||||
import os
|
import os
|
||||||
@@ -32,6 +71,7 @@ import sys
|
|||||||
path = os.path.normpath(os.path.join(os.path.dirname(os.path.abspath(__file__)), "..", "..", "client", "cli"))
|
path = os.path.normpath(os.path.join(os.path.dirname(os.path.abspath(__file__)), "..", "..", "client", "cli"))
|
||||||
if path not in sys.path:
|
if path not in sys.path:
|
||||||
sys.path.append(path)
|
sys.path.append(path)
|
||||||
|
|
||||||
del os, sys, 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):
|
def iter_files(path, filename_check=None):
|
||||||
for dirpath, dirnames, filenames in sorted(os.walk(path)):
|
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")
|
yield from iter_files(path, filename_check=lambda f: os.path.splitext(f)[1].lower() == ".blend")
|
||||||
|
|
||||||
|
|
||||||
def pack_blend_test(blendfile):
|
def pack_blend_test(blendfile_src, log, blender_bin):
|
||||||
TEMP_ZIP = "temp.zip"
|
|
||||||
|
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 = (
|
argv = (
|
||||||
"pack", blendfile,
|
"pack", blendfile_src,
|
||||||
"--output", TEMP_ZIP,
|
"--output", TEMP_ZIP,
|
||||||
"--quiet",
|
"--quiet",
|
||||||
)
|
)
|
||||||
|
|
||||||
import bam
|
import bam
|
||||||
print("bam", " ".join(argv))
|
log.info("bam " + " ".join(argv))
|
||||||
bam.main(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(
|
def pack_blend_recursive_test(
|
||||||
paths,
|
paths,
|
||||||
|
log,
|
||||||
blender_bin="blender",
|
blender_bin="blender",
|
||||||
):
|
):
|
||||||
|
num_blend = 0
|
||||||
|
num_blend_fail = 0
|
||||||
for path in paths:
|
for path in paths:
|
||||||
for f in iter_blends(path):
|
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():
|
def create_argparse():
|
||||||
@@ -104,6 +277,11 @@ def create_argparse():
|
|||||||
|
|
||||||
|
|
||||||
def main(argv=None):
|
def main(argv=None):
|
||||||
|
import logging
|
||||||
|
|
||||||
|
logging.basicConfig(level=logging.DEBUG)
|
||||||
|
log = logging.getLogger("PACK")
|
||||||
|
log.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
if argv is None:
|
if argv is None:
|
||||||
argv = sys.argv[1:]
|
argv = sys.argv[1:]
|
||||||
@@ -113,10 +291,10 @@ def main(argv=None):
|
|||||||
|
|
||||||
pack_blend_recursive_test(
|
pack_blend_recursive_test(
|
||||||
args.paths,
|
args.paths,
|
||||||
|
log,
|
||||||
blender_bin=args.blender_bin or "blender",
|
blender_bin=args.blender_bin or "blender",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
main()
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user