sharedlibraries/osinfo/osinfo.go (64 lines of code) (raw):

/* Copyright 2025 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 osinfo can be used to read OS and vendor related information from a // known source file. For Linux-based systems, the /etc/os-release file is // used. // // This package does not support Windows based systems. package osinfo import ( "context" "fmt" "io" "runtime" "strings" "github.com/zieckey/goini" "github.com/GoogleCloudPlatform/workloadagentplatform/sharedlibraries/log" ) const ( // OSName is given by runtime.GOOS. OSName = runtime.GOOS // OSReleaseFilePath lists the location of the os-release file in the Linux system. OSReleaseFilePath = "/etc/os-release" ) type ( // Data contains the OS related data. Data struct { OSName string OSVendor string OSVersion string } // FileReadCloser is a function which reads an input file and returns the // io.ReadCloser. It is created to facilitate testing. FileReadCloser func(string) (io.ReadCloser, error) ) // ReadData returns the OS related information. In case of Windows, the vendor // and version are empty. func ReadData(ctx context.Context, rc FileReadCloser, os, filePath string) (Data, error) { res := Data{OSName: os} // TODO: b/401025374 - Add support for Windows. if res.OSName == "windows" { log.CtxLogger(ctx).Info("Reading OS data for Windows is not supported") return res, nil } var err error res.OSVendor, res.OSVersion, err = readReleaseInfo(ctx, rc, filePath) if err != nil { log.CtxLogger(ctx).Errorw("Failed to read OS release info", "error", err) } return res, err } // readReleaseInfo parses the OS release file and retrieves the values for the // osVendorID and osVersion. func readReleaseInfo(ctx context.Context, cfr FileReadCloser, filePath string) (osVendorID, osVersion string, err error) { if cfr == nil || filePath == "" { return "", "", fmt.Errorf("both ConfigFileReader and OSReleaseFilePath must be set") } file, err := cfr(filePath) if err != nil { return "", "", err } defer file.Close() ini := goini.New() if err := ini.ParseFrom(file, "\n", "="); err != nil { return "", "", fmt.Errorf("failed to parse %s: %v", filePath, err) } id, ok := ini.Get("ID") if !ok { log.CtxLogger(ctx).Warn(fmt.Sprintf("Could not read ID from %s", filePath)) id = "" } osVendorID = strings.ReplaceAll(strings.TrimSpace(id), `"`, "") version, ok := ini.Get("VERSION") if !ok { log.CtxLogger(ctx).Warn(fmt.Sprintf("Could not read VERSION from %s", filePath)) version = "" } if vf := strings.Fields(version); len(vf) > 0 { osVersion = strings.ReplaceAll(strings.TrimSpace(vf[0]), `"`, "") } return osVendorID, osVersion, nil }