有没有什么方法可以使用 GRPC 在 Go 中提供文件,就像gin-gonic 的变体一样:
router.Static("/static", "/var/www")
你不能完全像那样做。
但是您可以使用原型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或其他内容)。
不确定这里的用例到底是什么。