mirror of
https://github.com/neilotoole/sq.git
synced 2024-12-19 14:11:45 +03:00
211 lines
5.6 KiB
Go
211 lines
5.6 KiB
Go
|
// Package render provides the mechanism for rendering ast into SQL.
|
||
|
package render
|
||
|
|
||
|
import (
|
||
|
"strings"
|
||
|
|
||
|
"github.com/neilotoole/sq/libsq/driver/dialect"
|
||
|
|
||
|
"github.com/neilotoole/sq/libsq/core/errz"
|
||
|
|
||
|
"github.com/neilotoole/sq/libsq/ast"
|
||
|
)
|
||
|
|
||
|
// Context contains context for rendering a query.
|
||
|
type Context struct {
|
||
|
// Renderer holds the rendering functions.
|
||
|
Renderer *Renderer
|
||
|
|
||
|
// Dialect is the driver dialect.
|
||
|
Dialect dialect.Dialect
|
||
|
|
||
|
// The args map contains predefined variables that are
|
||
|
// substituted into the query. It may be empty or nil.
|
||
|
Args map[string]string
|
||
|
}
|
||
|
|
||
|
// Renderer is a set of functions for rendering ast elements into SQL.
|
||
|
// Use NewDefaultRenderer to get a new instance. Each function can be
|
||
|
// swapped with a custom implementation for a SQL dialect.
|
||
|
type Renderer struct {
|
||
|
// FromTable renders a FROM table fragment.
|
||
|
FromTable func(rc *Context, tblSel *ast.TblSelectorNode) (string, error)
|
||
|
|
||
|
// SelectCols renders a column names/expression fragment.
|
||
|
// It shouldn't render the actual SELECT keyword. Example return value:
|
||
|
//
|
||
|
// "first_name" AS "given_name", "last name" AS "family_name"
|
||
|
SelectCols func(rc *Context, cols []ast.ResultColumn) (string, error)
|
||
|
|
||
|
// Range renders a row range fragment.
|
||
|
Range func(rc *Context, rr *ast.RowRangeNode) (string, error)
|
||
|
|
||
|
// OrderBy renders the ORDER BY fragment.
|
||
|
OrderBy func(rc *Context, ob *ast.OrderByNode) (string, error)
|
||
|
|
||
|
// GroupBy renders the GROUP BY fragment.
|
||
|
GroupBy func(rc *Context, gb *ast.GroupByNode) (string, error)
|
||
|
|
||
|
// Join renders a join fragment.
|
||
|
Join func(rc *Context, fnJoin *ast.JoinNode) (string, error)
|
||
|
|
||
|
// Function renders a function fragment.
|
||
|
Function func(rc *Context, fn *ast.FuncNode) (string, error)
|
||
|
|
||
|
// Literal renders a literal fragment.
|
||
|
Literal func(rc *Context, lit *ast.LiteralNode) (string, error)
|
||
|
|
||
|
// Where renders a WHERE fragment.
|
||
|
Where func(rc *Context, where *ast.WhereNode) (string, error)
|
||
|
|
||
|
// Expr renders an expression fragment.
|
||
|
Expr func(rc *Context, expr *ast.ExprNode) (string, error)
|
||
|
|
||
|
// Operator renders an operator fragment.
|
||
|
Operator func(rc *Context, op *ast.OperatorNode) (string, error)
|
||
|
|
||
|
// Distinct renders the DISTINCT fragment. Returns an
|
||
|
// empty string if n is nil.
|
||
|
Distinct func(rc *Context, n *ast.UniqueNode) (string, error)
|
||
|
|
||
|
// PreRender is a hook that is called before Render. It is a final
|
||
|
// opportunity to customize f before rendering. It is nil by default.
|
||
|
PreRender func(rc *Context, f *Fragments) error
|
||
|
|
||
|
// Render renders f into a SQL query.
|
||
|
Render func(rc *Context, f *Fragments) (string, error)
|
||
|
}
|
||
|
|
||
|
// NewDefaultRenderer returns a Renderer that works for most SQL dialects.
|
||
|
// Driver implementations can override specific rendering functions
|
||
|
// as needed.
|
||
|
func NewDefaultRenderer() *Renderer {
|
||
|
return &Renderer{
|
||
|
FromTable: doFromTable,
|
||
|
SelectCols: doSelectCols,
|
||
|
Range: doRange,
|
||
|
OrderBy: doOrderBy,
|
||
|
GroupBy: doGroupBy,
|
||
|
Join: doJoin,
|
||
|
Function: doFunction,
|
||
|
Literal: doLiteral,
|
||
|
Where: doWhere,
|
||
|
Expr: doExpr,
|
||
|
Operator: doOperator,
|
||
|
Distinct: doDistinct,
|
||
|
Render: doRender,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Fragments holds the fragments of a SQL query.
|
||
|
// It is passed to Renderer.PreRender and Renderer.Render.
|
||
|
type Fragments struct {
|
||
|
Distinct string
|
||
|
Columns string
|
||
|
From string
|
||
|
Where string
|
||
|
GroupBy string
|
||
|
OrderBy string
|
||
|
Range string
|
||
|
}
|
||
|
|
||
|
// Render implements QueryBuilder.
|
||
|
func doRender(_ *Context, f *Fragments) (string, error) {
|
||
|
sb := strings.Builder{}
|
||
|
|
||
|
sb.WriteString("SELECT")
|
||
|
|
||
|
if f.Distinct != "" {
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.Distinct)
|
||
|
}
|
||
|
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.Columns)
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.From)
|
||
|
|
||
|
if f.Where != "" {
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.Where)
|
||
|
}
|
||
|
|
||
|
if f.OrderBy != "" {
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.OrderBy)
|
||
|
}
|
||
|
|
||
|
if f.GroupBy != "" {
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.GroupBy)
|
||
|
}
|
||
|
|
||
|
if f.Range != "" {
|
||
|
sb.WriteRune(sp)
|
||
|
sb.WriteString(f.Range)
|
||
|
}
|
||
|
|
||
|
return sb.String(), nil
|
||
|
}
|
||
|
|
||
|
const (
|
||
|
singleQuote = '\''
|
||
|
sp = ' '
|
||
|
)
|
||
|
|
||
|
// renderSelectorNode renders a selector such as ".actor.first_name"
|
||
|
// or ".last_name".
|
||
|
func renderSelectorNode(d dialect.Dialect, node ast.Node) (string, error) {
|
||
|
// FIXME: switch to using enquote
|
||
|
switch node := node.(type) {
|
||
|
case *ast.ColSelectorNode:
|
||
|
return d.Enquote(node.ColName()), nil
|
||
|
case *ast.TblColSelectorNode:
|
||
|
return d.Enquote(node.TblName()) + "." + d.Enquote(node.ColName()), nil
|
||
|
case *ast.TblSelectorNode:
|
||
|
return d.Enquote(node.TblName()), nil
|
||
|
default:
|
||
|
return "", errz.Errorf(
|
||
|
"expected selector node type, but got %T: %s",
|
||
|
node,
|
||
|
node.Text(),
|
||
|
)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// sqlAppend is a convenience function for building the SQL string.
|
||
|
// The main purpose is to ensure that there's always a consistent amount
|
||
|
// of whitespace. Thus, if existing has a space suffix and add has a
|
||
|
// space prefix, the returned string will only have one space. If add
|
||
|
// is the empty string or just whitespace, this function simply
|
||
|
// returns existing.
|
||
|
func sqlAppend(existing, add string) string {
|
||
|
add = strings.TrimSpace(add)
|
||
|
if add == "" {
|
||
|
return existing
|
||
|
}
|
||
|
|
||
|
existing = strings.TrimSpace(existing)
|
||
|
return existing + " " + add
|
||
|
}
|
||
|
|
||
|
// unquoteLiteral returns true if s is a double-quoted string, and also returns
|
||
|
// the value with the quotes stripped. An error is returned if the string
|
||
|
// is malformed.
|
||
|
func unquoteLiteral(s string) (val string, ok bool, err error) {
|
||
|
hasPrefix := strings.HasPrefix(s, `"`)
|
||
|
hasSuffix := strings.HasSuffix(s, `"`)
|
||
|
|
||
|
if hasPrefix && hasSuffix {
|
||
|
val = strings.TrimPrefix(s, `"`)
|
||
|
val = strings.TrimSuffix(val, `"`)
|
||
|
return val, true, nil
|
||
|
}
|
||
|
|
||
|
if hasPrefix != hasSuffix {
|
||
|
return "", false, errz.Errorf("malformed literal: %s", s)
|
||
|
}
|
||
|
|
||
|
return s, false, nil
|
||
|
}
|