ASExchange.go 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. package client
  2. import (
  3. "errors"
  4. "fmt"
  5. "github.com/jcmturner/gokrb5/crypto"
  6. "github.com/jcmturner/gokrb5/iana/errorcode"
  7. "github.com/jcmturner/gokrb5/iana/patype"
  8. "github.com/jcmturner/gokrb5/messages"
  9. "github.com/jcmturner/gokrb5/types"
  10. "sort"
  11. )
  12. func (cl *Client) Login() error {
  13. return cl.ASExchange()
  14. }
  15. func (cl *Client) ASExchange() error {
  16. if !cl.IsConfigured() {
  17. return errors.New("Client is not configured correctly.")
  18. }
  19. a := messages.NewASReq(cl.Config, cl.Credentials.Username)
  20. b, err := a.Marshal()
  21. if err != nil {
  22. return fmt.Errorf("Error marshalling AS_REQ: %v", err)
  23. }
  24. rb, err := cl.SendToKDC(b)
  25. if err != nil {
  26. return fmt.Errorf("Error sending AS_REQ to KDC: %v", err)
  27. }
  28. var ar messages.ASRep
  29. err = ar.Unmarshal(rb)
  30. if err != nil {
  31. //An KRBError may have been returned instead.
  32. var krberr messages.KRBError
  33. err = krberr.Unmarshal(rb)
  34. if err != nil {
  35. return fmt.Errorf("Could not unmarshal data returned from KDC: %v", err)
  36. }
  37. if krberr.ErrorCode == errorcode.KDC_ERR_PREAUTH_REQUIRED {
  38. paTSb, err := types.GetPAEncTSEncAsnMarshalled()
  39. if err != nil {
  40. return fmt.Errorf("Error creating PAEncTSEnc for Pre-Authentication: %v", err)
  41. }
  42. sort.Sort(sort.Reverse(sort.IntSlice(cl.Config.LibDefaults.Default_tkt_enctype_ids)))
  43. etype, err := crypto.GetEtype(cl.Config.LibDefaults.Default_tkt_enctype_ids[0])
  44. if err != nil {
  45. return fmt.Errorf("Error creating etype: %v", err)
  46. }
  47. //paEncTS, err := crypto.GetEncryptedData(paTSb, etype, cl.Config.LibDefaults.Default_realm, cl.Credentials.Username, cl.Credentials.Keytab, 1)
  48. key, err := cl.Credentials.Keytab.GetEncryptionKey(cl.Credentials.Username, cl.Config.LibDefaults.Default_realm, 1, etype.GetETypeID())
  49. paEncTS, err := crypto.GetEncryptedData(paTSb, key, 0, 1)
  50. if err != nil {
  51. return fmt.Errorf("Error encrypting pre-authentication timestamp: %v", err)
  52. }
  53. pa := types.PAData{
  54. PADataType: patype.PA_ENC_TIMESTAMP,
  55. PADataValue: paEncTS.Cipher,
  56. }
  57. a.PAData = append(a.PAData, pa)
  58. b, err := a.Marshal()
  59. if err != nil {
  60. return fmt.Errorf("Error marshalling AS_REQ: %v", err)
  61. }
  62. rb, err := cl.SendToKDC(b)
  63. if err != nil {
  64. return fmt.Errorf("Error sending AS_REQ to KDC: %v", err)
  65. }
  66. err = ar.Unmarshal(rb)
  67. if err != nil {
  68. return fmt.Errorf("Could not unmarshal data returned from KDC: %v", err)
  69. }
  70. }
  71. return krberr
  72. }
  73. err = ar.DecryptEncPart(cl.Credentials)
  74. if err != nil {
  75. return fmt.Errorf("Error decrypting EncPart of AS_REP: %v", err)
  76. }
  77. if ok, err := ar.IsValid(cl.Config, a); !ok {
  78. return fmt.Errorf("AS_REP is not valid: %v", err)
  79. }
  80. cl.Session = &Session{
  81. AuthTime: ar.DecryptedEncPart.AuthTime,
  82. EndTime: ar.DecryptedEncPart.EndTime,
  83. RenewTill: ar.DecryptedEncPart.RenewTill,
  84. TGT: ar.Ticket,
  85. SessionKey: ar.DecryptedEncPart.Key,
  86. SessionKeyExpiration: ar.DecryptedEncPart.KeyExpiration,
  87. }
  88. return nil
  89. }