Merge branch 'master' of git.onelab.eu:/git/plewww
[plewww.git] / planetlab / slices / slice.php
1 <?php
2
3 // $Id$
4
5 // Require login
6 require_once 'plc_login.php';
7
8 // Get session and API handles
9 require_once 'plc_session.php';
10 global $plc, $api;
11
12 // Print header
13 require_once 'plc_drupal.php';
14 include 'plc_header.php';
15
16 // Common functions
17 require_once 'plc_functions.php';
18 require_once 'plc_peers.php';
19 require_once 'plc_objects.php';
20 require_once 'plc_visibletags2.php';
21 require_once 'linetabs.php';
22 require_once 'table2.php';
23 require_once 'details.php';
24 require_once 'toggle.php';
25 require_once 'form.php';
26 require_once 'raphael.php';
27 require_once 'columns.php';
28
29 // keep css separate for now
30 drupal_set_html_head('
31 <link href="/planetlab/css/my_slice.css" rel="stylesheet" type="text/css" />
32 <script src="/planetlab/slices/leases.js" type="text/javascript" charset="utf-8"></script>
33 ');
34
35 // -------------------- admins potentially need to get full list of users
36 ini_set('memory_limit','32M');
37
38 $profiling=false;
39 if ($_GET['profiling']) $profiling=true;
40
41 if ($profiling)  plc_debug_prof_start();
42
43 // -------------------- 
44 // recognized URL arguments
45 $slice_id=intval($_GET['id']);
46 if ( ! $slice_id ) { plc_error('Malformed URL - id not set'); return; }
47
48 ////////////////////
49 // have to name columns b/c we need the non-native 'omf_control' column
50 $slice_columns=array('slice_id','name','peer_id','site_id','person_ids','node_ids','expires',
51                      'url','description','instantiation','omf_control');
52 $slices= $api->GetSlices( array($slice_id), $slice_columns);
53
54 if (empty($slices)) {
55   drupal_set_message ("Slice " . $slice_id . " not found");
56   return;
57  }
58
59 $slice=$slices[0];
60
61 if ($profiling) plc_debug_prof('2: slice',count($slices));
62 // pull all node info to vars
63 $name= $slice['name'];
64 $expires = date( "d/m/Y", $slice['expires'] );
65 $site_id= $slice['site_id'];
66
67 $person_ids=$slice['person_ids'];
68
69 // get peers
70 $peer_id= $slice['peer_id'];
71 $peers=new Peers ($api);
72 $local_peer = ! $peer_id;
73
74 if ($profiling) plc_debug_prof('3: peers',count($peers));
75
76 // gets site info
77 $sites= $api->GetSites( array( $site_id ) );
78 $site=$sites[0];
79 $site_name= $site['name'];
80
81 if ($profiling) plc_debug_prof('4: sites',count($sites));
82 //////////////////////////////////////// building blocks for the renew area
83 // Constants
84 global $DAY;            $DAY = 24*60*60;
85 global $WEEK;           $WEEK = 7 * $DAY; 
86 global $MAX_WEEKS;      $MAX_WEEKS= 8;          // weeks from today
87 global $GRACE_DAYS;     $GRACE_DAYS=10;         // days for renewal promoted on top
88 global $NOW;            $NOW=mktime();
89
90 ////////////////////////////////////////////////////////////
91 // make the renew area on top and open if the expiration time is less than 10 days from now
92 function renew_needed ($slice) {
93   global $DAY, $NOW, $GRACE_DAYS;
94   $current_exp=$slice['expires'];
95
96   $time_left = $current_exp - $NOW;
97   $visible = $time_left/$DAY <= $GRACE_DAYS;
98   return $visible;
99 }
100
101 function renew_area ($slice,$site,$visible) {
102   global $DAY, $WEEK, $MAX_WEEKS, $GRACE_DAYS, $NOW;
103  
104   $current_exp=$slice['expires'];
105   $current_text = gmstrftime("%A %b-%d-%y %T %Z", $current_exp);
106   $max_exp= $NOW + ($MAX_WEEKS * $WEEK); // seconds since epoch
107   $max_text = gmstrftime("%A %b-%d-%y %T %Z", $max_exp);
108
109   // xxx some extra code needed to enable this area only if the slice description is OK:
110   // description and url must be non void
111   $toggle=
112     new PlekitToggle('renew',"Expires $current_text - Renew this slice",
113                      array("bubble"=>
114                            "Enter this zone if you wish to renew your slice",
115                            'visible'=>$visible));
116   $toggle->start();
117
118   // xxx message could take roles into account
119   if ($site['max_slices']<=0) {
120      $message= <<< EOF
121 <p class='my-slice-renewal'>Slice creation and renewal have been temporarily disabled for your
122 <site. This may have occurred because your site's nodes have been down
123 or unreachable for several weeks, and multiple attempts to contact
124 your site's PI(s) and Technical Contact(s) have all failed. If so,
125 contact your site's PI(s) and Technical Contact(s) and ask them to
126 bring up your site's nodes. Please visit your <a
127 href='/db/sites/index.php?id=$site_id'>site details</a> page to find
128 out more about your site's nodes, and how to contact your site's PI(s)
129 and Technical Contact(s).</p>
130 EOF;
131      echo $message;
132  
133   } else {
134     // xxx this is a rough cut and paste from the former UI
135     // showing a datepicker view could be considered as well with some extra work
136     // calculate possible extension lengths
137     $selectors = array();
138     foreach ( array ( 1 => "One more week", 
139                       2 => "Two more weeks", 
140                       3 => "Three more weeks", 
141                       4 => "One more month" ) as $weeks => $text ) {
142       $candidate_exp = $current_exp + $weeks*$WEEK;
143       if ( $candidate_exp < $max_exp) {
144         $selectors []= array('display'=>"$text (" . gmstrftime("%A %b-%d-%y %T %Z", $candidate_exp) . ")",
145                              'value'=>$candidate_exp);
146         $max_renewal_weeks=$weeks;
147         $max_renewal_date= gmstrftime("%A %b-%d-%y %T %Z", $candidate_exp);
148       }
149     }
150
151     if ( empty( $selectors ) ) {
152       print <<< EOF
153 <div class='my-slice-renewal'>
154 Slices annot be renewed more than $MAX_WEEKS weeks from now, i.e. not beyond $max_text. 
155 For this reason, the current slice cannot be renewed any further into the future, try again closer to expiration date.
156 </div>
157 EOF;
158      } else {
159       print <<< EOF
160 <div class='my-slice-renewal'>
161 <p>You <span class='bold'>must</span> provide a short description, 
162 as well as a link to a project website, before renewing it.
163
164 <br/> Please make sure to provide reasonable details on <span class='bold'>
165 the kind of traffic</span>, and <span class='bold'>copyrights</span> if relevant. 
166 Do <span class='bold'>not</span> provide bogus information; if a complaint is lodged against 
167 your slice  and your PlanetLab Operations Center is unable to determine what the normal behavior 
168 of your slice is, your slice may be deleted to resolve the complaint.</p>
169
170 <p><span class='bold'>NOTE:</span> 
171 Slices cannot be renewed beyond another $max_renewal_weeks week(s) ($max_renewal_date).
172 </p>
173 </div>
174 EOF;
175
176       $form = new PlekitForm (l_actions(),
177                               array('action'=>'renew-slice',
178                                     'slice_id'=>$slice['slice_id']));
179       $form->start();
180       print $form->label_html('expires','Duration');
181       print $form->select_html('expires',$selectors,array('label'=>'Pick one'));
182       print $form->submit_html('renew-button','Renew');
183       $form->end();
184     }
185   }
186  
187   $toggle->end();
188 }
189
190 ////////////////////////////////////////////////////////////
191
192 $am_in_slice = in_array(plc_my_person_id(),$person_ids);
193
194 if ($am_in_slice) {
195   drupal_set_title("My slice " . $name);
196  } else {
197   drupal_set_title("Slice " . $name);
198 }
199
200 $privileges = ( $local_peer && (plc_is_admin()  || plc_is_pi() || $am_in_slice));
201 $tags_privileges = $privileges || plc_is_admin();
202
203 $tabs=array();
204 $tabs [] = tab_nodes_slice($slice_id);
205 $tabs [] = tab_site($site);
206
207 // are these the right privileges for deletion ?
208 if ($privileges) {
209   $tabs["Events"]=array_merge(tablook_event(),
210                               array('url'=>l_event("Slice","slice",$slice_id),
211                                     'bubble'=>"Events for slice $name"));
212   $tabs ['Delete']= array('url'=>l_actions(),
213                           'method'=>'post',
214                           'values'=>array('action'=>'delete-slice','slice_id'=>$slice_id),
215                           'bubble'=>"Delete slice $name",
216                           'confirm'=>"Are you sure to delete slice $name");
217
218   //$tabs["Comon"]=array_merge(tablook_comon(),
219                              //array('url'=>l_comon("slice_id",$slice_id),
220                                    //'bubble'=>"Comon page about slice $name"));
221 }
222
223 plekit_linetabs($tabs);
224
225 ////////////////////////////////////////
226 $peers->block_start($peer_id);
227
228 //////////////////////////////////////// renewal area 
229 // (1) close to expiration : show on top and open
230
231 if ($local_peer ) {
232   $renew_visible = renew_needed ($slice);
233   if ($renew_visible) renew_area ($slice,$site,true);
234  }
235
236
237 //////////////////// details
238 // default for opening the details section or not ?
239 if ($local_peer) {
240   $default_show_details = true;
241  } else {
242   $default_show_details = ! $renew_visible;
243  }
244   
245 $toggle = 
246   new PlekitToggle ('my-slice-details',"Details",
247                     array('bubble'=>
248                           'Display and modify details for that slice',
249                           'visible'=>get_arg('show_details',$default_show_details)));
250 $toggle->start();
251
252 $details=new PlekitDetails($privileges);
253 $details->form_start(l_actions(),array('action'=>'update-slice',
254                                        'slice_id'=>$slice_id,
255                                        'name'=>$name));
256
257 $details->start();
258 if (! $local_peer) {
259   $details->th_td("Peer",$peers->peer_link($peer_id));
260   $details->space();
261  }
262
263
264 $details->th_td('Name',$slice['name']);
265 $details->th_td('Description',$slice['description'],'description',
266                 array('input_type'=>'textarea',
267                       'width'=>50,'height'=>5));
268 $details->th_td('URL',$slice['url'],'url',array('width'=>50));
269 $details->tr_submit("submit","Update Slice");
270 $details->th_td('Expires',$expires);
271 $details->th_td('Instantiation',$slice['instantiation']);
272 $details->th_td("OMF-friendly", ($slice['omf_control'] ? 'Yes' : 'No') . " [to change: see 'omf_control' in the tags section below]");
273 $details->th_td('Site',l_site_obj($site));
274 // xxx show the PIs here
275 //$details->th_td('PIs',...);
276 $details->end();
277
278 $details->form_end();
279 $toggle->end();
280
281 //////////////////// persons
282 $person_columns = array('email','person_id','first_name','last_name','roles');
283 // get persons in slice
284 if (!empty($person_ids))
285   $persons=$api->GetPersons(array('person_id'=>$slice['person_ids']),$person_columns);
286 // just propose to add everyone else
287 // xxx this is maybe too much for admins as it slows stuff down 
288 // as regular persons can see only a fraction of the db anyway
289 $potential_persons=
290   $api->GetPersons(array('~person_id'=>$slice['person_ids'],
291                          'peer_id'=>NULL,
292                          'enabled'=>true),
293                    $person_columns);
294 $count=count($persons);
295
296 if ($profiling) plc_debug_prof('4: persons',count($persons));
297 $toggle=
298   new PlekitToggle ('my-slice-persons',"$count users",
299                     array('bubble'=>
300                           'Manage accounts attached to this slice',
301                           'visible'=>get_arg('show_persons',false)));
302 $toggle->start();
303
304 ////////// people currently in
305 // visible:
306 // hide if both current+add are included
307 // so user can chose which section is of interest
308 // show otherwise
309 $toggle_persons = new PlekitToggle ('my-slice-persons-current',
310                                     "$count people currently in $name",
311                                     array('visible'=>get_arg('show_persons_current',!$privileges)));
312 $toggle_persons->start();
313
314 $headers=array();
315 $headers['email']='string';
316 $headers['first']='string';
317 $headers['last']='string';
318 $headers['R']='string';
319 if ($privileges) $headers[plc_delete_icon()]="none";
320 $table=new PlekitTable('persons',$headers,'0',
321                        array('notes_area'=>false));
322 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
323 $form->start();
324 $table->start();
325 if ($persons) foreach ($persons as $person) {
326   $table->row_start();
327   $table->cell(l_person_obj($person));
328   $table->cell($person['first_name']);
329   $table->cell($person['last_name']);
330   $table->cell(plc_vertical_table ($person['roles']));
331   if ($privileges) $table->cell ($form->checkbox_html('person_ids[]',$person['person_id']));
332   $table->row_end();
333 }
334 // actions area
335 if ($privileges) {
336
337   // remove persons
338   $table->tfoot_start();
339
340   $table->row_start();
341   $table->cell($form->submit_html ("remove-persons-from-slice","Remove selected"),
342                array('hfill'=>true,'align'=>'right'));
343   $table->row_end();
344  }
345 $table->end();
346 $toggle_persons->end();
347
348 ////////// people to add
349 if ($privileges) {
350   $count=count($potential_persons);
351   $toggle_persons = new PlekitToggle ('my-slice-persons-add',
352                                       "$count people may be added to $name",
353                                       array('visible'=>get_arg('show_persons_add',false)));
354   $toggle_persons->start();
355   if ( ! $potential_persons ) {
356     // xxx improve style
357     echo "<p class='not-relevant'>No person to add</p>";
358   } else {
359     $headers=array();
360     $headers['email']='string';
361     $headers['first']='string';
362     $headers['last']='string';
363     $headers['R']='string';
364     $headers['+']="none";
365     $options = array('notes_area'=>false,
366                      'search_width'=>15,
367                      'pagesize'=>8);
368     // show search for admins only as other people won't get that many names to add
369     if ( ! plc_is_admin() ) $options['search_area']=false;
370     
371     $table=new PlekitTable('add_persons',$headers,'0',$options);
372     $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
373     $form->start();
374     $table->start();
375     if ($potential_persons) foreach ($potential_persons as $person) {
376         $table->row_start();
377         $table->cell(l_person_obj($person));
378         $table->cell($person['first_name']);
379         $table->cell($person['last_name']);
380         $table->cell(plc_vertical_table ($person['roles']));
381         $table->cell ($form->checkbox_html('person_ids[]',$person['person_id']));
382         $table->row_end();
383       }
384     // add users
385     $table->tfoot_start();
386     $table->row_start();
387     $table->cell($form->submit_html ("add-persons-in-slice","Add selected"),
388                  array('hfill'=>true,'align'=>'right'));
389     $table->row_end();
390     $table->end();
391     $form->end();
392   }
393   $toggle_persons->end();
394 }
395 $toggle->end();
396
397 //////////////////////////////////////////////////////////// Nodes
398 // the nodes details to display here
399 // (1) we search for the tag types for which 'category' matches 'node*/ui*'
400 // all these tags will then be tentatively displayed in this area
401 // (2) further information can also be optionally specified in the category:
402 //     (.) we split the category with '/' and search for assignments of the form var=value
403 //     (.) header can be set to supersede the column header (default is tagname)
404 //     (.) rank can be used for ordering the columns (default is tagname)
405 //     (.) type is passed to the javascript table, for sorting (default is 'string')
406
407 // minimal list as a start
408 $node_fixed_columns = array('hostname','node_id','peer_id','slice_ids_whitelist', 'site_id',
409                             'run_level','boot_state','last_contact','node_type');
410 // create a VisibleTags object : basically the list of tag columns to show
411 //$visibletags = new VisibleTags ($api, 'node');
412 //$visiblecolumns = $visibletags->column_names();
413
414 // optimizing calls to GetNodes
415 //$all_nodes=$api->GetNodes(NULL,$node_columns);
416 //$slice_nodes=$api->GetNodes(array('node_id'=>$slice['node_ids']),$node_columns);
417 //$potential_nodes=$api->GetNodes(array('~node_id'=>$slice['node_ids']),$node_columns);
418
419
420 //NEW CODE FOR ENABLING COLUMN CONFIGURATION
421
422 //prepare fix and configurable columns
423
424 $fix_columns = array();
425 $fix_columns[]=array('tagname'=>'hostname', 'header'=>'hostname', 'type'=>'string', 'title'=>'The name of the node');
426 $fix_columns[]=array('tagname'=>'peer_id', 'header'=>'AU', 'type'=>'string', 'title'=>'Authority');
427 $fix_columns[]=array('tagname'=>'run_level', 'header'=>'ST', 'type'=>'string', 'title'=>'Status');
428 $fix_columns[]=array('tagname'=>'node_type', 'header'=>'RES', 'type'=>'string', 'title'=>'Reservable');
429
430 // columns that correspond to the visible tags for nodes (*node/ui*)
431 $visibletags = new VisibleTags ($api, 'node');
432 $visibletags->columns();
433 $tag_columns = $visibletags->headers();
434
435 // extra columns that are not tags (for the moment not sorted correctly)
436
437 $extra_columns = array();
438 $extra_columns[]=array('tagname'=>'sitename', 'header'=>'SN', 'type'=>'string', 'title'=>'Site name', 'fetched'=>true);
439 $extra_columns[]=array('tagname'=>'domain', 'header'=>'DN', 'type'=>'string', 'title'=>'Toplevel domain name', 'fetched'=>true);
440 $extra_columns[]=array('tagname'=>'ipaddress', 'header'=>'IP', 'type'=>'string', 'title'=>'IP Address', 'fetched'=>true);
441 $extra_columns[]=array('tagname'=>'fcdistro', 'header'=>'OS', 'type'=>'string', 'title'=>'Operating system', 'fetched'=>false);
442
443 //Get user's column configuration
444
445 $default_configuration = "hostname:f|ST:f|AU:f|RES:f|R|L|OS|MS";
446 $column_configuration = "";
447 $slice_column_configuration = "";
448
449 $show_configuration = "";
450 $show_reservable_message = '1';
451 $show_columns_message = true;
452
453
454 //$PersonTags=$api->GetPersonTags (array('person_id'=>$plc->person['person_id']));
455 $PersonTags=$api->GetPersonTags (array('person_id'=>$plc->person['person_id']));
456 //print_r($PersonTags);
457 foreach ($PersonTags as $ptag) {
458         if ($ptag['tagname'] == 'columnconf')
459         {
460                 $column_configuration = $ptag['value'];
461                 $conf_tag_id = $ptag['person_tag_id'];
462         }
463         if ($ptag['tagname'] == 'showconf')
464         {
465                 $show_configuration = $ptag['value'];
466                 $show_tag_id = $ptag['person_tag_id'];
467         }
468 }
469
470 //print("<br>person column configuration = ".$column_configuration);
471 //print("<br>person show configuration = ".$show_configuration);
472
473 $sliceconf_exists = false;
474 if ($column_configuration == "")
475 {
476         $column_configuration = $slice_id.";default";
477         $sliceconf_exists = true;
478 }
479 else {
480         $slice_conf = explode(";",$column_configuration);
481         for ($i=0; $i<count($slice_conf); $i++ ) {
482                 if ($slice_conf[$i] == $slice_id)
483                 {
484                         $i++;
485                         $slice_column_configuration = $slice_conf[$i];
486                         $sliceconf_exists = true;
487                         break;
488                 }
489                 else
490                 {
491                         $i++;
492                         $slice_column_configuration = $slice_conf[$i];
493                 }
494         }        
495 }
496
497 if ($sliceconf_exists == false)
498         $column_configuration = $column_configuration.";".$slice_id.";default";
499
500 //print("<br>slice configuration = ".$slice_column_configuration);
501
502
503 if ($slice_column_configuration == "")
504         $full_configuration = $default_configuration;
505 else
506         $full_configuration = $default_configuration."|".$slice_column_configuration;
507
508
509 //instantiate the column configuration class, which prepares the headers array
510 $ConfigureColumns =new PlekitColumns($full_configuration, $fix_columns, $tag_columns, $extra_columns);
511
512 $visiblecolumns = $ConfigureColumns->node_tags();
513
514 $node_columns=array_merge($node_fixed_columns,$visiblecolumns);
515 //print_r($node_columns);
516 $all_nodes=$api->GetNodes(NULL,$node_columns);
517
518 //print("<br>person show configuration = ".$show_configuration);
519
520 $show_conf = explode(";",$show_configuration);
521 foreach ($show_conf as $ss) {
522         if ($ss =="reservable")
523                 $show_reservable_message = '0';
524         if ($ss =="columns")
525                 $show_columns_message = false;
526 }        
527
528 //print("res:".$show_reservable_message." - cols:".$show_columns_message);
529
530 $slice_nodes=array();
531 $potential_nodes=array();
532 $reservable_nodes=array();
533 foreach ($all_nodes as $node) {
534   if (in_array($node['node_id'],$slice['node_ids'])) {
535     $slice_nodes[]=$node;
536     if ($node['node_type']=='reservable') $reservable_nodes[]=$node;
537   } else {
538     $potential_nodes[]=$node;
539   }
540 }
541 if ($profiling) plc_debug_prof('5: nodes',count($slice_nodes));
542 ////////////////////
543 // outline the number of reservable nodes
544 $nodes_message=count_english($slice_nodes,"node");
545 if (count($reservable_nodes)) $nodes_message .= " (" . count($reservable_nodes) . " reservable)";
546 $toggle=new PlekitToggle ('my-slice-nodes',$nodes_message,
547                           array('bubble'=>
548                                 'Manage nodes attached to this slice',
549                                 'visible'=>get_arg('show_nodes',false)));
550 $toggle->start();
551
552
553 //////////////////// reservable nodes area
554
555 $count=count($reservable_nodes);
556 if ($count && $privileges) {
557   // having reservable nodes in white lists looks a bit off scope for now...
558   $toggle_nodes=new PlekitToggle('my-slice-nodes-reserve',
559                                  "Leases - " . count($reservable_nodes) . " reservable node(s)",
560                                  array('visible'=>$show_reservable_message, 'info_div'=>'note_reservable_div'));
561   $toggle_nodes->start();
562
563 if ($show_reservable_message) 
564 $note_display = "";
565 else
566 $note_display = "display:none;";
567
568 ////////// show a notice to people having attached a reservable node
569 if (count($reservable_nodes) && $privileges) {
570   $mark=reservable_mark();
571   print <<<EOF
572 <br>
573 <div id='note_reservable_div' style="align:center; background-color:#CAE8EA; padding:4px; width:800px; $note_display">
574 <table align=center><tr><td valign=top>
575 You have attached one or more <span class='bold'>reservable nodes</span> to your slice. 
576 Reservable nodes show up with the '$mark' mark. 
577 Your slivers will be available <span class='bold'>only during timeslots
578 where you have obtained leases</span>. 
579 You can manage your leases in the tab below.
580 <br>
581 Please note that as of August 2010 this feature is experimental. 
582 Feedback is appreciated at <a href="mailto:devel@planet-lab.org">devel@planet-lab.org</a>
583 </td><td valign=top><span onClick=closeMessage('reservable')><img class='reset' src="/planetlab/icons/clear.png" alt="hide message"></span>
584 </td></tr></table>
585 </div>
586 EOF;
587 }  
588
589   $grain=$api->GetLeaseGranularity();
590   if ($profiling) plc_debug_prof('6 granul',$grain);
591   // where to start from, expressed as an offset in hours from now
592   $resa_offset=$_GET['resa_offset'];
593   if ( ! $resa_offset ) $resa_offset=0;
594   $rough_start=time()+$resa_offset*3600;
595   // xxx should be configurable
596   $resa_slots=$_GET['resa_slots'];
597   if ( ! $resa_slots ) $resa_slots = 36;
598   // for now, show the next 72 hours, or 72 grains, which ever is smaller
599   $duration=$resa_slots*$grain;
600   $steps=$duration/$grain;
601   $start=intval($rough_start/$grain)*$grain;
602   $end=$rough_start+$duration;
603   $lease_columns=array('lease_id','name','t_from','t_until','hostname','name');
604   $leases=$api->GetLeases(array(']t_until'=>$rough_start,'[t_from'=>$end,'-SORT'=>'t_from'),$lease_columns);
605   if ($profiling) plc_debug_prof('7 leases',count($leases));
606   // hash nodes -> leases
607   $host_hash=array();
608   foreach ($leases as $lease) {
609     $hostname=$lease['hostname'];
610     if ( ! $host_hash[$hostname] ) {
611         $host_hash[$hostname]=array();
612     }
613     // resync within the table
614     $lease['nfrom']=($lease['t_from']-$start)/$grain;
615     $lease['nuntil']=($lease['t_until']-$start)/$grain;
616     $host_hash[$hostname] []= $lease;
617   }
618   # leases_data is the name used by leases.js to locate this table
619   echo "<table id='leases_data'>";
620   # pass (slice_id,slicename) as the [0,0] coordinate as thead>tr>td
621   echo "<thead><tr><td>" . $slice['slice_id'] . '&' . $slice['name'] . "</td>";
622   # the timeslot headers read (timestamp,label)
623   $day_names=array('Su','M','Tu','W','Th','F','Sa');
624   for ($i=0; $i<$steps; $i++) {
625     $timestamp=($start+$i*$grain);
626     $day=$day_names[intval(strftime("%w",$timestamp))];
627     $label=$day . strftime(" %H:%M",$timestamp);
628     // expose in each header cell the full timestamp, and how to display it - use & as a separator*/
629     echo "<th>" . implode("&",array($timestamp,$label)) . "</th>";
630   }
631   echo "</tr></thead><tbody>";
632   // todo - sort on hostnames
633   function sort_hostname ($a,$b) { return ($a['hostname']<$b['hostname'])?-1:1;}
634   usort($reservable_nodes,sort_hostname);
635   foreach ($reservable_nodes as $node) {
636     echo "<tr><th scope='row'>". $node['hostname'] . "</th>";
637     $hostname=$node['hostname'];
638     $leases=$host_hash[$hostname];
639     $counter=0;
640     while ($counter<$steps) {
641       if ($leases && ($leases[0]['nfrom']<=$counter)) {
642         $lease=array_shift($leases);
643         /* nicer display, merge two consecutive leases for the same slice 
644            avoid doing that for now, as it might makes things confusing */
645         /* while ($leases && ($leases[0]['name']==$lease['name']) && ($leases[0]['nfrom']==$lease['nuntil'])) {
646           $lease['nuntil']=$leases[0]['nuntil'];
647           array_shift($leases);
648           }*/
649         $duration=$lease['nuntil']-$counter;
650         echo "<td colspan='$duration'>" . $lease['lease_id'] . '&' . $lease['name'] . "</td>";
651         $counter=$lease['nuntil']; 
652       } else {
653         echo "<td></td>";
654         $counter+=1;
655       }
656     }
657     echo "</tr>";
658   }
659   echo "</tbody></table>\n";
660
661   // the general layout for the scheduler
662   echo <<< EOF
663 <div id='leases_area'></div>
664
665 <div id='leases_buttons'>
666   <button id='leases_clear' type='submit'>Clear</button>
667   <button id='leases_submit' type='submit'>Submit</button>
668 </div>
669 EOF;
670
671   $toggle_nodes->end();
672  }
673
674
675 //////////////////// node configuration panel
676
677 if ($show_columns_message) 
678 $column_conf_visible = '1';
679 else
680 $column_conf_visible = '0';
681
682
683 $toggle_nodes=new PlekitToggle('my-slice-nodes-configuration',
684                                "Node table layout",
685                                array('visible'=>$column_conf_visible, 'info_div'=>'note_columns_div'));
686 $toggle_nodes->start();
687
688 //usort ($table_headers, create_function('$col1,$col2','return strcmp($col1["header"],$col2["header"]);'));
689 //print("<p>TABLE HEADERS<p>");
690 //print_r($table_headers);
691
692 print("<div id='debug'></div>");
693 print("<input type='hidden' id='slice_id' value='".$slice['slice_id']."' />");
694 print("<input type='hidden' id='person_id' value='".$plc->person['person_id']."' />");
695 print("<input type='hidden' id='conf_tag_id' value='".$conf_tag_id."' />");
696 print("<input type='hidden' id='show_tag_id' value='".$show_tag_id."' />");
697 print("<input type='hidden' id='show_configuration' value='".$show_configuration."' />");
698 print("<input type='hidden' id='column_configuration' value='".$slice_column_configuration."' />");
699 print("<br><input type='hidden' size=80 id='full_column_configuration' value='".$column_configuration."' />");
700 print("<input type='hidden' id='previousConf' value='".$slice_column_configuration."'></input>");
701 print("<input type='hidden' id='defaultConf' value='".$default_configuration."'></input>");
702
703 //print ("showing column message = ".$show_columns_message);
704 if ($show_columns_message) 
705 $note_display = "";
706 else
707 $note_display = "display:none;";
708
709   print <<<EOF
710 <div id='note_columns_div' style="align:center; background-color:#CAE8EA; padding:4px; width:800px; $note_display">
711 <table align=center><tr><td valign=top>
712 This tab allows you to customize the columns in the node tables, below. Information on the nodes comes from a variety of monitoring sources. If you, as either a user or a provider of monitoring data, would like to see additional columns made available, please send us your request in mail to <a href="mailto:devel@planet-lab.org">devel@planet-lab.org</a>
713 </td><td valign=top><span onClick=closeMessage('columns')><img class='reset' src="/planetlab/icons/clear.png" alt="hide message permanently"></span>
714 </td></tr></table>
715 </div>
716 EOF;
717
718 $ConfigureColumns->configuration_panel_html(true);
719
720 $ConfigureColumns->javascript_init();
721
722 $toggle_nodes->end();
723
724
725 $all_sites=$api->GetSites(NULL, array('site_id','login_base'));
726 $site_hash=array();
727 foreach ($all_sites as $tmp_site) $site_hash[$tmp_site['site_id']]=$tmp_site['login_base'];
728
729 $interface_columns=array('ip','node_id','interface_id');
730 $interface_filter=array('is_primary'=>TRUE);
731 $interfaces=$api->GetInterfaces($interface_filter,$interface_columns);
732
733 $interface_hash=array();
734 foreach ($interfaces as $interface) $interface_hash[$interface['node_id']]=$interface;
735
736
737
738
739
740 //////////////////// nodes currently in
741 $toggle_nodes=new PlekitToggle('my-slice-nodes-current',
742                                count_english($slice_nodes,"node") . " currently in $name",
743                                array('visible'=>get_arg('show_nodes_current',!$privileges)));
744 $toggle_nodes->start();
745
746 $headers=array();
747 $notes=array();
748 //$notes=array_merge($notes,$visibletags->notes());
749 $notes [] = "For information about the different columns please see the <b>node table layout</b> tab above or <b>mouse over</b> the column headers";
750
751 /*
752 $headers['peer']='string';
753 $headers['hostname']='string';
754 $short="-S-"; $long=Node::status_footnote(); $type='string'; 
755         $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
756 $short=reservable_mark(); $long=reservable_legend(); $type='string';
757         $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
758 // the extra tags, configured for the UI
759 $headers=array_merge($headers,$visibletags->headers());
760
761 if ($privileges) $headers[plc_delete_icon()]="none";
762 */
763
764 $edit_header = array();
765 if ($privileges) $edit_header[plc_delete_icon()]="none";
766 $headers = array_merge($ConfigureColumns->get_headers(),$edit_header);
767
768 //print("<p>HEADERS<p>");
769 //print_r($headers);
770
771 $table_options = array('notes'=>$notes,
772                        'search_width'=>15,
773                        'pagesize'=>20,
774                         'configurable'=>true);
775
776 $table=new PlekitTable('nodes',$headers,NULL,$table_options);
777
778 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
779 $form->start();
780 $table->start();
781 if ($slice_nodes) foreach ($slice_nodes as $node) {
782   $table->row_start();
783
784 $table->cell($node['node_id'], array('display'=>'none'));
785
786   $table->cell(l_node_obj($node));
787   $peers->cell($table,$node['peer_id']);
788   $run_level=$node['run_level'];
789   list($label,$class) = Node::status_label_class_($node);
790   $table->cell ($label,array('class'=>$class));
791   $table->cell( ($node['node_type']=='reservable')?reservable_mark():"" );
792
793   $hostname=$node['hostname'];
794   $ip=$interface_hash[$node['node_id']]['ip'];
795   $interface_id=$interface_hash[$node['node_id']]['interface_id'];
796
797 //extra columns
798 $node['domain'] = topdomain($hostname);
799 $node['sitename'] = l_site_t($node['site_id'],$site_hash[$node['site_id']]);
800 $node['ipaddress'] = l_interface_t($interface_id,$ip);
801
802
803  //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
804  $ConfigureColumns->cells($table, $node);
805
806   if ($privileges) $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
807   $table->row_end();
808 }
809 // actions area
810 if ($privileges) {
811
812   // remove nodes
813   $table->tfoot_start();
814
815   $table->row_start();
816   $table->cell($form->submit_html ("remove-nodes-from-slice","Remove selected"),
817                array('hfill'=>true,'align'=>'right'));
818   $table->row_end();
819  }
820 $table->end();
821 $toggle_nodes->end();
822
823 //////////////////// nodes to add
824 if ($privileges) {
825   $new_potential_nodes = array();
826   if ($potential_nodes) foreach ($potential_nodes as $node) {
827       $emptywl=empty($node['slice_ids_whitelist']);
828       $inwl = (!emptywl) and in_array($slice['slice_id'],$node['slice_ids_whitelist']);
829       if ($emptywl or $inwl)
830         $new_potential_nodes[]=$node;
831   }
832   $potential_nodes=$new_potential_nodes;
833
834   $count=count($potential_nodes);
835   $toggle_nodes=new PlekitToggle('my-slice-nodes-add',
836                                  count_english($potential_nodes,"more node") . " available",
837                                  array('visible'=>get_arg('show_nodes_add',false)));
838   $toggle_nodes->start();
839
840   if ( $potential_nodes ) {
841     $headers=array();
842     $notes=array();
843
844
845 /*
846     $headers['peer']='string';
847     $headers['hostname']='string';
848     $short="-S-"; $long=Node::status_footnote(); $type='string'; 
849         $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
850         $short=reservable_mark(); $long=reservable_legend(); $type='string';
851         $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
852     // the extra tags, configured for the UI
853     $headers=array_merge($headers,$visibletags->headers());
854     $headers['+']="none";
855 */
856
857     $add_header = array();
858     $add_header['+']="none";
859     $headers = array_merge($ConfigureColumns->get_headers(),$add_header);
860
861     //$notes=array_merge($notes,$visibletags->notes());
862 $notes [] = "For information about the different columns please see the <b>node table layout</b> tab above or <b>mouse over</b> the column headers";
863     
864     $table=new PlekitTable('add_nodes',$headers,NULL, $table_options);
865     $form=new PlekitForm(l_actions(),
866                          array('slice_id'=>$slice['slice_id']));
867     $form->start();
868     $table->start();
869     if ($potential_nodes) foreach ($potential_nodes as $node) {
870         $table->row_start();
871
872 $table->cell($node['node_id'], array('display'=>'none'));
873
874         $table->cell(l_node_obj($node));
875         $peers->cell($table,$node['peer_id']);
876         list($label,$class) = Node::status_label_class_($node);
877         $table->cell ($label,array('class'=>$class));
878         $table->cell( ($node['node_type']=='reservable')?reservable_mark():"" );
879
880         //extra columns
881           $hostname=$node['hostname'];
882           $ip=$interface_hash[$node['node_id']]['ip'];
883           $interface_id=$interface_hash[$node['node_id']]['interface_id'];
884         $node['domain'] = topdomain($hostname);
885         $node['sitename'] = l_site_t($node['site_id'],$site_hash[$node['site_id']]);
886         $node['ipaddress'] = l_interface_t($interface_id,$ip);
887
888         //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
889         $ConfigureColumns->cells($table, $node);
890
891         $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
892         $table->row_end();
893       }
894     // add nodes
895     $table->tfoot_start();
896     $table->row_start();
897     $table->cell($form->submit_html ("add-nodes-in-slice","Add selected"),
898                  array('hfill'=>true,'align'=>'right'));
899     $table->row_end();
900     $table->end();
901     $form->end();
902   }
903   $toggle_nodes->end();
904 }
905
906 $toggle->end();
907
908 // very wide values get abbreviated
909 $tag_value_threshold=24;
910 //////////////////////////////////////////////////////////// Tags
911 //if ( $local_peer ) {
912   $tags=$api->GetSliceTags (array('slice_id'=>$slice_id));
913   if ($profiling) plc_debug_prof('8 slice tags',count($tags));
914   function get_tagname ($tag) { return $tag['tagname'];}
915   $tagnames = array_map ("get_tagname",$tags);
916   
917   $toggle = new PlekitToggle ('slice-tags',count_english_warning($tags,'tag'),
918                               array('bubble'=>'Inspect and set tags on tat slice',
919                                     'visible'=>get_arg('show_tags',false)));
920   $toggle->start();
921   
922   $headers=array(
923     "Name"=>"string",
924     "Value"=>"string",
925     "Node"=>"string",
926     "NodeGroup"=>"string");
927   if ($tags_privileges) $headers[plc_delete_icon()]="none";
928   
929   $table_options=array("notes_area"=>false,"pagesize_area"=>false,"search_width"=>10);
930   $table=new PlekitTable("slice_tags",$headers,'0',$table_options);
931   $form=new PlekitForm(l_actions(),
932                        array('slice_id'=>$slice['slice_id']));
933   $form->start();
934   $table->start();
935   if ($tags) {
936     foreach ($tags as $tag) {
937       $node_name = "ALL";
938       if ($tag['node_id']) {
939         $tag_nodes = $api->GetNodes(array('node_id'=>$tag['node_id']));
940         if ($profiling) plc_debug_prof('9 node for slice tag',count($tag_nodes));
941         if($tag_nodes) {
942           $node = $tag_nodes[0];
943           $node_name = $node['hostname'];
944         }
945       }
946       $nodegroup_name="n/a";
947       if ($tag['nodegroup_id']) { 
948         $nodegroups=$api->GetNodeGroups(array('nodegroup_id'=>$tag['nodegroup_id']));
949         if ($profiling) plc_debug_prof('10 nodegroup for slice tag',$nodegroup);
950         if ($nodegroup) {
951           $nodegroup = $nodegroups[0];
952           $nodegroup_name = $nodegroup['groupname'];
953         }
954       }
955       $table->row_start();
956       $table->cell(l_tag_obj($tag));
957       // very wide values get abbreviated
958       $table->cell(truncate_and_popup($tag['value'],$tag_value_threshold));
959       $table->cell($node_name);
960       $table->cell($nodegroup_name);
961       if ($tags_privileges) $table->cell ($form->checkbox_html('slice_tag_ids[]',$tag['slice_tag_id']));
962       $table->row_end();
963     }
964   }
965   if ($tags_privileges) {
966     $table->tfoot_start();
967     $table->row_start();
968     $table->cell($form->submit_html ("delete-slice-tags","Remove selected"),
969                  array('hfill'=>true,'align'=>'right'));
970     $table->row_end();
971     
972     $table->row_start();
973     function tag_selector ($tag) {
974       return array("display"=>$tag['tagname'],"value"=>$tag['tag_type_id']);
975     }
976     $all_tags= $api->GetTagTypes( array ("category"=>"slice*","-SORT"=>"+tagname"), array("tagname","tag_type_id"));
977     if ($profiling) plc_debug_prof('11 tagtypes',count($all_tags));
978     $selector_tag=array_map("tag_selector",$all_tags);
979     
980     function node_selector($node) { 
981       return array("display"=>$node["hostname"],"value"=>$node['node_id']);
982     }
983     $selector_node=array_map("node_selector",$slice_nodes);
984     
985     function nodegroup_selector($ng) {
986       return array("display"=>$ng["groupname"],"value"=>$ng['nodegroup_id']);
987     }
988     $all_nodegroups = $api->GetNodeGroups( array("groupname"=>"*"), array("groupname","nodegroup_id"));
989     if ($profiling) plc_debug_prof('13 nodegroups',count($all_nodegroups));
990     $selector_nodegroup=array_map("nodegroup_selector",$all_nodegroups);
991     
992     $table->cell($form->select_html("tag_type_id",$selector_tag,array('label'=>"Choose Tag")));
993     $table->cell($form->text_html("value","",array('width'=>8)));
994     $table->cell($form->select_html("node_id",$selector_node,array('label'=>"All Nodes")));
995     $table->cell($form->select_html("nodegroup_id",$selector_nodegroup,array('label'=>"No Nodegroup")));
996     $table->cell($form->submit_html("add-slice-tag","Set Tag"),array('columns'=>2,'align'=>'left'));
997     $table->row_end();
998   }
999     
1000   $table->end();
1001   $form->end();
1002   $toggle->end();
1003 //}
1004
1005
1006 //////////////////////// renew slice
1007 if ($local_peer ) {
1008   if ( ! $renew_visible) renew_area ($slice,$site,false);
1009  }
1010
1011 $peers->block_end($peer_id);
1012
1013 if ($profiling) plc_debug_prof_end();
1014
1015 // Print footer
1016 include 'plc_footer.php';
1017
1018 ?>