2022-09-02 20:50:09 +03:00
|
|
|
-- | Metadata API Actions relating to Source Kinds
|
2022-08-18 01:13:32 +03:00
|
|
|
module Hasura.RQL.DDL.SourceKinds
|
|
|
|
( -- * List Source Kinds
|
|
|
|
ListSourceKinds (..),
|
|
|
|
runListSourceKinds,
|
2022-11-18 07:17:50 +03:00
|
|
|
agentSourceKinds,
|
2022-08-18 01:13:32 +03:00
|
|
|
|
|
|
|
-- * Source Kind Info
|
|
|
|
SourceKindInfo (..),
|
|
|
|
SourceType (..),
|
2022-09-02 20:50:09 +03:00
|
|
|
|
|
|
|
-- * List Capabilities
|
|
|
|
GetSourceKindCapabilities (..),
|
|
|
|
runGetSourceKindCapabilities,
|
2022-08-18 01:13:32 +03:00
|
|
|
)
|
|
|
|
where
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
2022-11-18 07:17:50 +03:00
|
|
|
import Data.Aeson (FromJSON, ToJSON, (.:), (.:?), (.=))
|
2022-08-18 01:13:32 +03:00
|
|
|
import Data.Aeson qualified as Aeson
|
2022-09-02 20:50:09 +03:00
|
|
|
import Data.HashMap.Strict qualified as HashMap
|
2022-08-18 01:13:32 +03:00
|
|
|
import Data.HashMap.Strict.InsOrd qualified as InsOrdHashMap
|
|
|
|
import Data.Text.Extended (ToTxt (..))
|
2022-09-02 20:50:09 +03:00
|
|
|
import Data.Text.Extended qualified as Text.E
|
|
|
|
import Data.Text.NonEmpty (NonEmptyText)
|
2022-08-18 01:13:32 +03:00
|
|
|
import Data.Text.NonEmpty qualified as NE.Text
|
|
|
|
import Hasura.Backends.DataConnector.Adapter.Types qualified as DC.Types
|
2022-09-02 20:50:09 +03:00
|
|
|
import Hasura.Base.Error qualified as Error
|
2022-08-18 01:13:32 +03:00
|
|
|
import Hasura.EncJSON (EncJSON)
|
|
|
|
import Hasura.EncJSON qualified as EncJSON
|
|
|
|
import Hasura.Prelude
|
|
|
|
import Hasura.RQL.Types.Metadata qualified as Metadata
|
2022-09-02 20:50:09 +03:00
|
|
|
import Hasura.RQL.Types.SchemaCache qualified as SchemaCache
|
2022-10-18 07:17:57 +03:00
|
|
|
import Hasura.SQL.AnyBackend qualified as AB
|
2022-08-18 01:13:32 +03:00
|
|
|
import Hasura.SQL.Backend qualified as Backend
|
|
|
|
import Hasura.SQL.BackendMap qualified as BackendMap
|
2022-10-18 07:17:57 +03:00
|
|
|
import Language.GraphQL.Draft.Syntax qualified as GQL
|
2022-08-18 01:13:32 +03:00
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
data ListSourceKinds = ListSourceKinds
|
|
|
|
|
|
|
|
instance FromJSON ListSourceKinds where
|
|
|
|
parseJSON = Aeson.withObject "ListSourceKinds" (const $ pure ListSourceKinds)
|
|
|
|
|
|
|
|
instance ToJSON ListSourceKinds where
|
|
|
|
toJSON ListSourceKinds = Aeson.object []
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
data SourceKindInfo = SourceKindInfo
|
|
|
|
{ _skiSourceKind :: Text,
|
2022-11-18 07:17:50 +03:00
|
|
|
_skiDisplayName :: Maybe Text,
|
|
|
|
_skiReleaseName :: Maybe Text,
|
2023-01-19 08:18:44 +03:00
|
|
|
_skiBuiltin :: SourceType,
|
|
|
|
_skiAvailable :: Bool
|
2022-08-18 01:13:32 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
instance FromJSON SourceKindInfo where
|
|
|
|
parseJSON = Aeson.withObject "SourceKindInfo" \o -> do
|
|
|
|
_skiSourceKind <- o .: "kind"
|
2022-11-18 07:17:50 +03:00
|
|
|
_skiDisplayName <- o .:? "display_name"
|
|
|
|
_skiReleaseName <- o .:? "release_name"
|
2022-08-18 01:13:32 +03:00
|
|
|
_skiBuiltin <- o .: "builtin"
|
2023-01-19 08:18:44 +03:00
|
|
|
_skiAvailable <- o .: "available"
|
2022-08-18 01:13:32 +03:00
|
|
|
pure SourceKindInfo {..}
|
|
|
|
|
|
|
|
instance ToJSON SourceKindInfo where
|
2022-11-18 07:17:50 +03:00
|
|
|
toJSON SourceKindInfo {..} =
|
|
|
|
Aeson.object $
|
|
|
|
[ "kind" .= _skiSourceKind,
|
2023-01-19 08:18:44 +03:00
|
|
|
"builtin" .= _skiBuiltin,
|
|
|
|
"available" .= _skiAvailable
|
2022-11-18 07:17:50 +03:00
|
|
|
]
|
2023-02-03 04:45:36 +03:00
|
|
|
++ ["display_name" .= _skiDisplayName | has _skiDisplayName]
|
|
|
|
++ ["release_name" .= _skiReleaseName | has _skiReleaseName]
|
2022-11-18 07:17:50 +03:00
|
|
|
where
|
2023-02-03 04:45:36 +03:00
|
|
|
has :: Maybe Text -> Bool
|
|
|
|
has x = not $ isNothing x || x == Just ""
|
2022-08-18 01:13:32 +03:00
|
|
|
|
|
|
|
data SourceType = Builtin | Agent
|
|
|
|
|
|
|
|
instance FromJSON SourceType where
|
|
|
|
parseJSON = Aeson.withBool "source type" \case
|
|
|
|
True -> pure Builtin
|
|
|
|
False -> pure Agent
|
|
|
|
|
|
|
|
instance ToJSON SourceType where
|
|
|
|
toJSON Builtin = Aeson.Bool True
|
|
|
|
toJSON Agent = Aeson.Bool False
|
|
|
|
|
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
agentSourceKinds :: (Metadata.MetadataM m) => m [SourceKindInfo]
|
|
|
|
agentSourceKinds = do
|
|
|
|
agentsM <- BackendMap.lookup @'Backend.DataConnector . Metadata._metaBackendConfigs <$> Metadata.getMetadata
|
|
|
|
case agentsM of
|
|
|
|
Nothing -> pure []
|
|
|
|
Just (Metadata.BackendConfigWrapper agents) ->
|
2022-11-18 07:17:50 +03:00
|
|
|
pure $ fmap mkAgentSource $ InsOrdHashMap.toList agents
|
|
|
|
|
|
|
|
mkAgentSource :: (DC.Types.DataConnectorName, DC.Types.DataConnectorOptions) -> SourceKindInfo
|
|
|
|
mkAgentSource (dcName, DC.Types.DataConnectorOptions {_dcoDisplayName}) =
|
|
|
|
SourceKindInfo
|
|
|
|
{ _skiSourceKind = skiKind,
|
|
|
|
_skiDisplayName = _dcoDisplayName,
|
|
|
|
_skiReleaseName = Nothing,
|
2023-01-19 08:18:44 +03:00
|
|
|
_skiBuiltin = Agent,
|
|
|
|
_skiAvailable = True
|
2022-11-18 07:17:50 +03:00
|
|
|
}
|
|
|
|
where
|
|
|
|
skiKind = GQL.unName (DC.Types.unDataConnectorName dcName)
|
2022-08-18 01:13:32 +03:00
|
|
|
|
|
|
|
mkNativeSource :: Backend.BackendType -> Maybe SourceKindInfo
|
|
|
|
mkNativeSource = \case
|
|
|
|
Backend.DataConnector -> Nothing
|
2022-11-18 07:17:50 +03:00
|
|
|
b ->
|
|
|
|
Just
|
|
|
|
SourceKindInfo
|
|
|
|
{ _skiSourceKind = fromMaybe (toTxt b) (Backend.backendShortName b),
|
|
|
|
_skiBuiltin = Builtin,
|
|
|
|
_skiDisplayName = Nothing,
|
2023-01-19 08:18:44 +03:00
|
|
|
_skiReleaseName = Nothing,
|
|
|
|
_skiAvailable = True
|
2022-11-18 07:17:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
runListSourceKinds'' ::
|
|
|
|
forall m.
|
|
|
|
( Metadata.MetadataM m,
|
|
|
|
MonadError Error.QErr m,
|
|
|
|
SchemaCache.CacheRM m
|
|
|
|
) =>
|
|
|
|
ListSourceKinds ->
|
|
|
|
m [SourceKindInfo]
|
|
|
|
runListSourceKinds'' x = do
|
|
|
|
sks <- runListSourceKinds' x
|
|
|
|
mapM setNames sks
|
|
|
|
where
|
|
|
|
suffixKey :: Text -> Text -> Text
|
|
|
|
suffixKey a b = b <> " (" <> a <> ")"
|
|
|
|
|
|
|
|
setNames :: SourceKindInfo -> m SourceKindInfo
|
2023-01-19 08:18:44 +03:00
|
|
|
setNames ski@SourceKindInfo {_skiSourceKind, _skiDisplayName} =
|
|
|
|
-- If there are issues fetching the capabilities for an agent, then list it as unavailable.
|
|
|
|
flip catchError (const $ pure $ ski {_skiAvailable = False}) do
|
|
|
|
ci <- getCapabilities ski
|
|
|
|
-- Prefer metadata, then capabilities, then source-kind key
|
|
|
|
pure
|
|
|
|
ski
|
|
|
|
{ _skiReleaseName = DC.Types._dciReleaseName =<< ci,
|
|
|
|
_skiDisplayName =
|
|
|
|
(suffixKey _skiSourceKind <$> _skiDisplayName) -- Question: Should we suffix the key if the user explicitly sets a name?
|
|
|
|
<|> (suffixKey _skiSourceKind <$> (DC.Types._dciDisplayName =<< ci))
|
|
|
|
<|> Just _skiSourceKind
|
|
|
|
}
|
2022-11-18 07:17:50 +03:00
|
|
|
|
|
|
|
getCapabilities :: SourceKindInfo -> m (Maybe DC.Types.DataConnectorInfo)
|
|
|
|
getCapabilities SourceKindInfo {_skiSourceKind, _skiBuiltin} = case (_skiBuiltin, NE.Text.mkNonEmptyText _skiSourceKind) of
|
|
|
|
(Builtin, _) -> pure Nothing
|
|
|
|
(Agent, Nothing) -> pure Nothing
|
|
|
|
(Agent, Just nesk) -> Just <$> runGetSourceKindCapabilities' (GetSourceKindCapabilities nesk)
|
|
|
|
|
|
|
|
runListSourceKinds ::
|
|
|
|
( MonadError Error.QErr m,
|
|
|
|
Metadata.MetadataM m,
|
|
|
|
SchemaCache.CacheRM m
|
|
|
|
) =>
|
|
|
|
ListSourceKinds ->
|
|
|
|
m EncJSON
|
|
|
|
runListSourceKinds x = do
|
|
|
|
sks <- runListSourceKinds'' x
|
|
|
|
pure $ EncJSON.encJFromJValue $ Aeson.object ["sources" .= sks]
|
|
|
|
|
|
|
|
-- TODO: This kind of direct encoding seems unsafe.
|
|
|
|
-- Perhaps we chould have these functions defined as ToJSON j => ... -> j
|
|
|
|
-- Then wrap them with an encoder at invocation?
|
|
|
|
-- Or even existentailly quantify them over the ToJSON class?
|
|
|
|
runListSourceKinds' :: Metadata.MetadataM m => ListSourceKinds -> m [SourceKindInfo]
|
|
|
|
runListSourceKinds' ListSourceKinds = do
|
2022-11-15 03:39:38 +03:00
|
|
|
let builtins = mapMaybe mkNativeSource (filter (/= Backend.DataConnector) Backend.supportedBackends)
|
2022-08-18 01:13:32 +03:00
|
|
|
agents <- agentSourceKinds
|
2022-11-15 03:39:38 +03:00
|
|
|
pure (builtins <> agents)
|
|
|
|
|
2022-09-02 20:50:09 +03:00
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
newtype GetSourceKindCapabilities = GetSourceKindCapabilities {_gskcKind :: NonEmptyText}
|
|
|
|
|
|
|
|
instance FromJSON GetSourceKindCapabilities where
|
|
|
|
parseJSON = Aeson.withObject "GetSourceKindCapabilities" \o -> do
|
|
|
|
_gskcKind <- o .: "name"
|
|
|
|
pure $ GetSourceKindCapabilities {..}
|
|
|
|
|
|
|
|
-- | List Backend Capabilities. Currently this only supports Data Connector Backends.
|
|
|
|
runGetSourceKindCapabilities ::
|
2022-11-18 07:17:50 +03:00
|
|
|
( MonadError Error.QErr m,
|
2022-09-02 20:50:09 +03:00
|
|
|
SchemaCache.CacheRM m
|
|
|
|
) =>
|
|
|
|
GetSourceKindCapabilities ->
|
|
|
|
m EncJSON
|
2022-11-18 07:17:50 +03:00
|
|
|
runGetSourceKindCapabilities x = EncJSON.encJFromJValue <$> runGetSourceKindCapabilities' x
|
|
|
|
|
|
|
|
-- Main implementation of runGetSourceKindCapabilities that actually returns the DataConnectorInfo
|
|
|
|
-- and defers json encoding to `runGetSourceKindCapabilities`. This allows reuse and ensures a
|
|
|
|
-- correct assembly of DataConnectorInfo
|
|
|
|
runGetSourceKindCapabilities' ::
|
|
|
|
( MonadError Error.QErr m,
|
|
|
|
SchemaCache.CacheRM m
|
|
|
|
) =>
|
|
|
|
GetSourceKindCapabilities ->
|
|
|
|
m DC.Types.DataConnectorInfo
|
|
|
|
runGetSourceKindCapabilities' GetSourceKindCapabilities {..} = do
|
2022-10-18 07:17:57 +03:00
|
|
|
case AB.backendSourceKindFromText $ NE.Text.unNonEmptyText _gskcKind of
|
|
|
|
Just backendSourceKind ->
|
|
|
|
case AB.unpackAnyBackend @'Backend.DataConnector backendSourceKind of
|
|
|
|
Just (Backend.DataConnectorKind dataConnectorName) -> do
|
|
|
|
backendCache <- fmap SchemaCache.scBackendCache $ SchemaCache.askSchemaCache
|
|
|
|
let capabilitiesMap = maybe mempty SchemaCache.unBackendInfoWrapper $ BackendMap.lookup @'Backend.DataConnector backendCache
|
2022-11-18 07:17:50 +03:00
|
|
|
HashMap.lookup dataConnectorName capabilitiesMap
|
|
|
|
`onNothing` Error.throw400 Error.DataConnectorError ("Source Kind " <> Text.E.toTxt dataConnectorName <> " was not found")
|
2022-10-18 07:17:57 +03:00
|
|
|
Nothing ->
|
|
|
|
-- Must be a native backend
|
|
|
|
Error.throw400 Error.DataConnectorError (Text.E.toTxt _gskcKind <> " does not support Capabilities")
|
|
|
|
Nothing ->
|
|
|
|
Error.throw400 Error.DataConnectorError ("Source Kind " <> Text.E.toTxt _gskcKind <> " was not found")
|