-
Notifications
You must be signed in to change notification settings - Fork 23
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add functions for validating gov action metadata
- Loading branch information
Showing
17 changed files
with
948 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
135 changes: 135 additions & 0 deletions
135
cardano-api/internal/Cardano/Api/Governance/Metadata/DrepRegistration.hs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,135 @@ | ||
{-# LANGUAGE DeriveGeneric #-} | ||
{-# LANGUAGE FlexibleInstances #-} | ||
{-# LANGUAGE InstanceSigs #-} | ||
{-# LANGUAGE LambdaCase #-} | ||
{-# LANGUAGE TypeFamilies #-} | ||
|
||
module Cardano.Api.Governance.Metadata.DrepRegistration | ||
( -- * DRep off-chain metadata | ||
|
||
-- | This module implements validation of metadata for DRep registration and | ||
-- update actions, as specified bt the CIP-119 (https://cips.cardano.org/cip/CIP-0119). | ||
-- | ||
-- The constraints implemented in this module can be tested against a JSON | ||
-- 'ByteString' by using the function 'validateGovActionAnchorData' in | ||
-- "Cardano.Api.Governance.Metadata.Validation" with the parameter 'DrepRegistrationMetadata'. | ||
CIP119 (..) | ||
) | ||
where | ||
|
||
import Cardano.Api.Governance.Metadata.Parsers (textWithMaxLength) | ||
import Cardano.Api.Governance.Metadata.Validation (Authors, Body, GovActionMetadata (..), | ||
HashAlgorithm) | ||
|
||
import Data.Aeson (FromJSON, withObject, (.:), (.:?)) | ||
import qualified Data.Aeson as Aeson | ||
import Data.Aeson.Types (Parser) | ||
import Data.Text (Text) | ||
import GHC.Generics (Generic) | ||
|
||
data CIP119 = DrepRegistrationMetadata | ||
|
||
instance FromJSON (GovActionMetadata CIP119) where | ||
parseJSON :: Aeson.Value -> Parser (GovActionMetadata CIP119) | ||
parseJSON = withObject "CIP119Common" $ \v -> | ||
GovActionMetadata | ||
<$> v .: "hashAlgorithm" | ||
<*> pure Absent | ||
<*> v .: "body" | ||
|
||
-- Hash Algorithm (Enum) | ||
data instance HashAlgorithm CIP119 = Blake2b256 | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON (HashAlgorithm CIP119) where | ||
parseJSON :: Aeson.Value -> Parser (HashAlgorithm CIP119) | ||
parseJSON = Aeson.withText "HashAlgorithm" $ | ||
\case | ||
"blake2b-256" -> return Blake2b256 | ||
_ -> fail "Invalid hashAlgorithm, it must be: blake2b-256" | ||
|
||
-- Body of the metadata document | ||
data instance Body CIP119 = Body | ||
{ paymentAddress :: Maybe Text | ||
, givenName :: Text | ||
, image :: Maybe ImageObject | ||
, objectives :: Maybe Text | ||
, motivations :: Maybe Text | ||
, qualifications :: Maybe Text | ||
, doNotList :: Maybe DoNotList | ||
, references :: Maybe [Reference] | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON (Body CIP119) where | ||
parseJSON :: Aeson.Value -> Parser (Body CIP119) | ||
parseJSON = withObject "Body" $ \v -> | ||
Body | ||
<$> v .:? "paymentAddress" | ||
<*> (v .: "givenName" >>= textWithMaxLength "givenName" 80) | ||
<*> v .:? "image" | ||
<*> (v .:? "objectives" >>= traverse (textWithMaxLength "objectives" 1000)) | ||
<*> (v .:? "motivations" >>= traverse (textWithMaxLength "motivations" 1000)) | ||
<*> (v .:? "qualifications" >>= traverse (textWithMaxLength "qualifications" 1000)) | ||
<*> v .:? "doNotList" | ||
<*> v .:? "references" | ||
|
||
-- Profile picture | ||
data ImageObject = ImageObject | ||
{ contentUrl :: Text -- Base64 encoded image or URL | ||
, sha256 :: Maybe Text -- Only present for URL images | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON ImageObject where | ||
parseJSON :: Aeson.Value -> Parser ImageObject | ||
parseJSON = withObject "ImageObject" $ \v -> | ||
ImageObject | ||
<$> v .: "contentUrl" | ||
<*> v .:? "sha256" | ||
|
||
-- DoNotList Enum | ||
data DoNotList = DoNotListTrue | DoNotListFalse | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON DoNotList where | ||
parseJSON :: Aeson.Value -> Parser DoNotList | ||
parseJSON = Aeson.withText "DoNotList" $ | ||
\case | ||
"true" -> return DoNotListTrue | ||
"false" -> return DoNotListFalse | ||
_ -> fail "Invalid doNotList value, must be one of: true, false" | ||
|
||
-- Reference type | ||
data Reference = Reference | ||
{ refType :: ReferenceType | ||
, label :: Text | ||
, uri :: Text | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON Reference where | ||
parseJSON :: Aeson.Value -> Parser Reference | ||
parseJSON = withObject "Reference" $ \v -> | ||
Reference | ||
<$> v .: "@type" | ||
<*> v .: "label" | ||
<*> v .: "uri" | ||
|
||
-- ReferenceType Enum | ||
data ReferenceType = GovernanceMetadata | Other | Link | Identity | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON ReferenceType where | ||
parseJSON :: Aeson.Value -> Parser ReferenceType | ||
parseJSON = Aeson.withText "ReferenceType" $ | ||
\case | ||
"GovernanceMetadata" -> return GovernanceMetadata | ||
"Other" -> return Other | ||
"Link" -> return Link | ||
"Identity" -> return Identity | ||
_ -> | ||
fail "Invalid reference type, must be one of: GovernanceMetadata, Other, Link, Identity" | ||
|
||
-- We don't need to validate Authors because it is optional in CIP-119 | ||
data instance Authors CIP119 = Absent |
167 changes: 167 additions & 0 deletions
167
cardano-api/internal/Cardano/Api/Governance/Metadata/GovAction.hs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,167 @@ | ||
{-# LANGUAGE DeriveGeneric #-} | ||
{-# LANGUAGE FlexibleInstances #-} | ||
{-# LANGUAGE InstanceSigs #-} | ||
{-# LANGUAGE LambdaCase #-} | ||
{-# LANGUAGE TypeFamilies #-} | ||
|
||
module Cardano.Api.Governance.Metadata.GovAction | ||
( -- * Government action metadata | ||
|
||
-- | This module implements validation of metadata for Government Actions in | ||
-- general, as specified bt the CIP-108 (https://cips.cardano.org/cip/CIP-0108), | ||
-- except for Government Actions covered by other CIPs. | ||
-- | ||
-- The constraints implemented in this module can be tested against a JSON | ||
-- 'ByteString' by using the function 'validateGovActionAnchorData' in | ||
-- "Cardano.Api.Governance.Metadata.Validation" with the parameter 'BaseGovActionMetadata'. | ||
CIP108 (..) | ||
) | ||
where | ||
|
||
import Cardano.Api.Governance.Metadata.Parsers (textWithMaxLength) | ||
import Cardano.Api.Governance.Metadata.Validation (Authors, Body, GovActionMetadata (..), | ||
HashAlgorithm) | ||
|
||
import Data.Aeson (FromJSON, withArray, withObject, withText, (.:), (.:?)) | ||
import qualified Data.Aeson as Aeson | ||
import Data.Aeson.Types (Parser, Value (..)) | ||
import Data.Text (Text) | ||
import GHC.Generics (Generic) | ||
|
||
data CIP108 = BaseGovActionMetadata | ||
|
||
instance FromJSON (GovActionMetadata CIP108) where | ||
parseJSON :: Value -> Parser (GovActionMetadata CIP108) | ||
parseJSON = withObject "CIP108Common" $ \v -> | ||
GovActionMetadata | ||
<$> v .: "hashAlgorithm" | ||
<*> v .: "authors" | ||
<*> v .: "body" | ||
|
||
-- Enum for HashAlgorithm | ||
|
||
data instance HashAlgorithm CIP108 = Blake2b256 | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON (HashAlgorithm CIP108) where | ||
parseJSON :: Value -> Parser (HashAlgorithm CIP108) | ||
parseJSON = withText "HashAlgorithm" $ | ||
\case | ||
"blake2b-256" -> return Blake2b256 | ||
_ -> fail "Invalid hashAlgorithm value, must be: blake2b-256" | ||
|
||
-- Author object | ||
|
||
newtype instance Authors CIP108 = Authors [Author] | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON (Authors CIP108) where | ||
parseJSON :: Value -> Parser (Authors CIP108) | ||
parseJSON = withArray "Authors" $ \arr -> | ||
Authors <$> Aeson.parseJSON (Array arr) | ||
|
||
data Author = Author | ||
{ name :: Maybe Text | ||
, witness :: Witness | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON Author where | ||
parseJSON :: Value -> Parser Author | ||
parseJSON = withObject "Author" $ \v -> | ||
Author | ||
<$> v .:? "name" | ||
<*> v .: "witness" | ||
|
||
-- Witness object | ||
data Witness = Witness | ||
{ witnessAlgorithm :: Maybe WitnessAlgorithm | ||
, publicKey :: Maybe Text | ||
, signature :: Maybe Text | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON Witness where | ||
parseJSON :: Value -> Parser Witness | ||
parseJSON = withObject "Witness" $ \v -> | ||
Witness | ||
<$> v .:? "witnessAlgorithm" | ||
<*> v .:? "publicKey" | ||
<*> v .:? "signature" | ||
|
||
-- Enum for WitnessAlgorithm | ||
data WitnessAlgorithm = Ed25519 | CIP0008 | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON WitnessAlgorithm where | ||
parseJSON :: Value -> Parser WitnessAlgorithm | ||
parseJSON = withText "WitnessAlgorithm" $ | ||
\case | ||
"ed25519" -> return Ed25519 | ||
"CIP-0008" -> return CIP0008 | ||
_ -> fail "Invalid witnessAlgorithm value, must be: ed25519 or CIP-0008" | ||
|
||
-- Body of the metadata document | ||
|
||
data instance Body CIP108 = Body | ||
{ title :: Text | ||
, abstract :: Text | ||
, motivation :: Text | ||
, rationale :: Text | ||
, references :: Maybe [Reference] | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON (Body CIP108) where | ||
parseJSON :: Value -> Parser (Body CIP108) | ||
parseJSON = withObject "Body" $ \v -> | ||
Body | ||
<$> (v .: "title" >>= textWithMaxLength "title" 80) | ||
<*> (v .: "abstract" >>= textWithMaxLength "abstract" 2500) | ||
<*> v .: "motivation" | ||
<*> v .: "rationale" | ||
<*> v .:? "references" | ||
|
||
-- Reference object | ||
data Reference = Reference | ||
{ refType :: ReferenceType | ||
, label :: Text | ||
, uri :: Text | ||
, referenceHash :: Maybe ReferenceHash | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON Reference where | ||
parseJSON :: Value -> Parser Reference | ||
parseJSON = withObject "Reference" $ \v -> | ||
Reference | ||
<$> v .: "@type" | ||
<*> v .: "label" | ||
<*> v .: "uri" | ||
<*> v .:? "referenceHash" | ||
|
||
-- Enum for ReferenceType | ||
data ReferenceType = GovernanceMetadata | Other | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON ReferenceType where | ||
parseJSON :: Value -> Parser ReferenceType | ||
parseJSON = withText "ReferenceType" $ | ||
\case | ||
"GovernanceMetadata" -> return GovernanceMetadata | ||
"Other" -> return Other | ||
_ -> fail "Invalid reference type, must be one of: GovernanceMetadata, Other" | ||
|
||
-- ReferenceHash object | ||
data ReferenceHash = ReferenceHash | ||
{ referenceHashDigest :: Text | ||
, referenceHashAlgorithm :: HashAlgorithm CIP108 | ||
} | ||
deriving (Show, Generic) | ||
|
||
instance FromJSON ReferenceHash where | ||
parseJSON :: Value -> Parser ReferenceHash | ||
parseJSON = withObject "ReferenceHash" $ \v -> | ||
ReferenceHash | ||
<$> v .: "hashDigest" | ||
<*> v .: "hashAlgorithm" |
22 changes: 22 additions & 0 deletions
22
cardano-api/internal/Cardano/Api/Governance/Metadata/Parsers.hs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
module Cardano.Api.Governance.Metadata.Parsers (textWithMaxLength) where | ||
|
||
import Data.Aeson.Types (Parser, Value, parseJSON) | ||
import Data.Text (Text) | ||
import qualified Data.Text as T | ||
|
||
-- | Parser for 'Text' that validates that the number of characters is | ||
-- under a given maximum. The 'String' parameter is meant to be the name | ||
-- of the field in order to be able to give context in case of error. | ||
textWithMaxLength :: String -> Int -> Value -> Parser Text | ||
textWithMaxLength fieldName maxLen value = do | ||
txt <- parseJSON value | ||
if T.length txt <= maxLen | ||
then pure txt | ||
else | ||
fail $ | ||
"key \"" | ||
++ fieldName | ||
++ "\" exceeds maximum length of " | ||
++ show maxLen | ||
++ " characters. Got length: " | ||
++ show (T.length txt) |
Oops, something went wrong.