// +build openbsd package host import ( "bytes" "encoding/binary" "io/ioutil" "os" "os/exec" "runtime" "strconv" "strings" "time" "unsafe" "github.com/shirou/gopsutil/internal/common" "github.com/shirou/gopsutil/process" ) const ( UTNameSize = 32 /* see MAXLOGNAME in */ UTLineSize = 8 UTHostSize = 16 ) func Info() (*InfoStat, error) { ret := &InfoStat{ OS: runtime.GOOS, PlatformFamily: "openbsd", } hostname, err := os.Hostname() if err == nil { ret.Hostname = hostname } platform, family, version, err := PlatformInformation() if err == nil { ret.Platform = platform ret.PlatformFamily = family ret.PlatformVersion = version } system, role, err := Virtualization() if err == nil { ret.VirtualizationSystem = system ret.VirtualizationRole = role } procs, err := process.Pids() if err == nil { ret.Procs = uint64(len(procs)) } boot, err := BootTime() if err == nil { ret.BootTime = boot ret.Uptime = uptime(boot) } return ret, nil } func BootTime() (uint64, error) { val, err := common.DoSysctrl("kern.boottime") if err != nil { return 0, err } boottime, err := strconv.ParseUint(val[0], 10, 64) if err != nil { return 0, err } return boottime, nil } func uptime(boot uint64) uint64 { return uint64(time.Now().Unix()) - boot } func Uptime() (uint64, error) { boot, err := BootTime() if err != nil { return 0, err } return uptime(boot), nil } func PlatformInformation() (string, string, string, error) { platform := "" family := "" version := "" uname, err := exec.LookPath("uname") if err != nil { return "", "", "", err } out, err := invoke.Command(uname, "-s") if err == nil { platform = strings.ToLower(strings.TrimSpace(string(out))) } out, err = invoke.Command(uname, "-r") if err == nil { version = strings.ToLower(strings.TrimSpace(string(out))) } return platform, family, version, nil } func Virtualization() (string, string, error) { system := "" role := "" return system, role, nil } func Users() ([]UserStat, error) { var ret []UserStat utmpfile := "/var/run/utmp" file, err := os.Open(utmpfile) if err != nil { return ret, err } buf, err := ioutil.ReadAll(file) if err != nil { return ret, err } u := Utmp{} entrySize := int(unsafe.Sizeof(u)) count := len(buf) / entrySize for i := 0; i < count; i++ { b := buf[i*entrySize : i*entrySize+entrySize] var u Utmp br := bytes.NewReader(b) err := binary.Read(br, binary.LittleEndian, &u) if err != nil || u.Time == 0 { continue } user := UserStat{ User: common.IntToString(u.Name[:]), Terminal: common.IntToString(u.Line[:]), Host: common.IntToString(u.Host[:]), Started: int(u.Time), } ret = append(ret, user) } return ret, nil }