Rename serializers to modifiers

pull/29/head
Vinícius Garcia 2022-09-21 21:38:30 -03:00
parent f95cd2b7b2
commit cb15295e46
6 changed files with 107 additions and 107 deletions

66
attr_modifiers.go Normal file
View File

@ -0,0 +1,66 @@
package ksql
import (
"context"
"database/sql/driver"
"fmt"
)
// Here we keep all the registered modifier
var modifiers = map[string]AttrModifier{
"json": jsonModifier{},
}
// RegisterAttrModifier allow users to add custom modifiers on startup
// it is recommended to do this inside an init() function.
func RegisterAttrModifier(key string, modifier AttrModifier) {
_, found := modifiers[key]
if found {
panic(fmt.Errorf("KSQL: cannot register modifier '%s' name is already in use", key))
}
modifiers[key] = modifier
}
// AttrModifier describes the two operations required to serialize and deserialize an object from the database.
type AttrModifier interface {
AttrScan(ctx context.Context, opInfo OpInfo, attrPtr interface{}, dbValue interface{}) error
AttrValue(ctx context.Context, opInfo OpInfo, inputValue interface{}) (outputValue interface{}, _ error)
}
// OpInfo contains information that might be used by a modifier to determine how it should behave.
type OpInfo struct {
// A string version of the name of one of
// the methods of the `ksql.Provider` interface, e.g. `Insert` or `Query`
Method string
// The string representing the current underlying database, e.g.:
// "postgres", "sqlite3", "mysql" or "sqlserver".
DriverName string
}
// attrModifier is the wrapper that allow us to intercept the Scan and Value processes
// so we can run the modifiers instead of allowing the database driver to use
// its default behavior.
//
// For that this struct implements both the `sql.Scanner` and `sql.Valuer` interfaces.
type attrModifier struct {
ctx context.Context
// When Scanning this value should be a pointer to the attribute
// and when "Valuing" it should just be the actual value
attr interface{}
modifierName string
opInfo OpInfo
}
// Scan implements the sql.Scanner interface
func (a attrModifier) Scan(dbValue interface{}) error {
return modifiers[a.modifierName].AttrScan(a.ctx, a.opInfo, a.attr, dbValue)
}
// Value implements the sql.Valuer interface
func (a attrModifier) Value() (driver.Value, error) {
return modifiers[a.modifierName].AttrValue(a.ctx, a.opInfo, a.attr)
}

View File

@ -1,66 +0,0 @@
package ksql
import (
"context"
"database/sql/driver"
"fmt"
)
// Here we keep all the registered serializers
var serializers = map[string]AttrSerializer{
"json": jsonSerializer{},
}
// RegisterAttrSerializer allow users to add custom serializers on startup
// it is recommended to do this inside an init() function.
func RegisterAttrSerializer(key string, serializer AttrSerializer) {
_, found := serializers[key]
if found {
panic(fmt.Errorf("KSQL: cannot register serializer '%s' name is already in use", key))
}
serializers[key] = serializer
}
// AttrSerializer describes the two operations required to serialize and deserialize an object from the database.
type AttrSerializer interface {
AttrScan(ctx context.Context, opInfo OpInfo, attrPtr interface{}, dbValue interface{}) error
AttrValue(ctx context.Context, opInfo OpInfo, inputValue interface{}) (outputValue interface{}, _ error)
}
// OpInfo contains information that might be used by a serializer to determine how it should behave.
type OpInfo struct {
// A string version of the name of one of
// the methods of the `ksql.Provider` interface, e.g. `Insert` or `Query`
Method string
// The string representing the current underlying database, e.g.:
// "postgres", "sqlite3", "mysql" or "sqlserver".
DriverName string
}
// attrSerializer is the wrapper that allow us to intercept the Scan and Value processes
// so we can run the serializers instead of allowing the database driver to use
// its default behavior.
//
// For that this struct implements both the `sql.Scanner` and `sql.Valuer` interfaces.
type attrSerializer struct {
ctx context.Context
// When Scanning this value should be a pointer to the attribute
// and when "Valuing" it should just be the actual value
attr interface{}
serializerName string
opInfo OpInfo
}
// Scan implements the sql.Scanner interface
func (a attrSerializer) Scan(dbValue interface{}) error {
return serializers[a.serializerName].AttrScan(a.ctx, a.opInfo, a.attr, dbValue)
}
// Value implements the sql.Valuer interface
func (a attrSerializer) Value() (driver.Value, error) {
return serializers[a.serializerName].AttrValue(a.ctx, a.opInfo, a.attr)
}

View File

@ -20,10 +20,10 @@ type StructInfo struct {
// information regarding a specific field // information regarding a specific field
// of a struct. // of a struct.
type FieldInfo struct { type FieldInfo struct {
Name string Name string
Index int Index int
Valid bool Valid bool
SerializerName string ModifierName string
} }
// ByIndex returns either the *FieldInfo of a valid // ByIndex returns either the *FieldInfo of a valid
@ -249,10 +249,10 @@ func getTagNames(t reflect.Type) (StructInfo, error) {
} }
tags := strings.Split(name, ",") tags := strings.Split(name, ",")
var serializerName string var modifierName string
if len(tags) > 1 { if len(tags) > 1 {
name = tags[0] name = tags[0]
serializerName = tags[1] modifierName = tags[1]
} }
if _, found := info.byName[name]; found { if _, found := info.byName[name]; found {
@ -263,9 +263,9 @@ func getTagNames(t reflect.Type) (StructInfo, error) {
} }
info.add(FieldInfo{ info.add(FieldInfo{
Name: name, Name: name,
Index: i, Index: i,
SerializerName: serializerName, ModifierName: modifierName,
}) })
} }

View File

@ -10,11 +10,11 @@ import (
// This type was created to make it easier to adapt // This type was created to make it easier to adapt
// input attributes to be convertible to and from JSON // input attributes to be convertible to and from JSON
// before sending or receiving it from the database. // before sending or receiving it from the database.
type jsonSerializer struct{} type jsonModifier struct{}
// Scan Implements the Scanner interface in order to load // Scan Implements the Scanner interface in order to load
// this field from the JSON stored in the database // this field from the JSON stored in the database
func (j jsonSerializer) AttrScan(ctx context.Context, opInfo OpInfo, attrPtr interface{}, dbValue interface{}) error { func (j jsonModifier) AttrScan(ctx context.Context, opInfo OpInfo, attrPtr interface{}, dbValue interface{}) error {
if dbValue == nil { if dbValue == nil {
v := reflect.ValueOf(attrPtr) v := reflect.ValueOf(attrPtr)
// Set the struct to its 0 value just like json.Unmarshal // Set the struct to its 0 value just like json.Unmarshal
@ -37,7 +37,7 @@ func (j jsonSerializer) AttrScan(ctx context.Context, opInfo OpInfo, attrPtr int
// Value Implements the Valuer interface in order to save // Value Implements the Valuer interface in order to save
// this field as JSON on the database. // this field as JSON on the database.
func (j jsonSerializer) AttrValue(ctx context.Context, opInfo OpInfo, inputValue interface{}) (outputValue interface{}, _ error) { func (j jsonModifier) AttrValue(ctx context.Context, opInfo OpInfo, inputValue interface{}) (outputValue interface{}, _ error) {
b, err := json.Marshal(inputValue) b, err := json.Marshal(inputValue)
if opInfo.DriverName == "sqlserver" { if opInfo.DriverName == "sqlserver" {
return string(b), err return string(b), err

48
ksql.go
View File

@ -718,12 +718,12 @@ func buildInsertQuery(
recordValue := recordMap[col] recordValue := recordMap[col]
params[i] = recordValue params[i] = recordValue
serializerName := info.ByName(col).SerializerName modifierName := info.ByName(col).ModifierName
if serializerName != "" { if modifierName != "" {
params[i] = attrSerializer{ params[i] = attrModifier{
ctx: ctx, ctx: ctx,
attr: recordValue, attr: recordValue,
serializerName: serializerName, modifierName: modifierName,
opInfo: OpInfo{ opInfo: OpInfo{
DriverName: dialect.DriverName(), DriverName: dialect.DriverName(),
Method: "Insert", Method: "Insert",
@ -827,12 +827,12 @@ func buildUpdateQuery(
for i, k := range keys { for i, k := range keys {
recordValue := recordMap[k] recordValue := recordMap[k]
serializerName := info.ByName(k).SerializerName modifierName := info.ByName(k).ModifierName
if serializerName != "" { if modifierName != "" {
recordValue = attrSerializer{ recordValue = attrModifier{
ctx: ctx, ctx: ctx,
attr: recordValue, attr: recordValue,
serializerName: serializerName, modifierName: modifierName,
opInfo: OpInfo{ opInfo: OpInfo{
DriverName: dialect.DriverName(), DriverName: dialect.DriverName(),
Method: "Update", Method: "Update",
@ -1032,15 +1032,15 @@ func getScanArgsForNestedStructs(
if fieldInfo.Valid { if fieldInfo.Valid {
valueScanner = nestedStructValue.Field(fieldInfo.Index).Addr().Interface() valueScanner = nestedStructValue.Field(fieldInfo.Index).Addr().Interface()
if fieldInfo.SerializerName != "" { if fieldInfo.ModifierName != "" {
valueScanner = &attrSerializer{ valueScanner = &attrModifier{
ctx: ctx, ctx: ctx,
attr: valueScanner, attr: valueScanner,
serializerName: fieldInfo.SerializerName, modifierName: fieldInfo.ModifierName,
opInfo: OpInfo{ opInfo: OpInfo{
DriverName: dialect.DriverName(), DriverName: dialect.DriverName(),
// We will not differentiate between Query, QueryOne and QueryChunks // We will not differentiate between Query, QueryOne and QueryChunks
// if we did this could lead users to make very strange serializers // if we did this could lead users to make very strange modifiers
Method: "Query", Method: "Query",
}, },
} }
@ -1062,15 +1062,15 @@ func getScanArgsFromNames(ctx context.Context, dialect Dialect, names []string,
valueScanner := nopScannerValue valueScanner := nopScannerValue
if fieldInfo.Valid { if fieldInfo.Valid {
valueScanner = v.Field(fieldInfo.Index).Addr().Interface() valueScanner = v.Field(fieldInfo.Index).Addr().Interface()
if fieldInfo.SerializerName != "" { if fieldInfo.ModifierName != "" {
valueScanner = &attrSerializer{ valueScanner = &attrModifier{
ctx: ctx, ctx: ctx,
attr: valueScanner, attr: valueScanner,
serializerName: fieldInfo.SerializerName, modifierName: fieldInfo.ModifierName,
opInfo: OpInfo{ opInfo: OpInfo{
DriverName: dialect.DriverName(), DriverName: dialect.DriverName(),
// We will not differentiate between Query, QueryOne and QueryChunks // We will not differentiate between Query, QueryOne and QueryChunks
// if we did this could lead users to make very strange serializers // if we did this could lead users to make very strange modifiers
Method: "Query", Method: "Query",
}, },
} }

View File

@ -2799,14 +2799,14 @@ func getUserByID(db DBAdapter, dialect Dialect, result *user, id uint) error {
return sql.ErrNoRows return sql.ErrNoRows
} }
value := attrSerializer{ value := attrModifier{
ctx: context.TODO(), ctx: context.TODO(),
attr: &result.Address, attr: &result.Address,
serializerName: "json", modifierName: "json",
opInfo: OpInfo{ opInfo: OpInfo{
DriverName: dialect.DriverName(), DriverName: dialect.DriverName(),
// We will not differentiate between Query, QueryOne and QueryChunks // We will not differentiate between Query, QueryOne and QueryChunks
// if we did this could lead users to make very strange serializers // if we did this could lead users to make very strange modifiers
Method: "Query", Method: "Query",
}, },
} }