| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257 |
- package service
- import (
- "encoding/hex"
- "fmt"
- "github.com/stretchr/testify/assert"
- "gopkg.in/jcmturner/gokrb5.v2/client"
- "gopkg.in/jcmturner/gokrb5.v2/credentials"
- "gopkg.in/jcmturner/gokrb5.v2/iana/nametype"
- "gopkg.in/jcmturner/gokrb5.v2/keytab"
- "gopkg.in/jcmturner/gokrb5.v2/messages"
- "gopkg.in/jcmturner/gokrb5.v2/testdata"
- "gopkg.in/jcmturner/gokrb5.v2/types"
- "io/ioutil"
- "log"
- "net/http"
- "net/http/httptest"
- "sync"
- "testing"
- "time"
- )
- func TestService_SPNEGOKRB_NoAuthHeader(t *testing.T) {
- s := httpServer()
- defer s.Close()
- r, _ := http.NewRequest("GET", s.URL, nil)
- httpResp, err := http.DefaultClient.Do(r)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusUnauthorized, httpResp.StatusCode, "Status code in response to client with no SPNEGO not as expected")
- assert.Equal(t, "Negotiate", httpResp.Header.Get("WWW-Authenticate"), "Negitation header not set by server.")
- }
- func TestService_SPNEGOKRB_ValidUser(t *testing.T) {
- s := httpServer()
- defer s.Close()
- cl := getClient()
- sname := types.PrincipalName{
- NameType: nametype.KRB_NT_PRINCIPAL,
- NameString: []string{"HTTP", "host.test.gokrb5"},
- }
- b, _ := hex.DecodeString(testdata.HTTP_KEYTAB)
- kt, _ := keytab.Parse(b)
- st := time.Now().UTC()
- tkt, sessionKey, err := messages.NewTicket(cl.Credentials.CName, cl.Credentials.Realm,
- sname, "TEST.GOKRB5",
- types.NewKrbFlags(),
- kt,
- 18,
- 1,
- st,
- st,
- st.Add(time.Duration(24)*time.Hour),
- st.Add(time.Duration(48)*time.Hour),
- )
- if err != nil {
- t.Fatalf("Error getting test ticket: %v", err)
- }
- r, _ := http.NewRequest("GET", s.URL, nil)
- err = client.SetSPNEGOHeader(*cl.Credentials, tkt, sessionKey, r)
- if err != nil {
- t.Fatalf("Error setting client SPNEGO header: %v", err)
- }
- httpResp, err := http.DefaultClient.Do(r)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusOK, httpResp.StatusCode, "Status code in response to client SPNEGO request not as expected")
- }
- func TestService_SPNEGOKRB_Replay(t *testing.T) {
- s := httpServer()
- defer s.Close()
- cl := getClient()
- sname := types.PrincipalName{
- NameType: nametype.KRB_NT_PRINCIPAL,
- NameString: []string{"HTTP", "host.test.gokrb5"},
- }
- b, _ := hex.DecodeString(testdata.HTTP_KEYTAB)
- kt, _ := keytab.Parse(b)
- st := time.Now().UTC()
- tkt, sessionKey, err := messages.NewTicket(cl.Credentials.CName, cl.Credentials.Realm,
- sname, "TEST.GOKRB5",
- types.NewKrbFlags(),
- kt,
- 18,
- 1,
- st,
- st,
- st.Add(time.Duration(24)*time.Hour),
- st.Add(time.Duration(48)*time.Hour),
- )
- if err != nil {
- t.Fatalf("Error getting test ticket: %v", err)
- }
- r1, _ := http.NewRequest("GET", s.URL, nil)
- err = client.SetSPNEGOHeader(*cl.Credentials, tkt, sessionKey, r1)
- if err != nil {
- t.Fatalf("Error setting client SPNEGO header: %v", err)
- }
- // First request with this ticket should be accepted
- httpResp, err := http.DefaultClient.Do(r1)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusOK, httpResp.StatusCode, "Status code in response to client SPNEGO request not as expected")
- // Use ticket again should be rejected
- httpResp, err = http.DefaultClient.Do(r1)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusUnauthorized, httpResp.StatusCode, "Status code in response to client with no SPNEGO not as expected. Expected a replay to be detected.")
- // Form a 2nd ticket
- st = time.Now().UTC()
- tkt2, sessionKey2, err := messages.NewTicket(cl.Credentials.CName, cl.Credentials.Realm,
- sname, "TEST.GOKRB5",
- types.NewKrbFlags(),
- kt,
- 18,
- 1,
- st,
- st,
- st.Add(time.Duration(24)*time.Hour),
- st.Add(time.Duration(48)*time.Hour),
- )
- if err != nil {
- t.Fatalf("Error getting test ticket: %v", err)
- }
- r2, _ := http.NewRequest("GET", s.URL, nil)
- err = client.SetSPNEGOHeader(*cl.Credentials, tkt2, sessionKey2, r2)
- if err != nil {
- t.Fatalf("Error setting client SPNEGO header: %v", err)
- }
- // First use of 2nd ticket should be accepted
- httpResp, err = http.DefaultClient.Do(r2)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusOK, httpResp.StatusCode, "Status code in response to client SPNEGO request not as expected")
- // Using the 1st ticket again should still be rejected
- httpResp, err = http.DefaultClient.Do(r1)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusUnauthorized, httpResp.StatusCode, "Status code in response to client with no SPNEGO not as expected. Expected a replay to be detected.")
- // Using the 2nd again should be rejected as replay
- httpResp, err = http.DefaultClient.Do(r2)
- if err != nil {
- t.Fatalf("Request error: %v\n", err)
- }
- assert.Equal(t, http.StatusUnauthorized, httpResp.StatusCode, "Status code in response to client with no SPNEGO not as expected. Expected a replay to be detected.")
- }
- func TestService_SPNEGOKRB_ReplayCache_Concurrency(t *testing.T) {
- s := httpServer()
- defer s.Close()
- cl := getClient()
- sname := types.PrincipalName{
- NameType: nametype.KRB_NT_PRINCIPAL,
- NameString: []string{"HTTP", "host.test.gokrb5"},
- }
- b, _ := hex.DecodeString(testdata.HTTP_KEYTAB)
- kt, _ := keytab.Parse(b)
- st := time.Now().UTC()
- tkt, sessionKey, err := messages.NewTicket(cl.Credentials.CName, cl.Credentials.Realm,
- sname, "TEST.GOKRB5",
- types.NewKrbFlags(),
- kt,
- 18,
- 1,
- st,
- st,
- st.Add(time.Duration(24)*time.Hour),
- st.Add(time.Duration(48)*time.Hour),
- )
- if err != nil {
- t.Fatalf("Error getting test ticket: %v", err)
- }
- r1, _ := http.NewRequest("GET", s.URL, nil)
- err = client.SetSPNEGOHeader(*cl.Credentials, tkt, sessionKey, r1)
- if err != nil {
- t.Fatalf("Error setting client SPNEGO header: %v", err)
- }
- // Form a 2nd ticket
- st = time.Now().UTC()
- tkt2, sessionKey2, err := messages.NewTicket(cl.Credentials.CName, cl.Credentials.Realm,
- sname, "TEST.GOKRB5",
- types.NewKrbFlags(),
- kt,
- 18,
- 1,
- st,
- st,
- st.Add(time.Duration(24)*time.Hour),
- st.Add(time.Duration(48)*time.Hour),
- )
- if err != nil {
- t.Fatalf("Error getting test ticket: %v", err)
- }
- r2, _ := http.NewRequest("GET", s.URL, nil)
- err = client.SetSPNEGOHeader(*cl.Credentials, tkt2, sessionKey2, r2)
- if err != nil {
- t.Fatalf("Error setting client SPNEGO header: %v", err)
- }
- // Concurrent 1st requests should be OK
- var wg sync.WaitGroup
- wg.Add(2)
- go httpGet(r1, &wg)
- go httpGet(r2, &wg)
- wg.Wait()
- // A number of concurrent requests with the same ticket should be rejected due to replay
- var wg2 sync.WaitGroup
- noReq := 10
- wg2.Add(noReq * 2)
- for i := 0; i < noReq; i++ {
- go httpGet(r1, &wg2)
- go httpGet(r2, &wg2)
- }
- wg2.Wait()
- }
- func httpGet(r *http.Request, wg *sync.WaitGroup) {
- defer wg.Done()
- http.DefaultClient.Do(r)
- }
- func httpServer() *httptest.Server {
- l := log.New(ioutil.Discard, "GOKRB5 Service Tests: ", log.Ldate|log.Ltime|log.Lshortfile)
- b, _ := hex.DecodeString(testdata.HTTP_KEYTAB)
- kt, _ := keytab.Parse(b)
- th := http.HandlerFunc(testAppHandler)
- s := httptest.NewServer(SPNEGOKRB5Authenticate(th, kt, "", l))
- return s
- }
- func testAppHandler(w http.ResponseWriter, r *http.Request) {
- w.WriteHeader(http.StatusOK)
- ctx := r.Context()
- fmt.Fprintf(w, "<html>\nTEST.GOKRB5 Handler\nAuthenticed user: %s\nUser's realm: %s\n</html>", ctx.Value(CTXKeyCredentials).(credentials.Credentials).Username, ctx.Value(CTXKeyCredentials).(credentials.Credentials).Realm)
- return
- }
|