mirror of
				https://github.com/go-delve/delve.git
				synced 2025-10-31 18:57:18 +08:00 
			
		
		
		
	 1758f8523a
			
		
	
	1758f8523a
	
	
	
		
			
			Adds a configuration option (show-location-expr) that when activated will cause the whatis command to also print the DWARF location expression for a variable.
		
			
				
	
	
		
			192 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package terminal
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"reflect"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"text/tabwriter"
 | |
| 
 | |
| 	"github.com/derekparker/delve/pkg/config"
 | |
| )
 | |
| 
 | |
| func configureCmd(t *Term, ctx callContext, args string) error {
 | |
| 	switch args {
 | |
| 	case "-list":
 | |
| 		return configureList(t)
 | |
| 	case "-save":
 | |
| 		return config.SaveConfig(t.conf)
 | |
| 	case "":
 | |
| 		return fmt.Errorf("wrong number of arguments to \"config\"")
 | |
| 	default:
 | |
| 		return configureSet(t, args)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type configureIterator struct {
 | |
| 	cfgValue reflect.Value
 | |
| 	cfgType  reflect.Type
 | |
| 	i        int
 | |
| }
 | |
| 
 | |
| func iterateConfiguration(conf *config.Config) *configureIterator {
 | |
| 	cfgValue := reflect.ValueOf(conf).Elem()
 | |
| 	cfgType := cfgValue.Type()
 | |
| 
 | |
| 	return &configureIterator{cfgValue, cfgType, -1}
 | |
| }
 | |
| 
 | |
| func (it *configureIterator) Next() bool {
 | |
| 	it.i++
 | |
| 	return it.i < it.cfgValue.NumField()
 | |
| }
 | |
| 
 | |
| func (it *configureIterator) Field() (name string, field reflect.Value) {
 | |
| 	name = it.cfgType.Field(it.i).Tag.Get("yaml")
 | |
| 	if comma := strings.Index(name, ","); comma >= 0 {
 | |
| 		name = name[:comma]
 | |
| 	}
 | |
| 	field = it.cfgValue.Field(it.i)
 | |
| 	return
 | |
| }
 | |
| 
 | |
| func configureFindFieldByName(conf *config.Config, name string) reflect.Value {
 | |
| 	it := iterateConfiguration(conf)
 | |
| 	for it.Next() {
 | |
| 		fieldName, field := it.Field()
 | |
| 		if fieldName == name {
 | |
| 			return field
 | |
| 		}
 | |
| 	}
 | |
| 	return reflect.ValueOf(nil)
 | |
| }
 | |
| 
 | |
| func configureList(t *Term) error {
 | |
| 	w := new(tabwriter.Writer)
 | |
| 	w.Init(os.Stdout, 0, 8, 1, ' ', 0)
 | |
| 
 | |
| 	it := iterateConfiguration(t.conf)
 | |
| 	for it.Next() {
 | |
| 		fieldName, field := it.Field()
 | |
| 		if fieldName == "" {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		if field.Kind() == reflect.Ptr {
 | |
| 			if !field.IsNil() {
 | |
| 				fmt.Fprintf(w, "%s\t%v\n", fieldName, field.Elem())
 | |
| 			} else {
 | |
| 				fmt.Fprintf(w, "%s\t<not defined>\n", fieldName)
 | |
| 			}
 | |
| 		} else {
 | |
| 			fmt.Fprintf(w, "%s\t%v\n", fieldName, field)
 | |
| 		}
 | |
| 	}
 | |
| 	return w.Flush()
 | |
| }
 | |
| 
 | |
| func configureSet(t *Term, args string) error {
 | |
| 	v := strings.SplitN(args, " ", 2)
 | |
| 
 | |
| 	cfgname := v[0]
 | |
| 	var rest string
 | |
| 	if len(v) == 2 {
 | |
| 		rest = v[1]
 | |
| 	}
 | |
| 
 | |
| 	if cfgname == "alias" {
 | |
| 		return configureSetAlias(t, rest)
 | |
| 	}
 | |
| 
 | |
| 	field := configureFindFieldByName(t.conf, cfgname)
 | |
| 	if !field.CanAddr() {
 | |
| 		return fmt.Errorf("%q is not a configuration parameter", cfgname)
 | |
| 	}
 | |
| 
 | |
| 	if field.Kind() == reflect.Slice && field.Type().Elem().Name() == "SubstitutePathRule" {
 | |
| 		return configureSetSubstituePath(t, rest)
 | |
| 	}
 | |
| 
 | |
| 	simpleArg := func(typ reflect.Type) (reflect.Value, error) {
 | |
| 		switch typ.Kind() {
 | |
| 		case reflect.Int:
 | |
| 			n, err := strconv.Atoi(rest)
 | |
| 			if err != nil {
 | |
| 				return reflect.ValueOf(nil), fmt.Errorf("argument to %q must be a number", cfgname)
 | |
| 			}
 | |
| 			return reflect.ValueOf(&n), nil
 | |
| 		case reflect.Bool:
 | |
| 			v := rest == "true"
 | |
| 			return reflect.ValueOf(&v), nil
 | |
| 		default:
 | |
| 			return reflect.ValueOf(nil), fmt.Errorf("unsupported type for configuration key %q", cfgname)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if field.Kind() == reflect.Ptr {
 | |
| 		val, err := simpleArg(field.Type().Elem())
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		field.Set(val)
 | |
| 	} else {
 | |
| 		val, err := simpleArg(field.Type())
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		field.Set(val.Elem())
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func configureSetSubstituePath(t *Term, rest string) error {
 | |
| 	argv := config.SplitQuotedFields(rest, '"')
 | |
| 	switch len(argv) {
 | |
| 	case 1: // delete substitute-path rule
 | |
| 		for i := range t.conf.SubstitutePath {
 | |
| 			if t.conf.SubstitutePath[i].From == argv[0] {
 | |
| 				copy(t.conf.SubstitutePath[i:], t.conf.SubstitutePath[i+1:])
 | |
| 				t.conf.SubstitutePath = t.conf.SubstitutePath[:len(t.conf.SubstitutePath)-1]
 | |
| 				return nil
 | |
| 			}
 | |
| 		}
 | |
| 		return fmt.Errorf("could not find rule for %q", argv[0])
 | |
| 	case 2: // add substitute-path rule
 | |
| 		for i := range t.conf.SubstitutePath {
 | |
| 			if t.conf.SubstitutePath[i].From == argv[0] {
 | |
| 				t.conf.SubstitutePath[i].To = argv[1]
 | |
| 				return nil
 | |
| 			}
 | |
| 		}
 | |
| 		t.conf.SubstitutePath = append(t.conf.SubstitutePath, config.SubstitutePathRule{argv[0], argv[1]})
 | |
| 	default:
 | |
| 		return fmt.Errorf("too many arguments to \"config substitute-path\"")
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func configureSetAlias(t *Term, rest string) error {
 | |
| 	argv := config.SplitQuotedFields(rest, '"')
 | |
| 	switch len(argv) {
 | |
| 	case 1: // delete alias rule
 | |
| 		for k := range t.conf.Aliases {
 | |
| 			v := t.conf.Aliases[k]
 | |
| 			for i := range v {
 | |
| 				if v[i] == argv[0] {
 | |
| 					copy(v[i:], v[i+1:])
 | |
| 					t.conf.Aliases[k] = v[:len(v)-1]
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	case 2: // add alias rule
 | |
| 		alias, cmd := argv[1], argv[0]
 | |
| 		if t.conf.Aliases == nil {
 | |
| 			t.conf.Aliases = make(map[string][]string)
 | |
| 		}
 | |
| 		t.conf.Aliases[cmd] = append(t.conf.Aliases[cmd], alias)
 | |
| 	}
 | |
| 	t.cmds.Merge(t.conf.Aliases)
 | |
| 	return nil
 | |
| }
 |