minijinja


Nameminijinja JSON
Version 1.0.16 PyPI version JSON
download
home_pageNone
SummaryAn experimental Python binding of the Rust MiniJinja template engine.
upload_time2024-03-27 19:53:05
maintainerNone
docs_urlNone
authorNone
requires_python>=3.8
licenseNone
keywords jinja template-engine
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            <div align="center">
  <img src="https://github.com/mitsuhiko/minijinja/raw/main/artwork/logo.png" alt="" width=320>
  <p><strong>MiniJinja for Python: a powerful template engine for Rust and Python</strong></p>

[![Build Status](https://github.com/mitsuhiko/minijinja/workflows/Tests/badge.svg?branch=main)](https://github.com/mitsuhiko/minijinja/actions?query=workflow%3ATests)
[![License](https://img.shields.io/github/license/mitsuhiko/minijinja)](https://github.com/mitsuhiko/minijinja/blob/main/LICENSE)
[![Crates.io](https://img.shields.io/crates/d/minijinja.svg)](https://crates.io/crates/minijinja)
[![rustc 1.61.0](https://img.shields.io/badge/rust-1.61%2B-orange.svg)](https://img.shields.io/badge/rust-1.61%2B-orange.svg)
[![Documentation](https://docs.rs/minijinja/badge.svg)](https://docs.rs/minijinja)

</div>

`minijinja-py` is an experimental binding of
[MiniJinja](https://github.com/mitsuhiko/minijinja) to Python.  It has somewhat
limited functionality compared to the Rust version.  These bindings use
[maturin](https://www.maturin.rs/) and [pyo3](https://pyo3.rs/).

You might want to use MiniJinja instead of Jinja2 when the full feature set
of Jinja2 is not required and you want to have the same rendering experience
of a data set between Rust and Python.

With these bindings MiniJinja can render some Python objects and values
that are passed to templates, but there are clear limitations with regards
to what can be done.

To install MiniJinja for Python you can fetch the package [from PyPI](https://pypi.org/project/minijinja/):

```
$ pip install minijinja
```

## Basic API

The basic API is hidden behind the `Environment` object.  It behaves almost entirely
like in `minijinja` with some Python specific changes.  For instance instead of
`env.set_debug(True)` you use `env.debug = True`.  Additionally instead of using
`add_template` or attaching a `source` you either pass a dictionary of templates
directly to the environment or a `loader` function.

```python
from minijinja import Environment

env = Environment(templates={
    "template_name": "Template source"
})
```

To render a template you can use the `render_template` method:

```python
result = env.render_template('template_name', var1="value 1", var2="value 2")
print(result)
```

## Purpose

MiniJinja attempts a certain level of compatibility with Jinja2, but it does not
try to achieve this at all costs.  As a result you will notice that quite a few
templates will refuse to render with MiniJinja despite the fact that they probably
look quite innocent.  It is however possible to write templates that render to the
same results for both Jinja2 and MiniJinja.  This raises the question why you might
want to use MiniJinja.

The main benefit would be to achieve the exact same results in both Rust and Python.
Additionally MiniJinja has a stronger sandbox than Jinja2 and might perform ever so
slightly better in some situations.  However you should be aware that due to the
marshalling that needs to happen in either direction there is a certain amount of
loss of information.

## Dynamic Template Loading

MiniJinja's Python bindings inherit the underlying behavior of how MiniJinja loads
templates.  Templates are loaded on first use and then cached.  The templates are
loaded via a loader.  To trigger a reload you can call `env.reload()` or
alternatively set `env.reload_before_render` to `True`.

```python
def my_loader(name):
    segments = []
    for segment in name.split("/"):
        if "\\" in segment or segment in (".", ".."):
            return None
        segments.append(segment)
    try:
        with open(os.path.join(TEMPLATES, *segments)) as f:
            return f.read()
    except (IOError, OSError):
        pass

env = Environment(loader=my_loader)
env.reload_before_render = True
print(env.render_template("index.html"))
```

Alternatively templates can manually be loaded and unloaded with `env.add_template`
and `env.remove_template`.

## Auto Escaping

The default behavior is to use auto escaping file files ending in `.html`.  You can
customize this behavior by overriding the `auto_escape_callback`:

```python
env = Environment(auto_escape_callback=lambda x: x.endswith((".html", ".foo")))
```

MiniJinja uses [markupsafe](https://github.com/pallets/markupsafe) if it's available
on the Python side.  It will honor `__html__`.

## Finalizers

Instead of custom formatters like in MiniJinja, you can define a finalizer instead
which is similar to how it works in Jinja2.  It's passed a value (or optional also
the state as first argument when `pass_state` is used) and can return a new value.
If the special `NotImplemented` value is returned, the original value is rendered
without any modification:

```
from minijinja import Environment

def finalizer(value):
    if value is None:
	return ""
    return NotImplemented

env = Environment(finalizer=finalizer)
assert env.render_str("{{ none }}") == ""
```

## State Access

Functions passed to the environment such as filters or global functions can
optionally have the template state passed by using the `pass_state` parameter.
This is similar to `pass_context` in Jinja2.  It can be used to look at the
name of the template or to look up variables in the context.

```python
from minijinja import pass_state

@pass_state
def my_filter(state, value):
    return state.lookup("a_variable") + value

env.add_filter("add_a_variable", my_filter)
```

## Runtime Behavior

MiniJinja uses it's own runtime model which is not matching the Python
runtime model.  As a result there are clear gaps in behavior between the
two and only limited effort is made to bridge them.  For instance you will
be able to call some methods of types, but for instance builtins such as
dicts and lists do not expose their methods on the MiniJinja side.  This
means that it's very intentional that if you pass a dictionary to MiniJinja,
the Python `.items()` method is unavailable.

Here is what this means for some basic types:

* Python dictionaries and lists (as well as other objects that behave as sequences)
  appear in the MiniJinja side as native lists.  They do not expose any specific
  other behavior and when they move back to the Python side they will appear as basic
  lists.  Specifically this means that a tuple (which does not exist in MiniJinja)
  when moving from Python to MiniJinja turns into a list and will remain a list when
  it moves back.
* Python objects are represented in MiniJinja similarly to dicts, but they retain all
  their meaningful Python APIs.  This means they stringify via `__str__` and they
  allow the MiniJinja code to call their non-underscored methods.  Note that there is
  no extra security layer in use at the moment so take care of what you pass there.
* MiniJinja's python binding understand what `__html__` is when it exists on a string
  subclass.  This means that a `markupsafe.Markup` object will appear as safe string in
  MiniJinja.  This information can also flow back to Python again.

## Sponsor

If you like the project and find it useful you can [become a
sponsor](https://github.com/sponsors/mitsuhiko).

## License and Links

- [Documentation](https://docs.rs/minijinja/)
- [Examples](https://github.com/mitsuhiko/minijinja/tree/main/examples)
- [Issue Tracker](https://github.com/mitsuhiko/minijinja/issues)
- [MiniJinja Playground](https://mitsuhiko.github.io/minijinja-playground/)
- License: [Apache-2.0](https://github.com/mitsuhiko/minijinja/blob/main/LICENSE)


            

Raw data

            {
    "_id": null,
    "home_page": null,
    "name": "minijinja",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.8",
    "maintainer_email": "Armin Ronacher <armin.ronacher@active-4.com>",
    "keywords": "jinja, template-engine",
    "author": null,
    "author_email": "Armin Ronacher <armin.ronacher@active-4.com>",
    "download_url": "https://files.pythonhosted.org/packages/b6/9a/bcd2c658f8c81196e1c958be3a7fbcb2ae5f3a77afeddf5522c43839e062/minijinja-1.0.16.tar.gz",
    "platform": null,
    "description": "<div align=\"center\">\n  <img src=\"https://github.com/mitsuhiko/minijinja/raw/main/artwork/logo.png\" alt=\"\" width=320>\n  <p><strong>MiniJinja for Python: a powerful template engine for Rust and Python</strong></p>\n\n[![Build Status](https://github.com/mitsuhiko/minijinja/workflows/Tests/badge.svg?branch=main)](https://github.com/mitsuhiko/minijinja/actions?query=workflow%3ATests)\n[![License](https://img.shields.io/github/license/mitsuhiko/minijinja)](https://github.com/mitsuhiko/minijinja/blob/main/LICENSE)\n[![Crates.io](https://img.shields.io/crates/d/minijinja.svg)](https://crates.io/crates/minijinja)\n[![rustc 1.61.0](https://img.shields.io/badge/rust-1.61%2B-orange.svg)](https://img.shields.io/badge/rust-1.61%2B-orange.svg)\n[![Documentation](https://docs.rs/minijinja/badge.svg)](https://docs.rs/minijinja)\n\n</div>\n\n`minijinja-py` is an experimental binding of\n[MiniJinja](https://github.com/mitsuhiko/minijinja) to Python.  It has somewhat\nlimited functionality compared to the Rust version.  These bindings use\n[maturin](https://www.maturin.rs/) and [pyo3](https://pyo3.rs/).\n\nYou might want to use MiniJinja instead of Jinja2 when the full feature set\nof Jinja2 is not required and you want to have the same rendering experience\nof a data set between Rust and Python.\n\nWith these bindings MiniJinja can render some Python objects and values\nthat are passed to templates, but there are clear limitations with regards\nto what can be done.\n\nTo install MiniJinja for Python you can fetch the package [from PyPI](https://pypi.org/project/minijinja/):\n\n```\n$ pip install minijinja\n```\n\n## Basic API\n\nThe basic API is hidden behind the `Environment` object.  It behaves almost entirely\nlike in `minijinja` with some Python specific changes.  For instance instead of\n`env.set_debug(True)` you use `env.debug = True`.  Additionally instead of using\n`add_template` or attaching a `source` you either pass a dictionary of templates\ndirectly to the environment or a `loader` function.\n\n```python\nfrom minijinja import Environment\n\nenv = Environment(templates={\n    \"template_name\": \"Template source\"\n})\n```\n\nTo render a template you can use the `render_template` method:\n\n```python\nresult = env.render_template('template_name', var1=\"value 1\", var2=\"value 2\")\nprint(result)\n```\n\n## Purpose\n\nMiniJinja attempts a certain level of compatibility with Jinja2, but it does not\ntry to achieve this at all costs.  As a result you will notice that quite a few\ntemplates will refuse to render with MiniJinja despite the fact that they probably\nlook quite innocent.  It is however possible to write templates that render to the\nsame results for both Jinja2 and MiniJinja.  This raises the question why you might\nwant to use MiniJinja.\n\nThe main benefit would be to achieve the exact same results in both Rust and Python.\nAdditionally MiniJinja has a stronger sandbox than Jinja2 and might perform ever so\nslightly better in some situations.  However you should be aware that due to the\nmarshalling that needs to happen in either direction there is a certain amount of\nloss of information.\n\n## Dynamic Template Loading\n\nMiniJinja's Python bindings inherit the underlying behavior of how MiniJinja loads\ntemplates.  Templates are loaded on first use and then cached.  The templates are\nloaded via a loader.  To trigger a reload you can call `env.reload()` or\nalternatively set `env.reload_before_render` to `True`.\n\n```python\ndef my_loader(name):\n    segments = []\n    for segment in name.split(\"/\"):\n        if \"\\\\\" in segment or segment in (\".\", \"..\"):\n            return None\n        segments.append(segment)\n    try:\n        with open(os.path.join(TEMPLATES, *segments)) as f:\n            return f.read()\n    except (IOError, OSError):\n        pass\n\nenv = Environment(loader=my_loader)\nenv.reload_before_render = True\nprint(env.render_template(\"index.html\"))\n```\n\nAlternatively templates can manually be loaded and unloaded with `env.add_template`\nand `env.remove_template`.\n\n## Auto Escaping\n\nThe default behavior is to use auto escaping file files ending in `.html`.  You can\ncustomize this behavior by overriding the `auto_escape_callback`:\n\n```python\nenv = Environment(auto_escape_callback=lambda x: x.endswith((\".html\", \".foo\")))\n```\n\nMiniJinja uses [markupsafe](https://github.com/pallets/markupsafe) if it's available\non the Python side.  It will honor `__html__`.\n\n## Finalizers\n\nInstead of custom formatters like in MiniJinja, you can define a finalizer instead\nwhich is similar to how it works in Jinja2.  It's passed a value (or optional also\nthe state as first argument when `pass_state` is used) and can return a new value.\nIf the special `NotImplemented` value is returned, the original value is rendered\nwithout any modification:\n\n```\nfrom minijinja import Environment\n\ndef finalizer(value):\n    if value is None:\n\treturn \"\"\n    return NotImplemented\n\nenv = Environment(finalizer=finalizer)\nassert env.render_str(\"{{ none }}\") == \"\"\n```\n\n## State Access\n\nFunctions passed to the environment such as filters or global functions can\noptionally have the template state passed by using the `pass_state` parameter.\nThis is similar to `pass_context` in Jinja2.  It can be used to look at the\nname of the template or to look up variables in the context.\n\n```python\nfrom minijinja import pass_state\n\n@pass_state\ndef my_filter(state, value):\n    return state.lookup(\"a_variable\") + value\n\nenv.add_filter(\"add_a_variable\", my_filter)\n```\n\n## Runtime Behavior\n\nMiniJinja uses it's own runtime model which is not matching the Python\nruntime model.  As a result there are clear gaps in behavior between the\ntwo and only limited effort is made to bridge them.  For instance you will\nbe able to call some methods of types, but for instance builtins such as\ndicts and lists do not expose their methods on the MiniJinja side.  This\nmeans that it's very intentional that if you pass a dictionary to MiniJinja,\nthe Python `.items()` method is unavailable.\n\nHere is what this means for some basic types:\n\n* Python dictionaries and lists (as well as other objects that behave as sequences)\n  appear in the MiniJinja side as native lists.  They do not expose any specific\n  other behavior and when they move back to the Python side they will appear as basic\n  lists.  Specifically this means that a tuple (which does not exist in MiniJinja)\n  when moving from Python to MiniJinja turns into a list and will remain a list when\n  it moves back.\n* Python objects are represented in MiniJinja similarly to dicts, but they retain all\n  their meaningful Python APIs.  This means they stringify via `__str__` and they\n  allow the MiniJinja code to call their non-underscored methods.  Note that there is\n  no extra security layer in use at the moment so take care of what you pass there.\n* MiniJinja's python binding understand what `__html__` is when it exists on a string\n  subclass.  This means that a `markupsafe.Markup` object will appear as safe string in\n  MiniJinja.  This information can also flow back to Python again.\n\n## Sponsor\n\nIf you like the project and find it useful you can [become a\nsponsor](https://github.com/sponsors/mitsuhiko).\n\n## License and Links\n\n- [Documentation](https://docs.rs/minijinja/)\n- [Examples](https://github.com/mitsuhiko/minijinja/tree/main/examples)\n- [Issue Tracker](https://github.com/mitsuhiko/minijinja/issues)\n- [MiniJinja Playground](https://mitsuhiko.github.io/minijinja-playground/)\n- License: [Apache-2.0](https://github.com/mitsuhiko/minijinja/blob/main/LICENSE)\n\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "An experimental Python binding of the Rust MiniJinja template engine.",
    "version": "1.0.16",
    "project_urls": {
        "Donate": "https://github.com/sponsors/mitsuhiko",
        "Issue Tracker": "https://github.com/mitsuhiko/minijinja/issues",
        "Repository": "https://github.com/mitsuhiko/minijinja"
    },
    "split_keywords": [
        "jinja",
        " template-engine"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "ce660e98b4d2903a4591e168b973ade3df7450566872b72bb13209f3d698455f",
                "md5": "5a68e720bc388b1a17541a37bb9a3ac4",
                "sha256": "b7c1640d23ad719da999ab434fd58e895125e067d310d512c9a45ec7e7f301dc"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl",
            "has_sig": false,
            "md5_digest": "5a68e720bc388b1a17541a37bb9a3ac4",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 1496413,
            "upload_time": "2024-03-27T19:52:45",
            "upload_time_iso_8601": "2024-03-27T19:52:45.312103Z",
            "url": "https://files.pythonhosted.org/packages/ce/66/0e98b4d2903a4591e168b973ade3df7450566872b72bb13209f3d698455f/minijinja-1.0.16-cp38-abi3-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "a07bce0c38b4b3e4ff5efc197632ee040e7427b5ea9d57c2336d035453b7a414",
                "md5": "15667a3e4d94a5b56abea386fbb1b41a",
                "sha256": "f83ca4a37d73b57b87cee4d50f7ce2ade2d6c04fbede49faabc37f7244161bd6"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl",
            "has_sig": false,
            "md5_digest": "15667a3e4d94a5b56abea386fbb1b41a",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 771827,
            "upload_time": "2024-03-27T19:52:48",
            "upload_time_iso_8601": "2024-03-27T19:52:48.766563Z",
            "url": "https://files.pythonhosted.org/packages/a0/7b/ce0c38b4b3e4ff5efc197632ee040e7427b5ea9d57c2336d035453b7a414/minijinja-1.0.16-cp38-abi3-manylinux_2_17_aarch64.manylinux2014_aarch64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "4a2d27af85b3da81085b35ef02f927db6b6729e08772d8b7baedb63a3569a160",
                "md5": "9cc50f01e0371e460ed0b60fd7a3f36d",
                "sha256": "87c61c94c014c3a6d2ef83ee2f67229ee23b13b3586c8d4d3a7d090ba19f1b6c"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-manylinux_2_17_armv7l.manylinux2014_armv7l.whl",
            "has_sig": false,
            "md5_digest": "9cc50f01e0371e460ed0b60fd7a3f36d",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 761945,
            "upload_time": "2024-03-27T19:52:52",
            "upload_time_iso_8601": "2024-03-27T19:52:52.366803Z",
            "url": "https://files.pythonhosted.org/packages/4a/2d/27af85b3da81085b35ef02f927db6b6729e08772d8b7baedb63a3569a160/minijinja-1.0.16-cp38-abi3-manylinux_2_17_armv7l.manylinux2014_armv7l.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "347351050688e7b1a4446e174a0bf3430dfc0f9b88dd9bc969c37c37f5e60c4f",
                "md5": "4d395e52c82862b3bf8cb2ca5df9061d",
                "sha256": "beeb7fd7552e351b071b1d01fa80600a332b364c83f0f4a5c96b96d74d162434"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "has_sig": false,
            "md5_digest": "4d395e52c82862b3bf8cb2ca5df9061d",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 804589,
            "upload_time": "2024-03-27T19:52:55",
            "upload_time_iso_8601": "2024-03-27T19:52:55.318558Z",
            "url": "https://files.pythonhosted.org/packages/34/73/51050688e7b1a4446e174a0bf3430dfc0f9b88dd9bc969c37c37f5e60c4f/minijinja-1.0.16-cp38-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "d5bb2fbe18975ecab6663128f9a45db89b3185319532d5f3c35eb1fd52916a7c",
                "md5": "a05418eaa5f7c85dde1e07903da2fa9f",
                "sha256": "38cc378dd256bd3766f1c83754c9a4407356b6d992defc725eab8a7c203e07b9"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-manylinux_2_5_i686.manylinux1_i686.whl",
            "has_sig": false,
            "md5_digest": "a05418eaa5f7c85dde1e07903da2fa9f",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 879626,
            "upload_time": "2024-03-27T19:52:58",
            "upload_time_iso_8601": "2024-03-27T19:52:58.409886Z",
            "url": "https://files.pythonhosted.org/packages/d5/bb/2fbe18975ecab6663128f9a45db89b3185319532d5f3c35eb1fd52916a7c/minijinja-1.0.16-cp38-abi3-manylinux_2_5_i686.manylinux1_i686.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "4804dc4295268dd05499cef2307f9b9b2d5ab4effbe9c3236ad8c3abda1a21ca",
                "md5": "db813cf6790c0c1985356456628cbb3b",
                "sha256": "f7f0da6c0d2b78bce9aa0f2bbfebc0ece18cdf8dae9430b4001055b8eb77911c"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-win32.whl",
            "has_sig": false,
            "md5_digest": "db813cf6790c0c1985356456628cbb3b",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 670877,
            "upload_time": "2024-03-27T19:53:01",
            "upload_time_iso_8601": "2024-03-27T19:53:01.175765Z",
            "url": "https://files.pythonhosted.org/packages/48/04/dc4295268dd05499cef2307f9b9b2d5ab4effbe9c3236ad8c3abda1a21ca/minijinja-1.0.16-cp38-abi3-win32.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8a43e044bf6fa4c6f1e6e514ca25d48d8942f78d567f3cea7ea78fc36564388c",
                "md5": "06bf9ece364f4715d55aff76853cb826",
                "sha256": "0a41541c28fd7ce64b38ddc60974930f81c163440c348393bd5143debeb309b3"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16-cp38-abi3-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "06bf9ece364f4715d55aff76853cb826",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.8",
            "size": 697217,
            "upload_time": "2024-03-27T19:53:03",
            "upload_time_iso_8601": "2024-03-27T19:53:03.511576Z",
            "url": "https://files.pythonhosted.org/packages/8a/43/e044bf6fa4c6f1e6e514ca25d48d8942f78d567f3cea7ea78fc36564388c/minijinja-1.0.16-cp38-abi3-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "b69abcd2c658f8c81196e1c958be3a7fbcb2ae5f3a77afeddf5522c43839e062",
                "md5": "179b650b241212fe052b94f9b6830e63",
                "sha256": "57f6e98bc735794eb87648f251e68fbfd9cf50333f6a4053b398751483350826"
            },
            "downloads": -1,
            "filename": "minijinja-1.0.16.tar.gz",
            "has_sig": false,
            "md5_digest": "179b650b241212fe052b94f9b6830e63",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.8",
            "size": 190254,
            "upload_time": "2024-03-27T19:53:05",
            "upload_time_iso_8601": "2024-03-27T19:53:05.912733Z",
            "url": "https://files.pythonhosted.org/packages/b6/9a/bcd2c658f8c81196e1c958be3a7fbcb2ae5f3a77afeddf5522c43839e062/minijinja-1.0.16.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-03-27 19:53:05",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "sponsors",
    "github_project": "mitsuhiko",
    "github_not_found": true,
    "lcname": "minijinja"
}
        
Elapsed time: 0.21371s