jet/mysql/select_statement.go

191 lines
5 KiB
Go
Raw Normal View History

2019-08-11 12:13:59 +02:00
package mysql
import (
2020-06-27 18:48:19 +02:00
"github.com/go-jet/jet/v2/internal/jet"
)
2019-08-11 12:13:59 +02:00
2019-08-17 10:43:16 +02:00
// RowLock is interface for SELECT statement row lock types
type RowLock = jet.RowLock
2019-08-11 12:13:59 +02:00
2019-08-17 10:43:16 +02:00
// Row lock types
2019-08-11 12:13:59 +02:00
var (
2019-08-17 18:32:01 +02:00
UPDATE = jet.NewRowLock("UPDATE")
SHARE = jet.NewRowLock("SHARE")
2019-08-11 12:13:59 +02:00
)
// Window function clauses
var (
PARTITION_BY = jet.PARTITION_BY
ORDER_BY = jet.ORDER_BY
UNBOUNDED = jet.UNBOUNDED
CURRENT_ROW = jet.CURRENT_ROW
)
// PRECEDING window frame clause
func PRECEDING(offset interface{}) jet.FrameExtent {
return jet.PRECEDING(toJetFrameOffset(offset))
}
// FOLLOWING window frame clause
func FOLLOWING(offset interface{}) jet.FrameExtent {
return jet.FOLLOWING(toJetFrameOffset(offset))
}
// Window is used to specify window reference from WINDOW clause
var Window = jet.WindowName
2019-08-17 10:43:16 +02:00
// SelectStatement is interface for MySQL SELECT statement
2019-08-11 12:13:59 +02:00
type SelectStatement interface {
2019-08-15 13:54:05 +02:00
Statement
2019-08-11 12:13:59 +02:00
jet.HasProjections
2019-08-15 13:54:05 +02:00
Expression
2019-08-11 12:13:59 +02:00
DISTINCT() SelectStatement
2021-05-03 18:48:15 +02:00
FROM(tables ...ReadableTable) SelectStatement
2019-08-11 12:13:59 +02:00
WHERE(expression BoolExpression) SelectStatement
GROUP_BY(groupByClauses ...jet.GroupByClause) SelectStatement
HAVING(boolExpression BoolExpression) SelectStatement
WINDOW(name string) windowExpand
ORDER_BY(orderByClauses ...OrderByClause) SelectStatement
2019-08-11 12:13:59 +02:00
LIMIT(limit int64) SelectStatement
OFFSET(offset int64) SelectStatement
2019-08-17 10:43:16 +02:00
FOR(lock RowLock) SelectStatement
2019-08-15 11:59:17 +02:00
LOCK_IN_SHARE_MODE() SelectStatement
2019-08-11 12:13:59 +02:00
2019-08-17 10:43:16 +02:00
UNION(rhs SelectStatement) setStatement
UNION_ALL(rhs SelectStatement) setStatement
2019-08-11 12:13:59 +02:00
AsTable(alias string) SelectTable
}
//SELECT creates new SelectStatement with list of projections
2019-08-15 13:54:05 +02:00
func SELECT(projection Projection, projections ...Projection) SelectStatement {
return newSelectStatement(nil, append([]Projection{projection}, projections...))
2019-08-11 12:13:59 +02:00
}
2019-08-15 13:54:05 +02:00
func newSelectStatement(table ReadableTable, projections []Projection) SelectStatement {
2019-08-11 12:13:59 +02:00
newSelect := &selectStatementImpl{}
2019-08-17 18:32:01 +02:00
newSelect.ExpressionStatement = jet.NewExpressionStatementImpl(Dialect, jet.SelectStatementType, newSelect, &newSelect.Select,
&newSelect.From, &newSelect.Where, &newSelect.GroupBy, &newSelect.Having, &newSelect.Window, &newSelect.OrderBy,
2019-08-15 11:59:17 +02:00
&newSelect.Limit, &newSelect.Offset, &newSelect.For, &newSelect.ShareLock)
2019-08-11 12:13:59 +02:00
newSelect.Select.ProjectionList = projections
2021-05-03 18:48:15 +02:00
if table != nil {
newSelect.From.Tables = []jet.Serializer{table}
}
2019-08-11 12:13:59 +02:00
newSelect.Limit.Count = -1
newSelect.Offset.Count = -1
2019-08-15 11:59:17 +02:00
newSelect.ShareLock.Name = "LOCK IN SHARE MODE"
newSelect.ShareLock.InNewLine = true
2019-08-11 12:13:59 +02:00
newSelect.setOperatorsImpl.parent = newSelect
return newSelect
}
type selectStatementImpl struct {
2019-08-17 18:32:01 +02:00
jet.ExpressionStatement
2019-08-11 12:13:59 +02:00
setOperatorsImpl
2019-08-15 11:59:17 +02:00
Select jet.ClauseSelect
From jet.ClauseFrom
Where jet.ClauseWhere
GroupBy jet.ClauseGroupBy
Having jet.ClauseHaving
Window jet.ClauseWindow
2019-08-15 11:59:17 +02:00
OrderBy jet.ClauseOrderBy
Limit jet.ClauseLimit
Offset jet.ClauseOffset
For jet.ClauseFor
ShareLock jet.ClauseOptional
2019-08-11 12:13:59 +02:00
}
func (s *selectStatementImpl) DISTINCT() SelectStatement {
s.Select.Distinct = true
return s
}
2021-05-03 18:48:15 +02:00
func (s *selectStatementImpl) FROM(tables ...ReadableTable) SelectStatement {
for _, table := range tables {
s.From.Tables = append(s.From.Tables, table)
}
2019-08-11 12:13:59 +02:00
return s
}
func (s *selectStatementImpl) WHERE(condition BoolExpression) SelectStatement {
s.Where.Condition = condition
return s
}
func (s *selectStatementImpl) GROUP_BY(groupByClauses ...jet.GroupByClause) SelectStatement {
s.GroupBy.List = groupByClauses
return s
}
func (s *selectStatementImpl) HAVING(boolExpression BoolExpression) SelectStatement {
s.Having.Condition = boolExpression
return s
}
func (s *selectStatementImpl) WINDOW(name string) windowExpand {
s.Window.Definitions = append(s.Window.Definitions, jet.WindowDefinition{Name: name})
return windowExpand{selectStatement: s}
}
func (s *selectStatementImpl) ORDER_BY(orderByClauses ...OrderByClause) SelectStatement {
2019-08-11 12:13:59 +02:00
s.OrderBy.List = orderByClauses
return s
}
func (s *selectStatementImpl) LIMIT(limit int64) SelectStatement {
s.Limit.Count = limit
return s
}
func (s *selectStatementImpl) OFFSET(offset int64) SelectStatement {
s.Offset.Count = offset
return s
}
2019-08-17 10:43:16 +02:00
func (s *selectStatementImpl) FOR(lock RowLock) SelectStatement {
2019-08-11 12:13:59 +02:00
s.For.Lock = lock
return s
}
2019-08-15 11:59:17 +02:00
func (s *selectStatementImpl) LOCK_IN_SHARE_MODE() SelectStatement {
s.ShareLock.Show = true
return s
}
2019-08-11 12:13:59 +02:00
func (s *selectStatementImpl) AsTable(alias string) SelectTable {
return newSelectTable(s, alias)
}
//-----------------------------------------------------
type windowExpand struct {
selectStatement *selectStatementImpl
}
func (w windowExpand) AS(window ...jet.Window) SelectStatement {
if len(window) == 0 {
return w.selectStatement
}
windowsDefinition := w.selectStatement.Window.Definitions
windowsDefinition[len(windowsDefinition)-1].Window = window[0]
return w.selectStatement
}
func toJetFrameOffset(offset interface{}) jet.Serializer {
if offset == UNBOUNDED {
return jet.UNBOUNDED
}
// check for interval expression
//if exp, ok := offset.(Expression); ok {
// return exp
//}
return jet.FixedLiteral(offset)
}