4 require_once 'plc_login.php';
6 // Get session and API handles
7 require_once 'plc_session.php';
11 require_once 'plc_drupal.php';
12 include 'plc_header.php';
15 require_once 'plc_functions.php';
16 require_once 'plc_peers.php';
17 require_once 'plc_objects.php';
18 require_once 'plc_visibletags2.php';
19 require_once 'linetabs.php';
20 require_once 'table2.php';
21 require_once 'details.php';
22 require_once 'toggle.php';
23 require_once 'form.php';
24 require_once 'raphael.php';
25 require_once 'columns.php';
27 // keep css separate for now
28 drupal_set_html_head('
29 <link href="/planetlab/css/my_slice.css" rel="stylesheet" type="text/css" />
30 <script src="/planetlab/slices/leases.js" type="text/javascript" charset="utf-8"></script>
33 // -------------------- admins potentially need to get full list of users
34 ini_set('memory_limit','32M');
37 if ($_GET['profiling']) $profiling=true;
39 if ($profiling) plc_debug_prof_start();
41 // --------------------
42 // recognized URL arguments
43 $slice_id=intval($_GET['id']);
44 if ( ! $slice_id ) { plc_error('Malformed URL - id not set'); return; }
47 // have to name columns b/c we need the non-native 'omf_control' column
48 $slice_columns=array('slice_id','name','peer_id','site_id','person_ids','node_ids','expires',
49 'url','description','instantiation','omf_control');
50 $slices= $api->GetSlices( array($slice_id), $slice_columns);
53 drupal_set_message ("Slice " . $slice_id . " not found");
59 if ($profiling) plc_debug_prof('2: slice',count($slices));
60 // pull all node info to vars
61 $name= $slice['name'];
62 $expires = date( "d/m/Y", $slice['expires'] );
63 $site_id= $slice['site_id'];
65 $person_ids=$slice['person_ids'];
68 $peer_id= $slice['peer_id'];
69 $peers=new Peers ($api);
70 $local_peer = ! $peer_id;
72 if ($profiling) plc_debug_prof('3: peers',count($peers));
75 $sites= $api->GetSites( array( $site_id ) );
77 $site_name= $site['name'];
78 $max_slices = $site['max_slices'];
80 if ($profiling) plc_debug_prof('4: sites',count($sites));
81 //////////////////////////////////////// building blocks for the renew area
83 global $DAY; $DAY = 24*60*60;
84 global $WEEK; $WEEK = 7 * $DAY;
85 global $MAX_WEEKS; $MAX_WEEKS= 8; // weeks from today
86 global $GRACE_DAYS; $GRACE_DAYS=10; // days for renewal promoted on top
87 global $NOW; $NOW=mktime();
89 ////////////////////////////////////////////////////////////
90 // make the renew area on top and open if the expiration time is less than 10 days from now
91 function renew_needed ($slice) {
92 global $DAY, $NOW, $GRACE_DAYS;
93 $current_exp=$slice['expires'];
95 $time_left = $current_exp - $NOW;
96 $visible = $time_left/$DAY <= $GRACE_DAYS;
100 function renew_area ($slice,$site,$visible) {
101 global $DAY, $WEEK, $MAX_WEEKS, $GRACE_DAYS, $NOW;
103 $current_exp=$slice['expires'];
104 $current_text = gmstrftime("%A %b-%d-%y %T %Z", $current_exp);
105 $max_exp= $NOW + ($MAX_WEEKS * $WEEK); // seconds since epoch
106 $max_text = gmstrftime("%A %b-%d-%y %T %Z", $max_exp);
108 // xxx some extra code needed to enable this area only if the slice description is OK:
109 // description and url must be non void
111 new PlekitToggle('renew',"Expires $current_text - Renew this slice",
113 "Enter this zone if you wish to renew your slice",
114 'visible'=>$visible));
117 // xxx message could take roles into account
118 if ($site['max_slices']<=0) {
120 <p class='my-slice-renewal'>Slice creation and renewal have been temporarily disabled for your
121 <site. This may have occurred because your site's nodes have been down
122 or unreachable for several weeks, and multiple attempts to contact
123 your site's PI(s) and Technical Contact(s) have all failed. If so,
124 contact your site's PI(s) and Technical Contact(s) and ask them to
125 bring up your site's nodes. Please visit your <a
126 href='/db/sites/index.php?id=$site_id'>site details</a> page to find
127 out more about your site's nodes, and how to contact your site's PI(s)
128 and Technical Contact(s).</p>
133 // xxx this is a rough cut and paste from the former UI
134 // showing a datepicker view could be considered as well with some extra work
135 // calculate possible extension lengths
136 $selectors = array();
137 foreach ( array ( 1 => "One more week",
138 2 => "Two more weeks",
139 3 => "Three more weeks",
140 4 => "One more month" ) as $weeks => $text ) {
141 $candidate_exp = $current_exp + $weeks*$WEEK;
142 if ( $candidate_exp < $max_exp) {
143 $selectors []= array('display'=>"$text (" . gmstrftime("%A %b-%d-%y %T %Z", $candidate_exp) . ")",
144 'value'=>$candidate_exp);
145 $max_renewal_weeks=$weeks;
146 $max_renewal_date= gmstrftime("%A %b-%d-%y %T %Z", $candidate_exp);
150 if ( empty( $selectors ) ) {
152 <div class='my-slice-renewal'>
153 Slices cannot be renewed more than $MAX_WEEKS weeks from now, i.e. not beyond $max_text.
154 For this reason, the current slice cannot be renewed any further into the future, try again closer to expiration date.
159 <div class='my-slice-renewal'>
160 <span class='bold'>Important:</span> Please take this opportunity to review and update your slice information in the Details tab.
162 PlanetLab's security model requires that anyone who is concerned about a slice's activity be able to immediately learn about that slice. The details that you provide are your public explanation about why the slice behaves as it does. Be sure to describe the <span class='bold'>kind of traffic</span> that your slice generates, and how it handles material that is under <span class='bold'>copyright</span>, if relevant.
164 The PlanetLab Operations Centres regularly respond to concerns raised by third parties about site behaviour. Most incidents are resolved rapidly based upon the publicly posted slice details. However, when these details are not sufficiently clear or accurate, and we cannot immediately reach the slice owner, we must delete the slice.
168 $form = new PlekitForm (l_actions(),
169 array('action'=>'renew-slice',
170 'slice_id'=>$slice['slice_id']));
172 print $form->label_html('expires','Duration: ');
173 print $form->select_html('expires',$selectors,array('label'=>'Pick one'));
174 print $form->submit_html('renew-button','Renew');
177 print("<p><i>NOTE: Slices cannot be renewed beyond another $max_renewal_weeks week(s) ($max_renewal_date).</i> </p>");
185 ////////////////////////////////////////////////////////////
187 $am_in_slice = in_array(plc_my_person_id(),$person_ids);
190 drupal_set_title("My slice " . $name);
192 drupal_set_title("Slice " . $name);
195 $privileges = ( $local_peer && (plc_is_admin() || plc_is_pi() || $am_in_slice));
196 $tags_privileges = $privileges || plc_is_admin();
199 $tabs [] = tab_nodes_slice($slice_id);
200 $tabs [] = tab_site($site);
202 // are these the right privileges for deletion ?
204 $tabs ['Delete']= array('url'=>l_actions(),
206 'values'=>array('action'=>'delete-slice','slice_id'=>$slice_id),
207 'bubble'=>"Delete slice $name",
208 'confirm'=>"Are you sure to delete slice $name");
210 $tabs["Events"]=array_merge(tablook_event(),
211 array('url'=>l_event("Slice","slice",$slice_id),
212 'bubble'=>"Events for slice $name"));
213 $tabs["Comon"]=array_merge(tablook_comon(),
214 array('url'=>l_comon("slice_id",$slice_id),
215 'bubble'=>"Comon page about slice $name"));
218 plekit_linetabs($tabs);
220 ////////////////////////////////////////
221 $peers->block_start($peer_id);
223 //////////////////////////////////////// renewal area
224 // (1) close to expiration : show on top and open
227 $renew_visible = renew_needed ($slice);
228 if ($renew_visible) renew_area ($slice,$site,true);
232 //////////////////// details
233 // default for opening the details section or not ?
235 $default_show_details = true;
237 $default_show_details = ! $renew_visible;
241 new PlekitToggle ('my-slice-details',"Details",
243 'Display and modify details for that slice',
244 'visible'=>get_arg('show_details',$default_show_details)));
247 $details=new PlekitDetails($privileges);
248 $details->form_start(l_actions(),array('action'=>'update-slice',
249 'slice_id'=>$slice_id,
254 $details->th_td("Peer",$peers->peer_link($peer_id));
259 $details->th_td('Name',$slice['name']);
260 $details->th_td('Description',$slice['description'],'description',
261 array('input_type'=>'textarea',
262 'width'=>50,'height'=>5));
263 $details->th_td('URL',$slice['url'],'url',array('width'=>50));
264 $details->tr_submit("submit","Update Slice");
265 $details->th_td('Expires',$expires);
266 $details->th_td('Instantiation',$slice['instantiation']);
267 $details->th_td("OMF-friendly", ($slice['omf_control'] ? 'Yes' : 'No') . " [to change: see 'omf_control' in the tags section below]");
268 $details->th_td('Site',l_site_obj($site));
269 // xxx show the PIs here
270 //$details->th_td('PIs',...);
273 $details->form_end();
276 //////////////////// persons
277 $person_columns = array('email','person_id','first_name','last_name','roles');
278 // get persons in slice
279 if (!empty($person_ids))
280 $persons=$api->GetPersons(array('person_id'=>$slice['person_ids']),$person_columns);
281 // just propose to add everyone else
282 // xxx this is maybe too much for admins as it slows stuff down
283 // as regular persons can see only a fraction of the db anyway
285 $api->GetPersons(array('~person_id'=>$slice['person_ids'],
289 $count=count($persons);
291 if ($profiling) plc_debug_prof('4: persons',count($persons));
293 new PlekitToggle ('my-slice-persons',"$count users",
295 'Manage accounts attached to this slice',
296 'visible'=>get_arg('show_persons',false)));
299 ////////// people currently in
301 // hide if both current+add are included
302 // so user can chose which section is of interest
304 $toggle_persons = new PlekitToggle ('my-slice-persons-current',
305 "$count people currently in $name",
306 array('visible'=>get_arg('show_persons_current',!$privileges)));
307 $toggle_persons->start();
310 $headers['email']='string';
311 $headers['first']='string';
312 $headers['last']='string';
313 $headers['R']='string';
314 if ($privileges) $headers[plc_delete_icon()]="none";
315 $table=new PlekitTable('persons',$headers,'0',
316 array('notes_area'=>false));
317 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
320 if ($persons) foreach ($persons as $person) {
322 $table->cell(l_person_obj($person));
323 $table->cell($person['first_name']);
324 $table->cell($person['last_name']);
325 $table->cell(plc_vertical_table ($person['roles']));
326 if ($privileges) $table->cell ($form->checkbox_html('person_ids[]',$person['person_id']));
333 $table->tfoot_start();
336 $table->cell($form->submit_html ("remove-persons-from-slice","Remove selected"),
337 array('hfill'=>true,'align'=>'right'));
341 $toggle_persons->end();
343 ////////// people to add
345 $count=count($potential_persons);
346 $toggle_persons = new PlekitToggle ('my-slice-persons-add',
347 "$count people may be added to $name",
348 array('visible'=>get_arg('show_persons_add',false)));
349 $toggle_persons->start();
350 if ( ! $potential_persons ) {
352 echo "<p class='not-relevant'>No person to add</p>";
355 $headers['email']='string';
356 $headers['first']='string';
357 $headers['last']='string';
358 $headers['R']='string';
359 $headers['+']="none";
360 $options = array('notes_area'=>false,
363 // show search for admins only as other people won't get that many names to add
364 if ( ! plc_is_admin() ) $options['search_area']=false;
366 $table=new PlekitTable('add_persons',$headers,'0',$options);
367 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
370 if ($potential_persons) foreach ($potential_persons as $person) {
372 $table->cell(l_person_obj($person));
373 $table->cell($person['first_name']);
374 $table->cell($person['last_name']);
375 $table->cell(plc_vertical_table ($person['roles']));
376 $table->cell ($form->checkbox_html('person_ids[]',$person['person_id']));
380 $table->tfoot_start();
382 $table->cell($form->submit_html ("add-persons-in-slice","Add selected"),
383 array('hfill'=>true,'align'=>'right'));
388 $toggle_persons->end();
392 //////////////////////////////////////////////////////////// Nodes
393 // the nodes details to display here
394 // (1) we search for the tag types for which 'category' matches 'node*/ui*'
395 // all these tags will then be tentatively displayed in this area
396 // (2) further information can also be optionally specified in the category:
397 // (.) we split the category with '/' and search for assignments of the form var=value
398 // (.) header can be set to supersede the column header (default is tagname)
399 // (.) rank can be used for ordering the columns (default is tagname)
400 // (.) type is passed to the javascript table, for sorting (default is 'string')
402 // minimal list as a start
403 $node_fixed_columns = array('hostname','node_id','peer_id','slice_ids_whitelist', 'site_id',
404 'run_level','boot_state','last_contact','node_type');
405 // create a VisibleTags object : basically the list of tag columns to show
406 //$visibletags = new VisibleTags ($api, 'node');
407 //$visiblecolumns = $visibletags->column_names();
409 // optimizing calls to GetNodes
410 //$all_nodes=$api->GetNodes(NULL,$node_columns);
411 //$slice_nodes=$api->GetNodes(array('node_id'=>$slice['node_ids']),$node_columns);
412 //$potential_nodes=$api->GetNodes(array('~node_id'=>$slice['node_ids']),$node_columns);
415 //NEW CODE FOR ENABLING COLUMN CONFIGURATION
417 //prepare fix and configurable columns
419 $fix_columns = array();
420 $fix_columns[]=array('tagname'=>'hostname', 'header'=>'hostname', 'type'=>'string', 'title'=>'The name of the node');
421 $fix_columns[]=array('tagname'=>'peer_id', 'header'=>'AU', 'type'=>'string', 'title'=>'Authority');
422 $fix_columns[]=array('tagname'=>'run_level', 'header'=>'ST', 'type'=>'string', 'title'=>'Status');
423 $fix_columns[]=array('tagname'=>'node_type', 'header'=>'RES', 'type'=>'string', 'title'=>'Reservable');
425 // columns that correspond to the visible tags for nodes (*node/ui*)
426 $visibletags = new VisibleTags ($api, 'node');
427 $visibletags->columns();
428 $tag_columns = $visibletags->headers();
430 // extra columns that are not tags (for the moment not sorted correctly)
432 $extra_columns = array();
433 $extra_columns[]=array('tagname'=>'sitename', 'header'=>'SN', 'type'=>'string', 'title'=>'Site name', 'fetched'=>true);
434 $extra_columns[]=array('tagname'=>'domain', 'header'=>'DN', 'type'=>'string', 'title'=>'Toplevel domain name', 'fetched'=>true);
435 $extra_columns[]=array('tagname'=>'ipaddress', 'header'=>'IP', 'type'=>'string', 'title'=>'IP Address', 'fetched'=>true);
436 $extra_columns[]=array('tagname'=>'fcdistro', 'header'=>'OS', 'type'=>'string', 'title'=>'Operating system', 'fetched'=>false);
438 //Get user's column configuration
440 $first_time_configuration = false;
441 $default_configuration = "hostname:f|ST:f|AU:f|RES:f";
442 //$extra_default = "";
443 $column_configuration = "";
444 $slice_column_configuration = "";
446 $show_configuration = "";
447 $show_reservable_message = '1';
448 $show_columns_message = '1';
451 //$PersonTags=$api->GetPersonTags (array('person_id'=>$plc->person['person_id']));
452 $PersonTags=$api->GetPersonTags (array('person_id'=>$plc->person['person_id']));
453 //print_r($PersonTags);
454 foreach ($PersonTags as $ptag) {
455 if ($ptag['tagname'] == 'columnconf')
457 $column_configuration = $ptag['value'];
458 $conf_tag_id = $ptag['person_tag_id'];
460 if ($ptag['tagname'] == 'showconf')
462 $show_configuration = $ptag['value'];
463 $show_tag_id = $ptag['person_tag_id'];
467 //print("<br>person column configuration = ".$column_configuration);
468 //print("<br>person show configuration = ".$show_configuration);
470 $sliceconf_exists = false;
471 if ($column_configuration == "")
473 $first_time_configuration = true;
474 $column_configuration = $slice_id.";default";
475 $sliceconf_exists = true;
478 $slice_conf = explode(";",$column_configuration);
479 for ($i=0; $i<count($slice_conf); $i++ ) {
480 if ($slice_conf[$i] == $slice_id)
483 $slice_column_configuration = $slice_conf[$i];
484 $sliceconf_exists = true;
490 $slice_column_configuration = $slice_conf[$i];
495 if ($sliceconf_exists == false)
496 $column_configuration = $column_configuration.";".$slice_id.";default";
498 //print("<br>slice configuration = ".$slice_column_configuration);
501 if ($slice_column_configuration == "")
502 $full_configuration = $default_configuration;
504 $full_configuration = $default_configuration."|".$slice_column_configuration;
507 //instantiate the column configuration class, which prepares the headers array
508 $ConfigureColumns =new PlekitColumns($full_configuration, $fix_columns, $tag_columns, $extra_columns);
510 $visiblecolumns = $ConfigureColumns->node_tags();
512 $node_columns=array_merge($node_fixed_columns,$visiblecolumns);
513 //print_r($node_columns);
514 $all_nodes=$api->GetNodes(NULL,$node_columns);
516 //print("<br>person show configuration = ".$show_configuration);
518 $show_conf = explode(";",$show_configuration);
519 foreach ($show_conf as $ss) {
520 if ($ss =="reservable")
521 $show_reservable_message = '0';
523 $show_columns_message = '0';
526 //print("res:".$show_reservable_message." - cols:".$show_columns_message);
528 $slice_nodes=array();
529 $potential_nodes=array();
530 $reservable_nodes=array();
531 foreach ($all_nodes as $node) {
532 if (in_array($node['node_id'],$slice['node_ids'])) {
533 $slice_nodes[]=$node;
534 if ($node['node_type']=='reservable') $reservable_nodes[]=$node;
536 $potential_nodes[]=$node;
539 if ($profiling) plc_debug_prof('5: nodes',count($slice_nodes));
541 // outline the number of reservable nodes
542 $nodes_message=count_english($slice_nodes,"node");
543 if (count($reservable_nodes)) $nodes_message .= " (" . count($reservable_nodes) . " reservable)";
544 $toggle=new PlekitToggle ('my-slice-nodes',$nodes_message,
546 'Manage nodes attached to this slice',
547 'visible'=>get_arg('show_nodes',false)));
551 //////////////////// reservable nodes area
553 $count=count($reservable_nodes);
554 if ($count && $privileges) {
555 // having reservable nodes in white lists looks a bit off scope for now...
556 $toggle_nodes=new PlekitToggle('my-slice-nodes-reserve',
557 "Leases - " . count($reservable_nodes) . " reservable node(s)",
558 array('visible'=>$show_reservable_message, 'info_div'=>'note_reservable_div'));
559 $toggle_nodes->start();
561 if ($show_reservable_message)
564 $note_display = "display:none;";
566 ////////// show a notice to people having attached a reservable node
567 if (count($reservable_nodes) && $privileges) {
568 $mark=reservable_mark();
571 <div id='note_reservable_div' style="align:center; background-color:#CAE8EA; padding:4px; width:800px; $note_display">
572 <table align=center><tr><td valign=top>
573 You have attached one or more reservable nodes to your slice.
574 Reservable nodes show up with the '$mark' mark.
575 Your slivers will be available only during timeslots
576 where you have obtained leases.
577 You can manage your leases in the tab below.
579 Please note that as of August 2010 this feature is experimental.
580 Feedback is appreciated at <a href="mailto:devel@planet-lab.org">devel@planet-lab.org</a>
581 </td><td valign=top><span onClick=closeMessage('reservable')><img class='reset' src="/planetlab/icons/clear.png" alt="hide message"></span>
587 $grain=$api->GetLeaseGranularity();
588 if ($profiling) plc_debug_prof('6 granul',$grain);
589 // where to start from, expressed as an offset in hours from now
590 $resa_offset=$_GET['resa_offset'];
591 if ( ! $resa_offset ) $resa_offset=0;
592 $rough_start=time()+$resa_offset*3600;
593 // xxx should be configurable
594 $resa_slots=$_GET['resa_slots'];
595 if ( ! $resa_slots ) $resa_slots = 36;
596 // for now, show the next 72 hours, or 72 grains, which ever is smaller
597 $duration=$resa_slots*$grain;
598 $steps=$duration/$grain;
599 $start=intval($rough_start/$grain)*$grain;
600 $end=$rough_start+$duration;
601 $lease_columns=array('lease_id','name','t_from','t_until','hostname','name');
602 $leases=$api->GetLeases(array(']t_until'=>$rough_start,'[t_from'=>$end,'-SORT'=>'t_from'),$lease_columns);
603 if ($profiling) plc_debug_prof('7 leases',count($leases));
604 // hash nodes -> leases
606 foreach ($leases as $lease) {
607 $hostname=$lease['hostname'];
608 if ( ! $host_hash[$hostname] ) {
609 $host_hash[$hostname]=array();
611 // resync within the table
612 $lease['nfrom']=($lease['t_from']-$start)/$grain;
613 $lease['nuntil']=($lease['t_until']-$start)/$grain;
614 $host_hash[$hostname] []= $lease;
616 # leases_data is the name used by leases.js to locate this table
617 echo "<table id='leases_data'>";
618 # pass (slice_id,slicename) as the [0,0] coordinate as thead>tr>td
619 echo "<thead><tr><td>" . $slice['slice_id'] . '&' . $slice['name'] . "</td>";
620 # the timeslot headers read (timestamp,label)
621 $day_names=array('Su','M','Tu','W','Th','F','Sa');
622 for ($i=0; $i<$steps; $i++) {
623 $timestamp=($start+$i*$grain);
624 $day=$day_names[intval(strftime("%w",$timestamp))];
625 $label=$day . strftime(" %H:%M",$timestamp);
626 // expose in each header cell the full timestamp, and how to display it - use & as a separator*/
627 echo "<th>" . implode("&",array($timestamp,$label)) . "</th>";
629 echo "</tr></thead><tbody>";
630 // todo - sort on hostnames
631 function sort_hostname ($a,$b) { return ($a['hostname']<$b['hostname'])?-1:1;}
632 usort($reservable_nodes,sort_hostname);
633 foreach ($reservable_nodes as $node) {
634 echo "<tr><th scope='row'>". $node['hostname'] . "</th>";
635 $hostname=$node['hostname'];
636 $leases=$host_hash[$hostname];
638 while ($counter<$steps) {
639 if ($leases && ($leases[0]['nfrom']<=$counter)) {
640 $lease=array_shift($leases);
641 /* nicer display, merge two consecutive leases for the same slice
642 avoid doing that for now, as it might makes things confusing */
643 /* while ($leases && ($leases[0]['name']==$lease['name']) && ($leases[0]['nfrom']==$lease['nuntil'])) {
644 $lease['nuntil']=$leases[0]['nuntil'];
645 array_shift($leases);
647 $duration=$lease['nuntil']-$counter;
648 echo "<td colspan='$duration'>" . $lease['lease_id'] . '&' . $lease['name'] . "</td>";
649 $counter=$lease['nuntil'];
657 echo "</tbody></table>\n";
659 // the general layout for the scheduler
661 <div id='leases_area'></div>
663 <div id='leases_buttons'>
664 <button id='leases_clear' type='submit'>Clear</button>
665 <button id='leases_submit' type='submit'>Submit</button>
669 $toggle_nodes->end();
673 //////////////////// node configuration panel
675 if ($first_time_configuration)
676 $column_conf_visible = '1';
678 $column_conf_visible = '0';
681 $toggle_nodes=new PlekitToggle('my-slice-nodes-configuration',
683 array('visible'=>$column_conf_visible, 'info_div'=>'note_columns_div'));
684 $toggle_nodes->start();
686 //usort ($table_headers, create_function('$col1,$col2','return strcmp($col1["header"],$col2["header"]);'));
687 //print("<p>TABLE HEADERS<p>");
688 //print_r($table_headers);
690 print("<div id='debug'></div>");
691 print("<input type='hidden' id='slice_id' value='".$slice['slice_id']."' />");
692 print("<input type='hidden' id='person_id' value='".$plc->person['person_id']."' />");
693 print("<input type='hidden' id='conf_tag_id' value='".$conf_tag_id."' />");
694 print("<input type='hidden' id='show_tag_id' value='".$show_tag_id."' />");
695 print("<input type='hidden' id='show_configuration' value='".$show_configuration."' />");
696 print("<input type='hidden' id='column_configuration' value='".$slice_column_configuration."' />");
697 print("<br><input type='hidden' size=80 id='full_column_configuration' value='".$column_configuration."' />");
698 print("<input type='hidden' id='previousConf' value='".$slice_column_configuration."'></input>");
699 print("<input type='hidden' id='defaultConf' value='".$default_configuration."'></input>");
701 //print ("showing column message = ".$show_columns_message);
702 if ($show_columns_message == '0')
703 $note_display = "display:none;";
708 <div id='note_columns_div' style="align:center; background-color:#CAE8EA; padding:4px; width:800px; $note_display">
709 <table align=center><tr><td valign=top>
710 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>
711 </td><td valign=top><span onClick=closeMessage('columns')><img class='reset' src="/planetlab/icons/clear.png" alt="hide message permanently"></span>
716 $ConfigureColumns->configuration_panel_html(true);
718 $ConfigureColumns->javascript_init();
720 $toggle_nodes->end();
723 $all_sites=$api->GetSites(NULL, array('site_id','login_base'));
725 foreach ($all_sites as $tmp_site) $site_hash[$tmp_site['site_id']]=$tmp_site['login_base'];
727 $interface_columns=array('ip','node_id','interface_id');
728 $interface_filter=array('is_primary'=>TRUE);
729 $interfaces=$api->GetInterfaces($interface_filter,$interface_columns);
731 $interface_hash=array();
732 foreach ($interfaces as $interface) $interface_hash[$interface['node_id']]=$interface;
738 //////////////////// nodes currently in
739 $toggle_nodes=new PlekitToggle('my-slice-nodes-current',
740 count_english($slice_nodes,"node") . " currently in $name",
741 array('visible'=>get_arg('show_nodes_current',!$privileges)));
742 $toggle_nodes->start();
746 //$notes=array_merge($notes,$visibletags->notes());
747 $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 $headers['peer']='string';
751 $headers['hostname']='string';
752 $short="-S-"; $long=Node::status_footnote(); $type='string';
753 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
754 $short=reservable_mark(); $long=reservable_legend(); $type='string';
755 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
756 // the extra tags, configured for the UI
757 $headers=array_merge($headers,$visibletags->headers());
759 if ($privileges) $headers[plc_delete_icon()]="none";
762 $edit_header = array();
763 if ($privileges) $edit_header[plc_delete_icon()]="none";
764 $headers = array_merge($ConfigureColumns->get_headers(),$edit_header);
766 //print("<p>HEADERS<p>");
769 $table_options = array('notes'=>$notes,
772 'configurable'=>true);
774 $table=new PlekitTable('nodes',$headers,NULL,$table_options);
776 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
779 if ($slice_nodes) foreach ($slice_nodes as $node) {
782 $table->cell($node['node_id'], array('display'=>'none'));
784 $table->cell(l_node_obj($node));
785 $peers->cell($table,$node['peer_id']);
786 $run_level=$node['run_level'];
787 list($label,$class) = Node::status_label_class_($node);
788 $table->cell ($label,array('class'=>$class));
789 $table->cell( ($node['node_type']=='reservable')?reservable_mark():"" );
791 $hostname=$node['hostname'];
792 $ip=$interface_hash[$node['node_id']]['ip'];
793 $interface_id=$interface_hash[$node['node_id']]['interface_id'];
796 $node['domain'] = topdomain($hostname);
797 $node['sitename'] = l_site_t($node['site_id'],$site_hash[$node['site_id']]);
798 $node['ipaddress'] = l_interface_t($interface_id,$ip);
801 //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
802 $ConfigureColumns->cells($table, $node);
804 if ($privileges) $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
811 $table->tfoot_start();
814 $table->cell($form->submit_html ("remove-nodes-from-slice","Remove selected"),
815 array('hfill'=>true,'align'=>'right'));
819 $toggle_nodes->end();
821 //////////////////// nodes to add
823 $new_potential_nodes = array();
824 if ($potential_nodes) foreach ($potential_nodes as $node) {
825 $emptywl=empty($node['slice_ids_whitelist']);
826 $inwl = (!emptywl) and in_array($slice['slice_id'],$node['slice_ids_whitelist']);
827 if ($emptywl or $inwl)
828 $new_potential_nodes[]=$node;
830 $potential_nodes=$new_potential_nodes;
832 $count=count($potential_nodes);
833 $toggle_nodes=new PlekitToggle('my-slice-nodes-add',
834 count_english($potential_nodes,"more node") . " available",
835 array('visible'=>get_arg('show_nodes_add',false)));
836 $toggle_nodes->start();
838 if ( $potential_nodes ) {
844 $headers['peer']='string';
845 $headers['hostname']='string';
846 $short="-S-"; $long=Node::status_footnote(); $type='string';
847 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
848 $short=reservable_mark(); $long=reservable_legend(); $type='string';
849 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
850 // the extra tags, configured for the UI
851 $headers=array_merge($headers,$visibletags->headers());
852 $headers['+']="none";
855 $add_header = array();
856 $add_header['+']="none";
857 $headers = array_merge($ConfigureColumns->get_headers(),$add_header);
859 //$notes=array_merge($notes,$visibletags->notes());
860 $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";
862 $table=new PlekitTable('add_nodes',$headers,NULL, $table_options);
863 $form=new PlekitForm(l_actions(),
864 array('slice_id'=>$slice['slice_id']));
867 if ($potential_nodes) foreach ($potential_nodes as $node) {
870 $table->cell($node['node_id'], array('display'=>'none'));
872 $table->cell(l_node_obj($node));
873 $peers->cell($table,$node['peer_id']);
874 list($label,$class) = Node::status_label_class_($node);
875 $table->cell ($label,array('class'=>$class));
876 $table->cell( ($node['node_type']=='reservable')?reservable_mark():"" );
879 $hostname=$node['hostname'];
880 $ip=$interface_hash[$node['node_id']]['ip'];
881 $interface_id=$interface_hash[$node['node_id']]['interface_id'];
882 $node['domain'] = topdomain($hostname);
883 $node['sitename'] = l_site_t($node['site_id'],$site_hash[$node['site_id']]);
884 $node['ipaddress'] = l_interface_t($interface_id,$ip);
886 //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
887 $ConfigureColumns->cells($table, $node);
889 $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
893 $table->tfoot_start();
895 $table->cell($form->submit_html ("add-nodes-in-slice","Add selected"),
896 array('hfill'=>true,'align'=>'right'));
901 $toggle_nodes->end();
906 // very wide values get abbreviated
907 $tag_value_threshold=24;
908 //////////////////////////////////////////////////////////// Tags
909 //if ( $local_peer ) {
910 $tags=$api->GetSliceTags (array('slice_id'=>$slice_id));
911 if ($profiling) plc_debug_prof('8 slice tags',count($tags));
912 function get_tagname ($tag) { return $tag['tagname'];}
913 $tagnames = array_map ("get_tagname",$tags);
915 $toggle = new PlekitToggle ('slice-tags',count_english_warning($tags,'tag'),
916 array('bubble'=>'Inspect and set tags on tat slice',
917 'visible'=>get_arg('show_tags',false)));
924 "NodeGroup"=>"string");
925 if ($tags_privileges) $headers[plc_delete_icon()]="none";
927 $table_options=array("notes_area"=>false,"pagesize_area"=>false,"search_width"=>10);
928 $table=new PlekitTable("slice_tags",$headers,'0',$table_options);
929 $form=new PlekitForm(l_actions(),
930 array('slice_id'=>$slice['slice_id']));
934 // Get hostnames for nodes in a single pass
935 $_node_ids = array();
936 foreach ($tags as $tag) {
937 if ($tag['node_id']) {
938 array_push($_node_ids, $tag['node_id']);
941 $_nodes = $api->GetNodes(array('node_id' => $_node_ids), array('node_id', 'hostname'));
942 $_hostnames = array();
943 foreach ($_nodes as $_node) {
944 $_hostnames[$_node['node_id']] = $_node['hostname'];
947 // Loop through tags again to display
948 foreach ($tags as $tag) {
950 if ($tag['node_id']) {
951 $node_name = $_hostnames[$tag['node_id']];
953 $nodegroup_name="n/a";
954 if ($tag['nodegroup_id']) {
955 $nodegroups=$api->GetNodeGroups(array('nodegroup_id'=>$tag['nodegroup_id']));
956 if ($profiling) plc_debug_prof('10 nodegroup for slice tag',$nodegroup);
958 $nodegroup = $nodegroups[0];
959 $nodegroup_name = $nodegroup['groupname'];
963 $table->cell(l_tag_obj($tag));
964 // very wide values get abbreviated
965 $table->cell(truncate_and_popup($tag['value'],$tag_value_threshold));
966 $table->cell($node_name);
967 $table->cell($nodegroup_name);
968 if ($tags_privileges) $table->cell ($form->checkbox_html('slice_tag_ids[]',$tag['slice_tag_id']));
972 if ($tags_privileges) {
973 $table->tfoot_start();
975 $table->cell($form->submit_html ("delete-slice-tags","Remove selected"),
976 array('hfill'=>true,'align'=>'right'));
980 function tag_selector ($tag) {
981 return array("display"=>$tag['tagname'],"value"=>$tag['tag_type_id']);
983 $all_tags= $api->GetTagTypes( array ("category"=>"slice*","-SORT"=>"+tagname"), array("tagname","tag_type_id"));
984 if ($profiling) plc_debug_prof('11 tagtypes',count($all_tags));
985 $selector_tag=array_map("tag_selector",$all_tags);
987 function node_selector($node) {
988 return array("display"=>$node["hostname"],"value"=>$node['node_id']);
990 $selector_node=array_map("node_selector",$slice_nodes);
992 function nodegroup_selector($ng) {
993 return array("display"=>$ng["groupname"],"value"=>$ng['nodegroup_id']);
995 $all_nodegroups = $api->GetNodeGroups( array("groupname"=>"*"), array("groupname","nodegroup_id"));
996 if ($profiling) plc_debug_prof('13 nodegroups',count($all_nodegroups));
997 $selector_nodegroup=array_map("nodegroup_selector",$all_nodegroups);
999 $table->cell($form->select_html("tag_type_id",$selector_tag,array('label'=>"Choose Tag")));
1000 $table->cell($form->text_html("value","",array('width'=>8)));
1001 $table->cell($form->select_html("node_id",$selector_node,array('label'=>"All Nodes")));
1002 $table->cell($form->select_html("nodegroup_id",$selector_nodegroup,array('label'=>"No Nodegroup")));
1003 $table->cell($form->submit_html("add-slice-tag","Set Tag"),array('columns'=>2,'align'=>'left'));
1013 //////////////////////// renew slice
1015 if ( ! $renew_visible) renew_area ($slice,$site,false);
1018 $peers->block_end($peer_id);
1020 if ($profiling) plc_debug_prof_end();
1023 include 'plc_footer.php';