-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
- Loading branch information
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,74 @@ | ||
# Go SafeCast | ||
|
||
## Origin of this project | ||
|
||
In Go, integer type conversion can lead to unexpected behavior and errors if not handled carefully. | ||
|
||
Issues can happen when converting between signed and unsigned integers, or when converting to a smaller integer type. | ||
|
||
The gosec project raised this to my attention when the gosec [G115 rule was added](https://github.com/securego/gosec/pull/1149) | ||
|
||
> G115: Potential integer overflow when converting between integer types | ||
This issue was way more complex than expected, and required multiple fixes. | ||
|
||
## Example | ||
|
||
This code seems OK | ||
|
||
```go | ||
package main | ||
|
||
import ( | ||
"fmt" | ||
Check failure on line 23 in README.md GitHub Actions / ci-lintHard tabs
|
||
) | ||
|
||
func main() { | ||
var a uint64 | ||
Check failure on line 27 in README.md GitHub Actions / ci-lintHard tabs
|
||
a = 42 | ||
Check failure on line 28 in README.md GitHub Actions / ci-lintHard tabs
|
||
b := int32(a) | ||
Check failure on line 29 in README.md GitHub Actions / ci-lintHard tabs
|
||
fmt.Println(b) // 42 | ||
Check failure on line 30 in README.md GitHub Actions / ci-lintHard tabs
|
||
} | ||
``` | ||
|
||
But the conversion to int32 will behave differently depending on the value | ||
|
||
```go | ||
package main | ||
|
||
import ( | ||
"fmt" | ||
Check failure on line 40 in README.md GitHub Actions / ci-lintHard tabs
|
||
) | ||
|
||
func main() { | ||
var a uint64 | ||
Check failure on line 44 in README.md GitHub Actions / ci-lintHard tabs
|
||
a = 2147483647 | ||
Check failure on line 45 in README.md GitHub Actions / ci-lintHard tabs
|
||
b := int32(a) | ||
Check failure on line 46 in README.md GitHub Actions / ci-lintHard tabs
|
||
fmt.Println(b) // 2147483647 | ||
Check failure on line 47 in README.md GitHub Actions / ci-lintHard tabs
|
||
|
||
a = 2147483647 + 1 | ||
b = int32(a) | ||
fmt.Println(b) // -2147483648 Stack overflow | ||
|
||
c := -1 | ||
d := uint32(c) | ||
fmt.Println(d) // 4294967295 | ||
} | ||
``` | ||
|
||
https://go.dev/play/p/9PRWI7e0x1T | ||
|
||
|
||
## Motivation | ||
|
||
The gosec G115 will now report issues in a lot of project. | ||
|
||
Some libraries existed (See [alternatives](#alternatives) section), but they were not able to cover all the use cases. | ||
|
||
## Alternatives | ||
|
||
- [github.com/rung/go-safecast](https://github.com/rung/go-safecast): | ||
Unmaintained, not architecture agnostic, do not support uint -> int conversion | ||
|
||
- [github.com/cybergarage/go-safecast](https://github.com/cybergarage/go-safecast) | ||
Work with pointer like json.Marshall |