|
|
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 "" }
|