2020-11-07 05:14:29 +00:00
|
|
|
package system
|
|
|
|
|
|
|
|
import (
|
2020-12-26 01:04:18 +00:00
|
|
|
"bufio"
|
|
|
|
"bytes"
|
2020-12-25 19:21:09 +00:00
|
|
|
"context"
|
2020-12-26 01:04:18 +00:00
|
|
|
"encoding/json"
|
2020-12-25 21:32:41 +00:00
|
|
|
"fmt"
|
2020-12-26 01:04:18 +00:00
|
|
|
"io"
|
2021-01-13 05:14:57 +00:00
|
|
|
"strconv"
|
2020-12-26 01:04:18 +00:00
|
|
|
"strings"
|
2020-11-07 06:22:33 +00:00
|
|
|
"sync"
|
2020-12-25 19:21:09 +00:00
|
|
|
"time"
|
2021-01-13 05:14:57 +00:00
|
|
|
|
|
|
|
"emperror.dev/errors"
|
2020-11-07 05:14:29 +00:00
|
|
|
)
|
|
|
|
|
2020-12-26 01:05:01 +00:00
|
|
|
var cr = []byte(" \r")
|
|
|
|
var crr = []byte("\r\n")
|
|
|
|
|
2021-01-13 05:14:57 +00:00
|
|
|
// FirstNotEmpty returns the first string passed in that is not an empty value.
|
|
|
|
func FirstNotEmpty(v ...string) string {
|
|
|
|
for _, val := range v {
|
|
|
|
if val != "" {
|
|
|
|
return val
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func MustInt(v string) int {
|
|
|
|
i, err := strconv.Atoi(v)
|
|
|
|
if err != nil {
|
|
|
|
panic(errors.Wrap(err, "system/utils: could not parse int"))
|
|
|
|
}
|
|
|
|
return i
|
|
|
|
}
|
|
|
|
|
2020-12-26 01:05:01 +00:00
|
|
|
func ScanReader(r io.Reader, callback func(line string)) error {
|
|
|
|
br := bufio.NewReader(r)
|
|
|
|
// Avoid constantly re-allocating memory when we're flooding lines through this
|
|
|
|
// function by using the same buffer for the duration of the call and just truncating
|
|
|
|
// the value back to 0 every loop.
|
|
|
|
var str strings.Builder
|
|
|
|
for {
|
|
|
|
str.Reset()
|
|
|
|
var err error
|
|
|
|
var line []byte
|
|
|
|
var isPrefix bool
|
|
|
|
|
|
|
|
for {
|
|
|
|
// Read the line and write it to the buffer.
|
|
|
|
line, isPrefix, err = br.ReadLine()
|
|
|
|
// Certain games like Minecraft output absolutely random carriage returns in the output seemingly
|
|
|
|
// in line with that it thinks is the terminal size. Those returns break a lot of output handling,
|
|
|
|
// so we'll just replace them with proper new-lines and then split it later and send each line as
|
|
|
|
// its own event in the response.
|
|
|
|
str.Write(bytes.Replace(line, cr, crr, -1))
|
|
|
|
// Finish this loop and begin outputting the line if there is no prefix (the line fit into
|
|
|
|
// the default buffer), or if we hit the end of the line.
|
|
|
|
if !isPrefix || err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
// If we encountered an error with something in ReadLine that was not an EOF just abort
|
|
|
|
// the entire process here.
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Publish the line for this loop. Break on new-line characters so every line is sent as a single
|
|
|
|
// output event, otherwise you get funky handling in the browser console.
|
|
|
|
for _, line := range strings.Split(str.String(), "\r\n") {
|
|
|
|
callback(line)
|
|
|
|
}
|
|
|
|
// If the error we got previously that lead to the line being output is an io.EOF we want to
|
|
|
|
// exit the entire looping process.
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-12-25 19:21:09 +00:00
|
|
|
// Runs a given work function every "d" duration until the provided context is canceled.
|
|
|
|
func Every(ctx context.Context, d time.Duration, work func(t time.Time)) {
|
|
|
|
ticker := time.NewTicker(d)
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
ticker.Stop()
|
|
|
|
return
|
|
|
|
case t := <-ticker.C:
|
|
|
|
work(t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2020-12-25 21:32:41 +00:00
|
|
|
func FormatBytes(b int64) string {
|
|
|
|
if b < 1024 {
|
|
|
|
return fmt.Sprintf("%d B", b)
|
|
|
|
}
|
|
|
|
div, exp := int64(1024), 0
|
|
|
|
for n := b / 1024; n >= 1024; n /= 1024 {
|
|
|
|
div *= 1024
|
|
|
|
exp++
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%.1f %ciB", float64(b)/float64(div), "KMGTPE"[exp])
|
|
|
|
}
|
|
|
|
|
2020-11-07 05:14:29 +00:00
|
|
|
type AtomicBool struct {
|
2020-12-26 01:04:18 +00:00
|
|
|
v bool
|
|
|
|
mu sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewAtomicBool(v bool) *AtomicBool {
|
|
|
|
return &AtomicBool{v: v}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ab *AtomicBool) Store(v bool) {
|
|
|
|
ab.mu.Lock()
|
|
|
|
ab.v = v
|
|
|
|
ab.mu.Unlock()
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2020-12-26 01:04:18 +00:00
|
|
|
// Stores the value "v" if the current value stored in the AtomicBool is the opposite
|
|
|
|
// boolean value. If successfully swapped, the response is "true", otherwise "false"
|
|
|
|
// is returned.
|
|
|
|
func (ab *AtomicBool) SwapIf(v bool) bool {
|
|
|
|
ab.mu.Lock()
|
|
|
|
defer ab.mu.Unlock()
|
|
|
|
if ab.v != v {
|
|
|
|
ab.v = v
|
|
|
|
return true
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
2020-12-26 01:04:18 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ab *AtomicBool) Load() bool {
|
|
|
|
ab.mu.RLock()
|
|
|
|
defer ab.mu.RUnlock()
|
|
|
|
return ab.v
|
|
|
|
}
|
2020-11-07 05:14:29 +00:00
|
|
|
|
2020-12-26 01:04:18 +00:00
|
|
|
func (ab *AtomicBool) UnmarshalJSON(b []byte) error {
|
|
|
|
ab.mu.Lock()
|
|
|
|
defer ab.mu.Unlock()
|
|
|
|
return json.Unmarshal(b, &ab.v)
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2020-12-26 01:04:18 +00:00
|
|
|
func (ab *AtomicBool) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(ab.Load())
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AtomicString allows for reading/writing to a given struct field without having to worry
|
|
|
|
// about a potential race condition scenario. Under the hood it uses a simple sync.RWMutex
|
|
|
|
// to control access to the value.
|
|
|
|
type AtomicString struct {
|
2020-11-07 06:22:33 +00:00
|
|
|
v string
|
|
|
|
mu sync.RWMutex
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2020-12-26 01:04:18 +00:00
|
|
|
func NewAtomicString(v string) *AtomicString {
|
|
|
|
return &AtomicString{v: v}
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Stores the string value passed atomically.
|
|
|
|
func (as *AtomicString) Store(v string) {
|
2020-11-07 06:22:33 +00:00
|
|
|
as.mu.Lock()
|
|
|
|
as.v = v
|
|
|
|
as.mu.Unlock()
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Loads the string value and returns it.
|
|
|
|
func (as *AtomicString) Load() string {
|
2020-11-07 06:22:33 +00:00
|
|
|
as.mu.RLock()
|
|
|
|
defer as.mu.RUnlock()
|
|
|
|
return as.v
|
|
|
|
}
|
|
|
|
|
2020-12-26 01:04:18 +00:00
|
|
|
func (as *AtomicString) UnmarshalJSON(b []byte) error {
|
|
|
|
as.mu.Lock()
|
|
|
|
defer as.mu.Unlock()
|
|
|
|
return json.Unmarshal(b, &as.v)
|
2020-11-07 06:22:33 +00:00
|
|
|
}
|
|
|
|
|
2020-12-26 01:04:18 +00:00
|
|
|
func (as *AtomicString) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(as.Load())
|
2020-11-07 05:14:29 +00:00
|
|
|
}
|