#!/usr/bin/env python3
# This file is placed in the Public Domain.
#
# pylint: disable=C,R,W0212,W0611,W0613,W0105,E0401,E0402


"python3 irc bot"


import getpass
import inspect
import os
import pwd
import readline
import sys
import termios
import time
import _thread


sys.path.insert(0, os.getcwd())


from objx import Object
from botl import Broker, Client, Command, Default, Error, Event, Handler, Storage
from botl import cdir, cmnd, debug, forever, launch, parse_cmd, scan, spl


Cfg         = Default()
Cfg.mod     = "cmd,irc,rss,mod"
Cfg.name    = "zbot"
Cfg.wd      = os.path.expanduser(f"~/.{Cfg.name}")
Cfg.pidfile = os.path.join(Cfg.wd, f"{Cfg.name}.pid")
Cfg.user    = getpass.getuser()
Storage.wd  = Cfg.wd


import zbot


if os.path.exists("mods"):
    import mods
else:
    mods = None


class Console(Client):

    def announce(self, txt):
        pass

    def callback(self, evt):
        Client.callback(self, evt)
        evt.wait()

    def poll(self):
        evt = Event()
        evt.orig = object.__repr__(self)
        evt.txt = input("> ")
        evt.type = "command"
        return evt

    def say(self, channel, txt):
        txt = txt.encode('utf-8', 'replace').decode()
        print(txt)


def daemon(pidfile, verbose=False):
    pid = os.fork()
    if pid != 0:
        os._exit(0)
    os.setsid()
    pid2 = os.fork()
    if pid2 != 0:
        os._exit(0)
    if not verbose:
        with open('/dev/null', 'r', encoding="utf-8") as sis:
            os.dup2(sis.fileno(), sys.stdin.fileno())
        with open('/dev/null', 'a+', encoding="utf-8") as sos:
            os.dup2(sos.fileno(), sys.stdout.fileno())
        with open('/dev/null', 'a+', encoding="utf-8") as ses:
            os.dup2(ses.fileno(), sys.stderr.fileno())
    os.umask(0)
    os.chdir("/")
    if os.path.exists(pidfile):
        os.unlink(pidfile)
    cdir(os.path.dirname(pidfile))
    with open(pidfile, "w", encoding="utf-8") as fds:
        fds.write(str(os.getpid()))


def daemoned():
    daemon(Cfg.pidfile)
    privileges(Cfg.user)
    scan(zbot, Cfg.mod, True)
    forever()


def privileges(username):
    pwnam = pwd.getpwnam(username)
    os.setgid(pwnam.pw_gid)
    os.setuid(pwnam.pw_uid)


def wrap(func):
    old2 = None
    try:
        old2 = termios.tcgetattr(sys.stdin.fileno())
    except termios.error:
        pass
    try:
        func()
    except (KeyboardInterrupt, EOFError):
        print("")
    finally:
        if old2:
            termios.tcsetattr(sys.stdin.fileno(), termios.TCSADRAIN, old2)


def main():
    readline.redisplay()
    Storage.skel()
    parse_cmd(Cfg, " ".join(sys.argv[1:]))
    hasmod = Cfg.hasmods or "a" in Cfg.opts or False
    if "a" in Cfg.opts:
        Cfg.mod = ",".join(zbot.__dir__())
    if mods:
        Cfg.mod += "," + ",".join(mods.__dir__())
    print(Cfg.mod)
    if "v" in Cfg.opts:
        Error.output = print
        dte = time.ctime(time.time()).replace("  ", " ")
        debug(f"{Cfg.name.upper()} {Cfg.opts.upper()} started {dte}")
    if "d" in Cfg.opts:
        daemoned()
    if "h" in Cfg.opts:
        return print(__doc__)
    if "c" in Cfg.opts:
        scan(zbot, Cfg.mod, hasmod, Cfg.sets.dis, hasmod)
        scan(mods, Cfg.mod, hasmod, Cfg.sets.dis, hasmod)
        csl = Console()
        csl.start()
        forever()
    if Cfg.otxt:
        scan(zbot, Cfg.mod)
        scan(mods, Cfg.mod)
        return cmnd(Cfg.otxt, print)


def wrapped():
    wrap(main)
    Error.show()


if __name__ == "__main__":
    wrapped()
