tackle


Nametackle JSON
Version 0.5.1 PyPI version JSON
download
home_pagehttps://github.com/robcxyz/tackle
SummaryTackle is a declarative DSL for building modular workflows and code generators. Tool is plugins based and can easily be extended by writing additional hooks or importing external providers that can be turned into a self documenting CLI, all out of yaml, json, toml.
upload_time2023-07-17 13:44:36
maintainer
docs_urlNone
authorRob Cannon
requires_python>=3.7
licenseBSD
keywords cookiecutter tackle tackle-box tacklebox tackle box python projects project templates jinja2 skeleton scaffolding project directory package packaging kubernetes
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            <img align="right" width="280" height="280" src="https://raw.githubusercontent.com/sudoblockio/tackle/main/docs/assets/logo-box.png">

# tackle

[![pypi](https://img.shields.io/pypi/v/tackle.svg)](https://pypi.python.org/pypi/tackle)
[![python](https://img.shields.io/pypi/pyversions/tackle.svg)](https://pypi.python.org/pypi/tackle)
[![codecov](https://codecov.io/gh/sudoblockio/tackle/branch/main/graphs/badge.svg?branch=main)](https://codecov.io/github/sudoblockio/tackle?branch=main)
[![codeql](https://github.com/sudoblockio/tackle/actions/workflows/codeql.yml/badge.svg)](https://github.com/sudoblockio/tackle/actions/workflows/codeql.yml)

[//]: # ([![main-tests]&#40;https://github.com/sudoblockio/tackle/actions/workflows/main.yml/badge.svg&#41;]&#40;https://github.com/sudoblockio/tackle/actions&#41;)

* [Documentation](https://sudoblockio.github.io/tackle)
* [Discord](https://discord.gg/7uVUfUVD7K)
* [PyPI](https://pypi.org/project/tackle/)
* [BSD License](LICENSE)

[//]: # (* [Slack]&#40;https://join.slack.com/t/slack-y748219/shared_invite/zt-1cqreswyd-5qDBE53QlY97mQOI6DhcKw&#41;)

Tackle is an experimental general purpose configuration language for building modular code generators and declarative CLIs. Built as a fork of [cookiecutter](https://github.com/cookiecutter/cookiecutter), it can make any config file into a CLI with both strong and weakly typed programmable flow control common to a general purpose programming language. Basically you can write a fully functional Turing-complete program out of a config file. It's wild.

**With tackle, you can build:**
- Modular code generators / repo scaffolding tools that can be updated over time
- Interactive glue code for infrastructure-as-code deployment strategies
- Generic utilities like SSH helpers and dotfile managers
- Combinations of all of the above and anything else you can think of

[//]: # (- Declarative makefile alternatives for advanced toolchain management)

> If this project gets enough adoption / stars, it will be re-written in a compiled language, either Go or Rust. Give it a star if you'd like to see that.

### Features

- Makes arbitrary yaml / json / toml [dynamic](https://sudoblockio.github.io/tackle/hook-methods/)
  - Embed loops, conditionals, and other custom logic
  - Self documenting CLI to call logic
- Ships with a collection of over [100 hooks](https://sudoblockio.github.io/tackle) that act like plugins within your config file
  - [Prompt user for inputs](https://sudoblockio.github.io/tackle)
  - [Generate code from templates](https://sudoblockio.github.io/tackle/providers/Generate/)
  - Read and write [yaml](https://sudoblockio.github.io/tackle/providers/Yaml/) / [toml](https://sudoblockio.github.io/tackle/providers/Toml/) / [json](https://sudoblockio.github.io/tackle/providers/Json/) [files](https://sudoblockio.github.io/tackle/providers/Files/)
  - [Make http calls](https://sudoblockio.github.io/tackle/providers/Web/)
  - [Run arbitrary system commands](https://sudoblockio.github.io/tackle/providers/Command/)
  - [Manipulate the context](https://sudoblockio.github.io/tackle/providers/Context/)
  - [Run other tackle files](https://sudoblockio.github.io/tackle/providers/Tackle/tackle/)
- Modular design allows creating / importing new hooks easy
  - Supports both [python](https://sudoblockio.github.io/tackle/python-hooks/) and [declarative](https://sudoblockio.github.io/tackle/declarative-hooks/) hooks which can be imported / called / defined in-line or within jinja templates
  - Hooks can be composed of other hooks allowing complex objects to be validated and operated against

### Install

> Note: tackle can install dependencies on its own. Check [docs](https://sudoblockio.github.io/tackle/installation#best-installation-method) for advanced installation methods to isolate tackle from your system python.

```shell
python -m venv env && source env/bin/activate
pip install tackle
```

**Quick Demo:** `tackle sudoblockio/tackle-hello-world`

### Hello world

Check out the [docs](https://sudoblockio.github.io/tackle/hello-worlds/) for >10 hello worlds that demonstrate the various aspects of the syntax with the simplest one using the [print](https://sudoblockio.github.io/tackle/providers/Console/print/) hook, one of [>100 hooks](https://sudoblockio.github.io/tackle/providers/Collections/).

**hello.yaml**
```yaml
# Any time a key ends with `->`, we are calling a hook
hw->: print Hello world!
```

To run, call `tackle hello.yaml`. Can also be [version controlled](https://sudoblockio.github.io/tackle/creating-providers/) -> [`tackle sudoblockio/tackle-hello-world`](https://github.com/sudoblockio/tackle-hello-world).

Can also use [loops, conditionals, and other base methods](https://sudoblockio.github.io/tackle/hook-methods/).

**hello.yaml**
```yaml
the:
  words:
    - Hello
    - cruel
    - world!
one liner->: print {{item}} --for the.words --if "item != 'cruel'"
multiple lines:
  ->: print
  objects: {{item}}
  for:
    - Hello
    - world!
  if: item != 'cruel'
# Or combinations of the above with other methods like try/except
```

New hooks can be [made in python](https://sudoblockio.github.io/tackle/python-hooks/) which under the hood is a [pydantic](https://github.com/pydantic/pydantic) model.

**.hooks/hello.py**
```python
from tackle import BaseHook

class Greeter(BaseHook):
    hook_type: str = "greeter"
    target: str
    args: list = ['target']
    def exec(self):
      expression = f"Hello {self.target}"
      print(expression)
      return expression
```

Or can be [defined inline within your tackle file, imported remotely, or in a `hooks` directory.](https://sudoblockio.github.io/tackle/declarative-hooks/).

**.hooks/hello.yaml**
```yaml
# Keys ending with `<-` mean we are creating a hook / method
greeter<-:
  target: str
  args: ['target']
  exec<-:
    expression->: var Hello {{target}}  # var hook renders variables
    p->: print {{expression}}
  return: expression
```

And both can be [called the same way](https://sudoblockio.github.io/tackle/writing-tackle-files/).

**tackle.yaml**
```yaml
hello: world!
With a flag->: greeter --target {{hello}}
Target in argument->: greeter {{hello}}
Expanded fields:
  ->: greeter
  target: {{hello}}
Jinja template->: {{ greeter(hello) }}
# Or combinations jinja and compact / expanded hooks allowing chaining of hook calls.
```

With the declarative hooks being [callable from the command line](https://sudoblockio.github.io/tackle/declarative-cli/):

```shell
tackle hello.yaml greeter --target world!
# Or from a github repo
tackle sudoblockio/tackle-hello-world greeter --target world!
```

Documentation can be embedded into the hooks.

**hello.yaml**
```yaml
<-:
  help: This is the default hook
  target:
    type: union[str, int]
    default->: input
    description: The thing to say hello to
  exec<-:
    greeting->: select Who to greet? --choices ['world',target]
    hi->: greeter --target {{greeting}}
  greeting-method<-:
    help: A method that greets
    # ... Greeting options / logic
    extends: greeter
greeter<-:
  help: A reusable greeter object
  target: str
  exec<-:
    hi->: print Hello {{target}}
```

Which when running `tackle hello.yaml help` produces its own help screen.

```text
usage: tackle hello.yaml [--target]

This is the default hook

options:
    --target [str] The thing to say hello to
methods:
    greeting-method     A method that greets
    greeter     A reusable greeter object
```

Hooks can be imported [within a tackle provider](https://sudoblockio.github.io/tackle/declarative-cli/#importing-hooks) or [through hooks](https://sudoblockio.github.io/tackle/providers/Tackle/import/), [linked](https://sudoblockio.github.io/tackle/providers/Tackle/tackle/), and/or combined with [inheritance](https://sudoblockio.github.io/tackle/declarative-hooks/#extending-hooks) or [composition](https://sudoblockio.github.io/tackle/declarative-hooks/#extending-hooks) creating a web of CLIs.

### Use Cases

- [Code Generation](https://sudoblockio.github.io/tackle/tutorials/code-generation/)

**WIP Tutorials**

- Declarative Utilities
- Infrastructure-as-Code Management
- Kubernetes Manifest Management
- Toolchain Management

[//]: # (- [Repo Management]&#40;&#41; - wip)

### Topics

- [Writing Tackle Files](https://sudoblockio.github.io/tackle/writing-tackle-files/)
- [Creating Providers](https://sudoblockio.github.io/tackle/creating-providers/)
- [Python Hooks](https://sudoblockio.github.io/tackle/python-hooks/)
- [Declarative Hooks](https://sudoblockio.github.io/tackle/declarative-hooks/)
- [Blocks](https://sudoblockio.github.io/tackle/writing-tackle-files/#blocks) and [Flow Control](https://sudoblockio.github.io/tackle/hook-methods/)
- [Memory Management](https://sudoblockio.github.io/tackle/memory-management/)
- [Special Variables](https://sudoblockio.github.io/tackle/special-variables/)
- [Declarative CLIs](https://sudoblockio.github.io/tackle/declarative-cli/)

### Known Issues

- **Windows Support**
  - tackle is lacking some windows support as shown in the [failed tests](https://github.com/sudoblockio/tackle/actions/workflows/main-windows.yml). If you are a windows user, it is highly recommended to use WSL. **Please get in touch** if you are motivated to fix these tests to make tackle fully cross-platform. It probably isn't that hard to fix them as they mostly are due to differences in how windows handles paths.
- **Whitespaces**
  - tackle relies heavily on parsing based on whitespaces which if you are not careful can easily bite you. Whenever you need to have some whitespaces preserved, make sure to quote the entire expression. Future work will be put in to overhaul the [regex based parser](https://github.com/sudoblockio/tackle/blob/main/tackle/utils/command.py#L52) with a PEG parser like [parsimonious](https://github.com/erikrose/parsimonious).

### Contributing

Contributions are welcome but please be advised of the following notes.

- This project uses [conventional commits](https://www.conventionalcommits.org/) which generates the [changelog](./CHANGELOG.md) with [release-please-action](https://github.com/google-github-actions/release-please-action) in the [release](https://github.com/sudoblockio/tackle/blob/main/.github/workflows/release.yml) CI workflow. If commits have been made outside of this convention they will be squashed accordingly.
- For making changes to providers, please include test coverage using the existing fixtures and patterns from prior tests or communicate any suggestions that deviate from this style. It definitely can be improved but consistency is more important than making directed improvements. Tests should be runnable from the test's directory and via `make test`.
- For making changes to the core parser, please create a proposal first outlining your suggestions with examples before spending time working on code.

It is very easy to create new providers / hooks with tackle. Over time, it will adopt the same import pattern of what Ansible does where all provider / hooks (modules) are stored in version controlled locations. In the meantime, please feel free to contribute to this repository for hooks that have general applicability or create your own hooks in other repositories that are more bespoke / opinionated in nature.

### Code of Conduct

Everyone interacting in the tackle project's codebases, issue trackers, chat rooms, and mailing lists is expected to follow the [PyPA Code of Conduct](https://www.pypa.io/en/latest/code-of-conduct/).

## Credit

Special thanks to the [cookiecutter](https://github.com/cookiecutter/cookiecutter) community for laying the basis for this project.



            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/robcxyz/tackle",
    "name": "tackle",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": "",
    "keywords": "cookiecutter,tackle,tackle-box,tacklebox,tackle box,Python,projects,project templates,Jinja2,skeleton,scaffolding,project directory,package,packaging,kubernetes",
    "author": "Rob Cannon",
    "author_email": "robc.io.opensource@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/cb/e3/fe2c90b7162ab34bc6afecbacaefe3f313b5d875529170814e109dda9018/tackle-0.5.1.tar.gz",
    "platform": null,
    "description": "<img align=\"right\" width=\"280\" height=\"280\" src=\"https://raw.githubusercontent.com/sudoblockio/tackle/main/docs/assets/logo-box.png\">\n\n# tackle\n\n[![pypi](https://img.shields.io/pypi/v/tackle.svg)](https://pypi.python.org/pypi/tackle)\n[![python](https://img.shields.io/pypi/pyversions/tackle.svg)](https://pypi.python.org/pypi/tackle)\n[![codecov](https://codecov.io/gh/sudoblockio/tackle/branch/main/graphs/badge.svg?branch=main)](https://codecov.io/github/sudoblockio/tackle?branch=main)\n[![codeql](https://github.com/sudoblockio/tackle/actions/workflows/codeql.yml/badge.svg)](https://github.com/sudoblockio/tackle/actions/workflows/codeql.yml)\n\n[//]: # ([![main-tests]&#40;https://github.com/sudoblockio/tackle/actions/workflows/main.yml/badge.svg&#41;]&#40;https://github.com/sudoblockio/tackle/actions&#41;)\n\n* [Documentation](https://sudoblockio.github.io/tackle)\n* [Discord](https://discord.gg/7uVUfUVD7K)\n* [PyPI](https://pypi.org/project/tackle/)\n* [BSD License](LICENSE)\n\n[//]: # (* [Slack]&#40;https://join.slack.com/t/slack-y748219/shared_invite/zt-1cqreswyd-5qDBE53QlY97mQOI6DhcKw&#41;)\n\nTackle is an experimental general purpose configuration language for building modular code generators and declarative CLIs. Built as a fork of [cookiecutter](https://github.com/cookiecutter/cookiecutter), it can make any config file into a CLI with both strong and weakly typed programmable flow control common to a general purpose programming language. Basically you can write a fully functional Turing-complete program out of a config file. It's wild.\n\n**With tackle, you can build:**\n- Modular code generators / repo scaffolding tools that can be updated over time\n- Interactive glue code for infrastructure-as-code deployment strategies\n- Generic utilities like SSH helpers and dotfile managers\n- Combinations of all of the above and anything else you can think of\n\n[//]: # (- Declarative makefile alternatives for advanced toolchain management)\n\n> If this project gets enough adoption / stars, it will be re-written in a compiled language, either Go or Rust. Give it a star if you'd like to see that.\n\n### Features\n\n- Makes arbitrary yaml / json / toml [dynamic](https://sudoblockio.github.io/tackle/hook-methods/)\n  - Embed loops, conditionals, and other custom logic\n  - Self documenting CLI to call logic\n- Ships with a collection of over [100 hooks](https://sudoblockio.github.io/tackle) that act like plugins within your config file\n  - [Prompt user for inputs](https://sudoblockio.github.io/tackle)\n  - [Generate code from templates](https://sudoblockio.github.io/tackle/providers/Generate/)\n  - Read and write [yaml](https://sudoblockio.github.io/tackle/providers/Yaml/) / [toml](https://sudoblockio.github.io/tackle/providers/Toml/) / [json](https://sudoblockio.github.io/tackle/providers/Json/) [files](https://sudoblockio.github.io/tackle/providers/Files/)\n  - [Make http calls](https://sudoblockio.github.io/tackle/providers/Web/)\n  - [Run arbitrary system commands](https://sudoblockio.github.io/tackle/providers/Command/)\n  - [Manipulate the context](https://sudoblockio.github.io/tackle/providers/Context/)\n  - [Run other tackle files](https://sudoblockio.github.io/tackle/providers/Tackle/tackle/)\n- Modular design allows creating / importing new hooks easy\n  - Supports both [python](https://sudoblockio.github.io/tackle/python-hooks/) and [declarative](https://sudoblockio.github.io/tackle/declarative-hooks/) hooks which can be imported / called / defined in-line or within jinja templates\n  - Hooks can be composed of other hooks allowing complex objects to be validated and operated against\n\n### Install\n\n> Note: tackle can install dependencies on its own. Check [docs](https://sudoblockio.github.io/tackle/installation#best-installation-method) for advanced installation methods to isolate tackle from your system python.\n\n```shell\npython -m venv env && source env/bin/activate\npip install tackle\n```\n\n**Quick Demo:** `tackle sudoblockio/tackle-hello-world`\n\n### Hello world\n\nCheck out the [docs](https://sudoblockio.github.io/tackle/hello-worlds/) for >10 hello worlds that demonstrate the various aspects of the syntax with the simplest one using the [print](https://sudoblockio.github.io/tackle/providers/Console/print/) hook, one of [>100 hooks](https://sudoblockio.github.io/tackle/providers/Collections/).\n\n**hello.yaml**\n```yaml\n# Any time a key ends with `->`, we are calling a hook\nhw->: print Hello world!\n```\n\nTo run, call `tackle hello.yaml`. Can also be [version controlled](https://sudoblockio.github.io/tackle/creating-providers/) -> [`tackle sudoblockio/tackle-hello-world`](https://github.com/sudoblockio/tackle-hello-world).\n\nCan also use [loops, conditionals, and other base methods](https://sudoblockio.github.io/tackle/hook-methods/).\n\n**hello.yaml**\n```yaml\nthe:\n  words:\n    - Hello\n    - cruel\n    - world!\none liner->: print {{item}} --for the.words --if \"item != 'cruel'\"\nmultiple lines:\n  ->: print\n  objects: {{item}}\n  for:\n    - Hello\n    - world!\n  if: item != 'cruel'\n# Or combinations of the above with other methods like try/except\n```\n\nNew hooks can be [made in python](https://sudoblockio.github.io/tackle/python-hooks/) which under the hood is a [pydantic](https://github.com/pydantic/pydantic) model.\n\n**.hooks/hello.py**\n```python\nfrom tackle import BaseHook\n\nclass Greeter(BaseHook):\n    hook_type: str = \"greeter\"\n    target: str\n    args: list = ['target']\n    def exec(self):\n      expression = f\"Hello {self.target}\"\n      print(expression)\n      return expression\n```\n\nOr can be [defined inline within your tackle file, imported remotely, or in a `hooks` directory.](https://sudoblockio.github.io/tackle/declarative-hooks/).\n\n**.hooks/hello.yaml**\n```yaml\n# Keys ending with `<-` mean we are creating a hook / method\ngreeter<-:\n  target: str\n  args: ['target']\n  exec<-:\n    expression->: var Hello {{target}}  # var hook renders variables\n    p->: print {{expression}}\n  return: expression\n```\n\nAnd both can be [called the same way](https://sudoblockio.github.io/tackle/writing-tackle-files/).\n\n**tackle.yaml**\n```yaml\nhello: world!\nWith a flag->: greeter --target {{hello}}\nTarget in argument->: greeter {{hello}}\nExpanded fields:\n  ->: greeter\n  target: {{hello}}\nJinja template->: {{ greeter(hello) }}\n# Or combinations jinja and compact / expanded hooks allowing chaining of hook calls.\n```\n\nWith the declarative hooks being [callable from the command line](https://sudoblockio.github.io/tackle/declarative-cli/):\n\n```shell\ntackle hello.yaml greeter --target world!\n# Or from a github repo\ntackle sudoblockio/tackle-hello-world greeter --target world!\n```\n\nDocumentation can be embedded into the hooks.\n\n**hello.yaml**\n```yaml\n<-:\n  help: This is the default hook\n  target:\n    type: union[str, int]\n    default->: input\n    description: The thing to say hello to\n  exec<-:\n    greeting->: select Who to greet? --choices ['world',target]\n    hi->: greeter --target {{greeting}}\n  greeting-method<-:\n    help: A method that greets\n    # ... Greeting options / logic\n    extends: greeter\ngreeter<-:\n  help: A reusable greeter object\n  target: str\n  exec<-:\n    hi->: print Hello {{target}}\n```\n\nWhich when running `tackle hello.yaml help` produces its own help screen.\n\n```text\nusage: tackle hello.yaml [--target]\n\nThis is the default hook\n\noptions:\n    --target [str] The thing to say hello to\nmethods:\n    greeting-method     A method that greets\n    greeter     A reusable greeter object\n```\n\nHooks can be imported [within a tackle provider](https://sudoblockio.github.io/tackle/declarative-cli/#importing-hooks) or [through hooks](https://sudoblockio.github.io/tackle/providers/Tackle/import/), [linked](https://sudoblockio.github.io/tackle/providers/Tackle/tackle/), and/or combined with [inheritance](https://sudoblockio.github.io/tackle/declarative-hooks/#extending-hooks) or [composition](https://sudoblockio.github.io/tackle/declarative-hooks/#extending-hooks) creating a web of CLIs.\n\n### Use Cases\n\n- [Code Generation](https://sudoblockio.github.io/tackle/tutorials/code-generation/)\n\n**WIP Tutorials**\n\n- Declarative Utilities\n- Infrastructure-as-Code Management\n- Kubernetes Manifest Management\n- Toolchain Management\n\n[//]: # (- [Repo Management]&#40;&#41; - wip)\n\n### Topics\n\n- [Writing Tackle Files](https://sudoblockio.github.io/tackle/writing-tackle-files/)\n- [Creating Providers](https://sudoblockio.github.io/tackle/creating-providers/)\n- [Python Hooks](https://sudoblockio.github.io/tackle/python-hooks/)\n- [Declarative Hooks](https://sudoblockio.github.io/tackle/declarative-hooks/)\n- [Blocks](https://sudoblockio.github.io/tackle/writing-tackle-files/#blocks) and [Flow Control](https://sudoblockio.github.io/tackle/hook-methods/)\n- [Memory Management](https://sudoblockio.github.io/tackle/memory-management/)\n- [Special Variables](https://sudoblockio.github.io/tackle/special-variables/)\n- [Declarative CLIs](https://sudoblockio.github.io/tackle/declarative-cli/)\n\n### Known Issues\n\n- **Windows Support**\n  - tackle is lacking some windows support as shown in the [failed tests](https://github.com/sudoblockio/tackle/actions/workflows/main-windows.yml). If you are a windows user, it is highly recommended to use WSL. **Please get in touch** if you are motivated to fix these tests to make tackle fully cross-platform. It probably isn't that hard to fix them as they mostly are due to differences in how windows handles paths.\n- **Whitespaces**\n  - tackle relies heavily on parsing based on whitespaces which if you are not careful can easily bite you. Whenever you need to have some whitespaces preserved, make sure to quote the entire expression. Future work will be put in to overhaul the [regex based parser](https://github.com/sudoblockio/tackle/blob/main/tackle/utils/command.py#L52) with a PEG parser like [parsimonious](https://github.com/erikrose/parsimonious).\n\n### Contributing\n\nContributions are welcome but please be advised of the following notes.\n\n- This project uses [conventional commits](https://www.conventionalcommits.org/) which generates the [changelog](./CHANGELOG.md) with [release-please-action](https://github.com/google-github-actions/release-please-action) in the [release](https://github.com/sudoblockio/tackle/blob/main/.github/workflows/release.yml) CI workflow. If commits have been made outside of this convention they will be squashed accordingly.\n- For making changes to providers, please include test coverage using the existing fixtures and patterns from prior tests or communicate any suggestions that deviate from this style. It definitely can be improved but consistency is more important than making directed improvements. Tests should be runnable from the test's directory and via `make test`.\n- For making changes to the core parser, please create a proposal first outlining your suggestions with examples before spending time working on code.\n\nIt is very easy to create new providers / hooks with tackle. Over time, it will adopt the same import pattern of what Ansible does where all provider / hooks (modules) are stored in version controlled locations. In the meantime, please feel free to contribute to this repository for hooks that have general applicability or create your own hooks in other repositories that are more bespoke / opinionated in nature.\n\n### Code of Conduct\n\nEveryone interacting in the tackle project's codebases, issue trackers, chat rooms, and mailing lists is expected to follow the [PyPA Code of Conduct](https://www.pypa.io/en/latest/code-of-conduct/).\n\n## Credit\n\nSpecial thanks to the [cookiecutter](https://github.com/cookiecutter/cookiecutter) community for laying the basis for this project.\n\n\n",
    "bugtrack_url": null,
    "license": "BSD",
    "summary": "Tackle is a declarative DSL for building modular workflows and code generators. Tool is plugins based and can easily be extended by writing additional hooks or importing external providers that can be turned into a self documenting CLI, all out of yaml, json, toml.",
    "version": "0.5.1",
    "project_urls": {
        "Homepage": "https://github.com/robcxyz/tackle"
    },
    "split_keywords": [
        "cookiecutter",
        "tackle",
        "tackle-box",
        "tacklebox",
        "tackle box",
        "python",
        "projects",
        "project templates",
        "jinja2",
        "skeleton",
        "scaffolding",
        "project directory",
        "package",
        "packaging",
        "kubernetes"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "87760653c43444878a77c82e41db3a1b1e32b15d48890c9b05af2a41848bbafa",
                "md5": "e0ba91bb8216a96ad8dbf55197826a81",
                "sha256": "23d0c4937a2ea8e3db4927f6dcb0cc7887c637fcba9e4aceef8a3f23a91e3c5f"
            },
            "downloads": -1,
            "filename": "tackle-0.5.1-py2.py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "e0ba91bb8216a96ad8dbf55197826a81",
            "packagetype": "bdist_wheel",
            "python_version": "py2.py3",
            "requires_python": ">=3.7",
            "size": 268182,
            "upload_time": "2023-07-17T13:44:35",
            "upload_time_iso_8601": "2023-07-17T13:44:35.378804Z",
            "url": "https://files.pythonhosted.org/packages/87/76/0653c43444878a77c82e41db3a1b1e32b15d48890c9b05af2a41848bbafa/tackle-0.5.1-py2.py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "cbe3fe2c90b7162ab34bc6afecbacaefe3f313b5d875529170814e109dda9018",
                "md5": "615f77e102ed3c1b65d18f5c8100f748",
                "sha256": "31cabb31c704cf6aadd16bf558c2ffb096db5a9497c310c0b361ae40dc282fd9"
            },
            "downloads": -1,
            "filename": "tackle-0.5.1.tar.gz",
            "has_sig": false,
            "md5_digest": "615f77e102ed3c1b65d18f5c8100f748",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 198309,
            "upload_time": "2023-07-17T13:44:36",
            "upload_time_iso_8601": "2023-07-17T13:44:36.735221Z",
            "url": "https://files.pythonhosted.org/packages/cb/e3/fe2c90b7162ab34bc6afecbacaefe3f313b5d875529170814e109dda9018/tackle-0.5.1.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-07-17 13:44:36",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "robcxyz",
    "github_project": "tackle",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "tox": true,
    "lcname": "tackle"
}
        
Elapsed time: 0.09852s