mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-10 18:43:40 +00:00
77b4fc5487
* beginning with postgres statements * try pgx * use pgx * database * init works for postgres * arrays working * init for cockroach * init * start tests * tests * TESTS * ch * ch * chore: use go 1.18 * read stmts * fix typo * tests * connection string * add missing error handler * cleanup * start all apis * go mod tidy * old update * switch back to minute * on conflict * replace string slice with `database.StringArray` in db models * fix tests and start * update go version in dockerfile * setup go * clean up * remove notification migration * update * docs: add deploy guide for postgres * fix: revert sonyflake * use `database.StringArray` for daos * use `database.StringArray` every where * new tables * index naming, metadata primary key, project grant role key type * docs(postgres): change to beta * chore: correct compose * fix(defaults): add empty postgres config * refactor: remove unused code * docs: add postgres to self hosted * fix broken link * so? * change title * add mdx to link * fix stmt * update goreleaser in test-code * docs: improve postgres example * update more projections * fix: add beta log for postgres * revert index name change * prerelease * fix: add sequence to v1 "reduce paniced" * log if nil * add logging * fix: log output * fix(import): check if org exists and user * refactor: imports * fix(user): ignore malformed events * refactor: method naming * fix: test * refactor: correct errors.Is call * ci: don't build dev binaries on main * fix(go releaser): update version to 1.11.0 * fix(user): projection should not break * fix(user): handle error properly * docs: correct config example * Update .releaserc.js * Update .releaserc.js Co-authored-by: Livio Amstutz <livio.a@gmail.com> Co-authored-by: Elio Bischof <eliobischof@gmail.com>
432 lines
9.0 KiB
Go
432 lines
9.0 KiB
Go
package query
|
|
|
|
import (
|
|
"errors"
|
|
"reflect"
|
|
|
|
sq "github.com/Masterminds/squirrel"
|
|
|
|
"github.com/zitadel/zitadel/internal/domain"
|
|
)
|
|
|
|
type SearchResponse struct {
|
|
Count uint64
|
|
*LatestSequence
|
|
}
|
|
|
|
type SearchRequest struct {
|
|
Offset uint64
|
|
Limit uint64
|
|
SortingColumn Column
|
|
Asc bool
|
|
}
|
|
|
|
func (req *SearchRequest) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
if req.Offset > 0 {
|
|
query = query.Offset(req.Offset)
|
|
}
|
|
if req.Limit > 0 {
|
|
query = query.Limit(req.Limit)
|
|
}
|
|
|
|
if !req.SortingColumn.isZero() {
|
|
clause := req.SortingColumn.orderBy()
|
|
if !req.Asc {
|
|
clause += " DESC"
|
|
}
|
|
query = query.OrderByClause(clause)
|
|
}
|
|
|
|
return query
|
|
}
|
|
|
|
type SearchQuery interface {
|
|
toQuery(sq.SelectBuilder) sq.SelectBuilder
|
|
comp() sq.Sqlizer
|
|
}
|
|
|
|
type NotNullQuery struct {
|
|
Column Column
|
|
}
|
|
|
|
func NewNotNullQuery(col Column) (*NotNullQuery, error) {
|
|
if col.isZero() {
|
|
return nil, ErrMissingColumn
|
|
}
|
|
return &NotNullQuery{
|
|
Column: col,
|
|
}, nil
|
|
}
|
|
|
|
func (q *NotNullQuery) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (q *NotNullQuery) comp() sq.Sqlizer {
|
|
return sq.NotEq{q.Column.identifier(): nil}
|
|
}
|
|
|
|
type orQuery struct {
|
|
queries []SearchQuery
|
|
}
|
|
|
|
func newOrQuery(queries ...SearchQuery) (*orQuery, error) {
|
|
if len(queries) == 0 {
|
|
return nil, ErrMissingColumn
|
|
}
|
|
return &orQuery{queries: queries}, nil
|
|
}
|
|
|
|
func (q *orQuery) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (q *orQuery) comp() sq.Sqlizer {
|
|
or := make(sq.Or, len(q.queries))
|
|
for i, query := range q.queries {
|
|
or[i] = query.comp()
|
|
}
|
|
return or
|
|
}
|
|
|
|
type TextQuery struct {
|
|
Column Column
|
|
Text string
|
|
Compare TextComparison
|
|
}
|
|
|
|
var (
|
|
ErrInvalidCompare = errors.New("invalid compare")
|
|
ErrMissingColumn = errors.New("missing column")
|
|
ErrInvalidNumber = errors.New("value is no number")
|
|
)
|
|
|
|
func NewTextQuery(col Column, value string, compare TextComparison) (*TextQuery, error) {
|
|
if compare < 0 || compare >= textCompareMax {
|
|
return nil, ErrInvalidCompare
|
|
}
|
|
if col.isZero() {
|
|
return nil, ErrMissingColumn
|
|
}
|
|
return &TextQuery{
|
|
Column: col,
|
|
Text: value,
|
|
Compare: compare,
|
|
}, nil
|
|
}
|
|
|
|
func (q *TextQuery) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (s *TextQuery) comp() sq.Sqlizer {
|
|
switch s.Compare {
|
|
case TextEquals:
|
|
return sq.Eq{s.Column.identifier(): s.Text}
|
|
case TextEqualsIgnoreCase:
|
|
return sq.ILike{s.Column.identifier(): s.Text}
|
|
case TextStartsWith:
|
|
return sq.Like{s.Column.identifier(): s.Text + "%"}
|
|
case TextStartsWithIgnoreCase:
|
|
return sq.ILike{s.Column.identifier(): s.Text + "%"}
|
|
case TextEndsWith:
|
|
return sq.Like{s.Column.identifier(): "%" + s.Text}
|
|
case TextEndsWithIgnoreCase:
|
|
return sq.ILike{s.Column.identifier(): "%" + s.Text}
|
|
case TextContains:
|
|
return sq.Like{s.Column.identifier(): "%" + s.Text + "%"}
|
|
case TextContainsIgnoreCase:
|
|
return sq.ILike{s.Column.identifier(): "%" + s.Text + "%"}
|
|
case TextListContains:
|
|
return &listContains{col: s.Column, args: []interface{}{s.Text}}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type TextComparison int
|
|
|
|
const (
|
|
TextEquals TextComparison = iota
|
|
TextEqualsIgnoreCase
|
|
TextStartsWith
|
|
TextStartsWithIgnoreCase
|
|
TextEndsWith
|
|
TextEndsWithIgnoreCase
|
|
TextContains
|
|
TextContainsIgnoreCase
|
|
TextListContains
|
|
TextNotEquals
|
|
|
|
textCompareMax
|
|
)
|
|
|
|
//Deprecated: Use TextComparison, will be removed as soon as all calls are changed to query
|
|
func TextComparisonFromMethod(m domain.SearchMethod) TextComparison {
|
|
switch m {
|
|
case domain.SearchMethodEquals:
|
|
return TextEquals
|
|
case domain.SearchMethodEqualsIgnoreCase:
|
|
return TextEqualsIgnoreCase
|
|
case domain.SearchMethodStartsWith:
|
|
return TextStartsWith
|
|
case domain.SearchMethodStartsWithIgnoreCase:
|
|
return TextStartsWithIgnoreCase
|
|
case domain.SearchMethodContains:
|
|
return TextContains
|
|
case domain.SearchMethodContainsIgnoreCase:
|
|
return TextContainsIgnoreCase
|
|
case domain.SearchMethodEndsWith:
|
|
return TextEndsWith
|
|
case domain.SearchMethodEndsWithIgnoreCase:
|
|
return TextEndsWithIgnoreCase
|
|
case domain.SearchMethodListContains:
|
|
return TextListContains
|
|
default:
|
|
return textCompareMax
|
|
}
|
|
}
|
|
|
|
type NumberQuery struct {
|
|
Column Column
|
|
Number interface{}
|
|
Compare NumberComparison
|
|
}
|
|
|
|
func NewNumberQuery(c Column, value interface{}, compare NumberComparison) (*NumberQuery, error) {
|
|
if compare < 0 || compare >= numberCompareMax {
|
|
return nil, ErrInvalidCompare
|
|
}
|
|
if c.isZero() {
|
|
return nil, ErrMissingColumn
|
|
}
|
|
switch reflect.TypeOf(value).Kind() {
|
|
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Float32, reflect.Float64:
|
|
//everything fine
|
|
default:
|
|
return nil, ErrInvalidNumber
|
|
}
|
|
return &NumberQuery{
|
|
Column: c,
|
|
Number: value,
|
|
Compare: compare,
|
|
}, nil
|
|
}
|
|
|
|
func (q *NumberQuery) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (s *NumberQuery) comp() sq.Sqlizer {
|
|
switch s.Compare {
|
|
case NumberEquals:
|
|
return sq.Eq{s.Column.identifier(): s.Number}
|
|
case NumberNotEquals:
|
|
return sq.NotEq{s.Column.identifier(): s.Number}
|
|
case NumberLess:
|
|
return sq.Lt{s.Column.identifier(): s.Number}
|
|
case NumberGreater:
|
|
return sq.Gt{s.Column.identifier(): s.Number}
|
|
case NumberListContains:
|
|
return &listContains{col: s.Column, args: []interface{}{s.Number}}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type NumberComparison int
|
|
|
|
const (
|
|
NumberEquals NumberComparison = iota
|
|
NumberNotEquals
|
|
NumberLess
|
|
NumberGreater
|
|
NumberListContains
|
|
|
|
numberCompareMax
|
|
)
|
|
|
|
//Deprecated: Use NumberComparison, will be removed as soon as all calls are changed to query
|
|
func NumberComparisonFromMethod(m domain.SearchMethod) NumberComparison {
|
|
switch m {
|
|
case domain.SearchMethodEquals:
|
|
return NumberEquals
|
|
case domain.SearchMethodNotEquals:
|
|
return NumberNotEquals
|
|
case domain.SearchMethodGreaterThan:
|
|
return NumberGreater
|
|
case domain.SearchMethodLessThan:
|
|
return NumberLess
|
|
case domain.SearchMethodListContains:
|
|
return NumberListContains
|
|
default:
|
|
return numberCompareMax
|
|
}
|
|
}
|
|
|
|
type ListQuery struct {
|
|
Column Column
|
|
List []interface{}
|
|
Compare ListComparison
|
|
}
|
|
|
|
func NewListQuery(column Column, value []interface{}, compare ListComparison) (*ListQuery, error) {
|
|
if compare < 0 || compare >= listCompareMax {
|
|
return nil, ErrInvalidCompare
|
|
}
|
|
if column.isZero() {
|
|
return nil, ErrMissingColumn
|
|
}
|
|
return &ListQuery{
|
|
Column: column,
|
|
List: value,
|
|
Compare: compare,
|
|
}, nil
|
|
}
|
|
|
|
func (q *ListQuery) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (s *ListQuery) comp() sq.Sqlizer {
|
|
switch s.Compare {
|
|
case ListIn:
|
|
return sq.Eq{s.Column.identifier(): s.List}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type ListComparison int
|
|
|
|
const (
|
|
ListIn ListComparison = iota
|
|
|
|
listCompareMax
|
|
)
|
|
|
|
func ListComparisonFromMethod(m domain.SearchMethod) ListComparison {
|
|
switch m {
|
|
case domain.SearchMethodEquals:
|
|
return ListIn
|
|
default:
|
|
return listCompareMax
|
|
}
|
|
}
|
|
|
|
type or struct {
|
|
queries []SearchQuery
|
|
}
|
|
|
|
func Or(queries ...SearchQuery) *or {
|
|
return &or{
|
|
queries: queries,
|
|
}
|
|
}
|
|
|
|
func (q *or) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (q *or) comp() sq.Sqlizer {
|
|
queries := make([]sq.Sqlizer, 0)
|
|
for _, query := range q.queries {
|
|
queries = append(queries, query.comp())
|
|
}
|
|
return sq.Or(queries)
|
|
}
|
|
|
|
type BoolQuery struct {
|
|
Column Column
|
|
Value bool
|
|
}
|
|
|
|
func NewBoolQuery(c Column, value bool) (*BoolQuery, error) {
|
|
return &BoolQuery{
|
|
Column: c,
|
|
Value: value,
|
|
}, nil
|
|
}
|
|
|
|
func (q *BoolQuery) toQuery(query sq.SelectBuilder) sq.SelectBuilder {
|
|
return query.Where(q.comp())
|
|
}
|
|
|
|
func (s *BoolQuery) comp() sq.Sqlizer {
|
|
return sq.Eq{s.Column.identifier(): s.Value}
|
|
}
|
|
|
|
var (
|
|
//countColumn represents the default counter for search responses
|
|
countColumn = Column{
|
|
name: "COUNT(*) OVER ()",
|
|
}
|
|
//uniqueColumn shows if there are any results
|
|
uniqueColumn = Column{
|
|
name: "COUNT(*) = 0",
|
|
}
|
|
)
|
|
|
|
type table struct {
|
|
name string
|
|
alias string
|
|
}
|
|
|
|
func (t table) setAlias(a string) table {
|
|
t.alias = a
|
|
return t
|
|
}
|
|
|
|
func (t table) identifier() string {
|
|
if t.alias == "" {
|
|
return t.name
|
|
}
|
|
return t.name + " AS " + t.alias
|
|
}
|
|
|
|
func (t table) isZero() bool {
|
|
return t.name == ""
|
|
}
|
|
|
|
type Column struct {
|
|
name string
|
|
table table
|
|
isOrderByLower bool
|
|
}
|
|
|
|
func (c Column) identifier() string {
|
|
if c.table.alias != "" {
|
|
return c.table.alias + "." + c.name
|
|
}
|
|
if c.table.name != "" {
|
|
return c.table.name + "." + c.name
|
|
}
|
|
return c.name
|
|
}
|
|
|
|
func (c Column) orderBy() string {
|
|
if !c.isOrderByLower {
|
|
return c.identifier()
|
|
}
|
|
return "LOWER(" + c.identifier() + ")"
|
|
}
|
|
|
|
func (c Column) setTable(t table) Column {
|
|
c.table = t
|
|
return c
|
|
}
|
|
|
|
func (c Column) isZero() bool {
|
|
return c.table.isZero() || c.name == ""
|
|
}
|
|
|
|
func join(join, from Column) string {
|
|
return join.table.identifier() + " ON " + from.identifier() + " = " + join.identifier()
|
|
}
|
|
|
|
type listContains struct {
|
|
col Column
|
|
args interface{}
|
|
}
|
|
|
|
func (q *listContains) ToSql() (string, []interface{}, error) {
|
|
return q.col.identifier() + " @> ? ", []interface{}{q.args}, nil
|
|
}
|