mirror of
https://github.com/hasura/graphql-engine.git
synced 2024-12-17 20:41:49 +03:00
e3c2bf53a5
### Description This PR is a first step in a series of cleanups of action relationships. This first step does not contain any behavioral change, and it simply reorganizes / prunes / rearranges / documents the code. Mainly: - it divides some files in RQL.Types between metadata types, schema cache types, execution types; - it renames some types for consistency; - it minimizes exports and prunes unnecessary types; - it moves some types in places where they make more sense; - it replaces uses of `DMap BackendTag` with `BackendMap`. Most of the "movement" within files re-organizes declarations in a "top-down" fashion, by moving all TH splices to the end of the file, which avoids order or declarations mattering. ### Optional list types One main type change this PR makes is a replacement of variant list types in `CustomTypes.hs`; we had `Maybe [a]`, or sometimes `Maybe (NonEmpty a)`. This PR harmonizes all of them to `[a]`, as most of the code would use them as such, by doing `fromMaybe []` or `maybe [] toList`. PR-URL: https://github.com/hasura/graphql-engine-mono/pull/4613 GitOrigin-RevId: bc624e10df587eba862ff27a5e8021b32d0d78a2
180 lines
6.9 KiB
Haskell
180 lines
6.9 KiB
Haskell
{-# LANGUAGE DuplicateRecordFields #-}
|
|
|
|
module Hasura.Backends.BigQuery.DDL.Source
|
|
( resolveSource,
|
|
postDropSourceHook,
|
|
resolveSourceConfig,
|
|
restTypeToScalarType,
|
|
)
|
|
where
|
|
|
|
import Data.Aeson qualified as J
|
|
import Data.ByteString.Lazy qualified as L
|
|
import Data.Environment qualified as Env
|
|
import Data.HashMap.Strict.Extended qualified as HM
|
|
import Data.Int qualified as Int
|
|
import Data.Text qualified as T
|
|
import Data.Text.Encoding qualified as T
|
|
import Data.Time.Clock.System
|
|
import Hasura.Backends.BigQuery.Connection
|
|
import Hasura.Backends.BigQuery.Meta
|
|
import Hasura.Backends.BigQuery.Source
|
|
import Hasura.Backends.BigQuery.Types
|
|
import Hasura.Base.Error
|
|
import Hasura.Prelude
|
|
import Hasura.RQL.Types.Backend (BackendConfig)
|
|
import Hasura.RQL.Types.Column
|
|
import Hasura.RQL.Types.Common
|
|
import Hasura.RQL.Types.Source
|
|
import Hasura.RQL.Types.SourceCustomization
|
|
import Hasura.RQL.Types.Table
|
|
import Hasura.SQL.Backend
|
|
|
|
defaultGlobalSelectLimit :: Int.Int64
|
|
defaultGlobalSelectLimit = 1000
|
|
|
|
defaultRetryLimit :: Int
|
|
defaultRetryLimit = 5
|
|
|
|
defaultRetryBaseDelay :: Microseconds
|
|
defaultRetryBaseDelay = 500000
|
|
|
|
resolveSourceConfig ::
|
|
MonadIO m =>
|
|
SourceName ->
|
|
BigQueryConnSourceConfig ->
|
|
BackendSourceKind 'BigQuery ->
|
|
BackendConfig 'BigQuery ->
|
|
Env.Environment ->
|
|
m (Either QErr BigQuerySourceConfig)
|
|
resolveSourceConfig _name BigQueryConnSourceConfig {..} _backendKind _backendConfig env = runExceptT $ do
|
|
eSA <- resolveConfigurationJson env _cscServiceAccount
|
|
case eSA of
|
|
Left e -> throw400 Unexpected $ T.pack e
|
|
Right serviceAccount -> do
|
|
projectId <- resolveConfigurationInput env _cscProjectId
|
|
retryOptions <- do
|
|
numRetries <-
|
|
resolveConfigurationInput env `mapM` _cscRetryLimit >>= \case
|
|
Nothing -> pure defaultRetryLimit
|
|
Just v -> readNonNegative v "retry limit"
|
|
if numRetries == 0
|
|
then pure Nothing
|
|
else do
|
|
let _retryNumRetries = numRetries
|
|
_retryBaseDelay <-
|
|
resolveConfigurationInput env `mapM` _cscRetryBaseDelay >>= \case
|
|
Nothing -> pure defaultRetryBaseDelay
|
|
Just v -> fromInteger <$> readNonNegative v "retry base delay"
|
|
pure $ Just RetryOptions {..}
|
|
_scConnection <- initConnection serviceAccount projectId retryOptions
|
|
_scDatasets <- resolveConfigurationInputs env _cscDatasets
|
|
_scGlobalSelectLimit <-
|
|
resolveConfigurationInput env `mapM` _cscGlobalSelectLimit >>= \case
|
|
Nothing -> pure defaultGlobalSelectLimit
|
|
Just v -> readNonNegative v "global select limit"
|
|
pure BigQuerySourceConfig {..}
|
|
|
|
readNonNegative :: (MonadError QErr m, Num a, Ord a, J.FromJSON a, Read a) => Text -> Text -> m a
|
|
readNonNegative i paramName =
|
|
-- This works around the inconsistency between JSON and
|
|
-- environment variables. The config handling module should be
|
|
-- reworked to handle non-text values better.
|
|
case readMaybe (T.unpack i) <|> J.decode (L.fromStrict (T.encodeUtf8 i)) of
|
|
Nothing -> throw400 Unexpected $ "Need a non-negative integer for " <> paramName
|
|
Just i' -> do
|
|
when (i' < 0) $ throw400 Unexpected $ "Need the integer for the " <> paramName <> " to be non-negative"
|
|
pure i'
|
|
|
|
resolveSource ::
|
|
(MonadIO m) =>
|
|
BigQuerySourceConfig ->
|
|
SourceTypeCustomization ->
|
|
m (Either QErr (ResolvedSource 'BigQuery))
|
|
resolveSource sourceConfig customization =
|
|
runExceptT $ do
|
|
tables <- getTables sourceConfig
|
|
routines <- getRoutines sourceConfig
|
|
let result = (,) <$> tables <*> routines
|
|
case result of
|
|
Left err ->
|
|
throw400 Unexpected $
|
|
"unexpected exception while connecting to database: " <> tshow err
|
|
Right (restTables, restRoutines) -> do
|
|
seconds <- liftIO $ fmap systemSeconds getSystemTime
|
|
let functions = HM.groupOn (routineReferenceToFunctionName . routineReference) restRoutines
|
|
pure
|
|
( ResolvedSource
|
|
{ _rsConfig = sourceConfig,
|
|
_rsCustomization = customization,
|
|
_rsTables =
|
|
HM.fromList
|
|
[ ( restTableReferenceToTableName tableReference,
|
|
DBTableMetadata
|
|
{ _ptmiOid = OID (fromIntegral seconds + index :: Int), -- TODO: The seconds are used for uniqueness. BigQuery doesn't support a "stable" ID for a table.
|
|
_ptmiColumns =
|
|
[ RawColumnInfo
|
|
{ rciName = ColumnName name,
|
|
rciPosition = position,
|
|
rciType = restTypeToScalarType type',
|
|
rciIsNullable =
|
|
case mode of
|
|
Nullable -> True
|
|
_ -> False,
|
|
rciDescription = Nothing,
|
|
rciMutability = ColumnMutability {_cmIsInsertable = True, _cmIsUpdatable = True}
|
|
}
|
|
| (position, RestFieldSchema {name, type', mode}) <-
|
|
zip [1 ..] fields -- TODO: Same trouble as Oid above.
|
|
],
|
|
_ptmiPrimaryKey = Nothing,
|
|
_ptmiUniqueConstraints = mempty,
|
|
_ptmiForeignKeys = mempty,
|
|
_ptmiViewInfo = Just $ ViewInfo False False False,
|
|
_ptmiDescription = Nothing,
|
|
_ptmiExtraTableMetadata = ()
|
|
}
|
|
)
|
|
| (index, RestTable {tableReference, schema}) <-
|
|
zip [0 ..] restTables,
|
|
let RestTableSchema fields = schema
|
|
],
|
|
_rsFunctions = functions,
|
|
_rsScalars = mempty
|
|
}
|
|
)
|
|
|
|
restTypeToScalarType :: RestType -> ScalarType
|
|
restTypeToScalarType =
|
|
\case
|
|
STRING -> StringScalarType
|
|
BYTES -> BytesScalarType
|
|
INTEGER -> IntegerScalarType
|
|
FLOAT -> FloatScalarType
|
|
BOOL -> BoolScalarType
|
|
TIMESTAMP -> TimestampScalarType
|
|
DATE -> DateScalarType
|
|
TIME -> TimeScalarType
|
|
DATETIME -> DatetimeScalarType
|
|
GEOGRAPHY -> GeographyScalarType
|
|
STRUCT -> StructScalarType
|
|
BIGDECIMAL -> BigDecimalScalarType
|
|
DECIMAL -> DecimalScalarType
|
|
|
|
-- Hierarchy: Project / Dataset / Table
|
|
-- see <https://cloud.google.com/bigquery/docs/datasets-intro>
|
|
restTableReferenceToTableName :: RestTableReference -> TableName
|
|
restTableReferenceToTableName RestTableReference {..} =
|
|
TableName {tableName = tableId, tableNameSchema = datasetId}
|
|
|
|
-- We ignore project id and push that requirement up to the top to
|
|
-- the data source level.
|
|
|
|
postDropSourceHook ::
|
|
(MonadIO m) =>
|
|
BigQuerySourceConfig ->
|
|
m ()
|
|
postDropSourceHook _ =
|
|
-- On BigQuery we don't keep connections open.
|
|
pure ()
|