--- name: password_verify kind: function lang: go domain: infra version: "1.0.0" purity: impure signature: "func PasswordVerify(password string, hash string) error" description: "Verifica un password en texto plano contra un hash bcrypt. Retorna nil si hacen match, error si no coinciden o si el hash es invalido." tags: [password, verify, bcrypt, auth, infra] uses_functions: [] uses_types: [] returns: [] returns_optional: false error_type: error_go_core imports: [golang.org/x/crypto/bcrypt] params: - name: password desc: "password en texto plano a verificar" - name: hash desc: "hash bcrypt obtenido previamente de PasswordHash (guardado en BD)" output: "nil si el password coincide con el hash; error si no coincide o hash invalido" tested: true tests: ["verifica password correcto", "rechaza password incorrecto", "rechaza hash malformado"] test_file_path: "functions/infra/password_verify_test.go" file_path: "functions/infra/password_verify.go" --- ## Ejemplo ```go row := db.QueryRow("SELECT password_hash FROM users WHERE email = ?", email) var stored string if err := row.Scan(&stored); err != nil { HTTPErrorResponse(w, HTTPError{Status: 401, Code: "invalid_credentials", Message: "email o password incorrectos"}) return } if err := PasswordVerify(input, stored); err != nil { HTTPErrorResponse(w, HTTPError{Status: 401, Code: "invalid_credentials", Message: "email o password incorrectos"}) return } // OK, emitir token ``` ## Notas Impura — bcrypt.CompareHashAndPassword es constant-time internamente (mitiga timing attacks). En respuestas HTTP al usuario NO distinguir entre "email no existe" y "password incorrecto": ambos casos deben retornar el mismo mensaje generico para no filtrar existencia de cuentas. El error real se puede loguear internamente con log_info/log_warn sin problema.