2024-10-28 23:35:36 +00:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
|
2024-10-30 21:03:11 +00:00
|
|
|
module Track
|
|
|
|
( Track,
|
2024-11-10 15:43:13 +00:00
|
|
|
module Point,
|
2024-10-31 16:18:40 +00:00
|
|
|
BadFile,
|
2024-10-30 21:03:11 +00:00
|
|
|
parse,
|
2024-10-31 18:29:57 +00:00
|
|
|
parseFile,
|
2024-11-03 18:40:31 +00:00
|
|
|
parseBS,
|
2024-10-30 21:03:11 +00:00
|
|
|
Track.length,
|
|
|
|
)
|
|
|
|
where
|
|
|
|
|
|
|
|
import Control.Exception
|
2024-11-05 23:32:01 +00:00
|
|
|
import Data.Aeson
|
|
|
|
import Data.ByteString.Lazy.Char8 qualified as L
|
2024-11-03 18:40:31 +00:00
|
|
|
import Data.Either
|
2024-10-31 16:18:40 +00:00
|
|
|
import Data.Functor ((<&>))
|
2024-10-30 21:03:11 +00:00
|
|
|
import Data.List as List
|
|
|
|
import Data.List qualified
|
|
|
|
import Data.Map as Map
|
2024-10-30 23:32:13 +00:00
|
|
|
import Data.Maybe
|
2024-10-30 21:03:11 +00:00
|
|
|
import Data.Text qualified
|
|
|
|
import Data.Text.Lazy as T
|
2024-10-27 23:13:39 +00:00
|
|
|
import Data.Time
|
2024-10-30 21:03:11 +00:00
|
|
|
import Data.Time.ISO8601 qualified
|
|
|
|
import Debug.Trace (trace, traceShow)
|
2024-11-10 22:17:18 +00:00
|
|
|
import Point
|
2024-10-31 17:13:50 +00:00
|
|
|
import Text.Read (readMaybe)
|
2024-10-28 23:35:36 +00:00
|
|
|
import Text.XML
|
|
|
|
import Text.XML.Cursor as Cursor
|
2024-11-05 23:32:01 +00:00
|
|
|
|
2024-10-29 19:35:17 +00:00
|
|
|
-- TODO do we even need this type?
|
2024-10-27 23:13:39 +00:00
|
|
|
type Track = [Point]
|
|
|
|
|
2024-10-30 17:17:48 +00:00
|
|
|
gpxNS localName =
|
|
|
|
Name localName (Just "http://www.topografix.com/GPX/1/1") Nothing
|
|
|
|
|
|
|
|
tpxNS localName =
|
|
|
|
Name localName (Just "http://www.garmin.com/xmlschemas/TrackPointExtension/v2") Nothing
|
2024-10-30 13:30:33 +00:00
|
|
|
|
2024-10-31 16:18:40 +00:00
|
|
|
data BadFile = BadFile String deriving (Show)
|
2024-10-31 00:35:19 +00:00
|
|
|
|
|
|
|
instance Exception BadFile
|
|
|
|
|
|
|
|
elToPoint :: Cursor -> Either SomeException Point
|
2024-10-29 21:22:49 +00:00
|
|
|
elToPoint c =
|
2024-11-04 23:36:43 +00:00
|
|
|
let lat = listToMaybe (attribute "lat" c) >>= asDouble
|
|
|
|
lon = listToMaybe (attribute "lon" c) >>= asDouble
|
2024-10-31 17:25:01 +00:00
|
|
|
ts =
|
|
|
|
listToMaybe (child c >>= element (gpxNS "time") >>= child >>= content)
|
|
|
|
>>= (Data.Time.ISO8601.parseISO8601 . Data.Text.unpack)
|
2024-11-04 23:36:43 +00:00
|
|
|
ele = listToMaybe (child c >>= element (gpxNS "ele") >>= child >>= content) >>= asDouble
|
2024-10-31 17:25:01 +00:00
|
|
|
gpxtpx =
|
|
|
|
child c
|
|
|
|
>>= element (gpxNS "extensions")
|
|
|
|
>>= child
|
|
|
|
>>= element (tpxNS "TrackPointExtension")
|
|
|
|
>>= child
|
|
|
|
extn n =
|
|
|
|
gpxtpx >>= element n >>= child >>= content
|
|
|
|
|
|
|
|
cadence = extn (tpxNS "cad")
|
|
|
|
hr = extn (tpxNS "hr")
|
|
|
|
power = extn "{http://www.garmin.com/xmlschemas/PowerExtension/v1}PowerInWatts"
|
|
|
|
in if isJust lat && isJust lon && isJust ts
|
|
|
|
then
|
|
|
|
Right $
|
|
|
|
Point
|
|
|
|
(Pos (fromJust lat) (fromJust lon) ele)
|
|
|
|
(fromJust ts)
|
|
|
|
(listToMaybe cadence >>= asInt)
|
|
|
|
(listToMaybe power >>= asInt)
|
|
|
|
(listToMaybe hr >>= asInt)
|
|
|
|
else Left (toException (BadFile "missing a required attribute"))
|
|
|
|
where
|
2024-11-04 23:36:43 +00:00
|
|
|
asDouble v = (readMaybe :: String -> Maybe Double) (Data.Text.unpack v)
|
2024-10-31 17:25:01 +00:00
|
|
|
asInt v = (readMaybe :: String -> Maybe Int) (Data.Text.unpack v)
|
2024-10-28 23:35:36 +00:00
|
|
|
|
2024-10-31 00:35:19 +00:00
|
|
|
getPoints :: Cursor -> Either SomeException [Point]
|
2024-10-28 23:35:36 +00:00
|
|
|
getPoints c =
|
2024-10-30 21:03:11 +00:00
|
|
|
let trkpts =
|
|
|
|
element (gpxNS "gpx") c
|
|
|
|
>>= child
|
|
|
|
>>= element (gpxNS "trk")
|
|
|
|
>>= descendant
|
|
|
|
>>= element (gpxNS "trkpt")
|
2024-10-31 00:35:19 +00:00
|
|
|
in traverse elToPoint trkpts
|
2024-10-28 23:35:36 +00:00
|
|
|
|
2024-10-29 19:33:06 +00:00
|
|
|
parse :: String -> Either SomeException [Point]
|
2024-10-29 21:44:49 +00:00
|
|
|
parse str = do
|
|
|
|
gpx <- parseText def (T.pack str)
|
2024-10-31 00:35:19 +00:00
|
|
|
getPoints (fromDocument gpx)
|
2024-10-28 23:35:36 +00:00
|
|
|
|
2024-10-27 23:13:39 +00:00
|
|
|
length :: Track -> Int
|
2024-10-30 14:25:54 +00:00
|
|
|
length = Data.List.length
|
2024-10-31 18:29:57 +00:00
|
|
|
|
|
|
|
-- parseFile :: FilePath -> IO [Point]
|
|
|
|
parseFile name = do
|
|
|
|
gpx <- Text.XML.readFile def name
|
|
|
|
return $ case getPoints (fromDocument gpx) of
|
|
|
|
Left err -> []
|
|
|
|
Right points -> points
|
2024-11-03 18:40:31 +00:00
|
|
|
|
|
|
|
parseBS bs = do
|
|
|
|
gpx <- parseLBS def bs
|
|
|
|
getPoints (fromDocument gpx)
|