| 
									
										
										
										
											2024-09-30 00:23:19 +08:00
										 |  |  | package core | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"slices" | 
					
						
							|  |  |  | 	"strconv" | 
					
						
							|  |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	validation "github.com/go-ozzo/ozzo-validation/v4" | 
					
						
							|  |  |  | 	"github.com/go-ozzo/ozzo-validation/v4/is" | 
					
						
							|  |  |  | 	"github.com/pocketbase/pocketbase/core/validators" | 
					
						
							|  |  |  | 	"github.com/pocketbase/pocketbase/tools/types" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func init() { | 
					
						
							|  |  |  | 	Fields[FieldTypeJSON] = func() Field { | 
					
						
							|  |  |  | 		return &JSONField{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const FieldTypeJSON = "json" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const DefaultJSONFieldMaxSize int64 = 5 << 20 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	_ Field                 = (*JSONField)(nil) | 
					
						
							|  |  |  | 	_ MaxBodySizeCalculator = (*JSONField)(nil) | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // JSONField defines "json" type field for storing any serialized JSON value.
 | 
					
						
							| 
									
										
										
										
											2024-10-24 13:37:22 +08:00
										 |  |  | //
 | 
					
						
							|  |  |  | // The respective zero record field value is the zero [types.JSONRaw].
 | 
					
						
							| 
									
										
										
										
											2024-09-30 00:23:19 +08:00
										 |  |  | type JSONField struct { | 
					
						
							| 
									
										
										
										
											2024-10-24 13:37:22 +08:00
										 |  |  | 	// Name (required) is the unique name of the field.
 | 
					
						
							|  |  |  | 	Name string `form:"name" json:"name"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Id is the unique stable field identifier.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// It is automatically generated from the name when adding to a collection FieldsList.
 | 
					
						
							|  |  |  | 	Id string `form:"id" json:"id"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// System prevents the renaming and removal of the field.
 | 
					
						
							|  |  |  | 	System bool `form:"system" json:"system"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Hidden hides the field from the API response.
 | 
					
						
							|  |  |  | 	Hidden bool `form:"hidden" json:"hidden"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Presentable hints the Dashboard UI to use the underlying
 | 
					
						
							|  |  |  | 	// field record value in the relation preview label.
 | 
					
						
							|  |  |  | 	Presentable bool `form:"presentable" json:"presentable"` | 
					
						
							| 
									
										
										
										
											2024-09-30 00:23:19 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// ---
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// MaxSize specifies the maximum size of the allowed field value (in bytes).
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// If zero, a default limit of 5MB is applied.
 | 
					
						
							|  |  |  | 	MaxSize int64 `form:"maxSize" json:"maxSize"` | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Required will require the field value to be non-empty JSON value
 | 
					
						
							|  |  |  | 	// (aka. not "null", `""`, "[]", "{}").
 | 
					
						
							|  |  |  | 	Required bool `form:"required" json:"required"` | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Type implements [Field.Type] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) Type() string { | 
					
						
							|  |  |  | 	return FieldTypeJSON | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetId implements [Field.GetId] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) GetId() string { | 
					
						
							|  |  |  | 	return f.Id | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SetId implements [Field.SetId] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) SetId(id string) { | 
					
						
							|  |  |  | 	f.Id = id | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetName implements [Field.GetName] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) GetName() string { | 
					
						
							|  |  |  | 	return f.Name | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SetName implements [Field.SetName] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) SetName(name string) { | 
					
						
							|  |  |  | 	f.Name = name | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetSystem implements [Field.GetSystem] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) GetSystem() bool { | 
					
						
							|  |  |  | 	return f.System | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SetSystem implements [Field.SetSystem] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) SetSystem(system bool) { | 
					
						
							|  |  |  | 	f.System = system | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetHidden implements [Field.GetHidden] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) GetHidden() bool { | 
					
						
							|  |  |  | 	return f.Hidden | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SetHidden implements [Field.SetHidden] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) SetHidden(hidden bool) { | 
					
						
							|  |  |  | 	f.Hidden = hidden | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ColumnType implements [Field.ColumnType] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) ColumnType(app App) string { | 
					
						
							|  |  |  | 	return "JSON DEFAULT NULL" | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // PrepareValue implements [Field.PrepareValue] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) PrepareValue(record *Record, raw any) (any, error) { | 
					
						
							|  |  |  | 	if str, ok := raw.(string); ok { | 
					
						
							|  |  |  | 		// in order to support seamlessly both json and multipart/form-data requests,
 | 
					
						
							|  |  |  | 		// the following normalization rules are applied for plain string values:
 | 
					
						
							|  |  |  | 		// - "true" is converted to the json `true`
 | 
					
						
							|  |  |  | 		// - "false" is converted to the json `false`
 | 
					
						
							|  |  |  | 		// - "null" is converted to the json `null`
 | 
					
						
							|  |  |  | 		// - "[1,2,3]" is converted to the json `[1,2,3]`
 | 
					
						
							|  |  |  | 		// - "{\"a\":1,\"b\":2}" is converted to the json `{"a":1,"b":2}`
 | 
					
						
							|  |  |  | 		// - numeric strings are converted to json number
 | 
					
						
							|  |  |  | 		// - double quoted strings are left as they are (aka. without normalizations)
 | 
					
						
							|  |  |  | 		// - any other string (empty string too) is double quoted
 | 
					
						
							|  |  |  | 		if str == "" { | 
					
						
							|  |  |  | 			raw = strconv.Quote(str) | 
					
						
							|  |  |  | 		} else if str == "null" || str == "true" || str == "false" { | 
					
						
							|  |  |  | 			raw = str | 
					
						
							|  |  |  | 		} else if ((str[0] >= '0' && str[0] <= '9') || | 
					
						
							|  |  |  | 			str[0] == '-' || | 
					
						
							|  |  |  | 			str[0] == '"' || | 
					
						
							|  |  |  | 			str[0] == '[' || | 
					
						
							|  |  |  | 			str[0] == '{') && | 
					
						
							|  |  |  | 			is.JSON.Validate(str) == nil { | 
					
						
							|  |  |  | 			raw = str | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			raw = strconv.Quote(str) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return types.ParseJSONRaw(raw) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var emptyJSONValues = []string{ | 
					
						
							|  |  |  | 	"null", `""`, "[]", "{}", "", | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ValidateValue implements [Field.ValidateValue] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) ValidateValue(ctx context.Context, app App, record *Record) error { | 
					
						
							|  |  |  | 	raw, ok := record.GetRaw(f.Name).(types.JSONRaw) | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		return validators.ErrUnsupportedValueType | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	maxSize := f.CalculateMaxBodySize() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if int64(len(raw)) > maxSize { | 
					
						
							|  |  |  | 		return validation.NewError( | 
					
						
							|  |  |  | 			"validation_json_size_limit", | 
					
						
							|  |  |  | 			fmt.Sprintf("The maximum allowed JSON size is %v bytes", maxSize), | 
					
						
							|  |  |  | 		).SetParams(map[string]any{"maxSize": maxSize}) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if is.JSON.Validate(raw) != nil { | 
					
						
							|  |  |  | 		return validation.NewError("validation_invalid_json", "Must be a valid json value") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	rawStr := strings.TrimSpace(raw.String()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if f.Required && slices.Contains(emptyJSONValues, rawStr) { | 
					
						
							|  |  |  | 		return validation.ErrRequired | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ValidateSettings implements [Field.ValidateSettings] interface method.
 | 
					
						
							|  |  |  | func (f *JSONField) ValidateSettings(ctx context.Context, app App, collection *Collection) error { | 
					
						
							|  |  |  | 	return validation.ValidateStruct(f, | 
					
						
							|  |  |  | 		validation.Field(&f.Id, validation.By(DefaultFieldIdValidationRule)), | 
					
						
							|  |  |  | 		validation.Field(&f.Name, validation.By(DefaultFieldNameValidationRule)), | 
					
						
							|  |  |  | 		validation.Field(&f.MaxSize, validation.Min(0)), | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // CalculateMaxBodySize implements the [MaxBodySizeCalculator] interface.
 | 
					
						
							|  |  |  | func (f *JSONField) CalculateMaxBodySize() int64 { | 
					
						
							|  |  |  | 	if f.MaxSize <= 0 { | 
					
						
							|  |  |  | 		return DefaultJSONFieldMaxSize | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return f.MaxSize | 
					
						
							|  |  |  | } |