package response import ( "context" "encoding/json" "errors" "fmt" "net/http" "strings" "github.com/beckn/beckn-onix/pkg/model" ) type Error struct { Code string `json:"code,omitempty"` Message string `json:"message,omitempty"` Paths string `json:"paths,omitempty"` } // SchemaValidationErr represents a collection of schema validation failures. type SchemaValidationErr struct { Errors []Error } // Error implements the error interface for SchemaValidationErr. func (e *SchemaValidationErr) Error() string { var errorMessages []string for _, err := range e.Errors { errorMessages = append(errorMessages, fmt.Sprintf("%s: %s", err.Paths, err.Message)) } return strings.Join(errorMessages, "; ") } type Message struct { Ack struct { Status string `json:"status,omitempty"` } `json:"ack,omitempty"` Error *Error `json:"error,omitempty"` } func SendAck(w http.ResponseWriter) { resp := &model.Response{ Message: model.Message{ Ack: model.Ack{ Status: model.StatusACK, }, }, } data, _ := json.Marshal(resp) //should not fail here w.Header().Set("Content-Type", "application/json") w.WriteHeader(http.StatusOK) _, err := w.Write(data) if err != nil { http.Error(w, "failed to write response", http.StatusInternalServerError) return } } func nack(w http.ResponseWriter, err *model.Error, status int, ctx context.Context) { resp := &model.Response{ Message: model.Message{ Ack: model.Ack{ Status: model.StatusNACK, }, Error: err, }, } data, _ := json.Marshal(resp) //should not fail here w.Header().Set("Content-Type", "application/json") w.WriteHeader(status) _, er := w.Write(data) if er != nil { fmt.Printf("Error writing response: %v, MessageID: %s", er, ctx.Value(model.MsgIDKey)) http.Error(w, fmt.Sprintf("Internal server error, MessageID: %s", ctx.Value(model.MsgIDKey)), http.StatusInternalServerError) return } } func internalServerError(ctx context.Context) *model.Error { return &model.Error{ Code: http.StatusText(http.StatusInternalServerError), Message: fmt.Sprintf("Internal server error, MessageID: %s", ctx.Value(model.MsgIDKey)), } } func SendNack(ctx context.Context, w http.ResponseWriter, err error) { var schemaErr *model.SchemaValidationErr var signErr *model.SignValidationErr var badReqErr *model.BadReqErr var notFoundErr *model.NotFoundErr switch { case errors.As(err, &schemaErr): nack(w, schemaErr.BecknError(), http.StatusBadRequest, ctx) return case errors.As(err, &signErr): nack(w, signErr.BecknError(), http.StatusUnauthorized, ctx) return case errors.As(err, &badReqErr): nack(w, badReqErr.BecknError(), http.StatusBadRequest, ctx) return case errors.As(err, ¬FoundErr): nack(w, notFoundErr.BecknError(), http.StatusNotFound, ctx) return default: nack(w, internalServerError(ctx), http.StatusInternalServerError, ctx) return } }