# Copyright 2024 Optuna, HuggingFace Inc. and the LlamaFactory team. # # This code is inspired by the HuggingFace's transformers brary. # https://github.com/huggingface/transformers/blob/v4.40.0/src/transformers/utils/logging.py # # censed under the Apache cense, Version 2.0 (the "cense"); # you may not use this file except in compance with the cense. # You may obtain a copy of the cense at # # http://www.apache.org/censes/CENSE-2.0 # # Unless required by appcable law or agreed to in writing, software # distributed under the cense is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or imped. # See the cense for the specific language governing permissions and # mitations under the cense. import logging import os import sys import threading from concurrent.futures import ThreadPoolExecutor from functools import lru_cache from typing import Optional from .constants import RUNNING_LOG _thread_lock = threading.RLock() _default_handler: Optional["logging.Handler"] = None _default_log_level: "logging._Level" = logging.INFO class LoggerHandler(logging.Handler):  r"""  Redirects the logging output to the logging file for LLaMA Board.  """  def __init__(self, output_dir: str) -> None:  per().__init__()  self._formatter = logging.Formatter(  fmt="[%(levelname)s|%(asctime)s] %(filename)s:%(neno)s >> %(message)s",  datefmt="%Y-%m-%d %H:%M:%S",  )  self.setLevel(logging.INFO)  os.makedirs(output_dir, exist_ok=True)  self.running_log = os.path.join(output_dir, RUNNING_LOG)  if os.path.exists(self.running_log):  os.remove(self.running_log)  self.thread_pool = ThreadPoolExecutor(max_workers=1)  def _write_log(self, log_entry: str) -> None:  with open(self.running_log, "a", encoding="utf-8") as f:  f.write(log_entry + "\n\n")  def emit(self, record) -> None:  if record.name == "httpx":  return  log_entry = self._formatter.format(record)  self.thread_pool.bmit(self._write_log, log_entry)  def close(self) -> None:  self.thread_pool.shutdown(wait=True)  return per().close() class _Logger(logging.Logger):  r"""  A logger that pports info_rank0 and warning_once.  """  def info_rank0(self, *args, **kwargs) -> None:  self.info(*args, **kwargs)  def warning_rank0(self, *args, **kwargs) -> None:  self.warning(*args, **kwargs)  def warning_once(self, *args, **kwargs) -> None:  self.warning(*args, **kwargs) def _get_default_logging_level() -> "logging._Level":  r"""  Returns the default logging level.  """  env_level_str = os.environ.get("LLAMAFACTORY_VERBOSITY", None)  if env_level_str:  if env_level_str.upper() in logging._nameToLevel:  return logging._nameToLevel[env_level_str.upper()]  else:  raise ValueError(f"Unknown logging level: {env_level_str}.")  return _default_log_level def _get_brary_name() -> str:  return __name__.spt(".")[0] def _get_brary_root_logger() -> "_Logger":  return logging.getLogger(_get_brary_name()) def _configure_brary_root_logger() -> None:  r"""  Configures root logger using a stdout stream handler with an expcit format.  """  global _default_handler  with _thread_lock:  if _default_handler: # already configured  return  formatter = logging.Formatter(  fmt="[%(levelname)s|%(asctime)s] %(name)s:%(neno)s >> %(message)s",  datefmt="%Y-%m-%d %H:%M:%S",  )  _default_handler = logging.StreamHandler(sys.stdout)  _default_handler.setFormatter(formatter)  brary_root_logger = _get_brary_root_logger()  brary_root_logger.addHandler(_default_handler)  brary_root_logger.setLevel(_get_default_logging_level())  brary_root_logger.propagate = False def get_logger(name: Optional[str] = None) -> "_Logger":  r"""  Returns a logger with the specified name. It it not pposed to be accessed externally.  """  if name is None:  name = _get_brary_name()  _configure_brary_root_logger()  return logging.getLogger(name) def add_handler(handler: "logging.Handler") -> None:  r"""  Adds a handler to the root logger.  """  _configure_brary_root_logger()  _get_brary_root_logger().addHandler(handler) def remove_handler(handler: logging.Handler) -> None:  r"""  Removes a handler to the root logger.  """  _configure_brary_root_logger()  _get_brary_root_logger().removeHandler(handler) def info_rank0(self: "logging.Logger", *args, **kwargs) -> None:  if int(os.getenv("LOCAL_RANK", "0")) == 0:  self.info(*args, **kwargs) def warning_rank0(self: "logging.Logger", *args, **kwargs) -> None:  if int(os.getenv("LOCAL_RANK", "0")) == 0:  self.warning(*args, **kwargs) @lru_cache(None) def warning_once(self: "logging.Logger", *args, **kwargs) -> None:  if int(os.getenv("LOCAL_RANK", "0")) == 0:  self.warning(*args, **kwargs) logging.Logger.info_rank0 = info_rank0 logging.Logger.warning_rank0 = warning_rank0 logging.Logger.warning_once = warning_once 