Compare commits
	
		
			337 Commits
		
	
	
		
			show
			...
			v3.0.0-alp
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 563768b58a | ||
|  | 9dfeb98111 | ||
|  | 5f4491bb86 | ||
|  | fbdf1f2c1c | ||
|  | a3a7434f2c | ||
|  | 592179c0a2 | ||
|  | 9b74bc52d6 | ||
|  | 05f3e1a125 | ||
|  | 16c591d741 | ||
|  | 755b816086 | ||
|  | 7aa92fa8b5 | ||
|  | 5077683b70 | ||
|  | 7f6cefd9c9 | ||
|  | 647ce61dec | ||
|  | d3326efd4b | ||
|  | 6920677f1e | ||
|  | 1838e4a1ee | ||
|  | 96233b2d9b | ||
|  | e082ac42a0 | ||
|  | d7ef224447 | ||
|  | 8c6f4062ef | ||
|  | 0d860c53a6 | ||
| bcc890e47c | |||
|  | f9bf562393 | ||
| dfa50a888d | |||
|  | e63b9015ae | ||
| 3627e47f04 | |||
|  | 7d41c2224e | ||
|  | 68927e875b | ||
|  | 0c19a87c89 | ||
|  | f73ec65ac3 | ||
|  | b27e71ae64 | ||
|  | 0299517f0d | ||
|  | e1404a1100 | ||
|  | 057d61063f | ||
|  | 73a3f596e8 | ||
|  | 0287ab8751 | ||
|  | 42c28f2b6d | ||
|  | a2bb0bea2d | ||
|  | 9f9c748f9b | ||
|  | a5e9dc21ca | ||
|  | 3f4b58b58c | ||
|  | 0a79db498c | ||
|  | 7c5e3b0f30 | ||
|  | 07fbb06ed8 | ||
|  | a4252ba69c | ||
|  | 8fe4f1f2c3 | ||
|  | 2e04fcd718 | ||
|  | 7355455020 | ||
|  | 040577fb74 | ||
|  | 4e7621da18 | ||
|  | 8e30ede8c7 | ||
|  | 630ceb5dad | ||
|  | 85ae232936 | ||
|  | 13ea0eec02 | ||
|  | 09ec20fded | ||
|  | 3480e0a64e | ||
|  | 318a80f824 | ||
|  | 6d9a38a747 | ||
|  | 58d6726380 | ||
|  | e5db6ea8a7 | ||
|  | 3468331506 | ||
|  | 1bac08cc0e | ||
|  | 333320dcb8 | ||
|  | ce12c040fa | ||
|  | ee36e26edc | ||
|  | 3ffb899951 | ||
|  | 00bd2bc65f | ||
|  | 86f4235aaf | ||
|  | b37f9c94b8 | ||
|  | 0ed1c70d29 | ||
|  | db8e10834b | ||
|  | 0a937745cd | ||
|  | f5ed7e5833 | ||
|  | 859b9e7786 | ||
|  | 2b033b6495 | ||
|  | 51caf2a24e | ||
|  | eaa46c2de7 | ||
|  | 90dca65f55 | ||
| 97ae2979ad | |||
|  | 6f309dada3 | ||
|  | f99b436ec2 | ||
|  | c817f29d6e | ||
|  | f744c6248f | ||
|  | 4ff114e798 | ||
|  | c58ac35dfc | ||
|  | b5314829fa | ||
|  | 41c7688697 | ||
|  | b021546c09 | ||
|  | 6898a65508 | ||
|  | d577dd6abe | ||
|  | 3c633e3577 | ||
|  | 174e44b846 | ||
|  | a63480a81a | ||
|  | 6d9d94b105 | ||
|  | 64e9185386 | ||
|  | 1b5c83f3cc | ||
|  | 979af853b9 | ||
|  | a64078b5c3 | ||
|  | 58845d7012 | ||
|  | dcf01ebbf0 | ||
|  | 355ad2a1af | ||
|  | b882ff3df9 | ||
|  | 6337c92cd0 | ||
|  | a95accad56 | ||
|  | 6532b6208b | ||
|  | 0f5c53b6e4 | ||
|  | deea8fecf4 | ||
|  | df3e5364ca | ||
|  | 132c1e35fe | ||
|  | 5967a68e78 | ||
|  | 104b7d8f8d | ||
|  | 4f0f4326df | ||
|  | ee02511658 | ||
|  | a8fc5590a8 | ||
|  | bc60f23ff6 | ||
|  | 2000da6fd8 | ||
|  | 5ab475636a | ||
|  | 51b4ab0abc | ||
|  | 687a5e2e58 | ||
|  | fcd307d902 | ||
|  | 00cd07a3a6 | ||
|  | a2a1f4dfbd | ||
|  | 2b506b1a2a | ||
|  | a2550820d3 | ||
|  | c940961574 | ||
|  | 695cc9d526 | ||
|  | 87543b2c8a | ||
|  | 5f9c3a6efd | ||
|  | 2b889087bd | ||
|  | ece02a6d21 | ||
|  | 58c6bbbf6b | ||
|  | c16f4b741c | ||
|  | 83cecdb294 | ||
|  | 8c7c27c573 | ||
|  | 5fd36d6cc0 | ||
|  | 3b40fde68b | ||
|  | 9d3365c4be | ||
|  | 2efb459c66 | ||
|  | 6add74b4f6 | ||
|  | c67d78f1ef | ||
|  | a89610ffea | ||
|  | da9bb11240 | ||
|  | a3a1a84172 | ||
|  | 1179d7e89a | ||
|  | a5df913926 | ||
|  | 9ce706191b | ||
|  | 0327f30e3c | ||
|  | 0ce132eb8f | ||
|  | 00b76e0a64 | ||
|  | aec27be9b4 | ||
|  | 86dfcb819b | ||
|  | d613804b0a | ||
| 92e9d05432 | |||
|  | 8dfd93e915 | ||
|  | e5136332e3 | ||
|  | f10fd4b479 | ||
|  | 74368026a5 | ||
|  | fde1aa9d6a | ||
|  | f45cdba9ba | ||
|  | 73c2f25935 | ||
|  | b270860b79 | ||
|  | 8e81cea96f | ||
|  | cdd8f9fd82 | ||
|  | e7ba930236 | ||
|  | a346064eaf | ||
|  | 47bdd5c993 | ||
|  | 9af12ff9df | ||
| 6c7bcf3883 | |||
|  | bbc3b7040b | ||
|  | 582f2e8b94 | ||
|  | bd3ef67328 | ||
|  | 1ccd4cd940 | ||
|  | aa679f7a73 | ||
|  | 003731ace9 | ||
|  | 7b379bf1f1 | ||
|  | b6f3e8b715 | ||
|  | 8f6ec21b91 | ||
|  | e4e56b0f3f | ||
|  | 219d29f664 | ||
|  | 8fb138af06 | ||
|  | a39e6515da | ||
|  | 2c7fd286de | ||
|  | 8aa2712b4d | ||
|  | b5c2121cef | ||
|  | ca9b877646 | ||
|  | ff49b4fc71 | ||
|  | 222431b57a | ||
|  | ddb51529a7 | ||
|  | 7c048f331a | ||
|  | 8475183bbb | ||
|  | 10f35db3ed | ||
|  | b68af8ab63 | ||
|  | 266602a3d6 | ||
|  | 15d5142d9b | ||
|  | 0d88650511 | ||
|  | 8660370dc9 | ||
|  | 73339dde85 | ||
|  | 3f354f3c30 | ||
|  | c08eb5f892 | ||
|  | 27e41c4ad5 | ||
|  | 1da8a640da | ||
|  | e7ad031eb8 | ||
|  | 192f548c83 | ||
|  | d85b4197b4 | ||
|  | bb5f2e5525 | ||
|  | f00b696282 | ||
|  | e2d662608c | ||
|  | 9e9773c9c7 | ||
|  | 2f8e2487f7 | ||
|  | d6c1fbf841 | ||
|  | c3b404bab0 | ||
|  | cd283654eb | ||
|  | 5712cc9c62 | ||
|  | be5a10a4d4 | ||
|  | b53a2c67f1 | ||
|  | cc79692d68 | ||
|  | 796a598b37 | ||
|  | 73b4423682 | ||
|  | 198e942889 | ||
|  | 95703e4565 | ||
|  | 2729569f66 | ||
|  | 67146ecdc2 | ||
|  | bd049a51e6 | ||
|  | ffd89599a0 | ||
|  | 496293afa1 | ||
|  | 7d7f4046e8 | ||
|  | c800070477 | ||
|  | 877fe5fb0a | ||
|  | dad011cab4 | ||
|  | f939200b34 | ||
|  | 9c072a372c | ||
|  | fbb91c6cb7 | ||
|  | b2cf501952 | ||
|  | 1fce0f02b6 | ||
|  | 12061bd006 | ||
|  | 856c73b341 | ||
|  | 4de19805ba | ||
|  | c09b871a6b | ||
|  | e876cb917d | ||
|  | 8f5ef012ff | ||
|  | 287992cef3 | ||
|  | 344ce061ce | ||
|  | 5d14970a55 | ||
|  | 0615fe825f | ||
|  | 6a661fd08c | ||
|  | f6d9416a9e | ||
|  | a29676b86a | ||
|  | dc10f88c12 | ||
|  | e61edf6280 | ||
|  | 3410a0949b | ||
|  | 9216a47724 | ||
|  | cf37d64819 | ||
|  | f0c0f3d4c4 | ||
|  | c4e3f8c336 | ||
|  | 8875719619 | ||
|  | c19b349e96 | ||
|  | 14155c7e02 | ||
|  | 3d36398818 | ||
|  | 56a7897c91 | ||
|  | 5efb386224 | ||
|  | 4d2de923cd | ||
|  | f64b1468a5 | ||
|  | 56f281002b | ||
|  | 0d7250352f | ||
|  | ef43f01da4 | ||
|  | c9e5ae6a2b | ||
|  | 8a802d8f7a | ||
|  | 331ab3715c | ||
|  | 6b451a2197 | ||
|  | b4c0224746 | ||
|  | 500d793fc4 | ||
|  | 16af265e8b | ||
|  | b222cf8e13 | ||
|  | f549e20fa2 | ||
|  | 83e9c1fad2 | ||
|  | c220686c29 | ||
|  | 1b18730d54 | ||
|  | 5764519f5b | ||
|  | 957001f8ad | ||
|  | 0955671e45 | ||
|  | 57b060bac5 | ||
|  | 3136e1409e | ||
|  | ca791d7e8d | ||
|  | 05858b746c | ||
|  | 09d1450d7d | ||
|  | 1ca1fd411a | ||
|  | a2d4d62f1c | ||
|  | 8ab20f501c | ||
|  | 366fb228e5 | ||
|  | bba8c254d7 | ||
|  | ebd53794af | ||
|  | 2299244332 | ||
|  | cf61d98635 | ||
|  | 15d1967aaf | ||
|  | 410fec8ee4 | ||
|  | c831b6c03a | ||
|  | 290595f88e | ||
|  | ba64518ebd | ||
|  | b14d63b4a1 | ||
|  | af2db0a0d9 | ||
|  | fb255a7e5a | ||
|  | 47c1cb433e | ||
|  | 3fac7d79ab | ||
|  | 25c937fd0e | ||
|  | e5c1fbc591 | ||
|  | d781c9ae2d | ||
|  | 54951740bf | ||
|  | 0fb4734e67 | ||
|  | 346e034d0a | ||
|  | 116cc1e9ee | ||
|  | 762a5bc9e8 | ||
|  | d39b723511 | ||
|  | 5494e935f4 | ||
|  | e0863bb7eb | ||
|  | 89f86167ad | ||
|  | dfec1ad6b1 | ||
|  | 66d3e4a595 | ||
|  | 19a03babc4 | ||
|  | ee24b4f083 | ||
|  | 937ecc8d2f | ||
|  | 6078adb8bc | ||
|  | 39f18b0b70 | ||
|  | efb64b7dbb | ||
|  | f892b41299 | ||
|  | 1eb63635b5 | ||
|  | 688228377b | ||
|  | 506006f0fa | ||
|  | 22de001a80 | ||
|  | d90cc8bf2f | ||
|  | 5a8f19589b | ||
|  | d61df6363b | ||
|  | f062013a7b | ||
|  | fea93a5b7a | ||
|  | 30dc29e17f | ||
|  | 5387f73b5d | ||
| 90dd1f63c8 | 
							
								
								
									
										2
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +1,3 @@ | ||||
| # These are supported funding model platforms | ||||
|  | ||||
| github: asim | ||||
| github: micro | ||||
|   | ||||
							
								
								
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| ## Pull Request template | ||||
| Please, go through these steps before clicking submit on this PR. | ||||
|  | ||||
| 1. Give a descriptive title to your PR. | ||||
| 2. Provide a description of your changes. | ||||
| 3. Make sure you have some relevant tests. | ||||
| 4. Put `closes #XXXX` in your comment to auto-close the issue that your PR fixes (if applicable). | ||||
|  | ||||
| **PLEASE REMOVE THIS TEMPLATE BEFORE SUBMITTING** | ||||
							
								
								
									
										1
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/docker.yml
									
									
									
									
										vendored
									
									
								
							| @@ -19,3 +19,4 @@ jobs: | ||||
|            name: micro/go-micro | ||||
|            username: ${{ secrets.DOCKER_USERNAME }} | ||||
|            password: ${{ secrets.DOCKER_PASSWORD }} | ||||
|            tag_names: true | ||||
							
								
								
									
										41
									
								
								.github/workflows/scripts/build-all-examples.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/scripts/build-all-examples.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| #!/bin/bash | ||||
| # set -x | ||||
|  | ||||
| function build_binary { | ||||
|     echo building $1 | ||||
|     pushd $1 | ||||
|     go build -o _main | ||||
|     local ret=$? | ||||
|     if [ $ret -gt 0 ]; then  | ||||
|         failed=1 | ||||
|         failed_arr+=($1) | ||||
|     fi | ||||
|     popd | ||||
| } | ||||
|  | ||||
| function is_main { | ||||
|     grep "package main" -l -dskip $1/*.go > /dev/null 2>&1 | ||||
| } | ||||
|  | ||||
|  | ||||
| function check_dir { | ||||
|     is_main $1 | ||||
|     local ret=$? | ||||
|     if [ $ret == 0 ]; then | ||||
|         build_binary $1 $2 | ||||
|     fi | ||||
|     for filename in $1/*; do | ||||
|         if [ -d $filename ]; then | ||||
|             check_dir $filename $2 | ||||
|         fi | ||||
|     done | ||||
| } | ||||
| failed_arr=() | ||||
| failed=0 | ||||
| go mod edit -replace github.com/micro/go-micro/v2=github.com/$2/v2@$1  | ||||
| check_dir . $1 | ||||
| if [ $failed -gt 0 ]; then | ||||
|     echo Some builds failed | ||||
|     printf '%s\n' "${failed_arr[@]}" | ||||
| fi | ||||
| exit $failed | ||||
							
								
								
									
										19
									
								
								.github/workflows/scripts/build-micro.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/scripts/build-micro.sh
									
									
									
									
										vendored
									
									
										Executable file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| #!/bin/bash | ||||
| # set -x | ||||
|  | ||||
| failed=0 | ||||
| go mod edit -replace github.com/micro/go-micro/v2=github.com/$2/v2@$1  | ||||
| # basic test, build the binary | ||||
| go install | ||||
| failed=$? | ||||
| if [ $failed -gt 0 ]; then | ||||
|     exit $failed | ||||
| fi | ||||
| # unit tests | ||||
| IN_TRAVIS_CI=yes go test -v ./... | ||||
|  | ||||
| ./scripts/test-docker.sh | ||||
| # Generate keys for JWT tests | ||||
| ssh-keygen -f /tmp/sshkey -m pkcs8 -q -N "" | ||||
| ssh-keygen -f /tmp/sshkey -e  -m pkcs8 > /tmp/sshkey.pub | ||||
| go clean -testcache && IN_TRAVIS_CI=yes go test --tags=integration -v ./test | ||||
							
								
								
									
										37
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,23 +1,27 @@ | ||||
| # Go Micro [](https://opensource.org/licenses/Apache-2.0) [](https://pkg.go.dev/github.com/micro/go-micro?tab=doc) [](https://travis-ci.org/micro/go-micro) [](https://goreportcard.com/report/github.com/micro/go-micro) | ||||
| # Go Micro [](https://opensource.org/licenses/Apache-2.0) [](https://pkg.go.dev/github.com/micro/go-micro?tab=doc) [](https://travis-ci.org/micro/go-micro) [](https://goreportcard.com/report/github.com/micro/go-micro)  | ||||
|  | ||||
| Go Micro is a framework for distributed systems development. | ||||
| Go Micro is a standard library for distributed systems development. | ||||
|  | ||||
| ## Overview | ||||
|  | ||||
| Go Micro provides the core requirements for distributed systems development including RPC and Event driven communication.  | ||||
| The **micro** philosophy is sane defaults with a pluggable architecture. We provide defaults to get you started quickly  | ||||
| The **Micro** philosophy is sane defaults with a pluggable architecture. We provide defaults to get you started quickly  | ||||
| but everything can be easily swapped out.  | ||||
|  | ||||
| <img src="https://micro.mu/docs/images/go-micro.svg" /> | ||||
|  | ||||
| Plugins are available at [github.com/micro/go-plugins](https://github.com/micro/go-plugins). | ||||
|  | ||||
| Follow us on [Twitter](https://twitter.com/microhq) or join the [Community](https://micro.mu/slack). | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| Go Micro abstracts away the details of distributed systems. Here are the main features. | ||||
|  | ||||
| - **Authentication** - Auth is built in as a first class citizen. Authentication and authorization enable secure  | ||||
| zero trust networking by providing every service an identity and certificates. This additionally includes rule  | ||||
| based access control. | ||||
|  | ||||
| - **Dynamic Config** - Load and hot reload dynamic config from anywhere. The config interface provides a way to load application  | ||||
| level config from any source such as env vars, file, etcd. You can merge the sources and even define fallbacks. | ||||
|  | ||||
| - **Data Storage** - A simple data store interface to read, write and delete records. It includes support for memory, file and  | ||||
| CockroachDB by default. State and persistence becomes a core requirement beyond prototyping and Micro looks to build that into the framework. | ||||
|  | ||||
| - **Service Discovery** - Automatic service registration and name resolution. Service discovery is at the core of micro service  | ||||
| development. When service A needs to speak to service B it needs the location of that service. The default discovery mechanism is  | ||||
| multicast DNS (mdns), a zeroconf system. | ||||
| @@ -30,26 +34,21 @@ across the services and retry a different node if there's a problem. | ||||
| to seamlessly encode and decode Go types for you. Any variety of messages could be encoded and sent from different clients. The client  | ||||
| and server handle this by default. This includes protobuf and json by default. | ||||
|  | ||||
| - **Request/Response** - RPC based request/response with support for bidirectional streaming. We provide an abstraction for synchronous  | ||||
| communication. A request made to a service will be automatically resolved, load balanced, dialled and streamed. The default  | ||||
| transport is [gRPC](https://grpc.io/). | ||||
| - **gRPC Transport** - gRPC based request/response with support for bidirectional streaming. We provide an abstraction for synchronous communication. A request made to a service will be automatically resolved, load balanced, dialled and streamed. | ||||
|  | ||||
| - **Async Messaging** - PubSub is built in as a first class citizen for asynchronous communication and event driven architectures.  | ||||
| Event notifications are a core pattern in micro service development. The default messaging system is a HTTP event message broker. | ||||
|  | ||||
| - **Synchronization** - Distributed systems are often built in an eventually consistent manner. Support for distributed locking and  | ||||
| leadership are built in as a Sync interface. When using an eventually consistent database or scheduling use the Sync interface. | ||||
|  | ||||
| - **Pluggable Interfaces** - Go Micro makes use of Go interfaces for each distributed system abstraction. Because of this these interfaces  | ||||
| are pluggable and allows Go Micro to be runtime agnostic. You can plugin any underlying technology. Find plugins in  | ||||
| [github.com/micro/go-plugins](https://github.com/micro/go-plugins). | ||||
|  | ||||
| ## Getting Started | ||||
|  | ||||
| To make use of Go Micro | ||||
|  | ||||
| ```golang | ||||
| import "github.com/micro/go-micro/v2" | ||||
| ``` | ||||
|  | ||||
| See the [docs](https://micro.mu/docs/framework.html) for detailed information on the architecture, installation and use of go-micro. | ||||
| See [pkg.go.dev](https://pkg.go.dev/github.com/micro/go-micro?tab=doc) for usage. | ||||
|  | ||||
| ## License | ||||
|  | ||||
|   | ||||
| @@ -1,36 +0,0 @@ | ||||
| # Go Micro [](https://opensource.org/licenses/Apache-2.0) [](https://pkg.go.dev/github.com/micro/go-micro?tab=doc) [](https://travis-ci.org/micro/go-micro) [](https://goreportcard.com/report/github.com/micro/go-micro) | ||||
|  | ||||
| Go Micro是基于Golang的微服务开发框架。 | ||||
|  | ||||
| ## 概览 | ||||
|  | ||||
| Go Micro提供分布式系统开发的核心库,包含RPC与事件驱动的通信机制。 | ||||
|  | ||||
| **micro**的设计哲学是可插拔的架构理念,她提供可快速构建系统的组件,并且可以根据自身的需求剥离默认实现并自行定制。 | ||||
|  | ||||
| <img src="https://micro.mu/docs/images/go-micro.svg" /> | ||||
|  | ||||
| 所有插件可在仓库[github.com/micro/go-plugins](https://github.com/micro/go-plugins)中找到。 | ||||
|  | ||||
| 可以订阅我们的[Twitter](https://twitter.com/microhq)或者加入[Slack](http://slack.micro.mu/)论坛。 | ||||
|  | ||||
| ## 特性 | ||||
|  | ||||
| Go Micro把分布式系统的各种细节抽象出来。下面是它的主要特性。 | ||||
|  | ||||
| - **服务发现(Service Discovery)** - 自动服务注册与名称解析。服务发现是微服务开发中的核心。当服务A要与服务B协作时,它得知道B在哪里。默认的服务发现系统是Consul,而multicast DNS (mdns,组播)机制作为本地解决方案,或者零依赖的P2P网络中的SWIM协议(gossip)。  | ||||
|  | ||||
| - **负载均衡(Load Balancing)** - 在服务发现之上构建了负载均衡机制。当我们得到一个服务的任意多个的实例节点时,我们要一个机制去决定要路由到哪一个节点。我们使用随机处理过的哈希负载均衡机制来保证对服务请求颁发的均匀分布,并且在发生问题时进行重试。 | ||||
|  | ||||
| - **消息编码(Message Encoding)** - 支持基于内容类型(content-type)动态编码消息。客户端和服务端会一起使用content-type的格式来对Go进行无缝编/解码。各种各样的消息被编码会发送到不同的客户端,客户端服服务端默认会处理这些消息。content-type默认包含proto-rpc和json-rpc。 | ||||
|  | ||||
| - **Request/Response** - RPC通信基于支持双向流的请求/响应方式,我们提供有抽象的同步通信机制。请求发送到服务时,会自动解析、负载均衡、拨号、转成字节流。默认的传输协议是http/1.1,而tls下使用http2协议。 | ||||
|  | ||||
| - **异步消息(Async Messaging)** - 发布订阅(PubSub)头等功能内置在异步通信与事件驱动架构中。事件通知在微服务开发中处于核心位置。默认的消息传送使用点到点http/1.1,激活tls时则使用http2。 | ||||
|  | ||||
| - **可插拔接口(Pluggable Interfaces)** - Go Micro为每个分布式系统抽象出接口。因此,Go Micro的接口都是可插拔的,允许其在运行时不可知的情况下仍可支持。所以只要实现接口,可以在内部使用任何的技术。更多插件请参考:[github.com/micro/go-plugins](https://github.com/micro/go-plugins)。 | ||||
|  | ||||
| ## 快速上手 | ||||
|  | ||||
| 更多关于架构、安装的资料可以查看[文档](https://micro.mu/docs/cn/)。 | ||||
|  | ||||
| @@ -2,7 +2,7 @@ | ||||
| package command | ||||
|  | ||||
| var ( | ||||
| 	// Commmands keyed by golang/regexp patterns | ||||
| 	// Commands keyed by golang/regexp patterns | ||||
| 	// regexp.Match(key, input) is used to match | ||||
| 	Commands = map[string]Command{} | ||||
| ) | ||||
|   | ||||
| @@ -6,8 +6,8 @@ import ( | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/bwmarrin/discordgo" | ||||
| 	"github.com/micro/go-micro/v2/agent/input" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/agent/input" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| type discordConn struct { | ||||
| @@ -70,12 +70,32 @@ func (dc *discordConn) Recv(event *input.Event) error { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ChunkString(s string, chunkSize int) []string { | ||||
| 	var chunks []string | ||||
| 	runes := []rune(s) | ||||
|  | ||||
| 	if len(runes) == 0 { | ||||
| 		return []string{s} | ||||
| 	} | ||||
|  | ||||
| 	for i := 0; i < len(runes); i += chunkSize { | ||||
| 		nn := i + chunkSize | ||||
| 		if nn > len(runes) { | ||||
| 			nn = len(runes) | ||||
| 		} | ||||
| 		chunks = append(chunks, string(runes[i:nn])) | ||||
| 	} | ||||
| 	return chunks | ||||
| } | ||||
|  | ||||
| func (dc *discordConn) Send(e *input.Event) error { | ||||
| 	fields := strings.Split(e.To, ":") | ||||
| 	_, err := dc.master.session.ChannelMessageSend(fields[0], string(e.Data)) | ||||
| 	if err != nil { | ||||
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) { | ||||
| 			logger.Error("[bot][loop][send]", err) | ||||
| 	for _, chunk := range ChunkString(string(e.Data), 2000) { | ||||
| 		_, err := dc.master.session.ChannelMessageSend(fields[0], chunk) | ||||
| 		if err != nil { | ||||
| 			if logger.V(logger.ErrorLevel, logger.DefaultLogger) { | ||||
| 				logger.Error("[bot][loop][send]", err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
|   | ||||
| @@ -9,7 +9,7 @@ import ( | ||||
|  | ||||
| 	"github.com/bwmarrin/discordgo" | ||||
| 	"github.com/micro/cli/v2" | ||||
| 	"github.com/micro/go-micro/v2/agent/input" | ||||
| 	"github.com/micro/go-micro/v3/agent/input" | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
|   | ||||
| @@ -7,7 +7,7 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/agent/input" | ||||
| 	"github.com/micro/go-micro/v3/agent/input" | ||||
| 	"github.com/nlopes/slack" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/micro/cli/v2" | ||||
| 	"github.com/micro/go-micro/v2/agent/input" | ||||
| 	"github.com/micro/go-micro/v3/agent/input" | ||||
| 	"github.com/nlopes/slack" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -6,8 +6,8 @@ import ( | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/forestgiant/sliceutil" | ||||
| 	"github.com/micro/go-micro/v2/agent/input" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/agent/input" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	tgbotapi "gopkg.in/telegram-bot-api.v4" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/micro/cli/v2" | ||||
| 	"github.com/micro/go-micro/v2/agent/input" | ||||
| 	"github.com/micro/go-micro/v3/agent/input" | ||||
| 	tgbotapi "gopkg.in/telegram-bot-api.v4" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -11,9 +11,9 @@ import ( | ||||
|  | ||||
| import ( | ||||
| 	context "context" | ||||
| 	api "github.com/micro/go-micro/v2/api" | ||||
| 	client "github.com/micro/go-micro/v2/client" | ||||
| 	server "github.com/micro/go-micro/v2/server" | ||||
| 	api "github.com/micro/go-micro/v3/api" | ||||
| 	client "github.com/micro/go-micro/v3/client" | ||||
| 	server "github.com/micro/go-micro/v3/server" | ||||
| ) | ||||
|  | ||||
| // Reference imports to suppress errors if they are not otherwise used. | ||||
|   | ||||
| @@ -5,8 +5,8 @@ import ( | ||||
| 	"regexp" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/server" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/server" | ||||
| ) | ||||
|  | ||||
| type Api interface { | ||||
| @@ -18,7 +18,7 @@ type Api interface { | ||||
| 	Register(*Endpoint) error | ||||
| 	// Register a route | ||||
| 	Deregister(*Endpoint) error | ||||
| 	// Implemenation of api | ||||
| 	// Implementation of api | ||||
| 	String() string | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -4,13 +4,13 @@ package api | ||||
| import ( | ||||
| 	"net/http" | ||||
|  | ||||
| 	goapi "github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	api "github.com/micro/go-micro/v2/api/proto" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	"github.com/micro/go-micro/v2/errors" | ||||
| 	"github.com/micro/go-micro/v2/util/ctx" | ||||
| 	goapi "github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	"github.com/micro/go-micro/v3/api/handler/util" | ||||
| 	api "github.com/micro/go-micro/v3/api/proto" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"github.com/micro/go-micro/v3/errors" | ||||
| 	"github.com/micro/go-micro/v3/util/ctx" | ||||
| ) | ||||
|  | ||||
| type apiHandler struct { | ||||
| @@ -71,10 +71,8 @@ func (a *apiHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
|  | ||||
| 	// create the context from headers | ||||
| 	cx := ctx.FromRequest(r) | ||||
| 	// create strategy | ||||
| 	so := selector.WithStrategy(strategy(service.Services)) | ||||
|  | ||||
| 	if err := c.Call(cx, req, rsp, client.WithSelectOption(so)); err != nil { | ||||
| 	if err := c.Call(cx, req, rsp, client.WithRouter(util.Router(service.Services))); err != nil { | ||||
| 		w.Header().Set("Content-Type", "application/json") | ||||
| 		ce := errors.Parse(err.Error()) | ||||
| 		switch ce.Code { | ||||
|   | ||||
| @@ -7,9 +7,7 @@ import ( | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
|  | ||||
| 	api "github.com/micro/go-micro/v2/api/proto" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	api "github.com/micro/go-micro/v3/api/proto" | ||||
| 	"github.com/oxtoacart/bpool" | ||||
| ) | ||||
|  | ||||
| @@ -109,11 +107,3 @@ func requestToProto(r *http.Request) (*api.Request, error) { | ||||
|  | ||||
| 	return req, nil | ||||
| } | ||||
|  | ||||
| // strategy is a hack for selection | ||||
| func strategy(services []*registry.Service) selector.Strategy { | ||||
| 	return func(_ []*registry.Service) selector.Next { | ||||
| 		// ignore input to this function, use services above | ||||
| 		return selector.Random(services) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -11,9 +11,9 @@ import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	proto "github.com/micro/go-micro/v2/api/proto" | ||||
| 	"github.com/micro/go-micro/v2/util/ctx" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	proto "github.com/micro/go-micro/v3/api/proto" | ||||
| 	"github.com/micro/go-micro/v3/util/ctx" | ||||
| 	"github.com/oxtoacart/bpool" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -4,13 +4,14 @@ package http | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math/rand" | ||||
| 	"net/http" | ||||
| 	"net/http/httputil" | ||||
| 	"net/url" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| @@ -20,7 +21,7 @@ const ( | ||||
| type httpHandler struct { | ||||
| 	options handler.Options | ||||
|  | ||||
| 	// set with different initialiser | ||||
| 	// set with different initializer | ||||
| 	s *api.Service | ||||
| } | ||||
|  | ||||
| @@ -64,16 +65,19 @@ func (h *httpHandler) getService(r *http.Request) (string, error) { | ||||
| 		return "", errors.New("no route found") | ||||
| 	} | ||||
|  | ||||
| 	// create a random selector | ||||
| 	next := selector.Random(service.Services) | ||||
|  | ||||
| 	// get the next node | ||||
| 	s, err := next() | ||||
| 	if err != nil { | ||||
| 		return "", nil | ||||
| 	// get the nodes for this service | ||||
| 	var nodes []*registry.Node | ||||
| 	for _, srv := range service.Services { | ||||
| 		nodes = append(nodes, srv.Nodes...) | ||||
| 	} | ||||
|  | ||||
| 	return fmt.Sprintf("http://%s", s.Address), nil | ||||
| 	// select a random node | ||||
| 	if len(nodes) == 0 { | ||||
| 		return "", errors.New("no route found") | ||||
| 	} | ||||
| 	node := nodes[rand.Int()%len(nodes)] | ||||
|  | ||||
| 	return fmt.Sprintf("http://%s", node.Address), nil | ||||
| } | ||||
|  | ||||
| func (h *httpHandler) String() string { | ||||
|   | ||||
| @@ -6,13 +6,13 @@ import ( | ||||
| 	"net/http/httptest" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v2/api/resolver/vpath" | ||||
| 	"github.com/micro/go-micro/v2/api/router" | ||||
| 	regRouter "github.com/micro/go-micro/v2/api/router/registry" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/registry/memory" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver/vpath" | ||||
| 	"github.com/micro/go-micro/v3/api/router" | ||||
| 	regRouter "github.com/micro/go-micro/v3/api/router/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry/memory" | ||||
| ) | ||||
|  | ||||
| func testHttp(t *testing.T, path, service, ns string) { | ||||
| @@ -58,7 +58,7 @@ func testHttp(t *testing.T, path, service, ns string) { | ||||
| 		router.WithHandler("http"), | ||||
| 		router.WithRegistry(r), | ||||
| 		router.WithResolver(vpath.NewResolver( | ||||
| 			resolver.WithNamespace(resolver.StaticNamespace(ns)), | ||||
| 			resolver.WithServicePrefix(ns), | ||||
| 		)), | ||||
| 	) | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,9 @@ | ||||
| package handler | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/api/router" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/client/grpc" | ||||
| 	"github.com/micro/go-micro/v3/api/router" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"github.com/micro/go-micro/v3/client/grpc" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| @@ -62,7 +62,7 @@ func WithClient(c client.Client) Option { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithmaxRecvSize specifies max body size | ||||
| // WithMaxRecvSize specifies max body size | ||||
| func WithMaxRecvSize(size int64) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.MaxRecvSize = size | ||||
|   | ||||
| @@ -5,25 +5,23 @@ import ( | ||||
| 	"encoding/json" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"net/textproto" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | ||||
| 	jsonpatch "github.com/evanphx/json-patch/v5" | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	"github.com/micro/go-micro/v2/api/internal/proto" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	"github.com/micro/go-micro/v2/codec" | ||||
| 	"github.com/micro/go-micro/v2/codec/jsonrpc" | ||||
| 	"github.com/micro/go-micro/v2/codec/protorpc" | ||||
| 	"github.com/micro/go-micro/v2/errors" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v2/metadata" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/util/ctx" | ||||
| 	"github.com/micro/go-micro/v2/util/qson" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	"github.com/micro/go-micro/v3/api/handler/util" | ||||
| 	"github.com/micro/go-micro/v3/api/internal/proto" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"github.com/micro/go-micro/v3/codec" | ||||
| 	"github.com/micro/go-micro/v3/codec/jsonrpc" | ||||
| 	"github.com/micro/go-micro/v3/codec/protorpc" | ||||
| 	"github.com/micro/go-micro/v3/errors" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	"github.com/micro/go-micro/v3/metadata" | ||||
| 	"github.com/micro/go-micro/v3/util/ctx" | ||||
| 	"github.com/micro/go-micro/v3/util/qson" | ||||
| 	"github.com/oxtoacart/bpool" | ||||
| ) | ||||
|  | ||||
| @@ -65,14 +63,6 @@ func (b *buffer) Write(_ []byte) (int, error) { | ||||
| 	return 0, nil | ||||
| } | ||||
|  | ||||
| // strategy is a hack for selection | ||||
| func strategy(services []*registry.Service) selector.Strategy { | ||||
| 	return func(_ []*registry.Service) selector.Next { | ||||
| 		// ignore input to this function, use services above | ||||
| 		return selector.Random(services) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (h *rpcHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	bsize := handler.DefaultMaxRecvSize | ||||
| 	if h.opts.MaxRecvSize > 0 { | ||||
| @@ -113,36 +103,17 @@ func (h *rpcHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
|  | ||||
| 	// create context | ||||
| 	cx := ctx.FromRequest(r) | ||||
| 	// get context from http handler wrappers | ||||
| 	md, ok := metadata.FromContext(r.Context()) | ||||
| 	if !ok { | ||||
| 		md = make(metadata.Metadata) | ||||
| 	} | ||||
| 	// fill contex with http headers | ||||
| 	md["Host"] = r.Host | ||||
| 	md["Method"] = r.Method | ||||
| 	// get canonical headers | ||||
| 	for k, _ := range r.Header { | ||||
| 		// may be need to get all values for key like r.Header.Values() provide in go 1.14 | ||||
| 		md[textproto.CanonicalMIMEHeaderKey(k)] = r.Header.Get(k) | ||||
| 	} | ||||
|  | ||||
| 	// merge context with overwrite | ||||
| 	cx = metadata.MergeContext(cx, md, true) | ||||
|  | ||||
| 	// set merged context to request | ||||
| 	*r = *r.Clone(cx) | ||||
| 	// if stream we currently only support json | ||||
| 	if isStream(r, service) { | ||||
| 		// drop older context as it can have timeouts and create new | ||||
| 		//		md, _ := metadata.FromContext(cx) | ||||
| 		//serveWebsocket(context.TODO(), w, r, service, c) | ||||
| 		serveWebsocket(cx, w, r, service, c) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	// create strategy | ||||
| 	so := selector.WithStrategy(strategy(service.Services)) | ||||
| 	// create custom router | ||||
| 	callOpt := client.WithRouter(util.Router(service.Services)) | ||||
|  | ||||
| 	// walk the standard call path | ||||
| 	// get payload | ||||
| @@ -174,7 +145,7 @@ func (h *rpcHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 		) | ||||
|  | ||||
| 		// make the call | ||||
| 		if err := c.Call(cx, req, response, client.WithSelectOption(so)); err != nil { | ||||
| 		if err := c.Call(cx, req, response, callOpt); err != nil { | ||||
| 			writeError(w, r, err) | ||||
| 			return | ||||
| 		} | ||||
| @@ -209,7 +180,7 @@ func (h *rpcHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 			client.WithContentType(ct), | ||||
| 		) | ||||
| 		// make the call | ||||
| 		if err := c.Call(cx, req, &response, client.WithSelectOption(so)); err != nil { | ||||
| 		if err := c.Call(cx, req, &response, callOpt); err != nil { | ||||
| 			writeError(w, r, err) | ||||
| 			return | ||||
| 		} | ||||
| @@ -294,7 +265,7 @@ func requestPayload(r *http.Request) ([]byte, error) { | ||||
|  | ||||
| 	// otherwise as per usual | ||||
| 	ctx := r.Context() | ||||
| 	// dont user meadata.FromContext as it mangles names | ||||
| 	// dont user metadata.FromContext as it mangles names | ||||
| 	md, ok := metadata.FromContext(ctx) | ||||
| 	if !ok { | ||||
| 		md = make(map[string]string) | ||||
| @@ -397,13 +368,29 @@ func requestPayload(r *http.Request) ([]byte, error) { | ||||
| 				return out, nil | ||||
| 			} | ||||
| 		} | ||||
| 		var jsonbody map[string]interface{} | ||||
| 		if json.Valid(bodybuf) { | ||||
| 			if err = json.Unmarshal(bodybuf, &jsonbody); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 		} | ||||
| 		dstmap := make(map[string]interface{}) | ||||
| 		ps := strings.Split(bodydst, ".") | ||||
| 		if len(ps) == 1 { | ||||
| 			dstmap[ps[0]] = bodybuf | ||||
| 			if jsonbody != nil { | ||||
| 				dstmap[ps[0]] = jsonbody | ||||
| 			} else { | ||||
| 				// old unexpected behaviour | ||||
| 				dstmap[ps[0]] = bodybuf | ||||
| 			} | ||||
| 		} else { | ||||
| 			em := make(map[string]interface{}) | ||||
| 			em[ps[len(ps)-1]] = bodybuf | ||||
| 			if jsonbody != nil { | ||||
| 				em[ps[len(ps)-1]] = jsonbody | ||||
| 			} else { | ||||
| 				// old unexpected behaviour | ||||
| 				em[ps[len(ps)-1]] = bodybuf | ||||
| 			} | ||||
| 			for i := len(ps) - 2; i > 0; i-- { | ||||
| 				nm := make(map[string]interface{}) | ||||
| 				nm[ps[i]] = em | ||||
|   | ||||
| @@ -7,7 +7,7 @@ import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	go_api "github.com/micro/go-micro/v2/api/proto" | ||||
| 	go_api "github.com/micro/go-micro/v3/api/proto" | ||||
| ) | ||||
|  | ||||
| func TestRequestPayloadFromRequest(t *testing.T) { | ||||
|   | ||||
| @@ -12,11 +12,11 @@ import ( | ||||
| 	"github.com/gobwas/httphead" | ||||
| 	"github.com/gobwas/ws" | ||||
| 	"github.com/gobwas/ws/wsutil" | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	raw "github.com/micro/go-micro/v2/codec/bytes" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/handler/util" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	raw "github.com/micro/go-micro/v3/codec/bytes" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| // serveWebsocket will stream rpc back over websockets assuming json | ||||
| @@ -110,9 +110,11 @@ func serveWebsocket(ctx context.Context, w http.ResponseWriter, r *http.Request, | ||||
| 		client.StreamingRequest(), | ||||
| 	) | ||||
|  | ||||
| 	so := selector.WithStrategy(strategy(service.Services)) | ||||
| 	// create custom router | ||||
| 	callOpt := client.WithRouter(util.Router(service.Services)) | ||||
|  | ||||
| 	// create a new stream | ||||
| 	stream, err := c.Stream(ctx, req, client.WithSelectOption(so)) | ||||
| 	stream, err := c.Stream(ctx, req, callOpt) | ||||
| 	if err != nil { | ||||
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) { | ||||
| 			logger.Error(err) | ||||
| @@ -185,7 +187,11 @@ func writeLoop(rw io.ReadWriter, stream client.Stream) { | ||||
| 			if err != nil { | ||||
| 				if wserr, ok := err.(wsutil.ClosedError); ok { | ||||
| 					switch wserr.Code { | ||||
| 					case ws.StatusGoingAway: | ||||
| 						// this happens when user leave the page | ||||
| 						return | ||||
| 					case ws.StatusNormalClosure, ws.StatusNoStatusRcvd: | ||||
| 						// this happens when user close ws connection, or we don't get any status | ||||
| 						return | ||||
| 					} | ||||
| 				} | ||||
|   | ||||
							
								
								
									
										32
									
								
								api/handler/util/router.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								api/handler/util/router.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| package util | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/router" | ||||
| ) | ||||
|  | ||||
| // Router is a hack for API routing | ||||
| func Router(srvs []*registry.Service) router.Router { | ||||
| 	var routes []router.Route | ||||
|  | ||||
| 	for _, srv := range srvs { | ||||
| 		for _, n := range srv.Nodes { | ||||
| 			routes = append(routes, router.Route{Address: n.Address, Metadata: n.Metadata}) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return &apiRouter{routes: routes} | ||||
| } | ||||
|  | ||||
| func (r *apiRouter) Lookup(...router.QueryOption) ([]router.Route, error) { | ||||
| 	return r.routes, nil | ||||
| } | ||||
|  | ||||
| type apiRouter struct { | ||||
| 	routes []router.Route | ||||
| 	router.Router | ||||
| } | ||||
|  | ||||
| func (r *apiRouter) String() string { | ||||
| 	return "api" | ||||
| } | ||||
| @@ -5,15 +5,16 @@ import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"math/rand" | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"net/http/httputil" | ||||
| 	"net/url" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| @@ -70,16 +71,19 @@ func (wh *webHandler) getService(r *http.Request) (string, error) { | ||||
| 		return "", errors.New("no route found") | ||||
| 	} | ||||
|  | ||||
| 	// create a random selector | ||||
| 	next := selector.Random(service.Services) | ||||
|  | ||||
| 	// get the next node | ||||
| 	s, err := next() | ||||
| 	if err != nil { | ||||
| 		return "", nil | ||||
| 	// get the nodes | ||||
| 	var nodes []*registry.Node | ||||
| 	for _, srv := range service.Services { | ||||
| 		nodes = append(nodes, srv.Nodes...) | ||||
| 	} | ||||
| 	if len(nodes) == 0 { | ||||
| 		return "", errors.New("no route found") | ||||
| 	} | ||||
|  | ||||
| 	return fmt.Sprintf("http://%s", s.Address), nil | ||||
| 	// select a random node | ||||
| 	node := nodes[rand.Int()%len(nodes)] | ||||
|  | ||||
| 	return fmt.Sprintf("http://%s", node.Address), nil | ||||
| } | ||||
|  | ||||
| // serveWebSocket used to serve a web socket proxied connection | ||||
|   | ||||
| @@ -6,12 +6,17 @@ import ( | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| ) | ||||
|  | ||||
| type Resolver struct{} | ||||
| type Resolver struct { | ||||
| 	opts resolver.Options | ||||
| } | ||||
|  | ||||
| func (r *Resolver) Resolve(req *http.Request, opts ...resolver.ResolveOption) (*resolver.Endpoint, error) { | ||||
| 	// parse options | ||||
| 	options := resolver.NewResolveOptions(opts...) | ||||
|  | ||||
| func (r *Resolver) Resolve(req *http.Request) (*resolver.Endpoint, error) { | ||||
| 	// /foo.Bar/Service | ||||
| 	if req.URL.Path == "/" { | ||||
| 		return nil, errors.New("unknown name") | ||||
| @@ -26,6 +31,7 @@ func (r *Resolver) Resolve(req *http.Request) (*resolver.Endpoint, error) { | ||||
| 		Host:   req.Host, | ||||
| 		Method: req.Method, | ||||
| 		Path:   req.URL.Path, | ||||
| 		Domain: options.Domain, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| @@ -34,5 +40,5 @@ func (r *Resolver) String() string { | ||||
| } | ||||
|  | ||||
| func NewResolver(opts ...resolver.Option) resolver.Resolver { | ||||
| 	return &Resolver{} | ||||
| 	return &Resolver{opts: resolver.NewOptions(opts...)} | ||||
| } | ||||
|   | ||||
| @@ -4,19 +4,23 @@ package host | ||||
| import ( | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| ) | ||||
|  | ||||
| type Resolver struct { | ||||
| 	opts resolver.Options | ||||
| } | ||||
|  | ||||
| func (r *Resolver) Resolve(req *http.Request) (*resolver.Endpoint, error) { | ||||
| func (r *Resolver) Resolve(req *http.Request, opts ...resolver.ResolveOption) (*resolver.Endpoint, error) { | ||||
| 	// parse options | ||||
| 	options := resolver.NewResolveOptions(opts...) | ||||
|  | ||||
| 	return &resolver.Endpoint{ | ||||
| 		Name:   req.Host, | ||||
| 		Host:   req.Host, | ||||
| 		Method: req.Method, | ||||
| 		Path:   req.URL.Path, | ||||
| 		Domain: options.Domain, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -1,25 +1,16 @@ | ||||
| package resolver | ||||
|  | ||||
| import ( | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/auth" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| ) | ||||
|  | ||||
| // NewOptions returns new initialised options | ||||
| func NewOptions(opts ...Option) Options { | ||||
| 	var options Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	if options.Namespace == nil { | ||||
| 		options.Namespace = StaticNamespace(auth.DefaultNamespace) | ||||
| 	} | ||||
|  | ||||
| 	return options | ||||
| type Options struct { | ||||
| 	Handler       string | ||||
| 	ServicePrefix string | ||||
| } | ||||
|  | ||||
| type Option func(o *Options) | ||||
|  | ||||
| // WithHandler sets the handler being used | ||||
| func WithHandler(h string) Option { | ||||
| 	return func(o *Options) { | ||||
| @@ -27,9 +18,46 @@ func WithHandler(h string) Option { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithNamespace sets the function which determines the namespace for a request | ||||
| func WithNamespace(n func(*http.Request) string) Option { | ||||
| // WithServicePrefix sets the ServicePrefix option | ||||
| func WithServicePrefix(p string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Namespace = n | ||||
| 		o.ServicePrefix = p | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // NewOptions returns new initialised options | ||||
| func NewOptions(opts ...Option) Options { | ||||
| 	var options Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
| 	return options | ||||
| } | ||||
|  | ||||
| // ResolveOptions are used when resolving a request | ||||
| type ResolveOptions struct { | ||||
| 	Domain string | ||||
| } | ||||
|  | ||||
| // ResolveOption sets an option | ||||
| type ResolveOption func(*ResolveOptions) | ||||
|  | ||||
| // Domain sets the resolve Domain option | ||||
| func Domain(n string) ResolveOption { | ||||
| 	return func(o *ResolveOptions) { | ||||
| 		o.Domain = n | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // NewResolveOptions returns new initialised resolve options | ||||
| func NewResolveOptions(opts ...ResolveOption) ResolveOptions { | ||||
| 	var options ResolveOptions | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
| 	if len(options.Domain) == 0 { | ||||
| 		options.Domain = registry.DefaultDomain | ||||
| 	} | ||||
|  | ||||
| 	return options | ||||
| } | ||||
|   | ||||
| @@ -5,26 +5,29 @@ import ( | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| ) | ||||
|  | ||||
| type Resolver struct { | ||||
| 	opts resolver.Options | ||||
| } | ||||
|  | ||||
| func (r *Resolver) Resolve(req *http.Request) (*resolver.Endpoint, error) { | ||||
| func (r *Resolver) Resolve(req *http.Request, opts ...resolver.ResolveOption) (*resolver.Endpoint, error) { | ||||
| 	// parse options | ||||
| 	options := resolver.NewResolveOptions(opts...) | ||||
|  | ||||
| 	if req.URL.Path == "/" { | ||||
| 		return nil, resolver.ErrNotFound | ||||
| 	} | ||||
|  | ||||
| 	parts := strings.Split(req.URL.Path[1:], "/") | ||||
| 	ns := r.opts.Namespace(req) | ||||
|  | ||||
| 	return &resolver.Endpoint{ | ||||
| 		Name:   ns + "." + parts[0], | ||||
| 		Name:   r.opts.ServicePrefix + "." + parts[0], | ||||
| 		Host:   req.Host, | ||||
| 		Method: req.Method, | ||||
| 		Path:   req.URL.Path, | ||||
| 		Domain: options.Domain, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -13,7 +13,7 @@ var ( | ||||
|  | ||||
| // Resolver resolves requests to endpoints | ||||
| type Resolver interface { | ||||
| 	Resolve(r *http.Request) (*Endpoint, error) | ||||
| 	Resolve(r *http.Request, opts ...ResolveOption) (*Endpoint, error) | ||||
| 	String() string | ||||
| } | ||||
|  | ||||
| @@ -27,18 +27,6 @@ type Endpoint struct { | ||||
| 	Method string | ||||
| 	// HTTP Path e.g /greeter. | ||||
| 	Path string | ||||
| } | ||||
|  | ||||
| type Options struct { | ||||
| 	Handler   string | ||||
| 	Namespace func(*http.Request) string | ||||
| } | ||||
|  | ||||
| type Option func(o *Options) | ||||
|  | ||||
| // StaticNamespace returns the same namespace for each request | ||||
| func StaticNamespace(ns string) func(*http.Request) string { | ||||
| 	return func(*http.Request) string { | ||||
| 		return ns | ||||
| 	} | ||||
| 	// Domain endpoint exists within | ||||
| 	Domain string | ||||
| } | ||||
|   | ||||
							
								
								
									
										80
									
								
								api/resolver/subdomain/subdomain.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								api/resolver/subdomain/subdomain.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| // Package subdomain is a resolver which uses the subdomain to determine the domain to route to. It | ||||
| // offloads the endpoint resolution to a child resolver which is provided in New. | ||||
| package subdomain | ||||
|  | ||||
| import ( | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	"golang.org/x/net/publicsuffix" | ||||
| ) | ||||
|  | ||||
| func NewResolver(parent resolver.Resolver, opts ...resolver.Option) resolver.Resolver { | ||||
| 	options := resolver.NewOptions(opts...) | ||||
| 	return &Resolver{options, parent} | ||||
| } | ||||
|  | ||||
| type Resolver struct { | ||||
| 	opts resolver.Options | ||||
| 	resolver.Resolver | ||||
| } | ||||
|  | ||||
| func (r *Resolver) Resolve(req *http.Request, opts ...resolver.ResolveOption) (*resolver.Endpoint, error) { | ||||
| 	if dom := r.Domain(req); len(dom) > 0 { | ||||
| 		opts = append(opts, resolver.Domain(dom)) | ||||
| 	} | ||||
|  | ||||
| 	return r.Resolver.Resolve(req, opts...) | ||||
| } | ||||
|  | ||||
| func (r *Resolver) Domain(req *http.Request) string { | ||||
| 	// determine the host, e.g. foobar.m3o.app | ||||
| 	host := req.URL.Hostname() | ||||
| 	if len(host) == 0 { | ||||
| 		if h, _, err := net.SplitHostPort(req.Host); err == nil { | ||||
| 			host = h // host does contain a port | ||||
| 		} else if strings.Contains(err.Error(), "missing port in address") { | ||||
| 			host = req.Host // host does not contain a port | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// check for an ip address | ||||
| 	if net.ParseIP(host) != nil { | ||||
| 		return "" | ||||
| 	} | ||||
|  | ||||
| 	// check for dev enviroment | ||||
| 	if host == "localhost" || host == "127.0.0.1" { | ||||
| 		return "" | ||||
| 	} | ||||
|  | ||||
| 	// extract the top level domain plus one (e.g. 'myapp.com') | ||||
| 	domain, err := publicsuffix.EffectiveTLDPlusOne(host) | ||||
| 	if err != nil { | ||||
| 		logger.Debugf("Unable to extract domain from %v", host) | ||||
| 		return "" | ||||
| 	} | ||||
|  | ||||
| 	// there was no subdomain | ||||
| 	if host == domain { | ||||
| 		return "" | ||||
| 	} | ||||
|  | ||||
| 	// remove the domain from the host, leaving the subdomain, e.g. "staging.foo.myapp.com" => "staging.foo" | ||||
| 	subdomain := strings.TrimSuffix(host, "."+domain) | ||||
|  | ||||
| 	// return the reversed subdomain as the namespace, e.g. "staging.foo" => "foo-staging" | ||||
| 	comps := strings.Split(subdomain, ".") | ||||
| 	for i := len(comps)/2 - 1; i >= 0; i-- { | ||||
| 		opp := len(comps) - 1 - i | ||||
| 		comps[i], comps[opp] = comps[opp], comps[i] | ||||
| 	} | ||||
| 	return strings.Join(comps, "-") | ||||
| } | ||||
|  | ||||
| func (r *Resolver) String() string { | ||||
| 	return "subdomain" | ||||
| } | ||||
							
								
								
									
										71
									
								
								api/resolver/subdomain/subdomain_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								api/resolver/subdomain/subdomain_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| package subdomain | ||||
|  | ||||
| import ( | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v3/api/resolver/vpath" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
|  | ||||
| func TestResolve(t *testing.T) { | ||||
| 	tt := []struct { | ||||
| 		Name   string | ||||
| 		Host   string | ||||
| 		Result string | ||||
| 	}{ | ||||
| 		{ | ||||
| 			Name:   "Top level domain", | ||||
| 			Host:   "micro.mu", | ||||
| 			Result: "micro", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "Effective top level domain", | ||||
| 			Host:   "micro.com.au", | ||||
| 			Result: "micro", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "Subdomain dev", | ||||
| 			Host:   "dev.micro.mu", | ||||
| 			Result: "dev", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "Subdomain foo", | ||||
| 			Host:   "foo.micro.mu", | ||||
| 			Result: "foo", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "Multi-level subdomain", | ||||
| 			Host:   "staging.myapp.m3o.app", | ||||
| 			Result: "myapp-staging", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "Dev host", | ||||
| 			Host:   "127.0.0.1", | ||||
| 			Result: "micro", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "Localhost", | ||||
| 			Host:   "localhost", | ||||
| 			Result: "micro", | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:   "IP host", | ||||
| 			Host:   "81.151.101.146", | ||||
| 			Result: "micro", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tt { | ||||
| 		t.Run(tc.Name, func(t *testing.T) { | ||||
| 			r := NewResolver(vpath.NewResolver()) | ||||
| 			result, err := r.Resolve(&http.Request{URL: &url.URL{Host: tc.Host, Path: "foo/bar"}}) | ||||
| 			assert.Nil(t, err, "Expecter err to be nil") | ||||
| 			if result != nil { | ||||
| 				assert.Equal(t, tc.Result, result.Domain, "Expected %v but got %v", tc.Result, result.Domain) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| @@ -3,12 +3,11 @@ package vpath | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| 	"regexp" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| ) | ||||
|  | ||||
| func NewResolver(opts ...resolver.Option) resolver.Resolver { | ||||
| @@ -23,38 +22,41 @@ var ( | ||||
| 	re = regexp.MustCompile("^v[0-9]+$") | ||||
| ) | ||||
|  | ||||
| func (r *Resolver) Resolve(req *http.Request) (*resolver.Endpoint, error) { | ||||
| func (r *Resolver) Resolve(req *http.Request, opts ...resolver.ResolveOption) (*resolver.Endpoint, error) { | ||||
| 	if req.URL.Path == "/" { | ||||
| 		return nil, errors.New("unknown name") | ||||
| 	} | ||||
|  | ||||
| 	fmt.Println(req.URL.Path) | ||||
| 	options := resolver.NewResolveOptions(opts...) | ||||
|  | ||||
| 	parts := strings.Split(req.URL.Path[1:], "/") | ||||
| 	if len(parts) == 1 { | ||||
| 		return &resolver.Endpoint{ | ||||
| 			Name:   r.withNamespace(req, parts...), | ||||
| 			Name:   r.withPrefix(parts...), | ||||
| 			Host:   req.Host, | ||||
| 			Method: req.Method, | ||||
| 			Path:   req.URL.Path, | ||||
| 			Domain: options.Domain, | ||||
| 		}, nil | ||||
| 	} | ||||
|  | ||||
| 	// /v1/foo | ||||
| 	if re.MatchString(parts[0]) { | ||||
| 		return &resolver.Endpoint{ | ||||
| 			Name:   r.withNamespace(req, parts[0:2]...), | ||||
| 			Name:   r.withPrefix(parts[0:2]...), | ||||
| 			Host:   req.Host, | ||||
| 			Method: req.Method, | ||||
| 			Path:   req.URL.Path, | ||||
| 			Domain: options.Domain, | ||||
| 		}, nil | ||||
| 	} | ||||
|  | ||||
| 	return &resolver.Endpoint{ | ||||
| 		Name:   r.withNamespace(req, parts[0]), | ||||
| 		Name:   r.withPrefix(parts[0]), | ||||
| 		Host:   req.Host, | ||||
| 		Method: req.Method, | ||||
| 		Path:   req.URL.Path, | ||||
| 		Domain: options.Domain, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| @@ -62,11 +64,12 @@ func (r *Resolver) String() string { | ||||
| 	return "path" | ||||
| } | ||||
|  | ||||
| func (r *Resolver) withNamespace(req *http.Request, parts ...string) string { | ||||
| 	ns := r.opts.Namespace(req) | ||||
| 	if len(ns) == 0 { | ||||
| 		return strings.Join(parts, ".") | ||||
| // withPrefix transforms "foo" into "go.micro.api.foo" | ||||
| func (r *Resolver) withPrefix(parts ...string) string { | ||||
| 	p := r.opts.ServicePrefix | ||||
| 	if len(p) > 0 { | ||||
| 		parts = append([]string{p}, parts...) | ||||
| 	} | ||||
|  | ||||
| 	return strings.Join(append([]string{ns}, parts...), ".") | ||||
| 	return strings.Join(parts, ".") | ||||
| } | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| package router | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v2/api/resolver/vpath" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver/vpath" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry/mdns" | ||||
| ) | ||||
|  | ||||
| type Options struct { | ||||
| @@ -17,7 +18,7 @@ type Option func(o *Options) | ||||
| func NewOptions(opts ...Option) Options { | ||||
| 	options := Options{ | ||||
| 		Handler:  "meta", | ||||
| 		Registry: registry.DefaultRegistry, | ||||
| 		Registry: mdns.NewRegistry(), | ||||
| 	} | ||||
|  | ||||
| 	for _, o := range opts { | ||||
|   | ||||
| @@ -10,13 +10,13 @@ import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/router" | ||||
| 	"github.com/micro/go-micro/v2/api/router/util" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v2/metadata" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/registry/cache" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/router" | ||||
| 	"github.com/micro/go-micro/v3/api/router/util" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	"github.com/micro/go-micro/v3/metadata" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry/cache" | ||||
| ) | ||||
|  | ||||
| // endpoint struct, that holds compiled pcre | ||||
| @@ -99,7 +99,7 @@ func (r *registryRouter) process(res *registry.Result) { | ||||
| 	service, err := r.rc.GetService(res.Service.Name) | ||||
| 	if err != nil { | ||||
| 		if logger.V(logger.ErrorLevel, logger.DefaultLogger) { | ||||
| 			logger.Errorf("unable to get service: %v", err) | ||||
| 			logger.Errorf("unable to get %v service: %v", res.Service.Name, err) | ||||
| 		} | ||||
| 		return | ||||
| 	} | ||||
| @@ -188,7 +188,7 @@ func (r *registryRouter) store(services []*registry.Service) { | ||||
| 		for _, p := range ep.Endpoint.Path { | ||||
| 			var pcreok bool | ||||
|  | ||||
| 			if p[0] == '^' && p[len(p)-1] != '$' { | ||||
| 			if p[0] == '^' && p[len(p)-1] == '$' { | ||||
| 				pcrereg, err := regexp.CompilePOSIX(p) | ||||
| 				if err == nil { | ||||
| 					cep.pcreregs = append(cep.pcreregs, pcrereg) | ||||
| @@ -260,7 +260,7 @@ func (r *registryRouter) watch() { | ||||
| 			res, err := w.Next() | ||||
| 			if err != nil { | ||||
| 				if logger.V(logger.ErrorLevel, logger.DefaultLogger) { | ||||
| 					logger.Errorf("error getting next endoint: %v", err) | ||||
| 					logger.Errorf("error getting next endpoint: %v", err) | ||||
| 				} | ||||
| 				close(ch) | ||||
| 				break | ||||
| @@ -434,7 +434,7 @@ func (r *registryRouter) Route(req *http.Request) (*api.Service, error) { | ||||
| 	name := rp.Name | ||||
|  | ||||
| 	// get service | ||||
| 	services, err := r.rc.GetService(name) | ||||
| 	services, err := r.rc.GetService(name, registry.GetDomain(rp.Domain)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										34
									
								
								api/router/registry/registry_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								api/router/registry/registry_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| package registry | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
|  | ||||
| func TestStoreRegex(t *testing.T) { | ||||
| 	router := newRouter() | ||||
| 	router.store([]*registry.Service{ | ||||
| 		{ | ||||
| 			Name:    "Foobar", | ||||
| 			Version: "latest", | ||||
| 			Endpoints: []*registry.Endpoint{ | ||||
| 				{ | ||||
| 					Name: "foo", | ||||
| 					Metadata: map[string]string{ | ||||
| 						"endpoint":    "FooEndpoint", | ||||
| 						"description": "Some description", | ||||
| 						"method":      "POST", | ||||
| 						"path":        "^/foo/$", | ||||
| 						"handler":     "rpc", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Metadata: map[string]string{}, | ||||
| 		}, | ||||
| 	}, | ||||
| 	) | ||||
|  | ||||
| 	assert.Len(t, router.ceps["Foobar.foo"].pcreregs, 1) | ||||
| } | ||||
| @@ -4,7 +4,7 @@ package router | ||||
| import ( | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| ) | ||||
|  | ||||
| // Router is used to determine an endpoint for a request | ||||
|   | ||||
| @@ -9,18 +9,18 @@ import ( | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/handler" | ||||
| 	"github.com/micro/go-micro/v2/api/handler/rpc" | ||||
| 	"github.com/micro/go-micro/v2/api/router" | ||||
| 	rregistry "github.com/micro/go-micro/v2/api/router/registry" | ||||
| 	rstatic "github.com/micro/go-micro/v2/api/router/static" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	gcli "github.com/micro/go-micro/v2/client/grpc" | ||||
| 	rmemory "github.com/micro/go-micro/v2/registry/memory" | ||||
| 	"github.com/micro/go-micro/v2/server" | ||||
| 	gsrv "github.com/micro/go-micro/v2/server/grpc" | ||||
| 	pb "github.com/micro/go-micro/v2/server/grpc/proto" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/handler" | ||||
| 	"github.com/micro/go-micro/v3/api/handler/rpc" | ||||
| 	"github.com/micro/go-micro/v3/api/router" | ||||
| 	rregistry "github.com/micro/go-micro/v3/api/router/registry" | ||||
| 	rstatic "github.com/micro/go-micro/v3/api/router/static" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	gcli "github.com/micro/go-micro/v3/client/grpc" | ||||
| 	rmemory "github.com/micro/go-micro/v3/registry/memory" | ||||
| 	"github.com/micro/go-micro/v3/server" | ||||
| 	gsrv "github.com/micro/go-micro/v3/server/grpc" | ||||
| 	pb "github.com/micro/go-micro/v3/server/grpc/proto" | ||||
| ) | ||||
|  | ||||
| // server is used to implement helloworld.GreeterServer. | ||||
|   | ||||
| @@ -8,13 +8,13 @@ import ( | ||||
| 	"strings" | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api" | ||||
| 	"github.com/micro/go-micro/v2/api/router" | ||||
| 	"github.com/micro/go-micro/v2/api/router/util" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v2/metadata" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	rutil "github.com/micro/go-micro/v2/util/registry" | ||||
| 	"github.com/micro/go-micro/v3/api" | ||||
| 	"github.com/micro/go-micro/v3/api/router" | ||||
| 	"github.com/micro/go-micro/v3/api/router/util" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	"github.com/micro/go-micro/v3/metadata" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	rutil "github.com/micro/go-micro/v3/util/registry" | ||||
| ) | ||||
|  | ||||
| type endpoint struct { | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| // InvalidTemplateError indicates that the path template is not valid. | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import ( | ||||
| 	"reflect" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| func TestTokenize(t *testing.T) { | ||||
|   | ||||
| @@ -7,7 +7,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
|   | ||||
| @@ -7,8 +7,8 @@ import ( | ||||
| 	"net" | ||||
| 	"os" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/server/acme" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/api/server/acme" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	"golang.org/x/crypto/acme/autocert" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| // Package certmagic is the ACME provider from github.com/mholt/certmagic | ||||
| // Package certmagic is the ACME provider from github.com/caddyserver/certmagic | ||||
| package certmagic | ||||
|  | ||||
| import ( | ||||
| @@ -7,9 +7,9 @@ import ( | ||||
| 	"net" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/mholt/certmagic" | ||||
| 	"github.com/micro/go-micro/v2/api/server/acme" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/caddyserver/certmagic" | ||||
| 	"github.com/micro/go-micro/v3/api/server/acme" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| type certmagicProvider struct { | ||||
| @@ -18,10 +18,10 @@ type certmagicProvider struct { | ||||
|  | ||||
| // TODO: set self-contained options | ||||
| func (c *certmagicProvider) setup() { | ||||
| 	certmagic.Default.CA = c.opts.CA | ||||
| 	certmagic.DefaultACME.CA = c.opts.CA | ||||
| 	if c.opts.ChallengeProvider != nil { | ||||
| 		// Enabling DNS Challenge disables the other challenges | ||||
| 		certmagic.Default.DNSProvider = c.opts.ChallengeProvider | ||||
| 		certmagic.DefaultACME.DNSProvider = c.opts.ChallengeProvider | ||||
| 	} | ||||
| 	if c.opts.OnDemand { | ||||
| 		certmagic.Default.OnDemand = new(certmagic.OnDemandConfig) | ||||
| @@ -32,9 +32,10 @@ func (c *certmagicProvider) setup() { | ||||
| 	} | ||||
| 	// If multiple instances of the provider are running, inject some | ||||
| 	// randomness so they don't collide | ||||
| 	// RenewalWindowRatio [0.33 - 0.50) | ||||
| 	rand.Seed(time.Now().UnixNano()) | ||||
| 	randomDuration := (7 * 24 * time.Hour) + (time.Duration(rand.Intn(504)) * time.Hour) | ||||
| 	certmagic.Default.RenewDurationBefore = randomDuration | ||||
| 	randomRatio := float64(rand.Intn(17)+33) * 0.01 | ||||
| 	certmagic.Default.RenewalWindowRatio = randomRatio | ||||
| } | ||||
|  | ||||
| func (c *certmagicProvider) Listen(hosts ...string) (net.Listener, error) { | ||||
|   | ||||
| @@ -9,9 +9,9 @@ import ( | ||||
| 	"strings" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/mholt/certmagic" | ||||
| 	"github.com/micro/go-micro/v2/store" | ||||
| 	"github.com/micro/go-micro/v2/sync" | ||||
| 	"github.com/caddyserver/certmagic" | ||||
| 	"github.com/micro/go-micro/v3/store" | ||||
| 	"github.com/micro/go-micro/v3/sync" | ||||
| ) | ||||
|  | ||||
| // File represents a "File" that will be stored in store.Store - the contents and last modified time | ||||
|   | ||||
| @@ -9,9 +9,9 @@ import ( | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/gorilla/handlers" | ||||
| 	"github.com/micro/go-micro/v2/api/server" | ||||
| 	"github.com/micro/go-micro/v2/api/server/cors" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v3/api/server" | ||||
| 	"github.com/micro/go-micro/v3/api/server/cors" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| ) | ||||
|  | ||||
| type httpServer struct { | ||||
|   | ||||
| @@ -4,8 +4,8 @@ import ( | ||||
| 	"crypto/tls" | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/api/resolver" | ||||
| 	"github.com/micro/go-micro/v2/api/server/acme" | ||||
| 	"github.com/micro/go-micro/v3/api/resolver" | ||||
| 	"github.com/micro/go-micro/v3/api/server/acme" | ||||
| ) | ||||
|  | ||||
| type Option func(o *Options) | ||||
|   | ||||
| @@ -11,9 +11,9 @@ import ( | ||||
|  | ||||
| import ( | ||||
| 	context "context" | ||||
| 	api "github.com/micro/go-micro/v2/api" | ||||
| 	client "github.com/micro/go-micro/v2/client" | ||||
| 	server "github.com/micro/go-micro/v2/server" | ||||
| 	api "github.com/micro/go-micro/v3/api" | ||||
| 	client "github.com/micro/go-micro/v3/client" | ||||
| 	server "github.com/micro/go-micro/v3/server" | ||||
| ) | ||||
|  | ||||
| // Reference imports to suppress errors if they are not otherwise used. | ||||
|   | ||||
							
								
								
									
										103
									
								
								auth/auth.go
									
									
									
									
									
								
							
							
						
						
									
										103
									
								
								auth/auth.go
									
									
									
									
									
								
							| @@ -7,20 +7,23 @@ import ( | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	// BearerScheme used for Authorization header | ||||
| 	BearerScheme = "Bearer " | ||||
| 	// ScopePublic is the scope applied to a rule to allow access to the public | ||||
| 	ScopePublic = "" | ||||
| 	// ScopeAccount is the scope applied to a rule to limit to users with any valid account | ||||
| 	ScopeAccount = "*" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	// ErrNotFound is returned when a resouce cannot be found | ||||
| 	ErrNotFound = errors.New("not found") | ||||
| 	// ErrEncodingToken is returned when the service encounters an error during encoding | ||||
| 	ErrEncodingToken = errors.New("error encoding the token") | ||||
| 	// ErrInvalidToken is returned when the token provided is not valid | ||||
| 	// ErrInvalidToken is when the token provided is not valid | ||||
| 	ErrInvalidToken = errors.New("invalid token provided") | ||||
| 	// ErrInvalidRole is returned when the role provided was invalid | ||||
| 	ErrInvalidRole = errors.New("invalid role") | ||||
| 	// ErrForbidden is returned when a user does not have the necessary roles to access a resource | ||||
| 	// ErrForbidden is when a user does not have the necessary scope to access a resource | ||||
| 	ErrForbidden = errors.New("resource forbidden") | ||||
| ) | ||||
|  | ||||
| // Auth providers authentication and authorization | ||||
| // Auth provides authentication and authorization | ||||
| type Auth interface { | ||||
| 	// Init the auth | ||||
| 	Init(opts ...Option) | ||||
| @@ -28,46 +31,34 @@ type Auth interface { | ||||
| 	Options() Options | ||||
| 	// Generate a new account | ||||
| 	Generate(id string, opts ...GenerateOption) (*Account, error) | ||||
| 	// Grant access to a resource | ||||
| 	Grant(role string, res *Resource) error | ||||
| 	// Revoke access to a resource | ||||
| 	Revoke(role string, res *Resource) error | ||||
| 	// Verify an account has access to a resource | ||||
| 	Verify(acc *Account, res *Resource) error | ||||
| 	// Verify an account has access to a resource using the rules | ||||
| 	Verify(acc *Account, res *Resource, opts ...VerifyOption) error | ||||
| 	// Inspect a token | ||||
| 	Inspect(token string) (*Account, error) | ||||
| 	// Token generated using refresh token | ||||
| 	// Token generated using refresh token or credentials | ||||
| 	Token(opts ...TokenOption) (*Token, error) | ||||
| 	// Grant access to a resource | ||||
| 	Grant(rule *Rule) error | ||||
| 	// Revoke access to a resource | ||||
| 	Revoke(rule *Rule) error | ||||
| 	// Rules returns all the rules used to verify requests | ||||
| 	Rules(...RulesOption) ([]*Rule, error) | ||||
| 	// String returns the name of the implementation | ||||
| 	String() string | ||||
| } | ||||
|  | ||||
| // Resource is an entity such as a user or | ||||
| type Resource struct { | ||||
| 	// Name of the resource | ||||
| 	Name string `json:"name"` | ||||
| 	// Type of resource, e.g. | ||||
| 	Type string `json:"type"` | ||||
| 	// Endpoint resource e.g NotesService.Create | ||||
| 	Endpoint string `json:"endpoint"` | ||||
| 	// Namespace the resource belongs to | ||||
| 	Namespace string `json:"namespace"` | ||||
| } | ||||
|  | ||||
| // Account provided by an auth provider | ||||
| type Account struct { | ||||
| 	// ID of the account e.g. email | ||||
| 	ID string `json:"id"` | ||||
| 	// Type of the account, e.g. service | ||||
| 	Type string `json:"type"` | ||||
| 	// Provider who issued the account | ||||
| 	Provider string `json:"provider"` | ||||
| 	// Roles associated with the Account | ||||
| 	Roles []string `json:"roles"` | ||||
| 	// Issuer of the account | ||||
| 	Issuer string `json:"issuer"` | ||||
| 	// Any other associated metadata | ||||
| 	Metadata map[string]string `json:"metadata"` | ||||
| 	// Namespace the account belongs to | ||||
| 	Namespace string `json:"namespace"` | ||||
| 	// Scopes the account has access to | ||||
| 	Scopes []string `json:"scopes"` | ||||
| 	// Secret for the account, e.g. the password | ||||
| 	Secret string `json:"secret"` | ||||
| } | ||||
| @@ -84,15 +75,47 @@ type Token struct { | ||||
| 	Expiry time.Time `json:"expiry"` | ||||
| } | ||||
|  | ||||
| // Expired returns a boolean indicating if the token needs to be refreshed | ||||
| func (t *Token) Expired() bool { | ||||
| 	return t.Expiry.Unix() < time.Now().Unix() | ||||
| } | ||||
|  | ||||
| // Resource is an entity such as a user or | ||||
| type Resource struct { | ||||
| 	// Name of the resource, e.g. go.micro.service.notes | ||||
| 	Name string `json:"name"` | ||||
| 	// Type of resource, e.g. service | ||||
| 	Type string `json:"type"` | ||||
| 	// Endpoint resource e.g NotesService.Create | ||||
| 	Endpoint string `json:"endpoint"` | ||||
| } | ||||
|  | ||||
| // Access defines the type of access a rule grants | ||||
| type Access int | ||||
|  | ||||
| const ( | ||||
| 	// DefaultNamespace used for auth | ||||
| 	DefaultNamespace = "go.micro" | ||||
| 	// TokenCookieName is the name of the cookie which stores the auth token | ||||
| 	TokenCookieName = "micro-token" | ||||
| 	// BearerScheme used for Authorization header | ||||
| 	BearerScheme = "Bearer " | ||||
| 	// AccessGranted to a resource | ||||
| 	AccessGranted Access = iota | ||||
| 	// AccessDenied to a resource | ||||
| 	AccessDenied | ||||
| ) | ||||
|  | ||||
| // Rule is used to verify access to a resource | ||||
| type Rule struct { | ||||
| 	// ID of the rule, e.g. "public" | ||||
| 	ID string | ||||
| 	// Scope the rule requires, a blank scope indicates open to the public and * indicates the rule | ||||
| 	// applies to any valid account | ||||
| 	Scope string | ||||
| 	// Resource the rule applies to | ||||
| 	Resource *Resource | ||||
| 	// Access determines if the rule grants or denies access to the resource | ||||
| 	Access Access | ||||
| 	// Priority the rule should take when verifying a request, the higher the value the sooner the | ||||
| 	// rule will be applied | ||||
| 	Priority int32 | ||||
| } | ||||
|  | ||||
| type accountKey struct{} | ||||
|  | ||||
| // AccountFromContext gets the account from the context, which | ||||
|   | ||||
| @@ -1,86 +0,0 @@ | ||||
| package auth | ||||
|  | ||||
| import ( | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v2/auth/provider/basic" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	DefaultAuth = NewAuth() | ||||
| ) | ||||
|  | ||||
| func NewAuth(opts ...Option) Auth { | ||||
| 	options := Options{ | ||||
| 		Provider: basic.NewProvider(), | ||||
| 	} | ||||
|  | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	return &noop{ | ||||
| 		opts: options, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type noop struct { | ||||
| 	opts Options | ||||
| } | ||||
|  | ||||
| // String returns the name of the implementation | ||||
| func (n *noop) String() string { | ||||
| 	return "noop" | ||||
| } | ||||
|  | ||||
| // Init the auth | ||||
| func (n *noop) Init(opts ...Option) { | ||||
| 	for _, o := range opts { | ||||
| 		o(&n.opts) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Options set for auth | ||||
| func (n *noop) Options() Options { | ||||
| 	return n.opts | ||||
| } | ||||
|  | ||||
| // Generate a new account | ||||
| func (n *noop) Generate(id string, opts ...GenerateOption) (*Account, error) { | ||||
| 	options := NewGenerateOptions(opts...) | ||||
|  | ||||
| 	return &Account{ | ||||
| 		ID:        id, | ||||
| 		Roles:     options.Roles, | ||||
| 		Secret:    options.Secret, | ||||
| 		Metadata:  options.Metadata, | ||||
| 		Namespace: DefaultNamespace, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // Grant access to a resource | ||||
| func (n *noop) Grant(role string, res *Resource) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Revoke access to a resource | ||||
| func (n *noop) Revoke(role string, res *Resource) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Verify an account has access to a resource | ||||
| func (n *noop) Verify(acc *Account, res *Resource) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Inspect a token | ||||
| func (n *noop) Inspect(token string) (*Account, error) { | ||||
| 	return &Account{ | ||||
| 		ID:        uuid.New().String(), | ||||
| 		Namespace: DefaultNamespace, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // Token generation using an account id and secret | ||||
| func (n *noop) Token(opts ...TokenOption) (*Token, error) { | ||||
| 	return &Token{}, nil | ||||
| } | ||||
							
								
								
									
										151
									
								
								auth/jwt/jwt.go
									
									
									
									
									
								
							
							
						
						
									
										151
									
								
								auth/jwt/jwt.go
									
									
									
									
									
								
							| @@ -1,38 +1,35 @@ | ||||
| // Package jwt is a jwt implementation of the auth interface | ||||
| package jwt | ||||
|  | ||||
| import ( | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/auth" | ||||
| 	"github.com/micro/go-micro/v2/auth/token" | ||||
| 	jwtToken "github.com/micro/go-micro/v2/auth/token/jwt" | ||||
| 	"github.com/micro/go-micro/v3/auth" | ||||
| 	"github.com/micro/go-micro/v3/util/token" | ||||
| 	"github.com/micro/go-micro/v3/util/token/jwt" | ||||
| ) | ||||
|  | ||||
| // NewAuth returns a new instance of the Auth service | ||||
| func NewAuth(opts ...auth.Option) auth.Auth { | ||||
| 	j := new(jwt) | ||||
| 	j := new(jwtAuth) | ||||
| 	j.Init(opts...) | ||||
| 	return j | ||||
| } | ||||
|  | ||||
| type rule struct { | ||||
| 	role     string | ||||
| 	resource *auth.Resource | ||||
| } | ||||
|  | ||||
| type jwt struct { | ||||
| type jwtAuth struct { | ||||
| 	options auth.Options | ||||
| 	jwt     token.Provider | ||||
| 	rules   []*rule | ||||
| 	token   token.Provider | ||||
| 	rules   []*auth.Rule | ||||
|  | ||||
| 	sync.Mutex | ||||
| } | ||||
|  | ||||
| func (j *jwt) String() string { | ||||
| func (j *jwtAuth) String() string { | ||||
| 	return "jwt" | ||||
| } | ||||
|  | ||||
| func (j *jwt) Init(opts ...auth.Option) { | ||||
| func (j *jwtAuth) Init(opts ...auth.Option) { | ||||
| 	j.Lock() | ||||
| 	defer j.Unlock() | ||||
|  | ||||
| @@ -40,36 +37,35 @@ func (j *jwt) Init(opts ...auth.Option) { | ||||
| 		o(&j.options) | ||||
| 	} | ||||
|  | ||||
| 	if len(j.options.Namespace) == 0 { | ||||
| 		j.options.Namespace = auth.DefaultNamespace | ||||
| 	} | ||||
|  | ||||
| 	j.jwt = jwtToken.NewTokenProvider( | ||||
| 	j.token = jwt.NewTokenProvider( | ||||
| 		token.WithPrivateKey(j.options.PrivateKey), | ||||
| 		token.WithPublicKey(j.options.PublicKey), | ||||
| 	) | ||||
| } | ||||
|  | ||||
| func (j *jwt) Options() auth.Options { | ||||
| func (j *jwtAuth) Options() auth.Options { | ||||
| 	j.Lock() | ||||
| 	defer j.Unlock() | ||||
| 	return j.options | ||||
| } | ||||
|  | ||||
| func (j *jwt) Generate(id string, opts ...auth.GenerateOption) (*auth.Account, error) { | ||||
| func (j *jwtAuth) Generate(id string, opts ...auth.GenerateOption) (*auth.Account, error) { | ||||
| 	options := auth.NewGenerateOptions(opts...) | ||||
| 	if len(options.Issuer) == 0 { | ||||
| 		options.Issuer = j.Options().Issuer | ||||
| 	} | ||||
|  | ||||
| 	account := &auth.Account{ | ||||
| 		ID:        id, | ||||
| 		Type:      options.Type, | ||||
| 		Roles:     options.Roles, | ||||
| 		Provider:  options.Provider, | ||||
| 		Metadata:  options.Metadata, | ||||
| 		Namespace: options.Namespace, | ||||
| 		ID:       id, | ||||
| 		Type:     options.Type, | ||||
| 		Scopes:   options.Scopes, | ||||
| 		Metadata: options.Metadata, | ||||
| 		Issuer:   options.Issuer, | ||||
| 	} | ||||
|  | ||||
| 	// generate a JWT secret which can be provided to the Token() method | ||||
| 	// and exchanged for an access token | ||||
| 	secret, err := j.jwt.Generate(account) | ||||
| 	secret, err := j.token.Generate(account, token.WithExpiry(time.Hour*24*365)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @@ -79,91 +75,51 @@ func (j *jwt) Generate(id string, opts ...auth.GenerateOption) (*auth.Account, e | ||||
| 	return account, nil | ||||
| } | ||||
|  | ||||
| func (j *jwt) Grant(role string, res *auth.Resource) error { | ||||
| func (j *jwtAuth) Grant(rule *auth.Rule) error { | ||||
| 	j.Lock() | ||||
| 	defer j.Unlock() | ||||
| 	j.rules = append(j.rules, &rule{role, res}) | ||||
| 	j.rules = append(j.rules, rule) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (j *jwt) Revoke(role string, res *auth.Resource) error { | ||||
| func (j *jwtAuth) Revoke(rule *auth.Rule) error { | ||||
| 	j.Lock() | ||||
| 	defer j.Unlock() | ||||
|  | ||||
| 	rules := make([]*rule, 0, len(j.rules)) | ||||
|  | ||||
| 	var ruleFound bool | ||||
| 	for _, r := range rules { | ||||
| 		if r.role == role && r.resource == res { | ||||
| 			ruleFound = true | ||||
| 		} else { | ||||
| 	rules := []*auth.Rule{} | ||||
| 	for _, r := range j.rules { | ||||
| 		if r.ID != rule.ID { | ||||
| 			rules = append(rules, r) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if !ruleFound { | ||||
| 		return auth.ErrNotFound | ||||
| 	} | ||||
|  | ||||
| 	j.rules = rules | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (j *jwt) Verify(acc *auth.Account, res *auth.Resource) error { | ||||
| func (j *jwtAuth) Verify(acc *auth.Account, res *auth.Resource, opts ...auth.VerifyOption) error { | ||||
| 	j.Lock() | ||||
| 	if len(res.Namespace) == 0 { | ||||
| 		res.Namespace = j.options.Namespace | ||||
| 	} | ||||
| 	rules := j.rules | ||||
| 	j.Unlock() | ||||
| 	defer j.Unlock() | ||||
|  | ||||
| 	for _, rule := range rules { | ||||
| 		// validate the rule applies to the requested resource | ||||
| 		if rule.resource.Namespace != "*" && rule.resource.Namespace != res.Namespace { | ||||
| 			continue | ||||
| 		} | ||||
| 		if rule.resource.Type != "*" && rule.resource.Type != res.Type { | ||||
| 			continue | ||||
| 		} | ||||
| 		if rule.resource.Name != "*" && rule.resource.Name != res.Name { | ||||
| 			continue | ||||
| 		} | ||||
| 		if rule.resource.Endpoint != "*" && rule.resource.Endpoint != res.Endpoint { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// a blank role indicates anyone can access the resource, even without an account | ||||
| 		if rule.role == "" { | ||||
| 			return nil | ||||
| 		} | ||||
|  | ||||
| 		// all furter checks require an account | ||||
| 		if acc == nil { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// this rule allows any account access, allow the request | ||||
| 		if rule.role == "*" { | ||||
| 			return nil | ||||
| 		} | ||||
|  | ||||
| 		// if the account has the necessary role, allow the request | ||||
| 		for _, r := range acc.Roles { | ||||
| 			if r == rule.role { | ||||
| 				return nil | ||||
| 			} | ||||
| 		} | ||||
| 	var options auth.VerifyOptions | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	// no rules matched, forbid the request | ||||
| 	return auth.ErrForbidden | ||||
| 	return auth.VerifyAccess(j.rules, acc, res) | ||||
| } | ||||
|  | ||||
| func (j *jwt) Inspect(token string) (*auth.Account, error) { | ||||
| 	return j.jwt.Inspect(token) | ||||
| func (j *jwtAuth) Rules(opts ...auth.RulesOption) ([]*auth.Rule, error) { | ||||
| 	j.Lock() | ||||
| 	defer j.Unlock() | ||||
| 	return j.rules, nil | ||||
| } | ||||
|  | ||||
| func (j *jwt) Token(opts ...auth.TokenOption) (*auth.Token, error) { | ||||
| func (j *jwtAuth) Inspect(token string) (*auth.Account, error) { | ||||
| 	return j.token.Inspect(token) | ||||
| } | ||||
|  | ||||
| func (j *jwtAuth) Token(opts ...auth.TokenOption) (*auth.Token, error) { | ||||
| 	options := auth.NewTokenOptions(opts...) | ||||
|  | ||||
| 	secret := options.RefreshToken | ||||
| @@ -171,20 +127,25 @@ func (j *jwt) Token(opts ...auth.TokenOption) (*auth.Token, error) { | ||||
| 		secret = options.Secret | ||||
| 	} | ||||
|  | ||||
| 	account, err := j.jwt.Inspect(secret) | ||||
| 	account, err := j.token.Inspect(secret) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	tok, err := j.jwt.Generate(account, token.WithExpiry(options.Expiry)) | ||||
| 	access, err := j.token.Generate(account, token.WithExpiry(options.Expiry)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	refresh, err := j.token.Generate(account, token.WithExpiry(options.Expiry+time.Hour)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &auth.Token{ | ||||
| 		Created:      tok.Created, | ||||
| 		Expiry:       tok.Expiry, | ||||
| 		AccessToken:  tok.Token, | ||||
| 		RefreshToken: tok.Token, | ||||
| 		Created:      access.Created, | ||||
| 		Expiry:       access.Expiry, | ||||
| 		AccessToken:  access.Token, | ||||
| 		RefreshToken: refresh.Token, | ||||
| 	}, nil | ||||
| } | ||||
|   | ||||
							
								
								
									
										81
									
								
								auth/noop/noop.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								auth/noop/noop.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| package noop | ||||
|  | ||||
| import ( | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v3/auth" | ||||
| ) | ||||
|  | ||||
| func NewAuth(opts ...auth.Option) auth.Auth { | ||||
| 	var options auth.Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	return &noop{ | ||||
| 		opts: options, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type noop struct { | ||||
| 	opts auth.Options | ||||
| } | ||||
|  | ||||
| // String returns the name of the implementation | ||||
| func (n *noop) String() string { | ||||
| 	return "noop" | ||||
| } | ||||
|  | ||||
| // Init the auth | ||||
| func (n *noop) Init(opts ...auth.Option) { | ||||
| 	for _, o := range opts { | ||||
| 		o(&n.opts) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Options set for auth | ||||
| func (n *noop) Options() auth.Options { | ||||
| 	return n.opts | ||||
| } | ||||
|  | ||||
| // Generate a new account | ||||
| func (n *noop) Generate(id string, opts ...auth.GenerateOption) (*auth.Account, error) { | ||||
| 	options := auth.NewGenerateOptions(opts...) | ||||
|  | ||||
| 	return &auth.Account{ | ||||
| 		ID:       id, | ||||
| 		Secret:   options.Secret, | ||||
| 		Metadata: options.Metadata, | ||||
| 		Scopes:   options.Scopes, | ||||
| 		Issuer:   n.Options().Issuer, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // Grant access to a resource | ||||
| func (n *noop) Grant(rule *auth.Rule) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Revoke access to a resource | ||||
| func (n *noop) Revoke(rule *auth.Rule) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Rules used to verify requests | ||||
| func (n *noop) Rules(opts ...auth.RulesOption) ([]*auth.Rule, error) { | ||||
| 	return []*auth.Rule{}, nil | ||||
| } | ||||
|  | ||||
| // Verify an account has access to a resource | ||||
| func (n *noop) Verify(acc *auth.Account, res *auth.Resource, opts ...auth.VerifyOption) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Inspect a token | ||||
| func (n *noop) Inspect(token string) (*auth.Account, error) { | ||||
| 	return &auth.Account{ID: uuid.New().String(), Issuer: n.Options().Issuer}, nil | ||||
| } | ||||
|  | ||||
| // Token generation using an account id and secret | ||||
| func (n *noop) Token(opts ...auth.TokenOption) (*auth.Token, error) { | ||||
| 	return &auth.Token{}, nil | ||||
| } | ||||
							
								
								
									
										120
									
								
								auth/options.go
									
									
									
									
									
								
							
							
						
						
									
										120
									
								
								auth/options.go
									
									
									
									
									
								
							| @@ -1,10 +1,10 @@ | ||||
| package auth | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/auth/provider" | ||||
| 	"github.com/micro/go-micro/v2/store" | ||||
| 	"github.com/micro/go-micro/v3/store" | ||||
| ) | ||||
|  | ||||
| func NewOptions(opts ...Option) Options { | ||||
| @@ -12,17 +12,12 @@ func NewOptions(opts ...Option) Options { | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	if len(options.Namespace) == 0 { | ||||
| 		options.Namespace = DefaultNamespace | ||||
| 	} | ||||
|  | ||||
| 	return options | ||||
| } | ||||
|  | ||||
| type Options struct { | ||||
| 	// Namespace the service belongs to | ||||
| 	Namespace string | ||||
| 	// Issuer of the service's account | ||||
| 	Issuer string | ||||
| 	// ID is the services auth ID | ||||
| 	ID string | ||||
| 	// Secret is used to authenticate the service | ||||
| @@ -33,20 +28,29 @@ type Options struct { | ||||
| 	PublicKey string | ||||
| 	// PrivateKey for encoding JWTs | ||||
| 	PrivateKey string | ||||
| 	// Provider is an auth provider | ||||
| 	Provider provider.Provider | ||||
| 	// LoginURL is the relative url path where a user can login | ||||
| 	LoginURL string | ||||
| 	// Store to back auth | ||||
| 	Store store.Store | ||||
| 	// Addrs sets the addresses of auth | ||||
| 	Addrs []string | ||||
| 	// Context to store other options | ||||
| 	Context context.Context | ||||
| } | ||||
|  | ||||
| type Option func(o *Options) | ||||
|  | ||||
| // Namespace the service belongs to | ||||
| func Namespace(n string) Option { | ||||
| // Addrs is the auth addresses to use | ||||
| func Addrs(addrs ...string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Namespace = n | ||||
| 		o.Addrs = addrs | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Issuer of the services account | ||||
| func Issuer(i string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Issuer = i | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -86,13 +90,6 @@ func ClientToken(token *Token) Option { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Provider set the auth provider | ||||
| func Provider(p provider.Provider) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Provider = p | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // LoginURL sets the auth LoginURL | ||||
| func LoginURL(url string) Option { | ||||
| 	return func(o *Options) { | ||||
| @@ -103,16 +100,16 @@ func LoginURL(url string) Option { | ||||
| type GenerateOptions struct { | ||||
| 	// Metadata associated with the account | ||||
| 	Metadata map[string]string | ||||
| 	// Roles/scopes associated with the account | ||||
| 	Roles []string | ||||
| 	// Namespace the account belongs too | ||||
| 	Namespace string | ||||
| 	// Scopes the account has access too | ||||
| 	Scopes []string | ||||
| 	// Provider of the account, e.g. oauth | ||||
| 	Provider string | ||||
| 	// Type of the account, e.g. user | ||||
| 	Type string | ||||
| 	// Secret used to authenticate the account | ||||
| 	Secret string | ||||
| 	// Issuer of the account, e.g. micro | ||||
| 	Issuer string | ||||
| } | ||||
|  | ||||
| type GenerateOption func(o *GenerateOptions) | ||||
| @@ -138,20 +135,6 @@ func WithMetadata(md map[string]string) GenerateOption { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithRoles for the generated account | ||||
| func WithRoles(rs ...string) GenerateOption { | ||||
| 	return func(o *GenerateOptions) { | ||||
| 		o.Roles = rs | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithNamespace for the generated account | ||||
| func WithNamespace(n string) GenerateOption { | ||||
| 	return func(o *GenerateOptions) { | ||||
| 		o.Namespace = n | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithProvider for the generated account | ||||
| func WithProvider(p string) GenerateOption { | ||||
| 	return func(o *GenerateOptions) { | ||||
| @@ -159,6 +142,20 @@ func WithProvider(p string) GenerateOption { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithScopes for the generated account | ||||
| func WithScopes(s ...string) GenerateOption { | ||||
| 	return func(o *GenerateOptions) { | ||||
| 		o.Scopes = s | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // WithIssuer for the generated account | ||||
| func WithIssuer(i string) GenerateOption { | ||||
| 	return func(o *GenerateOptions) { | ||||
| 		o.Issuer = i | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // NewGenerateOptions from a slice of options | ||||
| func NewGenerateOptions(opts ...GenerateOption) GenerateOptions { | ||||
| 	var options GenerateOptions | ||||
| @@ -177,6 +174,8 @@ type TokenOptions struct { | ||||
| 	RefreshToken string | ||||
| 	// Expiry is the time the token should live for | ||||
| 	Expiry time.Duration | ||||
| 	// Issuer of the account | ||||
| 	Issuer string | ||||
| } | ||||
|  | ||||
| type TokenOption func(o *TokenOptions) | ||||
| @@ -201,6 +200,12 @@ func WithToken(rt string) TokenOption { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func WithTokenIssuer(iss string) TokenOption { | ||||
| 	return func(o *TokenOptions) { | ||||
| 		o.Issuer = iss | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // NewTokenOptions from a slice of options | ||||
| func NewTokenOptions(opts ...TokenOption) TokenOptions { | ||||
| 	var options TokenOptions | ||||
| @@ -215,3 +220,40 @@ func NewTokenOptions(opts ...TokenOption) TokenOptions { | ||||
|  | ||||
| 	return options | ||||
| } | ||||
|  | ||||
| type VerifyOptions struct { | ||||
| 	Context   context.Context | ||||
| 	Namespace string | ||||
| } | ||||
|  | ||||
| type VerifyOption func(o *VerifyOptions) | ||||
|  | ||||
| func VerifyContext(ctx context.Context) VerifyOption { | ||||
| 	return func(o *VerifyOptions) { | ||||
| 		o.Context = ctx | ||||
| 	} | ||||
| } | ||||
| func VerifyNamespace(ns string) VerifyOption { | ||||
| 	return func(o *VerifyOptions) { | ||||
| 		o.Namespace = ns | ||||
| 	} | ||||
| } | ||||
|  | ||||
| type RulesOptions struct { | ||||
| 	Context   context.Context | ||||
| 	Namespace string | ||||
| } | ||||
|  | ||||
| type RulesOption func(o *RulesOptions) | ||||
|  | ||||
| func RulesContext(ctx context.Context) RulesOption { | ||||
| 	return func(o *RulesOptions) { | ||||
| 		o.Context = ctx | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func RulesNamespace(ns string) RulesOption { | ||||
| 	return func(o *RulesOptions) { | ||||
| 		o.Namespace = ns | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,34 +0,0 @@ | ||||
| package basic | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/auth/provider" | ||||
| ) | ||||
|  | ||||
| // NewProvider returns an initialised basic provider | ||||
| func NewProvider(opts ...provider.Option) provider.Provider { | ||||
| 	var options provider.Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
| 	return &basic{options} | ||||
| } | ||||
|  | ||||
| type basic struct { | ||||
| 	opts provider.Options | ||||
| } | ||||
|  | ||||
| func (b *basic) String() string { | ||||
| 	return "basic" | ||||
| } | ||||
|  | ||||
| func (b *basic) Options() provider.Options { | ||||
| 	return b.opts | ||||
| } | ||||
|  | ||||
| func (b *basic) Endpoint(...provider.EndpointOption) string { | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (b *basic) Redirect() string { | ||||
| 	return "" | ||||
| } | ||||
| @@ -1,65 +0,0 @@ | ||||
| package oauth | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net/url" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/auth/provider" | ||||
| ) | ||||
|  | ||||
| // NewProvider returns an initialised oauth provider | ||||
| func NewProvider(opts ...provider.Option) provider.Provider { | ||||
| 	var options provider.Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
| 	return &oauth{options} | ||||
| } | ||||
|  | ||||
| type oauth struct { | ||||
| 	opts provider.Options | ||||
| } | ||||
|  | ||||
| func (o *oauth) String() string { | ||||
| 	return "oauth" | ||||
| } | ||||
|  | ||||
| func (o *oauth) Options() provider.Options { | ||||
| 	return o.opts | ||||
| } | ||||
|  | ||||
| func (o *oauth) Endpoint(opts ...provider.EndpointOption) string { | ||||
| 	var options provider.EndpointOptions | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	params := make(url.Values) | ||||
| 	params.Add("response_type", "code") | ||||
|  | ||||
| 	if len(options.State) > 0 { | ||||
| 		params.Add("state", options.State) | ||||
| 	} | ||||
|  | ||||
| 	if len(options.LoginHint) > 0 { | ||||
| 		params.Add("login_hint", options.LoginHint) | ||||
| 	} | ||||
|  | ||||
| 	if clientID := o.opts.ClientID; len(clientID) > 0 { | ||||
| 		params.Add("client_id", clientID) | ||||
| 	} | ||||
|  | ||||
| 	if scope := o.opts.Scope; len(scope) > 0 { | ||||
| 		params.Add("scope", scope) | ||||
| 	} | ||||
|  | ||||
| 	if redir := o.Redirect(); len(redir) > 0 { | ||||
| 		params.Add("redirect_uri", redir) | ||||
| 	} | ||||
|  | ||||
| 	return fmt.Sprintf("%v?%v", o.opts.Endpoint, params.Encode()) | ||||
| } | ||||
|  | ||||
| func (o *oauth) Redirect() string { | ||||
| 	return o.opts.Redirect | ||||
| } | ||||
| @@ -1,47 +0,0 @@ | ||||
| package provider | ||||
|  | ||||
| // Option returns a function which sets an option | ||||
| type Option func(*Options) | ||||
|  | ||||
| // Options a provider can have | ||||
| type Options struct { | ||||
| 	// ClientID is the application's ID. | ||||
| 	ClientID string | ||||
| 	// ClientSecret is the application's secret. | ||||
| 	ClientSecret string | ||||
| 	// Endpoint for the provider | ||||
| 	Endpoint string | ||||
| 	// Redirect url incase of UI | ||||
| 	Redirect string | ||||
| 	// Scope of the oauth request | ||||
| 	Scope string | ||||
| } | ||||
|  | ||||
| // Credentials is an option which sets the client id and secret | ||||
| func Credentials(id, secret string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.ClientID = id | ||||
| 		o.ClientSecret = secret | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Endpoint sets the endpoint option | ||||
| func Endpoint(e string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Endpoint = e | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Redirect sets the Redirect option | ||||
| func Redirect(r string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Redirect = r | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Scope sets the oauth scope | ||||
| func Scope(s string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Scope = s | ||||
| 	} | ||||
| } | ||||
| @@ -1,49 +0,0 @@ | ||||
| // Package provider is an external auth provider e.g oauth | ||||
| package provider | ||||
|  | ||||
| import ( | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| // Provider is an auth provider | ||||
| type Provider interface { | ||||
| 	// String returns the name of the provider | ||||
| 	String() string | ||||
| 	// Options returns the options of a provider | ||||
| 	Options() Options | ||||
| 	// Endpoint for the provider | ||||
| 	Endpoint(...EndpointOption) string | ||||
| 	// Redirect url incase of UI | ||||
| 	Redirect() string | ||||
| } | ||||
|  | ||||
| // Grant is a granted authorisation | ||||
| type Grant struct { | ||||
| 	// token for reuse | ||||
| 	Token string | ||||
| 	// Expiry of the token | ||||
| 	Expiry time.Time | ||||
| 	// Scopes associated with grant | ||||
| 	Scopes []string | ||||
| } | ||||
|  | ||||
| type EndpointOptions struct { | ||||
| 	// State is a code to verify the req | ||||
| 	State string | ||||
| 	// LoginHint prefils the user id on oauth clients | ||||
| 	LoginHint string | ||||
| } | ||||
|  | ||||
| type EndpointOption func(*EndpointOptions) | ||||
|  | ||||
| func WithState(c string) EndpointOption { | ||||
| 	return func(o *EndpointOptions) { | ||||
| 		o.State = c | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func WithLoginHint(hint string) EndpointOption { | ||||
| 	return func(o *EndpointOptions) { | ||||
| 		o.LoginHint = hint | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										91
									
								
								auth/rules.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								auth/rules.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| package auth | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // VerifyAccess an account has access to a resource using the rules provided. If the account does not have | ||||
| // access an error will be returned. If there are no rules provided which match the resource, an error | ||||
| // will be returned | ||||
| func VerifyAccess(rules []*Rule, acc *Account, res *Resource) error { | ||||
| 	// the rule is only to be applied if the type matches the resource or is catch-all (*) | ||||
| 	validTypes := []string{"*", res.Type} | ||||
|  | ||||
| 	// the rule is only to be applied if the name matches the resource or is catch-all (*) | ||||
| 	validNames := []string{"*", res.Name} | ||||
|  | ||||
| 	// rules can have wildcard excludes on endpoints since this can also be a path for web services, | ||||
| 	// e.g. /foo/* would include /foo/bar. We also want to check for wildcards and the exact endpoint | ||||
| 	validEndpoints := []string{"*", res.Endpoint} | ||||
| 	if comps := strings.Split(res.Endpoint, "/"); len(comps) > 1 { | ||||
| 		for i := 1; i < len(comps)+1; i++ { | ||||
| 			wildcard := fmt.Sprintf("%v/*", strings.Join(comps[0:i], "/")) | ||||
| 			validEndpoints = append(validEndpoints, wildcard) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// filter the rules to the ones which match the criteria above | ||||
| 	filteredRules := make([]*Rule, 0) | ||||
| 	for _, rule := range rules { | ||||
| 		if !include(validTypes, rule.Resource.Type) { | ||||
| 			continue | ||||
| 		} | ||||
| 		if !include(validNames, rule.Resource.Name) { | ||||
| 			continue | ||||
| 		} | ||||
| 		if !include(validEndpoints, rule.Resource.Endpoint) { | ||||
| 			continue | ||||
| 		} | ||||
| 		filteredRules = append(filteredRules, rule) | ||||
| 	} | ||||
|  | ||||
| 	// sort the filtered rules by priority, highest to lowest | ||||
| 	sort.SliceStable(filteredRules, func(i, j int) bool { | ||||
| 		return filteredRules[i].Priority > filteredRules[j].Priority | ||||
| 	}) | ||||
|  | ||||
| 	// loop through the rules and check for a rule which applies to this account | ||||
| 	for _, rule := range filteredRules { | ||||
| 		// a blank scope indicates the rule applies to everyone, even nil accounts | ||||
| 		if rule.Scope == ScopePublic && rule.Access == AccessDenied { | ||||
| 			return ErrForbidden | ||||
| 		} else if rule.Scope == ScopePublic && rule.Access == AccessGranted { | ||||
| 			return nil | ||||
| 		} | ||||
|  | ||||
| 		// all further checks require an account | ||||
| 		if acc == nil { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		// this rule applies to any account | ||||
| 		if rule.Scope == ScopeAccount && rule.Access == AccessDenied { | ||||
| 			return ErrForbidden | ||||
| 		} else if rule.Scope == ScopeAccount && rule.Access == AccessGranted { | ||||
| 			return nil | ||||
| 		} | ||||
|  | ||||
| 		// if the account has the necessary scope | ||||
| 		if include(acc.Scopes, rule.Scope) && rule.Access == AccessDenied { | ||||
| 			return ErrForbidden | ||||
| 		} else if include(acc.Scopes, rule.Scope) && rule.Access == AccessGranted { | ||||
| 			return nil | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// if no rules matched then return forbidden | ||||
| 	return ErrForbidden | ||||
| } | ||||
|  | ||||
| // include is a helper function which checks to see if the slice contains the value. includes is | ||||
| // not case sensitive. | ||||
| func include(slice []string, val string) bool { | ||||
| 	for _, s := range slice { | ||||
| 		if strings.ToLower(s) == strings.ToLower(val) { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
							
								
								
									
										288
									
								
								auth/rules_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										288
									
								
								auth/rules_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,288 @@ | ||||
| package auth | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestVerify(t *testing.T) { | ||||
| 	srvResource := &Resource{ | ||||
| 		Type:     "service", | ||||
| 		Name:     "go.micro.service.foo", | ||||
| 		Endpoint: "Foo.Bar", | ||||
| 	} | ||||
|  | ||||
| 	webResource := &Resource{ | ||||
| 		Type:     "service", | ||||
| 		Name:     "go.micro.web.foo", | ||||
| 		Endpoint: "/foo/bar", | ||||
| 	} | ||||
|  | ||||
| 	catchallResource := &Resource{ | ||||
| 		Type:     "*", | ||||
| 		Name:     "*", | ||||
| 		Endpoint: "*", | ||||
| 	} | ||||
|  | ||||
| 	tt := []struct { | ||||
| 		Name     string | ||||
| 		Rules    []*Rule | ||||
| 		Account  *Account | ||||
| 		Resource *Resource | ||||
| 		Error    error | ||||
| 	}{ | ||||
| 		{ | ||||
| 			Name:     "NoRules", | ||||
| 			Rules:    []*Rule{}, | ||||
| 			Account:  nil, | ||||
| 			Resource: srvResource, | ||||
| 			Error:    ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallPublicAccount", | ||||
| 			Account:  &Account{}, | ||||
| 			Resource: srvResource, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "", | ||||
| 					Resource: catchallResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallPublicNoAccount", | ||||
| 			Resource: srvResource, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "", | ||||
| 					Resource: catchallResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallPrivateAccount", | ||||
| 			Account:  &Account{}, | ||||
| 			Resource: srvResource, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallPrivateNoAccount", | ||||
| 			Resource: srvResource, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallServiceRuleMatch", | ||||
| 			Resource: srvResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope: "*", | ||||
| 					Resource: &Resource{ | ||||
| 						Type:     srvResource.Type, | ||||
| 						Name:     srvResource.Name, | ||||
| 						Endpoint: "*", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallServiceRuleNoMatch", | ||||
| 			Resource: srvResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope: "*", | ||||
| 					Resource: &Resource{ | ||||
| 						Type:     srvResource.Type, | ||||
| 						Name:     "wrongname", | ||||
| 						Endpoint: "*", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "ExactRuleValidScope", | ||||
| 			Resource: srvResource, | ||||
| 			Account: &Account{ | ||||
| 				Scopes: []string{"neededscope"}, | ||||
| 			}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "neededscope", | ||||
| 					Resource: srvResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "ExactRuleInvalidScope", | ||||
| 			Resource: srvResource, | ||||
| 			Account: &Account{ | ||||
| 				Scopes: []string{"neededscope"}, | ||||
| 			}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "invalidscope", | ||||
| 					Resource: srvResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallDenyWithAccount", | ||||
| 			Resource: srvResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 					Access:   AccessDenied, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "CatchallDenyWithNoAccount", | ||||
| 			Resource: srvResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 					Access:   AccessDenied, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "RulePriorityGrantFirst", | ||||
| 			Resource: srvResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 					Access:   AccessGranted, | ||||
| 					Priority: 1, | ||||
| 				}, | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 					Access:   AccessDenied, | ||||
| 					Priority: 0, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "RulePriorityDenyFirst", | ||||
| 			Resource: srvResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 					Access:   AccessGranted, | ||||
| 					Priority: 0, | ||||
| 				}, | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: catchallResource, | ||||
| 					Access:   AccessDenied, | ||||
| 					Priority: 1, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "WebExactEndpointValid", | ||||
| 			Resource: webResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope:    "*", | ||||
| 					Resource: webResource, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "WebExactEndpointInalid", | ||||
| 			Resource: webResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope: "*", | ||||
| 					Resource: &Resource{ | ||||
| 						Type:     webResource.Type, | ||||
| 						Name:     webResource.Name, | ||||
| 						Endpoint: "invalidendpoint", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "WebWildcardEndpoint", | ||||
| 			Resource: webResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope: "*", | ||||
| 					Resource: &Resource{ | ||||
| 						Type:     webResource.Type, | ||||
| 						Name:     webResource.Name, | ||||
| 						Endpoint: "*", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "WebWildcardPathEndpointValid", | ||||
| 			Resource: webResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope: "*", | ||||
| 					Resource: &Resource{ | ||||
| 						Type:     webResource.Type, | ||||
| 						Name:     webResource.Name, | ||||
| 						Endpoint: "/foo/*", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			Name:     "WebWildcardPathEndpointInvalid", | ||||
| 			Resource: webResource, | ||||
| 			Account:  &Account{}, | ||||
| 			Rules: []*Rule{ | ||||
| 				&Rule{ | ||||
| 					Scope: "*", | ||||
| 					Resource: &Resource{ | ||||
| 						Type:     webResource.Type, | ||||
| 						Name:     webResource.Name, | ||||
| 						Endpoint: "/bar/*", | ||||
| 					}, | ||||
| 				}, | ||||
| 			}, | ||||
| 			Error: ErrForbidden, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range tt { | ||||
| 		t.Run(tc.Name, func(t *testing.T) { | ||||
| 			if err := VerifyAccess(tc.Rules, tc.Account, tc.Resource); err != tc.Error { | ||||
| 				t.Errorf("Expected %v but got %v", tc.Error, err) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,279 +0,0 @@ | ||||
| // Code generated by protoc-gen-micro. DO NOT EDIT. | ||||
| // source: auth/service/proto/auth.proto | ||||
|  | ||||
| package go_micro_auth | ||||
|  | ||||
| import ( | ||||
| 	fmt "fmt" | ||||
| 	proto "github.com/golang/protobuf/proto" | ||||
| 	math "math" | ||||
| ) | ||||
|  | ||||
| import ( | ||||
| 	context "context" | ||||
| 	api "github.com/micro/go-micro/v2/api" | ||||
| 	client "github.com/micro/go-micro/v2/client" | ||||
| 	server "github.com/micro/go-micro/v2/server" | ||||
| ) | ||||
|  | ||||
| // 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.ProtoPackageIsVersion3 // please upgrade the proto package | ||||
|  | ||||
| // Reference imports to suppress errors if they are not otherwise used. | ||||
| var _ api.Endpoint | ||||
| var _ context.Context | ||||
| var _ client.Option | ||||
| var _ server.Option | ||||
|  | ||||
| // Api Endpoints for Auth service | ||||
|  | ||||
| func NewAuthEndpoints() []*api.Endpoint { | ||||
| 	return []*api.Endpoint{} | ||||
| } | ||||
|  | ||||
| // Client API for Auth service | ||||
|  | ||||
| type AuthService interface { | ||||
| 	Generate(ctx context.Context, in *GenerateRequest, opts ...client.CallOption) (*GenerateResponse, error) | ||||
| 	Inspect(ctx context.Context, in *InspectRequest, opts ...client.CallOption) (*InspectResponse, error) | ||||
| 	Token(ctx context.Context, in *TokenRequest, opts ...client.CallOption) (*TokenResponse, error) | ||||
| } | ||||
|  | ||||
| type authService struct { | ||||
| 	c    client.Client | ||||
| 	name string | ||||
| } | ||||
|  | ||||
| func NewAuthService(name string, c client.Client) AuthService { | ||||
| 	return &authService{ | ||||
| 		c:    c, | ||||
| 		name: name, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *authService) Generate(ctx context.Context, in *GenerateRequest, opts ...client.CallOption) (*GenerateResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Auth.Generate", in) | ||||
| 	out := new(GenerateResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *authService) Inspect(ctx context.Context, in *InspectRequest, opts ...client.CallOption) (*InspectResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Auth.Inspect", in) | ||||
| 	out := new(InspectResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *authService) Token(ctx context.Context, in *TokenRequest, opts ...client.CallOption) (*TokenResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Auth.Token", in) | ||||
| 	out := new(TokenResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| // Server API for Auth service | ||||
|  | ||||
| type AuthHandler interface { | ||||
| 	Generate(context.Context, *GenerateRequest, *GenerateResponse) error | ||||
| 	Inspect(context.Context, *InspectRequest, *InspectResponse) error | ||||
| 	Token(context.Context, *TokenRequest, *TokenResponse) error | ||||
| } | ||||
|  | ||||
| func RegisterAuthHandler(s server.Server, hdlr AuthHandler, opts ...server.HandlerOption) error { | ||||
| 	type auth interface { | ||||
| 		Generate(ctx context.Context, in *GenerateRequest, out *GenerateResponse) error | ||||
| 		Inspect(ctx context.Context, in *InspectRequest, out *InspectResponse) error | ||||
| 		Token(ctx context.Context, in *TokenRequest, out *TokenResponse) error | ||||
| 	} | ||||
| 	type Auth struct { | ||||
| 		auth | ||||
| 	} | ||||
| 	h := &authHandler{hdlr} | ||||
| 	return s.Handle(s.NewHandler(&Auth{h}, opts...)) | ||||
| } | ||||
|  | ||||
| type authHandler struct { | ||||
| 	AuthHandler | ||||
| } | ||||
|  | ||||
| func (h *authHandler) Generate(ctx context.Context, in *GenerateRequest, out *GenerateResponse) error { | ||||
| 	return h.AuthHandler.Generate(ctx, in, out) | ||||
| } | ||||
|  | ||||
| func (h *authHandler) Inspect(ctx context.Context, in *InspectRequest, out *InspectResponse) error { | ||||
| 	return h.AuthHandler.Inspect(ctx, in, out) | ||||
| } | ||||
|  | ||||
| func (h *authHandler) Token(ctx context.Context, in *TokenRequest, out *TokenResponse) error { | ||||
| 	return h.AuthHandler.Token(ctx, in, out) | ||||
| } | ||||
|  | ||||
| // Api Endpoints for Accounts service | ||||
|  | ||||
| func NewAccountsEndpoints() []*api.Endpoint { | ||||
| 	return []*api.Endpoint{} | ||||
| } | ||||
|  | ||||
| // Client API for Accounts service | ||||
|  | ||||
| type AccountsService interface { | ||||
| 	List(ctx context.Context, in *ListAccountsRequest, opts ...client.CallOption) (*ListAccountsResponse, error) | ||||
| } | ||||
|  | ||||
| type accountsService struct { | ||||
| 	c    client.Client | ||||
| 	name string | ||||
| } | ||||
|  | ||||
| func NewAccountsService(name string, c client.Client) AccountsService { | ||||
| 	return &accountsService{ | ||||
| 		c:    c, | ||||
| 		name: name, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *accountsService) List(ctx context.Context, in *ListAccountsRequest, opts ...client.CallOption) (*ListAccountsResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Accounts.List", in) | ||||
| 	out := new(ListAccountsResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| // Server API for Accounts service | ||||
|  | ||||
| type AccountsHandler interface { | ||||
| 	List(context.Context, *ListAccountsRequest, *ListAccountsResponse) error | ||||
| } | ||||
|  | ||||
| func RegisterAccountsHandler(s server.Server, hdlr AccountsHandler, opts ...server.HandlerOption) error { | ||||
| 	type accounts interface { | ||||
| 		List(ctx context.Context, in *ListAccountsRequest, out *ListAccountsResponse) error | ||||
| 	} | ||||
| 	type Accounts struct { | ||||
| 		accounts | ||||
| 	} | ||||
| 	h := &accountsHandler{hdlr} | ||||
| 	return s.Handle(s.NewHandler(&Accounts{h}, opts...)) | ||||
| } | ||||
|  | ||||
| type accountsHandler struct { | ||||
| 	AccountsHandler | ||||
| } | ||||
|  | ||||
| func (h *accountsHandler) List(ctx context.Context, in *ListAccountsRequest, out *ListAccountsResponse) error { | ||||
| 	return h.AccountsHandler.List(ctx, in, out) | ||||
| } | ||||
|  | ||||
| // Api Endpoints for Rules service | ||||
|  | ||||
| func NewRulesEndpoints() []*api.Endpoint { | ||||
| 	return []*api.Endpoint{} | ||||
| } | ||||
|  | ||||
| // Client API for Rules service | ||||
|  | ||||
| type RulesService interface { | ||||
| 	Create(ctx context.Context, in *CreateRequest, opts ...client.CallOption) (*CreateResponse, error) | ||||
| 	Delete(ctx context.Context, in *DeleteRequest, opts ...client.CallOption) (*DeleteResponse, error) | ||||
| 	List(ctx context.Context, in *ListRequest, opts ...client.CallOption) (*ListResponse, error) | ||||
| } | ||||
|  | ||||
| type rulesService struct { | ||||
| 	c    client.Client | ||||
| 	name string | ||||
| } | ||||
|  | ||||
| func NewRulesService(name string, c client.Client) RulesService { | ||||
| 	return &rulesService{ | ||||
| 		c:    c, | ||||
| 		name: name, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *rulesService) Create(ctx context.Context, in *CreateRequest, opts ...client.CallOption) (*CreateResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Rules.Create", in) | ||||
| 	out := new(CreateResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *rulesService) Delete(ctx context.Context, in *DeleteRequest, opts ...client.CallOption) (*DeleteResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Rules.Delete", in) | ||||
| 	out := new(DeleteResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *rulesService) List(ctx context.Context, in *ListRequest, opts ...client.CallOption) (*ListResponse, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Rules.List", in) | ||||
| 	out := new(ListResponse) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| // Server API for Rules service | ||||
|  | ||||
| type RulesHandler interface { | ||||
| 	Create(context.Context, *CreateRequest, *CreateResponse) error | ||||
| 	Delete(context.Context, *DeleteRequest, *DeleteResponse) error | ||||
| 	List(context.Context, *ListRequest, *ListResponse) error | ||||
| } | ||||
|  | ||||
| func RegisterRulesHandler(s server.Server, hdlr RulesHandler, opts ...server.HandlerOption) error { | ||||
| 	type rules interface { | ||||
| 		Create(ctx context.Context, in *CreateRequest, out *CreateResponse) error | ||||
| 		Delete(ctx context.Context, in *DeleteRequest, out *DeleteResponse) error | ||||
| 		List(ctx context.Context, in *ListRequest, out *ListResponse) error | ||||
| 	} | ||||
| 	type Rules struct { | ||||
| 		rules | ||||
| 	} | ||||
| 	h := &rulesHandler{hdlr} | ||||
| 	return s.Handle(s.NewHandler(&Rules{h}, opts...)) | ||||
| } | ||||
|  | ||||
| type rulesHandler struct { | ||||
| 	RulesHandler | ||||
| } | ||||
|  | ||||
| func (h *rulesHandler) Create(ctx context.Context, in *CreateRequest, out *CreateResponse) error { | ||||
| 	return h.RulesHandler.Create(ctx, in, out) | ||||
| } | ||||
|  | ||||
| func (h *rulesHandler) Delete(ctx context.Context, in *DeleteRequest, out *DeleteResponse) error { | ||||
| 	return h.RulesHandler.Delete(ctx, in, out) | ||||
| } | ||||
|  | ||||
| func (h *rulesHandler) List(ctx context.Context, in *ListRequest, out *ListResponse) error { | ||||
| 	return h.RulesHandler.List(ctx, in, out) | ||||
| } | ||||
| @@ -1,136 +0,0 @@ | ||||
| syntax = "proto3"; | ||||
|  | ||||
| package go.micro.auth; | ||||
|  | ||||
| service Auth { | ||||
| 	rpc Generate(GenerateRequest) returns (GenerateResponse) {}; | ||||
| 	rpc Inspect(InspectRequest) returns (InspectResponse) {};		 | ||||
| 	rpc Token(TokenRequest) returns (TokenResponse) {}; | ||||
| } | ||||
|  | ||||
| service Accounts { | ||||
| 	rpc List(ListAccountsRequest) returns (ListAccountsResponse) {}; | ||||
| } | ||||
|  | ||||
| service Rules { | ||||
| 	rpc Create(CreateRequest) returns (CreateResponse) {}; | ||||
| 	rpc Delete(DeleteRequest) returns (DeleteResponse) {}; | ||||
| 	rpc List(ListRequest) returns (ListResponse) {}; | ||||
| } | ||||
|  | ||||
| message ListAccountsRequest { | ||||
| } | ||||
|  | ||||
| message ListAccountsResponse { | ||||
| 	repeated Account accounts = 1; | ||||
| } | ||||
|  | ||||
| message Token { | ||||
| 	string access_token = 1; | ||||
| 	string refresh_token = 2; | ||||
| 	int64 created = 3; | ||||
| 	int64 expiry = 4; | ||||
| } | ||||
|  | ||||
| message Account { | ||||
| 	string id = 1; | ||||
| 	string type = 2; | ||||
| 	repeated string roles = 3; | ||||
| 	map<string, string> metadata = 4; | ||||
| 	string namespace = 5; | ||||
| 	string provider = 6; | ||||
| 	string secret = 7; | ||||
| } | ||||
|  | ||||
| message Resource{ | ||||
| 	string name = 1; | ||||
| 	string type = 2; | ||||
| 	string endpoint = 3; | ||||
| 	string namespace = 4; | ||||
| } | ||||
|  | ||||
| message GenerateRequest { | ||||
| 	string id = 1; | ||||
| 	repeated string roles = 2; | ||||
| 	map<string, string> metadata = 3; | ||||
| 	string namespace = 4; | ||||
| 	string secret = 5; | ||||
| 	string type = 6; | ||||
| 	string provider = 7; | ||||
| } | ||||
|  | ||||
| message GenerateResponse { | ||||
| 	Account account = 1; | ||||
| } | ||||
|  | ||||
| message GrantRequest { | ||||
| 	string role = 1; | ||||
| 	Resource resource = 2; | ||||
| } | ||||
|  | ||||
| message GrantResponse {} | ||||
|  | ||||
| message RevokeRequest { | ||||
| 	string role = 1; | ||||
| 	Resource resource = 2; | ||||
| } | ||||
|  | ||||
| message RevokeResponse {} | ||||
|  | ||||
| message InspectRequest { | ||||
| 	string token = 1; | ||||
| } | ||||
|  | ||||
| message InspectResponse { | ||||
| 	Account account = 1; | ||||
| } | ||||
|  | ||||
| message TokenRequest { | ||||
| 	string id = 1; | ||||
| 	string secret = 2; | ||||
| 	string refresh_token = 3; | ||||
| 	int64 token_expiry = 4; | ||||
| } | ||||
|  | ||||
| message TokenResponse { | ||||
| 	Token token = 1; | ||||
| } | ||||
|  | ||||
| enum Access { | ||||
| 	UNKNOWN = 0; | ||||
| 	GRANTED = 1; | ||||
| 	DENIED = 2; | ||||
| } | ||||
|  | ||||
| message Rule { | ||||
| 	string id = 1; | ||||
| 	string role = 2; | ||||
| 	Resource resource = 3; | ||||
| 	Access access = 4; | ||||
| 	int32 priority = 5; | ||||
| } | ||||
|  | ||||
| message CreateRequest { | ||||
| 	string role = 1; | ||||
| 	Resource resource = 2; | ||||
| 	Access access = 3; | ||||
| 	int32 priority = 4; | ||||
| } | ||||
|  | ||||
| message CreateResponse {} | ||||
|  | ||||
| message DeleteRequest { | ||||
| 	string role = 1; | ||||
| 	Resource resource = 2; | ||||
| 	Access access = 3; | ||||
| 	int32 priority = 4; | ||||
| } | ||||
|  | ||||
| message DeleteResponse {} | ||||
|  | ||||
| message ListRequest { | ||||
| } | ||||
|  | ||||
| message ListResponse { | ||||
| 	repeated Rule rules = 1; | ||||
| } | ||||
| @@ -1,305 +0,0 @@ | ||||
| package service | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/auth" | ||||
| 	pb "github.com/micro/go-micro/v2/auth/service/proto" | ||||
| 	"github.com/micro/go-micro/v2/auth/token" | ||||
| 	"github.com/micro/go-micro/v2/auth/token/jwt" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	log "github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v2/util/jitter" | ||||
| ) | ||||
|  | ||||
| // NewAuth returns a new instance of the Auth service | ||||
| func NewAuth(opts ...auth.Option) auth.Auth { | ||||
| 	return &svc{options: auth.NewOptions(opts...)} | ||||
| } | ||||
|  | ||||
| // svc is the service implementation of the Auth interface | ||||
| type svc struct { | ||||
| 	options auth.Options | ||||
| 	auth    pb.AuthService | ||||
| 	rule    pb.RulesService | ||||
| 	jwt     token.Provider | ||||
|  | ||||
| 	rules []*pb.Rule | ||||
| 	sync.Mutex | ||||
| } | ||||
|  | ||||
| func (s *svc) String() string { | ||||
| 	return "service" | ||||
| } | ||||
|  | ||||
| func (s *svc) Init(opts ...auth.Option) { | ||||
| 	for _, o := range opts { | ||||
| 		o(&s.options) | ||||
| 	} | ||||
|  | ||||
| 	dc := client.DefaultClient | ||||
| 	s.auth = pb.NewAuthService("go.micro.auth", dc) | ||||
| 	s.rule = pb.NewRulesService("go.micro.auth", dc) | ||||
|  | ||||
| 	// if we have a JWT public key passed as an option, | ||||
| 	// we can decode tokens with the type "JWT" locally | ||||
| 	// and not have to make an RPC call | ||||
| 	if key := s.options.PublicKey; len(key) > 0 { | ||||
| 		s.jwt = jwt.NewTokenProvider(token.WithPublicKey(key)) | ||||
| 	} | ||||
|  | ||||
| 	// load rules periodically from the auth service | ||||
| 	go func() { | ||||
| 		ruleTimer := time.NewTicker(time.Second * 30) | ||||
|  | ||||
| 		// load rules immediately on startup | ||||
| 		s.loadRules() | ||||
|  | ||||
| 		for { | ||||
| 			<-ruleTimer.C | ||||
|  | ||||
| 			// jitter for up to 5 seconds, this stops | ||||
| 			// all the services calling the auth service | ||||
| 			// at the exact same time | ||||
| 			time.Sleep(jitter.Do(time.Second * 5)) | ||||
| 			s.loadRules() | ||||
| 		} | ||||
| 	}() | ||||
| } | ||||
|  | ||||
| func (s *svc) Options() auth.Options { | ||||
| 	s.Lock() | ||||
| 	defer s.Unlock() | ||||
| 	return s.options | ||||
| } | ||||
|  | ||||
| // Generate a new account | ||||
| func (s *svc) Generate(id string, opts ...auth.GenerateOption) (*auth.Account, error) { | ||||
| 	options := auth.NewGenerateOptions(opts...) | ||||
|  | ||||
| 	rsp, err := s.auth.Generate(context.TODO(), &pb.GenerateRequest{ | ||||
| 		Id:        id, | ||||
| 		Type:      options.Type, | ||||
| 		Secret:    options.Secret, | ||||
| 		Roles:     options.Roles, | ||||
| 		Metadata:  options.Metadata, | ||||
| 		Provider:  options.Provider, | ||||
| 		Namespace: options.Namespace, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return serializeAccount(rsp.Account), nil | ||||
| } | ||||
|  | ||||
| // Grant access to a resource | ||||
| func (s *svc) Grant(role string, res *auth.Resource) error { | ||||
| 	_, err := s.rule.Create(context.TODO(), &pb.CreateRequest{ | ||||
| 		Role:   role, | ||||
| 		Access: pb.Access_GRANTED, | ||||
| 		Resource: &pb.Resource{ | ||||
| 			Namespace: res.Namespace, | ||||
| 			Type:      res.Type, | ||||
| 			Name:      res.Name, | ||||
| 			Endpoint:  res.Endpoint, | ||||
| 		}, | ||||
| 	}) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // Revoke access to a resource | ||||
| func (s *svc) Revoke(role string, res *auth.Resource) error { | ||||
| 	_, err := s.rule.Delete(context.TODO(), &pb.DeleteRequest{ | ||||
| 		Role:   role, | ||||
| 		Access: pb.Access_GRANTED, | ||||
| 		Resource: &pb.Resource{ | ||||
| 			Namespace: res.Namespace, | ||||
| 			Type:      res.Type, | ||||
| 			Name:      res.Name, | ||||
| 			Endpoint:  res.Endpoint, | ||||
| 		}, | ||||
| 	}) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| // Verify an account has access to a resource | ||||
| func (s *svc) Verify(acc *auth.Account, res *auth.Resource) error { | ||||
| 	// set the namespace on the resource | ||||
| 	if len(res.Namespace) == 0 { | ||||
| 		res.Namespace = s.Options().Namespace | ||||
| 	} | ||||
|  | ||||
| 	queries := [][]string{ | ||||
| 		{res.Namespace, res.Type, res.Name, res.Endpoint}, // check for specific role, e.g. service.foo.ListFoo:admin (role is checked in accessForRule) | ||||
| 		{res.Namespace, res.Type, res.Name, "*"},          // check for wildcard endpoint, e.g. service.foo* | ||||
| 		{res.Namespace, res.Type, "*"},                    // check for wildcard name, e.g. service.* | ||||
| 		{res.Namespace, "*"},                              // check for wildcard type, e.g. * | ||||
| 		{"*"},                                             // check for wildcard namespace | ||||
| 	} | ||||
|  | ||||
| 	// endpoint is a url which can have wildcard excludes, e.g. | ||||
| 	// "/foo/*" will allow "/foo/bar" | ||||
| 	if comps := strings.Split(res.Endpoint, "/"); len(comps) > 1 { | ||||
| 		for i := 1; i < len(comps); i++ { | ||||
| 			wildcard := fmt.Sprintf("%v/*", strings.Join(comps[0:i], "/")) | ||||
| 			queries = append(queries, []string{res.Type, res.Name, wildcard}) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// set a default account id / namespace to log | ||||
| 	logID := acc.ID | ||||
| 	if len(logID) == 0 { | ||||
| 		logID = "[no account]" | ||||
| 	} | ||||
| 	logNamespace := acc.Namespace | ||||
| 	if len(logNamespace) == 0 { | ||||
| 		logNamespace = "[no namespace]" | ||||
| 	} | ||||
|  | ||||
| 	for _, q := range queries { | ||||
| 		for _, rule := range s.listRules(q...) { | ||||
| 			switch accessForRule(rule, acc, res) { | ||||
| 			case pb.Access_UNKNOWN: | ||||
| 				continue // rule did not specify access, check the next rule | ||||
| 			case pb.Access_GRANTED: | ||||
| 				log.Tracef("%v:%v granted access to %v:%v:%v:%v by rule %v", logNamespace, logID, res.Namespace, res.Type, res.Name, res.Endpoint, rule.Id) | ||||
| 				return nil // rule grants the account access to the resource | ||||
| 			case pb.Access_DENIED: | ||||
| 				log.Tracef("%v:%v denied access to %v:%v:%v:%v by rule %v", logNamespace, logID, res.Namespace, res.Type, res.Name, res.Endpoint, rule.Id) | ||||
| 				return auth.ErrForbidden // rule denies access to the resource | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// no rules were found for the resource, default to denying access | ||||
| 	log.Tracef("%v:%v denied access to %v:%v:%v:%v by lack of rule (%v rules found for namespace)", logNamespace, logID, res.Namespace, res.Type, res.Name, res.Endpoint, len(s.listRules(res.Namespace))) | ||||
| 	return auth.ErrForbidden | ||||
| } | ||||
|  | ||||
| // Inspect a token | ||||
| func (s *svc) Inspect(token string) (*auth.Account, error) { | ||||
| 	// try to decode JWT locally and fall back to srv if an error occurs | ||||
| 	if len(strings.Split(token, ".")) == 3 && s.jwt != nil { | ||||
| 		return s.jwt.Inspect(token) | ||||
| 	} | ||||
|  | ||||
| 	// the token is not a JWT or we do not have the keys to decode it, | ||||
| 	// fall back to the auth service | ||||
| 	rsp, err := s.auth.Inspect(context.TODO(), &pb.InspectRequest{Token: token}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return serializeAccount(rsp.Account), nil | ||||
| } | ||||
|  | ||||
| // Token generation using an account ID and secret | ||||
| func (s *svc) Token(opts ...auth.TokenOption) (*auth.Token, error) { | ||||
| 	options := auth.NewTokenOptions(opts...) | ||||
|  | ||||
| 	rsp, err := s.auth.Token(context.Background(), &pb.TokenRequest{ | ||||
| 		Id:           options.ID, | ||||
| 		Secret:       options.Secret, | ||||
| 		RefreshToken: options.RefreshToken, | ||||
| 		TokenExpiry:  int64(options.Expiry.Seconds()), | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return serializeToken(rsp.Token), nil | ||||
| } | ||||
|  | ||||
| var ruleJoinKey = ":" | ||||
|  | ||||
| // accessForRule returns a rule status, indicating if a rule permits access to a | ||||
| // resource for a given account | ||||
| func accessForRule(rule *pb.Rule, acc *auth.Account, res *auth.Resource) pb.Access { | ||||
| 	if rule.Role == "*" { | ||||
| 		return rule.Access | ||||
| 	} | ||||
|  | ||||
| 	for _, role := range acc.Roles { | ||||
| 		if rule.Role == role { | ||||
| 			return rule.Access | ||||
| 		} | ||||
|  | ||||
| 		// allow user.anything if role is user.* | ||||
| 		if strings.HasSuffix(rule.Role, ".*") && strings.HasPrefix(rule.Role, role+".") { | ||||
| 			return rule.Access | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return pb.Access_UNKNOWN | ||||
| } | ||||
|  | ||||
| // listRules gets all the rules from the store which match the filters. | ||||
| // filters are namespace, type, name and then endpoint. | ||||
| func (s *svc) listRules(filters ...string) []*pb.Rule { | ||||
| 	s.Lock() | ||||
| 	defer s.Unlock() | ||||
|  | ||||
| 	var rules []*pb.Rule | ||||
| 	for _, r := range s.rules { | ||||
| 		if len(filters) > 0 && r.Resource.Namespace != filters[0] { | ||||
| 			continue | ||||
| 		} | ||||
| 		if len(filters) > 1 && r.Resource.Type != filters[1] { | ||||
| 			continue | ||||
| 		} | ||||
| 		if len(filters) > 2 && r.Resource.Name != filters[2] { | ||||
| 			continue | ||||
| 		} | ||||
| 		if len(filters) > 3 && r.Resource.Endpoint != filters[3] { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		rules = append(rules, r) | ||||
| 	} | ||||
|  | ||||
| 	// sort rules by priority | ||||
| 	sort.Slice(rules, func(i, j int) bool { | ||||
| 		return rules[i].Priority < rules[j].Priority | ||||
| 	}) | ||||
|  | ||||
| 	return rules | ||||
| } | ||||
|  | ||||
| // loadRules retrieves the rules from the auth service | ||||
| func (s *svc) loadRules() { | ||||
| 	rsp, err := s.rule.List(context.TODO(), &pb.ListRequest{}) | ||||
| 	s.Lock() | ||||
| 	defer s.Unlock() | ||||
|  | ||||
| 	if err != nil { | ||||
| 		log.Errorf("Error listing rules: %v", err) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	s.rules = rsp.Rules | ||||
| } | ||||
|  | ||||
| func serializeToken(t *pb.Token) *auth.Token { | ||||
| 	return &auth.Token{ | ||||
| 		AccessToken:  t.AccessToken, | ||||
| 		RefreshToken: t.RefreshToken, | ||||
| 		Created:      time.Unix(t.Created, 0), | ||||
| 		Expiry:       time.Unix(t.Expiry, 0), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func serializeAccount(a *pb.Account) *auth.Account { | ||||
| 	return &auth.Account{ | ||||
| 		ID:        a.Id, | ||||
| 		Roles:     a.Roles, | ||||
| 		Secret:    a.Secret, | ||||
| 		Metadata:  a.Metadata, | ||||
| 		Provider:  a.Provider, | ||||
| 		Namespace: a.Namespace, | ||||
| 	} | ||||
| } | ||||
| @@ -1,26 +0,0 @@ | ||||
| package service | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	pb "github.com/micro/go-micro/v2/auth/service/proto" | ||||
| ) | ||||
|  | ||||
| func TestListRulesSorting(t *testing.T) { | ||||
| 	s := &svc{ | ||||
| 		rules: []*pb.Rule{ | ||||
| 			&pb.Rule{Priority: 1}, | ||||
| 			&pb.Rule{Priority: 3}, | ||||
| 			&pb.Rule{Priority: 2}, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	var priorities []int32 | ||||
| 	for _, r := range s.listRules() { | ||||
| 		priorities = append(priorities, r.Priority) | ||||
| 	} | ||||
|  | ||||
| 	if priorities[0] != 1 || priorities[1] != 2 || priorities[2] != 3 { | ||||
| 		t.Errorf("Incorrect Rule Sequence") | ||||
| 	} | ||||
| } | ||||
| @@ -37,31 +37,3 @@ type Subscriber interface { | ||||
| 	Topic() string | ||||
| 	Unsubscribe() error | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	DefaultBroker Broker = NewBroker() | ||||
| ) | ||||
|  | ||||
| func Init(opts ...Option) error { | ||||
| 	return DefaultBroker.Init(opts...) | ||||
| } | ||||
|  | ||||
| func Connect() error { | ||||
| 	return DefaultBroker.Connect() | ||||
| } | ||||
|  | ||||
| func Disconnect() error { | ||||
| 	return DefaultBroker.Disconnect() | ||||
| } | ||||
|  | ||||
| func Publish(topic string, msg *Message, opts ...PublishOption) error { | ||||
| 	return DefaultBroker.Publish(topic, msg, opts...) | ||||
| } | ||||
|  | ||||
| func Subscribe(topic string, handler Handler, opts ...SubscribeOption) (Subscriber, error) { | ||||
| 	return DefaultBroker.Subscribe(topic, handler, opts...) | ||||
| } | ||||
|  | ||||
| func String() string { | ||||
| 	return DefaultBroker.String() | ||||
| } | ||||
|   | ||||
							
								
								
									
										711
									
								
								broker/http.go
									
									
									
									
									
								
							
							
						
						
									
										711
									
								
								broker/http.go
									
									
									
									
									
								
							| @@ -1,711 +0,0 @@ | ||||
| // Package http provides a http based message broker | ||||
| package broker | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"context" | ||||
| 	"crypto/tls" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	"math/rand" | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"runtime" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v2/codec/json" | ||||
| 	merr "github.com/micro/go-micro/v2/errors" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/registry/cache" | ||||
| 	maddr "github.com/micro/go-micro/v2/util/addr" | ||||
| 	mnet "github.com/micro/go-micro/v2/util/net" | ||||
| 	mls "github.com/micro/go-micro/v2/util/tls" | ||||
| 	"golang.org/x/net/http2" | ||||
| ) | ||||
|  | ||||
| // HTTP Broker is a point to point async broker | ||||
| type httpBroker struct { | ||||
| 	id      string | ||||
| 	address string | ||||
| 	opts    Options | ||||
|  | ||||
| 	mux *http.ServeMux | ||||
|  | ||||
| 	c *http.Client | ||||
| 	r registry.Registry | ||||
|  | ||||
| 	sync.RWMutex | ||||
| 	subscribers map[string][]*httpSubscriber | ||||
| 	running     bool | ||||
| 	exit        chan chan error | ||||
|  | ||||
| 	// offline message inbox | ||||
| 	mtx   sync.RWMutex | ||||
| 	inbox map[string][][]byte | ||||
| } | ||||
|  | ||||
| type httpSubscriber struct { | ||||
| 	opts  SubscribeOptions | ||||
| 	id    string | ||||
| 	topic string | ||||
| 	fn    Handler | ||||
| 	svc   *registry.Service | ||||
| 	hb    *httpBroker | ||||
| } | ||||
|  | ||||
| type httpEvent struct { | ||||
| 	m   *Message | ||||
| 	t   string | ||||
| 	err error | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	DefaultPath      = "/" | ||||
| 	DefaultAddress   = "127.0.0.1:0" | ||||
| 	serviceName      = "micro.http.broker" | ||||
| 	broadcastVersion = "ff.http.broadcast" | ||||
| 	registerTTL      = time.Minute | ||||
| 	registerInterval = time.Second * 30 | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
| 	rand.Seed(time.Now().Unix()) | ||||
| } | ||||
|  | ||||
| func newTransport(config *tls.Config) *http.Transport { | ||||
| 	if config == nil { | ||||
| 		config = &tls.Config{ | ||||
| 			InsecureSkipVerify: true, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	dialTLS := func(network string, addr string) (net.Conn, error) { | ||||
| 		return tls.Dial(network, addr, config) | ||||
| 	} | ||||
|  | ||||
| 	t := &http.Transport{ | ||||
| 		Proxy: http.ProxyFromEnvironment, | ||||
| 		Dial: (&net.Dialer{ | ||||
| 			Timeout:   30 * time.Second, | ||||
| 			KeepAlive: 30 * time.Second, | ||||
| 		}).Dial, | ||||
| 		TLSHandshakeTimeout: 10 * time.Second, | ||||
| 		DialTLS:             dialTLS, | ||||
| 	} | ||||
| 	runtime.SetFinalizer(&t, func(tr **http.Transport) { | ||||
| 		(*tr).CloseIdleConnections() | ||||
| 	}) | ||||
|  | ||||
| 	// setup http2 | ||||
| 	http2.ConfigureTransport(t) | ||||
|  | ||||
| 	return t | ||||
| } | ||||
|  | ||||
| func newHttpBroker(opts ...Option) Broker { | ||||
| 	options := Options{ | ||||
| 		Codec:    json.Marshaler{}, | ||||
| 		Context:  context.TODO(), | ||||
| 		Registry: registry.DefaultRegistry, | ||||
| 	} | ||||
|  | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	// set address | ||||
| 	addr := DefaultAddress | ||||
|  | ||||
| 	if len(options.Addrs) > 0 && len(options.Addrs[0]) > 0 { | ||||
| 		addr = options.Addrs[0] | ||||
| 	} | ||||
|  | ||||
| 	h := &httpBroker{ | ||||
| 		id:          uuid.New().String(), | ||||
| 		address:     addr, | ||||
| 		opts:        options, | ||||
| 		r:           options.Registry, | ||||
| 		c:           &http.Client{Transport: newTransport(options.TLSConfig)}, | ||||
| 		subscribers: make(map[string][]*httpSubscriber), | ||||
| 		exit:        make(chan chan error), | ||||
| 		mux:         http.NewServeMux(), | ||||
| 		inbox:       make(map[string][][]byte), | ||||
| 	} | ||||
|  | ||||
| 	// specify the message handler | ||||
| 	h.mux.Handle(DefaultPath, h) | ||||
|  | ||||
| 	// get optional handlers | ||||
| 	if h.opts.Context != nil { | ||||
| 		handlers, ok := h.opts.Context.Value("http_handlers").(map[string]http.Handler) | ||||
| 		if ok { | ||||
| 			for pattern, handler := range handlers { | ||||
| 				h.mux.Handle(pattern, handler) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return h | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Ack() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Error() error { | ||||
| 	return h.err | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Message() *Message { | ||||
| 	return h.m | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Topic() string { | ||||
| 	return h.t | ||||
| } | ||||
|  | ||||
| func (h *httpSubscriber) Options() SubscribeOptions { | ||||
| 	return h.opts | ||||
| } | ||||
|  | ||||
| func (h *httpSubscriber) Topic() string { | ||||
| 	return h.topic | ||||
| } | ||||
|  | ||||
| func (h *httpSubscriber) Unsubscribe() error { | ||||
| 	return h.hb.unsubscribe(h) | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) saveMessage(topic string, msg []byte) { | ||||
| 	h.mtx.Lock() | ||||
| 	defer h.mtx.Unlock() | ||||
|  | ||||
| 	// get messages | ||||
| 	c := h.inbox[topic] | ||||
|  | ||||
| 	// save message | ||||
| 	c = append(c, msg) | ||||
|  | ||||
| 	// max length 64 | ||||
| 	if len(c) > 64 { | ||||
| 		c = c[:64] | ||||
| 	} | ||||
|  | ||||
| 	// save inbox | ||||
| 	h.inbox[topic] = c | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) getMessage(topic string, num int) [][]byte { | ||||
| 	h.mtx.Lock() | ||||
| 	defer h.mtx.Unlock() | ||||
|  | ||||
| 	// get messages | ||||
| 	c, ok := h.inbox[topic] | ||||
| 	if !ok { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	// more message than requests | ||||
| 	if len(c) >= num { | ||||
| 		msg := c[:num] | ||||
| 		h.inbox[topic] = c[num:] | ||||
| 		return msg | ||||
| 	} | ||||
|  | ||||
| 	// reset inbox | ||||
| 	h.inbox[topic] = nil | ||||
|  | ||||
| 	// return all messages | ||||
| 	return c | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) subscribe(s *httpSubscriber) error { | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	if err := h.r.Register(s.svc, registry.RegisterTTL(registerTTL)); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	h.subscribers[s.topic] = append(h.subscribers[s.topic], s) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) unsubscribe(s *httpSubscriber) error { | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	//nolint:prealloc | ||||
| 	var subscribers []*httpSubscriber | ||||
|  | ||||
| 	// look for subscriber | ||||
| 	for _, sub := range h.subscribers[s.topic] { | ||||
| 		// deregister and skip forward | ||||
| 		if sub == s { | ||||
| 			_ = h.r.Deregister(sub.svc) | ||||
| 			continue | ||||
| 		} | ||||
| 		// keep subscriber | ||||
| 		subscribers = append(subscribers, sub) | ||||
| 	} | ||||
|  | ||||
| 	// set subscribers | ||||
| 	h.subscribers[s.topic] = subscribers | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) run(l net.Listener) { | ||||
| 	t := time.NewTicker(registerInterval) | ||||
| 	defer t.Stop() | ||||
|  | ||||
| 	for { | ||||
| 		select { | ||||
| 		// heartbeat for each subscriber | ||||
| 		case <-t.C: | ||||
| 			h.RLock() | ||||
| 			for _, subs := range h.subscribers { | ||||
| 				for _, sub := range subs { | ||||
| 					_ = h.r.Register(sub.svc, registry.RegisterTTL(registerTTL)) | ||||
| 				} | ||||
| 			} | ||||
| 			h.RUnlock() | ||||
| 		// received exit signal | ||||
| 		case ch := <-h.exit: | ||||
| 			ch <- l.Close() | ||||
| 			h.RLock() | ||||
| 			for _, subs := range h.subscribers { | ||||
| 				for _, sub := range subs { | ||||
| 					_ = h.r.Deregister(sub.svc) | ||||
| 				} | ||||
| 			} | ||||
| 			h.RUnlock() | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) ServeHTTP(w http.ResponseWriter, req *http.Request) { | ||||
| 	if req.Method != "POST" { | ||||
| 		err := merr.BadRequest("go.micro.broker", "Method not allowed") | ||||
| 		http.Error(w, err.Error(), http.StatusMethodNotAllowed) | ||||
| 		return | ||||
| 	} | ||||
| 	defer req.Body.Close() | ||||
|  | ||||
| 	req.ParseForm() | ||||
|  | ||||
| 	b, err := ioutil.ReadAll(req.Body) | ||||
| 	if err != nil { | ||||
| 		errr := merr.InternalServerError("go.micro.broker", "Error reading request body: %v", err) | ||||
| 		w.WriteHeader(500) | ||||
| 		w.Write([]byte(errr.Error())) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	var m *Message | ||||
| 	if err = h.opts.Codec.Unmarshal(b, &m); err != nil { | ||||
| 		errr := merr.InternalServerError("go.micro.broker", "Error parsing request body: %v", err) | ||||
| 		w.WriteHeader(500) | ||||
| 		w.Write([]byte(errr.Error())) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	topic := m.Header["Micro-Topic"] | ||||
| 	//delete(m.Header, ":topic") | ||||
|  | ||||
| 	if len(topic) == 0 { | ||||
| 		errr := merr.InternalServerError("go.micro.broker", "Topic not found") | ||||
| 		w.WriteHeader(500) | ||||
| 		w.Write([]byte(errr.Error())) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	p := &httpEvent{m: m, t: topic} | ||||
| 	id := req.Form.Get("id") | ||||
|  | ||||
| 	//nolint:prealloc | ||||
| 	var subs []Handler | ||||
|  | ||||
| 	h.RLock() | ||||
| 	for _, subscriber := range h.subscribers[topic] { | ||||
| 		if id != subscriber.id { | ||||
| 			continue | ||||
| 		} | ||||
| 		subs = append(subs, subscriber.fn) | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	// execute the handler | ||||
| 	for _, fn := range subs { | ||||
| 		p.err = fn(p) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Address() string { | ||||
| 	h.RLock() | ||||
| 	defer h.RUnlock() | ||||
| 	return h.address | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Connect() error { | ||||
| 	h.RLock() | ||||
| 	if h.running { | ||||
| 		h.RUnlock() | ||||
| 		return nil | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	var l net.Listener | ||||
| 	var err error | ||||
|  | ||||
| 	if h.opts.Secure || h.opts.TLSConfig != nil { | ||||
| 		config := h.opts.TLSConfig | ||||
|  | ||||
| 		fn := func(addr string) (net.Listener, error) { | ||||
| 			if config == nil { | ||||
| 				hosts := []string{addr} | ||||
|  | ||||
| 				// check if its a valid host:port | ||||
| 				if host, _, err := net.SplitHostPort(addr); err == nil { | ||||
| 					if len(host) == 0 { | ||||
| 						hosts = maddr.IPs() | ||||
| 					} else { | ||||
| 						hosts = []string{host} | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				// generate a certificate | ||||
| 				cert, err := mls.Certificate(hosts...) | ||||
| 				if err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 				config = &tls.Config{Certificates: []tls.Certificate{cert}} | ||||
| 			} | ||||
| 			return tls.Listen("tcp", addr, config) | ||||
| 		} | ||||
|  | ||||
| 		l, err = mnet.Listen(h.address, fn) | ||||
| 	} else { | ||||
| 		fn := func(addr string) (net.Listener, error) { | ||||
| 			return net.Listen("tcp", addr) | ||||
| 		} | ||||
|  | ||||
| 		l, err = mnet.Listen(h.address, fn) | ||||
| 	} | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	addr := h.address | ||||
| 	h.address = l.Addr().String() | ||||
|  | ||||
| 	go http.Serve(l, h.mux) | ||||
| 	go func() { | ||||
| 		h.run(l) | ||||
| 		h.Lock() | ||||
| 		h.opts.Addrs = []string{addr} | ||||
| 		h.address = addr | ||||
| 		h.Unlock() | ||||
| 	}() | ||||
|  | ||||
| 	// get registry | ||||
| 	reg := h.opts.Registry | ||||
| 	if reg == nil { | ||||
| 		reg = registry.DefaultRegistry | ||||
| 	} | ||||
| 	// set cache | ||||
| 	h.r = cache.New(reg) | ||||
|  | ||||
| 	// set running | ||||
| 	h.running = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Disconnect() error { | ||||
| 	h.RLock() | ||||
| 	if !h.running { | ||||
| 		h.RUnlock() | ||||
| 		return nil | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	// stop cache | ||||
| 	rc, ok := h.r.(cache.Cache) | ||||
| 	if ok { | ||||
| 		rc.Stop() | ||||
| 	} | ||||
|  | ||||
| 	// exit and return err | ||||
| 	ch := make(chan error) | ||||
| 	h.exit <- ch | ||||
| 	err := <-ch | ||||
|  | ||||
| 	// set not running | ||||
| 	h.running = false | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Init(opts ...Option) error { | ||||
| 	h.RLock() | ||||
| 	if h.running { | ||||
| 		h.RUnlock() | ||||
| 		return errors.New("cannot init while connected") | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	for _, o := range opts { | ||||
| 		o(&h.opts) | ||||
| 	} | ||||
|  | ||||
| 	if len(h.opts.Addrs) > 0 && len(h.opts.Addrs[0]) > 0 { | ||||
| 		h.address = h.opts.Addrs[0] | ||||
| 	} | ||||
|  | ||||
| 	if len(h.id) == 0 { | ||||
| 		h.id = "go.micro.http.broker-" + uuid.New().String() | ||||
| 	} | ||||
|  | ||||
| 	// get registry | ||||
| 	reg := h.opts.Registry | ||||
| 	if reg == nil { | ||||
| 		reg = registry.DefaultRegistry | ||||
| 	} | ||||
|  | ||||
| 	// get cache | ||||
| 	if rc, ok := h.r.(cache.Cache); ok { | ||||
| 		rc.Stop() | ||||
| 	} | ||||
|  | ||||
| 	// set registry | ||||
| 	h.r = cache.New(reg) | ||||
|  | ||||
| 	// reconfigure tls config | ||||
| 	if c := h.opts.TLSConfig; c != nil { | ||||
| 		h.c = &http.Client{ | ||||
| 			Transport: newTransport(c), | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Options() Options { | ||||
| 	return h.opts | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Publish(topic string, msg *Message, opts ...PublishOption) error { | ||||
| 	// create the message first | ||||
| 	m := &Message{ | ||||
| 		Header: make(map[string]string), | ||||
| 		Body:   msg.Body, | ||||
| 	} | ||||
|  | ||||
| 	for k, v := range msg.Header { | ||||
| 		m.Header[k] = v | ||||
| 	} | ||||
|  | ||||
| 	m.Header["Micro-Topic"] = topic | ||||
|  | ||||
| 	// encode the message | ||||
| 	b, err := h.opts.Codec.Marshal(m) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// save the message | ||||
| 	h.saveMessage(topic, b) | ||||
|  | ||||
| 	// now attempt to get the service | ||||
| 	h.RLock() | ||||
| 	s, err := h.r.GetService(serviceName) | ||||
| 	if err != nil { | ||||
| 		h.RUnlock() | ||||
| 		return err | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	pub := func(node *registry.Node, t string, b []byte) error { | ||||
| 		scheme := "http" | ||||
|  | ||||
| 		// check if secure is added in metadata | ||||
| 		if node.Metadata["secure"] == "true" { | ||||
| 			scheme = "https" | ||||
| 		} | ||||
|  | ||||
| 		vals := url.Values{} | ||||
| 		vals.Add("id", node.Id) | ||||
|  | ||||
| 		uri := fmt.Sprintf("%s://%s%s?%s", scheme, node.Address, DefaultPath, vals.Encode()) | ||||
| 		r, err := h.c.Post(uri, "application/json", bytes.NewReader(b)) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		// discard response body | ||||
| 		io.Copy(ioutil.Discard, r.Body) | ||||
| 		r.Body.Close() | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	srv := func(s []*registry.Service, b []byte) { | ||||
| 		for _, service := range s { | ||||
| 			var nodes []*registry.Node | ||||
|  | ||||
| 			for _, node := range service.Nodes { | ||||
| 				// only use nodes tagged with broker http | ||||
| 				if node.Metadata["broker"] != "http" { | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				// look for nodes for the topic | ||||
| 				if node.Metadata["topic"] != topic { | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				nodes = append(nodes, node) | ||||
| 			} | ||||
|  | ||||
| 			// only process if we have nodes | ||||
| 			if len(nodes) == 0 { | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			switch service.Version { | ||||
| 			// broadcast version means broadcast to all nodes | ||||
| 			case broadcastVersion: | ||||
| 				var success bool | ||||
|  | ||||
| 				// publish to all nodes | ||||
| 				for _, node := range nodes { | ||||
| 					// publish async | ||||
| 					if err := pub(node, topic, b); err == nil { | ||||
| 						success = true | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				// save if it failed to publish at least once | ||||
| 				if !success { | ||||
| 					h.saveMessage(topic, b) | ||||
| 				} | ||||
| 			default: | ||||
| 				// select node to publish to | ||||
| 				node := nodes[rand.Int()%len(nodes)] | ||||
|  | ||||
| 				// publish async to one node | ||||
| 				if err := pub(node, topic, b); err != nil { | ||||
| 					// if failed save it | ||||
| 					h.saveMessage(topic, b) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// do the rest async | ||||
| 	go func() { | ||||
| 		// get a third of the backlog | ||||
| 		messages := h.getMessage(topic, 8) | ||||
| 		delay := (len(messages) > 1) | ||||
|  | ||||
| 		// publish all the messages | ||||
| 		for _, msg := range messages { | ||||
| 			// serialize here | ||||
| 			srv(s, msg) | ||||
|  | ||||
| 			// sending a backlog of messages | ||||
| 			if delay { | ||||
| 				time.Sleep(time.Millisecond * 100) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Subscribe(topic string, handler Handler, opts ...SubscribeOption) (Subscriber, error) { | ||||
| 	var err error | ||||
| 	var host, port string | ||||
| 	options := NewSubscribeOptions(opts...) | ||||
|  | ||||
| 	// parse address for host, port | ||||
| 	host, port, err = net.SplitHostPort(h.Address()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	addr, err := maddr.Extract(host) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var secure bool | ||||
|  | ||||
| 	if h.opts.Secure || h.opts.TLSConfig != nil { | ||||
| 		secure = true | ||||
| 	} | ||||
|  | ||||
| 	// register service | ||||
| 	node := ®istry.Node{ | ||||
| 		Id:      topic + "-" + h.id, | ||||
| 		Address: mnet.HostPort(addr, port), | ||||
| 		Metadata: map[string]string{ | ||||
| 			"secure": fmt.Sprintf("%t", secure), | ||||
| 			"broker": "http", | ||||
| 			"topic":  topic, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	// check for queue group or broadcast queue | ||||
| 	version := options.Queue | ||||
| 	if len(version) == 0 { | ||||
| 		version = broadcastVersion | ||||
| 	} | ||||
|  | ||||
| 	service := ®istry.Service{ | ||||
| 		Name:    serviceName, | ||||
| 		Version: version, | ||||
| 		Nodes:   []*registry.Node{node}, | ||||
| 	} | ||||
|  | ||||
| 	// generate subscriber | ||||
| 	subscriber := &httpSubscriber{ | ||||
| 		opts:  options, | ||||
| 		hb:    h, | ||||
| 		id:    node.Id, | ||||
| 		topic: topic, | ||||
| 		fn:    handler, | ||||
| 		svc:   service, | ||||
| 	} | ||||
|  | ||||
| 	// subscribe now | ||||
| 	if err := h.subscribe(subscriber); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// return the subscriber | ||||
| 	return subscriber, nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) String() string { | ||||
| 	return "http" | ||||
| } | ||||
|  | ||||
| // NewBroker returns a new http broker | ||||
| func NewBroker(opts ...Option) Broker { | ||||
| 	return newHttpBroker(opts...) | ||||
| } | ||||
| @@ -2,10 +2,712 @@ | ||||
| package http | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"bytes" | ||||
| 	"context" | ||||
| 	"crypto/tls" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
| 	"math/rand" | ||||
| 	"net" | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"runtime" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	"github.com/micro/go-micro/v3/codec/json" | ||||
| 	merr "github.com/micro/go-micro/v3/errors" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry/cache" | ||||
| 	"github.com/micro/go-micro/v3/registry/mdns" | ||||
| 	maddr "github.com/micro/go-micro/v3/util/addr" | ||||
| 	mnet "github.com/micro/go-micro/v3/util/net" | ||||
| 	mls "github.com/micro/go-micro/v3/util/tls" | ||||
| 	"golang.org/x/net/http2" | ||||
| ) | ||||
|  | ||||
| // HTTP Broker is a point to point async broker | ||||
| type httpBroker struct { | ||||
| 	id      string | ||||
| 	address string | ||||
| 	opts    broker.Options | ||||
|  | ||||
| 	mux *http.ServeMux | ||||
|  | ||||
| 	c *http.Client | ||||
| 	r registry.Registry | ||||
|  | ||||
| 	sync.RWMutex | ||||
| 	subscribers map[string][]*httpSubscriber | ||||
| 	running     bool | ||||
| 	exit        chan chan error | ||||
|  | ||||
| 	// offline message inbox | ||||
| 	mtx   sync.RWMutex | ||||
| 	inbox map[string][][]byte | ||||
| } | ||||
|  | ||||
| type httpSubscriber struct { | ||||
| 	opts  broker.SubscribeOptions | ||||
| 	id    string | ||||
| 	topic string | ||||
| 	fn    broker.Handler | ||||
| 	svc   *registry.Service | ||||
| 	hb    *httpBroker | ||||
| } | ||||
|  | ||||
| type httpEvent struct { | ||||
| 	m   *broker.Message | ||||
| 	t   string | ||||
| 	err error | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	DefaultPath      = "/" | ||||
| 	DefaultAddress   = "127.0.0.1:0" | ||||
| 	serviceName      = "micro.http.broker" | ||||
| 	broadcastVersion = "ff.http.broadcast" | ||||
| 	registerTTL      = time.Minute | ||||
| 	registerInterval = time.Second * 30 | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
| 	rand.Seed(time.Now().Unix()) | ||||
| } | ||||
|  | ||||
| func newTransport(config *tls.Config) *http.Transport { | ||||
| 	if config == nil { | ||||
| 		config = &tls.Config{ | ||||
| 			InsecureSkipVerify: true, | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	dialTLS := func(network string, addr string) (net.Conn, error) { | ||||
| 		return tls.Dial(network, addr, config) | ||||
| 	} | ||||
|  | ||||
| 	t := &http.Transport{ | ||||
| 		Proxy: http.ProxyFromEnvironment, | ||||
| 		Dial: (&net.Dialer{ | ||||
| 			Timeout:   30 * time.Second, | ||||
| 			KeepAlive: 30 * time.Second, | ||||
| 		}).Dial, | ||||
| 		TLSHandshakeTimeout: 10 * time.Second, | ||||
| 		DialTLS:             dialTLS, | ||||
| 	} | ||||
| 	runtime.SetFinalizer(&t, func(tr **http.Transport) { | ||||
| 		(*tr).CloseIdleConnections() | ||||
| 	}) | ||||
|  | ||||
| 	// setup http2 | ||||
| 	http2.ConfigureTransport(t) | ||||
|  | ||||
| 	return t | ||||
| } | ||||
|  | ||||
| func newHttpBroker(opts ...broker.Option) broker.Broker { | ||||
| 	options := broker.Options{ | ||||
| 		Codec:    json.Marshaler{}, | ||||
| 		Context:  context.TODO(), | ||||
| 		Registry: mdns.NewRegistry(), | ||||
| 	} | ||||
|  | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	// set address | ||||
| 	addr := DefaultAddress | ||||
|  | ||||
| 	if len(options.Addrs) > 0 && len(options.Addrs[0]) > 0 { | ||||
| 		addr = options.Addrs[0] | ||||
| 	} | ||||
|  | ||||
| 	h := &httpBroker{ | ||||
| 		id:          uuid.New().String(), | ||||
| 		address:     addr, | ||||
| 		opts:        options, | ||||
| 		r:           options.Registry, | ||||
| 		c:           &http.Client{Transport: newTransport(options.TLSConfig)}, | ||||
| 		subscribers: make(map[string][]*httpSubscriber), | ||||
| 		exit:        make(chan chan error), | ||||
| 		mux:         http.NewServeMux(), | ||||
| 		inbox:       make(map[string][][]byte), | ||||
| 	} | ||||
|  | ||||
| 	// specify the message handler | ||||
| 	h.mux.Handle(DefaultPath, h) | ||||
|  | ||||
| 	// get optional handlers | ||||
| 	if h.opts.Context != nil { | ||||
| 		handlers, ok := h.opts.Context.Value("http_handlers").(map[string]http.Handler) | ||||
| 		if ok { | ||||
| 			for pattern, handler := range handlers { | ||||
| 				h.mux.Handle(pattern, handler) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return h | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Ack() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Error() error { | ||||
| 	return h.err | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Message() *broker.Message { | ||||
| 	return h.m | ||||
| } | ||||
|  | ||||
| func (h *httpEvent) Topic() string { | ||||
| 	return h.t | ||||
| } | ||||
|  | ||||
| func (h *httpSubscriber) Options() broker.SubscribeOptions { | ||||
| 	return h.opts | ||||
| } | ||||
|  | ||||
| func (h *httpSubscriber) Topic() string { | ||||
| 	return h.topic | ||||
| } | ||||
|  | ||||
| func (h *httpSubscriber) Unsubscribe() error { | ||||
| 	return h.hb.unsubscribe(h) | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) saveMessage(topic string, msg []byte) { | ||||
| 	h.mtx.Lock() | ||||
| 	defer h.mtx.Unlock() | ||||
|  | ||||
| 	// get messages | ||||
| 	c := h.inbox[topic] | ||||
|  | ||||
| 	// save message | ||||
| 	c = append(c, msg) | ||||
|  | ||||
| 	// max length 64 | ||||
| 	if len(c) > 64 { | ||||
| 		c = c[:64] | ||||
| 	} | ||||
|  | ||||
| 	// save inbox | ||||
| 	h.inbox[topic] = c | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) getMessage(topic string, num int) [][]byte { | ||||
| 	h.mtx.Lock() | ||||
| 	defer h.mtx.Unlock() | ||||
|  | ||||
| 	// get messages | ||||
| 	c, ok := h.inbox[topic] | ||||
| 	if !ok { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	// more message than requests | ||||
| 	if len(c) >= num { | ||||
| 		msg := c[:num] | ||||
| 		h.inbox[topic] = c[num:] | ||||
| 		return msg | ||||
| 	} | ||||
|  | ||||
| 	// reset inbox | ||||
| 	h.inbox[topic] = nil | ||||
|  | ||||
| 	// return all messages | ||||
| 	return c | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) subscribe(s *httpSubscriber) error { | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	if err := h.r.Register(s.svc, registry.RegisterTTL(registerTTL)); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	h.subscribers[s.topic] = append(h.subscribers[s.topic], s) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) unsubscribe(s *httpSubscriber) error { | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	//nolint:prealloc | ||||
| 	var subscribers []*httpSubscriber | ||||
|  | ||||
| 	// look for subscriber | ||||
| 	for _, sub := range h.subscribers[s.topic] { | ||||
| 		// deregister and skip forward | ||||
| 		if sub == s { | ||||
| 			_ = h.r.Deregister(sub.svc) | ||||
| 			continue | ||||
| 		} | ||||
| 		// keep subscriber | ||||
| 		subscribers = append(subscribers, sub) | ||||
| 	} | ||||
|  | ||||
| 	// set subscribers | ||||
| 	h.subscribers[s.topic] = subscribers | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) run(l net.Listener) { | ||||
| 	t := time.NewTicker(registerInterval) | ||||
| 	defer t.Stop() | ||||
|  | ||||
| 	for { | ||||
| 		select { | ||||
| 		// heartbeat for each subscriber | ||||
| 		case <-t.C: | ||||
| 			h.RLock() | ||||
| 			for _, subs := range h.subscribers { | ||||
| 				for _, sub := range subs { | ||||
| 					_ = h.r.Register(sub.svc, registry.RegisterTTL(registerTTL)) | ||||
| 				} | ||||
| 			} | ||||
| 			h.RUnlock() | ||||
| 		// received exit signal | ||||
| 		case ch := <-h.exit: | ||||
| 			ch <- l.Close() | ||||
| 			h.RLock() | ||||
| 			for _, subs := range h.subscribers { | ||||
| 				for _, sub := range subs { | ||||
| 					_ = h.r.Deregister(sub.svc) | ||||
| 				} | ||||
| 			} | ||||
| 			h.RUnlock() | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) ServeHTTP(w http.ResponseWriter, req *http.Request) { | ||||
| 	if req.Method != "POST" { | ||||
| 		err := merr.BadRequest("go.micro.broker", "Method not allowed") | ||||
| 		http.Error(w, err.Error(), http.StatusMethodNotAllowed) | ||||
| 		return | ||||
| 	} | ||||
| 	defer req.Body.Close() | ||||
|  | ||||
| 	req.ParseForm() | ||||
|  | ||||
| 	b, err := ioutil.ReadAll(req.Body) | ||||
| 	if err != nil { | ||||
| 		errr := merr.InternalServerError("go.micro.broker", "Error reading request body: %v", err) | ||||
| 		w.WriteHeader(500) | ||||
| 		w.Write([]byte(errr.Error())) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	var m *broker.Message | ||||
| 	if err = h.opts.Codec.Unmarshal(b, &m); err != nil { | ||||
| 		errr := merr.InternalServerError("go.micro.broker", "Error parsing request body: %v", err) | ||||
| 		w.WriteHeader(500) | ||||
| 		w.Write([]byte(errr.Error())) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	topic := m.Header["Micro-Topic"] | ||||
| 	//delete(m.Header, ":topic") | ||||
|  | ||||
| 	if len(topic) == 0 { | ||||
| 		errr := merr.InternalServerError("go.micro.broker", "Topic not found") | ||||
| 		w.WriteHeader(500) | ||||
| 		w.Write([]byte(errr.Error())) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	p := &httpEvent{m: m, t: topic} | ||||
| 	id := req.Form.Get("id") | ||||
|  | ||||
| 	//nolint:prealloc | ||||
| 	var subs []broker.Handler | ||||
|  | ||||
| 	h.RLock() | ||||
| 	for _, subscriber := range h.subscribers[topic] { | ||||
| 		if id != subscriber.id { | ||||
| 			continue | ||||
| 		} | ||||
| 		subs = append(subs, subscriber.fn) | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	// execute the handler | ||||
| 	for _, fn := range subs { | ||||
| 		p.err = fn(p) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Address() string { | ||||
| 	h.RLock() | ||||
| 	defer h.RUnlock() | ||||
| 	return h.address | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Connect() error { | ||||
| 	h.RLock() | ||||
| 	if h.running { | ||||
| 		h.RUnlock() | ||||
| 		return nil | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	var l net.Listener | ||||
| 	var err error | ||||
|  | ||||
| 	if h.opts.Secure || h.opts.TLSConfig != nil { | ||||
| 		config := h.opts.TLSConfig | ||||
|  | ||||
| 		fn := func(addr string) (net.Listener, error) { | ||||
| 			if config == nil { | ||||
| 				hosts := []string{addr} | ||||
|  | ||||
| 				// check if its a valid host:port | ||||
| 				if host, _, err := net.SplitHostPort(addr); err == nil { | ||||
| 					if len(host) == 0 { | ||||
| 						hosts = maddr.IPs() | ||||
| 					} else { | ||||
| 						hosts = []string{host} | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				// generate a certificate | ||||
| 				cert, err := mls.Certificate(hosts...) | ||||
| 				if err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 				config = &tls.Config{Certificates: []tls.Certificate{cert}} | ||||
| 			} | ||||
| 			return tls.Listen("tcp", addr, config) | ||||
| 		} | ||||
|  | ||||
| 		l, err = mnet.Listen(h.address, fn) | ||||
| 	} else { | ||||
| 		fn := func(addr string) (net.Listener, error) { | ||||
| 			return net.Listen("tcp", addr) | ||||
| 		} | ||||
|  | ||||
| 		l, err = mnet.Listen(h.address, fn) | ||||
| 	} | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	addr := h.address | ||||
| 	h.address = l.Addr().String() | ||||
|  | ||||
| 	go http.Serve(l, h.mux) | ||||
| 	go func() { | ||||
| 		h.run(l) | ||||
| 		h.Lock() | ||||
| 		h.opts.Addrs = []string{addr} | ||||
| 		h.address = addr | ||||
| 		h.Unlock() | ||||
| 	}() | ||||
|  | ||||
| 	// get registry | ||||
| 	reg := h.opts.Registry | ||||
| 	if reg == nil { | ||||
| 		reg = mdns.NewRegistry() | ||||
| 	} | ||||
| 	// set cache | ||||
| 	h.r = cache.New(reg) | ||||
|  | ||||
| 	// set running | ||||
| 	h.running = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Disconnect() error { | ||||
| 	h.RLock() | ||||
| 	if !h.running { | ||||
| 		h.RUnlock() | ||||
| 		return nil | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	// stop cache | ||||
| 	rc, ok := h.r.(cache.Cache) | ||||
| 	if ok { | ||||
| 		rc.Stop() | ||||
| 	} | ||||
|  | ||||
| 	// exit and return err | ||||
| 	ch := make(chan error) | ||||
| 	h.exit <- ch | ||||
| 	err := <-ch | ||||
|  | ||||
| 	// set not running | ||||
| 	h.running = false | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Init(opts ...broker.Option) error { | ||||
| 	h.RLock() | ||||
| 	if h.running { | ||||
| 		h.RUnlock() | ||||
| 		return errors.New("cannot init while connected") | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	h.Lock() | ||||
| 	defer h.Unlock() | ||||
|  | ||||
| 	for _, o := range opts { | ||||
| 		o(&h.opts) | ||||
| 	} | ||||
|  | ||||
| 	if len(h.opts.Addrs) > 0 && len(h.opts.Addrs[0]) > 0 { | ||||
| 		h.address = h.opts.Addrs[0] | ||||
| 	} | ||||
|  | ||||
| 	if len(h.id) == 0 { | ||||
| 		h.id = "go.micro.http.broker-" + uuid.New().String() | ||||
| 	} | ||||
|  | ||||
| 	// get registry | ||||
| 	reg := h.opts.Registry | ||||
| 	if reg == nil { | ||||
| 		reg = mdns.NewRegistry() | ||||
| 	} | ||||
|  | ||||
| 	// get cache | ||||
| 	if rc, ok := h.r.(cache.Cache); ok { | ||||
| 		rc.Stop() | ||||
| 	} | ||||
|  | ||||
| 	// set registry | ||||
| 	h.r = cache.New(reg) | ||||
|  | ||||
| 	// reconfigure tls config | ||||
| 	if c := h.opts.TLSConfig; c != nil { | ||||
| 		h.c = &http.Client{ | ||||
| 			Transport: newTransport(c), | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Options() broker.Options { | ||||
| 	return h.opts | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Publish(topic string, msg *broker.Message, opts ...broker.PublishOption) error { | ||||
| 	// create the message first | ||||
| 	m := &broker.Message{ | ||||
| 		Header: make(map[string]string), | ||||
| 		Body:   msg.Body, | ||||
| 	} | ||||
|  | ||||
| 	for k, v := range msg.Header { | ||||
| 		m.Header[k] = v | ||||
| 	} | ||||
|  | ||||
| 	m.Header["Micro-Topic"] = topic | ||||
|  | ||||
| 	// encode the message | ||||
| 	b, err := h.opts.Codec.Marshal(m) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// save the message | ||||
| 	h.saveMessage(topic, b) | ||||
|  | ||||
| 	// now attempt to get the service | ||||
| 	h.RLock() | ||||
| 	s, err := h.r.GetService(serviceName) | ||||
| 	if err != nil { | ||||
| 		h.RUnlock() | ||||
| 		return err | ||||
| 	} | ||||
| 	h.RUnlock() | ||||
|  | ||||
| 	pub := func(node *registry.Node, t string, b []byte) error { | ||||
| 		scheme := "http" | ||||
|  | ||||
| 		// check if secure is added in metadata | ||||
| 		if node.Metadata["secure"] == "true" { | ||||
| 			scheme = "https" | ||||
| 		} | ||||
|  | ||||
| 		vals := url.Values{} | ||||
| 		vals.Add("id", node.Id) | ||||
|  | ||||
| 		uri := fmt.Sprintf("%s://%s%s?%s", scheme, node.Address, DefaultPath, vals.Encode()) | ||||
| 		r, err := h.c.Post(uri, "application/json", bytes.NewReader(b)) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		// discard response body | ||||
| 		io.Copy(ioutil.Discard, r.Body) | ||||
| 		r.Body.Close() | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	srv := func(s []*registry.Service, b []byte) { | ||||
| 		for _, service := range s { | ||||
| 			var nodes []*registry.Node | ||||
|  | ||||
| 			for _, node := range service.Nodes { | ||||
| 				// only use nodes tagged with broker http | ||||
| 				if node.Metadata["broker"] != "http" { | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				// look for nodes for the topic | ||||
| 				if node.Metadata["topic"] != topic { | ||||
| 					continue | ||||
| 				} | ||||
|  | ||||
| 				nodes = append(nodes, node) | ||||
| 			} | ||||
|  | ||||
| 			// only process if we have nodes | ||||
| 			if len(nodes) == 0 { | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			switch service.Version { | ||||
| 			// broadcast version means broadcast to all nodes | ||||
| 			case broadcastVersion: | ||||
| 				var success bool | ||||
|  | ||||
| 				// publish to all nodes | ||||
| 				for _, node := range nodes { | ||||
| 					// publish async | ||||
| 					if err := pub(node, topic, b); err == nil { | ||||
| 						success = true | ||||
| 					} | ||||
| 				} | ||||
|  | ||||
| 				// save if it failed to publish at least once | ||||
| 				if !success { | ||||
| 					h.saveMessage(topic, b) | ||||
| 				} | ||||
| 			default: | ||||
| 				// select node to publish to | ||||
| 				node := nodes[rand.Int()%len(nodes)] | ||||
|  | ||||
| 				// publish async to one node | ||||
| 				if err := pub(node, topic, b); err != nil { | ||||
| 					// if failed save it | ||||
| 					h.saveMessage(topic, b) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// do the rest async | ||||
| 	go func() { | ||||
| 		// get a third of the backlog | ||||
| 		messages := h.getMessage(topic, 8) | ||||
| 		delay := (len(messages) > 1) | ||||
|  | ||||
| 		// publish all the messages | ||||
| 		for _, msg := range messages { | ||||
| 			// serialize here | ||||
| 			srv(s, msg) | ||||
|  | ||||
| 			// sending a backlog of messages | ||||
| 			if delay { | ||||
| 				time.Sleep(time.Millisecond * 100) | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) Subscribe(topic string, handler broker.Handler, opts ...broker.SubscribeOption) (broker.Subscriber, error) { | ||||
| 	var err error | ||||
| 	var host, port string | ||||
| 	options := broker.NewSubscribeOptions(opts...) | ||||
|  | ||||
| 	// parse address for host, port | ||||
| 	host, port, err = net.SplitHostPort(h.Address()) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	addr, err := maddr.Extract(host) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var secure bool | ||||
|  | ||||
| 	if h.opts.Secure || h.opts.TLSConfig != nil { | ||||
| 		secure = true | ||||
| 	} | ||||
|  | ||||
| 	// register service | ||||
| 	node := ®istry.Node{ | ||||
| 		Id:      topic + "-" + h.id, | ||||
| 		Address: mnet.HostPort(addr, port), | ||||
| 		Metadata: map[string]string{ | ||||
| 			"secure": fmt.Sprintf("%t", secure), | ||||
| 			"broker": "http", | ||||
| 			"topic":  topic, | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	// check for queue group or broadcast queue | ||||
| 	version := options.Queue | ||||
| 	if len(version) == 0 { | ||||
| 		version = broadcastVersion | ||||
| 	} | ||||
|  | ||||
| 	service := ®istry.Service{ | ||||
| 		Name:    serviceName, | ||||
| 		Version: version, | ||||
| 		Nodes:   []*registry.Node{node}, | ||||
| 	} | ||||
|  | ||||
| 	// generate subscriber | ||||
| 	subscriber := &httpSubscriber{ | ||||
| 		opts:  options, | ||||
| 		hb:    h, | ||||
| 		id:    node.Id, | ||||
| 		topic: topic, | ||||
| 		fn:    handler, | ||||
| 		svc:   service, | ||||
| 	} | ||||
|  | ||||
| 	// subscribe now | ||||
| 	if err := h.subscribe(subscriber); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// return the subscriber | ||||
| 	return subscriber, nil | ||||
| } | ||||
|  | ||||
| func (h *httpBroker) String() string { | ||||
| 	return "http" | ||||
| } | ||||
|  | ||||
| // NewBroker returns a new http broker | ||||
| func NewBroker(opts ...broker.Option) broker.Broker { | ||||
| 	return broker.NewBroker(opts...) | ||||
| 	return newHttpBroker(opts...) | ||||
| } | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| package broker_test | ||||
| package http | ||||
| 
 | ||||
| import ( | ||||
| 	"sync" | ||||
| @@ -6,9 +6,9 @@ import ( | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/registry/memory" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry/memory" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| @@ -61,7 +61,7 @@ func sub(be *testing.B, c int) { | ||||
| 	be.StopTimer() | ||||
| 	m := newTestRegistry() | ||||
| 
 | ||||
| 	b := broker.NewBroker(broker.Registry(m)) | ||||
| 	b := NewBroker(broker.Registry(m)) | ||||
| 	topic := uuid.New().String() | ||||
| 
 | ||||
| 	if err := b.Init(); err != nil { | ||||
| @@ -120,7 +120,7 @@ func sub(be *testing.B, c int) { | ||||
| func pub(be *testing.B, c int) { | ||||
| 	be.StopTimer() | ||||
| 	m := newTestRegistry() | ||||
| 	b := broker.NewBroker(broker.Registry(m)) | ||||
| 	b := NewBroker(broker.Registry(m)) | ||||
| 	topic := uuid.New().String() | ||||
| 
 | ||||
| 	if err := b.Init(); err != nil { | ||||
| @@ -189,7 +189,7 @@ func pub(be *testing.B, c int) { | ||||
| 
 | ||||
| func TestBroker(t *testing.T) { | ||||
| 	m := newTestRegistry() | ||||
| 	b := broker.NewBroker(broker.Registry(m)) | ||||
| 	b := NewBroker(broker.Registry(m)) | ||||
| 
 | ||||
| 	if err := b.Init(); err != nil { | ||||
| 		t.Fatalf("Unexpected init error: %v", err) | ||||
| @@ -236,7 +236,7 @@ func TestBroker(t *testing.T) { | ||||
| 
 | ||||
| func TestConcurrentSubBroker(t *testing.T) { | ||||
| 	m := newTestRegistry() | ||||
| 	b := broker.NewBroker(broker.Registry(m)) | ||||
| 	b := NewBroker(broker.Registry(m)) | ||||
| 
 | ||||
| 	if err := b.Init(); err != nil { | ||||
| 		t.Fatalf("Unexpected init error: %v", err) | ||||
| @@ -293,7 +293,7 @@ func TestConcurrentSubBroker(t *testing.T) { | ||||
| 
 | ||||
| func TestConcurrentPubBroker(t *testing.T) { | ||||
| 	m := newTestRegistry() | ||||
| 	b := broker.NewBroker(broker.Registry(m)) | ||||
| 	b := NewBroker(broker.Registry(m)) | ||||
| 
 | ||||
| 	if err := b.Init(); err != nil { | ||||
| 		t.Fatalf("Unexpected init error: %v", err) | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"context" | ||||
| 	"net/http" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| ) | ||||
|  | ||||
| // Handle registers the handler for the given pattern. | ||||
|   | ||||
| @@ -9,10 +9,10 @@ import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	maddr "github.com/micro/go-micro/v2/util/addr" | ||||
| 	mnet "github.com/micro/go-micro/v2/util/net" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	maddr "github.com/micro/go-micro/v3/util/addr" | ||||
| 	mnet "github.com/micro/go-micro/v3/util/net" | ||||
| ) | ||||
|  | ||||
| type memoryBroker struct { | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| ) | ||||
|  | ||||
| func TestMemoryBroker(t *testing.T) { | ||||
|   | ||||
| @@ -3,7 +3,7 @@ package nats | ||||
| import ( | ||||
| 	"context" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| ) | ||||
|  | ||||
| // setBrokerOption returns a function to setup a context with given value | ||||
|   | ||||
| @@ -7,10 +7,10 @@ import ( | ||||
| 	"strings" | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v2/codec/json" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	"github.com/micro/go-micro/v3/codec/json" | ||||
| 	"github.com/micro/go-micro/v3/logger" | ||||
| 	"github.com/micro/go-micro/v3/registry/mdns" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| ) | ||||
|  | ||||
| @@ -131,6 +131,10 @@ func (n *natsBroker) Connect() error { | ||||
|  | ||||
| 		c, err := opts.Connect() | ||||
| 		if err != nil { | ||||
| 			if logger.V(logger.WarnLevel, logger.DefaultLogger) { | ||||
| 				logger.Warnf("Error connecting to broker: %v", err) | ||||
| 			} | ||||
|  | ||||
| 			return err | ||||
| 		} | ||||
| 		n.conn = c | ||||
| @@ -306,7 +310,7 @@ func NewBroker(opts ...broker.Option) broker.Broker { | ||||
| 		// Default codec | ||||
| 		Codec:    json.Marshaler{}, | ||||
| 		Context:  context.Background(), | ||||
| 		Registry: registry.DefaultRegistry, | ||||
| 		Registry: mdns.NewRegistry(), | ||||
| 	} | ||||
|  | ||||
| 	n := &natsBroker{ | ||||
|   | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| package nats | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	nats "github.com/nats-io/nats.go" | ||||
| ) | ||||
|  | ||||
|   | ||||
| @@ -4,8 +4,8 @@ import ( | ||||
| 	"context" | ||||
| 	"crypto/tls" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/codec" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v3/codec" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| ) | ||||
|  | ||||
| type Options struct { | ||||
| @@ -13,7 +13,7 @@ type Options struct { | ||||
| 	Secure bool | ||||
| 	Codec  codec.Marshaler | ||||
|  | ||||
| 	// Handler executed when error happens in broker mesage | ||||
| 	// Handler executed when error happens in broker message | ||||
| 	// processing | ||||
| 	ErrorHandler Handler | ||||
|  | ||||
|   | ||||
| @@ -1,374 +0,0 @@ | ||||
| // Code generated by protoc-gen-go. DO NOT EDIT. | ||||
| // source: broker/service/proto/broker.proto | ||||
|  | ||||
| package go_micro_broker | ||||
|  | ||||
| import ( | ||||
| 	context "context" | ||||
| 	fmt "fmt" | ||||
| 	proto "github.com/golang/protobuf/proto" | ||||
| 	grpc "google.golang.org/grpc" | ||||
| 	codes "google.golang.org/grpc/codes" | ||||
| 	status "google.golang.org/grpc/status" | ||||
| 	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.ProtoPackageIsVersion3 // please upgrade the proto package | ||||
|  | ||||
| type Empty struct { | ||||
| 	XXX_NoUnkeyedLiteral struct{} `json:"-"` | ||||
| 	XXX_unrecognized     []byte   `json:"-"` | ||||
| 	XXX_sizecache        int32    `json:"-"` | ||||
| } | ||||
|  | ||||
| func (m *Empty) Reset()         { *m = Empty{} } | ||||
| func (m *Empty) String() string { return proto.CompactTextString(m) } | ||||
| func (*Empty) ProtoMessage()    {} | ||||
| func (*Empty) Descriptor() ([]byte, []int) { | ||||
| 	return fileDescriptor_df4d8f04292cf3fe, []int{0} | ||||
| } | ||||
|  | ||||
| func (m *Empty) XXX_Unmarshal(b []byte) error { | ||||
| 	return xxx_messageInfo_Empty.Unmarshal(m, b) | ||||
| } | ||||
| func (m *Empty) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { | ||||
| 	return xxx_messageInfo_Empty.Marshal(b, m, deterministic) | ||||
| } | ||||
| func (m *Empty) XXX_Merge(src proto.Message) { | ||||
| 	xxx_messageInfo_Empty.Merge(m, src) | ||||
| } | ||||
| func (m *Empty) XXX_Size() int { | ||||
| 	return xxx_messageInfo_Empty.Size(m) | ||||
| } | ||||
| func (m *Empty) XXX_DiscardUnknown() { | ||||
| 	xxx_messageInfo_Empty.DiscardUnknown(m) | ||||
| } | ||||
|  | ||||
| var xxx_messageInfo_Empty proto.InternalMessageInfo | ||||
|  | ||||
| type PublishRequest struct { | ||||
| 	Topic                string   `protobuf:"bytes,1,opt,name=topic,proto3" json:"topic,omitempty"` | ||||
| 	Message              *Message `protobuf:"bytes,2,opt,name=message,proto3" json:"message,omitempty"` | ||||
| 	XXX_NoUnkeyedLiteral struct{} `json:"-"` | ||||
| 	XXX_unrecognized     []byte   `json:"-"` | ||||
| 	XXX_sizecache        int32    `json:"-"` | ||||
| } | ||||
|  | ||||
| func (m *PublishRequest) Reset()         { *m = PublishRequest{} } | ||||
| func (m *PublishRequest) String() string { return proto.CompactTextString(m) } | ||||
| func (*PublishRequest) ProtoMessage()    {} | ||||
| func (*PublishRequest) Descriptor() ([]byte, []int) { | ||||
| 	return fileDescriptor_df4d8f04292cf3fe, []int{1} | ||||
| } | ||||
|  | ||||
| func (m *PublishRequest) XXX_Unmarshal(b []byte) error { | ||||
| 	return xxx_messageInfo_PublishRequest.Unmarshal(m, b) | ||||
| } | ||||
| func (m *PublishRequest) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { | ||||
| 	return xxx_messageInfo_PublishRequest.Marshal(b, m, deterministic) | ||||
| } | ||||
| func (m *PublishRequest) XXX_Merge(src proto.Message) { | ||||
| 	xxx_messageInfo_PublishRequest.Merge(m, src) | ||||
| } | ||||
| func (m *PublishRequest) XXX_Size() int { | ||||
| 	return xxx_messageInfo_PublishRequest.Size(m) | ||||
| } | ||||
| func (m *PublishRequest) XXX_DiscardUnknown() { | ||||
| 	xxx_messageInfo_PublishRequest.DiscardUnknown(m) | ||||
| } | ||||
|  | ||||
| var xxx_messageInfo_PublishRequest proto.InternalMessageInfo | ||||
|  | ||||
| func (m *PublishRequest) GetTopic() string { | ||||
| 	if m != nil { | ||||
| 		return m.Topic | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (m *PublishRequest) GetMessage() *Message { | ||||
| 	if m != nil { | ||||
| 		return m.Message | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| type SubscribeRequest struct { | ||||
| 	Topic                string   `protobuf:"bytes,1,opt,name=topic,proto3" json:"topic,omitempty"` | ||||
| 	Queue                string   `protobuf:"bytes,2,opt,name=queue,proto3" json:"queue,omitempty"` | ||||
| 	XXX_NoUnkeyedLiteral struct{} `json:"-"` | ||||
| 	XXX_unrecognized     []byte   `json:"-"` | ||||
| 	XXX_sizecache        int32    `json:"-"` | ||||
| } | ||||
|  | ||||
| func (m *SubscribeRequest) Reset()         { *m = SubscribeRequest{} } | ||||
| func (m *SubscribeRequest) String() string { return proto.CompactTextString(m) } | ||||
| func (*SubscribeRequest) ProtoMessage()    {} | ||||
| func (*SubscribeRequest) Descriptor() ([]byte, []int) { | ||||
| 	return fileDescriptor_df4d8f04292cf3fe, []int{2} | ||||
| } | ||||
|  | ||||
| func (m *SubscribeRequest) XXX_Unmarshal(b []byte) error { | ||||
| 	return xxx_messageInfo_SubscribeRequest.Unmarshal(m, b) | ||||
| } | ||||
| func (m *SubscribeRequest) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { | ||||
| 	return xxx_messageInfo_SubscribeRequest.Marshal(b, m, deterministic) | ||||
| } | ||||
| func (m *SubscribeRequest) XXX_Merge(src proto.Message) { | ||||
| 	xxx_messageInfo_SubscribeRequest.Merge(m, src) | ||||
| } | ||||
| func (m *SubscribeRequest) XXX_Size() int { | ||||
| 	return xxx_messageInfo_SubscribeRequest.Size(m) | ||||
| } | ||||
| func (m *SubscribeRequest) XXX_DiscardUnknown() { | ||||
| 	xxx_messageInfo_SubscribeRequest.DiscardUnknown(m) | ||||
| } | ||||
|  | ||||
| var xxx_messageInfo_SubscribeRequest proto.InternalMessageInfo | ||||
|  | ||||
| func (m *SubscribeRequest) GetTopic() string { | ||||
| 	if m != nil { | ||||
| 		return m.Topic | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (m *SubscribeRequest) GetQueue() string { | ||||
| 	if m != nil { | ||||
| 		return m.Queue | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| type Message struct { | ||||
| 	Header               map[string]string `protobuf:"bytes,1,rep,name=header,proto3" json:"header,omitempty" protobuf_key:"bytes,1,opt,name=key,proto3" protobuf_val:"bytes,2,opt,name=value,proto3"` | ||||
| 	Body                 []byte            `protobuf:"bytes,2,opt,name=body,proto3" json:"body,omitempty"` | ||||
| 	XXX_NoUnkeyedLiteral struct{}          `json:"-"` | ||||
| 	XXX_unrecognized     []byte            `json:"-"` | ||||
| 	XXX_sizecache        int32             `json:"-"` | ||||
| } | ||||
|  | ||||
| func (m *Message) Reset()         { *m = Message{} } | ||||
| func (m *Message) String() string { return proto.CompactTextString(m) } | ||||
| func (*Message) ProtoMessage()    {} | ||||
| func (*Message) Descriptor() ([]byte, []int) { | ||||
| 	return fileDescriptor_df4d8f04292cf3fe, []int{3} | ||||
| } | ||||
|  | ||||
| func (m *Message) XXX_Unmarshal(b []byte) error { | ||||
| 	return xxx_messageInfo_Message.Unmarshal(m, b) | ||||
| } | ||||
| func (m *Message) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { | ||||
| 	return xxx_messageInfo_Message.Marshal(b, m, deterministic) | ||||
| } | ||||
| func (m *Message) XXX_Merge(src proto.Message) { | ||||
| 	xxx_messageInfo_Message.Merge(m, src) | ||||
| } | ||||
| func (m *Message) XXX_Size() int { | ||||
| 	return xxx_messageInfo_Message.Size(m) | ||||
| } | ||||
| func (m *Message) XXX_DiscardUnknown() { | ||||
| 	xxx_messageInfo_Message.DiscardUnknown(m) | ||||
| } | ||||
|  | ||||
| var xxx_messageInfo_Message proto.InternalMessageInfo | ||||
|  | ||||
| func (m *Message) GetHeader() map[string]string { | ||||
| 	if m != nil { | ||||
| 		return m.Header | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *Message) GetBody() []byte { | ||||
| 	if m != nil { | ||||
| 		return m.Body | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func init() { | ||||
| 	proto.RegisterType((*Empty)(nil), "go.micro.broker.Empty") | ||||
| 	proto.RegisterType((*PublishRequest)(nil), "go.micro.broker.PublishRequest") | ||||
| 	proto.RegisterType((*SubscribeRequest)(nil), "go.micro.broker.SubscribeRequest") | ||||
| 	proto.RegisterType((*Message)(nil), "go.micro.broker.Message") | ||||
| 	proto.RegisterMapType((map[string]string)(nil), "go.micro.broker.Message.HeaderEntry") | ||||
| } | ||||
|  | ||||
| func init() { proto.RegisterFile("broker/service/proto/broker.proto", fileDescriptor_df4d8f04292cf3fe) } | ||||
|  | ||||
| var fileDescriptor_df4d8f04292cf3fe = []byte{ | ||||
| 	// 299 bytes of a gzipped FileDescriptorProto | ||||
| 	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x7c, 0x51, 0x4d, 0x4b, 0xc3, 0x40, | ||||
| 	0x14, 0xec, 0xb6, 0xb6, 0xa1, 0xaf, 0xa2, 0x65, 0x29, 0x12, 0x7a, 0x31, 0x0d, 0x1e, 0x72, 0xda, | ||||
| 	0x48, 0xbc, 0xa8, 0x88, 0x07, 0xb1, 0xe0, 0x41, 0x41, 0xd6, 0x9b, 0xb7, 0x6c, 0xfa, 0x68, 0x43, | ||||
| 	0x1b, 0x37, 0xdd, 0x4d, 0x0a, 0xf9, 0x23, 0x9e, 0xfc, 0xb1, 0xd2, 0xdd, 0xf8, 0xd5, 0x50, 0x6f, | ||||
| 	0x6f, 0xde, 0xce, 0xce, 0x1b, 0x66, 0x60, 0x22, 0x94, 0x5c, 0xa2, 0x0a, 0x35, 0xaa, 0x4d, 0x9a, | ||||
| 	0x60, 0x98, 0x2b, 0x59, 0xc8, 0xd0, 0x2e, 0x99, 0x01, 0xf4, 0x78, 0x2e, 0x59, 0x96, 0x26, 0x4a, | ||||
| 	0x32, 0xbb, 0xf6, 0x1d, 0xe8, 0x4e, 0xb3, 0xbc, 0xa8, 0xfc, 0x57, 0x38, 0x7a, 0x2e, 0xc5, 0x2a, | ||||
| 	0xd5, 0x0b, 0x8e, 0xeb, 0x12, 0x75, 0x41, 0x47, 0xd0, 0x2d, 0x64, 0x9e, 0x26, 0x2e, 0xf1, 0x48, | ||||
| 	0xd0, 0xe7, 0x16, 0xd0, 0x08, 0x9c, 0x0c, 0xb5, 0x8e, 0xe7, 0xe8, 0xb6, 0x3d, 0x12, 0x0c, 0x22, | ||||
| 	0x97, 0xed, 0x68, 0xb2, 0x27, 0xfb, 0xce, 0xbf, 0x88, 0xfe, 0x2d, 0x0c, 0x5f, 0x4a, 0xa1, 0x13, | ||||
| 	0x95, 0x0a, 0xfc, 0x5f, 0x7d, 0x04, 0xdd, 0x75, 0x89, 0xa5, 0xd5, 0xee, 0x73, 0x0b, 0xfc, 0x77, | ||||
| 	0x02, 0x4e, 0x2d, 0x4a, 0x6f, 0xa0, 0xb7, 0xc0, 0x78, 0x86, 0xca, 0x25, 0x5e, 0x27, 0x18, 0x44, | ||||
| 	0x67, 0xfb, 0xce, 0xb3, 0x07, 0x43, 0x9b, 0xbe, 0x15, 0xaa, 0xe2, 0xf5, 0x1f, 0x4a, 0xe1, 0x40, | ||||
| 	0xc8, 0x59, 0x65, 0xe4, 0x0f, 0xb9, 0x99, 0xc7, 0x57, 0x30, 0xf8, 0x45, 0xa5, 0x43, 0xe8, 0x2c, | ||||
| 	0xb1, 0xaa, 0x6d, 0x6d, 0xc7, 0xad, 0xa9, 0x4d, 0xbc, 0xfa, 0x31, 0x65, 0xc0, 0x75, 0xfb, 0x92, | ||||
| 	0x44, 0x1f, 0x04, 0x7a, 0x77, 0xe6, 0x2a, 0xbd, 0x07, 0xa7, 0xce, 0x8f, 0x9e, 0x36, 0x2c, 0xfd, | ||||
| 	0x4d, 0x76, 0x7c, 0xd2, 0x20, 0xd8, 0x0e, 0x5a, 0xf4, 0x11, 0xfa, 0xdf, 0x49, 0xd1, 0x49, 0x83, | ||||
| 	0xb6, 0x9b, 0xe2, 0x78, 0x6f, 0xf8, 0x7e, 0xeb, 0x9c, 0x88, 0x9e, 0x29, 0xfd, 0xe2, 0x33, 0x00, | ||||
| 	0x00, 0xff, 0xff, 0x19, 0x9f, 0x10, 0x75, 0x19, 0x02, 0x00, 0x00, | ||||
| } | ||||
|  | ||||
| // Reference imports to suppress errors if they are not otherwise used. | ||||
| var _ context.Context | ||||
| var _ grpc.ClientConn | ||||
|  | ||||
| // 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.SupportPackageIsVersion4 | ||||
|  | ||||
| // BrokerClient is the client API for Broker service. | ||||
| // | ||||
| // For semantics around ctx use and closing/ending streaming RPCs, please refer to https://godoc.org/google.golang.org/grpc#ClientConn.NewStream. | ||||
| type BrokerClient interface { | ||||
| 	Publish(ctx context.Context, in *PublishRequest, opts ...grpc.CallOption) (*Empty, error) | ||||
| 	Subscribe(ctx context.Context, in *SubscribeRequest, opts ...grpc.CallOption) (Broker_SubscribeClient, error) | ||||
| } | ||||
|  | ||||
| type brokerClient struct { | ||||
| 	cc *grpc.ClientConn | ||||
| } | ||||
|  | ||||
| func NewBrokerClient(cc *grpc.ClientConn) BrokerClient { | ||||
| 	return &brokerClient{cc} | ||||
| } | ||||
|  | ||||
| func (c *brokerClient) Publish(ctx context.Context, in *PublishRequest, opts ...grpc.CallOption) (*Empty, error) { | ||||
| 	out := new(Empty) | ||||
| 	err := c.cc.Invoke(ctx, "/go.micro.broker.Broker/Publish", in, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *brokerClient) Subscribe(ctx context.Context, in *SubscribeRequest, opts ...grpc.CallOption) (Broker_SubscribeClient, error) { | ||||
| 	stream, err := c.cc.NewStream(ctx, &_Broker_serviceDesc.Streams[0], "/go.micro.broker.Broker/Subscribe", opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	x := &brokerSubscribeClient{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 Broker_SubscribeClient interface { | ||||
| 	Recv() (*Message, error) | ||||
| 	grpc.ClientStream | ||||
| } | ||||
|  | ||||
| type brokerSubscribeClient struct { | ||||
| 	grpc.ClientStream | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeClient) Recv() (*Message, error) { | ||||
| 	m := new(Message) | ||||
| 	if err := x.ClientStream.RecvMsg(m); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return m, nil | ||||
| } | ||||
|  | ||||
| // BrokerServer is the server API for Broker service. | ||||
| type BrokerServer interface { | ||||
| 	Publish(context.Context, *PublishRequest) (*Empty, error) | ||||
| 	Subscribe(*SubscribeRequest, Broker_SubscribeServer) error | ||||
| } | ||||
|  | ||||
| // UnimplementedBrokerServer can be embedded to have forward compatible implementations. | ||||
| type UnimplementedBrokerServer struct { | ||||
| } | ||||
|  | ||||
| func (*UnimplementedBrokerServer) Publish(ctx context.Context, req *PublishRequest) (*Empty, error) { | ||||
| 	return nil, status.Errorf(codes.Unimplemented, "method Publish not implemented") | ||||
| } | ||||
| func (*UnimplementedBrokerServer) Subscribe(req *SubscribeRequest, srv Broker_SubscribeServer) error { | ||||
| 	return status.Errorf(codes.Unimplemented, "method Subscribe not implemented") | ||||
| } | ||||
|  | ||||
| func RegisterBrokerServer(s *grpc.Server, srv BrokerServer) { | ||||
| 	s.RegisterService(&_Broker_serviceDesc, srv) | ||||
| } | ||||
|  | ||||
| func _Broker_Publish_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { | ||||
| 	in := new(PublishRequest) | ||||
| 	if err := dec(in); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if interceptor == nil { | ||||
| 		return srv.(BrokerServer).Publish(ctx, in) | ||||
| 	} | ||||
| 	info := &grpc.UnaryServerInfo{ | ||||
| 		Server:     srv, | ||||
| 		FullMethod: "/go.micro.broker.Broker/Publish", | ||||
| 	} | ||||
| 	handler := func(ctx context.Context, req interface{}) (interface{}, error) { | ||||
| 		return srv.(BrokerServer).Publish(ctx, req.(*PublishRequest)) | ||||
| 	} | ||||
| 	return interceptor(ctx, in, info, handler) | ||||
| } | ||||
|  | ||||
| func _Broker_Subscribe_Handler(srv interface{}, stream grpc.ServerStream) error { | ||||
| 	m := new(SubscribeRequest) | ||||
| 	if err := stream.RecvMsg(m); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return srv.(BrokerServer).Subscribe(m, &brokerSubscribeServer{stream}) | ||||
| } | ||||
|  | ||||
| type Broker_SubscribeServer interface { | ||||
| 	Send(*Message) error | ||||
| 	grpc.ServerStream | ||||
| } | ||||
|  | ||||
| type brokerSubscribeServer struct { | ||||
| 	grpc.ServerStream | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeServer) Send(m *Message) error { | ||||
| 	return x.ServerStream.SendMsg(m) | ||||
| } | ||||
|  | ||||
| var _Broker_serviceDesc = grpc.ServiceDesc{ | ||||
| 	ServiceName: "go.micro.broker.Broker", | ||||
| 	HandlerType: (*BrokerServer)(nil), | ||||
| 	Methods: []grpc.MethodDesc{ | ||||
| 		{ | ||||
| 			MethodName: "Publish", | ||||
| 			Handler:    _Broker_Publish_Handler, | ||||
| 		}, | ||||
| 	}, | ||||
| 	Streams: []grpc.StreamDesc{ | ||||
| 		{ | ||||
| 			StreamName:    "Subscribe", | ||||
| 			Handler:       _Broker_Subscribe_Handler, | ||||
| 			ServerStreams: true, | ||||
| 		}, | ||||
| 	}, | ||||
| 	Metadata: "broker/service/proto/broker.proto", | ||||
| } | ||||
| @@ -1,185 +0,0 @@ | ||||
| // Code generated by protoc-gen-micro. DO NOT EDIT. | ||||
| // source: broker/service/proto/broker.proto | ||||
|  | ||||
| package go_micro_broker | ||||
|  | ||||
| import ( | ||||
| 	fmt "fmt" | ||||
| 	proto "github.com/golang/protobuf/proto" | ||||
| 	math "math" | ||||
| ) | ||||
|  | ||||
| import ( | ||||
| 	context "context" | ||||
| 	api "github.com/micro/go-micro/v2/api" | ||||
| 	client "github.com/micro/go-micro/v2/client" | ||||
| 	server "github.com/micro/go-micro/v2/server" | ||||
| ) | ||||
|  | ||||
| // 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.ProtoPackageIsVersion3 // please upgrade the proto package | ||||
|  | ||||
| // Reference imports to suppress errors if they are not otherwise used. | ||||
| var _ api.Endpoint | ||||
| var _ context.Context | ||||
| var _ client.Option | ||||
| var _ server.Option | ||||
|  | ||||
| // Api Endpoints for Broker service | ||||
|  | ||||
| func NewBrokerEndpoints() []*api.Endpoint { | ||||
| 	return []*api.Endpoint{} | ||||
| } | ||||
|  | ||||
| // Client API for Broker service | ||||
|  | ||||
| type BrokerService interface { | ||||
| 	Publish(ctx context.Context, in *PublishRequest, opts ...client.CallOption) (*Empty, error) | ||||
| 	Subscribe(ctx context.Context, in *SubscribeRequest, opts ...client.CallOption) (Broker_SubscribeService, error) | ||||
| } | ||||
|  | ||||
| type brokerService struct { | ||||
| 	c    client.Client | ||||
| 	name string | ||||
| } | ||||
|  | ||||
| func NewBrokerService(name string, c client.Client) BrokerService { | ||||
| 	return &brokerService{ | ||||
| 		c:    c, | ||||
| 		name: name, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *brokerService) Publish(ctx context.Context, in *PublishRequest, opts ...client.CallOption) (*Empty, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Broker.Publish", in) | ||||
| 	out := new(Empty) | ||||
| 	err := c.c.Call(ctx, req, out, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return out, nil | ||||
| } | ||||
|  | ||||
| func (c *brokerService) Subscribe(ctx context.Context, in *SubscribeRequest, opts ...client.CallOption) (Broker_SubscribeService, error) { | ||||
| 	req := c.c.NewRequest(c.name, "Broker.Subscribe", &SubscribeRequest{}) | ||||
| 	stream, err := c.c.Stream(ctx, req, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err := stream.Send(in); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &brokerServiceSubscribe{stream}, nil | ||||
| } | ||||
|  | ||||
| type Broker_SubscribeService interface { | ||||
| 	Context() context.Context | ||||
| 	SendMsg(interface{}) error | ||||
| 	RecvMsg(interface{}) error | ||||
| 	Close() error | ||||
| 	Recv() (*Message, error) | ||||
| } | ||||
|  | ||||
| type brokerServiceSubscribe struct { | ||||
| 	stream client.Stream | ||||
| } | ||||
|  | ||||
| func (x *brokerServiceSubscribe) Close() error { | ||||
| 	return x.stream.Close() | ||||
| } | ||||
|  | ||||
| func (x *brokerServiceSubscribe) Context() context.Context { | ||||
| 	return x.stream.Context() | ||||
| } | ||||
|  | ||||
| func (x *brokerServiceSubscribe) SendMsg(m interface{}) error { | ||||
| 	return x.stream.Send(m) | ||||
| } | ||||
|  | ||||
| func (x *brokerServiceSubscribe) RecvMsg(m interface{}) error { | ||||
| 	return x.stream.Recv(m) | ||||
| } | ||||
|  | ||||
| func (x *brokerServiceSubscribe) Recv() (*Message, error) { | ||||
| 	m := new(Message) | ||||
| 	err := x.stream.Recv(m) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return m, nil | ||||
| } | ||||
|  | ||||
| // Server API for Broker service | ||||
|  | ||||
| type BrokerHandler interface { | ||||
| 	Publish(context.Context, *PublishRequest, *Empty) error | ||||
| 	Subscribe(context.Context, *SubscribeRequest, Broker_SubscribeStream) error | ||||
| } | ||||
|  | ||||
| func RegisterBrokerHandler(s server.Server, hdlr BrokerHandler, opts ...server.HandlerOption) error { | ||||
| 	type broker interface { | ||||
| 		Publish(ctx context.Context, in *PublishRequest, out *Empty) error | ||||
| 		Subscribe(ctx context.Context, stream server.Stream) error | ||||
| 	} | ||||
| 	type Broker struct { | ||||
| 		broker | ||||
| 	} | ||||
| 	h := &brokerHandler{hdlr} | ||||
| 	return s.Handle(s.NewHandler(&Broker{h}, opts...)) | ||||
| } | ||||
|  | ||||
| type brokerHandler struct { | ||||
| 	BrokerHandler | ||||
| } | ||||
|  | ||||
| func (h *brokerHandler) Publish(ctx context.Context, in *PublishRequest, out *Empty) error { | ||||
| 	return h.BrokerHandler.Publish(ctx, in, out) | ||||
| } | ||||
|  | ||||
| func (h *brokerHandler) Subscribe(ctx context.Context, stream server.Stream) error { | ||||
| 	m := new(SubscribeRequest) | ||||
| 	if err := stream.Recv(m); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return h.BrokerHandler.Subscribe(ctx, m, &brokerSubscribeStream{stream}) | ||||
| } | ||||
|  | ||||
| type Broker_SubscribeStream interface { | ||||
| 	Context() context.Context | ||||
| 	SendMsg(interface{}) error | ||||
| 	RecvMsg(interface{}) error | ||||
| 	Close() error | ||||
| 	Send(*Message) error | ||||
| } | ||||
|  | ||||
| type brokerSubscribeStream struct { | ||||
| 	stream server.Stream | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeStream) Close() error { | ||||
| 	return x.stream.Close() | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeStream) Context() context.Context { | ||||
| 	return x.stream.Context() | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeStream) SendMsg(m interface{}) error { | ||||
| 	return x.stream.Send(m) | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeStream) RecvMsg(m interface{}) error { | ||||
| 	return x.stream.Recv(m) | ||||
| } | ||||
|  | ||||
| func (x *brokerSubscribeStream) Send(m *Message) error { | ||||
| 	return x.stream.Send(m) | ||||
| } | ||||
| @@ -1,25 +0,0 @@ | ||||
| syntax = "proto3"; | ||||
|  | ||||
| package go.micro.broker; | ||||
|  | ||||
| service Broker { | ||||
| 	rpc Publish(PublishRequest) returns (Empty) {}; | ||||
| 	rpc Subscribe(SubscribeRequest) returns (stream Message) {}; | ||||
| } | ||||
|  | ||||
| message Empty {} | ||||
|  | ||||
| message PublishRequest { | ||||
| 	string topic = 1; | ||||
| 	Message message = 2; | ||||
| } | ||||
|  | ||||
| message SubscribeRequest { | ||||
| 	string topic = 1; | ||||
| 	string queue = 2; | ||||
| } | ||||
|  | ||||
| message Message { | ||||
| 	map<string,string> header = 1; | ||||
| 	bytes body = 2; | ||||
| } | ||||
| @@ -1,146 +0,0 @@ | ||||
| // Package service provides the broker service client | ||||
| package service | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	pb "github.com/micro/go-micro/v2/broker/service/proto" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| ) | ||||
|  | ||||
| type serviceBroker struct { | ||||
| 	Addrs   []string | ||||
| 	Client  pb.BrokerService | ||||
| 	options broker.Options | ||||
| } | ||||
|  | ||||
| var ( | ||||
| 	DefaultName = "go.micro.broker" | ||||
| ) | ||||
|  | ||||
| func (b *serviceBroker) Address() string { | ||||
| 	return b.Addrs[0] | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) Connect() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) Disconnect() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) Init(opts ...broker.Option) error { | ||||
| 	for _, o := range opts { | ||||
| 		o(&b.options) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) Options() broker.Options { | ||||
| 	return b.options | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) Publish(topic string, msg *broker.Message, opts ...broker.PublishOption) error { | ||||
| 	if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 		logger.Debugf("Publishing to topic %s broker %v", topic, b.Addrs) | ||||
| 	} | ||||
| 	_, err := b.Client.Publish(context.TODO(), &pb.PublishRequest{ | ||||
| 		Topic: topic, | ||||
| 		Message: &pb.Message{ | ||||
| 			Header: msg.Header, | ||||
| 			Body:   msg.Body, | ||||
| 		}, | ||||
| 	}, client.WithAddress(b.Addrs...)) | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) Subscribe(topic string, handler broker.Handler, opts ...broker.SubscribeOption) (broker.Subscriber, error) { | ||||
| 	var options broker.SubscribeOptions | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
| 	if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 		logger.Debugf("Subscribing to topic %s queue %s broker %v", topic, options.Queue, b.Addrs) | ||||
| 	} | ||||
| 	stream, err := b.Client.Subscribe(context.TODO(), &pb.SubscribeRequest{ | ||||
| 		Topic: topic, | ||||
| 		Queue: options.Queue, | ||||
| 	}, client.WithAddress(b.Addrs...), client.WithRequestTimeout(time.Hour)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	sub := &serviceSub{ | ||||
| 		topic:   topic, | ||||
| 		queue:   options.Queue, | ||||
| 		handler: handler, | ||||
| 		stream:  stream, | ||||
| 		closed:  make(chan bool), | ||||
| 		options: options, | ||||
| 	} | ||||
|  | ||||
| 	go func() { | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-sub.closed: | ||||
| 				if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 					logger.Debugf("Unsubscribed from topic %s", topic) | ||||
| 				} | ||||
| 				return | ||||
| 			default: | ||||
| 				if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 					// run the subscriber | ||||
| 					logger.Debugf("Streaming from broker %v to topic [%s] queue [%s]", b.Addrs, topic, options.Queue) | ||||
| 				} | ||||
| 				if err := sub.run(); err != nil { | ||||
| 					if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 						logger.Debugf("Resubscribing to topic %s broker %v", topic, b.Addrs) | ||||
| 					} | ||||
| 					stream, err := b.Client.Subscribe(context.TODO(), &pb.SubscribeRequest{ | ||||
| 						Topic: topic, | ||||
| 						Queue: options.Queue, | ||||
| 					}, client.WithAddress(b.Addrs...), client.WithRequestTimeout(time.Hour)) | ||||
| 					if err != nil { | ||||
| 						if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 							logger.Debugf("Failed to resubscribe to topic %s: %v", topic, err) | ||||
| 						} | ||||
| 						time.Sleep(time.Second) | ||||
| 						continue | ||||
| 					} | ||||
| 					// new stream | ||||
| 					sub.stream = stream | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	return sub, nil | ||||
| } | ||||
|  | ||||
| func (b *serviceBroker) String() string { | ||||
| 	return "service" | ||||
| } | ||||
|  | ||||
| func NewBroker(opts ...broker.Option) broker.Broker { | ||||
| 	var options broker.Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	addrs := options.Addrs | ||||
| 	if len(addrs) == 0 { | ||||
| 		addrs = []string{"127.0.0.1:8001"} | ||||
| 	} | ||||
|  | ||||
| 	cli := client.DefaultClient | ||||
|  | ||||
| 	return &serviceBroker{ | ||||
| 		Addrs:   addrs, | ||||
| 		Client:  pb.NewBrokerService(DefaultName, cli), | ||||
| 		options: options, | ||||
| 	} | ||||
| } | ||||
| @@ -1,108 +0,0 @@ | ||||
| package service | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	pb "github.com/micro/go-micro/v2/broker/service/proto" | ||||
| 	"github.com/micro/go-micro/v2/logger" | ||||
| ) | ||||
|  | ||||
| type serviceSub struct { | ||||
| 	topic   string | ||||
| 	queue   string | ||||
| 	handler broker.Handler | ||||
| 	stream  pb.Broker_SubscribeService | ||||
| 	closed  chan bool | ||||
| 	options broker.SubscribeOptions | ||||
| } | ||||
|  | ||||
| type serviceEvent struct { | ||||
| 	topic   string | ||||
| 	err     error | ||||
| 	message *broker.Message | ||||
| } | ||||
|  | ||||
| func (s *serviceEvent) Topic() string { | ||||
| 	return s.topic | ||||
| } | ||||
|  | ||||
| func (s *serviceEvent) Message() *broker.Message { | ||||
| 	return s.message | ||||
| } | ||||
|  | ||||
| func (s *serviceEvent) Ack() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (s *serviceEvent) Error() error { | ||||
| 	return s.err | ||||
| } | ||||
|  | ||||
| func (s *serviceSub) isClosed() bool { | ||||
| 	select { | ||||
| 	case <-s.closed: | ||||
| 		return true | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *serviceSub) run() error { | ||||
| 	exit := make(chan bool) | ||||
| 	go func() { | ||||
| 		select { | ||||
| 		case <-exit: | ||||
| 		case <-s.closed: | ||||
| 		} | ||||
|  | ||||
| 		// close the stream | ||||
| 		s.stream.Close() | ||||
| 	}() | ||||
|  | ||||
| 	for { | ||||
| 		// TODO: do not fail silently | ||||
| 		msg, err := s.stream.Recv() | ||||
| 		if err != nil { | ||||
| 			if logger.V(logger.DebugLevel, logger.DefaultLogger) { | ||||
| 				logger.Debugf("Streaming error for subcription to topic %s: %v", s.Topic(), err) | ||||
| 			} | ||||
|  | ||||
| 			// close the exit channel | ||||
| 			close(exit) | ||||
|  | ||||
| 			// don't return an error if we unsubscribed | ||||
| 			if s.isClosed() { | ||||
| 				return nil | ||||
| 			} | ||||
|  | ||||
| 			// return stream error | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		p := &serviceEvent{ | ||||
| 			topic: s.topic, | ||||
| 			message: &broker.Message{ | ||||
| 				Header: msg.Header, | ||||
| 				Body:   msg.Body, | ||||
| 			}, | ||||
| 		} | ||||
| 		p.err = s.handler(p) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (s *serviceSub) Options() broker.SubscribeOptions { | ||||
| 	return s.options | ||||
| } | ||||
|  | ||||
| func (s *serviceSub) Topic() string { | ||||
| 	return s.topic | ||||
| } | ||||
|  | ||||
| func (s *serviceSub) Unsubscribe() error { | ||||
| 	select { | ||||
| 	case <-s.closed: | ||||
| 		return nil | ||||
| 	default: | ||||
| 		close(s.closed) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										29
									
								
								cache/cache.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								cache/cache.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Package cache is a caching interface | ||||
| package cache | ||||
|  | ||||
| // Cache is an interface for caching | ||||
| type Cache interface { | ||||
| 	// Initialise options | ||||
| 	Init(...Option) error | ||||
| 	// Get a value | ||||
| 	Get(key string) (interface{}, error) | ||||
| 	// Set a value | ||||
| 	Set(key string, val interface{}) error | ||||
| 	// Delete a value | ||||
| 	Delete(key string) error | ||||
| 	// Name of the implementation | ||||
| 	String() string | ||||
| } | ||||
|  | ||||
| type Options struct { | ||||
| 	Nodes []string | ||||
| } | ||||
|  | ||||
| type Option func(o *Options) | ||||
|  | ||||
| // Nodes sets the nodes for the cache | ||||
| func Nodes(v ...string) Option { | ||||
| 	return func(o *Options) { | ||||
| 		o.Nodes = v | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										80
									
								
								cache/memcache/memcache.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								cache/memcache/memcache.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| // Package memcache is a memcache implementation of the Cache | ||||
| package memcache | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
|  | ||||
| 	"github.com/bradfitz/gomemcache/memcache" | ||||
| 	"github.com/micro/go-micro/v3/cache" | ||||
| ) | ||||
|  | ||||
| type memcacheCache struct { | ||||
| 	options cache.Options | ||||
| 	client  *memcache.Client | ||||
| } | ||||
|  | ||||
| type memcacheItem struct { | ||||
| 	Key   string | ||||
| 	Value interface{} | ||||
| } | ||||
|  | ||||
| func (m *memcacheCache) Init(opts ...cache.Option) error { | ||||
| 	for _, o := range opts { | ||||
| 		o(&m.options) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *memcacheCache) Get(key string) (interface{}, error) { | ||||
| 	item, err := m.client.Get(key) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var mc *memcacheItem | ||||
|  | ||||
| 	if err := json.Unmarshal(item.Value, &mc); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return mc.Value, nil | ||||
| } | ||||
|  | ||||
| func (m *memcacheCache) Set(key string, val interface{}) error { | ||||
| 	b, err := json.Marshal(val) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return m.client.Set(&memcache.Item{ | ||||
| 		Key:   key, | ||||
| 		Value: b, | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func (m *memcacheCache) Delete(key string) error { | ||||
| 	return m.client.Delete(key) | ||||
| } | ||||
|  | ||||
| func (m *memcacheCache) String() string { | ||||
| 	return "memcache" | ||||
| } | ||||
|  | ||||
| // NewCache returns a new memcache Cache | ||||
| func NewCache(opts ...cache.Option) cache.Cache { | ||||
| 	var options cache.Options | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
|  | ||||
| 	// get and set the nodes | ||||
| 	nodes := options.Nodes | ||||
| 	if len(nodes) == 0 { | ||||
| 		nodes = []string{"localhost:11211"} | ||||
| 	} | ||||
|  | ||||
| 	return &memcacheCache{ | ||||
| 		options: options, | ||||
| 		client:  memcache.New(nodes...), | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										56
									
								
								cache/memory/memory.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								cache/memory/memory.go
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| // Package memory is an in memory cache | ||||
| package memory | ||||
|  | ||||
| import ( | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v3/cache" | ||||
| 	"github.com/micro/go-micro/v3/errors" | ||||
| ) | ||||
|  | ||||
| type memoryCache struct { | ||||
| 	// TODO: use a decent caching library | ||||
| 	sync.RWMutex | ||||
| 	values map[string]interface{} | ||||
| } | ||||
|  | ||||
| func (m *memoryCache) Init(opts ...cache.Option) error { | ||||
| 	// TODO: implement | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *memoryCache) Get(key string) (interface{}, error) { | ||||
| 	m.RLock() | ||||
| 	defer m.RUnlock() | ||||
|  | ||||
| 	v, ok := m.values[key] | ||||
| 	if !ok { | ||||
| 		return nil, errors.NotFound("go.micro.cache", key+" not found") | ||||
| 	} | ||||
|  | ||||
| 	return v, nil | ||||
| } | ||||
|  | ||||
| func (m *memoryCache) Set(key string, val interface{}) error { | ||||
| 	m.Lock() | ||||
| 	m.values[key] = val | ||||
| 	m.Unlock() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *memoryCache) Delete(key string) error { | ||||
| 	m.Lock() | ||||
| 	delete(m.values, key) | ||||
| 	m.Unlock() | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (m *memoryCache) String() string { | ||||
| 	return "memory" | ||||
| } | ||||
|  | ||||
| func NewCache(opts ...cache.Option) cache.Cache { | ||||
| 	return &memoryCache{ | ||||
| 		values: make(map[string]interface{}), | ||||
| 	} | ||||
| } | ||||
| @@ -4,7 +4,7 @@ import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/util/backoff" | ||||
| 	"github.com/micro/go-micro/v3/util/backoff" | ||||
| ) | ||||
|  | ||||
| type BackoffFunc func(ctx context.Context, req Request, attempts int) (time.Duration, error) | ||||
|   | ||||
| @@ -16,10 +16,13 @@ func TestBackoff(t *testing.T) { | ||||
| 		7900 * time.Millisecond, | ||||
| 	} | ||||
|  | ||||
| 	c := NewClient() | ||||
| 	r := &testRequest{ | ||||
| 		service: "test", | ||||
| 		method:  "test", | ||||
| 	} | ||||
|  | ||||
| 	for i := 0; i < 5; i++ { | ||||
| 		d, err := exponentialBackoff(context.TODO(), c.NewRequest("test", "test", nil), i) | ||||
| 		d, err := exponentialBackoff(context.TODO(), r, i) | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|   | ||||
							
								
								
									
										66
									
								
								client/cache.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								client/cache.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | ||||
| package client | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"hash/fnv" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v3/metadata" | ||||
| 	cache "github.com/patrickmn/go-cache" | ||||
| ) | ||||
|  | ||||
| // NewCache returns an initialised cache. | ||||
| func NewCache() *Cache { | ||||
| 	return &Cache{ | ||||
| 		cache: cache.New(cache.NoExpiration, 30*time.Second), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Cache for responses | ||||
| type Cache struct { | ||||
| 	cache *cache.Cache | ||||
| } | ||||
|  | ||||
| // Get a response from the cache | ||||
| func (c *Cache) Get(ctx context.Context, req Request) (interface{}, bool) { | ||||
| 	return c.cache.Get(key(ctx, req)) | ||||
| } | ||||
|  | ||||
| // Set a response in the cache | ||||
| func (c *Cache) Set(ctx context.Context, req Request, rsp interface{}, expiry time.Duration) { | ||||
| 	c.cache.Set(key(ctx, req), rsp, expiry) | ||||
| } | ||||
|  | ||||
| // List the key value pairs in the cache | ||||
| func (c *Cache) List() map[string]string { | ||||
| 	items := c.cache.Items() | ||||
|  | ||||
| 	rsp := make(map[string]string, len(items)) | ||||
| 	for k, v := range items { | ||||
| 		bytes, _ := json.Marshal(v.Object) | ||||
| 		rsp[k] = string(bytes) | ||||
| 	} | ||||
|  | ||||
| 	return rsp | ||||
| } | ||||
|  | ||||
| // key returns a hash for the context and request | ||||
| func key(ctx context.Context, req Request) string { | ||||
| 	ns, _ := metadata.Get(ctx, "Micro-Namespace") | ||||
|  | ||||
| 	bytes, _ := json.Marshal(map[string]interface{}{ | ||||
| 		"namespace": ns, | ||||
| 		"request": map[string]interface{}{ | ||||
| 			"service":  req.Service(), | ||||
| 			"endpoint": req.Endpoint(), | ||||
| 			"method":   req.Method(), | ||||
| 			"body":     req.Body(), | ||||
| 		}, | ||||
| 	}) | ||||
|  | ||||
| 	h := fnv.New64() | ||||
| 	h.Write(bytes) | ||||
| 	return fmt.Sprintf("%x", h.Sum(nil)) | ||||
| } | ||||
							
								
								
									
										77
									
								
								client/cache_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								client/cache_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| package client | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v3/metadata" | ||||
| ) | ||||
|  | ||||
| func TestCache(t *testing.T) { | ||||
| 	ctx := context.TODO() | ||||
| 	req := &testRequest{service: "go.micro.service.foo", method: "Foo.Bar"} | ||||
|  | ||||
| 	t.Run("CacheMiss", func(t *testing.T) { | ||||
| 		if _, ok := NewCache().Get(ctx, req); ok { | ||||
| 			t.Errorf("Expected to get no result from Get") | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("CacheHit", func(t *testing.T) { | ||||
| 		c := NewCache() | ||||
|  | ||||
| 		rsp := "theresponse" | ||||
| 		c.Set(ctx, req, rsp, time.Minute) | ||||
|  | ||||
| 		if res, ok := c.Get(ctx, req); !ok { | ||||
| 			t.Errorf("Expected a result, got nothing") | ||||
| 		} else if res != rsp { | ||||
| 			t.Errorf("Expected '%v' result, got '%v'", rsp, res) | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestCacheKey(t *testing.T) { | ||||
| 	ctx := context.TODO() | ||||
|  | ||||
| 	req1 := &testRequest{service: "go.micro.service.foo", method: "Foo.Bar"} | ||||
| 	req2 := &testRequest{service: "go.micro.service.foo", method: "Foo.Baz"} | ||||
| 	req3 := &testRequest{service: "go.micro.service.foo", method: "Foo.Bar", body: "customquery"} | ||||
|  | ||||
| 	t.Run("IdenticalRequests", func(t *testing.T) { | ||||
| 		key1 := key(ctx, req1) | ||||
| 		key2 := key(ctx, req1) | ||||
| 		if key1 != key2 { | ||||
| 			t.Errorf("Expected the keys to match for identical requests and context") | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("DifferentRequestEndpoints", func(t *testing.T) { | ||||
| 		key1 := key(ctx, req1) | ||||
| 		key2 := key(ctx, req2) | ||||
|  | ||||
| 		if key1 == key2 { | ||||
| 			t.Errorf("Expected the keys to differ for different request endpoints") | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("DifferentRequestBody", func(t *testing.T) { | ||||
| 		key1 := key(ctx, req2) | ||||
| 		key2 := key(ctx, req3) | ||||
|  | ||||
| 		if key1 == key2 { | ||||
| 			t.Errorf("Expected the keys to differ for different request bodies") | ||||
| 		} | ||||
| 	}) | ||||
|  | ||||
| 	t.Run("DifferentMetadata", func(t *testing.T) { | ||||
| 		mdCtx := metadata.Set(context.TODO(), "Micro-Namespace", "bar") | ||||
| 		key1 := key(mdCtx, req1) | ||||
| 		key2 := key(ctx, req1) | ||||
|  | ||||
| 		if key1 == key2 { | ||||
| 			t.Errorf("Expected the keys to differ for different metadata") | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"context" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/codec" | ||||
| 	"github.com/micro/go-micro/v3/codec" | ||||
| ) | ||||
|  | ||||
| // Client is the interface used to make requests to services. | ||||
| @@ -22,11 +22,6 @@ type Client interface { | ||||
| 	String() string | ||||
| } | ||||
|  | ||||
| // Router manages request routing | ||||
| type Router interface { | ||||
| 	SendRequest(context.Context, Request) (Response, error) | ||||
| } | ||||
|  | ||||
| // Message is the interface for publishing asynchronously | ||||
| type Message interface { | ||||
| 	Topic() string | ||||
| @@ -62,7 +57,7 @@ type Response interface { | ||||
| 	Read() ([]byte, error) | ||||
| } | ||||
|  | ||||
| // Stream is the inteface for a bidirectional synchronous stream | ||||
| // Stream is the interface for a bidirectional synchronous stream | ||||
| type Stream interface { | ||||
| 	// Context for the stream | ||||
| 	Context() context.Context | ||||
| @@ -96,8 +91,6 @@ type MessageOption func(*MessageOptions) | ||||
| type RequestOption func(*RequestOptions) | ||||
|  | ||||
| var ( | ||||
| 	// DefaultClient is a default client to use out of the box | ||||
| 	DefaultClient Client = newRpcClient() | ||||
| 	// DefaultBackoff is the default backoff function for retries | ||||
| 	DefaultBackoff = exponentialBackoff | ||||
| 	// DefaultRetry is the default check-for-retry function for retries | ||||
| @@ -110,39 +103,4 @@ var ( | ||||
| 	DefaultPoolSize = 100 | ||||
| 	// DefaultPoolTTL sets the connection pool ttl | ||||
| 	DefaultPoolTTL = time.Minute | ||||
|  | ||||
| 	// NewClient returns a new client | ||||
| 	NewClient func(...Option) Client = newRpcClient | ||||
| ) | ||||
|  | ||||
| // Makes a synchronous call to a service using the default client | ||||
| func Call(ctx context.Context, request Request, response interface{}, opts ...CallOption) error { | ||||
| 	return DefaultClient.Call(ctx, request, response, opts...) | ||||
| } | ||||
|  | ||||
| // Publishes a publication using the default client. Using the underlying broker | ||||
| // set within the options. | ||||
| func Publish(ctx context.Context, msg Message, opts ...PublishOption) error { | ||||
| 	return DefaultClient.Publish(ctx, msg, opts...) | ||||
| } | ||||
|  | ||||
| // Creates a new message using the default client | ||||
| func NewMessage(topic string, payload interface{}, opts ...MessageOption) Message { | ||||
| 	return DefaultClient.NewMessage(topic, payload, opts...) | ||||
| } | ||||
|  | ||||
| // Creates a new request using the default client. Content Type will | ||||
| // be set to the default within options and use the appropriate codec | ||||
| func NewRequest(service, endpoint string, request interface{}, reqOpts ...RequestOption) Request { | ||||
| 	return DefaultClient.NewRequest(service, endpoint, request, reqOpts...) | ||||
| } | ||||
|  | ||||
| // Creates a streaming connection with a service and returns responses on the | ||||
| // channel passed in. It's up to the user to close the streamer. | ||||
| func NewStream(ctx context.Context, request Request, opts ...CallOption) (Stream, error) { | ||||
| 	return DefaultClient.Stream(ctx, request, opts...) | ||||
| } | ||||
|  | ||||
| func String() string { | ||||
| 	return DefaultClient.String() | ||||
| } | ||||
|   | ||||
| @@ -9,8 +9,8 @@ import ( | ||||
|  | ||||
| 	"github.com/golang/protobuf/jsonpb" | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/micro/go-micro/v2/codec" | ||||
| 	"github.com/micro/go-micro/v2/codec/bytes" | ||||
| 	"github.com/micro/go-micro/v3/codec" | ||||
| 	"github.com/micro/go-micro/v3/codec/bytes" | ||||
| 	"github.com/oxtoacart/bpool" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/encoding" | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| package grpc | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/errors" | ||||
| 	"github.com/micro/go-micro/v3/errors" | ||||
| 	"google.golang.org/grpc/status" | ||||
| ) | ||||
|  | ||||
| @@ -24,7 +24,9 @@ func microError(err error) error { | ||||
|  | ||||
| 	// return first error from details | ||||
| 	if details := s.Details(); len(details) > 0 { | ||||
| 		return microError(details[0].(error)) | ||||
| 		if verr, ok := details[0].(error); ok { | ||||
| 			return microError(verr) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// try to decode micro *errors.Error | ||||
|   | ||||
| @@ -6,18 +6,17 @@ import ( | ||||
| 	"crypto/tls" | ||||
| 	"fmt" | ||||
| 	"net" | ||||
| 	"reflect" | ||||
| 	"strings" | ||||
| 	"sync/atomic" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/broker" | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	raw "github.com/micro/go-micro/v2/codec/bytes" | ||||
| 	"github.com/micro/go-micro/v2/errors" | ||||
| 	"github.com/micro/go-micro/v2/metadata" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	pnet "github.com/micro/go-micro/v2/util/net" | ||||
| 	"github.com/micro/go-micro/v3/broker" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	raw "github.com/micro/go-micro/v3/codec/bytes" | ||||
| 	"github.com/micro/go-micro/v3/errors" | ||||
| 	"github.com/micro/go-micro/v3/metadata" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
|  | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/credentials" | ||||
| @@ -71,35 +70,9 @@ func (g *grpcClient) secure(addr string) grpc.DialOption { | ||||
| 	return grpc.WithInsecure() | ||||
| } | ||||
|  | ||||
| func (g *grpcClient) next(request client.Request, opts client.CallOptions) (selector.Next, error) { | ||||
| 	service, address, _ := pnet.Proxy(request.Service(), opts.Address) | ||||
|  | ||||
| 	// return remote address | ||||
| 	if len(address) > 0 { | ||||
| 		return func() (*registry.Node, error) { | ||||
| 			return ®istry.Node{ | ||||
| 				Address: address[0], | ||||
| 			}, nil | ||||
| 		}, nil | ||||
| 	} | ||||
|  | ||||
| 	// get next nodes from the selector | ||||
| 	next, err := g.opts.Selector.Select(service, opts.SelectOptions...) | ||||
| 	if err != nil { | ||||
| 		if err == selector.ErrNotFound { | ||||
| 			return nil, errors.InternalServerError("go.micro.client", "service %s: %s", service, err.Error()) | ||||
| 		} | ||||
| 		return nil, errors.InternalServerError("go.micro.client", "error selecting %s node: %s", service, err.Error()) | ||||
| 	} | ||||
|  | ||||
| 	return next, nil | ||||
| } | ||||
|  | ||||
| func (g *grpcClient) call(ctx context.Context, node *registry.Node, req client.Request, rsp interface{}, opts client.CallOptions) error { | ||||
| 	var header map[string]string | ||||
|  | ||||
| 	address := node.Address | ||||
|  | ||||
| 	header = make(map[string]string) | ||||
| 	if md, ok := metadata.FromContext(ctx); ok { | ||||
| 		header = make(map[string]string, len(md)) | ||||
| @@ -130,7 +103,7 @@ func (g *grpcClient) call(ctx context.Context, node *registry.Node, req client.R | ||||
|  | ||||
| 	grpcDialOptions := []grpc.DialOption{ | ||||
| 		grpc.WithTimeout(opts.DialTimeout), | ||||
| 		g.secure(address), | ||||
| 		g.secure(node.Address), | ||||
| 		grpc.WithDefaultCallOptions( | ||||
| 			grpc.MaxCallRecvMsgSize(maxRecvMsgSize), | ||||
| 			grpc.MaxCallSendMsgSize(maxSendMsgSize), | ||||
| @@ -141,13 +114,13 @@ func (g *grpcClient) call(ctx context.Context, node *registry.Node, req client.R | ||||
| 		grpcDialOptions = append(grpcDialOptions, opts...) | ||||
| 	} | ||||
|  | ||||
| 	cc, err := g.pool.getConn(address, grpcDialOptions...) | ||||
| 	cc, err := g.pool.getConn(node.Address, grpcDialOptions...) | ||||
| 	if err != nil { | ||||
| 		return errors.InternalServerError("go.micro.client", fmt.Sprintf("Error sending request: %v", err)) | ||||
| 	} | ||||
| 	defer func() { | ||||
| 		// defer execution of release | ||||
| 		g.pool.release(address, cc, grr) | ||||
| 		g.pool.release(node.Address, cc, grr) | ||||
| 	}() | ||||
|  | ||||
| 	ch := make(chan error, 1) | ||||
| @@ -173,11 +146,9 @@ func (g *grpcClient) call(ctx context.Context, node *registry.Node, req client.R | ||||
| 	return grr | ||||
| } | ||||
|  | ||||
| func (g *grpcClient) stream(ctx context.Context, node *registry.Node, req client.Request, opts client.CallOptions) (client.Stream, error) { | ||||
| func (g *grpcClient) stream(ctx context.Context, node *registry.Node, req client.Request, rsp interface{}, opts client.CallOptions) error { | ||||
| 	var header map[string]string | ||||
|  | ||||
| 	address := node.Address | ||||
|  | ||||
| 	if md, ok := metadata.FromContext(ctx); ok { | ||||
| 		header = make(map[string]string, len(md)) | ||||
| 		for k, v := range md { | ||||
| @@ -199,7 +170,7 @@ func (g *grpcClient) stream(ctx context.Context, node *registry.Node, req client | ||||
|  | ||||
| 	cf, err := g.newGRPCCodec(req.ContentType()) | ||||
| 	if err != nil { | ||||
| 		return nil, errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 		return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 	} | ||||
|  | ||||
| 	var dialCtx context.Context | ||||
| @@ -215,16 +186,16 @@ func (g *grpcClient) stream(ctx context.Context, node *registry.Node, req client | ||||
|  | ||||
| 	grpcDialOptions := []grpc.DialOption{ | ||||
| 		grpc.WithTimeout(opts.DialTimeout), | ||||
| 		g.secure(address), | ||||
| 		g.secure(node.Address), | ||||
| 	} | ||||
|  | ||||
| 	if opts := g.getGrpcDialOptions(); opts != nil { | ||||
| 		grpcDialOptions = append(grpcDialOptions, opts...) | ||||
| 	} | ||||
|  | ||||
| 	cc, err := grpc.DialContext(dialCtx, address, grpcDialOptions...) | ||||
| 	cc, err := grpc.DialContext(dialCtx, node.Address, grpcDialOptions...) | ||||
| 	if err != nil { | ||||
| 		return nil, errors.InternalServerError("go.micro.client", fmt.Sprintf("Error sending request: %v", err)) | ||||
| 		return errors.InternalServerError("go.micro.client", fmt.Sprintf("Error sending request: %v", err)) | ||||
| 	} | ||||
|  | ||||
| 	desc := &grpc.StreamDesc{ | ||||
| @@ -252,7 +223,7 @@ func (g *grpcClient) stream(ctx context.Context, node *registry.Node, req client | ||||
| 		// close the connection | ||||
| 		cc.Close() | ||||
| 		// now return the error | ||||
| 		return nil, errors.InternalServerError("go.micro.client", fmt.Sprintf("Error creating stream: %v", err)) | ||||
| 		return errors.InternalServerError("go.micro.client", fmt.Sprintf("Error creating stream: %v", err)) | ||||
| 	} | ||||
|  | ||||
| 	codec := &grpcCodec{ | ||||
| @@ -265,21 +236,25 @@ func (g *grpcClient) stream(ctx context.Context, node *registry.Node, req client | ||||
| 		r.codec = codec | ||||
| 	} | ||||
|  | ||||
| 	rsp := &response{ | ||||
| 		conn:   cc, | ||||
| 	// setup the stream response | ||||
| 	stream := &grpcStream{ | ||||
| 		context: ctx, | ||||
| 		request: req, | ||||
| 		response: &response{ | ||||
| 			conn:   cc, | ||||
| 			stream: st, | ||||
| 			codec:  cf, | ||||
| 			gcodec: codec, | ||||
| 		}, | ||||
| 		stream: st, | ||||
| 		codec:  cf, | ||||
| 		gcodec: codec, | ||||
| 		conn:   cc, | ||||
| 		cancel: cancel, | ||||
| 	} | ||||
|  | ||||
| 	return &grpcStream{ | ||||
| 		context:  ctx, | ||||
| 		request:  req, | ||||
| 		response: rsp, | ||||
| 		stream:   st, | ||||
| 		conn:     cc, | ||||
| 		cancel:   cancel, | ||||
| 	}, nil | ||||
| 	// set the stream as the response | ||||
| 	val := reflect.ValueOf(rsp).Elem() | ||||
| 	val.Set(reflect.ValueOf(stream).Elem()) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (g *grpcClient) poolMaxStreams() int { | ||||
| @@ -385,11 +360,6 @@ func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface | ||||
| 		opt(&callOpts) | ||||
| 	} | ||||
|  | ||||
| 	next, err := g.next(req, callOpts) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// check if we already have a deadline | ||||
| 	d, ok := ctx.Deadline() | ||||
| 	if !ok { | ||||
| @@ -432,19 +402,33 @@ func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface | ||||
| 			time.Sleep(t) | ||||
| 		} | ||||
|  | ||||
| 		// select next node | ||||
| 		node, err := next() | ||||
| 		service := req.Service() | ||||
| 		if err != nil { | ||||
| 			if err == selector.ErrNotFound { | ||||
| 				return errors.InternalServerError("go.micro.client", "service %s: %s", service, err.Error()) | ||||
| 			} | ||||
| 			return errors.InternalServerError("go.micro.client", "error selecting %s node: %s", service, err.Error()) | ||||
| 		// use the router passed as a call option, or fallback to the rpc clients router | ||||
| 		if callOpts.Router == nil { | ||||
| 			callOpts.Router = g.opts.Router | ||||
| 		} | ||||
| 		// use the selector passed as a call option, or fallback to the rpc clients selector | ||||
| 		if callOpts.Selector == nil { | ||||
| 			callOpts.Selector = g.opts.Selector | ||||
| 		} | ||||
|  | ||||
| 		// lookup the route to send the reques to | ||||
| 		route, err := client.LookupRoute(req, callOpts) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		// pass a node to enable backwards compatability as changing the | ||||
| 		// call func would be a breaking change. | ||||
| 		// todo v3: change the call func to accept a route | ||||
| 		node := ®istry.Node{Address: route.Address} | ||||
|  | ||||
| 		// make the call | ||||
| 		err = gcall(ctx, node, req, rsp, callOpts) | ||||
| 		g.opts.Selector.Mark(service, node, err) | ||||
|  | ||||
| 		// record the result of the call to inform future routing decisions | ||||
| 		g.opts.Selector.Record(*route, err) | ||||
|  | ||||
| 		// try and transform the error to a go-micro error | ||||
| 		if verr, ok := err.(*errors.Error); ok { | ||||
| 			return verr | ||||
| 		} | ||||
| @@ -492,11 +476,6 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli | ||||
| 		opt(&callOpts) | ||||
| 	} | ||||
|  | ||||
| 	next, err := g.next(req, callOpts) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// #200 - streams shouldn't have a request timeout set on the context | ||||
|  | ||||
| 	// should we noop right here? | ||||
| @@ -506,6 +485,14 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli | ||||
| 	default: | ||||
| 	} | ||||
|  | ||||
| 	// make a copy of stream | ||||
| 	gstream := g.stream | ||||
|  | ||||
| 	// wrap the call in reverse | ||||
| 	for i := len(callOpts.CallWrappers); i > 0; i-- { | ||||
| 		gstream = callOpts.CallWrappers[i-1](gstream) | ||||
| 	} | ||||
|  | ||||
| 	call := func(i int) (client.Stream, error) { | ||||
| 		// call backoff first. Someone may want an initial start delay | ||||
| 		t, err := callOpts.Backoff(ctx, req, i) | ||||
| @@ -518,17 +505,39 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli | ||||
| 			time.Sleep(t) | ||||
| 		} | ||||
|  | ||||
| 		node, err := next() | ||||
| 		service := req.Service() | ||||
| 		if err != nil { | ||||
| 			if err == selector.ErrNotFound { | ||||
| 				return nil, errors.InternalServerError("go.micro.client", "service %s: %s", service, err.Error()) | ||||
| 			} | ||||
| 			return nil, errors.InternalServerError("go.micro.client", "error selecting %s node: %s", service, err.Error()) | ||||
| 		// use the router passed as a call option, or fallback to the rpc clients router | ||||
| 		if callOpts.Router == nil { | ||||
| 			callOpts.Router = g.opts.Router | ||||
| 		} | ||||
| 		// use the selector passed as a call option, or fallback to the rpc clients selector | ||||
| 		if callOpts.Selector == nil { | ||||
| 			callOpts.Selector = g.opts.Selector | ||||
| 		} | ||||
|  | ||||
| 		stream, err := g.stream(ctx, node, req, callOpts) | ||||
| 		g.opts.Selector.Mark(service, node, err) | ||||
| 		// lookup the route to send the reques to | ||||
| 		route, err := client.LookupRoute(req, callOpts) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		// pass a node to enable backwards compatability as changing the | ||||
| 		// call func would be a breaking change. | ||||
| 		// todo v3: change the call func to accept a route | ||||
| 		node := ®istry.Node{Address: route.Address} | ||||
|  | ||||
| 		// make the call | ||||
| 		stream := &grpcStream{} | ||||
| 		err = g.stream(ctx, node, req, stream, callOpts) | ||||
|  | ||||
| 		// record the result of the call to inform future routing decisions | ||||
| 		g.opts.Selector.Record(*route, err) | ||||
|  | ||||
| 		// try and transform the error to a go-micro error | ||||
| 		if verr, ok := err.(*errors.Error); ok { | ||||
| 			return nil, verr | ||||
| 		} | ||||
|  | ||||
| 		g.opts.Selector.Record(*route, err) | ||||
| 		return stream, err | ||||
| 	} | ||||
|  | ||||
| @@ -555,7 +564,7 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli | ||||
| 				return rsp.stream, nil | ||||
| 			} | ||||
|  | ||||
| 			retry, rerr := callOpts.Retry(ctx, req, i, err) | ||||
| 			retry, rerr := callOpts.Retry(ctx, req, i, grr) | ||||
| 			if rerr != nil { | ||||
| 				return nil, rerr | ||||
| 			} | ||||
| @@ -573,6 +582,16 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli | ||||
|  | ||||
| func (g *grpcClient) Publish(ctx context.Context, p client.Message, opts ...client.PublishOption) error { | ||||
| 	var options client.PublishOptions | ||||
| 	var body []byte | ||||
|  | ||||
| 	// fail early on connect error | ||||
| 	if !g.once.Load().(bool) { | ||||
| 		if err := g.opts.Broker.Connect(); err != nil { | ||||
| 			return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 		} | ||||
| 		g.once.Store(true) | ||||
| 	} | ||||
|  | ||||
| 	for _, o := range opts { | ||||
| 		o(&options) | ||||
| 	} | ||||
| @@ -584,17 +603,15 @@ func (g *grpcClient) Publish(ctx context.Context, p client.Message, opts ...clie | ||||
| 	md["Content-Type"] = p.ContentType() | ||||
| 	md["Micro-Topic"] = p.Topic() | ||||
|  | ||||
| 	cf, err := g.newGRPCCodec(p.ContentType()) | ||||
| 	if err != nil { | ||||
| 		return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 	} | ||||
|  | ||||
| 	var body []byte | ||||
|  | ||||
| 	// passed in raw data | ||||
| 	if d, ok := p.Payload().(*raw.Frame); ok { | ||||
| 		body = d.Data | ||||
| 	} else { | ||||
| 		// use codec for payload | ||||
| 		cf, err := g.newGRPCCodec(p.ContentType()) | ||||
| 		if err != nil { | ||||
| 			return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 		} | ||||
| 		// set the body | ||||
| 		b, err := cf.Marshal(p.Payload()) | ||||
| 		if err != nil { | ||||
| @@ -603,13 +620,6 @@ func (g *grpcClient) Publish(ctx context.Context, p client.Message, opts ...clie | ||||
| 		body = b | ||||
| 	} | ||||
|  | ||||
| 	if !g.once.Load().(bool) { | ||||
| 		if err = g.opts.Broker.Connect(); err != nil { | ||||
| 			return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 		} | ||||
| 		g.once.Store(true) | ||||
| 	} | ||||
|  | ||||
| 	topic := p.Topic() | ||||
|  | ||||
| 	// get the exchange | ||||
|   | ||||
| @@ -5,11 +5,12 @@ import ( | ||||
| 	"net" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/client/selector" | ||||
| 	"github.com/micro/go-micro/v2/errors" | ||||
| 	"github.com/micro/go-micro/v2/registry" | ||||
| 	"github.com/micro/go-micro/v2/registry/memory" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"github.com/micro/go-micro/v3/errors" | ||||
| 	"github.com/micro/go-micro/v3/registry" | ||||
| 	"github.com/micro/go-micro/v3/registry/memory" | ||||
| 	"github.com/micro/go-micro/v3/router" | ||||
| 	regRouter "github.com/micro/go-micro/v3/router/registry" | ||||
| 	pgrpc "google.golang.org/grpc" | ||||
| 	pb "google.golang.org/grpc/examples/helloworld/helloworld" | ||||
| ) | ||||
| @@ -56,16 +57,11 @@ func TestGRPCClient(t *testing.T) { | ||||
| 		}, | ||||
| 	}) | ||||
|  | ||||
| 	// create selector | ||||
| 	se := selector.NewSelector( | ||||
| 		selector.Registry(r), | ||||
| 	) | ||||
| 	// create router | ||||
| 	rtr := regRouter.NewRouter(router.Registry(r)) | ||||
|  | ||||
| 	// create client | ||||
| 	c := NewClient( | ||||
| 		client.Registry(r), | ||||
| 		client.Selector(se), | ||||
| 	) | ||||
| 	c := NewClient(client.Router(rtr)) | ||||
|  | ||||
| 	testMethods := []string{ | ||||
| 		"/helloworld.Greeter/SayHello", | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| package grpc | ||||
|  | ||||
| import ( | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| ) | ||||
|  | ||||
| type grpcEvent struct { | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"context" | ||||
| 	"crypto/tls" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/encoding" | ||||
| ) | ||||
|   | ||||
| @@ -4,8 +4,8 @@ import ( | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v2/codec" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"github.com/micro/go-micro/v3/codec" | ||||
| ) | ||||
|  | ||||
| type grpcRequest struct { | ||||
|   | ||||
| @@ -3,8 +3,8 @@ package grpc | ||||
| import ( | ||||
| 	"strings" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/codec" | ||||
| 	"github.com/micro/go-micro/v2/codec/bytes" | ||||
| 	"github.com/micro/go-micro/v3/codec" | ||||
| 	"github.com/micro/go-micro/v3/codec/bytes" | ||||
| 	"google.golang.org/grpc" | ||||
| 	"google.golang.org/grpc/encoding" | ||||
| ) | ||||
|   | ||||
| @@ -5,7 +5,7 @@ import ( | ||||
| 	"io" | ||||
| 	"sync" | ||||
|  | ||||
| 	"github.com/micro/go-micro/v2/client" | ||||
| 	"github.com/micro/go-micro/v3/client" | ||||
| 	"google.golang.org/grpc" | ||||
| ) | ||||
|  | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user