6 require_once 'plc_login.php';
8 // Get session and API handles
9 require_once 'plc_session.php';
13 require_once 'plc_drupal.php';
14 include 'plc_header.php';
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';
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>
35 // -------------------- admins potentially need to get full list of users
36 ini_set('memory_limit','32M');
39 if ($_GET['profiling']) $profiling=true;
41 if ($profiling) plc_debug_prof_start();
43 // --------------------
44 // recognized URL arguments
45 $slice_id=intval($_GET['id']);
46 if ( ! $slice_id ) { plc_error('Malformed URL - id not set'); return; }
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);
55 drupal_set_message ("Slice " . $slice_id . " not found");
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'];
67 $person_ids=$slice['person_ids'];
70 $peer_id= $slice['peer_id'];
71 $peers=new Peers ($api);
72 $local_peer = ! $peer_id;
74 if ($profiling) plc_debug_prof('3: peers',count($peers));
77 $sites= $api->GetSites( array( $site_id ) );
79 $site_name= $site['name'];
81 if ($profiling) plc_debug_prof('4: sites',count($sites));
82 //////////////////////////////////////// building blocks for the renew area
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();
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'];
96 $time_left = $current_exp - $NOW;
97 $visible = $time_left/$DAY <= $GRACE_DAYS;
101 function renew_area ($slice,$site,$visible) {
102 global $DAY, $WEEK, $MAX_WEEKS, $GRACE_DAYS, $NOW;
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);
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
112 new PlekitToggle('renew',"Expires $current_text - Renew this slice",
114 "Enter this zone if you wish to renew your slice",
115 'visible'=>$visible));
118 // xxx message could take roles into account
119 if ($site['max_slices']<=0) {
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>
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);
151 if ( empty( $selectors ) ) {
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.
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.
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>
170 <p><span class='bold'>NOTE:</span>
171 Slices cannot be renewed beyond another $max_renewal_weeks week(s) ($max_renewal_date).
176 $form = new PlekitForm (l_actions(),
177 array('action'=>'renew-slice',
178 'slice_id'=>$slice['slice_id']));
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');
190 ////////////////////////////////////////////////////////////
192 $am_in_slice = in_array(plc_my_person_id(),$person_ids);
195 drupal_set_title("My slice " . $name);
197 drupal_set_title("Slice " . $name);
200 $privileges = ( $local_peer && (plc_is_admin() || plc_is_pi() || $am_in_slice));
201 $tags_privileges = $privileges || plc_is_admin();
204 $tabs [] = tab_nodes_slice($slice_id);
205 $tabs [] = tab_site($site);
207 // are these the right privileges for deletion ?
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(),
214 'values'=>array('action'=>'delete-slice','slice_id'=>$slice_id),
215 'bubble'=>"Delete slice $name",
216 'confirm'=>"Are you sure to delete slice $name");
218 //$tabs["Comon"]=array_merge(tablook_comon(),
219 //array('url'=>l_comon("slice_id",$slice_id),
220 //'bubble'=>"Comon page about slice $name"));
223 plekit_linetabs($tabs);
225 ////////////////////////////////////////
226 $peers->block_start($peer_id);
228 //////////////////////////////////////// renewal area
229 // (1) close to expiration : show on top and open
232 $renew_visible = renew_needed ($slice);
233 if ($renew_visible) renew_area ($slice,$site,true);
237 //////////////////// details
238 // default for opening the details section or not ?
240 $default_show_details = true;
242 $default_show_details = ! $renew_visible;
246 new PlekitToggle ('my-slice-details',"Details",
248 'Display and modify details for that slice',
249 'visible'=>get_arg('show_details',$default_show_details)));
252 $details=new PlekitDetails($privileges);
253 $details->form_start(l_actions(),array('action'=>'update-slice',
254 'slice_id'=>$slice_id,
259 $details->th_td("Peer",$peers->peer_link($peer_id));
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',...);
278 $details->form_end();
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
290 $api->GetPersons(array('~person_id'=>$slice['person_ids'],
294 $count=count($persons);
296 if ($profiling) plc_debug_prof('4: persons',count($persons));
298 new PlekitToggle ('my-slice-persons',"$count users",
300 'Manage accounts attached to this slice',
301 'visible'=>get_arg('show_persons',false)));
304 ////////// people currently in
306 // hide if both current+add are included
307 // so user can chose which section is of interest
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();
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']));
325 if ($persons) foreach ($persons as $person) {
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']));
338 $table->tfoot_start();
341 $table->cell($form->submit_html ("remove-persons-from-slice","Remove selected"),
342 array('hfill'=>true,'align'=>'right'));
346 $toggle_persons->end();
348 ////////// people to add
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 ) {
357 echo "<p class='not-relevant'>No person to add</p>";
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,
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;
371 $table=new PlekitTable('add_persons',$headers,'0',$options);
372 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
375 if ($potential_persons) foreach ($potential_persons as $person) {
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']));
385 $table->tfoot_start();
387 $table->cell($form->submit_html ("add-persons-in-slice","Add selected"),
388 array('hfill'=>true,'align'=>'right'));
393 $toggle_persons->end();
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')
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();
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);
420 //NEW CODE FOR ENABLING COLUMN CONFIGURATION
422 //prepare fix and configurable columns
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');
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();
435 // extra columns that are not tags (for the moment not sorted correctly)
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);
443 //Get user's column configuration
445 $default_configuration = "hostname:f|ST:f|AU:f|RES:f|R|L|OS|MS";
446 $column_configuration = "";
447 $slice_column_configuration = "";
449 $show_configuration = "";
450 $show_reservable_message = '1';
451 $show_columns_message = true;
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')
460 $column_configuration = $ptag['value'];
461 $conf_tag_id = $ptag['person_tag_id'];
463 if ($ptag['tagname'] == 'showconf')
465 $show_configuration = $ptag['value'];
466 $show_tag_id = $ptag['person_tag_id'];
470 //print("<br>person column configuration = ".$column_configuration);
471 //print("<br>person show configuration = ".$show_configuration);
473 $sliceconf_exists = false;
474 if ($column_configuration == "")
476 $column_configuration = $slice_id.";default";
477 $sliceconf_exists = true;
480 $slice_conf = explode(";",$column_configuration);
481 for ($i=0; $i<count($slice_conf); $i++ ) {
482 if ($slice_conf[$i] == $slice_id)
485 $slice_column_configuration = $slice_conf[$i];
486 $sliceconf_exists = true;
492 $slice_column_configuration = $slice_conf[$i];
497 if ($sliceconf_exists == false)
498 $column_configuration = $column_configuration.";".$slice_id.";default";
500 //print("<br>slice configuration = ".$slice_column_configuration);
503 if ($slice_column_configuration == "")
504 $full_configuration = $default_configuration;
506 $full_configuration = $default_configuration."|".$slice_column_configuration;
509 //instantiate the column configuration class, which prepares the headers array
510 $ConfigureColumns =new PlekitColumns($full_configuration, $fix_columns, $tag_columns, $extra_columns);
512 $visiblecolumns = $ConfigureColumns->node_tags();
514 $node_columns=array_merge($node_fixed_columns,$visiblecolumns);
515 //print_r($node_columns);
516 $all_nodes=$api->GetNodes(NULL,$node_columns);
518 //print("<br>person show configuration = ".$show_configuration);
520 $show_conf = explode(";",$show_configuration);
521 foreach ($show_conf as $ss) {
522 if ($ss =="reservable")
523 $show_reservable_message = '0';
525 $show_columns_message = false;
528 //print("res:".$show_reservable_message." - cols:".$show_columns_message);
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;
538 $potential_nodes[]=$node;
541 if ($profiling) plc_debug_prof('5: nodes',count($slice_nodes));
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,
548 'Manage nodes attached to this slice',
549 'visible'=>get_arg('show_nodes',false)));
553 //////////////////// reservable nodes area
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();
563 if ($show_reservable_message)
566 $note_display = "display:none;";
568 ////////// show a notice to people having attached a reservable node
569 if (count($reservable_nodes) && $privileges) {
570 $mark=reservable_mark();
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.
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>
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
608 foreach ($leases as $lease) {
609 $hostname=$lease['hostname'];
610 if ( ! $host_hash[$hostname] ) {
611 $host_hash[$hostname]=array();
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;
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>";
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];
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);
649 $duration=$lease['nuntil']-$counter;
650 echo "<td colspan='$duration'>" . $lease['lease_id'] . '&' . $lease['name'] . "</td>";
651 $counter=$lease['nuntil'];
659 echo "</tbody></table>\n";
661 // the general layout for the scheduler
663 <div id='leases_area'></div>
665 <div id='leases_buttons'>
666 <button id='leases_clear' type='submit'>Clear</button>
667 <button id='leases_submit' type='submit'>Submit</button>
671 $toggle_nodes->end();
675 //////////////////// node configuration panel
677 if ($show_columns_message)
678 $column_conf_visible = '1';
680 $column_conf_visible = '0';
683 $toggle_nodes=new PlekitToggle('my-slice-nodes-configuration',
685 array('visible'=>$column_conf_visible, 'info_div'=>'note_columns_div'));
686 $toggle_nodes->start();
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);
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>");
703 //print ("showing column message = ".$show_columns_message);
704 if ($show_columns_message)
707 $note_display = "display:none;";
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>
718 $ConfigureColumns->configuration_panel_html(true);
720 $ConfigureColumns->javascript_init();
722 $toggle_nodes->end();
725 $all_sites=$api->GetSites(NULL, array('site_id','login_base'));
727 foreach ($all_sites as $tmp_site) $site_hash[$tmp_site['site_id']]=$tmp_site['login_base'];
729 $interface_columns=array('ip','node_id','interface_id');
730 $interface_filter=array('is_primary'=>TRUE);
731 $interfaces=$api->GetInterfaces($interface_filter,$interface_columns);
733 $interface_hash=array();
734 foreach ($interfaces as $interface) $interface_hash[$interface['node_id']]=$interface;
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();
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";
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());
761 if ($privileges) $headers[plc_delete_icon()]="none";
764 $edit_header = array();
765 if ($privileges) $edit_header[plc_delete_icon()]="none";
766 $headers = array_merge($ConfigureColumns->get_headers(),$edit_header);
768 //print("<p>HEADERS<p>");
771 $table_options = array('notes'=>$notes,
774 'configurable'=>true);
776 $table=new PlekitTable('nodes',$headers,NULL,$table_options);
778 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
781 if ($slice_nodes) foreach ($slice_nodes as $node) {
784 $table->cell($node['node_id'], array('display'=>'none'));
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():"" );
793 $hostname=$node['hostname'];
794 $ip=$interface_hash[$node['node_id']]['ip'];
795 $interface_id=$interface_hash[$node['node_id']]['interface_id'];
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);
803 //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
804 $ConfigureColumns->cells($table, $node);
806 if ($privileges) $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
813 $table->tfoot_start();
816 $table->cell($form->submit_html ("remove-nodes-from-slice","Remove selected"),
817 array('hfill'=>true,'align'=>'right'));
821 $toggle_nodes->end();
823 //////////////////// nodes to add
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;
832 $potential_nodes=$new_potential_nodes;
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();
840 if ( $potential_nodes ) {
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";
857 $add_header = array();
858 $add_header['+']="none";
859 $headers = array_merge($ConfigureColumns->get_headers(),$add_header);
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";
864 $table=new PlekitTable('add_nodes',$headers,NULL, $table_options);
865 $form=new PlekitForm(l_actions(),
866 array('slice_id'=>$slice['slice_id']));
869 if ($potential_nodes) foreach ($potential_nodes as $node) {
872 $table->cell($node['node_id'], array('display'=>'none'));
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():"" );
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);
888 //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
889 $ConfigureColumns->cells($table, $node);
891 $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
895 $table->tfoot_start();
897 $table->cell($form->submit_html ("add-nodes-in-slice","Add selected"),
898 array('hfill'=>true,'align'=>'right'));
903 $toggle_nodes->end();
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);
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)));
926 "NodeGroup"=>"string");
927 if ($tags_privileges) $headers[plc_delete_icon()]="none";
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']));
936 foreach ($tags as $tag) {
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));
942 $node = $tag_nodes[0];
943 $node_name = $node['hostname'];
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);
951 $nodegroup = $nodegroups[0];
952 $nodegroup_name = $nodegroup['groupname'];
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']));
965 if ($tags_privileges) {
966 $table->tfoot_start();
968 $table->cell($form->submit_html ("delete-slice-tags","Remove selected"),
969 array('hfill'=>true,'align'=>'right'));
973 function tag_selector ($tag) {
974 return array("display"=>$tag['tagname'],"value"=>$tag['tag_type_id']);
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);
980 function node_selector($node) {
981 return array("display"=>$node["hostname"],"value"=>$node['node_id']);
983 $selector_node=array_map("node_selector",$slice_nodes);
985 function nodegroup_selector($ng) {
986 return array("display"=>$ng["groupname"],"value"=>$ng['nodegroup_id']);
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);
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'));
1006 //////////////////////// renew slice
1008 if ( ! $renew_visible) renew_area ($slice,$site,false);
1011 $peers->block_end($peer_id);
1013 if ($profiling) plc_debug_prof_end();
1016 include 'plc_footer.php';