2022-12-05 22:57:39 +02:00
|
|
|
// Package jsvm implements optional utilities for binding a JS goja runtime
|
|
|
|
// to the PocketBase instance (loading migrations, attaching to app hooks, etc.).
|
|
|
|
//
|
|
|
|
// Currently it provides the following plugins:
|
|
|
|
//
|
|
|
|
// 1. JS Migrations loader:
|
|
|
|
//
|
2023-02-23 21:51:42 +02:00
|
|
|
// jsvm.MustRegisterMigrations(app, &jsvm.MigrationsOptions{
|
|
|
|
// Dir: "custom_js_migrations_dir_path", // default to "pb_data/../pb_migrations"
|
|
|
|
// })
|
2022-11-26 09:05:52 +02:00
|
|
|
package jsvm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2022-11-30 17:23:00 +02:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
"unicode"
|
2022-11-26 09:05:52 +02:00
|
|
|
|
|
|
|
"github.com/dop251/goja"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
|
|
"github.com/pocketbase/pocketbase/apis"
|
|
|
|
"github.com/pocketbase/pocketbase/daos"
|
|
|
|
"github.com/pocketbase/pocketbase/models"
|
2022-11-27 23:00:58 +02:00
|
|
|
"github.com/pocketbase/pocketbase/models/schema"
|
2022-11-26 09:05:52 +02:00
|
|
|
)
|
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
func NewBaseVM() *goja.Runtime {
|
2022-11-26 09:05:52 +02:00
|
|
|
vm := goja.New()
|
2022-12-05 13:57:09 +02:00
|
|
|
vm.SetFieldNameMapper(FieldMapper{})
|
2022-11-26 09:05:52 +02:00
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
baseBinds(vm)
|
|
|
|
dbxBinds(vm)
|
2022-11-28 21:56:30 +02:00
|
|
|
|
|
|
|
return vm
|
|
|
|
}
|
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
func baseBinds(vm *goja.Runtime) {
|
2022-11-26 09:05:52 +02:00
|
|
|
vm.Set("unmarshal", func(src map[string]any, dest any) (any, error) {
|
|
|
|
raw, err := json.Marshal(src)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(raw, &dest); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return dest, nil
|
|
|
|
})
|
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
vm.Set("Record", func(call goja.ConstructorCall) *goja.Object {
|
|
|
|
var instance *models.Record
|
|
|
|
|
|
|
|
collection, ok := call.Argument(0).Export().(*models.Collection)
|
|
|
|
if ok {
|
|
|
|
instance = models.NewRecord(collection)
|
|
|
|
data, ok := call.Argument(1).Export().(map[string]any)
|
|
|
|
if ok {
|
|
|
|
if raw, err := json.Marshal(data); err == nil {
|
|
|
|
json.Unmarshal(raw, instance)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
instance = &models.Record{}
|
|
|
|
}
|
|
|
|
|
2022-11-26 09:05:52 +02:00
|
|
|
instanceValue := vm.ToValue(instance).(*goja.Object)
|
|
|
|
instanceValue.SetPrototype(call.This.Prototype())
|
2022-12-05 13:57:09 +02:00
|
|
|
|
2022-11-26 09:05:52 +02:00
|
|
|
return instanceValue
|
|
|
|
})
|
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
vm.Set("Collection", func(call goja.ConstructorCall) *goja.Object {
|
|
|
|
instance := &models.Collection{}
|
|
|
|
return defaultConstructor(vm, call, instance)
|
2022-11-26 09:05:52 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
vm.Set("Admin", func(call goja.ConstructorCall) *goja.Object {
|
|
|
|
instance := &models.Admin{}
|
2022-12-05 13:57:09 +02:00
|
|
|
return defaultConstructor(vm, call, instance)
|
2022-11-26 09:05:52 +02:00
|
|
|
})
|
|
|
|
|
2022-11-27 23:00:58 +02:00
|
|
|
vm.Set("Schema", func(call goja.ConstructorCall) *goja.Object {
|
|
|
|
instance := &schema.Schema{}
|
2022-12-05 13:57:09 +02:00
|
|
|
return defaultConstructor(vm, call, instance)
|
2022-11-27 23:00:58 +02:00
|
|
|
})
|
2022-11-28 21:56:30 +02:00
|
|
|
|
2022-11-27 23:00:58 +02:00
|
|
|
vm.Set("SchemaField", func(call goja.ConstructorCall) *goja.Object {
|
|
|
|
instance := &schema.SchemaField{}
|
2022-12-05 13:57:09 +02:00
|
|
|
return defaultConstructor(vm, call, instance)
|
2022-11-27 23:00:58 +02:00
|
|
|
})
|
|
|
|
|
2022-11-26 09:05:52 +02:00
|
|
|
vm.Set("Dao", func(call goja.ConstructorCall) *goja.Object {
|
|
|
|
db, ok := call.Argument(0).Export().(dbx.Builder)
|
|
|
|
if !ok || db == nil {
|
|
|
|
panic("missing required Dao(db) argument")
|
|
|
|
}
|
|
|
|
|
|
|
|
instance := daos.New(db)
|
|
|
|
instanceValue := vm.ToValue(instance).(*goja.Object)
|
|
|
|
instanceValue.SetPrototype(call.This.Prototype())
|
2022-12-05 13:57:09 +02:00
|
|
|
|
2022-11-26 09:05:52 +02:00
|
|
|
return instanceValue
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
func defaultConstructor(vm *goja.Runtime, call goja.ConstructorCall, instance any) *goja.Object {
|
|
|
|
if data := call.Argument(0).Export(); data != nil {
|
|
|
|
if raw, err := json.Marshal(data); err == nil {
|
|
|
|
json.Unmarshal(raw, instance)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
instanceValue := vm.ToValue(instance).(*goja.Object)
|
|
|
|
instanceValue.SetPrototype(call.This.Prototype())
|
|
|
|
|
|
|
|
return instanceValue
|
|
|
|
}
|
|
|
|
|
|
|
|
func dbxBinds(vm *goja.Runtime) {
|
2022-11-26 09:05:52 +02:00
|
|
|
obj := vm.NewObject()
|
|
|
|
vm.Set("$dbx", obj)
|
|
|
|
|
|
|
|
obj.Set("exp", dbx.NewExp)
|
|
|
|
obj.Set("hashExp", func(data map[string]any) dbx.HashExp {
|
|
|
|
exp := dbx.HashExp{}
|
|
|
|
for k, v := range data {
|
|
|
|
exp[k] = v
|
|
|
|
}
|
|
|
|
return exp
|
|
|
|
})
|
|
|
|
obj.Set("not", dbx.Not)
|
|
|
|
obj.Set("and", dbx.And)
|
|
|
|
obj.Set("or", dbx.Or)
|
|
|
|
obj.Set("in", dbx.In)
|
|
|
|
obj.Set("notIn", dbx.NotIn)
|
|
|
|
obj.Set("like", dbx.Like)
|
|
|
|
obj.Set("orLike", dbx.OrLike)
|
|
|
|
obj.Set("notLike", dbx.NotLike)
|
|
|
|
obj.Set("orNotLike", dbx.OrNotLike)
|
|
|
|
obj.Set("exists", dbx.Exists)
|
|
|
|
obj.Set("notExists", dbx.NotExists)
|
|
|
|
obj.Set("between", dbx.Between)
|
|
|
|
obj.Set("notBetween", dbx.NotBetween)
|
|
|
|
}
|
|
|
|
|
|
|
|
func apisBind(vm *goja.Runtime) {
|
|
|
|
obj := vm.NewObject()
|
|
|
|
vm.Set("$apis", obj)
|
|
|
|
|
|
|
|
// middlewares
|
|
|
|
obj.Set("requireRecordAuth", apis.RequireRecordAuth)
|
|
|
|
obj.Set("requireRecordAuth", apis.RequireRecordAuth)
|
|
|
|
obj.Set("requireSameContextRecordAuth", apis.RequireSameContextRecordAuth)
|
|
|
|
obj.Set("requireAdminAuth", apis.RequireAdminAuth)
|
|
|
|
obj.Set("requireAdminAuthOnlyIfAny", apis.RequireAdminAuthOnlyIfAny)
|
|
|
|
obj.Set("requireAdminOrRecordAuth", apis.RequireAdminOrRecordAuth)
|
|
|
|
obj.Set("requireAdminOrOwnerAuth", apis.RequireAdminOrOwnerAuth)
|
|
|
|
obj.Set("activityLogger", apis.ActivityLogger)
|
|
|
|
|
|
|
|
// api errors
|
|
|
|
obj.Set("notFoundError", apis.NewNotFoundError)
|
|
|
|
obj.Set("badRequestError", apis.NewBadRequestError)
|
|
|
|
obj.Set("forbiddenError", apis.NewForbiddenError)
|
|
|
|
obj.Set("unauthorizedError", apis.NewUnauthorizedError)
|
|
|
|
|
|
|
|
// record helpers
|
|
|
|
obj.Set("requestData", apis.RequestData)
|
|
|
|
obj.Set("enrichRecord", apis.EnrichRecord)
|
|
|
|
obj.Set("enrichRecords", apis.EnrichRecords)
|
|
|
|
}
|
2022-11-30 17:23:00 +02:00
|
|
|
|
2022-12-05 13:57:09 +02:00
|
|
|
// FieldMapper provides custom mapping between Go and JavaScript property names.
|
2022-11-30 17:23:00 +02:00
|
|
|
//
|
|
|
|
// It is similar to the builtin "uncapFieldNameMapper" but also converts
|
|
|
|
// all uppercase identifiers to their lowercase equivalent (eg. "GET" -> "get").
|
2022-12-05 13:57:09 +02:00
|
|
|
type FieldMapper struct {
|
2022-11-30 17:23:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// FieldName implements the [FieldNameMapper.FieldName] interface method.
|
2022-12-05 13:57:09 +02:00
|
|
|
func (u FieldMapper) FieldName(_ reflect.Type, f reflect.StructField) string {
|
2022-11-30 17:23:00 +02:00
|
|
|
return convertGoToJSName(f.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
// MethodName implements the [FieldNameMapper.MethodName] interface method.
|
2022-12-05 13:57:09 +02:00
|
|
|
func (u FieldMapper) MethodName(_ reflect.Type, m reflect.Method) string {
|
2022-11-30 17:23:00 +02:00
|
|
|
return convertGoToJSName(m.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
func convertGoToJSName(name string) string {
|
|
|
|
allUppercase := true
|
|
|
|
for _, c := range name {
|
2022-12-05 13:57:09 +02:00
|
|
|
if c != '_' && !unicode.IsUpper(c) {
|
2022-11-30 17:23:00 +02:00
|
|
|
allUppercase = false
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// eg. "JSON" -> "json"
|
|
|
|
if allUppercase {
|
|
|
|
return strings.ToLower(name)
|
|
|
|
}
|
|
|
|
|
|
|
|
// eg. "GetField" -> "getField"
|
|
|
|
return strings.ToLower(name[0:1]) + name[1:]
|
|
|
|
}
|