module Swish.RDF.RDFDatatypeXsdString
( rdfDatatypeXsdString
, rdfDatatypeValXsdString
, typeNameXsdString, namespaceXsdString
, axiomsXsdString, rulesXsdString
, prefixXsdString
)
where
import Swish.RDF.RDFRuleset
( RDFFormula, RDFRule, RDFRuleset
, makeRDFGraphFromN3String
, makeRDFFormula
, makeN3ClosureRule
)
import Swish.RDF.RDFVarBinding
( RDFVarBindingModify
)
import Swish.RDF.RDFDatatype
( RDFDatatype
, RDFDatatypeVal
, RDFDatatypeMod
, makeRdfDtOpenVarBindingModifiers
)
import Swish.RDF.RDFGraph
( RDFLabel(..) )
import Swish.RDF.ClassRestrictionRule
( makeRDFDatatypeRestrictionRules
)
import Swish.RDF.Datatype
( Datatype(..)
, DatatypeVal(..)
, DatatypeMap(..)
, DatatypeRel(..), DatatypeRelPr
, altArgs
, UnaryFnTable, unaryFnApp
, DatatypeMod(..)
, makeVmod_2_0
)
import Swish.RDF.Ruleset
( makeRuleset
)
import Swish.Utils.Namespace
( Namespace(..)
, ScopedName(..)
)
import Swish.RDF.Vocabulary
( namespaceRDF
, namespaceRDFS
, namespaceRDFD
, namespaceXSD
, namespaceXsdType
)
import Swish.RDF.VarBinding
( VarBinding(..)
, addVarBinding
, VarBindingModify(..)
)
nameXsdString :: String
nameXsdString = "string"
typeNameXsdString :: ScopedName
typeNameXsdString = ScopedName namespaceXSD nameXsdString
namespaceXsdString :: Namespace
namespaceXsdString = namespaceXsdType nameXsdString
infixr 5 +++
(+++) :: String -> ShowS
(+++) str = ((str++"\n")++)
rdfDatatypeXsdString :: RDFDatatype
rdfDatatypeXsdString = Datatype rdfDatatypeValXsdString
rdfDatatypeValXsdString :: RDFDatatypeVal String
rdfDatatypeValXsdString = DatatypeVal
{ tvalName = typeNameXsdString
, tvalRules = rdfRulesetXsdString
, tvalMkRules = makeRDFDatatypeRestrictionRules rdfDatatypeValXsdString
, tvalMkMods = makeRdfDtOpenVarBindingModifiers rdfDatatypeValXsdString
, tvalMap = mapXsdString
, tvalRel = relXsdString
, tvalMod = modXsdString
}
mapXsdString :: DatatypeMap String
mapXsdString = DatatypeMap
{
mapL2V = Just
, mapV2L = Just
}
relXsdString :: [DatatypeRel String]
relXsdString =
[ relXsdStringEq
, relXsdStringNe
]
mkStrRel2 ::
String -> DatatypeRelPr String -> UnaryFnTable String
-> DatatypeRel String
mkStrRel2 nam pr fns = DatatypeRel
{ dtRelName = ScopedName namespaceXsdString nam
, dtRelFunc = altArgs pr fns unaryFnApp
}
liftL2 :: (a->a->Bool) -> ([a]->a) -> ([a]->a) -> [a] -> Bool
liftL2 p i1 i2 as = p (i1 as) (i2 as)
lcomp :: (a->a->Bool) -> [a] -> Bool
lcomp p = liftL2 p head (head . tail)
relXsdStringEq :: DatatypeRel String
relXsdStringEq = mkStrRel2 "eq" (lcomp (==))
( repeat (const True, []) )
relXsdStringNe :: DatatypeRel String
relXsdStringNe = mkStrRel2 "ne" (lcomp (/=))
( repeat (const True, []) )
modXsdString :: [RDFDatatypeMod String]
modXsdString =
[ modXsdStringEq
, modXsdStringNe
]
modXsdStringEq, modXsdStringNe :: RDFDatatypeMod String
modXsdStringEq = modXsdStringCompare "eq" (==)
modXsdStringNe = modXsdStringCompare "ne" (/=)
modXsdStringCompare ::
String -> (String->String->Bool) -> RDFDatatypeMod String
modXsdStringCompare nam rel = DatatypeMod
{ dmName = ScopedName namespaceXsdString nam
, dmModf = [ f0 ]
, dmAppf = makeVmod_2_0
}
where
f0 vs@[v1,v2] = if rel v1 v2 then vs else []
f0 _ = []
rdfRulesetXsdString :: RDFRuleset
rdfRulesetXsdString =
makeRuleset namespaceXsdString axiomsXsdString rulesXsdString
mkPrefix :: Namespace -> String
mkPrefix ns =
"@prefix " ++ nsPrefix ns ++ ": <" ++ nsURI ns ++ "> . \n"
prefixXsdString :: String
prefixXsdString =
mkPrefix namespaceRDF ++
mkPrefix namespaceRDFS ++
mkPrefix namespaceRDFD ++
mkPrefix namespaceXSD ++
mkPrefix namespaceXsdString ++
" \n"
mkAxiom :: String -> String -> RDFFormula
mkAxiom local gr =
makeRDFFormula namespaceXsdString local (prefixXsdString++gr)
axiomsXsdString :: [RDFFormula]
axiomsXsdString =
[ mkAxiom "dt" "xsd:string rdf:type rdfs:Datatype ."
]
rulesXsdString :: [RDFRule]
rulesXsdString = rulesXsdStringClosure ++ rulesXsdStringRestriction
rulesXsdStringRestriction :: [RDFRule]
rulesXsdStringRestriction =
makeRDFDatatypeRestrictionRules rdfDatatypeValXsdString gr
where
gr = makeRDFGraphFromN3String rulesXsdStringStr
rulesXsdStringStr :: String
rulesXsdStringStr = prefixXsdString
+++ "xsd_string:Eq a rdfd:GeneralRestriction ; "
+++ " rdfd:onProperties (rdf:_1 rdf:_2) ; "
+++ " rdfd:constraint xsd_string:eq ; "
+++ " rdfd:maxCardinality \"1\"^^xsd:nonNegativeInteger . "
+++ "xsd_string:Ne a rdfd:GeneralRestriction ; "
+++ " rdfd:onProperties (rdf:_1 rdf:_2) ; "
+++ " rdfd:constraint xsd_string:ne ; "
+++ " rdfd:maxCardinality \"1\"^^xsd:nonNegativeInteger . "
rulesXsdStringClosure :: [RDFRule]
rulesXsdStringClosure =
[ xsdstrls
, xsdstrsl
]
xsdstrls :: RDFRule
xsdstrls = makeN3ClosureRule namespaceXsdString "ls"
"?a ?p ?l ."
"?a ?p ?s ."
(stringPlain "s" "l")
xsdstrsl :: RDFRule
xsdstrsl = makeN3ClosureRule namespaceXsdString "sl"
"?a ?p ?s ."
"?a ?p ?l ."
(stringPlain "s" "l")
stringPlain :: String -> String -> RDFVarBindingModify
stringPlain svar lvar = stringPlainValue (Var svar) (Var lvar)
stringPlainValue ::
RDFLabel -> RDFLabel -> RDFVarBindingModify
stringPlainValue svar lvar = VarBindingModify
{ vbmName = ScopedName namespaceRDFD "stringPlain"
, vbmApply = concatMap app1
, vbmVocab = [svar,lvar]
, vbmUsage = [[svar],[lvar],[]]
}
where
app1 vbind = app2 (vbMap vbind svar) (vbMap vbind lvar) vbind
app2 (Just (Lit s (Just _)))
(Just (Lit l Nothing))
vbind
| s == l
= [vbind]
app2 (Just (Lit s (Just _)))
Nothing
vbind
= [addVarBinding lvar (Lit s Nothing) vbind]
app2 Nothing
(Just (Lit l Nothing))
vbind
= [addVarBinding svar (Lit l (Just typeNameXsdString)) vbind]
app2 _ _ _ = []