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 -- . L.nub --too expensive 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 --hm- no error-checking here addAttribute :: Attribute -> Attributes -> Attributes addAttribute attr attrs = attrs `V.snoc` attr --if some attribute names overlap but the types do not, then spit back an error 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 --checks if set s1 is wholly contained in the set s2 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 --returns the disjunction of the AttributeNameSets 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 -- this is sorted so the tuples know in which order to output- the ordering is arbitrary sortedAttributeNameList :: S.Set AttributeName -> [AttributeName] sortedAttributeNameList attrNameSet= L.sort $ S.toList attrNameSet -- take two attribute sets and return an attribute set with the attributes which do not match 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 --check that each attribute only appears once 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) -- | Left-biased union of attributes. 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)