Demo OpenLab Debug off in manifold/static/js/manifold.js
[unfold.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              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.hrn', 'resource.hostname', 'resource.type', 
55                 'resource.network_hrn',
56                 #'lease.urn',
57                 'user.user_hrn',
58                 #'application.measurement_point.counter'
59         )
60     
61         query_resource_all = Query.get('resource').select(resource_fields)
62         if do_query_users:
63             query_user_all = Query.get('user').select(user_fields)
64     
65         aq = AnalyzedQuery(main_query, metadata=metadata)
66         page.enqueue_query(main_query, analyzed_query=aq)
67         page.enqueue_query(query_resource_all)
68         if do_query_users:
69             page.enqueue_query(query_user_all)
70     
71         # ... and for the relations
72         # XXX Let's hardcode resources for now
73         sq_resource    = aq.subquery('resource')
74         sq_user        = aq.subquery('user')
75         sq_lease       = aq.subquery('lease')
76         sq_measurement = aq.subquery('measurement')
77         
78     
79         # Prepare the display according to all metadata
80         # (some parts will be pending, others can be triggered by users).
81         # 
82         # For example slice measurements will not be requested by default...
83     
84         # Create the base layout (Stack)...
85         main_stack = Stack (
86             page=page,
87             title="Slice %s"%slicename,
88             sons=[],
89         )
90     
91         # ... responsible for the slice properties...
92     
93         # a nice header
94         main_stack.insert (
95             Raw (page=page,
96                  togglable=False, 
97                  toggled=True,
98                  html="<h2 class='well well-lg'> Slice %s</h2>"%slicename)
99         )
100     
101         # --------------------------------------------------------------------------
102         # ResourcesSelected (Pending Operations)
103
104         main_stack.insert(ResourcesSelected(
105             page                = page,
106             title               = 'Pending operations',
107             query               = main_query,
108             togglable           = True,
109             # start turned off, it will open up itself when stuff comes in
110             toggled             = False, 
111             domid               = 'pending',
112             outline_complete    = True,
113         ))
114
115         # --------------------------------------------------------------------------
116         # Filter Resources
117        
118         filter_query_editor = QueryEditor(
119             page  = page,
120             query = sq_resource, 
121             query_all = query_resource_all,
122             title = "Select Columns",
123             domid = 'select-columns',
124             )
125         filter_active_filters = ActiveFilters(
126             page  = page,
127             query = sq_resource,
128             title = "Active Filters",
129             )
130         filters_area = Stack(
131             page                = page,
132             title               = 'Filter Resources',
133             domid               = 'filters',
134             sons                = [filter_query_editor, filter_active_filters],
135             togglable           = True,
136             toggled             = 'persistent',
137             outline_complete    = True, 
138         )
139         main_stack.insert (filters_area)
140
141         # --------------------------------------------------------------------------
142         # RESOURCES
143         # the resources part is made of a Tabs (Geographic, List), 
144
145         resources_as_map = GoogleMap(
146             page       = page,
147             title      = 'Geographic view',
148             domid      = 'resources-map',
149             # tab's sons preferably turn this off
150             togglable  = False,
151             query      = sq_resource,
152             query_all  = query_resource_all,
153             checkboxes = True,
154             # center on Paris
155             latitude   = 49.,
156             longitude  = 9,
157             zoom       = 4,
158         )
159
160         resources_as_list = Hazelnut( 
161             page       = page,
162             domid      = 'resources-list',
163             title      = 'List view',
164             # this is the query at the core of the slice list
165             query      = sq_resource,
166             query_all  = query_resource_all,
167             checkboxes = True,
168             datatables_options = { 
169                 'iDisplayLength': 25,
170                 'bLengthChange' : True,
171                 'bAutoWidth'    : True,
172                 },
173             )
174
175         # with the new 'Filter' stuff on top, no need for anything but the hazelnut
176         resources_as_list_area = resources_as_list 
177
178         resources_area = Tabs ( page=page, 
179                                 domid="resources",
180                                 togglable=True,
181                                 title="Resources",
182                                 outline_complete=True,
183                                 sons=[ resources_as_map, resources_as_list_area, ],
184                                 active_domid = 'resources-map',
185                                 )
186         main_stack.insert (resources_area)
187
188
189         # --------------------------------------------------------------------------
190         # USERS
191     
192         if do_query_users:
193             tab_users = Tabs(
194                 page                = page,
195                 domid               = 'users',
196                 outline_complete    = True,
197                 togglable           = True,
198                 title               = 'Users',
199                 active_domid        = 'users-list',
200                 )
201             main_stack.insert(tab_users)
202     
203             tab_users.insert(Hazelnut( 
204                 page        = page,
205                 title       = 'Users List',
206                 domid       = 'users-list',
207                 # tab's sons preferably turn this off
208                 togglable   = False,
209                 # this is the query at the core of the slice list
210                 query       = sq_user,
211                 query_all  = query_user_all,
212                 checkboxes  = True,
213                 datatables_options = { 
214                     'iDisplayLength' : 25,
215                     'bLengthChange'  : True,
216                     'bAutoWidth'     : True,
217                 },
218             ))
219 # DEMO    
220         # --------------------------------------------------------------------------
221         # MEASUREMENTS
222 #        tab_measurements = Tabs (
223 #            page                = page,
224 #            active_domid        = 'measurements-list',
225 #            outline_complete    = True,
226 #            togglable           = True,
227 #            title               = 'Measurements',
228 #            domid               = 'measurements',
229 #        )
230 #        main_stack.insert(tab_measurements)
231 #    
232 #        tab_measurements.insert(Hazelnut( 
233 #            page        = page,
234 #            title       = 'Measurements',
235 #            domid       = 'measurements-list',
236 #            # tab's sons preferably turn this off
237 #            togglable   = False,
238 #            # this is the query at the core of the slice list
239 #            query       = sq_measurement,
240 #            # do NOT set checkboxes to False
241 #            # this table being otherwise empty, it just does not fly with dataTables
242 #            checkboxes  = True,
243 #            datatables_options = { 
244 #                'iDisplayLength' : 25,
245 #                'bLengthChange'  : True,
246 #                'bAutoWidth'     : True,
247 #            },
248 #        ))
249 #    
250 #        # --------------------------------------------------------------------------
251 #        # MESSAGES (we use transient=False for now)
252 #        main_stack.insert(Messages(
253 #            page   = page,
254 #            title  = "Runtime messages for slice %s"%slicename,
255 #            domid  = "msgs-pre",
256 #            levels = "ALL",
257 #            # plain messages are probably less nice for production but more reliable for development for now
258 #            transient = False,
259 #            # these make sense only in non-transient mode..
260 #            togglable = True,
261 #            toggled = 'persistent',
262 #            outline_complete = True,
263 #        ))
264 #    
265     
266         # variables that will get passed to the view-unfold1.html template
267         template_env = {}
268         
269         # define 'unfold1_main' to the template engine - the main contents
270         template_env [ 'unfold1_main' ] = main_stack.render(request)
271     
272         # more general variables expected in the template
273         template_env [ 'title' ] = '%(slicename)s'%locals()
274         # the menu items on the top
275         template_env [ 'topmenu_items' ] = topmenu_items('Slice', request) 
276         # so we can sho who is logged
277         template_env [ 'username' ] = the_user (request) 
278     
279         # don't forget to run the requests
280         page.expose_queries ()
281     
282         # xxx create another plugin with the same query and a different layout (with_datatables)
283         # show that it worls as expected, one single api call to backend and 2 refreshed views
284     
285         # the prelude object in page contains a summary of the requirements() for all plugins
286         # define {js,css}_{files,chunks}
287         prelude_env = page.prelude_env()
288         template_env.update(prelude_env)
289         result=render_to_response ('view-unfold1.html',template_env,
290                                    context_instance=RequestContext(request))
291         return result