it's messy but it works
Implement construction of integer values and characters. One-dimensional arrays are handled but are not constructed. I've also added an inspect() method to the BlendStruct class, which prints a nicely formatted dict with json.
This commit is contained in:
parent
6739b9d471
commit
c618ab5d35
@ -1,6 +1,8 @@
|
|||||||
# A purpose-built class to parse certain information from .blend files.
|
# A purpose-built class to parse certain information from .blend files.
|
||||||
|
|
||||||
|
import json
|
||||||
import io
|
import io
|
||||||
|
import re
|
||||||
|
|
||||||
class BlendStruct:
|
class BlendStruct:
|
||||||
"""
|
"""
|
||||||
@ -34,6 +36,12 @@ class BlendStruct:
|
|||||||
else:
|
else:
|
||||||
return str(self._structure)
|
return str(self._structure)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
if self._structure is None:
|
||||||
|
return f"<Blender Structure {self._type} (unloaded)>"
|
||||||
|
else:
|
||||||
|
return f"<Blender Structure {self._type} (loaded)>"
|
||||||
|
|
||||||
def __getitem__(self, item):
|
def __getitem__(self, item):
|
||||||
if self._structure is None:
|
if self._structure is None:
|
||||||
self._structure = self._load_cb()
|
self._structure = self._load_cb()
|
||||||
@ -42,7 +50,18 @@ class BlendStruct:
|
|||||||
def __iter__(self):
|
def __iter__(self):
|
||||||
if self._structure is None:
|
if self._structure is None:
|
||||||
self._structure = self._load_cb()
|
self._structure = self._load_cb()
|
||||||
return self._structure
|
return self._structure.__iter__()
|
||||||
|
|
||||||
|
def inspect(self):
|
||||||
|
"""
|
||||||
|
Return a human readable representation of the struct.
|
||||||
|
"""
|
||||||
|
summary = {}
|
||||||
|
if self._structure is None:
|
||||||
|
self._structure = self._load_cb()
|
||||||
|
for field, value in self._structure.items():
|
||||||
|
summary[field] = repr(value)
|
||||||
|
return json.dumps(summary, indent=4)
|
||||||
|
|
||||||
# A class for opening and reading data from .blend files
|
# A class for opening and reading data from .blend files
|
||||||
class Blendfile(io.FileIO):
|
class Blendfile(io.FileIO):
|
||||||
@ -68,6 +87,39 @@ class Blendfile(io.FileIO):
|
|||||||
return res.decode("utf-8")
|
return res.decode("utf-8")
|
||||||
res += char
|
res += char
|
||||||
|
|
||||||
|
def _construct_value(self, type, is_ptr, length):
|
||||||
|
"""
|
||||||
|
Construct a value from a type.
|
||||||
|
|
||||||
|
This is gonna be messy; documentation can come later.
|
||||||
|
"""
|
||||||
|
if length > 1:
|
||||||
|
arr = []
|
||||||
|
# Skip for now.
|
||||||
|
size = self._sdna["tlen"][type]
|
||||||
|
for _ in range(length):
|
||||||
|
self.seek(size, io.SEEK_CUR)
|
||||||
|
#arr.append(self._construct_value(type, False, 1))
|
||||||
|
return arr
|
||||||
|
|
||||||
|
INT_TYPES = ("short", "int", "long", "long long")
|
||||||
|
if type in self._sdna["structs"]:
|
||||||
|
offset = self.tell()
|
||||||
|
self.seek(self._sdna["tlen"][type], io.SEEK_CUR)
|
||||||
|
return BlendStruct(self._struct_loader(type, offset), type)
|
||||||
|
elif type in INT_TYPES:
|
||||||
|
size = self._sdna["tlen"][type]
|
||||||
|
return int.from_bytes(self.read(size), self.endianness)
|
||||||
|
elif type == "char":
|
||||||
|
if is_ptr:
|
||||||
|
return self._read_c_string()
|
||||||
|
else:
|
||||||
|
return self.read(1)
|
||||||
|
else:
|
||||||
|
size = self._sdna["tlen"][type]
|
||||||
|
self.seek(size, io.SEEK_CUR)
|
||||||
|
return type
|
||||||
|
|
||||||
def _read_block_offsets(self):
|
def _read_block_offsets(self):
|
||||||
"""
|
"""
|
||||||
Cache the offset to each file block.
|
Cache the offset to each file block.
|
||||||
@ -146,9 +198,10 @@ class Blendfile(io.FileIO):
|
|||||||
# Type length identifier; should be "TLEN"
|
# Type length identifier; should be "TLEN"
|
||||||
len_identifier = self.read(4).decode("utf-8")
|
len_identifier = self.read(4).decode("utf-8")
|
||||||
assert(len_identifier == "TLEN")
|
assert(len_identifier == "TLEN")
|
||||||
type_lengths = []
|
type_lengths = {}
|
||||||
for _ in range(total_types):
|
for i in range(total_types):
|
||||||
type_lengths.append(int.from_bytes(self.read(2), self.endianness))
|
length = int.from_bytes(self.read(2), self.endianness)
|
||||||
|
type_lengths[_types[i]] = length
|
||||||
sdna["tlen"] = type_lengths
|
sdna["tlen"] = type_lengths
|
||||||
|
|
||||||
# Align at 4 bytes.
|
# Align at 4 bytes.
|
||||||
@ -185,8 +238,19 @@ class Blendfile(io.FileIO):
|
|||||||
:param offset: The byte offset at which to begin loading.
|
:param offset: The byte offset at which to begin loading.
|
||||||
:return The loaded structure.
|
:return The loaded structure.
|
||||||
"""
|
"""
|
||||||
|
structure = {}
|
||||||
fields = self._sdna["structs"][struct_name]
|
fields = self._sdna["structs"][struct_name]
|
||||||
return {}
|
for name, type in fields.items():
|
||||||
|
lengths = re.findall(r"\[([0-9]+)\]", name)
|
||||||
|
if len(lengths) > 1:
|
||||||
|
raise ValueError(f"Can't handle nested array {name}.")
|
||||||
|
elif len(lengths) == 1:
|
||||||
|
length = int(lengths[0])
|
||||||
|
else:
|
||||||
|
length = 1
|
||||||
|
is_ptr = name.startswith("*")
|
||||||
|
structure[name] = self._construct_value(type, is_ptr, length)
|
||||||
|
return structure
|
||||||
|
|
||||||
def read_header(self):
|
def read_header(self):
|
||||||
self.seek(0)
|
self.seek(0)
|
||||||
@ -226,6 +290,12 @@ class Blendfile(io.FileIO):
|
|||||||
matched_blocks[identifier] = create_loader(offset)
|
matched_blocks[identifier] = create_loader(offset)
|
||||||
return matched_blocks
|
return matched_blocks
|
||||||
|
|
||||||
|
# Helper for creating a callback to load a given structure.
|
||||||
|
def _struct_loader(self, name, at_offset):
|
||||||
|
def load_struct():
|
||||||
|
return self._load_struct(name, at_offset)
|
||||||
|
return load_struct
|
||||||
|
|
||||||
def _load_block(self, offset):
|
def _load_block(self, offset):
|
||||||
"""
|
"""
|
||||||
Load a file block at a given offset to the beginning of the blend file.
|
Load a file block at a given offset to the beginning of the blend file.
|
||||||
@ -243,13 +313,7 @@ class Blendfile(io.FileIO):
|
|||||||
sdna_index = int.from_bytes(self.read(4), self.endianness)
|
sdna_index = int.from_bytes(self.read(4), self.endianness)
|
||||||
count = int.from_bytes(self.read(4), self.endianness)
|
count = int.from_bytes(self.read(4), self.endianness)
|
||||||
|
|
||||||
# Helper for creating a callback to load a given structure.
|
|
||||||
def create_loader(name, at_offset):
|
|
||||||
def load_struct():
|
|
||||||
return self._load_struct(name, at_offset)
|
|
||||||
return load_struct
|
|
||||||
|
|
||||||
# The type of the structure.
|
# The type of the structure.
|
||||||
name = list(self._sdna["structs"])[sdna_index]
|
name = list(self._sdna["structs"])[sdna_index]
|
||||||
for _ in range(count):
|
for _ in range(count):
|
||||||
yield BlendStruct(create_loader(name, self.tell()), name)
|
yield BlendStruct(self._struct_loader(name, self.tell()), name)
|
2
test.py
2
test.py
@ -14,4 +14,4 @@ with blendparse.Blendfile(blendfile) as blend:
|
|||||||
results = [block() for block in blend.get_blocks("SC").values()]
|
results = [block() for block in blend.get_blocks("SC").values()]
|
||||||
for structure in results[0]:
|
for structure in results[0]:
|
||||||
print(structure)
|
print(structure)
|
||||||
print(structure.load())
|
print(structure.inspect())
|
Loading…
x
Reference in New Issue
Block a user