2024-07-07 16:59:12 +00:00
|
|
|
{- P2P protocol over HTTP
|
|
|
|
-
|
|
|
|
- https://git-annex.branchable.com/design/p2p_protocol_over_http/
|
|
|
|
-
|
|
|
|
- Copyright 2024 Joey Hess <id@joeyh.name>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU AGPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
{-# LANGUAGE DataKinds #-}
|
|
|
|
{-# LANGUAGE TypeOperators #-}
|
2024-07-08 01:20:50 +00:00
|
|
|
{-# LANGUAGE TypeFamilies #-}
|
|
|
|
{-# LANGUAGE FlexibleInstances #-}
|
|
|
|
{-# LANGUAGE DeriveAnyClass #-}
|
|
|
|
{-# LANGUAGE DeriveGeneric #-}
|
2024-07-07 18:48:20 +00:00
|
|
|
{-# LANGUAGE MultiParamTypeClasses #-}
|
2024-07-07 16:59:12 +00:00
|
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
|
|
|
|
|
|
module P2P.Http where
|
|
|
|
|
|
|
|
import Annex.Common
|
|
|
|
import qualified P2P.Protocol as P2P
|
|
|
|
import Utility.Base64
|
|
|
|
import Utility.MonotonicClock
|
|
|
|
|
|
|
|
import Servant
|
2024-07-07 20:08:05 +00:00
|
|
|
import Servant.Client.Streaming
|
2024-07-08 01:20:50 +00:00
|
|
|
import Servant.Client.Core.RunClient
|
2024-07-08 01:51:30 +00:00
|
|
|
import qualified Servant.Client.Core.Request
|
2024-07-07 18:48:20 +00:00
|
|
|
import qualified Servant.Types.SourceT as S
|
2024-07-07 16:59:12 +00:00
|
|
|
import Servant.API.WebSocket
|
2024-07-07 18:48:20 +00:00
|
|
|
import qualified Network.WebSockets as Websocket
|
|
|
|
import Network.Wai
|
|
|
|
import Network.Wai.Handler.Warp
|
2024-07-08 01:20:50 +00:00
|
|
|
import Network.HTTP.Client (newManager, defaultManagerSettings)
|
2024-07-07 18:48:20 +00:00
|
|
|
import qualified Data.Text as T
|
2024-07-07 16:59:12 +00:00
|
|
|
import qualified Data.Text.Encoding as TE
|
|
|
|
import qualified Data.ByteString as B
|
2024-07-08 01:20:50 +00:00
|
|
|
import Text.Read (readMaybe)
|
2024-07-07 18:48:20 +00:00
|
|
|
import Data.Aeson hiding (Key)
|
2024-07-08 01:20:50 +00:00
|
|
|
import Data.Maybe
|
|
|
|
import Control.DeepSeq
|
|
|
|
import GHC.Generics
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
type P2PHttpAPI
|
2024-07-07 20:08:05 +00:00
|
|
|
= "git-annex" :> "v3" :> "key" :> CaptureKey
|
2024-07-07 16:59:12 +00:00
|
|
|
:> GetAPI '[DataLengthHeader]
|
|
|
|
:<|> "git-annex" :> "v2" :> "key" :> CaptureKey
|
|
|
|
:> GetAPI '[DataLengthHeader]
|
|
|
|
:<|> "git-annex" :> "v1" :> "key" :> CaptureKey
|
|
|
|
:> GetAPI '[DataLengthHeader]
|
|
|
|
:<|> "git-annex" :> "v0" :> "key" :> CaptureKey
|
|
|
|
:> GetAPI '[]
|
|
|
|
:<|> "git-annex" :> "v3" :> "checkpresent" :> CheckPresentAPI
|
|
|
|
:<|> "git-annex" :> "v2" :> "checkpresent" :> CheckPresentAPI
|
|
|
|
:<|> "git-annex" :> "v1" :> "checkpresent" :> CheckPresentAPI
|
|
|
|
:<|> "git-annex" :> "v0" :> "checkpresent" :> CheckPresentAPI
|
|
|
|
:<|> "git-annex" :> "v3" :> "remove" :> RemoveAPI RemoveResultPlus
|
|
|
|
:<|> "git-annex" :> "v2" :> "remove" :> RemoveAPI RemoveResultPlus
|
|
|
|
:<|> "git-annex" :> "v1" :> "remove" :> RemoveAPI RemoveResult
|
|
|
|
:<|> "git-annex" :> "v0" :> "remove" :> RemoveAPI RemoveResult
|
|
|
|
:<|> "git-annex" :> "v3" :> "remove-before" :> RemoveBeforeAPI
|
|
|
|
:<|> "git-annex" :> "v3" :> "gettimestamp" :> GetTimestampAPI
|
|
|
|
:<|> "git-annex" :> "v3" :> "put" :> DataLengthHeader
|
|
|
|
:> PutAPI PutResultPlus
|
|
|
|
:<|> "git-annex" :> "v2" :> "put" :> DataLengthHeader
|
|
|
|
:> PutAPI PutResultPlus
|
|
|
|
:<|> "git-annex" :> "v1" :> "put" :> DataLengthHeader
|
|
|
|
:> PutAPI PutResult
|
|
|
|
:<|> "git-annex" :> "v0" :> "put"
|
|
|
|
:> PutAPI PutResult
|
|
|
|
:<|> "git-annex" :> "v3" :> "putoffset"
|
|
|
|
:> PutOffsetAPI PutOffsetResultPlus
|
|
|
|
:<|> "git-annex" :> "v2" :> "putoffset"
|
|
|
|
:> PutOffsetAPI PutOffsetResultPlus
|
|
|
|
:<|> "git-annex" :> "v1" :> "putoffset"
|
|
|
|
:> PutOffsetAPI PutOffsetResult
|
2024-07-08 01:20:50 +00:00
|
|
|
:<|> "git-annex" :> "v3" :> "lockcontent" :> LockContentAPI
|
|
|
|
:<|> "git-annex" :> "v2" :> "lockcontent" :> LockContentAPI
|
|
|
|
:<|> "git-annex" :> "v1" :> "lockcontent" :> LockContentAPI
|
|
|
|
:<|> "git-annex" :> "v0" :> "lockcontent" :> LockContentAPI
|
2024-07-07 20:08:05 +00:00
|
|
|
:<|> "git-annex" :> "key" :> CaptureKey :> GetAPI '[]
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
p2pHttpAPI :: Proxy P2PHttpAPI
|
|
|
|
p2pHttpAPI = Proxy
|
|
|
|
|
|
|
|
p2pHttp :: Application
|
|
|
|
p2pHttp = serve p2pHttpAPI serveP2pHttp
|
|
|
|
|
|
|
|
serveP2pHttp :: Server P2PHttpAPI
|
|
|
|
serveP2pHttp
|
2024-07-07 20:08:05 +00:00
|
|
|
= serveGet
|
2024-07-07 18:48:20 +00:00
|
|
|
:<|> serveGet
|
|
|
|
:<|> serveGet
|
|
|
|
:<|> serveGet0
|
|
|
|
:<|> serveCheckPresent
|
|
|
|
:<|> serveCheckPresent
|
|
|
|
:<|> serveCheckPresent
|
|
|
|
:<|> serveCheckPresent
|
|
|
|
:<|> serveRemove id
|
|
|
|
:<|> serveRemove id
|
|
|
|
:<|> serveRemove dePlus
|
|
|
|
:<|> serveRemove dePlus
|
|
|
|
:<|> serveRemoveBefore
|
|
|
|
:<|> serveGetTimestamp
|
|
|
|
:<|> servePut id
|
|
|
|
:<|> servePut id
|
|
|
|
:<|> servePut dePlus
|
2024-07-07 20:08:05 +00:00
|
|
|
:<|> servePut dePlus Nothing
|
2024-07-07 18:48:20 +00:00
|
|
|
:<|> servePutOffset id
|
|
|
|
:<|> servePutOffset id
|
|
|
|
:<|> servePutOffset dePlus
|
2024-07-08 01:20:50 +00:00
|
|
|
:<|> serveLockContent
|
|
|
|
:<|> serveLockContent
|
|
|
|
:<|> serveLockContent
|
|
|
|
:<|> serveLockContent
|
2024-07-07 20:08:05 +00:00
|
|
|
:<|> serveGet0
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type GetAPI headers
|
2024-07-07 18:48:20 +00:00
|
|
|
= ClientUUID Optional
|
|
|
|
:> ServerUUID Optional
|
|
|
|
:> BypassUUIDs
|
2024-07-07 16:59:12 +00:00
|
|
|
:> AssociatedFileParam
|
|
|
|
:> OffsetParam
|
|
|
|
:> StreamGet NoFraming OctetStream
|
|
|
|
(Headers headers (SourceIO B.ByteString))
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
serveGet
|
|
|
|
:: B64Key
|
|
|
|
-> Maybe (B64UUID ClientSide)
|
|
|
|
-> Maybe (B64UUID ServerSide)
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Maybe B64FilePath
|
|
|
|
-> Maybe Offset
|
|
|
|
-> Handler (Headers '[DataLengthHeader] (S.SourceT IO B.ByteString))
|
|
|
|
serveGet = undefined
|
|
|
|
|
|
|
|
serveGet0
|
|
|
|
:: B64Key
|
|
|
|
-> Maybe (B64UUID ClientSide)
|
|
|
|
-> Maybe (B64UUID ServerSide)
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Maybe B64FilePath
|
|
|
|
-> Maybe Offset
|
|
|
|
-> Handler (Headers '[] (S.SourceT IO B.ByteString))
|
|
|
|
serveGet0 = undefined
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
clientGet
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> B64Key
|
|
|
|
-> Maybe (B64UUID ClientSide)
|
|
|
|
-> Maybe (B64UUID ServerSide)
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Maybe B64FilePath
|
|
|
|
-> Maybe Offset
|
|
|
|
-> ClientM (Headers '[DataLengthHeader] (S.SourceT IO B.ByteString))
|
|
|
|
clientGet (P2P.ProtocolVersion ver) k cu su bypass af o = case ver of
|
|
|
|
3 -> v3 k cu su bypass af o
|
|
|
|
2 -> v2 k cu su bypass af o
|
|
|
|
1 -> error "XXX" -- TODO v1
|
|
|
|
0 -> error "XXX" -- TODO v0
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> v3 :<|> v2 :<|> v1 :<|> v0 :<|> _ = client p2pHttpAPI
|
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type CheckPresentAPI
|
|
|
|
= KeyParam
|
2024-07-07 18:48:20 +00:00
|
|
|
:> ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
2024-07-07 16:59:12 +00:00
|
|
|
:> Post '[JSON] CheckPresentResult
|
2024-07-07 18:48:20 +00:00
|
|
|
|
|
|
|
serveCheckPresent
|
|
|
|
:: B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Handler CheckPresentResult
|
|
|
|
serveCheckPresent = undefined
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
clientCheckPresent
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> ClientM CheckPresentResult
|
|
|
|
clientCheckPresent (P2P.ProtocolVersion ver) = case ver of
|
|
|
|
3 -> v3
|
|
|
|
2 -> v2
|
|
|
|
1 -> v1
|
|
|
|
0 -> v0
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
v3 :<|> v2 :<|> v1 :<|> v0 :<|> _ = client p2pHttpAPI
|
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type RemoveAPI result
|
|
|
|
= KeyParam
|
2024-07-07 18:48:20 +00:00
|
|
|
:> ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
2024-07-07 16:59:12 +00:00
|
|
|
:> Post '[JSON] result
|
2024-07-07 18:48:20 +00:00
|
|
|
|
|
|
|
serveRemove
|
|
|
|
:: (RemoveResultPlus -> t)
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Handler t
|
|
|
|
serveRemove = undefined
|
2024-07-07 20:08:05 +00:00
|
|
|
|
|
|
|
clientRemove
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> ClientM RemoveResultPlus
|
|
|
|
clientRemove (P2P.ProtocolVersion ver) k cu su bypass = case ver of
|
|
|
|
3 -> v3 k cu su bypass
|
|
|
|
2 -> v2 k cu su bypass
|
|
|
|
1 -> plus <$> v1 k cu su bypass
|
|
|
|
0 -> plus <$> v0 k cu su bypass
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
v3 :<|> v2 :<|> v1 :<|> v0 :<|> _ = client p2pHttpAPI
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type RemoveBeforeAPI
|
|
|
|
= KeyParam
|
2024-07-07 18:48:20 +00:00
|
|
|
:> ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
|
|
|
:> QueryParam' '[Required] "timestamp" Timestamp
|
2024-07-07 16:59:12 +00:00
|
|
|
:> Post '[JSON] RemoveResult
|
2024-07-07 18:48:20 +00:00
|
|
|
|
|
|
|
serveRemoveBefore
|
|
|
|
:: B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Timestamp
|
|
|
|
-> Handler RemoveResult
|
|
|
|
serveRemoveBefore = undefined
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
clientRemoveBefore
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Timestamp
|
|
|
|
-> ClientM RemoveResult
|
|
|
|
clientRemoveBefore (P2P.ProtocolVersion ver) = case ver of
|
|
|
|
3 -> v3
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
v3 :<|> _ = client p2pHttpAPI
|
|
|
|
|
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type GetTimestampAPI
|
2024-07-07 18:48:20 +00:00
|
|
|
= ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
2024-07-07 16:59:12 +00:00
|
|
|
:> Post '[JSON] GetTimestampResult
|
2024-07-07 18:48:20 +00:00
|
|
|
|
|
|
|
serveGetTimestamp
|
|
|
|
:: B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Handler GetTimestampResult
|
|
|
|
serveGetTimestamp = undefined
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
clientGetTimestamp
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> ClientM GetTimestampResult
|
|
|
|
clientGetTimestamp (P2P.ProtocolVersion ver) = case ver of
|
|
|
|
3 -> v3
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|>
|
|
|
|
v3 :<|> _ = client p2pHttpAPI
|
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type PutAPI result
|
|
|
|
= KeyParam
|
2024-07-07 18:48:20 +00:00
|
|
|
:> ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
2024-07-07 16:59:12 +00:00
|
|
|
:> AssociatedFileParam
|
|
|
|
:> OffsetParam
|
|
|
|
:> Header' '[Required] "X-git-annex-data-length" DataLength
|
|
|
|
:> StreamBody NoFraming OctetStream (SourceIO B.ByteString)
|
|
|
|
:> Post '[JSON] result
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
servePut
|
|
|
|
:: (PutResultPlus -> t)
|
|
|
|
-> Maybe Integer
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Maybe B64FilePath
|
|
|
|
-> Maybe Offset
|
|
|
|
-> DataLength
|
|
|
|
-> S.SourceT IO B.ByteString
|
|
|
|
-> Handler t
|
|
|
|
servePut = undefined
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
clientPut
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> Maybe Integer
|
|
|
|
-> B64Key
|
2024-07-07 18:48:20 +00:00
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Maybe B64FilePath
|
|
|
|
-> Maybe Offset
|
|
|
|
-> DataLength
|
|
|
|
-> S.SourceT IO B.ByteString
|
2024-07-07 20:08:05 +00:00
|
|
|
-> ClientM PutResultPlus
|
|
|
|
clientPut (P2P.ProtocolVersion ver) sz k cu su bypass af o l src = case ver of
|
|
|
|
3 -> v3 sz k cu su bypass af o l src
|
|
|
|
2 -> v2 sz k cu su bypass af o l src
|
|
|
|
1 -> plus <$> v1 sz k cu su bypass af o l src
|
|
|
|
0 -> plus <$> v0 k cu su bypass af o l src
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|>
|
|
|
|
_ :<|>
|
|
|
|
v3 :<|> v2 :<|> v1 :<|> v0 :<|> _ = client p2pHttpAPI
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
type PutOffsetAPI result
|
|
|
|
= KeyParam
|
2024-07-07 18:48:20 +00:00
|
|
|
:> ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
2024-07-07 16:59:12 +00:00
|
|
|
:> Post '[JSON] result
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
servePutOffset
|
|
|
|
:: (PutOffsetResultPlus -> t)
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Handler t
|
|
|
|
servePutOffset = undefined
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
clientPutOffset
|
|
|
|
:: P2P.ProtocolVersion
|
|
|
|
-> B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> ClientM PutOffsetResultPlus
|
|
|
|
clientPutOffset (P2P.ProtocolVersion ver) = case ver of
|
|
|
|
3 -> v3
|
|
|
|
2 -> v2
|
|
|
|
_ -> error "unsupported protocol version"
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|>
|
|
|
|
_ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
v3 :<|> v2 :<|> _ = client p2pHttpAPI
|
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
type LockContentAPI
|
|
|
|
= KeyParam
|
|
|
|
:> ClientUUID Required
|
|
|
|
:> ServerUUID Required
|
|
|
|
:> BypassUUIDs
|
|
|
|
:> WebSocket
|
2024-07-07 20:08:05 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
serveLockContent
|
|
|
|
:: B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> Websocket.Connection
|
|
|
|
-> Handler ()
|
|
|
|
serveLockContent = undefined
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-08 01:51:30 +00:00
|
|
|
data WebSocketClient = WebSocketClient Servant.Client.Core.Request.Request
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
-- XXX this is enough to let servant-client work, but it's not yet
|
|
|
|
-- possible to run a WebSocketClient.
|
|
|
|
instance RunClient m => HasClient m WebSocket where
|
|
|
|
type Client m WebSocket = WebSocketClient
|
2024-07-08 01:51:30 +00:00
|
|
|
clientWithRoute _pm Proxy req = WebSocketClient req
|
|
|
|
hoistClientMonad _ _ _ w = w
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
clientLockContent
|
|
|
|
:: B64Key
|
|
|
|
-> B64UUID ClientSide
|
|
|
|
-> B64UUID ServerSide
|
|
|
|
-> [B64UUID Bypass]
|
|
|
|
-> WebSocketClient
|
|
|
|
clientLockContent = v3
|
|
|
|
where
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|>
|
|
|
|
_ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|> _ :<|>
|
|
|
|
_ :<|> _ :<|> _ :<|>
|
|
|
|
v3 :<|> _ = client p2pHttpAPI
|
|
|
|
-- XXX add other protocol versions
|
|
|
|
|
|
|
|
--XXX test code
|
|
|
|
query :: ClientM PutOffsetResultPlus
|
|
|
|
query = do
|
|
|
|
clientPutOffset (P2P.ProtocolVersion 3)
|
|
|
|
(B64Key (fromJust $ deserializeKey "WORM--foo"))
|
|
|
|
(B64UUID (toUUID ("client" :: String)))
|
|
|
|
(B64UUID (toUUID ("server" :: String)))
|
|
|
|
[]
|
|
|
|
|
|
|
|
--XXX test code
|
|
|
|
query' :: WebSocketClient
|
|
|
|
query' = do
|
|
|
|
clientLockContent (B64Key (fromJust $ deserializeKey "WORM--foo"))
|
|
|
|
(B64UUID (toUUID ("client" :: String)))
|
|
|
|
(B64UUID (toUUID ("server" :: String)))
|
|
|
|
[]
|
|
|
|
|
|
|
|
--XXX test code
|
|
|
|
run :: IO ()
|
|
|
|
run = do
|
|
|
|
manager' <- newManager defaultManagerSettings
|
2024-07-08 01:51:30 +00:00
|
|
|
let WebSocketClient wscreq = query'
|
|
|
|
res <- runClientM (runRequestAcceptStatus Nothing wscreq)
|
|
|
|
(mkClientEnv manager' (BaseUrl Http "localhost" 8081 ""))
|
|
|
|
-- res <- runClientM query (mkClientEnv manager' (BaseUrl Http "localhost" 8081 ""))
|
2024-07-08 01:20:50 +00:00
|
|
|
case res of
|
|
|
|
Left err -> putStrLn $ "Error: " ++ show err
|
|
|
|
Right res' -> do
|
|
|
|
print res'
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
type ClientUUID req = QueryParam' '[req] "clientuuid" (B64UUID ClientSide)
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
type ServerUUID req = QueryParam' '[req] "serveruuid" (B64UUID ServerSide)
|
|
|
|
|
|
|
|
type BypassUUIDs = QueryParams "bypass" (B64UUID Bypass)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
type CaptureKey = Capture "key" B64Key
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
type KeyParam = QueryParam' '[Required] "key" B64Key
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
type AssociatedFileParam = QueryParam "associatedfile" B64FilePath
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
type OffsetParam = QueryParam "offset" Offset
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
type DataLengthHeader = Header "X-git-annex-data-length" Integer
|
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
-- Phantom types for B64UIID
|
|
|
|
data ClientSide
|
|
|
|
data ServerSide
|
|
|
|
data Bypass
|
|
|
|
data Plus
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
-- Keys, UUIDs, and filenames are base64 encoded since Servant uses
|
|
|
|
-- Text and so needs UTF-8.
|
|
|
|
newtype B64Key = B64Key Key
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
newtype B64UUID t = B64UUID UUID
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show, Generic, NFData)
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
newtype B64FilePath = B64FilePath RawFilePath
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
newtype DataLength = DataLength Integer
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
newtype CheckPresentResult = CheckPresentResult Bool
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
newtype RemoveResult = RemoveResult Bool
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
data RemoveResultPlus = RemoveResultPlus Bool [B64UUID Plus]
|
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
newtype GetTimestampResult = GetTimestampResult Timestamp
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
newtype PutResult = PutResult Bool
|
2024-07-07 18:48:20 +00:00
|
|
|
deriving (Eq, Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
data PutResultPlus = PutResultPlus Bool [B64UUID Plus]
|
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
newtype PutOffsetResult = PutOffsetResult Offset
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-08 01:20:50 +00:00
|
|
|
data PutOffsetResultPlus = PutOffsetResultPlus Offset [B64UUID Plus]
|
|
|
|
deriving (Show, Generic, NFData)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
newtype Offset = Offset P2P.Offset
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show, Generic, NFData)
|
2024-07-07 18:48:20 +00:00
|
|
|
|
|
|
|
newtype Timestamp = Timestamp MonotonicTimestamp
|
2024-07-08 01:20:50 +00:00
|
|
|
deriving (Show)
|
2024-07-07 20:08:05 +00:00
|
|
|
|
|
|
|
instance ToHttpApiData B64Key where
|
|
|
|
toUrlPiece (B64Key k) = TE.decodeUtf8Lenient $
|
|
|
|
toB64 (serializeKey' k)
|
2024-07-07 16:59:12 +00:00
|
|
|
|
|
|
|
instance FromHttpApiData B64Key where
|
|
|
|
parseUrlPiece t = case fromB64Maybe (TE.encodeUtf8 t) of
|
|
|
|
Nothing -> Left "unable to base64 decode key"
|
|
|
|
Just b -> maybe (Left "key parse error") (Right . B64Key)
|
|
|
|
(deserializeKey' b)
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
instance ToHttpApiData (B64UUID t) where
|
|
|
|
toUrlPiece (B64UUID u) = TE.decodeUtf8Lenient $
|
|
|
|
toB64 (fromUUID u)
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
instance FromHttpApiData (B64UUID t) where
|
2024-07-07 16:59:12 +00:00
|
|
|
parseUrlPiece t = case fromB64Maybe (TE.encodeUtf8 t) of
|
|
|
|
Nothing -> Left "unable to base64 decode UUID"
|
|
|
|
Just b -> case toUUID b of
|
|
|
|
u@(UUID _) -> Right (B64UUID u)
|
|
|
|
NoUUID -> Left "empty UUID"
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
instance ToHttpApiData B64FilePath where
|
|
|
|
toUrlPiece (B64FilePath f) = TE.decodeUtf8Lenient $ toB64 f
|
|
|
|
|
2024-07-07 16:59:12 +00:00
|
|
|
instance FromHttpApiData B64FilePath where
|
|
|
|
parseUrlPiece t = case fromB64Maybe (TE.encodeUtf8 t) of
|
|
|
|
Nothing -> Left "unable to base64 decode filename"
|
|
|
|
Just b -> Right (B64FilePath b)
|
2024-07-07 18:48:20 +00:00
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
instance ToHttpApiData Offset where
|
|
|
|
toUrlPiece (Offset (P2P.Offset n)) = T.pack (show n)
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
instance FromHttpApiData Offset where
|
|
|
|
parseUrlPiece t = case readMaybe (T.unpack t) of
|
|
|
|
Nothing -> Left "offset parse error"
|
|
|
|
Just n -> Right (Offset (P2P.Offset n))
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
instance ToHttpApiData Timestamp where
|
|
|
|
toUrlPiece (Timestamp (MonotonicTimestamp n)) = T.pack (show n)
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
instance FromHttpApiData Timestamp where
|
|
|
|
parseUrlPiece t = case readMaybe (T.unpack t) of
|
|
|
|
Nothing -> Left "timestamp parse error"
|
|
|
|
Just n -> Right (Timestamp (MonotonicTimestamp n))
|
|
|
|
|
2024-07-07 20:08:05 +00:00
|
|
|
instance ToHttpApiData DataLength where
|
|
|
|
toUrlPiece (DataLength n) = T.pack (show n)
|
|
|
|
|
2024-07-07 18:48:20 +00:00
|
|
|
instance FromHttpApiData DataLength where
|
|
|
|
parseUrlPiece t = case readMaybe (T.unpack t) of
|
|
|
|
Nothing -> Left "X-git-annex-data-length parse error"
|
|
|
|
Just n -> Right (DataLength n)
|
|
|
|
|
|
|
|
instance ToJSON PutResult where
|
|
|
|
toJSON (PutResult b) =
|
|
|
|
object ["stored" .= b]
|
|
|
|
|
|
|
|
instance FromJSON PutResult where
|
|
|
|
parseJSON = withObject "PutResult" $ \v -> PutResult
|
|
|
|
<$> v .: "stored"
|
|
|
|
|
|
|
|
instance ToJSON PutResultPlus where
|
|
|
|
toJSON (PutResultPlus b us) = object
|
|
|
|
[ "stored" .= b
|
2024-07-08 01:20:50 +00:00
|
|
|
, "plusuuids" .= plusList us
|
2024-07-07 18:48:20 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
instance FromJSON PutResultPlus where
|
|
|
|
parseJSON = withObject "PutResultPlus" $ \v -> PutResultPlus
|
|
|
|
<$> v .: "stored"
|
|
|
|
<*> v .: "plusuuids"
|
|
|
|
|
|
|
|
instance ToJSON CheckPresentResult where
|
|
|
|
toJSON (CheckPresentResult b) = object
|
|
|
|
["present" .= b]
|
|
|
|
|
|
|
|
instance FromJSON CheckPresentResult where
|
|
|
|
parseJSON = withObject "CheckPresentResult" $ \v -> CheckPresentResult
|
|
|
|
<$> v .: "present"
|
|
|
|
|
|
|
|
instance ToJSON RemoveResult where
|
|
|
|
toJSON (RemoveResult b) = object
|
|
|
|
["removed" .= b]
|
|
|
|
|
|
|
|
instance FromJSON RemoveResult where
|
|
|
|
parseJSON = withObject "RemoveResult" $ \v -> RemoveResult
|
|
|
|
<$> v .: "removed"
|
|
|
|
|
|
|
|
instance ToJSON RemoveResultPlus where
|
|
|
|
toJSON (RemoveResultPlus b us) = object
|
|
|
|
[ "removed" .= b
|
2024-07-08 01:20:50 +00:00
|
|
|
, "plusuuids" .= plusList us
|
2024-07-07 18:48:20 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
instance FromJSON RemoveResultPlus where
|
|
|
|
parseJSON = withObject "RemoveResultPlus" $ \v -> RemoveResultPlus
|
|
|
|
<$> v .: "removed"
|
|
|
|
<*> v .: "plusuuids"
|
|
|
|
|
|
|
|
instance ToJSON GetTimestampResult where
|
|
|
|
toJSON (GetTimestampResult (Timestamp (MonotonicTimestamp t))) = object
|
|
|
|
["timestamp" .= t]
|
|
|
|
|
|
|
|
instance FromJSON GetTimestampResult where
|
|
|
|
parseJSON = withObject "GetTimestampResult" $ \v ->
|
|
|
|
GetTimestampResult . Timestamp . MonotonicTimestamp
|
|
|
|
<$> v .: "timestamp"
|
|
|
|
|
|
|
|
instance ToJSON PutOffsetResult where
|
|
|
|
toJSON (PutOffsetResult (Offset (P2P.Offset o))) = object
|
|
|
|
["offset" .= o]
|
|
|
|
|
|
|
|
instance FromJSON PutOffsetResult where
|
|
|
|
parseJSON = withObject "PutOffsetResult" $ \v ->
|
|
|
|
PutOffsetResult
|
|
|
|
<$> (Offset . P2P.Offset <$> v .: "offset")
|
|
|
|
|
|
|
|
instance ToJSON PutOffsetResultPlus where
|
|
|
|
toJSON (PutOffsetResultPlus (Offset (P2P.Offset o)) us) = object
|
|
|
|
[ "offset" .= o
|
2024-07-08 01:20:50 +00:00
|
|
|
, "plusuuids" .= plusList us
|
2024-07-07 18:48:20 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
instance FromJSON PutOffsetResultPlus where
|
|
|
|
parseJSON = withObject "PutOffsetResultPlus" $ \v ->
|
|
|
|
PutOffsetResultPlus
|
|
|
|
<$> (Offset . P2P.Offset <$> v .: "offset")
|
|
|
|
<*> v .: "plusuuids"
|
2024-07-08 01:20:50 +00:00
|
|
|
|
|
|
|
instance FromJSON (B64UUID t) where
|
|
|
|
parseJSON (String t) = case fromB64Maybe (TE.encodeUtf8 t) of
|
|
|
|
Just s -> pure (B64UUID (toUUID s))
|
|
|
|
_ -> mempty
|
|
|
|
parseJSON _ = mempty
|
|
|
|
|
|
|
|
plusList :: [B64UUID Plus] -> [String]
|
|
|
|
plusList = map (\(B64UUID u) -> fromUUID u)
|
|
|
|
|
|
|
|
class PlusClass plus unplus where
|
|
|
|
dePlus :: plus -> unplus
|
|
|
|
plus :: unplus -> plus
|
|
|
|
|
|
|
|
instance PlusClass RemoveResultPlus RemoveResult where
|
|
|
|
dePlus (RemoveResultPlus b _) = RemoveResult b
|
|
|
|
plus (RemoveResult b) = RemoveResultPlus b mempty
|
|
|
|
|
|
|
|
instance PlusClass PutResultPlus PutResult where
|
|
|
|
dePlus (PutResultPlus b _) = PutResult b
|
|
|
|
plus (PutResult b) = PutResultPlus b mempty
|
|
|
|
|
|
|
|
instance PlusClass PutOffsetResultPlus PutOffsetResult where
|
|
|
|
dePlus (PutOffsetResultPlus o _) = PutOffsetResult o
|
|
|
|
plus (PutOffsetResult o) = PutOffsetResultPlus o mempty
|