mirror of
https://github.com/pocketbase/pocketbase.git
synced 2025-01-24 14:26:58 +02:00
773 lines
23 KiB
Go
773 lines
23 KiB
Go
package migratecmd
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/pocketbase/pocketbase/models"
|
|
)
|
|
|
|
const (
|
|
TemplateLangJS = "js"
|
|
TemplateLangGo = "go"
|
|
)
|
|
|
|
var emptyTemplateErr = errors.New("empty template")
|
|
|
|
// -------------------------------------------------------------------
|
|
// JavaScript templates
|
|
// -------------------------------------------------------------------
|
|
|
|
func (p *plugin) jsBlankTemplate() (string, error) {
|
|
const template = `migrate((db) => {
|
|
// add up queries...
|
|
}, (db) => {
|
|
// add down queries...
|
|
})
|
|
`
|
|
|
|
return template, nil
|
|
}
|
|
|
|
func (p *plugin) jsSnapshotTemplate(collections []*models.Collection) (string, error) {
|
|
jsonData, err := marhshalWithoutEscape(collections, " ", " ")
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to serialize collections list: %w", err)
|
|
}
|
|
|
|
const template = `migrate((db) => {
|
|
const snapshot = %s;
|
|
|
|
const collections = snapshot.map((item) => new Collection(item));
|
|
|
|
return Dao(db).importCollections(collections, true, null);
|
|
}, (db) => {
|
|
return null;
|
|
})
|
|
`
|
|
|
|
return fmt.Sprintf(template, string(jsonData)), nil
|
|
}
|
|
|
|
func (p *plugin) jsCreateTemplate(collection *models.Collection) (string, error) {
|
|
jsonData, err := marhshalWithoutEscape(collection, " ", " ")
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to serialize collections list: %w", err)
|
|
}
|
|
|
|
const template = `migrate((db) => {
|
|
const collection = new Collection(%s);
|
|
|
|
return Dao(db).saveCollection(collection);
|
|
}, (db) => {
|
|
const dao = new Dao(db);
|
|
const collection = dao.findCollectionByNameOrId(%q);
|
|
|
|
return dao.deleteCollection(collection);
|
|
})
|
|
`
|
|
|
|
return fmt.Sprintf(template, string(jsonData), collection.Id), nil
|
|
}
|
|
|
|
func (p *plugin) jsDeleteTemplate(collection *models.Collection) (string, error) {
|
|
jsonData, err := marhshalWithoutEscape(collection, " ", " ")
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to serialize collections list: %w", err)
|
|
}
|
|
|
|
const template = `migrate((db) => {
|
|
const dao = new Dao(db);
|
|
const collection = dao.findCollectionByNameOrId(%q);
|
|
|
|
return dao.deleteCollection(collection);
|
|
}, (db) => {
|
|
const collection = new Collection(%s);
|
|
|
|
return Dao(db).saveCollection(collection);
|
|
})
|
|
`
|
|
|
|
return fmt.Sprintf(template, collection.Id, string(jsonData)), nil
|
|
}
|
|
|
|
func (p *plugin) jsDiffTemplate(new *models.Collection, old *models.Collection) (string, error) {
|
|
if new == nil && old == nil {
|
|
return "", errors.New("the diff template require at least one of the collection to be non-nil")
|
|
}
|
|
|
|
if new == nil {
|
|
return p.jsDeleteTemplate(old)
|
|
}
|
|
|
|
if old == nil {
|
|
return p.jsCreateTemplate(new)
|
|
}
|
|
|
|
upParts := []string{}
|
|
downParts := []string{}
|
|
varName := "collection"
|
|
|
|
if old.Name != new.Name {
|
|
upParts = append(upParts, fmt.Sprintf("%s.name = %q", varName, new.Name))
|
|
downParts = append(downParts, fmt.Sprintf("%s.name = %q", varName, old.Name))
|
|
}
|
|
|
|
if old.Type != new.Type {
|
|
upParts = append(upParts, fmt.Sprintf("%s.type = %q", varName, new.Type))
|
|
downParts = append(downParts, fmt.Sprintf("%s.type = %q", varName, old.Type))
|
|
}
|
|
|
|
if old.System != new.System {
|
|
upParts = append(upParts, fmt.Sprintf("%s.system = %t", varName, new.System))
|
|
downParts = append(downParts, fmt.Sprintf("%s.system = %t", varName, old.System))
|
|
}
|
|
|
|
// ---
|
|
// note: strconv.Quote is used because %q converts the rule operators in unicode char codes
|
|
// ---
|
|
|
|
if old.ListRule != new.ListRule {
|
|
if old.ListRule != nil && new.ListRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.listRule = null", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.listRule = %s", varName, strconv.Quote(*old.ListRule)))
|
|
} else if old.ListRule == nil && new.ListRule != nil || *old.ListRule != *new.ListRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.listRule = %s", varName, strconv.Quote(*new.ListRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.listRule = null", varName))
|
|
}
|
|
}
|
|
|
|
if old.ViewRule != new.ViewRule {
|
|
if old.ViewRule != nil && new.ViewRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.viewRule = null", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.viewRule = %s", varName, strconv.Quote(*old.ViewRule)))
|
|
} else if old.ViewRule == nil && new.ViewRule != nil || *old.ViewRule != *new.ViewRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.viewRule = %s", varName, strconv.Quote(*new.ViewRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.viewRule = null", varName))
|
|
}
|
|
}
|
|
|
|
if old.CreateRule != new.CreateRule {
|
|
if old.CreateRule != nil && new.CreateRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.createRule = null", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.createRule = %s", varName, strconv.Quote(*old.CreateRule)))
|
|
} else if old.CreateRule == nil && new.CreateRule != nil || *old.CreateRule != *new.CreateRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.createRule = %s", varName, strconv.Quote(*new.CreateRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.createRule = null", varName))
|
|
}
|
|
}
|
|
|
|
if old.UpdateRule != new.UpdateRule {
|
|
if old.UpdateRule != nil && new.UpdateRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.updateRule = null", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.updateRule = %s", varName, strconv.Quote(*old.UpdateRule)))
|
|
} else if old.UpdateRule == nil && new.UpdateRule != nil || *old.UpdateRule != *new.UpdateRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.updateRule = %s", varName, strconv.Quote(*new.UpdateRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.updateRule = null", varName))
|
|
}
|
|
}
|
|
|
|
if old.DeleteRule != new.DeleteRule {
|
|
if old.DeleteRule != nil && new.DeleteRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.deleteRule = null", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.deleteRule = %s", varName, strconv.Quote(*old.DeleteRule)))
|
|
} else if old.DeleteRule == nil && new.DeleteRule != nil || *old.DeleteRule != *new.DeleteRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.deleteRule = %s", varName, strconv.Quote(*new.DeleteRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.deleteRule = null", varName))
|
|
}
|
|
}
|
|
|
|
// Options
|
|
rawNewOptions, err := marhshalWithoutEscape(new.Options, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
rawOldOptions, err := marhshalWithoutEscape(old.Options, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if !bytes.Equal(rawNewOptions, rawOldOptions) {
|
|
upParts = append(upParts, fmt.Sprintf("%s.options = %s", varName, rawNewOptions))
|
|
downParts = append(downParts, fmt.Sprintf("%s.options = %s", varName, rawOldOptions))
|
|
}
|
|
|
|
// Indexes
|
|
rawNewIndexes, err := marhshalWithoutEscape(new.Indexes, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
rawOldIndexes, err := marhshalWithoutEscape(old.Indexes, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if !bytes.Equal(rawNewIndexes, rawOldIndexes) {
|
|
upParts = append(upParts, fmt.Sprintf("%s.indexes = %s", varName, rawNewIndexes))
|
|
downParts = append(downParts, fmt.Sprintf("%s.indexes = %s", varName, rawOldIndexes))
|
|
}
|
|
|
|
// ensure new line between regular and collection fields
|
|
if len(upParts) > 0 {
|
|
upParts[len(upParts)-1] += "\n"
|
|
}
|
|
if len(downParts) > 0 {
|
|
downParts[len(downParts)-1] += "\n"
|
|
}
|
|
|
|
// Schema
|
|
// -----------------------------------------------------------------
|
|
|
|
// deleted fields
|
|
for _, oldField := range old.Schema.Fields() {
|
|
if new.Schema.GetFieldById(oldField.Id) != nil {
|
|
continue // exist
|
|
}
|
|
|
|
rawOldField, err := marhshalWithoutEscape(oldField, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
upParts = append(upParts, "// remove")
|
|
upParts = append(upParts, fmt.Sprintf("%s.schema.removeField(%q)\n", varName, oldField.Id))
|
|
|
|
downParts = append(downParts, "// add")
|
|
downParts = append(downParts, fmt.Sprintf("%s.schema.addField(new SchemaField(%s))\n", varName, rawOldField))
|
|
}
|
|
|
|
// created fields
|
|
for _, newField := range new.Schema.Fields() {
|
|
if old.Schema.GetFieldById(newField.Id) != nil {
|
|
continue // exist
|
|
}
|
|
|
|
rawNewField, err := marhshalWithoutEscape(newField, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
upParts = append(upParts, "// add")
|
|
upParts = append(upParts, fmt.Sprintf("%s.schema.addField(new SchemaField(%s))\n", varName, rawNewField))
|
|
|
|
downParts = append(downParts, "// remove")
|
|
downParts = append(downParts, fmt.Sprintf("%s.schema.removeField(%q)\n", varName, newField.Id))
|
|
}
|
|
|
|
// modified fields
|
|
for _, newField := range new.Schema.Fields() {
|
|
oldField := old.Schema.GetFieldById(newField.Id)
|
|
if oldField == nil {
|
|
continue
|
|
}
|
|
|
|
rawNewField, err := marhshalWithoutEscape(newField, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
rawOldField, err := marhshalWithoutEscape(oldField, " ", " ")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if bytes.Equal(rawNewField, rawOldField) {
|
|
continue // no change
|
|
}
|
|
|
|
upParts = append(upParts, "// update")
|
|
upParts = append(upParts, fmt.Sprintf("%s.schema.addField(new SchemaField(%s))\n", varName, rawNewField))
|
|
|
|
downParts = append(downParts, "// update")
|
|
downParts = append(downParts, fmt.Sprintf("%s.schema.addField(new SchemaField(%s))\n", varName, rawOldField))
|
|
}
|
|
|
|
// -----------------------------------------------------------------
|
|
|
|
if len(upParts) == 0 && len(downParts) == 0 {
|
|
return "", emptyTemplateErr
|
|
}
|
|
|
|
up := strings.Join(upParts, "\n ")
|
|
down := strings.Join(downParts, "\n ")
|
|
|
|
const template = `migrate((db) => {
|
|
const dao = new Dao(db)
|
|
const collection = dao.findCollectionByNameOrId(%q)
|
|
|
|
%s
|
|
|
|
return dao.saveCollection(collection)
|
|
}, (db) => {
|
|
const dao = new Dao(db)
|
|
const collection = dao.findCollectionByNameOrId(%q)
|
|
|
|
%s
|
|
|
|
return dao.saveCollection(collection)
|
|
})
|
|
`
|
|
|
|
return fmt.Sprintf(
|
|
template,
|
|
old.Id, strings.TrimSpace(up),
|
|
new.Id, strings.TrimSpace(down),
|
|
), nil
|
|
}
|
|
|
|
// -------------------------------------------------------------------
|
|
// Go templates
|
|
// -------------------------------------------------------------------
|
|
|
|
func (p *plugin) goBlankTemplate() (string, error) {
|
|
const template = `package %s
|
|
|
|
import (
|
|
"github.com/pocketbase/dbx"
|
|
m "github.com/pocketbase/pocketbase/migrations"
|
|
)
|
|
|
|
func init() {
|
|
m.Register(func(db dbx.Builder) error {
|
|
// add up queries...
|
|
|
|
return nil
|
|
}, func(db dbx.Builder) error {
|
|
// add down queries...
|
|
|
|
return nil
|
|
})
|
|
}
|
|
`
|
|
|
|
return fmt.Sprintf(template, filepath.Base(p.options.Dir)), nil
|
|
}
|
|
|
|
func (p *plugin) goSnapshotTemplate(collections []*models.Collection) (string, error) {
|
|
jsonData, err := marhshalWithoutEscape(collections, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to serialize collections list: %w", err)
|
|
}
|
|
|
|
const template = `package %s
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
"github.com/pocketbase/pocketbase/daos"
|
|
m "github.com/pocketbase/pocketbase/migrations"
|
|
"github.com/pocketbase/pocketbase/models"
|
|
)
|
|
|
|
func init() {
|
|
m.Register(func(db dbx.Builder) error {
|
|
jsonData := ` + "`%s`" + `
|
|
|
|
collections := []*models.Collection{}
|
|
if err := json.Unmarshal([]byte(jsonData), &collections); err != nil {
|
|
return err
|
|
}
|
|
|
|
return daos.New(db).ImportCollections(collections, true, nil)
|
|
}, func(db dbx.Builder) error {
|
|
return nil
|
|
})
|
|
}
|
|
`
|
|
return fmt.Sprintf(
|
|
template,
|
|
filepath.Base(p.options.Dir),
|
|
escapeBacktick(string(jsonData)),
|
|
), nil
|
|
}
|
|
|
|
func (p *plugin) goCreateTemplate(collection *models.Collection) (string, error) {
|
|
jsonData, err := marhshalWithoutEscape(collection, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to serialize collections list: %w", err)
|
|
}
|
|
|
|
const template = `package %s
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
"github.com/pocketbase/pocketbase/daos"
|
|
m "github.com/pocketbase/pocketbase/migrations"
|
|
"github.com/pocketbase/pocketbase/models"
|
|
)
|
|
|
|
func init() {
|
|
m.Register(func(db dbx.Builder) error {
|
|
jsonData := ` + "`%s`" + `
|
|
|
|
collection := &models.Collection{}
|
|
if err := json.Unmarshal([]byte(jsonData), &collection); err != nil {
|
|
return err
|
|
}
|
|
|
|
return daos.New(db).SaveCollection(collection)
|
|
}, func(db dbx.Builder) error {
|
|
dao := daos.New(db);
|
|
|
|
collection, err := dao.FindCollectionByNameOrId(%q)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return dao.DeleteCollection(collection)
|
|
})
|
|
}
|
|
`
|
|
|
|
return fmt.Sprintf(
|
|
template,
|
|
filepath.Base(p.options.Dir),
|
|
escapeBacktick(string(jsonData)),
|
|
collection.Id,
|
|
), nil
|
|
}
|
|
|
|
func (p *plugin) goDeleteTemplate(collection *models.Collection) (string, error) {
|
|
jsonData, err := marhshalWithoutEscape(collection, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to serialize collections list: %w", err)
|
|
}
|
|
|
|
const template = `package %s
|
|
|
|
import (
|
|
"encoding/json"
|
|
|
|
"github.com/pocketbase/dbx"
|
|
"github.com/pocketbase/pocketbase/daos"
|
|
m "github.com/pocketbase/pocketbase/migrations"
|
|
"github.com/pocketbase/pocketbase/models"
|
|
)
|
|
|
|
func init() {
|
|
m.Register(func(db dbx.Builder) error {
|
|
dao := daos.New(db);
|
|
|
|
collection, err := dao.FindCollectionByNameOrId(%q)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return dao.DeleteCollection(collection)
|
|
}, func(db dbx.Builder) error {
|
|
jsonData := ` + "`%s`" + `
|
|
|
|
collection := &models.Collection{}
|
|
if err := json.Unmarshal([]byte(jsonData), &collection); err != nil {
|
|
return err
|
|
}
|
|
|
|
return daos.New(db).SaveCollection(collection)
|
|
})
|
|
}
|
|
`
|
|
|
|
return fmt.Sprintf(
|
|
template,
|
|
filepath.Base(p.options.Dir),
|
|
collection.Id,
|
|
escapeBacktick(string(jsonData)),
|
|
), nil
|
|
}
|
|
|
|
func (p *plugin) goDiffTemplate(new *models.Collection, old *models.Collection) (string, error) {
|
|
if new == nil && old == nil {
|
|
return "", errors.New("the diff template require at least one of the collection to be non-nil")
|
|
}
|
|
|
|
if new == nil {
|
|
return p.goDeleteTemplate(old)
|
|
}
|
|
|
|
if old == nil {
|
|
return p.goCreateTemplate(new)
|
|
}
|
|
|
|
upParts := []string{}
|
|
downParts := []string{}
|
|
varName := "collection"
|
|
if old.Name != new.Name {
|
|
upParts = append(upParts, fmt.Sprintf("%s.Name = %q\n", varName, new.Name))
|
|
downParts = append(downParts, fmt.Sprintf("%s.Name = %q\n", varName, old.Name))
|
|
}
|
|
|
|
if old.Type != new.Type {
|
|
upParts = append(upParts, fmt.Sprintf("%s.Type = %q\n", varName, new.Type))
|
|
downParts = append(downParts, fmt.Sprintf("%s.Type = %q\n", varName, old.Type))
|
|
}
|
|
|
|
if old.System != new.System {
|
|
upParts = append(upParts, fmt.Sprintf("%s.System = %t\n", varName, new.System))
|
|
downParts = append(downParts, fmt.Sprintf("%s.System = %t\n", varName, old.System))
|
|
}
|
|
|
|
// ---
|
|
// note: strconv.Quote is used because %q converts the rule operators in unicode char codes
|
|
// ---
|
|
|
|
if old.ListRule != new.ListRule {
|
|
if old.ListRule != nil && new.ListRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.ListRule = nil\n", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.ListRule = types.Pointer(%s)\n", varName, strconv.Quote(*old.ListRule)))
|
|
} else if old.ListRule == nil && new.ListRule != nil || *old.ListRule != *new.ListRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.ListRule = types.Pointer(%s)\n", varName, strconv.Quote(*new.ListRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.ListRule = nil\n", varName))
|
|
}
|
|
}
|
|
|
|
if old.ViewRule != new.ViewRule {
|
|
if old.ViewRule != nil && new.ViewRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.ViewRule = nil\n", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.ViewRule = types.Pointer(%s)\n", varName, strconv.Quote(*old.ViewRule)))
|
|
} else if old.ViewRule == nil && new.ViewRule != nil || *old.ViewRule != *new.ViewRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.ViewRule = types.Pointer(%s)\n", varName, strconv.Quote(*new.ViewRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.ViewRule = nil\n", varName))
|
|
}
|
|
}
|
|
|
|
if old.CreateRule != new.CreateRule {
|
|
if old.CreateRule != nil && new.CreateRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.CreateRule = nil\n", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.CreateRule = types.Pointer(%s)\n", varName, strconv.Quote(*old.CreateRule)))
|
|
} else if old.CreateRule == nil && new.CreateRule != nil || *old.CreateRule != *new.CreateRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.CreateRule = types.Pointer(%s)\n", varName, strconv.Quote(*new.CreateRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.CreateRule = nil\n", varName))
|
|
}
|
|
}
|
|
|
|
if old.UpdateRule != new.UpdateRule {
|
|
if old.UpdateRule != nil && new.UpdateRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.UpdateRule = nil\n", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.UpdateRule = types.Pointer(%s)\n", varName, strconv.Quote(*old.UpdateRule)))
|
|
} else if old.UpdateRule == nil && new.UpdateRule != nil || *old.UpdateRule != *new.UpdateRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.UpdateRule = types.Pointer(%s)\n", varName, strconv.Quote(*new.UpdateRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.UpdateRule = nil\n", varName))
|
|
}
|
|
}
|
|
|
|
if old.DeleteRule != new.DeleteRule {
|
|
if old.DeleteRule != nil && new.DeleteRule == nil {
|
|
upParts = append(upParts, fmt.Sprintf("%s.DeleteRule = nil\n", varName))
|
|
downParts = append(downParts, fmt.Sprintf("%s.DeleteRule = types.Pointer(%s)\n", varName, strconv.Quote(*old.DeleteRule)))
|
|
} else if old.DeleteRule == nil && new.DeleteRule != nil || *old.DeleteRule != *new.DeleteRule {
|
|
upParts = append(upParts, fmt.Sprintf("%s.DeleteRule = types.Pointer(%s)\n", varName, strconv.Quote(*new.DeleteRule)))
|
|
downParts = append(downParts, fmt.Sprintf("%s.DeleteRule = nil\n", varName))
|
|
}
|
|
}
|
|
|
|
// Options
|
|
rawNewOptions, err := marhshalWithoutEscape(new.Options, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
rawOldOptions, err := marhshalWithoutEscape(old.Options, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if !bytes.Equal(rawNewOptions, rawOldOptions) {
|
|
upParts = append(upParts, "options := map[string]any{}")
|
|
upParts = append(upParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), &options)", escapeBacktick(string(rawNewOptions))))
|
|
upParts = append(upParts, fmt.Sprintf("%s.SetOptions(options)\n", varName))
|
|
// ---
|
|
downParts = append(downParts, "options := map[string]any{}")
|
|
downParts = append(downParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), &options)", escapeBacktick(string(rawOldOptions))))
|
|
downParts = append(downParts, fmt.Sprintf("%s.SetOptions(options)\n", varName))
|
|
}
|
|
|
|
// Indexes
|
|
rawNewIndexes, err := marhshalWithoutEscape(new.Indexes, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
rawOldIndexes, err := marhshalWithoutEscape(old.Indexes, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if !bytes.Equal(rawNewIndexes, rawOldIndexes) {
|
|
upParts = append(upParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), &%s.Indexes)\n", escapeBacktick(string(rawNewIndexes)), varName))
|
|
// ---
|
|
downParts = append(downParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), &%s.Indexes)\n", escapeBacktick(string(rawOldIndexes)), varName))
|
|
}
|
|
|
|
// Schema
|
|
// ---------------------------------------------------------------
|
|
// deleted fields
|
|
for _, oldField := range old.Schema.Fields() {
|
|
if new.Schema.GetFieldById(oldField.Id) != nil {
|
|
continue // exist
|
|
}
|
|
|
|
rawOldField, err := marhshalWithoutEscape(oldField, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
fieldVar := fmt.Sprintf("del_%s", oldField.Name)
|
|
|
|
upParts = append(upParts, "// remove")
|
|
upParts = append(upParts, fmt.Sprintf("%s.Schema.RemoveField(%q)\n", varName, oldField.Id))
|
|
|
|
downParts = append(downParts, "// add")
|
|
downParts = append(downParts, fmt.Sprintf("%s := &schema.SchemaField{}", fieldVar))
|
|
downParts = append(downParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), %s)", escapeBacktick(string(rawOldField)), fieldVar))
|
|
downParts = append(downParts, fmt.Sprintf("%s.Schema.AddField(%s)\n", varName, fieldVar))
|
|
}
|
|
|
|
// created fields
|
|
for _, newField := range new.Schema.Fields() {
|
|
if old.Schema.GetFieldById(newField.Id) != nil {
|
|
continue // exist
|
|
}
|
|
|
|
rawNewField, err := marhshalWithoutEscape(newField, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
fieldVar := fmt.Sprintf("new_%s", newField.Name)
|
|
|
|
upParts = append(upParts, "// add")
|
|
upParts = append(upParts, fmt.Sprintf("%s := &schema.SchemaField{}", fieldVar))
|
|
upParts = append(upParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), %s)", escapeBacktick(string(rawNewField)), fieldVar))
|
|
upParts = append(upParts, fmt.Sprintf("%s.Schema.AddField(%s)\n", varName, fieldVar))
|
|
|
|
downParts = append(downParts, "// remove")
|
|
downParts = append(downParts, fmt.Sprintf("%s.Schema.RemoveField(%q)\n", varName, newField.Id))
|
|
}
|
|
|
|
// modified fields
|
|
for _, newField := range new.Schema.Fields() {
|
|
oldField := old.Schema.GetFieldById(newField.Id)
|
|
if oldField == nil {
|
|
continue
|
|
}
|
|
|
|
rawNewField, err := marhshalWithoutEscape(newField, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
rawOldField, err := marhshalWithoutEscape(oldField, "\t\t", "\t")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if bytes.Equal(rawNewField, rawOldField) {
|
|
continue // no change
|
|
}
|
|
|
|
fieldVar := fmt.Sprintf("edit_%s", newField.Name)
|
|
|
|
upParts = append(upParts, "// update")
|
|
upParts = append(upParts, fmt.Sprintf("%s := &schema.SchemaField{}", fieldVar))
|
|
upParts = append(upParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), %s)", escapeBacktick(string(rawNewField)), fieldVar))
|
|
upParts = append(upParts, fmt.Sprintf("%s.Schema.AddField(%s)\n", varName, fieldVar))
|
|
|
|
downParts = append(downParts, "// update")
|
|
downParts = append(downParts, fmt.Sprintf("%s := &schema.SchemaField{}", fieldVar))
|
|
downParts = append(downParts, fmt.Sprintf("json.Unmarshal([]byte(`%s`), %s)", escapeBacktick(string(rawOldField)), fieldVar))
|
|
downParts = append(downParts, fmt.Sprintf("%s.Schema.AddField(%s)\n", varName, fieldVar))
|
|
}
|
|
// ---------------------------------------------------------------
|
|
|
|
if len(upParts) == 0 && len(downParts) == 0 {
|
|
return "", emptyTemplateErr
|
|
}
|
|
|
|
up := strings.Join(upParts, "\n\t\t")
|
|
down := strings.Join(downParts, "\n\t\t")
|
|
combined := up + down
|
|
|
|
// generate imports
|
|
// ---
|
|
var imports string
|
|
|
|
if strings.Contains(combined, "json.Unmarshal(") ||
|
|
strings.Contains(combined, "json.Marshal(") {
|
|
imports += "\n\t\"encoding/json\"\n"
|
|
}
|
|
|
|
imports += "\n\t\"github.com/pocketbase/dbx\""
|
|
imports += "\n\t\"github.com/pocketbase/pocketbase/daos\""
|
|
imports += "\n\tm \"github.com/pocketbase/pocketbase/migrations\""
|
|
|
|
if strings.Contains(combined, "schema.SchemaField{") {
|
|
imports += "\n\t\"github.com/pocketbase/pocketbase/models/schema\""
|
|
}
|
|
|
|
if strings.Contains(combined, "types.Pointer(") {
|
|
imports += "\n\t\"github.com/pocketbase/pocketbase/tools/types\""
|
|
}
|
|
// ---
|
|
|
|
const template = `package %s
|
|
|
|
import (%s
|
|
)
|
|
|
|
func init() {
|
|
m.Register(func(db dbx.Builder) error {
|
|
dao := daos.New(db);
|
|
|
|
collection, err := dao.FindCollectionByNameOrId(%q)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
%s
|
|
|
|
return dao.SaveCollection(collection)
|
|
}, func(db dbx.Builder) error {
|
|
dao := daos.New(db);
|
|
|
|
collection, err := dao.FindCollectionByNameOrId(%q)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
%s
|
|
|
|
return dao.SaveCollection(collection)
|
|
})
|
|
}
|
|
`
|
|
|
|
return fmt.Sprintf(
|
|
template,
|
|
filepath.Base(p.options.Dir),
|
|
imports,
|
|
old.Id, strings.TrimSpace(up),
|
|
new.Id, strings.TrimSpace(down),
|
|
), nil
|
|
}
|
|
|
|
func marhshalWithoutEscape(v any, prefix string, indent string) ([]byte, error) {
|
|
raw, err := json.MarshalIndent(v, prefix, indent)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// unescape escaped unicode characters
|
|
unescaped, err := strconv.Unquote(strings.ReplaceAll(strconv.Quote(string(raw)), `\\u`, `\u`))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return []byte(unescaped), nil
|
|
}
|
|
|
|
func escapeBacktick(v string) string {
|
|
return strings.ReplaceAll(v, "`", "` + \"`\" + `")
|
|
}
|