bwinterface


Namebwinterface JSON
Version 0.2.0 PyPI version JSON
download
home_pagehttps://www.github.com/towalink/bwinterface
Summaryaccess the Bitwarden/Vaultwarden CLI ("bw") conveniently
upload_time2024-10-20 20:01:20
maintainerNone
docs_urlNone
authorDirk Henrici
requires_python>=3.7
licenseNone
keywords bitwarden vaultwarden wrapper interface
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # bwinterface

A Python wrapper around the "bw" (Bitwarden) CLI tool

The "bw" command line utility is used for accessing Bitwarden vaults or Vaultwarden vaults. At time of writing, there is no stable official module for accessing such vaults using Python. Therefore, this module provides a low-level Python wrapper around the "bw" CLI utility. A bunch of higher-level methods and facilities are provided for increased convenience, too.

---

## Features

- Provide a low-level wrapper around the "bw" CLI utility.
- Parse the "bw" tool output into Python data structures.
- Provide certain results in more convenient data structures.
- Caches certain results for increased performance in practical applications.
- Provide convenience methods to access the "bw" functionality in a simple manner.
- No other external libraries are needed, i.e. no dependencies except the "bw" utility.

---

## Installation

Install using PyPi:

```shell
pip3 install bwinterface
```

---

## Quickstart

### Accessing the bw CLI using bwinterface

The following code snippets provide examples on how `bwinterface` is used:

```python
import bwinterface

# Instantiate a wrapper object while providing the path to the "bw" CLI
bw = bwinterface.BWInterface(bw_cli='/opt/bw')

# Execute any bw command using "execute"
result = bw.execute('--help')
print(result)

# Login using API key
if not bw.check_login():
    print(bw.set_config_server('https://vaultwarden.mydomain.local'))
    result = bw.login_apikey(clientid='user.aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee', clientsecret='ffffffffffffffffffffffffffffff')
    print(result)

# Unlock the local vault
result = bw.unlock('MyMasterPassword')
print(result)

# There are useful properties
print(bw.organizations_asdictbyid)
print(bw.organizations_asdictbyname)

# There are get methods with filtering capabilities. Examples:
print(bw.get_collections('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee').data)
print(bw.get_collections('MyCollection').data)
print(bw.get_collections_asdictbyid('MyCollection'))
print(bw.get_collections_asdictbyname('MyCollection'))
print(bw.get_collectionid('MyCollection'))
print(bw.get_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee'))
print(bw.get_item('"MyItem"'))
print(bw.get_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee').data.get('notes'))
print(bw.get_item_notes('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee'))
print(bw.get_items_asdictbyid(organization=''))  # default organization
print(bw.get_items_asdictbyname(organization='MyOrganization'))

# You may create, change and delete objects. Examples:
print(bw.create_item('MyNewItem', 'myuser', 'mypassword', notes='This is a note'))
print(bw.edit_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee', password='MyNewPassword'))
print(bw.delete_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee', permanent=True))

# And there are many other convenience methods. Examples:
print(bw.sync())
print(bw.get_status())
print(bw.generate_password(uppercase=True, lowercase=True, number=True, special=True, passphrase=None, length=20, words=None, min_number=None, min_special=None, separator=None, capitalize=None, include_number=None, avoid_ambiguous=True))
```

Please see the section below for full API documentation.

---

## Detailed API description

### Properties

#### organizations

*Returns a cached list of organizations as returned by "bw"*

#### organizations_asdictbyid

*Returns a dictionary of (cached) organizations with organization identifiers as keys*

Note: This just differently formats the data provided by the `organization` property.

#### organizations_asdictbyname

*Returns a dictionary of (cached) organizations with organization names as keys*

Note: This just differently formats the data provided by the `organization` property.

### Methods for interaction (in alphabetical order)

#### `__init__(bw_cli='/opt/bitwarden/bw', print_bwcommands=True, print_resultdata=False, print_indent=None, sparse_output=False, suppress_output=True, suppress_errors=False)`

*Initializes the instance*

Parameters:
* "bw_cli" (str, optional, default: "/opt/bitwarden/bw"): Path to the "bw" command line utility
    The bwinterface module depends on the "bw" CLI utility for its operation. Provide the full path to the tool that shall be used.
* "print_bwcommands" (boolean, optional, default: True): Echo "bw" commands
    If True, each "bw" command executed is printed to stdout to show what is going on. JSON data is shown without encoding.
* "print_resultdata" (boolean, optional, default: True): Echo "bw" commands
    If True, the JSON data returned by each "bw" is printed to stdout.
* "print_indent" (boolean, optional, default: None): Indentation for JSON pretty printing
    If provided, JSON data is printed prettily with the specified indentation (try 2).
* "sparse_output" (boolean, optional, default: False): Reduce amount of output
    If True, "--raw" output of "bw" ise provided only.
* "suppress_output" (boolean, optional, default: True): Don't show output of "bw" utility
    If True, the full output to stdout of the "bw" utility is printed.
* "suppress_errors" (boolean, optional, default: False): Don't show error output of "bw" utility
    If True, the full output to stderr of the "bw" utility is printed.

Examples:
* `bw = bwinterface.BWInterface(bw_cli='/opt/bw')`
* `bw = bwinterface.BWInterface(bw_cli='/opt/bw', suppressoutput=False)`

#### `check_login()`

*Checks whether we are logged in*

#### `create_collection(name, organization=None, external_id=None, otherfields=None)`

*Create a collection with the given data*

#### `create_item(name, username, password, organization=None, collection=None, folder=None, totp=None, uris=None, type=1, notes=None, favorite=False, fields=None, otherfields=None)`

*Create an item with the given data*

#### `delete_collection(collection, organization, permanent=False)`

*Delete collection with the provided identifier ('permanent' does not use trash)*

#### `delete_item(itemid, permanent=False)`

*Delete item with the provided identifier ('permanent' does not use trash)*

#### `dict2base64(datadict)`

*Encodes a dictionary into a base64-encoded JSON notation*

#### `edit_item(itemid, name=None, username=None, password=None, organization=None, collection=None, folder=None, totp=None, uris=None, type=None, notes=None, favorite=None, fields=None, otherfields=None, create_if_not_exists=False, use_cache=True)`

*Update an item with the given data*

#### `execute(command, env=None, datadict=None, nojson=False, sparse_output=None, pretty=None)`

*Execute a bw command and return result*

#### `generate(uppercase=None, lowercase=None, number=None, special=None, passphrase=None, length=None, words=None, min_number=None, min_special=None, separator=None, capitalize=None, include_number=None, avoid_ambiguous=None)`

*Generates a new password/passphrase*

#### `generate_password(uppercase=None, lowercase=None, number=None, special=None, passphrase=None, length=None, words=None, min_number=None, min_special=None, separator=None, capitalize=None, include_number=None, avoid_ambiguous=None)`

*Returns a new password/passphrase*

#### `get_collectionid(collection, organization=None, use_cache=True)`

*Converts a string identifying a collection into the collection's UUID*

#### `get_collections(organization=None)`

*Gets a list of collections, optionally filtered*

#### `get_collections_aslist(organization=None)`

*Gets a list of collections, optionally filtered; always returns a list (empty list on invalid filters)*

#### `get_collections_asdictbyid(organization=None, use_cache=True)`

*Dictionary of collections with identifiers as keys, optionally filtered*

#### `get_collections_asdictbyname(organization=None, use_cache=True)`

*Dictionary of collections with names as keys, optionally filtered*

#### `get_folderid(folder)`

*Converts a string identifying a folder into the folder's UUID*

#### `get_item(itemid)`

*Get item with the provided identifier*

Note: it is possible to enter a search term (use double quotes) instead of an item id

#### `get_item_notes(itemid)`

*Get notes of item with the provided identifier (result is provided in 'out')*

#### `get_items(organization=None, collection=None, folder=None)`

*Gets a list of items, optionally filtered*

#### `get_items_asdict(organization=None, collection=None, folder=None, byname=False)`

*Dictionary of items, uncached, optionally filtered*

#### `get_items_asdictbyid(organization=None, collection=None, folder=None, use_cache=True)`

*Dictionary of items with identifiers as keys, optionally filtered*

#### `get_items_asdictbyname(organization=None, collection=None, folder=None, use_cache=True)`

*Dictionary of items with names as keys, optionally filtered*

#### `get_items_aslist(organization=None, collection=None, folder=None)`

*Gets a list of items, optionally filtered; always returns a list (empty list on invalid filters)*

#### `get_organizationid(organization)`

*Converts a string identifying an organization into the organization's UUID*

#### `get_organizations()`

*Gets a list of organizations*

#### `get_status()`

*Gets status information*

#### `invalidate_collection_cache()`

*Clears the collection cache*

#### `invalidate_item_cache()`

*Clears the item cache*

#### `invalidate_organization_cache()`

*Clears the organization cache*

#### `is_uuid(s)`

*Checks whether the given string is a valid UUID*

#### `login_apikey(clientid, clientsecret)`

*Logs in using the provided API credentials*

#### `logout()`

*Logout from vault*

#### `run_process(command, env=None)`

*Execute a command and return result*

#### `set_config_server(server)`

*Configures the server to use*

#### `sync()`

*Gets updates from the remote vault*

#### `unlock(pwd)`

*Unlocks the vault with the provided password*

#### `update_collection_cache(data, organization)`

*Updates the collection cache with the new or updated collection in case organization matches*

#### `update_item_cache(data, organization)`

*Updates the item cache with the new or updated item in case organization matches*

---

## Reporting bugs

In case you encounter any bugs, please report the expected behavior and the actual behavior so that the issue can be reproduced and fixed.

---
## Developers

### Clone repository

Clone this repo to your local machine using `https://github.com/towalink/bwinterface.git`

Install the module temporarily to make it available in your Python installation:
```shell
pip3 install -e <path to root of "src" directory>
```

---

## License

[![License](http://img.shields.io/:license-mit-blue.svg?style=flat-square)](https://opensource.org/licenses/MIT)

- **[MIT license](https://opensource.org/licenses/MIT)**
- Copyright 2024 © <a href="https://github.com/towalink/bwinterface" target="_blank">Dirk Henrici</a>.
- This project is not associated with Bitwarden or Vaultwarden.

            

Raw data

            {
    "_id": null,
    "home_page": "https://www.github.com/towalink/bwinterface",
    "name": "bwinterface",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.7",
    "maintainer_email": null,
    "keywords": "Bitwarden Vaultwarden wrapper interface",
    "author": "Dirk Henrici",
    "author_email": "towalink.bwinterface@henrici.name",
    "download_url": "https://files.pythonhosted.org/packages/e0/a5/8660124e79d30c9b20bdd628fd00ec7ec1f7edbf9dcb585bf90fb559db3a/bwinterface-0.2.0.tar.gz",
    "platform": null,
    "description": "# bwinterface\n\nA Python wrapper around the \"bw\" (Bitwarden) CLI tool\n\nThe \"bw\" command line utility is used for accessing Bitwarden vaults or Vaultwarden vaults. At time of writing, there is no stable official module for accessing such vaults using Python. Therefore, this module provides a low-level Python wrapper around the \"bw\" CLI utility. A bunch of higher-level methods and facilities are provided for increased convenience, too.\n\n---\n\n## Features\n\n- Provide a low-level wrapper around the \"bw\" CLI utility.\n- Parse the \"bw\" tool output into Python data structures.\n- Provide certain results in more convenient data structures.\n- Caches certain results for increased performance in practical applications.\n- Provide convenience methods to access the \"bw\" functionality in a simple manner.\n- No other external libraries are needed, i.e. no dependencies except the \"bw\" utility.\n\n---\n\n## Installation\n\nInstall using PyPi:\n\n```shell\npip3 install bwinterface\n```\n\n---\n\n## Quickstart\n\n### Accessing the bw CLI using bwinterface\n\nThe following code snippets provide examples on how `bwinterface` is used:\n\n```python\nimport bwinterface\n\n# Instantiate a wrapper object while providing the path to the \"bw\" CLI\nbw = bwinterface.BWInterface(bw_cli='/opt/bw')\n\n# Execute any bw command using \"execute\"\nresult = bw.execute('--help')\nprint(result)\n\n# Login using API key\nif not bw.check_login():\n    print(bw.set_config_server('https://vaultwarden.mydomain.local'))\n    result = bw.login_apikey(clientid='user.aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee', clientsecret='ffffffffffffffffffffffffffffff')\n    print(result)\n\n# Unlock the local vault\nresult = bw.unlock('MyMasterPassword')\nprint(result)\n\n# There are useful properties\nprint(bw.organizations_asdictbyid)\nprint(bw.organizations_asdictbyname)\n\n# There are get methods with filtering capabilities. Examples:\nprint(bw.get_collections('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee').data)\nprint(bw.get_collections('MyCollection').data)\nprint(bw.get_collections_asdictbyid('MyCollection'))\nprint(bw.get_collections_asdictbyname('MyCollection'))\nprint(bw.get_collectionid('MyCollection'))\nprint(bw.get_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee'))\nprint(bw.get_item('\"MyItem\"'))\nprint(bw.get_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee').data.get('notes'))\nprint(bw.get_item_notes('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee'))\nprint(bw.get_items_asdictbyid(organization=''))  # default organization\nprint(bw.get_items_asdictbyname(organization='MyOrganization'))\n\n# You may create, change and delete objects. Examples:\nprint(bw.create_item('MyNewItem', 'myuser', 'mypassword', notes='This is a note'))\nprint(bw.edit_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee', password='MyNewPassword'))\nprint(bw.delete_item('aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee', permanent=True))\n\n# And there are many other convenience methods. Examples:\nprint(bw.sync())\nprint(bw.get_status())\nprint(bw.generate_password(uppercase=True, lowercase=True, number=True, special=True, passphrase=None, length=20, words=None, min_number=None, min_special=None, separator=None, capitalize=None, include_number=None, avoid_ambiguous=True))\n```\n\nPlease see the section below for full API documentation.\n\n---\n\n## Detailed API description\n\n### Properties\n\n#### organizations\n\n*Returns a cached list of organizations as returned by \"bw\"*\n\n#### organizations_asdictbyid\n\n*Returns a dictionary of (cached) organizations with organization identifiers as keys*\n\nNote: This just differently formats the data provided by the `organization` property.\n\n#### organizations_asdictbyname\n\n*Returns a dictionary of (cached) organizations with organization names as keys*\n\nNote: This just differently formats the data provided by the `organization` property.\n\n### Methods for interaction (in alphabetical order)\n\n#### `__init__(bw_cli='/opt/bitwarden/bw', print_bwcommands=True, print_resultdata=False, print_indent=None, sparse_output=False, suppress_output=True, suppress_errors=False)`\n\n*Initializes the instance*\n\nParameters:\n* \"bw_cli\" (str, optional, default: \"/opt/bitwarden/bw\"): Path to the \"bw\" command line utility\n    The bwinterface module depends on the \"bw\" CLI utility for its operation. Provide the full path to the tool that shall be used.\n* \"print_bwcommands\" (boolean, optional, default: True): Echo \"bw\" commands\n    If True, each \"bw\" command executed is printed to stdout to show what is going on. JSON data is shown without encoding.\n* \"print_resultdata\" (boolean, optional, default: True): Echo \"bw\" commands\n    If True, the JSON data returned by each \"bw\" is printed to stdout.\n* \"print_indent\" (boolean, optional, default: None): Indentation for JSON pretty printing\n    If provided, JSON data is printed prettily with the specified indentation (try 2).\n* \"sparse_output\" (boolean, optional, default: False): Reduce amount of output\n    If True, \"--raw\" output of \"bw\" ise provided only.\n* \"suppress_output\" (boolean, optional, default: True): Don't show output of \"bw\" utility\n    If True, the full output to stdout of the \"bw\" utility is printed.\n* \"suppress_errors\" (boolean, optional, default: False): Don't show error output of \"bw\" utility\n    If True, the full output to stderr of the \"bw\" utility is printed.\n\nExamples:\n* `bw = bwinterface.BWInterface(bw_cli='/opt/bw')`\n* `bw = bwinterface.BWInterface(bw_cli='/opt/bw', suppressoutput=False)`\n\n#### `check_login()`\n\n*Checks whether we are logged in*\n\n#### `create_collection(name, organization=None, external_id=None, otherfields=None)`\n\n*Create a collection with the given data*\n\n#### `create_item(name, username, password, organization=None, collection=None, folder=None, totp=None, uris=None, type=1, notes=None, favorite=False, fields=None, otherfields=None)`\n\n*Create an item with the given data*\n\n#### `delete_collection(collection, organization, permanent=False)`\n\n*Delete collection with the provided identifier ('permanent' does not use trash)*\n\n#### `delete_item(itemid, permanent=False)`\n\n*Delete item with the provided identifier ('permanent' does not use trash)*\n\n#### `dict2base64(datadict)`\n\n*Encodes a dictionary into a base64-encoded JSON notation*\n\n#### `edit_item(itemid, name=None, username=None, password=None, organization=None, collection=None, folder=None, totp=None, uris=None, type=None, notes=None, favorite=None, fields=None, otherfields=None, create_if_not_exists=False, use_cache=True)`\n\n*Update an item with the given data*\n\n#### `execute(command, env=None, datadict=None, nojson=False, sparse_output=None, pretty=None)`\n\n*Execute a bw command and return result*\n\n#### `generate(uppercase=None, lowercase=None, number=None, special=None, passphrase=None, length=None, words=None, min_number=None, min_special=None, separator=None, capitalize=None, include_number=None, avoid_ambiguous=None)`\n\n*Generates a new password/passphrase*\n\n#### `generate_password(uppercase=None, lowercase=None, number=None, special=None, passphrase=None, length=None, words=None, min_number=None, min_special=None, separator=None, capitalize=None, include_number=None, avoid_ambiguous=None)`\n\n*Returns a new password/passphrase*\n\n#### `get_collectionid(collection, organization=None, use_cache=True)`\n\n*Converts a string identifying a collection into the collection's UUID*\n\n#### `get_collections(organization=None)`\n\n*Gets a list of collections, optionally filtered*\n\n#### `get_collections_aslist(organization=None)`\n\n*Gets a list of collections, optionally filtered; always returns a list (empty list on invalid filters)*\n\n#### `get_collections_asdictbyid(organization=None, use_cache=True)`\n\n*Dictionary of collections with identifiers as keys, optionally filtered*\n\n#### `get_collections_asdictbyname(organization=None, use_cache=True)`\n\n*Dictionary of collections with names as keys, optionally filtered*\n\n#### `get_folderid(folder)`\n\n*Converts a string identifying a folder into the folder's UUID*\n\n#### `get_item(itemid)`\n\n*Get item with the provided identifier*\n\nNote: it is possible to enter a search term (use double quotes) instead of an item id\n\n#### `get_item_notes(itemid)`\n\n*Get notes of item with the provided identifier (result is provided in 'out')*\n\n#### `get_items(organization=None, collection=None, folder=None)`\n\n*Gets a list of items, optionally filtered*\n\n#### `get_items_asdict(organization=None, collection=None, folder=None, byname=False)`\n\n*Dictionary of items, uncached, optionally filtered*\n\n#### `get_items_asdictbyid(organization=None, collection=None, folder=None, use_cache=True)`\n\n*Dictionary of items with identifiers as keys, optionally filtered*\n\n#### `get_items_asdictbyname(organization=None, collection=None, folder=None, use_cache=True)`\n\n*Dictionary of items with names as keys, optionally filtered*\n\n#### `get_items_aslist(organization=None, collection=None, folder=None)`\n\n*Gets a list of items, optionally filtered; always returns a list (empty list on invalid filters)*\n\n#### `get_organizationid(organization)`\n\n*Converts a string identifying an organization into the organization's UUID*\n\n#### `get_organizations()`\n\n*Gets a list of organizations*\n\n#### `get_status()`\n\n*Gets status information*\n\n#### `invalidate_collection_cache()`\n\n*Clears the collection cache*\n\n#### `invalidate_item_cache()`\n\n*Clears the item cache*\n\n#### `invalidate_organization_cache()`\n\n*Clears the organization cache*\n\n#### `is_uuid(s)`\n\n*Checks whether the given string is a valid UUID*\n\n#### `login_apikey(clientid, clientsecret)`\n\n*Logs in using the provided API credentials*\n\n#### `logout()`\n\n*Logout from vault*\n\n#### `run_process(command, env=None)`\n\n*Execute a command and return result*\n\n#### `set_config_server(server)`\n\n*Configures the server to use*\n\n#### `sync()`\n\n*Gets updates from the remote vault*\n\n#### `unlock(pwd)`\n\n*Unlocks the vault with the provided password*\n\n#### `update_collection_cache(data, organization)`\n\n*Updates the collection cache with the new or updated collection in case organization matches*\n\n#### `update_item_cache(data, organization)`\n\n*Updates the item cache with the new or updated item in case organization matches*\n\n---\n\n## Reporting bugs\n\nIn case you encounter any bugs, please report the expected behavior and the actual behavior so that the issue can be reproduced and fixed.\n\n---\n## Developers\n\n### Clone repository\n\nClone this repo to your local machine using `https://github.com/towalink/bwinterface.git`\n\nInstall the module temporarily to make it available in your Python installation:\n```shell\npip3 install -e <path to root of \"src\" directory>\n```\n\n---\n\n## License\n\n[![License](http://img.shields.io/:license-mit-blue.svg?style=flat-square)](https://opensource.org/licenses/MIT)\n\n- **[MIT license](https://opensource.org/licenses/MIT)**\n- Copyright 2024 \u00a9 <a href=\"https://github.com/towalink/bwinterface\" target=\"_blank\">Dirk Henrici</a>.\n- This project is not associated with Bitwarden or Vaultwarden.\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "access the Bitwarden/Vaultwarden CLI (\"bw\") conveniently",
    "version": "0.2.0",
    "project_urls": {
        "Homepage": "https://www.github.com/towalink/bwinterface",
        "PyPi": "https://pypi.org/project/bwinterface/",
        "Repository": "https://www.github.com/towalink/bwinterface"
    },
    "split_keywords": [
        "bitwarden",
        "vaultwarden",
        "wrapper",
        "interface"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "328dd77a1e7ba7879b1dc621627aa345f12898051ff76e5f819bde66fb3f9720",
                "md5": "f845c184cb967130d6c768e16003afcd",
                "sha256": "96e9196fe9ba74cdf0a48556c4cc23d3df8044df4e5295777b12577010429312"
            },
            "downloads": -1,
            "filename": "bwinterface-0.2.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "f845c184cb967130d6c768e16003afcd",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.7",
            "size": 10495,
            "upload_time": "2024-10-20T20:01:18",
            "upload_time_iso_8601": "2024-10-20T20:01:18.636340Z",
            "url": "https://files.pythonhosted.org/packages/32/8d/d77a1e7ba7879b1dc621627aa345f12898051ff76e5f819bde66fb3f9720/bwinterface-0.2.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "e0a58660124e79d30c9b20bdd628fd00ec7ec1f7edbf9dcb585bf90fb559db3a",
                "md5": "10ee7070ad1319fdc8926ede73283984",
                "sha256": "33b2127ddab02138909c735c95270db53910ef7ca99d3b12ca8f44e5f40d1c59"
            },
            "downloads": -1,
            "filename": "bwinterface-0.2.0.tar.gz",
            "has_sig": false,
            "md5_digest": "10ee7070ad1319fdc8926ede73283984",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.7",
            "size": 12614,
            "upload_time": "2024-10-20T20:01:20",
            "upload_time_iso_8601": "2024-10-20T20:01:20.759799Z",
            "url": "https://files.pythonhosted.org/packages/e0/a5/8660124e79d30c9b20bdd628fd00ec7ec1f7edbf9dcb585bf90fb559db3a/bwinterface-0.2.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-10-20 20:01:20",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "towalink",
    "github_project": "bwinterface",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "bwinterface"
}
        
Elapsed time: 0.57143s