hi

For an experiment, I'm trying to rewrite a method call to use a different 
package.

for example 

package main

import "github.com/pkg/errors"

func main() {
    err := errors.New("original error")
    errors.Wrap(err, "wrapped error")
}

into 

package main

import (
    "errors"
    "fmt"
)

func main() {
    err := errors.New("original err")
    fmt.Errorf("wrapped error: +%v", err)
}

So far I've been using ast functions like this

func processFile(file *ast.File, info *types.Info) {
    ast.Inspect(file, func(n ast.Node) bool {
        if n, ok := n.(*ast.CallExpr); ok {
            if selExpr, ok := n.Fun.(*ast.SelectorExpr); ok {
                // check if the call uses pkgErrors path
                obj := info.Uses[selExpr.Sel]
                if obj.Pkg().Path() != pkgErrors {
                    return true
                }

                // rename caller package to 'fmt'
                if x, ok := selExpr.X.(*ast.Ident); ok {
                    x.Name = "fmt"
                }

                // change method call
                switch selExpr.Sel.Name {
                case "Wrap":
                    selExpr.Sel.Name = "Printf"
                case "Wrapf":
                    selExpr.Sel.Name = "Printf"
                }

                // reverse the arguments
                n.Args = append(n.Args[1:], n.Args[0])

                // TODO: format first argument by appending ": %w" in the 
end
            }
        }

        return true // recur
    })

    astutil.AddImport(fset, file, "errors")
    astutil.DeleteImport(fset, file, pkgErrors)
}

And using this function to test it

func TestProcessFile(t *testing.T) {
    const src = `package main

import "github.com/pkg/errors"

func main() {
    err := errors.New("original error")
    errors.Wrap(err, "wrapped error")
}`

    file, err := parser.ParseFile(fset, "input.go", src, parser.AllErrors)
    if err != nil {
        t.Fatal(err)
    }

    conf := types.Config{Importer: importer.Default()}
    info := &types.Info{
        Types:      map[ast.Expr]types.TypeAndValue{},
        Defs:       map[*ast.Ident]types.Object{},
        Uses:       map[*ast.Ident]types.Object{},
        Implicits:  map[ast.Node]types.Object{},
        Selections: map[*ast.SelectorExpr]*types.Selection{},
        Scopes:     map[ast.Node]*types.Scope{},
    }

    _, err = conf.Check("cmd/hello", fset, []*ast.File{file}, info)
    if err != nil {
        t.Fatal(err)
    }

    processFile(file, info)

    //ast.Print(fset, file)
    _ = format.Node(os.Stdout, fset, file)

}

So far it gives me the correct output. But I'm wondering if this is the 
correct approach. It feels like by changing the value of Name I messed up 
the tokens positions.
Also I've been reading the `gofmt` and `eg`. that uses reflections but I'm 
not sure if I should use it.

Is there a better approach to achieve the desire result? looking for some 
pointers to do this.

Regards

-- 
You received this message because you are subscribed to the Google Groups 
"golang-nuts" group.
To unsubscribe from this group and stop receiving emails from it, send an email 
to golang-nuts+unsubscr...@googlegroups.com.
To view this discussion on the web visit 
https://groups.google.com/d/msgid/golang-nuts/2abc0171-9f1e-4574-b115-5f2ff905a76e%40googlegroups.com.

Reply via email to