{-# LANGUAGE UndecidableInstances #-}
module Hackage.Security.TUF.Snapshot (
Snapshot(..)
) where
import MyPrelude
import Control.Monad.Except
import Control.Monad.Reader
import Hackage.Security.JSON
import Hackage.Security.TUF.Header
import Hackage.Security.TUF.FileInfo
import Hackage.Security.TUF.FileMap
import Hackage.Security.TUF.Layout.Repo
import Hackage.Security.TUF.Signed
import qualified Hackage.Security.TUF.FileMap as FileMap
import Hackage.Security.Util.Pretty (pretty)
data Snapshot = Snapshot {
Snapshot -> FileVersion
snapshotVersion :: FileVersion
, Snapshot -> FileExpires
snapshotExpires :: FileExpires
, Snapshot -> FileInfo
snapshotInfoRoot :: FileInfo
, Snapshot -> FileInfo
snapshotInfoMirrors :: FileInfo
, Snapshot -> FileInfo
snapshotInfoTarGz :: FileInfo
, Snapshot -> Maybe FileInfo
snapshotInfoTar :: Maybe FileInfo
}
instance HasHeader Snapshot where
fileVersion :: Lens' Snapshot FileVersion
fileVersion FileVersion -> f FileVersion
f Snapshot
x = (\FileVersion
y -> Snapshot
x { snapshotVersion :: FileVersion
snapshotVersion = FileVersion
y }) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> FileVersion -> f FileVersion
f (Snapshot -> FileVersion
snapshotVersion Snapshot
x)
fileExpires :: Lens' Snapshot FileExpires
fileExpires FileExpires -> f FileExpires
f Snapshot
x = (\FileExpires
y -> Snapshot
x { snapshotExpires :: FileExpires
snapshotExpires = FileExpires
y }) forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> FileExpires -> f FileExpires
f (Snapshot -> FileExpires
snapshotExpires Snapshot
x)
instance MonadReader RepoLayout m => ToJSON m Snapshot where
toJSON :: Snapshot -> m JSValue
toJSON Snapshot{Maybe FileInfo
FileExpires
FileVersion
FileInfo
snapshotInfoTar :: Maybe FileInfo
snapshotInfoTarGz :: FileInfo
snapshotInfoMirrors :: FileInfo
snapshotInfoRoot :: FileInfo
snapshotExpires :: FileExpires
snapshotVersion :: FileVersion
snapshotInfoTar :: Snapshot -> Maybe FileInfo
snapshotInfoTarGz :: Snapshot -> FileInfo
snapshotInfoMirrors :: Snapshot -> FileInfo
snapshotInfoRoot :: Snapshot -> FileInfo
snapshotExpires :: Snapshot -> FileExpires
snapshotVersion :: Snapshot -> FileVersion
..} = do
RepoLayout
repoLayout <- forall r (m :: * -> *). MonadReader r m => m r
ask
forall (m :: * -> *). Monad m => [(String, m JSValue)] -> m JSValue
mkObject [
(String
"_type" , forall (m :: * -> *) a. Monad m => a -> m a
return forall a b. (a -> b) -> a -> b
$ String -> JSValue
JSString String
"Snapshot")
, (String
"version" , forall (m :: * -> *) a. ToJSON m a => a -> m JSValue
toJSON FileVersion
snapshotVersion)
, (String
"expires" , forall (m :: * -> *) a. ToJSON m a => a -> m JSValue
toJSON FileExpires
snapshotExpires)
, (String
"meta" , forall (m :: * -> *) a. ToJSON m a => a -> m JSValue
toJSON (RepoLayout -> FileMap
snapshotMeta RepoLayout
repoLayout))
]
where
snapshotMeta :: RepoLayout -> FileMap
snapshotMeta RepoLayout
repoLayout = [(TargetPath, FileInfo)] -> FileMap
FileMap.fromList forall a b. (a -> b) -> a -> b
$ [
(RepoLayout -> TargetPath
pathRoot RepoLayout
repoLayout , FileInfo
snapshotInfoRoot)
, (RepoLayout -> TargetPath
pathMirrors RepoLayout
repoLayout , FileInfo
snapshotInfoMirrors)
, (RepoLayout -> TargetPath
pathIndexTarGz RepoLayout
repoLayout , FileInfo
snapshotInfoTarGz)
] forall a. [a] -> [a] -> [a]
++
[ (RepoLayout -> TargetPath
pathIndexTar RepoLayout
repoLayout , FileInfo
infoTar) | Just FileInfo
infoTar <- [Maybe FileInfo
snapshotInfoTar] ]
instance ( MonadReader RepoLayout m
, MonadError DeserializationError m
, ReportSchemaErrors m
) => FromJSON m Snapshot where
fromJSON :: JSValue -> m Snapshot
fromJSON JSValue
enc = do
forall (m :: * -> *).
(ReportSchemaErrors m, MonadError DeserializationError m) =>
JSValue -> String -> m ()
verifyType JSValue
enc String
"Snapshot"
RepoLayout
repoLayout <- forall r (m :: * -> *). MonadReader r m => m r
ask
FileVersion
snapshotVersion <- forall (m :: * -> *) a.
(ReportSchemaErrors m, FromJSON m a) =>
JSValue -> String -> m a
fromJSField JSValue
enc String
"version"
FileExpires
snapshotExpires <- forall (m :: * -> *) a.
(ReportSchemaErrors m, FromJSON m a) =>
JSValue -> String -> m a
fromJSField JSValue
enc String
"expires"
FileMap
snapshotMeta <- forall (m :: * -> *) a.
(ReportSchemaErrors m, FromJSON m a) =>
JSValue -> String -> m a
fromJSField JSValue
enc String
"meta"
let lookupMeta :: TargetPath -> m FileInfo
lookupMeta TargetPath
k = case TargetPath -> FileMap -> Maybe FileInfo
FileMap.lookup TargetPath
k FileMap
snapshotMeta of
Maybe FileInfo
Nothing -> forall (m :: * -> *) a.
ReportSchemaErrors m =>
String -> Maybe String -> m a
expected (String
"\"" forall a. [a] -> [a] -> [a]
++ forall a. Pretty a => a -> String
pretty TargetPath
k forall a. [a] -> [a] -> [a]
++ String
"\" entry in .meta object") forall a. Maybe a
Nothing
Just FileInfo
v -> forall (f :: * -> *) a. Applicative f => a -> f a
pure FileInfo
v
FileInfo
snapshotInfoRoot <- TargetPath -> m FileInfo
lookupMeta (RepoLayout -> TargetPath
pathRoot RepoLayout
repoLayout)
FileInfo
snapshotInfoMirrors <- TargetPath -> m FileInfo
lookupMeta (RepoLayout -> TargetPath
pathMirrors RepoLayout
repoLayout)
FileInfo
snapshotInfoTarGz <- TargetPath -> m FileInfo
lookupMeta (RepoLayout -> TargetPath
pathIndexTarGz RepoLayout
repoLayout)
let snapshotInfoTar :: Maybe FileInfo
snapshotInfoTar = TargetPath -> FileMap -> Maybe FileInfo
FileMap.lookup (RepoLayout -> TargetPath
pathIndexTar RepoLayout
repoLayout) FileMap
snapshotMeta
forall (m :: * -> *) a. Monad m => a -> m a
return Snapshot{Maybe FileInfo
FileExpires
FileVersion
FileInfo
snapshotInfoTar :: Maybe FileInfo
snapshotInfoTarGz :: FileInfo
snapshotInfoMirrors :: FileInfo
snapshotInfoRoot :: FileInfo
snapshotExpires :: FileExpires
snapshotVersion :: FileVersion
snapshotInfoTar :: Maybe FileInfo
snapshotInfoTarGz :: FileInfo
snapshotInfoMirrors :: FileInfo
snapshotInfoRoot :: FileInfo
snapshotExpires :: FileExpires
snapshotVersion :: FileVersion
..}
instance (MonadKeys m, MonadReader RepoLayout m) => FromJSON m (Signed Snapshot) where
fromJSON :: JSValue -> m (Signed Snapshot)
fromJSON = forall (m :: * -> *) a.
(MonadKeys m, FromJSON m a) =>
JSValue -> m (Signed a)
signedFromJSON
pathRoot, pathMirrors, pathIndexTarGz, pathIndexTar :: RepoLayout -> TargetPath
pathRoot :: RepoLayout -> TargetPath
pathRoot = RepoPath -> TargetPath
TargetPathRepo forall b c a. (b -> c) -> (a -> b) -> a -> c
. RepoLayout -> RepoPath
repoLayoutRoot
pathMirrors :: RepoLayout -> TargetPath
pathMirrors = RepoPath -> TargetPath
TargetPathRepo forall b c a. (b -> c) -> (a -> b) -> a -> c
. RepoLayout -> RepoPath
repoLayoutMirrors
pathIndexTarGz :: RepoLayout -> TargetPath
pathIndexTarGz = RepoPath -> TargetPath
TargetPathRepo forall b c a. (b -> c) -> (a -> b) -> a -> c
. RepoLayout -> RepoPath
repoLayoutIndexTarGz
pathIndexTar :: RepoLayout -> TargetPath
pathIndexTar = RepoPath -> TargetPath
TargetPathRepo forall b c a. (b -> c) -> (a -> b) -> a -> c
. RepoLayout -> RepoPath
repoLayoutIndexTar