使用GRPC时如何提供文件

时间:2019-10-25 21:35:11

标签: go grpc

有什么方法可以像使用gin-gonic's variant一样在GRPC Go中提供文件:

router.Static("/static", "/var/www")

1 个答案:

答案 0 :(得分:1)

您不能完全那样做。
但是,您可以使用原始bytes类型并将文件字节放入该字段中。

也(如注释中所指出)对于大文件,您应该使用流传输而不是一元调用。 (大多数GRPC实现的每条消息限制为4MB)。

原始示例:

syntax = "proto3";

message Response {
    bytes fileChunk = 1;
}
message Request {
    string fileName = 1;
}

service TestService {
    rpc Download(Request) returns (stream Response);
}

服务器实现示例:

func (srv *Server) Download(req *pbgo.Request, responseStream pbgo.TestService_DownloadServer) error {
    bufferSize := 64 *1024 //64KiB, tweak this as desired
    file, err := os.Open(req.GetFileName())
    if err != nil {
        fmt.Println(err)
        return err
    }
    defer file.Close()
    buff := make([]byte, bufferSize)
    for {
        bytesRead, err := file.Read(buff)
        if err != nil {
            if err != io.EOF {
                fmt.Println(err)
            }
            break
        }
        resp := &pbgo.Response{
            FileChunk: buff[:bytesRead],
        }
        err = responseStream.Send(resp)
        if err != nil {
            log.Println("error while sending chunk:", err)
            return err
        }
    }
    return nil
}

客户会这样称呼它:

conn, err := grpc.Dial("localhost:9090", grpc.WithInsecure())
if err != nil {
    log.Fatal("client could connect to grpc service:", err)
}
c := pbgo.NewTestServiceClient(conn)
fileStreamResponse, err := c.Download(context.TODO(), &pbgo.Request{
    FileName: "test.txt",
})
if err != nil {
    log.Println("error downloading:", err)
    return
}
for {
    chunkResponse, err := fileStreamResponse.Recv()
    if err == io.EOF {
        log.Println("received all chunks")
        break
    }
    if err != nil {
        log.Println("err receiving chunk:", err)
        break
    }
    log.Printf("got new chunk with data: %s \n", chunkResponse.FileChunk)
}

如果您需要能够提供任意文件,则需要处理允许提供哪些文件(例如,有人请求文件/etc/passwd或其他内容)。
不知道这里的用例是什么。