package http_test import ( "bytes" "context" "sync" "testing" "github.com/fasthttp/router" json "github.com/goccy/go-json" http "github.com/valyala/fasthttp" "source.toby3d.me/website/indieauth/internal/common" "source.toby3d.me/website/indieauth/internal/domain" "source.toby3d.me/website/indieauth/internal/profile" profilerepo "source.toby3d.me/website/indieauth/internal/profile/repository/memory" "source.toby3d.me/website/indieauth/internal/session" sessionrepo "source.toby3d.me/website/indieauth/internal/session/repository/memory" "source.toby3d.me/website/indieauth/internal/testing/httptest" "source.toby3d.me/website/indieauth/internal/ticket" ticketrepo "source.toby3d.me/website/indieauth/internal/ticket/repository/memory" ticketucase "source.toby3d.me/website/indieauth/internal/ticket/usecase" "source.toby3d.me/website/indieauth/internal/token" delivery "source.toby3d.me/website/indieauth/internal/token/delivery/http" tokenrepo "source.toby3d.me/website/indieauth/internal/token/repository/memory" tokenucase "source.toby3d.me/website/indieauth/internal/token/usecase" ) type Dependencies struct { client *http.Client config *domain.Config profiles profile.Repository sessions session.Repository store *sync.Map tickets ticket.Repository ticketService ticket.UseCase token *domain.Token tokens token.Repository tokenService token.UseCase } /* TODO(toby3d) func TestExchange(t *testing.T) { t.Parallel() } */ func TestIntrospection(t *testing.T) { t.Parallel() deps := NewDependencies(t) r := router.New() delivery.NewRequestHandler(deps.tokenService, deps.ticketService, deps.config).Register(r) client, _, cleanup := httptest.New(t, r.Handler) t.Cleanup(cleanup) const requestURL = "https://app.example.com/introspect" req := httptest.NewRequest(http.MethodPost, requestURL, []byte("token="+deps.token.AccessToken)) defer http.ReleaseRequest(req) req.Header.Set(http.HeaderAccept, common.MIMEApplicationJSON) req.Header.SetContentType(common.MIMEApplicationForm) resp := http.AcquireResponse() defer http.ReleaseResponse(resp) if err := client.Do(req, resp); err != nil { t.Fatal(err) } if result := resp.StatusCode(); result != http.StatusOK { t.Errorf("GET %s = %d, want %d", requestURL, result, http.StatusOK) } result := new(delivery.TokenIntrospectResponse) if err := json.Unmarshal(resp.Body(), result); err != nil { e := err.(*json.SyntaxError) t.Logf("%s\noffset: %d", resp.Body(), e.Offset) t.Fatal(err) } deps.token.AccessToken = "" if result.ClientID != deps.token.ClientID.String() || result.Me != deps.token.Me.String() || result.Scope != deps.token.Scope.String() { t.Errorf("GET %s = %+v, want %+v", requestURL, result, deps.token) } } func TestRevocation(t *testing.T) { t.Parallel() deps := NewDependencies(t) r := router.New() delivery.NewRequestHandler(deps.tokenService, deps.ticketService, deps.config).Register(r) client, _, cleanup := httptest.New(t, r.Handler) t.Cleanup(cleanup) const requestURL = "https://app.example.com/revocation" req := httptest.NewRequest(http.MethodPost, requestURL, []byte("token="+deps.token.AccessToken)) defer http.ReleaseRequest(req) req.Header.Set(http.HeaderAccept, common.MIMEApplicationJSON) req.Header.SetContentType(common.MIMEApplicationForm) resp := http.AcquireResponse() defer http.ReleaseResponse(resp) if err := client.Do(req, resp); err != nil { t.Fatal(err) } if result := resp.StatusCode(); result != http.StatusOK { t.Errorf("POST %s = %d, want %d", requestURL, result, http.StatusOK) } expBody := []byte("{}") //nolint: ifshort if result := bytes.TrimSpace(resp.Body()); !bytes.Equal(result, expBody) { t.Errorf("POST %s = %s, want %s", requestURL, result, expBody) } result, err := deps.tokens.Get(context.TODO(), deps.token.AccessToken) if err != nil { t.Fatal(err) } if result.String() != deps.token.String() { t.Errorf("Get(%+v) = %s, want %s", deps.token.AccessToken, result, deps.token) } } func NewDependencies(tb testing.TB) Dependencies { tb.Helper() store := new(sync.Map) client := new(http.Client) config := domain.TestConfig(tb) token := domain.TestToken(tb) profiles := profilerepo.NewMemoryProfileRepository(store) sessions := sessionrepo.NewMemorySessionRepository(store, config) tickets := ticketrepo.NewMemoryTicketRepository(store, config) tokens := tokenrepo.NewMemoryTokenRepository(store) ticketService := ticketucase.NewTicketUseCase(tickets, client, config) tokenService := tokenucase.NewTokenUseCase(tokenucase.Config{ Config: config, Profiles: profiles, Sessions: sessions, Tokens: tokens, }) return Dependencies{ client: client, config: config, profiles: profiles, sessions: sessions, store: store, tickets: tickets, ticketService: ticketService, token: token, tokens: tokens, tokenService: tokenService, } }