stupidly tried to set checkboxes=False .. on an otherwise empty table - dataTables...
[myslice.git] / portal / sliceview.py
1 from django.template                 import RequestContext
2 from django.shortcuts                import render_to_response
3
4 from unfold.loginrequired            import LoginRequiredAutoLogoutView
5
6 from unfold.page                     import Page
7 from manifold.core.query             import Query, AnalyzedQuery
8
9 from ui.topmenu                      import topmenu_items, the_user
10
11 from plugins.raw                     import Raw
12 from plugins.stack                   import Stack
13 from plugins.tabs                    import Tabs
14 from plugins.hazelnut                import Hazelnut 
15 from plugins.resources_selected      import ResourcesSelected
16 from plugins.googlemap               import GoogleMap
17 from plugins.senslabmap.senslabmap   import SensLabMap
18 from plugins.querycode               import QueryCode
19 from plugins.query_editor            import QueryEditor
20 from plugins.active_filters          import ActiveFilters
21 from plugins.quickfilter             import QuickFilter
22 from plugins.messages                import Messages
23
24 from myslice.config                  import Config
25
26 tmp_default_slice='ple.upmc.myslicedemo'
27
28 # temporary : turn off the users part to speed things up
29 do_query_users=True
30
31 class SliceView (LoginRequiredAutoLogoutView):
32
33     def get (self,request, slicename=tmp_default_slice):
34     
35         page = Page(request)
36         page.add_css_files ('css/slice-view.css')
37         page.add_js_files  ( [ "js/common.functions.js" ] )
38         page.add_js_chunks ('$(function() { messages.debug("sliceview: jQuery version " + $.fn.jquery); });')
39         page.add_js_chunks ('$(function() { messages.debug("sliceview: users turned %s"); });'%("on" if do_query_users else "off"))
40         page.add_js_chunks ('$(function() { messages.debug("manifold URL %s"); });'%(Config.manifold_url()))
41         page.expose_js_metadata()
42     
43         metadata = page.get_metadata()
44         resource_md = metadata.details_by_object('resource')
45         resource_fields = [column['name'] for column in resource_md['column']]
46     
47         user_md = metadata.details_by_object('user')
48         user_fields = ['user_hrn'] # [column['name'] for column in user_md['column']]
49     
50         # TODO The query to run is embedded in the URL
51         main_query = Query.get('slice').filter_by('slice_hrn', '=', slicename)
52         main_query.select(
53                 'slice_hrn',
54                 'resource.resource_hrn', 'resource.hostname', 'resource.type', 'resource.network_hrn',
55                 #'lease.urn',
56                 'user.user_hrn',
57                 #'application.measurement_point.counter'
58         )
59     
60         query_resource_all = Query.get('resource').select(resource_fields)
61         if do_query_users:
62             query_user_all = Query.get('user').select(user_fields)
63     
64         aq = AnalyzedQuery(main_query, metadata=metadata)
65         page.enqueue_query(main_query, analyzed_query=aq)
66         page.enqueue_query(query_resource_all)
67         if do_query_users:
68             page.enqueue_query(query_user_all)
69     
70         # ... and for the relations
71         # XXX Let's hardcode resources for now
72         sq_resource    = aq.subquery('resource')
73         sq_user        = aq.subquery('user')
74         sq_lease       = aq.subquery('lease')
75         sq_measurement = aq.subquery('measurement')
76         
77     
78         # Prepare the display according to all metadata
79         # (some parts will be pending, others can be triggered by users).
80         # 
81         # For example slice measurements will not be requested by default...
82     
83         # Create the base layout (Stack)...
84         main_stack = Stack (
85             page=page,
86             title="Slice %s"%slicename,
87             sons=[],
88         )
89     
90         # ... responsible for the slice properties...
91     
92         # a nice header
93         main_stack.insert (
94             Raw (page=page,
95                  togglable=False, 
96                  toggled=True,
97                  html="<h2 class='well well-lg'> Slice %s</h2>"%slicename)
98         )
99     
100         # --------------------------------------------------------------------------
101         # ResourcesSelected (Pending Operations)
102
103         main_stack.insert(ResourcesSelected(
104             page                = page,
105             title               = 'Pending operations',
106             query               = main_query,
107             togglable           = True,
108             domid               = 'pending',
109             outline_complete    = True,
110         ))
111     
112         # --------------------------------------------------------------------------
113         # RESOURCES
114         # the resources part is made of a Tabs (Geographic, List), 
115
116         resources_as_map = GoogleMap(
117             page       = page,
118             title      = 'Geographic view',
119             domid      = 'resources-map',
120             # tab's sons preferably turn this off
121             togglable  = False,
122             query      = sq_resource,
123             query_all  = query_resource_all,
124             checkboxes = True,
125             # center on Paris
126             latitude   = 49.,
127             longitude  = 9,
128             zoom       = 4,
129         )
130
131         resources_as_list = Hazelnut( 
132             page       = page,
133             domid      = 'resources-list',
134             # this is the query at the core of the slice list
135             query      = sq_resource,
136             query_all  = query_resource_all,
137             checkboxes = True,
138             datatables_options = { 
139                 'iDisplayLength': 25,
140                 'bLengthChange' : True,
141                 'bAutoWidth'    : True,
142                 },
143             )
144
145         resources_query_editor = QueryEditor(
146             page  = page,
147             query = query_resource_all,
148             title = "Select Columns",
149             )
150         resources_active_filters = ActiveFilters(
151             page  = page,
152             query = sq_resource,
153             title = "Active Filters ?",
154             )
155
156         # List area itself is a Stack with hazelnut on top,
157         # and a togglable tabs for customization plugins 
158         resources_as_list_area = Stack(
159             page        = page,
160             title       = 'Resources as a List',
161             domid       = 'resources-list-area',
162             sons= [ resources_as_list, 
163                     Tabs ( page=page,
164                            title="Customize Resources layout",
165                            togglable=True,
166                            toggled='persistent',
167                            domid="customize-resources",
168                            outline_complete=True,
169                            sons = [ resources_query_editor, resources_active_filters, ],
170                            ),
171                     ],
172             )
173         resources_area = Tabs ( page=page, 
174                                 domid="resources",
175                                 togglable=True,
176                                 title="Resources",
177                                 outline_complete=True,
178                                 sons=[ resources_as_map, resources_as_list_area, ],
179                                 active_domid = 'resources-map',
180                                 )
181         main_stack.insert (resources_area)
182
183
184         # --------------------------------------------------------------------------
185         # USERS
186     
187         if do_query_users:
188             tab_users = Tabs(
189                 page                = page,
190                 domid               = 'users',
191                 outline_complete    = True,
192                 togglable           = True,
193                 title               = 'Users',
194                 active_domid        = 'users-list',
195                 )
196             main_stack.insert(tab_users)
197     
198             tab_users.insert(Hazelnut( 
199                 page        = page,
200                 title       = 'Users List',
201                 domid       = 'users-list',
202                 # tab's sons preferably turn this off
203                 togglable   = False,
204                 # this is the query at the core of the slice list
205                 query       = sq_user,
206                 query_all  = query_user_all,
207                 checkboxes  = True,
208                 datatables_options = { 
209                     'iDisplayLength' : 25,
210                     'bLengthChange'  : True,
211                     'bAutoWidth'     : True,
212                 },
213             ))
214     
215         # --------------------------------------------------------------------------
216         # MEASUREMENTS
217         tab_measurements = Tabs (
218             page                = page,
219             active_domid        = 'measurements-list',
220             outline_complete    = True,
221             togglable           = True,
222             title               = 'Measurements',
223             domid               = 'measurements',
224         )
225         main_stack.insert(tab_measurements)
226     
227         tab_measurements.insert(Hazelnut( 
228             page        = page,
229             title       = 'Measurements',
230             domid       = 'measurements-list',
231             # tab's sons preferably turn this off
232             togglable   = False,
233             # this is the query at the core of the slice list
234             query       = sq_measurement,
235             # do NOT set checkboxes to False
236             # this table being otherwise empty, it just does not fly with dataTables
237             checkboxes  = True,
238             datatables_options = { 
239                 'iDisplayLength' : 25,
240                 'bLengthChange'  : True,
241                 'bAutoWidth'     : True,
242             },
243         ))
244     
245         # --------------------------------------------------------------------------
246         # MESSAGES (we use transient=False for now)
247         main_stack.insert(Messages(
248             page   = page,
249             title  = "Runtime messages for slice %s"%slicename,
250             domid  = "msgs-pre",
251             levels = "ALL",
252             # plain messages are probably less nice for production but more reliable for development for now
253             transient = False,
254             # these make sense only in non-transient mode..
255             togglable = True,
256             toggled = 'persistent',
257             outline_complete = True,
258         ))
259     
260     
261         # variables that will get passed to the view-unfold1.html template
262         template_env = {}
263         
264         # define 'unfold1_main' to the template engine - the main contents
265         template_env [ 'unfold1_main' ] = main_stack.render(request)
266     
267         # more general variables expected in the template
268         template_env [ 'title' ] = '%(slicename)s'%locals()
269         # the menu items on the top
270         template_env [ 'topmenu_items' ] = topmenu_items('Slice', request) 
271         # so we can sho who is logged
272         template_env [ 'username' ] = the_user (request) 
273     
274         # don't forget to run the requests
275         page.expose_queries ()
276     
277         # xxx create another plugin with the same query and a different layout (with_datatables)
278         # show that it worls as expected, one single api call to backend and 2 refreshed views
279     
280         # the prelude object in page contains a summary of the requirements() for all plugins
281         # define {js,css}_{files,chunks}
282         prelude_env = page.prelude_env()
283         template_env.update(prelude_env)
284         result=render_to_response ('view-unfold1.html',template_env,
285                                    context_instance=RequestContext(request))
286         return result