2014-04-10 09:25:17 +04:00
|
|
|
{-# LANGUAGE DeriveGeneric #-}
|
2014-04-07 03:29:46 +04:00
|
|
|
|
2014-04-05 07:55:53 +04:00
|
|
|
module Main where
|
|
|
|
|
|
|
|
import Database.Bloodhound.Client
|
2014-04-07 03:29:46 +04:00
|
|
|
import Data.Aeson
|
|
|
|
import Data.DeriveTH
|
2014-04-10 07:24:46 +04:00
|
|
|
import Data.Either (Either(..))
|
2014-04-07 03:29:46 +04:00
|
|
|
import Data.Maybe (fromJust)
|
2014-04-07 04:53:53 +04:00
|
|
|
import Data.Time.Calendar (Day(..))
|
|
|
|
import Data.Time.Clock (secondsToDiffTime, UTCTime(..))
|
2014-04-07 03:29:46 +04:00
|
|
|
import Data.Text (Text)
|
|
|
|
import GHC.Generics (Generic)
|
2014-04-05 07:55:53 +04:00
|
|
|
import Network.HTTP.Conduit
|
|
|
|
import qualified Network.HTTP.Types.Status as NHTS
|
|
|
|
import Test.Hspec
|
|
|
|
|
2014-04-11 01:39:10 +04:00
|
|
|
testServer = Server "http://localhost:9200"
|
|
|
|
testIndex = "twitter"
|
|
|
|
testMapping = "tweet"
|
2014-04-11 04:54:40 +04:00
|
|
|
|
|
|
|
validateStatus resp expected =
|
|
|
|
(NHTS.statusCode $ responseStatus resp)
|
|
|
|
`shouldBe` (expected :: Int)
|
|
|
|
|
2014-04-07 03:29:46 +04:00
|
|
|
createExampleIndex = createIndex testServer defaultIndexSettings testIndex
|
|
|
|
deleteExampleIndex = deleteIndex testServer testIndex
|
|
|
|
|
2014-04-11 01:39:10 +04:00
|
|
|
data Location = Location { lat :: Double
|
|
|
|
, lon :: Double } deriving (Eq, Generic, Show)
|
|
|
|
|
2014-04-10 09:25:17 +04:00
|
|
|
data Tweet = Tweet { user :: Text
|
2014-04-07 03:29:46 +04:00
|
|
|
, postDate :: UTCTime
|
2014-04-11 01:39:10 +04:00
|
|
|
, message :: Text
|
|
|
|
, location :: Location }
|
2014-04-07 03:29:46 +04:00
|
|
|
deriving (Eq, Generic, Show)
|
|
|
|
|
2014-04-11 01:39:10 +04:00
|
|
|
instance ToJSON Tweet
|
2014-04-07 03:29:46 +04:00
|
|
|
instance FromJSON Tweet
|
2014-04-11 01:39:10 +04:00
|
|
|
instance ToJSON Location
|
|
|
|
instance FromJSON Location
|
|
|
|
|
|
|
|
data TweetMapping = TweetMapping deriving (Eq, Show)
|
|
|
|
|
|
|
|
instance ToJSON TweetMapping where
|
|
|
|
toJSON TweetMapping =
|
|
|
|
object ["tweet" .=
|
|
|
|
object ["properties" .=
|
|
|
|
object ["location" .= object ["type" .= ("geo_point" :: Text)]]]]
|
2014-04-07 03:29:46 +04:00
|
|
|
|
2014-04-07 04:53:53 +04:00
|
|
|
exampleTweet = Tweet { user = "bitemyapp"
|
2014-04-07 07:51:30 +04:00
|
|
|
, postDate = UTCTime
|
|
|
|
(ModifiedJulianDay 55000)
|
2014-04-07 04:53:53 +04:00
|
|
|
(secondsToDiffTime 10)
|
2014-04-11 01:39:10 +04:00
|
|
|
, message = "Use haskell!"
|
|
|
|
, location = Location 40.12 (-71.34) }
|
2014-04-05 07:55:53 +04:00
|
|
|
|
2014-04-10 09:25:17 +04:00
|
|
|
insertData :: IO ()
|
|
|
|
insertData = do
|
|
|
|
let encoded = encode exampleTweet
|
|
|
|
_ <- deleteExampleIndex
|
|
|
|
created <- createExampleIndex
|
2014-04-11 01:39:10 +04:00
|
|
|
mappingCreated <- createMapping testServer testIndex testMapping TweetMapping
|
|
|
|
docCreated <- indexDocument testServer testIndex testMapping exampleTweet "1"
|
|
|
|
_ <- refreshIndex testServer testIndex
|
2014-04-10 09:25:17 +04:00
|
|
|
return ()
|
|
|
|
|
2014-04-11 01:39:10 +04:00
|
|
|
searchTweet :: Search -> IO (Either String Tweet)
|
|
|
|
searchTweet search = do
|
2014-04-11 01:49:15 +04:00
|
|
|
reply <- searchByIndex testServer testIndex search
|
2014-04-10 09:25:17 +04:00
|
|
|
let result = eitherDecode (responseBody reply) :: Either String (SearchResult Tweet)
|
|
|
|
let myTweet = fmap (hitSource . head . hits . searchHits) result
|
|
|
|
return myTweet
|
|
|
|
|
2014-04-11 04:15:10 +04:00
|
|
|
searchExpectNoResults :: Search -> IO ()
|
|
|
|
searchExpectNoResults search = do
|
|
|
|
reply <- searchByIndex testServer testIndex search
|
|
|
|
let result = eitherDecode (responseBody reply) :: Either String (SearchResult Tweet)
|
|
|
|
let emptyHits = fmap (hits . searchHits) result
|
|
|
|
emptyHits `shouldBe` Right []
|
|
|
|
|
2014-04-05 07:55:53 +04:00
|
|
|
main :: IO ()
|
|
|
|
main = hspec $ do
|
2014-04-11 04:54:40 +04:00
|
|
|
|
2014-04-10 10:41:21 +04:00
|
|
|
describe "index create/delete API" $ do
|
|
|
|
it "creates and then deletes the requested index" $ do
|
|
|
|
-- priming state.
|
|
|
|
_ <- deleteExampleIndex
|
|
|
|
resp <- createExampleIndex
|
|
|
|
deleteResp <- deleteExampleIndex
|
|
|
|
validateStatus resp 200
|
|
|
|
validateStatus deleteResp 200
|
2014-04-10 09:25:17 +04:00
|
|
|
|
2014-04-07 03:29:46 +04:00
|
|
|
describe "document API" $ do
|
|
|
|
it "indexes, gets, and then deletes the generated document" $ do
|
2014-04-10 10:41:21 +04:00
|
|
|
_ <- insertData
|
2014-04-11 01:49:15 +04:00
|
|
|
docInserted <- getDocument testServer testIndex testMapping "1"
|
2014-04-11 04:54:40 +04:00
|
|
|
let newTweet = eitherDecode
|
|
|
|
(responseBody docInserted) :: Either String (EsResult Tweet)
|
2014-04-10 09:25:17 +04:00
|
|
|
fmap _source newTweet `shouldBe` Right exampleTweet
|
|
|
|
|
2014-04-11 01:39:10 +04:00
|
|
|
describe "filtering API" $ do
|
|
|
|
it "returns document for composed boolmatch and identity" $ do
|
|
|
|
_ <- insertData
|
|
|
|
let queryFilter = BoolFilter (MustMatch (Term "user" "bitemyapp") False)
|
|
|
|
<&&> IdentityFilter
|
|
|
|
let search = Search Nothing (Just queryFilter)
|
|
|
|
myTweet <- searchTweet search
|
|
|
|
myTweet `shouldBe` Right exampleTweet
|
|
|
|
|
2014-04-11 04:54:40 +04:00
|
|
|
it "returns document for existential filter" $ do
|
2014-04-11 01:39:10 +04:00
|
|
|
_ <- insertData
|
|
|
|
let search = Search Nothing (Just (ExistsFilter "user"))
|
|
|
|
myTweet <- searchTweet search
|
|
|
|
myTweet `shouldBe` Right exampleTweet
|
|
|
|
|
2014-04-11 04:54:40 +04:00
|
|
|
it "returns document for geo boundingbox filter" $ do
|
2014-04-10 09:25:17 +04:00
|
|
|
_ <- insertData
|
2014-04-11 01:39:10 +04:00
|
|
|
let box = GeoBoundingBox (LatLon 40.73 (-74.1)) (LatLon 40.10 (-71.12))
|
|
|
|
let bbConstraint = GeoBoundingBoxConstraint "tweet.location" box False
|
|
|
|
let geoFilter = GeoBoundingBoxFilter bbConstraint GeoFilterMemory
|
|
|
|
let search = Search Nothing (Just geoFilter)
|
|
|
|
myTweet <- searchTweet search
|
2014-04-10 09:25:17 +04:00
|
|
|
myTweet `shouldBe` Right exampleTweet
|
2014-04-11 01:49:15 +04:00
|
|
|
|
2014-04-11 04:54:40 +04:00
|
|
|
it "doesn't return document for nonsensical boundingbox filter" $ do
|
2014-04-11 01:49:15 +04:00
|
|
|
_ <- insertData
|
|
|
|
let box = GeoBoundingBox (LatLon 0.73 (-4.1)) (LatLon 0.10 (-1.12))
|
|
|
|
let bbConstraint = GeoBoundingBoxConstraint "tweet.location" box False
|
|
|
|
let geoFilter = GeoBoundingBoxFilter bbConstraint GeoFilterMemory
|
|
|
|
let search = Search Nothing (Just geoFilter)
|
2014-04-11 04:15:10 +04:00
|
|
|
searchExpectNoResults search
|
2014-04-11 04:01:42 +04:00
|
|
|
|
2014-04-11 04:54:40 +04:00
|
|
|
it "returns document for geo distance filter" $ do
|
2014-04-11 04:01:42 +04:00
|
|
|
_ <- insertData
|
2014-04-11 04:15:10 +04:00
|
|
|
let geoPoint = GeoPoint "tweet.location" (LatLon 40.12 (-71.34))
|
2014-04-11 04:01:42 +04:00
|
|
|
let distance = Distance 10.0 Miles
|
|
|
|
let optimizeBbox = OptimizeGeoFilterType GeoFilterMemory
|
2014-04-11 04:15:10 +04:00
|
|
|
let geoFilter = GeoDistanceFilter geoPoint distance SloppyArc optimizeBbox False
|
2014-04-11 04:01:42 +04:00
|
|
|
let search = Search Nothing (Just geoFilter)
|
|
|
|
myTweet <- searchTweet search
|
|
|
|
myTweet `shouldBe` Right exampleTweet
|
2014-04-11 04:15:10 +04:00
|
|
|
|
2014-04-11 04:54:40 +04:00
|
|
|
it "returns document for geo distance range filter" $ do
|
2014-04-11 04:15:10 +04:00
|
|
|
_ <- insertData
|
|
|
|
let geoPoint = GeoPoint "tweet.location" (LatLon 40.12 (-71.34))
|
|
|
|
let distanceRange = DistanceRange (Distance 0.0 Miles) (Distance 10.0 Miles)
|
|
|
|
let geoFilter = GeoDistanceRangeFilter geoPoint distanceRange
|
|
|
|
let search = Search Nothing (Just geoFilter)
|
|
|
|
myTweet <- searchTweet search
|
|
|
|
myTweet `shouldBe` Right exampleTweet
|
|
|
|
|
2014-04-11 04:54:40 +04:00
|
|
|
it "doesn't return document for wild geo distance range filter" $ do
|
2014-04-11 04:15:10 +04:00
|
|
|
_ <- insertData
|
|
|
|
let geoPoint = GeoPoint "tweet.location" (LatLon 40.12 (-71.34))
|
|
|
|
let distanceRange = DistanceRange (Distance 100.0 Miles) (Distance 1000.0 Miles)
|
|
|
|
let geoFilter = GeoDistanceRangeFilter geoPoint distanceRange
|
|
|
|
let search = Search Nothing (Just geoFilter)
|
|
|
|
searchExpectNoResults search
|
2014-04-11 04:54:40 +04:00
|
|
|
|
|
|
|
it "returns document for geo polygon filter" $ do
|
|
|
|
_ <- insertData
|
|
|
|
let points = [LatLon 40.0 (-70.00),
|
|
|
|
LatLon 40.0 (-72.00),
|
|
|
|
LatLon 41.0 (-70.00),
|
|
|
|
LatLon 41.0 (-72.00)]
|
|
|
|
let geoFilter = GeoPolygonFilter "tweet.location" points
|
|
|
|
let search = Search Nothing (Just geoFilter)
|
|
|
|
myTweet <- searchTweet search
|
|
|
|
myTweet `shouldBe` Right exampleTweet
|
|
|
|
|
|
|
|
it "doesn't return document for bad geo polygon filter" $ do
|
|
|
|
_ <- insertData
|
|
|
|
let points = [LatLon 40.0 (-70.00),
|
|
|
|
LatLon 40.0 (-71.00),
|
|
|
|
LatLon 41.0 (-70.00),
|
|
|
|
LatLon 41.0 (-71.00)]
|
|
|
|
let geoFilter = GeoPolygonFilter "tweet.location" points
|
|
|
|
let search = Search Nothing (Just geoFilter)
|
|
|
|
searchExpectNoResults search
|
2014-04-11 05:04:24 +04:00
|
|
|
|
|
|
|
it "returns document for ids filter" $ do
|
|
|
|
_ <- insertData
|
|
|
|
let filter = IdsFilter "tweet" ["1"]
|
|
|
|
let search = Search Nothing (Just filter)
|
|
|
|
myTweet <- searchTweet search
|
|
|
|
myTweet `shouldBe` Right exampleTweet
|