mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2025-01-03 15:38:00 +08:00
183 lines
3.5 KiB
Go
183 lines
3.5 KiB
Go
package needle
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
)
|
|
|
|
const (
|
|
// stored unit types
|
|
Empty byte = iota
|
|
Minute
|
|
Hour
|
|
Day
|
|
Week
|
|
Month
|
|
Year
|
|
)
|
|
|
|
type TTL struct {
|
|
Count byte
|
|
Unit byte
|
|
}
|
|
|
|
var EMPTY_TTL = &TTL{}
|
|
|
|
// translate a readable ttl to internal ttl
|
|
// Supports format example:
|
|
// 3m: 3 minutes
|
|
// 4h: 4 hours
|
|
// 5d: 5 days
|
|
// 6w: 6 weeks
|
|
// 7M: 7 months
|
|
// 8y: 8 years
|
|
func ReadTTL(ttlString string) (*TTL, error) {
|
|
if ttlString == "" {
|
|
return EMPTY_TTL, nil
|
|
}
|
|
ttlBytes := []byte(ttlString)
|
|
unitByte := ttlBytes[len(ttlBytes)-1]
|
|
countBytes := ttlBytes[0 : len(ttlBytes)-1]
|
|
if '0' <= unitByte && unitByte <= '9' {
|
|
countBytes = ttlBytes
|
|
unitByte = 'm'
|
|
}
|
|
count, err := strconv.Atoi(string(countBytes))
|
|
unit := toStoredByte(unitByte)
|
|
return &TTL{Count: byte(count), Unit: unit}, err
|
|
}
|
|
|
|
// read stored bytes to a ttl
|
|
func LoadTTLFromBytes(input []byte) (t *TTL) {
|
|
if input[0] == 0 && input[1] == 0 {
|
|
return EMPTY_TTL
|
|
}
|
|
return &TTL{Count: input[0], Unit: input[1]}
|
|
}
|
|
|
|
// read stored bytes to a ttl
|
|
func LoadTTLFromUint32(ttl uint32) (t *TTL) {
|
|
input := make([]byte, 2)
|
|
input[1] = byte(ttl)
|
|
input[0] = byte(ttl >> 8)
|
|
return LoadTTLFromBytes(input)
|
|
}
|
|
|
|
// save stored bytes to an output with 2 bytes
|
|
func (t *TTL) ToBytes(output []byte) {
|
|
output[0] = t.Count
|
|
output[1] = t.Unit
|
|
}
|
|
|
|
func (t *TTL) ToUint32() (output uint32) {
|
|
if t == nil || t.Count == 0 {
|
|
return 0
|
|
}
|
|
output = uint32(t.Count) << 8
|
|
output += uint32(t.Unit)
|
|
return output
|
|
}
|
|
|
|
func (t *TTL) String() string {
|
|
if t == nil || t.Count == 0 {
|
|
return ""
|
|
}
|
|
if t.Unit == Empty {
|
|
return ""
|
|
}
|
|
countString := strconv.Itoa(int(t.Count))
|
|
switch t.Unit {
|
|
case Minute:
|
|
return countString + "m"
|
|
case Hour:
|
|
return countString + "h"
|
|
case Day:
|
|
return countString + "d"
|
|
case Week:
|
|
return countString + "w"
|
|
case Month:
|
|
return countString + "M"
|
|
case Year:
|
|
return countString + "y"
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func toStoredByte(readableUnitByte byte) byte {
|
|
switch readableUnitByte {
|
|
case 'm':
|
|
return Minute
|
|
case 'h':
|
|
return Hour
|
|
case 'd':
|
|
return Day
|
|
case 'w':
|
|
return Week
|
|
case 'M':
|
|
return Month
|
|
case 'y':
|
|
return Year
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func (t TTL) Minutes() uint32 {
|
|
switch t.Unit {
|
|
case Empty:
|
|
return 0
|
|
case Minute:
|
|
return uint32(t.Count)
|
|
case Hour:
|
|
return uint32(t.Count) * 60
|
|
case Day:
|
|
return uint32(t.Count) * 60 * 24
|
|
case Week:
|
|
return uint32(t.Count) * 60 * 24 * 7
|
|
case Month:
|
|
return uint32(t.Count) * 60 * 24 * 30
|
|
case Year:
|
|
return uint32(t.Count) * 60 * 24 * 365
|
|
}
|
|
return 0
|
|
}
|
|
|
|
func SecondsToTTL(seconds int32) string {
|
|
if seconds == 0 {
|
|
return ""
|
|
}
|
|
if seconds%(3600*24*365) == 0 && seconds/(3600*24*365) < 256 {
|
|
return fmt.Sprintf("%dy", seconds/(3600*24*365))
|
|
}
|
|
if seconds%(3600*24*30) == 0 && seconds/(3600*24*30) < 256 {
|
|
return fmt.Sprintf("%dM", seconds/(3600*24*30))
|
|
}
|
|
if seconds%(3600*24*7) == 0 && seconds/(3600*24*7) < 256 {
|
|
return fmt.Sprintf("%dw", seconds/(3600*24*7))
|
|
}
|
|
if seconds%(3600*24) == 0 && seconds/(3600*24) < 256 {
|
|
return fmt.Sprintf("%dd", seconds/(3600*24))
|
|
}
|
|
if seconds%(3600) == 0 && seconds/(3600) < 256 {
|
|
return fmt.Sprintf("%dh", seconds/(3600))
|
|
}
|
|
if seconds/60 < 256 {
|
|
return fmt.Sprintf("%dm", seconds/60)
|
|
}
|
|
if seconds/(3600) < 256 {
|
|
return fmt.Sprintf("%dh", seconds/(3600))
|
|
}
|
|
if seconds/(3600*24) < 256 {
|
|
return fmt.Sprintf("%dd", seconds/(3600*24))
|
|
}
|
|
if seconds/(3600*24*7) < 256 {
|
|
return fmt.Sprintf("%dw", seconds/(3600*24*7))
|
|
}
|
|
if seconds/(3600*24*30) < 256 {
|
|
return fmt.Sprintf("%dM", seconds/(3600*24*30))
|
|
}
|
|
if seconds/(3600*24*365) < 256 {
|
|
return fmt.Sprintf("%dy", seconds/(3600*24*365))
|
|
}
|
|
return ""
|
|
}
|