internal/onetime/instancemetadata/instancemetadata.go (90 lines of code) (raw):

/* Copyright 2024 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 instancemetadata implements the one time execution mode for fetching the metadata of the // instance on which the agent is running. package instancemetadata import ( "context" "fmt" "io" "os" "flag" "google.golang.org/protobuf/encoding/prototext" "github.com/google/subcommands" "github.com/GoogleCloudPlatform/sapagent/internal/configuration" "github.com/GoogleCloudPlatform/sapagent/internal/onetime" "github.com/GoogleCloudPlatform/workloadagentplatform/sharedlibraries/osinfo" impb "github.com/GoogleCloudPlatform/sapagent/protos/instancemetadata" ) type ( // InstanceMetadata stores the arguments for the instancemetadata subcommand. InstanceMetadata struct { LogLevel string `json:"loglevel"` LogPath string `json:"log-path"` Help bool `json:"help,string"` OSReleasePath string `json:"os-release-path"` oteLogger *onetime.OTELogger `json:"-"` // RC is a field of type ReadCloser, a custom implementation can be provided if needed else the // default implementation will be used. RC ReadCloser `json:"-"` } // ReadCloser is function which takes the path of file and returns an io.ReadCloser interface and // an error. ReadCloser func(path string) (io.ReadCloser, error) ) const ( defaultLogLevel = "info" ) // Name implements the subcommand interface for instancemetadata. func (*InstanceMetadata) Name() string { return "instancemetadata" } // Synopsis implements the subcommand interface for instancemetadata. func (*InstanceMetadata) Synopsis() string { return "fetch the metadata of the instance" } // Usage implements the subcommand interface for instancemetadata. func (*InstanceMetadata) Usage() string { return "Usage: instancemetadata [-loglevel=<debug|info|warn|error>] [-log-path=<log-path>] [-h]\n" } // SetFlags implements the subcommand interface for instancemetadata. func (m *InstanceMetadata) SetFlags(fs *flag.FlagSet) { fs.BoolVar(&m.Help, "h", false, "Display help") fs.BoolVar(&m.Help, "help", false, "Display help") fs.StringVar(&m.LogLevel, "loglevel", defaultLogLevel, "Sets the logging level for a log file") fs.StringVar(&m.LogPath, "log-path", "", "The log path to write the log file (optional), default value is /var/log/google-cloud-sap-agent/instancemetadata.log") fs.StringVar(&m.OSReleasePath, "os-release-path", osinfo.OSReleaseFilePath, "The path to file which contains OS release information, default value is /etc/os-release") } // Execute implements the subcommand interface for instancemetadata. func (m *InstanceMetadata) Execute(ctx context.Context, f *flag.FlagSet, args ...any) subcommands.ExitStatus { _, cp, exitStatus, completed := onetime.Init(ctx, onetime.InitOptions{ Name: m.Name(), Help: m.Help, LogLevel: m.LogLevel, LogPath: m.LogPath, Fs: f, }, args...) if !completed { return exitStatus } _, _, status := m.Run(ctx, onetime.CreateRunOptions(cp, false)) if status != subcommands.ExitSuccess { m.oteLogger.LogMessageToFileAndConsole(ctx, "Failed to fetch instance metadata") } return status } // Run executes the MetadataHandler and returns the response. func (m *InstanceMetadata) Run(ctx context.Context, opts *onetime.RunOptions) (*impb.Metadata, string, subcommands.ExitStatus) { m.oteLogger = onetime.CreateOTELogger(opts.DaemonMode) m.setDefaults() return m.metadataHandler(ctx) } func (m *InstanceMetadata) setDefaults() { if m.OSReleasePath == "" { m.OSReleasePath = osinfo.OSReleaseFilePath } if m.RC == nil { m.RC = func(path string) (io.ReadCloser, error) { file, err := os.Open(path) var f io.ReadCloser = file return f, err } } } func (m *InstanceMetadata) metadataHandler(ctx context.Context) (*impb.Metadata, string, subcommands.ExitStatus) { osData, err := osinfo.ReadData(ctx, osinfo.FileReadCloser(m.RC), osinfo.OSName, m.OSReleasePath) if err != nil { m.oteLogger.LogMessageToFileAndConsole(ctx, fmt.Sprintf("Could not read OS release info from %s", m.OSReleasePath)) return nil, fmt.Sprintf("could not read OS release info, error: %s", err.Error()), subcommands.ExitFailure } response := &impb.Metadata{ OsName: osData.OSName, OsVendor: osData.OSVendor, OsVersion: osData.OSVersion, AgentVersion: configuration.AgentVersion, AgentBuildChange: configuration.AgentBuildChange, } m.oteLogger.LogMessageToFileAndConsole(ctx, fmt.Sprintf("Instance Metadata: %s", prototext.Format(response))) return response, "", subcommands.ExitSuccess }