123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- from collections import Counter
- from datetime import datetime
- from nicegui import ui
- from . import doc
- counter = Counter() # type: ignore
- start = datetime.now().strftime(r'%H:%M, %d %B %Y')
- doc.title('Storage')
- @doc.demo('Storage', '''
- NiceGUI offers a straightforward mechanism for data persistence within your application.
- It features five built-in storage types:
- - `app.storage.tab`:
- Stored server-side in memory, this dictionary is unique to each tab session and can hold arbitrary objects.
- Data will be lost when restarting the server until <https://github.com/zauberzeug/nicegui/discussions/2841> is implemented.
- This storage is only available within [page builder functions](/documentation/page)
- and requires an established connection, obtainable via [`await client.connected()`](/documentation/page#wait_for_client_connection).
- - `app.storage.client`:
- Also stored server-side in memory, this dictionary is unique to each client connection and can hold arbitrary objects.
- Data will be discarded when the page is reloaded or the user navigates to another page.
- Unlike data stored in `app.storage.tab` which can be persisted on the server even for days,
- `app.storage.client` helps caching resource-hungry objects such as a streaming or database connection you need to keep alive
- for dynamic site updates but would like to discard as soon as the user leaves the page or closes the browser.
- This storage is only available within [page builder functions](/documentation/page).
- - `app.storage.user`:
- Stored server-side, each dictionary is associated with a unique identifier held in a browser session cookie.
- Unique to each user, this storage is accessible across all their browser tabs.
- `app.storage.browser['id']` is used to identify the user.
- - `app.storage.general`:
- Also stored server-side, this dictionary provides a shared storage space accessible to all users.
- - `app.storage.browser`:
- Unlike the previous types, this dictionary is stored directly as the browser session cookie, shared among all browser tabs for the same user.
- However, `app.storage.user` is generally preferred due to its advantages in reducing data payload, enhancing security, and offering larger storage capacity.
- By default, NiceGUI holds a unique identifier for the browser session in `app.storage.browser['id']`.
- The user storage and browser storage are only available within `page builder functions </documentation/page>`_
- because they are accessing the underlying `Request` object from FastAPI.
- Additionally these two types require the `storage_secret` parameter in`ui.run()` to encrypt the browser session cookie.
- | Storage type | `tab` | `client` | `user` | `general` | `browser` |
- |-----------------------------|--------|----------|--------|-----------|-----------|
- | Location | Server | Server | Server | Server | Browser |
- | Across tabs | No | No | Yes | Yes | Yes |
- | Across browsers | No | No | No | Yes | No |
- | Across server restarts | No | No | No | Yes | No |
- | Across page reloads | Yes | No | Yes | Yes | Yes |
- | Needs page builder function | Yes | Yes | Yes | No | Yes |
- | Needs client connection | Yes | No | No | No | No |
- | Write only before response | No | No | No | No | Yes |
- | Needs serializable data | No | No | Yes | Yes | Yes |
- ''')
- def storage_demo():
- from nicegui import app
- # @ui.page('/')
- # def index():
- # app.storage.user['count'] = app.storage.user.get('count', 0) + 1
- # with ui.row():
- # ui.label('your own page visits:')
- # ui.label().bind_text_from(app.storage.user, 'count')
- #
- # ui.run(storage_secret='private key to secure the browser session cookie')
- # END OF DEMO
- app.storage.user['count'] = app.storage.user.get('count', 0) + 1
- with ui.row():
- ui.label('your own page visits:')
- ui.label().bind_text_from(app.storage.user, 'count')
- @doc.demo('Counting page visits', '''
- Here we are using the automatically available browser-stored session ID to count the number of unique page visits.
- ''')
- def page_visits():
- from collections import Counter
- from datetime import datetime
- from nicegui import app
- # counter = Counter()
- # start = datetime.now().strftime('%H:%M, %d %B %Y')
- #
- # @ui.page('/')
- # def index():
- # counter[app.storage.browser['id']] += 1
- # ui.label(f'{len(counter)} unique views ({sum(counter.values())} overall) since {start}')
- #
- # ui.run(storage_secret='private key to secure the browser session cookie')
- # END OF DEMO
- counter[app.storage.browser['id']] += 1
- ui.label(f'{len(counter)} unique views ({sum(counter.values())} overall) since {start}')
- @doc.demo('Storing UI state', '''
- Storage can also be used in combination with [`bindings`](/documentation/bindings).
- Here we are storing the value of a textarea between visits.
- The note is also shared between all tabs of the same user.
- ''')
- def ui_state():
- from nicegui import app
- # @ui.page('/')
- # def index():
- # ui.textarea('This note is kept between visits') \
- # .classes('w-full').bind_value(app.storage.user, 'note')
- # END OF DEMO
- ui.textarea('This note is kept between visits').classes('w-full').bind_value(app.storage.user, 'note')
- @doc.demo('Storing data per browser tab', '''
- When storing data in `app.storage.tab`, a single user can open multiple tabs of the same app, each with its own storage data.
- This may be beneficial in certain scenarios like search or when performing data analysis.
- It is also more secure to use such a volatile storage for scenarios like logging into a bank account or accessing a password manager.
- ''')
- def tab_storage():
- from nicegui import app
- # @ui.page('/')
- # async def index():
- # await ui.context.client.connected()
- with ui.column(): # HIDE
- app.storage.tab['count'] = app.storage.tab.get('count', 0) + 1
- ui.label(f'Tab reloaded {app.storage.tab["count"]} times')
- ui.button('Reload page', on_click=ui.navigate.reload)
- @doc.demo('Short-term memory', '''
- The goal of `app.storage.client` is to store data only for the duration of the current page visit.
- In difference to data stored in `app.storage.tab`
- - which is persisted between page changes and even browser restarts as long as the tab is kept open -
- the data in `app.storage.client` will be discarded if the user closes the browser, reloads the page or navigates to another page.
- This is beneficial for resource-hungry, intentionally short-lived or sensitive data.
- An example is a database connection, which should be closed as soon as the user leaves the page.
- Additionally, this storage useful if you want to return a page with default settings every time a user reloads.
- Meanwhile, it keeps the data alive during in-page navigation.
- This is also helpful when updating elements on the site at intervals, such as a live feed.
- ''')
- def short_term_memory():
- from nicegui import app
- # @ui.page('/')
- # async def index():
- with ui.column(): # HIDE
- cache = app.storage.client
- cache['count'] = 0
- ui.label().bind_text_from(cache, 'count', lambda n: f'Updated {n} times')
- ui.button('Update content',
- on_click=lambda: cache.update(count=cache['count'] + 1))
- ui.button('Reload page', on_click=ui.navigate.reload)
|