2024-09-29 19:23:19 +03:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"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 08:37:22 +03:00
|
|
|
//
|
|
|
|
// The respective zero record field value is the zero [types.JSONRaw].
|
2024-09-29 19:23:19 +03:00
|
|
|
type JSONField struct {
|
2024-10-24 08:37:22 +03: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-29 19:23:19 +03: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",
|
2024-11-25 11:28:20 +02:00
|
|
|
"The maximum allowed JSON size is {{.maxSize}} bytes",
|
2024-09-29 19:23:19 +03:00
|
|
|
).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
|
|
|
|
}
|