package web import "errors" // Interface for a request and its response. type Context interface { Bytes([]byte) error Error(...any) error Next() error Redirect(int, string) error Request() Request Response() Response Status(int) Context String(string) error } // Contains the request and response data. type context struct { request response server *server handlerCount uint8 } // Adds a raw byte slice to the response body. func (ctx *context) Bytes(body []byte) error { ctx.response.body = append(ctx.response.body, body...) return nil } // Provides a convenient way to wrap multiple errors. func (ctx *context) Error(messages ...any) error { var combined []error for _, msg := range messages { switch err := msg.(type) { case error: combined = append(combined, err) case string: combined = append(combined, errors.New(err)) } } return errors.Join(combined...) } // Executes the next handler in the middleware chain. func (ctx *context) Next() error { ctx.handlerCount++ return ctx.server.handlers[ctx.handlerCount](ctx) } // Redirects the client to a different location with the specified status code. func (ctx *context) Redirect(status int, location string) error { ctx.response.SetStatus(status) ctx.response.SetHeader("Location", location) return nil } // Returns the HTTP request. func (ctx *context) Request() Request { return &ctx.request } // Returns the HTTP response. func (ctx *context) Response() Response { return &ctx.response } // Sets the HTTP status of the response and returns the context for method chaining. func (ctx *context) Status(status int) Context { ctx.response.SetStatus(status) return ctx } // Adds the given string to the response body. func (ctx *context) String(body string) error { ctx.response.body = append(ctx.response.body, body...) return nil }