diff --git a/server/controller/http/router/common/controller.go b/server/controller/http/router/common/controller.go index 6234a386243..c117b62390d 100644 --- a/server/controller/http/router/common/controller.go +++ b/server/controller/http/router/common/controller.go @@ -26,7 +26,7 @@ import ( "github.com/op/go-logging" ) -var log = logging.MustGetLogger("common/controller") +var log = logging.MustGetLogger("http.router.common") func ForwardMasterController(c *gin.Context, masterControllerName string, port int) { requestHosts := strings.Split(c.Request.Host, ":") diff --git a/server/controller/http/router/common/response.go b/server/controller/http/router/common/response.go index d15bba1fec8..2ec81fd88dd 100644 --- a/server/controller/http/router/common/response.go +++ b/server/controller/http/router/common/response.go @@ -17,6 +17,7 @@ package common import ( + "encoding/json" "errors" "fmt" "net/http" @@ -31,14 +32,54 @@ type Response struct { OptStatus string `json:"OPT_STATUS"` Description string `json:"DESCRIPTION"` Data interface{} `json:"DATA"` + Page Page `json:"PAGE"` } -func HttpResponse(c *gin.Context, httpCode int, data interface{}, optStatus string, description string) { - c.JSON(httpCode, Response{ - OptStatus: optStatus, - Description: description, - Data: data, - }) +// String returns the string representation of the response when Data is a byte slice. +func (r Response) String() string { + return fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s,"PAGE":%s}`, r.OptStatus, r.Description, string(r.Data.([]byte)), r.Page.String()) +} + +// Bytes returns the byte slice representation of the response when Data is a byte slice. +func (r Response) Bytes() []byte { + return []byte(r.String()) +} + +type Page struct { + Index int `json:"INDEX"` + Size int `json:"SIZE"` + Total int `json:"TOTAL"` + TotalItem int `json:"TOTAL_ITEM"` +} + +func (p Page) String() string { + bytes, _ := json.Marshal(p) + return string(bytes) +} + +type ResponseOption func(resp *Response) + +func WithPage(p Page) ResponseOption { + return func(resp *Response) { + resp.Page = p + } +} + +func WithDescription(description string) ResponseOption { + return func(resp *Response) { + resp.Description = description + } +} + +func HttpResponse(c *gin.Context, httpCode int, data interface{}, optStatus string, options ...ResponseOption) { + resp := Response{ + OptStatus: optStatus, + Data: data, + } + for _, option := range options { + option(&resp) + } + c.JSON(httpCode, resp) } func BadRequestResponse(c *gin.Context, optStatus string, description string) { @@ -79,9 +120,10 @@ func StatusForbiddenResponse(c *gin.Context, description string) { }) } -func JsonResponse(c *gin.Context, data interface{}, err error) { +// TODO refactor, use ResponseOption +func JsonResponse(c *gin.Context, data interface{}, err error, respOptions ...ResponseOption) { if _, ok := data.([]byte); ok { - bytesResponse(c, data, err) + bytesResponse(c, data, err, respOptions...) return } @@ -110,47 +152,47 @@ func JsonResponse(c *gin.Context, data interface{}, err error) { InternalErrorResponse(c, data, httpcommon.FAIL, err.Error()) } } else { - HttpResponse(c, 200, data, httpcommon.SUCCESS, "") + HttpResponse(c, 200, data, httpcommon.SUCCESS, respOptions...) } } -func bytesResponse(c *gin.Context, data interface{}, err error) { +func bytesResponse(c *gin.Context, data interface{}, err error, respOptions ...ResponseOption) { + resp := Response{Data: data} + for _, opt := range respOptions { + opt(&resp) + } if err != nil { switch t := err.(type) { case *servicecommon.ServiceError: + resp.OptStatus = t.Status + resp.Description = t.Message switch t.Status { case httpcommon.NO_PERMISSIONS: - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusForbidden, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusForbidden, gin.MIMEJSON, resp.Bytes()) case httpcommon.RESOURCE_NOT_FOUND, httpcommon.INVALID_POST_DATA, httpcommon.RESOURCE_NUM_EXCEEDED, httpcommon.SELECTED_RESOURCES_NUM_EXCEEDED, httpcommon.RESOURCE_ALREADY_EXIST, httpcommon.PARAMETER_ILLEGAL, httpcommon.INVALID_PARAMETERS: - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusBadRequest, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusBadRequest, gin.MIMEJSON, resp.Bytes()) case httpcommon.SERVER_ERROR, httpcommon.CONFIG_PENDING: - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusInternalServerError, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusInternalServerError, gin.MIMEJSON, resp.Bytes()) case httpcommon.SERVICE_UNAVAILABLE: - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusServiceUnavailable, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusServiceUnavailable, gin.MIMEJSON, resp.Bytes()) case httpcommon.STATUES_PARTIAL_CONTENT: - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusPartialContent, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusPartialContent, gin.MIMEJSON, resp.Bytes()) default: if errors.Is(err, httpcommon.ERR_NO_PERMISSIONS) { - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusForbidden, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusForbidden, gin.MIMEJSON, resp.Bytes()) return } - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, t.Status, t.Message, string(data.([]byte))) - c.Data(http.StatusBadRequest, gin.MIMEJSON, []byte(d)) + c.Data(http.StatusBadRequest, gin.MIMEJSON, resp.Bytes()) } default: - d := fmt.Sprintf(`{"OPT_STATUS":%s,"DESCRIPTION":%s,"DATA":%s}`, httpcommon.FAIL, err.Error(), string(data.([]byte))) - c.Data(http.StatusInternalServerError, gin.MIMEJSON, []byte(d)) + resp.OptStatus = httpcommon.FAIL + resp.Description = err.Error() + c.Data(http.StatusInternalServerError, gin.MIMEJSON, resp.Bytes()) } } else { - d := fmt.Sprintf(`{"OPT_STATUS":"SUCCESS","DESCRIPTION":"","DATA": %v}`, string(data.([]byte))) - c.Data(http.StatusOK, gin.MIMEJSON, []byte(d)) + resp.OptStatus = httpcommon.SUCCESS + c.Data(http.StatusOK, gin.MIMEJSON, resp.Bytes()) } } diff --git a/server/controller/http/router/common/response_test.go b/server/controller/http/router/common/response_test.go new file mode 100644 index 00000000000..2ee6ddc72de --- /dev/null +++ b/server/controller/http/router/common/response_test.go @@ -0,0 +1,63 @@ +/* + * Copyright (c) 2024 Yunshan Networks + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package common + +import ( + "fmt" + "net/http/httptest" + "testing" + + "github.com/gin-gonic/gin" +) + +func TestHttpResponse(t *testing.T) { + gin.SetMode(gin.TestMode) + type args struct { + c *gin.Context + code int + status string + data interface{} + opts []ResponseOption + } + w := httptest.NewRecorder() + c, _ := gin.CreateTestContext(w) + tests := []struct { + name string + args args + }{ + { + name: "TestHttpResponse", + args: args{ + c: c, + code: 200, + status: "success", + data: nil, + opts: []ResponseOption{WithPage(Page{Index: 1, Size: 10, Total: 1, TotalItem: 1})}, + }, + }, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + HttpResponse(tt.args.c, tt.args.code, tt.args.data, tt.args.status, tt.args.opts...) + }) + // check response + if w.Code != tt.args.code { + t.Errorf("HttpResponse() code = %d, want %d", w.Code, tt.args.code) + } + fmt.Println(w.Body.String()) + } +}