2023-03-12 18:00:57 +03:00
|
|
|
// GoToSocial
|
|
|
|
// Copyright (C) GoToSocial Authors admin@gotosocial.org
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2021-08-25 16:34:33 +03:00
|
|
|
|
|
|
|
package bundb
|
|
|
|
|
|
|
|
import (
|
2023-07-24 15:14:13 +03:00
|
|
|
"strings"
|
|
|
|
|
2021-09-09 17:15:25 +03:00
|
|
|
"github.com/superseriousbusiness/gotosocial/internal/db"
|
2021-08-25 16:34:33 +03:00
|
|
|
"github.com/uptrace/bun"
|
|
|
|
)
|
|
|
|
|
2023-07-24 15:14:13 +03:00
|
|
|
// likeEscaper is a thread-safe string replacer which escapes
|
|
|
|
// common SQLite + Postgres `LIKE` wildcard chars using the
|
|
|
|
// escape character `\`. Initialized as a var in this package
|
|
|
|
// so it can be reused.
|
|
|
|
var likeEscaper = strings.NewReplacer(
|
|
|
|
`\`, `\\`, // Escape char.
|
|
|
|
`%`, `\%`, // Zero or more char.
|
|
|
|
`_`, `\_`, // Exactly one char.
|
|
|
|
)
|
|
|
|
|
2023-07-31 16:47:35 +03:00
|
|
|
// whereLike appends a WHERE clause to the
|
|
|
|
// given SelectQuery, which searches for
|
|
|
|
// matches of `search` in the given subQuery
|
|
|
|
// using LIKE.
|
2023-07-24 15:14:13 +03:00
|
|
|
func whereLike(
|
|
|
|
query *bun.SelectQuery,
|
|
|
|
subject interface{},
|
|
|
|
search string,
|
|
|
|
) *bun.SelectQuery {
|
|
|
|
// Escape existing wildcard + escape
|
|
|
|
// chars in the search query string.
|
|
|
|
search = likeEscaper.Replace(search)
|
|
|
|
|
|
|
|
// Add our own wildcards back in; search
|
|
|
|
// zero or more chars around the query.
|
|
|
|
search = `%` + search + `%`
|
|
|
|
|
|
|
|
// Append resulting WHERE
|
|
|
|
// clause to the main query.
|
|
|
|
return query.Where(
|
|
|
|
"(?) LIKE ? ESCAPE ?",
|
|
|
|
subject, search, `\`,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2023-07-31 16:47:35 +03:00
|
|
|
// whereStartsLike is like whereLike,
|
|
|
|
// but only searches for strings that
|
|
|
|
// START WITH `search`.
|
|
|
|
func whereStartsLike(
|
|
|
|
query *bun.SelectQuery,
|
|
|
|
subject interface{},
|
|
|
|
search string,
|
|
|
|
) *bun.SelectQuery {
|
|
|
|
// Escape existing wildcard + escape
|
|
|
|
// chars in the search query string.
|
|
|
|
search = likeEscaper.Replace(search)
|
|
|
|
|
|
|
|
// Add our own wildcards back in; search
|
|
|
|
// zero or more chars after the query.
|
|
|
|
search += `%`
|
|
|
|
|
|
|
|
// Append resulting WHERE
|
|
|
|
// clause to the main query.
|
|
|
|
return query.Where(
|
|
|
|
"(?) LIKE ? ESCAPE ?",
|
|
|
|
subject, search, `\`,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-09-09 17:15:25 +03:00
|
|
|
// updateWhere parses []db.Where and adds it to the given update query.
|
|
|
|
func updateWhere(q *bun.UpdateQuery, where []db.Where) {
|
|
|
|
for _, w := range where {
|
|
|
|
query, args := parseWhere(w)
|
|
|
|
q = q.Where(query, args...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// selectWhere parses []db.Where and adds it to the given select query.
|
|
|
|
func selectWhere(q *bun.SelectQuery, where []db.Where) {
|
|
|
|
for _, w := range where {
|
|
|
|
query, args := parseWhere(w)
|
|
|
|
q = q.Where(query, args...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// deleteWhere parses []db.Where and adds it to the given where query.
|
|
|
|
func deleteWhere(q *bun.DeleteQuery, where []db.Where) {
|
|
|
|
for _, w := range where {
|
|
|
|
query, args := parseWhere(w)
|
|
|
|
q = q.Where(query, args...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// parseWhere looks through the options on a single db.Where entry, and
|
|
|
|
// returns the appropriate query string and arguments.
|
|
|
|
func parseWhere(w db.Where) (query string, args []interface{}) {
|
|
|
|
if w.Not {
|
|
|
|
if w.Value == nil {
|
|
|
|
query = "? IS NOT NULL"
|
|
|
|
args = []interface{}{bun.Ident(w.Key)}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
query = "? != ?"
|
2022-10-08 14:50:48 +03:00
|
|
|
args = []interface{}{bun.Ident(w.Key), w.Value}
|
2021-09-09 17:15:25 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.Value == nil {
|
|
|
|
query = "? IS NULL"
|
|
|
|
args = []interface{}{bun.Ident(w.Key)}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
query = "? = ?"
|
2022-10-08 14:50:48 +03:00
|
|
|
args = []interface{}{bun.Ident(w.Key), w.Value}
|
2021-09-09 17:15:25 +03:00
|
|
|
return
|
|
|
|
}
|