fix templates after the db mod

This commit is contained in:
deflax 2018-01-25 02:00:06 +02:00
parent 5fbfe5f33a
commit 426143a5cc
9 changed files with 22 additions and 260 deletions

View file

@ -7,7 +7,7 @@ from .forms import ChargeForm, Addr2PoolForm
from .. import db from .. import db
from ..email import send_email from ..email import send_email
from ..models import User, Transaction, Server, Deployment, Service, Region, Bridge, Router, Address, Domain, contact_proxmaster from ..models import User, Transaction, Server, Deployment, Service, Region, Address, Domain, contact_proxmaster
from ..decorators import admin_required, permission_required from ..decorators import admin_required, permission_required
import base64 import base64
@ -94,8 +94,7 @@ def charge(user_pid=0):
@admin_required @admin_required
def list_addresses(): def list_addresses():
alladdresses = Address.query.order_by(Address.ip.asc()).all() alladdresses = Address.query.order_by(Address.ip.asc()).all()
allrouters = Router.query.all() return render_template('admin/list_addresses.html', addresses=alladdresses)
return render_template('admin/list_addresses.html', addresses=alladdresses, routers=allrouters)
@admin.route("/addr2pool", methods=['GET', 'POST']) @admin.route("/addr2pool", methods=['GET', 'POST'])
@fresh_login_required @fresh_login_required
@ -161,16 +160,11 @@ def dashboard(user_pid=0):
cuser = User.query.filter_by(pid=user_pid).first() cuser = User.query.filter_by(pid=user_pid).first()
inv_deployments = cuser.inv_deployments.filter_by(deleted=False).order_by(Deployment.date_created.desc()).all() inv_deployments = cuser.inv_deployments.filter_by(deleted=False).order_by(Deployment.date_created.desc()).all()
inv_legacy = cuser.inv_deployments.filter_by(deleted=False).filter_by(bridge_id=None).order_by(Deployment.date_created.desc()).all()
inv_deploycubeids = [] inv_deploycubeids = []
inv_deployments_list = [] inv_deployments_list = []
for invcls in inv_deployments: for invcls in inv_deployments:
inv_deploycubeids.extend([invcls.machine_id]) inv_deploycubeids.extend([invcls.machine_id])
inv_deployments_list.extend([invcls.machine_alias]) inv_deployments_list.extend([invcls.machine_alias])
for invcls in inv_legacy:
if invcls.user_id == cuser.pid:
inv_deploycubeids.extend([invcls.machine_id])
inv_services = cuser.inv_services.filter_by(deleted=False).order_by(Service.date_last_charge.asc()).all() inv_services = cuser.inv_services.filter_by(deleted=False).order_by(Service.date_last_charge.asc()).all()
inv_services_list = [] inv_services_list = []
@ -187,11 +181,6 @@ def dashboard(user_pid=0):
for invcls in inv_addresses: for invcls in inv_addresses:
inv_addresses_list.extend([invcls.ip]) inv_addresses_list.extend([invcls.ip])
inv_bridges = cuser.inv_bridges.order_by(Bridge.bridge_id.asc()).all()
inv_bridges_list = []
for invcls in inv_bridges:
inv_bridges_list.extend([invcls.bridge_id])
sys_regions = Region.query.all() sys_regions = Region.query.all()
regions = {} regions = {}
for region in sys_regions: for region in sys_regions:
@ -216,7 +205,7 @@ def dashboard(user_pid=0):
rrd[unit_id][graph] = base64.b64encode(raw).decode() rrd[unit_id][graph] = base64.b64encode(raw).decode()
status = { unit_id : query['status'] } status = { unit_id : query['status'] }
statuses.update(status) statuses.update(status)
current_app.logger.info('[{}] Enabled deployments: {}, services: {}, domains: {}, bridges: {}, addresses: {}'.format(cuser.email, inv_deployments_list, inv_services_list, inv_domains_list, inv_bridges_list, inv_addresses_list)) current_app.logger.info('[{}] Enabled deployments: {}, services: {}, domains: {}, addresses: {}'.format(cuser.email, inv_deployments_list, inv_services_list, inv_domains_list, inv_addresses_list))
return render_template('panel/dashboard.html', rrd=rrd, status=statuses, inv_deployments=inv_deployments, inv_legacy=inv_legacy, inv_services=inv_services, inv_domains=inv_domains, inv_bridges=inv_bridges, inv_addresses=inv_addresses, regions=regions) return render_template('panel/dashboard.html', rrd=rrd, status=statuses, inv_deployments=inv_deployments, inv_services=inv_services, inv_domains=inv_domains, inv_addresses=inv_addresses, regions=regions)

View file

@ -5,7 +5,7 @@ from flask_sqlalchemy import get_debug_queries
from . import main from . import main
from .. import db from .. import db
from ..email import send_email from ..email import send_email
from ..models import User, Permission, Server, Deployment, Service, Region, Address, Bridge, Router, Domain, contact_proxmaster from ..models import User, Permission, Server, Deployment, Service, Region, Address, Domain, contact_proxmaster
import base64 import base64

View file

@ -86,18 +86,14 @@ class User(db.Model, UserMixin):
language = db.Column(db.String, default='BG') language = db.Column(db.String, default='BG')
wallet = db.Column(db.Float) wallet = db.Column(db.Float)
currency = db.Column(db.String, default='BGN') currency = db.Column(db.String, default='BGN')
inv_transactions = db.relationship('Transaction', backref='owner', lazy='dynamic')
inv_servers = db.relationship('Server', backref='owner', lazy='dynamic') inv_servers = db.relationship('Server', backref='owner', lazy='dynamic')
inv_routers = db.relationship('Router', backref='owner', lazy='dynamic')
inv_addresses = db.relationship('Address', backref='owner', lazy='dynamic')
inv_bridges = db.relationship('Bridge', backref='owner', lazy='dynamic')
inv_deployments = db.relationship('Deployment', backref='owner', lazy='dynamic') inv_deployments = db.relationship('Deployment', backref='owner', lazy='dynamic')
inv_services = db.relationship('Service', backref='owner', lazy='dynamic') inv_services = db.relationship('Service', backref='owner', lazy='dynamic')
inv_domains = db.relationship('Domain', backref='owner', lazy='dynamic') inv_domains = db.relationship('Domain', backref='owner', lazy='dynamic')
inv_addresses = db.relationship('Address', backref='owner', lazy='dynamic')
inv_transactions = db.relationship('Transaction', backref='owner', lazy='dynamic')
def __init__(self, **kwargs): def __init__(self, **kwargs):
super(User, self).__init__(**kwargs) super(User, self).__init__(**kwargs)

View file

@ -6,7 +6,7 @@ from . import panel
from .forms import OrderForm from .forms import OrderForm
from .. import db from .. import db
from ..email import send_email from ..email import send_email
from ..models import User, Permission, Server, Deployment, Service, Region, Address, Bridge, Router, Domain, contact_proxmaster from ..models import User, Permission, Server, Deployment, Service, Region, Address, Domain, contact_proxmaster
import base64 import base64
@ -31,12 +31,8 @@ def dashboard():
sys_regions = Region.query.all() sys_regions = Region.query.all()
cuser = current_user cuser = current_user
inv_bridges = cuser.inv_bridges.order_by(Bridge.bridge_id.asc()).all()
inv_addresses = cuser.inv_addresses.order_by(Address.ip.asc()).all() inv_addresses = cuser.inv_addresses.order_by(Address.ip.asc()).all()
inv_routers = cuser.inv_routers.order_by(Router.date_created.asc()).all()
inv_deployments = cuser.inv_deployments.filter_by(deleted=False).order_by(Deployment.date_created.desc()).all() inv_deployments = cuser.inv_deployments.filter_by(deleted=False).order_by(Deployment.date_created.desc()).all()
inv_legacy = cuser.inv_deployments.filter_by(deleted=False).filter_by(bridge_id=None).order_by(Deployment.date_created.desc()).all()
regions = {} regions = {}
for region in sys_regions: for region in sys_regions:
regions[region.pid] = region.description regions[region.pid] = region.description
@ -45,9 +41,6 @@ def dashboard():
for invcls in inv_deployments: for invcls in inv_deployments:
if invcls.user_id == cuser.pid: if invcls.user_id == cuser.pid:
inv_deploycubeids.extend([invcls.machine_id]) inv_deploycubeids.extend([invcls.machine_id])
for invcls in inv_legacy:
if invcls.user_id == cuser.pid:
inv_deploycubeids.extend([invcls.machine_id])
inv_services = cuser.inv_services.filter_by(deleted=False).order_by(Service.date_last_charge.asc()).all() inv_services = cuser.inv_services.filter_by(deleted=False).order_by(Service.date_last_charge.asc()).all()
inv_domains = cuser.inv_domains.filter_by(deleted=False).order_by(Domain.date_created.desc()).all() inv_domains = cuser.inv_domains.filter_by(deleted=False).order_by(Domain.date_created.desc()).all()
@ -83,5 +76,5 @@ def dashboard():
send_email(current_app.config['MAIL_USERNAME'], 'Cube {} is unreachable'.format(unit_id), send_email(current_app.config['MAIL_USERNAME'], 'Cube {} is unreachable'.format(unit_id),
'vmanager/email/adm_unreachable', user=current_user, unit_id=unit_id, error=str(e)) 'vmanager/email/adm_unreachable', user=current_user, unit_id=unit_id, error=str(e))
return render_template('panel/dashboard.html', sys_regions=sys_regions, inv_bridges=inv_bridges, inv_deployments=inv_deployments, inv_legacy=inv_legacy, inv_services=inv_services, inv_domains=inv_domains, inv_addresses=inv_addresses, rrd=rrd, status=statuses, regions=regions) return render_template('panel/dashboard.html', sys_regions=sys_regions, inv_deployments=inv_deployments, inv_services=inv_services, inv_domains=inv_domains, inv_addresses=inv_addresses, rrd=rrd, status=statuses, regions=regions)

View file

@ -17,7 +17,7 @@
<thead> <thead>
<tr> <tr>
<th>IP</th> <th>IP</th>
<th>Asignee</th> <th>Assignee</th>
<th>VLAN</th> <th>VLAN</th>
<th>Slave</th> <th>Slave</th>
<th>Region</th> <th>Region</th>
@ -29,9 +29,15 @@
<tr> <tr>
{% if address.enabled == False %}<tr class="danger">{% else %}<tr>{% endif %} {% if address.enabled == False %}<tr class="danger">{% else %}<tr>{% endif %}
<td>{{ address.ip }}</td> <td>{{ address.ip }}</td>
{% if address.assignee != None %}
<td>{{ address.assignee.machine_alias }}</td> <td>{{ address.assignee.machine_alias }}</td>
<td>{{ address.assignee.vlan }}</td> <td>{{ address.assignee.vlan }}</td>
<td>{{ address.server.name }}</td> <td>{{ address.assignee.server.name }}</td>
{% else %}
<td>None</td>
<td></td>
<td></td>
{% endif %}
<td>{{ address.region.name }}</td> <td>{{ address.region.name }}</td>
<td>{{ address.rdns }}</td> <td>{{ address.rdns }}</td>
{% endfor %} {% endfor %}

View file

@ -139,34 +139,19 @@ addEventListener("DOMContentLoaded", function() {
<div class="row"> <div class="row">
{% for bridge in inv_bridges %} {% if inv_deployments != [] %}
<div class="col-md-12"> <div class="col-md-12">
<div class="panel panel-info" id="deployments"> <div class="panel panel-info" id="deployments">
<div class="panel-heading">Private Network Manager #{{ bridge.bridge_id }}</div> <div class="panel-heading">Datacenter Manager</div>
<div class="panel-body"><p> <div class="panel-body"><p>
<div class="panel-group" id="deploycubes" role="tablist" aria-multiselectable="true"> <div class="panel-group" id="deploycubes" role="tablist" aria-multiselectable="true">
{% for deploy in bridge.inv_deployments %} {% for deploy in inv_deployments %}
{% include "panel/vdc_pool.html" %} {% include "panel/vdc_pool.html" %}
{% endfor %} {% endfor %}
</div> </div>
</div> </div>
</div> </div>
</div> </div>
{% endfor %}
{% if inv_legacy != [] %}
<div class="col-md-12">
<div class="panel panel-info" id="legacy_deployments">
<div class="panel-heading">Network Manager</div>
<div class="panel-body"><p>
<div class="panel-group" id="deploylegacycubes" role="tablist" aria-multiselectable="true">
{% for deploy in inv_legacy %}
{% include "panel/vdc_pool2.html" %}
{% endfor %}
</div>
</div>
</div>
</div>
{% endif %} {% endif %}
{% if inv_services != [] %} {% if inv_services != [] %}

View file

@ -1,105 +0,0 @@
{% block vdc_pool %}
{% if deploy.enabled == False %}
<div class="panel panel-danger" style="margin-top: 2px">
{% else %}
{% if deploy.warning == True %}
<div class="panel panel-warning" style="margin-top: 2px">
{% else %}
<div class="panel panel-success" style="margin-top: 2px">
{% endif %}
{% endif %}
<div class="panel-heading" data-toggle="collapse" data-parent="#deploycubes" href="#cube{{ deploy.machine_id }}" aria-expanded="false" aria-controls="cube{{ deploy.machine_id }}" role="tab" id="dpanel{{ deploy.machine_id }}">
{% if deploy.protected == True %}<img class="icon icons8-Security-Checked" width="24" height="24" src="">{% endif %} <a href="#"><b>{% if status[deploy.machine_id] == 'running' %}<font color="green">{% else %}<font color="red">{% endif %}{{ deploy.machine_alias }}</font></b></a>
</div></div> <!-- end of heading -->
<!-- START OF HIDDEN PANEL -->
{% if deploy.enabled == False or deploy.warning == True %}
<div id="cube{{ deploy.machine_id }}" class="panel-collapse collapse in" role="tabpanel" aria-labelledby="heading{{ deploy.machine_id }}" style="border:3px solid #faebcc; border-top: none; margin-bottom: 10px;">
{% else %}
<div id="cube{{ deploy.machine_id }}" class="panel-collapse collapse" role="tabpanel" aria-labelledby="heading{{ deploy.machine_id }}" style="border:3px solid #d6e9c6; border-top: none; margin-bottom: 10px;">
{% endif %}
<div class="panel-body">
<ul class="nav nav-tabs">
<li class="active"><a data-toggle="tab" href="#controls{{ deploy.machine_id }}">Control</a></li>
<li><a data-toggle="tab" href="#graphs{{ deploy.machine_id }}">Monitoring</a></li>
<li><a data-toggle="tab" href="#misc{{ deploy.machine_id }}">Misc.</a></li>
</ul>
<div class="tab-content">
<div id="controls{{ deploy.machine_id }}" class="tab-pane fade in active">
<p><br />
{% if deploy.enabled == True %}
<img class="icon icons8-Device-Manager" width="32" height="32" src="">
{% if status[deploy.machine_id] == 'running' %}
<button class="confirm command command-shutdown btn btn-default btn-warning" value="shutdown" vmid="{{ deploy.machine_id }}"><span class="glyphicon glyphicon-off" aria-hidden="true"></span> Shutdown</button>
<button class="confirm command command-stop btn btn-default btn-danger" value="stop" vmid="{{ deploy.machine_id }}"><span class="glyphicon glyphicon-alert" aria-hidden="true"></span> Force Stop</button>
</p>
<p>
<img class="icon icons8-Remote-Control" width="32" height="32" src="">
<button class="btn btn-default btn-info" onclick="window.open('/vmanager/command/vmvnc/{{ deploy.machine_id }}', '_blank');"><span class="glyphicon glyphicon-console" aria-hidden="true"></span> Console</button>
{% else %}
<button class="command command-start btn btn-default btn-success" value="start" vmid="{{ deploy.machine_id }}"><span class="glyphicon glyphicon-play" aria-hidden="true"></span> Start</button>
{% endif %}
{% endif %}
</p>
<p>
<img class="icon icons8-Time" width="32" height="32" src="">
{% if deploy.enabled == False %}
<button class="btn btn-default btn-success" onclick="window.open('/vmanager/activate/{{ deploy.machine_id }}','_self');"><span class="glyphicon glyphicon-bell" aria-hidden="true"></span> Activate</button></td>
{% else %}
{% if deploy.warning == True %}
<button class="btn btn-default btn-success" onclick="window.open('/vmanager/activate/{{ deploy.machine_id }}');"><span class="glyphicon glyphicon-bell" aria-hidden="true"></span> Activate ({{ deploy.daysleft }} days left)</button></td>
{% else %}
<td data-title="Time Left">{{ deploy.daysleft }} day(s)</td>
{% endif %}
{% endif %}
</p>
</div>
<div id="graphs{{ deploy.machine_id }}" class="tab-pane fade">
<p>
<div class="row">
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['cpu'] }}'>
<img class="icon icons8-Electronics" width="32" height="32" src="">
Processor {{ deploy.machine_cpu }} {% if deploy.machine_cpu > 1 %}cores{% else %}core{% endif %}
</div>
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['mem'] }}'>
<img class="icon icons8-Memory-Slot" width="32" height="32" src="">
Allocated memory {{ deploy.machine_mem }} MB</div>
</div>
<br />
<div class="row">
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['net'] }}'>
<img class="icon icons8-Flow-Chart" width="32" height="32" src="">
Network traffic
</div>
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['hdd'] }}'>
<img class="icon icons8-SSD" width="32" height="32" src="">
IOPS of {{ deploy.machine_hdd }} GB storage</div>
</div>
</p>
</div>
<div id="misc{{ deploy.machine_id }}" class="tab-pane fade">
<br />
<p>
Unit: {{ deploy.machine_id }}<br />
State: {{ status[deploy.machine_id] }}<br />
Protected: {{ deploy.protected }}<br />
</p>
</div>
</div>
</div>
</div>
<!-- END OF HIDDEN PANEL -->
<!--Cloud Storage icon by Icons8 -->
{% if inv_deployments != [] %}
<!-- TODO: Create new deployment within this pool! -->
{% else %}
<button class="btn btn-default btn-lg btn-block" onclick="window.open('{{ url_for('vmanager.createvm') }}','_self')"><img class="icon icons8-Cloud-Storage" width="48" height="48" src=""> Create</button>
{% endif %}
{% endblock %}

View file

@ -1,102 +0,0 @@
<div class="panel-group" id="deploycubes" role="tablist" aria-multiselectable="true">
{% for deploy in inv_deployments %}
{% if deploy.enabled == False %}
<div class="panel panel-danger" style="margin-top: 2px">
{% else %}
{% if deploy.warning == True %}
<div class="panel panel-warning" style="margin-top: 2px">
{% else %}
<div class="panel panel-success" style="margin-top: 2px">
{% endif %}
{% endif %}
<div class="panel-heading" data-toggle="collapse" data-parent="#deploycubes" href="#cube{{ deploy.machine_id }}" aria-expanded="false" aria-controls="cube{{ deploy.machine_id }}" role="tab" id="dpanel{{ deploy.machine_id }}">
{% if deploy.protected == True %}<img class="icon icons8-Security-Checked" width="24" height="24" src="">{% endif %} <a href="#"><b>{% if status[deploy.machine_id] == 'running' %}<font color="green">{% else %}<font color="red">{% endif %}{{ deploy.machine_alias }}</font></b></a>
</div></div> <!-- end of heading -->
<!-- START OF HIDDEN PANEL -->
{% if deploy.enabled == False or deploy.warning == True %}
<div id="cube{{ deploy.machine_id }}" class="panel-collapse collapse in" role="tabpanel" aria-labelledby="heading{{ deploy.machine_id }}" style="border:3px solid #faebcc; border-top: none; margin-bottom: 10px;">
{% else %}
<div id="cube{{ deploy.machine_id }}" class="panel-collapse collapse" role="tabpanel" aria-labelledby="heading{{ deploy.machine_id }}" style="border:3px solid #d6e9c6; border-top: none; margin-bottom: 10px;">
{% endif %}
<div class="panel-body">
<ul class="nav nav-tabs">
<li class="active"><a data-toggle="tab" href="#controls{{ deploy.machine_id }}">Control</a></li>
<li><a data-toggle="tab" href="#graphs{{ deploy.machine_id }}">Monitoring</a></li>
<li><a data-toggle="tab" href="#misc{{ deploy.machine_id }}">Misc.</a></li>
</ul>
<div class="tab-content">
<div id="controls{{ deploy.machine_id }}" class="tab-pane fade in active">
<p><br />
{% if deploy.enabled == True %}
<img class="icon icons8-Device-Manager" width="32" height="32" src="">
{% if status[deploy.machine_id] == 'running' %}
<button class="confirm command command-shutdown btn btn-default btn-warning" value="shutdown" vmid="{{ deploy.machine_id }}"><span class="glyphicon glyphicon-off" aria-hidden="true"></span> Shutdown</button>
<button class="confirm command command-stop btn btn-default btn-danger" value="stop" vmid="{{ deploy.machine_id }}"><span class="glyphicon glyphicon-alert" aria-hidden="true"></span> Force Stop</button>
</p>
<p>
<img class="icon icons8-Remote-Control" width="32" height="32" src="">
<button class="btn btn-default btn-info" onclick="window.open('/vmanager/command/vmvnc/{{ deploy.machine_id }}', '_blank');"><span class="glyphicon glyphicon-console" aria-hidden="true"></span> Console</button>
{% else %}
<button class="command command-start btn btn-default btn-success" value="start" vmid="{{ deploy.machine_id }}"><span class="glyphicon glyphicon-play" aria-hidden="true"></span> Start</button>
{% endif %}
{% endif %}
</p>
<p>
<img class="icon icons8-Time" width="32" height="32" src="">
{% if deploy.enabled == False %}
<button class="btn btn-default btn-success" onclick="window.open('/vmanager/activate/{{ deploy.machine_id }}','_self');"><span class="glyphicon glyphicon-bell" aria-hidden="true"></span> Activate</button></td>
{% else %}
{% if deploy.warning == True %}
<button class="btn btn-default btn-success" onclick="window.open('/vmanager/activate/{{ deploy.machine_id }}');"><span class="glyphicon glyphicon-bell" aria-hidden="true"></span> Activate ({{ deploy.daysleft }} days left)</button></td>
{% else %}
<td data-title="Time Left">{{ deploy.daysleft }} day(s)</td>
{% endif %}
{% endif %}
</p>
</div>
<div id="graphs{{ deploy.machine_id }}" class="tab-pane fade">
<p>
<div class="row">
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['cpu'] }}'>
<img class="icon icons8-Electronics" width="32" height="32" src="">
Processor {{ deploy.machine_cpu }} {% if deploy.machine_cpu > 1 %}cores{% else %}core{% endif %}
</div>
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['mem'] }}'>
<img class="icon icons8-Memory-Slot" width="32" height="32" src="">
Allocated memory {{ deploy.machine_mem }} MB</div>
</div>
<br />
<div class="row">
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['net'] }}'>
<img class="icon icons8-Flow-Chart" width="32" height="32" src="">
Network traffic
</div>
<div class="col-md-6"><img class="img-responsive" src='data:image/png;base64,{{ rrd[deploy.machine_id]['hdd'] }}'>
<img class="icon icons8-SSD" width="32" height="32" src="">
IOPS of {{ deploy.machine_hdd }} GB storage</div>
</div>
</p>
</div>
<div id="misc{{ deploy.machine_id }}" class="tab-pane fade">
<br />
<p>
Unit: {{ deploy.machine_id }}<br />
State: {{ status[deploy.machine_id] }}<br />
Protected: {{ deploy.protected }}<br />
</p>
</div>
</div>
</div>
</div>
<!-- END OF HIDDEN PANEL -->
{% endfor %}
</div>

View file

@ -6,7 +6,7 @@ from . import vmanager
from .forms import CreateForm, ActivateForm from .forms import CreateForm, ActivateForm
from .. import db from .. import db
from ..email import send_email from ..email import send_email
from ..models import User, Permission, Transaction, Bridge, Router, Deployment, Service, Region, Address, Domain, contact_proxmaster from ..models import User, Permission, Transaction, Deployment, Service, Region, Address, Domain, contact_proxmaster
from ..decorators import admin_required, permission_required from ..decorators import admin_required, permission_required
import base64 import base64