Source code for h2o_dashboard.wave_auth
import asyncio
import json
import os
import dotenv
import requests
from h2o_wave import Q, app, ui, on, data, run_on, AsyncSite # noqa F401
from firebase_tools.authenticate import authenticate_with_firebase, check_str_token_validity
from h2o_dashboard.pages.documentation_page import documentation_page
from h2o_dashboard.pages.okx_dashbaord_page.okx_dashboard_page import okx_dashboard_page
from h2o_dashboard.pages.overview_page.overview_page import overview_page
from h2o_dashboard.util import add_card, clear_cards
dotenv.load_dotenv(dotenv.find_dotenv())
FIREBASE_CONFIG = {
"apiKey": os.environ.get("FIREBASE_API_KEY"),
}
assert FIREBASE_CONFIG["apiKey"], "FIREBASE_API_KEY environment variable not set!"
[docs]
async def init(q: Q) -> None:
"""
Q Page Meta (meta_card) Arguments:
box
A string indicating how to place this component on the page.
title
The title of the page.
refresh
Refresh rate in seconds. A value of 0 turns off live-updates. Values != 0 are currently ignored (reserved for future use).
notification
Display a desktop notification.
notification_bar
Display an in-app notification bar.
redirect
Redirect the page to a new URL.
icon
Shortcut icon path. Preferably a .png file (.ico files may not work in mobile browsers). Not supported in Safari.
layouts
The layouts supported by this page.
dialog
Display a dialog on the page.
side_panel
Display a side panel on the page.
theme
Specify the name of the theme (color scheme) to use on this page. One of 'light', 'neon' or 'h2o-dark'.
themes
Themes (color schemes) that define color used in the app.
tracker
Configure a tracker for the page (for web analytics).
scripts
External Javascript files to load into the page.
script
Javascript code to execute on this page.
stylesheet
CSS stylesheet to be applied to this page.
stylesheets
External CSS files to load into the page.
commands
Contextual menu commands for this component.
"""
# Static Business Website
# index_file = open('static/html/index.html', 'r').read()
q.page['meta'] = ui.meta_card(box='',
title='AntBot',
layouts=[ui.layout(breakpoint='xs', min_height='100vh', name='default',
zones=[
ui.zone('main', size='1', direction=ui.ZoneDirection.ROW, zones=[
ui.zone('sidebar', size='180px'),
ui.zone('body', zones=[
ui.zone('header'),
ui.zone('content', zones=[
# Specify various zones and use the one that is currently needed. Empty zones are ignored.
ui.zone('first_context', size='0 0 1 4',
direction=ui.ZoneDirection.ROW,
zones=[
ui.zone('first_context_1', size='1 4 0 0'),
ui.zone('first_context_2', size='1 4 0 0'),
ui.zone('first_context_3', size='1 4 0 0'),
ui.zone('first_context_4', size='1 4 0 0'),
]),
ui.zone('details', size='4 4 4 4'),
ui.zone('horizontal', size='1', direction=ui.ZoneDirection.ROW),
ui.zone('centered', size='1 1 1 1', align='center'),
ui.zone('vertical', size='1'),
# Wrapping = 'start', 'end', 'center', 'between', 'around', 'stretch'
# Layout = 'start', 'end', 'center', 'between', 'around'
ui.zone('grid_1', direction=ui.ZoneDirection.ROW, wrap='stretch',
justify='center'),
ui.zone('grid_2', direction=ui.ZoneDirection.ROW, wrap='stretch',
justify='start'),
ui.zone('grid_3', direction=ui.ZoneDirection.ROW, wrap='stretch',
justify='center'),
ui.zone('grid_4', direction=ui.ZoneDirection.ROW, wrap='stretch',
justify='center'),
ui.zone('grid_5', direction=ui.ZoneDirection.ROW, wrap='stretch',
justify='center'),
ui.zone('grid_6', direction=ui.ZoneDirection.ROW, wrap='stretch',
justify='center'),
ui.zone('bot_manual_controls',
direction=ui.ZoneDirection.ROW,
# Cover the entire card
wrap='stretch',
),
ui.zone('second_context', size='0 0 1 4',
direction=ui.ZoneDirection.ROW,
zones=[
ui.zone('second_context_1', size='1 4 0 0'),
ui.zone('second_context_2', size='1 4 0 0'),
ui.zone('second_context_3', size='1 4 0 0',
direction=ui.ZoneDirection.ROW,
zones=[
ui.zone('second_context_3_1', size='1 4 0 0'),
ui.zone('second_context_3_2', size='1 4 0 0')
]),
]),
]),
]),
]),
ui.zone('footer', size='0 1 0 0', direction=ui.ZoneDirection.ROW),
]),
],
tracker=ui.tracker(type=ui.TrackerType.GA, id='G-Q21Z3V0G9K'),
themes=[
ui.theme(
name='my-awesome-theme',
primary='#8C1B11', # Header and Sidebaer - Color Light Red
text='#000000', #
card='#ffffff',
page='#F2F2F2',
# page='#D91A1A',
),
ui.theme(
name='my-awesome-theme1',
primary='#571305', # Header and Sidebaer - Color Light Red
text='#E0E0E0', #
card='#333333',
page='#121212',
),
ui.theme(
name='my-awesome-theme2',
primary='#2979FF', # Header and Sidebaer - Color Light Red
text='#E0E0E0', #
card='#212121',
page='#000000',
)
],
theme='my-awesome-theme2'
)
q.client.initialized = False
[docs]
async def initialize_client(q: Q):
q.client.cards = set()
await init(q)
q.client.initialized = True
q.client.token = None # Initially, no token is set.
[docs]
async def render_login_page(q: Q, error_message=None):
"""Render the login page."""
await clear_cards(q)
items = [
ui.message_bar(type='info', text='Please login to continue.', ),
ui.text_xl('Login'),
ui.textbox(name='email', label='Email', required=True),
ui.textbox(name='password', label='Password', required=True, password=True),
ui.buttons([ui.button(name='login', label='Login', primary=True, )]),
]
if error_message:
items.insert(0, ui.message_bar(type='error', text=error_message, ))
await add_card(q, 'login', ui.form_card(box='centered', items=items, ), )
# Lets add decorations to the login page
[docs]
async def add_application_sidebar(q):
image_address = None
# image_address = ''
if not q.args["#"]:
q.args['#'] = 'overview_page'
await add_card(q, 'Application_Sidebar', ui.nav_card(
box='sidebar', color='primary', title='AntBot',
subtitle="just a tiny bot here to help you out",
value=f'#{q.args["#"]}',
# image='https://wave.h2o.ai/img/h2o-logo.svg', items=[]
# Loading from local file rather than url
image=image_address,
items=[
ui.nav_group('', items=[
ui.nav_item(name='#overview_page', label='Overview', icon='Home'),
ui.nav_item(name='#okx_dashboard_page', label='OKX Dashboard', icon='eDiscoveryApp'),
ui.nav_item(name='#documentation_page', label='Documentation', icon='Document'),
]),
# ui.nav_group('Docs', items=[
# ui.nav_item(name='#admin_userdocs_link', label='Usage Documentation', icon='TextDocumentShared'),
# ui.nav_item(name='#admin_devdocs_link', label='Codebase Documentation', icon='TextDocumentEdit'),
# ]),
# ui.nav_group('Account', items=[
# ui.nav_item(name='#admin_account_page', label='Account', icon='AccountManagement'),
# ui.nav_item(name='#admin_plans_page', label='Plans', icon='PaymentCard'),
# ui.nav_item(name='logout', label='Logout', icon='Logout'),
# ]),
],
))
[docs]
async def serve_security(q: Q, bypass_security=False):
response = None
# If logout is triggered, clear token and show login page
if q.args.logout:
q.client.token = None
await render_login_page(q)
return # End the function after logging out
# If login is triggered, try to authenticate and decide what to render next
if q.args.login:
response = await authenticate_with_firebase(q.args.email, q.args.password)
if response['status'] == 'success':
print("Token already exists and is valid")
q.client.token = response['token']
q.client.user_id = response['user_id']
q.client.email = q.args.email
q.client.expires_in = response['expires_in']
q.client.password = q.args.password
await render_login_page(q, error_message=response['error_message'])
else:
q.client.token = None
q.client.user_id = None
q.client.email = None
q.client.expires_in = None
q.client.password = None
await render_login_page(q, error_message=response['error_message'])
return # End the function if login fails
if bypass_security:
q.client.token = 'bypass'
await render_hidden_content(q)
return
# If the client already has a token, check its validity and act accordingly
if q.client.token:
if await check_str_token_validity(q.client.token):
context = response # todo: implement better context handling
await render_hidden_content(q)
else:
q.client.token = None
await render_login_page(q)
else:
await render_login_page(q)