#include <unistd.h>

#include "components/logger.hpp"
#include "errors.hpp"
#include "utils/factory.hpp"
#include "utils/string.hpp"

POLYBAR_NS

/**
 * Create instance
 */
const logger& logger::make(loglevel level) {
  auto instance = factory_util::singleton<const logger>(level);
  return static_cast<const logger&>(*instance);
}

/**
 * Construct logger
 */
logger::logger(loglevel level) : m_level(level) {
  // clang-format off
  if (isatty(m_fd)) {
    m_prefixes[loglevel::TRACE]   = "\r\033[0;90m- ";
    m_prefixes[loglevel::INFO]    = "\r\033[1;32m* \033[0m";
    m_prefixes[loglevel::WARNING] = "\r\033[1;33mwarn: \033[0m";
    m_prefixes[loglevel::ERROR]   = "\r\033[1;31merror: \033[0m";
    m_suffixes[loglevel::TRACE]   = "\033[0m";
    m_suffixes[loglevel::INFO]    = "\033[0m";
    m_suffixes[loglevel::WARNING] = "\033[0m";
    m_suffixes[loglevel::ERROR]   = "\033[0m";
  } else {
    m_prefixes.emplace(make_pair(loglevel::TRACE,   "polybar|trace  "));
    m_prefixes.emplace(make_pair(loglevel::INFO,    "polybar|infoe  "));
    m_prefixes.emplace(make_pair(loglevel::WARNING, "polybar|warne  "));
    m_prefixes.emplace(make_pair(loglevel::ERROR,   "polybar|error  "));
    m_suffixes.emplace(make_pair(loglevel::TRACE,   ""));
    m_suffixes.emplace(make_pair(loglevel::INFO,    ""));
    m_suffixes.emplace(make_pair(loglevel::WARNING, ""));
    m_suffixes.emplace(make_pair(loglevel::ERROR,   ""));
  }
  // clang-format on
}

/**
 * Set output verbosity
 */
void logger::verbosity(loglevel level) {
#ifndef DEBUG
  if (level == loglevel::TRACE) {
    throw application_error("not a debug build: trace disabled...");
  }
#endif
  m_level = level;
}

/**
 * Set output verbosity by loglevel name
 */
void logger::verbosity(string level) {
  verbosity(parse_loglevel_name(move(level)));
}

/**
 * Convert given loglevel name to its enum type counterpart
 */
loglevel parse_loglevel_name(const string& name) {
  if (string_util::compare(name, "error")) {
    return loglevel::ERROR;
  } else if (string_util::compare(name, "warning")) {
    return loglevel::WARNING;
  } else if (string_util::compare(name, "info")) {
    return loglevel::INFO;
  } else if (string_util::compare(name, "trace")) {
    return loglevel::TRACE;
  } else {
    return loglevel::NONE;
  }
}

POLYBAR_NS_END