mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-31 09:32:25 -05:00 
			
		
		
		
	
		
			
	
	
		
			296 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
		
		
			
		
	
	
			296 lines
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
|  | // Copyright 2011 The Go Authors. All rights reserved. | ||
|  | // Use of this source code is governed by a BSD-style | ||
|  | // license that can be found in the LICENSE file. | ||
|  | 
 | ||
|  | // Package bcrypt implements Provos and Mazières's bcrypt adaptive hashing | ||
|  | // algorithm. See http://www.usenix.org/event/usenix99/provos/provos.pdf | ||
|  | package bcrypt // import "golang.org/x/crypto/bcrypt" | ||
|  | 
 | ||
|  | // The code is a port of Provos and Mazières's C implementation. | ||
|  | import ( | ||
|  | 	"crypto/rand" | ||
|  | 	"crypto/subtle" | ||
|  | 	"errors" | ||
|  | 	"fmt" | ||
|  | 	"io" | ||
|  | 	"strconv" | ||
|  | 
 | ||
|  | 	"golang.org/x/crypto/blowfish" | ||
|  | ) | ||
|  | 
 | ||
|  | const ( | ||
|  | 	MinCost     int = 4  // the minimum allowable cost as passed in to GenerateFromPassword | ||
|  | 	MaxCost     int = 31 // the maximum allowable cost as passed in to GenerateFromPassword | ||
|  | 	DefaultCost int = 10 // the cost that will actually be set if a cost below MinCost is passed into GenerateFromPassword | ||
|  | ) | ||
|  | 
 | ||
|  | // The error returned from CompareHashAndPassword when a password and hash do | ||
|  | // not match. | ||
|  | var ErrMismatchedHashAndPassword = errors.New("crypto/bcrypt: hashedPassword is not the hash of the given password") | ||
|  | 
 | ||
|  | // The error returned from CompareHashAndPassword when a hash is too short to | ||
|  | // be a bcrypt hash. | ||
|  | var ErrHashTooShort = errors.New("crypto/bcrypt: hashedSecret too short to be a bcrypted password") | ||
|  | 
 | ||
|  | // The error returned from CompareHashAndPassword when a hash was created with | ||
|  | // a bcrypt algorithm newer than this implementation. | ||
|  | type HashVersionTooNewError byte | ||
|  | 
 | ||
|  | func (hv HashVersionTooNewError) Error() string { | ||
|  | 	return fmt.Sprintf("crypto/bcrypt: bcrypt algorithm version '%c' requested is newer than current version '%c'", byte(hv), majorVersion) | ||
|  | } | ||
|  | 
 | ||
|  | // The error returned from CompareHashAndPassword when a hash starts with something other than '$' | ||
|  | type InvalidHashPrefixError byte | ||
|  | 
 | ||
|  | func (ih InvalidHashPrefixError) Error() string { | ||
|  | 	return fmt.Sprintf("crypto/bcrypt: bcrypt hashes must start with '$', but hashedSecret started with '%c'", byte(ih)) | ||
|  | } | ||
|  | 
 | ||
|  | type InvalidCostError int | ||
|  | 
 | ||
|  | func (ic InvalidCostError) Error() string { | ||
|  | 	return fmt.Sprintf("crypto/bcrypt: cost %d is outside allowed range (%d,%d)", int(ic), int(MinCost), int(MaxCost)) | ||
|  | } | ||
|  | 
 | ||
|  | const ( | ||
|  | 	majorVersion       = '2' | ||
|  | 	minorVersion       = 'a' | ||
|  | 	maxSaltSize        = 16 | ||
|  | 	maxCryptedHashSize = 23 | ||
|  | 	encodedSaltSize    = 22 | ||
|  | 	encodedHashSize    = 31 | ||
|  | 	minHashSize        = 59 | ||
|  | ) | ||
|  | 
 | ||
|  | // magicCipherData is an IV for the 64 Blowfish encryption calls in | ||
|  | // bcrypt(). It's the string "OrpheanBeholderScryDoubt" in big-endian bytes. | ||
|  | var magicCipherData = []byte{ | ||
|  | 	0x4f, 0x72, 0x70, 0x68, | ||
|  | 	0x65, 0x61, 0x6e, 0x42, | ||
|  | 	0x65, 0x68, 0x6f, 0x6c, | ||
|  | 	0x64, 0x65, 0x72, 0x53, | ||
|  | 	0x63, 0x72, 0x79, 0x44, | ||
|  | 	0x6f, 0x75, 0x62, 0x74, | ||
|  | } | ||
|  | 
 | ||
|  | type hashed struct { | ||
|  | 	hash  []byte | ||
|  | 	salt  []byte | ||
|  | 	cost  int // allowed range is MinCost to MaxCost | ||
|  | 	major byte | ||
|  | 	minor byte | ||
|  | } | ||
|  | 
 | ||
|  | // GenerateFromPassword returns the bcrypt hash of the password at the given | ||
|  | // cost. If the cost given is less than MinCost, the cost will be set to | ||
|  | // DefaultCost, instead. Use CompareHashAndPassword, as defined in this package, | ||
|  | // to compare the returned hashed password with its cleartext version. | ||
|  | func GenerateFromPassword(password []byte, cost int) ([]byte, error) { | ||
|  | 	p, err := newFromPassword(password, cost) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 	return p.Hash(), nil | ||
|  | } | ||
|  | 
 | ||
|  | // CompareHashAndPassword compares a bcrypt hashed password with its possible | ||
|  | // plaintext equivalent. Returns nil on success, or an error on failure. | ||
|  | func CompareHashAndPassword(hashedPassword, password []byte) error { | ||
|  | 	p, err := newFromHash(hashedPassword) | ||
|  | 	if err != nil { | ||
|  | 		return err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	otherHash, err := bcrypt(password, p.cost, p.salt) | ||
|  | 	if err != nil { | ||
|  | 		return err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	otherP := &hashed{otherHash, p.salt, p.cost, p.major, p.minor} | ||
|  | 	if subtle.ConstantTimeCompare(p.Hash(), otherP.Hash()) == 1 { | ||
|  | 		return nil | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return ErrMismatchedHashAndPassword | ||
|  | } | ||
|  | 
 | ||
|  | // Cost returns the hashing cost used to create the given hashed | ||
|  | // password. When, in the future, the hashing cost of a password system needs | ||
|  | // to be increased in order to adjust for greater computational power, this | ||
|  | // function allows one to establish which passwords need to be updated. | ||
|  | func Cost(hashedPassword []byte) (int, error) { | ||
|  | 	p, err := newFromHash(hashedPassword) | ||
|  | 	if err != nil { | ||
|  | 		return 0, err | ||
|  | 	} | ||
|  | 	return p.cost, nil | ||
|  | } | ||
|  | 
 | ||
|  | func newFromPassword(password []byte, cost int) (*hashed, error) { | ||
|  | 	if cost < MinCost { | ||
|  | 		cost = DefaultCost | ||
|  | 	} | ||
|  | 	p := new(hashed) | ||
|  | 	p.major = majorVersion | ||
|  | 	p.minor = minorVersion | ||
|  | 
 | ||
|  | 	err := checkCost(cost) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 	p.cost = cost | ||
|  | 
 | ||
|  | 	unencodedSalt := make([]byte, maxSaltSize) | ||
|  | 	_, err = io.ReadFull(rand.Reader, unencodedSalt) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	p.salt = base64Encode(unencodedSalt) | ||
|  | 	hash, err := bcrypt(password, p.cost, p.salt) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 	p.hash = hash | ||
|  | 	return p, err | ||
|  | } | ||
|  | 
 | ||
|  | func newFromHash(hashedSecret []byte) (*hashed, error) { | ||
|  | 	if len(hashedSecret) < minHashSize { | ||
|  | 		return nil, ErrHashTooShort | ||
|  | 	} | ||
|  | 	p := new(hashed) | ||
|  | 	n, err := p.decodeVersion(hashedSecret) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 	hashedSecret = hashedSecret[n:] | ||
|  | 	n, err = p.decodeCost(hashedSecret) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 	hashedSecret = hashedSecret[n:] | ||
|  | 
 | ||
|  | 	// The "+2" is here because we'll have to append at most 2 '=' to the salt | ||
|  | 	// when base64 decoding it in expensiveBlowfishSetup(). | ||
|  | 	p.salt = make([]byte, encodedSaltSize, encodedSaltSize+2) | ||
|  | 	copy(p.salt, hashedSecret[:encodedSaltSize]) | ||
|  | 
 | ||
|  | 	hashedSecret = hashedSecret[encodedSaltSize:] | ||
|  | 	p.hash = make([]byte, len(hashedSecret)) | ||
|  | 	copy(p.hash, hashedSecret) | ||
|  | 
 | ||
|  | 	return p, nil | ||
|  | } | ||
|  | 
 | ||
|  | func bcrypt(password []byte, cost int, salt []byte) ([]byte, error) { | ||
|  | 	cipherData := make([]byte, len(magicCipherData)) | ||
|  | 	copy(cipherData, magicCipherData) | ||
|  | 
 | ||
|  | 	c, err := expensiveBlowfishSetup(password, uint32(cost), salt) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	for i := 0; i < 24; i += 8 { | ||
|  | 		for j := 0; j < 64; j++ { | ||
|  | 			c.Encrypt(cipherData[i:i+8], cipherData[i:i+8]) | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Bug compatibility with C bcrypt implementations. We only encode 23 of | ||
|  | 	// the 24 bytes encrypted. | ||
|  | 	hsh := base64Encode(cipherData[:maxCryptedHashSize]) | ||
|  | 	return hsh, nil | ||
|  | } | ||
|  | 
 | ||
|  | func expensiveBlowfishSetup(key []byte, cost uint32, salt []byte) (*blowfish.Cipher, error) { | ||
|  | 	csalt, err := base64Decode(salt) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	// Bug compatibility with C bcrypt implementations. They use the trailing | ||
|  | 	// NULL in the key string during expansion. | ||
|  | 	// We copy the key to prevent changing the underlying array. | ||
|  | 	ckey := append(key[:len(key):len(key)], 0) | ||
|  | 
 | ||
|  | 	c, err := blowfish.NewSaltedCipher(ckey, csalt) | ||
|  | 	if err != nil { | ||
|  | 		return nil, err | ||
|  | 	} | ||
|  | 
 | ||
|  | 	var i, rounds uint64 | ||
|  | 	rounds = 1 << cost | ||
|  | 	for i = 0; i < rounds; i++ { | ||
|  | 		blowfish.ExpandKey(ckey, c) | ||
|  | 		blowfish.ExpandKey(csalt, c) | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return c, nil | ||
|  | } | ||
|  | 
 | ||
|  | func (p *hashed) Hash() []byte { | ||
|  | 	arr := make([]byte, 60) | ||
|  | 	arr[0] = '$' | ||
|  | 	arr[1] = p.major | ||
|  | 	n := 2 | ||
|  | 	if p.minor != 0 { | ||
|  | 		arr[2] = p.minor | ||
|  | 		n = 3 | ||
|  | 	} | ||
|  | 	arr[n] = '$' | ||
|  | 	n++ | ||
|  | 	copy(arr[n:], []byte(fmt.Sprintf("%02d", p.cost))) | ||
|  | 	n += 2 | ||
|  | 	arr[n] = '$' | ||
|  | 	n++ | ||
|  | 	copy(arr[n:], p.salt) | ||
|  | 	n += encodedSaltSize | ||
|  | 	copy(arr[n:], p.hash) | ||
|  | 	n += encodedHashSize | ||
|  | 	return arr[:n] | ||
|  | } | ||
|  | 
 | ||
|  | func (p *hashed) decodeVersion(sbytes []byte) (int, error) { | ||
|  | 	if sbytes[0] != '$' { | ||
|  | 		return -1, InvalidHashPrefixError(sbytes[0]) | ||
|  | 	} | ||
|  | 	if sbytes[1] > majorVersion { | ||
|  | 		return -1, HashVersionTooNewError(sbytes[1]) | ||
|  | 	} | ||
|  | 	p.major = sbytes[1] | ||
|  | 	n := 3 | ||
|  | 	if sbytes[2] != '$' { | ||
|  | 		p.minor = sbytes[2] | ||
|  | 		n++ | ||
|  | 	} | ||
|  | 	return n, nil | ||
|  | } | ||
|  | 
 | ||
|  | // sbytes should begin where decodeVersion left off. | ||
|  | func (p *hashed) decodeCost(sbytes []byte) (int, error) { | ||
|  | 	cost, err := strconv.Atoi(string(sbytes[0:2])) | ||
|  | 	if err != nil { | ||
|  | 		return -1, err | ||
|  | 	} | ||
|  | 	err = checkCost(cost) | ||
|  | 	if err != nil { | ||
|  | 		return -1, err | ||
|  | 	} | ||
|  | 	p.cost = cost | ||
|  | 	return 3, nil | ||
|  | } | ||
|  | 
 | ||
|  | func (p *hashed) String() string { | ||
|  | 	return fmt.Sprintf("&{hash: %#v, salt: %#v, cost: %d, major: %c, minor: %c}", string(p.hash), p.salt, p.cost, p.major, p.minor) | ||
|  | } | ||
|  | 
 | ||
|  | func checkCost(cost int) error { | ||
|  | 	if cost < MinCost || cost > MaxCost { | ||
|  | 		return InvalidCostError(cost) | ||
|  | 	} | ||
|  | 	return nil | ||
|  | } |