add/remove/edit schedule UI working

Once I built the basic widget, it turned out to be rather easy to replicate
it once per scheduled activity and wire it all up to a fully working UI.

This does abuse yesod's form handling a bit, but I think it's ok.
And it would be nice to have it all ajax-y, so that saving one modified
form won't lose any modifications to other forms. But for now, a nice
simple 115 line of code implementation is a win.

This late night hack session commit was sponsored by Andrea Rota.
This commit is contained in:
Joey Hess 2013-10-11 02:57:36 -04:00
parent aa5b2f1c3c
commit 88ec6eff15
8 changed files with 103 additions and 20 deletions

View file

@ -11,6 +11,7 @@
module Assistant.WebApp.Configurators.Fsck where module Assistant.WebApp.Configurators.Fsck where
import qualified Data.Map as M import qualified Data.Map as M
import qualified Data.Set as S
import qualified Data.Text as T import qualified Data.Text as T
import qualified Text.Hamlet as Hamlet import qualified Text.Hamlet as Hamlet
@ -18,18 +19,44 @@ import Assistant.WebApp.Common
import Types.ScheduledActivity import Types.ScheduledActivity
import Utility.HumanTime import Utility.HumanTime
import Utility.Scheduled import Utility.Scheduled
import Logs.Schedule
import Annex.UUID
fsckForm :: ScheduledActivity -> Hamlet.Html -> MkMForm ScheduledActivity {- This adds a form to the page. It does not handle posting of the form,
fsckForm (ScheduledSelfFsck (Schedule r t) d) msg = do - because unlike a typical yesod form that posts using the same url
- that generated it, this form posts using one of two other routes. -}
fsckForm :: Bool -> ScheduledActivity -> Widget
fsckForm new activity = do
u <- liftAnnex getUUID
let action = if new
then AddActivityR u
else ChangeActivityR u activity
((res, form), enctype) <- liftH $ runFormPost $ fsckForm' new activity
case res of
FormSuccess _ -> noop
_ -> $(widgetFile "configurators/fsck/form")
{- This does not display a form, but it does get it from a post, and run
- some Annex action on it. -}
withFsckForm :: (ScheduledActivity -> Annex ()) -> Handler ()
withFsckForm a = do
((res, _form), _enctype) <- runFormPost $ fsckForm' False defaultFsck
case res of
FormSuccess activity -> liftAnnex $ a activity
_ -> noop
fsckForm' :: Bool -> ScheduledActivity -> Hamlet.Html -> MkMForm ScheduledActivity
fsckForm' new activity@(ScheduledSelfFsck (Schedule r t) d) msg = do
(durationRes, durationView) <- mreq intField "" (Just $ durationSeconds d `quot` 60 ) (durationRes, durationView) <- mreq intField "" (Just $ durationSeconds d `quot` 60 )
(timeRes, timeView) <- mreq (selectFieldList times) "" (Just t) (timeRes, timeView) <- mreq (selectFieldList times) "" (Just t)
(recurranceRes, recurranceView) <- mreq (selectFieldList recurrances) "" (Just r) (recurranceRes, recurranceView) <- mreq (selectFieldList recurrances) "" (Just r)
let form = do let form = do
webAppFormAuthToken webAppFormAuthToken
$(widgetFile "configurators/fsck/form") u <- liftAnnex getUUID
$(widgetFile "configurators/fsck/formcontent")
let formresult = ScheduledSelfFsck let formresult = ScheduledSelfFsck
<$> (Schedule <$> recurranceRes <*> timeRes) <$> (Schedule <$> recurranceRes <*> timeRes)
<*> (Duration <$> durationRes) <*> (Duration <$> ((60 *) <$> durationRes))
return (formresult, form) return (formresult, form)
where where
times :: [(Text, ScheduledTime)] times :: [(Text, ScheduledTime)]
@ -55,7 +82,7 @@ fsckForm (ScheduledSelfFsck (Schedule r t) d) msg = do
ensurevalue v desc l = case M.lookup v (M.fromList $ map (\(x,y) -> (y,x)) l) of ensurevalue v desc l = case M.lookup v (M.fromList $ map (\(x,y) -> (y,x)) l) of
Just _ -> l Just _ -> l
Nothing -> (desc, v) : l Nothing -> (desc, v) : l
fsckForm (ScheduledRemoteFsck u s d) _msg = error "TODO" fsckForm' new (ScheduledRemoteFsck u s d) _msg = error "TODO"
defaultFsck :: ScheduledActivity defaultFsck :: ScheduledActivity
defaultFsck = ScheduledSelfFsck (Schedule Daily AnyTime) (Duration $ 60*60) defaultFsck = ScheduledSelfFsck (Schedule Daily AnyTime) (Duration $ 60*60)
@ -64,7 +91,25 @@ getConfigFsckR :: Handler Html
getConfigFsckR = postConfigFsckR getConfigFsckR = postConfigFsckR
postConfigFsckR :: Handler Html postConfigFsckR :: Handler Html
postConfigFsckR = page "Consistency checks" (Just Configuration) $ do postConfigFsckR = page "Consistency checks" (Just Configuration) $ do
((res, form), enctype) <- liftH $ runFormPost $ fsckForm defaultFsck checks <- liftAnnex $ S.toList <$> (scheduleGet =<< getUUID)
case res of $(widgetFile "configurators/fsck")
FormSuccess s -> error "TODO"
_ -> $(widgetFile "configurators/fsck") getRemoveActivityR :: UUID -> ScheduledActivity -> Handler Html
getRemoveActivityR u activity = do
liftAnnex $ scheduleRemove u activity
redirect ConfigFsckR
getAddActivityR :: UUID -> Handler Html
getAddActivityR = postAddActivityR
postAddActivityR :: UUID -> Handler Html
postAddActivityR u = do
withFsckForm $ scheduleAdd u
redirect ConfigFsckR
getChangeActivityR :: UUID -> ScheduledActivity -> Handler Html
getChangeActivityR = postChangeActivityR
postChangeActivityR :: UUID -> ScheduledActivity -> Handler Html
postChangeActivityR u oldactivity = do
withFsckForm $ \newactivity -> scheduleChange u $
S.insert newactivity . S.delete oldactivity
redirect ConfigFsckR

View file

@ -23,6 +23,7 @@ import Utility.Yesod
import Logs.Transfer import Logs.Transfer
import Utility.Gpg (KeyId) import Utility.Gpg (KeyId)
import Build.SysConfig (packageversion) import Build.SysConfig (packageversion)
import Types.ScheduledActivity
import Yesod.Static import Yesod.Static
import Text.Hamlet import Text.Hamlet
@ -211,3 +212,8 @@ instance PathPiece RepoSelector where
instance PathPiece ThreadName where instance PathPiece ThreadName where
toPathPiece = pack . show toPathPiece = pack . show
fromPathPiece = readish . unpack fromPathPiece = readish . unpack
instance PathPiece ScheduledActivity where
toPathPiece = pack . show
fromPathPiece = readish . unpack

View file

@ -84,6 +84,10 @@
/config/repository/delete/finish/#UUID FinishDeleteRepositoryR GET /config/repository/delete/finish/#UUID FinishDeleteRepositoryR GET
/config/repository/delete/here DeleteCurrentRepositoryR GET POST /config/repository/delete/here DeleteCurrentRepositoryR GET POST
/config/activity/add/#UUID AddActivityR GET POST
/config/activity/change/#UUID/#ScheduledActivity ChangeActivityR GET POST
/config/activity/remove/#UUID/#ScheduledActivity RemoveActivityR GET
/transfers/#NotificationId TransfersR GET /transfers/#NotificationId TransfersR GET
/notifier/transfers NotifierTransfersR GET /notifier/transfers NotifierTransfersR GET

View file

@ -8,6 +8,9 @@
module Logs.Schedule ( module Logs.Schedule (
scheduleLog, scheduleLog,
scheduleSet, scheduleSet,
scheduleAdd,
scheduleRemove,
scheduleChange,
scheduleGet, scheduleGet,
scheduleMap, scheduleMap,
getLastRunTimes, getLastRunTimes,
@ -46,6 +49,15 @@ scheduleGet u = do
m <- scheduleMap m <- scheduleMap
return $ maybe S.empty S.fromList (M.lookup u m) return $ maybe S.empty S.fromList (M.lookup u m)
scheduleRemove :: UUID -> ScheduledActivity -> Annex ()
scheduleRemove u activity = scheduleChange u $ S.delete activity
scheduleAdd :: UUID -> ScheduledActivity -> Annex ()
scheduleAdd u activity = scheduleChange u $ S.insert activity
scheduleChange :: UUID -> (S.Set ScheduledActivity -> S.Set ScheduledActivity) -> Annex ()
scheduleChange u a = scheduleSet u . S.toList . a =<< scheduleGet u
getLastRunTimes :: Annex (M.Map ScheduledActivity LocalTime) getLastRunTimes :: Annex (M.Map ScheduledActivity LocalTime)
getLastRunTimes = do getLastRunTimes = do
f <- fromRepo gitAnnexScheduleState f <- fromRepo gitAnnexScheduleState

View file

@ -0,0 +1,2 @@
input[type=number]
width: 5em

View file

@ -9,6 +9,11 @@
Running the consistency check involves reading all the files in the # Running the consistency check involves reading all the files in the #
repository, which can take a long time if it's large. Running just a # repository, which can take a long time if it's large. Running just a #
little at a time will eventually check the whole repository. little at a time will eventually check the whole repository.
$if (not (null checks))
<p> <p>
<form method="post" .form-inline enctype=#{enctype}> Currently scheduled checks:
^{form} $forall check <- checks
^{fsckForm False check}
<p>
Add a check:
^{fsckForm True defaultFsck}

View file

@ -1,9 +1,2 @@
#{msg} <form method="post" .form-inline enctype=#{enctype} action="@{action}">
<p> ^{form}
<div .input-prepend .input-append>
Check this repository for #
^{fvInput durationView} minutes #
starting at ^{fvInput timeView}
^{fvInput recurranceView}
<button type=submit .btn .btn-primary>
Add

View file

@ -0,0 +1,16 @@
#{msg}
<p>
<div .input-prepend .input-append>
Check this repository for #
^{fvInput durationView} minutes #
^{fvInput recurranceView} #
starting at ^{fvInput timeView} #
$if new
<button type=submit .btn .btn-primary>
Add
$else
<button type=submit .btn>
Save
<a .btn href="@{RemoveActivityR u activity}">
Remove