module ProjectM36.Attribute where
import ProjectM36.Base
import ProjectM36.Error
import qualified Data.Set as S
import qualified Data.List as L
import qualified Data.Vector as V
import qualified Data.Hashable as Hash
import qualified Data.HashSet as HS
import qualified Data.Map as M
arity :: Attributes -> Int
arity = V.length
emptyAttributes :: Attributes
emptyAttributes = V.empty
null :: Attributes -> Bool
null = V.null
attributesFromList :: [Attribute] -> Attributes
attributesFromList = V.fromList
attributeName :: Attribute -> AttributeName
attributeName (Attribute name _) = name
atomType :: Attribute -> AtomType
atomType (Attribute _ atype) = atype
atomTypes :: Attributes -> V.Vector AtomType
atomTypes attrs = V.map atomType attrs
addAttribute :: Attribute -> Attributes -> Attributes
addAttribute attr attrs = attrs `V.snoc` attr
joinAttributes :: Attributes -> Attributes -> Either RelationalError Attributes
joinAttributes attrs1 attrs2 = if V.length uniqueOverlappingAttributes /= V.length overlappingAttributes then
Left (TupleAttributeTypeMismatchError overlappingAttributes)
else if V.length overlappingAttrsDifferentTypes > 0 then
Left (TupleAttributeTypeMismatchError overlappingAttrsDifferentTypes)
else
Right $ vectorUniqueify (attrs1 V.++ attrs2)
where
overlappingAttrsDifferentTypes = V.filter (\attr -> V.elem (attributeName attr) attrNames2 && V.notElem attr attrs2) attrs1
attrNames2 = V.map attributeName attrs2
uniqueOverlappingAttributes = vectorUniqueify overlappingAttributes
overlappingAttributes = V.filter (\attr -> V.elem attr attrs2) attrs1
addAttributes :: Attributes -> Attributes -> Attributes
addAttributes = (V.++)
deleteAttributeName :: AttributeName -> Attributes -> Attributes
deleteAttributeName attrName = V.filter (\attr -> attributeName attr /= attrName)
renameAttribute :: AttributeName -> Attribute -> Attribute
renameAttribute newAttrName (Attribute _ typeo) = Attribute newAttrName typeo
renameAttributes :: AttributeName -> AttributeName -> Attributes -> Attributes
renameAttributes oldAttrName newAttrName attrs = V.map renamer attrs
where
renamer attr = if attributeName attr == oldAttrName then
renameAttribute newAttrName attr
else
attr
atomTypeForAttributeName :: AttributeName -> Attributes -> Either RelationalError AtomType
atomTypeForAttributeName attrName attrs = do
(Attribute _ atype) <- attributeForName attrName attrs
return atype
attributeForName :: AttributeName -> Attributes -> Either RelationalError Attribute
attributeForName attrName attrs = case V.find (\attr -> attributeName attr == attrName) attrs of
Nothing -> Left $ NoSuchAttributeNamesError (S.singleton attrName)
Just attr -> Right attr
attributesForNames :: S.Set AttributeName -> Attributes -> Attributes
attributesForNames attrNameSet attrs = V.filter filt attrs
where
filt attr = S.member (attributeName attr) attrNameSet
attributeNameSet :: Attributes -> S.Set AttributeName
attributeNameSet attrVec = S.fromList $ V.toList $ V.map (\(Attribute name _) -> name) attrVec
attributeNames :: Attributes -> V.Vector AttributeName
attributeNames = V.map attributeName
attributesContained :: Attributes -> Attributes -> Bool
attributesContained attrs1 attrs2 = attributeNamesContained (attributeNameSet attrs1) (attributeNameSet attrs2)
attributeNamesContained :: S.Set AttributeName -> S.Set AttributeName -> Bool
attributeNamesContained attrs1 attrs2 = S.isSubsetOf attrs1 attrs2
nonMatchingAttributeNameSet :: S.Set AttributeName -> S.Set AttributeName -> S.Set AttributeName
nonMatchingAttributeNameSet a1 a2 = S.difference (S.union a1 a2) (S.intersection a1 a2)
matchingAttributeNameSet :: S.Set AttributeName -> S.Set AttributeName -> S.Set AttributeName
matchingAttributeNameSet = S.intersection
attributeNamesNotContained :: S.Set AttributeName -> S.Set AttributeName -> S.Set AttributeName
attributeNamesNotContained subset superset = S.filter (flip S.notMember superset) subset
sortedAttributeNameList :: S.Set AttributeName -> [AttributeName]
sortedAttributeNameList attrNameSet= L.sort $ S.toList attrNameSet
attributesDifference :: Attributes -> Attributes -> Attributes
attributesDifference attrsA attrsB = V.fromList $ diff (V.toList attrsA) (V.toList attrsB)
where
diff a b = (a L.\\ b) ++ (b L.\\ a)
vectorUniqueify :: (Hash.Hashable a, Eq a) => V.Vector a -> V.Vector a
vectorUniqueify vecIn = V.fromList $ HS.toList $ HS.fromList $ V.toList vecIn
verifyAttributes :: Attributes -> Either RelationalError Attributes
verifyAttributes attrs = if collapsedAttrs /= attrs then
Left (TupleAttributeTypeMismatchError (attributesDifference collapsedAttrs attrs))
else
Right attrs
where
collapsedAttrs = vectorUniqueify attrs
attributesEqual :: Attributes -> Attributes -> Bool
attributesEqual attrs1 attrs2 = V.null (attributesDifference attrs1 attrs2)
attributesAsMap :: Attributes -> M.Map AttributeName Attribute
attributesAsMap attrs = (M.fromList . V.toList) (V.map (\attr -> (attributeName attr, attr)) attrs)
union :: Attributes -> Attributes -> Attributes
union attrsA attrsB = V.fromList (M.elems unioned)
where
unioned = M.union (attributesAsMap attrsA) (attributesAsMap attrsB)
intersection :: Attributes -> Attributes -> Attributes
intersection attrsA attrsB = V.fromList (M.elems intersected)
where
intersected = M.intersection (attributesAsMap attrsA) (attributesAsMap attrsB)