| #!/usr/bin/perl -w |
| ############################################################################### |
| # $Id$ |
| ############################################################################### |
| # Licensed to the Apache Software Foundation (ASF) under one or more |
| # contributor license agreements. See the NOTICE file distributed with |
| # this work for additional information regarding copyright ownership. |
| # The ASF licenses this file to You under the Apache License, Version 2.0 |
| # (the "License"); you may not use this file except in compliance with |
| # the License. You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| ############################################################################### |
| |
| =head1 NAME |
| |
| VCL::new - Perl module for the VCL new state |
| |
| =head1 SYNOPSIS |
| |
| use VCL::new; |
| use VCL::utils; |
| |
| # Set variables containing the IDs of the request and reservation |
| my $request_id = 5; |
| my $reservation_id = 6; |
| |
| # Call the VCL::utils::get_request_info subroutine to populate a hash |
| my %request_info = get_request_info($request_id); |
| |
| # Set the reservation ID in the hash |
| $request_info{RESERVATIONID} = $reservation_id; |
| |
| # Create a new VCL::new object based on the request information |
| my $new = VCL::new->new(%request_info); |
| |
| =head1 DESCRIPTION |
| |
| This module supports the VCL "new" state. |
| |
| =cut |
| |
| ############################################################################## |
| package VCL::new; |
| |
| # Specify the lib path using FindBin |
| use FindBin; |
| use lib "$FindBin::Bin/.."; |
| |
| # Configure inheritance |
| use base qw(VCL::Module::State); |
| |
| # Specify the version of this module |
| our $VERSION = '2.00'; |
| |
| # Specify the version of Perl to use |
| use 5.008000; |
| |
| use strict; |
| use warnings; |
| use diagnostics; |
| |
| use VCL::utils; |
| |
| ############################################################################## |
| |
| =head1 OBJECT METHODS |
| |
| =cut |
| |
| #///////////////////////////////////////////////////////////////////////////// |
| |
| =head2 process |
| |
| Parameters : |
| Returns : |
| Description : |
| |
| =cut |
| |
| sub process { |
| my $self = shift; |
| |
| my $request_data = $self->data->get_request_data(); |
| my $request_id = $self->data->get_request_id(); |
| my $request_logid = $self->data->get_request_log_id(); |
| my $request_state_name = $self->data->get_request_state_name(); |
| my $request_laststate_name = $self->data->get_request_laststate_name(); |
| my $request_forimaging = $self->data->get_request_forimaging(); |
| my $request_preload_only = $self->data->get_request_preload_only(); |
| my $reservation_count = $self->data->get_reservation_count(); |
| my $reservation_id = $self->data->get_reservation_id(); |
| my $reservation_is_parent = $self->data->is_parent_reservation; |
| my $computer_id = $self->data->get_computer_id(); |
| my $computer_host_name = $self->data->get_computer_host_name(); |
| my $computer_short_name = $self->data->get_computer_short_name(); |
| my $computer_type = $self->data->get_computer_type(); |
| my $computer_ip_address = $self->data->get_computer_ip_address(); |
| my $computer_state_name = $self->data->get_computer_state_name(); |
| my $computer_next_image_id = $self->data->get_computer_nextimage_id(); |
| my $computer_next_image_name = $self->data->get_computer_nextimage_name(); |
| my $image_id = $self->data->get_image_id(); |
| my $image_os_name = $self->data->get_image_os_name(); |
| my $image_name = $self->data->get_image_name(); |
| my $image_prettyname = $self->data->get_image_prettyname(); |
| my $image_project = $self->data->get_image_project(); |
| my $image_reloadtime = $self->data->get_image_reload_time(); |
| my $image_architecture = $self->data->get_image_architecture(); |
| my $image_os_type = $self->data->get_image_os_type(); |
| my $imagemeta_checkuser = $self->data->get_imagemeta_checkuser(); |
| my $imagemeta_usergroupid = $self->data->get_imagemeta_usergroupid(); |
| my $imagemeta_usergroupmembercount = $self->data->get_imagemeta_usergroupmembercount(); |
| my $imagemeta_usergroupmembers = $self->data->get_imagemeta_usergroupmembers(); |
| my $imagerevision_id = $self->data->get_imagerevision_id(); |
| my $managementnode_id = $self->data->get_management_node_id(); |
| my $managementnode_hostname = $self->data->get_management_node_hostname(); |
| my $user_unityid = $self->data->get_user_login_id(); |
| my $user_uid = $self->data->get_user_uid(); |
| my $user_affiliation_sitewwwaddress = $self->data->get_user_affiliation_sitewwwaddress(); |
| my $user_affiliation_helpaddress = $self->data->get_user_affiliation_helpaddress(); |
| my $user_standalone = $self->data->get_user_standalone(); |
| my $user_email = $self->data->get_user_email(); |
| my $user_emailnotices = $self->data->get_user_emailnotices(); |
| my $user_imtype_name = $self->data->get_user_imtype_name(); |
| my $user_im_id = $self->data->get_user_im_id(); |
| |
| notify($ERRORS{'OK'}, 0, "reservation is parent = $reservation_is_parent"); |
| notify($ERRORS{'OK'}, 0, "preload only = $request_preload_only"); |
| notify($ERRORS{'OK'}, 0, "originating request state = $request_state_name"); |
| notify($ERRORS{'OK'}, 0, "originating request laststate = $request_laststate_name"); |
| notify($ERRORS{'OK'}, 0, "originating computer state = $computer_state_name"); |
| notify($ERRORS{'OK'}, 0, "originating computer type = $computer_type"); |
| |
| # If state is tomaintenance, place machine into maintenance state and set request to complete |
| if ($request_state_name =~ /tomaintenance/) { |
| notify($ERRORS{'OK'}, 0, "this is a 'tomaintenance' request"); |
| |
| # Update the request state to complete, update the computer state to maintenance, exit |
| # Do not update log.ending for tomaintenance reservations |
| if (switch_state($request_data, 'complete', 'maintenance', '', '0')) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name set to maintenance"); |
| } |
| |
| |
| if ($self->provisioner->can("post_maintenance_action")) { |
| if ($self->provisioner->post_maintenance_action()) { |
| notify($ERRORS{'OK'}, 0, "post action completed $computer_short_name"); |
| } |
| } |
| else { |
| notify($ERRORS{'OK'}, 0, "post action skipped, post_maintenance_action not implemented by " . ref($self->provisioner) . ", assuming no steps required"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } ## end if ($request_state_name =~ /tomaintenance/) |
| |
| #If reload state is not new (reload) and computer is part of block allocation |
| #confirm imagerevisionid is the production image. |
| if($request_state_name ne 'new') { |
| notify($ERRORS{'OK'}, 0, "request_state_name is not new"); |
| if(is_inblockrequest($computer_id)){ |
| notify($ERRORS{'OK'}, 0, "computer_id $computer_id is in blockrequest"); |
| my %imagerev_info = get_production_imagerevision_info($image_id); |
| unless($imagerevision_id == $imagerev_info{id}){ |
| notify($ERRORS{'OK'}, 0, "imagerevision_id does not match imagerevision_id= $imagerevision_id imagerev_info $imagerev_info{id}"); |
| $self->data->set_imagerevision_id($imagerev_info{id}); |
| $self->data->set_sublog_imagerevisionid($imagerev_info{id}); |
| $self->data->set_image_name($imagerev_info{imagename}); |
| $self->data->set_imagerevision_revision($imagerev_info{revision}); |
| |
| #reset variables in this scope |
| $imagerevision_id = $imagerev_info{id}; |
| $image_name = $imagerev_info{imagename}; |
| } |
| } |
| } |
| |
| # Confirm requested computer is available |
| if ($self->computer_not_being_used()) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name is not being used"); |
| } |
| elsif ($request_state_name ne 'new') { |
| # Computer is not available, not a new request (most likely a simple reload) |
| notify($ERRORS{'WARNING'}, 0, "request state=$request_state_name, $computer_short_name is NOT available"); |
| |
| # Set the computer next image so it gets loaded if/when other reservations are complete |
| if (!defined($computer_next_image_name) || $image_name ne $computer_next_image_name) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name is not available, setting computer next image to $image_name"); |
| if (setnextimage($computer_id, $image_id)) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name next image set to $image_name"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set $computer_short_name next image to $image_name"); |
| } |
| } |
| else { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name is not available, computer next image is already set to $image_name"); |
| } |
| |
| # Update request state to complete |
| if (update_request_state($request_id, "complete", $request_state_name)) { |
| notify($ERRORS{'OK'}, 0, "request state updated to 'complete'/'$request_state_name'"); |
| } |
| else { |
| notify($ERRORS{'CRITICAL'}, 0, "failed to update the request state to 'complete'/'$request_state_name'"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } ## end elsif ($request_state_name ne 'new') [ if ($self->computer_not_being_used()) |
| elsif ($request_preload_only) { |
| # Computer is not available, preload only = true |
| notify($ERRORS{'WARNING'}, 0, "preload reservation, $computer_short_name is NOT available"); |
| |
| # Set the computer next image so it gets loaded if/when other reservations are complete |
| if (!defined($computer_next_image_name) || $image_name ne $computer_next_image_name) { |
| notify($ERRORS{'OK'}, 0, "preload only request, $computer_short_name is not available, setting computer next image to $image_name"); |
| if (setnextimage($computer_id, $image_id)) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name next image set to $image_name"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set $computer_short_name next image to $image_name"); |
| } |
| } |
| else { |
| notify($ERRORS{'OK'}, 0, "preload only request, $computer_short_name is not available, computer next image is already set to $image_name"); |
| } |
| |
| # Only the parent reservation is allowed to modify the request state in this module |
| if (!$reservation_is_parent) { |
| notify($ERRORS{'OK'}, 0, "child preload reservation, computer is not available, states will be changed by the parent"); |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } |
| |
| # Return back to original states |
| notify($ERRORS{'OK'}, 0, "parent preload reservation, returning states back to original"); |
| |
| # Set the preload flag back to 1 so it will be processed again |
| if (update_preload_flag($request_id, 1)) { |
| notify($ERRORS{'OK'}, 0, "updated preload flag to 1"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to update preload flag to 1"); |
| } |
| |
| # Return request state back to the original |
| if (update_request_state($request_id, $request_state_name, $request_state_name)) { |
| notify($ERRORS{'OK'}, 0, "request state set back to '$request_state_name'/'$request_state_name'"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set request state back to '$request_state_name'/'$request_state_name'"); |
| } |
| |
| # Return computer state back to the original |
| if (update_computer_state($computer_id, $computer_state_name)) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name state set back to '$computer_state_name'"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set $computer_short_name state back to '$computer_state_name'"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } ## end elsif ($request_preload_only) [ if ($self->computer_not_being_used()) |
| else { |
| # Computer not available, state=new, PRELOADONLY = false |
| notify($ERRORS{'WARNING'}, 0, "$computer_short_name is NOT available"); |
| |
| # Call reservation_failed |
| $self->reservation_failed("process failed because computer is not available"); |
| } |
| |
| # Confirm requested resouces are available |
| if ($self->reload_image()) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name is loaded with $image_name"); |
| } |
| elsif ($request_preload_only) { |
| # Load failed preload only = true |
| notify($ERRORS{'WARNING'}, 0, "preload reservation, failed to load $computer_short_name with $image_name"); |
| |
| # Check if parent, only the parent is allowed to modify the request state in this module |
| if (!$reservation_is_parent) { |
| notify($ERRORS{'OK'}, 0, "this is a child preload reservation, states will be changed by the parent"); |
| |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } |
| |
| # Return back to original states |
| notify($ERRORS{'OK'}, 0, "this is a parent preload reservation, returning states back to original"); |
| |
| # Set the preload flag back to 1 so it will be processed again |
| if (update_preload_flag($request_id, 1)) { |
| notify($ERRORS{'OK'}, 0, "updated preload flag to 1"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to update preload flag to 1"); |
| } |
| |
| # Return request state back to the original |
| if (update_request_state($request_id, $request_state_name, $request_state_name)) { |
| notify($ERRORS{'OK'}, 0, "request state set back to '$request_state_name'/'$request_state_name'"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set request state back to '$request_state_name'/'$request_state_name'"); |
| } |
| |
| # Return computer state back to the original |
| if (update_computer_state($computer_id, $computer_state_name)) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name state set back to '$computer_state_name'"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set $computer_short_name state back to '$computer_state_name'"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } ## end elsif ($request_preload_only) [ if ($self->reload_image()) |
| else { |
| # Load failed, PRELOADONLY = false |
| notify($ERRORS{'WARNING'}, 0, "failed to load $computer_short_name with $image_name"); |
| |
| # Call reservation_failed, problem computer not opened for reservation |
| $self->reservation_failed("process failed after trying to load or make available"); |
| } |
| |
| |
| # Parent only checks and waits for any other images to complete and checkin |
| if ($reservation_is_parent && $reservation_count > 1) { |
| insertloadlog($reservation_id, $computer_id, "info", "cluster based reservation"); |
| |
| # Wait on child reservations |
| if ($self->wait_for_child_reservations()) { |
| notify($ERRORS{'OK'}, 0, "done waiting for child reservations, they are all ready"); |
| } |
| else { |
| # Call reservation_failed, problem computer not opened for reservation |
| $self->reservation_failed("child reservations never all became ready"); |
| } |
| } ## end if ($reservation_is_parent && $reservation_count... |
| |
| |
| # Check if request has been deleted |
| if (is_request_deleted($request_id)) { |
| notify($ERRORS{'OK'}, 0, "request has been deleted, setting computer state to 'available' and exiting"); |
| |
| # Update state of computer and exit |
| switch_state($request_data, '', 'available', '', '1'); |
| } |
| |
| my $next_computer_state; |
| my $next_request_state; |
| |
| # Attempt to reserve the computer if this is a 'new' reservation |
| # These steps are not done for simple reloads |
| if ($request_state_name eq 'new') { |
| # Set the computer next image to the one for this reservation |
| if (!defined($computer_next_image_name) || $image_name ne $computer_next_image_name) { |
| if (setnextimage($computer_id, $image_id)) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name next image set to $image_name"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set $computer_short_name next image to $image_name"); |
| } |
| } |
| else { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name next image is already set to $image_name"); |
| } |
| |
| if ($request_preload_only) { |
| # Return back to original states |
| notify($ERRORS{'OK'}, 0, "this is a preload reservation, returning states back to original"); |
| |
| # Set the preload flag back to 1 so it will be processed again |
| if (update_preload_flag($request_id, 1)) { |
| notify($ERRORS{'OK'}, 0, "updated preload flag to 1"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to update preload flag to 1"); |
| } |
| |
| # Set variables for the next states |
| $next_computer_state = $computer_state_name; |
| $next_request_state = $request_state_name; |
| |
| } ## end if ($request_preload_only) |
| else { |
| # Perform the steps necessary to prepare the computer for a user |
| if ($self->reserve_computer()) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name with $image_name successfully reserved"); |
| } |
| else { |
| # reserve_computer() returned false |
| notify($ERRORS{'OK'}, 0, "$computer_short_name with $image_name could NOT be reserved"); |
| |
| # Call reservation_failed, problem computer not opened for reservation |
| $self->reservation_failed("process failed after attempting to reserve the computer"); |
| } |
| |
| # Insert a row into the computerloadlog table |
| if (insertloadlog($reservation_id, $computer_id, "reserved", "$computer_short_name successfully reserved with $image_name")) { |
| notify($ERRORS{'OK'}, 0, "inserted computerloadlog entry, load state=reserved"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to insert computerloadlog entry, load state=reserved"); |
| } |
| |
| # Set variables for the next states |
| $next_computer_state = "reserved"; |
| $next_request_state = "reserved"; |
| } ## end else [ if ($request_preload_only) |
| } ## end if ($request_state_name eq 'new') |
| elsif ($request_state_name eq 'tovmhostinuse') { |
| # Set variables for the next states |
| $next_computer_state = "vmhostinuse"; |
| $next_request_state = "complete"; |
| } |
| else { |
| # Set variables for the next states |
| $next_computer_state = "available"; |
| $next_request_state = "complete"; |
| } |
| |
| # Update the computer state |
| if (update_computer_state($computer_id, $next_computer_state)) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name state set to '$next_computer_state'"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set $computer_short_name state to '$next_computer_state'"); |
| } |
| |
| # Update request state if this is the parent reservation |
| # Only parent reservations should modify the request state |
| if ($reservation_is_parent && update_request_state($request_id, $next_request_state, $request_state_name)) { |
| notify($ERRORS{'OK'}, 0, "request state set to '$next_request_state'/'$request_state_name'"); |
| } |
| elsif ($reservation_is_parent) { |
| notify($ERRORS{'CRITICAL'}, 0, "failed to set request state to '$next_request_state'/'$request_state_name'"); |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| } |
| else { |
| notify($ERRORS{'OK'}, 0, "this is a child image, request state NOT changed to '$next_request_state'"); |
| } |
| |
| # Insert a row into the computerloadlog table |
| if (insertloadlog($reservation_id, $computer_id, "info", "$computer_short_name successfully set to $next_computer_state with $image_name")) { |
| notify($ERRORS{'OK'}, 0, "inserted computerloadlog entry: $computer_short_name successfully set to $next_computer_state with $image_name"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to insert computerloadlog entry: $computer_short_name successfully set to $next_computer_state with $image_name"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "exiting"); |
| exit; |
| |
| } ## end sub process |
| |
| #///////////////////////////////////////////////////////////////////////////// |
| |
| =head2 reload_image |
| |
| Parameters : |
| Returns : |
| Description : |
| |
| =cut |
| |
| sub reload_image { |
| my $self = shift; |
| |
| my $request_data = $self->data->get_request_data(); |
| my $request_id = $self->data->get_request_id(); |
| my $request_logid = $self->data->get_request_log_id(); |
| my $request_state_name = $self->data->get_request_state_name(); |
| my $request_laststate_name = $self->data->get_request_laststate_name(); |
| my $request_forimaging = $self->data->get_request_forimaging(); |
| my $request_preload_only = $self->data->get_request_preload_only(); |
| my $reservation_count = $self->data->get_reservation_count(); |
| my $reservation_id = $self->data->get_reservation_id(); |
| my $reservation_is_parent = $self->data->is_parent_reservation; |
| my $computer_id = $self->data->get_computer_id(); |
| my $computer_host_name = $self->data->get_computer_host_name(); |
| my $computer_short_name = $self->data->get_computer_short_name(); |
| my $computer_type = $self->data->get_computer_type(); |
| my $computer_ip_address = $self->data->get_computer_ip_address(); |
| my $computer_state_name = $self->data->get_computer_state_name(); |
| my $computer_next_image_id = $self->data->get_computer_nextimage_id(); |
| my $computer_next_image_name = $self->data->get_computer_nextimage_name(); |
| my $computer_currentimage_name = $self->data->get_computer_currentimage_name(); |
| my $image_id = $self->data->get_image_id(); |
| my $image_os_name = $self->data->get_image_os_name(); |
| my $image_name = $self->data->get_image_name(); |
| my $image_prettyname = $self->data->get_image_prettyname(); |
| my $image_project = $self->data->get_image_project(); |
| my $image_reloadtime = $self->data->get_image_reload_time(); |
| my $image_architecture = $self->data->get_image_architecture(); |
| my $image_os_install_type = $self->data->get_image_os_install_type(); |
| my $image_os_type = $self->data->get_image_os_type(); |
| my $imagemeta_checkuser = $self->data->get_imagemeta_checkuser(); |
| my $imagemeta_usergroupid = $self->data->get_imagemeta_usergroupid(); |
| my $imagemeta_usergroupmembercount = $self->data->get_imagemeta_usergroupmembercount(); |
| my $imagemeta_usergroupmembers = $self->data->get_imagemeta_usergroupmembers(); |
| my $imagerevision_id = $self->data->get_imagerevision_id(); |
| my $managementnode_id = $self->data->get_management_node_id(); |
| my $managementnode_hostname = $self->data->get_management_node_hostname(); |
| my $user_unityid = $self->data->get_user_login_id(); |
| my $user_uid = $self->data->get_user_uid(); |
| my $user_affiliation_sitewwwaddress = $self->data->get_user_affiliation_sitewwwaddress(); |
| my $user_affiliation_helpaddress = $self->data->get_user_affiliation_helpaddress(); |
| my $user_standalone = $self->data->get_user_standalone(); |
| my $user_email = $self->data->get_user_email(); |
| my $user_emailnotices = $self->data->get_user_emailnotices(); |
| my $user_imtype_name = $self->data->get_user_imtype_name(); |
| my $user_im_id = $self->data->get_user_im_id(); |
| |
| # Try to get the node status if the provisioning engine has implemented a node_status() subroutine |
| my $node_status; |
| my $node_status_string = ''; |
| if ($self->provisioner->can("node_status")) { |
| notify($ERRORS{'DEBUG'}, 0, "calling " . ref($self->provisioner) . "->node_status()"); |
| insertloadlog($reservation_id, $computer_id, "statuscheck", "checking status of node"); |
| |
| # Call node_status(), check the return value |
| $node_status = $self->provisioner->node_status(); |
| |
| # Make sure a return value is defined, an error occurred if it is undefined |
| if (!defined($node_status)) { |
| notify($ERRORS{'CRITICAL'}, 0, ref($self->provisioner) . "->node_status() returned an undefined value, returning"); |
| return; |
| } |
| |
| # Check what node_status returned and try to get the "status" string |
| # First see if it returned a hashref |
| if (ref($node_status) eq 'HASH') { |
| notify($ERRORS{'DEBUG'}, 0, "node_status returned a hash reference"); |
| |
| # Check if the hash contains a key called "status" |
| if (defined $node_status->{status}) { |
| $node_status_string = $node_status->{status}; |
| notify($ERRORS{'DEBUG'}, 0, "node_status hash reference contains key {status}=$node_status_string"); |
| } |
| else { |
| notify($ERRORS{'DEBUG'}, 0, "node_status hash reference does not contain a key called 'status'"); |
| } |
| } ## end if (ref($node_status) eq 'HASH') |
| |
| # Check if node_status returned an array ref |
| elsif (ref($node_status) eq 'ARRAY') { |
| notify($ERRORS{'DEBUG'}, 0, "node_status returned an array reference"); |
| |
| # Check if the hash contains a key called "status" |
| if (defined((@{$node_status})[0])) { |
| $node_status_string = (@{$node_status})[0]; |
| notify($ERRORS{'DEBUG'}, 0, "node_status array reference contains index [0]=$node_status_string"); |
| } |
| else { |
| notify($ERRORS{'DEBUG'}, 0, "node_status array reference is empty"); |
| } |
| } ## end elsif (ref($node_status) eq 'ARRAY') [ if (ref($node_status) eq 'HASH') |
| |
| # Check if node_status didn't return a reference |
| # Assume string was returned |
| elsif (!ref($node_status)) { |
| # Use scalar value of node_status's return value |
| $node_status_string = $node_status; |
| notify($ERRORS{'DEBUG'}, 0, "node_status returned a scalar: $node_status"); |
| } |
| |
| else { |
| notify($ERRORS{'CRITICAL'}, 0, ref($self->provisioner) . "->node_status() returned an unsupported reference type: " . ref($node_status) . ", returning"); |
| insertloadlog($reservation_id, $computer_id, "failed", "node_status() returned an undefined value"); |
| return; |
| } |
| } ## end if ($self->provisioner->can("node_status")) |
| else { |
| notify($ERRORS{'OK'}, 0, "node status not checked, node_status() not implemented by " . ref($self->provisioner) . ", assuming load=true"); |
| } |
| |
| if ($computer_state_name eq 'reload') { |
| # Always call load() if state is reload regardless of node_status() |
| # Admin-initiated reloads will always cause node to be reloaded |
| notify($ERRORS{'OK'}, 0, "request state is $request_state_name, node will be reloaded regardless of status"); |
| $node_status_string = 'reload'; |
| } |
| |
| # Check if the status string returned by node_status = 'ready' |
| if ($node_status_string =~ /^ready/i) { |
| # node_status returned 'ready' |
| notify($ERRORS{'OK'}, 0, "node_status returned '$node_status_string', $computer_short_name will not be reloaded"); |
| insertloadlog($reservation_id, $computer_id, "info", "node status is $node_status_string, $computer_short_name will not be reloaded"); |
| } |
| |
| elsif ($node_status_string =~ /^post_load/i) { |
| notify($ERRORS{'OK'}, 0, "node_status returned '$node_status_string', OS post_load tasks will be performed on $computer_short_name"); |
| |
| # Check if the OS module implements a post_load subroutine and that post_load has been run |
| if ($self->os->can('post_load')) { |
| if ($self->os->post_load()) { |
| # Add the vcld_post_load line to currentimage.txt |
| $self->os->set_vcld_post_load_status(); |
| $node_status_string = 'READY'; |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to execute OS module's post_load() subroutine, $computer_short_name will be reloaded"); |
| $node_status_string = 'POST_LOAD_FAILED'; |
| } |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "provisioning module's node_status subroutine returned '$node_status' but OS module " . ref($self->os) . " does not implement a post_load() subroutine, $computer_short_name will not be reloaded"); |
| } |
| } |
| |
| # Provisioning module's node_status subroutine did not return 'ready' |
| if ($node_status_string !~ /^ready/i) { |
| notify($ERRORS{'OK'}, 0, "node status is $node_status_string, $computer_short_name will be reloaded"); |
| insertloadlog($reservation_id, $computer_id, "loadimageblade", "$computer_short_name must be reloaded with $image_name"); |
| |
| # Make sure provisioning module's load() subroutine exists |
| if (!$self->provisioner->can("load")) { |
| notify($ERRORS{'CRITICAL'}, 0, ref($self->provisioner) . "->load() subroutine does not exist, returning"); |
| insertloadlog($reservation_id, $computer_id, "failed", ref($self->provisioner) . "->load() subroutine does not exist"); |
| return; |
| } |
| |
| |
| # Make sure the image exists on this management node's local disks |
| # Attempt to retrieve it if necessary |
| if ($self->provisioner->can("does_image_exist")) { |
| notify($ERRORS{'DEBUG'}, 0, "calling " . ref($self->provisioner) . "->does_image_exist()"); |
| |
| if ($self->provisioner->does_image_exist($image_name)) { |
| notify($ERRORS{'OK'}, 0, "$image_name exists on this management node"); |
| insertloadlog($reservation_id, $computer_id, "doesimageexists", "confirmed image exists"); |
| } |
| else { |
| notify($ERRORS{'OK'}, 0, "$image_name does not exist on this management node"); |
| |
| # Try to retrieve the image files from another management node |
| if ($self->provisioner->can("retrieve_image")) { |
| notify($ERRORS{'DEBUG'}, 0, "calling " . ref($self->provisioner) . "->retrieve_image()"); |
| |
| if ($self->provisioner->retrieve_image($image_name)) { |
| notify($ERRORS{'OK'}, 0, "$image_name was retrieved from another management node"); |
| } |
| else { |
| notify($ERRORS{'CRITICAL'}, 0, "$image_name does not exist on management node and could not be retrieved"); |
| insertloadlog($reservation_id, $computer_id, "failed", "requested image does not exist on management node and could not be retrieved"); |
| return; |
| } |
| } ## end if ($self->provisioner->can("retrieve_image"... |
| else { |
| notify($ERRORS{'CRITICAL'}, 0, "unable to retrieve image from another management node, retrieve_image() is not implemented by " . ref($self->provisioner)); |
| insertloadlog($reservation_id, $computer_id, "failed", "failed requested image does not exist on management node, retrieve_image() is not implemented"); |
| return; |
| } |
| } ## end else [ if ($self->provisioner->does_image_exist($image_name... |
| } ## end if ($self->provisioner->can("does_image_exist"... |
| else { |
| notify($ERRORS{'OK'}, 0, "unable to check if image exists, does_image_exist() not implemented by " . ref($self->provisioner)); |
| } |
| |
| |
| # Update the computer state to reloading |
| if (update_computer_state($computer_id, "reloading")) { |
| notify($ERRORS{'OK'}, 0, "computer $computer_short_name state set to reloading"); |
| insertloadlog($reservation_id, $computer_id, "info", "computer state updated to reloading"); |
| } |
| else { |
| notify($ERRORS{'CRITICAL'}, 0, "unable to set $computer_short_name into reloading state, returning"); |
| insertloadlog($reservation_id, $computer_id, "failed", "unable to set computer $computer_short_name state to reloading"); |
| return; |
| } |
| |
| |
| # Call provisioning module's load() subroutine |
| notify($ERRORS{'OK'}, 0, "calling " . ref($self->provisioner) . "->load() subroutine"); |
| insertloadlog($reservation_id, $computer_id, "info", "calling " . ref($self->provisioner) . "->load() subroutine"); |
| if ($self->provisioner->load($node_status)) { |
| notify($ERRORS{'OK'}, 0, "$image_name was successfully reloaded on $computer_short_name"); |
| insertloadlog($reservation_id, $computer_id, "loadimagecomplete", "$image_name was successfully reloaded on $computer_short_name"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "$image_name failed to load on $computer_short_name, returning"); |
| insertloadlog($reservation_id, $computer_id, "loadimagefailed", "$image_name failed to load on $computer_short_name"); |
| return; |
| } |
| |
| #Post operations not to be handled by provisioning modules |
| if($image_os_install_type eq "kickstart"){ |
| notify($ERRORS{'OK'}, 0, "detected kickstart install on $computer_short_name, writing current_image.txt"); |
| if(write_currentimage_txt($self->data)){ |
| notify($ERRORS{'OK'}, 0, "Successfully wrote current_image.txt on $computer_short_name"); |
| } |
| } |
| |
| notify($ERRORS{'OK'}, 0, "node ready: successfully reloaded $computer_short_name with $image_name"); |
| insertloadlog($reservation_id, $computer_id, "nodeready", "$computer_short_name was reloaded with $image_name"); |
| } |
| |
| # Update the current image ID in the computer table |
| if (update_currentimage($computer_id, $image_id, $imagerevision_id, $image_id)) { |
| notify($ERRORS{'OK'}, 0, "updated computer table for $computer_short_name: currentimageid=$image_id"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to update computer table for $computer_short_name: currentimageid=$image_id"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "returning 1"); |
| return 1; |
| } ## end sub reload_image |
| |
| #///////////////////////////////////////////////////////////////////////////// |
| |
| =head2 computer_not_being_used |
| |
| Parameters : |
| Returns : |
| Description : |
| |
| =cut |
| |
| sub computer_not_being_used { |
| my $self = shift; |
| |
| my $request_data = $self->data->get_request_data(); |
| my $request_id = $self->data->get_request_id(); |
| my $request_logid = $self->data->get_request_log_id(); |
| my $request_state_name = $self->data->get_request_state_name(); |
| my $request_laststate_name = $self->data->get_request_laststate_name(); |
| my $request_forimaging = $self->data->get_request_forimaging(); |
| my $request_preload_only = $self->data->get_request_preload_only(); |
| my $reservation_count = $self->data->get_reservation_count(); |
| my $reservation_id = $self->data->get_reservation_id(); |
| my $reservation_is_parent = $self->data->is_parent_reservation; |
| my $computer_id = $self->data->get_computer_id(); |
| my $computer_host_name = $self->data->get_computer_host_name(); |
| my $computer_short_name = $self->data->get_computer_short_name(); |
| my $computer_type = $self->data->get_computer_type(); |
| my $computer_ip_address = $self->data->get_computer_ip_address(); |
| my $computer_state_name = $self->data->get_computer_state_name(); |
| my $computer_next_image_id = $self->data->get_computer_nextimage_id(); |
| my $computer_next_image_name = $self->data->get_computer_nextimage_name(); |
| my $image_id = $self->data->get_image_id(); |
| my $image_os_name = $self->data->get_image_os_name(); |
| my $image_name = $self->data->get_image_name(); |
| my $image_prettyname = $self->data->get_image_prettyname(); |
| my $image_project = $self->data->get_image_project(); |
| my $image_reloadtime = $self->data->get_image_reload_time(); |
| my $image_architecture = $self->data->get_image_architecture(); |
| my $image_os_type = $self->data->get_image_os_type(); |
| my $imagemeta_checkuser = $self->data->get_imagemeta_checkuser(); |
| my $imagemeta_usergroupid = $self->data->get_imagemeta_usergroupid(); |
| my $imagemeta_usergroupmembercount = $self->data->get_imagemeta_usergroupmembercount(); |
| my $imagemeta_usergroupmembers = $self->data->get_imagemeta_usergroupmembers(); |
| my $imagerevision_id = $self->data->get_computer_imagerevision_id(); |
| my $managementnode_id = $self->data->get_management_node_id(); |
| my $managementnode_hostname = $self->data->get_management_node_hostname(); |
| my $user_unityid = $self->data->get_user_login_id(); |
| my $user_uid = $self->data->get_user_uid(); |
| my $user_affiliation_sitewwwaddress = $self->data->get_user_affiliation_sitewwwaddress(); |
| my $user_affiliation_helpaddress = $self->data->get_user_affiliation_helpaddress(); |
| my $user_standalone = $self->data->get_user_standalone(); |
| my $user_email = $self->data->get_user_email(); |
| my $user_emailnotices = $self->data->get_user_emailnotices(); |
| my $user_imtype_name = $self->data->get_user_imtype_name(); |
| my $user_im_id = $self->data->get_user_im_id(); |
| |
| # Possible computer states: |
| # available |
| # deleted |
| # failed |
| # inuse |
| # maintenance |
| # reloading |
| # reserved |
| # vmhostinuse |
| |
| notify($ERRORS{'DEBUG'}, 0, "$computer_short_name state is $computer_state_name"); |
| |
| # Return 0 if computer state is maintenance or deleted |
| if ($computer_state_name =~ /^(deleted|maintenance)$/) { |
| notify($ERRORS{'WARNING'}, 0, "$computer_short_name is NOT available, its state is $computer_state_name"); |
| return 0; |
| } |
| |
| # Check if computer state is available |
| if ($computer_state_name =~ /^(available|reload)$/) { |
| notify($ERRORS{'OK'}, 0, "$computer_short_name is available, its state is $computer_state_name"); |
| return 1; |
| } |
| # Warn if computer state is failed, proceed to check for neighbor reservations |
| else { |
| notify($ERRORS{'WARNING'}, 0, "$computer_short_name state is $computer_state_name, checking if any conflicting requests are active"); |
| } |
| |
| # Set variables to control how may attempts are made to wait for an existing inuse reservation to end |
| my $inuse_loop_attempts = 4; |
| my $inuse_loop_wait = 30; |
| |
| INUSE_LOOP: for (my $inuse_loop_count = 0; $inuse_loop_count < $inuse_loop_attempts; $inuse_loop_count++) { |
| |
| # Check if this isn't the first iteration meaning something conflicting was found |
| if ($inuse_loop_count > 0) { |
| notify($ERRORS{'OK'}, 0, "attempt $inuse_loop_count/$inuse_loop_attempts: waiting for $inuse_loop_wait seconds before checking neighbor requests again"); |
| sleep $inuse_loop_wait; |
| } |
| |
| # Check if there is another request using this machine |
| # Get a hash containing all of the reservations for the computer |
| notify($ERRORS{'OK'}, 0, "checking neighbor reservations for $computer_short_name"); |
| my %neighbor_requests = get_request_by_computerid($computer_id); |
| |
| # There should be at least 1 request -- the one being processed |
| if (!%neighbor_requests) { |
| notify($ERRORS{'WARNING'}, 0, "failed to retrieve any requests for computer id=$computer_id, there should be at least 1"); |
| return; |
| } |
| |
| notify($ERRORS{'OK'}, 0, "found " . scalar keys(%neighbor_requests) . " total reservations for $computer_short_name"); |
| |
| # Loop through the neighbor requests |
| NEIGHBOR_REQUESTS: foreach my $neighbor_request_key (keys %neighbor_requests) { |
| my $neighbor_request_id = $neighbor_requests{$neighbor_request_key}{requestid}; |
| my $neighbor_reservation_id = $neighbor_requests{$neighbor_request_key}{reservationid}; |
| my $neighbor_state_name = $neighbor_requests{$neighbor_request_key}{currentstate}; |
| my $neighbor_laststate_name = $neighbor_requests{$neighbor_request_key}{laststate}; |
| my $neighbor_request_start = $neighbor_requests{$neighbor_request_key}{requeststart}; |
| |
| my $neighbor_request_start_epoch = convert_to_epoch_seconds($neighbor_request_start); |
| my $now_epoch = time(); |
| my $neighbor_start_diff = $neighbor_request_start_epoch - $now_epoch; |
| |
| # Ignore the request currently being processed and any complete requests |
| if ($neighbor_reservation_id == $reservation_id) { |
| next NEIGHBOR_REQUESTS; |
| } |
| |
| notify($ERRORS{'DEBUG'}, 0, "checking neighbor request=$neighbor_request_id, reservation=$neighbor_reservation_id, state=$neighbor_state_name, laststate=$neighbor_laststate_name"); |
| notify($ERRORS{'DEBUG'}, 0, "neighbor start time: $neighbor_request_start ($neighbor_start_diff)"); |
| |
| # Ignore any complete requests |
| if ($neighbor_state_name eq "complete") { |
| notify($ERRORS{'OK'}, 0, "neighbor request is complete: id=$neighbor_request_id, state=$neighbor_state_name"); |
| next NEIGHBOR_REQUESTS; |
| } |
| |
| # Check for overlapping reservations which user is involved or image is being created |
| # Don't check for state = new, it could be a future reservation |
| if ($neighbor_state_name =~ /^(maintenance|reserved|inuse|image)$/) { |
| notify($ERRORS{'WARNING'}, 0, "detected overlapping reservation on $computer_short_name: req=$neighbor_request_id, res=$neighbor_reservation_id, request state=$neighbor_state_name, laststate=$neighbor_laststate_name, computer state=$computer_state_name"); |
| return 0; |
| } |
| |
| # Check for other currently pending requests |
| elsif ($neighbor_state_name eq "pending") { |
| |
| # Make sure neighbor request process is actually running |
| my $neighbor_process_count = checkonprocess($neighbor_laststate_name, $neighbor_request_id); |
| if ($neighbor_process_count) { |
| notify($ERRORS{'OK'}, 0, "detected neighbor request $neighbor_request_id is active"); |
| } |
| elsif ($neighbor_process_count == 0) { |
| notify($ERRORS{'OK'}, 0, "detected neighbor request $neighbor_request_id is NOT active, setting its state to 'complete'"); |
| # Process was not found, set neighbor request to complete |
| if (update_request_state($neighbor_request_id, "complete", $neighbor_laststate_name)) { |
| notify($ERRORS{'OK'}, 0, "neighbor request $neighbor_request_id state set to 'complete'"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set neighbor request $neighbor_request_id state to 'complete'"); |
| } |
| # Check other neighbor requests |
| next NEIGHBOR_REQUESTS; |
| } ## end elsif ($neighbor_process_count == 0) [ if ($neighbor_process_count) |
| else { |
| # Undefined was returned from checkonprocess(), meaning error occurred |
| notify($ERRORS{'CRITICAL'}, 0, "error occurred while checking if neighbor request $neighbor_request_id process is running"); |
| |
| # Wait then try again |
| next INUSE_LOOP; |
| } |
| |
| # Check for state = pending and laststate = new, reserved, inuse, or image |
| # Just return 0 for these, don't bother waiting |
| if ($neighbor_laststate_name =~ /^(new|reserved|inuse|image)$/) { |
| notify($ERRORS{'WARNING'}, 0, "detected overlapping reservation on $computer_short_name: req=$neighbor_request_id, res=$neighbor_reservation_id, request state=$neighbor_state_name, laststate=$neighbor_laststate_name, computer state=$computer_state_name"); |
| return 0; |
| } |
| |
| # Neighbor request state is pending and process is actively running |
| # Neighbor request state should be deleted|timeout|reload|reclaim |
| if ($neighbor_laststate_name !~ /^(deleted|timeout|reload|reclaim)$/) { |
| notify($ERRORS{'WARNING'}, 0, "unexpected neighbor request laststate: $neighbor_laststate_name"); |
| } |
| |
| # Computer should be loading |
| if (monitorloading($neighbor_reservation_id, $image_name, $computer_id, $computer_short_name, $image_reloadtime)) { |
| # Returns 1 if specified image has been successfully loaded |
| # Returns 0 if another image is being loaded or if loading fails |
| notify($ERRORS{'OK'}, 0, "$image_name should have been loaded on $computer_short_name by reservation $neighbor_reservation_id"); |
| |
| # Check other neighbor requests |
| next NEIGHBOR_REQUESTS; |
| } |
| |
| # Computer is not being loaded with the correct image or loading failed |
| # Take evasive action - recheck on neighbor process |
| if (checkonprocess($neighbor_laststate_name, $neighbor_request_id)) { |
| notify($ERRORS{'OK'}, 0, "neighbor request=$neighbor_request_id, reservation=$neighbor_reservation_id owning $computer_short_name is not loading correct image or taking too long, attempting to kill process for reservation $neighbor_reservation_id"); |
| |
| # Kill competing neighbor process - set it's state to complete |
| if (kill_reservation_process($neighbor_reservation_id)) { |
| notify($ERRORS{'OK'}, 0, "killed competing process for reservation $neighbor_reservation_id"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to kill competing process for reservation $neighbor_reservation_id"); |
| } |
| } ## end if (checkonprocess($neighbor_laststate_name... |
| |
| # Either neighbor process was not found or competing process was just killed |
| # Set neighbor request to complete |
| if (update_request_state($neighbor_request_id, "deleted", $neighbor_laststate_name)) { |
| notify($ERRORS{'OK'}, 0, "neighbor request $neighbor_request_id state set to 'deleted'"); |
| # Check other neighbor requests |
| next NEIGHBOR_REQUESTS; |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "failed to set neighbor request $neighbor_request_id state to 'deleted'"); |
| } |
| } ## end elsif ($neighbor_state_name eq "pending") [ if ($neighbor_state_name =~ /^(reserved|inuse|image)$/) |
| |
| # Check for other requests |
| else { |
| notify($ERRORS{'OK'}, 0, "neighbor request state is OK: $neighbor_state_name/$neighbor_laststate_name"); |
| } |
| |
| } ## end foreach my $neighbor_request_key (keys %neighbor_requests) |
| |
| # Checked all neighbor requests and didn't find any conflicting reservations |
| notify($ERRORS{'OK'}, 0, "checked neighbor requests and didn't find any conflicting reservations for $computer_short_name"); |
| return 1; |
| |
| } ## end for (my $inuse_loop_count = 0; $inuse_loop_count... |
| |
| # Checked all neighbor requests several times and find something conflicting every time |
| notify($ERRORS{'WARNING'}, 0, "$computer_short_name does not appear to be available"); |
| return 0; |
| |
| } ## end sub computer_not_being_used |
| |
| #///////////////////////////////////////////////////////////////////////////// |
| |
| =head2 reserve_computer |
| |
| Parameters : |
| Returns : |
| Description : |
| |
| =cut |
| |
| sub reserve_computer { |
| my $self = shift; |
| |
| my $request_data = $self->data->get_request_data(); |
| my $request_id = $self->data->get_request_id(); |
| my $request_logid = $self->data->get_request_log_id(); |
| my $request_state_name = $self->data->get_request_state_name(); |
| my $request_laststate_name = $self->data->get_request_laststate_name(); |
| my $request_forimaging = $self->data->get_request_forimaging(); |
| my $request_preload_only = $self->data->get_request_preload_only(); |
| my $reservation_count = $self->data->get_reservation_count(); |
| my $reservation_id = $self->data->get_reservation_id(); |
| my $reservation_is_parent = $self->data->is_parent_reservation; |
| my $computer_id = $self->data->get_computer_id(); |
| my $computer_host_name = $self->data->get_computer_host_name(); |
| my $computer_short_name = $self->data->get_computer_short_name(); |
| my $computer_type = $self->data->get_computer_type(); |
| my $computer_ip_address = $self->data->get_computer_ip_address(); |
| my $computer_state_name = $self->data->get_computer_state_name(); |
| my $computer_next_image_id = $self->data->get_computer_nextimage_id(); |
| my $computer_next_image_name = $self->data->get_computer_nextimage_name(); |
| my $image_id = $self->data->get_image_id(); |
| my $image_os_name = $self->data->get_image_os_name(); |
| my $image_name = $self->data->get_image_name(); |
| my $image_prettyname = $self->data->get_image_prettyname(); |
| my $image_project = $self->data->get_image_project(); |
| my $image_reloadtime = $self->data->get_image_reload_time(); |
| my $image_architecture = $self->data->get_image_architecture(); |
| my $image_os_type = $self->data->get_image_os_type(); |
| my $imagemeta_checkuser = $self->data->get_imagemeta_checkuser(); |
| my $imagemeta_usergroupid = $self->data->get_imagemeta_usergroupid(); |
| my $imagemeta_usergroupmembercount = $self->data->get_imagemeta_usergroupmembercount(); |
| my $imagemeta_usergroupmembers = $self->data->get_imagemeta_usergroupmembers(); |
| my $imagerevision_id = $self->data->get_computer_imagerevision_id(); |
| my $managementnode_id = $self->data->get_management_node_id(); |
| my $managementnode_hostname = $self->data->get_management_node_hostname(); |
| my $user_unityid = $self->data->get_user_login_id(); |
| my $user_uid = $self->data->get_user_uid(); |
| my $user_affiliation_sitewwwaddress = $self->data->get_user_affiliation_sitewwwaddress(); |
| my $user_affiliation_helpaddress = $self->data->get_user_affiliation_helpaddress(); |
| my $user_standalone = $self->data->get_user_standalone(); |
| my $user_email = $self->data->get_user_email(); |
| my $user_emailnotices = $self->data->get_user_emailnotices(); |
| my $user_imtype_name = $self->data->get_user_imtype_name(); |
| my $user_im_id = $self->data->get_user_im_id(); |
| my $ip_configuration = $self->data->get_management_node_public_ip_configuration(); |
| |
| notify($ERRORS{'OK'}, 0, "user_standalone=$user_standalone, image OS type=$image_os_type"); |
| |
| my ($mailstring, $subject, $r); |
| |
| # check for deletion |
| if (is_request_deleted($request_id)) { |
| notify($ERRORS{'OK'}, 0, "user has deleted, quietly exiting"); |
| #return 0 and let process routine handle reset computer state |
| return 0; |
| } |
| |
| if ($computer_type =~ /blade|virtualmachine/) { |
| |
| if (!$self->os->update_public_ip_address()) { |
| $self->reservation_failed("failed to update private IP address"); |
| } |
| |
| insertloadlog($reservation_id, $computer_id, "info", "node ready adding user account"); |
| |
| # Create a random password and update the reservation table unless the reservation if for a Linux non-standalone image |
| unless ($image_os_type =~ /linux/ && !$user_standalone) { |
| # Create a random password for the reservation |
| my $reservation_password = getpw(); |
| |
| # Update the password in the reservation table |
| if (update_request_password($reservation_id, $reservation_password)) { |
| notify($ERRORS{'DEBUG'}, 0, "updated password in the reservation table"); |
| } |
| else { |
| $self->reservation_failed("failed to update password in the reservation table"); |
| } |
| |
| # Set the password in the DataStructure object |
| $self->data->set_reservation_password($reservation_password); |
| } |
| |
| # Check if OS module implements a reserve() subroutine |
| if ($self->os->can('reserve')) { |
| # Call the OS module's reserve() subroutine |
| notify($ERRORS{'DEBUG'}, 0, "calling OS module's reserve() subroutine"); |
| if ($self->os->reserve()) { |
| notify($ERRORS{'DEBUG'}, 0, "OS module successfully reserved resources for this reservation"); |
| } |
| else { |
| $self->reservation_failed("OS module failed to reserve resources for this reservation"); |
| } |
| } |
| |
| # Check if this is a parent reservation, only the parent reservation handles notifications |
| if (!$reservation_is_parent) { |
| return 1; |
| } |
| |
| # Assemble the message subject based on whether this is a cluster based or normal request |
| if ($request_forimaging) { |
| $subject = "VCL -- $image_prettyname imaging reservation"; |
| } |
| elsif ($reservation_count > 1) { |
| $subject = "VCL -- Cluster-based reservation"; |
| } |
| else { |
| $subject = "VCL -- $image_prettyname reservation"; |
| } |
| |
| # Assemble the message body |
| if ($request_forimaging) { |
| $mailstring = <<"EOF"; |
| |
| The resources for your VCL image creation request have been successfully reserved. |
| |
| EOF |
| } |
| else { |
| $mailstring = <<"EOF"; |
| |
| The resources for your VCL request have been successfully reserved. |
| |
| EOF |
| } |
| |
| # Add the image name and IP address information |
| $mailstring .= "Reservation Information:\n"; |
| foreach $r (keys %{$request_data->{reservation}}) { |
| my $reservation_image_name = $request_data->{reservation}{$r}{image}{prettyname}; |
| my $computer_ip_address = $request_data->{reservation}{$r}{computer}{IPaddress}; |
| $mailstring .= "Image Name: $reservation_image_name\n"; |
| $mailstring .= "IP Address: $computer_ip_address\n\n"; |
| } |
| |
| $mailstring .= <<"EOF"; |
| Connection will not be allowed until you acknowledge using the VCL web interface. You must acknowledge the reservation within the next 15 minutes or the resources will be reclaimed for other VCL users. |
| |
| -Visit $user_affiliation_sitewwwaddress |
| -Select "Current Reservations" |
| -Click the "Connect" button |
| |
| Upon acknowledgement, all of the remaining connection details will be displayed. |
| |
| EOF |
| |
| if ($request_forimaging) { |
| $mailstring .= <<"EOF"; |
| You have up to 8 hours to complete the new image. Once you have completed preparing the new image: |
| |
| -Visit $user_affiliation_sitewwwaddress |
| -Select "Current Reservations" |
| -Click the "Create Image" button and follow the instuctions |
| |
| EOF |
| } ## end if ($request_forimaging) |
| |
| $mailstring .= <<"EOF"; |
| Thank You, |
| VCL Team |
| |
| |
| ****************************************************************** |
| This is an automated notice. If you need assistance please respond |
| with detailed information on the issue and a help ticket will be |
| generated. |
| |
| To disable email notices |
| -Visit $user_affiliation_sitewwwaddress |
| -Select User Preferences |
| -Select General Preferences |
| |
| ****************************************************************** |
| EOF |
| if ($user_emailnotices) { |
| mail($user_email, $subject, $mailstring, $user_affiliation_helpaddress); |
| } |
| else { |
| #just for our email record keeping, might be overkill |
| notify($ERRORS{'MAILMASTERS'}, 0, " $user_email\n$mailstring"); |
| } |
| |
| notify($ERRORS{'DEBUG'}, 0, "IMTYPE_name= $user_imtype_name calling notify_via"); |
| if ($user_imtype_name ne "none") { |
| notify_via_IM($user_imtype_name, $user_im_id, $mailstring, $user_affiliation_helpaddress); |
| } |
| |
| |
| |
| } ## end if ($computer_type =~ /blade|virtualmachine/) |
| |
| elsif ($computer_type eq "lab") { |
| if ($image_os_name =~ /sun4x_|rhel/) { |
| # i can't really do anything here |
| # because I need the remoteIP the user |
| # will be accessing the machine from |
| $subject = "VCL -- $image_prettyname reservation"; |
| |
| $mailstring = <<"EOF"; |
| |
| A machine with $image_prettyname has been reserved. Use ssh to connect to $computer_ip_address. |
| |
| Username: your Unity ID |
| Password: your Unity password |
| |
| Connection will not be allowed until you acknowledge using the VCL web interface. |
| -Visit $user_affiliation_sitewwwaddress |
| -Select Current Reservations |
| -Click the Connect button to acknowledge |
| |
| |
| Thank You, |
| VCL Team |
| |
| |
| |
| ****************************************************************** |
| This is an automated notice. If you need assistance please respond |
| with detailed information on the issue and a help ticket will be |
| generated. |
| |
| To disable email notices |
| -Visit $user_affiliation_sitewwwaddress |
| -Select User Preferences |
| -Select General Preferences |
| |
| ****************************************************************** |
| EOF |
| |
| if ($user_emailnotices) { |
| #if "0" user does not care to get additional notices |
| mail($user_email, $subject, $mailstring, $user_affiliation_helpaddress); |
| } |
| else { |
| #just for our record keeping |
| notify($ERRORS{'MAILMASTERS'}, 0, "$user_email\n$mailstring"); |
| } |
| if ($user_imtype_name ne "none") { |
| notify_via_IM($user_imtype_name, $user_im_id, $mailstring, $user_affiliation_helpaddress); |
| } |
| } ## end if ($image_os_name =~ /sun4x_|rhel/) |
| elsif ($image_os_name =~ /realm/) { |
| #same as above |
| return 1; |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "hrmm found an OS I am not set to handle $image_os_name"); |
| return 0; |
| } |
| } ## end elsif ($computer_type eq "lab") [ if ($computer_type =~ /blade|virtualmachine/) |
| |
| #update log table with the IPaddress of the machine |
| if (update_sublog_ipaddress($request_logid, $computer_ip_address)) { |
| notify($ERRORS{'OK'}, 0, "updated sublog $request_logid for node $computer_short_name IPaddress $computer_ip_address"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "could not update sublog $request_logid for node $computer_short_name IPaddress $computer_ip_address"); |
| } |
| |
| return 1; |
| } ## end sub reserve_computer |
| |
| #///////////////////////////////////////////////////////////////////////////// |
| |
| =head2 wait_for_child_reservations |
| |
| Parameters : |
| Returns : |
| Description : |
| |
| =cut |
| |
| sub wait_for_child_reservations { |
| my $self = shift; |
| my $request_data = $self->data->get_request_data(); |
| my $request_id = $self->data->get_request_id(); |
| my $reservation_count = $self->data->get_reservation_count(); |
| my $reservation_id = $self->data->get_reservation_id(); |
| my @reservation_ids = $self->data->get_reservation_ids(); |
| |
| # Set limits on how many attempts to make and how long to wait between attempts |
| # Wait a long time - 20 minutes |
| my $loop_iteration_limit = 40; |
| my $loop_iteration_delay = 30; |
| |
| WAITING_LOOP: for (my $loop_iteration = 1; $loop_iteration <= $loop_iteration_limit; $loop_iteration++) { |
| if ($loop_iteration > 1) { |
| notify($ERRORS{'OK'}, 0, "waiting for $loop_iteration_delay seconds"); |
| sleep $loop_iteration_delay; |
| } |
| |
| # Check if request has been deleted |
| if (is_request_deleted($request_id)) { |
| notify($ERRORS{'OK'}, 0, "request has been deleted, setting computer state to 'available' and exiting"); |
| |
| # Update state of computer and exit |
| switch_state($request_data, '', 'available', '', '1'); |
| } |
| |
| # Check if all of the reservations are ready according to the computerloadlog table |
| my $computerloadlog_reservations_ready = reservations_ready($request_id); |
| if ($computerloadlog_reservations_ready) { |
| notify($ERRORS{'OK'}, 0, "ready: all child reservations are ready according to computerloadlog, returning 1"); |
| return 1; |
| } |
| elsif (defined $computerloadlog_reservations_ready) { |
| notify($ERRORS{'OK'}, 0, "not ready: all child reservations are NOT ready according to computerloadlog"); |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "error occurred checking if child reservations are ready according to computerloadlog"); |
| } |
| |
| notify($ERRORS{'OK'}, 0, "attempt $loop_iteration/$loop_iteration_limit: waiting for child reservations to become ready"); |
| |
| RESERVATION_LOOP: foreach my $child_reservation_id (@reservation_ids) { |
| # Don't bother checking this reservation |
| if ($child_reservation_id == $reservation_id) { |
| next RESERVATION_LOOP; |
| } |
| |
| # Get the computer ID of the child reservation |
| my $child_computer_id = $request_data->{reservation}{$child_reservation_id}{computer}{id}; |
| notify($ERRORS{'DEBUG'}, 0, "checking reservation $child_reservation_id: computer ID=$child_computer_id"); |
| |
| # Get the child reservation's current computer state |
| my $child_computer_state = get_computer_current_state_name($child_computer_id); |
| notify($ERRORS{'DEBUG'}, 0, "reservation $child_reservation_id: computer state=$child_computer_state"); |
| |
| # Check child reservation's computer state, is it reserved? |
| if ($child_computer_state eq "reserved") { |
| notify($ERRORS{'OK'}, 0, "ready: reservation $child_reservation_id computer state is reserved"); |
| } |
| elsif ($child_computer_state eq "reloading") { |
| notify($ERRORS{'OK'}, 0, "not ready: reservation $child_reservation_id is still reloading"); |
| next WAITING_LOOP; |
| } |
| elsif ($child_computer_state eq "available" && $loop_iteration > 2) { |
| # Child computer may still be in the available state if the request start is recent |
| # Warn if still in available state after this subroutine has iterated a couple times |
| notify($ERRORS{'WARNING'}, 0, "not ready: reservation $child_reservation_id: computer state is still $child_computer_state"); |
| next WAITING_LOOP; |
| } |
| elsif ($child_computer_state eq "available") { |
| notify($ERRORS{'OK'}, 0, "not ready: reservation $child_reservation_id: reloading has not begun yet"); |
| next WAITING_LOOP; |
| } |
| elsif ($child_computer_state =~ /^(failed|maintenance|deleted)$/) { |
| notify($ERRORS{'WARNING'}, 0, "abort: reservation $child_reservation_id: computer was put into maintenance, returning"); |
| return; |
| } |
| else { |
| notify($ERRORS{'WARNING'}, 0, "unexpected: reservation $child_reservation_id: computer in unexpected state: $child_computer_state"); |
| next WAITING_LOOP; |
| } |
| } ## end foreach my $child_reservation_id (@reservation_ids) |
| |
| notify($ERRORS{'OK'}, 0, "all child reservations are ready, returning 1"); |
| return 1; |
| } ## end for (my $loop_iteration = 1; $loop_iteration... |
| |
| # If out of main loop, waited maximum amount of time |
| notify($ERRORS{'WARNING'}, 0, "waited maximum amount of time for child reservations to become ready, returning 0"); |
| return 0; |
| |
| } ## end sub wait_for_child_reservations |
| |
| #///////////////////////////////////////////////////////////////////////////// |
| |
| 1; |
| __END__ |
| |
| =head1 SEE ALSO |
| |
| L<http://cwiki.apache.org/VCL/> |
| |
| =cut |