# 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"
}