module Text.XML.HaXml.DtdToHaskell.Convert
( dtd2TypeDef
) where
import List (intersperse)
import Text.XML.HaXml.Types hiding (Name)
import Text.XML.HaXml.DtdToHaskell.TypeDef
data Record = R [AttDef] ContentSpec
type Db = [(String,Record)]
dtd2TypeDef :: [MarkupDecl] -> [TypeDef]
dtd2TypeDef mds =
(concatMap convert . reverse . database []) mds
where
database db [] = db
database db (m:ms) =
case m of
(Element (ElementDecl n cs)) ->
case lookup n db of
Nothing -> database ((n, R [] cs):db) ms
(Just (R as _)) -> database (replace n (R as cs) db) ms
(AttList (AttListDecl n as)) ->
case lookup n db of
Nothing -> database ((n, R as EMPTY):db) ms
(Just (R a cs)) -> database (replace n (R (a++as) cs) db) ms
_ -> database db ms
replace n v [] = error "dtd2TypeDef.replace: no element to replace"
replace n v (x@(n0,_):db)
| n==n0 = (n,v): db
| otherwise = x: replace n v db
convert :: (String, Record) -> [TypeDef]
convert (n, R as cs) =
case cs of
EMPTY -> modifier None []
ANY -> modifier None [[Any]]
(Mixed PCDATA) -> modifier None [[String]]
(Mixed (PCDATAplus ns)) -> modifier Star ([String]: map ((:[]) . Defined . name) ns)
(ContentSpec cp) ->
case cp of
(TagName n' m) -> modifier m [[Defined (name n')]]
(Choice cps m) -> modifier m (map ((:[]).inner) cps)
(Seq cps m) -> modifier m [map inner cps]
++ concatMap (mkAttrDef n) as
where
attrs :: AttrFields
attrs = map (mkAttrField n) as
modifier None sts = mkData sts attrs False (name n)
modifier m [[st]] = mkData [[modf m st]] attrs False (name n)
modifier m sts = mkData [[modf m (Defined (name_ n))]]
attrs False (name n) ++
mkData sts [] True (name_ n)
inner :: CP -> StructType
inner (TagName n' m) = modf m (Defined (name n'))
inner (Choice cps m) = modf m (OneOf (map inner cps))
inner (Seq cps None) = Tuple (map inner cps)
inner (Seq cps m) = modf m (Tuple (map inner cps))
modf None x = x
modf Query x = Maybe x
modf Star x = List x
modf Plus x = List1 x
mkData :: [[StructType]] -> AttrFields -> Bool -> Name -> [TypeDef]
mkData [] fs aux n = [DataDef aux n fs []]
mkData [ts] fs aux n = [DataDef aux n fs [(n, ts)]]
mkData tss fs aux n = [DataDef aux n fs (map (mkConstr n) tss)]
where
mkConstr n ts = (mkConsName n ts, ts)
mkConsName (Name x n) sts = Name x (n++concat (intersperse "_" (map flatten sts)))
flatten (Maybe st) = flatten st
flatten (List st) = flatten st
flatten (List1 st) = flatten st
flatten (Tuple sts) =
concat (intersperse "_" (map flatten sts))
flatten String = "Str"
flatten (OneOf sts) =
concat (intersperse "_" (map flatten sts))
flatten Any = "Any"
flatten (Defined (Name _ n)) = n
mkAttrDef e (AttDef n StringType def) =
[]
mkAttrDef e (AttDef n (TokenizedType t) def) =
[]
mkAttrDef e (AttDef n (EnumeratedType (NotationType nt)) def) =
[EnumDef (name_a e n) (map (name_ac e n) nt)]
mkAttrDef e (AttDef n (EnumeratedType (Enumeration es)) def) =
[EnumDef (name_a e n) (map (name_ac e n) es)]
mkAttrField :: String -> AttDef -> (Name,StructType)
mkAttrField e (AttDef n typ req) = (name_f e n, mkType typ req)
where
mkType StringType REQUIRED = String
mkType StringType IMPLIED = Maybe String
mkType StringType (DefaultTo (AttValue [Left s]) f) = Defaultable String s
mkType (TokenizedType _) REQUIRED = String
mkType (TokenizedType _) IMPLIED = Maybe String
mkType (TokenizedType _) (DefaultTo (AttValue [Left s]) f) =
Defaultable String s
mkType (EnumeratedType _) REQUIRED = Defined (name_a e n)
mkType (EnumeratedType _) IMPLIED = Maybe (Defined (name_a e n))
mkType (EnumeratedType _) (DefaultTo (AttValue [Left s]) f) =
Defaultable (Defined (name_a e n)) (hName (name_ac e n s))