souplesse/lib/Track.hs

108 lines
2.9 KiB
Haskell
Raw Normal View History

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,
parseFile,
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
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
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)
instance Exception BadFile
elToPoint :: Cursor -> Either SomeException Point
2024-10-29 21:22:49 +00:00
elToPoint c =
let lat = listToMaybe (attribute "lat" c) >>= asDouble
lon = listToMaybe (attribute "lon" c) >>= asDouble
ts =
listToMaybe (child c >>= element (gpxNS "time") >>= child >>= content)
>>= (Data.Time.ISO8601.parseISO8601 . Data.Text.unpack)
ele = listToMaybe (child c >>= element (gpxNS "ele") >>= child >>= content) >>= asDouble
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
asDouble v = (readMaybe :: String -> Maybe Double) (Data.Text.unpack v)
asInt v = (readMaybe :: String -> Maybe Int) (Data.Text.unpack v)
2024-10-28 23:35:36 +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")
in traverse elToPoint trkpts
2024-10-28 23:35:36 +00:00
parse :: String -> Either SomeException [Point]
parse str = do
gpx <- parseText def (T.pack str)
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
-- parseFile :: FilePath -> IO [Point]
parseFile name = do
gpx <- Text.XML.readFile def name
return $ case getPoints (fromDocument gpx) of
Left err -> []
Right points -> points
parseBS bs = do
gpx <- parseLBS def bs
getPoints (fromDocument gpx)