// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. package sqlstore import ( "database/sql" "fmt" sq "github.com/mattermost/squirrel" "github.com/pkg/errors" "github.com/mattermost/mattermost/server/public/model" "github.com/mattermost/mattermost/server/v8/channels/store" ) type SqlSystemStore struct { *SqlStore systemSelectQuery sq.SelectBuilder } func newSqlSystemStore(sqlStore *SqlStore) store.SystemStore { s := SqlSystemStore{SqlStore: sqlStore} s.systemSelectQuery = s.getQueryBuilder().Select("Name", "Value").From("Systems") return &s } func (s SqlSystemStore) Save(system *model.System) error { query := "INSERT INTO Systems (Name, Value) VALUES (:Name, :Value)" if _, err := s.GetMaster().NamedExec(query, system); err != nil { return errors.Wrapf(err, "failed to save system property with name=%s", system.Name) } return nil } func (s SqlSystemStore) SaveOrUpdate(system *model.System) error { query := s.getQueryBuilder(). Insert("Systems"). Columns("Name", "Value"). Values(system.Name, system.Value) query = query.SuffixExpr(sq.Expr("ON CONFLICT (name) DO UPDATE SET Value = ?", system.Value)) queryString, args, err := query.ToSql() if err != nil { return errors.Wrap(err, "system_tosql") } if _, err := s.GetMaster().Exec(queryString, args...); err != nil { return errors.Wrap(err, "failed to upsert system property") } return nil } func (s SqlSystemStore) Update(system *model.System) error { query := "UPDATE Systems SET Value=:Value WHERE Name=:Name" if _, err := s.GetMaster().NamedExec(query, system); err != nil { return errors.Wrapf(err, "failed to update system property with name=%s", system.Name) } return nil } func (s SqlSystemStore) Get() (model.StringMap, error) { systems := []model.System{} props := make(model.StringMap) query := s.systemSelectQuery if err := s.GetReplica().SelectBuilder(&systems, query); err != nil { return nil, errors.Wrap(err, "failed to get System list") } for _, prop := range systems { props[prop.Name] = prop.Value } return props, nil } func (s SqlSystemStore) GetByName(name string) (*model.System, error) { var system model.System query := s.systemSelectQuery.Where(sq.Eq{"Name": name}) if err := s.GetMaster().GetBuilder(&system, query); err != nil { if err == sql.ErrNoRows { return nil, store.NewErrNotFound("System", fmt.Sprintf("name=%s", system.Name)) } return nil, errors.Wrapf(err, "failed to get system property with name=%s", system.Name) } return &system, nil } func (s SqlSystemStore) PermanentDeleteByName(name string) (*model.System, error) { var system model.System if _, err := s.GetMaster().Exec("DELETE FROM Systems WHERE Name = ?", name); err != nil { return nil, errors.Wrapf(err, "failed to permanent delete system property with name=%s", system.Name) } return &system, nil } // InsertIfExists inserts a given system value if it does not already exist. If a value // already exists, it returns the old one, else returns the new one. func (s SqlSystemStore) InsertIfExists(system *model.System) (_ *model.System, err error) { tx, err := s.GetMaster().BeginXWithIsolation(&sql.TxOptions{ Isolation: sql.LevelSerializable, }) if err != nil { return nil, errors.Wrap(err, "begin_transaction") } defer finalizeTransactionX(tx, &err) var origSystem model.System query := s.systemSelectQuery.Where(sq.Eq{"Name": system.Name}) if err := tx.GetBuilder(&origSystem, query); err != nil && err != sql.ErrNoRows { return nil, errors.Wrapf(err, "failed to get system property with name=%s", system.Name) } if origSystem.Value != "" { // Already a value exists, return that. return &origSystem, nil } // Key does not exist, need to insert. if _, err := tx.NamedExec("INSERT INTO Systems (Name, Value) VALUES (:Name, :Value)", system); err != nil { return nil, errors.Wrapf(err, "failed to save system property with name=%s", system.Name) } if err := tx.Commit(); err != nil { return nil, errors.Wrap(err, "commit_transaction") } return system, nil }