haskell-relational-record/doc/examples/Query.hs

218 lines
5.6 KiB
Haskell
Raw Normal View History

2013-07-03 15:52:28 +04:00
{-# LANGUAGE MonadComprehensions #-}
{-# LANGUAGE FlexibleContexts #-}
import Database.Record
import Database.Relational.Query
import Database.HDBC (IConnection, SqlValue)
import Data.Int (Int32)
import qualified Account
import Account (Account(..), account)
import qualified Customer
import Customer (Customer, customer)
import qualified Individual
import Individual (Individual, individual)
import qualified ProductType
import ProductType (ProductType, productType)
import qualified Branch
import Branch (Branch, Branch)
import qualified Officer
import Officer (Officer, Officer)
import qualified Transaction
import Transaction (Transaction, transaction)
import qualified Business
import Business (Business, business)
import qualified Department
import Department (Department, department)
import qualified Product
import Product (Product, product)
import qualified Employee
import Employee (Employee, employee)
import DataSource (connect)
import Database.HDBC.Record.Query
(ExecutedStatement,
runQuery, prepare, bindTo, execute, fetchUnique, fetchUnique')
import Database.HDBC.Session (withConnectionIO, handleSqlError')
allAccount :: Relation () (Account)
allAccount =
relation
2013-07-03 15:52:28 +04:00
[ a
| a <- query account
]
2013-08-02 22:56:28 +04:00
-- sql/4.3.3a.sh
--
-- @
-- SELECT account_id, product_cd, cust_id, avail_balance
-- FROM LEARNINGSQL.account
-- WHERE product_cd IN ('CHK', 'SAV', 'CD', 'MM')
-- @
--
account1 :: Relation () Account
account1 =
relation
2013-08-02 22:56:28 +04:00
[ a
| a <- query account
, () <- wheres $ a ! Account.productCd' `in'` values ["CHK", "SAV", "CD", "MM"]
]
account1' :: Relation () (((Int32, String), Int32), Maybe Double)
account1' =
relation
2013-08-02 22:56:28 +04:00
[ a ! Account.accountId' >< a ! Account.productCd' >< a ! Account.custId' >< a ! Account.availBalance'
| a <- query account
, () <- wheres $ a ! Account.productCd' `in'` values ["CHK", "SAV", "CD", "MM"]
]
-- | sql/5.1.2a.sh
--
-- @
-- SELECT e.fname, e.lname, d.name
-- FROM LEARNINGSQL.employee e INNER JOIN LEARNINGSQL.department d
-- USING (dept_id)
-- @
--
2013-07-03 17:58:35 +04:00
join1 :: Relation () (Employee, Department)
2013-07-03 15:52:28 +04:00
join1 =
relation
2013-07-03 15:52:28 +04:00
[ e >< d
2013-08-02 22:56:28 +04:00
| e <- query employee
, d <- query department
2013-07-03 17:58:35 +04:00
, () <- on $ e ! Employee.deptId' .=. just (d ! Department.deptId')
2013-07-03 15:52:28 +04:00
]
2013-08-02 22:56:28 +04:00
join1' :: Relation () ((String, String), String)
join1' =
relation
2013-08-02 22:56:28 +04:00
[ e ! Employee.fname' >< e ! Employee.lname' >< d ! Department.name'
| e <- query employee
, d <- query department
, () <- on $ e ! Employee.deptId' .=. just (d ! Department.deptId')
]
-- | sql/5.3a.sh
--
-- @
-- SELECT e.fname, e.lname, e_mgr.fname mgr_fname, e_mgr.lname mgr_lname
-- FROM LEARNINGSQL.employee e INNER JOIN LEARNINGSQL.employee e_mgr
-- ON e.superior_emp_id = e_mgr.emp_id
-- @
--
selfJoin1 :: Relation () (Employee, Employee)
selfJoin1 =
relation
2013-08-02 22:56:28 +04:00
[ e >< m
| e <- query employee
, m <- query employee
, () <- on $ e ! Employee.superiorEmpId' .=. just (m ! Employee.empId')
]
selfJoin1' :: Relation () ((String, String), (String, String))
selfJoin1' =
relation
2013-08-02 22:56:28 +04:00
[ emp >< mgr
| e <- query employee
, m <- query employee
, () <- on $ e ! Employee.superiorEmpId' .=. just (m ! Employee.empId')
, let emp = e ! Employee.fname' >< e ! Employee.lname'
, let mgr = m ! Employee.fname' >< m ! Employee.lname'
]
-- | sql/6.4.1a.sh
--
-- @
-- SELECT emp_id, assigned_branch_id
-- FROM LEARNINGSQL.employee
-- WHERE title = 'Teller'
-- UNION
-- SELECT open_emp_id, open_branch_id
-- FROM LEARNINGSQL.account
-- WHERE product_cd = 'SAV'
-- ORDER BY open_emp_id
-- @
--
employee1 :: Relation () (Maybe Int32, Maybe Int32)
employee1 =
relation
2013-08-02 22:56:28 +04:00
[ just (e ! Employee.empId') >< e ! Employee.assignedBranchId'
| e <- query employee
, () <- wheres $ e ! Employee.title' .=. just (value "Teller")
]
account2 :: Relation () (Maybe Int32, Maybe Int32)
account2 =
relation
2013-08-02 22:56:28 +04:00
[ a ! Account.openEmpId' >< a ! Account.openBranchId'
| a <- query account
, () <- wheres $ a ! Account.productCd' .=. value "SAV"
]
union1 :: Relation () (Maybe Int32, Maybe Int32)
union1 =
relation
2013-08-02 22:56:28 +04:00
[ ea
| ea <- query $ employee1 `union` account2
, () <- asc $ ea ! fst'
]
union1' :: Relation () (Maybe Int32, Maybe Int32)
union1' =
relation
[ ea
| ea <- query $ relation
[ just (e ! Employee.empId') >< e ! Employee.assignedBranchId'
| e <- query employee
, () <- wheres $ e ! Employee.title' .=. just (value "Teller")
]
`union` relation
[ a ! Account.openEmpId' >< a ! Account.openBranchId'
| a <- query account
, () <- wheres $ a ! Account.productCd' .=. value "SAV"
]
, () <- asc $ ea ! fst'
]
2013-08-02 22:56:28 +04:00
-- | sql/8.1a.sh
--
-- @
-- SELECT open_emp_id, COUNT(*) how_many
-- FROM LEARNINGSQL.account
-- GROUP BY open_emp_id
-- ORDER BY open_emp_id
-- @
--
group1 :: Relation () (Maybe Int32, Int32)
group1 =
aggregateRelation
2013-08-02 22:56:28 +04:00
[ g >< count a
| a <- query account
, g <- groupBy $ a ! Account.openEmpId'
, () <- asc $ g <!> id'
2013-08-02 22:56:28 +04:00
]
runAndPrint :: (Show a, IConnection conn, FromSql SqlValue a, ToSql SqlValue p)
=> conn -> Relation p a -> p -> IO ()
runAndPrint conn rel param = do
putStrLn $ "SQL: " ++ sqlFromRelation rel
records <- runQuery conn param (fromRelation rel)
mapM_ print records
putStrLn ""
2013-07-03 15:52:28 +04:00
main :: IO ()
2013-08-02 22:56:28 +04:00
main = handleSqlError' $ withConnectionIO connect $ \conn -> do
let run :: (Show a, FromSql SqlValue a, ToSql SqlValue p) => Relation p a -> p -> IO ()
run = runAndPrint conn
run allAccount ()
run account1 ()
run account1' ()
run join1 ()
run join1' ()
run selfJoin1 ()
run selfJoin1' ()
run union1 ()
run union1' ()
2013-08-02 22:56:28 +04:00
run group1 ()