Compare commits
	
		
			6 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| e5818146a8 | |||
| 1310054121 | |||
| 49d423915c | |||
| 1962cb3c52 | |||
| 84f124dd4d | |||
| ff8e066135 | 
| @@ -3,8 +3,8 @@ package cursortoken | |||||||
| import ( | import ( | ||||||
| 	"encoding/base32" | 	"encoding/base32" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"errors" |  | ||||||
| 	"go.mongodb.org/mongo-driver/bson/primitive" | 	"go.mongodb.org/mongo-driver/bson/primitive" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| @@ -127,7 +127,7 @@ func Decode(tok string) (CursorToken, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if !strings.HasPrefix(tok, "tok_") { | 	if !strings.HasPrefix(tok, "tok_") { | ||||||
| 		return CursorToken{}, errors.New("could not decode token, missing prefix") | 		return CursorToken{}, exerr.New(exerr.TypeCursorTokenDecode, "could not decode token, missing prefix").Str("token", tok).Build() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	body, err := base32.StdEncoding.DecodeString(tok[len("tok_"):]) | 	body, err := base32.StdEncoding.DecodeString(tok[len("tok_"):]) | ||||||
| @@ -138,7 +138,7 @@ func Decode(tok string) (CursorToken, error) { | |||||||
| 	var tokenDeserialize cursorTokenSerialize | 	var tokenDeserialize cursorTokenSerialize | ||||||
| 	err = json.Unmarshal(body, &tokenDeserialize) | 	err = json.Unmarshal(body, &tokenDeserialize) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return CursorToken{}, err | 		return CursorToken{}, exerr.Wrap(err, "failed to deserialize token").Str("token", tok).Build() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	token := CursorToken{Mode: CTMNormal} | 	token := CursorToken{Mode: CTMNormal} | ||||||
|   | |||||||
| @@ -3,13 +3,17 @@ package ginext | |||||||
| import ( | import ( | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"strings" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func CorsMiddleware() gin.HandlerFunc { | func CorsMiddleware(allowheader []string, exposeheader []string) gin.HandlerFunc { | ||||||
| 	return func(c *gin.Context) { | 	return func(c *gin.Context) { | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Origin", "*") | 		c.Writer.Header().Set("Access-Control-Allow-Origin", "*") | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Credentials", "true") | 		c.Writer.Header().Set("Access-Control-Allow-Credentials", "true") | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Headers", "Content-Type, Content-Length, Accept-Encoding, X-CSRF-Token, Authorization, accept, origin, Cache-Control, X-Requested-With") | 		c.Writer.Header().Set("Access-Control-Allow-Headers", strings.Join(allowheader, ", ")) | ||||||
|  | 		if len(exposeheader) > 0 { | ||||||
|  | 			c.Writer.Header().Set("Access-Control-Expose-Headers", strings.Join(exposeheader, ", ")) | ||||||
|  | 		} | ||||||
| 		c.Writer.Header().Set("Access-Control-Allow-Methods", "OPTIONS, GET, POST, PUT, PATCH, DELETE, COUNT") | 		c.Writer.Header().Set("Access-Control-Allow-Methods", "OPTIONS, GET, POST, PUT, PATCH, DELETE, COUNT") | ||||||
|  |  | ||||||
| 		if c.Request.Method == "OPTIONS" { | 		if c.Request.Method == "OPTIONS" { | ||||||
|   | |||||||
| @@ -21,12 +21,16 @@ type GinWrapper struct { | |||||||
|  |  | ||||||
| 	opt                   Options | 	opt                   Options | ||||||
| 	allowCors             bool | 	allowCors             bool | ||||||
|  | 	corsAllowHeader       []string | ||||||
|  | 	corsExposeHeader      []string | ||||||
| 	ginDebug              bool | 	ginDebug              bool | ||||||
| 	bufferBody            bool | 	bufferBody            bool | ||||||
| 	requestTimeout        time.Duration | 	requestTimeout        time.Duration | ||||||
| 	listenerBeforeRequest []func(g *gin.Context) | 	listenerBeforeRequest []func(g *gin.Context) | ||||||
| 	listenerAfterRequest  []func(g *gin.Context, resp HTTPResponse) | 	listenerAfterRequest  []func(g *gin.Context, resp HTTPResponse) | ||||||
|  |  | ||||||
|  | 	buildRequestBindError func(g *gin.Context, fieldtype string, err error) HTTPResponse | ||||||
|  |  | ||||||
| 	routeSpecs []ginRouteSpec | 	routeSpecs []ginRouteSpec | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -38,15 +42,18 @@ type ginRouteSpec struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| type Options struct { | type Options struct { | ||||||
| 	AllowCors                *bool                                     // Add cors handler to allow all CORS requests on the default http methods | 	AllowCors                *bool                                                          // Add cors handler to allow all CORS requests on the default http methods | ||||||
| 	GinDebug                 *bool                                     // Set gin.debug to true (adds more logs) | 	CorsAllowHeader          *[]string                                                      // override the default values of Access-Control-Allow-Headers (AllowCors must be true) | ||||||
| 	SuppressGinLogs          *bool                                     // Suppress our custom gin logs (even if GinDebug == true) | 	CorsExposeHeader         *[]string                                                      // return Access-Control-Expose-Headers (AllowCors must be true) | ||||||
| 	BufferBody               *bool                                     // Buffers the input body stream, this way the ginext error handler can later include the whole request body | 	GinDebug                 *bool                                                          // Set gin.debug to true (adds more logs) | ||||||
| 	Timeout                  *time.Duration                            // The default handler timeout | 	SuppressGinLogs          *bool                                                          // Suppress our custom gin logs (even if GinDebug == true) | ||||||
| 	ListenerBeforeRequest    []func(g *gin.Context)                    // Register listener that are called before the handler method | 	BufferBody               *bool                                                          // Buffers the input body stream, this way the ginext error handler can later include the whole request body | ||||||
| 	ListenerAfterRequest     []func(g *gin.Context, resp HTTPResponse) // Register listener that are called after the handler method | 	Timeout                  *time.Duration                                                 // The default handler timeout | ||||||
| 	DebugTrimHandlerPrefixes []string                                  // Trim these prefixes from the handler names in the debug print | 	ListenerBeforeRequest    []func(g *gin.Context)                                         // Register listener that are called before the handler method | ||||||
| 	DebugReplaceHandlerNames map[string]string                         // Replace handler names in debug output | 	ListenerAfterRequest     []func(g *gin.Context, resp HTTPResponse)                      // Register listener that are called after the handler method | ||||||
|  | 	DebugTrimHandlerPrefixes []string                                                       // Trim these prefixes from the handler names in the debug print | ||||||
|  | 	DebugReplaceHandlerNames map[string]string                                              // Replace handler names in debug output | ||||||
|  | 	BuildRequestBindError    func(g *gin.Context, fieldtype string, err error) HTTPResponse // Override function which generates the HTTPResponse errors that are returned by the preContext..Start() methids | ||||||
| } | } | ||||||
|  |  | ||||||
| // NewEngine creates a new (wrapped) ginEngine | // NewEngine creates a new (wrapped) ginEngine | ||||||
| @@ -72,18 +79,21 @@ func NewEngine(opt Options) *GinWrapper { | |||||||
| 		opt:                   opt, | 		opt:                   opt, | ||||||
| 		suppressGinLogs:       langext.Coalesce(opt.SuppressGinLogs, false), | 		suppressGinLogs:       langext.Coalesce(opt.SuppressGinLogs, false), | ||||||
| 		allowCors:             langext.Coalesce(opt.AllowCors, false), | 		allowCors:             langext.Coalesce(opt.AllowCors, false), | ||||||
|  | 		corsAllowHeader:       langext.Coalesce(opt.CorsAllowHeader, []string{"Content-Type", "Content-Length", "Accept-Encoding", "X-CSRF-Token", "Authorization", "accept", "origin", "Cache-Control", "X-Requested-With"}), | ||||||
|  | 		corsExposeHeader:      langext.Coalesce(opt.CorsExposeHeader, []string{}), | ||||||
| 		ginDebug:              ginDebug, | 		ginDebug:              ginDebug, | ||||||
| 		bufferBody:            langext.Coalesce(opt.BufferBody, false), | 		bufferBody:            langext.Coalesce(opt.BufferBody, false), | ||||||
| 		requestTimeout:        langext.Coalesce(opt.Timeout, 24*time.Hour), | 		requestTimeout:        langext.Coalesce(opt.Timeout, 24*time.Hour), | ||||||
| 		listenerBeforeRequest: opt.ListenerBeforeRequest, | 		listenerBeforeRequest: opt.ListenerBeforeRequest, | ||||||
| 		listenerAfterRequest:  opt.ListenerAfterRequest, | 		listenerAfterRequest:  opt.ListenerAfterRequest, | ||||||
|  | 		buildRequestBindError: langext.Conditional(opt.BuildRequestBindError == nil, defaultBuildRequestBindError, opt.BuildRequestBindError), | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	engine.RedirectFixedPath = false | 	engine.RedirectFixedPath = false | ||||||
| 	engine.RedirectTrailingSlash = false | 	engine.RedirectTrailingSlash = false | ||||||
|  |  | ||||||
| 	if wrapper.allowCors { | 	if wrapper.allowCors { | ||||||
| 		engine.Use(CorsMiddleware()) | 		engine.Use(CorsMiddleware(wrapper.corsAllowHeader, wrapper.corsExposeHeader)) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if ginDebug && !wrapper.suppressGinLogs { | 	if ginDebug && !wrapper.suppressGinLogs { | ||||||
| @@ -222,3 +232,7 @@ func (w *GinWrapper) ForwardRequest(writer http.ResponseWriter, req *http.Reques | |||||||
| func (w *GinWrapper) ListRoutes() []gin.RouteInfo { | func (w *GinWrapper) ListRoutes() []gin.RouteInfo { | ||||||
| 	return w.engine.Routes() | 	return w.engine.Routes() | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func defaultBuildRequestBindError(g *gin.Context, fieldtype string, err error) HTTPResponse { | ||||||
|  | 	return Error(err) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -77,14 +77,14 @@ func (pctx *PreContext) IgnoreWrongContentType() *PreContext { | |||||||
| 	return pctx | 	return pctx | ||||||
| } | } | ||||||
|  |  | ||||||
| func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPErrorResponse) { | func (pctx PreContext) Start() (*AppContext, *gin.Context, *HTTPResponse) { | ||||||
| 	if pctx.uri != nil { | 	if pctx.uri != nil { | ||||||
| 		if err := pctx.ginCtx.ShouldBindUri(pctx.uri); err != nil { | 		if err := pctx.ginCtx.ShouldBindUri(pctx.uri); err != nil { | ||||||
| 			err = exerr.Wrap(err, "Failed to read uri"). | 			err = exerr.Wrap(err, "Failed to read uri"). | ||||||
| 				WithType(exerr.TypeBindFailURI). | 				WithType(exerr.TypeBindFailURI). | ||||||
| 				Str("struct_type", fmt.Sprintf("%T", pctx.uri)). | 				Str("struct_type", fmt.Sprintf("%T", pctx.uri)). | ||||||
| 				Build() | 				Build() | ||||||
| 			return nil, nil, langext.Ptr(Error(err)) | 			return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "URI", err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -94,7 +94,7 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 				WithType(exerr.TypeBindFailQuery). | 				WithType(exerr.TypeBindFailQuery). | ||||||
| 				Str("struct_type", fmt.Sprintf("%T", pctx.query)). | 				Str("struct_type", fmt.Sprintf("%T", pctx.query)). | ||||||
| 				Build() | 				Build() | ||||||
| 			return nil, nil, langext.Ptr(Error(err)) | 			return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "QUERY", err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -105,14 +105,14 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 					WithType(exerr.TypeBindFailJSON). | 					WithType(exerr.TypeBindFailJSON). | ||||||
| 					Str("struct_type", fmt.Sprintf("%T", pctx.body)). | 					Str("struct_type", fmt.Sprintf("%T", pctx.body)). | ||||||
| 					Build() | 					Build() | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "JSON", err)) | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			if !pctx.ignoreWrongContentType { | 			if !pctx.ignoreWrongContentType { | ||||||
| 				err := exerr.New(exerr.TypeBindFailJSON, "missing JSON body"). | 				err := exerr.New(exerr.TypeBindFailJSON, "missing JSON body"). | ||||||
| 					Str("struct_type", fmt.Sprintf("%T", pctx.body)). | 					Str("struct_type", fmt.Sprintf("%T", pctx.body)). | ||||||
| 					Build() | 					Build() | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "JSON", err)) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -121,14 +121,14 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 		if brc, ok := pctx.ginCtx.Request.Body.(dataext.BufferedReadCloser); ok { | 		if brc, ok := pctx.ginCtx.Request.Body.(dataext.BufferedReadCloser); ok { | ||||||
| 			v, err := brc.BufferedAll() | 			v, err := brc.BufferedAll() | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "BODY", err)) | ||||||
| 			} | 			} | ||||||
| 			*pctx.rawbody = v | 			*pctx.rawbody = v | ||||||
| 		} else { | 		} else { | ||||||
| 			buf := &bytes.Buffer{} | 			buf := &bytes.Buffer{} | ||||||
| 			_, err := io.Copy(buf, pctx.ginCtx.Request.Body) | 			_, err := io.Copy(buf, pctx.ginCtx.Request.Body) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "BODY", err)) | ||||||
| 			} | 			} | ||||||
| 			*pctx.rawbody = buf.Bytes() | 			*pctx.rawbody = buf.Bytes() | ||||||
| 		} | 		} | ||||||
| @@ -141,7 +141,7 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 					WithType(exerr.TypeBindFailFormData). | 					WithType(exerr.TypeBindFailFormData). | ||||||
| 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | ||||||
| 					Build() | 					Build() | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "FORM", err)) | ||||||
| 			} | 			} | ||||||
| 		} else if pctx.ginCtx.ContentType() == "application/x-www-form-urlencoded" { | 		} else if pctx.ginCtx.ContentType() == "application/x-www-form-urlencoded" { | ||||||
| 			if err := pctx.ginCtx.ShouldBindWith(pctx.form, binding.Form); err != nil { | 			if err := pctx.ginCtx.ShouldBindWith(pctx.form, binding.Form); err != nil { | ||||||
| @@ -149,14 +149,14 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 					WithType(exerr.TypeBindFailFormData). | 					WithType(exerr.TypeBindFailFormData). | ||||||
| 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | ||||||
| 					Build() | 					Build() | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "FORM", err)) | ||||||
| 			} | 			} | ||||||
| 		} else { | 		} else { | ||||||
| 			if !pctx.ignoreWrongContentType { | 			if !pctx.ignoreWrongContentType { | ||||||
| 				err := exerr.New(exerr.TypeBindFailFormData, "missing form body"). | 				err := exerr.New(exerr.TypeBindFailFormData, "missing form body"). | ||||||
| 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | 					Str("struct_type", fmt.Sprintf("%T", pctx.form)). | ||||||
| 					Build() | 					Build() | ||||||
| 				return nil, nil, langext.Ptr(Error(err)) | 				return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "FORM", err)) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @@ -167,7 +167,7 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 				WithType(exerr.TypeBindFailHeader). | 				WithType(exerr.TypeBindFailHeader). | ||||||
| 				Str("struct_type", fmt.Sprintf("%T", pctx.query)). | 				Str("struct_type", fmt.Sprintf("%T", pctx.query)). | ||||||
| 				Build() | 				Build() | ||||||
| 			return nil, nil, langext.Ptr(Error(err)) | 			return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "HEADER", err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -179,7 +179,7 @@ func (pctx PreContext) Start() (*AppContext, *gin.Context, *InspectableHTTPError | |||||||
| 		err := pctx.persistantData.sessionObj.Init(pctx.ginCtx, actx) | 		err := pctx.persistantData.sessionObj.Init(pctx.ginCtx, actx) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			actx.Cancel() | 			actx.Cancel() | ||||||
| 			return nil, nil, langext.Ptr(Error(exerr.Wrap(err, "Failed to init session").Build())) | 			return nil, nil, langext.Ptr(pctx.wrapper.buildRequestBindError(pctx.ginCtx, "INIT", err)) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -36,9 +36,8 @@ type InspectableHTTPResponse interface { | |||||||
| 	Headers() []string | 	Headers() []string | ||||||
| } | } | ||||||
|  |  | ||||||
| type InspectableHTTPErrorResponse interface { | type HTTPErrorResponse interface { | ||||||
| 	HTTPResponse | 	HTTPResponse | ||||||
| 	InspectableHTTPResponse |  | ||||||
|  |  | ||||||
| 	Error() error | 	Error() error | ||||||
| } | } | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ func (j jsonAPIErrResponse) Unwrap() error { | |||||||
| 	return j.err | 	return j.err | ||||||
| } | } | ||||||
|  |  | ||||||
| func Error(e error) InspectableHTTPErrorResponse { | func Error(e error) HTTPResponse { | ||||||
| 	return &jsonAPIErrResponse{ | 	return &jsonAPIErrResponse{ | ||||||
| 		err: exerr.FromError(e), | 		err: exerr.FromError(e), | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| package goext | package goext | ||||||
|  |  | ||||||
| const GoextVersion = "0.0.483" | const GoextVersion = "0.0.489" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2024-07-16T15:08:37+0200" | const GoextVersionTimestamp = "2024-07-23T14:21:03+0200" | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ import ( | |||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"image" | 	"image" | ||||||
| 	"image/color" | 	"image/color" | ||||||
|  | 	"image/draw" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -284,6 +285,13 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		if dataimg.ColorModel() != color.RGBAModel && dataimg.ColorModel() != color.NRGBAModel { | ||||||
|  | 			// the image cannto be 16bpp or similar - otherwise fpdf errors out | ||||||
|  | 			dataImgRGBA := image.NewNRGBA(image.Rect(0, 0, dataimg.Bounds().Dx(), dataimg.Bounds().Dy())) | ||||||
|  | 			draw.Draw(dataImgRGBA, dataImgRGBA.Bounds(), dataimg, dataimg.Bounds().Min, draw.Src) | ||||||
|  | 			dataimg = dataImgRGBA | ||||||
|  | 		} | ||||||
|  |  | ||||||
| 		bfr, imgMime, err := imageext.EncodeImage(dataimg, compression) | 		bfr, imgMime, err := imageext.EncodeImage(dataimg, compression) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			b.b.SetError(err) | 			b.b.SetError(err) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user