import logging

from django.utils.safestring import SafeData, mark_safe
from lxml import html
from lxml.etree import ParserError, XMLSyntaxError

logger = logging.getLogger("judge.html")


class HTMLTreeString(SafeData):
    def __init__(self, str):
        try:
            self._tree = html.fromstring(str, parser=html.HTMLParser(recover=True))
        except (XMLSyntaxError, ParserError) as e:
            if str and (
                not isinstance(e, ParserError) or e.args[0] != "Document is empty"
            ):
                logger.exception("Failed to parse HTML string")
            self._tree = html.Element("div")

    def __getattr__(self, attr):
        try:
            return getattr(self._tree, attr)
        except AttributeError:
            return getattr(str(self), attr)

    def __setattr__(self, key, value):
        if key[0] == "_":
            super(HTMLTreeString, self).__setattr__(key, value)
        setattr(self._tree, key, value)

    def __repr__(self):
        return "<HTMLTreeString %r>" % str(self)

    def __str__(self):
        return mark_safe(html.tostring(self._tree, encoding="unicode"))

    def __radd__(self, other):
        return other + str(self)

    def __add__(self, other):
        return str(self) + other

    def __getitem__(self, item):
        return str(self)[item]

    def __getstate__(self):
        return str(self)

    def __setstate__(self, state):
        self._tree = html.fromstring(state)

    @property
    def tree(self):
        return self._tree


def fromstring(str):
    if isinstance(str, HTMLTreeString):
        return str
    return HTMLTreeString(str)