auth.go 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. package main
  2. import (
  3. "bufio"
  4. "errors"
  5. "os"
  6. "strings"
  7. "golang.org/x/crypto/bcrypt"
  8. )
  9. var (
  10. filename string
  11. )
  12. type AuthUser struct {
  13. username string
  14. passwordHash string
  15. allowedAddresses []string
  16. }
  17. func AuthLoadFile(file string) error {
  18. f, err := os.Open(file)
  19. if err != nil {
  20. return err
  21. }
  22. f.Close()
  23. filename = file
  24. return nil
  25. }
  26. func AuthReady() bool {
  27. return (filename != "")
  28. }
  29. // Split a string and ignore empty results
  30. // https://stackoverflow.com/a/46798310/119527
  31. func splitstr(s string, sep rune) []string {
  32. return strings.FieldsFunc(s, func(c rune) bool { return c == sep })
  33. }
  34. func parseLine(line string) *AuthUser {
  35. parts := strings.Fields(line)
  36. if len(parts) < 2 || len(parts) > 3 {
  37. return nil
  38. }
  39. user := AuthUser{
  40. username: parts[0],
  41. passwordHash: parts[1],
  42. allowedAddresses: nil,
  43. }
  44. if len(parts) >= 3 {
  45. user.allowedAddresses = splitstr(parts[2], ',')
  46. }
  47. return &user
  48. }
  49. func AuthFetch(username string) (*AuthUser, error) {
  50. if !AuthReady() {
  51. return nil, errors.New("Authentication file not specified. Call LoadFile() first")
  52. }
  53. file, err := os.Open(filename)
  54. if err != nil {
  55. return nil, err
  56. }
  57. defer file.Close()
  58. scanner := bufio.NewScanner(file)
  59. for scanner.Scan() {
  60. user := parseLine(scanner.Text())
  61. if user == nil {
  62. continue
  63. }
  64. if strings.ToLower(username) != strings.ToLower(user.username) {
  65. continue
  66. }
  67. return user, nil
  68. }
  69. return nil, errors.New("User not found")
  70. }
  71. func AuthCheckPassword(username string, secret string) error {
  72. user, err := AuthFetch(username)
  73. if err != nil {
  74. return err
  75. }
  76. if bcrypt.CompareHashAndPassword([]byte(user.passwordHash), []byte(secret)) == nil {
  77. return nil
  78. }
  79. return errors.New("Password invalid")
  80. }