"logging.handlers" --- Gestores de *logging*
********************************************

**Código fuente** Lib/logging/handlers.py


Important
^^^^^^^^^

Esta página contiene solo información de referencia. Para tutoriales
por favor véase

* Tutorial Básico

* Tutorial avanzado

* Libro de cocina de *Logging*

======================================================================

Estos gestores son muy útiles y están provistos en este paquete. Nota
que tres de los gestores de las clases ("StreamHandler", "FileHandler"
and "NullHandler") están definidos en propio módulo "logging" pero
fueron documentados aquí junto con los otros gestores.


StreamHandler
=============

La clase "StreamHandler" ubicada en el paquete núcleo "logging" envía
la salida del *logging* a un *stream* como *sys.stdout*, *sys.stderr*
o cualquier objeto tipo archivo (o mas precisamente cualquier objeto
que soporte los métodos "write()" y "flush()").

class logging.StreamHandler(stream=None)

   Retorna una nueva instancia de la clase "StreamHandler". Si
   *stream* esta especificado, la instancia lo usará para la salida
   del registro, sino se usará *sys.stderr*.

   emit(record)

      Si esta especificado un formateador se usará para formatear el
      registro. Luego se escribe el registro al *stream* con un
      terminador. Si hay información de excepción presente, se
      formateará usando "traceback.print_exception()" y se añadirá al
      *stream*.

   flush()

      Descarga el *stream* llamando a su método "flush()". Nota que el
      método "close()" es heredado de la clase "Handler" y por lo
      tanto no produce ninguna salida. Por eso muchas veces será
      necesario invocar al método explícito "flush()".

   setStream(stream)

      Establece el *stream* de la instancia a un valor especifico, si
      este es diferente. El anterior *stream* es vaciado antes de que
      el nuevo *stream* sea establecido.

      Parámetros:
         **stream** -- El *stream* que el gestor debe usar.

      Devuelve:
         el anterior *stream*. si el *stream* cambió o *None* si no
         cambió.

   Nuevo en la versión 3.7.

Distinto en la versión 3.2: La clase "StreamHandler" ahora tiene un
atributo "terminador` con valor por defecto ``'\n'", que se usa como
terminador cuando se escribe un registro formateado a un *stream*. Si
no se quiere esta terminación de nueva línea, se puede establecer el
atributo de la instancia del gestor``terminator`` para vaciar la
cadena de caracteres. En versiones anteriores el terminador era
codificado como "'\n'".


FileHandler
===========

La clase "FileHandler" está localizada en el paquete núcleo "logging",
envía la salida del *logging* a un archivo de disco. Hereda la
funcionalidad de salida de la clase "StreamHandler".

class logging.FileHandler(filename, mode='a', encoding=None, delay=False)

   Retorna una nueva instancia de la clase "FileHandler". Abre el
   archivo especificado y se usa como un *stream* para *logging*. si
   *mode* no se especifica se usa "'a'" . Si *encoding* no es "None",
   se usará para abrir el archivo con esa codificación. Si *delay* es
   *true* entonces se difiere la apertura del archivo hasta la primer
   llamada al método "emit()". Por defecto el archivo crece
   indefinidamente.

   Distinto en la versión 3.6: Así como valores de cadena de
   caracteres, también se aceptan objetos de la clase "Path" para el
   argumento "*filename*".

   close()

      Cierra el archivo.

   emit(record)

      Da la salida del registro al archivo.


NullHandler
===========

Nuevo en la versión 3.1.

La clase "NullHandler" está ubicada en el núcleo biblioteca "logging"
. No realiza ningún formateo o salida. Es en esencia un gestor 'no-op'
para uso de desarrolladores de bibliotecas.

class logging.NullHandler

   Retorna una nueva instancia de la clase "NullHandler".

   emit(record)

      Este método no realiza ninguna acción.

   handle(record)

      Este método no realiza ninguna acción.

   createLock()

      Este método retorna "None" para el bloqueo , dado que no hay una
      E/S subyacente cuyo acceso se necesite serializar.

Véase Configurando Logging para una biblioteca para mas información en
como usar la clase "NullHandler".


WatchedFileHandler
==================

La clase "WatchedFileHandler" está ubicada en el módulo
"logging.handlers", es una clase "FileHandler" que vigila a que
archivo se está enviando el *logging*. Si el archivo cambia , este se
cerrará y se volverá a abrir usando el nombre de archivo.

Puede suceder que haya un cambio de archivo por uso de programas como
*newsyslog* y *logrotate* que realizan una rotación del archivo log.
Este gestor destinado para uso bajo Unix/Linux controla el archivo
para ver si hubo cambios desde la última emisión. (Un archivo se
considera que cambió si su dispositivo o nodo índice cambió). Si el
archivo cambió entonces el anterior *stream* de archivo se cerrará, y
se abrirá el nuevo para obtener un nuevo *stream*.

Este gestor no es apropiado para uso bajo Windows porque bajo Windows
los archivos log abiertos no se pueden mover o renombrar. *Logging*
abre los archivos con bloqueos exclusivos y entonces no hay necesidad
de usar el gestor. Por otra parte *ST_INO* no es soportado bajo
Windows. La función "stat()" siempre retorna cero para este valor.

class logging.handlers.WatchedFileHandler(filename, mode='a', encoding=None, delay=False)

   Retorna una nueva instancia de la clase "WatchedFileHandler". El
   archivo especificado se abre y usa como el *stream* para *logging*.
   Si *mode* no se especifica se usará "'a'". Si *encoding* no es
   "None", se usa para abrir el archivo con esa codificación. Si
   *delay* es true se diferirá la apertura del archivo hasta tanto se
   llame al método "emit()" . Por defecto el archivo crecerá
   indefinidamente.

   Distinto en la versión 3.6: Así como valores de cadena de
   caracteres, también se aceptan objetos de la clase "Path" para el
   argumento "*filename*".

   reopenIfNeeded()

      Revisa si el archivo cambió. Si hubo cambio, el *stream*
      existente se vacía y cierra y el archivo se abre nuevamente.
      Típicamente es un precursor para dar salida del registro a un
      archivo.

      Nuevo en la versión 3.6.

   emit(record)

      Da salida al registro a un archivo, pero primero invoca al
      método "reopenIfNeeded()" para reabrir el archivo si es que
      cambió.


BaseRotatingHandler
===================

La clase "BaseRotatingHandler" ubicada en el módulo "logging.handlers"
es la clase base para rotar los gestores de archivos de clases
"RotatingFileHandler" y "TimedRotatingFileHandler". No debería ser
necesario instanciar esta clase, pero tiene métodos y atributos que
quizá se necesiten sobrescribir (*override*).

class logging.handlers.BaseRotatingHandler(filename, mode, encoding=None, delay=False)

   Los parámetros son como los de la clase "FileHandler". Los
   atributos son:

   namer

      Si este atributo se establece como invocable, el método
      "rotation_filename()" delega a este invocable. Los parámetros
      pasados al invocable son aquellos pasados al método
      "rotation_filename()".

      Nota:

        La función de nombrado es invocada unas cuantas veces durante
        el volcado (*rollover*) , entonces debe ser tan simple y
        rápida como sea posible. Debe también retornar siempre la
        misma salida para una misma entrada, de otra manera el volcado
        puede no funcionar como se espera.

      Nuevo en la versión 3.3.

   rotator

      Si este atributo se estableció como invocable, el método
      "rotate()" delega a este invocable. Los parámetros pasados al
      invocable son aquellos pasados al método "rotate()".

      Nuevo en la versión 3.3.

   rotation_filename(default_name)

      Modifica el nombre de un archivo log cuando esta rotando.

      Esto esta previsto para que pueda usarse un nombre de archivo
      personalizado.

      La implementación por defecto llama al atributo 'namer' del
      gestor, si este es invocable, pasando el nombre por defecto a
      él. Si el atributo no es invocable (por defecto es "None") el
      nombre se retorna sin cambios.

      Parámetros:
         **default_name** -- El nombre por defecto para el archivo de
         log.

      Nuevo en la versión 3.3.

   rotate(source, dest)

      Cuando está rotando, rotar el actual log.

      La implementación por defecto llama al atributo 'rotator' del
      gestor, si es invocable, pasando los argumentos de origen y
      destino a él. Si no se puede invocar (porque el atributo por
      defecto es 'None') el origen es simplemente renombrado al
      destino.

      Parámetros:
         * **source** -- El nombre de archivo origen . Normalmente el
           nombre de archivo base, por ejemplo 'test.log'.

         * **dest** -- El nombre de archivo de destino. Normalmente es
           el nombre al que se rota el archivo origen por ejemplo
           'test.log.1'.

      Nuevo en la versión 3.3.

La razón de que existen los atributos es para evitar tener que usar
una subclase - puedes usar los mismos invocadores para instancias de
clase "RotatingFileHandler" y "TimedRotatingFileHandler". Si el
rotador invocable o la función de nombrado plantean una excepción esta
se manejará de la misma manera que cualquier otra excepción durante
una llamada al método "emit()" por ejemplo a través del método
"handleError()" del gestor.

Si necesitas hacer cambios mas significativos al proceso de rotación
puedes obviar los métodos.

Para un ejemplo véase Using a rotator and namer to customize log
rotation processing.


RotatingFileHandler
===================

La clase "RotatingFileHandler", localizada en el módulo
"logging.handlers" soporta la rotación de archivos log de disco.

class logging.handlers.RotatingFileHandler(filename, mode='a', maxBytes=0, backupCount=0, encoding=None, delay=False)

   Retorna una nueva instancia de la clase "RotatingFileHandler". El
   archivo especificado es abierto y usado como un *stream* para
   *logging*. Si *mode* no se especifica , se usará "'a'" . Si
   *encoding* no es "None" se usará para abrir el archivo con esa
   codificación. Si *delay* es verdadero entonces la apertura del
   archivo se diferirá hasta la primer llamada al método "emit()". Por
   defecto el archivo crece indefinidamente.

   Se pueden usar los valores *maxBytes* y *backupCount* para permitir
   que el archivo *rollover* tenga un tamaño predeterminado. Cuando el
   tamaño del archivo está a punto de excederse, se cerrará y un nuevo
   archivo se abrirá silenciosamente para salida. El volcado
   (*rollover*) ocurre cada vez que el actual archivo log esta cerca
   de *maxBytes* en tamaño , pero si cualquiera *maxBytes* o
   *backupCount* es cero, el volcado (*rollover*) no ocurre. Por eso
   generalmente necesitas establecer *backupCount* por lo menos en 1 y
   no tener cero en *maxBytes*. Cuando *backupCount* no es cero, el
   sistema guardará los anteriores archivos log agregando las
   extensiones '.1', '.2' etc. al nombre del archivo. Por ejemplo con
   un *backupCount* de 5 y un nombre de archivo base de "app.log",
   tendrás como nombre de archivo t "app.log", "app.log.1",
   "app.log.2", hasta "app.log.5". El archivo que esta siendo escrito
   es siempre "app.log". Cuando este se completa , se cierra y se
   renombra a "app.log.1" y si ya existen "app.log.1", "app.log.2",
   etc. Entonces se renombrará como "app.log.2", "app.log.3" etc.
   respectivamente.

   Distinto en la versión 3.6: Así como valores de cadena de
   caracteres, también se aceptan objetos de la clase "Path" para el
   argumento "*filename*".

   doRollover()

      Realiza un volcado (*rollover*) como se describe arriba.

   emit(record)

      Da la salida del registro al archivo , dando suministro para el
      volcado (*rollover*) como está descripto anteriormente.


TimedRotatingFileHandler
========================

La clase "TimedRotatingFileHandler" está ubicada en el módulo
"logging.handlers". Soporta la rotación de archivos de log a ciertos
intervalos de tiempo.

class logging.handlers.TimedRotatingFileHandler(filename, when='h', interval=1, backupCount=0, encoding=None, delay=False, utc=False, atTime=None)

   Retorna una nueva instancia de la clase "TimedRotatingFileHandler"
   . El archivo especificado es abierto y usado como *stream* para el
   historial de log. En la rotación también establece el sufijo del
   nombre de archivo. La rotación ocurre basada en el producto de
   *when* y *interval*.

   Puedes usar el *when* para especificar el tipo de intervalo
   *interval*. La lista de posibles valores esta debajo. Nota que no
   distingue mayúsculas y minúsculas.

   +------------------+------------------------------+---------------------------+
   | Valor            | Tipo de intervalo            | Si/como *atTime* es usado |
   |==================|==============================|===========================|
   | "'S'"            | Segundos                     | Ignorado                  |
   +------------------+------------------------------+---------------------------+
   | "'M'"            | Minutos                      | Ignorado                  |
   +------------------+------------------------------+---------------------------+
   | "'H'"            | Horas                        | Ignorado                  |
   +------------------+------------------------------+---------------------------+
   | "'D'"            | Días                         | Ignorado                  |
   +------------------+------------------------------+---------------------------+
   | "'W0'-'W6'"      | Día de la semana (0=Lunes)   | Usado para calcular la    |
   |                  |                              | hora inicial del volcado  |
   |                  |                              | *rollover*                |
   +------------------+------------------------------+---------------------------+
   | "'midnight'"     | Volcado (*rollover*) a       | Usado para calcular la    |
   |                  | medianoche , si *atTime* no  | hora inicial del volcado  |
   |                  | está especificado, sino el   | *rollover*                |
   |                  | volcado se hará *atTime*     |                           |
   +------------------+------------------------------+---------------------------+

   Cuando se usa rotación basada en día de la semana, especifica 'W0'
   para Lunes, 'W1' para Martes y así, hasta 'W6' para Domingo. en
   este caso el valor pasado para *Interval* no se usa.

   El sistema guardará los archivos de log anteriores agregándoles
   extensiones al nombre de archivo. Las extensiones están basadas en
   día-hora usando el formato "%Y-%m-%d_%H-%M-%S" o un prefijo
   respecto el intervalo del volcado (*rollover*).

   Cuando se calcula la hora del siguiente volcado (*rollover*) por
   primera vez (cuando el gestor es creado), la última hora de
   modificación de un archivo log existente o sino la hora actual, se
   usa para calcular cuando será la próxima rotación.

   Si el argumento *utc* es *true* se usará la hora en UTC, sino se
   usará la hora local.

   Si *backupCount* no es cero, se conservará como máximo una cantidad
   de archivos especificada en *backupCount*,y si son creados más,
   cuando ocurre el volcado (*rollover*) se borrará el último. La
   lógica de borrado usa el intervalo para determinar que archivos
   borrar, pues entonces cambiando el intervalo puede dejar viejos
   archivos abandonados.

   Si *delay* es *true* entonces la apertura del archivo se demorará
   hasta la primer llamada a "emit()".

   Si *atTime* no es "None", debe haber una instancia "datetime.time"
   que especifica la hora que ocurre el volcado (*rollover*) , para
   los casos en que el volcado esta establecido para ocurrir "a
   medianoche" o "un día en particular". Nótese que en estos casos el
   valor *atTime* se usa para calcular el valor *initial* del volcado
   (*rollover*) y los subsecuentes volcados serán calculados a través
   del calculo normal de intervalos.

   Nota:

     El cálculo de la hora en que se realizara el volcado (*rollover*)
     inicial cuando se inicializa el gestor. El cálculo de la hora de
     los siguientes volcados (*rollovers*) se realiza solo cuando este
     ocurre, y el volcado ocurre solo cuando se emite una salida. Si
     esto no se tiene en cuenta puede generar cierta confusión. Por
     ejemplo si se establece un intervalo de "cada minuto" eso no
     significa que siempre se verán archivos log con hora (en el
     nombre del archivo) separados por un minuto. Si durante la
     ejecución de la aplicación el *logging* se genera con mayor
     frecuencia que un minuto entonces se pueden esperar archivos log
     separados por un minuto. Si por otro lado los mensajes *logging*
     son establecidos cada digamos cinco minutos, entonces habrá
     brechas de tiempo en los archivos correspondientes a los minutos
     que no hubo salida (y no ocurrió por tanto volcado alguno).

   Distinto en la versión 3.4: Se agregó el parámetro *atTime*.

   Distinto en la versión 3.6: Así como valores de cadena de
   caracteres, también se aceptan objetos de la clase "Path" para el
   argumento "*filename*".

   doRollover()

      Realiza un volcado (*rollover*) como se describe arriba.

   emit(record)

      Da la salida del registro a un archivo , proveyendo la
      información para el volcado (*rollover*) como esta descripto
      anteriormente.


SocketHandler
=============

La clase "SocketHandler" esta localizada en el módulo
"logging.handlers", envía el *logging* a un socket de la red. La clase
base usa *sockets* TCP.

class logging.handlers.SocketHandler(host, port)

   Retorna una nueva instancia de la clase "SocketHandler" destinada
   para comunicarse con una terminal remota cuya dirección esta dada
   por *host* y *port*.

   Distinto en la versión 3.4: Si "port" se especifica como``None`` se
   crea un socket de dominio Unix, usando el valor en``host`` - de
   otra manera se creará un socket TCP.

   close()

      Cierra el socket.

   emit()

      Serializa (*Pickles*) el registro del diccionario de atributos y
      lo escribe en el socket en formato binario. Si hay un error con
      el socket, silenciosamente descarta el paquete. Si la conexión
      se perdió previamente, la restablece. Para deserializar
      (*unpickle*) un registro en el extremo receptor a una clase
      "LogRecord", usa la función "makeLogRecord()".

   handleError()

      Maneja un error que ocurrió durante el método "emit()". La causa
      mas común es una perdida de conexión. Cierra el socket para que
      podamos reintentar en el próximo evento.

   makeSocket()

      Este es un método patrón que permite subclases para definir el
      tipo preciso de socket que se necesita. La implementación por
      defecto crea un socket TCP("socket.SOCK_STREAM").

   makePickle(record)

      Serializa (*pickles*) el registro del diccionario de atributos
      en formato binario con un prefijo de tamaño, y lo retorna listo
      para transmitir a través del socket. Los detalles de esta
      operación son equivalentes a:

         data = pickle.dumps(record_attr_dict, 1)
         datalen = struct.pack('>L', len(data))
         return datalen + data

      Nota que los serializados (*pickles*) no son totalmente seguros.
      Si te preocupa la seguridad desearás evitar este método para
      implementar un mecanismo mas seguro. Por ejemplo puedes firmar
      *pickles* usando HMAC y verificarlos después en el extremo
      receptor. O alternativamente puedes deshabilitar la
      deserialización (*unpickling*) de objetos globales en el extremo
      receptor.

   send(packet)

      Envía un paquete serializado (*pickled*) de cadena de caracteres
      al socket. El formato de la cadena de bytes enviada es tal como
      se describe en la documentación de "makePickle()".

      Esta función permite envíos parciales, que pueden ocurrir cuando
      la red esta ocupada.

   createSocket()

      Intenta crear un socket, si hay una falla usa un algoritmo de
      marcha atrás exponencial. En el fallo inicial el gestor
      desechará el mensaje que intentaba enviar. Cuando los siguientes
      mensajes sean gestionados por la misma instancia no intentará
      conectarse hasta que haya transcurrido cierto tiempo. Los
      parámetros por defecto son tales que el retardo inicial es un
      segundo y si después del retardo la conexión todavía no se puede
      realizar, el gestor doblará el retardo cada vez hasta un máximo
      de 30 segundos.

      Este comportamiento es controlado por los siguientes atributos
      del gestor:

      * "retryStart" (retardo inicial por defecto 1.0 segundos)

      * "retryFactor" (multiplicador por defecto 2.0)

      * "retryMax" (máximo retardo por defecto 30.0 segundos)

      Esto significa que si el oyente remoto (*listener*) comienza
      después de que se usó el gestor , pueden perderse mensajes (dado
      que el gestor no puede siquiera intentar una conexión hasta que
      se haya cumplido el retardo, y silenciosamente desechará los
      mensajes mientras se cumpla el retardo).


DatagramHandler
===============

La clase "DatagramHandler" está ubicada en el módulo
"logging.handlers", hereda de la clase "SocketHandler" para realizar
el soporte de mensajes *logging* por los *sockets* UDP.

class logging.handlers.DatagramHandler(host, port)

   Retorna una nueva instancia de la clase "DatagramHandler" destinada
   para comunicarse con la terminal remota cuya dirección es dada por
   *host* y *port*.

   Distinto en la versión 3.4: Si 'port' se especifica como "None", se
   crea un socket de dominio Unix, usando el valor en "host" - de otra
   manera se crea un socket UDP.

   emit()

      Serializa (*pickles*) el registro del diccionario de atributos y
      lo escribe en el socket en formato binario. Si hay un error con
      el socket, silenciosamente desecha el paquete. Para deserializar
      (*unpickle*) el registro en el extremo de recepción a una clase
      "LogRecord", usa la función "makeLogRecord()".

   makeSocket()

      El método original de la clase "SocketHandler" se omite para
      crear un socket UDP ("socket.SOCK_DGRAM").

   send(s)

      Enviar una cadena de caracteres serializada (*pickled*) a un
      socket de red. El formato de la cadena de *bytes* enviado es tal
      como se describe en la documentación para
      "SocketHandler.makePickle()".


Gestor *SysLog* (*SysLogHandler*)
=================================

La clase "SysLogHandler" está ubicada en el módulo "logging.handlers".
Realiza el soporte de los mensajes de *logging* a un *syslog* Unix
local o remoto.

class logging.handlers.SysLogHandler(address=('localhost', SYSLOG_UDP_PORT), facility=LOG_USER, socktype=socket.SOCK_DGRAM)

   Retorna una nueva instancia de la clase "SysLogHandler" concebida
   para comunicarse con una terminal remota Unix cuya dirección esta
   dada por *address* en la forma de una tupla "(host, port)" . Si
   *address* no se especifica se usará "('localhost', 514)". La
   dirección se usa para abrir el socket. Una alternativa a consignar
   una tupla "(host, port)" es proveer una dirección como cadena de
   caracteres, por ejemplo '/dev/log'. En este caso se usa un socket
   de dominio Unix para enviar el mensaje al syslog. Si *facility* no
   se especifica se usara "LOG_USER" . El tipo de socket abierto usado
   depende del argumento *socktype* , que por defecto es
   "socket.SOCK_DGRAM" y por lo tanto abre un socket UDP . Para abrir
   un socket TCP (para usar con los nuevos *daemons syslog* como
   Rsyslog) se debe especificar un valor de "socket.SOCK_STREAM".

   Nótese que si el servidor no esta escuchando el puerto UDP 514, la
   clase "SysLogHandler" puede parecer no funcionar. En ese caso
   chequea que dirección deberías usar para un socket de dominio . Es
   sistema-dependiente. Por ejemplo en Linux generalmente es
   '/dev/log' pero en OS/X es '/var/run/syslog'. Será necesario
   chequear tu plataforma y usar la dirección adecuada (quizá sea
   necesario realizar este chequeo mientras corre la aplicación si
   necesita correr en diferentes plataformas). En Windows seguramente
   tengas que usar la opción UDP.

   Distinto en la versión 3.2: Se agregó *socktype*.

   close()

      Cierra el socket del host remoto.

   emit(record)

      El registro es formateado, y luego enviado al servidor *syslog*.
      Si hay información de excepción presente entonces no se enviará
      al servidor.

      Distinto en la versión 3.2.1: (Véase el bpo-12168.) En versiones
      anteriores , los mensajes enviados a los *daemons syslog*
      siempre terminaban con un byte NUL ya que versiones anteriores
      de estos *daemons* esperaban un mensaje NUL de terminación.
      Incluso a pesar que no es relevante la especificación (**RFC
      5424**). Versiones mas recientes de estos *daemons* no esperan
      el byte NUL pero lo quitan si esta ahí. Versiones aún mas
      recientes que están mas cercanas a la especificación RFC 5424
      pasan el byte NUL como parte del mensaje.Para habilitar una
      gestión mas sencilla de los mensajes *syslog* respecto de todos
      esos *daemons* de diferentes comportamientos el agregado del
      byte NUL es configurable a través del uso del atributo de nivel
      de clase 'append_nul'. Este es por defecto '"True" (preservando
      el comportamiento ya existente) pero se puede establecer a
      'False' en una instancia "SysLogHandler" como para que esa
      instancia no añada el terminador NUL.

      Distinto en la versión 3.3: (Véase: issue '12419') en versiones
      anteriores, no había posibilidades para un prefijo 'ident' o
      'tag' para identificar el origen del mensaje. Esto ahora se
      puede especificar usando un atributo de nivel de clase, que por
      defecto será "''" para preservar el comportamiento existente ,
      pero puede ser sorteado en una instancia "SysLogHandler" para
      que esta instancia anteponga el *ident* a todos los mensajes
      gestionados. Nótese que el *ident* provisto debe ser texto, no
      bytes y se antepone al mensaje tal como es.

   encodePriority(facility, priority)

      Codifica la funcionalidad y prioridad en un entero. Puedes pasar
      cadenas de caracteres o enteros, si pasas cadenas de caracteres
      se usarán los diccionarios de mapeo interno para convertirlos en
      enteros.

      Los valores simbólicos "LOG_" están definidos en "SysLogHandler"
      e invierten los valores definidos en el archivo de encabezado
      *sys/syslog.h`*.

      **Prioridades**

      +----------------------------+-----------------+
      | Nombre (cadena de          | Valor simbólico |
      | caracteres)                |                 |
      |============================|=================|
      | "alert"                    | LOG_ALERT       |
      +----------------------------+-----------------+
      | "crit" or "critical"       | LOG_CRIT        |
      +----------------------------+-----------------+
      | "debug"                    | LOG_DEBUG       |
      +----------------------------+-----------------+
      | "emerg" or "panic"         | LOG_EMERG       |
      +----------------------------+-----------------+
      | "err" or "error"           | LOG_ERR         |
      +----------------------------+-----------------+
      | "info"                     | LOG_INFO        |
      +----------------------------+-----------------+
      | "notice"                   | LOG_NOTICE      |
      +----------------------------+-----------------+
      | "warn" or "warning"        | LOG_WARNING     |
      +----------------------------+-----------------+

      **Facilities**

      +-----------------+-----------------+
      | Nombre (cadena  | Valor simbólico |
      | de caracteres)  |                 |
      |=================|=================|
      | "auth"          | LOG_AUTH        |
      +-----------------+-----------------+
      | "authpriv"      | LOG_AUTHPRIV    |
      +-----------------+-----------------+
      | "cron"          | LOG_CRON        |
      +-----------------+-----------------+
      | "daemon"        | LOG_DAEMON      |
      +-----------------+-----------------+
      | "ftp"           | LOG_FTP         |
      +-----------------+-----------------+
      | "kern"          | LOG_KERN        |
      +-----------------+-----------------+
      | "lpr"           | LOG_LPR         |
      +-----------------+-----------------+
      | "mail"          | LOG_MAIL        |
      +-----------------+-----------------+
      | "news"          | LOG_NEWS        |
      +-----------------+-----------------+
      | "syslog"        | LOG_SYSLOG      |
      +-----------------+-----------------+
      | "user"          | LOG_USER        |
      +-----------------+-----------------+
      | "uucp"          | LOG_UUCP        |
      +-----------------+-----------------+
      | "local0"        | LOG_LOCAL0      |
      +-----------------+-----------------+
      | "local1"        | LOG_LOCAL1      |
      +-----------------+-----------------+
      | "local2"        | LOG_LOCAL2      |
      +-----------------+-----------------+
      | "local3"        | LOG_LOCAL3      |
      +-----------------+-----------------+
      | "local4"        | LOG_LOCAL4      |
      +-----------------+-----------------+
      | "local5"        | LOG_LOCAL5      |
      +-----------------+-----------------+
      | "local6"        | LOG_LOCAL6      |
      +-----------------+-----------------+
      | "local7"        | LOG_LOCAL7      |
      +-----------------+-----------------+

   mapPriority(levelname)

      Mapea un nombre de nivel *logging* a un nombre de prioridad
      *syslog*. Puedes necesitar omitir esto si estas usando niveles
      personalizados, o si el algoritmo por defecto no es aplicable a
      tus necesidades. El algoritmo por defecto mapea "DEBUG", "INFO",
      "WARNING", "ERROR" y "CRITICAL" a sus nombres equivalentes
      *syslog*, y todos los demás nombres de nivel a 'warning'.


Gestor de eventos *NTELog* (NTEventLogHandler)
==============================================

La clase "NTEventLogHandler" esta localizada en el módulo
"logging.handlers", soporta el envío de mensajes de *logging* a un log
de eventos local Windows NT, Windows 2000 o Windows XP. Antes de que
puedas usarlo, necesitarás tener instaladas las extensiones de Win32
de Mark Hammond para Python.

class logging.handlers.NTEventLogHandler(appname, dllname=None, logtype='Application')

   Retorna una nueva instancia de la clase "NTEventLogHandler" la
   *appname* se usa para definir el nombre de la aplicación tal como
   aparece en el log de eventos. Se crea una entrada de registro
   apropiada usando este nombre. El *dllname* debe dar la ruta
   completa calificada de un .dll o .exe que contiene definiciones de
   mensaje para conservar en el log. (si no esta especificada, se
   usara``'win32service.pyd'`` esto es instalado con las extensiones
   de Win32 y contiene algunas definiciones básicas de mensajes de
   conservación de lugar. Nótese que el uso de estos conservadores de
   lugar harán tu log de eventos extenso, dado que el origen completo
   del mensaje es guardado en el log. Si quieres *logs* menos extensos
   deberás pasar el nombre de tu propio .dll o .exe que contiene la
   definición de mensajes que quieres usar en el log. El *logtype*
   puede ser de "'Application'", "'System'" or "'Security'" y sino por
   defecto será de "'Application'".

   close()

      Llegado a este punto puedes remover el nombre de aplicación del
      registro como origen de entrada de log de eventos. Sin embargo
      si haces esto no te será posible ver los eventos tal como has
      propuesto en el visor del log de eventos - necesita ser capaz de
      acceder al registro para tomar el nombre .dll. Esto no lo hace
      la versión actual.

   emit(record)

      Determina el ID del mensaje, categoría y tipo de evento y luego
      registra el mensaje en el log de eventos NT.

   getEventCategory(record)

      Retorna la categoría de evento del registro. Evita esto si
      quieres especificar tus propias categorías. Esta versión retorna
      0.

   getEventType(record)

      Retorna el tipo de evento del registro. Haz caso omiso de esto
      si quieres especificar tus propios tipos. Esta versión realiza
      un mapeo usando el atributo *typemap* del gestor, que se
      establece en "__init__()" a un diccionario que contiene mapeo
      para "DEBUG", "INFO", "WARNING", "ERROR" y "CRITICAL". Si estas
      usando tus propios niveles, necesitarás omitir este método o
      colocar un diccionario a medida en el atributo *typemap* del
      gestor.

   getMessageID(record)

      Retorna el ID de mensaje para el registro. Si estas usando tus
      propios mensajes, podrás hacerlo pasando el *msg* al *logger*
      siendo un ID mas que un formato de cadena de caracteres. Luego
      aquí puedes usar una búsqueda de diccionario para obtener el ID
      de mensaje. Esta versión retorna 1, que es el ID de mensaje base
      en "win32service.pyd".


SMTPHandler
===========

La clase "SMTPHandler" esta ubicada en el módulo "logging.handlers" y
soporta el envío de mensajes de *logging* a un a dirección de correo
electrónico a través de SMTP.

class logging.handlers.SMTPHandler(mailhost, fromaddr, toaddrs, subject, credentials=None, secure=None, timeout=1.0)

   Retorna una nueva instancia de la clase "SMTPHandler" . Esta
   instancia se inicializa con la dirección de: y para: y asunto: del
   correo electrónico. El *toaddrs* debe ser una lista de cadena de
   caracteres. Para especificar un puerto SMTP no estandarizado usa el
   formato de tupla (host, puerto) para el argumento *mailhost*. Si
   usas una cadena de caracteres, se utiliza el puerto estándar SMTP.
   Si tu servidor SMTP necesita autenticación, puedes especificar una
   tupla (usuario, contraseña) para el argumento de *credentials*.

   Para especificar el uso de un protocolo de seguridad (TLS), pasa
   una tupla al argumento *secure*. Esto solo se utilizará cuando sean
   provistas las credenciales de autenticación. La tupla deberá ser
   una tupla vacía o una tupla con único valor con el nombre de un
   archivo-clave *keyfile*, o una tupla de 2 valores con el nombre del
   archivo-clave *keyfile* y archivo certificado. (Esta tupla se pasa
   al método "smtplib.SMTP.starttls()" method.).

   Se puede especificar un tiempo de espera para comunicación con el
   servidor SMTP usando el argumento *timeout*.

   Nuevo en la versión 3.3: Se agregó el argumento *timeout*.

   emit(record)

      Formatea el registro y lo envía a las direcciones especificadas.

   getSubject(record)

      Si quieres especificar una línea de argumento que es registro-
      dependiente, sobrescribe (*override*) este método.


MemoryHandler
=============

La clase "MemoryHandler" esta ubicada en el módulo "logging.handlers"
.Soporta el almacenamiento temporal de registros *logging* en memoria.
Periódicamente los descarga al gestor *target*. Esto ocurre cuando el
búfer está lleno o cuando ocurre un evento de cierta importancia.

"MemoryHandler" es una subclase de la clase mas general
"BufferingHandler", que es una clase abstracta. Este almacena
temporalmente registros *logging* en la memoria. Cada vez que un
registro es agregado al búfer, se realiza una comprobación llamando al
método "shouldFlush()" para ver si dicho búfer debe ser descargado. Si
debiera, entonces el método "flush()" se espera que realice la
descarga.

class logging.handlers.BufferingHandler(capacity)

   Inicializa el gestor con un búfer de una capacidad especifica. Aquí
   capacidad significa el número de registros *logging* en el
   almacenamiento temporal.

   emit(record)

      Añade un registro al búfer. Si el método "shouldFlush()" retorna
      *true* , entonces llama al método "flush()" para procesar el
      búfer.

   flush()

      Puedes sobrescribir (*override*) esto para implementar un
      comportamiento 'a medida' de la descarga. Esta versión solo
      vacía el búfer.

   shouldFlush(record)

      Retorna "True" si el búfer tiene aún capacidad. Este método
      puede ser omitido para implementar estrategias a medida de
      vaciado.

class logging.handlers.MemoryHandler(capacity, flushLevel=ERROR, target=None, flushOnClose=True)

   Retorna una nueva instancia de la clase "MemoryHandler" . La
   instancia se inicializa con un búfer del tamaño *capacity*. Si el
   *flushLevel* no se especifica, se usará "ERROR" . Si no se
   especifica *target* el objetivo deberá especificarse usando el
   método "setTarget()" -Antes de esto el gestor no realizará nada
   útil. Si se especifica *flushOnClose* como "False", entonces el
   búfer no se vaciará cuando el gestor se cierra. Si no se especifica
   o se especifica como "True", el comportamiento previo de vaciado
   del búfer sucederá cuando se cierre el gestor.

   Distinto en la versión 3.6: Se añadió el parámetro *flushOnClose*.

   close()

      Invoca al método "flush()" y establece el objetivo a 'None' y
      vacía el búfer.

   flush()

      Para la clase "MemoryHandler" el vaciado significa simplemente
      enviar los registros del búfer al objetivo, si es que hay uno.
      El búfer además se vacía cuando esto ocurre. Sobrescribe
      (*override*) si deseas un comportamiento diferente.

   setTarget(target)

      Establece el gestor de objetivo para este gestor.

   shouldFlush(record)

      Comprueba si el búfer esta lleno o un registro igual o mas alto
      que *flushLevel*.


HTTPHandler
===========

La clase "HTTPHandler" está ubicada en el módulo
"logging.handlers".Soporta el envío de mensajes *logging* a un
servidor Web, usando la semántica "GET" o *POST`*.

class logging.handlers.HTTPHandler(host, url, method='GET', secure=False, credentials=None, context=None)

   Retorna una nueva instancia de la clase "HTTPHandler". el *host*
   puede ser de la forma "host:puerto", y necesitarás usar un numero
   de puerto especifico. Si no se especifica *method* se usará "GET" .
   Si *secure* es true se usará una conexión HTTPS. El parámetro
   *context* puede establecerse a una instancia "ssl.SSLContext" para
   establecer la configuración de SSL usado en la conexión HTTPS. Si
   se especifica *credentials* debe ser una tupla doble, consistente
   en usuario y contraseña, que se colocará en un encabezado de
   autorización HTTP usando autenticación básica. Si especificas
   credenciales también deberás especificar *secure=True* así tu
   usuario y contraseña no son pasados como texto en blanco por la
   conexión.

   Distinto en la versión 3.5: Se agregó el parámetro *context*.

   mapLogRecord(record)

      Provee un diccionario, basado en "record" para ser codificado en
      forma URL y enviado al servidor web. La implementación por
      defecto retorna "record.__dict__". Este método puede omitirse si
      por ejemplo solo se enviará al servidor web un subconjunto de la
      clase "LogRecord" o si se requiere enviar al servidor algo mas
      específico.

   emit(record)

      Envía el registro al servidor Web como un diccionario con
      codificación URL. Se usa el método "mapLogRecord()" para
      convertir el registro al diccionario que debe ser enviado.

   Nota:

     Dado que preparar un registro para enviar a un servidor Web no es
     lo mismo que una operación genérica de formato, usando el método
     "setFormatter()" para especificar una clase "Formatter" por una
     clase "HTTPHandler" no tiene efecto.  En vez de llamar al método
     "format()" este gestor llama al método "mapLogRecord()" y después
     a la función "urllib.parse.urlencode()" para codificar el
     diccionario en una forma que sea adecuada para enviar a un
     servidor Web.


QueueHandler
============

Nuevo en la versión 3.2.

La clase "QueueHandler" localizada en el módulo "logging.handlers"
soporta el envío de mensajes de *logging* a una cola, tal como los
implementados en los módulos "queue" o "multiprocessing" .

Junto con la clase "QueueListener" la clase "QueueHandler" puede
usarse para permitir a los gestores realizar su tarea en un hilo
separado del que realiza el *logging*. Esto es importante en
aplicaciones Web y también otras aplicaciones donde los clientes
servidos por los hilos necesitan responder tan rápido como sea
posible, mientras que cualquier operación potencialmente lenta (tal
como enviar un correo electrónico a través la clase "SMTPHandler") se
realizan por un hilo diferente.

class logging.handlers.QueueHandler(queue)

   Retorna una nueva instancia de la clase "QueueHandler". La
   instancia se inicializa con la cola a la que se enviarán los
   mensajes. La cola puede ser cualquier objeto tipo-cola; es usado
   tal como por el método "enqueue()" que necesita saber como enviar
   los mensajes a ella. La cola no es *requerida* para tener una API
   de rastreo de tareas, lo que significa que puedes usar instancias
   de "SimpleQueue" como *queue*.

   emit(record)

      Pone en la cola el resultado de preparar el registro historial
      de log. Si ocurre una excepción (por ejemplo por que una cola de
      destino se llenó) entonces se llama al método "handleError()".
      Esto puede resultar en que el registro se descarte (si
      "logging.raiseExceptions" es "False") o que se imprima un
      mensaje a "sys.stderr" (si "logging.raiseExceptions" es "True").

   prepare(record)

      Prepara un registro para poner en la cola. El objeto que retorna
      este método se colocará en cola.

      La implementación base da formato al registro para unir la
      información de los mensajes, argumentos y excepciones, si están
      presentes. También remueve los elementos que no se pueden
      serializar (*unpickleables*) de los registros in-situ.

      Puedes querer hacer caso omiso de este método si quieres
      convertir el registro en un diccionario o cadena de caracteres
      JSON, o enviar una copia modificada del registro mientras dejas
      el original intacto.

   enqueue(record)

      Coloca en la cola al registro usando "put_nowait()"; puede que
      quieras sobrescribe (*override*) esto si quieres usar una acción
      de bloqueo, o un tiempo de espera, o una implementación de cola
      a medida.


QueueListener
=============

Nuevo en la versión 3.2.

La clase "QueueListener" esta localizada en el módulo
"logging.handlers". Soporta la recepción de mensajes *logging* de una
cola, tal como los implementados en los módulos "queue" or
"multiprocessing" . Los mensajes son recibidos de una cola en un hilo
interno y se pasan en el mismo hilo, a uno o mas gestores para
procesarlos. Mientras la clase "QueueListener" no es en si misma un
gestor, esta documentada aquí porque trabaja mano a mano con la clase
"QueueHandler".

Junto con la clase "QueueHandler" , la clase "QueueListener" puede
usarse para permitir a los gestores hacer su labor en un hilo separado
del que hace el *logging*. Esto es importante en aplicaciones Web y
también otras aplicaciones de servicio donde los clientes servidos por
los hilos necesitan una respuesta tan rápida como sea posible,
mientras cualquier operación potencialmente lenta (tal como enviar un
correo electrónico a través de la clase "SMTPHandler") son atendidas
por un hilo diferente.

class logging.handlers.QueueListener(queue, *handlers, respect_handler_level=False)

   Retorna una nueva instancia de la clase "QueueListener". La
   instancia se inicializa con la cola para enviar mensajes a una
   lista de gestores que manejarán entradas colocadas en la cola. La
   cola puede ser cualquier objeto de tipo-cola, es usado tal como
   está por el método "dequeue()" que necesita saber como tomar los
   mensajes de esta. La cola no es *obligatoria* para tener la API de
   seguimiento de tareas (aunque se usa si está disponible), lo que
   significa que puede usar instancias "SimpleQueue" para *queue*.

   Si "respect_handler_level" es "True", se respeta un nivel de gestor
   (comparado con el nivel del mensaje) cuando decide si pasar el
   mensajes al gestor; de otra manera, el comportamiento es como en
   versiones anteriores de Python - de pasar cada mensaje a cada
   gestor.

   Distinto en la versión 3.5: Se agregó el argumento
   "respect_handler_levels".

   dequeue(block)

      Extrae de la cola un registro y lo retorna, con opción a
      bloquearlo.

      La implementación base usa "get()". Puedes sobrescribir
      (*override*) este método si quieres usar tiempos de espera o
      trabajar con colas implementadas a medida.

   prepare(record)

      Prepara un registro para ser gestionado.

      Esta implementación solo retorna el registro que fue pasado.
      Puedes sobrescribir (*override*) este método para hacer una
      serialización a medida o una manipulación del registro antes de
      pasarlo a los gestores.

   handle(record)

      Manejar un registro.

      Esto solo realiza un bucle a través de los gestores
      ofreciéndoles el registro para ser gestionado. El objeto actual
      pasado a los gestores es aquel que es retornado por el método
      "prepare()".

   start()

      Da comienzo al oyente (*listener*).

      Esto da comienzo a un hilo en segundo plano para supervisar la
      cola de registros log a procesar.

   stop()

      Detiene el oyente (*listener*).

      Esto solicita terminar al hilo, y luego espera hasta que
      termine. Nota que si no llamas a esto antes de que tu aplicación
      salga, puede haber algunos registros que aun están en la cola,
      que no serán procesados.

   enqueue_sentinel()

      Escribe un centinela (*sentinel*) en la cola para decir al
      oyente (*listener*) de salir. Esta implementación usa
      "put_nowait()". Puedes sobrescribir (*override*) este método si
      quieres usar tiempos de espera o trabajar con implementaciones
      de cola a tu medida.

      Nuevo en la versión 3.3.

Ver también:

  Módulo "logging"
     Referencia API para el módulo de *logging*.

  Módulo "logging.config"
     Configuración API para el módulo de *logging*.
