// Copyright 2023 The Gitea Authors. All rights reserved. // SPDX-License-Identifier: MIT package hash import ( "crypto/sha256" "encoding/hex" "strings" "code.gitea.io/gitea/modules/log" "golang.org/x/crypto/pbkdf2" ) func init() { MustRegister("pbkdf2", NewPBKDF2Hasher) } // PBKDF2Hasher implements PasswordHasher // and uses the PBKDF2 key derivation function. type PBKDF2Hasher struct { iter, keyLen int } // HashWithSaltBytes a provided password and salt func (hasher *PBKDF2Hasher) HashWithSaltBytes(password string, salt []byte) string { if hasher == nil { return "" } return hex.EncodeToString(pbkdf2.Key([]byte(password), salt, hasher.iter, hasher.keyLen, sha256.New)) } // NewPBKDF2Hasher is a factory method to create an PBKDF2Hasher // config should be either empty or of the form: // "<iter>$<keyLen>", where <x> is the string representation // of an integer func NewPBKDF2Hasher(config string) *PBKDF2Hasher { // This default configuration uses the following parameters: // iter=10000, keyLen=50. // This matches the original configuration for `pbkdf2` prior to storing parameters // in the database. // THESE VALUES MUST NOT BE CHANGED OR BACKWARDS COMPATIBILITY WILL BREAK hasher := &PBKDF2Hasher{ iter: 10_000, keyLen: 50, } if config == "" { return hasher } vals := strings.SplitN(config, "$", 2) if len(vals) != 2 { log.Error("invalid pbkdf2 hash spec %s", config) return nil } var err error hasher.iter, err = parseIntParam(vals[0], "iter", "pbkdf2", config, nil) hasher.keyLen, err = parseIntParam(vals[1], "keyLen", "pbkdf2", config, err) if err != nil { return nil } return hasher }