Newer
Older
# Copyright 2021-2023 H2020 TeraFlow (https://www.teraflow-h2020.eu/)
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
Lluis Gifre Renom
committed
from flask import jsonify, redirect, render_template, Blueprint, flash, session, url_for, request
from common.proto.context_pb2 import Connection, Context, Device, Empty, Link, Service, Slice, Topology, ContextIdList
from common.tools.grpc.Tools import grpc_message_to_json_string
from context.client.ContextClient import ContextClient
Lluis Gifre Renom
committed
from device.client.DeviceClient import DeviceClient
Lluis Gifre Renom
committed
from webui.service.main.forms import ContextForm, DescriptorForm
context_client = ContextClient()
device_client = DeviceClient()
'context' : ('Context', 'Contexts' ),
'topology' : ('Topology', 'Topologies' ),
'device' : ('Device', 'Devices' ),
'link' : ('Link', 'Links' ),
'service' : ('Service', 'Services' ),
'slice' : ('Slice', 'Slices' ),
'connection': ('Connection', 'Connections'),
# action => infinitive, past
'add' : ('Add', 'Added'),
'update' : ('Update', 'Updated'),
'config' : ('Configure', 'Configured'),
def process_descriptor(entity_name, action_name, grpc_method, grpc_class, entities):
entity_name_singluar,entity_name_plural = ENTITY_TO_TEXT[entity_name]
action_infinitive, action_past = ACTION_TO_TEXT[action_name]
Lluis Gifre Renom
committed
num_ok, num_err = 0, 0
Lluis Gifre Renom
committed
try:
Lluis Gifre Renom
committed
num_ok += 1
except Exception as e: # pylint: disable=broad-except
flash(f'Unable to {action_infinitive} {entity_name_singluar} {str(entity)}: {str(e)}', 'error')
Lluis Gifre Renom
committed
num_err += 1
if num_ok : flash(f'{str(num_ok)} {entity_name_plural} {action_past}', 'success')
if num_err: flash(f'{str(num_err)} {entity_name_plural} failed', 'danger')
Lluis Gifre Renom
committed
def process_descriptors(descriptors):
try:
descriptors_file = request.files[descriptors.name]
descriptors_data = descriptors_file.read()
descriptors = json.loads(descriptors_data)
except Exception as e: # pylint: disable=broad-except
flash(f'Unable to load descriptor file: {str(e)}', 'danger')
return
dummy_mode = descriptors.get('dummy_mode' , False)
contexts = descriptors.get('contexts' , [])
topologies = descriptors.get('topologies' , [])
devices = descriptors.get('devices' , [])
links = descriptors.get('links' , [])
services = descriptors.get('services' , [])
slices = descriptors.get('slices' , [])
connections = descriptors.get('connections', [])
if dummy_mode:
# Dummy Mode: used to pre-load databases (WebUI debugging purposes) with no smart or automated tasks.
context_client.connect()
contexts_add = copy.deepcopy(contexts)
for context in contexts_add:
context['topology_ids'] = []
context['service_ids'] = []
topologies_add = copy.deepcopy(topologies)
for topology in topologies_add:
topology['device_ids'] = []
topology['link_ids'] = []
process_descriptor('context', 'add', context_client.SetContext, Context, contexts_add )
process_descriptor('topology', 'add', context_client.SetTopology, Topology, topologies_add)
process_descriptor('device', 'add', context_client.SetDevice, Device, devices )
process_descriptor('link', 'add', context_client.SetLink, Link, links )
process_descriptor('service', 'add', context_client.SetService, Service, services )
process_descriptor('context', 'update', context_client.SetContext, Context, contexts )
process_descriptor('topology', 'update', context_client.SetTopology, Topology, topologies )
process_descriptor('slice', 'add', context_client.SetSlice, Slice, slices )
process_descriptor('connection', 'add', context_client.SetConnection, Connection, connections )
context_client.close()
return
# Normal mode: follows the automated workflows in the different components
# in normal mode, connections should not be set
assert len(connections) == 0
devices_add = []
devices_config = []
for device in devices:
connect_rules = []
config_rules = []
for config_rule in device.get('device_config', {}).get('config_rules', []):
custom_resource_key : Optional[str] = config_rule.get('custom', {}).get('resource_key')
if custom_resource_key is not None and custom_resource_key.startswith('_connect/'):
connect_rules.append(config_rule)
else:
config_rules.append(config_rule)
if len(connect_rules) > 0:
device_add = copy.deepcopy(device)
device_add['device_endpoints'] = []
device_add['device_config'] = {'config_rules': connect_rules}
devices_add.append(device_add)
if len(config_rules) > 0:
device['device_config'] = {'config_rules': config_rules}
devices_config.append(device)
services_add = []
for service in services:
service_copy = copy.deepcopy(service)
service_copy['service_endpoint_ids'] = []
service_copy['service_constraints'] = []
service_copy['service_config'] = {'config_rules': []}
services_add.append(service_copy)
slices_add = []
for slice in slices:
slice_copy = copy.deepcopy(slice)
slice_copy['slice_endpoint_ids'] = []
slice_copy['slice_constraints'] = []
slice_copy['slice_config'] = {'config_rules': []}
slices_add.append(slice_copy)
context_client.connect()
device_client.connect()
process_descriptor('context', 'add', context_client.SetContext, Context, contexts )
process_descriptor('topology', 'add', context_client.SetTopology, Topology, topologies )
process_descriptor('device', 'add', device_client .AddDevice, Device, devices_add )
process_descriptor('device', 'config', device_client .ConfigureDevice, Device, devices_config)
process_descriptor('link', 'add', context_client.SetLink, Link, links )
process_descriptor('service', 'add', service_client.CreateService, Service, services_add )
process_descriptor('service', 'update', service_client.UpdateService, Service, services )
process_descriptor('slice', 'add', slice_client .CreateSlice, Slice, slices_add )
process_descriptor('slice', 'update', slice_client .UpdateSlice, Slice, slices )
device_client.close()
context_client.close()
@main.route('/', methods=['GET', 'POST'])
def home():
context_client.connect()
Lluis Gifre Renom
committed
device_client.connect()
response: ContextIdList = context_client.ListContextIds(Empty())
context_form: ContextForm = ContextForm()
context_form.context.choices.append(('', 'Select...'))
for context in response.context_ids:
context_form.context.choices.append((context.context_uuid.uuid, context.context_uuid))
if context_form.validate_on_submit():
session['context_uuid'] = context_form.context.data
flash(f'The context was successfully set to `{context_form.context.data}`.', 'success')
Lluis Gifre Renom
committed
return redirect(url_for("main.home"))
if 'context_uuid' in session:
context_form.context.data = session['context_uuid']
Lluis Gifre Renom
committed
descriptor_form: DescriptorForm = DescriptorForm()
try:
if descriptor_form.validate_on_submit():
process_descriptors(descriptor_form.descriptors)
return redirect(url_for("main.home"))
except Exception as e:
logger.exception('Descriptor load failed')
flash(f'Descriptor load failed: `{str(e)}`', 'danger')
finally:
context_client.close()
device_client.close()
Lluis Gifre Renom
committed
return render_template('main/home.html', context_form=context_form, descriptor_form=descriptor_form)
Lluis Gifre Renom
committed
@main.route('/topology', methods=['GET'])
def topology():
context_client.connect()
try:
response = context_client.ListDevices(Empty())
devices = [{
'id': device.device_id.device_uuid.uuid,
'name': device.device_id.device_uuid.uuid,
'type': device.device_type,
} for device in response.devices]
Lluis Gifre Renom
committed
response = context_client.ListLinks(Empty())
links = []
for link in response.links:
if len(link.link_endpoint_ids) != 2:
str_link = grpc_message_to_json_string(link)
logger.warning('Unexpected link with len(endpoints) != 2: {:s}'.format(str_link))
continue
links.append({
'id': link.link_id.link_uuid.uuid,
'source': link.link_endpoint_ids[0].device_id.device_uuid.uuid,
'target': link.link_endpoint_ids[1].device_id.device_uuid.uuid,
})
Lluis Gifre Renom
committed
return jsonify({'devices': devices, 'links': links})
except:
logger.exception('Error retrieving topology')
finally:
context_client.close()
@main.get('/about')
def about():
return render_template('main/about.html')
@main.get('/debug')
def debug():
return render_template('main/debug.html')
Lluis Gifre Renom
committed
@main.get('/resetsession')
def reset_session():
session.clear()