import inspect
import pkgutil
import importlib
from typing import IO, Callable, Iterator, Union, Dict, Any, Tuple
from struct import unpack, pack
from itertools import repeat
from jawa.constants import UTF8
from jawa.util.stream import BufferStreamReader
[docs]class Attribute(object):
ADDED_IN: int = None
MINIMUM_CLASS_VERSION: Tuple[int, int] = None
def __init__(self, parent: 'AttributeTable', name_index: int):
self.parent = parent
self.name_index = name_index
@property
def name(self) -> UTF8:
"""
The name of this attribute.
"""
return self.cf.constants[self.name_index]
@property
def cf(self):
"""
The ClassFile that owns this attribute, if any.
"""
return self.parent.cf
[docs] def unpack(self, info: Union[bytes, BufferStreamReader]):
"""
Parses an instance of this attribute from the blob `info`.
"""
raise NotImplementedError()
[docs] def pack(self) -> bytes:
"""
This attribute packed into its on-disk representation.
"""
raise NotImplementedError()
[docs]class UnknownAttribute(Attribute):
def __init__(self, parent: 'AttributeTable', name_index: int):
super().__init__(parent, name_index)
self.info = None
[docs] def unpack(self, info: Union[bytes, BufferStreamReader]):
self.info = info
[docs] def pack(self) -> bytes:
return self.info
[docs]class AttributeTable(object):
def __init__(self, cf, parent: Attribute=None):
#: The ClassFile that ultimately owns this AttributeTable.
self.cf = cf
#: The parent Attribute, if one exists.
self.parent = parent
self._table = []
[docs] def unpack(self, source: IO):
"""
Read the ConstantPool from the file-like object `source`.
.. note::
Advanced usage only. You will typically never need to call this
method as it will be called for you when loading a ClassFile.
:param source: Any file-like object providing `read()`
"""
count = unpack('>H', source.read(2))[0]
for _ in repeat(None, count):
name_index, length = unpack('>HI', source.read(6))
info_blob = source.read(length)
self._table.append((name_index, info_blob))
def __getitem__(self, key):
attr = self._table[key]
if not isinstance(attr, Attribute):
name_index, info = attr[0], attr[1]
name = self.cf.constants[name_index].value
attribute_type = ATTRIBUTE_CLASSES.get(name, UnknownAttribute)
self._table[key] = attr = attribute_type(self, name_index)
if attribute_type is UnknownAttribute:
attr.unpack(info)
else:
attr.unpack(BufferStreamReader(info))
return attr
def __len__(self):
return len(self._table)
[docs] def pack(self, out: IO):
"""
Write the AttributeTable to the file-like object `out`.
.. note::
Advanced usage only. You will typically never need to call this
method as it will be called for you when saving a ClassFile.
:param out: Any file-like object providing `write()`
"""
out.write(pack('>H', len(self._table)))
for attribute in self:
info = attribute.pack()
out.write(pack(
'>HI',
attribute.name.index,
len(info)
))
out.write(info)
[docs] def create(self, type_, *args, **kwargs) -> Any:
"""
Creates a new attribute of `type_`, appending it to the attribute
table and returning it.
"""
attribute = type_(self, *args, **kwargs)
self._table.append(attribute)
return attribute
[docs] def find(self, *, name: str=None, f: Callable=None) -> Iterator[Any]:
for idx, attribute in enumerate(self._table):
if name is not None:
# Optimization to filter solely on name without causing
# a full attribute load.
if not isinstance(attribute, Attribute) and f is None:
attr_name = self.cf.constants[attribute[0]].value
if attr_name != name:
continue
elif name != attribute.name.value:
continue
# Force an attribute load.
if not isinstance(attribute, Attribute):
attribute = self[idx]
if f is not None and not f(attribute):
continue
yield attribute
[docs] def find_one(self, **kwargs) -> Any:
"""
Same as ``find()`` but returns only the first result.
"""
return next(self.find(**kwargs), None)
[docs]def get_attribute_classes() -> Dict[str, Attribute]:
"""
Lookup all builtin Attribute subclasses, load them, and return a dict
"""
attribute_children = pkgutil.iter_modules(
importlib.import_module('jawa.attributes').__path__,
prefix='jawa.attributes.'
)
result = {}
for _, name, _ in attribute_children:
classes = inspect.getmembers(
importlib.import_module(name),
lambda c: (
inspect.isclass(c) and issubclass(c, Attribute) and
c is not Attribute
)
)
for class_name, class_ in classes:
attribute_name = getattr(class_, 'ATTRIBUTE_NAME', class_name[:-9])
result[attribute_name] = class_
return result
#: A dictionary of known attribute subclasses at the time this module
#: was loaded.
ATTRIBUTE_CLASSES = get_attribute_classes()