2021-11-07 20:44:48 +01:00
|
|
|
// Copyright 2018 The Xorm Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package builder
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (b *Builder) limitWriteTo(w Writer) error {
|
|
|
|
if strings.TrimSpace(b.dialect) == "" {
|
|
|
|
return ErrDialectNotSetUp
|
|
|
|
}
|
|
|
|
|
|
|
|
if b.limitation != nil {
|
|
|
|
limit := b.limitation
|
|
|
|
if limit.offset < 0 || limit.limitN <= 0 {
|
|
|
|
return ErrInvalidLimitation
|
|
|
|
}
|
2023-09-10 20:41:55 +02:00
|
|
|
// unset limit condition to prevent final.WriteTo from recursing forever
|
2021-11-07 20:44:48 +01:00
|
|
|
b.limitation = nil
|
|
|
|
defer func() {
|
|
|
|
b.limitation = limit
|
|
|
|
}()
|
|
|
|
|
|
|
|
switch strings.ToLower(strings.TrimSpace(b.dialect)) {
|
|
|
|
case ORACLE:
|
|
|
|
if len(b.selects) == 0 {
|
|
|
|
b.selects = append(b.selects, "*")
|
|
|
|
}
|
|
|
|
|
|
|
|
var final *Builder
|
|
|
|
selects := b.selects
|
|
|
|
b.selects = append(selects, "ROWNUM RN")
|
|
|
|
|
|
|
|
var wb *Builder
|
|
|
|
if b.optype == setOpType {
|
|
|
|
wb = Dialect(b.dialect).Select("at.*", "ROWNUM RN").
|
|
|
|
From(b, "at")
|
|
|
|
} else {
|
|
|
|
wb = b
|
|
|
|
}
|
|
|
|
|
|
|
|
if limit.offset == 0 {
|
|
|
|
final = Dialect(b.dialect).Select(selects...).From(wb, "at").
|
|
|
|
Where(Lte{"at.RN": limit.limitN})
|
|
|
|
} else {
|
|
|
|
sub := Dialect(b.dialect).Select("*").
|
|
|
|
From(b, "at").Where(Lte{"at.RN": limit.offset + limit.limitN})
|
|
|
|
|
|
|
|
final = Dialect(b.dialect).Select(selects...).From(sub, "att").
|
|
|
|
Where(Gt{"att.RN": limit.offset})
|
|
|
|
}
|
|
|
|
|
2023-09-10 20:41:55 +02:00
|
|
|
return final.WriteTo(w)
|
2021-11-07 20:44:48 +01:00
|
|
|
case SQLITE, MYSQL, POSTGRES:
|
|
|
|
// if type UNION, we need to write previous content back to current writer
|
|
|
|
if b.optype == setOpType {
|
2023-09-10 20:41:55 +02:00
|
|
|
if err := b.WriteTo(w); err != nil {
|
2021-11-07 20:44:48 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if limit.offset == 0 {
|
2023-09-10 20:41:55 +02:00
|
|
|
fmt.Fprint(w, " LIMIT ", limit.limitN)
|
2021-11-07 20:44:48 +01:00
|
|
|
} else {
|
2023-09-10 20:41:55 +02:00
|
|
|
fmt.Fprintf(w, " LIMIT %v OFFSET %v", limit.limitN, limit.offset)
|
2021-11-07 20:44:48 +01:00
|
|
|
}
|
|
|
|
case MSSQL:
|
|
|
|
if len(b.selects) == 0 {
|
|
|
|
b.selects = append(b.selects, "*")
|
|
|
|
}
|
|
|
|
|
|
|
|
var final *Builder
|
|
|
|
selects := b.selects
|
|
|
|
b.selects = append(append([]string{fmt.Sprintf("TOP %d %v", limit.limitN+limit.offset, b.selects[0])},
|
|
|
|
b.selects[1:]...), "ROW_NUMBER() OVER (ORDER BY (SELECT 1)) AS RN")
|
|
|
|
|
|
|
|
var wb *Builder
|
|
|
|
if b.optype == setOpType {
|
|
|
|
wb = Dialect(b.dialect).Select("*", "ROW_NUMBER() OVER (ORDER BY (SELECT 1)) AS RN").
|
|
|
|
From(b, "at")
|
|
|
|
} else {
|
|
|
|
wb = b
|
|
|
|
}
|
|
|
|
|
|
|
|
if limit.offset == 0 {
|
|
|
|
final = Dialect(b.dialect).Select(selects...).From(wb, "at")
|
|
|
|
} else {
|
|
|
|
final = Dialect(b.dialect).Select(selects...).From(wb, "at").Where(Gt{"at.RN": limit.offset})
|
|
|
|
}
|
|
|
|
|
2023-09-10 20:41:55 +02:00
|
|
|
return final.WriteTo(w)
|
2021-11-07 20:44:48 +01:00
|
|
|
default:
|
|
|
|
return ErrNotSupportType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|