mirror of
https://github.com/hasura/graphql-engine.git
synced 2025-01-05 14:27:59 +03:00
6ad050a29c
PR-URL: https://github.com/hasura/graphql-engine-mono/pull/3983 GitOrigin-RevId: 6fa30a68b1f3d450e9da7b722b9e4938c71e7091
119 lines
3.7 KiB
Haskell
119 lines
3.7 KiB
Haskell
{-# LANGUAGE TemplateHaskell #-}
|
||
|
||
-- | Postgres SQL Error
|
||
--
|
||
-- Functions and datatypes for interpreting Postgres errors.
|
||
module Hasura.Backends.Postgres.SQL.Error
|
||
( PGErrorType (..),
|
||
_PGDataException,
|
||
_PGIntegrityConstraintViolation,
|
||
_PGSyntaxErrorOrAccessRuleViolation,
|
||
_PGTransactionRollback,
|
||
pgErrorType,
|
||
PGErrorCode (..),
|
||
_PGErrorGeneric,
|
||
_PGErrorSpecific,
|
||
PGDataException (..),
|
||
PGIntegrityConstraintViolation (..),
|
||
PGSyntaxErrorOrAccessRuleViolation (..),
|
||
PGTransactionRollback (..),
|
||
)
|
||
where
|
||
|
||
import Control.Lens.TH (makePrisms)
|
||
import Data.Text qualified as T
|
||
import Database.PG.Query.Connection qualified as Q
|
||
import Hasura.Prelude
|
||
|
||
-- | The top-level error code type. Errors in Postgres are divided into different /classes/, which
|
||
-- are further subdivided into individual error codes. Even if a particular status code is not known
|
||
-- to the application, it’s possible to determine its class and handle it appropriately.
|
||
data PGErrorType
|
||
= PGDataException !(Maybe (PGErrorCode PGDataException))
|
||
| PGIntegrityConstraintViolation !(Maybe (PGErrorCode PGIntegrityConstraintViolation))
|
||
| PGSyntaxErrorOrAccessRuleViolation !(Maybe (PGErrorCode PGSyntaxErrorOrAccessRuleViolation))
|
||
| PGTransactionRollback !(Maybe (PGErrorCode PGTransactionRollback))
|
||
deriving (Show, Eq)
|
||
|
||
data PGErrorCode a
|
||
= -- | represents errors that have the non-specific @000@ status code
|
||
PGErrorGeneric
|
||
| -- | represents errors with a known, more specific status code
|
||
PGErrorSpecific !a
|
||
deriving (Show, Eq, Functor)
|
||
|
||
data PGDataException
|
||
= PGInvalidDatetimeFormat
|
||
| PGInvalidParameterValue
|
||
| PGInvalidEscapeSequence
|
||
| PGInvalidTextRepresentation
|
||
deriving (Show, Eq)
|
||
|
||
data PGIntegrityConstraintViolation
|
||
= PGRestrictViolation
|
||
| PGNotNullViolation
|
||
| PGForeignKeyViolation
|
||
| PGUniqueViolation
|
||
| PGCheckViolation
|
||
| PGExclusionViolation
|
||
deriving (Show, Eq)
|
||
|
||
data PGSyntaxErrorOrAccessRuleViolation
|
||
= PGUndefinedObject
|
||
| PGInvalidColumnReference
|
||
deriving (Show, Eq)
|
||
|
||
data PGTransactionRollback
|
||
= PGSerializationFailure
|
||
deriving (Show, Eq)
|
||
|
||
$(makePrisms ''PGErrorType)
|
||
$(makePrisms ''PGErrorCode)
|
||
|
||
pgErrorType :: Q.PGStmtErrDetail -> Maybe PGErrorType
|
||
pgErrorType errorDetails = do
|
||
parseTypes =<< Q.edStatusCode errorDetails
|
||
where
|
||
parseTypes fullCodeText =
|
||
choice
|
||
[ withClass
|
||
"22"
|
||
PGDataException
|
||
[ code "007" PGInvalidDatetimeFormat,
|
||
code "023" PGInvalidParameterValue,
|
||
code "025" PGInvalidEscapeSequence,
|
||
code "P02" PGInvalidTextRepresentation
|
||
],
|
||
withClass
|
||
"23"
|
||
PGIntegrityConstraintViolation
|
||
[ code "001" PGRestrictViolation,
|
||
code "502" PGNotNullViolation,
|
||
code "503" PGForeignKeyViolation,
|
||
code "505" PGUniqueViolation,
|
||
code "514" PGCheckViolation,
|
||
code "P01" PGExclusionViolation
|
||
],
|
||
withClass
|
||
"40"
|
||
PGTransactionRollback
|
||
[ code "001" PGSerializationFailure
|
||
],
|
||
withClass
|
||
"42"
|
||
PGSyntaxErrorOrAccessRuleViolation
|
||
[ code "704" PGUndefinedObject,
|
||
code "P10" PGInvalidColumnReference
|
||
]
|
||
]
|
||
where
|
||
(classText, codeText) = T.splitAt 2 fullCodeText
|
||
|
||
withClass :: Text -> (Maybe a -> b) -> [Maybe a] -> Maybe b
|
||
withClass expectedClassText mkClass codes =
|
||
guard (classText == expectedClassText) $> mkClass (choice codes)
|
||
|
||
code :: Text -> a -> Maybe (PGErrorCode a)
|
||
code expectedCodeText codeValue =
|
||
guard (codeText == expectedCodeText) $> PGErrorSpecific codeValue
|