From: Ciro Scognamiglio Date: Tue, 25 Feb 2014 16:05:19 +0000 (+0100) Subject: merged changes X-Git-Tag: myslice-1.1~308 X-Git-Url: http://git.onelab.eu/?a=commitdiff_plain;h=d5867ebb7c85a07fd5b9dad155c9a82048b3e706;hp=da0b290c0d86afe3d388f4b1a8be60a70b39ed92;p=myslice.git merged changes --- diff --git a/myslice/urls.py b/myslice/urls.py index 67bb75d9..1e84db59 100644 --- a/myslice/urls.py +++ b/myslice/urls.py @@ -63,7 +63,7 @@ urls = [ # #(r'^view/?', include('view.urls')), #(r'^list/slices', 'view.list.slices'), - #(r'^list/(?P[^/]+)', 'view.list.default'), + (r'^list/(?P[^/]+)', 'rest.list'), # # # Portal diff --git a/portal/sliceview.py b/portal/sliceview.py index 744d2b34..d2b40550 100644 --- a/portal/sliceview.py +++ b/portal/sliceview.py @@ -1,444 +1,23 @@ -import json from django.template import RequestContext from django.shortcuts import render_to_response -from unfold.loginrequired import LoginRequiredAutoLogoutView - -from unfold.page import Page from manifold.core.query import Query, AnalyzedQuery from manifoldapi.manifoldapi import execute_query -from ui.topmenu import topmenu_items_live, the_user +from django.views.generic.base import TemplateView -from plugins.raw import Raw -from plugins.stack import Stack -from plugins.tabs import Tabs -from plugins.querytable import QueryTable -from plugins.querygrid import QueryGrid -from plugins.queryupdater import QueryUpdater -from plugins.googlemap import GoogleMap -from plugins.senslabmap import SensLabMap -from plugins.scheduler import Scheduler -from plugins.querycode import QueryCode -# Thierry -# stay away from query editor for now as it seems to make things go very slow -# see https://lists.myslice.info/pipermail/devel-myslice/2013-December/000221.html -from plugins.query_editor import QueryEditor -from plugins.active_filters import ActiveFilters -from plugins.quickfilter import QuickFilter -from plugins.messages import Messages -from plugins.slicestat import SliceStat +from unfold.loginrequired import LoginRequiredView +from django.http import HttpResponse +from django.shortcuts import render -from myslice.configengine import ConfigEngine +from unfold.page import Page +from manifold.core.query import Query, AnalyzedQuery +from manifold.manifoldapi import execute_query from theme import ThemeView -tmp_default_slice='ple.upmc.myslicedemo' - -# temporary : turn off the users part to speed things up -do_query_users=True -#do_query_users=False - -#do_query_leases=True -do_query_leases=False - -insert_grid=False -#insert_grid=True - -insert_messages=False -#insert_messages=True - -class SliceView (LoginRequiredAutoLogoutView, ThemeView): - - def get (self,request, slicename=tmp_default_slice): - - page = Page(request) - page.add_css_files ('css/slice-view.css') - page.add_js_files ( [ "js/common.functions.js" ] ) - page.add_js_chunks ('$(function() { messages.debug("sliceview: jQuery version " + $.fn.jquery); });') - page.add_js_chunks ('$(function() { messages.debug("sliceview: users turned %s"); });'%("on" if do_query_users else "off")) - page.add_js_chunks ('$(function() { messages.debug("sliceview: leases turned %s"); });'%("on" if do_query_leases else "off")) - page.add_js_chunks ('$(function() { messages.debug("manifold URL %s"); });'%(ConfigEngine().manifold_url())) - - metadata = page.get_metadata() - resource_md = metadata.details_by_object('resource') - resource_fields = [column['name'] for column in resource_md['column']] - - user_md = metadata.details_by_object('user') - user_fields = ['user_hrn'] # [column['name'] for column in user_md['column']] - - # TODO The query to run is embedded in the URL - main_query = Query.get('slice').filter_by('slice_hrn', '=', slicename) - main_query.select( - 'slice_hrn', - #'resource.hrn', 'resource.urn', - 'resource.hostname', 'resource.type', - 'resource.network_hrn', - 'lease.urn', - 'user.user_hrn', - #'application.measurement_point.counter' - ) - # for internal use in the querytable plugin; - # needs to be a unique column present for each returned record - main_query_init_key = 'hostname' - - query_resource_all = Query.get('resource').select(resource_fields) - - aq = AnalyzedQuery(main_query, metadata=metadata) - page.enqueue_query(main_query, analyzed_query=aq) - page.enqueue_query(query_resource_all) - if do_query_users: - # Required: the user must have an authority in its user.config - # XXX Temporary solution - user_query = Query().get('local:user').select('config','email') - user_details = execute_query(self.request, user_query) - - # not always found in user_details... - config={} -# for user_detail in user_details: -# #email = user_detail['email'] -# if user_detail['config']: -# config = json.loads(user_detail['config']) -# user_detail['authority'] = config.get('authority',"Unknown Authority") -# -# if user_detail['authority'] is not None: -# sub_authority = user_detail['authority'].split('.') -# root_authority = sub_authority[0] -# query_user_all = Query.get(root_authority+':user').select(user_fields) -# -# # XXX TODO this filter doesn't work - to be improved in Manifold -# #.filter_by('authority.authority_hrn', '=', user_detail['authority']) -# -# page.enqueue_query(query_user_all) -# else: -# print "authority of the user is not in local:user db" - query_user_all = Query.get('user').select(user_fields).filter_by('parent_authority','==','ple.upmc') - page.enqueue_query(query_user_all) - # query_user_all = None - - # ... and for the relations - # XXX Let's hardcode resources for now - sq_resource = aq.subquery('resource') - sq_user = aq.subquery('user') - sq_lease = aq.subquery('lease') - sq_measurement = aq.subquery('measurement') - - - # Prepare the display according to all metadata - # (some parts will be pending, others can be triggered by users). - # - # For example slice measurements will not be requested by default... - - # Create the base layout (Stack)... - main_stack = Stack ( - page=page, - title="Slice %s"%slicename, - sons=[], - ) - - # ... responsible for the slice properties... - - # a nice header - main_stack.insert ( - Raw (page=page, - togglable=False, - toggled=True, - html="

Slice %s

"%slicename) - ) - - # -------------------------------------------------------------------------- - # QueryUpdater (Pending Operations) - - main_stack.insert(QueryUpdater( - page = page, - title = 'Pending operations', - query = main_query, - togglable = True, - # start turned off, it will open up itself when stuff comes in - toggled = False, - domid = 'pending', - outline_complete = True, - )) - - # -------------------------------------------------------------------------- - # Filter Resources - -# turn off for now -- see above - filter_query_editor = QueryEditor( - page = page, - query = sq_resource, - query_all = query_resource_all, - title = "Select Columns", - domid = 'select-columns', - ) - filter_active_filters = ActiveFilters( - page = page, - query = sq_resource, - title = "Active Filters", - ) - filters_area = Stack( - page = page, - title = 'Filter Resources', - domid = 'filters', - sons = [filter_query_editor, - filter_active_filters], - togglable = True, - toggled = 'persistent', - outline_complete = True, - ) - main_stack.insert (filters_area) - - # -------------------------------------------------------------------------- - # RESOURCES - # the resources part is made of a Tabs (Geographic, List), - - resources_as_gmap = GoogleMap( - page = page, - title = 'Geographic view', - domid = 'resources-map', - # tab's sons preferably turn this off - togglable = False, - query = sq_resource, - query_all = query_resource_all, - # this key is the one issued by google - googlemap_api_key = ConfigEngine().googlemap_api_key(), - # the key to use at init-time - init_key = main_query_init_key, - checkboxes = True, - # center on Paris - latitude = 49., - longitude = 9, - zoom = 4, - ) - - resources_as_3dmap = SensLabMap( - page = page, - title = '3D Map', - domid = 'senslabmap', - query = sq_resource, - query_all = query_resource_all, - ) - - resources_as_list = QueryTable( - page = page, - domid = 'resources-list', - title = 'List view', - # this is the query at the core of the slice list - query = sq_resource, - query_all = query_resource_all, - init_key = main_query_init_key, - checkboxes = True, - datatables_options = { - 'iDisplayLength': 25, - 'bLengthChange' : True, - 'bAutoWidth' : True, - }, - ) - - if insert_grid: - resources_as_grid = QueryGrid( - page = page, - domid = 'resources-grid', - title = 'Grid view', - # this is the query at the core of the slice list - query = sq_resource, - query_all = query_resource_all, - init_key = main_query_init_key, - checkboxes = True, - ) - - if do_query_leases: - resources_as_scheduler = Scheduler( - page = page, - title = 'Scheduler', - domid = 'scheduler', - query = sq_resource, - query_all_resources = query_resource_all, - query_lease = sq_lease, - ) - - # with the new 'Filter' stuff on top, no need for anything but the querytable - resources_as_list_area = resources_as_list - - resources_sons = [ - resources_as_gmap, - resources_as_3dmap, - resources_as_scheduler, - resources_as_list_area, - ] if do_query_leases else [ - resources_as_gmap, - resources_as_3dmap, - resources_as_list_area, - ] - if insert_grid: - resources_sons.append(resources_as_grid) - - resources_area = Tabs ( page=page, - domid="resources", - togglable=True, - title="Resources", - outline_complete=True, - sons= resources_sons, - active_domid = 'resources-map', - persistent_active=True, - ) - main_stack.insert (resources_area) - - # -------------------------------------------------------------------------- - # USERS - - if do_query_users and query_user_all is not None: - tab_users = Tabs( - page = page, - domid = 'users', - outline_complete = True, - togglable = True, - title = 'Users', - active_domid = 'users-list', - ) - main_stack.insert(tab_users) +class SliceView (LoginRequiredView, ThemeView): + template_name = "slice-view.html" - tab_users.insert(QueryTable( - page = page, - title = 'Users List', - domid = 'users-list', - # tab's sons preferably turn this off - togglable = False, - # this is the query at the core of the slice list - query = sq_user, - query_all = query_user_all, - init_key = 'user_hrn', - checkboxes = True, - datatables_options = { - 'iDisplayLength' : 25, - 'bLengthChange' : True, - 'bAutoWidth' : True, - }, - )) - -# DEMO - # -------------------------------------------------------------------------- - # MEASUREMENTS - measurements_stats_cpu = SliceStat( - title = "CPU Usage", - domid = 'resources-stats-cpu', - page = page, - stats = 'slice', - key = 'hrn', - query = 'none', - slicename = slicename, - o = 'cpu' - ) - - measurements_stats_mem = SliceStat( - title = "Memory Usage", - domid = 'resources-stats-mem', - page = page, - stats = 'slice', - key = 'hrn', - query = 'none', - slicename = slicename, - o = 'mem' - ) - - measurements_stats_asb = SliceStat( - title = "Traffic Sent", - domid = 'resources-stats-asb', - page = page, - stats = 'slice', - key = 'hrn', - query = 'none', - slicename = slicename, - o = 'asb' - ) - - measurements_stats_arb = SliceStat( - title = "Traffic Received", - domid = 'resources-stats-arb', - page = page, - stats = 'slice', - key = 'hrn', - query = 'none', - slicename = slicename, - o = 'arb' - ) - - tab_measurements = Tabs ( page=page, - domid = "measurements", - togglable = True, - toggled = 'persistent', - title = "Measurements", - outline_complete=True, - sons = [ measurements_stats_cpu, measurements_stats_mem, measurements_stats_asb, measurements_stats_arb ], - active_domid = 'resources-stats-cpu', - persistent_active = True, - ) - main_stack.insert (tab_measurements) - -# tab_measurements = Tabs ( -# page = page, -# active_domid = 'measurements-list', -# outline_complete = True, -# togglable = True, -# title = 'Measurements', -# domid = 'measurements', -# ) -# main_stack.insert(tab_measurements) -# -# tab_measurements.insert(QueryTable( -# page = page, -# title = 'Measurements', -# domid = 'measurements-list', -# # tab's sons preferably turn this off -# togglable = False, -# # this is the query at the core of the slice list -# query = sq_measurement, -# # do NOT set checkboxes to False -# # this table being otherwise empty, it just does not fly with dataTables -# checkboxes = True, -# datatables_options = { -# 'iDisplayLength' : 25, -# 'bLengthChange' : True, -# 'bAutoWidth' : True, -# }, -# )) -# -# # -------------------------------------------------------------------------- -# # MESSAGES (we use transient=False for now) - if insert_messages: - main_stack.insert(Messages( - page = page, - title = "Runtime messages for slice %s"%slicename, - domid = "msgs-pre", - levels = "ALL", - # plain messages are probably less nice for production but more reliable for development for now - transient = False, - # these make sense only in non-transient mode.. - togglable = True, - toggled = 'persistent', - outline_complete = True, - )) - - # variables that will get passed to the view-unfold1.html template - template_env = {} - - # define 'unfold_main' to the template engine - the main contents - template_env [ 'unfold_main' ] = main_stack.render(request) - - # more general variables expected in the template - template_env [ 'title' ] = '%(slicename)s'%locals() - # the menu items on the top - template_env [ 'topmenu_items' ] = topmenu_items_live('Slice', page) - # so we can sho who is logged - template_env [ 'username' ] = the_user (request) - - template_env ['theme'] = self.theme - - # don't forget to run the requests - page.expose_js_metadata() - # the prelude object in page contains a summary of the requirements() for all plugins - # define {js,css}_{files,chunks} - template_env.update(page.prelude_env()) - - return render_to_response ('view-unfold1.html',template_env, - context_instance=RequestContext(request)) + def get(self, request, slicename): + return render_to_response(self.template, {"slice": slicename, "theme": self.theme, "username": request.user}, context_instance=RequestContext(request)) diff --git a/portal/templates/onelab/onelab_home-view.html b/portal/templates/onelab/onelab_home-view.html index 812b36fc..b2970241 100644 --- a/portal/templates/onelab/onelab_home-view.html +++ b/portal/templates/onelab/onelab_home-view.html @@ -102,8 +102,8 @@ {% if person %} $.getJSON("rest/user", function( data ) { var items = []; - $.each( data[0].slices, function( key, val ) { - items.push( "
  • " + val + "
  • " ); + $.each( data[0].slice, function( key, val ) { + items.push( "
  • " + val.slice_hrn + "
  • " ); }); $("div#home-slice-list").html($( "
      ", { html: items.join( "" ) })); diff --git a/rest/__init__.py b/rest/__init__.py index 71fe44dc..701b6249 100644 --- a/rest/__init__.py +++ b/rest/__init__.py @@ -24,6 +24,7 @@ def dispatch(request, object_type, object_name): if (object_type == 'platform') : object_type = 'local:platform' object_properties = ['platform', 'platform_longname', 'platform_url', 'platform_description','gateway_type']; + return switch.get('platform', error)(request, object_name, object_properties) else : query = Query.get('local:object').filter_by('table', '==', object_type).select('column.name') results = execute_query(request, query) @@ -33,30 +34,31 @@ def dispatch(request, object_type, object_name): object_properties.append(r['name']) else : return error() - - return switch.get(object_type, error)(request, object_type, object_name, object_properties) + return switch.get(object_type, error)(request, object_name, object_properties) # if request.method == 'GET': # return switch.get(request, object_type, object_name, object_properties) # elif request.method == 'POST': # return post(request, object_type, object_name) -def platform(): - return HttpResponse(json.dumps({'user' : 'error message'}), content_type="application/json") +def platform(request, object_name, object_properties): + query = Query().get('local:platform').filter_by('disabled', '==', '0').select(object_properties) + return send(request, execute_query(request, query), object_properties) -def slice(): - return HttpResponse(json.dumps({'user' : 'error message'}), content_type="application/json") +def slice(request, object_name, object_properties): + query = Query().get('slice').filter_by('user.user_hrn', '==', '$user_hrn').select(object_properties) + return send(request, execute_query(request, query), object_properties) -def user(request, object_type, object_name, object_properties): +def user(request, object_name, object_properties): query = Query().get('user').filter_by('user_hrn', '==', '$user_hrn').select(object_properties) - #.select('slice.slice_hrn') - return send(request, execute_query(request, query), object_properties) def send(request, response, object_properties): if request.path.split('/')[1] == 'rest' : response_data = response - else : + elif request.path.split('/')[1] == 'table' : + response_data = response + elif request.path.split('/')[1] == 'datatable' : response_data = {} response_data['columns'] = object_properties response_data['labels'] = object_properties @@ -74,37 +76,5 @@ def send(request, response, object_properties): return HttpResponse(json.dumps(response_data), content_type="application/json") - - - - -def get(request, object_type, object_name, object_properties): - - query = Query().get(object_type).select(object_properties).filter_by('user_hrn', '=', '$user_hrn') - if (object_name) : - query = query.filter_by(object_type + '_hrn', '=', object_name) - - response = execute_query(request, query) - - response_data = {} - response_data['columns'] = object_properties - response_data['labels'] = object_properties - #response_data['labels'] = [ 'Platform', 'Name', 'Url', 'Description','Gateway Type' ] - response_data['data'] = [] - response_data['total'] = len(response) - for r in response : - d = [] - for p in object_properties : - d.append(r[p]) - print d - - response_data['data'].append(d) - #response_data['data'].append([ r['platform'], r['platform_longname'], r['platform_url'], r['platform_description'], r['gateway_type'] ]) - - return HttpResponse(json.dumps(response_data), content_type="application/json") - -def post(request, object_type, object_name): - pass - -def error(): - return HttpResponse(json.dumps({'error' : 'error message'}), content_type="application/json") +def error(request, object_name, object_properties): + return HttpResponse(json.dumps({'error' : 'an error has occurred'}), content_type="application/json")