-
-
Save encryptblockr/43891700c751356899b6642e64027377 to your computer and use it in GitHub Desktop.
redis pub sub vs redis streams vs grpc streams
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
version: "3.7" | |
services: | |
redis: | |
image: "redis:6.0.6-alpine" | |
ports: | |
- "6379:6379" | |
nats: | |
image: "nats:2.1.9-alpine3.12" | |
ports: | |
- "4222:4222" | |
- "8222:8222" | |
nats-streaming: | |
image: "nats-streaming:0.19.0-alpine3.12" | |
command: --store=FILE --dir=/tmp/nats-streaming-store | |
ports: | |
- "4223:4222" | |
- "8223:8222" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
module redis-pubsub | |
go 1.14 | |
require ( | |
github.com/go-redis/redis/v8 v8.0.0-beta.7 | |
github.com/golang/protobuf v1.4.2 | |
github.com/nats-io/nats.go v1.10.0 | |
github.com/nats-io/stan.go v0.7.0 | |
google.golang.org/grpc v1.30.0 | |
google.golang.org/grpc/cmd/protoc-gen-go-grpc v0.0.0-20200812184716-7d8921505e1b // indirect | |
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | |
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | |
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | |
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | |
github.com/DataDog/sketches-go v0.0.0-20190923095040-43f19ad77ff7/go.mod h1:Q5DbzQ+3AkgGwymQO7aZFNP7ns2lZKGtvRBzRXfdi60= | |
github.com/benbjohnson/clock v1.0.3/go.mod h1:bGMdMPoPVvcYyt1gHDf4J2KE153Yf9BuiUKYMaxlTDM= | |
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | |
github.com/cespare/xxhash/v2 v2.1.1 h1:6MnRN8NT7+YBpUIWxHtefFZOKTAPgGjpQSxqLNn0+qY= | |
github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | |
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | |
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= | |
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | |
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | |
github.com/dgryski/go-rendezvous v0.0.0-20200624174652-8d2f3be8b2d9 h1:h2Ul3Ym2iVZWMQGYmulVUJ4LSkBm1erp9mUkPwtMoLg= | |
github.com/dgryski/go-rendezvous v0.0.0-20200624174652-8d2f3be8b2d9/go.mod h1:cuUVRXasLTGF7a8hSLbxyZXjz+1KgoB3wDUb6vlszIc= | |
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= | |
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= | |
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= | |
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | |
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= | |
github.com/go-redis/redis v6.15.9+incompatible h1:K0pv1D7EQUjfyoMql+r/jZqCLizCGKFlFgcHWWmHQjg= | |
github.com/go-redis/redis/v8 v8.0.0-beta.7 h1:4HiY+qfsyz8OUr9zyAP2T1CJ0SFRY4mKFvm9TEznuv8= | |
github.com/go-redis/redis/v8 v8.0.0-beta.7/go.mod h1:FGJAWDWFht1sQ4qxyJHZZbVyvnVcKQN0E3u5/5lRz+g= | |
github.com/gogo/protobuf v1.3.1 h1:DqDEcV5aeaTmdFBePNpYsp3FlcVH/2ISVVM9Qf8PSls= | |
github.com/gogo/protobuf v1.3.1/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o= | |
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | |
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | |
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | |
github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | |
github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= | |
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | |
github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | |
github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | |
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | |
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | |
github.com/golang/protobuf v1.4.2 h1:+Z5KGCizgyZCbGh1KZqA0fcLLkwbsjIzS4aV2v7wJX0= | |
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | |
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | |
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | |
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | |
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | |
github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | |
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | |
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | |
github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00= | |
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= | |
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | |
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | |
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | |
github.com/nats-io/jwt v0.3.2 h1:+RB5hMpXUUA2dfxuhBTEkMOrYmM+gKIZYS1KjSostMI= | |
github.com/nats-io/jwt v0.3.2/go.mod h1:/euKqTS1ZD+zzjYrY7pseZrTtWQSjujC7xjPc8wL6eU= | |
github.com/nats-io/nats.go v1.10.0 h1:L8qnKaofSfNFbXg0C5F71LdjPRnmQwSsA4ukmkt1TvY= | |
github.com/nats-io/nats.go v1.10.0/go.mod h1:AjGArbfyR50+afOUotNX2Xs5SYHf+CoOa5HH1eEl2HE= | |
github.com/nats-io/nkeys v0.1.3/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= | |
github.com/nats-io/nkeys v0.1.4 h1:aEsHIssIk6ETN5m2/MD8Y4B2X7FfXrBAUdkyRvbVYzA= | |
github.com/nats-io/nkeys v0.1.4/go.mod h1:XdZpAbhgyyODYqjTawOnIOI7VlbKSarI9Gfy1tqEu/s= | |
github.com/nats-io/nuid v1.0.1 h1:5iA8DT8V7q8WK2EScv2padNa/rTESc1KdnPw4TC2paw= | |
github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= | |
github.com/nats-io/stan.go v0.7.0 h1:sMVHD9RkxPOl6PJfDVBQd+gbxWkApeYl6GrH+10msO4= | |
github.com/nats-io/stan.go v0.7.0/go.mod h1:Ci6mUIpGQTjl++MqK2XzkWI/0vF+Bl72uScx7ejSYmU= | |
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | |
github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | |
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= | |
github.com/opentracing/opentracing-go v1.1.1-0.20190913142402-a7454ce5950e/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= | |
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | |
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | |
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | |
github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | |
go.opentelemetry.io/otel v0.7.0 h1:u43jukpwqR8EsyeJOMgrsUgZwVI1e1eVw7yuzRkD1l0= | |
go.opentelemetry.io/otel v0.7.0/go.mod h1:aZMyHG5TqDOXEgH2tyLiXSUKly1jT3yqE9PmrzIeCdo= | |
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | |
golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | |
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | |
golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59 h1:3zb4D3T4G8jdExgVU/95+vQXfpEPiMdCaZgmGVxjNHM= | |
golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | |
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | |
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= | |
golang.org/x/exp v0.0.0-20200513190911-00229845015e h1:rMqLP+9XLy+LdbCXHjJHAmTfXCr93W7oruWA6Hq1Alc= | |
golang.org/x/exp v0.0.0-20200513190911-00229845015e/go.mod h1:4M0jN8W1tt0AVLNr8HDosyJCDCDuyL9N9+3m7wDWgKw= | |
golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= | |
golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | |
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | |
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= | |
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= | |
golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= | |
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= | |
golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= | |
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | |
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | |
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | |
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | |
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | |
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | |
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | |
golang.org/x/net v0.0.0-20190923162816-aa69164e4478 h1:l5EDrHhldLYb3ZRHDUhXF7Om7MvYXnkV9/iQNo1lX6g= | |
golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | |
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | |
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | |
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | |
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | |
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | |
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | |
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | |
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | |
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | |
golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | |
golang.org/x/sys v0.0.0-20191010194322-b09406accb47 h1:/XfQ9z7ib8eEJX2hdgFTZJ/ntt0swNk5oYBziWeTCvY= | |
golang.org/x/sys v0.0.0-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | |
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | |
golang.org/x/text v0.3.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs= | |
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | |
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | |
golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | |
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | |
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= | |
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= | |
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= | |
golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= | |
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | |
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | |
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | |
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= | |
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | |
google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= | |
google.golang.org/genproto v0.0.0-20191009194640-548a555dbc03 h1:4HYDjxeNXAOTv3o1N2tjo8UUSlhQgAD52FVkwxnWgM8= | |
google.golang.org/genproto v0.0.0-20191009194640-548a555dbc03/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc= | |
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= | |
google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= | |
google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= | |
google.golang.org/grpc v1.30.0 h1:M5a8xTlYTxwMn5ZFkwhRabsygDY5G8TYLyQDBxJNAxE= | |
google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= | |
google.golang.org/grpc v1.31.0 h1:T7P4R73V3SSDPhH7WW7ATbfViLtmamH0DKrP3f9AuDI= | |
google.golang.org/grpc/cmd/protoc-gen-go-grpc v0.0.0-20200812184716-7d8921505e1b h1:0VfKveOcfZ8TYURzoY1yA9fVTURpTycZXoK3g0En0xs= | |
google.golang.org/grpc/cmd/protoc-gen-go-grpc v0.0.0-20200812184716-7d8921505e1b/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= | |
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | |
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | |
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | |
google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | |
google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | |
google.golang.org/protobuf v1.23.0 h1:4MY060fB1DLGMB/7MBTLnwQUY6+F09GEiz6SsrNqyzM= | |
google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | |
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | |
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | |
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | |
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | |
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | |
gopkg.in/yaml.v2 v2.2.7/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | |
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | |
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | |
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//go:generate protoc -I=. --go_out=. --go-grpc_out=. --go-grpc_opt=requireUnimplementedServers=false ./stream.proto | |
package main | |
import ( | |
"context" | |
"flag" | |
"log" | |
"net" | |
"os" | |
"runtime/pprof" | |
"time" | |
"github.com/go-redis/redis/v8" | |
nats "github.com/nats-io/nats.go" | |
stan "github.com/nats-io/stan.go" | |
"google.golang.org/grpc" | |
) | |
const KEY = "test-topic-name" | |
const MSG = "hello world" | |
type profiler = func(func()) | |
func main() { | |
flagN := flag.Int("n", 10, "iterations to run") | |
verbose := flag.Bool("v", false, "verbose mode") | |
mode := flag.String("mode", "pubsub", "mode of running: pubsub/stream/grpc/nats/nats-streaming") | |
cpuprofile := flag.String("cpuprofile", "", "write cpu profile to `file`") | |
flag.Parse() | |
ctx := context.Background() | |
var withProfile profiler | |
if *cpuprofile != "" { | |
f, err := os.Create(*cpuprofile) | |
if err != nil { | |
log.Fatal("could not create CPU profile: ", err) | |
} | |
defer f.Close() | |
withProfile = func(block func()) { | |
if err := pprof.StartCPUProfile(f); err != nil { | |
log.Panic("could not start CPU profile: ", err) | |
} | |
defer pprof.StopCPUProfile() | |
block() | |
} | |
} else { | |
withProfile = func(block func()) { | |
block() | |
} | |
} | |
switch *mode { | |
case "pubsub": | |
runPubSub(ctx, redisClient(), *flagN, *verbose, withProfile) | |
case "stream": | |
runStream(ctx, redisClient(), *flagN, *verbose, withProfile) | |
case "grpc": | |
runGrpc(ctx, *flagN, *verbose, withProfile) | |
case "nats": | |
runNats(ctx, *flagN, *verbose, withProfile) | |
case "nats-streaming": | |
runNatsStreaming(ctx, *flagN, *verbose, withProfile) | |
default: | |
log.Println("Unsupported mode:", *mode) | |
} | |
} | |
func redisClient() *redis.Client { | |
return redis.NewClient(&redis.Options{ | |
Addr: "localhost:6379", | |
Password: "", // no password set | |
DB: 0, // use default DB | |
}) | |
} | |
func runPubSub(ctx context.Context, client *redis.Client, n int, verbose bool, withProfile profiler) { | |
pubSub := client.Subscribe(ctx, KEY) | |
ch := pubSub.Channel() | |
pong, err := client.Ping(ctx).Result() | |
log.Println(pong, err) | |
totalDuration := time.Duration(0) | |
withProfile(func() { | |
for i := 0; i < n; i++ { | |
err := client.Publish(ctx, KEY, MSG).Err() | |
if err != nil { | |
log.Panic(err) | |
} | |
start := time.Now() | |
<-ch | |
took := time.Since(start) | |
if verbose { | |
log.Println(took) | |
} | |
totalDuration += took | |
} | |
}) | |
log.Println("Avg:", totalDuration/time.Duration(n)) | |
} | |
func runStream(ctx context.Context, client *redis.Client, n int, verbose bool, withProfile profiler) { | |
received := make(chan string, 10) | |
go func() { | |
lastId := "$" | |
for { | |
cmd := client.XRead(ctx, &redis.XReadArgs{ | |
Streams: []string{KEY, lastId}, | |
Count: 1, | |
Block: 100 * time.Millisecond, | |
}) | |
res, err := cmd.Result() | |
if err != nil { | |
log.Panic(err) | |
} | |
msg := res[0].Messages[0] | |
received <- msg.Values[KEY].(string) | |
lastId = msg.ID | |
} | |
}() | |
time.Sleep(10 * time.Millisecond) | |
totalDuration := time.Duration(0) | |
withProfile(func() { | |
for i := 0; i < n; i++ { | |
args := &redis.XAddArgs{ | |
ID: "*", | |
Stream: KEY, | |
MaxLenApprox: 10, | |
Values: map[string]interface{}{KEY: MSG}, | |
} | |
start := time.Now() | |
err := client.XAdd(ctx, args).Err() | |
if err != nil { | |
log.Panic(err) | |
} | |
<-received | |
took := time.Since(start) | |
if verbose { | |
log.Println(took) | |
} | |
totalDuration += took | |
} | |
}) | |
log.Println("Avg:", totalDuration/time.Duration(n)) | |
} | |
func runGrpc(ctx context.Context, n int, verbose bool, withProfile profiler) { | |
lis, err := net.Listen("tcp", "127.0.0.1:50051") | |
if err != nil { | |
log.Fatalf("failed to listen: %v", err) | |
} | |
s := grpc.NewServer() | |
streamer := newStreamer() | |
RegisterStreamServer(s, streamer) | |
go func() { | |
err := s.Serve(lis) | |
if err != nil { | |
log.Fatalf("failed to serve: %v", err) | |
} | |
}() | |
time.Sleep(100 * time.Millisecond) | |
conn, err := grpc.Dial("127.0.0.1:50051", grpc.WithInsecure(), grpc.WithBlock()) | |
if err != nil { | |
log.Fatalf("did not connect: %v", err) | |
} | |
defer conn.Close() | |
client := NewStreamClient(conn) | |
streamClient, err := client.Read(ctx, &ReadRequest{}) | |
if err != nil { | |
log.Panic(err) | |
} | |
stream := make(chan *Entry, 1) | |
go func() { | |
for { | |
entry, err := streamClient.Recv() | |
if err != nil { | |
log.Panic(err) | |
} | |
stream <- entry | |
} | |
}() | |
totalDuration := time.Duration(0) | |
withProfile(func() { | |
for i := 0; i < n; i++ { | |
streamer.publish(MSG) | |
start := time.Now() | |
<-stream | |
took := time.Since(start) | |
if verbose { | |
log.Println(took) | |
} | |
totalDuration += took | |
} | |
}) | |
log.Println("Avg:", totalDuration/time.Duration(n)) | |
} | |
type streamer struct { | |
entries chan *Entry | |
} | |
func newStreamer() *streamer { | |
return &streamer{make(chan *Entry, 10)} | |
} | |
func (s *streamer) publish(msg string) { | |
s.entries <- &Entry{Msg: msg} | |
} | |
func (s *streamer) Read(req *ReadRequest, stream Stream_ReadServer) error { | |
for entry := range s.entries { | |
stream.Send(entry) | |
} | |
return nil | |
} | |
var _ StreamServer = &streamer{} | |
func runNats(ctx context.Context, n int, verbose bool, withProfile profiler) { | |
nc, err := nats.Connect(nats.DefaultURL) | |
if err != nil { | |
log.Panic(err) | |
} | |
defer nc.Close() | |
ch := make(chan *nats.Msg, 1) | |
sub, err := nc.ChanSubscribe(KEY, ch) | |
if err != nil { | |
log.Panic(err) | |
} | |
defer sub.Drain() | |
msg := []byte(MSG) | |
totalDuration := time.Duration(0) | |
withProfile(func() { | |
for i := 0; i < n; i++ { | |
err := nc.Publish(KEY, msg) | |
if err != nil { | |
log.Panic(err) | |
} | |
start := time.Now() | |
<-ch | |
took := time.Since(start) | |
if verbose { | |
log.Println(took) | |
} | |
totalDuration += took | |
} | |
}) | |
log.Println("Avg:", totalDuration/time.Duration(n)) | |
} | |
func runNatsStreaming(ctx context.Context, n int, verbose bool, withProfile profiler) { | |
sc, err := stan.Connect("test-cluster", "-", stan.NatsURL("nats://localhost:4223")) | |
if err != nil { | |
log.Panic(err) | |
} | |
defer sc.Close() | |
ch := make(chan *stan.Msg, 1) | |
sub, err := sc.Subscribe(KEY, func(m *stan.Msg) { | |
ch <- m | |
}) | |
if err != nil { | |
log.Panic(err) | |
} | |
defer sub.Close() | |
msg := []byte(MSG) | |
totalDuration := time.Duration(0) | |
withProfile(func() { | |
for i := 0; i < n; i++ { | |
err := sc.Publish(KEY, msg) | |
if err != nil { | |
log.Panic(err) | |
} | |
start := time.Now() | |
<-ch | |
took := time.Since(start) | |
if verbose { | |
log.Println(took) | |
} | |
totalDuration += took | |
} | |
}) | |
log.Println("Avg:", totalDuration/time.Duration(n)) | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Code generated by protoc-gen-go. DO NOT EDIT. | |
// source: stream.proto | |
package main // import "." | |
import proto "github.com/golang/protobuf/proto" | |
import fmt "fmt" | |
import math "math" | |
// Reference imports to suppress errors if they are not otherwise used. | |
var _ = proto.Marshal | |
var _ = fmt.Errorf | |
var _ = math.Inf | |
// This is a compile-time assertion to ensure that this generated file | |
// is compatible with the proto package it is being compiled against. | |
// A compilation error at this line likely means your copy of the | |
// proto package needs to be updated. | |
const _ = proto.ProtoPackageIsVersion2 // please upgrade the proto package | |
type ReadRequest struct { | |
LastId int64 `protobuf:"varint,1,opt,name=last_id,json=lastId,proto3" json:"last_id,omitempty"` | |
XXX_NoUnkeyedLiteral struct{} `json:"-"` | |
XXX_unrecognized []byte `json:"-"` | |
XXX_sizecache int32 `json:"-"` | |
} | |
func (m *ReadRequest) Reset() { *m = ReadRequest{} } | |
func (m *ReadRequest) String() string { return proto.CompactTextString(m) } | |
func (*ReadRequest) ProtoMessage() {} | |
func (*ReadRequest) Descriptor() ([]byte, []int) { | |
return fileDescriptor_stream_987174f751fe2b93, []int{0} | |
} | |
func (m *ReadRequest) XXX_Unmarshal(b []byte) error { | |
return xxx_messageInfo_ReadRequest.Unmarshal(m, b) | |
} | |
func (m *ReadRequest) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { | |
return xxx_messageInfo_ReadRequest.Marshal(b, m, deterministic) | |
} | |
func (dst *ReadRequest) XXX_Merge(src proto.Message) { | |
xxx_messageInfo_ReadRequest.Merge(dst, src) | |
} | |
func (m *ReadRequest) XXX_Size() int { | |
return xxx_messageInfo_ReadRequest.Size(m) | |
} | |
func (m *ReadRequest) XXX_DiscardUnknown() { | |
xxx_messageInfo_ReadRequest.DiscardUnknown(m) | |
} | |
var xxx_messageInfo_ReadRequest proto.InternalMessageInfo | |
func (m *ReadRequest) GetLastId() int64 { | |
if m != nil { | |
return m.LastId | |
} | |
return 0 | |
} | |
type Entry struct { | |
Id int64 `protobuf:"varint,1,opt,name=id,proto3" json:"id,omitempty"` | |
Msg string `protobuf:"bytes,2,opt,name=msg,proto3" json:"msg,omitempty"` | |
XXX_NoUnkeyedLiteral struct{} `json:"-"` | |
XXX_unrecognized []byte `json:"-"` | |
XXX_sizecache int32 `json:"-"` | |
} | |
func (m *Entry) Reset() { *m = Entry{} } | |
func (m *Entry) String() string { return proto.CompactTextString(m) } | |
func (*Entry) ProtoMessage() {} | |
func (*Entry) Descriptor() ([]byte, []int) { | |
return fileDescriptor_stream_987174f751fe2b93, []int{1} | |
} | |
func (m *Entry) XXX_Unmarshal(b []byte) error { | |
return xxx_messageInfo_Entry.Unmarshal(m, b) | |
} | |
func (m *Entry) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { | |
return xxx_messageInfo_Entry.Marshal(b, m, deterministic) | |
} | |
func (dst *Entry) XXX_Merge(src proto.Message) { | |
xxx_messageInfo_Entry.Merge(dst, src) | |
} | |
func (m *Entry) XXX_Size() int { | |
return xxx_messageInfo_Entry.Size(m) | |
} | |
func (m *Entry) XXX_DiscardUnknown() { | |
xxx_messageInfo_Entry.DiscardUnknown(m) | |
} | |
var xxx_messageInfo_Entry proto.InternalMessageInfo | |
func (m *Entry) GetId() int64 { | |
if m != nil { | |
return m.Id | |
} | |
return 0 | |
} | |
func (m *Entry) GetMsg() string { | |
if m != nil { | |
return m.Msg | |
} | |
return "" | |
} | |
func init() { | |
proto.RegisterType((*ReadRequest)(nil), "main.ReadRequest") | |
proto.RegisterType((*Entry)(nil), "main.Entry") | |
} | |
func init() { proto.RegisterFile("stream.proto", fileDescriptor_stream_987174f751fe2b93) } | |
var fileDescriptor_stream_987174f751fe2b93 = []byte{ | |
// 160 bytes of a gzipped FileDescriptorProto | |
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xe2, 0xe2, 0x29, 0x2e, 0x29, 0x4a, | |
0x4d, 0xcc, 0xd5, 0x2b, 0x28, 0xca, 0x2f, 0xc9, 0x17, 0x62, 0xc9, 0x4d, 0xcc, 0xcc, 0x53, 0x52, | |
0xe3, 0xe2, 0x0e, 0x4a, 0x4d, 0x4c, 0x09, 0x4a, 0x2d, 0x2c, 0x4d, 0x2d, 0x2e, 0x11, 0x12, 0xe7, | |
0x62, 0xcf, 0x49, 0x2c, 0x2e, 0x89, 0xcf, 0x4c, 0x91, 0x60, 0x54, 0x60, 0xd4, 0x60, 0x0e, 0x62, | |
0x03, 0x71, 0x3d, 0x53, 0x94, 0x34, 0xb9, 0x58, 0x5d, 0xf3, 0x4a, 0x8a, 0x2a, 0x85, 0xf8, 0xb8, | |
0x98, 0xe0, 0x92, 0x4c, 0x99, 0x29, 0x42, 0x02, 0x5c, 0xcc, 0xb9, 0xc5, 0xe9, 0x12, 0x4c, 0x0a, | |
0x8c, 0x1a, 0x9c, 0x41, 0x20, 0xa6, 0x91, 0x09, 0x17, 0x5b, 0x30, 0xd8, 0x22, 0x21, 0x2d, 0x2e, | |
0x16, 0x90, 0xe1, 0x42, 0x82, 0x7a, 0x20, 0xbb, 0xf4, 0x90, 0x2c, 0x92, 0xe2, 0x86, 0x08, 0x81, | |
0xcd, 0x54, 0x62, 0x30, 0x60, 0x74, 0xe2, 0x88, 0x62, 0xd3, 0xb3, 0x06, 0x89, 0x25, 0xb1, 0x81, | |
0xdd, 0x67, 0x0c, 0x08, 0x00, 0x00, 0xff, 0xff, 0x44, 0xbb, 0xf3, 0x59, 0xaf, 0x00, 0x00, 0x00, | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
syntax = "proto3"; | |
package main; | |
option go_package = ".;main"; | |
service Stream { | |
rpc Read (ReadRequest) returns (stream Entry) {} | |
} | |
message ReadRequest { | |
int64 last_id = 1; | |
} | |
message Entry { | |
int64 id = 1; | |
string msg = 2; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// Code generated by protoc-gen-go-grpc. DO NOT EDIT. | |
package main | |
import ( | |
context "context" | |
grpc "google.golang.org/grpc" | |
codes "google.golang.org/grpc/codes" | |
status "google.golang.org/grpc/status" | |
) | |
// This is a compile-time assertion to ensure that this generated file | |
// is compatible with the grpc package it is being compiled against. | |
const _ = grpc.SupportPackageIsVersion6 | |
// StreamClient is the client API for Stream service. | |
// | |
// For semantics around ctx use and closing/ending streaming RPCs, please refer to https://pkg.go.dev/google.golang.org/grpc/?tab=doc#ClientConn.NewStream. | |
type StreamClient interface { | |
Read(ctx context.Context, in *ReadRequest, opts ...grpc.CallOption) (Stream_ReadClient, error) | |
} | |
type streamClient struct { | |
cc grpc.ClientConnInterface | |
} | |
func NewStreamClient(cc grpc.ClientConnInterface) StreamClient { | |
return &streamClient{cc} | |
} | |
func (c *streamClient) Read(ctx context.Context, in *ReadRequest, opts ...grpc.CallOption) (Stream_ReadClient, error) { | |
stream, err := c.cc.NewStream(ctx, &_Stream_serviceDesc.Streams[0], "/main.Stream/Read", opts...) | |
if err != nil { | |
return nil, err | |
} | |
x := &streamReadClient{stream} | |
if err := x.ClientStream.SendMsg(in); err != nil { | |
return nil, err | |
} | |
if err := x.ClientStream.CloseSend(); err != nil { | |
return nil, err | |
} | |
return x, nil | |
} | |
type Stream_ReadClient interface { | |
Recv() (*Entry, error) | |
grpc.ClientStream | |
} | |
type streamReadClient struct { | |
grpc.ClientStream | |
} | |
func (x *streamReadClient) Recv() (*Entry, error) { | |
m := new(Entry) | |
if err := x.ClientStream.RecvMsg(m); err != nil { | |
return nil, err | |
} | |
return m, nil | |
} | |
// StreamServer is the server API for Stream service. | |
// All implementations should embed UnimplementedStreamServer | |
// for forward compatibility | |
type StreamServer interface { | |
Read(*ReadRequest, Stream_ReadServer) error | |
} | |
// UnimplementedStreamServer should be embedded to have forward compatible implementations. | |
type UnimplementedStreamServer struct { | |
} | |
func (*UnimplementedStreamServer) Read(*ReadRequest, Stream_ReadServer) error { | |
return status.Errorf(codes.Unimplemented, "method Read not implemented") | |
} | |
func RegisterStreamServer(s *grpc.Server, srv StreamServer) { | |
s.RegisterService(&_Stream_serviceDesc, srv) | |
} | |
func _Stream_Read_Handler(srv interface{}, stream grpc.ServerStream) error { | |
m := new(ReadRequest) | |
if err := stream.RecvMsg(m); err != nil { | |
return err | |
} | |
return srv.(StreamServer).Read(m, &streamReadServer{stream}) | |
} | |
type Stream_ReadServer interface { | |
Send(*Entry) error | |
grpc.ServerStream | |
} | |
type streamReadServer struct { | |
grpc.ServerStream | |
} | |
func (x *streamReadServer) Send(m *Entry) error { | |
return x.ServerStream.SendMsg(m) | |
} | |
var _Stream_serviceDesc = grpc.ServiceDesc{ | |
ServiceName: "main.Stream", | |
HandlerType: (*StreamServer)(nil), | |
Methods: []grpc.MethodDesc{}, | |
Streams: []grpc.StreamDesc{ | |
{ | |
StreamName: "Read", | |
Handler: _Stream_Read_Handler, | |
ServerStreams: true, | |
}, | |
}, | |
Metadata: "stream.proto", | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment