| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- package GSSAPI
- import (
- "encoding/binary"
- "encoding/hex"
- "fmt"
- "github.com/jcmturner/asn1"
- "github.com/jcmturner/gokrb5/asn1tools"
- "github.com/jcmturner/gokrb5/config"
- "github.com/jcmturner/gokrb5/iana/chksumtype"
- "github.com/jcmturner/gokrb5/messages"
- "github.com/jcmturner/gokrb5/types"
- "math/rand"
- )
- const (
- TOK_ID_KRB_AP_REQ = "0100"
- TOK_ID_KRB_AP_REP = "0200"
- TOK_ID_KRB_ERROR = "0300"
- GSS_C_DELEG_FLAG = 1
- GSS_C_MUTUAL_FLAG = 2
- GSS_C_REPLAY_FLAG = 4
- GSS_C_SEQUENCE_FLAG = 8
- GSS_C_CONF_FLAG = 16
- GSS_C_INTEG_FLAG = 32
- )
- func NewKRB5APREQMechToken(c config.Config, cname types.PrincipalName, tkt types.Ticket, sessionKey types.EncryptionKey) ([]byte, error) {
- // Create the header
- tb, _ := hex.DecodeString(TOK_ID_KRB_AP_REQ)
- b, _ := asn1.Marshal(MechTypeOID_Krb5)
- b = append(b, tb...)
- // Add the token
- APReq, err := messages.NewAPReq(
- tkt,
- sessionKey,
- newAuthenticator(c, cname),
- )
- tb, err = APReq.Marshal()
- if err != nil {
- return []byte{}, fmt.Errorf("Could not marshal AP_REQ: %v", err)
- }
- b = append(b, tb...)
- return asn1tools.AddASNAppTag(b, 0), nil
- }
- func newAuthenticator(c config.Config, username types.PrincipalName) types.Authenticator {
- //RFC 4121 Section 4.1.1
- auth := types.NewAuthenticator(c.LibDefaults.Default_realm, username)
- auth.Cksum = types.Checksum{
- CksumType: chksumtype.GSSAPI,
- Checksum: newAuthenticatorChksum([]int{GSS_C_INTEG_FLAG, GSS_C_CONF_FLAG}),
- }
- auth.SeqNumber = int(rand.Int63())
- return auth
- }
- func newAuthenticatorChksum(flags []int) []byte {
- a := make([]byte, 24)
- for i := range flags {
- if i == GSS_C_DELEG_FLAG {
- x := make([]byte, 28-len(a))
- a = append(a, x...)
- }
- setAuthenticatorChksumFlag(a, uint32(i))
- }
- return a
- }
- func setAuthenticatorChksumFlag(a []byte, i uint32) {
- f := binary.LittleEndian.Uint32(a[20:24])
- f |= i
- binary.LittleEndian.PutUint32(a[20:24], f)
- }
- /*
- The authenticator checksum field SHALL have the following format:
- Octet Name Description
- -----------------------------------------------------------------
- 0..3 Lgth Number of octets in Bnd field; Represented
- in little-endian order; Currently contains
- hex value 10 00 00 00 (16).
- 4..19 Bnd Channel binding information, as described in
- section 4.1.1.2.
- 20..23 Flags Four-octet context-establishment flags in
- little-endian order as described in section
- 4.1.1.1.
- 24..25 DlgOpt The delegation option identifier (=1) in
- little-endian order [optional]. This field
- and the next two fields are present if and
- only if GSS_C_DELEG_FLAG is set as described
- in section 4.1.1.1.
- 26..27 Dlgth The length of the Deleg field in little-endian order [optional].
- 28..(n-1) Deleg A KRB_CRED message (n = Dlgth + 28) [optional].
- n..last Exts Extensions [optional].
- */
|