{-# LANGUAGE OverloadedStrings #-} module MainTest where import qualified Control.Exception as Exception import Control.Monad import qualified Data.Either as Either import qualified Data.Monoid as Monoid import Data.Text (Text) import qualified Data.Text as T import Control.Exception (assert) import qualified System.IO.Unsafe as Unsafe import qualified FastTags import FastTags (UnstrippedTokens(..), TokenVal(..), TagVal(..), Type(..), Tag, Pos(..)) import Test.Tasty import Test.Tasty.HUnit main :: IO () main = defaultMain tests tests :: TestTree tests = testGroup "tests" [ testTokenize , testSkipString , testStripComments , testBreakBlocks , testProcessAll , testProcess ] test :: (Show a, Eq b, Show b) => (a -> b) -> a -> b -> TestTree test f x expected = testCase (take 70 $ show x) $ f x @?= expected testTokenize :: TestTree testTokenize = testGroup "tokenize" [ "a::b->c" ==> ["a", "::", "b", "->", "c"] , "a∷b→c" ==> ["a", "∷", "b", "→", "c"] , "x{-\n bc#-}\n" ==> ["x", "{-", "nl 2", "bc#", "-}"] , "X.Y" ==> ["X.Y"] , "x9" ==> ["x9"] -- , "9x" ==> ["nl 0", "9", "x"] , "x :+: y" ==> ["x", ":+:", "y"] , "(#$)" ==> ["(", "#$", ")"] , "Data.Map.map" ==> ["Data.Map.map"] , "Map.map" ==> ["Map.map"] , "forall a. f a" ==> ["forall", "a", ".", "f", "a"] , "forall a . Foo" ==> ["forall", "a", ".", "Foo"] , "forall a. Foo" ==> ["forall", "a", ".", "Foo"] , "forall a .Foo" ==> ["forall", "a", ".", "Foo"] , "forall a.Foo" ==> ["forall", "a", ".", "Foo"] , "$#-- hi" ==> ["$#--", "hi"] , "(*), (-)" ==> ["(", "*", ")", ",", "(", "-", ")"] , "(.::)" ==> ["(", ".::", ")"] -- we rely on this behavior , "data (:+) a b" ==> ["data", "(", ":+", ")", "a", "b"] , "data (.::+::) a b" ==> ["data", "(", ".::+::", ")", "a", "b"] ] where (==>) = test f f = tail . extractTokens . tokenize testSkipString :: TestTree testSkipString = testGroup "skipString" [ "hi \" there" ==> " there" , "hi \\a \" there" ==> " there" , "hi \\\" there\"" ==> "" , "hi" ==> "" -- String continuation , test f' "hi \\\n \\bar\" baz" ("hi bar\"", " baz") ] where (==>) = test f f = snd . FastTags.breakString f' = FastTags.breakString testStripComments :: TestTree testStripComments = testGroup "stripComments" [ "hello -- there" ==> ["nl 0", "hello"] , "hello --there" ==> ["nl 0", "hello"] , "hello {- there -} fred" ==> ["nl 0", "hello", "fred"] , "hello -- {- there -}\nfred" ==> ["nl 0", "hello", "nl 0", "fred"] , "{-# LANG #-} hello {- there {- nested -} comment -} fred" ==> ["nl 0", "hello", "fred"] , "hello {-\nthere\n------}\n fred" ==> ["nl 0", "hello", "nl 1", "fred"] , "hello {- \nthere\n ------} \n fred" ==> ["nl 0", "hello", "nl 1", "fred"] , "hello {-\nthere\n-----}\n fred" ==> ["nl 0", "hello", "nl 1", "fred"] , "hello {- \nthere\n -----} \n fred" ==> ["nl 0", "hello", "nl 1", "fred"] , "hello {-\n-- there -}" ==> ["nl 0", "hello"] , "foo --- my comment\n--- my other comment\nbar" ==> ["nl 0", "foo", "nl 0", "nl 0", "bar"] ] where (==>) = test f f = extractTokens . FastTags.stripComments . tokenize testBreakBlocks :: TestTree testBreakBlocks = testGroup "breakBlocks" [ "1\n2\n" ==> [["1"], ["2"]] , "1\n 1\n2\n" ==> [["1", "1"], ["2"]] , "1\n 1\n 1\n2\n" ==> [["1", "1", "1"], ["2"]] -- intervening blank lines are ignored , "1\n 1\n\n 1\n2\n" ==> [["1", "1", "1"], ["2"]] , "1\n\n\n 1\n2\n" ==> [["1", "1"], ["2"]] , "1\n 11\n 11\n" ==> [["1", "11", "11"]] , " 11\n 11\n" ==> [["11"], ["11"]] ] where (==>) = test f f = map (extractTokens . UnstrippedTokens . FastTags.stripNewlines) . FastTags.breakBlocks . tokenize testProcessAll :: TestTree testProcessAll = testGroup "processAll" [ ["data X", "module X"] ==> ["fn0:1 X Type", "fn1:1 X Module"] -- Type goes ahead of Module. , ["module X\n\ \data X"] ==> ["fn0:2 X Type", "fn0:1 X Module"] -- Extra X was filtered. , ["module X\n\ \data X = X\n"] ==> ["fn0:2 X Type", "fn0:2 X Constructor", "fn0:1 X Module"] , ["module X\n\ \data X a =\n\ \ X a\n"] ==> ["fn0:2 X Type", "fn0:3 X Constructor", "fn0:1 X Module"] , ["newtype A f a b = A\n\ \ { unA :: f (a -> b) }"] ==> ["fn0:1 A Type", "fn0:1 A Constructor", "fn0:2 unA Function"] ] where (==>) = test f f = map showTag . FastTags.processAll . map (\(i, t) -> fst $ FastTags.process ("fn" ++ show i) False t) . zip [0..] showTag (Pos p (TagVal _ text typ)) = unwords [show p, T.unpack text, show typ] testProcess :: TestTree testProcess = testGroup "process" [ testMisc , testData , testGADT , testFamilies , testFunctions , testClass , testInstance , testLiterate , testPatterns , testFFI ] testMisc :: TestTree testMisc = testGroup "misc" [ "module Bar.Foo where\n" ==> [TagVal "module Bar.Foo" "Foo" Module] , "newtype Foo a b =\n\ \\tBar x y z\n" ==> [ TagVal "\tBar" "Bar" Constructor , TagVal "newtype Foo" "Foo" Type ] , "f :: A -> B\n\ \g :: C -> D\n\ \data D = C {\n\ \\tf :: A\n\ \\t}\n" ==> [ TagVal "data D = C" "C" Constructor , TagVal "data D" "D" Type , TagVal "\tf" "f" Function , TagVal "f" "f" Function , TagVal "g" "g" Function ] ] where (==>) = test f f = map valOf . fst . FastTags.process "fn.hs" False testData :: TestTree testData = testGroup "data" [ "data X\n" ==> ["X"] , "data X = X Int\n" ==> ["X", "X"] , "data Foo = Bar | Baz" ==> ["Bar", "Baz", "Foo"] , "data Foo =\n\tBar\n\t| Baz" ==> ["Bar", "Baz", "Foo"] -- Records. , "data Foo a = Bar { field :: Field }" ==> ["Bar", "Foo", "field"] , "data R = R { a::X, b::Y }" ==> ["R", "R", "a", "b"] , "data R = R { a∷X, b∷Y }" ==> ["R", "R", "a", "b"] , "data R = R {\n\ta::X\n\t, b::Y\n\t}" ==> ["R", "R", "a", "b"] , "data R = R {\n\ta,b::X\n\t}" ==> ["R", "R", "a", "b"] , "data R = R {\n\ \ a :: !RealTime\n\ \ , b :: !RealTime\n\ \}" ==> ["R", "R", "a", "b"] , "data Rec = Rec {\n\ \ a :: Int\n\ \, b :: !Double\n\ \, c :: Maybe Rec\ \\n\ \}" ==> ["Rec", "Rec", "a", "b", "c"] , "data X = X !Int" ==> ["X", "X"] , "data X = Y !Int !X | Z" ==> ["X", "Y", "Z"] , "data X = Y :+: !Z | !X `Mult` X" ==> [":+:", "Mult", "X"] , "data X = !Y `Add` !Z" ==> ["Add", "X"] , "data X = forall a. Y a" ==> ["X", "Y"] , "data X = forall a . Y a" ==> ["X", "Y"] , "data X = forall a .Y a" ==> ["X", "Y"] , "data X = forall a.Y a" ==> ["X", "Y"] , "data X = forall a. Eq a => Y a" ==> ["X", "Y"] , "data X = forall a. (Eq a) => Y a" ==> ["X", "Y"] , "data X = forall a. (Eq a, Ord a) => Y a" ==> ["X", "Y"] -- "data X = forall a. Ref :<: a => Y a" ==> ["X", "Y"] -- "data X = forall a. (:<:) Ref a => Y a" ==> ["X", "Y"] , "data X = forall a. ((:<:) Ref a) => Y a" ==> ["X", "Y"] , "data X = forall a. Y !a" ==> ["X", "Y"] , "data X = forall a. (Eq a, Ord a) => Y !a" ==> ["X", "Y"] , "data Foo a = \n\ \ Plain Int\n\ \ | forall a. Bar a Int\n\ \ | forall a b. Baz b a\n\ \ | forall a . Quux a \ \ | forall a .Quuz a" ==> ["Bar", "Baz", "Foo", "Plain", "Quux", "Quuz"] , "data X a = Add a " ==> ["Add", "X"] , "data Eq a => X a = Add a" ==> ["Add", "X"] , "data (Eq a) => X a = Add a" ==> ["Add", "X"] , "data (Eq a, Ord a) => X a = Add a" ==> ["Add", "X"] , "data (Eq (a), Ord (a)) => X a = Add a" ==> ["Add", "X"] -- These are hard-to-deal-with uses of contexts, which are probably not that -- common and therefoce can be ignored. -- "data Ref :<: f => X f = RRef f" ==> ["X", "RRef"] -- "data a :<: b => X a b = Add a" ==> ["X", "Add"] , "data (a :<: b) => X a b = Add a" ==> ["Add", "X"] , "newtype Eq a => X a = Add a" ==> ["Add", "X"] , "newtype (Eq a) => X a = Add a" ==> ["Add", "X"] , "newtype (Eq a, Ord a) => X a = Add a" ==> ["Add", "X"] , "newtype (u :*: v) z = X" ==> [":*:", "X"] , "data (u :*: v) z = X" ==> [":*:", "X"] , "type (u :*: v) z = (u, v, z)" ==> [":*:"] , "newtype ((u :: (* -> *) -> *) :*: v) z = X" ==> [":*:", "X"] , "data ((u :: (* -> *) -> *) :*: v) z = X" ==> [":*:", "X"] , "type ((u :: (* -> *) -> *) :*: v) z = (u, v, z)" ==> [":*:"] , "newtype (Eq (v z)) => ((u :: (* -> *) -> *) :*: v) z = X" ==> [":*:", "X"] , "data (Eq (v z)) => ((u :: (* -> *) -> *) :*: v) z = X" ==> [":*:", "X"] , "type (Eq (v z)) => ((u :: (* -> *) -> *) :*: v) z = (u, v, z)" ==> [":*:"] , "newtype Eq (v z) => ((u :: (* -> *) -> *) :*: v) z = X" ==> [":*:", "X"] , "data Eq (v z) => ((u :: (* -> *) -> *) :*: v) z = X" ==> [":*:", "X"] , "type Eq (v z) => ((u :: (* -> *) -> *) :*: v) z = (u, v, z)" ==> [":*:"] , "newtype (Eq (v z)) => ((u :: (* -> *) -> *) `Foo` v) z = X" ==> ["Foo", "X"] , "data (Eq (v z)) => ((u :: (* -> *) -> *) `Foo` v) z = X" ==> ["Foo", "X"] , "type (Eq (v z)) => ((u :: (* -> *) -> *) `Foo` v) z = (u, v, z)" ==> ["Foo"] , "type (Eq (v z)) => ((u ∷ (* -> *) -> *) `Foo` v) z = (u, v, z)" ==> ["Foo"] , "newtype Eq (v z) => ((u :: (* -> *) -> *) `Foo` v) z = X" ==> ["Foo", "X"] , "data Eq (v z) => ((u :: (* -> *) -> *) `Foo` v) z = X" ==> ["Foo", "X"] , "type Eq (v z) => ((u :: (* -> *) -> *) `Foo` v) z = (u, v, z)" ==> ["Foo"] , "type Eq (v z) ⇒ ((u ∷ (* → *) → *) `Foo` v) z = (u, v, z)" ==> ["Foo"] , "data (:*:) u v z = X" ==> [":*:", "X"] , "data (Eq (u v), Ord (z)) => (:*:) u v z = X" ==> [":*:", "X"] , "data (u `W` v) z = X" ==> ["W", "X"] , "data (Eq (u v), Ord (z)) => (u `W` v) z = X" ==> ["W", "X"] , "newtype X a = Z {\n\ \ -- TODO blah\n\ \ foo :: [a] }" ==> ["X", "Z", "foo"] , "newtype (u :*: v) z = X {\n\ \ -- my insightful comment\n\ \ extract :: (u (v z)) }" ==> [":*:", "X", "extract"] , "data Hadron a b = Science { f :: a, g :: a, h :: b }" ==> ["Hadron", "Science", "f", "g", "h"] , "data Hadron a b = Science { f :: a, g :: (a, b), h :: b }" ==> ["Hadron", "Science", "f", "g", "h"] , "data Hadron a b = forall x. Science { f :: x, h :: b }" ==> ["Hadron", "Science", "f", "h"] , "data Hadron a b = forall x. Science { f :: [x], h :: b }" ==> ["Hadron", "Science", "f", "h"] , "data Hadron a b = forall x y. Science { f :: (x, y), h :: b }" ==> ["Hadron", "Science", "f", "h"] , "data Hadron a b = forall x y. Science { f :: [(x, y)], h :: b }" ==> ["Hadron", "Science", "f", "h"] , "data Hadron a b = forall x y. Science { f :: [(Box x, Map x y)], h :: b }" ==> ["Hadron", "Science", "f", "h"] , "data Hadron a b = forall x y. Science { f ∷ [(Box x, Map x y)], h ∷ b }" ==> ["Hadron", "Science", "f", "h"] , "data Hadron a b = forall x y z. Science\n\ \ { f :: x\n\ \ , g :: [(Box x, Map x y, z)] \ \ }" ==> ["Hadron", "Science", "f", "g"] , "data Hadron a b = forall x y z. Science\n\ \ { f :: x\n\ \ , g :: [(Box x, Map x y, z)] \ \ , h :: b\n\ \ }" ==> ["Hadron", "Science", "f", "g", "h"] , "data Hadron a b = Science { h :: b }" ==> ["Hadron", "Science", "h"] , "data Test a b =\n\ \ Foo a\n\ \ | Bar [(Maybe a, Map a b, b)]" ==> ["Bar", "Foo", "Test"] , "data Test a b =\n\ \ Foo a\n\ \ | [(Maybe b, Map b a, a)] `Bar` [(Maybe a, Map a b, b)]" ==> ["Bar", "Foo", "Test"] ] where (==>) = test process testGADT :: TestTree testGADT = testGroup "gadt" [ "data X where A :: X\n" ==> ["A", "X"] , "data X where\n\tA :: X\n" ==> ["A", "X"] , "data X where\n\tA :: X\n\tB :: X\n" ==> ["A", "B", "X"] , "data X where\n\tA, B :: X\n" ==> ["A", "B", "X"] , "data X :: * -> * -> * where\n\ \ A, B :: Int -> Int -> X\n" ==> ["A", "B", "X"] , "data X ∷ * → * → * where\n\ \ A, B ∷ Int → Int → X\n" ==> ["A", "B", "X"] , "data Vec ix where\n\ \ Nil :: Int -> Foo Int\n\ \ (:::) :: Int -> Vec Int -> Vec Int\n\ \ (.+.) :: Int -> Int -> Vec Int -> Vec Int\n" ==> [".+.", ":::", "Nil", "Vec"] , "data Vec ix where\n\ \ Nil :: Int -> Foo Int\n\ \ -- foo\n\ \ (:::) :: Int -> Vec Int -> Vec Int\n\ \-- bar\n\ \ (.+.) :: Int -> \n\ \ -- ^ baz\n\ \ Int -> \n\ \ Vec Int -> \n\ \Vec Int\n" ==> [".+.", ":::", "Nil", "Vec"] , "data NatSing (n :: Nat) where\n ZeroSing :: 'Zero\n SuccSing :: NatSing n -> NatSing ('Succ n)\n" ==> ["NatSing", "SuccSing", "ZeroSing"] ] where (==>) = test process testFamilies :: TestTree testFamilies = testGroup "families" [ "type family X :: *\n" ==> ["X"] , "data family X :: * -> *\n" ==> ["X"] , "data family a ** b" ==> ["**"] , "type family a :<: b\n" ==> [":<:"] , "type family (a :: Nat) :<: (b :: Nat) :: Nat\n" ==> [":<:"] , "type family (a :: Nat) `Family` (b :: Nat) :: Nat\n" ==> ["Family"] , "type family (m :: Nat) <=? (n :: Nat) :: Bool" ==> ["<=?"] , "type family (m ∷ Nat) <=? (n ∷ Nat) ∷ Bool" ==> ["<=?"] , "data instance X a b = Y a | Z { unZ :: b }" ==> ["Y", "Z", "unZ"] , "data instance (Eq a, Eq b) => X a b = Y a | Z { unZ :: b }" ==> ["Y", "Z", "unZ"] , "data instance XList Char = XCons !Char !(XList Char) | XNil" ==> ["XCons", "XNil"] , "newtype instance Cxt x => T [x] = A (B x) deriving (Z,W)" ==> ["A"] , "data instance G [a] b where\n\ \ G1 :: c -> G [Int] b\n\ \ G2 :: G [a] Bool" ==> ["G1", "G2"] , "class C where\n\ttype X y :: *\n" ==> ["C", "X"] , "class C where\n\tdata X y :: *\n" ==> ["C", "X"] , "class C where\n\ttype X y ∷ *\n" ==> ["C", "X"] , "class C where\n\tdata X y ∷ *\n" ==> ["C", "X"] ] where (==>) = test process testFunctions :: TestTree testFunctions = testGroup "functions" [ -- Multiple declarations. "a,b::X" ==> ["a", "b"] -- With an operator. , "(+), a :: X" ==> ["+", "a"] -- Don't get fooled by literals. , "1 :: Int" ==> [] -- plain functions and operators , "(.::) :: X -> Y" ==> [".::"] , "(:::) :: X -> Y" ==> [":::"] , "(->:) :: X -> Y" ==> ["->:"] , "(--+) :: X -> Y" ==> ["--+"] , "(=>>) :: X -> Y" ==> ["=>>"] , "_g :: X -> Y" ==> ["_g"] , toplevelFunctionsWithoutSignatures ] where (==>) = test process toplevelFunctionsWithoutSignatures = testGroup "toplevel functions without signatures" [ "infix 5 |+|" ==> [] , "infixl 5 |+|" ==> [] , "infixr 5 |+|" ==> [] , "f = g" ==> ["f"] , "f :: a -> b -> a\n\ \f x y = x" ==> ["f"] , "f x y = x" ==> ["f"] , "f (x :+: y) z = x" ==> ["f"] , "(x :+: y) `f` z = x" ==> ["f"] , "(x :+: y) :*: z = x" ==> [":*:"] , "((:+:) x y) :*: z = x" ==> [":*:"] , "(:*:) (x :+: y) z = x" ==> [":*:"] , "(:*:) ((:+:) x y) z = x" ==> [":*:"] , strictMatchTests , lazyMatchTests , atPatternsTests , "f x Nothing = x\n\ \f x (Just y) = y" ==> ["f"] , "f x y = g x\n\ \ where\n\ \ g _ = y" ==> ["f"] , "x `f` y = x" ==> ["f"] , "(|+|) x y = x" ==> ["|+|"] , "x |+| y = x" ==> ["|+|"] , "(!) x y = x" ==> ["!"] , "--- my comment" ==> [] ] strictMatchTests = testGroup "strict match (!)" [ "f !x y = x" ==> ["f"] , "f x !y = x" ==> ["f"] , "f !x !y = x" ==> ["f"] , "f ! x y = x" ==> ["f"] -- this one is a bit controversial but it seems to be the way ghc parses it , "f ! x = x" ==> ["f"] , "(:*:) !(x :+: y) z = x" ==> [":*:"] , "(:*:) !(!x :+: !y) !z = x" ==> [":*:"] , "(:*:) !((:+:) x y) z = x" ==> [":*:"] , "(:*:) !((:+:) !x !y) !z = x" ==> [":*:"] , "(!) :: a -> b -> a\n\ \(!) x y = x" ==> ["!"] -- this is a degenerate case since even ghc treats ! here as -- a BangPatterns instead of operator , "x ! y = x" ==> ["x"] ] lazyMatchTests = testGroup "lazy match (~)" [ "f ~x y = x" ==> ["f"] , "f x ~y = x" ==> ["f"] , "f ~x ~y = x" ==> ["f"] , "f ~ x y = x" ==> ["f"] -- this one is a bit controversial but it seems to be the way ghc parses it , "f ~ x = x" ==> ["f"] , "(:*:) ~(x :+: y) z = x" ==> [":*:"] , "(:*:) ~(~x :+: ~y) ~z = x" ==> [":*:"] , "(:*:) ~((:+:) x y) z = x" ==> [":*:"] , "(:*:) ~((:+:) ~x ~y) ~z = x" ==> [":*:"] , "(~) :: a -> b -> a\n\ \(~) x y = x" ==> ["~"] -- this is a degenerate case since even ghc treats ~ here as -- a BangPatterns instead of operator , "x ~ y = x" ==> ["x"] ] atPatternsTests = testGroup "at patterns (@)" [ "f z@x y = z" ==> ["f"] , "f x z'@y = z'" ==> ["f"] , "f z@x z'@y = z" ==> ["f"] , "f z@(Foo _) z'@y = z" ==> ["f"] , "f z@(Foo _) z'@(Bar _) = z" ==> ["f"] , "f z @ x y = z" ==> ["f"] , "f z @ (x : xs) = z: [x: xs]" ==> ["f"] , "f z @ (x : zs @ xs) = z: [x: zs]" ==> ["f"] , "f z @ (zz @x : zs @ xs) = z: [zz: zs]" ==> ["f"] , "(:*:) zzz@(x :+: y) z = x" ==> [":*:"] , "(:*:) zzz@(zx@x :+: zy@y) zz@z = x" ==> [":*:"] , "(:*:) zzz@((:+:) x y) z = x" ==> [":*:"] , "(:*:) zzz@((:+:) zs@x zs@y) zz@z = x" ==> [":*:"] , "f z@(!x) ~y = x" ==> ["f"] ] testClass :: TestTree testClass = testGroup "class" [ "class (X x) => C a b where\n\tm :: a->b\n\tn :: c\n" ==> ["C", "m", "n"] , "class (X x) ⇒ C a b where\n\tm ∷ a→b\n\tn ∷ c\n" ==> ["C", "m", "n"] , "class (X x) => C a b | a -> b where\n\tm :: a->b\n\tn :: c\n" ==> ["C", "m", "n"] , "class (X x) ⇒ C a b | a → b where\n\tm ∷ a→b\n\tn ∷ c\n" ==> ["C", "m", "n"] , "class A a where f :: X\n" ==> ["A", "f"] -- indented inside where , "class X where\n\ta, (+) :: X\n" ==> ["+", "X", "a"] , "class X where\n\ta :: X\n\tb, c :: Y" ==> ["X", "a", "b", "c"] , "class X\n\twhere\n\ta :: X\n\tb, c :: Y" ==> ["X", "a", "b", "c"] , "class X\n\twhere\n\ta ::\n\t\tX\n\tb :: Y" ==> ["X", "a", "b"] , "class a :<: b where\n f :: a -> b" ==> [":<:", "f"] , "class (:<:) a b where\n f :: a -> b" ==> [":<:", "f"] , "class Eq a => a :<: b where\n f :: a -> b" ==> [":<:", "f"] -- , "class a :<<<: b => a :<: b where\n f :: a -> b" -- ==> -- [":<:", "f"] , "class (a :<<<: b) => a :<: b where\n f :: a -> b" ==> [":<:", "f"] , "class (a :<<<: b) ⇒ a :<: b where\n f ∷ a → b" ==> [":<:", "f"] , "class (Eq a, Ord b) => a :<: b where\n f :: a -> b" ==> [":<:", "f"] , "class (Eq a, Ord b) => (a :: (* -> *) -> *) :<: b where\n f :: a -> b" ==> [":<:", "f"] -- this is bizzarre , "class (Eq (a), Ord (f a [a])) => f `Z` a" ==> ["Z"] , "class A f where\n data F f :: *\n g :: a -> f a\n h :: f a -> a" ==> ["A", "F", "g", "h"] , "class A f where\n data F f :: *\n mkF :: f -> F f\n getF :: F f -> f" ==> ["A", "F", "getF", "mkF"] , "class A f where\n\ \ data F f :: * -- foo\n\ \ -- bar\n\ \ -- baz\n\ \ mkF :: f -> F f\n\ \ getF :: F f -> f" ==> ["A", "F", "getF", "mkF"] -- Not confused by a class context on a method. , "class X a where\n\tfoo :: Eq a => a -> a\n" ==> ["X", "foo"] ] where (==>) = test process testInstance :: TestTree testInstance = testGroup "instance" [ "instance Foo Quux where\n\ \ data Bar Quux a = QBar { frob :: a }\n\ \ | QBaz { fizz :: String }\n\ \ deriving (Show)" ==> ["QBar", "QBaz", "fizz", "frob"] , "instance Foo Quux where\n\ \ data Bar Quux a = QBar a | QBaz String deriving (Show)" ==> ["QBar", "QBaz"] , "instance Foo Quux where\n\ \ data Bar Quux a = QBar { frob :: a }\n\ \ | QBaz { fizz :: String }\n\ \ deriving (Show)\n\ \ data IMRuunningOutOfNamesHere Quux = Whatever" ==> ["QBar", "QBaz", "Whatever", "fizz", "frob"] -- in this test foo function should not affect tags found , "instance Foo Quux where\n\ \ data Bar Quux a = QBar { frob :: a }\n\ \ | QBaz { fizz :: String }\n\ \ deriving (Show)\n\ \\n\ \ foo _ = QBaz \"hey there\"" ==> ["QBar", "QBaz", "fizz", "frob"] , "instance Foo Int where foo _ = 1" ==> [] , "instance Foo Quux where\n\ \ newtype Bar Quux a = QBar a \n\ \ deriving (Show)\n\ \\n\ \ foo _ = QBaz \"hey there\"" ==> ["QBar"] , "instance Foo Quux where\n\ \ newtype Bar Quux a = QBar { frob :: a }" ==> ["QBar", "frob"] , "instance (Monoid w,MonadBaseControl b m) => MonadBaseControl b (JournalT w m) where\n\ \ newtype StM (JournalT w m) a =\n\ \ StMJournal { unStMJournal :: ComposeSt (JournalT w) m a }\n\ \ liftBaseWith = defaultLiftBaseWith StMJournal\n\ \ restoreM = defaultRestoreM unStMJournal\n\ \ {-# INLINE liftBaseWith #-}\n\ \ {-# INLINE restoreM #-}\n\ \" ==> ["StMJournal", "unStMJournal"] ] where (==>) = test process testLiterate :: TestTree testLiterate = testGroup "literate" [ "> class (X x) => C a b where\n>\tm :: a->b\n>\tn :: c\n" ==> ["C", "m", "n"] , "Test\n\\begin{code}\nclass (X x) => C a b where\n\tm :: a->b\n\tn :: c\n\\end{code}" ==> ["C", "m", "n"] ] where (==>) = test f f = map untag . fst . FastTags.process "fn.lhs" False testPatterns :: TestTree testPatterns = testGroup "patterns" [ "pattern Arrow a b = ConsT \"->\" [a, b]" ==> ["Arrow"] , "pattern Arrow a b = ConsT \"->\" [a, b]\n\ \pattern Pair a b = [a, b]" ==> ["Arrow", "Pair"] ] where (==>) = test process testFFI :: TestTree testFFI = testGroup "ffi" [ "foreign import ccall foo :: Double -> IO Double" ==> ["foo"] , "foreign import unsafe java foo :: Double -> IO Double" ==> ["foo"] , "foreign import safe stdcall foo :: Double -> IO Double" ==> ["foo"] ] where (==>) = test process process :: Text -> [String] process = map untag . fst . FastTags.process "fn.hs" False untag :: Pos TagVal -> String untag (Pos _ (TagVal _ name _)) = T.unpack name tokenize :: Text -> UnstrippedTokens tokenize = Monoid.mconcat . map (FastTags.tokenize False) . FastTags.stripCpp . FastTags.annotate "fn" extractTokens :: UnstrippedTokens -> [Text] extractTokens = map (\token -> case FastTags.valOf token of Token _ name -> name Newline n -> T.pack ("nl " ++ show n)) . FastTags.unstrippedTokensOf