generated from kubernetes/kubernetes-template-project
-
Notifications
You must be signed in to change notification settings - Fork 521
/
Copy pathduration.go
121 lines (92 loc) · 3.15 KB
/
duration.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
/*
Copyright 2024 The Kubernetes Authors.
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.
*/
package main
import (
"errors"
"fmt"
"regexp"
"time"
)
var re = regexp.MustCompile(`^([0-9]{1,5}(h|m|s|ms)){1,4}$`)
func ParseDuration(s string) (*time.Duration, error) {
/*
parseDuration parses a GEP-2257 Duration format to a time object
Valid date units in time.Duration are "ns", "us" (or "µs"), "ms", "s", "m", "h"
Valid date units according to GEP-2257 are "h", "m", "s", "ms"
input: string
output: time.Duration
See https://gateway-api.sigs.k8s.io/geps/gep-2257/ for more details.
*/
if !re.MatchString(s) {
return nil, errors.New("Invalid duration format")
}
parsedTime, err := time.ParseDuration(s)
if err != nil {
return nil, err
}
return &parsedTime, nil
}
const maxDuration = 99999*time.Hour + 59*time.Minute + 59*time.Second + 999*time.Millisecond
func FormatDuration(duration time.Duration) (string, error) {
/*
formatDuration formats a time object to GEP-2257 Duration format to a GEP-2257 Duration Format
The time format from GEP-2257 must match the regex
"^([1-9]{1,5}(h|m|s|ms)){1,4}$"
A time.Duration allows for negative time, floating points, and allow for zero units
For example, -4h, 4.5h, and 4h0m0s are all valid in the golang time package
See https://gateway-api.sigs.k8s.io/geps/gep-2257/ for more details.
Input: time.Duration
Returns: string or error if duration cannot be expressed as a GEP-2257 Duration format.
*/
if duration == 0 {
return "0s", nil
}
// check if a negative value
if duration < 0 {
return "", errors.New("Invalid duration format. Cannot have negative durations")
}
// check for the maximum value allowed to be expressed
if duration > maxDuration {
return "", errors.New("Invalid duration format. Duration larger than maximum expression allowed in GEP-2257")
}
// time.Duration allows for floating point ms, which is not allowed in GEP-2257
durationMicroseconds := duration.Microseconds()
if durationMicroseconds%1000 != 0 {
return "", errors.New("Cannot express sub-milliseconds precision in GEP-2257")
}
output := ""
seconds := int(duration.Seconds())
// calculating the hours
hours := seconds / 3600
if hours > 0 {
output += fmt.Sprintf("%dh", hours)
seconds -= hours * 3600
}
// calculating the minutes
minutes := seconds / 60
if minutes > 0 {
output += fmt.Sprintf("%dm", minutes)
seconds -= minutes * 60
}
// calculating the seconds
if seconds > 0 {
output += fmt.Sprintf("%ds", seconds)
}
// calculating the milliseconds
durationMilliseconds := durationMicroseconds / 1000
ms := durationMilliseconds % 1000
if ms != 0 {
output += fmt.Sprintf("%dms", ms)
}
return output, nil
}