lassen


Namelassen JSON
Version 0.5.2 PyPI version JSON
download
home_page
SummaryCommon webapp scaffolding.
upload_time2024-01-15 17:46:26
maintainer
docs_urlNone
authorPierce Freeman
requires_python>=3.11,<4.0
license
keywords
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # lassen

**40.4881° N, 121.5049° W**

Core utilities for MonkeySee web applications. Not guaranteed to be backwards compatible, use at your own risk.

At its core, Lassen tries to:

- Provide a suite of conventions for the 99% case of CRUD backends: manipulating data, storing it, and serving to frontends.
- Create typehinted definitions for everything to provide robust mypy support up and down the stack.
- Configure settings by environment variables, with a standard set of keys for database connections and web services.
- Make common things trivial and hard things possible.

We also build connective tissue between fast, more tailored libraries:

- FastAPI for webapp routes
- Pydantic for data validation
- SQLAlchemy for database management
- Alembic for database migrations

## Design Philosophy

Backends revolve around a core set of data objects. In the early days of design, these typically mirror the business objectives almost 1:1 (User, Team, Project). As complexity grows, they might also start to include auxiliary metadata, such as state enums or locally cached copies of other tables. These objects might be further optimized for the database engine: indexes, refactored foreign keys, etc. This creates a divergence between the data objects and the API that hosts it to the outside world.

In some ways database tables are like choosing the best data structure. They should efficiently move data from disk to memory to remote clients, and back again. So long as the data conversion is fast and lossless, it doesn't matter as much how the sausage is made.

A web API on the other hand aims to provide semantic objects to clients. These should be the objects and actions that represent your domain. The API layer should intentionally contrain the state/action space to be context aware. Useful APIs don't just mirror the database.

In Lassen, we view CRUD actions as projections on top of the underlying data objects. They might involve field merges, field subset, etc. Most libraries solve for this divergence by forcing a forking of class definitions: a separate definition to Create, to Update, etc. This often creates redundent code that's hard to sift through and reason about when adding new behavior.

Rather than configuring this CRUD at a class level, we focus on the CRUD actions that users can perform on each field. The `Stub` class defined below specifies _one_ key that is backed by a database value, and then generates CRUD schemas for API use depending on the allowed permissions for each field. Through this key-wise definition, we aim to clearly delineate in code and API contracts what is permitted, while aligning access patterns with the data values themselves.

## Structure

**Stores:** Each datamodel is expected to have its own store. Base classes that provide standard logic are provided by `lassen.store`
- StoreBase: Base class for all stores
- StoreFilterMixin: Mixin for filtering stores that specify an additional schema to use to filter
- StoreS3Mixin: Mixin for stores that use S3 for external storage of a piece of data. Support compression on both upload and downloads.

**Schemas:** Each datamodel should define a Model class (SQLAlchemy base object) and a series of Schema objects (Pydantic) that allow the Store to serialize the models. These schemas are also often used for direct CRUD referencing in the API layer.

We use a base `Stub` file to generate these schemas from a centralized definition. When defining generators you should use a path that can be fully managed by lassen, since we will remove and regenerate these files on each run.

```python
STORE_GENERATOR = StoreGenerator("models/auto")
SCHEMA_GENERATOR = SchemaGenerator("schemas/auto")
```

```bash
poetry run generate-lassen
```

**Migrations:** Lassen includes a templated alembic.init and env.py file. Client applications just need to have a `migrations` folder within their project root. After this you can swap `poetry run alembic` with `poetry run migrate`.

```sh
poetry run migrate upgrade head
```

**Settings:** Application settings should subclass our core settings. This provides a standard way to load settings from environment variables and includes common database keys.

```python
from lassen.core.config import CoreSettings, register_settings

@register_settings
class ClientSettings(CoreSettings):
    pass
```

**Schemas:** For helper schemas when returning results via API, see [lassen.schema](./lassen/schema.py).

## Development

Install all the extra dependencies so you can fully run the unit test suite.

```sh
poetry install --extras "aws database datasets"

createuser lassen
createdb -O lassen lassen_db
createdb -O lassen lassen_test_db
```

Unit Tests:

```sh
poetry run pytest
```

            

Raw data

            {
    "_id": null,
    "home_page": "",
    "name": "lassen",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.11,<4.0",
    "maintainer_email": "",
    "keywords": "",
    "author": "Pierce Freeman",
    "author_email": "pierce@freeman.vc",
    "download_url": "https://files.pythonhosted.org/packages/59/f4/62b461ad2928f220cf71478771e90f5f0ee5cd09a40fd9287a62b9a3c241/lassen-0.5.2.tar.gz",
    "platform": null,
    "description": "# lassen\n\n**40.4881\u00b0 N, 121.5049\u00b0 W**\n\nCore utilities for MonkeySee web applications. Not guaranteed to be backwards compatible, use at your own risk.\n\nAt its core, Lassen tries to:\n\n- Provide a suite of conventions for the 99% case of CRUD backends: manipulating data, storing it, and serving to frontends.\n- Create typehinted definitions for everything to provide robust mypy support up and down the stack.\n- Configure settings by environment variables, with a standard set of keys for database connections and web services.\n- Make common things trivial and hard things possible.\n\nWe also build connective tissue between fast, more tailored libraries:\n\n- FastAPI for webapp routes\n- Pydantic for data validation\n- SQLAlchemy for database management\n- Alembic for database migrations\n\n## Design Philosophy\n\nBackends revolve around a core set of data objects. In the early days of design, these typically mirror the business objectives almost 1:1 (User, Team, Project). As complexity grows, they might also start to include auxiliary metadata, such as state enums or locally cached copies of other tables. These objects might be further optimized for the database engine: indexes, refactored foreign keys, etc. This creates a divergence between the data objects and the API that hosts it to the outside world.\n\nIn some ways database tables are like choosing the best data structure. They should efficiently move data from disk to memory to remote clients, and back again. So long as the data conversion is fast and lossless, it doesn't matter as much how the sausage is made.\n\nA web API on the other hand aims to provide semantic objects to clients. These should be the objects and actions that represent your domain. The API layer should intentionally contrain the state/action space to be context aware. Useful APIs don't just mirror the database.\n\nIn Lassen, we view CRUD actions as projections on top of the underlying data objects. They might involve field merges, field subset, etc. Most libraries solve for this divergence by forcing a forking of class definitions: a separate definition to Create, to Update, etc. This often creates redundent code that's hard to sift through and reason about when adding new behavior.\n\nRather than configuring this CRUD at a class level, we focus on the CRUD actions that users can perform on each field. The `Stub` class defined below specifies _one_ key that is backed by a database value, and then generates CRUD schemas for API use depending on the allowed permissions for each field. Through this key-wise definition, we aim to clearly delineate in code and API contracts what is permitted, while aligning access patterns with the data values themselves.\n\n## Structure\n\n**Stores:** Each datamodel is expected to have its own store. Base classes that provide standard logic are provided by `lassen.store`\n- StoreBase: Base class for all stores\n- StoreFilterMixin: Mixin for filtering stores that specify an additional schema to use to filter\n- StoreS3Mixin: Mixin for stores that use S3 for external storage of a piece of data. Support compression on both upload and downloads.\n\n**Schemas:** Each datamodel should define a Model class (SQLAlchemy base object) and a series of Schema objects (Pydantic) that allow the Store to serialize the models. These schemas are also often used for direct CRUD referencing in the API layer.\n\nWe use a base `Stub` file to generate these schemas from a centralized definition. When defining generators you should use a path that can be fully managed by lassen, since we will remove and regenerate these files on each run.\n\n```python\nSTORE_GENERATOR = StoreGenerator(\"models/auto\")\nSCHEMA_GENERATOR = SchemaGenerator(\"schemas/auto\")\n```\n\n```bash\npoetry run generate-lassen\n```\n\n**Migrations:** Lassen includes a templated alembic.init and env.py file. Client applications just need to have a `migrations` folder within their project root. After this you can swap `poetry run alembic` with `poetry run migrate`.\n\n```sh\npoetry run migrate upgrade head\n```\n\n**Settings:** Application settings should subclass our core settings. This provides a standard way to load settings from environment variables and includes common database keys.\n\n```python\nfrom lassen.core.config import CoreSettings, register_settings\n\n@register_settings\nclass ClientSettings(CoreSettings):\n    pass\n```\n\n**Schemas:** For helper schemas when returning results via API, see [lassen.schema](./lassen/schema.py).\n\n## Development\n\nInstall all the extra dependencies so you can fully run the unit test suite.\n\n```sh\npoetry install --extras \"aws database datasets\"\n\ncreateuser lassen\ncreatedb -O lassen lassen_db\ncreatedb -O lassen lassen_test_db\n```\n\nUnit Tests:\n\n```sh\npoetry run pytest\n```\n",
    "bugtrack_url": null,
    "license": "",
    "summary": "Common webapp scaffolding.",
    "version": "0.5.2",
    "project_urls": null,
    "split_keywords": [],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "6247f639645d7bd0907f1e754156c220c70095351be4c1181539b57fe7be603e",
                "md5": "ebac83573b162b42bb07cf54399336e4",
                "sha256": "4d548f09acb51b946a02138f0b3583803a07525cec003e89197787b44c9fdb31"
            },
            "downloads": -1,
            "filename": "lassen-0.5.2-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "ebac83573b162b42bb07cf54399336e4",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.11,<4.0",
            "size": 73451,
            "upload_time": "2024-01-15T17:46:24",
            "upload_time_iso_8601": "2024-01-15T17:46:24.213618Z",
            "url": "https://files.pythonhosted.org/packages/62/47/f639645d7bd0907f1e754156c220c70095351be4c1181539b57fe7be603e/lassen-0.5.2-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "59f462b461ad2928f220cf71478771e90f5f0ee5cd09a40fd9287a62b9a3c241",
                "md5": "729e93584a27cb4262acdfd44fbe328d",
                "sha256": "e70e6adf0dea9700669070182318acf409a094aa4d0a862090b233d420819ed1"
            },
            "downloads": -1,
            "filename": "lassen-0.5.2.tar.gz",
            "has_sig": false,
            "md5_digest": "729e93584a27cb4262acdfd44fbe328d",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.11,<4.0",
            "size": 65236,
            "upload_time": "2024-01-15T17:46:26",
            "upload_time_iso_8601": "2024-01-15T17:46:26.111386Z",
            "url": "https://files.pythonhosted.org/packages/59/f4/62b461ad2928f220cf71478771e90f5f0ee5cd09a40fd9287a62b9a3c241/lassen-0.5.2.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-01-15 17:46:26",
    "github": false,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "lcname": "lassen"
}
        
Elapsed time: 0.17542s