Metadata-Version: 2.4
Name: pyppmd-gentee
Version: 1.4.0
Summary: PPMd compression/decompression library with Gentee installer variant decoder
Author-email: Hiroshi Miura <miurahr@linux.com>
Maintainer-email: Joel Puig Rubio <joel.puig.rubio@gmail.com>
License: LGPL-2.1-or-later
Project-URL: Source, https://github.com/puigru/pyppmd-gentee
Project-URL: Upstream, https://github.com/miurahr/pyppmd
Project-URL: Bug Tracker, https://github.com/puigru/pyppmd-gentee/issues
Classifier: Development Status :: 5 - Production/Stable
Classifier: Operating System :: MacOS :: MacOS X
Classifier: Operating System :: Microsoft :: Windows
Classifier: Operating System :: POSIX
Classifier: Operating System :: POSIX :: Linux
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3
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 :: 3 :: Only
Classifier: Programming Language :: Python :: Implementation :: CPython
Classifier: Programming Language :: Python :: Implementation :: PyPy
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Requires-Python: >=3.10
Description-Content-Type: text/x-rst
License-File: LICENSE
Provides-Extra: test
Requires-Dist: pytest>=6.0; extra == "test"
Requires-Dist: pytest-benchmark; extra == "test"
Requires-Dist: pytest-cov; extra == "test"
Requires-Dist: pytest-timeout; extra == "test"
Requires-Dist: hypothesis; extra == "test"
Requires-Dist: coverage[toml]>=5.2; extra == "test"
Provides-Extra: check
Requires-Dist: mypy>=1.10.0; extra == "check"
Requires-Dist: check-manifest; extra == "check"
Requires-Dist: flake8; extra == "check"
Requires-Dist: flake8-isort; extra == "check"
Requires-Dist: flake8-black; extra == "check"
Requires-Dist: readme-renderer; extra == "check"
Requires-Dist: pygments; extra == "check"
Provides-Extra: docs
Requires-Dist: sphinx; extra == "docs"
Requires-Dist: sphinx_rtd_theme; extra == "docs"
Provides-Extra: fuzzer
Requires-Dist: atheris; extra == "fuzzer"
Requires-Dist: hypothesis; extra == "fuzzer"
Dynamic: license-file

PyPPMd-Gentee
=============

A fork of `pyppmd <https://github.com/miurahr/pyppmd>`_ that adds the Gentee PPMd-I
variant decoder (``Ppmd8gDecoder``). Installable alongside the official ``pyppmd`` package.

Install with ``pip install pyppmd-gentee``, then ``import pyppmd_gentee``.


Introduction
------------

``pyppmd_gentee`` provides the same classes and functions as ``pyppmd`` for compressing
and decompressing data using the PPMd algorithm, plus ``Ppmd8gDecoder`` — a variant
of PPMd-I that implements the Gentee installer's modified codec.


Development status
------------------

A project status is considered as ``Stable``.

Gentee PPMd-I variant
---------------------

``Ppmd8gDecoder`` decodes data compressed with the PPMd variant used in Gentee installers.
It differs from the standard PPMd-I codec in BinSumm initialization, frequency updates,
and model restore behaviour.

.. code-block:: python

    import pyppmd_gentee

    dec = pyppmd_gentee.Ppmd8gDecoder(max_order=6, mem_size=16 << 20)
    result = dec.decode(compressed_data, expected_length)

For streaming across multiple files (as in GEA archives), use ``lightweight_reset()``
or ``reinit()`` between entries to manage the decoder state:

.. code-block:: python

    dec = pyppmd_gentee.Ppmd8gDecoder(max_order=6, mem_size=16 << 20)

    first_file = dec.decode(chunk_0, length_0)

    dec.lightweight_reset()  # keeps model, resets context to root
    second_file = dec.decode(chunk_1, length_1)

    dec.reinit(6)  # full model rebuild
    third_file = dec.decode(chunk_2, length_2)

See ``examples/extract_stardef.py`` for a complete example that downloads and extracts
a Gentee-based installer using ``Ppmd8gDecoder``.

Extra input byte
----------------

``PPMd`` algorithm and implementation is designed to use ``Extra`` input byte.
The encoder will omit a last null (b"\0") byte when last byte is b"\0".
You may need to provide an extra null byte when you don't get expected size of
extracted data.

You can do like as:

.. code-block::

    dec = pyppmd.Ppmd7Decoder(max_order=6, mem_size=16 << 10)
    result = dec.decode(compressed, length)
    if len(result) < length:
        if dec.needs_input:
            # ppmd need an extra null byte
            result += dec.decode(b"\0", length - len(result))
        else:
            result += dec.decode(b"", length - len(result))


.. warning::
   When use it on MSYS2/MINGW64 environment, you should set environment variable ``SETUPTOOLS_USE_DISTUTILS=stdlib``

Copyright and License
---------------------

Some codes are derived from p7zip/7zip and pyzstd project.
Details are shown in LicenseNotices.rst

- SPDX-License-Identifier: LGPL-2.1-or-later
- SPDX-URL: https://spdx.org/licenses/LGPL-2.1-or-later.html

PyPPMd is licensed under GNU Lesser General Public License v2.1 or later.

- Copyright (C) 2020-2025 Hiroshi Miura
- Copyright (C) 2020-2021 Ma Lin
- Copyright (C) 2010-2012 Lockless Inc.
- Copyright (C) 1999-2017 Igor Pavlov

This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.

This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Lesser General Public License for more details.

You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
