mirror of
https://github.com/kataras/iris.git
synced 2025-01-26 03:56:34 +01:00
226 lines
7.5 KiB
Go
226 lines
7.5 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
|
|
"github.com/kataras/iris/v12"
|
|
"github.com/kataras/iris/v12/x/errors"
|
|
"github.com/kataras/iris/v12/x/errors/validation"
|
|
"github.com/kataras/iris/v12/x/pagination"
|
|
)
|
|
|
|
func main() {
|
|
app := iris.New()
|
|
|
|
// Create a new service and pass it to the handlers.
|
|
service := new(myService)
|
|
|
|
app.Post("/", createHandler(service)) // OR: errors.CreateHandler(service.Create)
|
|
app.Get("/", listAllHandler(service)) // OR errors.Handler(service.ListAll, errors.Value(ListRequest{}))
|
|
app.Post("/page", listHandler(service)) // OR: errors.ListHandler(service.ListPaginated)
|
|
app.Delete("/{id:string}", deleteHandler(service)) // OR: errors.NoContentOrNotModifiedHandler(service.DeleteWithFeedback, errors.PathParam[string]("id"))
|
|
|
|
app.Listen(":8080")
|
|
}
|
|
|
|
func createHandler(service *myService) iris.Handler {
|
|
return func(ctx iris.Context) {
|
|
// What it does?
|
|
// 1. Reads the request body and binds it to the CreateRequest struct.
|
|
// 2. Calls the service.Create function with the given request body.
|
|
// 3. If the service.Create returns an error, it sends an appropriate error response to the client.
|
|
// 4. If the service.Create returns a response, it sets the status code to 201 (Created) and sends the response as a JSON payload to the client.
|
|
//
|
|
// Useful for create operations.
|
|
errors.Create(ctx, service.Create)
|
|
}
|
|
}
|
|
|
|
func listAllHandler(service *myService) iris.Handler {
|
|
return func(ctx iris.Context) {
|
|
// What it does?
|
|
// 1. If the 3rd variadic (optional) parameter is empty (not our case here), it reads the request body and binds it to the ListRequest struct,
|
|
// otherwise (our case) it calls the service.ListAll function directly with the given input parameter (empty ListRequest struct value in our case).
|
|
// 2. Calls the service.ListAll function with the ListRequest value.
|
|
// 3. If the service.ListAll returns an error, it sends an appropriate error response to the client.
|
|
// 4. If the service.ListAll returns a response, it sets the status code to 200 (OK) and sends the response as a JSON payload to the client.
|
|
//
|
|
// Useful for get single, fetch multiple and search operations.
|
|
errors.OK(ctx, service.ListAll, ListRequest{})
|
|
}
|
|
}
|
|
|
|
func listHandler(service *myService) iris.Handler {
|
|
return func(ctx iris.Context) {
|
|
errors.List(ctx, service.ListPaginated)
|
|
}
|
|
}
|
|
|
|
func deleteHandler(service *myService) iris.Handler {
|
|
return func(ctx iris.Context) {
|
|
id := ctx.Params().Get("id")
|
|
// What it does?
|
|
// 1. Calls the service.DeleteWithFeedback function with the given input parameter.
|
|
// 2. If the service.DeleteWithFeedback returns an error, it sends an appropriate error response to the client.
|
|
// 3.If the service.DeleteWithFeedback doesn't return an error then it sets the status code to 204 (No Content) and
|
|
// sends the response as a JSON payload to the client.
|
|
// errors.NoContent(ctx, service.Delete, id)
|
|
// OR:
|
|
// 1. Calls the service.DeleteWithFeedback function with the given input parameter.
|
|
// 2. If the service.DeleteWithFeedback returns an error, it sends an appropriate error response to the client.
|
|
// 3. If the service.DeleteWithFeedback returns true, it sets the status code to 204 (No Content).
|
|
// 4. If the service.DeleteWithFeedback returns false, it sets the status code to 304 (Not Modified).
|
|
//
|
|
// Useful for update and delete operations.
|
|
errors.NoContentOrNotModified(ctx, service.DeleteWithFeedback, id)
|
|
}
|
|
}
|
|
|
|
type (
|
|
myService struct{}
|
|
|
|
CreateRequest struct {
|
|
Fullname string `json:"fullname"`
|
|
Age int `json:"age"`
|
|
Hobbies []string `json:"hobbies"`
|
|
}
|
|
|
|
CreateResponse struct {
|
|
ID string `json:"id"`
|
|
Firstname string `json:"firstname"`
|
|
Lastname string `json:"lastname"`
|
|
Age int `json:"age"`
|
|
Hobbies []string `json:"hobbies"`
|
|
}
|
|
)
|
|
|
|
// ValidateContext implements the errors.ContextValidator interface.
|
|
// It validates the request body and returns an error if the request body is invalid.
|
|
// You can also alter the "r" CreateRequest before calling the service method,
|
|
// e.g. give a default value to a field if it's empty or set an ID based on a path parameter.
|
|
// OR
|
|
// Custom function per route:
|
|
//
|
|
// r.Post("/", errors.Validation(validateCreateRequest), createHandler(service))
|
|
// [more code here...]
|
|
// func validateCreateRequest(ctx iris.Context, r *CreateRequest) error {
|
|
// return validation.Join(
|
|
// validation.String("fullname", r.Fullname).NotEmpty().Fullname().Length(3, 50),
|
|
// validation.Number("age", r.Age).InRange(18, 130),
|
|
// validation.Slice("hobbies", r.Hobbies).Length(1, 10),
|
|
// )
|
|
// }
|
|
func (r *CreateRequest) ValidateContext(ctx iris.Context) error {
|
|
// To pass custom validation functions:
|
|
// return validation.Join(
|
|
// validation.String("fullname", r.Fullname).Func(customStringFuncHere),
|
|
// OR
|
|
// validation.Field("any_field", r.AnyFieldValue).Func(customAnyFuncHere))
|
|
return validation.Join(
|
|
validation.String("fullname", r.Fullname).Fullname().Length(3, 50),
|
|
validation.Number("age", r.Age).InRange(18, 130),
|
|
validation.Slice("hobbies", r.Hobbies).Length(1, 10),
|
|
)
|
|
|
|
/* Example Output:
|
|
{
|
|
"http_error_code": {
|
|
"canonical_name": "INVALID_ARGUMENT",
|
|
"status": 400
|
|
},
|
|
"message": "validation failure",
|
|
"details": "fields were invalid",
|
|
"validation": [
|
|
{
|
|
"field": "fullname",
|
|
"value": "",
|
|
"reason": "must not be empty, must contain first and last name, must be between 3 and 50 characters"
|
|
},
|
|
{
|
|
"field": "age",
|
|
"value": 0,
|
|
"reason": "must be in range of [18, 130]"
|
|
},
|
|
{
|
|
"field": "hobbies",
|
|
"value": null,
|
|
"reason": "must be between 1 and 10 elements"
|
|
}
|
|
]
|
|
}
|
|
*/
|
|
}
|
|
|
|
func (s *myService) Create(ctx context.Context, in CreateRequest) (CreateResponse, error) {
|
|
arr := strings.Split(in.Fullname, " ")
|
|
firstname, lastname := arr[0], arr[1]
|
|
id := "test_id"
|
|
|
|
resp := CreateResponse{
|
|
ID: id,
|
|
Firstname: firstname,
|
|
Lastname: lastname,
|
|
Age: in.Age,
|
|
Hobbies: in.Hobbies,
|
|
}
|
|
return resp, nil // , errors.New("create: test error")
|
|
}
|
|
|
|
type ListRequest struct {
|
|
}
|
|
|
|
func (s *myService) ListAll(ctx context.Context, in ListRequest) ([]CreateResponse, error) {
|
|
resp := []CreateResponse{
|
|
{
|
|
ID: "test-id-1",
|
|
Firstname: "test first name 1",
|
|
Lastname: "test last name 1",
|
|
},
|
|
{
|
|
ID: "test-id-2",
|
|
Firstname: "test first name 2",
|
|
Lastname: "test last name 2",
|
|
},
|
|
{
|
|
ID: "test-id-3",
|
|
Firstname: "test first name 3",
|
|
Lastname: "test last name 3",
|
|
},
|
|
}
|
|
|
|
return resp, nil //, errors.New("list all: test error")
|
|
}
|
|
|
|
type ListFilter struct {
|
|
Firstname string `json:"firstname"`
|
|
}
|
|
|
|
func (s *myService) ListPaginated(ctx context.Context, opts pagination.ListOptions, filter ListFilter) ([]CreateResponse, int /* any number type */, error) {
|
|
all, err := s.ListAll(ctx, ListRequest{})
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
filteredResp := make([]CreateResponse, 0)
|
|
for _, v := range all {
|
|
if strings.Contains(v.Firstname, filter.Firstname) {
|
|
filteredResp = append(filteredResp, v)
|
|
}
|
|
|
|
if len(filteredResp) == opts.GetLimit() {
|
|
break
|
|
}
|
|
}
|
|
|
|
return filteredResp, len(all), nil // errors.New("list paginated: test error")
|
|
}
|
|
|
|
func (s *myService) Delete(ctx context.Context, id string) error {
|
|
return nil // errors.New("delete: test error")
|
|
}
|
|
|
|
func (s *myService) DeleteWithFeedback(ctx context.Context, id string) (bool, error) {
|
|
return true, nil // false, errors.New("delete: test error")
|
|
}
|