> -- | These is the pretty printing functions, which produce SQL > -- source from ASTs. The code attempts to format the output in a > -- readable way. > module Language.SQL.SimpleSQL.Pretty > (prettyQueryExpr > ,prettyValueExpr > ,prettyStatement > ,prettyStatements > ) where TODO: there should be more comments in this file, especially the bits which have been changed to try to improve the layout of the output. > import Language.SQL.SimpleSQL.Syntax > import Language.SQL.SimpleSQL.Dialect > import Text.PrettyPrint (render, vcat, text, (<>), (<+>), empty, parens, > nest, Doc, punctuate, comma, sep, quotes, > brackets,hcat) > import Data.Maybe (maybeToList, catMaybes) > import Data.List (intercalate) > -- | Convert a query expr ast to concrete syntax. > prettyQueryExpr :: Dialect -> QueryExpr -> String > prettyQueryExpr d = render . queryExpr d > -- | Convert a value expr ast to concrete syntax. > prettyValueExpr :: Dialect -> ValueExpr -> String > prettyValueExpr d = render . valueExpr d > -- | Convert a statement ast to concrete syntax. > prettyStatement :: Dialect -> Statement -> String > prettyStatement d = render . statement d > -- | Convert a list of statements to concrete syntax. A semi colon > -- is inserted after each statement. > prettyStatements :: Dialect -> [Statement] -> String > prettyStatements d = render . vcat . map ((<> text ";\n") . statement d) = value expressions > valueExpr :: Dialect -> ValueExpr -> Doc > valueExpr _ (StringLit s e t) = > text (s ++ (if '\'' `elem` s then doubleUpQuotes t else t) ++ e) > valueExpr _ (NumLit s) = text s > valueExpr _ (IntervalLit s v f t) = > text "interval" > <+> me (\x -> if x then text "+" else text "-") s > <+> quotes (text v) > <+> intervalTypeField f > <+> me (\x -> text "to" <+> intervalTypeField x) t > valueExpr _ (Iden i) = names i > valueExpr _ Star = text "*" > valueExpr _ Parameter = text "?" > valueExpr _ (HostParameter p i) = > text (':':p) > <+> me (\i' -> text "indicator" <+> text (':':i')) i > valueExpr d (App f es) = names f <> parens (commaSep (map (valueExpr d) es)) > valueExpr dia (AggregateApp f d es od fil) = > names f > <> parens ((case d of > Distinct -> text "distinct" > All -> text "all" > SQDefault -> empty) > <+> commaSep (map (valueExpr dia) es) > <+> orderBy dia od) > <+> me (\x -> text "filter" > <+> parens (text "where" <+> valueExpr dia x)) fil > valueExpr d (AggregateAppGroup f es od) = > names f > <> parens (commaSep (map (valueExpr d) es)) > <+> if null od > then empty > else text "within group" <+> parens (orderBy d od) > valueExpr d (WindowApp f es pb od fr) = > names f <> parens (commaSep $ map (valueExpr d) es) > <+> text "over" > <+> parens ((case pb of > [] -> empty > _ -> text "partition by" > <+> nest 13 (commaSep $ map (valueExpr d) pb)) > <+> orderBy d od > <+> me frd fr) > where > frd (FrameFrom rs fp) = rsd rs <+> fpd fp > frd (FrameBetween rs fps fpe) = > rsd rs <+> text "between" <+> fpd fps > <+> text "and" <+> fpd fpe > rsd rs = case rs of > FrameRows -> text "rows" > FrameRange -> text "range" > fpd UnboundedPreceding = text "unbounded preceding" > fpd UnboundedFollowing = text "unbounded following" > fpd Current = text "current row" > fpd (Preceding e) = valueExpr d e <+> text "preceding" > fpd (Following e) = valueExpr d e <+> text "following" > valueExpr dia (SpecialOp nm [a,b,c]) | nm `elem` [[Name Nothing "between"] > ,[Name Nothing "not between"]] = > sep [valueExpr dia a > ,names nm <+> valueExpr dia b > ,nest (length (unnames nm) + 1) $ text "and" <+> valueExpr dia c] > valueExpr d (SpecialOp [Name Nothing "rowctor"] as) = > parens $ commaSep $ map (valueExpr d) as > valueExpr d (SpecialOp nm es) = > names nm <+> parens (commaSep $ map (valueExpr d) es) > valueExpr d (SpecialOpK nm fs as) = > names nm <> parens (sep $ catMaybes > (fmap (valueExpr d) fs > : map (\(n,e) -> Just (text n <+> valueExpr d e)) as)) > valueExpr d (PrefixOp f e) = names f <+> valueExpr d e > valueExpr d (PostfixOp f e) = valueExpr d e <+> names f > valueExpr d e@(BinOp _ op _) | op `elem` [[Name Nothing "and"] > ,[Name Nothing "or"]] = > -- special case for and, or, get all the ands so we can vcat them > -- nicely > case ands e of > (e':es) -> vcat (valueExpr d e' > : map ((names op <+>) . valueExpr d) es) > [] -> empty -- shouldn't be possible > where > ands (BinOp a op' b) | op == op' = ands a ++ ands b > ands x = [x] > -- special case for . we don't use whitespace > valueExpr d (BinOp e0 [Name Nothing "."] e1) = > valueExpr d e0 <> text "." <> valueExpr d e1 > valueExpr d (BinOp e0 f e1) = > valueExpr d e0 <+> names f <+> valueExpr d e1 > valueExpr dia (Case t ws els) = > sep $ [text "case" <+> me (valueExpr dia) t] > ++ map w ws > ++ maybeToList (fmap e els) > ++ [text "end"] > where > w (t0,t1) = > text "when" <+> nest 5 (commaSep $ map (valueExpr dia) t0) > <+> text "then" <+> nest 5 (valueExpr dia t1) > e el = text "else" <+> nest 5 (valueExpr dia el) > valueExpr d (Parens e) = parens $ valueExpr d e > valueExpr d (Cast e tn) = > text "cast" <> parens (sep [valueExpr d e > ,text "as" > ,typeName tn]) > valueExpr _ (TypedLit tn s) = > typeName tn <+> quotes (text s) > valueExpr d (SubQueryExpr ty qe) = > (case ty of > SqSq -> empty > SqExists -> text "exists" > SqUnique -> text "unique" > ) <+> parens (queryExpr d qe) > valueExpr d (QuantifiedComparison v c cp sq) = > valueExpr d v > <+> names c > <+> (text $ case cp of > CPAny -> "any" > CPSome -> "some" > CPAll -> "all") > <+> parens (queryExpr d sq) > valueExpr d (Match v u sq) = > valueExpr d v > <+> text "match" > <+> (if u then text "unique" else empty) > <+> parens (queryExpr d sq) > valueExpr d (In b se x) = > valueExpr d se <+> > (if b then empty else text "not") > <+> text "in" > <+> parens (nest (if b then 3 else 7) $ > case x of > InList es -> commaSep $ map (valueExpr d) es > InQueryExpr qe -> queryExpr d qe) > valueExpr d (Array v es) = > valueExpr d v <> brackets (commaSep $ map (valueExpr d) es) > valueExpr d (ArrayCtor q) = > text "array" <> parens (queryExpr d q) > valueExpr d (MultisetCtor es) = > text "multiset" <> brackets (commaSep $ map (valueExpr d) es) > valueExpr d (MultisetQueryCtor q) = > text "multiset" <> parens (queryExpr d q) > valueExpr d (MultisetBinOp a c q b) = > sep > [valueExpr d a > ,text "multiset" > ,text $ case c of > Union -> "union" > Intersect -> "intersect" > Except -> "except" > ,case q of > SQDefault -> empty > All -> text "all" > Distinct -> text "distinct" > ,valueExpr d b] > {-valueExpr d (Escape v e) = > valueExpr d v <+> text "escape" <+> text [e] > valueExpr d (UEscape v e) = > valueExpr d v <+> text "uescape" <+> text [e]-} > valueExpr d (Collate v c) = > valueExpr d v <+> text "collate" <+> names c > valueExpr _ (NextValueFor ns) = > text "next value for" <+> names ns > valueExpr d (VEComment cmt v) = > vcat $ map comment cmt ++ [valueExpr d v] > doubleUpQuotes :: String -> String > doubleUpQuotes [] = [] > doubleUpQuotes ('\'':cs) = '\'':'\'':doubleUpQuotes cs > doubleUpQuotes (c:cs) = c:doubleUpQuotes cs > doubleUpDoubleQuotes :: String -> String > doubleUpDoubleQuotes [] = [] > doubleUpDoubleQuotes ('"':cs) = '"':'"':doubleUpDoubleQuotes cs > doubleUpDoubleQuotes (c:cs) = c:doubleUpDoubleQuotes cs > unname :: Name -> String > unname (Name Nothing n) = n > unname (Name (Just (s,e)) n) = > s ++ (if '"' `elem` s then doubleUpDoubleQuotes n else n) ++ e > unnames :: [Name] -> String > unnames ns = intercalate "." $ map unname ns > name :: Name -> Doc > name (Name Nothing n) = text n > name (Name (Just (s,e)) n) = > text s <> text (if '"' `elem` s then doubleUpDoubleQuotes n else n) <> text e > names :: [Name] -> Doc > names ns = hcat $ punctuate (text ".") $ map name ns > typeName :: TypeName -> Doc > typeName (TypeName t) = names t > typeName (PrecTypeName t a) = names t <+> parens (text $ show a) > typeName (PrecScaleTypeName t a b) = > names t <+> parens (text (show a) <+> comma <+> text (show b)) > typeName (PrecLengthTypeName t i m u) = > names t > <> parens (text (show i) > <> me (\x -> case x of > PrecK -> text "K" > PrecM -> text "M" > PrecG -> text "G" > PrecT -> text "T" > PrecP -> text "P") m > <+> me (\x -> case x of > PrecCharacters -> text "CHARACTERS" > PrecOctets -> text "OCTETS") u) > typeName (CharTypeName t i cs col) = > names t > <> me (\x -> parens (text $ show x)) i > <+> (if null cs > then empty > else text "character set" <+> names cs) > <+> (if null col > then empty > else text "collate" <+> names col) > typeName (TimeTypeName t i tz) = > names t > <> me (\x -> parens (text $ show x)) i > <+> text (if tz > then "with time zone" > else "without time zone") > typeName (RowTypeName cs) = > text "row" <> parens (commaSep $ map f cs) > where > f (n,t) = name n <+> typeName t > typeName (IntervalTypeName f t) = > text "interval" > <+> intervalTypeField f > <+> me (\x -> text "to" <+> intervalTypeField x) t > typeName (ArrayTypeName tn sz) = > typeName tn <+> text "array" <+> me (brackets . text . show) sz > typeName (MultisetTypeName tn) = > typeName tn <+> text "multiset" > intervalTypeField :: IntervalTypeField -> Doc > intervalTypeField (Itf n p) = > text n > <+> me (\(x,x1) -> > parens (text (show x) > <+> me (\y -> (sep [comma,text (show y)])) x1)) p = query expressions > queryExpr :: Dialect -> QueryExpr -> Doc > queryExpr dia (Select d sl fr wh gb hv od off fe) = > sep [text "select" > ,case d of > SQDefault -> empty > All -> text "all" > Distinct -> text "distinct" > ,nest 7 $ sep [selectList dia sl] > ,from dia fr > ,maybeValueExpr dia "where" wh > ,grpBy dia gb > ,maybeValueExpr dia "having" hv > ,orderBy dia od > ,me (\e -> text "offset" <+> valueExpr dia e <+> text "rows") off > ,fetchFirst > ] > where > fetchFirst = > me (\e -> if diSyntaxFlavour dia == MySQL > then text "limit" <+> valueExpr dia e > else text "fetch first" <+> valueExpr dia e > <+> text "rows only") fe > queryExpr dia (CombineQueryExpr q1 ct d c q2) = > sep [queryExpr dia q1 > ,text (case ct of > Union -> "union" > Intersect -> "intersect" > Except -> "except") > <+> case d of > SQDefault -> empty > All -> text "all" > Distinct -> text "distinct" > <+> case c of > Corresponding -> text "corresponding" > Respectively -> empty > ,queryExpr dia q2] > queryExpr d (With rc withs qe) = > text "with" <+> (if rc then text "recursive" else empty) > <+> vcat [nest 5 > (vcat $ punctuate comma $ flip map withs $ \(n,q) -> > alias n <+> text "as" <+> parens (queryExpr d q)) > ,queryExpr d qe] > queryExpr d (Values vs) = > text "values" > <+> nest 7 (commaSep (map (parens . commaSep . map (valueExpr d)) vs)) > queryExpr _ (Table t) = text "table" <+> names t > queryExpr d (QEComment cmt v) = > vcat $ map comment cmt ++ [queryExpr d v] > alias :: Alias -> Doc > alias (Alias nm cols) = > text "as" <+> name nm > <+> me (parens . commaSep . map name) cols > selectList :: Dialect -> [(ValueExpr,Maybe Name)] -> Doc > selectList d is = commaSep $ map si is > where > si (e,al) = valueExpr d e <+> me als al > als al = text "as" <+> name al > from :: Dialect -> [TableRef] -> Doc > from _ [] = empty > from d ts = > sep [text "from" > ,nest 5 $ vcat $ punctuate comma $ map tr ts] > where > tr (TRSimple t) = names t > tr (TRLateral t) = text "lateral" <+> tr t > tr (TRFunction f as) = > names f <> parens (commaSep $ map (valueExpr d) as) > tr (TRAlias t a) = sep [tr t, alias a] > tr (TRParens t) = parens $ tr t > tr (TRQueryExpr q) = parens $ queryExpr d q > tr (TRJoin t0 b jt t1 jc) = > sep [tr t0 > ,if b then text "natural" else empty > ,joinText jt <+> tr t1 > ,joinCond jc] > joinText jt = > sep [case jt of > JInner -> text "inner" > JLeft -> text "left" > JRight -> text "right" > JFull -> text "full" > JCross -> text "cross" > ,text "join"] > joinCond (Just (JoinOn e)) = text "on" <+> valueExpr d e > joinCond (Just (JoinUsing es)) = > text "using" <+> parens (commaSep $ map name es) > joinCond Nothing = empty > maybeValueExpr :: Dialect -> String -> Maybe ValueExpr -> Doc > maybeValueExpr d k = me > (\e -> sep [text k > ,nest (length k + 1) $ valueExpr d e]) > grpBy :: Dialect -> [GroupingExpr] -> Doc > grpBy _ [] = empty > grpBy d gs = sep [text "group by" > ,nest 9 $ commaSep $ map ge gs] > where > ge (SimpleGroup e) = valueExpr d e > ge (GroupingParens g) = parens (commaSep $ map ge g) > ge (Cube es) = text "cube" <> parens (commaSep $ map ge es) > ge (Rollup es) = text "rollup" <> parens (commaSep $ map ge es) > ge (GroupingSets es) = text "grouping sets" <> parens (commaSep $ map ge es) > orderBy :: Dialect -> [SortSpec] -> Doc > orderBy _ [] = empty > orderBy dia os = sep [text "order by" > ,nest 9 $ commaSep $ map f os] > where > f (SortSpec e d n) = > valueExpr dia e > <+> (case d of > Asc -> text "asc" > Desc -> text "desc" > DirDefault -> empty) > <+> (case n of > NullsOrderDefault -> empty > NullsFirst -> text "nulls" <+> text "first" > NullsLast -> text "nulls" <+> text "last") = statements > statement :: Dialect -> Statement -> Doc == ddl > statement _ (CreateSchema nm) = > text "create" <+> text "schema" <+> names nm > statement d (CreateTable nm cds) = > text "create" <+> text "table" <+> names nm > <+> parens (commaSep $ map cd cds) > where > cd (TableConstraintDef n con) = > maybe empty (\s -> text "constraint" <+> names s) n > <+> tableConstraint d con > cd (TableColumnDef cd') = columnDef d cd' > statement d (AlterTable t act) = > texts ["alter","table"] <+> names t > <+> alterTableAction d act > statement _ (DropSchema nm db) = > text "drop" <+> text "schema" <+> names nm <+> dropBehav db > statement d (CreateDomain nm ty def cs) = > text "create" <+> text "domain" <+> names nm > <+> typeName ty > <+> maybe empty (\def' -> text "default" <+> valueExpr d def') def > <+> sep (map con cs) > where > con (cn, e) = > maybe empty (\cn' -> text "constraint" <+> names cn') cn > <+> text "check" <> parens (valueExpr d e) > statement d (AlterDomain nm act) = > texts ["alter","domain"] > <+> names nm > <+> a act > where > a (ADSetDefault v) = texts ["set","default"] <+> valueExpr d v > a (ADDropDefault) = texts ["drop","default"] > a (ADAddConstraint cnm e) = > text "add" > <+> maybe empty (\cnm' -> text "constraint" <+> names cnm') cnm > <+> text "check" <> parens (valueExpr d e) > a (ADDropConstraint cnm) = texts ["drop", "constraint"] > <+> names cnm > statement _ (DropDomain nm db) = > text "drop" <+> text "domain" <+> names nm <+> dropBehav db > statement _ (CreateSequence nm sgos) = > texts ["create","sequence"] <+> names nm > <+> sep (map sequenceGeneratorOption sgos) > statement _ (AlterSequence nm sgos) = > texts ["alter","sequence"] <+> names nm > <+> sep (map sequenceGeneratorOption sgos) > statement _ (DropSequence nm db) = > text "drop" <+> text "sequence" <+> names nm <+> dropBehav db > statement d (CreateAssertion nm ex) = > texts ["create","assertion"] <+> names nm > <+> text "check" <+> parens (valueExpr d ex) > statement _ (DropAssertion nm db) = > text "drop" <+> text "assertion" <+> names nm <+> dropBehav db == dml > statement d (SelectStatement q) = queryExpr d q > statement d (Delete t a w) = > text "delete" <+> text "from" > <+> names t <+> maybe empty (\x -> text "as" <+> name x) a > <+> maybeValueExpr d "where" w > statement _ (Truncate t ir) = > text "truncate" <+> text "table" <+> names t > <+> case ir of > DefaultIdentityRestart -> empty > ContinueIdentity -> text "continue" <+> text "identity" > RestartIdentity -> text "restart" <+> text "identity" > statement d (Insert t cs s) = > text "insert" <+> text "into" <+> names t > <+> maybe empty (\cs' -> parens (commaSep $ map name cs')) cs > <+> case s of > DefaultInsertValues -> text "default" <+> text "values" > InsertQuery q -> queryExpr d q > statement d (Update t a sts whr) = > text "update" <+> names t > <+> maybe empty (\x -> text "as" <+> name x) a > <+> text "set" <+> commaSep (map sc sts) > <+> maybeValueExpr d "where" whr > where > sc (Set tg v) = names tg <+> text "=" <+> valueExpr d v > sc (SetMultiple ts vs) = parens (commaSep $ map names ts) <+> text "=" > <+> parens (commaSep $ map (valueExpr d) vs) > statement _ (DropTable n b) = > text "drop" <+> text "table" <+> names n <+> dropBehav b > statement d (CreateView r nm al q co) = > text "create" <+> (if r then text "recursive" else empty) > <+> text "view" <+> names nm > <+> (maybe empty (\al' -> parens $ commaSep $ map name al')) al > <+> text "as" > <+> queryExpr d q > <+> case co of > Nothing -> empty > Just DefaultCheckOption -> texts ["with", "check", "option"] > Just CascadedCheckOption -> texts ["with", "cascaded", "check", "option"] > Just LocalCheckOption -> texts ["with", "local", "check", "option"] > statement _ (DropView n b) = > text "drop" <+> text "view" <+> names n <+> dropBehav b == transactions > statement _ StartTransaction = > texts ["start", "transaction"] > statement _ (Savepoint nm) = > text "savepoint" <+> name nm > statement _ (ReleaseSavepoint nm) = > texts ["release", "savepoint"] <+> name nm > statement _ Commit = > text "commit" > statement _ (Rollback mn) = > text "rollback" > <+> maybe empty (\n -> texts ["to","savepoint"] <+> name n) mn == access control > statement _ (GrantPrivilege pas po rs go) = > text "grant" <+> commaSep (map privAct pas) > <+> text "on" <+> privObj po > <+> text "to" <+> commaSep (map name rs) > <+> grantOpt go > where > grantOpt WithGrantOption = texts ["with","grant","option"] > grantOpt WithoutGrantOption = empty > statement _ (GrantRole rs trs ao) = > text "grant" <+> commaSep (map name rs) > <+> text "to" <+> commaSep (map name trs) > <+> adminOpt ao > where > adminOpt WithAdminOption = texts ["with","admin","option"] > adminOpt WithoutAdminOption = empty > statement _ (CreateRole nm) = > texts ["create","role"] <+> name nm > statement _ (DropRole nm) = > texts ["drop","role"] <+> name nm > statement _ (RevokePrivilege go pas po rs db) = > text "revoke" > <+> grantOptFor go > <+> commaSep (map privAct pas) > <+> text "on" <+> privObj po > <+> text "from" <+> commaSep (map name rs) > <+> dropBehav db > where > grantOptFor GrantOptionFor = texts ["grant","option","for"] > grantOptFor NoGrantOptionFor = empty > statement _ (RevokeRole ao rs trs db) = > text "revoke" > <+> adminOptFor ao > <+> commaSep (map name rs) > <+> text "from" <+> commaSep (map name trs) > <+> dropBehav db > where > adminOptFor AdminOptionFor = texts ["admin","option","for"] > adminOptFor NoAdminOptionFor = empty == sessions == extras > dropBehav :: DropBehaviour -> Doc > dropBehav DefaultDropBehaviour = empty > dropBehav Cascade = text "cascade" > dropBehav Restrict = text "restrict" > columnDef :: Dialect -> ColumnDef -> Doc > columnDef d (ColumnDef n t mdef cons) = > name n <+> typeName t > <+> case mdef of > Nothing -> empty > Just (DefaultClause def) -> > text "default" <+> valueExpr d def > Just (GenerationClause e) -> > texts ["generated","always","as"] <+> parens (valueExpr d e) > Just (IdentityColumnSpec w o) -> > text "generated" > <+> (case w of > GeneratedAlways -> text "always" > GeneratedByDefault -> text "by" <+> text "default") > <+> text "as" <+> text "identity" > <+> (case o of > [] -> empty > os -> parens (sep $ map sequenceGeneratorOption os)) > <+> sep (map cdef cons) > where > cdef (ColConstraintDef cnm con) = > maybe empty (\s -> text "constraint" <+> names s) cnm > <+> pcon con > pcon ColNotNullConstraint = texts ["not","null"] > pcon ColUniqueConstraint = text "unique" > pcon ColPrimaryKeyConstraint = texts ["primary","key"] > pcon (ColCheckConstraint v) = text "check" <+> parens (valueExpr d v) > pcon (ColReferencesConstraint tb c m u del) = > text "references" > <+> names tb > <+> maybe empty (\c' -> parens (name c')) c > <+> refMatch m > <+> refAct "update" u > <+> refAct "delete" del > sequenceGeneratorOption :: SequenceGeneratorOption -> Doc > sequenceGeneratorOption (SGODataType t) = > text "as" <+> typeName t > sequenceGeneratorOption (SGORestart mi) = > text "restart" <+> maybe empty (\mi' -> texts ["with", show mi']) mi > sequenceGeneratorOption (SGOStartWith i) = texts ["start", "with", show i] > sequenceGeneratorOption (SGOIncrementBy i) = texts ["increment", "by", show i] > sequenceGeneratorOption (SGOMaxValue i) = texts ["maxvalue", show i] > sequenceGeneratorOption SGONoMaxValue = texts ["no", "maxvalue"] > sequenceGeneratorOption (SGOMinValue i) = texts ["minvalue", show i] > sequenceGeneratorOption SGONoMinValue = texts ["no", "minvalue"] > sequenceGeneratorOption SGOCycle = text "cycle" > sequenceGeneratorOption SGONoCycle = text "no cycle" > refMatch :: ReferenceMatch -> Doc > refMatch m = case m of > DefaultReferenceMatch -> empty > MatchFull -> texts ["match", "full"] > MatchPartial -> texts ["match","partial"] > MatchSimple -> texts ["match", "simple"] > refAct :: String -> ReferentialAction -> Doc > refAct t a = case a of > DefaultReferentialAction -> empty > RefCascade -> texts ["on", t, "cascade"] > RefSetNull -> texts ["on", t, "set", "null"] > RefSetDefault -> texts ["on", t, "set", "default"] > RefRestrict -> texts ["on", t, "restrict"] > RefNoAction -> texts ["on", t, "no", "action"] > alterTableAction :: Dialect -> AlterTableAction -> Doc > alterTableAction d (AddColumnDef cd) = > texts ["add", "column"] <+> columnDef d cd > alterTableAction d (AlterColumnSetDefault n v) = > texts ["alter", "column"] > <+> name n > <+> texts ["set","default"] <+> valueExpr d v > alterTableAction _ (AlterColumnDropDefault n) = > texts ["alter", "column"] > <+> name n > <+> texts ["drop","default"] > alterTableAction _ (AlterColumnSetNotNull n) = > texts ["alter", "column"] > <+> name n > <+> texts ["set","not","null"] > alterTableAction _ (AlterColumnDropNotNull n) = > texts ["alter", "column"] > <+> name n > <+> texts ["drop","not","null"] > alterTableAction _ (AlterColumnSetDataType n t) = > texts ["alter", "column"] > <+> name n > <+> texts ["set","data","Type"] > <+> typeName t > alterTableAction _ (DropColumn n b) = > texts ["drop", "column"] > <+> name n > <+> dropBehav b > alterTableAction d (AddTableConstraintDef n con) = > text "add" > <+> maybe empty (\s -> text "constraint" <+> names s) n > <+> tableConstraint d con > alterTableAction _ (DropTableConstraintDef n b) = > texts ["drop", "constraint"] > <+> names n > <+> dropBehav b > tableConstraint :: Dialect -> TableConstraint -> Doc > tableConstraint _ (TableUniqueConstraint ns) = > text "unique" <+> parens (commaSep $ map name ns) > tableConstraint _ (TablePrimaryKeyConstraint ns) = > texts ["primary","key"] <+> parens (commaSep $ map name ns) > tableConstraint _ (TableReferencesConstraint cs t tcs m u del) = > texts ["foreign", "key"] > <+> parens (commaSep $ map name cs) > <+> text "references" > <+> names t > <+> maybe empty (\c' -> parens (commaSep $ map name c')) tcs > <+> refMatch m > <+> refAct "update" u > <+> refAct "delete" del > tableConstraint d (TableCheckConstraint v) = text "check" <+> parens (valueExpr d v) > privAct :: PrivilegeAction -> Doc > privAct PrivAll = texts ["all","privileges"] > privAct (PrivSelect cs) = text "select" <+> maybeColList cs > privAct (PrivInsert cs) = text "insert" <+> maybeColList cs > privAct (PrivUpdate cs) = text "update" <+> maybeColList cs > privAct (PrivReferences cs) = text "references" <+> maybeColList cs > privAct PrivDelete = text "delete" > privAct PrivUsage = text "usage" > privAct PrivTrigger = text "trigger" > privAct PrivExecute = text "execute" > maybeColList :: [Name] -> Doc > maybeColList cs = > if null cs > then empty > else parens (commaSep $ map name cs) > privObj :: PrivilegeObject -> Doc > privObj (PrivTable nm) = names nm > privObj (PrivDomain nm) = text "domain" <+> names nm > privObj (PrivType nm) = text "type" <+> names nm > privObj (PrivSequence nm) = text "sequence" <+> names nm > privObj (PrivFunction nm) = texts ["specific", "function"] <+> names nm = utils > commaSep :: [Doc] -> Doc > commaSep ds = sep $ punctuate comma ds > me :: (a -> Doc) -> Maybe a -> Doc > me = maybe empty > comment :: Comment -> Doc > comment (BlockComment str) = text "/*" <+> text str <+> text "*/" > texts :: [String] -> Doc > texts ts = sep $ map text ts