package sqlbuilder import ( "github.com/dropbox/godropbox/errors" ) // An expression type expression interface { clause projection groupByClause orderByClause IS_NULL() boolExpression IS_NOT_NULL() boolExpression IS_DISTINCT_FROM(expression expression) boolExpression IS_NOT_DISTINCT_FROM(expression expression) boolExpression IN(subQuery selectStatement) boolExpression NOT_IN(subQuery selectStatement) boolExpression AS(alias string) projection ASC() orderByClause DESC() orderByClause } type expressionInterfaceImpl struct { parent expression } func (e *expressionInterfaceImpl) IS_NULL() boolExpression { return newPostifxBoolExpression(e.parent, "IS NULL") } func (e *expressionInterfaceImpl) IS_NOT_NULL() boolExpression { return newPostifxBoolExpression(e.parent, "IS NOT NULL") } func (e *expressionInterfaceImpl) IS_DISTINCT_FROM(expression expression) boolExpression { return newBinaryBoolExpression(e.parent, expression, "IS DISTINCT FROM") } func (e *expressionInterfaceImpl) IS_NOT_DISTINCT_FROM(expression expression) boolExpression { return newBinaryBoolExpression(e.parent, expression, "IS NOT DISTINCT FROM") } func (e *expressionInterfaceImpl) IN(subQuery selectStatement) boolExpression { return newBinaryBoolExpression(e.parent, subQuery, "IN") } func (e *expressionInterfaceImpl) NOT_IN(subQuery selectStatement) boolExpression { return newBinaryBoolExpression(e.parent, subQuery, "NOT IN") } func (e *expressionInterfaceImpl) AS(alias string) projection { return NewAlias(e.parent, alias) } func (e *expressionInterfaceImpl) ASC() orderByClause { return &orderByClauseImpl{expression: e.parent, ascent: true} } func (e *expressionInterfaceImpl) DESC() orderByClause { return &orderByClauseImpl{expression: e.parent, ascent: false} } func (e *expressionInterfaceImpl) serializeForGroupBy(statement statementType, out *queryData) error { return e.parent.serialize(statement, out) } func (e *expressionInterfaceImpl) serializeForProjection(statement statementType, out *queryData) error { return e.parent.serialize(statement, out) } func (e *expressionInterfaceImpl) serializeAsOrderBy(statement statementType, out *queryData) error { return e.parent.serialize(statement, out) } // Representation of binary operations (e.g. comparisons, arithmetic) type binaryOpExpression struct { lhs, rhs expression operator string } func newBinaryExpression(lhs, rhs expression, operator string, parent ...expression) binaryOpExpression { binaryExpression := binaryOpExpression{ lhs: lhs, rhs: rhs, operator: operator, } return binaryExpression } func isSimpleOperand(expression expression) bool { if _, ok := expression.(*literalExpression); ok { return true } if _, ok := expression.(column); ok { return true } if _, ok := expression.(*numericFunc); ok { return true } return false } func (c *binaryOpExpression) serialize(statement statementType, out *queryData) error { if c == nil { return errors.New("Binary expression is nil.") } if c.lhs == nil { return errors.Newf("nil lhs.") } if c.rhs == nil { return errors.Newf("nil rhs.") } wrap := !isSimpleOperand(c.lhs) && !isSimpleOperand(c.rhs) if wrap { out.writeString("(") } if err := c.lhs.serialize(statement, out); err != nil { return err } out.writeString(" " + c.operator + " ") if err := c.rhs.serialize(statement, out); err != nil { return err } if wrap { out.writeString(")") } return nil } // A prefix operator expression type prefixOpExpression struct { expression expression operator string } func newPrefixExpression(expression expression, operator string) prefixOpExpression { prefixExpression := prefixOpExpression{ expression: expression, operator: operator, } return prefixExpression } func (p *prefixOpExpression) serialize(statement statementType, out *queryData) error { if p == nil { return errors.New("Prefix expression is nil.") } out.writeString(p.operator + " ") if p.expression == nil { return errors.Newf("nil prefix expression.") } if err := p.expression.serialize(statement, out); err != nil { return err } return nil } // A postifx operator expression type postfixOpExpression struct { expression expression operator string } func newPostfixOpExpression(expression expression, operator string) postfixOpExpression { postfixOpExpression := postfixOpExpression{ expression: expression, operator: operator, } return postfixOpExpression } func (p *postfixOpExpression) serialize(statement statementType, out *queryData) error { if p == nil { return errors.New("Postifx operator expression is nil.") } if p.expression == nil { return errors.Newf("nil prefix expression.") } if err := p.expression.serialize(statement, out); err != nil { return err } out.writeString(p.operator) return nil }