feat(users/Profpatsch/whatcd-resolver): Add torrent & basic status

This is a bit dirty, ideally we have a single polling loop that uses
`hx-swap-oob` to fill all status fields in the table (to avoid O(n)
looping requests).

Change-Id: I78ab392964cf00e39424002fe48cb35a60af184a
Reviewed-on: https://cl.tvl.fyi/c/depot/+/8875
Tested-by: BuildkiteCI
Reviewed-by: Profpatsch <mail@profpatsch.de>
This commit is contained in:
Profpatsch 2023-06-30 01:01:41 +02:00
parent 9504914a59
commit 12d23b3e64
3 changed files with 159 additions and 93 deletions

View file

@ -35,7 +35,7 @@ in
pa-field-parser = hsSelf.callPackage ./extra-pkgs/pa-field-parser-0.1.0.1.nix { }; pa-field-parser = hsSelf.callPackage ./extra-pkgs/pa-field-parser-0.1.0.1.nix { };
pa-label = hsSelf.callPackage ./extra-pkgs/pa-label-0.1.0.1.nix { }; pa-label = hsSelf.callPackage ./extra-pkgs/pa-label-0.1.0.1.nix { };
pa-pretty = hsSelf.callPackage ./extra-pkgs/pa-pretty-0.1.1.0.nix { }; pa-pretty = hsSelf.callPackage ./extra-pkgs/pa-pretty-0.1.1.0.nix { };
pa-json = hsSelf.callPackage ./extra-pkgs/pa-json-0.2.0.0.nix { }; pa-json = hsSelf.callPackage ./extra-pkgs/pa-json-0.2.1.0.nix { };
pa-run-command = hsSelf.callPackage ./extra-pkgs/pa-run-command-0.1.0.0.nix { }; pa-run-command = hsSelf.callPackage ./extra-pkgs/pa-run-command-0.1.0.0.nix { };
}; };
}; };

View file

@ -3,6 +3,7 @@
, aeson-better-errors , aeson-better-errors
, aeson-pretty , aeson-pretty
, base , base
, base64-bytestring
, bytestring , bytestring
, containers , containers
, hspec-core , hspec-core
@ -18,13 +19,14 @@
}: }:
mkDerivation { mkDerivation {
pname = "pa-json"; pname = "pa-json";
version = "0.2.0.0"; version = "0.2.1.0";
sha256 = "b57ef3888b8ea3230925675eccd6affbc3d296fc8762f5937435af4bdbd276e4"; sha256 = "d0c274fa38c05d38e9c2c15ee9dd4ff3ac369650dbc918c973863457110646c8";
libraryHaskellDepends = [ libraryHaskellDepends = [
aeson aeson
aeson-better-errors aeson-better-errors
aeson-pretty aeson-pretty
base base
base64-bytestring
bytestring bytestring
containers containers
hspec-core hspec-core

View file

@ -65,7 +65,10 @@ htmlUi = do
respond (Wai.responseLBS Http.status500 [] "") respond (Wai.responseLBS Http.status500 [] "")
catchAppException $ do catchAppException $ do
let renderHtml = if debug then toLazyBytes . textToBytesUtf8 . stringToText . Html.Pretty.renderHtml else Html.renderHtml let renderHtml =
if debug
then Html.Pretty.renderHtml >>> stringToText >>> textToBytesUtf8 >>> toLazyBytes
else Html.renderHtml
let h act = do let h act = do
res <- runInIO act res <- runInIO act
respond . Wai.responseLBS Http.ok200 [("Content-Type", "text/html")] . renderHtml $ res respond . Wai.responseLBS Http.ok200 [("Content-Type", "text/html")] . renderHtml $ res
@ -76,6 +79,12 @@ htmlUi = do
parser parser
req req
let torrentIdMp =
mp
( do
label @"torrentId" <$> Multipart.field "torrent-id" ((Field.utf8 >>> Field.signedDecimal >>> Field.bounded @Int "int"))
)
case req & Wai.pathInfo & Text.intercalate "/" of case req & Wai.pathInfo & Text.intercalate "/" of
"" -> h mainHtml "" -> h mainHtml
"snips/redacted/search" -> do "snips/redacted/search" -> do
@ -87,23 +96,40 @@ htmlUi = do
) )
snipsRedactedSearch dat snipsRedactedSearch dat
"snips/redacted/torrentDataJson" -> h $ do "snips/redacted/torrentDataJson" -> h $ do
dat <- dat <- torrentIdMp
mp
( do
label @"torrentId" <$> Multipart.field "torrent-id" ((Field.utf8 >>> Field.signedDecimal >>> Field.bounded @Int "int"))
)
mkVal <$> (runTransaction $ getTorrentById dat) mkVal <$> (runTransaction $ getTorrentById dat)
"snips/redacted/getTorrentFile" -> h $ do "snips/redacted/getTorrentFile" -> h $ do
dat <- dat <- torrentIdMp
mp
( do
label @"torrentId" <$> Multipart.field "torrent-id" ((Field.utf8 >>> Field.signedDecimal >>> Field.bounded @Int "int"))
)
runTransaction $ do runTransaction $ do
redactedGetTorrentFileAndInsert dat inserted <- redactedGetTorrentFileAndInsert dat
pure [hsx|Got!|] running <-
lift @Transaction $
doTransmissionRequest' (transmissionRequestAddTorrent inserted)
pure $
everySecond
"snips/transmission/getTorrentState"
(Enc.object [("torrent-hash", Enc.text running.torrentHash)])
"Starting"
"snips/transmission/getTorrentState" -> h $ do
dat <- mp $ label @"torrentHash" <$> Multipart.field "torrent-hash" Field.utf8
status <-
doTransmissionRequest'
( transmissionRequestListOnlyTorrents
( T2
(label @"ids" [label @"torrentHash" dat.torrentHash])
(label @"fields" ["hashString"])
)
(Json.keyLabel @"torrentHash" "hashString" Json.asText)
)
<&> List.find (\torrent -> torrent.torrentHash == dat.torrentHash)
pure $
case status of
Nothing -> [hsx|ERROR unknown|]
Just _torrent -> [hsx|Running|]
_ -> h mainHtml _ -> h mainHtml
where where
everySecond :: Text -> Enc -> Html -> Html
everySecond call extraData innerHtml = [hsx|<div hx-trigger="every 1s" hx-swap="outerHTML" hx-post={call} hx-vals={Enc.encToBytesUtf8 extraData}>{innerHtml}</div>|]
mainHtml = runTransaction $ do mainHtml = runTransaction $ do
bestTorrentsTable <- getBestTorrentsTable bestTorrentsTable <- getBestTorrentsTable
transmissionTorrentsTable <- lift @Transaction getTransmissionTorrentsTable transmissionTorrentsTable <- lift @Transaction getTransmissionTorrentsTable
@ -198,35 +224,25 @@ getBestTorrentsTable = do
|] |]
getTransmissionTorrentsTable :: getTransmissionTorrentsTable ::
(MonadIO m, MonadTransmission m, MonadThrow m) => (MonadIO m, MonadTransmission m, MonadThrow m, MonadLogger m) =>
m Html m Html
getTransmissionTorrentsTable = do getTransmissionTorrentsTable = do
let fields = ["hashString", "name", "activity", "percentDone", "percentComplete", "eta"] let fields = ["hashString", "name", "activity", "percentDone", "percentComplete", "eta"]
resp <- doTransmissionRequest transmissionConnectionConfig (transmissionRequestListAllTorrents fields)
case resp.result of
TransmissionResponseFailure err -> appThrowTree (nestedError "Transmission RPC error" $ singleError $ newError err)
TransmissionResponseSuccess ->
resp.arguments
& Map.lookup "torrents"
& annotate [fmt|Missing field "torrents"|]
& orAppThrowTree
<&> Json.parseValue (Json.eachInArray (Json.asObject <&> KeyMap.toMapText))
<&> first (Json.parseErrorTree "Cannot parse transmission torrents")
>>= \case
Left err -> appThrowTree err
Right a ->
pure $
toTable
( a
& List.sortOn (\m -> m & Map.lookup "percentDone" & fromMaybe (Json.Number 0))
<&> Map.toList
-- TODO
& List.take 100
)
zipNonEmpty :: NonEmpty a -> NonEmpty b -> NonEmpty (a, b) doTransmissionRequest'
zipNonEmpty (a :| as) (b :| bs) = (a, b) :| zip as bs ( transmissionRequestListAllTorrents fields $ do
Json.asObject <&> KeyMap.toMapText
)
<&> \resp ->
toTable
( resp
& List.sortOn (\m -> m & Map.lookup "percentDone" & fromMaybe (Json.Number 0))
<&> Map.toList
-- TODO
& List.take 100
)
-- | Render an arbitrary json value to HTML in a more-or-less reasonable fashion.
mkVal :: Json.Value -> Html mkVal :: Json.Value -> Html
mkVal = \case mkVal = \case
Json.Number n -> Html.toHtml @Text $ showToText n Json.Number n -> Html.toHtml @Text $ showToText n
@ -245,6 +261,7 @@ mkVal = \case
& foldMap (\(k, v) -> Html.dt (Html.toHtml @Text k) <> Html.dd (mkVal v)) & foldMap (\(k, v) -> Html.dt (Html.toHtml @Text k) <> Html.dd (mkVal v))
& Html.dl & Html.dl
-- | Render a table-like structure of json values as an HTML table.
toTable :: [[(Text, Json.Value)]] -> Html toTable :: [[(Text, Json.Value)]] -> Html
toTable xs = toTable xs =
case xs & nonEmpty of case xs & nonEmpty of
@ -273,53 +290,73 @@ data TransmissionRequest = TransmissionRequest
} }
deriving stock (Show) deriving stock (Show)
testTransmission :: TransmissionRequest -> IO (Either TmpPg.StartError ()) testTransmission :: Show out => (TransmissionRequest, Json.Parse Error out) -> IO (Either TmpPg.StartError ())
testTransmission req = runAppWith $ doTransmissionRequest transmissionConnectionConfig req >>= liftIO . printPretty testTransmission req = runAppWith $ doTransmissionRequest transmissionConnectionConfig req >>= liftIO . printPretty
transmissionConnectionConfig :: T2 "host" Text "port" Text transmissionConnectionConfig :: T2 "host" Text "port" Text
transmissionConnectionConfig = (T2 (label @"host" "localhost") (label @"port" "9091")) transmissionConnectionConfig = (T2 (label @"host" "localhost") (label @"port" "9091"))
transmissionRequestListAllTorrents :: [Text] -> TransmissionRequest transmissionRequestListAllTorrents :: Monad m => [Text] -> Json.ParseT e m out -> (TransmissionRequest, Json.ParseT e m [out])
transmissionRequestListAllTorrents fields = transmissionRequestListAllTorrents fields parseTorrent =
TransmissionRequest ( TransmissionRequest
{ method = "torrent-get", { method = "torrent-get",
arguments = arguments =
Map.fromList Map.fromList
[ ("fields", Enc.list Enc.text fields) [ ("fields", Enc.list Enc.text fields)
], ],
tag = Nothing tag = Nothing
} },
Json.key "torrents" $ Json.eachInArray parseTorrent
)
transmissionRequestListOnlyTorrents :: transmissionRequestListOnlyTorrents ::
( HasField "ids" r1 [r2], ( HasField "ids" r1 [(Label "torrentHash" Text)],
HasField "fields" r1 [Text], HasField "fields" r1 [Text],
HasField "torrentSha" r2 Text Monad m
) => ) =>
r1 -> r1 ->
TransmissionRequest Json.ParseT e m out ->
transmissionRequestListOnlyTorrents dat = (TransmissionRequest, Json.ParseT e m [out])
TransmissionRequest transmissionRequestListOnlyTorrents dat parseTorrent =
{ method = "torrent-get", ( TransmissionRequest
arguments = { method = "torrent-get",
Map.fromList arguments =
[ ("ids", Enc.list (\i -> Enc.text i.torrentSha) dat.ids), Map.fromList
("fields", Enc.list Enc.text dat.fields) [ ("ids", Enc.list (\i -> Enc.text i.torrentHash) dat.ids),
], ("fields", Enc.list Enc.text dat.fields)
tag = Nothing ],
} tag = Nothing
},
Json.key "torrents" $ Json.eachInArray parseTorrent
)
-- transmissionRequestAddTorrent dat = transmissionRequestAddTorrent ::
-- TransmissionRequest { (HasField "torrentFile" r ByteString, Monad m) =>
-- method = "torrent-add", r ->
-- arguments = ( TransmissionRequest,
-- Map.fromList [ Json.ParseT err m (T2 "torrentHash" Text "torrentName" Text)
-- ("metainfo", Enc.text $) )
-- ] transmissionRequestAddTorrent dat =
-- } ( TransmissionRequest
{ method = "torrent-add",
arguments =
Map.fromList
[ ("metainfo", Enc.base64Bytes dat.torrentFile),
("paused", Enc.bool True)
],
tag = Nothing
},
do
let p method = Json.key method $ do
hash <- Json.keyLabel @"torrentHash" "hashString" Json.asText
name <- Json.keyLabel @"torrentName" "name" Json.asText
pure $ T2 hash name
p "torrent-duplicate" Json.<|> p "torrent-added"
)
data TransmissionResponse = TransmissionResponse data TransmissionResponse output = TransmissionResponse
{ result :: TransmissionResponseStatus, { result :: TransmissionResponseStatus,
arguments :: Map Text Json.Value, arguments :: Maybe output,
tag :: Maybe Int tag :: Maybe Int
} }
deriving stock (Show) deriving stock (Show)
@ -329,30 +366,53 @@ data TransmissionResponseStatus
| TransmissionResponseFailure Text | TransmissionResponseFailure Text
deriving stock (Show) deriving stock (Show)
doTransmissionRequest' ::
( MonadIO m,
MonadTransmission m,
MonadThrow m,
MonadLogger m
) =>
(TransmissionRequest, Json.Parse Error output) ->
m output
doTransmissionRequest' req = do
resp <-
doTransmissionRequest
transmissionConnectionConfig
req
case resp.result of
TransmissionResponseFailure err -> appThrowTree (nestedError "Transmission RPC error" $ singleError $ newError err)
TransmissionResponseSuccess -> case resp.arguments of
Nothing -> appThrowTree "Transmission RPC error: No `arguments` field in response"
Just out -> pure out
-- | Contact the transmission RPC, and do the CSRF protection dance.
--
-- Spec: https://github.com/transmission/transmission/blob/main/docs/rpc-spec.md
doTransmissionRequest :: doTransmissionRequest ::
( MonadIO m, ( MonadIO m,
MonadTransmission m, MonadTransmission m,
HasField "host" t1 Text, HasField "host" t1 Text,
HasField "port" t1 Text, HasField "port" t1 Text,
MonadThrow m MonadThrow m,
MonadLogger m
) => ) =>
t1 -> t1 ->
TransmissionRequest -> (TransmissionRequest, Json.Parse Error output) ->
m TransmissionResponse m (TransmissionResponse output)
doTransmissionRequest dat req = do doTransmissionRequest dat (req, parser) = do
sessionId <- getTransmissionId sessionId <- getTransmissionId
let body =
Enc.object
( [ ("method", req.method & Enc.text),
("arguments", Enc.map id req.arguments)
]
<> (req.tag & maybe [] (\t -> [("tag", t & Enc.int)]))
)
logDebug [fmt|transmission request: {showPrettyJsonEncoding body.unEnc}|]
let httpReq = let httpReq =
[fmt|http://{dat.host}:{dat.port}/transmission/rpc|] [fmt|http://{dat.host}:{dat.port}/transmission/rpc|]
& Http.setRequestMethod "POST" & Http.setRequestMethod "POST"
& Http.setRequestBodyLBS & Http.setRequestBodyLBS (Enc.encToBytesUtf8Lazy body)
( Enc.encToBytesUtf8Lazy $
Enc.object
( [ ("method", req.method & Enc.text),
("arguments", Enc.map id req.arguments)
]
<> (req.tag & maybe [] (\t -> [("tag", t & Enc.int)]))
)
)
& (sessionId & maybe id (Http.setRequestHeader "X-Transmission-Session-Id" . (: []))) & (sessionId & maybe id (Http.setRequestHeader "X-Transmission-Session-Id" . (: [])))
resp <- Http.httpBS httpReq resp <- Http.httpBS httpReq
-- Implement the CSRF protection thingy -- Implement the CSRF protection thingy
@ -367,7 +427,7 @@ doTransmissionRequest dat req = do
& liftIO & liftIO
<&> NonEmpty.head <&> NonEmpty.head
setTransmissionId tid setTransmissionId tid
doTransmissionRequest dat req doTransmissionRequest dat (req, parser)
200 -> 200 ->
resp resp
& Http.getResponseBody & Http.getResponseBody
@ -378,9 +438,7 @@ doTransmissionRequest dat req = do
"success" -> TransmissionResponseSuccess "success" -> TransmissionResponseSuccess
err -> TransmissionResponseFailure err err -> TransmissionResponseFailure err
arguments <- arguments <-
Json.keyMay "arguments" Json.asObject Json.keyMay "arguments" parser
<&> fromMaybe mempty
<&> KeyMap.toMapText
tag <- tag <-
Json.keyMay Json.keyMay
"tag" "tag"
@ -390,7 +448,11 @@ doTransmissionRequest dat req = do
& first (Json.parseErrorTree "Cannot parse transmission RPC response") & first (Json.parseErrorTree "Cannot parse transmission RPC response")
& \case & \case
Right a -> pure a Right a -> pure a
Left err -> appThrowTree err Left err -> do
case Json.eitherDecodeStrict' @Json.Value (resp & Http.getResponseBody) of
Left _err -> pure ()
Right val -> logInfo [fmt|failing transmission response: {showPrettyJson val}|]
appThrowTree err
_ -> liftIO $ unwrapIOError $ Left [fmt|Non-200 response: {showPretty resp}|] _ -> liftIO $ unwrapIOError $ Left [fmt|Non-200 response: {showPretty resp}|]
redactedSearch :: redactedSearch ::
@ -559,7 +621,7 @@ redactedGetTorrentFileAndInsert ::
MonadLogger m MonadLogger m
) => ) =>
r -> r ->
Transaction m () Transaction m (Label "torrentFile" ByteString)
redactedGetTorrentFileAndInsert dat = do redactedGetTorrentFileAndInsert dat = do
bytes <- redactedGetTorrentFile dat bytes <- redactedGetTorrentFile dat
execute execute
@ -572,6 +634,7 @@ redactedGetTorrentFileAndInsert dat = do
dat.torrentId dat.torrentId
) )
>>= assertOneUpdated "redactedGetTorrentFileAndInsert" >>= assertOneUpdated "redactedGetTorrentFileAndInsert"
>>= \() -> pure (label @"torrentFile" bytes)
assertOneUpdated :: assertOneUpdated ::
(HasField "numberOfRowsAffected" r Natural, MonadThrow m) => (HasField "numberOfRowsAffected" r Natural, MonadThrow m) =>
@ -720,6 +783,7 @@ unzipT3 xs = xs <&> toTup & unzip3 & fromTup
fromTup :: (a, b, c) -> T3 l1 a l2 b l3 c fromTup :: (a, b, c) -> T3 l1 a l2 b l3 c
fromTup (t1, t2, t3) = T3 (label @l1 t1) (label @l2 t2) (label @l3 t3) fromTup (t1, t2, t3) = T3 (label @l1 t1) (label @l2 t2) (label @l3 t3)
-- | Do a request to the redacted API. If you know what that is, you know how to find the API docs.
redactedApiRequest :: redactedApiRequest ::
( MonadThrow m, ( MonadThrow m,
MonadIO m, MonadIO m,