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');
38 if ($_GET['profiling']) $profiling=true;
40 if ($profiling) plc_debug_prof_start();
42 // --------------------
43 // recognized URL arguments
44 $slice_id=intval($_GET['id']);
45 if ( ! $slice_id ) { plc_error('Malformed URL - id not set'); return; }
48 // have to name columns b/c we need the non-native 'omf_control' column
49 $slice_columns=array('slice_id','name','peer_id','site_id','person_ids','node_ids','expires',
50 'url','description','instantiation','omf_control');
51 $slices= $api->GetSlices( array($slice_id), $slice_columns);
54 drupal_set_message ("Slice " . $slice_id . " not found");
60 if ($profiling) plc_debug_prof('1: slice',count($slices));
61 // pull all node info to vars
62 $name= $slice['name'];
63 $expires = date( "d/m/Y", $slice['expires'] );
64 $site_id= $slice['site_id'];
66 $person_ids=$slice['person_ids'];
69 $peer_id= $slice['peer_id'];
70 $peers=new Peers ($api);
71 $local_peer = ! $peer_id;
73 if ($profiling) plc_debug_prof('2: peers',count($peers));
76 $sites= $api->GetSites( array( $site_id ) );
78 $site_name= $site['name'];
79 $max_slices = $site['max_slices'];
81 if ($profiling) plc_debug_prof('3: 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 //////////////////////////////////////////////////////////// utility for the renew tab
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 cannot 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 <span class='bold'>Important:</span> Please take this opportunity to review and update your slice information in the Details tab.
163 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.
165 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.
169 $form = new PlekitForm (l_actions(),
170 array('action'=>'renew-slice',
171 'slice_id'=>$slice['slice_id']));
173 print $form->label_html('expires','Duration: ');
174 print $form->select_html('expires',$selectors,array('label'=>'Pick one'));
175 print $form->submit_html('renew-button','Renew');
178 print("<p><i>NOTE: Slices cannot be renewed beyond another $max_renewal_weeks week(s) ($max_renewal_date).</i> </p>");
186 ////////////////////////////////////////////////////////////
188 $am_in_slice = in_array(plc_my_person_id(),$person_ids);
191 drupal_set_title("My slice " . $name);
193 drupal_set_title("Slice " . $name);
196 $privileges = ( $local_peer && (plc_is_admin() || plc_is_pi() || $am_in_slice));
197 $tags_privileges = $privileges || plc_is_admin();
200 $tabs [] = tab_nodes_slice($slice_id);
201 $tabs [] = tab_site($site);
203 // are these the right privileges for deletion ?
205 $tabs ['Delete']= array('url'=>l_actions(),
207 'values'=>array('action'=>'delete-slice','slice_id'=>$slice_id),
208 'bubble'=>"Delete slice $name",
209 'confirm'=>"Are you sure to delete slice $name");
211 $tabs["Events"]=array_merge(tablook_event(),
212 array('url'=>l_event("Slice","slice",$slice_id),
213 'bubble'=>"Events for slice $name"));
214 $tabs["Comon"]=array_merge(tablook_comon(),
215 array('url'=>l_comon("slice_id",$slice_id),
216 'bubble'=>"Comon page about slice $name"));
219 plekit_linetabs($tabs);
221 ////////////////////////////////////////
222 $peers->block_start($peer_id);
224 //////////////////////////////////////// renewal area
225 // (1) close to expiration : show on top and open
228 $renew_visible = renew_needed ($slice);
229 if ($renew_visible) renew_area ($slice,$site,true);
233 //////////////////////////////////////////////////////////// tab:details
234 // default for opening the details section or not ?
236 $default_show_details = true;
238 $default_show_details = ! $renew_visible;
242 new PlekitToggle ('my-slice-details',"Details",
244 'Display and modify details for that slice',
245 'visible'=>get_arg('show_details',$default_show_details)));
248 $details=new PlekitDetails($privileges);
249 $details->form_start(l_actions(),array('action'=>'update-slice',
250 'slice_id'=>$slice_id,
255 $details->th_td("Peer",$peers->peer_link($peer_id));
260 $details->th_td('Name',$slice['name']);
261 $details->th_td('Description',$slice['description'],'description',
262 array('input_type'=>'textarea',
263 'width'=>50,'height'=>5));
264 $details->th_td('URL',$slice['url'],'url',array('width'=>50));
265 $details->tr_submit("submit","Update Slice");
266 $details->th_td('Expires',$expires);
267 $details->th_td('Instantiation',$slice['instantiation']);
268 $details->th_td("OMF-friendly", ($slice['omf_control'] ? 'Yes' : 'No') . " [to change: see 'omf_control' in the tags section below]");
269 $details->th_td('Site',l_site_obj($site));
270 // xxx show the PIs here
271 //$details->th_td('PIs',...);
274 $details->form_end();
277 //////////////////////////////////////////////////////////// tab:persons
278 $person_columns = array('email','person_id','first_name','last_name','roles');
279 // get persons in slice
280 if (!empty($person_ids))
281 $persons=$api->GetPersons(array('person_id'=>$slice['person_ids']),$person_columns);
282 // just propose to add everyone else
283 // xxx this is maybe too much for admins as it slows stuff down
284 // as regular persons can see only a fraction of the db anyway
286 $api->GetPersons(array('~person_id'=>$slice['person_ids'],
290 $count=count($persons);
292 if ($profiling) plc_debug_prof('4: persons',count($persons));
294 new PlekitToggle ('my-slice-persons',"$count users",
296 'Manage accounts attached to this slice',
297 'visible'=>get_arg('show_persons',false)));
300 ////////// people currently in
302 // hide if both current+add are included
303 // so user can chose which section is of interest
305 $toggle_persons = new PlekitToggle ('my-slice-persons-current',
306 "$count people currently in $name",
307 array('visible'=>get_arg('show_persons_current',!$privileges)));
308 $toggle_persons->start();
311 $headers['email']='string';
312 $headers['first']='string';
313 $headers['last']='string';
314 $headers['R']='string';
315 if ($privileges) $headers[plc_delete_icon()]="none";
316 $table=new PlekitTable('persons',$headers,'0',
317 array('notes_area'=>false));
318 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
321 if ($persons) foreach ($persons as $person) {
323 $table->cell(l_person_obj($person));
324 $table->cell($person['first_name']);
325 $table->cell($person['last_name']);
326 $table->cell(plc_vertical_table ($person['roles']));
327 if ($privileges) $table->cell ($form->checkbox_html('person_ids[]',$person['person_id']));
334 $table->tfoot_start();
337 $table->cell($form->submit_html ("remove-persons-from-slice","Remove selected"),
338 array('hfill'=>true,'align'=>'right'));
342 $toggle_persons->end();
344 ////////// people to add
346 $count=count($potential_persons);
347 $toggle_persons = new PlekitToggle ('my-slice-persons-add',
348 "$count people may be added to $name",
349 array('visible'=>get_arg('show_persons_add',false)));
350 $toggle_persons->start();
351 if ( ! $potential_persons ) {
353 echo "<p class='not-relevant'>No person to add</p>";
356 $headers['email']='string';
357 $headers['first']='string';
358 $headers['last']='string';
359 $headers['R']='string';
360 $headers['+']="none";
361 $options = array('notes_area'=>false,
364 // show search for admins only as other people won't get that many names to add
365 if ( ! plc_is_admin() ) $options['search_area']=false;
367 $table=new PlekitTable('add_persons',$headers,'0',$options);
368 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
371 if ($potential_persons) foreach ($potential_persons as $person) {
373 $table->cell(l_person_obj($person));
374 $table->cell($person['first_name']);
375 $table->cell($person['last_name']);
376 $table->cell(plc_vertical_table ($person['roles']));
377 $table->cell ($form->checkbox_html('person_ids[]',$person['person_id']));
381 $table->tfoot_start();
383 $table->cell($form->submit_html ("add-persons-in-slice","Add selected"),
384 array('hfill'=>true,'align'=>'right'));
389 $toggle_persons->end();
393 //////////////////////////////////////////////////////////// tab:nodes
394 // the nodes details to display here
395 // (1) we search for the tag types for which 'category' matches 'node*/ui*'
396 // all these tags will then be tentatively displayed in this area
397 // (2) further information can also be optionally specified in the category:
398 // (.) we split the category with '/' and search for assignments of the form var=value
399 // (.) header can be set to supersede the column header (default is tagname)
400 // (.) rank can be used for ordering the columns (default is tagname)
401 // (.) type is passed to the javascript table, for sorting (default is 'string')
403 // minimal list as a start
404 $node_fixed_columns = array('hostname','node_id','peer_id','slice_ids_whitelist', 'site_id',
405 'run_level','boot_state','last_contact','node_type');
406 // create a VisibleTags object : basically the list of tag columns to show
407 //$visibletags = new VisibleTags ($api, 'node');
408 //$visiblecolumns = $visibletags->column_names();
410 // optimizing calls to GetNodes
411 //$all_nodes=$api->GetNodes(NULL,$node_columns);
412 //$slice_nodes=$api->GetNodes(array('node_id'=>$slice['node_ids']),$node_columns);
413 //$potential_nodes=$api->GetNodes(array('~node_id'=>$slice['node_ids']),$node_columns);
416 //NEW CODE FOR ENABLING COLUMN CONFIGURATION
418 //prepare fix and configurable columns
420 $fix_columns = array();
421 $fix_columns[]=array('tagname'=>'hostname', 'header'=>'hostname', 'type'=>'string', 'title'=>'The name of the node');
422 $fix_columns[]=array('tagname'=>'peer_id', 'header'=>'AU', 'type'=>'string', 'title'=>'Authority');
423 $fix_columns[]=array('tagname'=>'run_level', 'header'=>'ST', 'type'=>'string', 'title'=>'Status');
424 $fix_columns[]=array('tagname'=>'node_type', 'header'=>'RES', 'type'=>'string', 'title'=>'Reservable');
426 // columns that correspond to the visible tags for nodes (*node/ui*)
427 $visibletags = new VisibleTags ($api, 'node');
428 $visibletags->columns();
429 $tag_columns = $visibletags->headers();
431 //columns that are not defined as extra myslice tags
432 $extra_columns = array();
434 $extra_columns[]=array('tagname'=>'sitename', 'header'=>'SN', 'type'=>'string', 'title'=>'Site name', 'fetched'=>true, 'source'=>'myplc');
435 $extra_columns[]=array('tagname'=>'domain', 'header'=>'DN', 'type'=>'string', 'title'=>'Toplevel domain name', 'fetched'=>true, 'source'=>'myplc');
436 $extra_columns[]=array('tagname'=>'ipaddress', 'header'=>'IP', 'type'=>'string', 'title'=>'IP Address', 'fetched'=>true, 'source'=>'myplc');
437 $extra_columns[]=array('tagname'=>'fcdistro', 'header'=>'OS', 'type'=>'string', 'title'=>'Operating system', 'fetched'=>false, 'source'=>'myplc');
438 $extra_columns[]=array('tagname'=>'date_created', 'header'=>'DA', 'source'=>'myplc', 'type'=>'date', 'title'=>'Date added', 'fetched'=>false);
439 $extra_columns[]=array('tagname'=>'arch', 'header'=>'A', 'source'=>'myplc', 'type'=>'string', 'title'=>'Architecture', 'fetched'=>false);
440 if (plc_is_admin()) {
441 $extra_columns[]=array('tagname'=>'deployment', 'header'=>'DL', 'source'=>'myplc', 'type'=>'string', 'title'=>'Deployment', 'fetched'=>false);
445 //NOTE: Uncomment these lines if CoMon provides information for your nodes
447 //$extra_columns[]=array('tagname'=>'bwlimit', 'header'=>'BW', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Bandwidth limit', 'fetched'=>false);
448 //$extra_columns[]=array('tagname'=>'numcores', 'header'=>'CC', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Number of CPU Cores', 'fetched'=>false);
449 //$extra_columns[]=array('tagname'=>'cpuspeed', 'header'=>'CR', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'CPU clock rate', 'fetched'=>false);
450 //$extra_columns[]=array('tagname'=>'disksize', 'header'=>'DS', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Disk size', 'fetched'=>false);
451 //$extra_columns[]=array('tagname'=>'gbfree', 'header'=>'DF', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Currently available disk space', 'fetched'=>false);
452 //$extra_columns[]=array('tagname'=>'memsize', 'header'=>'MS', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Memory size', 'fetched'=>false);
453 //$extra_columns[]=array('tagname'=>'numslices', 'header'=>'SM', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Number of slices in memory', 'fetched'=>false);
454 //$extra_columns[]=array('tagname'=>'uptime', 'header'=>'UT', 'source'=>'comon', 'type'=>'sortAlphaNumericTop', 'title'=>'Continuous uptime until now', 'fetched'=>false);
457 //NOTE: Uncomment these lines if TopHat provides information for your nodes
459 //$extra_columns[]=array('tagname'=>'asn', 'header'=>'AS', 'source'=>'tophat', 'type'=>'string', 'title'=>'AS Number', 'fetched'=>false);
460 //$extra_columns[]=array('tagname'=>'city', 'header'=>'LCY', 'source'=>'tophat', 'type'=>'string', 'title'=>'City', 'fetched'=>false);
461 //$extra_columns[]=array('tagname'=>'region', 'header'=>'LRN', 'source'=>'tophat', 'type'=>'string', 'title'=>'Region', 'fetched'=>false);
462 //$extra_columns[]=array('tagname'=>'country', 'header'=>'LCN', 'source'=>'tophat', 'type'=>'string', 'title'=>'Country', 'fetched'=>false);
463 //$extra_columns[]=array('tagname'=>'continent', 'header'=>'LCT', 'source'=>'tophat', 'type'=>'string', 'title'=>'Continent', 'fetched'=>false);
464 ////$extra_columns[]=array('tagname'=>'hopcount', 'header'=>'HC', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Hop count from reference node', 'fetched'=>false);
465 ////$extra_columns[]=array('tagname'=>'rtt', 'header'=>'RTT', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Round trip time from reference node', 'fetched'=>false);
466 //////$extra_columns[]=array('tagname'=>'agents', 'header'=>'MA', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Co-located measurement agents', 'fetched'=>true);
467 ////$extra_columns[]=array('tagname'=>'agents_sonoma', 'header'=>'MAS', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Co-located SONoMA agents', 'fetched'=>true);
468 ////$extra_columns[]=array('tagname'=>'agents_etomic', 'header'=>'MAE', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Co-located ETOMIC agents', 'fetched'=>true);
469 ////$extra_columns[]=array('tagname'=>'agents_tdmi', 'header'=>'MAT', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Co-located TDMI agents', 'fetched'=>true);
470 ////$extra_columns[]=array('tagname'=>'agents_dimes', 'header'=>'MAD', 'source'=>'tophat', 'type'=>'sortAlphaNumericTop', 'title'=>'Co-located DIMES agents', 'fetched'=>true);
474 //Get user's column configuration
476 $first_time_configuration = false;
477 $default_configuration = "hostname:f|ST:f|AU:f|RES:f";
478 //$extra_default = "";
479 $column_configuration = "";
480 $slice_column_configuration = "";
482 $show_configuration = "";
484 $PersonTags=$api->GetPersonTags (array('person_id'=>$plc->person['person_id']));
485 //plc_debug('ptags',$PersonTags);
486 foreach ($PersonTags as $ptag) {
487 if ($ptag['tagname'] == 'columnconf') {
488 $column_configuration = $ptag['value'];
489 $conf_tag_id = $ptag['person_tag_id'];
490 } else if ($ptag['tagname'] == 'showconf') {
491 $show_configuration = $ptag['value'];
492 $show_tag_id = $ptag['person_tag_id'];
496 $sliceconf_exists = false;
497 if ($column_configuration == "") {
498 $first_time_configuration = true;
499 $column_configuration = $slice_id.";default";
500 $sliceconf_exists = true;
502 $slice_conf = explode(";",$column_configuration);
503 for ($i=0; $i<count($slice_conf); $i++ ) {
504 if ($slice_conf[$i] == $slice_id) {
506 $slice_column_configuration = $slice_conf[$i];
507 $sliceconf_exists = true;
511 $slice_column_configuration = $slice_conf[$i];
516 if ($sliceconf_exists == false)
517 $column_configuration = $column_configuration.";".$slice_id.";default";
519 if ($slice_column_configuration == "")
520 $full_configuration = $default_configuration;
522 $full_configuration = $default_configuration."|".$slice_column_configuration;
525 //instantiate the column configuration class, which prepares the headers array
526 $ConfigureColumns =new PlekitColumns($full_configuration, $fix_columns, $tag_columns, $extra_columns);
528 $visiblecolumns = $ConfigureColumns->node_tags();
530 $node_columns=array_merge($node_fixed_columns,$visiblecolumns);
531 $all_nodes=$api->GetNodes(NULL,$node_columns);
533 $ConfigureColumns->fetch_live_data($all_nodes);
535 $show_reservable_info = TRUE;
536 $show_layout_info = '1';
537 $show_conf = explode(";",$show_configuration);
538 foreach ($show_conf as $ss) {
539 if ($ss =="reservable")
540 $show_reservable_info = FALSE;
541 else if ($ss =="columns")
542 $show_layout_info = '0';
545 $slice_nodes=array();
546 $potential_nodes=array();
547 $reservable_nodes=array();
548 foreach ($all_nodes as $node) {
549 if (in_array($node['node_id'],$slice['node_ids'])) {
550 $slice_nodes[]=$node;
551 if ($node['node_type']=='reservable') $reservable_nodes[]=$node;
553 $potential_nodes[]=$node;
556 if ($profiling) plc_debug_prof('5: nodes',count($slice_nodes));
558 // outline the number of reservable nodes
559 $nodes_message=count_english($slice_nodes,"node");
560 if (count($reservable_nodes)) $nodes_message .= " (" . count($reservable_nodes) . " reservable)";
561 $toggle=new PlekitToggle ('my-slice-nodes',$nodes_message,
563 'Manage nodes attached to this slice',
564 'visible'=>get_arg('show_nodes',false)));
568 //////////////////// reservable nodes area
570 You have attached one or more reservable nodes to your slice.
571 Reservable nodes show up with the '$mark' mark.
572 Your slivers will be available only during timeslots
573 where you have obtained leases.
574 You can manage your leases in the tab below.
576 This feature is still experimental; feedback is appreciated at <a href='mailto:devel@planet-lab.org'>devel@planet-lab.org</a>
578 $count=count($reservable_nodes);
579 if ($count && $privileges) {
580 // having reservable nodes in white lists looks a bit off scope for now...
581 $toggle_nodes=new PlekitToggle('my-slice-nodes-reserve',
582 "Leases - " . count($reservable_nodes) . " reservable node(s)",
583 array('visible'=>get_arg('show_nodes_resa',false),
584 'info_text'=>$leases_info,
585 'info_visible'=>$show_reservable_info));
586 $toggle_nodes->start();
588 // get settings from environment, otherwise set to defaults
589 // when to start, in hours in the future from now
590 $leases_offset=$_GET['leases_offset'];
591 if ( ! $leases_offset ) $leases_offset=0;
592 // how many timeslots to show
593 $leases_slots=$_GET['leases_slots'];
594 if ( ! $leases_slots ) $leases_slots = 48;
595 // offset in hours (in the future) from now
596 $leases_w = $_GET['leases_w'];
597 if ( ! $leases_w) $leases_w=14;
598 // number of timeslots to display
600 $granularity=$api->GetLeaseGranularity();
602 // these elements are for passing data to the javascript layer
603 echo "<span class='hidden' id='leases_slicename'>" . $slice['name'] . "</span>";
604 echo "<span class='hidden' id='leases_slice_id'>" . $slice['slice_id']. "</span>";
605 echo "<span class='hidden' id='leases_granularity'>" . $granularity . "</span>";
606 // ditto, and editable - very rough for now
607 echo "<div class='center' id='leases_settings'>";
608 echo "<label id='leases_offset_label' class='leases_label'>start, in hours from now</label>";
609 echo "<input type='text' class='leases_input' id='leases_offset_input' value='$leases_offset' />";
610 echo "<label id='leases_slots_label' class='leases_label'># of timeslots</label>";
611 echo "<input type='text' class='leases_input' id='leases_slots_input' value='$leases_slots' />";
612 echo "<label id='leases_w_label' class='leases_label'>slot width, in pixels</label>";
613 echo "<input type='text' class='leases_input' id='leases_w_input' value='$leases_w' />";
616 // leases_data is the name used by leases.js to locate this place
617 // first population will be triggered by init_scheduler from leases.js
618 echo "<table id='leases_data' class='hidden'></table>";
620 // the general layout for the scheduler
622 <div id='leases_area'></div>
624 <div id='leases_buttons'>
625 <button id='leases_refresh' type='submit'>Refresh</button>
626 <button id='leases_submit' type='submit'>Submit</button>
630 $toggle_nodes->end();
634 //////////////////// node configuration panel
635 if ($first_time_configuration)
636 $column_conf_visible = '1';
638 $column_conf_visible = '0';
641 This tab allows you to customize the columns in the node tables,
642 below. Information on the nodes comes from a variety of monitoring
643 sources. If you, as either a user or a provider of monitoring data,
644 would like to see additional columns made available, please send us
645 your request in mail to <a
646 href="mailto:support@myslice.info">support@myslice.info</a>. You can
647 find more information about the MySlice project at <a
648 href="http://trac.myslice.info">http://trac.myslice.info</a>.
650 $toggle_nodes=new PlekitToggle('my-slice-nodes-configuration',
652 array('visible'=>$column_conf_visible,
653 'info_text'=>$layout_info,
654 'info_visible'=>$show_layout_info));
655 $toggle_nodes->start();
657 //usort ($table_headers, create_function('$col1,$col2','return strcmp($col1["header"],$col2["header"]);'));
658 //print("<p>TABLE HEADERS<p>");
659 //print_r($table_headers);
661 print("<div id='debug'></div>");
662 print("<input type='hidden' id='slice_id' value='".$slice['slice_id']."' />");
663 print("<input type='hidden' id='person_id' value='".$plc->person['person_id']."' />");
664 print("<input type='hidden' id='conf_tag_id' value='".$conf_tag_id."' />");
665 print("<input type='hidden' id='show_tag_id' value='".$show_tag_id."' />");
666 print("<input type='hidden' id='show_configuration' value='".$show_configuration."' />");
667 print("<input type='hidden' id='column_configuration' value='".$slice_column_configuration."' />");
668 print("<br><input type='hidden' size=80 id='full_column_configuration' value='".$column_configuration."' />");
669 print("<input type='hidden' id='previousConf' value='".$slice_column_configuration."' />");
670 print("<input type='hidden' id='defaultConf' value='".$default_configuration."' />");
672 $ConfigureColumns->configuration_panel_html(true);
674 $ConfigureColumns->javascript_init();
676 $toggle_nodes->end();
679 $all_sites=$api->GetSites(NULL, array('site_id','login_base'));
681 foreach ($all_sites as $tmp_site) $site_hash[$tmp_site['site_id']]=$tmp_site['login_base'];
683 $interface_columns=array('ip','node_id','interface_id');
684 $interface_filter=array('is_primary'=>TRUE);
685 $interfaces=$api->GetInterfaces($interface_filter,$interface_columns);
687 $interface_hash=array();
688 foreach ($interfaces as $interface) $interface_hash[$interface['node_id']]=$interface;
694 //////////////////// nodes currently in
695 $toggle_nodes=new PlekitToggle('my-slice-nodes-current',
696 count_english($slice_nodes,"node") . " currently in $name",
697 array('visible'=>get_arg('show_nodes_current',!$privileges)));
698 $toggle_nodes->start();
702 //$notes=array_merge($notes,$visibletags->notes());
703 $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";
706 $headers['peer']='string';
707 $headers['hostname']='string';
708 $short="-S-"; $long=Node::status_footnote(); $type='string';
709 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
710 $short=reservable_mark(); $long=reservable_legend(); $type='string';
711 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
712 // the extra tags, configured for the UI
713 $headers=array_merge($headers,$visibletags->headers());
715 if ($privileges) $headers[plc_delete_icon()]="none";
718 $edit_header = array();
719 if ($privileges) $edit_header[plc_delete_icon()]="none";
720 $headers = array_merge($ConfigureColumns->get_headers(),$edit_header);
722 //print("<p>HEADERS<p>");
725 $table_options = array('notes'=>$notes,
728 'configurable'=>true);
730 $table=new PlekitTable('nodes',$headers,NULL,$table_options);
732 $form=new PlekitForm(l_actions(),array('slice_id'=>$slice['slice_id']));
735 if ($slice_nodes) foreach ($slice_nodes as $node) {
738 $table->cell($node['node_id'], array('display'=>'none'));
740 $table->cell(l_node_obj($node));
741 $peers->cell($table,$node['peer_id']);
742 $run_level=$node['run_level'];
743 list($label,$class) = Node::status_label_class_($node);
744 $table->cell ($label,array('class'=>$class));
745 $table->cell( ($node['node_type']=='reservable')?reservable_mark():"" );
747 $hostname=$node['hostname'];
748 $ip=$interface_hash[$node['node_id']]['ip'];
749 $interface_id=$interface_hash[$node['node_id']]['interface_id'];
752 $node['domain'] = topdomain($hostname);
753 $node['sitename'] = l_site_t($node['site_id'],$site_hash[$node['site_id']]);
755 $node['ipaddress'] = l_interface_t($interface_id,$ip);
757 $node['ipaddress'] = "n/a";
759 //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
760 $ConfigureColumns->cells($table, $node);
762 if ($privileges) $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
769 $table->tfoot_start();
772 $table->cell($form->submit_html ("remove-nodes-from-slice","Remove selected"),
773 array('hfill'=>true,'align'=>'right'));
777 $toggle_nodes->end();
779 //////////////////// nodes to add
781 $new_potential_nodes = array();
782 if ($potential_nodes) foreach ($potential_nodes as $node) {
783 $emptywl=empty($node['slice_ids_whitelist']);
784 $inwl = (!emptywl) and in_array($slice['slice_id'],$node['slice_ids_whitelist']);
785 if ($emptywl or $inwl)
786 $new_potential_nodes[]=$node;
788 $potential_nodes=$new_potential_nodes;
790 $count=count($potential_nodes);
791 $toggle_nodes=new PlekitToggle('my-slice-nodes-add',
792 count_english($potential_nodes,"more node") . " available",
793 array('visible'=>get_arg('show_nodes_add',false)));
794 $toggle_nodes->start();
796 if ( $potential_nodes ) {
802 $headers['peer']='string';
803 $headers['hostname']='string';
804 $short="-S-"; $long=Node::status_footnote(); $type='string';
805 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
806 $short=reservable_mark(); $long=reservable_legend(); $type='string';
807 $headers[$short]=array('type'=>$type,'title'=>$long); $notes []= "$short = $long";
808 // the extra tags, configured for the UI
809 $headers=array_merge($headers,$visibletags->headers());
810 $headers['+']="none";
813 $add_header = array();
814 $add_header['+']="none";
815 $headers = array_merge($ConfigureColumns->get_headers(),$add_header);
817 //$notes=array_merge($notes,$visibletags->notes());
818 $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";
820 $table=new PlekitTable('add_nodes',$headers,NULL, $table_options);
821 $form=new PlekitForm(l_actions(),
822 array('slice_id'=>$slice['slice_id']));
825 if ($potential_nodes) foreach ($potential_nodes as $node) {
828 $table->cell($node['node_id'], array('display'=>'none'));
830 $table->cell(l_node_obj($node));
831 $peers->cell($table,$node['peer_id']);
832 list($label,$class) = Node::status_label_class_($node);
833 $table->cell ($label,array('class'=>$class));
834 $table->cell( ($node['node_type']=='reservable')?reservable_mark():"" );
837 $hostname=$node['hostname'];
838 $ip=$interface_hash[$node['node_id']]['ip'];
839 $interface_id=$interface_hash[$node['node_id']]['interface_id'];
840 $node['domain'] = topdomain($hostname);
841 $node['sitename'] = l_site_t($node['site_id'],$site_hash[$node['site_id']]);
842 $node['ipaddress'] = l_interface_t($interface_id,$ip);
844 //foreach ($visiblecolumns as $tagname) $table->cell($node[$tagname]);
845 $ConfigureColumns->cells($table, $node);
847 $table->cell ($form->checkbox_html('node_ids[]',$node['node_id']));
851 $table->tfoot_start();
853 $table->cell($form->submit_html ("add-nodes-in-slice","Add selected"),
854 array('hfill'=>true,'align'=>'right'));
859 $toggle_nodes->end();
864 //////////////////////////////////////// retrieve all slice tags
865 $tags=$api->GetSliceTags (array('slice_id'=>$slice_id));
866 //////////////////////////////////////////////////////////// tab:initscripts
868 // * add a message on how to use this:
869 // * explain the 2 mechanisms (initscript_code, initscript)
870 // * explain the interface : initscript start|stop|restart slicename
874 There are two ways to attach an initscript to a slice:<ul>
876 <li> <span class='bold'> Shared initscripts </span> are global to the
877 MyPLC, and managed by the Operations Team. For that reason, regular
878 users cannot change these scripts, but can reference one of the
879 available names in the drop down below. </li>
881 <li> You also have the option to provide <span class='bold'> your own
882 code </span>, with the following conventions: <ul>
884 <li> Like regular initscripts, your script must except to receive as a
885 first argument <span class='bold'> start </span>, <span class='bold'>
886 stop </span> or <span class='bold'> restart </span>. It is important
887 to honor this argument, as your slice may be stopped and restarted at
888 any time; also this is used whenever the installed code gets changed.
891 <li> As a second argument, you will receive the slicename; in most
892 cases this can be safely ignored. </li>
897 The slice-specific setting has precedence on a shared initscript.
900 $shared_initscripts=$api->GetInitScripts(array('-SORT'=>'name'),array('name'));
901 //$shared_initscripts=$api->GetInitScripts();
902 if ($profiling) plc_debug_prof('6 initscripts',count($initscripts));
903 // xxx expose this even on foreign slices for now
907 if ($tags) foreach ($tags as $tag) {
908 if ($tag['tagname']=='initscript') {
909 if ($initscript!='') drupal_set_error("multiple occurrences of 'initscript' tag");
910 $initscript=$tag['value'];
912 if ($tag['tagname']=='initscript_code') {
913 if ($initscript_code!='') drupal_set_error("multiple occurrences of 'initscript_code' tag");
914 $initscript_code=$tag['value'];
915 // plc_debug_txt('retrieved body',$initscript_code);
918 $label="No initscript";
919 $trimmed=trim($initscript_code);
920 if (!empty($trimmed)) $label="Initscript : slice-specific (" . substr($initscript_code,0,20) . " ...)";
921 else if (!empty($initscript)) $label="Initscript: shared " . $initscript;
923 $toggle = new PlekitToggle('slice-initscripts',$label,
924 array('bubble'=>'Manage initscript on that slice',
925 'visible'=>get_arg('show_initscripts',false),
926 'info_text'=>$initscript_info
927 // not messing with persontags to guess whether this should be displayed or not
928 // hopefully some day toggle will know how to handle that using web storage
932 $details=new PlekitDetails(TRUE);
933 $details->form_start(l_actions(),array('action'=>'update-initscripts',
934 'slice_id'=>$slice_id,
936 'previous-initscript'=>$initscript,
937 'previous-initscript-code'=>$initscript_code));
939 // comppute a pulldown with available names
942 if ($shared_initscripts) foreach ($shared_initscripts as $is) {
943 $is_selector=array('display'=>$is['name'],'value'=>$is['name']);
944 if ($is['name']==$initscript) {
945 $is_selector['selected']=TRUE;
948 $selectors[]=$is_selector;
950 // display a warning when initscript references an unknown script
951 $details->tr_submit('unused','Update initscripts');
953 $details->th_td("shared initscript name",
954 $details->form()->select_html('initscript',$selectors,array('label'=>'none')),
956 array('input_type'=>'select'));
957 if ($initscript && ! $is_found)
958 // xxx better rendering ?
959 $details->th_td('WARNING',plc_warning_html("Current name '" . $initscript . "' is not a known shared initscript name"));
960 ////////// by contents
963 if ($initscript_code) {
964 $text=explode("\n",$initscript_code);
965 $script_height=count($text);
967 foreach ($text as $line) $script_width=max($script_width,strlen($line));
969 $details->th_td('slice initscript',$initscript_code,'initscript-code',
970 array('input_type'=>'textarea', 'width'=>$script_width,'height'=>$script_height));
971 $details->tr_submit('unused','Update initscripts');
972 $details->form_end();
977 //////////////////////////////////////////////////////////// tab:tags
978 // very wide values get abbreviated
979 $tag_value_threshold=24;
981 // * this area could use a help message about some special tags:
982 // * initscript-related should be taken out
983 // * sliverauth-related (ssh_key & hmac) should have a toggle to hide or show
986 // xxx expose this even on foreign slices for now
987 //if ( $local_peer ) {
988 if ($profiling) plc_debug_prof('7 slice tags',count($tags));
989 function get_tagname ($tag) { return $tag['tagname'];}
990 $tagnames = array_map ("get_tagname",$tags);
992 $toggle = new PlekitToggle ('slice-tags',count_english_warning($tags,'tag'),
993 array('bubble'=>'Inspect and set tags on that slice',
994 'visible'=>get_arg('show_tags',false)));
1001 "NodeGroup"=>"string");
1002 if ($tags_privileges) $headers[plc_delete_icon()]="none";
1004 $table_options=array("notes_area"=>false,"pagesize_area"=>false,"search_width"=>10);
1005 $table=new PlekitTable("slice_tags",$headers,'0',$table_options);
1006 $form=new PlekitForm(l_actions(),
1007 array('slice_id'=>$slice['slice_id']));
1011 // Get hostnames for nodes in a single pass
1012 $_node_ids = array();
1013 foreach ($tags as $tag) {
1014 if ($tag['node_id']) {
1015 array_push($_node_ids, $tag['node_id']);
1018 $_nodes = $api->GetNodes(array('node_id' => $_node_ids), array('node_id', 'hostname'));
1019 $_hostnames = array();
1020 foreach ($_nodes as $_node) {
1021 $_hostnames[$_node['node_id']] = $_node['hostname'];
1024 // Loop through tags again to display
1025 foreach ($tags as $tag) {
1027 if ($tag['node_id']) {
1028 $node_name = $_hostnames[$tag['node_id']];
1030 $nodegroup_name="n/a";
1031 if ($tag['nodegroup_id']) {
1032 $nodegroups=$api->GetNodeGroups(array('nodegroup_id'=>$tag['nodegroup_id']));
1033 if ($profiling) plc_debug_prof('8 nodegroup for slice tag',$nodegroup);
1035 $nodegroup = $nodegroups[0];
1036 $nodegroup_name = $nodegroup['groupname'];
1039 $table->row_start();
1040 $table->cell(l_tag_obj($tag));
1041 // very wide values get abbreviated
1042 $table->cell(truncate_and_popup($tag['value'],$tag_value_threshold));
1043 $table->cell($node_name);
1044 $table->cell($nodegroup_name);
1045 if ($tags_privileges) $table->cell ($form->checkbox_html('slice_tag_ids[]',$tag['slice_tag_id']));
1049 if ($tags_privileges) {
1050 $table->tfoot_start();
1051 $table->row_start();
1052 $table->cell($form->submit_html ("delete-slice-tags","Remove selected"),
1053 array('hfill'=>true,'align'=>'right'));
1056 $table->row_start();
1057 function tag_selector ($tag) {
1058 return array("display"=>$tag['tagname'],"value"=>$tag['tag_type_id']);
1060 $all_tags= $api->GetTagTypes( array ("category"=>"slice*","-SORT"=>"+tagname"), array("tagname","tag_type_id"));
1061 if ($profiling) plc_debug_prof('9 tagtypes',count($all_tags));
1062 $selector_tag=array_map("tag_selector",$all_tags);
1064 function node_selector($node) {
1065 return array("display"=>$node["hostname"],"value"=>$node['node_id']);
1067 $selector_node=array_map("node_selector",$slice_nodes);
1069 function nodegroup_selector($ng) {
1070 return array("display"=>$ng["groupname"],"value"=>$ng['nodegroup_id']);
1072 $all_nodegroups = $api->GetNodeGroups( array("groupname"=>"*"), array("groupname","nodegroup_id"));
1073 if ($profiling) plc_debug_prof('10 nodegroups',count($all_nodegroups));
1074 $selector_nodegroup=array_map("nodegroup_selector",$all_nodegroups);
1076 $table->cell($form->select_html("tag_type_id",$selector_tag,array('label'=>"Choose Tag")));
1077 $table->cell($form->text_html("value","",array('width'=>8)));
1078 $table->cell($form->select_html("node_id",$selector_node,array('label'=>"All Nodes")));
1079 $table->cell($form->select_html("nodegroup_id",$selector_nodegroup,array('label'=>"No Nodegroup")));
1080 $table->cell($form->submit_html("add-slice-tag","Set Tag"),array('columns'=>2,'align'=>'left'));
1090 //////////////////////////////////////////////////////////// tab:renew
1092 if ( ! $renew_visible) renew_area ($slice,$site,false);
1095 $peers->block_end($peer_id);
1097 if ($profiling) plc_debug_prof_end();
1100 include 'plc_footer.php';