first commit
This commit is contained in:
		
						commit
						79c590957c
					
				
							
								
								
									
										121
									
								
								base62.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								base62.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,121 @@ | |||||||
|  | package base62 | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"errors" | ||||||
|  | 
 | ||||||
|  | 	"github.com/samber/lo" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type Base62 struct { | ||||||
|  | 	characters    []byte       // base62 编码规则
 | ||||||
|  | 	charactersMap map[byte]int // base62 编码规则
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var base62 *Base62 | ||||||
|  | 
 | ||||||
|  | func init() { | ||||||
|  | 	base62 = &Base62{} | ||||||
|  | 	base62.SetCharacters([]byte("0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz")) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func GetInstance() *Base62 { | ||||||
|  | 	return base62 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (base62 *Base62) SetCharacters(characters []byte) (*Base62, error) { | ||||||
|  | 	characters = lo.FindUniques(characters) | ||||||
|  | 	charactersMap := make(map[byte]int) | ||||||
|  | 
 | ||||||
|  | 	if len(characters) != 62 { | ||||||
|  | 		return nil, errors.New("Invalid characters") | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for i, v := range characters { | ||||||
|  | 		charactersMap[v] = i | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	base62.characters = characters | ||||||
|  | 	base62.charactersMap = charactersMap | ||||||
|  | 
 | ||||||
|  | 	return base62, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Encode function receives a byte slice and encodes it to a string using the alphabet provided
 | ||||||
|  | func (e *Base62) Encode(source []byte) string { | ||||||
|  | 	if len(source) == 0 { | ||||||
|  | 		return "" | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	digits := []int{0} | ||||||
|  | 
 | ||||||
|  | 	for i := 0; i < len(source); i++ { | ||||||
|  | 		carry := int(source[i]) | ||||||
|  | 
 | ||||||
|  | 		for j := 0; j < len(digits); j++ { | ||||||
|  | 			carry += digits[j] << 8 | ||||||
|  | 			digits[j] = carry % 62 | ||||||
|  | 			carry = carry / 62 | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		for carry > 0 { | ||||||
|  | 			digits = append(digits, carry%62) | ||||||
|  | 			carry = carry / 62 | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var res bytes.Buffer | ||||||
|  | 
 | ||||||
|  | 	for k := 0; source[k] == 0 && k < len(source)-1; k++ { | ||||||
|  | 		res.WriteByte(base62.characters[0]) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for q := len(digits) - 1; q >= 0; q-- { | ||||||
|  | 		res.WriteByte(base62.characters[digits[q]]) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return res.String() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Decode function decodes a string previously obtained from Encode, using the same alphabet and returns a byte slice
 | ||||||
|  | // In case the input is not valid an arror will be returned
 | ||||||
|  | func (base62 *Base62) Decode(source string) ([]byte, error) { | ||||||
|  | 	if len(source) == 0 { | ||||||
|  | 		return []byte{}, nil | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	runes := []byte(source) | ||||||
|  | 
 | ||||||
|  | 	bytes := []byte{0} | ||||||
|  | 	for i := 0; i < len(runes); i++ { | ||||||
|  | 		value, ok := base62.charactersMap[runes[i]] | ||||||
|  | 
 | ||||||
|  | 		if !ok { | ||||||
|  | 			return nil, errors.New("Non Base Character") | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		carry := int(value) | ||||||
|  | 
 | ||||||
|  | 		for j := 0; j < len(bytes); j++ { | ||||||
|  | 			carry += int(bytes[j]) * 62 | ||||||
|  | 			bytes[j] = byte(carry & 0xff) | ||||||
|  | 			carry >>= 8 | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		for carry > 0 { | ||||||
|  | 			bytes = append(bytes, byte(carry&0xff)) | ||||||
|  | 			carry >>= 8 | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for k := 0; runes[k] == base62.characters[0] && k < len(runes)-1; k++ { | ||||||
|  | 		bytes = append(bytes, 0) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// Reverse bytes
 | ||||||
|  | 	for i, j := 0, len(bytes)-1; i < j; i, j = i+1, j-1 { | ||||||
|  | 		bytes[i], bytes[j] = bytes[j], bytes[i] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return bytes, nil | ||||||
|  | } | ||||||
							
								
								
									
										13
									
								
								base62_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								base62_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | |||||||
|  | package base62 | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"testing" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestEncode(t *testing.T) { | ||||||
|  | 	e := GetInstance().Encode([]byte("问问哇啊啊")) | ||||||
|  | 	t.Log(e) | ||||||
|  | 
 | ||||||
|  | 	d, err := GetInstance().Decode(e) | ||||||
|  | 	t.Log(string(d), err) | ||||||
|  | } | ||||||
							
								
								
									
										7
									
								
								go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								go.mod
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | module git.fsdpf.net/go/base62 | ||||||
|  | 
 | ||||||
|  | go 1.18 | ||||||
|  | 
 | ||||||
|  | require github.com/samber/lo v1.38.1 | ||||||
|  | 
 | ||||||
|  | require golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 // indirect | ||||||
							
								
								
									
										4
									
								
								go.sum
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								go.sum
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | github.com/samber/lo v1.38.1 h1:j2XEAqXKb09Am4ebOg31SpvzUTTs6EN3VfgeLUhPdXM= | ||||||
|  | github.com/samber/lo v1.38.1/go.mod h1:+m/ZKRl6ClXCE2Lgf3MsQlWfh4bn1bz6CXEOxnEXnEA= | ||||||
|  | golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17 h1:3MTrJm4PyNL9NBqvYDSj3DHl46qQakyfqfWo4jgfaEM= | ||||||
|  | golang.org/x/exp v0.0.0-20220303212507-bbda1eaf7a17/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE= | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user