{ "info": { "author": "Jean-Fran\u00e7ois Corbett", "author_email": "jeaco@orsted.dk", "bugtrack_url": null, "classifiers": [ "Development Status :: 4 - Beta", "Intended Audience :: Developers", "License :: OSI Approved :: BSD License", "Programming Language :: Python :: 3" ], "description": "# startables for Python\n\nPython package with data structures and functionality to read/write files in StarTable format and contain and manipulate the information therein.\n\n## This project\n\n### Talk to us\n\nFor feature requests and bugs relating specifically to this Python package, please refer to this GitHub project's issue tracker. \n\nFor issues relating to the StarTable format specification more broadly, please consult the [StarTable standard page](https://github.com/startable/startable-standard). \n\n### Contribute and explore\n\nContributions are welcome, especially if they relate to an issue that you have previously discussed or posted in the issue tracker. \n\n## Installation\n\nAvailable on PyPI:\n\n```\npip install startables\n```\n\nand on conda-forge:\n\n```\nconda install startables -c conda-forge\n```\n\n## Example usage\n\n```python\nimport startables as st # << recommended import idiom\nimport pandas as pd\n\n# Build table manually\ndf = pd.DataFrame(data=[[float('nan'), 'gnu', 3], [4, 'gnat', '{{(+ x y)}}']], columns=['a', 'b', 'c'])\ncol_specs = {n: ColumnMetadata(Unit(u)) for n, u in zip(['a', 'b', 'c'], ['-', 'text', 'm'])}\ntable_x = st.Table(df=df, name='some_table', col_specs=col_specs, destinations={'success', 'glory'})\n\n# Accessing Table contents: use Bundle.df pandas.DataFrame interface\nprint(table_x.df['b'])\n\n# Read bundle of tables from file\nb = st.read_csv('animal_farm_startable_file.csv')\n\n# Make new bundle containing a subset of this bundle's tables by name and/ or destination\nfarm_bundle = b.filter(name_pattern='farm') \n\n# Accessing tables in bundle: use Bundle.tables List interface\nfor t in b.tables:\n print(t.name)\n# Add tables to bundle\nb.tables.append(table_x)\n\n# Remove tables from bundle by name and/or destination\nremoved_tables = b.pop_tables(destination='my_farm') # tables now removed from b\n\n# ... More examples to come ...\n\n```\n\n\n\n## Object model\n\n### Table\n\n`Table` contains a single StarTable table block, including table name, destination field, and equal-length columns, with each column containing a list of values and having a name and metadata (currently, a unit field and a (not-fully-implemented) remark, both of them strings). \n\nTable contents are stored in a pandas DataFrame. `Table.df` grants the user read/write access to this DataFrame. Table column names are stored as the DataFrame column (Series) names. Column units are stored separately. \n\nThe user can modify the DataFrame at will using the pandas.DataFrame API, or even replace the DataFrame entirely. However, introducing columns with names not covered in the Table's column specification will break the Table. Other than that, removing columns, adding/removing rows, editing cells etc. are all fine and shouldn't break anything. \n\n### Bundle\n`Bundle` is a container for one or more `Table` that belong together, usually because they:\n- have a common origin e.g. come from the same file or file structure, and/or\n- are understood as having a common context, in particular when evaluating expressions\n\n`Bundle` is intended as the primary interface for file I/O. The `read_csv()` and `read_excel()` functions read StarTable files in CSV and Excel format, respectively, and both return a `Bundle`, while `Bundle.to_csv()` and `Bundle.to_excel()` writes a collection of tables to these same file formats. \n\n### Other classes\n`TableOrigin` contains an indication of where a given `Table` came from, intended for use in traceability. Currently it is just a stub that contains a single string. \n\n`ColumnMetadata` is a container class for a column's unit (and free-text remark, though this is not tied with read/write methods yet, so of limited utility). A Table's columns are specified by supplying a dict of column_name:ColumnMetadata that covers (at least) all column names present in the Table's child DataFrame. \n\n## Expressions\n\nTable cells are allowed to contain not only literal values, but also Lisp expressions. \n\n`Table.evaluate_expressions(context)` will return a `Table` with expressions (if there are any) evaluated based on the given context. Can also specify `inplace=True` to do this in-place. \n\n`Bundle.evaluate_expressions(context)` does the same thing, but for all its child tables. \n\n## Unit conversion\n\nThe units of a `Table` can be converted according to a `UnitPolicy`. \n\nA `UnitConversion` defines how a given source unit is converted to an associated reference unit. \n\n```python\nkm_m = ScaleUnitConversion(src_unit=Unit('km'), ref_unit=Unit('m'), ref_per_src=1000)\nkm_m.to_ref(42) # returns 42000\nkm_m.from_ref(2000) # returns 2\n```\n\nA `UnitPolicy` contains an arbitrary number of `UnitConversion`, with the restriction that any source unit is associated to one and only one reference unit, i.e. can't include a `UnitConversion` from `'mile'` to `'km'` and another from `'mile'` to `'m'` (but sure can include one from `'km'` to `'m'` and another from `'mile'` to `'m'`). Reference units themselves are automatically added as source units, with themselves as their own reference unit through an `IdentityUnitConversion`. Conversion is then possible between any two source units that share the same reference unit. \n\n```python\nC_K = LinearUnitConversion(Unit('\u00b0C'), Unit('K'), slope=1, intercept=273.15)\ncup = CustomUnitPolicy([\n ScaleUnitConversion(Unit('km'), Unit('m'), 1000),\n ScaleUnitConversion(Unit('mm'), Unit('m'), 0.001),\n IdentityUnitConversion(Unit('metre'), Unit('m')), # alias of a reference unit\n C_K, \n C_K.alias(src_unit_alias=Unit('deg_C')) # alias of a source unit\n ])\n\ncup.convert(42, from_unit=Unit('m'), to_unit=Unit('mm')) # returns 42000\ncup.convert(42, from_unit=Unit('mm'), to_unit=Unit('metre')) # returns 0.042\ncup.convert(42, from_unit=Unit('km'), to_unit=Unit('mm')) # returns 42000000\ncup.convert_to_ref(20, src_unit=Unit('deg_C')) # returns 293.15\ncup.ref_unit(src_unit=Unit('\u00b0C')) # returns Unit('K')\n```\n\nA `Table`'s units are converted column by column in accordance with the `UnitPolicy`. \n\n- `Table.convert_to_ref_units()` converts each column to its `UnitPolicy` reference unit by calling `UnitPolicy.convert_to_ref()`\n- `Table.convert_units()` converts to new units explicitly specified for each column.\n\n- `Table.convert_to_home_units()` is a special case of `Table.convert_units()` which converts back to the Table's \"home units\". \"Home units\" are saved in the Table's col_specs and are the column units with which the `Table` was created (whether manually or read from file), unless they are explicitly changed later. \n\nUnit conversion does not support expressions. Expressions must be evaluated prior to unit conversion. \n\n## Changelog\n\nThis project was migrated to GitHub from a private server at v0.8.0. Changes prior to this are not included in the GitHub repo; nevertheless, the pre-0.8.0 changelog is documented here. *PS-##* below refers to issue numbers on a legacy YouTrack issue tracker on a private server. These issue numbers are left as is for the historical record. \n\nThis project follows [semantic versioning](https://semver.org/). This changelog follows the guidelines published on [keepachangelog.com](https://keepachangelog.com/en/1.0.0/).\n\n### Unreleased\n\nIn a coming release, the following items are lined up to be...\n\n#### Added\n\n* {...crickets chirping...}\n\n\n### 0.8.5 - 2020-17-12\n\n#### Fixed\n\n* `pandas.read_excel()` is now called with `engine='openpyxl'` to ensure compalibility with `.xlsx` files\n\n### 0.8.4 - 2019-12-05\n\n#### Fixed\n\n* `Bundle.to_csv()` puts a separator on the blank line between the header and the remainder of the file, to facilitate round-trip read using `read_csv()`. This is a quick-fix workaround, because the real problem lies with the `read_csv()` parser.\n\n### 0.8.3 - 2019-10-30\n\n#### Added\n\n* [#2](https://github.com/startable/startables-python/issues/2): `read_bulk()` convenience function to read multiple StarTable files at once into a single `Bundle`. \n* `import_from_word()` utility function to parse table blocks from tables in Microsoft Word `*.docx` files. \n\n### 0.8.2 - 2019-10-30\n\n#### Added\n\n* [#5](https://github.com/startable/startables-python/issues/5): `ColumnMetadata` optional property `format_str`: format string specifying how the column's values will be formatted when writing to file using `Bundle.to_csv()` and `Bundle.to_excel()`.\n* [#6](https://github.com/startable/startables-python/issues/6): Optional `header` parameter to `Bundle.to_csv()` and `Bundle.to_excel()` to allow writing a free-text header at the top of the created file. Additionally, optional `header_sep` argument to `Bundle.to_excel()`, indicating a separator to split the header across multiple columns. \n\n### 0.8.1 - 2019-09-05\n\n#### Added\n\n* Updated setup.py and setup.cfd to enable proper release on PyPI.\n\n### 0.8.0 - 2019-09-03\n\n#### Changed\n\n* First parameter of `read_csv()` renamed from `stream` to `filepath_or_buffer`. The name `stream` was inconsistent with the expected type since 0.7.3, namely `str` or `pathlib.Path` (in addition to `TextIO` streams). Also, the new name `filepath_or_buffer` is consistent with `pandas.read_csv()`. This change will break code that has used `stream` as a named argument, though we are hopeful that this has rarely if ever been done by users of this API. \n* Removed restriction on `openpyxl` version (was previously restricted to < 2.6). This is a less crappy fix to [PS-49](https://youtrackncc/issue/PS-49) than had previously been implemented. \n\n#### Fixed\n\n* [PS-52](https://youtrackncc/issue/PS-52) read_csv() throws warning when given a stream as input; asks for a filename \n* [PS-53](https://youtrackncc/issue/PS-53) Bundle.to_csv() fails when column names are not strings\n\n### 0.7.3 - 2019-02-27\n\n#### Fixed \n\n- `openpyxl<26` dependency in environment.yml\n- [PS-19](https://youtrackncc/issue/PS-19) Reading from CSV can fail if not enough column delimiters on first line of CSV file\n- [PS-48](https://youtrackncc/issue/PS-48) Increase compatibility of startables python library by allowing non-standard formatted .csv files\n- [PS-50](https://youtrackncc/issue/PS-50) CSV files exported from Excel results in first table not being read due to UTF-8-BOM\n\n### 0.7.2 - 2019-02-18\n\n#### Fixed\n\nVersion [2.6.0](https://openpyxl.readthedocs.io/en/stable/changes.html#id1) of our `openpyxl` dependency, released a couple of weeks ago, contains major breaking changes (which kind of goes against the spirit of minor version updates in semantic versioning...) and these breaking changes do indeed break `startables`. To remedy this, the `openpyxl` version number is now fixed to `<2.6` in the `startables` conda package recipe. \n\n### 0.7.1 - 2018-11-30\n\n#### Changed \n\n- **Breaking changes** in methods `Bundle.filter()` and `Bundle.pop_tables()`: \n - Parameter `exact_name` renamed to `name` for consistency with the naming of destination-related parameters. \n - Ordering of parameters in signature changed to `(name, name_pattern, destination, destination_pattern, ignore_case)`\n - [PS-43](https://youtrackncc/issue/PS-43) Name and destination filters are now case-insensitive by default. Can be made case-sensitive again by setting parameter `ignore_case=False`.\n\n#### Added\n\n- [PS-41](https://youtrackncc/issue/PS-41) Filtering on destinations by regular expression\n\n### 0.7.0 - 2018-11-28\n\nAll of the changes in this version address [PS-27: Add/remove tables in Bundle](https://youtrackncc/issue/PS-27)\n\n#### Changed\n\n**Breaking changes** in `Bundle`:\n\n- Method `tables` renamed to `filter`. Instead of returning a `List[Table]`, now returns a `Bundle` containing the filtered tables (i.e. a subset of the original `Bundle`).\n- Property `tables` introduced (not to be confused with the former method of the same name). Returns the internal list of of tables stored in this `Bundle`. \n- All list-related operations are delegated to the list returned by the `tables` property. In particular: \n - **Can now add `Table`s to `Bundle`** (a main driver for this major change) by invoking `List`'s `append()` and `extend()` methods on `Bundle.tables`. \n - Magic methods `__getitem__`, `__iter__`, and `__len__` have been removed. \n\n#### Added\n\n- Method `Bundle.pop_tables()` to **remove a selection of `Bundle`'s member tables**, selected by name and/or destination. Returns the removed tables. (This was the other main driver for this major change.)\n\n### 0.6.1 - 2018-11-22\n\n#### Changed\n\n- [PS-2](https://youtrackncc/issue/PS-2) The ordering of destinations is now preserved. Table destinations can now be supplied as any `Iterable` (changed from `Set`) and are then stored internally as a `List`, thus preserving pre-existing order (if any). **Potentially breaking change**: A `ValueError` will be raised upon encountering any duplicates in the destinations supplied to a `Table`, either when read from file or programmatically. (Because duplicates are indeed nonsensical.) Duplicates were previously eliminated silently when read from file using `read_csv` and `read_excel`, and were not possible programmatically (pedants please refrain) as they had to be given as a `Set`.\n\n### 0.6.0 - 2018-11-20\n\n#### Added\n\n- Introducing: **Unit conversion machinery** [PS-10](https://youtrackncc/issue/PS-10)\n- Script that publishes this readme on windwiki\n- [PS-8](https://youtrackncc/issue/PS-8) A more helpful error message on syntax error raised while parsing an expression cell, guiding the user to the offending cell\n\n#### Changed\n\n- [PS-35](https://youtrackncc/issue/PS-35) Python version requirement relaxed to 3.6 and above (was strictly 3.6) \n\n#### Removed\n\n- [PS-33](https://youtrackncc/issue/PS-33) Logging. Now gone. Was generating too much noise in client code logs.\n\n\n### 0.5.5 - 2018-09-10\n\n#### Changed\n\n* [PS-15](https://youtrackncc/issue/PS-15) `read_csv()` and `read_excel()` now accepts `'nan'`, `'NaN'`, and `'NAN'` as valid no-data markers. Previously, only `'-'` was accepted. \n\n### **0.5.4** - 2018-09-07\n\n#### Fixed\n\n- [PS-28](https://youtrackncc/issue/PS-28) Numeric data in text columns doesn't get read. \n\n### **0.5.3** - 2018-09-07\n\n#### Fixed\n\n* [PS-5](https://youtrackncc/issue/PS-5) Table blocks with zero rows are ignored by read_excel() and read_csv(). \n\n### **0.5.2** - 2018-09-07\n\n#### Changed\n\n* [PS-14](https://youtrackncc/issue/PS-14) `read_csv()` now forwards `*args` and `*kwargs` to `pandas.read_csv()`, so user can now make use of `pandas.read_csv()`'s many useful arguments, not least `decimal` to control which decimal separator is used in numerical columns (typically either `'.'` or `','`). **Breaking change**: included in this forward is the previously explicitly implemented `sep` argument, which means that the default value of `sep` has now changed from `';'` to pandas' own default, `','`. This is a breaking change, but improves consistency with pandas' API. \n\n### **0.5.1** - 2018-08-28\n\n#### Changed\n\n- Column metadata (basically, just units for now) now stored in a separate field, rather than as monkeypatches on the child data frame's columns. The latter proved too fragile. \n\n#### Added\n\n - `Table.df` setter: user can now replace the Table's child dataframe, as long as all the columns of the new df are described in (and consistent with) the Table's column specification (dict of name:ColumnMetadata). If not, error is raised. \n - [PS-4](https://youtrackncc/issue/PS-4) Ability to get `Bundle.tables()` by destination\n - [PS-13](https://youtrackncc/issue/PS-13) Add exact_name option to `Bundle.tables()`\n - Column metadata now supports not only a unit, but also a free-text remark, but this is not yet used in the file readers and writers; until it is, this feature won't be very useful. \n\n#### Fixed\n\n- [PS-7](https://youtrackncc/issue/PS-7) After using `read_excel()`, `evaluate_expressions()` fails unless DataFrame index is manually reset\n- Other minor bug fixes.\n\n### **0.5.0** - 2018-08-10\n\nComplete redesign compared to the earlier 0.1 package. Total breaking change in the API. Pandas dataframes now lie at the heart of Table objects.\nRequires Python 3.6. \n\n\n\n\n", "description_content_type": "text/markdown", "docs_url": null, "download_url": "", "downloads": { "last_day": -1, "last_month": -1, "last_week": -1 }, "home_page": "https://github.com/startable/startables-python/", "keywords": "startable data-structure file-format table", "license": "BSD-3-Clause", "maintainer": "", "maintainer_email": "", "name": "startables", "package_url": "https://pypi.org/project/startables/", "platform": "", "project_url": "https://pypi.org/project/startables/", "project_urls": { "Homepage": "https://github.com/startable/startables-python/" }, "release_url": "https://pypi.org/project/startables/0.8.5/", "requires_dist": [ "numpy", "pandas", "openpyxl" ], "requires_python": ">=3.6", "summary": "Reads, writes, and manipulates data stored in StarTable format", "version": "0.8.5", "yanked": false, "yanked_reason": null }, "last_serial": 8931536, "releases": { "0.8.0": [ { "comment_text": "", "digests": { "md5": "4fc9d2d0df67316618faa07d350bda83", "sha256": "9ee8e9f185161cd7e45d2194e23c684d163eb503262dd2dea5ce02eef8d935fd" }, "downloads": -1, "filename": "startables-0.8.0.tar.gz", "has_sig": false, "md5_digest": "4fc9d2d0df67316618faa07d350bda83", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 33270, "upload_time": "2019-09-05T13:04:51", "upload_time_iso_8601": "2019-09-05T13:04:51.940611Z", "url": "https://files.pythonhosted.org/packages/02/29/8b4997e4895324abd5f47904c4046c8a73e439f6598f9da34a5514578dfe/startables-0.8.0.tar.gz", "yanked": false, "yanked_reason": null } ], "0.8.1": [ { "comment_text": "", "digests": { "md5": "f07c3024a37bfacea133f513c76eb784", "sha256": "186403164147fba3ee5e2e593290854645f29c7bf73cea5972107977a669774b" }, "downloads": -1, "filename": "startables-0.8.1.tar.gz", "has_sig": false, "md5_digest": "f07c3024a37bfacea133f513c76eb784", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 33409, "upload_time": "2019-09-05T13:19:37", "upload_time_iso_8601": "2019-09-05T13:19:37.430368Z", "url": "https://files.pythonhosted.org/packages/df/e7/a4723dafd3a7268a2bd08d848e1bf97db47faa5d66944d6167337f4de57d/startables-0.8.1.tar.gz", "yanked": false, "yanked_reason": null } ], "0.8.2": [ { "comment_text": "", "digests": { "md5": "d60fb31c8afa0894c6f8d4ad58d5e70d", "sha256": "fe9f56632a3f8ec56998c723ade039fa0dfa601dd751e8f44c0bb839a4027b2c" }, "downloads": -1, "filename": "startables-0.8.2.tar.gz", "has_sig": false, "md5_digest": "d60fb31c8afa0894c6f8d4ad58d5e70d", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 40866, "upload_time": "2019-10-30T11:50:51", "upload_time_iso_8601": "2019-10-30T11:50:51.803765Z", "url": "https://files.pythonhosted.org/packages/fa/62/7ab1430e04a6b203d77c783b858cc68de21a22f1a183c575d5c00a198313/startables-0.8.2.tar.gz", "yanked": false, "yanked_reason": null } ], "0.8.3": [ { "comment_text": "", "digests": { "md5": "e694fef8842138557e6c397a4adc6306", "sha256": "0b463acdacad1da53671fef2241dcffe4f539820243bb517584266f032198a07" }, "downloads": -1, "filename": "startables-0.8.3.tar.gz", "has_sig": false, "md5_digest": "e694fef8842138557e6c397a4adc6306", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 37503, "upload_time": "2019-10-30T15:01:29", "upload_time_iso_8601": "2019-10-30T15:01:29.421745Z", "url": "https://files.pythonhosted.org/packages/ad/10/d62af759e7b4d773f72e1c4efc0fa6cbb2bb574bcd4b03fc2f2a5cdd3c65/startables-0.8.3.tar.gz", "yanked": false, "yanked_reason": null } ], "0.8.4": [ { "comment_text": "", "digests": { "md5": "741474993b94068d11075ffe0dd242f2", "sha256": "f417d05ecf8bbb5d11a3662faccaca3fcb702e8e8ece4cce21922588d6271b49" }, "downloads": -1, "filename": "startables-0.8.4.tar.gz", "has_sig": false, "md5_digest": "741474993b94068d11075ffe0dd242f2", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 37832, "upload_time": "2019-12-05T13:45:10", "upload_time_iso_8601": "2019-12-05T13:45:10.409535Z", "url": "https://files.pythonhosted.org/packages/66/74/582aa599841ed665f91e4bed4af5b41fa9d3a5021fe0151e6210bebece86/startables-0.8.4.tar.gz", "yanked": false, "yanked_reason": null } ], "0.8.5": [ { "comment_text": "", "digests": { "md5": "1fb52e0dd4d9f8f8425cf005c61cbca7", "sha256": "a1b8abee3ea4dacec72da4d8c21672fbacbcbabe314e561e050e418d0e99ef8a" }, "downloads": -1, "filename": "startables-0.8.5-py3-none-any.whl", "has_sig": false, "md5_digest": "1fb52e0dd4d9f8f8425cf005c61cbca7", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": ">=3.6", "size": 36664, "upload_time": "2020-12-18T08:50:43", "upload_time_iso_8601": "2020-12-18T08:50:43.373259Z", "url": "https://files.pythonhosted.org/packages/d3/dc/1e8e9f74e277dff5e4548ee7bde4bc57afd10d857934d429cb5660ef816a/startables-0.8.5-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "f0fd4419790a536ecc7ebd1748dd8a6f", "sha256": "dc34f48f8194a71895f0502551845e7edee5cc88d123f6598461176c9d9a2ad9" }, "downloads": -1, "filename": "startables-0.8.5.tar.gz", "has_sig": false, "md5_digest": "f0fd4419790a536ecc7ebd1748dd8a6f", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 42972, "upload_time": "2020-12-18T08:50:44", "upload_time_iso_8601": "2020-12-18T08:50:44.932970Z", "url": "https://files.pythonhosted.org/packages/dd/b2/2a1feead9b0aece417050e832c6cc085d3242b2ac847259e4f2350d56b77/startables-0.8.5.tar.gz", "yanked": false, "yanked_reason": null } ] }, "urls": [ { "comment_text": "", "digests": { "md5": "1fb52e0dd4d9f8f8425cf005c61cbca7", "sha256": "a1b8abee3ea4dacec72da4d8c21672fbacbcbabe314e561e050e418d0e99ef8a" }, "downloads": -1, "filename": "startables-0.8.5-py3-none-any.whl", "has_sig": false, "md5_digest": "1fb52e0dd4d9f8f8425cf005c61cbca7", "packagetype": "bdist_wheel", "python_version": "py3", "requires_python": ">=3.6", "size": 36664, "upload_time": "2020-12-18T08:50:43", "upload_time_iso_8601": "2020-12-18T08:50:43.373259Z", "url": "https://files.pythonhosted.org/packages/d3/dc/1e8e9f74e277dff5e4548ee7bde4bc57afd10d857934d429cb5660ef816a/startables-0.8.5-py3-none-any.whl", "yanked": false, "yanked_reason": null }, { "comment_text": "", "digests": { "md5": "f0fd4419790a536ecc7ebd1748dd8a6f", "sha256": "dc34f48f8194a71895f0502551845e7edee5cc88d123f6598461176c9d9a2ad9" }, "downloads": -1, "filename": "startables-0.8.5.tar.gz", "has_sig": false, "md5_digest": "f0fd4419790a536ecc7ebd1748dd8a6f", "packagetype": "sdist", "python_version": "source", "requires_python": ">=3.6", "size": 42972, "upload_time": "2020-12-18T08:50:44", "upload_time_iso_8601": "2020-12-18T08:50:44.932970Z", "url": "https://files.pythonhosted.org/packages/dd/b2/2a1feead9b0aece417050e832c6cc085d3242b2ac847259e4f2350d56b77/startables-0.8.5.tar.gz", "yanked": false, "yanked_reason": null } ], "vulnerabilities": [] }