st-supabase-connection


Namest-supabase-connection JSON
Version 2.1.3 PyPI version JSON
download
home_pagehttps://github.com/SiddhantSadangi/st_supabase_connection
SummaryA Streamlit connection component for Supabase.
upload_time2025-11-02 19:11:00
maintainerNone
docs_urlNone
authorSiddhant Sadangi
requires_python>=3.10
licenseNone
keywords streamlit supabase connection integration
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # :electric_plug: Streamlit Supabase Connector

<div align="center">
  <a href="https://pepy.tech/project/st-supabase-connection">
    <img src="https://static.pepy.tech/personalized-badge/st-supabase-connection?period=total&units=international_system&left_color=black&right_color=brightgreen&left_text=Downloads" alt="Downloads">
  </a>
  <a href="https://badge.fury.io/py/st-supabase-connection">
    <img src="https://badge.fury.io/py/st-supabase-connection.svg" alt="PyPI version">
  </a>
  <a href="https://opensource.org/licenses/MIT">
    <img src="https://img.shields.io/badge/License-MIT-yellow.svg" alt="License: MIT">
  </a>
  <a href="https://github.com/SiddhantSadangi/st_supabase_connection/issues">
    <img src="https://img.shields.io/github/issues/SiddhantSadangi/st_supabase_connection.svg" alt="Issues">
  </a>
  <a href="https://github.com/SiddhantSadangi/st_supabase_connection/pulls">
    <img src="https://img.shields.io/badge/PRs-welcome-brightgreen.svg" alt="PRs Welcome">
  </a>
</div>

A Streamlit connection component to connect Streamlit to Supabase Storage, Database, and Auth.

## Contents
- [:electric\_plug: Streamlit Supabase Connector](#electric_plug-streamlit-supabase-connector)
  - [Contents](#contents)
  - [🚀 Quickstart](#-quickstart)
  - [:student: Interactive tutorial](#student-interactive-tutorial)
  - [:thinking: Why use this?](#thinking-why-use-this)
  - [:hammer\_and\_wrench: Setup](#hammer_and_wrench-setup)
  - [:magic\_wand: Usage](#magic_wand-usage)
  - [:ok\_hand: Supported methods](#ok_hand-supported-methods)
  - [:books: Examples](#books-examples)
    - [:package: Storage operations](#package-storage-operations)
      - [List existing buckets](#list-existing-buckets)
      - [Create a bucket](#create-a-bucket)
      - [Get bucket details](#get-bucket-details)
      - [Update a bucket](#update-a-bucket)
      - [Move files in a bucket](#move-files-in-a-bucket)
      - [List objects in a bucket](#list-objects-in-a-bucket)
      - [Empty a bucket](#empty-a-bucket)
      - [Delete a bucket](#delete-a-bucket)
    - [:file\_cabinet: Database operations](#file_cabinet-database-operations)
      - [Simple query](#simple-query)
      - [Query with join](#query-with-join)
      - [Filter through foreign tables](#filter-through-foreign-tables)
      - [Insert rows](#insert-rows)
    - [:lock: Auth operations](#lock-auth-operations)
      - [Create new user](#create-new-user)
      - [Sign in with password](#sign-in-with-password)
      - [Retrieve session](#retrieve-session)
      - [Retrieve user](#retrieve-user)
      - [Sign out](#sign-out)
  - [:star: Explore all options in a demo app](#star-explore-all-options-in-a-demo-app)
  - [:bow: Acknowledgements](#bow-acknowledgements)
  - [:hugs: Want to support my work?](#hugs-want-to-support-my-work)

## 🚀 Quickstart

1. Install the connector and its dependencies.

    ```bash
    pip install st-supabase-connection
    ```

2. Provide your Supabase credentials. In Streamlit Cloud, add them to `secrets.toml`:

    ```toml
    [connections.supabase_connection]
    url = "https://your-project.supabase.co"
    key = "service_role_or_anon_key"
    ```

    For local development you can use environment variables (`SUPABASE_URL`, `SUPABASE_KEY`) or pass the values directly during connection creation.

3. Create the cached connection in your app.

    ```python
    import streamlit as st
    from st_supabase_connection import SupabaseConnection

    st_supabase = st.connection(
        name="supabase_connection",
        type=SupabaseConnection,
        ttl=None,  # cache indefinitely; override when you need fresher data
    )

    # Example: list buckets without re-authenticating on every rerun
    buckets = st_supabase.list_buckets()
    st.write(buckets)
    ```

## :student: Interactive tutorial

<div align="center">
    <a href="https://st-supabase-connection.streamlit.app/">
        <img src="https://static.streamlit.io/badges/streamlit_badge_black_white.svg" alt="Open in Streamlit" style="height: 60px !important;width: 217px !important;">
    </a>
</div>

![Web capture_2-12-2023_124639_st-supabase-connection streamlit app](https://github.com/SiddhantSadangi/st_supabase_connection/assets/41324509/2870b021-48a0-4143-9693-c840880a28be)

## :thinking: Why use this?

- [x] Cache functionality to cache returned results. **Save time and money** on your API requests
- [x] Same method names as the Supabase Python API. **Minimum relearning required**
- [x] **Exposes more storage methods** than currently supported by the Supabase Python API. For example, `update()`, `create_signed_upload_url()`, and `upload_to_signed_url()`
- [x] Handles common Supabase quirks—leading slashes are normalised, MIME types inferred, and downloads streamed in memory—so you spend less time on glue code.
- [x] **Less keystrokes required** when integrating with your Streamlit app.

<details close>
<summary>Examples with and without the connector </summary>
<br>
<table>
<tr>
<td><b>Without connector</b></td><td><b>With connector</b></td>
<tr>
<td colspan="2"> Download file to local system from Supabase storage </td>
<tr>
<td valign="top">

```python
import mimetypes
import streamlit as st
from supabase import create_client

supabase_client = create_client(
    supabase_url="...", supabase_key="..."
)

bucket_id = st.text_input("Enter the bucket_id")
source_path = st.text_input("Enter source path")

file_name = source_path.split("/")[-1]

if st.button("Request download"):
    with open(file_name, "wb+") as f:
        response = supabase_client.storage.from_(
        bucket_id
        ).download(source_path)
        f.write(response)

    mime = mimetypes.guess_type(file_name)[0]
    data = open(file_name, "rb")

    st.download_button(
        "Download file", data=data,
        file_name=file_name, mime=mime,
    )
```

</td>
<td valign="top">

```python
import streamlit as st
from st_supabase_connection import SupabaseConnection

st_supabase_client = st.connection(
    name="supabase_connection", type=SupabaseConnection
)

bucket_id = st.text_input("Enter the bucket_id")
source_path = st.text_input("Enter source path")

if st.button("Request download"):
    file_name, mime, data = st_supabase_client.download(
        bucket_id,
        source_path,
    )  # returns (name, mime_type, bytes)

    st.download_button(
        "Download file", data=data,
        file_name=file_name, mime=mime,
    )

```

</td>
<tr>
<td colspan="2"> Upload file from local system to Supabase storage </td>
<tr>
<td valign="top">

```python
import streamlit as st
from supabase import create_client

supabase_client = create_client(
supabase_key="...", supabase_url="..."
)

bucket_id = st.text_input("Enter the bucket_id")
uploaded_file = st.file_uploader("Choose a file")
destination_path = st.text_input("Enter destination path")
overwrite = "true" if st.checkbox("Overwrite?") else "false"

with open(uploaded_file.name, "wb") as f:
    f.write(uploaded_file.getbuffer())

if st.button("Upload"):
    with open(uploaded_file.name, "rb") as f:
        supabase_client.storage.from_(bucket_id).upload(
            path=destination_path,
            file=f,
            file_options={
            "content-type": uploaded_file.type,
            "x-upsert": overwrite,
            },
        )

```

</td>
<td valign="top">

```python
import streamlit as st
from st_supabase_connection import SupabaseConnection

st_supabase_client = st.connection(
    name="supabase_connection", type=SupabaseConnection
)

bucket_id = st.text_input("Enter the bucket_id")
uploaded_file = st.file_uploader("Choose a file")
destination_path = st.text_input("Enter destination path")
overwrite = "true" if st.checkbox("Overwrite?") else "false"

if st.button("Upload"):
    st_supabase_client.upload(
        bucket_id, "local", uploaded_file,
        destination_path, overwrite,
    )
```

<tr>
</table>

</details>

## :hammer_and_wrench: Setup

1. Install `st-supabase-connection`

```sh
pip install st-supabase-connection
```

2. Set the `SUPABASE_URL` and `SUPABASE_KEY` Streamlit secrets as described [here](https://docs.streamlit.io/streamlit-community-cloud/get-started/deploy-an-app/connect-to-data-sources/secrets-management).

> [!NOTE]  
> For local development outside Streamlit, you can also set these as your environment variables (recommended), or pass these to the `url` and `key` args of `st.connection()`.

## :magic_wand: Usage

1. Import

```python
from st_supabase_connection import SupabaseConnection, execute_query
```

2. Initialize

```python
st_supabase_client = st.connection(
    name="YOUR_CONNECTION_NAME",
    type=SupabaseConnection,
    ttl=None,
)
```

3. Use the connection to work with Storage, Database, and Auth in a cached, Streamlit-friendly way:

    ```python
    # Storage
    file_name, mime, data = st_supabase.download("bucket", "path/to/report.csv", ttl=300)

    # Database (leverages postgrest-py under the hood)
    from st_supabase_connection import execute_query
    users = execute_query(
        st_supabase.table("users").select("name, email").order("created_at", desc=True),
        ttl="15m",
    )

    # Auth (cached helper)
    st_supabase.cached_sign_in_with_password({"email": email, "password": password})
    ```

## :ok_hand: Supported methods

<details close>
<summary> Storage </summary>
<ul>
    <li> <code>delete_bucket()</code> </li>
    <li> <code>empty_bucket()</code> </li>
    <li> <code>get_bucket()</code> </li>
    <li> <code>list_buckets()</code> </li>
    <li> <code>create_bucket()</code> </li>
    <li> <code>upload()</code> </li>
    <li> <code>download()</code> </li>
    <li> <code>update_bucket()</code> </li>
    <li> <code>move()</code> </li>
    <li> <code>list_objects()</code> </li>
    <li> <code>create_signed_urls()</code> </li>
    <li> <code>get_public_url()</code> </li>
    <li> <code>create_signed_upload_url()</code> </li>
    <li> <code>upload_to_signed_url()</code> </li>
</ul>

</details>

<details close>
<summary> Database </summary>
<ul>
    <li> <code>execute_query()</code> - Executes the passed query with caching enabled. </li>
    <li> All methods supported by <a href="https://postgrest-py.readthedocs.io/en/latest/api/request_builders.html">postgrest-py</a>.
</details>

<details>
<summary> Auth </summary>
<ul>
    <li> <code>cached_sign_in_with_password()</code> - Cached version of <code>sign_in_with_password()</code> for faster sign-in. </li>
    <li> All methods supported by <a href="https://supabase.com/docs/reference/python/auth-signup">Supabase's Python API </a>.
</details>

## :books: Examples

### :package: Storage operations

#### List existing buckets

```python
>>> st_supabase_client.list_buckets(ttl=None)
[
    SyncBucket(
        id="bucket1",
        name="bucket1",
        owner="",
        public=False,
        created_at=datetime.datetime(2023, 7, 31, 19, 56, 21, 518438, tzinfo=tzutc()),
        updated_at=datetime.datetime(2023, 7, 31, 19, 56, 21, 518438, tzinfo=tzutc()),
        file_size_limit=None,
        allowed_mime_types=None,
    ),
    SyncBucket(
        id="bucket2",
        name="bucket2",
        owner="",
        public=True,
        created_at=datetime.datetime(2023, 7, 31, 19, 56, 28, 203536, tzinfo=tzutc()),
        updated_at=datetime.datetime(2023, 7, 31, 19, 56, 28, 203536, tzinfo=tzutc()),
        file_size_limit=100,
        allowed_mime_types=["image/jpg", "image/png"],
    ),
]
```

#### Create a bucket

```python
>>> st_supabase_client.create_bucket("new_bucket")
{'name': 'new_bucket'}
```

#### Get bucket details

```python
>>> st_supabase_client.get_bucket("new_bucket")
SyncBucket(id='new_bucket', name='new_bucket', owner='', public=True, created_at=datetime.datetime(2023, 8, 2, 19, 41, 44, 810000, tzinfo=tzutc()), updated_at=datetime.datetime(2023, 8, 2, 19, 41, 44, 810000, tzinfo=tzutc()), file_size_limit=None, allowed_mime_types=None)
```

#### Update a bucket

```python
>>> st_supabase_client.update_bucket(
      "new_bucket",
      file_size_limit=100,
      allowed_mime_types=["image/jpg", "image/png"],
      public=True,
    )
{'message': 'Successfully updated'}
```

#### Move files in a bucket

```python
>>> st_supabase_client.move("new_bucket", "test.png", "folder1/new_test.png")
{'message': 'Successfully moved'}
```

#### List objects in a bucket

```python
>>> st_supabase_client.list_objects("new_bucket", path="folder1", ttl=0)
[
    {
        "name": "new_test.png",
        "id": "e506920e-2834-440e-85f1-1d5476927582",
        "updated_at": "2023-08-02T19:53:22.53986+00:00",
        "created_at": "2023-08-02T19:52:20.404391+00:00",
        "last_accessed_at": "2023-08-02T19:53:21.833+00:00",
        "metadata": {
            "eTag": '"814a0034f5549e957ee61360d87457e5"',
            "size": 473831,
            "mimetype": "image/png",
            "cacheControl": "max-age=3600",
            "lastModified": "2023-08-02T19:53:23.000Z",
            "contentLength": 473831,
            "httpStatusCode": 200,
        },
    }
]
```

#### Empty a bucket

```python
>>> st_supabase_client.empty_bucket("new_bucket")
{'message': 'Successfully emptied'}
```

#### Delete a bucket

```python
>>> st_supabase_client.delete_bucket("new_bucket")
{'message': 'Successfully deleted'}
```

### :file_cabinet: Database operations

#### Simple query

```python
>>> execute_query(st_supabase_client.table("countries").select("*"), ttl=0)
APIResponse(
    data=[
        {"id": 1, "name": "Afghanistan"},
        {"id": 2, "name": "Albania"},
        {"id": 3, "name": "Algeria"},
    ],
    count=None,
)
```

#### Query with join

```python
>>> execute_query(
        st_supabase_client.table("users").select("name, teams(name)", count="exact"),
        ttl="1h",
    )

APIResponse(
    data=[
        {"name": "Kiran", "teams": [{"name": "Green"}, {"name": "Blue"}]},
        {"name": "Evan", "teams": [{"name": "Blue"}]},
    ],
    count=2,
)
```

#### Filter through foreign tables

```python
>>> execute_query(
        st_supabase_client.table("cities").select("name, countries(*)", count="exact").eq("countries.name", "Curaçao"),
        ttl=None,
    )

APIResponse(
    data=[
        {
            "name": "Kralendijk",
            "countries": {
                "id": 2,
                "name": "Curaçao",
                "iso2": "CW",
                "iso3": "CUW",
                "local_name": None,
                "continent": None,
            },
        },
        {"name": "Willemstad", "countries": None},
    ],
    count=2,
)
```

#### Insert rows

```python
>>> execute_query(
        st_supabase_client.table("countries").insert(
            [{"name": "Wakanda", "iso2": "WK"}, {"name": "Wadiya", "iso2": "WD"}], count="None"
        ),
        ttl=0,
    )

APIResponse(
    data=[
        {
            "id": 250,
            "name": "Wakanda",
            "iso2": "WK",
            "iso3": None,
            "local_name": None,
            "continent": None,
        },
        {
            "id": 251,
            "name": "Wadiya",
            "iso2": "WD",
            "iso3": None,
            "local_name": None,
            "continent": None,
        },
    ],
    count=None,
)
```

### :lock: Auth operations

> [!NOTE]  
> If the call is valid, all Supabase Auth methods return the same response structure:
>
> ```json
> {
>   "user": {
>     "id": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544",
>     "app_metadata": {
>       "provider": "email",
>       "providers": ["email"]
>     },
>     "user_metadata": {
>       "attribution": "I made it :)",
>       "fname": "Siddhant"
>     },
>     "aud": "authenticated",
>     "confirmation_sent_at": null,
>     "recovery_sent_at": null,
>     "email_change_sent_at": null,
>     "new_email": null,
>     "invited_at": null,
>     "action_link": null,
>     "email": "test.user@abc.com",
>     "phone": "",
>     "created_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 365359, tzinfo=datetime.timezone.utc)",
>     "confirmed_at": null,
>     "email_confirmed_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 373966, tzinfo=datetime.timezone.utc)",
>     "phone_confirmed_at": null,
>     "last_sign_in_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 377070, tzinfo=datetime.timezone.utc)",
>     "role": "authenticated",
>     "updated_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 381584, tzinfo=datetime.timezone.utc)",
>     "identities": [
>       {
>         "id": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544",
>         "user_id": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544",
>         "identity_data": {
>           "email": "siddhant.sadangi@gmail.com",
>           "sub": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544"
>         },
>         "provider": "email",
>         "created_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)",
>         "last_sign_in_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 370002, tzinfo=datetime.timezone.utc)",
>         "updated_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)"
>       }
>     ],
>     "factors": null
>   },
>   "session": {
>     "provider_token": null,
>     "provider_refresh_token": null,
>     "access_token": "***",
>     "refresh_token": "***",
>     "expires_in": 3600,
>     "expires_at": 1696800390,
>     "token_type": "bearer",
>     "user": {
>       "id": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544",
>       "app_metadata": {
>         "provider": "email",
>         "providers": ["email"]
>       },
>       "user_metadata": {
>         "attribution": "I made it :)",
>         "fname": "Siddhant"
>       },
>       "aud": "authenticated",
>       "confirmation_sent_at": null,
>       "recovery_sent_at": null,
>       "email_change_sent_at": null,
>       "new_email": null,
>       "invited_at": null,
>       "action_link": null,
>       "email": "test.user@abc.com",
>       "phone": "",
>       "created_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 365359, tzinfo=datetime.timezone.utc)",
>       "confirmed_at": null,
>       "email_confirmed_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 373966, tzinfo=datetime.timezone.utc)",
>       "phone_confirmed_at": null,
>       "last_sign_in_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 377070, tzinfo=datetime.timezone.utc)",
>       "role": "authenticated",
>       "updated_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 381584, tzinfo=datetime.timezone.utc)",
>       "identities": [
>         {
>           "id": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544",
>           "user_id": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544",
>           "identity_data": {
>             "email": "siddhant.sadangi@gmail.com",
>             "sub": "e1f550fd-9cd1-44e4-bbe4-c04e91cf5544"
>           },
>           "provider": "email",
>           "created_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)",
>           "last_sign_in_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 370002, tzinfo=datetime.timezone.utc)",
>           "updated_at": "datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)"
>         }
>       ],
>       "factors": null
>     }
>   }
> }
> ```
>
> </details>

#### Create new user

```python
st_supabase_client.auth.sign_up(
    dict(
        email='test.user@abc.com',
        password='***',
        options=dict(
            data=dict(
                fname='Siddhant',
                attribution='I made it :)',
            )
        )
    )
)
```

#### Sign in with password

`SupabaseConnection()` offers a cached version of `sign_in_with_password()` for faster, request-free sign-ins.

```python
st_supabase_client.cached_sign_in_with_password(dict(email='test.user@abc.com', password='***'))
```

#### Retrieve session

```python
st_supabase_client.auth.get_session()
```

#### Retrieve user

```python
st_supabase_client.auth.get_user()
```

#### Sign out

```python
st_supabase_client.auth.sign_out()
```

> [!NOTE]  
> Check the [Supabase Python API reference](https://supabase.com/docs/reference/python/select) for more examples.

## :star: Explore all options in a demo app

[![Open in Streamlit](https://static.streamlit.io/badges/streamlit_badge_black_white.svg)](https://st-supabase-connection.streamlit.app/)

## :bow: Acknowledgements

This connector builds upon the awesome work done by the open-source community in general and the [Supabase Community](https://github.com/supabase-community) in particular. I cannot be more thankful to all the authors whose work I have used either directly or indirectly.

Thanks to all contributors to this project :bow:

<p align="center">
    <a href="https://github.com/SiddhantSadangi/st_supabase_connection/graphs/contributors">
        <img src="https://contrib.rocks/image?repo=SiddhantSadangi/st_supabase_connection" alt="Contributors" style="height: 60px !important;width: 217px !important;">
    </a>
</p>

## :hugs: Want to support my work?

<p align="center">
    <a href="https://www.buymeacoffee.com/siddhantsadangi" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 60px !important;">
    </a>
    <br>
    <a href="https://github.com/sponsors/SiddhantSadangi" target="_blank"><img src="https://img.shields.io/badge/Sponsor%20me%20on-GitHub-f34b7d?logo=github&style=flat" alt="Sponsor me on GitHub" style="height: 28px !important;">
</p>

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/SiddhantSadangi/st_supabase_connection",
    "name": "st-supabase-connection",
    "maintainer": null,
    "docs_url": null,
    "requires_python": ">=3.10",
    "maintainer_email": null,
    "keywords": "streamlit, supabase, connection, integration",
    "author": "Siddhant Sadangi",
    "author_email": "siddhant.sadangi@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/e2/b0/f7ce9add868cee6188e036406e7b81c1dd258958e535a7d7e052a8ec43f7/st_supabase_connection-2.1.3.tar.gz",
    "platform": null,
    "description": "# :electric_plug: Streamlit Supabase Connector\n\n<div align=\"center\">\n  <a href=\"https://pepy.tech/project/st-supabase-connection\">\n    <img src=\"https://static.pepy.tech/personalized-badge/st-supabase-connection?period=total&units=international_system&left_color=black&right_color=brightgreen&left_text=Downloads\" alt=\"Downloads\">\n  </a>\n  <a href=\"https://badge.fury.io/py/st-supabase-connection\">\n    <img src=\"https://badge.fury.io/py/st-supabase-connection.svg\" alt=\"PyPI version\">\n  </a>\n  <a href=\"https://opensource.org/licenses/MIT\">\n    <img src=\"https://img.shields.io/badge/License-MIT-yellow.svg\" alt=\"License: MIT\">\n  </a>\n  <a href=\"https://github.com/SiddhantSadangi/st_supabase_connection/issues\">\n    <img src=\"https://img.shields.io/github/issues/SiddhantSadangi/st_supabase_connection.svg\" alt=\"Issues\">\n  </a>\n  <a href=\"https://github.com/SiddhantSadangi/st_supabase_connection/pulls\">\n    <img src=\"https://img.shields.io/badge/PRs-welcome-brightgreen.svg\" alt=\"PRs Welcome\">\n  </a>\n</div>\n\nA Streamlit connection component to connect Streamlit to Supabase Storage, Database, and Auth.\n\n## Contents\n- [:electric\\_plug: Streamlit Supabase Connector](#electric_plug-streamlit-supabase-connector)\n  - [Contents](#contents)\n  - [\ud83d\ude80 Quickstart](#-quickstart)\n  - [:student: Interactive tutorial](#student-interactive-tutorial)\n  - [:thinking: Why use this?](#thinking-why-use-this)\n  - [:hammer\\_and\\_wrench: Setup](#hammer_and_wrench-setup)\n  - [:magic\\_wand: Usage](#magic_wand-usage)\n  - [:ok\\_hand: Supported methods](#ok_hand-supported-methods)\n  - [:books: Examples](#books-examples)\n    - [:package: Storage operations](#package-storage-operations)\n      - [List existing buckets](#list-existing-buckets)\n      - [Create a bucket](#create-a-bucket)\n      - [Get bucket details](#get-bucket-details)\n      - [Update a bucket](#update-a-bucket)\n      - [Move files in a bucket](#move-files-in-a-bucket)\n      - [List objects in a bucket](#list-objects-in-a-bucket)\n      - [Empty a bucket](#empty-a-bucket)\n      - [Delete a bucket](#delete-a-bucket)\n    - [:file\\_cabinet: Database operations](#file_cabinet-database-operations)\n      - [Simple query](#simple-query)\n      - [Query with join](#query-with-join)\n      - [Filter through foreign tables](#filter-through-foreign-tables)\n      - [Insert rows](#insert-rows)\n    - [:lock: Auth operations](#lock-auth-operations)\n      - [Create new user](#create-new-user)\n      - [Sign in with password](#sign-in-with-password)\n      - [Retrieve session](#retrieve-session)\n      - [Retrieve user](#retrieve-user)\n      - [Sign out](#sign-out)\n  - [:star: Explore all options in a demo app](#star-explore-all-options-in-a-demo-app)\n  - [:bow: Acknowledgements](#bow-acknowledgements)\n  - [:hugs: Want to support my work?](#hugs-want-to-support-my-work)\n\n## \ud83d\ude80 Quickstart\n\n1. Install the connector and its dependencies.\n\n    ```bash\n    pip install st-supabase-connection\n    ```\n\n2. Provide your Supabase credentials. In Streamlit Cloud, add them to `secrets.toml`:\n\n    ```toml\n    [connections.supabase_connection]\n    url = \"https://your-project.supabase.co\"\n    key = \"service_role_or_anon_key\"\n    ```\n\n    For local development you can use environment variables (`SUPABASE_URL`, `SUPABASE_KEY`) or pass the values directly during connection creation.\n\n3. Create the cached connection in your app.\n\n    ```python\n    import streamlit as st\n    from st_supabase_connection import SupabaseConnection\n\n    st_supabase = st.connection(\n        name=\"supabase_connection\",\n        type=SupabaseConnection,\n        ttl=None,  # cache indefinitely; override when you need fresher data\n    )\n\n    # Example: list buckets without re-authenticating on every rerun\n    buckets = st_supabase.list_buckets()\n    st.write(buckets)\n    ```\n\n## :student: Interactive tutorial\n\n<div align=\"center\">\n    <a href=\"https://st-supabase-connection.streamlit.app/\">\n        <img src=\"https://static.streamlit.io/badges/streamlit_badge_black_white.svg\" alt=\"Open in Streamlit\" style=\"height: 60px !important;width: 217px !important;\">\n    </a>\n</div>\n\n![Web capture_2-12-2023_124639_st-supabase-connection streamlit app](https://github.com/SiddhantSadangi/st_supabase_connection/assets/41324509/2870b021-48a0-4143-9693-c840880a28be)\n\n## :thinking: Why use this?\n\n- [x] Cache functionality to cache returned results. **Save time and money** on your API requests\n- [x] Same method names as the Supabase Python API. **Minimum relearning required**\n- [x] **Exposes more storage methods** than currently supported by the Supabase Python API. For example, `update()`, `create_signed_upload_url()`, and `upload_to_signed_url()`\n- [x] Handles common Supabase quirks\u2014leading slashes are normalised, MIME types inferred, and downloads streamed in memory\u2014so you spend less time on glue code.\n- [x] **Less keystrokes required** when integrating with your Streamlit app.\n\n<details close>\n<summary>Examples with and without the connector </summary>\n<br>\n<table>\n<tr>\n<td><b>Without connector</b></td><td><b>With connector</b></td>\n<tr>\n<td colspan=\"2\"> Download file to local system from Supabase storage </td>\n<tr>\n<td valign=\"top\">\n\n```python\nimport mimetypes\nimport streamlit as st\nfrom supabase import create_client\n\nsupabase_client = create_client(\n    supabase_url=\"...\", supabase_key=\"...\"\n)\n\nbucket_id = st.text_input(\"Enter the bucket_id\")\nsource_path = st.text_input(\"Enter source path\")\n\nfile_name = source_path.split(\"/\")[-1]\n\nif st.button(\"Request download\"):\n    with open(file_name, \"wb+\") as f:\n        response = supabase_client.storage.from_(\n        bucket_id\n        ).download(source_path)\n        f.write(response)\n\n    mime = mimetypes.guess_type(file_name)[0]\n    data = open(file_name, \"rb\")\n\n    st.download_button(\n        \"Download file\", data=data,\n        file_name=file_name, mime=mime,\n    )\n```\n\n</td>\n<td valign=\"top\">\n\n```python\nimport streamlit as st\nfrom st_supabase_connection import SupabaseConnection\n\nst_supabase_client = st.connection(\n    name=\"supabase_connection\", type=SupabaseConnection\n)\n\nbucket_id = st.text_input(\"Enter the bucket_id\")\nsource_path = st.text_input(\"Enter source path\")\n\nif st.button(\"Request download\"):\n    file_name, mime, data = st_supabase_client.download(\n        bucket_id,\n        source_path,\n    )  # returns (name, mime_type, bytes)\n\n    st.download_button(\n        \"Download file\", data=data,\n        file_name=file_name, mime=mime,\n    )\n\n```\n\n</td>\n<tr>\n<td colspan=\"2\"> Upload file from local system to Supabase storage </td>\n<tr>\n<td valign=\"top\">\n\n```python\nimport streamlit as st\nfrom supabase import create_client\n\nsupabase_client = create_client(\nsupabase_key=\"...\", supabase_url=\"...\"\n)\n\nbucket_id = st.text_input(\"Enter the bucket_id\")\nuploaded_file = st.file_uploader(\"Choose a file\")\ndestination_path = st.text_input(\"Enter destination path\")\noverwrite = \"true\" if st.checkbox(\"Overwrite?\") else \"false\"\n\nwith open(uploaded_file.name, \"wb\") as f:\n    f.write(uploaded_file.getbuffer())\n\nif st.button(\"Upload\"):\n    with open(uploaded_file.name, \"rb\") as f:\n        supabase_client.storage.from_(bucket_id).upload(\n            path=destination_path,\n            file=f,\n            file_options={\n            \"content-type\": uploaded_file.type,\n            \"x-upsert\": overwrite,\n            },\n        )\n\n```\n\n</td>\n<td valign=\"top\">\n\n```python\nimport streamlit as st\nfrom st_supabase_connection import SupabaseConnection\n\nst_supabase_client = st.connection(\n    name=\"supabase_connection\", type=SupabaseConnection\n)\n\nbucket_id = st.text_input(\"Enter the bucket_id\")\nuploaded_file = st.file_uploader(\"Choose a file\")\ndestination_path = st.text_input(\"Enter destination path\")\noverwrite = \"true\" if st.checkbox(\"Overwrite?\") else \"false\"\n\nif st.button(\"Upload\"):\n    st_supabase_client.upload(\n        bucket_id, \"local\", uploaded_file,\n        destination_path, overwrite,\n    )\n```\n\n<tr>\n</table>\n\n</details>\n\n## :hammer_and_wrench: Setup\n\n1. Install `st-supabase-connection`\n\n```sh\npip install st-supabase-connection\n```\n\n2. Set the `SUPABASE_URL` and `SUPABASE_KEY` Streamlit secrets as described [here](https://docs.streamlit.io/streamlit-community-cloud/get-started/deploy-an-app/connect-to-data-sources/secrets-management).\n\n> [!NOTE]  \n> For local development outside Streamlit, you can also set these as your environment variables (recommended), or pass these to the `url` and `key` args of `st.connection()`.\n\n## :magic_wand: Usage\n\n1. Import\n\n```python\nfrom st_supabase_connection import SupabaseConnection, execute_query\n```\n\n2. Initialize\n\n```python\nst_supabase_client = st.connection(\n    name=\"YOUR_CONNECTION_NAME\",\n    type=SupabaseConnection,\n    ttl=None,\n)\n```\n\n3. Use the connection to work with Storage, Database, and Auth in a cached, Streamlit-friendly way:\n\n    ```python\n    # Storage\n    file_name, mime, data = st_supabase.download(\"bucket\", \"path/to/report.csv\", ttl=300)\n\n    # Database (leverages postgrest-py under the hood)\n    from st_supabase_connection import execute_query\n    users = execute_query(\n        st_supabase.table(\"users\").select(\"name, email\").order(\"created_at\", desc=True),\n        ttl=\"15m\",\n    )\n\n    # Auth (cached helper)\n    st_supabase.cached_sign_in_with_password({\"email\": email, \"password\": password})\n    ```\n\n## :ok_hand: Supported methods\n\n<details close>\n<summary> Storage </summary>\n<ul>\n    <li> <code>delete_bucket()</code> </li>\n    <li> <code>empty_bucket()</code> </li>\n    <li> <code>get_bucket()</code> </li>\n    <li> <code>list_buckets()</code> </li>\n    <li> <code>create_bucket()</code> </li>\n    <li> <code>upload()</code> </li>\n    <li> <code>download()</code> </li>\n    <li> <code>update_bucket()</code> </li>\n    <li> <code>move()</code> </li>\n    <li> <code>list_objects()</code> </li>\n    <li> <code>create_signed_urls()</code> </li>\n    <li> <code>get_public_url()</code> </li>\n    <li> <code>create_signed_upload_url()</code> </li>\n    <li> <code>upload_to_signed_url()</code> </li>\n</ul>\n\n</details>\n\n<details close>\n<summary> Database </summary>\n<ul>\n    <li> <code>execute_query()</code> - Executes the passed query with caching enabled. </li>\n    <li> All methods supported by <a href=\"https://postgrest-py.readthedocs.io/en/latest/api/request_builders.html\">postgrest-py</a>.\n</details>\n\n<details>\n<summary> Auth </summary>\n<ul>\n    <li> <code>cached_sign_in_with_password()</code> - Cached version of <code>sign_in_with_password()</code> for faster sign-in. </li>\n    <li> All methods supported by <a href=\"https://supabase.com/docs/reference/python/auth-signup\">Supabase's Python API </a>.\n</details>\n\n## :books: Examples\n\n### :package: Storage operations\n\n#### List existing buckets\n\n```python\n>>> st_supabase_client.list_buckets(ttl=None)\n[\n    SyncBucket(\n        id=\"bucket1\",\n        name=\"bucket1\",\n        owner=\"\",\n        public=False,\n        created_at=datetime.datetime(2023, 7, 31, 19, 56, 21, 518438, tzinfo=tzutc()),\n        updated_at=datetime.datetime(2023, 7, 31, 19, 56, 21, 518438, tzinfo=tzutc()),\n        file_size_limit=None,\n        allowed_mime_types=None,\n    ),\n    SyncBucket(\n        id=\"bucket2\",\n        name=\"bucket2\",\n        owner=\"\",\n        public=True,\n        created_at=datetime.datetime(2023, 7, 31, 19, 56, 28, 203536, tzinfo=tzutc()),\n        updated_at=datetime.datetime(2023, 7, 31, 19, 56, 28, 203536, tzinfo=tzutc()),\n        file_size_limit=100,\n        allowed_mime_types=[\"image/jpg\", \"image/png\"],\n    ),\n]\n```\n\n#### Create a bucket\n\n```python\n>>> st_supabase_client.create_bucket(\"new_bucket\")\n{'name': 'new_bucket'}\n```\n\n#### Get bucket details\n\n```python\n>>> st_supabase_client.get_bucket(\"new_bucket\")\nSyncBucket(id='new_bucket', name='new_bucket', owner='', public=True, created_at=datetime.datetime(2023, 8, 2, 19, 41, 44, 810000, tzinfo=tzutc()), updated_at=datetime.datetime(2023, 8, 2, 19, 41, 44, 810000, tzinfo=tzutc()), file_size_limit=None, allowed_mime_types=None)\n```\n\n#### Update a bucket\n\n```python\n>>> st_supabase_client.update_bucket(\n      \"new_bucket\",\n      file_size_limit=100,\n      allowed_mime_types=[\"image/jpg\", \"image/png\"],\n      public=True,\n    )\n{'message': 'Successfully updated'}\n```\n\n#### Move files in a bucket\n\n```python\n>>> st_supabase_client.move(\"new_bucket\", \"test.png\", \"folder1/new_test.png\")\n{'message': 'Successfully moved'}\n```\n\n#### List objects in a bucket\n\n```python\n>>> st_supabase_client.list_objects(\"new_bucket\", path=\"folder1\", ttl=0)\n[\n    {\n        \"name\": \"new_test.png\",\n        \"id\": \"e506920e-2834-440e-85f1-1d5476927582\",\n        \"updated_at\": \"2023-08-02T19:53:22.53986+00:00\",\n        \"created_at\": \"2023-08-02T19:52:20.404391+00:00\",\n        \"last_accessed_at\": \"2023-08-02T19:53:21.833+00:00\",\n        \"metadata\": {\n            \"eTag\": '\"814a0034f5549e957ee61360d87457e5\"',\n            \"size\": 473831,\n            \"mimetype\": \"image/png\",\n            \"cacheControl\": \"max-age=3600\",\n            \"lastModified\": \"2023-08-02T19:53:23.000Z\",\n            \"contentLength\": 473831,\n            \"httpStatusCode\": 200,\n        },\n    }\n]\n```\n\n#### Empty a bucket\n\n```python\n>>> st_supabase_client.empty_bucket(\"new_bucket\")\n{'message': 'Successfully emptied'}\n```\n\n#### Delete a bucket\n\n```python\n>>> st_supabase_client.delete_bucket(\"new_bucket\")\n{'message': 'Successfully deleted'}\n```\n\n### :file_cabinet: Database operations\n\n#### Simple query\n\n```python\n>>> execute_query(st_supabase_client.table(\"countries\").select(\"*\"), ttl=0)\nAPIResponse(\n    data=[\n        {\"id\": 1, \"name\": \"Afghanistan\"},\n        {\"id\": 2, \"name\": \"Albania\"},\n        {\"id\": 3, \"name\": \"Algeria\"},\n    ],\n    count=None,\n)\n```\n\n#### Query with join\n\n```python\n>>> execute_query(\n        st_supabase_client.table(\"users\").select(\"name, teams(name)\", count=\"exact\"),\n        ttl=\"1h\",\n    )\n\nAPIResponse(\n    data=[\n        {\"name\": \"Kiran\", \"teams\": [{\"name\": \"Green\"}, {\"name\": \"Blue\"}]},\n        {\"name\": \"Evan\", \"teams\": [{\"name\": \"Blue\"}]},\n    ],\n    count=2,\n)\n```\n\n#### Filter through foreign tables\n\n```python\n>>> execute_query(\n        st_supabase_client.table(\"cities\").select(\"name, countries(*)\", count=\"exact\").eq(\"countries.name\", \"Cura\u00e7ao\"),\n        ttl=None,\n    )\n\nAPIResponse(\n    data=[\n        {\n            \"name\": \"Kralendijk\",\n            \"countries\": {\n                \"id\": 2,\n                \"name\": \"Cura\u00e7ao\",\n                \"iso2\": \"CW\",\n                \"iso3\": \"CUW\",\n                \"local_name\": None,\n                \"continent\": None,\n            },\n        },\n        {\"name\": \"Willemstad\", \"countries\": None},\n    ],\n    count=2,\n)\n```\n\n#### Insert rows\n\n```python\n>>> execute_query(\n        st_supabase_client.table(\"countries\").insert(\n            [{\"name\": \"Wakanda\", \"iso2\": \"WK\"}, {\"name\": \"Wadiya\", \"iso2\": \"WD\"}], count=\"None\"\n        ),\n        ttl=0,\n    )\n\nAPIResponse(\n    data=[\n        {\n            \"id\": 250,\n            \"name\": \"Wakanda\",\n            \"iso2\": \"WK\",\n            \"iso3\": None,\n            \"local_name\": None,\n            \"continent\": None,\n        },\n        {\n            \"id\": 251,\n            \"name\": \"Wadiya\",\n            \"iso2\": \"WD\",\n            \"iso3\": None,\n            \"local_name\": None,\n            \"continent\": None,\n        },\n    ],\n    count=None,\n)\n```\n\n### :lock: Auth operations\n\n> [!NOTE]  \n> If the call is valid, all Supabase Auth methods return the same response structure:\n>\n> ```json\n> {\n>   \"user\": {\n>     \"id\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\",\n>     \"app_metadata\": {\n>       \"provider\": \"email\",\n>       \"providers\": [\"email\"]\n>     },\n>     \"user_metadata\": {\n>       \"attribution\": \"I made it :)\",\n>       \"fname\": \"Siddhant\"\n>     },\n>     \"aud\": \"authenticated\",\n>     \"confirmation_sent_at\": null,\n>     \"recovery_sent_at\": null,\n>     \"email_change_sent_at\": null,\n>     \"new_email\": null,\n>     \"invited_at\": null,\n>     \"action_link\": null,\n>     \"email\": \"test.user@abc.com\",\n>     \"phone\": \"\",\n>     \"created_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 365359, tzinfo=datetime.timezone.utc)\",\n>     \"confirmed_at\": null,\n>     \"email_confirmed_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 373966, tzinfo=datetime.timezone.utc)\",\n>     \"phone_confirmed_at\": null,\n>     \"last_sign_in_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 377070, tzinfo=datetime.timezone.utc)\",\n>     \"role\": \"authenticated\",\n>     \"updated_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 381584, tzinfo=datetime.timezone.utc)\",\n>     \"identities\": [\n>       {\n>         \"id\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\",\n>         \"user_id\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\",\n>         \"identity_data\": {\n>           \"email\": \"siddhant.sadangi@gmail.com\",\n>           \"sub\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\"\n>         },\n>         \"provider\": \"email\",\n>         \"created_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)\",\n>         \"last_sign_in_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 370002, tzinfo=datetime.timezone.utc)\",\n>         \"updated_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)\"\n>       }\n>     ],\n>     \"factors\": null\n>   },\n>   \"session\": {\n>     \"provider_token\": null,\n>     \"provider_refresh_token\": null,\n>     \"access_token\": \"***\",\n>     \"refresh_token\": \"***\",\n>     \"expires_in\": 3600,\n>     \"expires_at\": 1696800390,\n>     \"token_type\": \"bearer\",\n>     \"user\": {\n>       \"id\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\",\n>       \"app_metadata\": {\n>         \"provider\": \"email\",\n>         \"providers\": [\"email\"]\n>       },\n>       \"user_metadata\": {\n>         \"attribution\": \"I made it :)\",\n>         \"fname\": \"Siddhant\"\n>       },\n>       \"aud\": \"authenticated\",\n>       \"confirmation_sent_at\": null,\n>       \"recovery_sent_at\": null,\n>       \"email_change_sent_at\": null,\n>       \"new_email\": null,\n>       \"invited_at\": null,\n>       \"action_link\": null,\n>       \"email\": \"test.user@abc.com\",\n>       \"phone\": \"\",\n>       \"created_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 365359, tzinfo=datetime.timezone.utc)\",\n>       \"confirmed_at\": null,\n>       \"email_confirmed_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 373966, tzinfo=datetime.timezone.utc)\",\n>       \"phone_confirmed_at\": null,\n>       \"last_sign_in_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 377070, tzinfo=datetime.timezone.utc)\",\n>       \"role\": \"authenticated\",\n>       \"updated_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 381584, tzinfo=datetime.timezone.utc)\",\n>       \"identities\": [\n>         {\n>           \"id\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\",\n>           \"user_id\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\",\n>           \"identity_data\": {\n>             \"email\": \"siddhant.sadangi@gmail.com\",\n>             \"sub\": \"e1f550fd-9cd1-44e4-bbe4-c04e91cf5544\"\n>           },\n>           \"provider\": \"email\",\n>           \"created_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)\",\n>           \"last_sign_in_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 370002, tzinfo=datetime.timezone.utc)\",\n>           \"updated_at\": \"datetime.datetime(2023, 10, 8, 20, 26, 30, 370040, tzinfo=datetime.timezone.utc)\"\n>         }\n>       ],\n>       \"factors\": null\n>     }\n>   }\n> }\n> ```\n>\n> </details>\n\n#### Create new user\n\n```python\nst_supabase_client.auth.sign_up(\n    dict(\n        email='test.user@abc.com',\n        password='***',\n        options=dict(\n            data=dict(\n                fname='Siddhant',\n                attribution='I made it :)',\n            )\n        )\n    )\n)\n```\n\n#### Sign in with password\n\n`SupabaseConnection()` offers a cached version of `sign_in_with_password()` for faster, request-free sign-ins.\n\n```python\nst_supabase_client.cached_sign_in_with_password(dict(email='test.user@abc.com', password='***'))\n```\n\n#### Retrieve session\n\n```python\nst_supabase_client.auth.get_session()\n```\n\n#### Retrieve user\n\n```python\nst_supabase_client.auth.get_user()\n```\n\n#### Sign out\n\n```python\nst_supabase_client.auth.sign_out()\n```\n\n> [!NOTE]  \n> Check the [Supabase Python API reference](https://supabase.com/docs/reference/python/select) for more examples.\n\n## :star: Explore all options in a demo app\n\n[![Open in Streamlit](https://static.streamlit.io/badges/streamlit_badge_black_white.svg)](https://st-supabase-connection.streamlit.app/)\n\n## :bow: Acknowledgements\n\nThis connector builds upon the awesome work done by the open-source community in general and the [Supabase Community](https://github.com/supabase-community) in particular. I cannot be more thankful to all the authors whose work I have used either directly or indirectly.\n\nThanks to all contributors to this project :bow:\n\n<p align=\"center\">\n    <a href=\"https://github.com/SiddhantSadangi/st_supabase_connection/graphs/contributors\">\n        <img src=\"https://contrib.rocks/image?repo=SiddhantSadangi/st_supabase_connection\" alt=\"Contributors\" style=\"height: 60px !important;width: 217px !important;\">\n    </a>\n</p>\n\n## :hugs: Want to support my work?\n\n<p align=\"center\">\n    <a href=\"https://www.buymeacoffee.com/siddhantsadangi\" target=\"_blank\"><img src=\"https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png\" alt=\"Buy Me A Coffee\" style=\"height: 60px !important;\">\n    </a>\n    <br>\n    <a href=\"https://github.com/sponsors/SiddhantSadangi\" target=\"_blank\"><img src=\"https://img.shields.io/badge/Sponsor%20me%20on-GitHub-f34b7d?logo=github&style=flat\" alt=\"Sponsor me on GitHub\" style=\"height: 28px !important;\">\n</p>\n",
    "bugtrack_url": null,
    "license": null,
    "summary": "A Streamlit connection component for Supabase.",
    "version": "2.1.3",
    "project_urls": {
        "Documentation": "https://github.com/SiddhantSadangi/st_supabase_connection/blob/main/README.md",
        "Funding": "https://www.buymeacoffee.com/siddhantsadangi",
        "Homepage": "https://github.com/SiddhantSadangi/st_supabase_connection"
    },
    "split_keywords": [
        "streamlit",
        " supabase",
        " connection",
        " integration"
    ],
    "urls": [
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "f861713b2a4d5063682622f46a128ef0e3eb6833d8a4492b06e4d974dfb57f53",
                "md5": "3258415365b7b159e792414689b25f58",
                "sha256": "6f6cc77c84d5f93e33027207ce0fab9c8c571a832dbf98a8ecaf101ed219e352"
            },
            "downloads": -1,
            "filename": "st_supabase_connection-2.1.3-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "3258415365b7b159e792414689b25f58",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.10",
            "size": 13042,
            "upload_time": "2025-11-02T19:10:59",
            "upload_time_iso_8601": "2025-11-02T19:10:59.403111Z",
            "url": "https://files.pythonhosted.org/packages/f8/61/713b2a4d5063682622f46a128ef0e3eb6833d8a4492b06e4d974dfb57f53/st_supabase_connection-2.1.3-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": null,
            "digests": {
                "blake2b_256": "e2b0f7ce9add868cee6188e036406e7b81c1dd258958e535a7d7e052a8ec43f7",
                "md5": "b59ac4565fc8632be220d4b14078f179",
                "sha256": "b1c560afdce8ccbf1ea70e2aa22425fb1a7788d75fb8c919d0b4fc3499e5c583"
            },
            "downloads": -1,
            "filename": "st_supabase_connection-2.1.3.tar.gz",
            "has_sig": false,
            "md5_digest": "b59ac4565fc8632be220d4b14078f179",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.10",
            "size": 18766,
            "upload_time": "2025-11-02T19:11:00",
            "upload_time_iso_8601": "2025-11-02T19:11:00.402754Z",
            "url": "https://files.pythonhosted.org/packages/e2/b0/f7ce9add868cee6188e036406e7b81c1dd258958e535a7d7e052a8ec43f7/st_supabase_connection-2.1.3.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2025-11-02 19:11:00",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "SiddhantSadangi",
    "github_project": "st_supabase_connection",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "st-supabase-connection"
}
        
Elapsed time: 1.45007s