Compare commits
	
		
			7 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| bfa8457e95 | |||
| 70106733d9 | |||
| ce7837b9ef | |||
| d0d72167eb | |||
| a55ee1a6ce | |||
| dfc319573c | |||
| 246e555f3f | 
| @@ -1,91 +1,14 @@ | |||||||
| package exerr | package exerr | ||||||
|  |  | ||||||
| import ( | type Method string | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" |  | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | const ( | ||||||
|  | 	MethodOutput Method = "OUTPUT" | ||||||
|  | 	MethodPrint  Method = "PRINT" | ||||||
|  | 	MethodBuild  Method = "BUILD" | ||||||
|  | 	MethodFatal  Method = "FATAL" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type ErrorCategory struct{ Category string } |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	CatWrap    = ErrorCategory{"Wrap"}    // The error is simply wrapping another error (e.g. when a grpc call returns an error) |  | ||||||
| 	CatSystem  = ErrorCategory{"System"}  // An internal system error (e.g. connection to db failed) |  | ||||||
| 	CatUser    = ErrorCategory{"User"}    // The user (the API caller) did something wrong (e.g. he has no permissions to do this) |  | ||||||
| 	CatForeign = ErrorCategory{"Foreign"} // A foreign error that some component threw (e.g. an unknown mongodb error), happens if we call Wrap(..) on an non-bmerror value |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| //goland:noinspection GoUnusedGlobalVariable |  | ||||||
| var AllCategories = []ErrorCategory{CatWrap, CatSystem, CatUser, CatForeign} |  | ||||||
|  |  | ||||||
| type ErrorSeverity struct{ Severity string } |  | ||||||
|  |  | ||||||
| var ( |  | ||||||
| 	SevTrace = ErrorSeverity{"Trace"} |  | ||||||
| 	SevDebug = ErrorSeverity{"Debug"} |  | ||||||
| 	SevInfo  = ErrorSeverity{"Info"} |  | ||||||
| 	SevWarn  = ErrorSeverity{"Warn"} |  | ||||||
| 	SevErr   = ErrorSeverity{"Err"} |  | ||||||
| 	SevFatal = ErrorSeverity{"Fatal"} |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| //goland:noinspection GoUnusedGlobalVariable |  | ||||||
| var AllSeverities = []ErrorSeverity{SevTrace, SevDebug, SevInfo, SevWarn, SevErr, SevFatal} |  | ||||||
|  |  | ||||||
| type ErrorType struct { |  | ||||||
| 	Key               string |  | ||||||
| 	DefaultStatusCode *int |  | ||||||
| } |  | ||||||
|  |  | ||||||
| //goland:noinspection GoUnusedGlobalVariable |  | ||||||
| var ( |  | ||||||
| 	TypeInternal       = NewType("INTERNAL_ERROR", langext.Ptr(500)) |  | ||||||
| 	TypePanic          = NewType("PANIC", langext.Ptr(500)) |  | ||||||
| 	TypeNotImplemented = NewType("NOT_IMPLEMENTED", langext.Ptr(500)) |  | ||||||
|  |  | ||||||
| 	TypeMongoQuery        = NewType("MONGO_QUERY", langext.Ptr(500)) |  | ||||||
| 	TypeCursorTokenDecode = NewType("CURSOR_TOKEN_DECODE", langext.Ptr(500)) |  | ||||||
| 	TypeMongoFilter       = NewType("MONGO_FILTER", langext.Ptr(500)) |  | ||||||
| 	TypeMongoReflection   = NewType("MONGO_REFLECTION", langext.Ptr(500)) |  | ||||||
| 	TypeMongoInvalidOpt   = NewType("MONGO_INVALIDOPT", langext.Ptr(500)) |  | ||||||
|  |  | ||||||
| 	TypeSQLQuery  = NewType("SQL_QUERY", langext.Ptr(500)) |  | ||||||
| 	TypeSQLBuild  = NewType("SQL_BUILD", langext.Ptr(500)) |  | ||||||
| 	TypeSQLDecode = NewType("SQL_DECODE", langext.Ptr(500)) |  | ||||||
|  |  | ||||||
| 	TypeWrap = NewType("Wrap", nil) |  | ||||||
|  |  | ||||||
| 	TypeBindFailURI      = NewType("BINDFAIL_URI", langext.Ptr(400)) |  | ||||||
| 	TypeBindFailQuery    = NewType("BINDFAIL_QUERY", langext.Ptr(400)) |  | ||||||
| 	TypeBindFailJSON     = NewType("BINDFAIL_JSON", langext.Ptr(400)) |  | ||||||
| 	TypeBindFailFormData = NewType("BINDFAIL_FORMDATA", langext.Ptr(400)) |  | ||||||
| 	TypeBindFailHeader   = NewType("BINDFAIL_HEADER", langext.Ptr(400)) |  | ||||||
|  |  | ||||||
| 	TypeMarshalEntityID = NewType("MARSHAL_ENTITY_ID", langext.Ptr(400)) |  | ||||||
| 	TypeInvalidCSID     = NewType("INVALID_CSID", langext.Ptr(400)) |  | ||||||
|  |  | ||||||
| 	TypeGoogleStatuscode = NewType("GOOGLE_STATUSCODE", langext.Ptr(400)) |  | ||||||
| 	TypeGoogleResponse   = NewType("GOOGLE_RESPONSE", langext.Ptr(400)) |  | ||||||
|  |  | ||||||
| 	TypeUnauthorized = NewType("UNAUTHORIZED", langext.Ptr(401)) |  | ||||||
| 	TypeAuthFailed   = NewType("AUTH_FAILED", langext.Ptr(401)) |  | ||||||
|  |  | ||||||
| 	// other values come from the downstream application that uses goext |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var registeredTypes = dataext.SyncMap[string, ErrorType]{} |  | ||||||
|  |  | ||||||
| func NewType(key string, defStatusCode *int) ErrorType { |  | ||||||
| 	et := ErrorType{key, defStatusCode} |  | ||||||
|  |  | ||||||
| 	registeredTypes.Set(key, et) |  | ||||||
|  |  | ||||||
| 	return et |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func ListRegisteredTypes() []ErrorType { |  | ||||||
| 	return registeredTypes.GetAllValues() |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type LogPrintLevel string | type LogPrintLevel string | ||||||
|  |  | ||||||
| const ( | const ( | ||||||
|   | |||||||
							
								
								
									
										89
									
								
								exerr/dataCategory.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								exerr/dataCategory.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | |||||||
|  | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsoncodec" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsonrw" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsontype" | ||||||
|  | 	"reflect" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ErrorCategory struct{ Category string } | ||||||
|  |  | ||||||
|  | var ( | ||||||
|  | 	CatWrap    = ErrorCategory{"Wrap"}    // The error is simply wrapping another error (e.g. when a grpc call returns an error) | ||||||
|  | 	CatSystem  = ErrorCategory{"System"}  // An internal system error (e.g. connection to db failed) | ||||||
|  | 	CatUser    = ErrorCategory{"User"}    // The user (the API caller) did something wrong (e.g. he has no permissions to do this) | ||||||
|  | 	CatForeign = ErrorCategory{"Foreign"} // A foreign error that some component threw (e.g. an unknown mongodb error), happens if we call Wrap(..) on an non-bmerror value | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (e *ErrorCategory) UnmarshalJSON(bytes []byte) error { | ||||||
|  | 	return json.Unmarshal(bytes, &e.Category) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorCategory) MarshalJSON() ([]byte, error) { | ||||||
|  | 	return json.Marshal(e.Category) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e *ErrorCategory) UnmarshalBSONValue(bt bsontype.Type, data []byte) error { | ||||||
|  | 	if bt == bson.TypeNull { | ||||||
|  | 		// we can't set nil in UnmarshalBSONValue (so we use default(struct)) | ||||||
|  | 		// Use mongoext.CreateGoExtBsonRegistry if you need to unmarsh pointer values | ||||||
|  | 		// https://stackoverflow.com/questions/75167597 | ||||||
|  | 		// https://jira.mongodb.org/browse/GODRIVER-2252 | ||||||
|  | 		*e = ErrorCategory{} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if bt != bson.TypeString { | ||||||
|  | 		return errors.New(fmt.Sprintf("cannot unmarshal %v into String", bt)) | ||||||
|  | 	} | ||||||
|  | 	var tt string | ||||||
|  | 	err := bson.RawValue{Type: bt, Value: data}.Unmarshal(&tt) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	*e = ErrorCategory{tt} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorCategory) MarshalBSONValue() (bsontype.Type, []byte, error) { | ||||||
|  | 	return bson.MarshalValue(e.Category) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorCategory) DecodeValue(dc bsoncodec.DecodeContext, vr bsonrw.ValueReader, val reflect.Value) error { | ||||||
|  | 	if val.Kind() == reflect.Ptr && val.IsNil() { | ||||||
|  | 		if !val.CanSet() { | ||||||
|  | 			return errors.New("ValueUnmarshalerDecodeValue") | ||||||
|  | 		} | ||||||
|  | 		val.Set(reflect.New(val.Type().Elem())) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tp, src, err := bsonrw.Copier{}.CopyValueToBytes(vr) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if val.Kind() == reflect.Ptr && len(src) == 0 { | ||||||
|  | 		val.Set(reflect.Zero(val.Type())) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = e.UnmarshalBSONValue(tp, src) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if val.Kind() == reflect.Ptr { | ||||||
|  | 		val.Set(reflect.ValueOf(&e)) | ||||||
|  | 	} else { | ||||||
|  | 		val.Set(reflect.ValueOf(e)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //goland:noinspection GoUnusedGlobalVariable | ||||||
|  | var AllCategories = []ErrorCategory{CatWrap, CatSystem, CatUser, CatForeign} | ||||||
							
								
								
									
										91
									
								
								exerr/dataSeverity.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								exerr/dataSeverity.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | |||||||
|  | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsoncodec" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsonrw" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsontype" | ||||||
|  | 	"reflect" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ErrorSeverity struct{ Severity string } | ||||||
|  |  | ||||||
|  | var ( | ||||||
|  | 	SevTrace = ErrorSeverity{"Trace"} | ||||||
|  | 	SevDebug = ErrorSeverity{"Debug"} | ||||||
|  | 	SevInfo  = ErrorSeverity{"Info"} | ||||||
|  | 	SevWarn  = ErrorSeverity{"Warn"} | ||||||
|  | 	SevErr   = ErrorSeverity{"Err"} | ||||||
|  | 	SevFatal = ErrorSeverity{"Fatal"} | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (e *ErrorSeverity) UnmarshalJSON(bytes []byte) error { | ||||||
|  | 	return json.Unmarshal(bytes, &e.Severity) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorSeverity) MarshalJSON() ([]byte, error) { | ||||||
|  | 	return json.Marshal(e.Severity) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e *ErrorSeverity) UnmarshalBSONValue(bt bsontype.Type, data []byte) error { | ||||||
|  | 	if bt == bson.TypeNull { | ||||||
|  | 		// we can't set nil in UnmarshalBSONValue (so we use default(struct)) | ||||||
|  | 		// Use mongoext.CreateGoExtBsonRegistry if you need to unmarsh pointer values | ||||||
|  | 		// https://stackoverflow.com/questions/75167597 | ||||||
|  | 		// https://jira.mongodb.org/browse/GODRIVER-2252 | ||||||
|  | 		*e = ErrorSeverity{} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if bt != bson.TypeString { | ||||||
|  | 		return errors.New(fmt.Sprintf("cannot unmarshal %v into String", bt)) | ||||||
|  | 	} | ||||||
|  | 	var tt string | ||||||
|  | 	err := bson.RawValue{Type: bt, Value: data}.Unmarshal(&tt) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	*e = ErrorSeverity{tt} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorSeverity) MarshalBSONValue() (bsontype.Type, []byte, error) { | ||||||
|  | 	return bson.MarshalValue(e.Severity) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorSeverity) DecodeValue(dc bsoncodec.DecodeContext, vr bsonrw.ValueReader, val reflect.Value) error { | ||||||
|  | 	if val.Kind() == reflect.Ptr && val.IsNil() { | ||||||
|  | 		if !val.CanSet() { | ||||||
|  | 			return errors.New("ValueUnmarshalerDecodeValue") | ||||||
|  | 		} | ||||||
|  | 		val.Set(reflect.New(val.Type().Elem())) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tp, src, err := bsonrw.Copier{}.CopyValueToBytes(vr) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if val.Kind() == reflect.Ptr && len(src) == 0 { | ||||||
|  | 		val.Set(reflect.Zero(val.Type())) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = e.UnmarshalBSONValue(tp, src) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if val.Kind() == reflect.Ptr { | ||||||
|  | 		val.Set(reflect.ValueOf(&e)) | ||||||
|  | 	} else { | ||||||
|  | 		val.Set(reflect.ValueOf(e)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //goland:noinspection GoUnusedGlobalVariable | ||||||
|  | var AllSeverities = []ErrorSeverity{SevTrace, SevDebug, SevInfo, SevWarn, SevErr, SevFatal} | ||||||
							
								
								
									
										155
									
								
								exerr/dataType.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										155
									
								
								exerr/dataType.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,155 @@ | |||||||
|  | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsoncodec" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsonrw" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/bsontype" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/dataext" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"reflect" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ErrorType struct { | ||||||
|  | 	Key               string | ||||||
|  | 	DefaultStatusCode *int | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //goland:noinspection GoUnusedGlobalVariable | ||||||
|  | var ( | ||||||
|  | 	TypeInternal       = NewType("INTERNAL_ERROR", langext.Ptr(500)) | ||||||
|  | 	TypePanic          = NewType("PANIC", langext.Ptr(500)) | ||||||
|  | 	TypeNotImplemented = NewType("NOT_IMPLEMENTED", langext.Ptr(500)) | ||||||
|  |  | ||||||
|  | 	TypeMongoQuery        = NewType("MONGO_QUERY", langext.Ptr(500)) | ||||||
|  | 	TypeCursorTokenDecode = NewType("CURSOR_TOKEN_DECODE", langext.Ptr(500)) | ||||||
|  | 	TypeMongoFilter       = NewType("MONGO_FILTER", langext.Ptr(500)) | ||||||
|  | 	TypeMongoReflection   = NewType("MONGO_REFLECTION", langext.Ptr(500)) | ||||||
|  | 	TypeMongoInvalidOpt   = NewType("MONGO_INVALIDOPT", langext.Ptr(500)) | ||||||
|  |  | ||||||
|  | 	TypeSQLQuery  = NewType("SQL_QUERY", langext.Ptr(500)) | ||||||
|  | 	TypeSQLBuild  = NewType("SQL_BUILD", langext.Ptr(500)) | ||||||
|  | 	TypeSQLDecode = NewType("SQL_DECODE", langext.Ptr(500)) | ||||||
|  |  | ||||||
|  | 	TypeWrap = NewType("Wrap", nil) | ||||||
|  |  | ||||||
|  | 	TypeBindFailURI      = NewType("BINDFAIL_URI", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailQuery    = NewType("BINDFAIL_QUERY", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailJSON     = NewType("BINDFAIL_JSON", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailFormData = NewType("BINDFAIL_FORMDATA", langext.Ptr(400)) | ||||||
|  | 	TypeBindFailHeader   = NewType("BINDFAIL_HEADER", langext.Ptr(400)) | ||||||
|  |  | ||||||
|  | 	TypeMarshalEntityID = NewType("MARSHAL_ENTITY_ID", langext.Ptr(400)) | ||||||
|  | 	TypeInvalidCSID     = NewType("INVALID_CSID", langext.Ptr(400)) | ||||||
|  |  | ||||||
|  | 	TypeGoogleStatuscode = NewType("GOOGLE_STATUSCODE", langext.Ptr(400)) | ||||||
|  | 	TypeGoogleResponse   = NewType("GOOGLE_RESPONSE", langext.Ptr(400)) | ||||||
|  |  | ||||||
|  | 	TypeUnauthorized = NewType("UNAUTHORIZED", langext.Ptr(401)) | ||||||
|  | 	TypeAuthFailed   = NewType("AUTH_FAILED", langext.Ptr(401)) | ||||||
|  |  | ||||||
|  | 	TypeInvalidImage    = NewType("IMAGEEXT_INVALID_IMAGE", langext.Ptr(400)) | ||||||
|  | 	TypeInvalidMimeType = NewType("IMAGEEXT_INVALID_MIMETYPE", langext.Ptr(400)) | ||||||
|  |  | ||||||
|  | 	// other values come from the downstream application that uses goext | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (e *ErrorType) UnmarshalJSON(bytes []byte) error { | ||||||
|  | 	var k string | ||||||
|  | 	err := json.Unmarshal(bytes, &k) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if d, ok := registeredTypes.Get(k); ok { | ||||||
|  | 		*e = d | ||||||
|  | 		return nil | ||||||
|  | 	} else { | ||||||
|  | 		*e = ErrorType{k, nil} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorType) MarshalJSON() ([]byte, error) { | ||||||
|  | 	return json.Marshal(e.Key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e *ErrorType) UnmarshalBSONValue(bt bsontype.Type, data []byte) error { | ||||||
|  | 	if bt == bson.TypeNull { | ||||||
|  | 		// we can't set nil in UnmarshalBSONValue (so we use default(struct)) | ||||||
|  | 		// Use mongoext.CreateGoExtBsonRegistry if you need to unmarsh pointer values | ||||||
|  | 		// https://stackoverflow.com/questions/75167597 | ||||||
|  | 		// https://jira.mongodb.org/browse/GODRIVER-2252 | ||||||
|  | 		*e = ErrorType{} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	if bt != bson.TypeString { | ||||||
|  | 		return errors.New(fmt.Sprintf("cannot unmarshal %v into String", bt)) | ||||||
|  | 	} | ||||||
|  | 	var tt string | ||||||
|  | 	err := bson.RawValue{Type: bt, Value: data}.Unmarshal(&tt) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if d, ok := registeredTypes.Get(tt); ok { | ||||||
|  | 		*e = d | ||||||
|  | 		return nil | ||||||
|  | 	} else { | ||||||
|  | 		*e = ErrorType{tt, nil} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorType) MarshalBSONValue() (bsontype.Type, []byte, error) { | ||||||
|  | 	return bson.MarshalValue(e.Key) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ErrorType) DecodeValue(dc bsoncodec.DecodeContext, vr bsonrw.ValueReader, val reflect.Value) error { | ||||||
|  | 	if val.Kind() == reflect.Ptr && val.IsNil() { | ||||||
|  | 		if !val.CanSet() { | ||||||
|  | 			return errors.New("ValueUnmarshalerDecodeValue") | ||||||
|  | 		} | ||||||
|  | 		val.Set(reflect.New(val.Type().Elem())) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	tp, src, err := bsonrw.Copier{}.CopyValueToBytes(vr) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if val.Kind() == reflect.Ptr && len(src) == 0 { | ||||||
|  | 		val.Set(reflect.Zero(val.Type())) | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = e.UnmarshalBSONValue(tp, src) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if val.Kind() == reflect.Ptr { | ||||||
|  | 		val.Set(reflect.ValueOf(&e)) | ||||||
|  | 	} else { | ||||||
|  | 		val.Set(reflect.ValueOf(e)) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var registeredTypes = dataext.SyncMap[string, ErrorType]{} | ||||||
|  |  | ||||||
|  | func NewType(key string, defStatusCode *int) ErrorType { | ||||||
|  | 	et := ErrorType{key, defStatusCode} | ||||||
|  |  | ||||||
|  | 	registeredTypes.Set(key, et) | ||||||
|  |  | ||||||
|  | 	return et | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ListRegisteredTypes() []ErrorType { | ||||||
|  | 	return registeredTypes.GetAllValues() | ||||||
|  | } | ||||||
							
								
								
									
										153
									
								
								exerr/data_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										153
									
								
								exerr/data_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,153 @@ | |||||||
|  | package exerr | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson" | ||||||
|  | 	"go.mongodb.org/mongo-driver/bson/primitive" | ||||||
|  | 	"go.mongodb.org/mongo-driver/mongo" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/tst" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func TestJSONMarshalErrorCategory(t *testing.T) { | ||||||
|  |  | ||||||
|  | 	c1 := CatSystem | ||||||
|  |  | ||||||
|  | 	jsonbin := tst.Must(json.Marshal(c1))(t) | ||||||
|  |  | ||||||
|  | 	var c2 ErrorCategory | ||||||
|  | 	tst.AssertNoErr(t, json.Unmarshal(jsonbin, &c2)) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, c1, c2) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, string(jsonbin), "\"System\"") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestJSONMarshalErrorSeverity(t *testing.T) { | ||||||
|  |  | ||||||
|  | 	c1 := SevErr | ||||||
|  |  | ||||||
|  | 	jsonbin := tst.Must(json.Marshal(c1))(t) | ||||||
|  |  | ||||||
|  | 	var c2 ErrorSeverity | ||||||
|  | 	tst.AssertNoErr(t, json.Unmarshal(jsonbin, &c2)) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, c1, c2) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, string(jsonbin), "\"Err\"") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestJSONMarshalErrorType(t *testing.T) { | ||||||
|  |  | ||||||
|  | 	c1 := TypeNotImplemented | ||||||
|  |  | ||||||
|  | 	jsonbin := tst.Must(json.Marshal(c1))(t) | ||||||
|  |  | ||||||
|  | 	var c2 ErrorType | ||||||
|  | 	tst.AssertNoErr(t, json.Unmarshal(jsonbin, &c2)) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, c1, c2) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, string(jsonbin), "\"NOT_IMPLEMENTED\"") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestBSONMarshalErrorCategory(t *testing.T) { | ||||||
|  | 	ctx, cancel := context.WithTimeout(context.Background(), 350*time.Millisecond) | ||||||
|  | 	defer cancel() | ||||||
|  |  | ||||||
|  | 	client, err := mongo.Connect(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Skip("Skip test - no local mongo found") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	err = client.Ping(ctx, nil) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Skip("Skip test - no local mongo found") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	primimd := primitive.NewObjectID() | ||||||
|  |  | ||||||
|  | 	_, err = client.Database("_test").Collection("goext-cicd").InsertOne(ctx, bson.M{"_id": primimd, "val": CatSystem}) | ||||||
|  | 	tst.AssertNoErr(t, err) | ||||||
|  |  | ||||||
|  | 	cursor := client.Database("_test").Collection("goext-cicd").FindOne(ctx, bson.M{"_id": primimd, "val": bson.M{"$type": "string"}}) | ||||||
|  |  | ||||||
|  | 	var c1 struct { | ||||||
|  | 		ID  primitive.ObjectID `bson:"_id"` | ||||||
|  | 		Val ErrorCategory      `bson:"val"` | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = cursor.Decode(&c1) | ||||||
|  | 	tst.AssertNoErr(t, err) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, c1.Val, CatSystem) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestBSONMarshalErrorSeverity(t *testing.T) { | ||||||
|  | 	ctx, cancel := context.WithTimeout(context.Background(), 350*time.Millisecond) | ||||||
|  | 	defer cancel() | ||||||
|  |  | ||||||
|  | 	client, err := mongo.Connect(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Skip("Skip test - no local mongo found") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	err = client.Ping(ctx, nil) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Skip("Skip test - no local mongo found") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	primimd := primitive.NewObjectID() | ||||||
|  |  | ||||||
|  | 	_, err = client.Database("_test").Collection("goext-cicd").InsertOne(ctx, bson.M{"_id": primimd, "val": SevErr}) | ||||||
|  | 	tst.AssertNoErr(t, err) | ||||||
|  |  | ||||||
|  | 	cursor := client.Database("_test").Collection("goext-cicd").FindOne(ctx, bson.M{"_id": primimd, "val": bson.M{"$type": "string"}}) | ||||||
|  |  | ||||||
|  | 	var c1 struct { | ||||||
|  | 		ID  primitive.ObjectID `bson:"_id"` | ||||||
|  | 		Val ErrorSeverity      `bson:"val"` | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = cursor.Decode(&c1) | ||||||
|  | 	tst.AssertNoErr(t, err) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, c1.Val, SevErr) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestBSONMarshalErrorType(t *testing.T) { | ||||||
|  | 	ctx, cancel := context.WithTimeout(context.Background(), 350*time.Millisecond) | ||||||
|  | 	defer cancel() | ||||||
|  |  | ||||||
|  | 	client, err := mongo.Connect(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Skip("Skip test - no local mongo found") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	err = client.Ping(ctx, nil) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Skip("Skip test - no local mongo found") | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	primimd := primitive.NewObjectID() | ||||||
|  |  | ||||||
|  | 	_, err = client.Database("_test").Collection("goext-cicd").InsertOne(ctx, bson.M{"_id": primimd, "val": TypeNotImplemented}) | ||||||
|  | 	tst.AssertNoErr(t, err) | ||||||
|  |  | ||||||
|  | 	cursor := client.Database("_test").Collection("goext-cicd").FindOne(ctx, bson.M{"_id": primimd, "val": bson.M{"$type": "string"}}) | ||||||
|  |  | ||||||
|  | 	var c1 struct { | ||||||
|  | 		ID  primitive.ObjectID `bson:"_id"` | ||||||
|  | 		Val ErrorType          `bson:"val"` | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = cursor.Decode(&c1) | ||||||
|  | 	tst.AssertNoErr(t, err) | ||||||
|  |  | ||||||
|  | 	tst.AssertEqual(t, c1.Val, TypeNotImplemented) | ||||||
|  | } | ||||||
| @@ -2,10 +2,19 @@ package exerr | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/tst" | 	"gogs.mikescher.com/BlackForestBytes/goext/tst" | ||||||
|  | 	"os" | ||||||
| 	"testing" | 	"testing" | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | func TestMain(m *testing.M) { | ||||||
|  | 	if !Initialized() { | ||||||
|  | 		Init(ErrorPackageConfigInit{ZeroLogErrTraces: langext.PFalse, ZeroLogAllTraces: langext.PFalse}) | ||||||
|  | 	} | ||||||
|  | 	os.Exit(m.Run()) | ||||||
|  | } | ||||||
|  |  | ||||||
| type golangErr struct { | type golangErr struct { | ||||||
| 	Message string | 	Message string | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,15 +4,6 @@ import ( | |||||||
| 	"sync" | 	"sync" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Method string |  | ||||||
|  |  | ||||||
| const ( |  | ||||||
| 	MethodOutput Method = "OUTPUT" |  | ||||||
| 	MethodPrint  Method = "PRINT" |  | ||||||
| 	MethodBuild  Method = "BUILD" |  | ||||||
| 	MethodFatal  Method = "FATAL" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| type Listener = func(method Method, v *ExErr) | type Listener = func(method Method, v *ExErr) | ||||||
|  |  | ||||||
| var listenerLock = sync.Mutex{} | var listenerLock = sync.Mutex{} | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							| @@ -15,6 +15,7 @@ require ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| require ( | require ( | ||||||
|  | 	github.com/disintegration/imaging v1.6.2 | ||||||
| 	github.com/jung-kurt/gofpdf v1.16.2 | 	github.com/jung-kurt/gofpdf v1.16.2 | ||||||
| 	golang.org/x/sync v0.7.0 | 	golang.org/x/sync v0.7.0 | ||||||
| ) | ) | ||||||
| @@ -53,6 +54,7 @@ require ( | |||||||
| 	github.com/xdg-go/stringprep v1.0.4 // indirect | 	github.com/xdg-go/stringprep v1.0.4 // indirect | ||||||
| 	github.com/youmark/pkcs8 v0.0.0-20240424034433-3c2c7870ae76 // indirect | 	github.com/youmark/pkcs8 v0.0.0-20240424034433-3c2c7870ae76 // indirect | ||||||
| 	golang.org/x/arch v0.8.0 // indirect | 	golang.org/x/arch v0.8.0 // indirect | ||||||
|  | 	golang.org/x/image v0.16.0 // indirect | ||||||
| 	golang.org/x/net v0.25.0 // indirect | 	golang.org/x/net v0.25.0 // indirect | ||||||
| 	golang.org/x/text v0.15.0 // indirect | 	golang.org/x/text v0.15.0 // indirect | ||||||
| 	google.golang.org/protobuf v1.34.1 // indirect | 	google.golang.org/protobuf v1.34.1 // indirect | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								go.sum
									
									
									
									
									
								
							| @@ -43,6 +43,8 @@ github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSV | |||||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
|  | github.com/disintegration/imaging v1.6.2 h1:w1LecBlG2Lnp8B3jk5zSuNqd7b4DXhcjwek1ei82L+c= | ||||||
|  | github.com/disintegration/imaging v1.6.2/go.mod h1:44/5580QXChDfwIclfc/PCwrr44amcmDAg8hxG0Ewe4= | ||||||
| github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= | github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY= | ||||||
| github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= | github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto= | ||||||
| github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0= | github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0= | ||||||
| @@ -222,6 +224,10 @@ golang.org/x/crypto v0.22.0/go.mod h1:vr6Su+7cTlO45qkww3VDJlzDn0ctJvRgYbC2NvXHt+ | |||||||
| golang.org/x/crypto v0.23.0 h1:dIJU/v2J8Mdglj/8rJ6UUOM3Zc9zLZxVZwwxMooUSAI= | golang.org/x/crypto v0.23.0 h1:dIJU/v2J8Mdglj/8rJ6UUOM3Zc9zLZxVZwwxMooUSAI= | ||||||
| golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= | golang.org/x/crypto v0.23.0/go.mod h1:CKFgDieR+mRhux2Lsu27y0fO304Db0wZe70UKqHu0v8= | ||||||
| golang.org/x/image v0.0.0-20190910094157-69e4b8554b2a/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | golang.org/x/image v0.0.0-20190910094157-69e4b8554b2a/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||||
|  | golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8 h1:hVwzHzIUGRjiF7EcUjqNxk3NCfkPxbDKRdnNE1Rpg0U= | ||||||
|  | golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||||
|  | golang.org/x/image v0.16.0 h1:9kloLAKhUufZhA12l5fwnx2NZW39/we1UhBesW433jw= | ||||||
|  | golang.org/x/image v0.16.0/go.mod h1:ugSZItdV4nOxyqp56HmXwH0Ry0nBCpjnZdpDaIHdoPs= | ||||||
| golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= | ||||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| package goext | package goext | ||||||
|  |  | ||||||
| const GoextVersion = "0.0.450" | const GoextVersion = "0.0.457" | ||||||
|  |  | ||||||
| const GoextVersionTimestamp = "2024-05-14T11:52:56+0200" | const GoextVersionTimestamp = "2024-05-20T00:07:33+0200" | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								imageext/enums.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								imageext/enums.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | package imageext | ||||||
|  |  | ||||||
|  | //go:generate go run ../_gen/enum-generate.go -- enums_gen.go | ||||||
							
								
								
									
										216
									
								
								imageext/enums_gen.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										216
									
								
								imageext/enums_gen.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,216 @@ | |||||||
|  | // Code generated by enum-generate.go DO NOT EDIT. | ||||||
|  |  | ||||||
|  | package imageext | ||||||
|  |  | ||||||
|  | import "gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | import "gogs.mikescher.com/BlackForestBytes/goext/enums" | ||||||
|  |  | ||||||
|  | const ChecksumEnumGenerator = "1da5383c33ee442fd0b899369053f66bdc85bed2dbf906949d3edfeedfe13340" // GoExtVersion: 0.0.449 | ||||||
|  |  | ||||||
|  | // ================================ ImageFit ================================ | ||||||
|  | // | ||||||
|  | // File:       image.go | ||||||
|  | // StringEnum: true | ||||||
|  | // DescrEnum:  false | ||||||
|  | // DataEnum:   false | ||||||
|  | // | ||||||
|  |  | ||||||
|  | var __ImageFitValues = []ImageFit{ | ||||||
|  | 	ImageFitStretch, | ||||||
|  | 	ImageFitCover, | ||||||
|  | 	ImageFitContainCenter, | ||||||
|  | 	ImageFitContainTopLeft, | ||||||
|  | 	ImageFitContainTopRight, | ||||||
|  | 	ImageFitContainBottomLeft, | ||||||
|  | 	ImageFitContainBottomRight, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var __ImageFitVarnames = map[ImageFit]string{ | ||||||
|  | 	ImageFitStretch:            "ImageFitStretch", | ||||||
|  | 	ImageFitCover:              "ImageFitCover", | ||||||
|  | 	ImageFitContainCenter:      "ImageFitContainCenter", | ||||||
|  | 	ImageFitContainTopLeft:     "ImageFitContainTopLeft", | ||||||
|  | 	ImageFitContainTopRight:    "ImageFitContainTopRight", | ||||||
|  | 	ImageFitContainBottomLeft:  "ImageFitContainBottomLeft", | ||||||
|  | 	ImageFitContainBottomRight: "ImageFitContainBottomRight", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) Valid() bool { | ||||||
|  | 	return langext.InArray(e, __ImageFitValues) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) Values() []ImageFit { | ||||||
|  | 	return __ImageFitValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) ValuesAny() []any { | ||||||
|  | 	return langext.ArrCastToAny(__ImageFitValues) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) ValuesMeta() []enums.EnumMetaValue { | ||||||
|  | 	return ImageFitValuesMeta() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) String() string { | ||||||
|  | 	return string(e) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) VarName() string { | ||||||
|  | 	if d, ok := __ImageFitVarnames[e]; ok { | ||||||
|  | 		return d | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) TypeName() string { | ||||||
|  | 	return "ImageFit" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) PackageName() string { | ||||||
|  | 	return "media" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageFit) Meta() enums.EnumMetaValue { | ||||||
|  | 	return enums.EnumMetaValue{VarName: e.VarName(), Value: e, Description: nil} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ParseImageFit(vv string) (ImageFit, bool) { | ||||||
|  | 	for _, ev := range __ImageFitValues { | ||||||
|  | 		if string(ev) == vv { | ||||||
|  | 			return ev, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return "", false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ImageFitValues() []ImageFit { | ||||||
|  | 	return __ImageFitValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ImageFitValuesMeta() []enums.EnumMetaValue { | ||||||
|  | 	return []enums.EnumMetaValue{ | ||||||
|  | 		ImageFitStretch.Meta(), | ||||||
|  | 		ImageFitCover.Meta(), | ||||||
|  | 		ImageFitContainCenter.Meta(), | ||||||
|  | 		ImageFitContainTopLeft.Meta(), | ||||||
|  | 		ImageFitContainTopRight.Meta(), | ||||||
|  | 		ImageFitContainBottomLeft.Meta(), | ||||||
|  | 		ImageFitContainBottomRight.Meta(), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ================================ ImageCompresson ================================ | ||||||
|  | // | ||||||
|  | // File:       image.go | ||||||
|  | // StringEnum: true | ||||||
|  | // DescrEnum:  false | ||||||
|  | // DataEnum:   false | ||||||
|  | // | ||||||
|  |  | ||||||
|  | var __ImageCompressonValues = []ImageCompresson{ | ||||||
|  | 	CompressionPNGNone, | ||||||
|  | 	CompressionPNGSpeed, | ||||||
|  | 	CompressionPNGBest, | ||||||
|  | 	CompressionJPEG100, | ||||||
|  | 	CompressionJPEG90, | ||||||
|  | 	CompressionJPEG80, | ||||||
|  | 	CompressionJPEG70, | ||||||
|  | 	CompressionJPEG60, | ||||||
|  | 	CompressionJPEG50, | ||||||
|  | 	CompressionJPEG25, | ||||||
|  | 	CompressionJPEG10, | ||||||
|  | 	CompressionJPEG1, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var __ImageCompressonVarnames = map[ImageCompresson]string{ | ||||||
|  | 	CompressionPNGNone:  "CompressionPNGNone", | ||||||
|  | 	CompressionPNGSpeed: "CompressionPNGSpeed", | ||||||
|  | 	CompressionPNGBest:  "CompressionPNGBest", | ||||||
|  | 	CompressionJPEG100:  "CompressionJPEG100", | ||||||
|  | 	CompressionJPEG90:   "CompressionJPEG90", | ||||||
|  | 	CompressionJPEG80:   "CompressionJPEG80", | ||||||
|  | 	CompressionJPEG70:   "CompressionJPEG70", | ||||||
|  | 	CompressionJPEG60:   "CompressionJPEG60", | ||||||
|  | 	CompressionJPEG50:   "CompressionJPEG50", | ||||||
|  | 	CompressionJPEG25:   "CompressionJPEG25", | ||||||
|  | 	CompressionJPEG10:   "CompressionJPEG10", | ||||||
|  | 	CompressionJPEG1:    "CompressionJPEG1", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) Valid() bool { | ||||||
|  | 	return langext.InArray(e, __ImageCompressonValues) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) Values() []ImageCompresson { | ||||||
|  | 	return __ImageCompressonValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) ValuesAny() []any { | ||||||
|  | 	return langext.ArrCastToAny(__ImageCompressonValues) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) ValuesMeta() []enums.EnumMetaValue { | ||||||
|  | 	return ImageCompressonValuesMeta() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) String() string { | ||||||
|  | 	return string(e) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) VarName() string { | ||||||
|  | 	if d, ok := __ImageCompressonVarnames[e]; ok { | ||||||
|  | 		return d | ||||||
|  | 	} | ||||||
|  | 	return "" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) TypeName() string { | ||||||
|  | 	return "ImageCompresson" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) PackageName() string { | ||||||
|  | 	return "media" | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (e ImageCompresson) Meta() enums.EnumMetaValue { | ||||||
|  | 	return enums.EnumMetaValue{VarName: e.VarName(), Value: e, Description: nil} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ParseImageCompresson(vv string) (ImageCompresson, bool) { | ||||||
|  | 	for _, ev := range __ImageCompressonValues { | ||||||
|  | 		if string(ev) == vv { | ||||||
|  | 			return ev, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return "", false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ImageCompressonValues() []ImageCompresson { | ||||||
|  | 	return __ImageCompressonValues | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ImageCompressonValuesMeta() []enums.EnumMetaValue { | ||||||
|  | 	return []enums.EnumMetaValue{ | ||||||
|  | 		CompressionPNGNone.Meta(), | ||||||
|  | 		CompressionPNGSpeed.Meta(), | ||||||
|  | 		CompressionPNGBest.Meta(), | ||||||
|  | 		CompressionJPEG100.Meta(), | ||||||
|  | 		CompressionJPEG90.Meta(), | ||||||
|  | 		CompressionJPEG80.Meta(), | ||||||
|  | 		CompressionJPEG70.Meta(), | ||||||
|  | 		CompressionJPEG60.Meta(), | ||||||
|  | 		CompressionJPEG50.Meta(), | ||||||
|  | 		CompressionJPEG25.Meta(), | ||||||
|  | 		CompressionJPEG10.Meta(), | ||||||
|  | 		CompressionJPEG1.Meta(), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ================================ ================= ================================ | ||||||
|  |  | ||||||
|  | func AllPackageEnums() []enums.Enum { | ||||||
|  | 	return []enums.Enum{ | ||||||
|  | 		ImageFitStretch,    // ImageFit | ||||||
|  | 		CompressionPNGNone, // ImageCompresson | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										317
									
								
								imageext/image.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										317
									
								
								imageext/image.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,317 @@ | |||||||
|  | package imageext | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
|  | 	"github.com/disintegration/imaging" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/mathext" | ||||||
|  | 	"image" | ||||||
|  | 	"image/color" | ||||||
|  | 	"image/draw" | ||||||
|  | 	"image/jpeg" | ||||||
|  | 	"image/png" | ||||||
|  | 	"io" | ||||||
|  | 	"math" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ImageFit string //@enum:type | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	ImageFitStretch            ImageFit = "STRETCH" | ||||||
|  | 	ImageFitCover              ImageFit = "COVER" | ||||||
|  | 	ImageFitContainCenter      ImageFit = "CONTAIN_CENTER" | ||||||
|  | 	ImageFitContainTopLeft     ImageFit = "CONTAIN_TOPLEFT" | ||||||
|  | 	ImageFitContainTopRight    ImageFit = "CONTAIN_TOPRIGHT" | ||||||
|  | 	ImageFitContainBottomLeft  ImageFit = "CONTAIN_BOTTOMLEFT" | ||||||
|  | 	ImageFitContainBottomRight ImageFit = "CONTAIN_BOTTOMRIGHT" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ImageCrop struct { // all crop values are percentages! | ||||||
|  |  | ||||||
|  | 	CropX      float64 `bson:"cropX"      json:"cropX"` | ||||||
|  | 	CropY      float64 `bson:"cropY"      json:"cropY"` | ||||||
|  | 	CropWidth  float64 `bson:"cropWidth"  json:"cropWidth"` | ||||||
|  | 	CropHeight float64 `bson:"cropHeight" json:"cropHeight"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type ImageCompresson string //@enum:type | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	CompressionPNGNone  ImageCompresson = "PNG_NONE" | ||||||
|  | 	CompressionPNGSpeed ImageCompresson = "PNG_SPEED" | ||||||
|  | 	CompressionPNGBest  ImageCompresson = "PNG_BEST" | ||||||
|  | 	CompressionJPEG100  ImageCompresson = "JPEG_100" | ||||||
|  | 	CompressionJPEG90   ImageCompresson = "JPEG_090" | ||||||
|  | 	CompressionJPEG80   ImageCompresson = "JPEG_080" | ||||||
|  | 	CompressionJPEG70   ImageCompresson = "JPEG_070" | ||||||
|  | 	CompressionJPEG60   ImageCompresson = "JPEG_060" | ||||||
|  | 	CompressionJPEG50   ImageCompresson = "JPEG_050" | ||||||
|  | 	CompressionJPEG25   ImageCompresson = "JPEG_025" | ||||||
|  | 	CompressionJPEG10   ImageCompresson = "JPEG_010" | ||||||
|  | 	CompressionJPEG1    ImageCompresson = "JPEG_001" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func CropImage(img image.Image, px float64, py float64, pw float64, ph float64) (image.Image, error) { | ||||||
|  |  | ||||||
|  | 	type subImager interface { | ||||||
|  | 		SubImage(r image.Rectangle) image.Image | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	x := int(float64(img.Bounds().Dx()) * px) | ||||||
|  | 	y := int(float64(img.Bounds().Dy()) * py) | ||||||
|  | 	w := int(float64(img.Bounds().Dx()) * pw) | ||||||
|  | 	h := int(float64(img.Bounds().Dy()) * ph) | ||||||
|  |  | ||||||
|  | 	if simg, ok := img.(subImager); ok { | ||||||
|  |  | ||||||
|  | 		return simg.SubImage(image.Rect(x, y, x+w, y+h)), nil | ||||||
|  |  | ||||||
|  | 	} else { | ||||||
|  |  | ||||||
|  | 		bfr1 := bytes.Buffer{} | ||||||
|  | 		err := png.Encode(&bfr1, img) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		imgPNG, err := png.Decode(&bfr1) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return imgPNG.(subImager).SubImage(image.Rect(x, y, w+w, y+h)), nil | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func EncodeImage(img image.Image, compression ImageCompresson) (bytes.Buffer, string, error) { | ||||||
|  | 	var err error | ||||||
|  |  | ||||||
|  | 	bfr := bytes.Buffer{} | ||||||
|  |  | ||||||
|  | 	switch compression { | ||||||
|  | 	case CompressionPNGNone: | ||||||
|  | 		enc := &png.Encoder{CompressionLevel: png.NoCompression} | ||||||
|  | 		err = enc.Encode(&bfr, img) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/png", nil | ||||||
|  | 	case CompressionPNGSpeed: | ||||||
|  | 		enc := &png.Encoder{CompressionLevel: png.BestSpeed} | ||||||
|  | 		err = enc.Encode(&bfr, img) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/png", nil | ||||||
|  | 	case CompressionPNGBest: | ||||||
|  | 		enc := &png.Encoder{CompressionLevel: png.BestCompression} | ||||||
|  | 		err = enc.Encode(&bfr, img) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/png", nil | ||||||
|  | 	case CompressionJPEG100: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 100}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG90: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 90}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG80: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 80}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG70: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 70}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG60: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 60}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG50: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 50}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG25: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 25}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG10: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 10}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	case CompressionJPEG1: | ||||||
|  | 		err = jpeg.Encode(&bfr, img, &jpeg.Options{Quality: 1}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return bytes.Buffer{}, "", exerr.Wrap(err, "").Build() | ||||||
|  | 		} | ||||||
|  | 		return bfr, "image/jpeg", nil | ||||||
|  | 	default: | ||||||
|  | 		return bytes.Buffer{}, "", exerr.New(exerr.TypeInternal, "unknown compression method: "+compression.String()).Build() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ObjectFitImage(img image.Image, bbw float64, bbh float64, fit ImageFit, fillColor color.Color) (image.Image, error) { | ||||||
|  |  | ||||||
|  | 	iw := img.Bounds().Size().X | ||||||
|  | 	ih := img.Bounds().Size().Y | ||||||
|  |  | ||||||
|  | 	// [iw, ih]   is the size of the image | ||||||
|  | 	// [bbw, bbh] is the target bounding box, | ||||||
|  | 	//             - it specifies the target ratio | ||||||
|  | 	//             - and the maximal target resolution | ||||||
|  |  | ||||||
|  | 	facW := float64(iw) / bbw | ||||||
|  | 	facH := float64(ih) / bbh | ||||||
|  |  | ||||||
|  | 	// facW is the ratio between iw and bbw | ||||||
|  | 	//  - it is the factor by which the bounding box must be multiplied to reach the image size (in the x-axis) | ||||||
|  | 	// | ||||||
|  | 	// (same is true for facH, but for the height and y-axis) | ||||||
|  |  | ||||||
|  | 	if fit == ImageFitCover { | ||||||
|  |  | ||||||
|  | 		// image-fit:cover completely fills the target-bounding-box, it potentially cuts parts of the image away | ||||||
|  |  | ||||||
|  | 		// we use the smaller (!) value of facW and facH, because we want to have the smallest possible destination rect (due to file size) | ||||||
|  | 		// and because the image is made to completely fill the bounding-box, the smaller factor (= teh dimension the image is stretched more) is relevant | ||||||
|  |  | ||||||
|  | 		// but we cap `fac` at 1 (can be larger than 1) | ||||||
|  | 		// a value >1 would mean the final image resolution is biger than the bounding box, which we do not want. | ||||||
|  |  | ||||||
|  | 		// if the initial image (iw, ih) is already bigger than the bounding box (bbw, bbh), facW and facH are always >1 and fac will be 1 | ||||||
|  | 		// which means we will simply use the bounding box as destination rect (and scale the image down) | ||||||
|  |  | ||||||
|  | 		fac := mathext.Clamp(mathext.Min(facW, facH), 0.0, 1.0) | ||||||
|  |  | ||||||
|  | 		// we scale the bounding box by fac (both dimension the same amount, to keep the bounding-box ratio) | ||||||
|  |  | ||||||
|  | 		w := int(math.Round(bbw * fac)) | ||||||
|  | 		h := int(math.Round(bbh * fac)) | ||||||
|  |  | ||||||
|  | 		img = imaging.Fill(img, w, h, imaging.Center, imaging.Lanczos) | ||||||
|  |  | ||||||
|  | 		newImg := image.NewRGBA(image.Rect(0, 0, w, h)) | ||||||
|  |  | ||||||
|  | 		draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||||
|  | 		draw.Draw(newImg, newImg.Bounds(), img, image.Pt(0, 0), draw.Over) | ||||||
|  |  | ||||||
|  | 		return newImg, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if fit == ImageFitContainCenter || fit == ImageFitContainTopLeft || fit == ImageFitContainTopRight || fit == ImageFitContainBottomLeft || fit == ImageFitContainBottomRight { | ||||||
|  |  | ||||||
|  | 		// image-fit:cover fills the target-bounding-box with the image, there is potentially empty-space, it potentially cuts parts of the image away | ||||||
|  |  | ||||||
|  | 		// we use the bigger (!) value of facW and facH, | ||||||
|  | 		// because the image is made to fit the bounding-box, the bigger factor (= the dimension the image is stretched less) is relevant | ||||||
|  |  | ||||||
|  | 		// but we cap `fac` at 1 (can be larger than 1) | ||||||
|  | 		// a value >1 would mean the final image resolution is biger than the bounding box, which we do not want. | ||||||
|  |  | ||||||
|  | 		// if the initial image (iw, ih) is already bigger than the bounding box (bbw, bbh), facW and facH are always >1 and fac will be 1 | ||||||
|  | 		// which means we will simply use the bounding box as destination rect (and scale the image down) | ||||||
|  |  | ||||||
|  | 		facOut := mathext.Clamp(mathext.Max(facW, facH), 0.0, 1.0) | ||||||
|  |  | ||||||
|  | 		// we scale the bounding box by fac (both dimension the same amount, to keep the bounding-box ratio) | ||||||
|  |  | ||||||
|  | 		ow := int(math.Round(bbw * facOut)) | ||||||
|  | 		oh := int(math.Round(bbh * facOut)) | ||||||
|  |  | ||||||
|  | 		facScale := mathext.Min(float64(ow)/float64(iw), float64(oh)/float64(ih)) | ||||||
|  |  | ||||||
|  | 		dw := int(math.Round(float64(iw) * facScale)) | ||||||
|  | 		dh := int(math.Round(float64(ih) * facScale)) | ||||||
|  |  | ||||||
|  | 		img = imaging.Resize(img, dw, dh, imaging.Lanczos) | ||||||
|  |  | ||||||
|  | 		var destBounds image.Rectangle | ||||||
|  | 		if fit == ImageFitContainCenter { | ||||||
|  | 			destBounds = image.Rect((ow-dw)/2, (oh-dh)/2, (ow-dw)/2+dw, (oh-dh)/2+dh) | ||||||
|  | 		} else if fit == ImageFitContainTopLeft { | ||||||
|  | 			destBounds = image.Rect(0, 0, iw, dh) | ||||||
|  | 		} else if fit == ImageFitContainTopRight { | ||||||
|  | 			destBounds = image.Rect(ow-iw, 0, ow, dh) | ||||||
|  | 		} else if fit == ImageFitContainBottomLeft { | ||||||
|  | 			destBounds = image.Rect(0, oh-dh, iw, oh) | ||||||
|  | 		} else if fit == ImageFitContainBottomRight { | ||||||
|  | 			destBounds = image.Rect(ow-dw, oh-dh, ow, oh) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		newImg := image.NewRGBA(image.Rect(0, 0, ow, oh)) | ||||||
|  |  | ||||||
|  | 		draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||||
|  | 		draw.Draw(newImg, destBounds, img, image.Pt(0, 0), draw.Over) | ||||||
|  |  | ||||||
|  | 		return newImg, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if fit == ImageFitStretch { | ||||||
|  |  | ||||||
|  | 		// image-fit:stretch simply stretches the image to the bounding box | ||||||
|  |  | ||||||
|  | 		// we use the bigger value of [facW;facH], to (potentially) scale the bounding box down before applying it | ||||||
|  | 		// theoretically we could directly use [bbw, bbh] in the call to imaging.Resize, | ||||||
|  | 		// but if the image is (a lot) smaller than the bouding box it is useful to scale it down to reduce final pdf filesize | ||||||
|  |  | ||||||
|  | 		// we also cap fac at 1, because we never want the final rect to be bigger than the inputted bounding box (see comments at start of method) | ||||||
|  |  | ||||||
|  | 		fac := mathext.Clamp(mathext.Max(facW, facH), 0.0, 1.0) | ||||||
|  |  | ||||||
|  | 		// we scale the bounding box by fac (both dimension the same amount, to keep the bounding-box ratio) | ||||||
|  |  | ||||||
|  | 		w := int(math.Round(bbw * fac)) | ||||||
|  | 		h := int(math.Round(bbh * fac)) | ||||||
|  |  | ||||||
|  | 		img = imaging.Resize(img, w, h, imaging.Lanczos) | ||||||
|  |  | ||||||
|  | 		newImg := image.NewRGBA(image.Rect(0, 0, w, h)) | ||||||
|  |  | ||||||
|  | 		draw.Draw(newImg, newImg.Bounds(), &image.Uniform{C: fillColor}, image.Pt(0, 0), draw.Src) | ||||||
|  | 		draw.Draw(newImg, newImg.Bounds(), img, image.Pt(0, 0), draw.Over) | ||||||
|  |  | ||||||
|  | 		return newImg, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil, exerr.New(exerr.TypeInternal, fmt.Sprintf("unknown image-fit: '%s'", fit)).Build() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func VerifyAndDecodeImage(data io.Reader, mime string) (image.Image, error) { | ||||||
|  |  | ||||||
|  | 	if mime == "image/jpeg" { | ||||||
|  | 		img, err := jpeg.Decode(data) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, exerr.Wrap(err, "failed to decode blob as jpeg").WithType(exerr.TypeInvalidImage).Build() | ||||||
|  | 		} | ||||||
|  | 		return img, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if mime == "image/png" { | ||||||
|  | 		img, err := png.Decode(data) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, exerr.Wrap(err, "failed to decode blob as png").WithType(exerr.TypeInvalidImage).Build() | ||||||
|  | 		} | ||||||
|  | 		return img, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return nil, exerr.New(exerr.TypeInvalidMimeType, fmt.Sprintf("unknown/invalid image mimetype: '%s'", mime)).Build() | ||||||
|  | } | ||||||
| @@ -59,6 +59,18 @@ func ArrUnique[T comparable](array []T) []T { | |||||||
| 	return result | 	return result | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func ArrUniqueStable[T comparable](array []T) []T { | ||||||
|  | 	hist := make(map[T]bool, len(array)) | ||||||
|  | 	result := make([]T, 0, len(array)) | ||||||
|  | 	for _, v := range array { | ||||||
|  | 		if _, ok := hist[v]; !ok { | ||||||
|  | 			hist[v] = true | ||||||
|  | 			result = append(result, v) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return result | ||||||
|  | } | ||||||
|  |  | ||||||
| func ArrEqualsExact[T comparable](arr1 []T, arr2 []T) bool { | func ArrEqualsExact[T comparable](arr1 []T, arr2 []T) bool { | ||||||
| 	if len(arr1) != len(arr2) { | 	if len(arr1) != len(arr2) { | ||||||
| 		return false | 		return false | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import ( | |||||||
| 	"go.mongodb.org/mongo-driver/bson/bsoncodec" | 	"go.mongodb.org/mongo-driver/bson/bsoncodec" | ||||||
| 	"go.mongodb.org/mongo-driver/bson/bsontype" | 	"go.mongodb.org/mongo-driver/bson/bsontype" | ||||||
| 	"go.mongodb.org/mongo-driver/bson/primitive" | 	"go.mongodb.org/mongo-driver/bson/primitive" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/exerr" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/rfctime" | 	"gogs.mikescher.com/BlackForestBytes/goext/rfctime" | ||||||
| 	"reflect" | 	"reflect" | ||||||
| @@ -34,6 +35,15 @@ func CreateGoExtBsonRegistry() *bsoncodec.Registry { | |||||||
| 	rb.RegisterTypeDecoder(reflect.TypeOf(rfctime.SecondsF64(0)), rfctime.SecondsF64(0)) | 	rb.RegisterTypeDecoder(reflect.TypeOf(rfctime.SecondsF64(0)), rfctime.SecondsF64(0)) | ||||||
| 	rb.RegisterTypeDecoder(reflect.TypeOf(langext.Ptr(rfctime.SecondsF64(0))), rfctime.SecondsF64(0)) | 	rb.RegisterTypeDecoder(reflect.TypeOf(langext.Ptr(rfctime.SecondsF64(0))), rfctime.SecondsF64(0)) | ||||||
|  |  | ||||||
|  | 	rb.RegisterTypeDecoder(reflect.TypeOf(exerr.ErrorCategory{}), exerr.ErrorCategory{}) | ||||||
|  | 	rb.RegisterTypeDecoder(reflect.TypeOf(langext.Ptr(exerr.ErrorCategory{})), exerr.ErrorCategory{}) | ||||||
|  |  | ||||||
|  | 	rb.RegisterTypeDecoder(reflect.TypeOf(exerr.ErrorSeverity{}), exerr.ErrorSeverity{}) | ||||||
|  | 	rb.RegisterTypeDecoder(reflect.TypeOf(langext.Ptr(exerr.ErrorSeverity{})), exerr.ErrorSeverity{}) | ||||||
|  |  | ||||||
|  | 	rb.RegisterTypeDecoder(reflect.TypeOf(exerr.ErrorType{}), exerr.ErrorType{}) | ||||||
|  | 	rb.RegisterTypeDecoder(reflect.TypeOf(langext.Ptr(exerr.ErrorType{})), exerr.ErrorType{}) | ||||||
|  |  | ||||||
| 	bsoncodec.DefaultValueEncoders{}.RegisterDefaultEncoders(rb) | 	bsoncodec.DefaultValueEncoders{}.RegisterDefaultEncoders(rb) | ||||||
| 	bsoncodec.DefaultValueDecoders{}.RegisterDefaultDecoders(rb) | 	bsoncodec.DefaultValueDecoders{}.RegisterDefaultDecoders(rb) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								wpdf/wpdf.go
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								wpdf/wpdf.go
									
									
									
									
									
								
							| @@ -9,6 +9,7 @@ type WPDFBuilder struct { | |||||||
| 	b           *gofpdf.Fpdf | 	b           *gofpdf.Fpdf | ||||||
| 	tr          func(string) string | 	tr          func(string) string | ||||||
| 	cellHeight  float64 | 	cellHeight  float64 | ||||||
|  | 	cellSpacing float64 | ||||||
| 	fontName    PDFFontFamily | 	fontName    PDFFontFamily | ||||||
| 	fontStyle   PDFFontStyle | 	fontStyle   PDFFontStyle | ||||||
| 	fontSize    float64 | 	fontSize    float64 | ||||||
| @@ -41,6 +42,7 @@ func NewPDFBuilder(orientation PDFOrientation, size PDFSize, unicode bool) *WPDF | |||||||
| 		b:           fpdfbuilder, | 		b:           fpdfbuilder, | ||||||
| 		tr:          tr, | 		tr:          tr, | ||||||
| 		cellHeight:  5, | 		cellHeight:  5, | ||||||
|  | 		cellSpacing: 1, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	b.SetMargins(PDFMargins{Left: 15, Top: 25, Right: 15}) // default values | 	b.SetMargins(PDFMargins{Left: 15, Top: 25, Right: 15}) // default values | ||||||
| @@ -49,6 +51,10 @@ func NewPDFBuilder(orientation PDFOrientation, size PDFSize, unicode bool) *WPDF | |||||||
| 	return b | 	return b | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *WPDFBuilder) FPDF() *gofpdf.Fpdf { | ||||||
|  | 	return b.b | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *WPDFBuilder) SetMargins(v PDFMargins) { | func (b *WPDFBuilder) SetMargins(v PDFMargins) { | ||||||
| 	b.b.SetMargins(v.Left, v.Top, v.Right) | 	b.b.SetMargins(v.Left, v.Top, v.Right) | ||||||
| } | } | ||||||
| @@ -99,6 +105,10 @@ func (b *WPDFBuilder) SetFont(fontName PDFFontFamily, fontStyle PDFFontStyle, fo | |||||||
| 	b.cellHeight = b.b.PointConvert(fontSize) | 	b.cellHeight = b.b.PointConvert(fontSize) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (b *WPDFBuilder) SetCellSpacing(h float64) { | ||||||
|  | 	b.cellSpacing = h | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *WPDFBuilder) Ln(h float64) { | func (b *WPDFBuilder) Ln(h float64) { | ||||||
| 	b.b.Ln(h) | 	b.b.Ln(h) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -115,7 +115,7 @@ func (b *WPDFBuilder) Cell(txt string, opts ...*PDFCellOpt) { | |||||||
| 	txtTR := b.tr(txt) | 	txtTR := b.tr(txt) | ||||||
|  |  | ||||||
| 	width := float64(0) | 	width := float64(0) | ||||||
| 	height := b.cellHeight | 	height := b.cellHeight + b.cellSpacing | ||||||
| 	border := BorderNone | 	border := BorderNone | ||||||
| 	ln := BreakToNextLine | 	ln := BreakToNextLine | ||||||
| 	align := AlignLeft | 	align := AlignLeft | ||||||
|   | |||||||
| @@ -3,13 +3,21 @@ package wpdf | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"github.com/jung-kurt/gofpdf" | 	"github.com/jung-kurt/gofpdf" | ||||||
|  | 	"gogs.mikescher.com/BlackForestBytes/goext/imageext" | ||||||
| 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | 	"gogs.mikescher.com/BlackForestBytes/goext/langext" | ||||||
|  | 	"image" | ||||||
|  | 	"image/color" | ||||||
|  | 	"image/png" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"os" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type PDFImageRef struct { | type PDFImageRef struct { | ||||||
| 	Info  *gofpdf.ImageInfoType | 	Info  *gofpdf.ImageInfoType | ||||||
| 	Name  string | 	Name  string | ||||||
|  | 	Bin   []byte | ||||||
|  | 	Image *image.Image | ||||||
|  | 	Mime  string | ||||||
| } | } | ||||||
|  |  | ||||||
| type PDFImageRegisterOpt struct { | type PDFImageRegisterOpt struct { | ||||||
| @@ -48,6 +56,7 @@ func (b *WPDFBuilder) RegisterImage(bin []byte, opts ...*PDFImageRegisterOpt) *P | |||||||
| 	imageType := "" | 	imageType := "" | ||||||
| 	readDpi := false | 	readDpi := false | ||||||
| 	allowNegativePosition := false | 	allowNegativePosition := false | ||||||
|  | 	mime := "application/octet-stream" | ||||||
|  |  | ||||||
| 	for _, opt := range opts { | 	for _, opt := range opts { | ||||||
| 		imageType = langext.Coalesce(opt.imageType, imageType) | 		imageType = langext.Coalesce(opt.imageType, imageType) | ||||||
| @@ -61,12 +70,26 @@ func (b *WPDFBuilder) RegisterImage(bin []byte, opts ...*PDFImageRegisterOpt) *P | |||||||
| 		switch ct { | 		switch ct { | ||||||
| 		case "image/jpg": | 		case "image/jpg": | ||||||
| 			imageType = "JPG" | 			imageType = "JPG" | ||||||
|  | 			mime = ct | ||||||
| 		case "image/jpeg": | 		case "image/jpeg": | ||||||
| 			imageType = "JPEG" | 			imageType = "JPEG" | ||||||
|  | 			mime = ct | ||||||
| 		case "image/png": | 		case "image/png": | ||||||
| 			imageType = "PNG" | 			imageType = "PNG" | ||||||
|  | 			mime = ct | ||||||
| 		case "image/gif": | 		case "image/gif": | ||||||
| 			imageType = "GIF" | 			imageType = "GIF" | ||||||
|  | 			mime = ct | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		switch imageType { | ||||||
|  | 		case "JPG": | ||||||
|  | 		case "JPEG": | ||||||
|  | 			mime = "image/jpeg" | ||||||
|  | 		case "PNG": | ||||||
|  | 			mime = "image/png" | ||||||
|  | 		case "GIF": | ||||||
|  | 			mime = "image/gif" | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -81,6 +104,9 @@ func (b *WPDFBuilder) RegisterImage(bin []byte, opts ...*PDFImageRegisterOpt) *P | |||||||
| 	return &PDFImageRef{ | 	return &PDFImageRef{ | ||||||
| 		Name:  imgName, | 		Name:  imgName, | ||||||
| 		Info:  info, | 		Info:  info, | ||||||
|  | 		Bin:   bin, | ||||||
|  | 		Image: nil, | ||||||
|  | 		Mime:  mime, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -95,6 +121,11 @@ type PDFImageOpt struct { | |||||||
| 	imageType             *string | 	imageType             *string | ||||||
| 	readDpi               *bool | 	readDpi               *bool | ||||||
| 	allowNegativePosition *bool | 	allowNegativePosition *bool | ||||||
|  | 	imageFit              *imageext.ImageFit | ||||||
|  | 	fillColor             *color.Color | ||||||
|  | 	compression           *imageext.ImageCompresson | ||||||
|  | 	reEncodePixelPerMM    *float64 | ||||||
|  | 	crop                  *imageext.ImageCrop | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewPDFImageOpt() *PDFImageOpt { | func NewPDFImageOpt() *PDFImageOpt { | ||||||
| @@ -151,7 +182,40 @@ func (opt *PDFImageOpt) AllowNegativePosition(v bool) *PDFImageOpt { | |||||||
| 	return opt | 	return opt | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (opt *PDFImageOpt) ImageFit(v imageext.ImageFit) *PDFImageOpt { | ||||||
|  | 	opt.imageFit = &v | ||||||
|  | 	return opt | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (opt *PDFImageOpt) FillColor(v color.Color) *PDFImageOpt { | ||||||
|  | 	opt.fillColor = &v | ||||||
|  | 	return opt | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (opt *PDFImageOpt) Compression(v imageext.ImageCompresson) *PDFImageOpt { | ||||||
|  | 	opt.compression = &v | ||||||
|  | 	return opt | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (opt *PDFImageOpt) ReEncodePixelPerMM(v float64) *PDFImageOpt { | ||||||
|  | 	opt.reEncodePixelPerMM = &v | ||||||
|  | 	return opt | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (opt *PDFImageOpt) Crop(cropX float64, cropY float64, cropWidth float64, cropHeight float64) *PDFImageOpt { | ||||||
|  | 	opt.crop = &imageext.ImageCrop{ | ||||||
|  | 		CropX:      cropX, | ||||||
|  | 		CropY:      cropY, | ||||||
|  | 		CropWidth:  cropWidth, | ||||||
|  | 		CropHeight: cropHeight, | ||||||
|  | 	} | ||||||
|  | 	return opt | ||||||
|  | } | ||||||
|  |  | ||||||
| func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | ||||||
|  | 	var err error | ||||||
|  |  | ||||||
| 	x := b.GetX() | 	x := b.GetX() | ||||||
| 	y := b.GetY() | 	y := b.GetY() | ||||||
| 	w := img.Info.Width() | 	w := img.Info.Width() | ||||||
| @@ -162,6 +226,11 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 	imageType := "" | 	imageType := "" | ||||||
| 	readDpi := false | 	readDpi := false | ||||||
| 	allowNegativePosition := false | 	allowNegativePosition := false | ||||||
|  | 	reEncodePixelPerMM := 15.0 | ||||||
|  | 	var imageFit *imageext.ImageFit = nil | ||||||
|  | 	var fillColor color.Color = color.Transparent | ||||||
|  | 	var compression *imageext.ImageCompresson = nil | ||||||
|  | 	var crop *imageext.ImageCrop = nil | ||||||
|  |  | ||||||
| 	for _, opt := range opts { | 	for _, opt := range opts { | ||||||
| 		x = langext.Coalesce(opt.x, x) | 		x = langext.Coalesce(opt.x, x) | ||||||
| @@ -174,6 +243,80 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 		imageType = langext.Coalesce(opt.imageType, imageType) | 		imageType = langext.Coalesce(opt.imageType, imageType) | ||||||
| 		readDpi = langext.Coalesce(opt.readDpi, readDpi) | 		readDpi = langext.Coalesce(opt.readDpi, readDpi) | ||||||
| 		allowNegativePosition = langext.Coalesce(opt.allowNegativePosition, allowNegativePosition) | 		allowNegativePosition = langext.Coalesce(opt.allowNegativePosition, allowNegativePosition) | ||||||
|  | 		imageFit = langext.CoalesceOpt(opt.imageFit, imageFit) | ||||||
|  | 		fillColor = langext.Coalesce(opt.fillColor, fillColor) | ||||||
|  | 		compression = langext.CoalesceOpt(opt.compression, compression) | ||||||
|  | 		reEncodePixelPerMM = langext.Coalesce(opt.reEncodePixelPerMM, reEncodePixelPerMM) | ||||||
|  | 		crop = langext.CoalesceOpt(opt.crop, crop) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	regName := img.Name | ||||||
|  |  | ||||||
|  | 	if imageFit != nil || fillColor != nil || crop != nil || compression != nil { | ||||||
|  |  | ||||||
|  | 		var dataimg image.Image | ||||||
|  | 		if img.Image != nil { | ||||||
|  | 			dataimg = *img.Image | ||||||
|  | 		} else { | ||||||
|  | 			dataimg, err = imageext.VerifyAndDecodeImage(bytes.NewReader(img.Bin), img.Mime) | ||||||
|  | 			if err != nil { | ||||||
|  | 				b.b.SetError(err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			img.Image = langext.Ptr(dataimg) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		_ = os.WriteFile("/tmp/a.png", img.Bin, 0755) | ||||||
|  |  | ||||||
|  | 		{ | ||||||
|  | 			bfr := bytes.Buffer{} | ||||||
|  | 			enc := &png.Encoder{CompressionLevel: png.NoCompression} | ||||||
|  | 			_ = enc.Encode(&bfr, dataimg) | ||||||
|  | 			_ = os.WriteFile("/tmp/b.png", img.Bin, 0755) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if crop != nil { | ||||||
|  | 			dataimg, err = imageext.CropImage(dataimg, crop.CropX, crop.CropY, crop.CropWidth, crop.CropHeight) | ||||||
|  | 			if err != nil { | ||||||
|  | 				b.b.SetError(err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if imageFit != nil { | ||||||
|  | 			pdfPixelPerMillimeter := 15.0 | ||||||
|  |  | ||||||
|  | 			pxw := w * pdfPixelPerMillimeter | ||||||
|  | 			pxh := h * pdfPixelPerMillimeter | ||||||
|  |  | ||||||
|  | 			dataimg, err = imageext.ObjectFitImage(dataimg, pxw, pxh, *imageFit, fillColor) | ||||||
|  | 			if err != nil { | ||||||
|  | 				b.b.SetError(err) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		bfr, imgMime, err := imageext.EncodeImage(dataimg, langext.Coalesce(compression, imageext.CompressionPNGSpeed)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			b.b.SetError(err) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		regName = regName + "_" + langext.MustRawHexUUID() | ||||||
|  |  | ||||||
|  | 		switch imgMime { | ||||||
|  | 		case "image/jpeg": | ||||||
|  | 			imageType = "JPEG" | ||||||
|  | 		case "image/png": | ||||||
|  | 			imageType = "PNG" | ||||||
|  | 		case "image/gif": | ||||||
|  | 			imageType = "GIF" | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		_ = os.WriteFile("/tmp/c.png", bfr.Bytes(), 0755) | ||||||
|  |  | ||||||
|  | 		b.b.RegisterImageOptionsReader(regName, gofpdf.ImageOptions{ImageType: imageType}, &bfr) | ||||||
|  |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	fpdfOpt := gofpdf.ImageOptions{ | 	fpdfOpt := gofpdf.ImageOptions{ | ||||||
| @@ -182,5 +325,5 @@ func (b *WPDFBuilder) Image(img *PDFImageRef, opts ...*PDFImageOpt) { | |||||||
| 		AllowNegativePosition: allowNegativePosition, | 		AllowNegativePosition: allowNegativePosition, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	b.b.ImageOptions(img.Name, x, y, w, h, flow, fpdfOpt, link, linkStr) | 	b.b.ImageOptions(regName, x, y, w, h, flow, fpdfOpt, link, linkStr) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -91,7 +91,7 @@ func (b *WPDFBuilder) MultiCell(txt string, opts ...*PDFMultiCellOpt) { | |||||||
| 	txtTR := b.tr(txt) | 	txtTR := b.tr(txt) | ||||||
|  |  | ||||||
| 	width := float64(0) | 	width := float64(0) | ||||||
| 	height := b.cellHeight | 	height := b.cellHeight + b.cellSpacing | ||||||
| 	border := BorderNone | 	border := BorderNone | ||||||
| 	align := AlignLeft | 	align := AlignLeft | ||||||
| 	fill := false | 	fill := false | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user