// Copyright 2022 ROC. All rights reserved. // Use of this source code is governed by a MIT style // license that can be found in the LICENSE file. package cfg import ( "strings" "github.com/rocboss/paopao-ce/pkg/types" ) // Features fetures info struct type Features struct { kv map[string]string suites map[string][]string features map[string]string } // Actions feature-func map alias type type Actions map[string]types.Fn // NewFeatures create new Features instance func NewFeatures(suites map[string][]string, kv map[string]string) *Features { f := newEmptyFeatures() for k, v := range suites { if len(k) > 0 { for i := 0; i < len(v); i++ { // ignore empty string if len(v[i]) == 0 { lastIdx := len(v) - 1 v[i] = v[lastIdx] v = v[:lastIdx] i-- } } if len(v) > 0 { f.suites[k] = v } } } for k, v := range kv { if len(k) > 0 && len(v) > 0 { f.kv[k] = v } } f.UseDefault() return f } func newEmptyFeatures() *Features { return &Features{ suites: make(map[string][]string), kv: make(map[string]string), features: make(map[string]string), } } // UseDefault use default suite for features func (f *Features) UseDefault() { f.Use([]string{"default"}, true) } // Use use custom suite for features func (f *Features) Use(suite []string, noDefault bool) { if noDefault && len(f.features) != 0 { f.features = make(map[string]string) } features := f.flatFeatures(suite) for _, feature := range features { if len(feature) == 0 { continue } f.features[feature] = f.kv[feature] } } func (f *Features) flatFeatures(suite []string) []string { features := make([]string, 0, len(suite)+10) for s := suite[:]; len(s) > 0; s = s[:len(s)-1] { item := strings.TrimSpace(strings.ToLower(s[0])) if len(item) > 0 { if items, exist := f.suites[item]; exist { s = append(s, items...) } features = append(features, item) } s[0] = s[len(s)-1] } return features } // Cfg get value by key if exist func (f *Features) Cfg(key string) (string, bool) { key = strings.ToLower(key) value, exist := f.features[key] return value, exist } // CfgIf check expression is true. if expression just have a string like // `Sms` is mean `Sms` whether define in suite feature settings. expression like // `Sms = SmsJuhe` is mean whether `Sms` define in suite feature settings and value // is `SmsJuhe` func (f *Features) CfgIf(expression string) bool { kv := strings.Split(expression, "=") key := strings.Trim(strings.ToLower(kv[0]), " ") v, ok := f.features[key] if len(kv) == 2 && ok && strings.Trim(kv[1], " ") == v { return true } else if len(kv) == 1 && ok { return true } return false } // CfgIn range actions to check item's expression is true then do the handle, defFn will handle // if all items are not matched, func (f *Features) CfgIn(actions Actions, defAct ...types.Fn) { itemMatched := false for expression, handle := range actions { if f.CfgIf(expression) && handle != nil { handle() itemMatched = true } } if !itemMatched && len(defAct) > 0 { for _, handle := range defAct { if handle != nil { handle() } } } } // CfgBe check expression is true then do the handle. if expression just have a string like // `Sms` is mean `Sms` whether defined in suite feature settings. expression like // `Sms = SmsJuhe` is mean whether `Sms` define in suite feature settings and value // is `SmsJuhe` func (f *Features) CfgBe(expression string, handle types.Fn) { if f.CfgIf(expression) && handle != nil { handle() } } // CfgNot check expression is not true then do the handle. if expression just have a string like // `Sms` is mean `Sms` whether defined in suite feature settings. expression like // `Sms = SmsJuhe` is mean whether `Sms` define in suite feature settings and value // is `SmsJuhe` func (f *Features) CfgNot(expression string, handle func()) { if !f.CfgIf(expression) { handle() } }