#!/bin/ksh93 # ALTRAN_PROLOG_BEGIN_TAG # This is an automatically generated prolog. # # Copyright (C) Altran ACT S.A.S. 2019,2021. All rights reserved. # # ALTRAN_PROLOG_END_TAG # # IBM_PROLOG_BEGIN_TAG # This is an automatically generated prolog. # # 61haes_r714 src/43haes/lib/ksh93/hacmp/KLIB_HACMP_modify_dependency.sh 1.3 # # Licensed Materials - Property of IBM # # COPYRIGHT International Business Machines Corp. 1990,2010 # All Rights Reserved # # US Government Users Restricted Rights - Use, duplication or # disclosure restricted by GSA ADP Schedule Contract with IBM Corp. # # IBM_PROLOG_END_TAG # @(#) 7d4c34b 43haes/lib/ksh93/hacmp/KLIB_HACMP_modify_dependency.sh, 726, 2147A_aha726, Feb 05 2021 09:50 PM #============================================================================ # # Name: KLIB_HACMP_modify_dependency # # Description: This is the main, FPATH function that is invoked by clmgr # to modify resource group dependencies in the cluster # configuration. # # Inputs: See the "devDoc()" function, below. # # Outputs: The properties hash is populated. The only other outputs are # any error messages that might be needed. # # Returns: Zero if no errors are detected. Otherwise, an appropriate # non-zero value is returned. Refer to the "RETURN" section # of the "devDoc()" function, below, for the standard return # code values/meanings for clmgr. # #============================================================================ function KLIB_HACMP_modify_dependency { . $HALIBROOT/log_entry "$0()" "$CL" : version=@(#) 7d4c34b 43haes/lib/ksh93/hacmp/KLIB_HACMP_modify_dependency.sh, 726, 2147A_aha726, Feb 05 2021 09:50 PM : INPUTS: $* typeset dependency=${1//\"/} # Required typeset explicit_type=${2//\"/} # Optional typeset parent=${3//\"/} typeset children=${4//\"/} typeset groups=${5//\"/} typeset high=${6//\"/} high=${high//,/ } typeset intermediate=${7//\"/} intermediate=${intermediate//,/ } typeset low=${8//\"/} low=${low//,/ } typeset -i i=0 for (( i=0; i<8; i++ )); do (( $# == 0 )) && break shift done typeset stop=${1//\"/} typeset start=${2//\"/} typeset after=${3//\"/} typeset serial=${4//\"/} typeset parallel=${5//\"/} [[ $CLMGR_LOGGING == 'med' ]] && set +x # Only trace param values #================================================================ : Check for a defined cluster. No need to continue without one. #================================================================ CL=$LINENO isClusterDefined if (( $? != RC_SUCCESS )); then log_return_msg "$RC_MISSING_DEPENDENCY" "$0()" "$LINENO" return $? fi #=================================== : Declare and initialize variables #=================================== typeset -i rc=$RC_UNKNOWN typeset implicit_type= typeset -u type= if [[ $dependency == *\(*\)* ]]; then implicit_type=${dependency##*\(} implicit_type=${implicit_type%%\)*} type=$implicit_type dependency=${dependency/\(*\)} fi if [[ -n $explicit_type ]]; then explicit_type=${explicit_type##*\(} explicit_type=${explicit_type%%\)*} type=$explicit_type fi if [[ -n $serial ]]; then serial=${serial//,/ } serial=${serial//+([[:space:]])/ } serial=${serial#+([[:space:]])} serial=${serial%+([[:space:]])} fi if [[ -n $parallel ]]; then parallel=${parallel//,/ } parallel=${parallel//+([[:space:]])/ } parallel=${parallel#+([[:space:]])} parallel=${parallel%+([[:space:]])} fi if [[ -n $parent ]]; then parent=${parent//,/ } parent=${parent//+([[:space:]])/ } parent=${parent#+([[:space:]])} parent=${parent%+([[:space:]])} fi if [[ -n $children ]]; then children=${children//,/ } children=${children//+([[:space:]])/ } children=${children#+([[:space:]])} children=${children%+([[:space:]])} fi if [[ -n $groups ]]; then groups=${groups//,/ } groups=${groups//+([[:space:]])/ } groups=${groups#+([[:space:]])} groups=${groups%+([[:space:]])} fi if [[ -n $high ]]; then high=${high//,/ } high=${high//+([[:space:]])/ } high=${high#+([[:space:]])} high=${high%+([[:space:]])} fi if [[ -n $intermediate ]]; then intermediate=${intermediate//,/ } intermediate=${intermediate//+([[:space:]])/ } intermediate=${intermediate#+([[:space:]])} intermediate=${intermediate%+([[:space:]])} fi if [[ -n $low ]]; then low=${low//,/ } low=${low//+([[:space:]])/ } low=${low#+([[:space:]])} low=${low%+([[:space:]])} fi if [[ -n $stop ]]; then stop=${stop//,/ } stop=${stop//+([[:space:]])/ } stop=${stop#+([[:space:]])} stop=${stop%+([[:space:]])} fi if [[ -n $start ]]; then start=${start//,/ } start=${start//+([[:space:]])/ } start=${start#+([[:space:]])} start=${start%+([[:space:]])} fi if [[ -n $after ]]; then after=${after//,/ } after=${after//+([[:space:]])/ } after=${after#+([[:space:]])} after=${after%+([[:space:]])} fi #================================================================ : Assuming an object was specified, see if it is a known object #================================================================ if [[ $dependency != *([[:space:]]) ]]; then CL=$LINENO KLIB_HACMP_is_known_dependency "$dependency" (( $? != RC_SUCCESS )) && rc=$RC_NOT_FOUND fi #================= : Validate input #================= if [[ -z $dependency && $type != @(A|R)* ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 100 "\nERROR: a name/label must be provided.\n\n" 1>&2 rc=$RC_MISSING_INPUT elif (( $rc == RC_NOT_FOUND )); then dspmsg -s $CLMGR_SET $CLMGR_MSGS 313 '\nERROR: no dependencies involving "%1$s"\n could be found.\n\n' "$dependency" 1>&2 typeset available CL=$LINENO KLIB_HACMP_list_dependencies available for (( i=0; i<${#available[*]}; i++ )); do if [[ ${available[$i]} != *([[:space:]]) ]]; then print -u2 "\t${available[$i]}" fi done print -u2 "" elif [[ -n $implicit_type && -n $explicit_type && \ $implicit_type != $explicit_type ]] then dspmsg -s $CLMGR_SET $CLMGR_MSGS 403 '\nERROR: the specified implicit dependency type, "%1$s", does\n not match the specified explicit dependency type, "%2$s".\n\n' "$implicit_type" "$explicit_type" 1>&2 rc=$RC_INCORRECT_INPUT elif [[ -z $type ]]; then : "DEPTYPE" gets set by KLIB_HACMP_is_known_dependency if [[ $DEPTYPE == *[[:space:]]* ]]; then rc=$RC_INCORRECT_INPUT dspmsg -s $CLMGR_SET $CLMGR_MSGS 34 '\nERROR: "%1$s" is ambiguous, and could match any of the following:\n %2$s\n\n' "$dependency" "$DEPTYPE" 1>&2 typeset available CL=$LINENO KLIB_HACMP_list_dependencies available for (( i=0; i<${#available[*]}; i++ )); do if [[ ${available[$i]} != *([[:space:]]) ]]; then print -u2 "\t${available[$i]}" fi done print -u2 "" else type=$DEPTYPE fi elif [[ -n $serial && -n $parallel ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 36 '\nERROR: conflicting options were provided, "%1$s" versus "%2$s".\n\n' "SERIAL" "PARALLEL" 1>&2 rc=$RC_MISSING_INPUT elif [[ -n $type && -n "$serial$parallel" && $type != @(A|R)* ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 110 '\nERROR: invalid value specified for "%1$s": "%2$s"\n\n' TYPE "$TYPE" 1>&2 dspmsg -s $CLMGR_SET $CLMGR_MSGS 3 'Valid values: %1$s\n\n' "ACQUIRE, RELEASE" 1>&2 rc=$RC_MISSING_INPUT elif [[ -n $parent && $parent == *[[:space:]]* ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 210 '\nERROR: only one resource group may be specified as the parent in a\n "parent/child" dependency: PARENT="%1$s"\n\n' "$parent" 1>&2 rc=$RC_INCORRECT_INPUT elif [[ -n $after && $after == *[[:space:]]* ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 210 '\nERROR: only one resource group may be specified as the primary resource group\n in a "STOP_AFTER" dependency: AFTER="%1$s"\n\n' "$after" 1>&2 rc=$RC_INCORRECT_INPUT elif [[ -n $stop && $stop == *[[:space:]]* ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 210 '\nERROR: only one resource group may be specified as the source in a\n "STOP_AFTER" dependency: STOP="%1$s"\n\n' "$stop" 1>&2 rc=$RC_INCORRECT_INPUT elif [[ -n $start && $start == *[[:space:]]* ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 210 '\nERROR: only one resource group may be specified as the source in a\n "START_AFTER" dependency: START="%1$s"\n\n' "$start" 1>&2 rc=$RC_INCORRECT_INPUT fi if (( $rc == RC_UNKNOWN )); then if [[ $type == "PARENT_CHILD" && -z "$parent$children" ]]; then dspmsg -s $CLMGR_SET $CLMGR_MSGS 401 '\nERROR: no valid modifications were specified for "%1$s".\n\n' "$dependency" 1>&2 rc=$RC_MISSING_INPUT elif [[ $type == SAME_@(SITE|NODE) || \ $type == @(SITE|NODE)COLLOCATION ]] && \ [[ -z $groups ]] then dspmsg -s $CLMGR_SET $CLMGR_MSGS 401 '\nERROR: no valid modifications were specified for "%1$s".\n\n' "$dependency" 1>&2 rc=$RC_MISSING_INPUT elif [[ $type == "DIFFERENT_NODES" || $type == ANTICOLLOCATION ]] && \ [[ -z "$high$intermediate$low" ]] then dspmsg -s $CLMGR_SET $CLMGR_MSGS 401 '\nERROR: no valid modifications were specified for "%1$s".\n\n' "$dependency" 1>&2 rc=$RC_MISSING_INPUT fi fi #============================================================== : Modify the dependency if no input errors have been detected #============================================================== if (( $rc == RC_UNKNOWN )); then case $type in AC*) typeset depList CL=$LINENO KLIB_HACMP_list_dependencies depList for (( i=0; i<${#depList[*]}; i++ )); do [[ ${depList[$i]} != *RELEASE_SERIALLY* ]] && continue break done typeset release=${depList[$i]} release=${release%%+([[:space:]])*} release=${release//,/ } if [[ -n $parallel ]]; then typeset -u uc=$parallel serial= if [[ $uc != "ALL" ]]; then typeset rgList CL=$LINENO KLIB_HACMP_list_resourcegroups rgList for (( i=0; i<${#rgList[*]}; i++ )); do if [[ " ${parallel//,/ } " != *\ ${rgList[$i]}\ * ]] then [[ -n $serial ]] && serial="$serial " serial="$serial${rgList[$i]}" fi done fi elif [[ -n $serial ]]; then typeset -u uc=$serial if [[ $uc == "ALL" ]]; then serial= typeset rgList CL=$LINENO KLIB_HACMP_list_resourcegroups rgList for (( i=0; i<${#rgList[*]}; i++ )); do [[ -n $serial ]] && serial="$serial " serial="$serial${rgList[$i]}" done fi fi print "$0()[$LINENO]($SECONDS): clrgorder -c -a \"${serial//,/ }\" -r \"$release\"" >>$CLMGR_TMPLOG # Always log commands clrgorder -c -a "${serial//,/ }" -r "$release" rc=$? print "$0()[$LINENO]($SECONDS): clrgorder RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; R*) typeset depList CL=$LINENO KLIB_HACMP_list_dependencies depList for (( i=0; i<${#depList[*]}; i++ )); do [[ ${depList[$i]} != *ACQUIRE_SERIALLY* ]] && continue break done typeset acquire=${depList[$i]} acquire=${acquire%%+([[:space:]])*} acquire=${acquire//,/ } if [[ -n $parallel ]]; then typeset -u uc=$parallel serial= if [[ $uc != "ALL" ]]; then typeset rgList CL=$LINENO KLIB_HACMP_list_resourcegroups rgList for (( i=0; i<${#rgList[*]}; i++ )); do if [[ " ${parallel//,/ } " != *\ ${rgList[$i]}\ * ]] then [[ -n $serial ]] && serial="$serial " serial="$serial${rgList[$i]}" fi done fi elif [[ -n $serial ]]; then typeset -u uc=$serial if [[ $uc == "ALL" ]]; then serial= typeset rgList CL=$LINENO KLIB_HACMP_list_resourcegroups rgList for (( i=0; i<${#rgList[*]}; i++ )); do [[ -n $serial ]] && serial="$serial " serial="$serial${rgList[$i]}" done fi fi print "$0()[$LINENO]($SECONDS): clrgorder -c -a \"$acquire\" -r \"${serial//,/ }\"" >>$CLMGR_TMPLOG # Always log commands clrgorder -c -a "$acquire" -r "${serial//,/ }" rc=$? print "$0()[$LINENO]($SECONDS): clrgorder RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; "PARENT_CHILD") current_parent=${dependency%%\+*} current_parent=${current_parent//,/ } current_children=${dependency##*\+} current_children=${current_children//,/ } [[ -z $parent ]] && parent=$current_parent [[ -z $children ]] && children=$current_children #============================================================== # Since the modify command does not support multiple children # (inexplicably, because the add command does), we can't use # it here. Instead, we have to delete the existing dependency, # then do an add operation. Whee! HACMP gymnastics! :-) #============================================================== CL=$LINENO KLIB_HACMP_delete_dependency "$dependency" if (( $? == RC_SUCCESS )); then CL=$LINENO KLIB_HACMP_add_dependency "$parent" "$children" rc=$? fi ;; "STOP_AFTER") current_after=${dependency%%\+*} current_after=${current_after//,/ } current_stop=${dependency##*\+} current_stop=${current_stop//,/ } [[ -z $after ]] && after=$current_after [[ -z $stop ]] && stop=$current_stop # IMPORTANT NOTE: the clrgdependency options "-op" and "-oc" # require *no* space between them and their # argument values! So don't "correct" that, # below, or you'll break this function!!! print "$0()[$LINENO]($SECONDS): clrgdependency -t \"$type\" -u -op\"$current_stop\" -p \"$stop\" -oc\"$current_after\" -c \"$after\"" >>$CLMGR_TMPLOG # Always log commands clrgdependency -t "$type" -u \ -op"$current_stop" \ -p "$stop" \ -oc"$current_after" \ -c "$after" rc=$? print "$0()[$LINENO]($SECONDS): clrgdependency RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; "START_AFTER") current_after=${dependency%%\+*} current_after=${current_after//,/ } current_start=${dependency##*\+} current_start=${current_start//,/ } [[ -z $after ]] && after=$current_after [[ -z $start ]] && start=$current_start # IMPORTANT NOTE: the clrgdependency options "-op" and "-oc" # require *no* space between them and their # argument values! So don't "correct" that, # below, or you'll break this function!!! print "$0()[$LINENO]($SECONDS): clrgdependency -t \"$type\" -u -oc\"$current_start\" -c \"$start\" -op\"$current_after\" -p \"$after\"" >>$CLMGR_TMPLOG # Always log commands clrgdependency -t "$type" -u \ -oc"$current_start" \ -c "$start" \ -op"$current_after" \ -p "$after" rc=$? print "$0()[$LINENO]($SECONDS): clrgdependency RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; @(SAME_NODE|NODECOLLOCATION)) # IMPORTANT NOTE: the clrgdependency option "-ol" requires *no* # space between it and its argument value! So # don't "correct" that, below! print "$0()[$LINENO]($SECONDS): clrgdependency -t \"NODECOLLOCATION\" -u -ol\"${dependency//,/ }\" -l\"$groups\"" >>$CLMGR_TMPLOG # Always log commands clrgdependency -t "NODECOLLOCATION" -u \ -ol"${dependency//,/ }" \ -l"$groups" rc=$? print "$0()[$LINENO]($SECONDS): clrgdependency RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; @(SAME_SITE|SITECOLLOCATION)) # IMPORTANT NOTE: the clrgdependency option "-ol" requires *no* # space between it and its argument value! So # don't "correct" that, below! print "$0()[$LINENO]($SECONDS): clrgdependency -t \"SITECOLLOCATION\" -u -ol\"${dependency//,/ }\" -l\"$groups\"" >>$CLMGR_TMPLOG # Always log commands clrgdependency -t "SITECOLLOCATION" -u \ -ol"${dependency//,/ }" \ -l"$groups" rc=$? print "$0()[$LINENO]($SECONDS): clrgdependency RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; @(DIFFERENT_NODES|ANTICOLLOCATION)) # IMPORTANT NOTE: the clrgdependency options "-hp", "-ip" and "-lc" # require *no* space between them and their # argument values! So don't "correct" that, below! print "$0()[$LINENO]($SECONDS): clrgdependency -t \"ANTICOLLOCATION\" -u -hp\"$high\" -ip\"$intermediate\" -lp\"$low\"" >>$CLMGR_TMPLOG # Always log commands clrgdependency -t "ANTICOLLOCATION" -u \ -hp"$high" \ -ip"$intermediate" \ -lp"$low" rc=$? print "$0()[$LINENO]($SECONDS): clrgdependency RC: $rc" >>$CLMGR_TMPLOG # Always log command result ;; esac (( $rc != RC_SUCCESS )) && rc=$RC_ERROR fi #======================================================================= : If a user input error was detected, provide some helpful suggestions #======================================================================= if (( $rc == RC_MISSING_INPUT || $rc == RC_INCORRECT_INPUT )) && \ [[ $CLMGR_GUI == *([[:space:]]) ]] then cl_dspmsg -s $CLMGR_SET $CLMGR_MSGS 104 "For more information about available options and syntax, try\n\"$HAUTILS/clmgr %1\$s\". As an\nalternative, if the PowerHA SystemMirror man pages have been installed, invoke\n\"$HAUTILS/clmgr -hv\" (or \"/usr/bin/man clmgr\"),\nsearching for \"%2\$s\" in the displayed text.\n\n" \ "modify dependency -h" "DEPENDENCY:" "$CLMGR_PROGNAME" 1>&2 fi log_return_msg "$rc" "$0()" "$LINENO" return $? } # End of "KLIB_HACMP_modify_dependency()" #============================================================================ # # Name: devDoc # # Description: This is a never-to-be-called, wrapper function that all the # clmgr FPATH functions implement in order to hide embedded # syntax from trace logging. This information is implemented # in POD format, and can be viewed in a number of ways using # POD tools. Some viewing suggestions for this function's POD- # formatted information are: # # perldoc # pod2text -c # pod2text -c --code # pod2html # # However, the more important use for this information is that # it is parsed by clmgr to display the syntax for this file's # operation. The information in the "SYNOPSIS" section is used # for this purpose. This feature was originally implemented # using the man page information. However, in a code review it # was pointed out that this approach had to be changed because # customers do not have to install the man pages! Therefore, a # built-in dependency on man page information would break the # automatic help feature of clmgr. So the SYNPOSIS section must # be used instead. # # IMPORTANT: As a result of this, it is imperative that the # information in this SYNOPSIS be kept in sync # with the man page information, which is owned # by the IDD team. # # Inputs: None. # # Outputs: None. # # Returns: n/a (not intended to be invoked) # #============================================================================ function devDoc { : <<'=cut' >/dev/null 2>&1 =head1 NAME KLIB_HACMP_modify_dependency =head1 SYNOPSIS # Temporal Dependency (parent ==> child) clmgr modify dependency \ [ TYPE=PARENT_CHILD ] \ [ PARENT= ] \ [ CHILD="[,,...]" ] # Temporal Dependency (start/stop after) clmgr modify dependency \ [ TYPE={STOP_AFTER|START_AFTER} ] \ [ {STOP|START}= ] \ [ AFTER= ] # Location Dependency (colocation) clmgr modify dependency \ [ TYPE=SAME_{NODE|SITE} ] \ GROUPS=",[,...]" # Location Dependency (anti-colocation) clmgr modify dependency \ [ TYPE=DIFFERENT_NODES ] \ [ HIGH=",,..." ] \ [ INTERMEDIATE=",,..." ] \ [ LOW=",,..." ] # Acquisition/Release Order clmgr modify dependency \ TYPE={ACQUIRE|RELEASE} \ { SERIAL="{,,...|ALL}" | PARALLEL="{,,...|ALL}" } NOTE: the alias for "dependency" is "de". =head1 DESCRIPTION Attempts to modify the specified resource group dependency to conform to the provided specifications. =head1 ARGUMENTS 1. dependency [REQUIRED] [string] The clmgr string/label of the dependency that is to be modified. 2. explicit_type [OPTIONAL] [string] The type of dependency that is being deleted. This is strictly a performance optimization, to eliminate having to divine the type from the dependency string itself. From the following set: PARENT_CHILD SAME_NODE or NODECOLLOCATION SAME_SITE or SITECOLLOCATION, DIFFERENT_NODES or ANTICOLLOCATION 3. parent [OPTIONAL] {string] The label of a resource group that will be the new parent. 4. children [OPTIONAL] {string] The labels of one or more resource groups that will be the new children of the specified parent resource group. 5. groups [OPTIONAL] {string] The labels of one or more resource groups to include in the colocation dependency 6. high [OPTIONAL] {string] The labels of one or more resource groups to include in an anti-colocation dependency, and give the highest startup priority. 7. intermediate [OPTIONAL] {string] The labels of one or more resource groups to include in an anti-colocation dependency, and give intermediate startup priority. 8. low [OPTIONAL] {string] The labels of one or more resource groups to include in an anti-colocation dependency, and give the lowest startup priority. 9. stop [OPTIONAL] {string] The label of a resource group that will be stopped after the specified "AFTER" resource group. 10. start [OPTIONAL] {string] The label of a resource group that will be started after the specified "AFTER" resource group. 11. after [OPTIONAL] {string] The label of a resource group that will be stopped or started *before* the list of other resource groups provided via "STOP" or "START". 12. serial [OPTIONAL] {string} The labels of one or more resource groups that are to be aquired/release serially, in the order specified. Any groups not specified here will continue to be acquired/released in parallel (the default behavior). 13. parallel [OPTIONAL] {string} The labels of one or more resource groups that are to be aquired/release in parallel (i.e. at the same time). Any groups not specified here will be acquired/released serially, in the order previously specified. =head1 RETURN 0: no errors were detected; the operation appears to have been successful 1: a general error has occurred 2: a specified resource does not exist, or could not be found 3: some required input was missing 4: some detected input was incorrect in some way 5: a required dependency does not exist 6: a specified search failed to match any data =cut } # End of "devDoc()" #============================================================================== # The following, comment block attempts to enforce coding standards when this # file is edited via emacs or vim. This block _must_ appear at the very end # of the file, or the editor will not find it, and it will be ignored. #============================================================================== # Local Variables: # indent-tabs-mode: nil # tab-width: 4 # End: #============================================================================== # vim: tabstop=4 shiftwidth=4 expandtab #==============================================================================