order phase 2
This commit is contained in:
parent
ef57acb3cb
commit
a43d2104ef
22 changed files with 500 additions and 85 deletions
|
@ -38,6 +38,9 @@ moment.init_app(app)
|
|||
from .main import main as main_blueprint
|
||||
app.register_blueprint(main_blueprint)
|
||||
|
||||
from .panel import panel as panel_blueprint
|
||||
app.register_blueprint(panel_blueprint, url_prefix='/panel')
|
||||
|
||||
from .auth import auth as auth_blueprint
|
||||
app.register_blueprint(auth_blueprint, url_prefix='/auth')
|
||||
|
||||
|
|
|
@ -217,6 +217,6 @@ def dashboard(user_pid=0):
|
|||
status = { unit_id : query['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))
|
||||
return render_template('main/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_legacy=inv_legacy, inv_services=inv_services, inv_domains=inv_domains, inv_bridges=inv_bridges, inv_addresses=inv_addresses, regions=regions)
|
||||
|
||||
|
||||
|
|
|
@ -72,7 +72,7 @@ def login():
|
|||
send_email(current_app.config['MAIL_USERNAME'], user.email + ' logged in.', 'auth/email/adm_loginnotify', user=user, ipaddr=lastip )
|
||||
#flash('Last Login: {} from {}'.format(user.last_seen.strftime("%a %d %B %Y %H:%M"), previp))
|
||||
flash('Last Login: {}'.format(user.last_seen.strftime("%a %d %B %Y %H:%M")))
|
||||
return redirect(request.args.get('next') or url_for('main.dashboard'))
|
||||
return redirect(request.args.get('next') or url_for('panel.dashboard'))
|
||||
else:
|
||||
flash('Invalid username or password.')
|
||||
|
||||
|
@ -104,7 +104,7 @@ def twofactor():
|
|||
db.session.add(user)
|
||||
db.session.commit()
|
||||
#send_email(current_app.config['MAIL_USERNAME'], user.email + ' logged in.', 'auth/email/adm_loginnotify', user=user, ipaddr=lastip )
|
||||
return redirect(request.args.get('next') or url_for('main.dashboard'))
|
||||
return redirect(request.args.get('next') or url_for('panel.dashboard'))
|
||||
else:
|
||||
flash('Invalid token.')
|
||||
return render_template('auth/2fa.html', page=page, form=form)
|
||||
|
|
|
@ -19,7 +19,7 @@ def after_request(response):
|
|||
#STATIC PAGES
|
||||
@main.route("/", methods=['GET'])
|
||||
def index():
|
||||
allservers = Server.query.all()
|
||||
allservers = Server.query.filter_by(enabled=True)
|
||||
return render_template('main/index.html', servers=allservers)
|
||||
|
||||
@main.route("/chat", methods=['GET'])
|
||||
|
@ -54,64 +54,3 @@ def market(group_id=0):
|
|||
abort(404)
|
||||
return render_template('main/marketgroup.html', groupname=allgroups[group_id], products=filtered_products)
|
||||
|
||||
#DASHBOARD
|
||||
@main.route("/dashboard", methods=['GET', 'POST'])
|
||||
@login_required
|
||||
def dashboard():
|
||||
sys_regions = Region.query.all()
|
||||
|
||||
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_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_legacy = cuser.inv_deployments.filter_by(deleted=False).filter_by(bridge_id=None).order_by(Deployment.date_created.desc()).all()
|
||||
|
||||
regions = {}
|
||||
for region in sys_regions:
|
||||
regions[region.pid] = region.description
|
||||
|
||||
inv_deploycubeids = []
|
||||
for invcls in inv_deployments:
|
||||
if invcls.user_id == cuser.pid:
|
||||
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_domains = cuser.inv_domains.filter_by(deleted=False).order_by(Domain.date_created.desc()).all()
|
||||
|
||||
#extract rrd and status from the deployments
|
||||
rrd = {}
|
||||
statuses = {}
|
||||
#current_app.logger.warning(str(inv_deploycubeids))
|
||||
for unit_id in inv_deploycubeids:
|
||||
rrd[unit_id] = {}
|
||||
data = { 'unit_id': int(unit_id),
|
||||
'type': 'kvm' }
|
||||
try:
|
||||
query = contact_proxmaster(data, 'vmrrd')
|
||||
except Exception as e:
|
||||
current_app.logger.error(e)
|
||||
flash('Support is notified.'.format(str(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))
|
||||
#current_app.logger.info('debug query:')
|
||||
#current_app.logger.info(query)
|
||||
|
||||
graphs_list = ['net', 'cpu', 'mem', 'hdd']
|
||||
try:
|
||||
for graph in graphs_list:
|
||||
raw = query[graph]['image'].encode('raw_unicode_escape')
|
||||
rrd[unit_id][graph] = base64.b64encode(raw).decode()
|
||||
status = { unit_id : query['status'] }
|
||||
statuses.update(status)
|
||||
except Exception as e:
|
||||
current_app.logger.error(e)
|
||||
flash('Support is notified.'.format(str(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))
|
||||
|
||||
return render_template('main/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)
|
||||
|
||||
|
|
|
@ -249,8 +249,11 @@ class Server(db.Model):
|
|||
user_id = db.Column(db.ForeignKey('users.pid')) #FK
|
||||
region_id = db.Column(db.ForeignKey('regions.pid')) #FK
|
||||
|
||||
enabled = db.Column(db.Boolean)
|
||||
name = db.Column(db.String)
|
||||
description = db.Column(db.String)
|
||||
cpu = db.Column(db.String)
|
||||
mem = db.Column(db.String)
|
||||
hdd = db.Column(db.String)
|
||||
address = db.Column(db.String)
|
||||
|
||||
class Bridge(db.Model):
|
||||
|
|
3
app/panel/__init__.py
Normal file
3
app/panel/__init__.py
Normal file
|
@ -0,0 +1,3 @@
|
|||
from flask import Blueprint
|
||||
panel = Blueprint('panel', __name__)
|
||||
from . import routes
|
14
app/panel/forms.py
Normal file
14
app/panel/forms.py
Normal file
|
@ -0,0 +1,14 @@
|
|||
from flask_wtf import FlaskForm
|
||||
from wtforms import StringField, PasswordField, BooleanField, SubmitField, SelectField, DecimalField
|
||||
from wtforms import validators, ValidationError
|
||||
from wtforms.fields.html5 import EmailField, DecimalRangeField
|
||||
|
||||
from .. import db
|
||||
|
||||
class OrderForm(FlaskForm):
|
||||
cpu = DecimalRangeField('Processor Cores', default=2)
|
||||
memory = DecimalRangeField('Memory', default=512)
|
||||
storage = DecimalRangeField('Storage', default=20)
|
||||
alias = StringField('Machine Alias:', [validators.Regexp(message='ex.: myservice1.com, myservice2.local', regex='^[a-zA-Z0-9][a-zA-Z0-9-_]{0,61}[a-zA-Z0-9]{0,1}\.([a-zA-Z]{1,6}|[a-zA-Z0-9-]{1,30}\.[a-zA-Z]{2,3})$'), validators.Length(6,64)])
|
||||
submit = SubmitField('Create')
|
||||
|
87
app/panel/routes.py
Normal file
87
app/panel/routes.py
Normal file
|
@ -0,0 +1,87 @@
|
|||
from flask import render_template, abort, redirect, url_for, abort, flash, request, current_app, make_response, g
|
||||
from flask_login import login_required, login_user, logout_user, current_user
|
||||
from flask_sqlalchemy import get_debug_queries
|
||||
|
||||
from . import panel
|
||||
from .forms import OrderForm
|
||||
from .. import db
|
||||
from ..email import send_email
|
||||
from ..models import User, Permission, Server, Deployment, Service, Region, Address, Bridge, Router, Domain, contact_proxmaster
|
||||
|
||||
import base64
|
||||
|
||||
@panel.after_app_request
|
||||
def after_request(response):
|
||||
for query in get_debug_queries():
|
||||
if query.duration >= current_app.config['SLOW_DB_QUERY_TIME']:
|
||||
current_app.logger.warning('Slow query: %s\nParameters: %s\nDuration: %fs\nContext: %s\n' % (query.statement, query.parameters, query.duration, query.context))
|
||||
return response
|
||||
|
||||
@panel.route("/order", methods=['GET', 'POST'])
|
||||
def order():
|
||||
form = OrderForm()
|
||||
if form.validate_on_submit():
|
||||
return redirect('main.index')
|
||||
return render_template('panel/order.html', form=form)
|
||||
|
||||
#DASHBOARD
|
||||
@panel.route("/dashboard", methods=['GET', 'POST'])
|
||||
@login_required
|
||||
def dashboard():
|
||||
sys_regions = Region.query.all()
|
||||
|
||||
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_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_legacy = cuser.inv_deployments.filter_by(deleted=False).filter_by(bridge_id=None).order_by(Deployment.date_created.desc()).all()
|
||||
|
||||
regions = {}
|
||||
for region in sys_regions:
|
||||
regions[region.pid] = region.description
|
||||
|
||||
inv_deploycubeids = []
|
||||
for invcls in inv_deployments:
|
||||
if invcls.user_id == cuser.pid:
|
||||
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_domains = cuser.inv_domains.filter_by(deleted=False).order_by(Domain.date_created.desc()).all()
|
||||
|
||||
#extract rrd and status from the deployments
|
||||
rrd = {}
|
||||
statuses = {}
|
||||
#current_app.logger.warning(str(inv_deploycubeids))
|
||||
for unit_id in inv_deploycubeids:
|
||||
rrd[unit_id] = {}
|
||||
data = { 'unit_id': int(unit_id),
|
||||
'type': 'kvm' }
|
||||
try:
|
||||
query = contact_proxmaster(data, 'vmrrd')
|
||||
except Exception as e:
|
||||
current_app.logger.error(e)
|
||||
flash('Support is notified.'.format(str(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))
|
||||
#current_app.logger.info('debug query:')
|
||||
#current_app.logger.info(query)
|
||||
|
||||
graphs_list = ['net', 'cpu', 'mem', 'hdd']
|
||||
try:
|
||||
for graph in graphs_list:
|
||||
raw = query[graph]['image'].encode('raw_unicode_escape')
|
||||
rrd[unit_id][graph] = base64.b64encode(raw).decode()
|
||||
status = { unit_id : query['status'] }
|
||||
statuses.update(status)
|
||||
except Exception as e:
|
||||
current_app.logger.error(e)
|
||||
flash('Support is notified.'.format(str(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))
|
||||
|
||||
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)
|
||||
|
|
@ -77,6 +77,6 @@ def activate(itemid=0):
|
|||
if owner.is_administrator:
|
||||
return redirect(url_for('admin.list_services'))
|
||||
else:
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
return render_template('smanager/activate.html', form=form, service=service, ppm=ppm, total=(ppm * service.period), currency=owner.currency)
|
||||
|
||||
|
|
86
app/static/css/range.css
Normal file
86
app/static/css/range.css
Normal file
|
@ -0,0 +1,86 @@
|
|||
input[type=range] {
|
||||
-webkit-appearance: none;
|
||||
width: 100%;
|
||||
margin: 13.8px 0;
|
||||
}
|
||||
input[type=range]:focus {
|
||||
outline: none;
|
||||
}
|
||||
input[type=range]::-webkit-slider-runnable-track {
|
||||
width: 100%;
|
||||
height: 8.4px;
|
||||
cursor: pointer;
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
background: #3071a9;
|
||||
border-radius: 1.3px;
|
||||
border: 0.2px solid #010101;
|
||||
}
|
||||
input[type=range]::-webkit-slider-thumb {
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
border: 1px solid #000000;
|
||||
height: 36px;
|
||||
width: 16px;
|
||||
border-radius: 3px;
|
||||
background: #f8ffff;
|
||||
cursor: pointer;
|
||||
-webkit-appearance: none;
|
||||
margin-top: -14px;
|
||||
}
|
||||
input[type=range]:focus::-webkit-slider-runnable-track {
|
||||
background: #367ebd;
|
||||
}
|
||||
input[type=range]::-moz-range-track {
|
||||
width: 100%;
|
||||
height: 8.4px;
|
||||
cursor: pointer;
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
background: #3071a9;
|
||||
border-radius: 1.3px;
|
||||
border: 0.2px solid #010101;
|
||||
}
|
||||
input[type=range]::-moz-range-thumb {
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
border: 1px solid #000000;
|
||||
height: 36px;
|
||||
width: 16px;
|
||||
border-radius: 3px;
|
||||
background: #f8ffff;
|
||||
cursor: pointer;
|
||||
}
|
||||
input[type=range]::-ms-track {
|
||||
width: 100%;
|
||||
height: 8.4px;
|
||||
cursor: pointer;
|
||||
background: transparent;
|
||||
border-color: transparent;
|
||||
color: transparent;
|
||||
}
|
||||
input[type=range]::-ms-fill-lower {
|
||||
background: #2a6495;
|
||||
border: 0.2px solid #010101;
|
||||
border-radius: 2.6px;
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
}
|
||||
input[type=range]::-ms-fill-upper {
|
||||
background: #3071a9;
|
||||
border: 0.2px solid #010101;
|
||||
border-radius: 2.6px;
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
}
|
||||
input[type=range]::-ms-thumb {
|
||||
box-shadow: 1px 1px 1px #000000, 0px 0px 1px #0d0d0d;
|
||||
border: 1px solid #000000;
|
||||
height: 36px;
|
||||
width: 16px;
|
||||
border-radius: 3px;
|
||||
background: #f8ffff;
|
||||
cursor: pointer;
|
||||
height: 8.4px;
|
||||
}
|
||||
input[type=range]:focus::-ms-fill-lower {
|
||||
background: #3071a9;
|
||||
}
|
||||
input[type=range]:focus::-ms-fill-upper {
|
||||
background: #367ebd;
|
||||
}
|
||||
|
BIN
app/static/images/server.png
Normal file
BIN
app/static/images/server.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 21 KiB |
|
@ -7,7 +7,7 @@
|
|||
|
||||
<div class="col-md-12">
|
||||
<div class="panel panel-info">
|
||||
<div class="panel-heading">Transactions</div>
|
||||
<div class="panel-heading">Servers</div>
|
||||
<div class="panel-body">
|
||||
|
||||
<div class="table-responsive">
|
||||
|
@ -15,7 +15,9 @@
|
|||
<thead>
|
||||
<tr>
|
||||
<th>Name</th>
|
||||
<th>Description</th>
|
||||
<th>CPU</th>
|
||||
<th>MEM</th>
|
||||
<th>HDD</th>
|
||||
<th>Address</th>
|
||||
<th>Region</th>
|
||||
<th>Seller</th>
|
||||
|
@ -26,7 +28,9 @@
|
|||
{% for server in servers %}
|
||||
<tr class="default">
|
||||
<td>{{ server.name }}</td>
|
||||
<td>{{ server.description }}</td>
|
||||
<td>{{ server.cpu }}</td>
|
||||
<td>{{ server.mem }}</td>
|
||||
<td>{{ server.hdd }}</td>
|
||||
<td>{{ server.address }}</td>
|
||||
<td>{{ server.region.name }}</td>
|
||||
<td><a href="{{ url_for('admin.dashboard', user_pid=server.owner.pid) }}">{{ server.owner.email }}</a></td>
|
||||
|
|
|
@ -16,6 +16,7 @@
|
|||
{{ super() }}
|
||||
<link href="{{ url_for('static', filename='css/navbar.css') }}" rel="stylesheet">
|
||||
<link href="{{ url_for('static', filename='css/style.css') }}" rel="stylesheet">
|
||||
<link href="{{ url_for('static', filename='css/range.css') }}" rel="stylesheet">
|
||||
{% endblock %}
|
||||
|
||||
{% block scripts %}
|
||||
|
|
|
@ -8,7 +8,6 @@
|
|||
|
||||
<div class="footerblock" style="float: right;" align="right">
|
||||
<h4>Contact</h4>
|
||||
<p>52 Volga str.<br>4002 Plovdiv, Bulgaria</p>
|
||||
<p>tel: +359 (0) 32 398 295 <br /> email: <a href="mailto:office@datapoint.bg">office@datapoint.bg</a></p>
|
||||
<p><a href="https://www.facebook.com/datapoint.bg/" target="_blank"><img src="{{ url_for('static', filename='images/fb.png') }}"></a></p>
|
||||
</div>
|
||||
|
|
|
@ -86,6 +86,16 @@
|
|||
<div class="col-md-12">
|
||||
<div class="panel panel-primary panel-transparent">
|
||||
<div class="panel-body">
|
||||
<div class="col-md-8">
|
||||
<img src="../../static/images/server.png" width="80%"><br /><br />
|
||||
<a href="/chat"><button type="button" class="btn btn-lg btn-success">Order VPS</button></a>
|
||||
</div>
|
||||
<div class="col-md-4">
|
||||
Alias: <b>{{ server.name }}</b><br />
|
||||
Processor: <b>{{ server.cpu }}</b><br />
|
||||
Memory: <b>{{ server.mem }}</b><br />
|
||||
Storage: <b>{{ server.hdd }}</b><br />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
@ -108,7 +118,7 @@
|
|||
<img src="../../static/images/VPS-Support.png" width="128" height="128" />
|
||||
<h2 class="media-heading">Поддръжка</h2>
|
||||
<p>Ще Ви помогнем във всички неприятни ситуации, по всяко време. </p>
|
||||
<a href="/chat"><button type="button" class="btn btn-success">Support</button></a>
|
||||
<a href="/chat"><button type="button" class="btn btn-lg btn-success">Support</button></a>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
|
|
@ -11,7 +11,7 @@
|
|||
{% if not current_user.is_authenticated %}
|
||||
<a class="navbar-brand" href="{{ url_for('main.index') }}" rel="home"></a>
|
||||
{% else %}
|
||||
<a class="navbar-brand" href="{{ url_for('main.dashboard') }}" rel="home"></a>
|
||||
<a class="navbar-brand" href="{{ url_for('panel.dashboard') }}" rel="home"></a>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
|
@ -49,7 +49,7 @@
|
|||
<li class="dropdown">
|
||||
<a href="#" class="dropdown-toggle" data-toggle="dropdown" role="button" aria-haspopup="true" aria-expanded="false"><img class="avatar" src="{{ current_user.gravatar(20) }}"> {{ current_user.email }} <span class="caret"></span></a>
|
||||
<ul class="dropdown-menu">
|
||||
<li><a href="{{ url_for('main.dashboard') }}"><span class="glyphicon glyphicon-modal-window"></span> Dashboard</a></li>
|
||||
<li><a href="{{ url_for('panel.dashboard') }}"><span class="glyphicon glyphicon-modal-window"></span> Dashboard</a></li>
|
||||
<li><a href="{{ url_for('uinvoice.transactions') }}"><span class="glyphicon glyphicon-list-alt"></span> Transactions</a></li>
|
||||
<li role="separator" class="divider"></li>
|
||||
<li><a href="{{ url_for('settings.profile') }}"><span class="glyphicon glyphicon-user"></span> Profile</a></li>
|
||||
|
|
|
@ -146,7 +146,7 @@ addEventListener("DOMContentLoaded", function() {
|
|||
<div class="panel-body"><p>
|
||||
<div class="panel-group" id="deploycubes" role="tablist" aria-multiselectable="true">
|
||||
{% for deploy in bridge.inv_deployments %}
|
||||
{% include "main/vdc_pool.html" %}
|
||||
{% include "panel/vdc_pool.html" %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
|
@ -161,7 +161,7 @@ addEventListener("DOMContentLoaded", function() {
|
|||
<div class="panel-body"><p>
|
||||
<div class="panel-group" id="deploylegacycubes" role="tablist" aria-multiselectable="true">
|
||||
{% for deploy in inv_legacy %}
|
||||
{% include "main/vdc_pool2.html" %}
|
||||
{% include "panel/vdc_pool2.html" %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
|
@ -263,7 +263,7 @@ addEventListener("DOMContentLoaded", function() {
|
|||
<!--<img width="32" height="32" src=""> No Public<br />-->
|
||||
|
||||
|
||||
<!--<button class="btn btn-default" onclick="window.open('{{ url_for('main.dashboard') }}','_self')"><span class="glyphicon glyphicon-plus" aria-hiddent="true"></span> Assign</button> -->
|
||||
<!--<button class="btn btn-default" onclick="window.open('{{ url_for('panel.dashboard') }}','_self')"><span class="glyphicon glyphicon-plus" aria-hiddent="true"></span> Assign</button> -->
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
59
app/templates/panel/order.html
Normal file
59
app/templates/panel/order.html
Normal file
|
@ -0,0 +1,59 @@
|
|||
{% extends "base.html" %}
|
||||
|
||||
{% block head %}
|
||||
{{ super() }}
|
||||
<script type="text/javascript">
|
||||
function getPeriod()
|
||||
{
|
||||
var period = 0;
|
||||
var periodselect = document.getElementById('period');
|
||||
period = periodselect.value;
|
||||
return period;
|
||||
}
|
||||
|
||||
function calculateTotal()
|
||||
{
|
||||
//var period = getPeriod();
|
||||
//var tpm = {{ tpm }};
|
||||
//var discount = {{ discount }};
|
||||
//var ppm = {{ ppm }};
|
||||
|
||||
var pricetotal = 0;
|
||||
//pricetotal = period * ppm;
|
||||
var priceselect = document.getElementById('totalPrice');
|
||||
priceselect.innerHTML = "Total: 20";
|
||||
}
|
||||
</script>
|
||||
{% endblock %}
|
||||
|
||||
{% block title %}Activate deployment{% endblock %}
|
||||
|
||||
{% block page_content %}
|
||||
|
||||
<div class="row">
|
||||
|
||||
<div class="col-md-12">
|
||||
<div class="panel panel-info">
|
||||
<div class="panel-heading">Order new deployment</div>
|
||||
<div class="panel-body">
|
||||
|
||||
<form method="POST" action="{{ url_for('panel.order') }}">
|
||||
{{ form.cpu.label }} {{ form.cpu(min=2, max=16, step=2) }}<br />
|
||||
{{ form.memory.label }} {{ form.memory(min=512, max=16384, step=512) }}<br />
|
||||
{{ form.storage.label }} {{ form.storage(min=10, max=100, step=10) }}<br />
|
||||
|
||||
<div id="totalPrice">Total: "</div>
|
||||
|
||||
<p>
|
||||
{{ form.csrf_token() }}
|
||||
{{ form.submit }}
|
||||
</p>
|
||||
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
{% endblock %}
|
||||
|
105
app/templates/panel/vdc_pool2.html
Normal file
105
app/templates/panel/vdc_pool2.html
Normal file
|
@ -0,0 +1,105 @@
|
|||
{% 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 %}
|
102
app/templates/panel/vdc_pool3.html
Normal file
102
app/templates/panel/vdc_pool3.html
Normal file
|
@ -0,0 +1,102 @@
|
|||
<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>
|
||||
|
||||
|
|
@ -38,7 +38,7 @@ def createvm():
|
|||
deployment_seeds = current_user.inv_deployments.filter_by(deleted=False).filter_by(protected=False).all()
|
||||
if deployment_seeds != []:
|
||||
flash('Offline deployments still exist.')
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
|
||||
form = CreateForm()
|
||||
if current_user.confirmed and form.validate_on_submit():
|
||||
|
@ -68,7 +68,7 @@ def createvm():
|
|||
flash('New point created successfully in region "{}".'.format(str(selected_region.description)))
|
||||
else:
|
||||
flash('Point could not be created! Please try again later...')
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
else:
|
||||
#bridge found. lets see on which slave it is so we can create the instance on the same slave.
|
||||
data = { 'unit_id': int(selected_bridge.bridge_id),
|
||||
|
@ -82,7 +82,7 @@ def createvm():
|
|||
bridge_phy_id = query['phy_id']
|
||||
else:
|
||||
flash('Point found but cannot be used!')
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
|
||||
###
|
||||
#create new machine...
|
||||
|
@ -101,20 +101,20 @@ def createvm():
|
|||
query = contact_proxmaster(data, 'create')
|
||||
except:
|
||||
flash('Region not available! Please try again later...')
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
|
||||
if query['status'] == 'kvm_created':
|
||||
deployment = Deployment(user_id=int(current_user.pid), machine_alias=str(form.servername.data), machine_id=query['unit_id'], machine_cpu=data['cpu'], machine_mem=data['mem'], machine_hdd=data['hdd'], enabled=True, protected=False, daysleft=15, warning=True, discount=0, bridge_id=int(selected_bridge.pid))
|
||||
db.session.add(deployment)
|
||||
db.session.commit()
|
||||
flash('New device created successfully in region "{}".'.format(str(selected_region.description)))
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
else:
|
||||
flash('Device could not be created! Please try again later...')
|
||||
|
||||
#TODO: cleanup bridge if the machine is new and we were not be able to create it
|
||||
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
return render_template('vmanager/create.html', form=form)
|
||||
|
||||
@vmanager.route('/activate/<int:itemid>', methods=['GET', 'POST'])
|
||||
|
@ -204,7 +204,7 @@ def activate(itemid=0):
|
|||
db.session.commit()
|
||||
else:
|
||||
flash('Router cannot be created.')
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
|
||||
today = datetime.utcnow()
|
||||
expiry = today + relativedelta(today, months=+(form.period.data))
|
||||
|
@ -231,7 +231,7 @@ def activate(itemid=0):
|
|||
if current_user.is_administrator():
|
||||
return redirect(url_for('admin.list_deployments'))
|
||||
else:
|
||||
return redirect(url_for('main.dashboard'))
|
||||
return redirect(url_for('panel.dashboard'))
|
||||
return render_template('vmanager/activate.html', form=form, deploy=deploy, cpu_cost=cpu_cost, mem_cost=mem_cost, hdd_cost=hdd_cost, tpm=tpm, ppm=ppm, discount=discount, total=total, currency=owner.currency)
|
||||
|
||||
@vmanager.route('/vmremove/<int:unit_id>', methods=['GET', 'POST'])
|
||||
|
|
Loading…
Add table
Reference in a new issue