"""
Olark template tags.
"""

import json
import re

from django.template import Library, Node, TemplateSyntaxError

from analytical.utils import get_identity, get_required_setting

SITE_ID_RE = re.compile(r'^\d+-\d+-\d+-\d+$')
SETUP_CODE = """
    <script type='text/javascript'>
      /*{literal}<![CDATA[*/ window.olark||(function(k){var g=window,j=document,a=g.location.protocol=="https:"?"https:":"http:",i=k.name,b="load",h="addEventListener";(function(){g[i]=function(){(c.s=c.s||[]).push(arguments)};var c=g[i]._={},f=k.methods.length;while(f--){(function(l){g[i][l]=function(){g[i]("call",l,arguments)}})(k.methods[f])}c.l=k.loader;c.i=arguments.callee;c.p={0:+new Date};c.P=function(l){c.p[l]=new Date-c.p[0]};function e(){c.P(b);g[i](b)}g[h]?g[h](b,e,false):g.attachEvent("on"+b,e);c.P(1);var d=j.createElement("script"),m=document.getElementsByTagName("script")[0];d.type="text/javascript";d.async=true;d.src=a+"//"+c.l;m.parentNode.insertBefore(d,m);c.P(2)})()})({loader:(function(a){return "static.olark.com/jsclient/loader1.js?ts="+(a?a[1]:(+new Date))})(document.cookie.match(/olarkld=([0-9]+)/)),name:"olark",methods:["configure","extend","declare","identify"]}); olark.identify('%(site_id)s');/*]]>{/literal}*/
      %(extra_code)s
    </script>
"""  # noqa
NICKNAME_CODE = "olark('api.chat.updateVisitorNickname', {snippet: '%s'});"
NICKNAME_CONTEXT_KEY = 'olark_nickname'
FULLNAME_CODE = "olark('api.visitor.updateFullName', {{fullName: '{0}'}});"
FULLNAME_CONTEXT_KEY = 'olark_fullname'
EMAIL_CODE = "olark('api.visitor.updateEmailAddress', {{emailAddress: '{0}'}});"
EMAIL_CONTEXT_KEY = 'olark_email'
STATUS_CODE = "olark('api.chat.updateVisitorStatus', {snippet: %s});"
STATUS_CONTEXT_KEY = 'olark_status'
MESSAGE_CODE = 'olark.configure(\'locale.%(key)s\', "%(msg)s");'
MESSAGE_KEYS = {
    'welcome_title',
    'chatting_title',
    'unavailable_title',
    'busy_title',
    'away_message',
    'loading_title',
    'welcome_message',
    'busy_message',
    'chat_input_text',
    'name_input_text',
    'email_input_text',
    'offline_note_message',
    'send_button_text',
    'offline_note_thankyou_text',
    'offline_note_error_text',
    'offline_note_sending_text',
    'operator_is_typing_text',
    'operator_has_stopped_typing_text',
    'introduction_error_text',
    'introduction_messages',
    'introduction_submit_button_text',
}

register = Library()


@register.tag
def olark(parser, token):
    """
    Olark set-up template tag.

    Renders Javascript code to set-up Olark chat.  You must supply
    your site ID in the ``OLARK_SITE_ID`` setting.
    """
    bits = token.split_contents()
    if len(bits) > 1:
        raise TemplateSyntaxError("'%s' takes no arguments" % bits[0])
    return OlarkNode()


class OlarkNode(Node):
    def __init__(self):
        self.site_id = get_required_setting(
            'OLARK_SITE_ID',
            SITE_ID_RE,
            "must be a string looking like 'XXXX-XXX-XX-XXXX'",
        )

    def render(self, context):
        extra_code = []
        try:
            extra_code.append(NICKNAME_CODE % context[NICKNAME_CONTEXT_KEY])
        except KeyError:
            identity = get_identity(context, 'olark', self._get_nickname)
            if identity is not None:
                extra_code.append(NICKNAME_CODE % identity)
        try:
            extra_code.append(FULLNAME_CODE.format(context[FULLNAME_CONTEXT_KEY]))
        except KeyError:
            pass
        try:
            extra_code.append(EMAIL_CODE.format(context[EMAIL_CONTEXT_KEY]))
        except KeyError:
            pass
        try:
            extra_code.append(
                STATUS_CODE % json.dumps(context[STATUS_CONTEXT_KEY], sort_keys=True)
            )
        except KeyError:
            pass
        extra_code.extend(self._get_configuration(context))
        html = SETUP_CODE % {
            'site_id': self.site_id,
            'extra_code': ' '.join(extra_code),
        }
        return html

    def _get_nickname(self, user):
        name = user.get_full_name()
        if name:
            return '%s (%s)' % (name, user.username)
        else:
            return user.username

    def _get_configuration(self, context):
        code = []
        for dict_ in context:
            for var, val in dict_.items():
                if var.startswith('olark_'):
                    key = var[6:]
                    if key in MESSAGE_KEYS:
                        code.append(MESSAGE_CODE % {'key': key, 'msg': val})
        return code


def contribute_to_analytical(add_node):
    OlarkNode()  # ensure properly configured
    add_node('body_bottom', OlarkNode)
