"pprint" — L’affichage élégant de données
*****************************************

**Code source :** Lib/pprint.py

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

Le module "pprint" permet « d’afficher élégamment » des structures de
données Python arbitraires sous une forme qui peut être utilisée
ensuite comme une entrée dans l’interpréteur. Si les structures
formatées incluent des objets qui ne sont pas des types Python
fondamentaux, leurs représentations peuvent ne pas être acceptables en
tant que telles par l’interpréteur. Cela peut être le cas si des
objets tels que des fichiers, des interfaces de connexion (*sockets*
en anglais) ou des classes sont inclus, c’est aussi valable pour
beaucoup d’autres types d’objets qui ne peuvent être représentés sous
forme littérale en Python.

L’affichage formaté affiche, tant que possible, les objets sur une
seule ligne et les sépare sur plusieurs lignes s’ils dépassent la
largeur autorisée par l’interpréteur. Cette largeur est ajustable par
le paramètre *width* qui est par défaut de 80 caractères.

Modifié dans la version 3.9: Prise en charge de l’affichage élégant de
"types.SimpleNamespace".

Modifié dans la version 3.10: Prise en charge de l’affichage élégant
de "dataclasses.dataclass".


Fonctions
=========

pprint.pp(object, stream=None, indent=1, width=80, depth=None, *, compact=False, sort_dicts=False, underscore_numbers=False)

   Affiche la représentation formatée de *object*, suivie d’un retour
   à la ligne. Cette fonction peut être utilisée dans l’interpréteur
   interactif au lieu de la fonction "print()" pour inspecter les
   valeurs. Astuce : vous pouvez réassigner "print = pprint.pp" pour
   l’utiliser dans une portée.

   Paramètres:
      * **object** -- L’objet à afficher.

      * **stream** (*file-like object* | None) -- Un objet fichier-
        compatible sur lequel la sortie sera écrite en appelant sa
        méthode "write()". Si la valeur est "None" (la valeur par
        défaut), "sys.stdout" est utilisé.

      * **indent** (*int*) -- La quantité d’indentation ajoutée pour
        chaque niveau d’imbrication.

      * **width** (*int*) -- Le nombre maximal de caractères souhaités
        par ligne dans la sortie. Si une structure ne peut pas être
        formatée dans la largeur limite donnée, un effort maximal sera
        fait.

      * **depth** (*int** | **None*) -- Le nombre de niveaux
        d’imbrication qui peuvent être affichés. Si la structure de
        données à afficher est trop profonde, le niveau suivant est
        remplacé par "...". Si la valeur est "None" (la valeur par
        défaut), il n’y a pas de contrainte sur la profondeur des
        objets formatés.

      * **compact** (*bool*) -- Contrôle la façon dont les longues
        *sequences* sont formatées. Si la valeur est "False" (la
        valeur par défaut), chaque élément d’une séquence sera formaté
        sur une ligne séparée, sinon autant d’éléments que possible
        qui peuvent tenir dans la *width* (la largeur) seront formatés
        sur chaque ligne de sortie.

      * **sort_dicts** (*bool*) -- Si la valeur est "True", les
        dictionnaires seront formatés avec leurs clés triées, sinon
        ils seront affichés dans l'ordre d'insertion (la valeur par
        défaut).

      * **underscore_numbers** (*bool*) -- Si la valeur est "True",
        les entiers seront formatés avec le caractère "_" comme
        séparateur de milliers, sinon les underscores ne sont pas
        affichés (le comportement par défaut).

   >>> import pprint
   >>> stuff = ['spam', 'eggs', 'lumberjack', 'knights', 'ni']
   >>> stuff.insert(0, stuff)
   >>> pprint.pp(stuff)
   [<Recursion on list with id=...>,
    'spam',
    'eggs',
    'lumberjack',
    'knights',
    'ni']

   Ajouté dans la version 3.8.

pprint.pprint(object, stream=None, indent=1, width=80, depth=None, *, compact=False, sort_dicts=True, underscore_numbers=False)

   Alias pour "pp()" avec *sort_dicts* initialisé à "True" par défaut
   (ce qui trie automatiquement les clés des dictionnaires), vous
   pourriez préférer utiliser "pp()" à la place où *sort_dicts* est
   initialisé à "False" par défaut.

pprint.pformat(object, indent=1, width=80, depth=None, *, compact=False, sort_dicts=True, underscore_numbers=False)

   Renvoie une représentation formatée de *object* sous forme de
   chaîne de caractères. *indent*, *width*, *depth*, *compact*,
   *sort_dicts* et  *underscore_numbers* sont passés au constructeur
   de "PrettyPrinter" comme paramètres de formatage et leurs
   significations sont telles que décrites dans la documentation ci-
   dessus.

pprint.isreadable(object)

   Détermine si la représentation formatée de *object* est « lisible
   », ou s’il peut être utilisé pour recomposer sa valeur en utilisant
   la fonction "eval()". Cela renvoie toujours "False" pour les objets
   récursifs.

   >>> pprint.isreadable(stuff)
   False

pprint.isrecursive(object)

   Détermine si *object* nécessite une représentation récursive. Cette
   fonction est soumise aux mêmes limitations que celles notées dans
   "saferepr()" ci-dessous et peut lever une exception
   "RecursionError" si elle échoue à détecter un objet récursif.

pprint.saferepr(object)

   Renvoie une représentation de *object*, protégée contre la
   récursivité dans certaines structures de données courantes, à
   savoir que les instances de "dict", "list" et "tuple" ou les sous-
   classes dont la méthode "__repr__" n’ont pas été surchargées. Si la
   représentation de *object* présente une entrée récursive, celle-ci
   sera représentée telle que "<Recursion on typename with
   id=number>". Par ailleurs, la représentation de l’objet n’est pas
   formatée.

   >>> pprint.saferepr(stuff)
   "[<Recursion on list with id=...>, 'spam', 'eggs', 'lumberjack', 'knights', 'ni']"


Les Objets PrettyPrinter
========================

class pprint.PrettyPrinter(indent=1, width=80, depth=None, stream=None, *, compact=False, sort_dicts=True, underscore_numbers=False)

   Crée une instance de "PrettyPrinter".

   Les arguments ont la même signification que pour "pp()". Notez
   qu’ils sont dans un ordre différent, et que *sort_dicts* est
   initialisé à "True" par défaut.

   >>> import pprint
   >>> stuff = ['spam', 'eggs', 'lumberjack', 'knights', 'ni']
   >>> stuff.insert(0, stuff[:])
   >>> pp = pprint.PrettyPrinter(indent=4)
   >>> pp.pprint(stuff)
   [   ['spam', 'eggs', 'lumberjack', 'knights', 'ni'],
       'spam',
       'eggs',
       'lumberjack',
       'knights',
       'ni']
   >>> pp = pprint.PrettyPrinter(width=41, compact=True)
   >>> pp.pprint(stuff)
   [['spam', 'eggs', 'lumberjack',
     'knights', 'ni'],
    'spam', 'eggs', 'lumberjack', 'knights',
    'ni']
   >>> tup = ('spam', ('eggs', ('lumberjack', ('knights', ('ni', ('dead',
   ... ('parrot', ('fresh fruit',))))))))
   >>> pp = pprint.PrettyPrinter(depth=6)
   >>> pp.pprint(tup)
   ('spam', ('eggs', ('lumberjack', ('knights', ('ni', ('dead', (...)))))))

   Modifié dans la version 3.4: Ajout du paramètre *compact*.

   Modifié dans la version 3.8: Ajout du paramètre *sort_dicts*.

   Modifié dans la version 3.10: Ajout du paramètre
   *underscore_numbers*.

   Modifié dans la version 3.11: N'essaie plus d’écrire sur
   "sys.stdout" lorsqu'il vaut "None".

Les instances de la classe "PrettyPrinter" ont les méthodes suivantes
:

PrettyPrinter.pformat(object)

   Renvoie la représentation formatée de *object*. Cela prend en
   compte les options passées au constructeur de la classe
   "PrettyPrinter".

PrettyPrinter.pprint(object)

   Affiche sur le flux configuré la représentation formatée de
   *object*, suivie d’une fin de ligne.

Les méthodes suivantes fournissent les implémentations pour les
fonctions correspondantes de mêmes noms. L’utilisation de ces méthodes
sur une instance est légèrement plus efficace, car les nouveaux objets
"PrettyPrinter" n’ont pas besoin d’être créés.

PrettyPrinter.isreadable(object)

   Détermine si la représentation formatée de *object* est « lisible
   », ou si elle peut être utilisée pour recomposer sa valeur en
   utilisant la fonction "eval()". Cela renvoie toujours "False" pour
   les objets récursifs. Si le paramètre *depth* de la classe
   "PrettyPrinter" est initialisé et que l’objet est plus « profond »
   que permis, cela renvoie "False".

PrettyPrinter.isrecursive(object)

   Détermine si l’objet nécessite une représentation récursive.

Cette méthode est fournie sous forme de point d’entrée ou méthode (à
déclenchement) automatique (*hook* en anglais) pour permettre aux
sous-classes de modifier la façon dont les objets sont convertis en
chaînes. L’implémentation par défaut est celle de la fonction
"saferepr()".

PrettyPrinter.format(object, context, maxlevels, level)

   Renvoie trois valeurs : la version formatée de *object* sous forme
   de chaîne de caractères, une option indiquant si le résultat est «
   lisible », et une option indiquant si une récursion a été détectée.
   Le premier argument est l’objet à représenter. Le deuxième est un
   dictionnaire qui contient l'"id()" des objets (conteneurs directs
   ou indirects de *objet* qui affectent sa représentation) qui font
   partie du contexte de représentation courant tel que les clés; si
   un objet doit être représenté, mais l’a déjà été dans ce contexte,
   le troisième argument renvoie "True". Des appels récursifs à la
   méthode "format()" doivent ajouter des entrés additionnelles aux
   conteneurs de ce dictionnaire. Le troisième argument *maxlevels*,
   donne la limite maximale de récursivité; la valeur par défaut est
   "0". Cet argument doit être passé non modifié pour des appels non
   récursifs. Le quatrième argument, *level*, donne le niveau de
   récursivité courant; les appels récursifs doivent être passés à une
   valeur inférieure à celle de l’appel courant.


Exemple
=======

Pour illustrer quelques cas pratiques de l’utilisation de la fonction
"pp()" et de ses paramètres, allons chercher des informations sur un
projet PyPI :

   >>> import json
   >>> import pprint
   >>> from urllib.request import urlopen
   >>> with urlopen('https://pypi.org/pypi/sampleproject/1.2.0/json') as resp:
   ...     project_info = json.load(resp)['info']

Dans sa forme basique, la fonction "pp()" affiche l’intégralité de
l’objet :

   >>> pprint.pp(project_info)
   {'author': 'The Python Packaging Authority',
    'author_email': 'pypa-dev@googlegroups.com',
    'bugtrack_url': None,
    'classifiers': ['Development Status :: 3 - Alpha',
                    'Intended Audience :: Developers',
                    'License :: OSI Approved :: MIT License',
                    'Programming Language :: Python :: 2',
                    'Programming Language :: Python :: 2.6',
                    'Programming Language :: Python :: 2.7',
                    'Programming Language :: Python :: 3',
                    'Programming Language :: Python :: 3.2',
                    'Programming Language :: Python :: 3.3',
                    'Programming Language :: Python :: 3.4',
                    'Topic :: Software Development :: Build Tools'],
    'description': 'A sample Python project\n'
                   '=======================\n'
                   '\n'
                   'This is the description file for the project.\n'
                   '\n'
                   'The file should use UTF-8 encoding and be written using '
                   'ReStructured Text. It\n'
                   'will be used to generate the project webpage on PyPI, and '
                   'should be written for\n'
                   'that purpose.\n'
                   '\n'
                   'Typical contents for this file would include an overview of '
                   'the project, basic\n'
                   'usage examples, etc. Generally, including the project '
                   'changelog in here is not\n'
                   'a good idea, although a simple "What\'s New" section for the '
                   'most recent version\n'
                   'may be appropriate.',
    'description_content_type': None,
    'docs_url': None,
    'download_url': 'UNKNOWN',
    'downloads': {'last_day': -1, 'last_month': -1, 'last_week': -1},
    'home_page': 'https://github.com/pypa/sampleproject',
    'keywords': 'sample setuptools development',
    'license': 'MIT',
    'maintainer': None,
    'maintainer_email': None,
    'name': 'sampleproject',
    'package_url': 'https://pypi.org/project/sampleproject/',
    'platform': 'UNKNOWN',
    'project_url': 'https://pypi.org/project/sampleproject/',
    'project_urls': {'Download': 'UNKNOWN',
                     'Homepage': 'https://github.com/pypa/sampleproject'},
    'release_url': 'https://pypi.org/project/sampleproject/1.2.0/',
    'requires_dist': None,
    'requires_python': None,
    'summary': 'A sample Python project',
    'version': '1.2.0'}

Le résultat peut être limité à une certaine profondeur en initialisant
*depth*. ( "…" est utilisé pour des contenus plus « profonds ») :

   >>> pprint.pp(project_info, depth=1)
   {'author': 'The Python Packaging Authority',
    'author_email': 'pypa-dev@googlegroups.com',
    'bugtrack_url': None,
    'classifiers': [...],
    'description': 'A sample Python project\n'
                   '=======================\n'
                   '\n'
                   'This is the description file for the project.\n'
                   '\n'
                   'The file should use UTF-8 encoding and be written using '
                   'ReStructured Text. It\n'
                   'will be used to generate the project webpage on PyPI, and '
                   'should be written for\n'
                   'that purpose.\n'
                   '\n'
                   'Typical contents for this file would include an overview of '
                   'the project, basic\n'
                   'usage examples, etc. Generally, including the project '
                   'changelog in here is not\n'
                   'a good idea, although a simple "What\'s New" section for the '
                   'most recent version\n'
                   'may be appropriate.',
    'description_content_type': None,
    'docs_url': None,
    'download_url': 'UNKNOWN',
    'downloads': {...},
    'home_page': 'https://github.com/pypa/sampleproject',
    'keywords': 'sample setuptools development',
    'license': 'MIT',
    'maintainer': None,
    'maintainer_email': None,
    'name': 'sampleproject',
    'package_url': 'https://pypi.org/project/sampleproject/',
    'platform': 'UNKNOWN',
    'project_url': 'https://pypi.org/project/sampleproject/',
    'project_urls': {...},
    'release_url': 'https://pypi.org/project/sampleproject/1.2.0/',
    'requires_dist': None,
    'requires_python': None,
    'summary': 'A sample Python project',
    'version': '1.2.0'}

De plus, une valeur maximale de caractères sur une ligne peut être
définie en initialisant le paramètre *width*. Si un long objet ne peut
être scindé, la valeur donnée à *width* sera outrepassée :

   >>> pprint.pp(project_info, depth=1, width=60)
   {'author': 'The Python Packaging Authority',
    'author_email': 'pypa-dev@googlegroups.com',
    'bugtrack_url': None,
    'classifiers': [...],
    'description': 'A sample Python project\n'
                   '=======================\n'
                   '\n'
                   'This is the description file for the '
                   'project.\n'
                   '\n'
                   'The file should use UTF-8 encoding and be '
                   'written using ReStructured Text. It\n'
                   'will be used to generate the project '
                   'webpage on PyPI, and should be written '
                   'for\n'
                   'that purpose.\n'
                   '\n'
                   'Typical contents for this file would '
                   'include an overview of the project, '
                   'basic\n'
                   'usage examples, etc. Generally, including '
                   'the project changelog in here is not\n'
                   'a good idea, although a simple "What\'s '
                   'New" section for the most recent version\n'
                   'may be appropriate.',
    'description_content_type': None,
    'docs_url': None,
    'download_url': 'UNKNOWN',
    'downloads': {...},
    'home_page': 'https://github.com/pypa/sampleproject',
    'keywords': 'sample setuptools development',
    'license': 'MIT',
    'maintainer': None,
    'maintainer_email': None,
    'name': 'sampleproject',
    'package_url': 'https://pypi.org/project/sampleproject/',
    'platform': 'UNKNOWN',
    'project_url': 'https://pypi.org/project/sampleproject/',
    'project_urls': {...},
    'release_url': 'https://pypi.org/project/sampleproject/1.2.0/',
    'requires_dist': None,
    'requires_python': None,
    'summary': 'A sample Python project',
    'version': '1.2.0'}
