# Heater/sensor verification code
#
# Copyright (C) 2018  Kevin O'Connor <kevin@koconnor.net>
#
# This file may be distributed under the terms of the GNU GPLv3 license.
import logging

HINT_THERMAL = """
See the 'verify_heater' section in config/example-extras.cfg
for the parameters that control this check.
"""

class HeaterCheck:
    def __init__(self, config):
        self.printer = config.get_printer()
        self.heater_name = config.get_name().split()[1]
        self.heater = None
        self.hysteresis = config.getfloat('hysteresis', 5., minval=0.)
        self.max_error = config.getfloat('max_error', 120., minval=0.)
        self.heating_gain = config.getfloat('heating_gain', 2., above=0.)
        default_gain_time = 20.
        if self.heater_name == 'heater_bed':
            default_gain_time = 60.
        self.check_gain_time = config.getfloat(
            'check_gain_time', default_gain_time, minval=1.)
        self.met_target = False
        self.last_target = self.goal_temp = self.error = 0.
        self.fault_systime = self.printer.get_reactor().NEVER
        self.check_timer = None
    def printer_state(self, state):
        if state == 'connect':
            if self.printer.get_start_args().get('debugoutput') is not None:
                # Disable verify_heater if outputting to a debug file
                return
            pheater = self.printer.lookup_object('heater')
            self.heater = pheater.lookup_heater(self.heater_name)
            logging.info("Starting heater checks for %s", self.heater_name)
            reactor = self.printer.get_reactor()
            self.check_timer = reactor.register_timer(self.check_event,
                                                      reactor.NOW)
        elif state == 'shutdown' and self.check_timer is not None:
            reactor = self.printer.get_reactor()
            reactor.update_timer(self.check_timer, reactor.NEVER)
    def check_event(self, eventtime):
        temp, target = self.heater.get_temp(eventtime)
        if temp >= target - self.hysteresis:
            # Temperature near target - reset checks
            if not self.met_target and target:
                logging.info("Heater %s within range of %.3f",
                             self.heater_name, target)
            self.met_target = True
            self.error = 0.
        elif self.met_target:
            self.error += (target - self.hysteresis) - temp
            if target != self.last_target:
                # Target changed - reset checks
                logging.info("Heater %s approaching new target of %.3f",
                             self.heater_name, target)
                self.met_target = False
                self.goal_temp = temp + self.heating_gain
                self.fault_systime = eventtime + self.check_gain_time
            elif self.error >= self.max_error:
                # Failure due to inability to maintain target temperature
                return self.heater_fault()
        elif temp >= self.goal_temp:
            # Temperature approaching target - reset checks
            self.goal_temp = temp + self.heating_gain
            self.fault_systime = eventtime + self.check_gain_time
        elif eventtime >= self.fault_systime:
            # Failure due to inability to approach target temperature
            return self.heater_fault()
        self.last_target = target
        return eventtime + 1.
    def heater_fault(self):
        msg = "Heater %s not heating at expected rate" % (self.heater_name,)
        logging.error(msg)
        self.printer.invoke_shutdown(msg + HINT_THERMAL)
        return self.printer.get_reactor().NEVER

def load_config_prefix(config):
    return HeaterCheck(config)