You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
153 lines
3.7 KiB
Go
153 lines
3.7 KiB
Go
// +build linux darwin unix
|
|
|
|
package staros
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os/user"
|
|
"strconv"
|
|
"strings"
|
|
"syscall"
|
|
"time"
|
|
)
|
|
|
|
var clockTicks = 100 // default value
|
|
|
|
// StartTime 开机时间
|
|
func StartTime() time.Time {
|
|
tmp, _ := readAsString("/proc/stat")
|
|
data := splitBy(ReplaceByte9(tmp), " ")
|
|
btime, _ := strconv.ParseInt(strings.TrimSpace(data["btime"]), 10, 64)
|
|
return time.Unix(btime, 0)
|
|
}
|
|
|
|
// IsRoot 当前是否是管理员用户
|
|
func IsRoot() bool {
|
|
uid, _ := user.Current()
|
|
if uid.Uid == "0" {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func Whoami() (uid, gid int, uname, gname, home string, err error) {
|
|
var me *user.User
|
|
var gup *user.Group
|
|
me, err = user.Current()
|
|
if err != nil {
|
|
return
|
|
}
|
|
uid, _ = strconv.Atoi(me.Uid)
|
|
gid, _ = strconv.Atoi(me.Uid)
|
|
home = me.HomeDir
|
|
uname = me.Username
|
|
gup, err = user.LookupGroupId(me.Gid)
|
|
if err != nil {
|
|
return
|
|
}
|
|
gname = gup.Name
|
|
return
|
|
}
|
|
|
|
func getCPUSample() (idle, total uint64) {
|
|
contents, err := ioutil.ReadFile("/proc/stat")
|
|
if err != nil {
|
|
return
|
|
}
|
|
lines := strings.Split(string(contents), "\n")
|
|
for _, line := range lines {
|
|
fields := strings.Fields(line)
|
|
if fields[0] == "cpu" {
|
|
numFields := len(fields)
|
|
for i := 1; i < numFields; i++ {
|
|
val, err := strconv.ParseUint(fields[i], 10, 64)
|
|
if err != nil {
|
|
fmt.Println("Error: ", i, fields[i], err)
|
|
}
|
|
total += val // tally up all the numbers to get total ticks
|
|
if i == 4 || i == 5 { // idle is the 5th field in the cpu line
|
|
idle += val
|
|
}
|
|
}
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
func splitProcStat(content []byte) []string {
|
|
nameStart := bytes.IndexByte(content, '(')
|
|
nameEnd := bytes.LastIndexByte(content, ')')
|
|
restFields := strings.Fields(string(content[nameEnd+2:])) // +2 skip ') '
|
|
name := content[nameStart+1 : nameEnd]
|
|
pid := strings.TrimSpace(string(content[:nameStart]))
|
|
fields := make([]string, 3, len(restFields)+3)
|
|
fields[1] = string(pid)
|
|
fields[2] = string(name)
|
|
fields = append(fields, restFields...)
|
|
return fields
|
|
}
|
|
|
|
func getCPUSampleByPid(pid int) float64 {
|
|
contents, err := ioutil.ReadFile("/proc/" + strconv.Itoa(pid) + "/stat")
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
fields := splitProcStat(contents)
|
|
utime, err := strconv.ParseFloat(fields[14], 64)
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
|
|
stime, err := strconv.ParseFloat(fields[15], 64)
|
|
if err != nil {
|
|
return 0
|
|
}
|
|
|
|
// There is no such thing as iotime in stat file. As an approximation, we
|
|
// will use delayacct_blkio_ticks (aggregated block I/O delays, as per Linux
|
|
// docs). Note: I am assuming at least Linux 2.6.18
|
|
var iotime float64
|
|
if len(fields) > 42 {
|
|
iotime, err = strconv.ParseFloat(fields[42], 64)
|
|
if err != nil {
|
|
iotime = 0 // Ancient linux version, most likely
|
|
}
|
|
} else {
|
|
iotime = 0 // e.g. SmartOS containers
|
|
}
|
|
return utime/float64(clockTicks) + stime/float64(clockTicks) + iotime/float64(clockTicks)
|
|
}
|
|
func CpuUsageByPid(pid int, sleep time.Duration) float64 {
|
|
total1 := getCPUSampleByPid(pid)
|
|
time.Sleep(sleep)
|
|
total2 := getCPUSampleByPid(pid)
|
|
return (total2 - total1) / sleep.Seconds() * 100
|
|
}
|
|
|
|
// CpuUsage 获取CPU使用量
|
|
func CpuUsage(sleep time.Duration) float64 {
|
|
idle0, total0 := getCPUSample()
|
|
time.Sleep(sleep)
|
|
idle1, total1 := getCPUSample()
|
|
idleTicks := float64(idle1 - idle0)
|
|
totalTicks := float64(total1 - total0)
|
|
cpuUsage := 100 * (totalTicks - idleTicks) / totalTicks
|
|
return cpuUsage
|
|
//fmt.Printf("CPU usage is %f%% [busy: %f, total: %f]\n", cpuUsage, totalTicks-idleTicks, totalTicks)
|
|
}
|
|
|
|
func DiskUsage(path string) (disk DiskStatus) {
|
|
fs := syscall.Statfs_t{}
|
|
err := syscall.Statfs(path, &fs)
|
|
if err != nil {
|
|
return
|
|
}
|
|
disk.All = fs.Blocks * uint64(fs.Bsize)
|
|
disk.Free = fs.Bfree * uint64(fs.Bsize)
|
|
disk.Available = fs.Bavail * uint64(fs.Bsize)
|
|
disk.Used = disk.All - disk.Free
|
|
return
|
|
}
|