From dda38ffb8d00dadf523d0ce810183df67d45ac0c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Cl=C3=A9ment=20Hurlin?= Date: Thu, 8 Aug 2024 17:56:08 +0200 Subject: [PATCH] Common.hs: avoid using Opt.auto to avoid overflows going silent --- .../Cardano/CLI/EraBased/Options/Common.hs | 22 +++++++++++++++++-- 1 file changed, 20 insertions(+), 2 deletions(-) diff --git a/cardano-cli/src/Cardano/CLI/EraBased/Options/Common.hs b/cardano-cli/src/Cardano/CLI/EraBased/Options/Common.hs index 652d35591c..d4519b6569 100644 --- a/cardano-cli/src/Cardano/CLI/EraBased/Options/Common.hs +++ b/cardano-cli/src/Cardano/CLI/EraBased/Options/Common.hs @@ -4,6 +4,7 @@ {-# LANGUAGE LambdaCase #-} {-# LANGUAGE RankNTypes #-} {-# LANGUAGE ScopedTypeVariables #-} +{-# LANGUAGE TypeApplications #-} {- HLINT ignore "Move brackets to avoid $" -} {- HLINT ignore "Use <$>" -} @@ -28,9 +29,11 @@ import qualified Ouroboros.Network.Protocol.LocalStateQuery.Type as Consensus import Control.Monad (mfilter) import qualified Data.Aeson as Aeson import Data.Bifunctor +import Data.Bits (Bits, toIntegralSized) import Data.ByteString (ByteString) import qualified Data.ByteString.Base16 as B16 import qualified Data.ByteString.Char8 as BSC +import Data.Data (Proxy (..), Typeable, typeRep) import Data.Foldable import Data.Functor (($>)) import qualified Data.IP as IP @@ -43,7 +46,7 @@ import qualified Data.Text as Text import Data.Time.Clock (UTCTime) import Data.Time.Format (defaultTimeLocale, parseTimeOrError) import Data.Word -import GHC.Natural (Natural) +import GHC.Natural (Natural, naturalToWordMaybe) import Network.Socket (PortNumber) import Options.Applicative hiding (help, str) import qualified Options.Applicative as Opt @@ -3181,9 +3184,24 @@ pMaxTransactionSize = , Opt.help "Maximum transaction size." ] +-- | @wordReader typeName@ is a reader for a word of type @a@. When it fails +-- parsing, it provides a nice error message. This custom reader is needed +-- to avoid the overflow issues of 'Opt.auto' described in https://github.com/IntersectMBO/cardano-cli/issues/860. +wordReader :: forall a. (Typeable a, Integral a, Bits a) => ReadM a +wordReader = + Opt.eitherReader parser + where + parser s = + case readMaybe s >>= naturalToWordMaybe >>= toIntegralSized of + Nothing -> + Left $ "Cannot parse " <> s <> " as a " <> typeName + Just a -> + Right a + typeName = show $ typeRep (Proxy @a) + pMaxBlockHeaderSize :: Parser Word16 pMaxBlockHeaderSize = - Opt.option Opt.auto $ + Opt.option wordReader $ mconcat [ Opt.long "max-block-header-size" , Opt.metavar "WORD16"