mirror of
https://github.com/stoatchat/livekit-protocol.git
synced 2026-07-01 22:04:52 -04:00
c9e9ae555f
* Add license headers & NOTICE * generated protobuf --------- Co-authored-by: github-actions <41898282+github-actions[bot]@users.noreply.github.com>
263 lines
5.3 KiB
Go
263 lines
5.3 KiB
Go
// Copyright 2023 LiveKit, Inc.
|
|
//
|
|
// 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
|
|
//
|
|
// http://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.
|
|
|
|
//go:build linux
|
|
|
|
package utils
|
|
|
|
import (
|
|
"errors"
|
|
"os"
|
|
"regexp"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/livekit/protocol/logger"
|
|
)
|
|
|
|
var (
|
|
usageRegex = regexp.MustCompile("usage_usec ([0-9]+)")
|
|
)
|
|
|
|
const (
|
|
cpuStatsPathV1 = "/sys/fs/cgroup/cpu,cpuacct/cpuacct.usage"
|
|
cpuStatsPathV2 = "/sys/fs/cgroup/cpu.stat"
|
|
|
|
numCPUPathV1Period = "/sys/fs/cgroup/cpu/cpu.cfs_period_us"
|
|
numCPUPathV1Quota = "/sys/fs/cgroup/cpu/cpu.cfs_quota_us"
|
|
numCPUPathV2 = "/sys/fs/cgroup/cpu.max"
|
|
)
|
|
|
|
type cpuInfoGetter interface {
|
|
getTotalCPUTime() (int64, error)
|
|
numCPU() (float64, error)
|
|
}
|
|
|
|
type cgroupCPUMonitor struct {
|
|
lastSampleTime int64
|
|
lastTotalCPUTime int64
|
|
nCPU float64
|
|
|
|
cg cpuInfoGetter
|
|
}
|
|
|
|
func newPlatformCPUMonitor() (platformCPUMonitor, error) {
|
|
// probe for the cgroup version
|
|
var cg cpuInfoGetter
|
|
for k, v := range map[string]func() cpuInfoGetter{
|
|
cpuStatsPathV1: newCpuInfoGetterV1,
|
|
cpuStatsPathV2: newCpuInfoGetterV2,
|
|
} {
|
|
e, err := fileExists(k)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if e {
|
|
cg = v()
|
|
break
|
|
}
|
|
}
|
|
if cg == nil {
|
|
logger.Infow("failed reading cgroup specific cpu stats, falling back to system wide implementation")
|
|
return newOSStatCPUMonitor()
|
|
}
|
|
|
|
cpu, err := cg.getTotalCPUTime()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
nCPU, err := cg.numCPU()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &cgroupCPUMonitor{
|
|
lastSampleTime: time.Now().UnixNano(),
|
|
lastTotalCPUTime: cpu,
|
|
nCPU: nCPU,
|
|
cg: cg,
|
|
}, nil
|
|
}
|
|
|
|
func (p *cgroupCPUMonitor) getCPUIdle() (float64, error) {
|
|
next, err := p.cg.getTotalCPUTime()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
t := time.Now().UnixNano()
|
|
|
|
duration := t - p.lastSampleTime
|
|
cpuTime := next - p.lastTotalCPUTime
|
|
|
|
busyRatio := float64(cpuTime) / float64(duration)
|
|
idleRatio := p.nCPU - busyRatio
|
|
|
|
// Clamp the value as we do not get all the timestamps at the same time
|
|
if idleRatio > p.nCPU {
|
|
idleRatio = p.nCPU
|
|
} else if idleRatio < 0 {
|
|
idleRatio = 0
|
|
}
|
|
|
|
p.lastSampleTime = t
|
|
p.lastTotalCPUTime = next
|
|
|
|
return idleRatio, nil
|
|
}
|
|
|
|
func (p *cgroupCPUMonitor) numCPU() float64 {
|
|
return p.nCPU
|
|
}
|
|
|
|
type cpuInfoGetterV1 struct{}
|
|
|
|
func newCpuInfoGetterV1() cpuInfoGetter {
|
|
return &cpuInfoGetterV1{}
|
|
}
|
|
|
|
func (cg *cpuInfoGetterV1) getTotalCPUTime() (int64, error) {
|
|
b, err := os.ReadFile(cpuStatsPathV1)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Skip the trailing EOL
|
|
i, err := strconv.ParseInt(string(b[:len(b)-1]), 10, 64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return i, nil
|
|
}
|
|
|
|
func (cg *cpuInfoGetterV1) numCPU() (float64, error) {
|
|
quota, err := readIntFromFile(numCPUPathV1Quota)
|
|
if err != nil {
|
|
if errors.Is(err, os.ErrNotExist) {
|
|
// File may not exist in case of no quota
|
|
return float64(runtime.NumCPU()), nil
|
|
}
|
|
|
|
return 0, err
|
|
}
|
|
|
|
if quota < 0 {
|
|
// default
|
|
return float64(runtime.NumCPU()), nil
|
|
}
|
|
|
|
period, err := readIntFromFile(numCPUPathV1Period)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
if period <= 0 {
|
|
// default
|
|
return float64(runtime.NumCPU()), nil
|
|
}
|
|
|
|
return float64(quota) / float64(period), nil
|
|
}
|
|
|
|
type cpuInfoGetterV2 struct{}
|
|
|
|
func newCpuInfoGetterV2() cpuInfoGetter {
|
|
return &cpuInfoGetterV2{}
|
|
}
|
|
|
|
func (cg *cpuInfoGetterV2) getTotalCPUTime() (int64, error) {
|
|
b, err := os.ReadFile(cpuStatsPathV2)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
m := usageRegex.FindSubmatch(b)
|
|
if len(m) <= 1 {
|
|
return 0, errors.New("could not parse cpu stats")
|
|
}
|
|
|
|
i, err := strconv.ParseInt(string(m[1]), 10, 64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Caller expects time in ns
|
|
return i * 1000, nil
|
|
}
|
|
|
|
func (cg *cpuInfoGetterV2) numCPU() (float64, error) {
|
|
b, err := os.ReadFile(numCPUPathV2)
|
|
if err != nil {
|
|
if errors.Is(err, os.ErrNotExist) {
|
|
// File may not exist in case of no quota
|
|
return float64(runtime.NumCPU()), nil
|
|
}
|
|
return 0, err
|
|
}
|
|
|
|
s := strings.TrimSuffix(string(b), "\n")
|
|
|
|
m := strings.Split(s, " ")
|
|
if len(m) <= 1 {
|
|
return 0, errors.New("could not parse cpu stats")
|
|
}
|
|
|
|
if m[0] == "max" {
|
|
// No quota
|
|
return float64(runtime.NumCPU()), nil
|
|
}
|
|
|
|
quota, err := strconv.ParseInt(string(m[0]), 10, 64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
period, err := strconv.ParseInt(string(m[1]), 10, 64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return float64(quota) / float64(period), nil
|
|
}
|
|
|
|
func fileExists(path string) (bool, error) {
|
|
_, err := os.Lstat(path)
|
|
switch {
|
|
case err == nil:
|
|
return true, nil
|
|
case errors.Is(err, os.ErrNotExist):
|
|
return false, nil
|
|
default:
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
func readIntFromFile(filename string) (int, error) {
|
|
b, err := os.ReadFile(filename)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Remove trailing new line if any
|
|
s := strings.TrimSuffix(string(b), "\n")
|
|
|
|
// Remove trailing space if any
|
|
s = strings.TrimSuffix(s, " ")
|
|
|
|
return strconv.Atoi(s)
|
|
}
|