fix noscript mode to not allocate notification ids on each refresh
Now the javascript does an ajax call at the start to request the url to use to poll, and the notification id is generated then, once we know javascript is working.
This commit is contained in:
parent
2821f9f976
commit
6e40aed948
4 changed files with 65 additions and 38 deletions
|
@ -36,6 +36,7 @@ import Text.Hamlet
|
||||||
import Network.Socket (PortNumber)
|
import Network.Socket (PortNumber)
|
||||||
import Text.Blaze.Renderer.String
|
import Text.Blaze.Renderer.String
|
||||||
import Data.Text (Text, pack, unpack)
|
import Data.Text (Text, pack, unpack)
|
||||||
|
import qualified Data.Text as T
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import Control.Concurrent.STM
|
import Control.Concurrent.STM
|
||||||
|
|
||||||
|
@ -93,6 +94,8 @@ mkYesod "WebApp" [parseRoutes|
|
||||||
/noscriptauto NoScriptAutoR GET
|
/noscriptauto NoScriptAutoR GET
|
||||||
/transfers/#NotificationId TransfersR GET
|
/transfers/#NotificationId TransfersR GET
|
||||||
/sidebar/#NotificationId SideBarR GET
|
/sidebar/#NotificationId SideBarR GET
|
||||||
|
/notifier/transfers NotifierTransfersR GET
|
||||||
|
/notifier/sidebar NotifierSideBarR GET
|
||||||
/closealert/#AlertId CloseAlert GET
|
/closealert/#AlertId CloseAlert GET
|
||||||
/config ConfigR GET
|
/config ConfigR GET
|
||||||
/addrepository AddRepositoryR GET
|
/addrepository AddRepositoryR GET
|
||||||
|
@ -137,18 +140,39 @@ instance Yesod WebApp where
|
||||||
- The widget should have a html element with an id=ident, which will be
|
- The widget should have a html element with an id=ident, which will be
|
||||||
- replaced when it's updated.
|
- replaced when it's updated.
|
||||||
-
|
-
|
||||||
- Updating is done by getting html from the gethtml route.
|
- The geturl route should return the notifier url to use for polling.
|
||||||
-
|
-
|
||||||
- ms_delay is how long to delay between AJAX updates
|
- ms_delay is how long to delay between AJAX updates
|
||||||
- ms_startdelay is how long to delay before updating with AJAX at the start
|
- ms_startdelay is how long to delay before updating with AJAX at the start
|
||||||
-}
|
-}
|
||||||
autoUpdate :: Text -> Route WebApp -> Int -> Int -> Widget
|
autoUpdate :: Text -> Route WebApp -> Int -> Int -> Widget
|
||||||
autoUpdate ident gethtml ms_delay ms_startdelay = do
|
autoUpdate ident geturl ms_delay ms_startdelay = do
|
||||||
let delay = show ms_delay
|
let delay = show ms_delay
|
||||||
let startdelay = show ms_startdelay
|
let startdelay = show ms_startdelay
|
||||||
addScript $ StaticR longpolling_js
|
addScript $ StaticR longpolling_js
|
||||||
$(widgetFile "longpolling")
|
$(widgetFile "longpolling")
|
||||||
|
|
||||||
|
{- Notifier urls are requested by the javascript, to avoid allocation
|
||||||
|
- of NotificationIds when noscript pages are loaded. This constructs a
|
||||||
|
- notifier url for a given Route and NotificationBroadcaster.
|
||||||
|
-}
|
||||||
|
notifierUrl :: (NotificationId -> Route WebApp) -> (DaemonStatus -> NotificationBroadcaster) -> Handler RepPlain
|
||||||
|
notifierUrl route selector = do
|
||||||
|
(urlbits, _params) <- renderRoute . route <$> newNotifier selector
|
||||||
|
webapp <- getYesod
|
||||||
|
return $ RepPlain $ toContent $ T.concat
|
||||||
|
[ "/"
|
||||||
|
, T.intercalate "/" urlbits
|
||||||
|
, "?auth="
|
||||||
|
, secretToken webapp
|
||||||
|
]
|
||||||
|
|
||||||
|
getNotifierTransfersR :: Handler RepPlain
|
||||||
|
getNotifierTransfersR = notifierUrl TransfersR transferNotifier
|
||||||
|
|
||||||
|
getNotifierSideBarR :: Handler RepPlain
|
||||||
|
getNotifierSideBarR = notifierUrl SideBarR alertNotifier
|
||||||
|
|
||||||
{- A display of currently running and queued transfers.
|
{- A display of currently running and queued transfers.
|
||||||
-
|
-
|
||||||
- Or, if there have never been any this run, an intro display. -}
|
- Or, if there have never been any this run, an intro display. -}
|
||||||
|
@ -159,7 +183,8 @@ transfersDisplay warnNoScript = do
|
||||||
M.toList . currentTransfers
|
M.toList . currentTransfers
|
||||||
<$> liftIO (getDaemonStatus $ daemonStatus webapp)
|
<$> liftIO (getDaemonStatus $ daemonStatus webapp)
|
||||||
queued <- liftIO $ getTransferQueue $ transferQueue webapp
|
queued <- liftIO $ getTransferQueue $ transferQueue webapp
|
||||||
let ident = transfersDisplayIdent
|
let ident = "transfers"
|
||||||
|
autoUpdate ident NotifierTransfersR (10 :: Int) (10 :: Int)
|
||||||
let transfers = current ++ queued
|
let transfers = current ++ queued
|
||||||
if null transfers
|
if null transfers
|
||||||
then ifM (lift $ showIntro <$> getWebAppState)
|
then ifM (lift $ showIntro <$> getWebAppState)
|
||||||
|
@ -168,9 +193,7 @@ transfersDisplay warnNoScript = do
|
||||||
)
|
)
|
||||||
else $(widgetFile "transfers")
|
else $(widgetFile "transfers")
|
||||||
|
|
||||||
transfersDisplayIdent :: Text
|
{- An intro message, and list of repositories. -}
|
||||||
transfersDisplayIdent = "transfers"
|
|
||||||
|
|
||||||
introDisplay :: Text -> Widget
|
introDisplay :: Text -> Widget
|
||||||
introDisplay ident = do
|
introDisplay ident = do
|
||||||
webapp <- lift getYesod
|
webapp <- lift getYesod
|
||||||
|
@ -206,8 +229,8 @@ getTransfersR nid = do
|
||||||
page <- widgetToPageContent $ transfersDisplay False
|
page <- widgetToPageContent $ transfersDisplay False
|
||||||
hamletToRepHtml $ [hamlet|^{pageBody page}|]
|
hamletToRepHtml $ [hamlet|^{pageBody page}|]
|
||||||
|
|
||||||
sideBarDisplay :: Bool -> Widget
|
sideBarDisplay :: Widget
|
||||||
sideBarDisplay noScript = do
|
sideBarDisplay = do
|
||||||
let content = do
|
let content = do
|
||||||
{- Any yesod message appears as the first alert. -}
|
{- Any yesod message appears as the first alert. -}
|
||||||
maybe noop rendermessage =<< lift getMessage
|
maybe noop rendermessage =<< lift getMessage
|
||||||
|
@ -218,14 +241,9 @@ sideBarDisplay noScript = do
|
||||||
<$> liftIO (getDaemonStatus $ daemonStatus webapp)
|
<$> liftIO (getDaemonStatus $ daemonStatus webapp)
|
||||||
mapM_ renderalert $
|
mapM_ renderalert $
|
||||||
take displayAlerts $ reverse $ sortAlertPairs alertpairs
|
take displayAlerts $ reverse $ sortAlertPairs alertpairs
|
||||||
ident <- lift newIdent
|
let ident = "sidebar"
|
||||||
$(widgetFile "sidebar")
|
$(widgetFile "sidebar")
|
||||||
|
autoUpdate ident NotifierSideBarR (10 :: Int) (10 :: Int)
|
||||||
unless noScript $ do
|
|
||||||
{- Set up automatic updates of the sidebar
|
|
||||||
- when alerts come in. -}
|
|
||||||
nid <- lift $ newNotifier alertNotifier
|
|
||||||
autoUpdate ident (SideBarR nid) (10 :: Int) (10 :: Int)
|
|
||||||
where
|
where
|
||||||
bootstrapclass Activity = "alert-info"
|
bootstrapclass Activity = "alert-info"
|
||||||
bootstrapclass Warning = "alert"
|
bootstrapclass Warning = "alert"
|
||||||
|
@ -264,7 +282,7 @@ getSideBarR :: NotificationId -> Handler RepHtml
|
||||||
getSideBarR nid = do
|
getSideBarR nid = do
|
||||||
waitNotifier alertNotifier nid
|
waitNotifier alertNotifier nid
|
||||||
|
|
||||||
page <- widgetToPageContent $ sideBarDisplay True
|
page <- widgetToPageContent sideBarDisplay
|
||||||
hamletToRepHtml $ [hamlet|^{pageBody page}|]
|
hamletToRepHtml $ [hamlet|^{pageBody page}|]
|
||||||
|
|
||||||
{- Called by the client to close an alert. -}
|
{- Called by the client to close an alert. -}
|
||||||
|
@ -273,43 +291,39 @@ getCloseAlert i = do
|
||||||
webapp <- getYesod
|
webapp <- getYesod
|
||||||
void $ liftIO $ removeAlert (daemonStatus webapp) i
|
void $ liftIO $ removeAlert (daemonStatus webapp) i
|
||||||
|
|
||||||
dashboard :: Bool -> Bool -> Widget
|
{- The main dashboard. -}
|
||||||
dashboard noScript warnNoScript = do
|
dashboard :: Bool -> Widget
|
||||||
sideBarDisplay noScript
|
dashboard warnNoScript = do
|
||||||
transfersDisplay warnNoScript
|
sideBarDisplay
|
||||||
|
let content = transfersDisplay warnNoScript
|
||||||
|
$(widgetFile "dashboard")
|
||||||
|
|
||||||
getHomeR :: Handler RepHtml
|
getHomeR :: Handler RepHtml
|
||||||
getHomeR = defaultLayout $ do
|
getHomeR = defaultLayout $ dashboard True
|
||||||
{- Set up automatic updates for the transfers display. -}
|
|
||||||
nid <- lift $ newNotifier transferNotifier
|
|
||||||
autoUpdate transfersDisplayIdent (TransfersR nid) (10 :: Int) (10 :: Int)
|
|
||||||
|
|
||||||
dashboard False True
|
{- Same as HomeR, except with autorefreshing via meta refresh. -}
|
||||||
|
|
||||||
{- Same as HomeR, except with no javascript, so it doesn't allocate
|
|
||||||
- new resources each time the page is refreshed, and with autorefreshing
|
|
||||||
- via meta refresh. -}
|
|
||||||
getNoScriptAutoR :: Handler RepHtml
|
getNoScriptAutoR :: Handler RepHtml
|
||||||
getNoScriptAutoR = defaultLayout $ do
|
getNoScriptAutoR = defaultLayout $ do
|
||||||
let ident = NoScriptR
|
let ident = NoScriptR
|
||||||
let delayseconds = 3 :: Int
|
let delayseconds = 3 :: Int
|
||||||
let this = NoScriptAutoR
|
let this = NoScriptAutoR
|
||||||
toWidgetHead $(hamletFile $ hamletTemplate "metarefresh")
|
toWidgetHead $(hamletFile $ hamletTemplate "metarefresh")
|
||||||
dashboard True False
|
dashboard False
|
||||||
|
|
||||||
|
{- Same as HomeR, except no autorefresh at all (and no noscript warning). -}
|
||||||
getNoScriptR :: Handler RepHtml
|
getNoScriptR :: Handler RepHtml
|
||||||
getNoScriptR = defaultLayout $
|
getNoScriptR = defaultLayout $
|
||||||
dashboard True True
|
dashboard False
|
||||||
|
|
||||||
getConfigR :: Handler RepHtml
|
getConfigR :: Handler RepHtml
|
||||||
getConfigR = defaultLayout $ do
|
getConfigR = defaultLayout $ do
|
||||||
sideBarDisplay False
|
sideBarDisplay
|
||||||
setTitle "Configuration"
|
setTitle "Configuration"
|
||||||
[whamlet|<a href="@{HomeR}">main|]
|
[whamlet|<a href="@{HomeR}">main|]
|
||||||
|
|
||||||
getAddRepositoryR :: Handler RepHtml
|
getAddRepositoryR :: Handler RepHtml
|
||||||
getAddRepositoryR = defaultLayout $ do
|
getAddRepositoryR = defaultLayout $ do
|
||||||
sideBarDisplay False
|
sideBarDisplay
|
||||||
setTitle "Add repository"
|
setTitle "Add repository"
|
||||||
[whamlet|<a href="@{HomeR}">main|]
|
[whamlet|<a href="@{HomeR}">main|]
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
// Uses long-polling to update a div with a specified id,
|
// Updates a div with a specified id, by polling an url,
|
||||||
// by polling an url, which should return a new div, with the same id.
|
// which should return a new div, with the same id.
|
||||||
|
|
||||||
connfails=0;
|
connfails=0;
|
||||||
|
|
||||||
|
|
10
templates/dashboard.hamlet
Normal file
10
templates/dashboard.hamlet
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
^{content}
|
||||||
|
$if warnNoScript
|
||||||
|
<noscript>
|
||||||
|
<div .navbar .navbar-fixed-bottom>
|
||||||
|
<div .navbar-inner>
|
||||||
|
<div .container>
|
||||||
|
Javascript is disabled; cannot update in real-time.
|
||||||
|
<div .btn-group>
|
||||||
|
<a .btn .btn-primary href="@{NoScriptAutoR}">Auto-refresh every 3 seconds #
|
||||||
|
<a .btn .btn-primary href="@{NoScriptR}">Manually refresh
|
|
@ -1,9 +1,12 @@
|
||||||
function longpoll_#{ident}() {
|
function longpoll_#{ident}() {
|
||||||
longpoll('@{gethtml}', '#{ident}'
|
longpoll(longpoll_#{ident}_url, '#{ident}'
|
||||||
, function() { setTimeout(longpoll_#{ident}, #{delay}); }
|
, function() { setTimeout(longpoll_#{ident}, #{delay}); }
|
||||||
, function() { webapp_disconnected(); }
|
, function() { webapp_disconnected(); }
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
$(function() {
|
$(function() {
|
||||||
|
$.get("@{geturl}", function(url){
|
||||||
|
longpoll_#{ident}_url = url;
|
||||||
setTimeout(longpoll_#{ident}, #{startdelay});
|
setTimeout(longpoll_#{ident}, #{startdelay});
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
Loading…
Reference in a new issue