2013-12-13 11:39:26 +01:00
|
|
|
|
|
|
|
This is the pretty printer code which takes AST values and turns them
|
|
|
|
back into SQL source text. It attempts to format the output nicely.
|
|
|
|
|
2013-12-13 15:04:48 +01:00
|
|
|
> module Language.SQL.SimpleSQL.Pretty
|
|
|
|
> (prettyQueryExpr
|
|
|
|
> ,prettyScalarExpr
|
|
|
|
> ) where
|
2013-12-13 11:39:26 +01:00
|
|
|
|
2013-12-13 15:04:48 +01:00
|
|
|
> import Language.SQL.SimpleSQL.Syntax
|
2013-12-13 14:05:32 +01:00
|
|
|
> import Text.PrettyPrint
|
|
|
|
> import Data.Maybe
|
2013-12-13 11:39:26 +01:00
|
|
|
|
2013-12-13 14:05:32 +01:00
|
|
|
> prettyQueryExpr :: QueryExpr -> String
|
|
|
|
> prettyQueryExpr = render . queryExpr
|
2013-12-13 11:39:26 +01:00
|
|
|
|
|
|
|
> prettyScalarExpr :: ScalarExpr -> String
|
2013-12-13 14:05:32 +01:00
|
|
|
> prettyScalarExpr = render . scalarExpr
|
|
|
|
|
|
|
|
|
|
|
|
= scalar expressions
|
|
|
|
|
|
|
|
> scalarExpr :: ScalarExpr -> Doc
|
2013-12-13 16:00:22 +01:00
|
|
|
> scalarExpr (StringLit s) = quotes $ text s
|
|
|
|
> scalarExpr (NumLit s) = text s
|
2013-12-13 23:07:45 +01:00
|
|
|
> scalarExpr (IntervalLit v u p) =
|
|
|
|
> text "interval" <+> quotes (text v)
|
|
|
|
> <+> text u
|
|
|
|
> <+> maybe empty (parens . text . show ) p
|
2013-12-13 16:00:22 +01:00
|
|
|
> scalarExpr (Iden i) = text i
|
|
|
|
> scalarExpr (Iden2 q i) = text q <> text "." <> text i
|
2013-12-13 14:05:32 +01:00
|
|
|
> scalarExpr Star = text "*"
|
|
|
|
> scalarExpr (Star2 q) = text q <> text "." <> text "*"
|
|
|
|
|
|
|
|
> scalarExpr (App f es) = text f <> parens (commaSep (map scalarExpr es))
|
2013-12-13 20:13:36 +01:00
|
|
|
|
2013-12-13 22:18:30 +01:00
|
|
|
> scalarExpr (AggregateApp f d es od) =
|
|
|
|
> text f
|
|
|
|
> <> parens ((case d of
|
|
|
|
> Just Distinct -> text "distinct"
|
|
|
|
> Just All -> text "all"
|
|
|
|
> Nothing -> empty)
|
|
|
|
> <+> commaSep (map scalarExpr es)
|
|
|
|
> <+> orderBy od)
|
|
|
|
|
2013-12-13 22:31:36 +01:00
|
|
|
> scalarExpr (WindowApp f es pb od) =
|
|
|
|
> text f <> parens (commaSep $ map scalarExpr es)
|
|
|
|
> <+> text "over"
|
|
|
|
> <+> parens ((case pb of
|
|
|
|
> [] -> empty
|
|
|
|
> _ -> text "partition by"
|
|
|
|
> <+> nest 4 (commaSep $ map scalarExpr pb))
|
|
|
|
> <+> orderBy od)
|
|
|
|
|
2013-12-13 20:26:14 +01:00
|
|
|
> scalarExpr (SpecialOp nm [a,b,c]) | nm `elem` ["between", "not between"] =
|
2013-12-13 23:34:05 +01:00
|
|
|
> scalarExpr a <+> text nm <+> scalarExpr b <+> text "and" <+> scalarExpr c
|
2013-12-13 20:13:36 +01:00
|
|
|
|
2013-12-13 21:38:43 +01:00
|
|
|
> scalarExpr (SpecialOp "extract" [a,n]) =
|
|
|
|
> text "extract" <> parens (scalarExpr a
|
|
|
|
> <+> text "from"
|
|
|
|
> <+> scalarExpr n)
|
|
|
|
|
2013-12-13 23:34:05 +01:00
|
|
|
> scalarExpr (SpecialOp "substring" [a,s,e]) =
|
|
|
|
> text "substring" <> parens (scalarExpr a
|
|
|
|
> <+> text "from"
|
|
|
|
> <+> scalarExpr s
|
|
|
|
> <+> text "for"
|
|
|
|
> <+> scalarExpr e)
|
|
|
|
|
2013-12-13 20:26:14 +01:00
|
|
|
> scalarExpr (SpecialOp nm es) =
|
|
|
|
> text nm <+> parens (commaSep $ map scalarExpr es)
|
2013-12-13 20:13:36 +01:00
|
|
|
|
2013-12-13 20:26:14 +01:00
|
|
|
> scalarExpr (PrefixOp f e) = text f <+> scalarExpr e
|
|
|
|
> scalarExpr (PostfixOp f e) = scalarExpr e <+> text f
|
2013-12-13 23:34:05 +01:00
|
|
|
> scalarExpr (BinOp "and" e0 e1) =
|
|
|
|
> sep [scalarExpr e0, text "and" <+> scalarExpr e1]
|
2013-12-13 20:26:14 +01:00
|
|
|
> scalarExpr (BinOp f e0 e1) =
|
2013-12-13 23:34:05 +01:00
|
|
|
> scalarExpr e0 <+> text f <+> scalarExpr e1
|
2013-12-13 14:05:32 +01:00
|
|
|
|
|
|
|
> scalarExpr (Case t ws els) =
|
2013-12-13 21:25:22 +01:00
|
|
|
> sep [text "case" <+> maybe empty scalarExpr t
|
|
|
|
> ,nest 4 (sep (map w ws
|
2013-12-13 14:05:32 +01:00
|
|
|
> ++ maybeToList (fmap e els)))
|
|
|
|
> ,text "end"]
|
|
|
|
> where
|
|
|
|
> w (t0,t1) = sep [text "when" <+> scalarExpr t0
|
|
|
|
> ,text "then" <+> scalarExpr t1]
|
|
|
|
> e el = text "else" <+> scalarExpr el
|
|
|
|
> scalarExpr (Parens e) = parens $ scalarExpr e
|
2013-12-13 17:50:41 +01:00
|
|
|
> scalarExpr (Cast e (TypeName tn)) =
|
|
|
|
> text "cast" <> parens (sep [scalarExpr e
|
|
|
|
> ,text "as"
|
|
|
|
> ,text tn])
|
|
|
|
|
2013-12-13 19:24:20 +01:00
|
|
|
> scalarExpr (CastOp (TypeName tn) s) =
|
2013-12-13 17:50:41 +01:00
|
|
|
> text tn <+> quotes (text s)
|
2013-12-13 14:05:32 +01:00
|
|
|
|
2013-12-13 19:43:28 +01:00
|
|
|
> scalarExpr (SubQueryExpr ty qe) =
|
|
|
|
> (case ty of
|
|
|
|
> SqSq -> empty
|
|
|
|
> SqExists -> text "exists"
|
|
|
|
> SqAll -> text "all"
|
|
|
|
> SqSome -> text "some"
|
|
|
|
> SqAny -> text "any"
|
|
|
|
> ) <+> parens (queryExpr qe)
|
|
|
|
|
2013-12-13 20:00:06 +01:00
|
|
|
> scalarExpr (In b se x) =
|
|
|
|
> sep [scalarExpr se
|
|
|
|
> ,if b then empty else text "not"
|
|
|
|
> ,text "in"
|
|
|
|
> ,parens (nest 4 $
|
|
|
|
> case x of
|
|
|
|
> InList es -> commaSep $ map scalarExpr es
|
|
|
|
> InQueryExpr qe -> queryExpr qe)]
|
|
|
|
|
2013-12-13 14:05:32 +01:00
|
|
|
= query expressions
|
|
|
|
|
|
|
|
> queryExpr :: QueryExpr -> Doc
|
2013-12-13 16:27:02 +01:00
|
|
|
> queryExpr (Select d sl fr wh gb hv od lm off) =
|
2013-12-13 14:05:32 +01:00
|
|
|
> sep [text "select"
|
2013-12-13 16:27:02 +01:00
|
|
|
> ,case d of
|
|
|
|
> All -> empty
|
|
|
|
> Distinct -> text "distinct"
|
2013-12-13 14:05:32 +01:00
|
|
|
> ,nest 4 $ sep [selectList sl]
|
|
|
|
> ,from fr
|
2013-12-13 16:27:02 +01:00
|
|
|
> ,maybeScalarExpr "where" wh
|
2013-12-13 14:05:32 +01:00
|
|
|
> ,grpBy gb
|
2013-12-13 16:27:02 +01:00
|
|
|
> ,maybeScalarExpr "having" hv
|
|
|
|
> ,orderBy od
|
|
|
|
> ,maybeScalarExpr "limit" lm
|
|
|
|
> ,maybeScalarExpr "offset" off
|
|
|
|
> ]
|
2013-12-13 22:49:22 +01:00
|
|
|
> queryExpr (CombineQueryExpr q1 ct d c q2) =
|
2013-12-13 22:41:12 +01:00
|
|
|
> sep [queryExpr q1
|
2013-12-13 22:49:22 +01:00
|
|
|
> ,text (case ct of
|
|
|
|
> Union -> "union"
|
|
|
|
> Intersect -> "intersect"
|
|
|
|
> Except -> "except")
|
|
|
|
> <+> case d of
|
|
|
|
> All -> empty
|
|
|
|
> Distinct -> text "distinct"
|
|
|
|
> <+> case c of
|
|
|
|
> Corresponding -> text "corresponding"
|
|
|
|
> Respectively -> empty
|
2013-12-13 22:41:12 +01:00
|
|
|
> ,queryExpr q2]
|
2013-12-13 14:05:32 +01:00
|
|
|
|
|
|
|
> selectList :: [(Maybe String, ScalarExpr)] -> Doc
|
|
|
|
> selectList is = commaSep $ map si is
|
|
|
|
> where
|
|
|
|
> si (al,e) = scalarExpr e <+> maybe empty alias al
|
|
|
|
> alias al = text "as" <+> text al
|
|
|
|
|
|
|
|
> from :: [TableRef] -> Doc
|
|
|
|
> from [] = empty
|
|
|
|
> from ts =
|
|
|
|
> sep [text "from"
|
|
|
|
> ,nest 4 $ commaSep $ map tr ts]
|
|
|
|
> where
|
|
|
|
> tr (SimpleTableRef t) = text t
|
2013-12-13 23:37:34 +01:00
|
|
|
> tr (JoinAlias t a cs) =
|
|
|
|
> tr t <+> text "as" <+> text a
|
|
|
|
> <+> maybe empty (\cs' -> parens $ commaSep $ map text cs') cs
|
2013-12-13 14:05:32 +01:00
|
|
|
> tr (JoinParens t) = parens $ tr t
|
|
|
|
> tr (JoinQueryExpr q) = parens $ queryExpr q
|
|
|
|
> tr (JoinTableRef jt t0 t1 jc) =
|
|
|
|
> sep [tr t0
|
|
|
|
> ,joinText jt jc
|
|
|
|
> ,tr t1
|
|
|
|
> ,joinCond jc]
|
|
|
|
> joinText jt jc =
|
|
|
|
> sep [case jc of
|
|
|
|
> Just JoinNatural -> text "natural"
|
|
|
|
> _ -> empty
|
|
|
|
> ,case jt of
|
|
|
|
> Inner -> text "inner"
|
|
|
|
> JLeft -> text "left"
|
|
|
|
> JRight -> text "right"
|
|
|
|
> Full -> text "full"
|
|
|
|
> Cross -> text "cross"
|
|
|
|
> ,text "join"]
|
|
|
|
> joinCond (Just (JoinOn e)) = text "on" <+> scalarExpr e
|
|
|
|
> joinCond (Just (JoinUsing es)) = text "using" <+> parens (commaSep $ map text es)
|
|
|
|
> joinCond Nothing = empty
|
|
|
|
> joinCond (Just JoinNatural) = empty
|
|
|
|
|
2013-12-13 16:27:02 +01:00
|
|
|
> maybeScalarExpr :: String -> Maybe ScalarExpr -> Doc
|
|
|
|
> maybeScalarExpr k = maybe empty
|
|
|
|
> (\e -> sep [text k
|
|
|
|
> ,nest 4 $ scalarExpr e])
|
2013-12-13 14:05:32 +01:00
|
|
|
|
|
|
|
> grpBy :: [ScalarExpr] -> Doc
|
|
|
|
> grpBy [] = empty
|
|
|
|
> grpBy gs = sep [text "group by"
|
|
|
|
> ,nest 4 $ commaSep $ map scalarExpr gs]
|
|
|
|
|
2013-12-13 16:08:10 +01:00
|
|
|
> orderBy :: [(ScalarExpr,Direction)] -> Doc
|
2013-12-13 14:05:32 +01:00
|
|
|
> orderBy [] = empty
|
|
|
|
> orderBy os = sep [text "order by"
|
2013-12-13 16:08:10 +01:00
|
|
|
> ,nest 4 $ commaSep $ map f os]
|
|
|
|
> where
|
|
|
|
> f (e,Asc) = scalarExpr e
|
|
|
|
> f (e,Desc) = scalarExpr e <+> text "desc"
|
2013-12-13 14:05:32 +01:00
|
|
|
|
|
|
|
|
|
|
|
= utils
|
|
|
|
|
|
|
|
> commaSep :: [Doc] -> Doc
|
|
|
|
> commaSep ds = sep $ punctuate comma ds
|