mirror of
https://github.com/mattermost/mattermost.git
synced 2026-02-18 18:18:23 -05:00
* Remove legacy quoteColumnName() utility Since Mattermost only supports PostgreSQL, the quoteColumnName() helper that was designed to handle database-specific column quoting is no longer needed. The function was a no-op that simply returned the column name unchanged. Remove the function from utils.go and update status_store.go to use the "Manual" column name directly. * Remove legacy driver checks from store.go Since Mattermost only supports PostgreSQL, remove conditional checks for different database drivers: - Simplify specialSearchChars() to always return PostgreSQL-compatible chars - Remove driver check from computeBinaryParam() - Remove driver check from computeDefaultTextSearchConfig() - Simplify GetDbVersion() to use PostgreSQL syntax directly - Remove switch statement from ensureMinimumDBVersion() - Remove unused driver parameter from versionString() * Remove MySQL alternatives for batch delete operations Since Mattermost only supports PostgreSQL, remove the MySQL-specific DELETE...LIMIT syntax and keep only the PostgreSQL array-based approach: - reaction_store.go: Use PostgreSQL array syntax for PermanentDeleteBatch - file_info_store.go: Use PostgreSQL array syntax for PermanentDeleteBatch - preference_store.go: Use PostgreSQL tuple IN subquery for DeleteInvalidVisibleDmsGms * Remove MySQL alternatives for UPDATE...FROM syntax Since Mattermost only supports PostgreSQL, remove the MySQL-specific UPDATE syntax that joins tables differently: - thread_store.go: Use PostgreSQL UPDATE...FROM syntax in MarkAllAsReadByChannels and MarkAllAsReadByTeam - post_store.go: Use PostgreSQL UPDATE...FROM syntax in deleteThreadFiles * Remove MySQL alternatives for JSON and subquery operations Since Mattermost only supports PostgreSQL, remove the MySQL-specific JSON and subquery syntax: - thread_store.go: Use PostgreSQL JSONB operators for updating participants - access_control_policy_store.go: Use PostgreSQL JSONB @> operator for querying JSON imports - session_store.go: Use PostgreSQL subquery syntax for Cleanup - job_store.go: Use PostgreSQL subquery syntax for Cleanup * Remove MySQL alternatives for CTE queries Since Mattermost only supports PostgreSQL, simplify code that uses CTEs (Common Table Expressions): - channel_store.go: Remove MySQL CASE-based fallback in UpdateLastViewedAt and use PostgreSQL CTE exclusively - draft_store.go: Remove driver checks in DeleteEmptyDraftsByCreateAtAndUserId, DeleteOrphanDraftsByCreateAtAndUserId, and determineMaxDraftSize * Remove driver checks in migrate.go and schema_dump.go Simplify migration code to use PostgreSQL driver directly since PostgreSQL is the only supported database. * Remove driver checks in sqlx_wrapper.go Always apply lowercase named parameter transformation since PostgreSQL is the only supported database. * Remove driver checks in user_store.go Simplify user store functions to use PostgreSQL-only code paths: - Remove isPostgreSQL parameter from helper functions - Use LEFT JOIN pattern instead of subqueries for bot filtering - Always use case-insensitive LIKE with lower() for search - Remove MySQL-specific role filtering alternatives * Remove driver checks in post_store.go Simplify post_store.go to use PostgreSQL-only code paths: - Inline getParentsPostsPostgreSQL into getParentsPosts - Use PostgreSQL TO_CHAR/TO_TIMESTAMP for date formatting in analytics - Use PostgreSQL array syntax for batch deletes - Simplify determineMaxPostSize to always use information_schema - Use PostgreSQL jsonb subtraction for thread participants - Always execute RefreshPostStats (PostgreSQL materialized views) - Use materialized views for AnalyticsPostCountsByDay - Simplify AnalyticsPostCountByTeam to always use countByTeam * Remove driver checks in channel_store.go Simplify channel_store.go to use PostgreSQL-only code paths: - Always use sq.Dollar.ReplacePlaceholders for UNION queries - Use PostgreSQL LEFT JOIN for retention policy exclusion - Use PostgreSQL jsonb @> operator for access control policy imports - Simplify buildLIKEClause to always use LOWER() for case-insensitive search - Simplify buildFulltextClauseX to always use PostgreSQL to_tsvector/to_tsquery - Simplify searchGroupChannelsQuery to use ARRAY_TO_STRING/ARRAY_AGG * Remove driver checks in file_info_store.go Simplify file_info_store.go to use PostgreSQL-only code paths: - Always use PostgreSQL to_tsvector/to_tsquery for file search - Use file_stats materialized view for CountAll() - Use file_stats materialized view for GetStorageUsage() when not including deleted - Always execute RefreshFileStats() for materialized view refresh * Remove driver checks in attributes_store.go Simplify attributes_store.go to use PostgreSQL-only code paths: - Always execute RefreshAttributes() for materialized view refresh - Remove isPostgreSQL parameter from generateSearchQueryForExpression - Always use PostgreSQL LOWER() LIKE LOWER() syntax for case-insensitive search * Remove driver checks in retention_policy_store.go Simplify retention_policy_store.go to use PostgreSQL-only code paths: - Remove isPostgres parameter from scanRetentionIdsForDeletion - Always use pq.Array for scanning retention IDs - Always use pq.Array for inserting retention IDs - Remove unused json import * Remove driver checks in property stores Simplify property_field_store.go and property_value_store.go to use PostgreSQL-only code paths: - Always use PostgreSQL type casts (::text, ::jsonb, ::bigint, etc.) - Remove isPostgres variable and conditionals * Remove driver checks in channel_member_history_store.go Simplify PermanentDeleteBatch to use PostgreSQL-only code path: - Always use ctid-based subquery for DELETE with LIMIT * Remove remaining driver checks in user_store.go Simplify user_store.go to use PostgreSQL-only code paths: - Use LEFT JOIN for bot exclusion in AnalyticsActiveCountForPeriod - Use LEFT JOIN for bot exclusion in IsEmpty * Simplify fulltext search by consolidating buildFulltextClause functions Remove convertMySQLFullTextColumnsToPostgres and consolidate buildFulltextClause and buildFulltextClauseX into a single function that takes variadic column arguments and returns sq.Sqlizer. * Simplify SQL stores leveraging PostgreSQL-only support - Simplify UpdateMembersRole in channel_store.go and team_store.go to use UPDATE...RETURNING instead of SELECT + UPDATE - Simplify GetPostReminders in post_store.go to use DELETE...RETURNING - Simplify DeleteOrphanedRows queries by removing MySQL workarounds for subquery locking issues - Simplify UpdateUserLastSyncAt to use UPDATE...FROM...RETURNING instead of fetching user first then updating - Remove MySQL index hint workarounds in ORDER BY clauses - Update outdated comments referencing MySQL - Consolidate buildFulltextClause and remove convertMySQLFullTextColumnsToPostgres * Remove MySQL-specific test artifacts - Delete unused MySQLStopWords variable and stop_word.go file - Remove redundant testSearchEmailAddressesWithQuotes test (already covered by testSearchEmailAddresses) - Update comment that referenced MySQL query planning * Remove MySQL references from server code outside sqlstore - Update config example and DSN parsing docs to reflect PostgreSQL-only support - Remove mysql:// scheme check from IsDatabaseDSN - Simplify SanitizeDataSource to only handle PostgreSQL - Remove outdated MySQL comments from model and plugin code * Remove MySQL references from test files - Update test DSNs to use PostgreSQL format - Remove dead mysql-replica flag and replicaFlag variable - Simplify tests that had MySQL/PostgreSQL branches * Update docs and test config to use PostgreSQL - Update mmctl config set example to use postgres driver - Update test-config.json to use PostgreSQL DSN format * Remove MySQL migration scripts, test data, and docker image Delete MySQL-related files that are no longer needed: - ESR upgrade scripts (esr.*.mysql.*.sql) - MySQL schema dumps (mattermost-mysql-*.sql) - MySQL replication test scripts (replica-*.sh, mysql-migration-test.sh) - MySQL test warmup data (mysql_migration_warmup.sql) - MySQL docker image reference from mirror-docker-images.json * Remove MySQL references from webapp - Simplify minimumHashtagLength description to remove MySQL-specific configuration note - Remove unused HIDE_MYSQL_STATS_NOTIFICATION preference constant - Update en.json i18n source file * clean up e2e-tests * rm server/tests/template.load * Use teamMemberSliceColumns() in UpdateMembersRole RETURNING clause Refactor to use the existing helper function instead of hardcoding the column names, ensuring consistency if the columns are updated. * u.id -> u.Id * address code review feedback --------- Co-authored-by: Mattermost Build <build@mattermost.com>
488 lines
13 KiB
Go
488 lines
13 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package sqlstore
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"errors"
|
|
"net"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"sync/atomic"
|
|
"time"
|
|
"unicode"
|
|
|
|
"github.com/jmoiron/sqlx"
|
|
|
|
"github.com/mattermost/mattermost/server/public/shared/mlog"
|
|
"github.com/mattermost/mattermost/server/v8/channels/store/storetest"
|
|
sq "github.com/mattermost/squirrel"
|
|
)
|
|
|
|
type StoreTestWrapper struct {
|
|
orig *SqlStore
|
|
}
|
|
|
|
func NewStoreTestWrapper(orig *SqlStore) *StoreTestWrapper {
|
|
return &StoreTestWrapper{orig}
|
|
}
|
|
|
|
func (w *StoreTestWrapper) GetMaster() storetest.SqlXExecutor {
|
|
return w.orig.GetMaster()
|
|
}
|
|
|
|
func (w *StoreTestWrapper) DriverName() string {
|
|
return w.orig.DriverName()
|
|
}
|
|
|
|
func (w *StoreTestWrapper) GetQueryPlaceholder() sq.PlaceholderFormat {
|
|
return w.orig.getQueryPlaceholder()
|
|
}
|
|
|
|
type Builder interface {
|
|
ToSql() (string, []any, error)
|
|
}
|
|
|
|
// sqlxExecutor exposes sqlx operations. It is used to enable some internal store methods to
|
|
// accept both transactions (*sqlxTxWrapper) and common db handlers (*sqlxDbWrapper).
|
|
type sqlxExecutor interface {
|
|
Get(dest any, query string, args ...any) error
|
|
GetBuilder(dest any, builder Builder) error
|
|
NamedExec(query string, arg any) (sql.Result, error)
|
|
Exec(query string, args ...any) (sql.Result, error)
|
|
ExecBuilder(builder Builder) (sql.Result, error)
|
|
ExecRaw(query string, args ...any) (sql.Result, error)
|
|
NamedQuery(query string, arg any) (*sqlx.Rows, error)
|
|
QueryRowX(query string, args ...any) *sqlx.Row
|
|
QueryX(query string, args ...any) (*sqlx.Rows, error)
|
|
Select(dest any, query string, args ...any) error
|
|
SelectBuilder(dest any, builder Builder) error
|
|
}
|
|
|
|
// namedParamRegex is used to capture all named parameters and convert them to lowercase.
|
|
// This will also lowercase any constant strings containing a :, but sqlx
|
|
// will fail the query, so it won't be checked in inadvertently.
|
|
var namedParamRegex = regexp.MustCompile(`:\w+`)
|
|
|
|
type sqlxDBWrapper struct {
|
|
*sqlx.DB
|
|
queryTimeout time.Duration
|
|
trace bool
|
|
isOnline *atomic.Bool
|
|
}
|
|
|
|
func newSqlxDBWrapper(db *sqlx.DB, timeout time.Duration, trace bool) *sqlxDBWrapper {
|
|
w := &sqlxDBWrapper{
|
|
DB: db,
|
|
queryTimeout: timeout,
|
|
trace: trace,
|
|
isOnline: &atomic.Bool{},
|
|
}
|
|
w.isOnline.Store(true)
|
|
return w
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) Stats() sql.DBStats {
|
|
return w.DB.Stats()
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) Beginx() (*sqlxTxWrapper, error) {
|
|
tx, err := w.DB.Beginx()
|
|
if err != nil {
|
|
return nil, w.checkErr(err)
|
|
}
|
|
|
|
return newSqlxTxWrapper(tx, w.queryTimeout, w.trace, w), nil
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) BeginXWithIsolation(opts *sql.TxOptions) (*sqlxTxWrapper, error) {
|
|
tx, err := w.DB.BeginTxx(context.Background(), opts)
|
|
if err != nil {
|
|
return nil, w.checkErr(err)
|
|
}
|
|
|
|
return newSqlxTxWrapper(tx, w.queryTimeout, w.trace, w), nil
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) Get(dest any, query string, args ...any) error {
|
|
query = w.DB.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErr(w.DB.GetContext(ctx, dest, query, args...))
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) GetBuilder(dest any, builder Builder) error {
|
|
query, args, err := builder.ToSql()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return w.Get(dest, query, args...)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) NamedExec(query string, arg any) (sql.Result, error) {
|
|
query = namedParamRegex.ReplaceAllStringFunc(query, strings.ToLower)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), arg)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErrWithResult(w.DB.NamedExecContext(ctx, query, arg))
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) Exec(query string, args ...any) (sql.Result, error) {
|
|
query = w.DB.Rebind(query)
|
|
|
|
return w.ExecRaw(query, args...)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) ExecBuilder(builder Builder) (sql.Result, error) {
|
|
query, args, err := builder.ToSql()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return w.Exec(query, args...)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) ExecNoTimeout(query string, args ...any) (sql.Result, error) {
|
|
query = w.DB.Rebind(query)
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErrWithResult(w.DB.ExecContext(context.Background(), query, args...))
|
|
}
|
|
|
|
// ExecRaw is like Exec but without any rebinding of params. You need to pass
|
|
// the exact param types of your target database.
|
|
func (w *sqlxDBWrapper) ExecRaw(query string, args ...any) (sql.Result, error) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErrWithResult(w.DB.ExecContext(ctx, query, args...))
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) NamedQuery(query string, arg any) (*sqlx.Rows, error) {
|
|
query = namedParamRegex.ReplaceAllStringFunc(query, strings.ToLower)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), arg)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErrWithRows(w.DB.NamedQueryContext(ctx, query, arg))
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) QueryRowX(query string, args ...any) *sqlx.Row {
|
|
query = w.DB.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.DB.QueryRowxContext(ctx, query, args...)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) QueryX(query string, args ...any) (*sqlx.Rows, error) {
|
|
query = w.DB.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErrWithRows(w.DB.QueryxContext(ctx, query, args...))
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) Select(dest any, query string, args ...any) error {
|
|
return w.SelectCtx(context.Background(), dest, query, args...)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) SelectCtx(ctx context.Context, dest any, query string, args ...any) error {
|
|
query = w.DB.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(ctx, w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.checkErr(w.DB.SelectContext(ctx, dest, query, args...))
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) SelectBuilder(dest any, builder Builder) error {
|
|
return w.SelectBuilderCtx(context.Background(), dest, builder)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) SelectBuilderCtx(ctx context.Context, dest any, builder Builder) error {
|
|
query, args, err := builder.ToSql()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return w.SelectCtx(ctx, dest, query, args...)
|
|
}
|
|
|
|
type sqlxTxWrapper struct {
|
|
*sqlx.Tx
|
|
queryTimeout time.Duration
|
|
trace bool
|
|
dbw *sqlxDBWrapper
|
|
}
|
|
|
|
func newSqlxTxWrapper(tx *sqlx.Tx, timeout time.Duration, trace bool, dbw *sqlxDBWrapper) *sqlxTxWrapper {
|
|
return &sqlxTxWrapper{
|
|
Tx: tx,
|
|
queryTimeout: timeout,
|
|
trace: trace,
|
|
dbw: dbw,
|
|
}
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) Get(dest any, query string, args ...any) error {
|
|
query = w.Tx.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.dbw.checkErr(w.Tx.GetContext(ctx, dest, query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) GetBuilder(dest any, builder Builder) error {
|
|
query, args, err := builder.ToSql()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return w.dbw.checkErr(w.Get(dest, query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) Exec(query string, args ...any) (sql.Result, error) {
|
|
query = w.Tx.Rebind(query)
|
|
|
|
return w.dbw.checkErrWithResult(w.ExecRaw(query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) ExecNoTimeout(query string, args ...any) (sql.Result, error) {
|
|
query = w.Tx.Rebind(query)
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.dbw.checkErrWithResult(w.Tx.ExecContext(context.Background(), query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) ExecBuilder(builder Builder) (sql.Result, error) {
|
|
query, args, err := builder.ToSql()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return w.Exec(query, args...)
|
|
}
|
|
|
|
// ExecRaw is like Exec but without any rebinding of params. You need to pass
|
|
// the exact param types of your target database.
|
|
func (w *sqlxTxWrapper) ExecRaw(query string, args ...any) (sql.Result, error) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.dbw.checkErrWithResult(w.Tx.ExecContext(ctx, query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) NamedExec(query string, arg any) (sql.Result, error) {
|
|
query = namedParamRegex.ReplaceAllStringFunc(query, strings.ToLower)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), arg)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.dbw.checkErrWithResult(w.Tx.NamedExecContext(ctx, query, arg))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) NamedQuery(query string, arg any) (*sqlx.Rows, error) {
|
|
query = namedParamRegex.ReplaceAllStringFunc(query, strings.ToLower)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), arg)
|
|
}(time.Now())
|
|
}
|
|
|
|
// There is no tx.NamedQueryContext support in the sqlx API. (https://github.com/jmoiron/sqlx/issues/447)
|
|
// So we need to implement this ourselves.
|
|
type result struct {
|
|
rows *sqlx.Rows
|
|
err error
|
|
}
|
|
|
|
// Need to add a buffer of 1 to prevent goroutine leak.
|
|
resChan := make(chan *result, 1)
|
|
go func() {
|
|
rows, err := w.Tx.NamedQuery(query, arg)
|
|
resChan <- &result{
|
|
rows: rows,
|
|
err: err,
|
|
}
|
|
}()
|
|
|
|
// staticcheck fails to check that res gets re-assigned later.
|
|
res := &result{} //nolint:staticcheck
|
|
select {
|
|
case res = <-resChan:
|
|
case <-ctx.Done():
|
|
res = &result{
|
|
rows: nil,
|
|
err: ctx.Err(),
|
|
}
|
|
}
|
|
|
|
return res.rows, w.dbw.checkErr(res.err)
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) QueryRowX(query string, args ...any) *sqlx.Row {
|
|
query = w.Tx.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.Tx.QueryRowxContext(ctx, query, args...)
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) QueryX(query string, args ...any) (*sqlx.Rows, error) {
|
|
query = w.Tx.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.dbw.checkErrWithRows(w.Tx.QueryxContext(ctx, query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) Select(dest any, query string, args ...any) error {
|
|
query = w.Tx.Rebind(query)
|
|
ctx, cancel := context.WithTimeout(context.Background(), w.queryTimeout)
|
|
defer cancel()
|
|
|
|
if w.trace {
|
|
defer func(then time.Time) {
|
|
printArgs(query, time.Since(then), args)
|
|
}(time.Now())
|
|
}
|
|
|
|
return w.dbw.checkErr(w.Tx.SelectContext(ctx, dest, query, args...))
|
|
}
|
|
|
|
func (w *sqlxTxWrapper) SelectBuilder(dest any, builder Builder) error {
|
|
query, args, err := builder.ToSql()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return w.Select(dest, query, args...)
|
|
}
|
|
|
|
func removeSpace(r rune) rune {
|
|
// Strip everything except ' '
|
|
// This also strips out more than one space,
|
|
// but we ignore it for now until someone complains.
|
|
if unicode.IsSpace(r) && r != ' ' {
|
|
return -1
|
|
}
|
|
return r
|
|
}
|
|
|
|
func printArgs(query string, dur time.Duration, args ...any) {
|
|
query = strings.Map(removeSpace, query)
|
|
fields := make([]mlog.Field, 0, len(args)+1)
|
|
fields = append(fields, mlog.Duration("duration", dur))
|
|
for i, arg := range args {
|
|
fields = append(fields, mlog.Any("arg"+strconv.Itoa(i), arg))
|
|
}
|
|
mlog.Debug(query, fields...)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) checkErrWithResult(res sql.Result, err error) (sql.Result, error) {
|
|
return res, w.checkErr(err)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) checkErrWithRows(res *sqlx.Rows, err error) (*sqlx.Rows, error) {
|
|
return res, w.checkErr(err)
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) checkErr(err error) error {
|
|
var netError *net.OpError
|
|
if errors.As(err, &netError) && (!netError.Temporary() && !netError.Timeout()) {
|
|
w.isOnline.Store(false)
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (w *sqlxDBWrapper) Online() bool {
|
|
return w.isOnline.Load()
|
|
}
|