"logging.config" --- 日志记录配置
*********************************

**原始碼：**Lib/logging/config.py


Important
^^^^^^^^^

此页面仅包含参考信息。有关教程，请参阅

* 基礎教學

* 進階教學

* 日志记录操作手册

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

这一节描述了用于配置 logging 模块的 API。


配置函数
========

下列函数可配置 logging 模块。 它们位于 "logging.config" 模块中。 它们
的使用是可选的 --- 要配置 logging 模块你可以使用这些函数，也可以通过调
用主 API (在 "logging" 本身定义) 并定义在 "logging" 或
"logging.handlers" 中声明的处理器。

logging.config.dictConfig(config)

   从一个字典获取日志记录配置。 字典的内容描述见下文的 配置字典架构。

   如果在配置期间遇到错误，此函数将引发 "ValueError", "TypeError",
   "AttributeError" 或 "ImportError" 并附带适当的描述性消息。 下面是将
   会引发错误的（可能不完整的）条件列表:

   * "level" 不是字符串或者不是对应于实际日志记录级别的字符串。

   * "propagate" 值不是布尔类型。

   * id 没有对应的目标。

   * 在增量调用期间发现不存在的处理器 id。

   * 无效的日志记录器名称。

   * 无法解析为内部或外部对象。

   解析由 "DictConfigurator" 类执行，该类的构造器可传入用于配置的字典
   ，并且具有 "configure()" 方法。 "logging.config" 模块具有可调用属性
   "dictConfigClass"，其初始值设为 "DictConfigurator"。 你可以使用你自
   己的适当实现来替换 "dictConfigClass" 的值。

   "dictConfig()" 会调用 "dictConfigClass" 并传入指定的字典，然后在所
   返回的对象上调用 "configure()" 方法以使配置生效:

      def dictConfig(config):
          dictConfigClass(config).configure()

   例如，"DictConfigurator" 的子类可以在它自己的 "__init__()" 中调用
   "DictConfigurator.__init__()"，然后设置可以在后续 "configure()" 调
   用中使用的自定义前缀。 "dictConfigClass" 将被绑定到这个新的子类，然
   后就可以与在默认的未定制状态下完全相同的方式调用 "dictConfig()"。

   3.2 版新加入.

logging.config.fileConfig(fname, defaults=None, disable_existing_loggers=True, encoding=None)

   从一个 "configparser" 格式文件中读取日志记录配置。 文件格式应当与
   配置文件格式 中的描述一致。 此函数可在应用程序中被多次调用，以允许
   最终用户在多个预设配置中进行选择（如果开发者提供了展示选项并加载选
   定配置的机制）。

   參數:
      * **fname** -- 一个文件名，或一个文件类对象，或是一个派生自
        "RawConfigParser" 的实例。 如果传入了一个派生自
        "RawConfigParser" 的实例，它会被原样使用。 否则，将会实例化一
        个 "Configparser"，并且它会从作为 "fname" 传入的对象中读取配置
        。 如果存在 "readline()" 方法，则它会被当作一个文件类对象并使
        用 "read_file()" 来读取；在其它情况下，它会被当作一个文件名并
        传递给 "read()"。

      * **defaults** -- 要传递给 ConfigParser 的默认值可在此参数中指定
        。

      * **disable_existing_loggers** -- 如果指定为 "False"，则当执行此
        调用时已存在的日志记录器会被保持启用。 默认值为 "True" 因为这
        将以向下兼容的方式启用旧有行为。 此行为是禁用任何已存在的非根
        日志记录器除非它们或它们的上级在日志配置中被显式地指定。
        :param encoding: 当 *fname* 为文件名时用于打开文件的编码格式。

   3.4 版更變: 现在接受 "RawConfigParser" 子类的实例作为 "fname" 的值
   。 这有助于:

      * 使用一个配置文件，其中日志记录配置只是全部应用程序配置的一部分
        。

      * 使用从一个文件读取的配置，它随后会在被传给 "fileConfig" 之前由
        使用配置的应用程序来修改（例如基于命令行参数或运行时环境的其他
        部分）。

   3.10 版新加入: 增加了 *encoding* 形参。

logging.config.listen(port=DEFAULT_LOGGING_CONFIG_PORT, verify=None)

   在指定的端口上启动套接字服务器，并监听新的配置。 如果未指定端口，则
   会使用模块默认的 "DEFAULT_LOGGING_CONFIG_PORT"。 日志记录配置将作为
   适合由 "dictConfig()" 或 "fileConfig()" 进行处理的文件来发送。 返回
   一个 "Thread" 实例，你可以在该实例上调用 "start()" 来启动服务器，对
   该服务器你可以在适当的时候执行 "join()"。 要停止该服务器，请调用
   "stopListening()"。

   如果指定 "verify" 参数，则它应当是一个可调用对象，该对象应当验证通
   过套接字接收的字节数据是否有效且应被处理。 这可以通过对通过套接字发
   送的内容进行加密和/或签名来完成，这样 "verify" 可调用对象就能执行签
   名验证和/或解密。 "verify" 可调用对象的调用会附带一个参数 —— 通过套
   接字接收的字节数据 —— 并应当返回要处理的字节数据，或者返回 "None"
   来指明这些字节数据应当被丢弃。 返回的字节数据可以与传入的字节数据相
   同（例如在只执行验证的时候），或者也可以完全不同（例如在可能执行了
   解密的时候）。

   要将配置发送到套接字，请读取配置文件并将其作为字节序列发送到套接字
   ，字节序列要以使用 "struct.pack('>L', n)" 打包为二进制格式的四字节
   长度的字符串打头。

   備註:

     因为配置的各部分是通过 "eval()" 传递的，使用此函数可能让用户面临
     安全风险。虽然此函数仅绑定到 "localhost" 上的套接字，因此并不接受
     来自远端机器的连接，但在某些场景中不受信任的代码可以在调用
     "listen()" 的进程的账户下运行。具体来说，如果如果调用 "listen()"
     的进程在用户无法彼此信任的多用户机器上运行，则恶意用户就能简单地
     通过连接到受害者的 "listen()" 套接字并发送运行攻击者想在受害者的
     进程上执行的任何代码的配置的方式，安排运行几乎任意的代码。如果是
     使用默认端口这会特别容易做到，即便使用了不同端口也不难做到。要避
     免发生这种情况的风险，请在 "listen()" 中使用 "verify" 参数来防止
     未经认可的配置被应用。

   3.4 版更變: 新增 "verify" 引數。

   備註:

     如果你希望将配置发送给未禁用现有日志记录器的监听器，你将需要使用
     JSON 格式的配置，该格式将使用 "dictConfig()" 进行配置。 此方法允
     许你在你发送的配置中将 "disable_existing_loggers" 指定为 "False"
     。

logging.config.stopListening()

   停止通过对 "listen()" 的调用所创建的监听服务器。 此函数的调用通常会
   先于在 "listen()" 的返回值上调用 "join()"。


安全考量
========

日志配置功能试图提供便利，从某种角度来说，这是通过将配置文件中的文本转
换为日志配置中使用的 Python 对象来完成的 —— 如 用户定义对象 中所述。但
是，这些相同的机制（从用户定义的模块中导入可调用对象并使用配置中的参数
调用它们）可用于调用您指定的任何代码，因此您应该*非常谨慎*地处理来自非
信任源的配置文件。并且在实际加载前，您应该确信加载不会导致坏事情。


配置字典架构
============

描述日志记录配置需要列出要创建的不同对象及它们之间的连接；例如，你可以
创建一个名为 'console' 的处理器，然后名为 'startup' 的日志记录器将可以
把它的消息发送给 'console' 处理器。 这些对象并不仅限于 "logging" 模块
所提供的对象，因为你还可以编写你自己的格式化或处理器类。 这些类的形参
可能还需要包括 "sys.stderr" 这样的外部对象。 描述这些对象和连接的语法
会在下面的 对象连接 中定义。


字典架构细节
------------

传给 "dictConfig()" 的字典必须包含以下的键:

* *version* - 应设为代表架构版本的整数值。 目前唯一有效的值是 1，使用
  此键可允许架构在继续演化的同时保持向下兼容性。

所有其他键都是可选项，但如存在它们将根据下面的描述来解读。 在下面提到
'configuring dict' 的所有情况下，都将检查它的特殊键 "'()'" 以确定是否
需要自定义实例化。 如果需要，则会使用下面 用户定义对象 所描述的机制来
创建一个实例；否则，会使用上下文来确定要实例化的对象。

* *formatters* - 对应的值将是一个字典，其中每个键是一个格式器 ID 而每
  个值则是一个描述如何配置相应 "Formatter" 实例的字典。

  在配置字典中搜索以下可选键，这些键对应于创建 "Formatter" 对象时传入
  的参数。:

     * "format"

     * "datefmt"

     * "style"

     * "validate" (从版本 >=3.8 起)

  可选的 "class" 键指定格式化器类的名称（形式为带点号的模块名和类名）
  。 实例化的参数与 "Formatter" 的相同，因此这个键对于实例化自定义的
  "Formatter" 子类最为有用。 如果，替代类可能 会以扩展和精简格式呈现异
  常回溯信息。 如果你的格式化器需要不同的或额外的配置键，你应当使用 用
  户定义对象。

* *filters* - 对应的值将是一个字典，其中每个键是一个过滤器 ID 而每个值
  则是一个描述如何配置相应 Filter 实例的字典。

  将在配置字典中搜索键 "name" (默认值为空字符串) 并且该键会被用于构造
  "logging.Filter" 实例。

* *handlers* - 对应的值将是一个字典，其中每个键是一个处理器 ID 而每个
  值则是一个描述如何配置相应 Handler 实例的字典。

  将在配置字典中搜索下列键:

  * "class" (强制)。 这是处理器类的完整限定名称。

  * "level" (可选)。 处理器的级别。

  * "formatter" (可选)。 处理器所对应格式化器的 ID。

  * "filters" (可选)。 由处理器所对应过滤器的 ID 组成的列表。

  所有 *其他* 键会被作为关键字参数传递给处理器类的构造器。 例如，给定
  如下配置:

     handlers:
       console:
         class : logging.StreamHandler
         formatter: brief
         level   : INFO
         filters: [allow_foo]
         stream  : ext://sys.stdout
       file:
         class : logging.handlers.RotatingFileHandler
         formatter: precise
         filename: logconfig.log
         maxBytes: 1024
         backupCount: 3

  ID 为 "console" 的处理器会被实例化为 "logging.StreamHandler"，并使用
  "sys.stdout" 作为下层流。 ID 为 "file" 的处理器会被实例化为
  "logging.handlers.RotatingFileHandler"，并附带关键字参数
  "filename='logconfig.log', maxBytes=1024, backupCount=3"。

* *loggers* - 对应的值将是一个字典，其中每个键是一个日志记录器名称而每
  个值则是一个描述如何配置相应 Logger 实例的字典。

  将在配置字典中搜索下列键:

  * "level" (可选)。 日志记录器的级别。

  * "propagate" (可选)。 日志记录器的传播设置。

  * "filters" (可选)。 由日志记录器对应过滤器的 ID 组成的列表。

  * "handlers" (可选)。 由日志记录器对应处理器的 ID 组成的列表。

  指定的记录器将根据指定的级别、传播、过滤器和处理器来配置。

* *root* - 这将成为根日志记录器对应的配置。 配置的处理方式将与所有日志
  记录器一致，除了 "propagate" 设置将不可用之外。

* *incremental* - 配置是否要被解读为在现有配置上新增。 该值默认为
  "False"，这意味着指定的配置将以与当前 "fileConfig()" API 所使用的相
  同语义来替代现有的配置。

  如果指定的值为 "True"，配置会按照 增量配置 部分所描述的方式来处理。

* *disable_existing_loggers* - 是否要禁用任何现有的非根日志记录器。 该
  设置对应于 "fileConfig()" 中的同名形参。 如果省略，则此形参默认为
  "True"。 如果 *incremental* 为 "True" 则该省会被忽略。


增量配置
--------

为增量配置提供完全的灵活性是很困难的。 例如，由于过滤器和格式化器这样
的对象是匿名的，一旦完成配置，在增加配置时就不可能引用这些匿名对象。

此外，一旦完成了配置，在运行时任意改变日志记录器、处理器、过滤器、格式
化器的对象图就不是很有必要；日志记录器和处理器的详细程度只需通过设置级
别即可实现控制（对于日志记录器则可设置传播旗标）。 在多线程环境中以安
全的方式任意改变对象图也许会导致问题；虽然并非不可能，但这样做的好处不
足以抵销其所增加的实现复杂度。

这样，当配置字典的 "incremental" 键存在且为 "True" 时，系统将完全忽略
任何 "formatters" 和 "filters" 条目，并仅会处理 "handlers" 条目中的
"level" 设置，以及 "loggers" 和 "root" 条目中的 "level" 和 "propagate"
设置。

使用配置字典中的值可让配置以封存字典对象的形式通过线路传送给套接字监听
器。 这样，长时间运行的应用程序的日志记录的详细程度可随时间改变而无须
停止并重新启动应用程序。


对象连接
--------

该架构描述了一组日志记录对象 —— 日志记录器、处理器、格式化器、过滤器
—— 它们在对象图中彼此连接。 因此，该架构需要能表示对象之间的连接。 例
如，在配置完成后，一个特定的日志记录器关联到了一个特定的处理器。 出于
讨论的目的，我们可以说该日志记录器代表两者间连接的源头，而处理器则代表
对应的目标。 当然在已配置对象中这是由包含对处理器的引用的日志记录器来
代表的。 在配置字典中，这是通过给每个目标对象一个 ID 来无歧义地标识它
，然后在源头对象中使用该 ID 来实现的。

因此，举例来说，考虑以下 YAML 代码段:

   formatters:
     brief:
       # configuration for formatter with id 'brief' goes here
     precise:
       # configuration for formatter with id 'precise' goes here
   handlers:
     h1: #This is an id
      # configuration of handler with id 'h1' goes here
      formatter: brief
     h2: #This is another id
      # configuration of handler with id 'h2' goes here
      formatter: precise
   loggers:
     foo.bar.baz:
       # other configuration for logger 'foo.bar.baz'
       handlers: [h1, h2]

（注：这里使用 YAML 是因为它的可读性比表示字典的等价 Python 源码形式更
好。）

日志记录器 ID 就是日志记录器的名称，它会在程序中被用来获取对日志记录器
的引用，例如 "foo.bar.baz"。 格式化器和过滤器的 ID 可以是任意字符串值
(例如上面的 "brief", "precise") 并且它们是瞬态的，因为它们仅对处理配置
字典有意义并会被用来确定对象之间的连接，而当配置调用完成时不会在任何地
方保留。

上面的代码片段指明名为 "foo.bar.baz" 的日志记录器应当关联到两个处理器
，它们的 ID 是 "h1" 和 "h2"。 "h1" 的格式化器的 ID 是 "brief"，而 "h2"
的格式化器的 ID 是 "precise"。


用户定义对象
------------

此架构支持用户定义对象作为处理器、过滤器和格式化器。 （日志记录器的不
同实例不需要具有不同类型，因此这个配置架构并不支持用户定义日志记录器类
。）

要配置的对象是由字典描述的，其中包含它们的配置详情。 在某些地方，日志
记录系统将能够从上下文中推断出如何实例化一个对象，但是当要实例化一个用
户自定义对象时，系统将不知道要如何做。 为了提供用户自定义对象实例化的
完全灵活性，用户需要提供一个‘工厂’函数 —— 即在调用时传入配置字典并返回
实例化对象的可调用对象。 这是用一个通过特殊键 "'()'" 来访问的工厂函数
的绝对导入路径来标示的。 下面是一个实际的例子:

   formatters:
     brief:
       format: '%(message)s'
     default:
       format: '%(asctime)s %(levelname)-8s %(name)-15s %(message)s'
       datefmt: '%Y-%m-%d %H:%M:%S'
     custom:
         (): my.package.customFormatterFactory
         bar: baz
         spam: 99.9
         answer: 42

上面的 YAML 代码片段定义了三个格式化器。 第一个的 ID 为 "brief"，是带
有特殊格式字符串的标准 "logging.Formatter" 实例。 第二个的 ID 为
"default"，具有更长的格式同时还显式地定义了时间格式，并将最终实例化一
个带有这两个格式字符串的 "logging.Formatter"。 以 Python 源代码形式显
示的 "brief" 和 "default" 格式化器分别具有下列配置子字典:

   {
     'format' : '%(message)s'
   }

和：

   {
     'format' : '%(asctime)s %(levelname)-8s %(name)-15s %(message)s',
     'datefmt' : '%Y-%m-%d %H:%M:%S'
   }

并且由于这些字典不包含特殊键 "'()'"，实例化方式是从上下文中推断出来的
：结果会创建标准的 "logging.Formatter" 实例。 第三个格式器的 ID 为
"custom"，对应配置子字典为:

   {
     '()' : 'my.package.customFormatterFactory',
     'bar' : 'baz',
     'spam' : 99.9,
     'answer' : 42
   }

并且它包含特殊键 "'()'"，这意味着需要用户自定义实例化方式。 在此情况下
，将使用指定的工厂可调用对象。 如果它本身就是一个可调用对象则将被直接
使用 —— 否则如果你指定了一个字符串（如这个例子所示）则将使用正常的导入
机制来定位实例的可调用对象。 调用该可调用对象将传入配置子字典中 **剩余
的** 条目作为关键字参数。 在上面的例子中，调用将预期返回 ID 为
"custom" 的格式化器:

   my.package.customFormatterFactory(bar='baz', spam=99.9, answer=42)

警告:

  上面示例中 "bar", "spam" 和 "answer" 等键的值不应是配置目录或引用如
  "cfg://foo" 或 "ext://bar"，因为它们将不会被配置机制所处理，而是被原
  样传给可调用对象。

将 "'()'" 用作特殊键是因为它不是一个有效的关键字形参名称，这样就不会与
调用中使用的关键字参数发生冲突。 "'()'" 还被用作表明对应值为可调用对象
的助记符。

你还可以指定一个特殊键 "'.'" 其值应为一个将属性名称映射到对应值的字典
。 如果找到，则指定的属性在被返回之前将在用户定义的对象上设置。 因此，
使用以下配置:

   {
     '()' : 'my.package.customFormatterFactory',
     'bar' : 'baz',
     'spam' : 99.9,
     'answer' : 42,
     '.' {
       'foo': 'bar',
       'baz': 'bozz'
     }
   }

被返回的格式化器的 "foo" 属性将设为 "'bar'" 而 "baz" 属性将设为
"'bozz'"。

警告:

  上面示例中 "foo" 和 "baz" 等属性的值不应是配置目录或引用如
  "cfg://foo" 或 "ext://bar"，因为它们将不会被配置机制所处理，而是被原
  样设置为属性。


处理器配置顺序
--------------

处理器按其键的字母顺序进行配置，而已配置的处理器将替换配置方案内部
"handlers" 字典（的一个工作副本）中的配置字典。 如果你使用
"cfg://handlers.foo" 这样的构造，那么在初始状态下 "handlers['foo']" 会
指向名为 "foo" 的处理器的配置字典，随后（一旦配置了该处理器）它将指向
已配置的处理器实例。 因此，"cfg://handlers.foo" 可以解析为一个字典或处
理器实例。 通常来说，对于带依赖的处理器采用在它们所依赖的任何处理器完
成配置 _之后_ 再进行配置的方式来命名处理器是一种明智的做法；这将允许使
用 "cfg://handlers.foo" 这样的构造来配置依赖于处理器 "foo" 的处理器。
如果这个带依赖的处理器被命名为 "bar"，则会导致问题，因为 "bar" 的配置
将在 "foo" 的配置之前被尝试使用，而 "foo" 将尚未配置完成。 但是，如果
带依赖的处理器被命名为 "foobar"，则它将在 "foo" 之后被配置，结果就是
"cfg://handlers.foo" 将被解析为已配置的处理器 "foo"，而不是其配置字典
。


访问外部对象
------------

有时一个配置需要引用配置以外的对象，例如 "sys.stderr"。 如果配置字典是
使用 Python 代码构造的，这会很直观，但是当配置是通过文本文件（例如
JSON, YAML）提供的时候就会引发问题。 在一个文本文件中，没有将
"sys.stderr" 与字符串字面值 "'sys.stderr'" 区分开来的标准方式。  为了
实现这种区分，配置系统会在字符串值中查找规定的特殊前缀并对其做特殊处理
。 例如，如果在配置中将字符串字面值 "'ext://sys.stderr'" 作为一个值来
提供，则 "ext://" 将被去除而该值的剩余部分将使用正常导入机制来处理。

此类前缀的处理方式类似于协议处理：存在一种通用机制来查找与正则表达式
"^(?P<prefix>[a-z]+)://(?P<suffix>.*)$" 相匹配的前缀，如果识别出了
"prefix"，则 "suffix" 会以与前缀相对应的方式来处理并且处理的结果将替代
原字符串值。 如果未识别出前缀，则原字符串将保持不变。


访问内部对象
------------

除了外部对象，有时还需要引用配置中的对象。 这将由配置系统针对它所了解
的内容隐式地完成。 例如，在日志记录器或处理器中表示 "level" 的字符串值
"'DEBUG'" 将被自动转换为值 "logging.DEBUG"，而 "handlers", "filters"
和 "formatter" 条目将接受一个对象 ID 并解析为适当的目标对象。

但是，对于 "logging" 模块所不了解的用户自定义对象则需要一种更通用的机
制。 例如，考虑 "logging.handlers.MemoryHandler"，它接受一个 "target"
参数即其所委托的另一个处理器。 由于系统已经知道存在该类，因而在配置中
，给定的 "target" 只需为相应目标处理器的的对象 ID 即可，而系统将根据该
ID 解析出处理器。 但是，如果用户定义了一个具有 "alternate" 处理器的
"my.package.MyHandler"，则配置程序将不知道 "alternate" 指向的是一个处
理器。 为了应对这种情况，通用解析系统允许用户指定:

   handlers:
     file:
       # configuration of file handler goes here

     custom:
       (): my.package.MyHandler
       alternate: cfg://handlers.file

字符串字面值 "'cfg://handlers.file'" 将按照与 "ext://" 前缀类似的方式
被解析为结果字符串，但查找操作是在配置自身而不是在导入命名空间中进行。
该机制允许按点号或按索引来访问，与 "str.format" 所提供的方式类似。 这
样，给定以下代码段:

   handlers:
     email:
       class: logging.handlers.SMTPHandler
       mailhost: localhost
       fromaddr: my_app@domain.tld
       toaddrs:
         - support_team@domain.tld
         - dev_team@domain.tld
       subject: Houston, we have a problem.

在该配置中，字符串 "'cfg://handlers'" 将解析为带有 "handlers" 键的字典
，字符串 "'cfg://handlers.email" 将解析为具有 "email" 键的 "handlers"
字典中的字典，依此类推。 字符串 "'cfg://handlers.email.toaddrs[1]" 将
解析为 "'dev_team@domain.tld'" 而字符串
"'cfg://handlers.email.toaddrs[0]'" 将解析为值
"'support_team@domain.tld'"。 "subject" 值 可以使用
"'cfg://handlers.email.subject'" 或者等价的
"'cfg://handlers.email[subject]'" 来访问。 后一种形式仅在键包含空格或
非字母类数字类字符的情况下才需要使用。 如果一个索引仅由十进制数码构成
，则将尝试使用相应的整数值来访问，如果有必要则将回退为字符串值。

给定字符串 "cfg://handlers.myhandler.mykey.123"，这将解析为
"config_dict['handlers']['myhandler']['mykey']['123']"。 如果字符串被
指定为 "cfg://handlers.myhandler.mykey[123]"，系统将尝试从
"config_dict['handlers']['myhandler']['mykey'][123]" 中提取值，并在尝
试失败时回退为 "config_dict['handlers']['myhandler']['mykey']['123']"
。


导入解析与定制导入器
--------------------

导入解析默认使用内置的 "__import__()" 函数来执行导入。 你可能想要将其
替换为你自己的导入机制：如果是这样的话，你可以替换 "DictConfigurator"
或其超类 "BaseConfigurator" 类的 "importer" 属性。 但是你必须小心谨慎
，因为函数是从类中通过描述器方式来访问的。 如果你使用 Python 可调用对
象来执行导入，并且你希望在类层级而不是在实例层级上定义它，则你需要用
"staticmethod()" 来装饰它。 例如:

   from importlib import import_module
   from logging.config import BaseConfigurator

   BaseConfigurator.importer = staticmethod(import_module)

如果你是在一个配置器的 *实例* 上设置导入可调用对象则你不需要用
"staticmethod()" 来装饰。


配置文件格式
============

"fileConfig()" 所能理解的配置文件格式是基于 "configparser" 功能的。 该
文件必须包含 "[loggers]", "[handlers]" 和 "[formatters]" 等小节，它们
通过名称来标识文件中定义的每种类型的实体。 对于每个这样的实体，都有单
独的小节来标识实体的配置方式。 因此，对于 "[loggers]" 小节中名为
"log01" 的日志记录器，相应的配置详情保存在 "[logger_log01]" 小节中。
类似地，对于 "[handlers]" 小节中名为 "hand01" 的处理器，其配置将保存在
名为 "[handler_hand01]" 的小节中，而对于 "[formatters]" 小节中名为
"form01" 的格式化器，其配置将在名为 "[formatter_form01]" 的小节中指定
。 根日志记录器的配置必须在名为 "[logger_root]" 的小节中指定。

備註:

  "fileConfig()" API 比 "dictConfig()" API 更旧因而没有提供涵盖日志记
  录特定方面的功能。 例如，你无法配置 "Filter" 对象，该对象使用
  "fileConfig()" 提供超出简单整数级别的消息过滤功能。 如果你想要在你的
  日志记录配置中包含 "Filter" 的实例，你将必须使用 "dictConfig()"。 请
  注意未来还将向 "dictConfig()" 添加对配置功能的强化，因此值得考虑在方
  便的时候转换到这个新 API。

在文件中这些小节的例子如下所示。

   [loggers]
   keys=root,log02,log03,log04,log05,log06,log07

   [handlers]
   keys=hand01,hand02,hand03,hand04,hand05,hand06,hand07,hand08,hand09

   [formatters]
   keys=form01,form02,form03,form04,form05,form06,form07,form08,form09

根日志记录器必须指定一个级别和一个处理器列表。 根日志小节的例子如下所
示。

   [logger_root]
   level=NOTSET
   handlers=hand01

"level" 条目可以为 "DEBUG, INFO, WARNING, ERROR, CRITICAL" 或 "NOTSET"
之一。 其中 "NOTSET" 仅适用于根日志记录器，表示将会记录所有消息。 级别
值会在 "logging" 包命名空间的上下文中通过 "eval()" 来得出。

"handlers" 条目是以逗号分隔的处理器名称列表，它必须出现于 "[handlers]"
小节并且在配置文件中有相应的小节。

对于根日志记录器以外的日志记录器，还需要某些附加信息。 下面的例子演示
了这些信息。

   [logger_parser]
   level=DEBUG
   handlers=hand01
   propagate=1
   qualname=compiler.parser

"level" 和 "handlers" 条目的解释方式与根日志记录器的一致，不同之处在于
如果一个非根日志记录器的级别被指定为 "NOTSET"，则系统会咨询更高层级的
日志记录器来确定该日志记录器的有效级别。 "propagate" 条目设为 1 表示消
息必须从此日志记录器传播到更高层级的处理器，设为 0 表示消息 **不会**
传播到更高层级的处理器。 "qualname" 条目是日志记录器的层级通道名称，也
就是应用程序获取日志记录器所用的名称。

指定处理器配置的小节说明如下。

   [handler_hand01]
   class=StreamHandler
   level=NOTSET
   formatter=form01
   args=(sys.stdout,)

"class" 条目指明处理器的类（由 "logging" 包命名空间中的 "eval()" 来确
定）。 "level" 会以与日志记录器相同的方式来解读，"NOTSET" 会被视为表示
‘记录一切消息’。

"formatter" 条目指明此处理器的格式化器的键名称。 如为空白，则会使用默
认的格式化器 ("logging._defaultFormatter")。 如果指定了名称，则它必须
出现于 "[formatters]" 小节并且在配置文件中有相应的小节。

"args" 条目，当在 "logging" 包命名空间的上下文中执行 "eval()" 时将是传
给处理器类构造器的参数列表。 请参阅相应处理器的构造器说明或者下面的示
例，以了解典型的条目是如何构造的。 如果未提供，则默认为 "()"。

可选的 "kwargs" 条目，当在 "logging" 包命名空间的上下文中执行 "eval()"
时将是传给处理器的构造器的关键字参数字典。 如果未提供，则默认为 "{}"。

   [handler_hand02]
   class=FileHandler
   level=DEBUG
   formatter=form02
   args=('python.log', 'w')

   [handler_hand03]
   class=handlers.SocketHandler
   level=INFO
   formatter=form03
   args=('localhost', handlers.DEFAULT_TCP_LOGGING_PORT)

   [handler_hand04]
   class=handlers.DatagramHandler
   level=WARN
   formatter=form04
   args=('localhost', handlers.DEFAULT_UDP_LOGGING_PORT)

   [handler_hand05]
   class=handlers.SysLogHandler
   level=ERROR
   formatter=form05
   args=(('localhost', handlers.SYSLOG_UDP_PORT), handlers.SysLogHandler.LOG_USER)

   [handler_hand06]
   class=handlers.NTEventLogHandler
   level=CRITICAL
   formatter=form06
   args=('Python Application', '', 'Application')

   [handler_hand07]
   class=handlers.SMTPHandler
   level=WARN
   formatter=form07
   args=('localhost', 'from@abc', ['user1@abc', 'user2@xyz'], 'Logger Subject')
   kwargs={'timeout': 10.0}

   [handler_hand08]
   class=handlers.MemoryHandler
   level=NOTSET
   formatter=form08
   target=
   args=(10, ERROR)

   [handler_hand09]
   class=handlers.HTTPHandler
   level=NOTSET
   formatter=form09
   args=('localhost:9022', '/log', 'GET')
   kwargs={'secure': True}

指定格式化器配置的小节说明如下。

   [formatter_form01]
   format=F1 %(asctime)s %(levelname)s %(message)s
   datefmt=
   style=%
   validate=True
   class=logging.Formatter

用于格式化器配置的参数与字典规范 格式化器部分 中的键相同。

備註:

  由于如上所述使用了 "eval()"，因此使用 "listen()" 通过套接字来发送和
  接收配置会导致潜在的安全风险。 此风险仅限于相互间没有信任的多个用户
  在同一台机器上运行代码的情况；请参阅 "listen()" 了解更多信息。

也參考:

  "logging" 模組
     日志记录模块的 API 参考。

  "logging.handlers" 模組
     日志记录模块附带的有用处理器。
