// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. package app import ( "context" "database/sql" "database/sql/driver" "sync" "github.com/mattermost/mattermost-server/v6/model" "github.com/mattermost/mattermost-server/v6/plugin" ) // DriverImpl implements the plugin.Driver interface on the server-side. // Each new request for a connection/statement/transaction etc, generates // a new entry tracked centrally in a map. Further requests operate on the // object ID. type DriverImpl struct { s *Server connMut sync.RWMutex connMap map[string]*sql.Conn txMut sync.Mutex txMap map[string]driver.Tx stMut sync.RWMutex stMap map[string]driver.Stmt rowsMut sync.RWMutex rowsMap map[string]driver.Rows } func NewDriverImpl(s *Server) *DriverImpl { return &DriverImpl{ s: s, connMap: make(map[string]*sql.Conn), txMap: make(map[string]driver.Tx), stMap: make(map[string]driver.Stmt), rowsMap: make(map[string]driver.Rows), } } func (d *DriverImpl) Conn(isMaster bool) (string, error) { dbFunc := d.s.sqlStore.GetMaster if !isMaster { dbFunc = d.s.sqlStore.GetReplica } conn, err := dbFunc().Db.Conn(context.Background()) if err != nil { return "", err } connID := model.NewId() d.connMut.Lock() d.connMap[connID] = conn d.connMut.Unlock() return connID, nil } // According to https://golang.org/pkg/database/sql/#Conn, a client can call // Close on a connection, concurrently while running a query. // // Therefore, we have to handle the case where the connection is no longer // present in the map because it has been closed. ErrBadConn is a good choice // here which indicates the sql package to retry on a new connection. // // ConnPing, ConnQuery, ConnClose, Tx, and Stmt do this. func (d *DriverImpl) ConnPing(connID string) error { d.connMut.RLock() conn, ok := d.connMap[connID] d.connMut.RUnlock() if !ok { return driver.ErrBadConn } return conn.Raw(func(innerConn interface{}) error { return innerConn.(driver.Pinger).Ping(context.Background()) }) } func (d *DriverImpl) ConnQuery(connID, q string, args []driver.NamedValue) (_ string, err error) { var rows driver.Rows d.connMut.RLock() conn, ok := d.connMap[connID] d.connMut.RUnlock() if !ok { return "", driver.ErrBadConn } err = conn.Raw(func(innerConn interface{}) error { rows, err = innerConn.(driver.QueryerContext).QueryContext(context.Background(), q, args) return err }) if err != nil { return "", err } rowsID := model.NewId() d.rowsMut.Lock() d.rowsMap[rowsID] = rows d.rowsMut.Unlock() return rowsID, nil } func (d *DriverImpl) ConnExec(connID, q string, args []driver.NamedValue) (_ plugin.ResultContainer, err error) { var res driver.Result var ret plugin.ResultContainer d.connMut.RLock() conn, ok := d.connMap[connID] d.connMut.RUnlock() if !ok { return ret, driver.ErrBadConn } err = conn.Raw(func(innerConn interface{}) error { res, err = innerConn.(driver.ExecerContext).ExecContext(context.Background(), q, args) return err }) if err != nil { return ret, err } ret.LastID, ret.LastIDError = res.LastInsertId() ret.RowsAffected, ret.RowsAffectedError = res.RowsAffected() return ret, nil } func (d *DriverImpl) ConnClose(connID string) error { d.connMut.Lock() conn, ok := d.connMap[connID] if !ok { d.connMut.Unlock() return driver.ErrBadConn } delete(d.connMap, connID) d.connMut.Unlock() return conn.Close() } func (d *DriverImpl) Tx(connID string, opts driver.TxOptions) (_ string, err error) { var tx driver.Tx d.connMut.RLock() conn, ok := d.connMap[connID] d.connMut.RUnlock() if !ok { return "", driver.ErrBadConn } err = conn.Raw(func(innerConn interface{}) error { tx, err = innerConn.(driver.ConnBeginTx).BeginTx(context.Background(), opts) return err }) if err != nil { return "", err } txID := model.NewId() d.txMut.Lock() d.txMap[txID] = tx d.txMut.Unlock() return txID, nil } func (d *DriverImpl) TxCommit(txID string) error { d.txMut.Lock() tx := d.txMap[txID] delete(d.txMap, txID) d.txMut.Unlock() return tx.Commit() } func (d *DriverImpl) TxRollback(txID string) error { d.txMut.Lock() tx := d.txMap[txID] delete(d.txMap, txID) d.txMut.Unlock() return tx.Rollback() } func (d *DriverImpl) Stmt(connID, q string) (_ string, err error) { var stmt driver.Stmt d.connMut.RLock() conn, ok := d.connMap[connID] d.connMut.RUnlock() if !ok { return "", driver.ErrBadConn } err = conn.Raw(func(innerConn interface{}) error { stmt, err = innerConn.(driver.Conn).Prepare(q) return err }) if err != nil { return "", err } stID := model.NewId() d.stMut.Lock() d.stMap[stID] = stmt d.stMut.Unlock() return stID, nil } func (d *DriverImpl) StmtClose(stID string) error { d.stMut.Lock() err := d.stMap[stID].Close() delete(d.stMap, stID) d.stMut.Unlock() return err } func (d *DriverImpl) StmtNumInput(stID string) int { d.stMut.RLock() defer d.stMut.RUnlock() return d.stMap[stID].NumInput() } func (d *DriverImpl) StmtQuery(stID string, args []driver.NamedValue) (string, error) { argVals := make([]driver.Value, len(args)) for i, a := range args { argVals[i] = a.Value } d.stMut.RLock() st := d.stMap[stID] d.stMut.RUnlock() rows, err := st.Query(argVals) //nolint:staticcheck if err != nil { return "", err } rowsID := model.NewId() d.rowsMut.Lock() d.rowsMap[rowsID] = rows d.rowsMut.Unlock() return rowsID, nil } func (d *DriverImpl) StmtExec(stID string, args []driver.NamedValue) (plugin.ResultContainer, error) { argVals := make([]driver.Value, len(args)) for i, a := range args { argVals[i] = a.Value } var ret plugin.ResultContainer d.stMut.RLock() st := d.stMap[stID] d.stMut.RUnlock() res, err := st.Exec(argVals) //nolint:staticcheck if err != nil { return ret, err } ret.LastID, ret.LastIDError = res.LastInsertId() ret.RowsAffected, ret.RowsAffectedError = res.RowsAffected() return ret, nil } func (d *DriverImpl) RowsColumns(rowsID string) []string { d.rowsMut.RLock() defer d.rowsMut.RUnlock() return d.rowsMap[rowsID].Columns() } func (d *DriverImpl) RowsClose(rowsID string) error { d.rowsMut.Lock() defer d.rowsMut.Unlock() err := d.rowsMap[rowsID].Close() delete(d.rowsMap, rowsID) return err } func (d *DriverImpl) RowsNext(rowsID string, dest []driver.Value) error { d.rowsMut.RLock() rows := d.rowsMap[rowsID] d.rowsMut.RUnlock() return rows.Next(dest) } func (d *DriverImpl) RowsHasNextResultSet(rowsID string) bool { d.rowsMut.RLock() defer d.rowsMut.RUnlock() return d.rowsMap[rowsID].(driver.RowsNextResultSet).HasNextResultSet() } func (d *DriverImpl) RowsNextResultSet(rowsID string) error { d.rowsMut.RLock() defer d.rowsMut.RUnlock() return d.rowsMap[rowsID].(driver.RowsNextResultSet).NextResultSet() } func (d *DriverImpl) RowsColumnTypeDatabaseTypeName(rowsID string, index int) string { d.rowsMut.RLock() defer d.rowsMut.RUnlock() return d.rowsMap[rowsID].(driver.RowsColumnTypeDatabaseTypeName).ColumnTypeDatabaseTypeName(index) } func (d *DriverImpl) RowsColumnTypePrecisionScale(rowsID string, index int) (int64, int64, bool) { d.rowsMut.RLock() defer d.rowsMut.RUnlock() return d.rowsMap[rowsID].(driver.RowsColumnTypePrecisionScale).ColumnTypePrecisionScale(index) }