2022-03-16 03:39:21 +03:00
|
|
|
{-# LANGUAGE TemplateHaskell #-}
|
|
|
|
|
2022-01-03 20:16:24 +03:00
|
|
|
-- | Metadata related types, functions and helpers.
|
2022-01-11 01:54:51 +03:00
|
|
|
--
|
|
|
|
-- Provides a single function which loads the MSSQL database metadata.
|
2022-04-21 10:19:37 +03:00
|
|
|
-- See the file at src-rsr/mssql/mssql_table_metadata.sql for the SQL we use to build
|
2022-01-11 01:54:51 +03:00
|
|
|
-- this metadata.
|
|
|
|
-- See 'Hasura.RQL.Types.Table.DBTableMetadata' for the Haskell type we use forall
|
|
|
|
-- storing this metadata.
|
2021-02-23 20:37:27 +03:00
|
|
|
module Hasura.Backends.MSSQL.Meta
|
2021-09-24 01:56:37 +03:00
|
|
|
( loadDBMetadata,
|
|
|
|
)
|
|
|
|
where
|
|
|
|
|
|
|
|
import Data.Aeson as Aeson
|
|
|
|
import Data.ByteString.UTF8 qualified as BSUTF8
|
|
|
|
import Data.FileEmbed (embedFile, makeRelativeToProject)
|
|
|
|
import Data.HashMap.Strict qualified as HM
|
2022-05-10 18:43:24 +03:00
|
|
|
import Data.HashMap.Strict.NonEmpty qualified as NEHashMap
|
2021-09-24 01:56:37 +03:00
|
|
|
import Data.HashSet qualified as HS
|
|
|
|
import Data.String
|
|
|
|
import Data.Text qualified as T
|
|
|
|
import Data.Text.Encoding qualified as T
|
2021-10-22 17:49:15 +03:00
|
|
|
import Database.MSSQL.Transaction qualified as Tx
|
2021-09-24 01:56:37 +03:00
|
|
|
import Database.ODBC.SQLServer qualified as ODBC
|
|
|
|
import Hasura.Backends.MSSQL.Instances.Types ()
|
2022-02-07 17:11:49 +03:00
|
|
|
import Hasura.Backends.MSSQL.SQL.Error
|
2021-11-26 16:47:12 +03:00
|
|
|
import Hasura.Backends.MSSQL.Types.Internal
|
2021-09-24 01:56:37 +03:00
|
|
|
import Hasura.Base.Error
|
|
|
|
import Hasura.Prelude
|
|
|
|
import Hasura.RQL.Types.Column
|
|
|
|
import Hasura.RQL.Types.Common (OID (..))
|
|
|
|
import Hasura.RQL.Types.Table
|
|
|
|
import Hasura.SQL.Backend
|
2021-06-15 18:05:41 +03:00
|
|
|
|
2021-02-23 20:37:27 +03:00
|
|
|
--------------------------------------------------------------------------------
|
2022-01-03 20:16:24 +03:00
|
|
|
|
|
|
|
-- * Loader
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-10-22 17:49:15 +03:00
|
|
|
loadDBMetadata :: (MonadIO m) => Tx.TxET QErr m (DBTablesMetadata 'MSSQL)
|
|
|
|
loadDBMetadata = do
|
2022-04-21 10:19:37 +03:00
|
|
|
let queryBytes = $(makeRelativeToProject "src-rsr/mssql/mssql_table_metadata.sql" >>= embedFile)
|
2021-09-24 01:56:37 +03:00
|
|
|
odbcQuery :: ODBC.Query = fromString . BSUTF8.toString $ queryBytes
|
2022-02-07 17:11:49 +03:00
|
|
|
sysTablesText <- runIdentity <$> Tx.singleRowQueryE defaultMSSQLTxErrorHandler odbcQuery
|
2021-02-25 21:15:55 +03:00
|
|
|
case Aeson.eitherDecodeStrict (T.encodeUtf8 sysTablesText) of
|
2021-09-24 01:56:37 +03:00
|
|
|
Left e -> throw500 $ T.pack $ "error loading sql server database schema: " <> e
|
2021-02-25 21:15:55 +03:00
|
|
|
Right sysTables -> pure $ HM.fromList $ map transformTable sysTables
|
2021-02-23 20:37:27 +03:00
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2022-01-03 20:16:24 +03:00
|
|
|
|
|
|
|
-- * Local types
|
2021-02-23 20:37:27 +03:00
|
|
|
|
|
|
|
data SysTable = SysTable
|
2021-09-24 01:56:37 +03:00
|
|
|
{ staName :: Text,
|
|
|
|
staObjectId :: Int,
|
|
|
|
staJoinedSysColumn :: [SysColumn],
|
2021-10-01 15:52:19 +03:00
|
|
|
staJoinedSysSchema :: SysSchema,
|
|
|
|
staJoinedSysPrimaryKey :: Maybe SysPrimaryKey
|
2021-09-24 01:56:37 +03:00
|
|
|
}
|
|
|
|
deriving (Show, Generic)
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-05-21 05:46:58 +03:00
|
|
|
instance FromJSON SysTable where
|
2021-02-23 20:37:27 +03:00
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
2021-10-01 15:52:19 +03:00
|
|
|
newtype SysPrimaryKeyColumn = SysPrimaryKeyColumn
|
|
|
|
{spkcName :: Text}
|
|
|
|
deriving (Show, Generic)
|
|
|
|
|
|
|
|
instance FromJSON SysPrimaryKeyColumn where
|
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
|
|
|
data SysPrimaryKey = SysPrimaryKey
|
|
|
|
{ spkName :: Text,
|
|
|
|
spkIndexId :: Int,
|
|
|
|
spkColumns :: NESeq SysPrimaryKeyColumn
|
|
|
|
}
|
|
|
|
deriving (Show, Generic)
|
|
|
|
|
|
|
|
instance FromJSON SysPrimaryKey where
|
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
2021-02-23 20:37:27 +03:00
|
|
|
data SysSchema = SysSchema
|
2021-09-24 01:56:37 +03:00
|
|
|
{ ssName :: Text,
|
|
|
|
ssSchemaId :: Int
|
|
|
|
}
|
|
|
|
deriving (Show, Generic)
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-05-21 05:46:58 +03:00
|
|
|
instance FromJSON SysSchema where
|
2021-02-23 20:37:27 +03:00
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
|
|
|
data SysColumn = SysColumn
|
2021-09-24 01:56:37 +03:00
|
|
|
{ scName :: Text,
|
|
|
|
scColumnId :: Int,
|
|
|
|
scUserTypeId :: Int,
|
|
|
|
scIsNullable :: Bool,
|
|
|
|
scIsIdentity :: Bool,
|
2022-02-03 17:14:33 +03:00
|
|
|
scIsComputed :: Bool,
|
2021-09-24 01:56:37 +03:00
|
|
|
scJoinedSysType :: SysType,
|
|
|
|
scJoinedForeignKeyColumns :: [SysForeignKeyColumn]
|
|
|
|
}
|
|
|
|
deriving (Show, Generic)
|
|
|
|
|
2021-02-23 20:37:27 +03:00
|
|
|
instance FromJSON SysColumn where
|
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
|
|
|
data SysType = SysType
|
2021-09-24 01:56:37 +03:00
|
|
|
{ styName :: Text,
|
|
|
|
stySchemaId :: Int,
|
|
|
|
styUserTypeId :: Int
|
|
|
|
}
|
|
|
|
deriving (Show, Generic)
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-05-21 05:46:58 +03:00
|
|
|
instance FromJSON SysType where
|
2021-02-23 20:37:27 +03:00
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
|
|
|
data SysForeignKeyColumn = SysForeignKeyColumn
|
2021-09-24 01:56:37 +03:00
|
|
|
{ sfkcConstraintObjectId :: Int,
|
|
|
|
sfkcConstraintColumnId :: Int,
|
|
|
|
sfkcParentObjectId :: Int,
|
|
|
|
sfkcParentColumnId :: Int,
|
|
|
|
sfkcReferencedObjectId :: Int,
|
|
|
|
sfkcReferencedColumnId :: Int,
|
|
|
|
sfkcJoinedReferencedTableName :: Text,
|
|
|
|
sfkcJoinedReferencedColumnName :: Text,
|
|
|
|
sfkcJoinedReferencedSysSchema :: SysSchema
|
|
|
|
}
|
|
|
|
deriving (Show, Generic)
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-05-21 05:46:58 +03:00
|
|
|
instance FromJSON SysForeignKeyColumn where
|
2021-02-23 20:37:27 +03:00
|
|
|
parseJSON = genericParseJSON hasuraJSON
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
2022-01-03 20:16:24 +03:00
|
|
|
|
|
|
|
-- * Transform
|
2021-02-23 20:37:27 +03:00
|
|
|
|
|
|
|
transformTable :: SysTable -> (TableName, DBTableMetadata 'MSSQL)
|
|
|
|
transformTable tableInfo =
|
2022-02-24 11:13:19 +03:00
|
|
|
let schemaName = SchemaName $ ssName $ staJoinedSysSchema tableInfo
|
2021-09-24 01:56:37 +03:00
|
|
|
tableName = TableName (staName tableInfo) schemaName
|
|
|
|
tableOID = OID $ staObjectId tableInfo
|
2021-03-05 13:52:40 +03:00
|
|
|
(columns, foreignKeys) = unzip $ transformColumn <$> staJoinedSysColumn tableInfo
|
2021-03-31 16:48:36 +03:00
|
|
|
foreignKeysMetadata = HS.fromList $ map ForeignKeyMetadata $ coalesceKeys $ concat foreignKeys
|
2021-10-01 15:52:19 +03:00
|
|
|
primaryKey = transformPrimaryKey <$> staJoinedSysPrimaryKey tableInfo
|
2021-09-24 01:56:37 +03:00
|
|
|
identityColumns =
|
|
|
|
map (ColumnName . scName) $
|
|
|
|
filter scIsIdentity $ staJoinedSysColumn tableInfo
|
|
|
|
in ( tableName,
|
|
|
|
DBTableMetadata
|
|
|
|
tableOID
|
|
|
|
columns
|
2021-10-01 15:52:19 +03:00
|
|
|
primaryKey
|
2021-09-24 01:56:37 +03:00
|
|
|
HS.empty -- no unique constraints?
|
|
|
|
foreignKeysMetadata
|
|
|
|
Nothing -- no views, only tables
|
|
|
|
Nothing -- no description
|
|
|
|
identityColumns
|
|
|
|
)
|
|
|
|
|
|
|
|
transformColumn ::
|
|
|
|
SysColumn ->
|
|
|
|
(RawColumnInfo 'MSSQL, [ForeignKey 'MSSQL])
|
2022-06-08 02:24:42 +03:00
|
|
|
transformColumn sysCol =
|
|
|
|
let rciName = ColumnName $ scName sysCol
|
|
|
|
rciPosition = scColumnId sysCol
|
2021-09-24 01:56:37 +03:00
|
|
|
|
2022-06-08 02:24:42 +03:00
|
|
|
rciIsNullable = scIsNullable sysCol
|
2022-01-19 11:37:50 +03:00
|
|
|
rciDescription = Nothing
|
2022-06-08 02:24:42 +03:00
|
|
|
rciType = parseScalarType $ styName $ scJoinedSysType sysCol
|
2021-09-24 01:56:37 +03:00
|
|
|
foreignKeys =
|
2022-06-08 02:24:42 +03:00
|
|
|
scJoinedForeignKeyColumns sysCol <&> \foreignKeyColumn ->
|
2021-10-01 15:52:19 +03:00
|
|
|
let _fkConstraint = Constraint "fk_mssql" $ OID $ sfkcConstraintObjectId foreignKeyColumn
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2022-02-24 11:13:19 +03:00
|
|
|
schemaName = SchemaName $ ssName $ sfkcJoinedReferencedSysSchema foreignKeyColumn
|
2021-09-24 01:56:37 +03:00
|
|
|
_fkForeignTable = TableName (sfkcJoinedReferencedTableName foreignKeyColumn) schemaName
|
2022-05-10 18:43:24 +03:00
|
|
|
_fkColumnMapping = NEHashMap.singleton rciName $ ColumnName $ sfkcJoinedReferencedColumnName foreignKeyColumn
|
2021-09-24 01:56:37 +03:00
|
|
|
in ForeignKey {..}
|
2021-12-01 14:33:14 +03:00
|
|
|
|
2022-06-08 02:24:42 +03:00
|
|
|
colIsImmutable = scIsComputed sysCol || scIsIdentity sysCol
|
2022-02-03 17:14:33 +03:00
|
|
|
rciMutability = ColumnMutability {_cmIsInsertable = not colIsImmutable, _cmIsUpdatable = not colIsImmutable}
|
2021-09-24 01:56:37 +03:00
|
|
|
in (RawColumnInfo {..}, foreignKeys)
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-10-01 15:52:19 +03:00
|
|
|
transformPrimaryKey :: SysPrimaryKey -> PrimaryKey 'MSSQL (Column 'MSSQL)
|
|
|
|
transformPrimaryKey (SysPrimaryKey {..}) =
|
|
|
|
let constraint = Constraint spkName $ OID spkIndexId
|
|
|
|
columns = (ColumnName . spkcName) <$> spkColumns
|
|
|
|
in PrimaryKey constraint columns
|
|
|
|
|
2021-02-23 20:37:27 +03:00
|
|
|
--------------------------------------------------------------------------------
|
2022-01-03 20:16:24 +03:00
|
|
|
|
|
|
|
-- * Helpers
|
2021-02-23 20:37:27 +03:00
|
|
|
|
2021-03-31 16:48:36 +03:00
|
|
|
coalesceKeys :: [ForeignKey 'MSSQL] -> [ForeignKey 'MSSQL]
|
|
|
|
coalesceKeys = HM.elems . foldl' coalesce HM.empty
|
|
|
|
where
|
|
|
|
coalesce mapping fk@(ForeignKey constraint tableName _) = HM.insertWith combine (constraint, tableName) fk mapping
|
2022-05-10 18:43:24 +03:00
|
|
|
combine oldFK newFK = oldFK {_fkColumnMapping = (NEHashMap.union `on` _fkColumnMapping) oldFK newFK}
|
2021-02-23 20:37:27 +03:00
|
|
|
|
|
|
|
parseScalarType :: Text -> ScalarType
|
|
|
|
parseScalarType = \case
|
2021-09-24 01:56:37 +03:00
|
|
|
"char" -> CharType
|
|
|
|
"numeric" -> NumericType
|
|
|
|
"decimal" -> DecimalType
|
|
|
|
"money" -> DecimalType
|
|
|
|
"smallmoney" -> DecimalType
|
|
|
|
"int" -> IntegerType
|
|
|
|
"smallint" -> SmallintType
|
|
|
|
"float" -> FloatType
|
|
|
|
"real" -> RealType
|
|
|
|
"date" -> DateType
|
|
|
|
"time" -> Ss_time2Type
|
|
|
|
"varchar" -> VarcharType
|
|
|
|
"nchar" -> WcharType
|
|
|
|
"nvarchar" -> WvarcharType
|
|
|
|
"ntext" -> WtextType
|
|
|
|
"timestamp" -> TimestampType
|
|
|
|
"text" -> TextType
|
|
|
|
"binary" -> BinaryType
|
|
|
|
"bigint" -> BigintType
|
|
|
|
"tinyint" -> TinyintType
|
|
|
|
"varbinary" -> VarbinaryType
|
|
|
|
"bit" -> BitType
|
2021-02-23 20:37:27 +03:00
|
|
|
"uniqueidentifier" -> GuidType
|
2021-09-24 01:56:37 +03:00
|
|
|
"geography" -> GeographyType
|
|
|
|
"geometry" -> GeometryType
|
|
|
|
t -> UnknownType t
|