Files

86 lines
2.7 KiB
Go

// Copyright 2023 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package fixcontrol
import (
"fmt"
"strconv"
"strings"
"github.com/pingcap/errors"
)
// ParseToMap parses the user input to an optimizer fix control map.
func ParseToMap(s string) (result map[uint64]string, warnMsgs []string, err error) {
m := make(map[uint64]string)
for ; len(s) > 0; s = strings.TrimSpace(s) {
// find the colon
colonIdx := strings.IndexByte(s, ':')
if colonIdx < 0 {
err = errors.New("invalid fix control: expected colon not found")
return nil, nil, err
}
// 1. the part before the colon is the fix control number and also the key in the map
keyStr := strings.TrimSpace(s[0:colonIdx])
key, err := strconv.ParseUint(keyStr, 10, 64)
if err != nil {
return nil, nil, err
}
// 2. the part after the colon is the value set for this fix
s = strings.TrimSpace(s[colonIdx+1:])
var value string
// if it starts with quote, we need to find the closing quote, and the content between is the value for this fix
if s[0] == '\'' || s[0] == '"' {
quote := s[0]
endIdx := strings.IndexByte(s[1:], quote)
if endIdx < 0 {
err = errors.New("invalid fix control: expected quote not found")
return nil, nil, err
}
endIdx = endIdx + 1
value = s[1:endIdx]
s = s[endIdx+1:]
}
// if there's no comma, the remaining content belong to this fix
endIdx := len(s)
nextStartIdx := len(s)
commaIdx := strings.IndexByte(s, ',')
// if we find a comma, we need to locate the end for this fix and the start of the next fix
if commaIdx >= 0 {
endIdx = commaIdx
nextStartIdx = commaIdx + 1
}
// if there's no quote, get the value for this fix now.
if len(value) == 0 {
value = strings.TrimSpace(s[:endIdx])
}
// 3. key and value are found, now handling repeated key and set into map
originalValue, ok := m[key]
if ok && originalValue != value {
warnMsg := fmt.Sprintf("repeated assignment for fix control: %d. existing value: %q. new value: %q.",
key,
originalValue,
value)
warnMsgs = append(warnMsgs, warnMsg)
}
m[key] = value
// 4. prepare to parse remaining content
s = s[nextStartIdx:]
}
return m, warnMsgs, nil
}