mirror of
https://github.com/wneessen/go-mail.git
synced 2024-11-23 14:10:50 +01:00
Winni Neessen
514e68be54
Changed the abbreviation 'idx' to 'i' in the 'random.go' file. This change makes the variable roles much clearer and aligns the naming with Go's best practice, enhancing readability and maintainability.
74 lines
2 KiB
Go
74 lines
2 KiB
Go
// SPDX-FileCopyrightText: 2022-2023 The go-mail Authors
|
|
//
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package mail
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"math/big"
|
|
"strings"
|
|
)
|
|
|
|
// Range of characters for the secure string generation
|
|
const cr = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890"
|
|
|
|
// Bitmask sizes for the string generators (based on 93 chars total)
|
|
const (
|
|
letterIdxBits = 7 // 7 bits to represent a letter index
|
|
letterIdxMask = 1<<letterIdxBits - 1 // All 1-bits, as many as letterIdxBits
|
|
letterIdxMax = 63 / letterIdxBits // # of letter indices fitting in 63 bits
|
|
)
|
|
|
|
// randomStringSecure returns a random, string of length characters. This method uses the
|
|
// crypto/random package and therfore is cryptographically secure
|
|
func randomStringSecure(length int) (string, error) {
|
|
randString := strings.Builder{}
|
|
randString.Grow(length)
|
|
charRangeLength := len(cr)
|
|
|
|
randPool := make([]byte, 8)
|
|
_, err := rand.Read(randPool)
|
|
if err != nil {
|
|
return randString.String(), err
|
|
}
|
|
for idx, char, rest := length-1, binary.BigEndian.Uint64(randPool), letterIdxMax; idx >= 0; {
|
|
if rest == 0 {
|
|
_, err = rand.Read(randPool)
|
|
if err != nil {
|
|
return randString.String(), err
|
|
}
|
|
char, rest = binary.BigEndian.Uint64(randPool), letterIdxMax
|
|
}
|
|
if i := int(char & letterIdxMask); i < charRangeLength {
|
|
randString.WriteByte(cr[i])
|
|
idx--
|
|
}
|
|
char >>= letterIdxBits
|
|
rest--
|
|
}
|
|
|
|
return randString.String(), nil
|
|
}
|
|
|
|
// randNum returns a random number with a maximum value of length
|
|
func randNum(length int) (int, error) {
|
|
if length <= 0 {
|
|
return 0, fmt.Errorf("provided number is <= 0: %d", length)
|
|
}
|
|
length64 := big.NewInt(int64(length))
|
|
if !length64.IsUint64() {
|
|
return 0, fmt.Errorf("big.NewInt() generation returned negative value: %d", length64)
|
|
}
|
|
randNum64, err := rand.Int(rand.Reader, length64)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
randomNum := int(randNum64.Int64())
|
|
if randomNum < 0 {
|
|
return 0, fmt.Errorf("generated random number does not fit as int64: %d", randNum64)
|
|
}
|
|
return randomNum, nil
|
|
}
|