1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383
|
module Database.HDBC.PostgreSQL.Statement where
import Database.HDBC.Types
import Database.HDBC
import Database.HDBC.PostgreSQL.Types
import Database.HDBC.PostgreSQL.Utils
import Foreign.C.Types
import Foreign.ForeignPtr
import Foreign.Ptr
import Control.Concurrent.MVar
import Foreign.C.String
import Control.Monad
import Data.List
import Data.Word
import Data.Ratio
import qualified Data.ByteString as B
import qualified Data.ByteString.UTF8 as BUTF8
import Database.HDBC.PostgreSQL.Parser(convertSQL)
import Database.HDBC.DriverUtils
import Database.HDBC.PostgreSQL.PTypeConv
import Data.Time.Format
#ifndef MIN_TIME_15
import System.Locale
#endif
l :: Monad m => t -> m ()
l _ = return ()
--l m = hPutStrLn stderr ("\n" ++ m)
#include <libpq-fe.h>
data SState =
SState { stomv :: MVar (Maybe Stmt),
nextrowmv :: MVar (CInt), -- -1 for no next row (empty); otherwise, next row to read.
dbo :: Conn,
squery :: String,
coldefmv :: MVar [(String, SqlColDesc)]}
-- FIXME: we currently do no prepare optimization whatsoever.
newSth :: Conn -> ChildList -> String -> IO Statement
newSth indbo mchildren query =
do l "in newSth"
newstomv <- newMVar Nothing
newnextrowmv <- newMVar (-1)
newcoldefmv <- newMVar []
usequery <- case convertSQL query of
Left errstr -> throwSqlError $ SqlError
{seState = "",
seNativeError = (-1),
seErrorMsg = "hdbc prepare: " ++
show errstr}
Right converted -> return converted
let sstate = SState {stomv = newstomv, nextrowmv = newnextrowmv,
dbo = indbo, squery = usequery,
coldefmv = newcoldefmv}
let retval =
Statement {execute = fexecute sstate,
executeMany = fexecutemany sstate,
executeRaw = fexecuteRaw sstate,
finish = public_ffinish sstate,
fetchRow = ffetchrow sstate,
originalQuery = query,
getColumnNames = fgetColumnNames sstate,
describeResult = fdescribeResult sstate}
addChild mchildren retval
return retval
fgetColumnNames :: SState -> IO [(String)]
fgetColumnNames sstate =
do c <- readMVar (coldefmv sstate)
return (map fst c)
fdescribeResult :: SState -> IO [(String, SqlColDesc)]
fdescribeResult sstate =
readMVar (coldefmv sstate)
{- For now, we try to just handle things as simply as possible.
FIXME lots of room for improvement here (types, etc). -}
fexecute :: (Num a, Read a) => SState -> [SqlValue] -> IO a
fexecute sstate args = withConnLocked (dbo sstate) $ \cconn ->
B.useAsCString (BUTF8.fromString (squery sstate)) $ \cquery ->
withCStringArr0 args $ \cargs -> -- wichSTringArr0 uses UTF-8
do l "in fexecute"
public_ffinish sstate -- Sets nextrowmv to -1
resptr <- pqexecParams cconn cquery
(genericLength args) nullPtr cargs nullPtr nullPtr 0
handleResultStatus cconn resptr sstate =<< pqresultStatus resptr
{- | Differs from fexecute in that it does not prepare its input
query, and the input query may contain multiple statements. This
is useful for issuing DDL or DML commands. -}
fexecuteRaw :: SState -> IO ()
fexecuteRaw sstate =
withConnLocked (dbo sstate) $ \cconn ->
B.useAsCString (BUTF8.fromString (squery sstate)) $ \cquery ->
do l "in fexecute"
public_ffinish sstate -- Sets nextrowmv to -1
resptr <- pqexec cconn cquery
_ <- handleResultStatus cconn resptr sstate =<< pqresultStatus resptr :: IO Int
return ()
handleResultStatus :: (Num a, Read a) => Ptr CConn -> Ptr CStmt -> SState -> ResultStatus -> IO a
handleResultStatus cconn resptr sstate status =
case status of
#{const PGRES_EMPTY_QUERY} ->
do l $ "PGRES_EMPTY_QUERY: " ++ squery sstate
pqclear_raw resptr
_ <- swapMVar (coldefmv sstate) []
return 0
#{const PGRES_COMMAND_OK} ->
do l $ "PGRES_COMMAND_OK: " ++ squery sstate
rowscs <- pqcmdTuples resptr
rows <- peekCString rowscs
pqclear_raw resptr
_ <- swapMVar (coldefmv sstate) []
return $ case rows of
"" -> 0
x -> read x
#{const PGRES_TUPLES_OK} ->
do l $ "PGRES_TUPLES_OK: " ++ squery sstate
_ <- fgetcoldef resptr >>= swapMVar (coldefmv sstate)
numrows <- pqntuples resptr
if numrows < 1 then (pqclear_raw resptr >> return 0) else
do
fresptr <- newForeignPtr pqclearptr resptr
_ <- swapMVar (nextrowmv sstate) 0
_ <- swapMVar (stomv sstate) (Just fresptr)
return 0
_ | resptr == nullPtr -> do
l $ "PGRES ERROR: " ++ squery sstate
errormsg <- peekCStringUTF8 =<< pqerrorMessage cconn
statusmsg <- peekCStringUTF8 =<< pqresStatus status
throwSqlError $ SqlError { seState = "E"
, seNativeError = fromIntegral status
, seErrorMsg = "execute: " ++ statusmsg ++
": " ++ errormsg}
_ -> do l $ "PGRES ERROR: " ++ squery sstate
errormsg <- peekCStringUTF8 =<< pqresultErrorMessage resptr
statusmsg <- peekCStringUTF8 =<< pqresStatus status
state <- peekCStringUTF8 =<<
pqresultErrorField resptr #{const PG_DIAG_SQLSTATE}
pqclear_raw resptr
throwSqlError $ SqlError { seState = state
, seNativeError = fromIntegral status
, seErrorMsg = "execute: " ++ statusmsg ++
": " ++ errormsg}
peekCStringUTF8 :: CString -> IO String
-- Marshal a NUL terminated C string into a Haskell string, decoding it
-- with UTF8.
peekCStringUTF8 str
| str == nullPtr = return ""
| otherwise = fmap BUTF8.toString (B.packCString str)
{- General algorithm: find out how many columns we have, check the type
of each to see if it's NULL. If it's not, fetch it as text and return that.
-}
ffetchrow :: SState -> IO (Maybe [SqlValue])
ffetchrow sstate = modifyMVar (nextrowmv sstate) dofetchrow
where dofetchrow (-1) = l "ffr -1" >> return ((-1), Nothing)
dofetchrow nextrow = modifyMVar (stomv sstate) $ \stmt ->
case stmt of
Nothing -> l "ffr nos" >> return (stmt, ((-1), Nothing))
Just cmstmt -> withStmt cmstmt $ \cstmt ->
do l $ "ffetchrow: " ++ show nextrow
numrows <- pqntuples cstmt
l $ "numrows: " ++ show numrows
if nextrow >= numrows
then do l "no more rows"
-- Don't use public_ffinish here
ffinish cmstmt
return (Nothing, ((-1), Nothing))
else do l "getting stuff"
ncols <- pqnfields cstmt
res <- mapM (getCol cstmt nextrow)
[0..(ncols - 1)]
return (stmt, (nextrow + 1, Just res))
getCol p row icol =
do isnull <- pqgetisnull p row icol
if isnull /= 0
then return SqlNull
else do text <- pqgetvalue p row icol
coltype <- liftM oidToColType $ pqftype p icol
s <- B.packCString text
makeSqlValue coltype s
fgetcoldef :: Ptr CStmt -> IO [(String, SqlColDesc)]
fgetcoldef cstmt =
do ncols <- pqnfields cstmt
mapM desccol [0..(ncols - 1)]
where desccol i =
do colname <- peekCStringUTF8 =<< pqfname cstmt i
coltype <- pqftype cstmt i
--coloctets <- pqfsize
let coldef = oidToColDef coltype
return (colname, coldef)
-- FIXME: needs a faster algorithm.
fexecutemany :: SState -> [[SqlValue]] -> IO ()
fexecutemany sstate arglist =
mapM_ (fexecute sstate :: [SqlValue] -> IO Int) arglist >> return ()
-- Finish and change state
public_ffinish :: SState -> IO ()
public_ffinish sstate =
do l "public_ffinish"
_ <- swapMVar (nextrowmv sstate) (-1)
modifyMVar_ (stomv sstate) worker
where worker Nothing = return Nothing
worker (Just sth) = ffinish sth >> return Nothing
ffinish :: Stmt -> IO ()
ffinish _ = pure ()
foreign import ccall unsafe "libpq-fe.h PQresultStatus"
pqresultStatus :: (Ptr CStmt) -> IO #{type ExecStatusType}
foreign import ccall safe "libpq-fe.h PQexecParams"
pqexecParams :: (Ptr CConn) -> CString -> CInt ->
(Ptr #{type Oid}) ->
(Ptr CString) ->
(Ptr CInt) ->
(Ptr CInt) ->
CInt ->
IO (Ptr CStmt)
foreign import ccall safe "libpq-fe.h PQexec"
pqexec :: (Ptr CConn) -> CString -> IO (Ptr CStmt)
foreign import ccall unsafe "libpq-fe.h &PQclear"
pqclearptr :: FunPtr (Ptr CStmt -> IO ())
foreign import ccall unsafe "libpq-fe.h PQclear"
pqclear_raw :: Ptr CStmt -> IO ()
foreign import ccall unsafe "libpq-fe.h PQcmdTuples"
pqcmdTuples :: Ptr CStmt -> IO CString
foreign import ccall unsafe "libpq-fe.h PQresStatus"
pqresStatus :: #{type ExecStatusType} -> IO CString
foreign import ccall unsafe "libpq-fe.h PQresultErrorMessage"
pqresultErrorMessage :: (Ptr CStmt) -> IO CString
foreign import ccall unsafe "libpq-fe.h PQresultErrorField"
pqresultErrorField :: (Ptr CStmt) -> CInt -> IO CString
foreign import ccall unsafe "libpq-fe.h PQntuples"
pqntuples :: Ptr CStmt -> IO CInt
foreign import ccall unsafe "libpq-fe.h PQnfields"
pqnfields :: Ptr CStmt -> IO CInt
foreign import ccall unsafe "libpq-fe.h PQgetisnull"
pqgetisnull :: Ptr CStmt -> CInt -> CInt -> IO CInt
foreign import ccall unsafe "libpq-fe.h PQgetvalue"
pqgetvalue :: Ptr CStmt -> CInt -> CInt -> IO CString
foreign import ccall unsafe "libpq-fe.h PQfname"
pqfname :: Ptr CStmt -> CInt -> IO CString
foreign import ccall unsafe "libpq-fe.h PQftype"
pqftype :: Ptr CStmt -> CInt -> IO #{type Oid}
-- SqlValue construction function and helpers
-- Make a SqlValue for the passed column type and string value, where it is assumed that the value represented is not the Sql null value.
-- The IO Monad is required only to obtain the local timezone for interpreting date/time values without an explicit timezone.
makeSqlValue :: SqlTypeId -> B.ByteString -> IO SqlValue
makeSqlValue sqltypeid bstrval =
let strval = BUTF8.toString bstrval
in
case sqltypeid of
tid | tid == SqlCharT ||
tid == SqlVarCharT ||
tid == SqlLongVarCharT ||
tid == SqlWCharT ||
tid == SqlWVarCharT ||
tid == SqlWLongVarCharT -> return $ SqlByteString bstrval
tid | tid == SqlDecimalT ||
tid == SqlNumericT -> return $ SqlRational (makeRationalFromDecimal strval)
tid | tid == SqlSmallIntT ||
tid == SqlTinyIntT ||
tid == SqlIntegerT -> return $ SqlInt32 (read strval)
SqlBigIntT -> return $ SqlInteger (read strval)
tid | tid == SqlRealT ||
tid == SqlFloatT ||
tid == SqlDoubleT -> return $ SqlDouble (read strval)
SqlBitT -> return $ case strval of
't':_ -> SqlBool True
'f':_ -> SqlBool False
'T':_ -> SqlBool True -- the rest of these are here "just in case", since they are legal as input
'y':_ -> SqlBool True
'Y':_ -> SqlBool True
"1" -> SqlBool True
_ -> SqlBool False
-- Dates and Date/Times
tid | tid == SqlDateT -> return $ SqlLocalDate (fromSql (toSql strval))
tid | tid == SqlTimestampWithZoneT -> return $ SqlZonedTime (fromSql (toSql (fixString strval)))
-- SqlUTCDateTimeT not actually generated by PostgreSQL
tid | tid == SqlTimestampT ||
tid == SqlUTCDateTimeT -> return $ SqlLocalTime (fromSql (toSql strval))
-- Times without dates
tid | tid == SqlTimeT ||
tid == SqlUTCTimeT -> return $ SqlLocalTimeOfDay (fromSql (toSql strval))
tid | tid == SqlTimeWithZoneT ->
(let (a, b) = case (parseTime' defaultTimeLocale "%T%Q %z" timestr,
parseTime' defaultTimeLocale "%T%Q %z" timestr) of
(Just x, Just y) -> (x, y)
x -> error $ "PostgreSQL Statement.hsc: Couldn't parse " ++ strval ++ " as SqlZonedLocalTimeOfDay: " ++ show x
timestr = fixString strval
in return $ SqlZonedLocalTimeOfDay a b)
SqlIntervalT _ -> return $ SqlDiffTime $ fromRational $
case split ':' strval of
[h, m, s] -> toRational (((read h)::Integer) * 60 * 60 +
((read m)::Integer) * 60) +
toRational ((read s)::Double)
_ -> error $ "PostgreSQL Statement.hsc: Couldn't parse interval: " ++ strval
-- TODO: For now we just map the binary types to SqlByteStrings. New SqlValue constructors are needed to handle these.
tid | tid == SqlBinaryT ||
tid == SqlVarBinaryT ||
tid == SqlLongVarBinaryT -> return $ SqlByteString bstrval
SqlGUIDT -> return $ SqlByteString bstrval
SqlUnknownT _ -> return $ SqlByteString bstrval
_ -> error $ "PostgreSQL Statement.hsc: unknown typeid: " ++ show sqltypeid
-- Convert "15:33:01.536+00" to "15:33:01.536 +0000"
fixString :: String -> String
fixString s =
let (strbase, zone) = splitAt (length s - 3) s
in
if (head zone) == '-' || (head zone) == '+'
then strbase ++ " " ++ zone ++ "00"
else -- It wasn't in the expected format; don't touch.
s
-- Make a rational number from a decimal string representation of the number.
makeRationalFromDecimal :: String -> Rational
makeRationalFromDecimal s =
case elemIndex '.' s of
Nothing -> toRational ((read s)::Integer)
Just dotix ->
let (nstr,'.':dstr) = splitAt dotix s
num = (read $ nstr ++ dstr)::Integer
den = 10^((genericLength dstr) :: Integer)
in
num % den
split :: Char -> String -> [String]
split delim inp =
lines . map (\x -> if x == delim then '\n' else x) $ inp
parseTime' :: ParseTime t => TimeLocale -> String -> String -> Maybe t
#if MIN_TIME_15
parseTime' = parseTimeM True
#else
parseTime' = parseTime
#endif
|