923093042a
Signed-off-by: Michael Mayer <michael@liquidbytes.net>
154 lines
3.6 KiB
Go
154 lines
3.6 KiB
Go
/*
|
|
Package entity contains models for data storage based on GORM.
|
|
|
|
See http://gorm.io/docs/ for more information about GORM.
|
|
|
|
Additional information concerning data storage can be found in our Developer Guide:
|
|
|
|
https://github.com/photoprism/photoprism/wiki/Storage
|
|
*/
|
|
package entity
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/jinzhu/gorm"
|
|
"github.com/photoprism/photoprism/internal/event"
|
|
)
|
|
|
|
var log = event.Log
|
|
|
|
func logError(result *gorm.DB) {
|
|
if result.Error != nil {
|
|
log.Error(result.Error.Error())
|
|
}
|
|
}
|
|
|
|
type Types map[string]interface{}
|
|
|
|
// List of database entities and their table names.
|
|
var Entities = Types{
|
|
"accounts": &Account{},
|
|
"files": &File{},
|
|
"files_share": &FileShare{},
|
|
"files_sync": &FileSync{},
|
|
"photos": &Photo{},
|
|
"descriptions": &Description{},
|
|
"places": &Place{},
|
|
"locations": &Location{},
|
|
"cameras": &Camera{},
|
|
"lenses": &Lens{},
|
|
"countries": &Country{},
|
|
"albums": &Album{},
|
|
"photos_albums": &PhotoAlbum{},
|
|
"labels": &Label{},
|
|
"categories": &Category{},
|
|
"photos_labels": &PhotoLabel{},
|
|
"keywords": &Keyword{},
|
|
"photos_keywords": &PhotoKeyword{},
|
|
"links": &Link{},
|
|
}
|
|
|
|
// WaitForMigration waits for the database migration to be successful.
|
|
func (list Types) WaitForMigration() {
|
|
attempts := 100
|
|
|
|
for name := range list {
|
|
for i := 0; i <= attempts; i++ {
|
|
if err := Db().Raw(fmt.Sprintf("DESCRIBE `%s`", name)).Scan(&struct{}{}).Error; err == nil {
|
|
// log.Debugf("entity: table %s migrated", name)
|
|
break
|
|
} else {
|
|
log.Debugf("entity: wait for migration %s (%s)", err.Error(), name)
|
|
}
|
|
|
|
if i == attempts {
|
|
panic("migration failed")
|
|
}
|
|
|
|
time.Sleep(50 * time.Millisecond)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Truncate removes all data from tables without dropping them.
|
|
func (list Types) Truncate() {
|
|
for name := range list {
|
|
if err := Db().Raw(fmt.Sprintf("TRUNCATE TABLE `%s`", name)).Scan(&struct{}{}).Error; err == nil {
|
|
log.Debugf("entity: removed all data from %s", name)
|
|
break
|
|
} else if err.Error() != "record not found" {
|
|
log.Debugf("entity: truncate %s (%s)", err.Error(), name)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Drop migrates all database tables of registered entities.
|
|
func (list Types) Migrate() {
|
|
for _, entity := range list {
|
|
if err := UnscopedDb().AutoMigrate(entity).Error; err != nil {
|
|
log.Debugf("entity: migrate %s (waiting 1s)", err.Error())
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
if err := UnscopedDb().AutoMigrate(entity).Error; err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Drop drops all database tables of registered entities.
|
|
func (list Types) Drop() {
|
|
for _, entity := range list {
|
|
if err := UnscopedDb().DropTableIfExists(entity).Error; err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Creates default database entries for test and production.
|
|
func CreateDefaultFixtures() {
|
|
CreateUnknownPlace()
|
|
CreateUnknownCountry()
|
|
CreateUnknownCamera()
|
|
CreateUnknownLens()
|
|
CreateViews()
|
|
}
|
|
|
|
// MigrateDb creates all tables and inserts default entities as needed.
|
|
func MigrateDb() {
|
|
Entities.Migrate()
|
|
Entities.WaitForMigration()
|
|
|
|
CreateDefaultFixtures()
|
|
}
|
|
|
|
// ResetTestFixtures drops database tables for all known entities and re-creates them with fixtures.
|
|
func ResetTestFixtures() {
|
|
Entities.Migrate()
|
|
Entities.WaitForMigration()
|
|
Entities.Truncate()
|
|
|
|
CreateDefaultFixtures()
|
|
|
|
CreateTestFixtures()
|
|
}
|
|
|
|
// InitTestDb connects to and completely initializes the test database incl fixtures.
|
|
func InitTestDb(dsn string) *Gorm {
|
|
if HasDbProvider() {
|
|
return nil
|
|
}
|
|
|
|
db := &Gorm{
|
|
Driver: "mysql",
|
|
Dsn: dsn,
|
|
}
|
|
|
|
SetDbProvider(db)
|
|
ResetTestFixtures()
|
|
|
|
return db
|
|
}
|