Skip to content

Commit cbee36f

Browse files
committed
Add utility function to convert uint64 to uint32 securely
1 parent 45b7622 commit cbee36f

File tree

2 files changed

+75
-0
lines changed

2 files changed

+75
-0
lines changed

types/conversion.go

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
// Copyright 2024 Red Hat, Inc
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// http://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
package types
16+
17+
import (
18+
"errors"
19+
)
20+
21+
// Uint64ToUint32 safely converts a uint64 into a uint32 without overflow
22+
func Uint64ToUint32(v uint64) (uint32, error) {
23+
// Check for overflow without casting
24+
if v > uint64(^uint32(0)) { // ^uint32(0) is the maximum uint32 value (4,294,967,295)
25+
return 0, errors.New("value exceeds uint32 range")
26+
}
27+
28+
return uint32(v), nil // #nosec G103: safe after bounds check
29+
}

types/conversion_test.go

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
// Copyright 2024 Red Hat, Inc
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// http://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
package types
15+
16+
import (
17+
"testing"
18+
)
19+
20+
func TestUint64ToUint32(t *testing.T) {
21+
tests := []struct {
22+
name string
23+
input uint64
24+
want uint32
25+
wantErr bool
26+
}{
27+
{"Zero", 0, 0, false},
28+
{"Max uint32", 4294967295, 4294967295, false},
29+
{"Overflow", 4294967296, 0, true},
30+
{"Large overflow", 18446744073709551615, 0, true},
31+
{"Mid-range value", 2147483648, 2147483648, false},
32+
}
33+
34+
for _, tt := range tests {
35+
t.Run(tt.name, func(t *testing.T) {
36+
got, err := Uint64ToUint32(tt.input)
37+
if (err != nil) != tt.wantErr {
38+
t.Errorf("Uint64ToUint32() error = %v, wantErr %v", err, tt.wantErr)
39+
return
40+
}
41+
if got != tt.want {
42+
t.Errorf("Uint64ToUint32() = %v, want %v", got, tt.want)
43+
}
44+
})
45+
}
46+
}

0 commit comments

Comments
 (0)