Metadata-Version: 2.4
Name: grimp
Version: 3.14
Classifier: Development Status :: 5 - Production/Stable
Classifier: Intended Audience :: Developers
Classifier: License :: OSI Approved :: BSD License
Classifier: Operating System :: Unix
Classifier: Operating System :: POSIX
Classifier: Operating System :: Microsoft :: Windows
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Classifier: Programming Language :: Python :: Implementation :: CPython
Classifier: Programming Language :: Rust
Classifier: Topic :: Utilities
Requires-Dist: typing-extensions>=3.10.0.0
Summary: Builds a queryable graph of the imports within one or more Python packages.
Author-email: David Seddon <david@seddonym.me>
License: BSD 2-Clause License
Requires-Python: >=3.10
Description-Content-Type: text/x-rst; charset=UTF-8
Project-URL: Documentation, https://grimp.readthedocs.io/
Project-URL: Source-code, https://github.com/python-grimp/grimp/

=====
Grimp
=====

.. image:: https://img.shields.io/pypi/v/grimp.svg
    :target: https://pypi.org/project/grimp

.. image:: https://img.shields.io/pypi/pyversions/grimp.svg
    :alt: Python versions
    :target: https://pypi.org/project/grimp/

.. image:: https://github.com/python-grimp/grimp/actions/workflows/main.yml/badge.svg
     :target: https://github.com/python-grimp/grimp/actions?workflow=CI
     :alt: CI Status

.. image:: https://img.shields.io/endpoint?url=https://codspeed.io/badge.json
     :target: https://codspeed.io/python-grimp/grimp?utm_source=badge
     :alt: Codspeed

.. image:: https://img.shields.io/badge/License-BSD_2--Clause-orange.svg
     :target: https://opensource.org/licenses/BSD-2-Clause
     :alt: BSD license

Builds a queryable graph of the imports within one or more Python packages.

Quick start
-----------

Install grimp::

    pip install grimp

Install the Python package you wish to analyse::

    pip install somepackage

In Python, build the import graph for the package::

    >>> import grimp
    >>> graph = grimp.build_graph('somepackage')

You may now use the graph object to analyse the package. Some examples::

    >>> graph.find_children('somepackage.foo')
    {
        'somepackage.foo.one',
        'somepackage.foo.two',
    }

    >>> graph.find_descendants('somepackage.foo')
    {
        'somepackage.foo.one',
        'somepackage.foo.two',
        'somepackage.foo.two.blue',
        'somepackage.foo.two.green',
    }

    >>> graph.find_modules_directly_imported_by('somepackage.foo')
    {
        'somepackage.bar.one',
    }

    >>> graph.find_upstream_modules('somepackage.foo')
    {
        'somepackage.bar.one',
        'somepackage.baz',
        'somepackage.foobar',
    }

    >>> graph.find_shortest_chain(importer='somepackage.foobar', imported='somepackage.foo')
    (
        'somepackage.foobar',
        'somepackage.baz',
        'somepackage.foo',
    )

    >>> graph.get_import_details(importer='somepackage.foobar', imported='somepackage.baz'))
    [
        {
            'importer': 'somepackage.foobar',
            'imported': 'somepackage.baz',
            'line_number': 5,
            'line_contents': 'from . import baz',
        },
    ]


External packages
-----------------

By default, external dependencies will not be included. This can be overridden like so::

    >>> graph = grimp.build_graph('somepackage', include_external_packages=True)
    >>> graph.find_modules_directly_imported_by('somepackage.foo')
    {
        'somepackage.bar.one',
        'os',
        'decimal',
        'sqlalchemy',
    }

Multiple packages
-----------------

You may analyse multiple root packages. To do this, pass each package name as a positional argument::

    >>> graph = grimp.build_graph('somepackage', 'anotherpackage')
    >>> graph.find_modules_directly_imported_by('somepackage.foo')
    {
        'somepackage.bar.one',
        'anotherpackage.baz',
    }

Namespace packages
------------------

Graphs can be built either from `namespace packages`_ or from their `portions`_.

What's a namespace package?
###########################

Namespace packages are a Python feature allows subpackages to be distributed independently, while
still importable under a shared namespace.

This is used by
`the Python client for Google's Cloud Logging API`_, for example. When installed, it is importable
in Python as ``google.cloud.logging``. The parent packages ``google`` and ``google.cloud`` are both namespace
packages, while ``google.cloud.logging`` is known as the 'portion'. Other portions in the same
namespace can be installed separately, for example ``google.cloud.secretmanager``.

Examples::

    # In this one, the portion is supplied. Neither "google" nor "google.cloud"
    # will appear in the graph.
    >>> graph = grimp.build_graph("google.cloud.logging")

    # In this one, a namespace is supplied.
    # Neither "google" nor "google.cloud" will appear in the graph,
    # as will other installed packages under the "google" namespace such
    # as "google.auth".
    >>> graph = grimp.build_graph("google")

    # This one supplies a subnamespace of "google" - it will include
    # "google.cloud.logging" and "google.cloud.secretmanager" but not "google.auth".
    >>> graph = grimp.build_graph("google.cloud")



.. _portions: https://docs.python.org/3/glossary.html#term-portion
.. _namespace packages: https://docs.python.org/3/glossary.html#term-namespace-package
.. _The Python client for Google's Cloud Logging API: https://pypi.org/project/google-cloud-logging/
