internal/system/sapdiscovery/sapdiscovery.go (485 lines of code) (raw):
/*
Copyright 2022 Google LLC
Licensed 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
https://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.
*/
// Package sapdiscovery discovers the SAP Applications and instances running on a given instance.
package sapdiscovery
import (
"context"
"fmt"
"regexp"
"strconv"
"strings"
"google.golang.org/protobuf/encoding/prototext"
"github.com/GoogleCloudPlatform/sapagent/internal/pacemaker"
"github.com/GoogleCloudPlatform/sapagent/internal/system"
"github.com/GoogleCloudPlatform/workloadagentplatform/sharedlibraries/commandlineexecutor"
"github.com/GoogleCloudPlatform/workloadagentplatform/sharedlibraries/log"
cpb "github.com/GoogleCloudPlatform/sapagent/protos/configuration"
sapb "github.com/GoogleCloudPlatform/sapagent/protos/sapapp"
spb "github.com/GoogleCloudPlatform/workloadagentplatform/sharedprotos/system"
)
var (
// hostMapPattern captures the site name and host name e.g.
// sapodb22 -> [HO2_22] sapodb22 -> Site: HO2_22 Host name: sapodb22
hostMapPattern = regexp.MustCompile(`\S*\s->\s\[([^]]+)\]\s(.*)`)
// sitePattern captures site name e.g.
// site name: HO2_22 -> site_name=HO2_22
sitePattern = regexp.MustCompile(`site name: (.*)`)
primaryMastersPattern = regexp.MustCompile(`PRIMARY_MASTERS=(.*)`)
sapInitRunningPattern = regexp.MustCompile(`running$`)
// netweaverProtocolPortPattern captures protocol and port number.
// Example: "ParameterValue\nOK\nPROT=HTTP,PORT=8100" is parsed as "PROT=HTTP,PORT=8100".
netweaverProtocolPortPattern = regexp.MustCompile(`PROT=([a-z|A-Z]+),PORT=([0-9]+)`)
// sapServicesStartsrvPattern captures the sapstartsrv path in /usr/sap/sapservices.
// Example: "/usr/sap/DEV/ASCS01/exe/sapstartsrv pf=/usr/sap/DEV/SYS/profile/DEV_ASCS01_dnwh75ldbci -D -u devadm"
// is parsed as "sapstartsrv pf=/usr/sap/DEV/SYS/profile/DEV_ASCS01".
// Additional possibilities include sapstartsrv pf=/sapmnt/PRD/profile/PRD_ASCS01_alidascs11
sapServicesStartsrvPattern = regexp.MustCompile(`sapstartsrv pf=/(usr/sap|sapmnt)/([A-Z][A-Z0-9]{2})[/a-zA-Z0-9]*/profile/([A-Z][A-Z0-9]{2})_([a-zA-Z]+)([0-9]+)`)
// sapServicesProfilePattern captures the sap profile path in /usr/sap/sapservices.
// Example: "/usr/sap/DEV/ASCS01/exe/sapstartsrv pf=/usr/sap/DEV/SYS/profile/DEV_ASCS01_dnwh75ldbci -D -u devadm"
// is parsed as "/usr/sap/DEV/SYS/profile/DEV_ASCS01_dnwh75ldbci".
sapServicesProfilePattern = regexp.MustCompile(`pf=(\S*)?`)
// libraryPathPattern captures the LD_LIBRARY_PATH from /usr/sap/sapservices.
// Example: "LD_LIBRARY_PATH=/usr/sap/DEV/ASCS01/exe:$LD_LIBRARY_PATH;export LD_LIBRARY_PATH" is parsed as
// "/usr/sap/DEV/ASCS01/exe".
libraryPathPattern = regexp.MustCompile("LD_LIBRARY_PATH=(/usr/sap/[A-Z][A-Z|0-9][A-Z|0-9]/[a-z|A-Z]+[0-9]+/exe)")
// systemReplicationStatus contains valid return codes for systemReplicationStatus.py.
// Return codes reference can be found in "SAP HANA System Replication" section in SAP docs.
// Any code from 10-15 is a valid return code. Anything else needs to be treated as failure.
// Reference: https://help.sap.com/docs/SAP_HANA_PLATFORM/4e9b18c116aa42fc84c7dbfd02111aba/f6b1bd1020984ee69e902b21b702c096.html?version=2.0.04
systemReplicationStatus = map[int64]string{
10: "No System Replication.",
11: "Error: Error occurred on the connection. Additional details on the error can be found in REPLICATION_STATUS_DETAILS.",
12: "Unknown: The secondary system did not connect to primary since last restart of the primary system.",
13: "Initializing: Initial data transfer is in progress. In this state, the secondary is not usable at all.",
14: "Syncing: The secondary system is syncing again (for example, after a temporary connection loss or restart of the secondary).",
15: "Active: Initialization or sync with the primary is complete and the secondary is continuously replicating. No data loss will occur in SYNC mode.",
}
siteMapPattern = regexp.MustCompile(`((\s+\|)*)(---)?([a-zA-Z0-9_\-]+)\s\([^\)]+\)`)
depthPattern = regexp.MustCompile(`\s+\|`)
modePattern = regexp.MustCompile(`mode: (primary|syncmem|async|sync)\n`)
)
type (
listInstances func(context.Context, commandlineexecutor.Execute) ([]*instanceInfo, error)
// ReplicationConfig is a function that returns the replication configuration information.
ReplicationConfig func(context.Context, string, string, string, system.SapSystemDiscoveryInterface) (int, int64, *sapb.HANAReplicaSite, error)
instanceInfo struct {
Sid, InstanceName, Snr, ProfilePath, LDLibraryPath string
}
// GCEInterface provides an easily testable translation to the secret manager API.
GCEInterface interface {
GetSecret(ctx context.Context, projectID, secretName string) (string, error)
}
)
// SAPApplications Discovers the SAP Application instances.
//
// Returns a sapb.SAPInstances which is an array of SAP instances running on the given machine.
func SAPApplications(ctx context.Context, sapSystemInterface system.SapSystemDiscoveryInterface) *sapb.SAPInstances {
data, err := pacemaker.Data(ctx)
if err != nil {
// could not collect data from crm_mon
log.CtxLogger(ctx).Debugw("Failure in reading crm_mon data from pacemaker", "err", err)
}
return instances(ctx, HANAReplicationConfig, listSAPInstances, commandlineexecutor.ExecuteCommand, data, sapSystemInterface)
}
// instances is a testable version of SAPApplications.
func instances(ctx context.Context, hrc ReplicationConfig, list listInstances, exec commandlineexecutor.Execute, crmdata *pacemaker.CRMMon, sapSystemInterface system.SapSystemDiscoveryInterface) *sapb.SAPInstances {
log.CtxLogger(ctx).Debug("Discovering SAP Applications.")
var sapInstances []*sapb.SAPInstance
hana, err := hanaInstances(ctx, hrc, list, exec, sapSystemInterface)
if err != nil {
log.CtxLogger(ctx).Infow("Unable to discover HANA instances", "err", err)
} else {
sapInstances = hana
}
netweaver, err := netweaverInstances(ctx, list, exec)
if err != nil {
log.CtxLogger(ctx).Infow("Unable to discover Netweaver instances", "err", err)
} else {
sapInstances = append(sapInstances, netweaver...)
}
return &sapb.SAPInstances{
Instances: sapInstances,
LinuxClusterMember: pacemaker.Enabled(ctx, crmdata),
}
}
// hanaInstances returns list of SAP HANA Instances present on the machine.
// Returns error in case of failures.
func hanaInstances(ctx context.Context, hrc ReplicationConfig, list listInstances, exec commandlineexecutor.Execute, sapSystemInterface system.SapSystemDiscoveryInterface) ([]*sapb.SAPInstance, error) {
log.CtxLogger(ctx).Debug("Discovering SAP HANA instances.")
sapServicesEntries, err := list(ctx, exec)
if err != nil {
return nil, err
}
var instances []*sapb.SAPInstance
for _, entry := range sapServicesEntries {
log.CtxLogger(ctx).Debugw("Processing SAP Instance", "instance", entry)
if entry.InstanceName != "HDB" {
log.CtxLogger(ctx).Debugw("Instance is not SAP HANA", "instance", entry)
continue
}
instanceID := entry.InstanceName + entry.Snr
user := strings.ToLower(entry.Sid) + "adm"
siteID, _, replicationSites, err := hrc(ctx, user, entry.Sid, instanceID, sapSystemInterface)
if err != nil {
log.CtxLogger(ctx).Debugw("Failed to get HANA HA configuration for instance", "instanceid", instanceID, "error", err)
siteID = -1 // INSTANCE_SITE_UNDEFINED
}
log.CtxLogger(ctx).Debugw("Got replication information", "siteID", siteID, "replicationSites", replicationSites)
instance := &sapb.SAPInstance{
Sapsid: entry.Sid,
InstanceNumber: entry.Snr,
Type: sapb.InstanceType_HANA,
Site: HANASite(siteID),
User: user,
InstanceId: instanceID,
ProfilePath: entry.ProfilePath,
LdLibraryPath: entry.LDLibraryPath,
SapcontrolPath: fmt.Sprintf("%s/sapcontrol", entry.LDLibraryPath),
HanaReplicationTree: replicationSites,
}
log.CtxLogger(ctx).Debugw("Found SAP HANA instance", "instance", prototext.Format(instance))
instances = append(instances, instance)
}
log.CtxLogger(ctx).Debugw("Found SAP HANA instances", "count", len(instances), "instances", instances)
return instances, nil
}
// HANAReplicationConfig discovers the HANA High Availability configuration.
// Returns:
// HANA HA site as int.
//
// 0 == STANDALONE MODE
// 1 == HANA PRIMARY
// 2 == HANA SECONDARY
// 3 == HANA DR
//
// Exit status of systemReplicationStatus.py as int64.
func HANAReplicationConfig(ctx context.Context, user, sid, instID string, sapSystemInterface system.SapSystemDiscoveryInterface) (site int, exitStatus int64, replicationSites *sapb.HANAReplicaSite, err error) {
return readReplicationConfig(ctx, user, sid, instID, commandlineexecutor.ExecuteCommand, sapSystemInterface)
}
// readReplicationConfig is a testable version of HANAReplicationConfig.
func readReplicationConfig(ctx context.Context, user, sid, instID string, exec commandlineexecutor.Execute, sapSystemInterface system.SapSystemDiscoveryInterface) (mode int, exitStatus int64, replicationSites *sapb.HANAReplicaSite, err error) {
cmd := "sudo"
args := "" +
fmt.Sprintf("-i -u %sadm /usr/sap/%s/%s/HDBSettings.sh ", strings.ToLower(sid), sid, instID) +
fmt.Sprintf(" /usr/sap/%s/%s/exe/python_support/systemReplicationStatus.py --sapcontrol=1", sid, instID)
result := exec(ctx, commandlineexecutor.Params{
Executable: cmd,
ArgsToSplit: args,
})
exitStatus = int64(result.ExitCode)
log.CtxLogger(ctx).Debugw("systemReplicationStatus.py returned", "result", result)
cmd = "sudo"
args = fmt.Sprintf("-i -u %sadm hdbnsutil -sr_state", strings.ToLower(sid))
result = exec(ctx, commandlineexecutor.Params{
Executable: cmd,
ArgsToSplit: args,
})
log.CtxLogger(ctx).Debugw("Tool hdbnsutil returned", "exitstatus", exitStatus)
log.CtxLogger(ctx).Debugw("SAP HANA Replication Config result", "stdout", result.StdOut)
if strings.Contains(result.StdOut, "mode: none") {
log.CtxLogger(ctx).Debugw("HANA instance is in standalone mode for instance", "instanceid", instID)
return 0, exitStatus, nil, nil
}
if exitStatus == 10 {
// Since this is a system with replication, as indicated by failing the "mode: none" check, the
// exit status of 10 indicates an error in the system. Since we know it is not standalone, we
// will return 12 to indicate a replication state error.
exitStatus = 12
}
match := sitePattern.FindStringSubmatch(result.StdOut)
if len(match) != 2 {
log.CtxLogger(ctx).Debugw("Error determining SAP HANA Site for instance", "instanceid", instID)
return 0, 0, nil, nil
}
site := match[1]
systems := sapSystemInterface.GetSAPSystems()
var sys *spb.SapDiscovery
for _, s := range systems {
if s.GetDatabaseLayer().GetSid() == sid {
sys = s
break
}
}
mode, err = readMode(ctx, result.StdOut, site, sys)
if err != nil {
return 0, exitStatus, nil, err
}
haHostMap, err := readHAMembers(ctx, result.StdOut, instID)
if err != nil {
log.CtxLogger(ctx).Infow("Error reading HA members", "instanceid", instID, "error", err)
return mode, exitStatus, nil, nil
}
log.CtxLogger(ctx).Debugw("HA Host Map", "haHostMap", haHostMap)
// site may be either the host name or the site name. Utilize the haHostMap to get a site name.
if s, ok := haHostMap[site]; ok {
site = s
}
// Separate sites into tiers.
log.CtxLogger(ctx).Debugw("Separating sites into tiers")
lines := strings.Split(result.StdOut, "\n")
inSiteMappings := false
siteStack := []*sapb.HANAReplicaSite{}
currentDepth := -1
var primarySite *sapb.HANAReplicaSite
for _, line := range lines {
if strings.Contains(line, "Site Mappings") {
inSiteMappings = true
continue
} else if !inSiteMappings {
continue
}
matches := siteMapPattern.FindStringSubmatch(line)
if len(matches) != 5 {
continue
}
siteName := matches[4]
var hostName string
for h, s := range haHostMap {
log.CtxLogger(ctx).Debugw("Checking host name", "siteName", s, "hostName", h)
if s == siteName {
hostName = h
log.CtxLogger(ctx).Debugw("Found host name", "siteName", s, "hostName", h)
break
}
}
log.CtxLogger(ctx).Debugw("Making site for host", "siteName", siteName, "hostName", hostName)
site := &sapb.HANAReplicaSite{Name: hostName}
depth := len(depthPattern.FindAllStringSubmatch(line, -1))
log.CtxLogger(ctx).Debugw("Depth of site", "site", site, "depth", depth, "currentDepth", currentDepth)
for depth <= currentDepth {
log.CtxLogger(ctx).Debugw("Popping site", "site", siteStack[len(siteStack)-1])
siteStack = siteStack[:len(siteStack)-1]
currentDepth--
}
if len(siteStack) > 0 {
currentSite := siteStack[len(siteStack)-1]
log.CtxLogger(ctx).Debugw("Appending site to parent's targets", "currentSite", currentSite, "site", site)
currentSite.Targets = append(currentSite.Targets, site)
}
siteStack = append(siteStack, site)
currentDepth = depth
if primarySite == nil {
primarySite = site
}
}
return mode, exitStatus, primarySite, nil
}
func readMode(ctx context.Context, stdOut, site string, sys *spb.SapDiscovery) (mode int, err error) {
log.CtxLogger(ctx).Debugw("Reading SAP HANA Replication Mode for instance", "siteID", site)
match := modePattern.FindStringSubmatch(stdOut)
if len(match) < 2 {
log.CtxLogger(ctx).Debugw("Error determining SAP HANA Replication Mode for instance", "siteID", site, "stdOut", stdOut)
return 0, fmt.Errorf("error determining SAP HANA Replication Mode for site: %s", site)
}
log.CtxLogger(ctx).Debugw("Match", "match", match)
if match[1] == "primary" {
mode = 1
log.CtxLogger(ctx).Debug("Current SAP HANA node is primary")
} else {
// How do we determine if this is secondary or DR?
// Node is DR if tier >= 3 or if it is not in the same load balancer group as the primary.
mode = 2
if sys.GetDatabaseLayer().GetReplicationSites() != nil {
// If this site is not part of the primary DB component, it is DR.
log.CtxLogger(ctx).Debugw("Checking if site is part of primary DB component")
mode = 3
for _, r := range sys.GetDatabaseLayer().GetResources() {
if r.GetResourceKind() == spb.SapDiscovery_Resource_RESOURCE_KIND_INSTANCE &&
r.GetInstanceProperties().GetVirtualHostname() == site {
mode = 2
break
}
}
}
log.CtxLogger(ctx).Debug("Current SAP HANA node is secondary")
}
return mode, nil
}
func readHAMembers(ctx context.Context, stdOut, instID string) (HAHostMap map[string]string, err error) {
HAHostMap = make(map[string]string)
haSites := hostMapPattern.FindAllStringSubmatch(stdOut, -1)
if len(haSites) < 1 {
log.CtxLogger(ctx).Debugw("Error determining SAP HANA HA members for instance", "instanceid", instID)
return nil, fmt.Errorf("error determining SAP HANA HA members for instance: %s", instID)
}
// Submatch should be a number of lines matching the pattern.
// The 0 element is the whole match, and the subsequent elements are the matches for the capture groups.
for _, match := range haSites {
if len(match) > 2 {
site := match[1]
host := match[2]
HAHostMap[host] = site
}
}
return HAHostMap, nil
}
// HANASite maps an integer to SAP Instance Site Type.
func HANASite(mode int) sapb.InstanceSite {
sites := map[int]sapb.InstanceSite{
0: sapb.InstanceSite_HANA_STANDALONE,
1: sapb.InstanceSite_HANA_PRIMARY,
2: sapb.InstanceSite_HANA_SECONDARY,
3: sapb.InstanceSite_HANA_DR,
}
if site, ok := sites[mode]; ok {
return site
}
return sapb.InstanceSite_INSTANCE_SITE_UNDEFINED
}
// listSAPInstances returns list of SAP Instances present on the machine.
// The list is derived from '/usr/sap/sapservices' file.
func listSAPInstances(ctx context.Context, exec commandlineexecutor.Execute) ([]*instanceInfo, error) {
var sapServicesEntries []*instanceInfo
result := exec(ctx, commandlineexecutor.Params{
Executable: "grep",
ArgsToSplit: "'pf=' /usr/sap/sapservices",
})
log.CtxLogger(ctx).Debugw("`grep 'pf=' /usr/sap/sapservices` returned", "stdout", result.StdOut, "stderr", result.StdErr, "error", result.Error)
if result.Error != nil {
return nil, result.Error
}
lines := strings.Split(strings.TrimSuffix(result.StdOut, "\n"), "\n")
for _, line := range lines {
if strings.HasPrefix(line, "#") {
log.CtxLogger(ctx).Debugw("Not processing the commented entry", "line", line)
continue
}
path := sapServicesStartsrvPattern.FindStringSubmatch(line)
if len(path) != 6 {
log.CtxLogger(ctx).Debugw("No SAP instance found", "line", line, "match", path)
continue
}
profile := sapServicesProfilePattern.FindStringSubmatch(line)
if len(profile) != 2 {
log.CtxLogger(ctx).Debugw("No SAP instance profile found", "line", line, "match", profile)
continue
}
number, err := strconv.Atoi(path[5])
if err != nil {
log.CtxLogger(ctx).Debugw("Failed to parse SAP instance number", "line", line, "match", path[5], "err", err)
continue
}
entry := &instanceInfo{
Sid: strings.ToUpper(path[2]),
InstanceName: path[4],
Snr: fmt.Sprintf("%02d", number),
ProfilePath: profile[1],
}
entry.LDLibraryPath = fmt.Sprintf("/usr/sap/%s/%s%s/exe", entry.Sid, entry.InstanceName, entry.Snr)
libraryPath := libraryPathPattern.FindStringSubmatch(line)
if len(libraryPath) == 2 {
log.CtxLogger(ctx).Debugw("Overriding SAP LD_LIBRARY_PATH with value found", "line", line)
entry.LDLibraryPath = libraryPath[1]
}
log.CtxLogger(ctx).Debugw("Found SAP Instance", "entry", entry)
sapServicesEntries = append(sapServicesEntries, entry)
}
return sapServicesEntries, nil
}
// netweaverInstances returns list of SAP Netweaver instances present on the machine.
func netweaverInstances(ctx context.Context, list listInstances, exec commandlineexecutor.Execute) ([]*sapb.SAPInstance, error) {
var instances []*sapb.SAPInstance
log.CtxLogger(ctx).Debug("Discovering SAP NetWeaver instances.")
sapServicesEntries, err := list(ctx, commandlineexecutor.ExecuteCommand)
if err != nil {
return nil, err
}
for _, entry := range sapServicesEntries {
log.CtxLogger(ctx).Debugw("Processing SAP Instance", "entry", entry)
instanceID := entry.InstanceName + entry.Snr
user := strings.ToLower(entry.Sid) + "adm"
instance := &sapb.SAPInstance{
Sapsid: entry.Sid,
InstanceNumber: entry.Snr,
User: user,
InstanceId: instanceID,
ProfilePath: entry.ProfilePath,
LdLibraryPath: entry.LDLibraryPath,
SapcontrolPath: fmt.Sprintf("%s/sapcontrol", entry.LDLibraryPath),
}
instance.NetweaverHttpPort, instance.Type, instance.Kind = findPort(ctx, instance, entry.InstanceName, exec)
if instance.GetType() == sapb.InstanceType_NETWEAVER {
instance.NetweaverHealthCheckUrl, instance.ServiceName, err = buildURLAndServiceName(entry.InstanceName, instance.NetweaverHttpPort)
if err != nil {
log.CtxLogger(ctx).Debugw("Could not build Netweaver URL for health check", "err", err)
}
log.CtxLogger(ctx).Debugw("Found SAP NetWeaver instance", "instance", prototext.Format(instance))
instances = append(instances, instance)
}
}
log.CtxLogger(ctx).Debugw("Found SAP NetWeaver instances", "count", len(instances), "instances", instances)
return instances, nil
}
// findPort uses the SAP instanceName to find the server HTTP port.
func findPort(ctx context.Context, instance *sapb.SAPInstance, instanceName string, exec commandlineexecutor.Execute) (string, sapb.InstanceType, sapb.InstanceKind) {
var (
httpPort string
instanceType sapb.InstanceType = sapb.InstanceType_INSTANCE_TYPE_UNDEFINED
instanceKind sapb.InstanceKind = sapb.InstanceKind_INSTANCE_KIND_UNDEFINED
err error
)
switch instanceName {
case "ASCS", "SCS":
instanceType = sapb.InstanceType_NETWEAVER
instanceKind = sapb.InstanceKind_CS
httpPort, err = serverPortFromSAPProfile(ctx, instance, "ms", exec)
if err != nil {
log.CtxLogger(ctx).Debugw("The ms HTTP port not found, set to default: '81<snr>.'", "instancename", instanceName)
httpPort = "81" + instance.GetInstanceNumber()
}
case "J", "JC", "D", "DVEBMGS":
instanceType = sapb.InstanceType_NETWEAVER
instanceKind = sapb.InstanceKind_APP
httpPort, err = serverPortFromSAPProfile(ctx, instance, "icm", exec)
if err != nil {
log.CtxLogger(ctx).Debugw("The icm HTTP port not found, set to default: '5<snr>00.'", "instancename", instanceName)
httpPort = "5" + instance.GetInstanceNumber() + "00"
}
case "ERS":
log.CtxLogger(ctx).Debugw("This is an Enqueue Replication System.", "instancename", instanceName)
instanceType = sapb.InstanceType_NETWEAVER
instanceKind = sapb.InstanceKind_ERS
case "HDB":
log.CtxLogger(ctx).Debugw("This is a HANA instance.", "instancename", instanceName)
instanceType = sapb.InstanceType_HANA
default:
if strings.HasPrefix(instanceName, "W") {
instanceKind = sapb.InstanceKind_APP
instanceType = sapb.InstanceType_NETWEAVER
} else {
log.CtxLogger(ctx).Debugw("Unknown instance", "instancename", instanceName)
}
}
return httpPort, instanceType, instanceKind
}
// serverPortFromSAPProfile returns the HTTP port using `sapcontrol -function ParameterValue`.
func serverPortFromSAPProfile(ctx context.Context, instance *sapb.SAPInstance, prefix string, exec commandlineexecutor.Execute) (string, error) {
// Check if any of server_port_0 thru server_port_9 are configured for HTTP.
// Reference: "Generic Profile Parameters with Ending _<xx>" section in SAP NetWeaver documentation.
// (link: https://help.sap.com/doc/saphelp_nw74/7.4.16/en-us/c4/1839a549b24fef92860134ce6af271/frameset.htm)
for i := 0; i < 10; i++ {
params := commandlineexecutor.Params{
User: instance.GetUser(),
Executable: instance.GetSapcontrolPath(),
ArgsToSplit: fmt.Sprintf("%s -nr %s -function ParameterValue %s/server_port_%d", instance.GetSapcontrolPath(), instance.GetInstanceNumber(), prefix, i),
Env: []string{"LD_LIBRARY_PATH=" + instance.GetLdLibraryPath()},
}
port, err := parseHTTPPort(ctx, params, exec)
if err != nil {
log.CtxLogger(ctx).Debugw("Server port is not configured for HTTP", "port", fmt.Sprintf("%s/server_port_%d", prefix, i), "error", err)
continue
}
return port, nil
}
return "", fmt.Errorf("the HTTP port is not configured for instance : %s", instance.GetInstanceId())
}
// parseHTTPPort parses the output of sapcontrol command for HTTP port.
// Returns HTTP port on success, error if current parameter is not configured for HTTP.
func parseHTTPPort(ctx context.Context, params commandlineexecutor.Params, exec commandlineexecutor.Execute) (port string, err error) {
result := exec(ctx, params)
log.CtxLogger(ctx).Debugw("Sapcontrol returned", "stdout", result.StdOut, "stderr", result.StdErr, "error", result.Error)
if result.Error != nil {
return "", result.Error
}
match := netweaverProtocolPortPattern.FindStringSubmatch(result.StdOut)
if len(match) != 3 {
return "", fmt.Errorf("the port is not configured for HTTP")
}
protocol, port := match[1], match[2]
log.CtxLogger(ctx).Debugw("Found protocol on port", "protocol", protocol, "port", port)
if protocol == "HTTP" && port != "0" {
return port, nil
}
return "", fmt.Errorf("the port is not configured for HTTP")
}
// buildURLAndServiceName builds the health check URLs bases on SAP Instance type.
func buildURLAndServiceName(instanceName, HTTPPort string) (url, serviceName string, err error) {
if HTTPPort == "" {
return "", "", fmt.Errorf("empty value for HTTP port")
}
switch instanceName {
case "ASCS", "SCS":
url = fmt.Sprintf("http://localhost:%s/msgserver/text/logon", HTTPPort)
serviceName = "SAP-CS" // Central Services
case "D", "DVEBMGS":
url = fmt.Sprintf("http://localhost:%s/sap/public/icman/ping", HTTPPort)
serviceName = "SAP-ICM-ABAP"
case "J", "JC":
url = fmt.Sprintf("http://localhost:%s/sap/admin/public/images/sap.png", HTTPPort)
serviceName = "SAP-ICM-Java"
default:
return "", "", fmt.Errorf("unknown SAP instance type")
}
return url, serviceName, nil
}
// sapInitRunning returns a bool indicating if sapinit is running.
// Returns an error in case of failures.
func sapInitRunning(ctx context.Context, exec commandlineexecutor.Execute) (bool, error) {
result := exec(ctx, commandlineexecutor.Params{
Executable: "/usr/sap/hostctrl/exe/sapinit",
Args: []string{"status"},
})
log.CtxLogger(ctx).Debugw("`/usr/sap/hostctrl/exe/sapinit status` returned", "stdout", result.StdOut, "stderr", result.StdErr, "error", result.Error)
if result.Error != nil {
return false, result.Error
}
match := sapInitRunningPattern.FindStringSubmatch(result.StdOut)
if match == nil {
return false, nil
}
return true, nil
}
// ReadHANACredentials returns either of a HANA DB user/password pair or a hdbuserstore key from configuration.
// To connect to a HANA instance either of a user/password pair or a hdbuserstore key is required
func ReadHANACredentials(ctx context.Context, projectID string, hanaConfig *cpb.HANAMetricsConfig, gceService GCEInterface) (user, password, hdbuserstoreKey string, err error) {
// Value hana_db_user must be set to collect HANA DB query metrics.
user = hanaConfig.GetHanaDbUser()
if user == "" {
log.CtxLogger(ctx).Debug("Using default value for hana_db_user.")
user = "SYSTEM"
}
// Either hdbuserstore_key, hana_db_password or hana_db_password_secret_name must be set for user/pass authentication.
if hanaConfig.GetHdbuserstoreKey() == "" && hanaConfig.GetHanaDbPassword() == "" && hanaConfig.GetHanaDbPasswordSecretName() == "" {
return "", "", "", fmt.Errorf("All of hana_db_password, hana_db_password_secret_name and hdbuserstore_key are empty")
}
// Return hdbuserstore credentials only, if set.
if hanaConfig.GetHdbuserstoreKey() != "" {
return user, "", hanaConfig.GetHdbuserstoreKey(), nil
}
if hanaConfig.GetHanaDbPassword() != "" {
return user, hanaConfig.GetHanaDbPassword(), "", nil
}
password, err = gceService.GetSecret(ctx, projectID, hanaConfig.GetHanaDbPasswordSecretName())
if err != nil {
return "", "", "", err
}
return user, password, "", nil
}