From a788e3376b603d29bfedef5ae11912c5c67a110e Mon Sep 17 00:00:00 2001 From: Ethen Pociask Date: Mon, 26 Aug 2024 02:28:47 -0400 Subject: [PATCH] feat: Support explict modes for read fallback and caching - updated cfg validation tests --- server/config.go | 66 ++++++++++++-------- server/config_test.go | 48 ++++++++++++++ store/s3.go | 11 ++++ utils/{parse_bytes.go => utils.go} | 20 ++++++ utils/{parse_bytes_test.go => utils_test.go} | 0 5 files changed, 119 insertions(+), 26 deletions(-) rename utils/{parse_bytes.go => utils.go} (81%) rename utils/{parse_bytes_test.go => utils_test.go} (100%) diff --git a/server/config.go b/server/config.go index 193f45fd..293eea35 100644 --- a/server/config.go +++ b/server/config.go @@ -54,12 +54,15 @@ const ( CacheTargets = "routing.cache-targets" ) -const BytesPerSymbol = 31 -const MaxCodingRatio = 8 - -var MaxSRSPoints = 1 << 28 // 2^28 +const ( + BytesPerSymbol = 31 + MaxCodingRatio = 8 +) -var MaxAllowedBlobSize = uint64(MaxSRSPoints * BytesPerSymbol / MaxCodingRatio) +var ( + MaxSRSPoints = 1 << 28 // 2^28 + MaxAllowedBlobSize = uint64(MaxSRSPoints * BytesPerSymbol / MaxCodingRatio) +) type Config struct { S3Config store.S3Config @@ -93,6 +96,7 @@ type Config struct { CacheTargets []string } +// GetMaxBlobLength ... returns the maximum blob length in bytes func (cfg *Config) GetMaxBlobLength() (uint64, error) { if cfg.maxBlobLengthBytes == 0 { numBytes, err := utils.ParseBytesAmount(cfg.MaxBlobLength) @@ -110,6 +114,7 @@ func (cfg *Config) GetMaxBlobLength() (uint64, error) { return cfg.maxBlobLengthBytes, nil } +// VerificationCfg ... returns certificate config used to verify blobs from eigenda func (cfg *Config) VerificationCfg() *verify.Config { numBytes, err := cfg.GetMaxBlobLength() if err != nil { @@ -141,11 +146,11 @@ func (cfg *Config) VerificationCfg() *verify.Config { } } -// ReadConfig parses the Config from the provided flags or environment variables. +// ReadConfig ... parses the Config from the provided flags or environment variables. func ReadConfig(ctx *cli.Context) Config { cfg := Config{ S3Config: store.S3Config{ - S3CredentialType: toS3CredentialType(ctx.String(S3CredentialTypeFlagName)), + S3CredentialType: store.StringToS3CredentialType(ctx.String(S3CredentialTypeFlagName)), Bucket: ctx.String(S3BucketFlagName), Path: ctx.String(S3PathFlagName), Endpoint: ctx.String(S3EndpointFlagName), @@ -182,15 +187,23 @@ func ReadConfig(ctx *cli.Context) Config { return cfg } -func toS3CredentialType(s string) store.S3CredentialType { - switch s { - case string(store.S3CredentialStatic): - return store.S3CredentialStatic - case string(store.S3CredentialIAM): - return store.S3CredentialIAM - default: - return store.S3CredentialUnknown +// checkTargets ... verifies that a backend target slice is constructed correctly +func (cfg *Config) checkTargets(targets []string) error { + if len(targets) == 0 { + return nil + } + + if utils.ContainsDuplicates(targets) { + return fmt.Errorf("duplicate fallback targets provided") + } + + for _, t := range targets { + if store.StringToBackend(t) == store.Unknown { + return fmt.Errorf("unknown fallback target provided: %s", t) + } } + + return nil } // Check ... verifies that configuration values are adequately set @@ -229,19 +242,20 @@ func (cfg *Config) Check() error { return fmt.Errorf("eigenda disperser rpc url is not set") } - if len(cfg.FallbackTargets) > 0 { - for _, t := range cfg.FallbackTargets { - if store.StringToBackend(t) == store.Unknown { - return fmt.Errorf("unknown fallback target provided: %s", t) - } - } + err = cfg.checkTargets(cfg.FallbackTargets) + if err != nil { + return err + } + + err = cfg.checkTargets(cfg.CacheTargets) + if err != nil { + return err } - if len(cfg.CacheTargets) > 0 { - for _, t := range cfg.CacheTargets { - if store.StringToBackend(t) == store.Unknown { - return fmt.Errorf("unknown cache target provided: %s", t) - } + // verify that same target is not in both fallback and cache targets + for _, t := range cfg.FallbackTargets { + if utils.Contains(cfg.CacheTargets, t) { + return fmt.Errorf("target %s is in both fallback and cache targets", t) } } diff --git a/server/config_test.go b/server/config_test.go index 2c13df74..f045fd33 100644 --- a/server/config_test.go +++ b/server/config_test.go @@ -126,4 +126,52 @@ func TestConfigVerification(t *testing.T) { err := cfg.Check() require.Error(t, err) }) + + t.Run("InvalidFallbackTarget", func(t *testing.T) { + cfg := validCfg() + cfg.FallbackTargets = []string{"postgres"} + + err := cfg.Check() + require.Error(t, err) + }) + + t.Run("InvalidCacheTarget", func(t *testing.T) { + cfg := validCfg() + cfg.CacheTargets = []string{"postgres"} + + err := cfg.Check() + require.Error(t, err) + }) + t.Run("InvalidCacheTarget", func(t *testing.T) { + cfg := validCfg() + cfg.CacheTargets = []string{"postgres"} + + err := cfg.Check() + require.Error(t, err) + }) + + t.Run("DuplicateCacheTargets", func(t *testing.T) { + cfg := validCfg() + cfg.CacheTargets = []string{"s3", "s3"} + + err := cfg.Check() + require.Error(t, err) + }) + + t.Run("DuplicateFallbackTargets", func(t *testing.T) { + cfg := validCfg() + cfg.FallbackTargets = []string{"s3", "s3"} + + err := cfg.Check() + require.Error(t, err) + }) + + t.Run("OverlappingCacheFallbackTargets", func(t *testing.T) { + cfg := validCfg() + cfg.FallbackTargets = []string{"s3"} + cfg.CacheTargets = []string{"s3"} + + err := cfg.Check() + require.Error(t, err) + }) } diff --git a/store/s3.go b/store/s3.go index fa04a958..12d5c1dc 100644 --- a/store/s3.go +++ b/store/s3.go @@ -21,6 +21,17 @@ const ( S3CredentialUnknown S3CredentialType = "unknown" ) +func StringToS3CredentialType(s string) S3CredentialType { + switch s { + case "static": + return S3CredentialStatic + case "iam": + return S3CredentialIAM + default: + return S3CredentialUnknown + } +} + var _ PrecomputedKeyStore = (*S3Store)(nil) type S3CredentialType string diff --git a/utils/parse_bytes.go b/utils/utils.go similarity index 81% rename from utils/parse_bytes.go rename to utils/utils.go index 8a90119a..0a0b7106 100644 --- a/utils/parse_bytes.go +++ b/utils/utils.go @@ -8,6 +8,26 @@ import ( // Helper utility functions // +func ContainsDuplicates[P comparable](s []P) bool { + seen := make(map[P]interface{}) + for _, v := range s { + if _, ok := seen[v]; ok { + return true + } + seen[v] = interface{}(nil) + } + return false +} + +func Contains[P comparable](s []P, e P) bool { + for _, v := range s { + if v == e { + return true + } + } + return false +} + func EqualSlices[P comparable](s1, s2 []P) bool { if len(s1) != len(s2) { return false diff --git a/utils/parse_bytes_test.go b/utils/utils_test.go similarity index 100% rename from utils/parse_bytes_test.go rename to utils/utils_test.go