diff options
-rw-r--r-- | bbcode/src/Text/BBCode.hs | 48 | ||||
-rw-r--r-- | bbcode/thermoprint-bbcode.cabal | 2 | ||||
-rw-r--r-- | bbcode/thermoprint-bbcode.nix | 10 |
3 files changed, 51 insertions, 9 deletions
diff --git a/bbcode/src/Text/BBCode.hs b/bbcode/src/Text/BBCode.hs index 7a328a8..dfa1db7 100644 --- a/bbcode/src/Text/BBCode.hs +++ b/bbcode/src/Text/BBCode.hs | |||
@@ -1,14 +1,52 @@ | |||
1 | {-# LANGUAGE OverloadedStrings #-} | 1 | {-# LANGUAGE OverloadedStrings #-} |
2 | {-# LANGUAGE DeriveGeneric #-} | ||
2 | 3 | ||
3 | module Text.BBCode | 4 | module Text.BBCode |
4 | ( | 5 | ( TreeError(..) |
6 | , rose | ||
7 | , matches | ||
5 | ) where | 8 | ) where |
6 | 9 | ||
7 | import Data.Attoparsec.Text | ||
8 | |||
9 | import Data.Text (Text) | 10 | import Data.Text (Text) |
10 | import qualified Data.Text as T (singleton, head, tail) | ||
11 | 11 | ||
12 | import Control.Applicative | 12 | import GHC.Generics (Generic) |
13 | import Control.Exception (Exception) | ||
14 | import Data.Typeable (Typeable) | ||
15 | |||
16 | import Control.Monad (unless) | ||
13 | 17 | ||
14 | import Text.BBCode.Lexer (BBToken(..), token) | 18 | import Text.BBCode.Lexer (BBToken(..), token) |
19 | |||
20 | import Data.Tree | ||
21 | import Data.Tree.Zipper (TreePos, Empty, Full) | ||
22 | import qualified Data.Tree.Zipper as Z | ||
23 | |||
24 | data TreeError = ImbalancedTags Text Text | ||
25 | | LeftoverClose Text | ||
26 | deriving (Show, Eq, Generic, Typeable) | ||
27 | |||
28 | instance Exception TreeError | ||
29 | |||
30 | matches :: Text -> Text -> Bool | ||
31 | -- ^ @`matches` "open" "close"@ should be true iff @[/close]@ is a valid closing tag for @[open]@ | ||
32 | -- | ||
33 | -- Until we allow for attributes this is equality according to `(==)` | ||
34 | matches = (==) | ||
35 | |||
36 | rose :: [BBToken] -> Either TreeError (Forest Text) | ||
37 | -- ^ Assuming that both tags and content have the same type (we use 'Text') bbcode is a flat representation of a rose tree | ||
38 | rose = fmap Z.toForest . flip rose' (Z.fromForest []) | ||
39 | where | ||
40 | rose' :: [BBToken] -> TreePos Empty Text -> Either TreeError (TreePos Empty Text) | ||
41 | rose' [] = return | ||
42 | rose' (x:xs) = (>>= rose' xs) . rose'' x | ||
43 | |||
44 | rose'' (BBStr t) = return . Z.nextSpace . Z.insert (Node t []) | ||
45 | rose'' (BBOpen t) = return . Z.children . Z.insert (Node t []) | ||
46 | rose'' (BBClose t) = close t -- for more pointless | ||
47 | |||
48 | close :: Text -> TreePos Empty Text -> Either TreeError (TreePos Empty Text) | ||
49 | close tag pos = do | ||
50 | pos' <- maybe (Left $ LeftoverClose tag) Right $ Z.parent pos | ||
51 | unless (Z.label pos' `matches` tag) . Left $ ImbalancedTags (Z.label pos') tag -- The structure shows that this mode of failure is not logically required -- it's just nice to have | ||
52 | return $ Z.nextSpace pos' | ||
diff --git a/bbcode/thermoprint-bbcode.cabal b/bbcode/thermoprint-bbcode.cabal index f1018a5..80a9da5 100644 --- a/bbcode/thermoprint-bbcode.cabal +++ b/bbcode/thermoprint-bbcode.cabal | |||
@@ -24,6 +24,8 @@ library | |||
24 | build-depends: base >=4.8 && <4.9 | 24 | build-depends: base >=4.8 && <4.9 |
25 | , attoparsec >=0.13.0 && <1 | 25 | , attoparsec >=0.13.0 && <1 |
26 | , text >=1.2.1 && <2 | 26 | , text >=1.2.1 && <2 |
27 | , containers >=0.4.0 && <1 | ||
28 | , rosezipper >=0.2 && <1 | ||
27 | hs-source-dirs: src | 29 | hs-source-dirs: src |
28 | default-language: Haskell2010 | 30 | default-language: Haskell2010 |
29 | 31 | ||
diff --git a/bbcode/thermoprint-bbcode.nix b/bbcode/thermoprint-bbcode.nix index 5521b10..c379053 100644 --- a/bbcode/thermoprint-bbcode.nix +++ b/bbcode/thermoprint-bbcode.nix | |||
@@ -1,14 +1,16 @@ | |||
1 | { mkDerivation, attoparsec, base, hspec, QuickCheck | 1 | { mkDerivation, attoparsec, base, containers, hspec, QuickCheck |
2 | , quickcheck-instances, stdenv, text | 2 | , quickcheck-instances, rosezipper, stdenv, text |
3 | }: | 3 | }: |
4 | mkDerivation { | 4 | mkDerivation { |
5 | pname = "thermoprint-bbcode"; | 5 | pname = "thermoprint-bbcode"; |
6 | version = "0.0.0"; | 6 | version = "0.0.0"; |
7 | src = ./.; | 7 | src = ./.; |
8 | libraryHaskellDepends = [ | 8 | libraryHaskellDepends = [ |
9 | attoparsec base QuickCheck quickcheck-instances text | 9 | attoparsec base containers rosezipper text |
10 | ]; | ||
11 | testHaskellDepends = [ | ||
12 | attoparsec base hspec QuickCheck quickcheck-instances text | ||
10 | ]; | 13 | ]; |
11 | testHaskellDepends = [ attoparsec base hspec QuickCheck text ]; | ||
12 | homepage = "http://dirty-haskell.org/tags/thermoprint.html"; | 14 | homepage = "http://dirty-haskell.org/tags/thermoprint.html"; |
13 | description = "A parser for a subset of bbcode compatible with thermoprint-spec"; | 15 | description = "A parser for a subset of bbcode compatible with thermoprint-spec"; |
14 | license = stdenv.lib.licenses.publicDomain; | 16 | license = stdenv.lib.licenses.publicDomain; |