about summary refs log tree commit diff
diff options
context:
space:
mode:
authorWilliam Carroll <wpcarro@gmail.com>2020-07-28T09·14+0100
committerWilliam Carroll <wpcarro@gmail.com>2020-07-28T09·14+0100
commit6d9e76313d1f89dcf4c1adb7bfabd811a65bd83a (patch)
tree65b41771610fb43c4f6cc81c6b958c9acf5ee4bf
parent0637da36ccac7e609041bc8999e3da348171f95f (diff)
Partially support DELETE /trips
Allow a user to delete a trip entry from the Trips table using the Primary
Key. While this type-checks and compiles, it doesn't appear to be working as
intended. Perhaps I should use an auto-incrementing integer as the Primary
Key. I'm not sure how I want to handle this, so I'm punting for now.
-rw-r--r--src/API.hs9
-rw-r--r--src/App.hs17
-rw-r--r--src/Types.hs18
3 files changed, 39 insertions, 5 deletions
diff --git a/src/API.hs b/src/API.hs
index a42bf804b471..545aa25be777 100644
--- a/src/API.hs
+++ b/src/API.hs
@@ -14,10 +14,15 @@ type API = "user"
            :> Post '[JSON] (Maybe T.Session)
       :<|> "user"
            :> Capture "name" Text
-           :> Get  '[JSON] (Maybe T.Account)
-      :<|> "trip"
+           :> Get '[JSON] (Maybe T.Account)
+      -- Create
+      :<|> "trips"
            :> ReqBody '[JSON] T.Trip
            :> Post '[JSON] NoContent
       -- Read
       :<|> "trips"
            :> Get '[JSON] [T.Trip]
+      -- Delete
+      :<|> "trips"
+           :> ReqBody '[JSON] T.TripPK
+           :> Delete '[JSON] NoContent
diff --git a/src/App.hs b/src/App.hs
index c4203137a4fc..7747951922fa 100644
--- a/src/App.hs
+++ b/src/App.hs
@@ -21,11 +21,13 @@ server dbFile = userAddH
            :<|> userGetH
            :<|> createTripH
            :<|> listTripsH
+           :<|> deleteTripH
   where
-    userAddH newUser = liftIO $ userAdd newUser
-    userGetH name    = liftIO $ userGet name
-    createTripH trip = liftIO $ createTrip trip
+    userAddH newUser   = liftIO $ userAdd newUser
+    userGetH name      = liftIO $ userGet name
+    createTripH trip   = liftIO $ createTrip trip
     listTripsH         = liftIO $ listTrips
+    deleteTripH tripPK = liftIO $ deleteTrip tripPK
 
     -- TODO(wpcarro): Handle failed CONSTRAINTs instead of sending 500s
     userAdd :: T.Account -> IO (Maybe T.Session)
@@ -53,6 +55,15 @@ server dbFile = userAddH
     listTrips :: IO [T.Trip]
     listTrips = withConnection dbFile $ \conn -> do
       query_ conn "SELECT * FROM Trips"
+
+    -- TODO(wpcarro): Validate incoming data like startDate.
+    deleteTrip :: T.TripPK -> IO NoContent
+    deleteTrip tripPK =
+      withConnection dbFile $ \conn -> do
+        execute conn "DELETE FROM Trips WHERE username = ? AND destination = ? and startDate = ?"
+          (tripPK & T.tripPKFields)
+        pure NoContent
+
 mkApp :: FilePath -> IO Application
 mkApp dbFile = do
   pure $ serve (Proxy @ API) $ server dbFile
diff --git a/src/Types.hs b/src/Types.hs
index 112b17c53ab2..6d6b83347931 100644
--- a/src/Types.hs
+++ b/src/Types.hs
@@ -199,6 +199,24 @@ instance FromRow Trip where
                  <*> field
                  <*> field
 
+-- | The fields used as the Primary Key for a Trip entry.
+data TripPK = TripPK
+  { tripPKUsername :: Username
+  , tripPKDestination :: Destination
+  , tripPKStartDate :: Date
+  } deriving (Eq, Show, Generic)
+
+tripPKFields :: TripPK -> (Username, Destination, Date)
+tripPKFields (TripPK{ tripPKUsername
+                    , tripPKDestination
+                    , tripPKStartDate
+                    })
+  = (tripPKUsername, tripPKDestination, tripPKStartDate)
+
+-- TODO(wpcarro): Prefer shorter JSON fields like username instead of
+-- tripPKUsername.
+instance FromJSON TripPK
+
 -- | Return the tuple representation of a Trip record for SQL.
 tripFields :: Trip -> (Username, Destination, Date, Date, Comment)
 tripFields (Trip{ tripUsername