{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE NamedFieldPuns #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeOperators #-}
module Data.Morpheus.Validation.Query.Selection
( validateSelectionSet
) where
import Data.Morpheus.Error.Selection (cannotQueryField, duplicateQuerySelections,
hasNoSubfields, subfieldsNotSelected)
import Data.Morpheus.Error.Variable (unknownType)
import Data.Morpheus.Types.Internal.AST.Operation (ValidVariables)
import Data.Morpheus.Types.Internal.AST.RawSelection (Fragment (..), FragmentLib, RawSelection (..),
RawSelection' (..), RawSelectionSet)
import Data.Morpheus.Types.Internal.AST.Selection (Selection (..), SelectionRec (..), SelectionSet)
import Data.Morpheus.Types.Internal.Base (EnhancedKey (..))
import Data.Morpheus.Types.Internal.Data (DataField (..), DataFullType (..), DataObject,
DataTyCon (..), DataTypeLib (..), TypeAlias (..),
allDataTypes)
import Data.Morpheus.Types.Internal.Validation (Validation)
import Data.Morpheus.Validation.Internal.Utils (checkNameCollision, lookupType)
import Data.Morpheus.Validation.Query.Arguments (validateArguments)
import Data.Morpheus.Validation.Query.Fragment (castFragmentType, resolveSpread)
import Data.Morpheus.Validation.Query.Utils.Selection (lookupFieldAsSelectionSet, lookupSelectionField,
lookupUnionTypes)
import Data.Text (Text)
checkDuplicatesOn :: DataObject -> SelectionSet -> Validation SelectionSet
checkDuplicatesOn DataTyCon {typeName = name'} keys = checkNameCollision enhancedKeys selError >> pure keys
where
selError = duplicateQuerySelections name'
enhancedKeys = map selToKey keys
selToKey (key', Selection {selectionPosition = position'}) = EnhancedKey key' position'
clusterUnionSelection ::
FragmentLib -> Text -> [DataObject] -> (Text, RawSelection) -> Validation ([Fragment], SelectionSet)
clusterUnionSelection fragments type' possibleTypes' = splitFrag
where
packFragment fragment = return ([fragment], [])
typeNames = map typeName possibleTypes'
splitFrag :: (Text, RawSelection) -> Validation ([Fragment], SelectionSet)
splitFrag (_, Spread ref) = resolveSpread fragments typeNames ref >>= packFragment
splitFrag ("__typename", RawSelectionField RawSelection' {rawSelectionPosition = position'}) =
return
( []
, [ ( "__typename"
, Selection {selectionRec = SelectionField, selectionArguments = [], selectionPosition = position'})
])
splitFrag (key, RawSelectionSet RawSelection' {rawSelectionPosition}) =
Left $ cannotQueryField key type' rawSelectionPosition
splitFrag (key, RawSelectionField RawSelection' {rawSelectionPosition}) =
Left $ cannotQueryField key type' rawSelectionPosition
splitFrag (key', RawAlias {rawAliasPosition = position'}) = Left $ cannotQueryField key' type' position'
splitFrag (_, InlineFragment fragment') =
castFragmentType Nothing (fragmentPosition fragment') typeNames fragment' >>= packFragment
categorizeTypes :: [DataObject] -> [Fragment] -> [(DataObject, [Fragment])]
categorizeTypes types fragments = filter notEmpty $ map categorizeType types
where
notEmpty = (0 /=) . length . snd
categorizeType :: DataObject -> (DataObject, [Fragment])
categorizeType datatype = (datatype, filter matches fragments)
where
matches fragment = fragmentType fragment == typeName datatype
flatTuple :: [([a], [b])] -> ([a], [b])
flatTuple list' = (concatMap fst list', concatMap snd list')
validateSelectionSet ::
DataTypeLib -> FragmentLib -> Text -> ValidVariables -> DataObject -> RawSelectionSet -> Validation SelectionSet
validateSelectionSet lib fragments' operatorName variables = __validate
where
__validate dataType'@DataTyCon {typeName = typeName'} selectionSet' =
concat <$> mapM validateSelection selectionSet' >>= checkDuplicatesOn dataType'
where
validateFragment Fragment {fragmentSelection = selection'} = __validate dataType' selection'
getValidationData key RawSelection' {rawSelectionArguments, rawSelectionPosition} = do
selectionField <- lookupSelectionField rawSelectionPosition key dataType'
arguments <-
validateArguments
lib
operatorName
variables
(key, selectionField)
rawSelectionPosition
rawSelectionArguments
fieldDataType <-
lookupType
(unknownType (aliasTyCon $fieldType selectionField) rawSelectionPosition)
(allDataTypes lib)
(aliasTyCon $ fieldType selectionField)
return (selectionField, fieldDataType, arguments)
validateSelection :: (Text, RawSelection) -> Validation SelectionSet
validateSelection (key', RawAlias {rawAliasSelection = rawSelection', rawAliasPosition = position'}) =
fmap processSingleSelection <$> validateSelection rawSelection'
where
processSingleSelection (selKey', selection') =
( key'
, selection'
{ selectionRec = SelectionAlias {aliasFieldName = selKey', aliasSelection = selectionRec selection'}
, selectionPosition = position'
})
validateSelection (key', RawSelectionSet fullRawSelection'@RawSelection' { rawSelectionRec = rawSelectors
, rawSelectionPosition = position'
}) = do
(dataField, dataType, arguments) <- getValidationData key' fullRawSelection'
case dataType of
Union _ -> do
(categories, __typename) <- clusterTypes
mapM (validateCluster __typename) categories >>= returnSelection arguments . UnionSelection
where clusterTypes = do
unionTypes <- lookupUnionTypes position' key' lib dataField
(spreads, __typename) <-
flatTuple <$> mapM (clusterUnionSelection fragments' typeName' unionTypes) rawSelectors
return (categorizeTypes unionTypes spreads, __typename)
validateCluster :: SelectionSet -> (DataObject, [Fragment]) -> Validation (Text, SelectionSet)
validateCluster sysSelection' (type', frags') = do
selection' <- __validate type' (concatMap fragmentSelection frags')
return (typeName type', sysSelection' ++ selection')
OutputObject _ -> do
fieldType' <- lookupFieldAsSelectionSet position' key' lib dataField
__validate fieldType' rawSelectors >>= returnSelection arguments . SelectionSet
_ -> Left $ hasNoSubfields key' (aliasTyCon $fieldType dataField) position'
where
returnSelection arguments' selection' =
pure
[ ( key'
, Selection
{selectionArguments = arguments', selectionRec = selection', selectionPosition = position'})
]
validateSelection (key, RawSelectionField fullRawSelection'@RawSelection' {rawSelectionPosition}) = do
(dataField, datatype, arguments) <- getValidationData key fullRawSelection'
isLeaf datatype dataField
pure
[ ( key
, Selection
{ selectionArguments = arguments
, selectionRec = SelectionField
, selectionPosition = rawSelectionPosition
})
]
where
isLeaf (Leaf _) _ = Right ()
isLeaf _ DataField {fieldType = TypeAlias {aliasTyCon}} =
Left $ subfieldsNotSelected key aliasTyCon rawSelectionPosition
validateSelection (_, Spread reference') = resolveSpread fragments' [typeName'] reference' >>= validateFragment
validateSelection (_, InlineFragment fragment') =
castFragmentType Nothing (fragmentPosition fragment') [typeName'] fragment' >>= validateFragment