agiledragon / gomonkey

gomonkey is a library to make monkey patching in unit tests easy
MIT License
1.93k stars 178 forks source link

ApplyMethod possibly not work with generic types #104

Open qiuchengxuan opened 1 year ago

qiuchengxuan commented 1 year ago

这里定义了一个泛型的结构体,大致是这样:

type Table[T any] struct {
    ...
}

func (t *Table[T]) Add(network net.IPNet, value T) bool {
    ...
}

这里内容无关紧要,在使用gomonkey patch了叫Add的方法后,发现单元测试并未成功,调用的依旧是原本的Add。

进一步定位发现,reflect包获取到的方法名和实际的Add方法的signature并不相同,实际上方法地址也并不相同

reflect获取的方法名和地址:

0x1c56aa0
XXX.(*Table[string]).Add(SB) <autogenerated>
        <autogenerated>:1       0x1c56aa0       493b6610                cmp rsp, qword ptr [r14+0x10]
        <autogenerated>:1       0x1c56aa4       0f86cd000000            jbe 0x1c56b77

实际调用的方法名和地址:

XXX.(*Table[go.shape.string_0]).Add(SB) .../cidrtable.go
=>      cidrtable.go:29 0x1c56020       4c8da42458ffffff        lea r12, ptr [rsp+0xffffff58]
        cidrtable.go:29 0x1c56028       4d3b6610                cmp r12, qword ptr [r14+0x10]

gomonkey是通过取reflect记录的地址来修改实际调用的,但是看来这个方式对于泛型方法暂且不适用

golang版本1.18.2和1.18.5均存在上述问题

agiledragon commented 1 year ago

请参考《gomonkey用户如何对泛型打桩》[https://www.jianshu.com/p/8a52eae7f786]

qiuchengxuan commented 1 year ago

请参考《gomonkey用户如何对泛型打桩》[https://www.jianshu.com/p/8a52eae7f786]

嗯,但是对于间接调用泛型方法的用例恐怕不可行

不过我似乎找到了解决办法,reflect拿到的方法地址,第一个调用将会是callsite调用的实际的泛型方法,那么这里找到这个第一个call指向的地址,就能正确的为泛型方法打桩,具体如下:

首先定义一个FindFirstCall

package gomonkey

import (
    "reflect"
    "unsafe"

    "golang.org/x/arch/x86/x86asm"
)

func FindFirstCall(ptr uintptr) uintptr {
    addr := ptr
    for addr < ptr+65535 {
        sliceHeader := reflect.SliceHeader{Data: addr, Len: 15, Cap: 15}
        code := *(*[]byte)(unsafe.Pointer(&sliceHeader))
        inst, err := x86asm.Decode(code, 64)
        if err != nil {
            return ptr
        }
        if inst.Op == x86asm.CALL {
            if v, ok := inst.Args[0].(x86asm.Rel); ok {
                return uintptr(int64(addr) + int64(inst.Len) + int64(v))
            }
        }
        addr += uintptr(inst.Len)
    }
    return ptr
}

然后ApplyMethod时根据类型名确定是否为泛型类型

func (this *Patches) ApplyMethod(target interface{}, methodName string, double interface{}) *Patches {
    rtype := castRType(target)
    m, ok := rtype.MethodByName(methodName)
    if !ok {
        panic("retrieve method by name failed")
    }
    d := reflect.ValueOf(double)
    generics := strings.ContainsRune(rtype.String(), '[')
    return this.ApplyCore(m.Func, d, generics)
}

最后ApplyCore时如果是泛型方法,那么需要找到实际的地址


func (this *Patches) ApplyCore(target, double reflect.Value, generics ...bool) *Patches {
    this.check(target, double)
    assTarget := *(*uintptr)(getPointer(target))
    if len(generics) > 0 && generics[0] {
        assTarget = FindFirstCall(assTarget)
    }
    original := replace(assTarget, uintptr(getPointer(double)))
    if _, ok := this.originals[assTarget]; !ok {
        this.originals[assTarget] = original
    }
    this.valueHolders[double] = double
    return this
}

这样是否可以?