Compare commits
	
		
			242 Commits
		
	
	
		
			v3.2.3
			...
			36c52990ee
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 36c52990ee | |||
| 7ad92816ef | |||
| c47d04db7f | |||
| 80a95aae8f | |||
| 5a9b5e23fc | |||
| 11e3ab33b5 | |||
| 1f8a64c7d6 | |||
| 7969cfd8c9 | |||
| cee07d11d4 | |||
| d5360e2804 | |||
| 2d23f347eb | |||
| 78664a34ed | |||
| bec0e310e9 | |||
| 92436e9016 | |||
| 9526345cd6 | |||
| 25618a3859 | |||
| b22f150601 | |||
| 71bcb63b60 | |||
| 698bfbc6f1 | |||
| 209042de3a | |||
| 990685628d | |||
| e3fe27105b | |||
| 0f8ead6acc | |||
| 2fbcee7b59 | |||
| 19a469c4e2 | |||
| 2a6a93a792 | |||
| 2adba9d0da | |||
| 9d70c4dd34 | |||
| b9704903f2 | |||
| 4eda58e404 | |||
| 0f8d0a1123 | |||
| 603b855e2a | |||
| f2cfd562c3 | |||
| 781cf3d719 | |||
| 1b65507fe5 | |||
| 40b5402aa5 | |||
| 
						 | 
					e8b7e30e4d | ||
| 55e7e3d61d | |||
| 
						 | 
					d75c36938a | ||
| a7cec360e9 | |||
| 
						 | 
					3978256931 | ||
| 6dfff28203 | |||
| 
						 | 
					373acda151 | ||
| 3f9ed4e83b | |||
| 
						 | 
					0486780f36 | ||
| 36788b5fbf | |||
| 
						 | 
					c79b83a3c1 | ||
| a6585fd6d4 | |||
| 
						 | 
					7fce5ccad9 | ||
| b5a5e98f9e | |||
| 
						 | 
					e419c6a67c | ||
| 17c46e63a2 | |||
| 
						 | 
					cdf1a2c3e3 | ||
| aba3df63df | |||
| 
						 | 
					337df46029 | ||
| bfe4839a00 | |||
| 
						 | 
					5474c37f8f | ||
| ba88a5568a | |||
| 
						 | 
					57f903a8c2 | ||
| 9f918bd3f2 | |||
| 
						 | 
					0d88050daf | ||
| 
						 | 
					d1ed5bed51 | ||
| 
						 | 
					42697318b1 | ||
| 28aea45725 | |||
| 
						 | 
					ed72c05645 | ||
| 
						 | 
					d239dcde63 | ||
| 6eb07dc351 | |||
| 
						 | 
					a0d704f845 | ||
| 4b8a761f30 | |||
| 
						 | 
					a2c711a1b5 | ||
| be564f50aa | |||
| 
						 | 
					b7b1eff81c | ||
| 671a6b6f7c | |||
| 27c1c4d86b | |||
| 76d37a99eb | |||
| 
						 | 
					4c2827172f | ||
| e3461dd23f | |||
| 
						 | 
					d8b5c011e5 | ||
| 
						 | 
					15abd38afd | ||
| 5865a0f388 | |||
| 
						 | 
					67da8d1165 | ||
| 
						 | 
					c7d24caa03 | ||
| 8c222c4715 | |||
| 6e91cd5cf5 | |||
| a9e673b2ef | |||
| 251f06cc31 | |||
| e97e4580a1 | |||
| bd7dbe94ca | |||
| 
						 | 
					0f32fad4c0 | ||
| 
						 | 
					858111106e | ||
| 
						 | 
					dc35dc6d3e | ||
| 
						 | 
					24b1abfb9a | ||
| 
						 | 
					205fd53047 | ||
| 
						 | 
					b3b7d1af13 | ||
| 80e2184bba | |||
| 
						 | 
					a920d15d95 | ||
| 5ecbfac164 | |||
| 
						 | 
					61d7a322de | ||
| 
						 | 
					b43c207f6a | ||
| 
						 | 
					d243b884c7 | ||
| 107b470b9a | |||
| 
						 | 
					878bac53ac | ||
| e28c584056 | |||
| b8dc821784 | |||
| 
						 | 
					907d2591df | ||
| 
						 | 
					8cc656eec7 | ||
| 
						 | 
					5792434604 | ||
| 402ccee5b9 | |||
| 
						 | 
					90986a26e2 | ||
| 6d41afd5a0 | |||
| 2e645748bb | |||
| f6b1c8d745 | |||
| ea503d0583 | |||
| 048773c669 | |||
| 
						 | 
					e3cb87ffe3 | ||
| b1af43c4b0 | |||
| 
						 | 
					f994df9e04 | ||
| c556f7157f | |||
| 
						 | 
					17be582d19 | ||
| 131a150d3d | |||
| 
						 | 
					7f874a286e | ||
| 2f40797303 | |||
| 
						 | 
					3e1b2b7c5d | ||
| 6442f4c474 | |||
| 
						 | 
					1dda7e6b83 | ||
| 2b85cabe1f | |||
| 02895dd712 | |||
| 42b93ce57e | |||
| 763c299ab7 | |||
| 
						 | 
					937a6d62b2 | ||
| fb6e2c8845 | |||
| 3e9a3a917d | |||
| 289a765784 | |||
| b13ad231d1 | |||
| 0793e84da6 | |||
| 
						 | 
					f2c6d7cc80 | ||
| 6969b228a7 | |||
| 84362e6dd9 | |||
| 
						 | 
					b83cc26ca0 | ||
| 
						 | 
					6591845ded | ||
| 7c7c93521f | |||
| 9e10237b97 | |||
| 
						 | 
					856c7dae7c | ||
| e85f2e1f45 | |||
| 
						 | 
					0a8ccce4e1 | ||
| 1b7e22442f | |||
| 
						 | 
					4c81ce6a9d | ||
| 89ffe47d06 | |||
| 
						 | 
					a03274011f | ||
| 186ec6bf00 | |||
| 
						 | 
					fefcc273d9 | ||
| a5ca5ec499 | |||
| 
						 | 
					7104528c7d | ||
| eaa61e254e | |||
| 
						 | 
					00aa5331f6 | ||
| c165c0f1a8 | |||
| 
						 | 
					fa6fe590bd | ||
| 8cae060e05 | |||
| a10275ad1f | |||
| 
						 | 
					f247a8c906 | ||
| 
						 | 
					015b2a4b05 | ||
| 
						 | 
					c9da9d36a0 | ||
| 1f69062916 | |||
| c4ca900a56 | |||
| 
						 | 
					569d95e3e3 | ||
| d2f1b7b3b9 | |||
| 
						 | 
					65c10f7b0a | ||
| 76537a045a | |||
| 
						 | 
					7f19288476 | ||
| c3d10b669b | |||
| 20c5840f47 | |||
| 8b2dd91711 | |||
| 
						 | 
					08621f7cf3 | ||
| 
						 | 
					8d7d802730 | ||
| 137fbae58e | |||
| 
						 | 
					13093dd404 | ||
| 12cf576b9c | |||
| 
						 | 
					b91014d287 | ||
| 6b867760e5 | |||
| 
						 | 
					f61f056a8e | ||
| 0525dae1cd | |||
| 
						 | 
					50d0d836e5 | ||
| a1e7560d6f | |||
| a510f982e2 | |||
| 1129a1e992 | |||
| 
						 | 
					033fcd3e2b | ||
| c2ef582962 | |||
| 4c9f1f21a9 | |||
| 
						 | 
					4585b73513 | ||
| 
						 | 
					728d3f2a8c | ||
| c1a2cfae67 | |||
| 8edc3d35bb | |||
| 
						 | 
					4d0bb35dd9 | ||
| 0539d08195 | |||
| f40591a520 | |||
| acfab7e10c | |||
| 6697ccddf2 | |||
| d536140a5b | |||
| db02559a00 | |||
| 0fae2e1bdd | |||
| 
						 | 
					7a1fdef33a | ||
| 2bb81ff232 | |||
| 
						 | 
					9ac268b2f0 | ||
| 510fa4b379 | |||
| 
						 | 
					08def4d244 | ||
| a4683c0b78 | |||
| 
						 | 
					637e3df24a | ||
| 
						 | 
					36c7fa6a23 | ||
| 2bef21a001 | |||
| 
						 | 
					78c32dc3e8 | ||
| 
						 | 
					4d5a2d1a4a | ||
| 
						 | 
					9817e0c1af | ||
| 
						 | 
					ca4d561efe | ||
| 
						 | 
					0ffbfc36d7 | ||
| 
						 | 
					8a1b5b1130 | ||
| 
						 | 
					1048869e6c | ||
| 
						 | 
					35e9b9b77e | ||
| 
						 | 
					faeaff358d | ||
| 
						 | 
					3d7a5f6122 | ||
| 9030637204 | |||
| 
						 | 
					225cdfd469 | ||
| 
						 | 
					d3696a4bf8 | ||
| 
						 | 
					88432401ac | ||
| 
						 | 
					e3383bae9b | ||
| 
						 | 
					60c340cc86 | ||
| 
						 | 
					8d237e1423 | ||
| 
						 | 
					be27263047 | ||
| 
						 | 
					41a60a8a16 | ||
| 2e09afca64 | |||
| 
						 | 
					aeb25fec3b | ||
| a4518a12eb | |||
| 2a6e2c72c1 | |||
| 
						 | 
					1d17c6c962 | ||
| 
						 | 
					11c265458c | ||
| 8d34fe5dc0 | |||
| 8c480b5f0e | |||
| c1e03ca61d | |||
| 
						 | 
					22dc300099 | ||
| 
						 | 
					8584261b44 | ||
| 4d31515547 | |||
| 9f36491023 | |||
| 
						 | 
					1378d624be | 
							
								
								
									
										29
									
								
								.gitea/workflows/job_lint.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								.gitea/workflows/job_lint.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,29 @@
 | 
			
		||||
name: lint
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    types: [opened, reopened, synchronize]
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
    - v3
 | 
			
		||||
    - v4
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  lint:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: checkout code
 | 
			
		||||
      uses: actions/checkout@v4
 | 
			
		||||
      with:
 | 
			
		||||
        filter: 'blob:none'
 | 
			
		||||
    - name: setup go
 | 
			
		||||
      uses: actions/setup-go@v5
 | 
			
		||||
      with:
 | 
			
		||||
        cache-dependency-path: "**/*.sum"
 | 
			
		||||
        go-version: 'stable' 
 | 
			
		||||
    - name: setup deps
 | 
			
		||||
      run: go get -v ./...
 | 
			
		||||
    - name: run lint
 | 
			
		||||
      uses: https://github.com/golangci/golangci-lint-action@v6
 | 
			
		||||
      with:
 | 
			
		||||
        version: 'latest'
 | 
			
		||||
							
								
								
									
										34
									
								
								.gitea/workflows/job_test.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.gitea/workflows/job_test.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,34 @@
 | 
			
		||||
name: test
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    types: [opened, reopened, synchronize]
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
    - v3
 | 
			
		||||
    - v4
 | 
			
		||||
  push:
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
    - v3
 | 
			
		||||
    - v4
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: checkout code
 | 
			
		||||
      uses: actions/checkout@v4
 | 
			
		||||
      with:
 | 
			
		||||
        filter: 'blob:none'
 | 
			
		||||
    - name: setup go
 | 
			
		||||
      uses: actions/setup-go@v5
 | 
			
		||||
      with:
 | 
			
		||||
        cache-dependency-path: "**/*.sum"
 | 
			
		||||
        go-version: 'stable'
 | 
			
		||||
    - name: setup deps
 | 
			
		||||
      run: go get -v ./...
 | 
			
		||||
    - name: run test
 | 
			
		||||
      env:
 | 
			
		||||
        INTEGRATION_TESTS: yes
 | 
			
		||||
      run: go test -mod readonly -v ./...
 | 
			
		||||
							
								
								
									
										53
									
								
								.gitea/workflows/job_tests.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								.gitea/workflows/job_tests.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,53 @@
 | 
			
		||||
name: test
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    types: [opened, reopened, synchronize]
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
    - v3
 | 
			
		||||
    - v4
 | 
			
		||||
  push:
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
    - v3
 | 
			
		||||
    - v4
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: checkout code
 | 
			
		||||
      uses: actions/checkout@v4
 | 
			
		||||
      with:
 | 
			
		||||
        filter: 'blob:none'
 | 
			
		||||
    - name: checkout tests
 | 
			
		||||
      uses: actions/checkout@v4
 | 
			
		||||
      with:
 | 
			
		||||
        ref: master
 | 
			
		||||
        filter: 'blob:none'
 | 
			
		||||
        repository: unistack-org/micro-tests
 | 
			
		||||
        path: micro-tests
 | 
			
		||||
    - name: setup go
 | 
			
		||||
      uses: actions/setup-go@v5
 | 
			
		||||
      with:
 | 
			
		||||
        cache-dependency-path: "**/*.sum"
 | 
			
		||||
        go-version: 'stable'
 | 
			
		||||
    - name: setup go work
 | 
			
		||||
      env:
 | 
			
		||||
        GOWORK: /workspace/${{ github.repository_owner }}/go.work
 | 
			
		||||
      run: |
 | 
			
		||||
        go work init
 | 
			
		||||
        go work use .
 | 
			
		||||
        go work use micro-tests        
 | 
			
		||||
    - name: setup deps
 | 
			
		||||
      env:
 | 
			
		||||
        GOWORK: /workspace/${{ github.repository_owner }}/go.work
 | 
			
		||||
      run: go get -v ./...
 | 
			
		||||
    - name: run tests
 | 
			
		||||
      env:
 | 
			
		||||
        INTEGRATION_TESTS: yes
 | 
			
		||||
        GOWORK: /workspace/${{ github.repository_owner }}/go.work
 | 
			
		||||
      run: |
 | 
			
		||||
        cd micro-tests
 | 
			
		||||
        go test -mod readonly -v ./... || true        
 | 
			
		||||
							
								
								
									
										13
									
								
								.github/stale.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.github/stale.sh
									
									
									
									
										vendored
									
									
								
							@@ -1,13 +0,0 @@
 | 
			
		||||
#!/bin/bash -ex
 | 
			
		||||
 | 
			
		||||
export PATH=$PATH:$(pwd)/bin
 | 
			
		||||
export GO111MODULE=on
 | 
			
		||||
export GOBIN=$(pwd)/bin
 | 
			
		||||
 | 
			
		||||
#go get github.com/rvflash/goup@v0.4.1
 | 
			
		||||
 | 
			
		||||
#goup -v ./...
 | 
			
		||||
#go get github.com/psampaz/go-mod-outdated@v0.6.0
 | 
			
		||||
go list -u -m -mod=mod -json all | go-mod-outdated -update -direct -ci || true
 | 
			
		||||
 | 
			
		||||
#go list -u -m -json all | go-mod-outdated -update
 | 
			
		||||
							
								
								
									
										46
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										46
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							@@ -1,46 +0,0 @@
 | 
			
		||||
name: build
 | 
			
		||||
on:
 | 
			
		||||
 push:
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    name: test
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: setup
 | 
			
		||||
      uses: actions/setup-go@v1
 | 
			
		||||
      with:
 | 
			
		||||
        go-version: 1.15
 | 
			
		||||
    - name: checkout
 | 
			
		||||
      uses: actions/checkout@v2
 | 
			
		||||
    - name: cache
 | 
			
		||||
      uses: actions/cache@v2
 | 
			
		||||
      with:
 | 
			
		||||
        path: ~/go/pkg/mod
 | 
			
		||||
        key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
 | 
			
		||||
        restore-keys: ${{ runner.os }}-go-
 | 
			
		||||
    - name: deps
 | 
			
		||||
      run: go get -v -t -d ./...
 | 
			
		||||
    - name: test
 | 
			
		||||
      env:
 | 
			
		||||
        INTEGRATION_TESTS: yes
 | 
			
		||||
      run: go test -mod readonly -v ./...
 | 
			
		||||
  lint:
 | 
			
		||||
    name: lint
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: checkout
 | 
			
		||||
        uses: actions/checkout@v2
 | 
			
		||||
      - name: lint
 | 
			
		||||
        uses: golangci/golangci-lint-action@v1
 | 
			
		||||
        continue-on-error: true
 | 
			
		||||
        with:
 | 
			
		||||
          # Required: the version of golangci-lint is required and must be specified without patch version: we always use the latest patch version.
 | 
			
		||||
          version: v1.30
 | 
			
		||||
          # Optional: working directory, useful for monorepos
 | 
			
		||||
          # working-directory: somedir
 | 
			
		||||
          # Optional: golangci-lint command line arguments.
 | 
			
		||||
          # args: --issues-exit-code=0
 | 
			
		||||
          # Optional: show only new issues if it's a pull request. The default value is `false`.
 | 
			
		||||
          # only-new-issues: true
 | 
			
		||||
							
								
								
									
										46
									
								
								.github/workflows/pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										46
									
								
								.github/workflows/pr.yml
									
									
									
									
										vendored
									
									
								
							@@ -1,46 +0,0 @@
 | 
			
		||||
name: prbuild
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    branches:
 | 
			
		||||
    - master
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    name: test
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
    - name: setup
 | 
			
		||||
      uses: actions/setup-go@v1
 | 
			
		||||
      with:
 | 
			
		||||
        go-version: 1.15
 | 
			
		||||
    - name: checkout
 | 
			
		||||
      uses: actions/checkout@v2
 | 
			
		||||
    - name: cache
 | 
			
		||||
      uses: actions/cache@v2
 | 
			
		||||
      with:
 | 
			
		||||
        path: ~/go/pkg/mod
 | 
			
		||||
        key: ${{ runner.os }}-go-${{ hashFiles('**/go.sum') }}
 | 
			
		||||
        restore-keys: ${{ runner.os }}-go-
 | 
			
		||||
    - name: deps
 | 
			
		||||
      run: go get -v -t -d ./...
 | 
			
		||||
    - name: test
 | 
			
		||||
      env:
 | 
			
		||||
        INTEGRATION_TESTS: yes
 | 
			
		||||
      run: go test -mod readonly -v ./...
 | 
			
		||||
  lint:
 | 
			
		||||
    name: lint
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: checkout
 | 
			
		||||
        uses: actions/checkout@v2
 | 
			
		||||
      - name: lint
 | 
			
		||||
        uses: golangci/golangci-lint-action@v1
 | 
			
		||||
        continue-on-error: true
 | 
			
		||||
        with:
 | 
			
		||||
          # Required: the version of golangci-lint is required and must be specified without patch version: we always use the latest patch version.
 | 
			
		||||
          version: v1.30
 | 
			
		||||
          # Optional: working directory, useful for monorepos
 | 
			
		||||
          # working-directory: somedir
 | 
			
		||||
          # Optional: golangci-lint command line arguments.
 | 
			
		||||
          # args: --issues-exit-code=0
 | 
			
		||||
          # Optional: show only new issues if it's a pull request. The default value is `false`.
 | 
			
		||||
          # only-new-issues: true
 | 
			
		||||
							
								
								
									
										24
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,24 @@
 | 
			
		||||
# Binaries for programs and plugins
 | 
			
		||||
*.exe
 | 
			
		||||
*.exe~
 | 
			
		||||
*.dll
 | 
			
		||||
*.so
 | 
			
		||||
*.dylib
 | 
			
		||||
bin
 | 
			
		||||
 | 
			
		||||
# Test binary, built with `go test -c`
 | 
			
		||||
*.test
 | 
			
		||||
 | 
			
		||||
# Output of the go coverage tool, specifically when used with LiteIDE
 | 
			
		||||
*.out
 | 
			
		||||
 | 
			
		||||
# Dependency directories (remove the comment below to include it)
 | 
			
		||||
# vendor/
 | 
			
		||||
 | 
			
		||||
# Go workspace file
 | 
			
		||||
go.work
 | 
			
		||||
 | 
			
		||||
# General
 | 
			
		||||
.DS_Store
 | 
			
		||||
.idea
 | 
			
		||||
.vscode
 | 
			
		||||
							
								
								
									
										5
									
								
								.golangci.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.golangci.yml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,5 @@
 | 
			
		||||
run:
 | 
			
		||||
  concurrency: 8
 | 
			
		||||
  deadline: 5m
 | 
			
		||||
  issues-exit-code: 1
 | 
			
		||||
  tests: true
 | 
			
		||||
							
								
								
									
										90
									
								
								codec.go
									
									
									
									
									
								
							
							
						
						
									
										90
									
								
								codec.go
									
									
									
									
									
								
							@@ -1,13 +1,16 @@
 | 
			
		||||
package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"io"
 | 
			
		||||
 | 
			
		||||
	"github.com/unistack-org/micro/v3/codec"
 | 
			
		||||
	"go.unistack.org/micro/v4/codec"
 | 
			
		||||
	"google.golang.org/grpc"
 | 
			
		||||
	"google.golang.org/grpc/encoding"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	_ encoding.Codec = &wrapMicroCodec{}
 | 
			
		||||
	_ codec.Codec    = &wrapGrpcCodec{}
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type wrapStream struct{ grpc.ClientStream }
 | 
			
		||||
 | 
			
		||||
func (w *wrapStream) Write(d []byte) (int, error) {
 | 
			
		||||
@@ -19,7 +22,7 @@ func (w *wrapStream) Write(d []byte) (int, error) {
 | 
			
		||||
func (w *wrapStream) Read(d []byte) (int, error) {
 | 
			
		||||
	m := &codec.Frame{}
 | 
			
		||||
	err := w.ClientStream.RecvMsg(m)
 | 
			
		||||
	d = m.Data
 | 
			
		||||
	copy(d, m.Data)
 | 
			
		||||
	return len(d), err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -29,89 +32,34 @@ func (w *wrapMicroCodec) Name() string {
 | 
			
		||||
	return w.Codec.String()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (w *wrapMicroCodec) Marshal(v interface{}) ([]byte, error) {
 | 
			
		||||
	return w.Codec.Marshal(v)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (w *wrapMicroCodec) Unmarshal(d []byte, v interface{}) error {
 | 
			
		||||
	return w.Codec.Unmarshal(d, v)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type wrapGrpcCodec struct{ encoding.Codec }
 | 
			
		||||
 | 
			
		||||
func (w *wrapGrpcCodec) String() string {
 | 
			
		||||
	return w.Codec.Name()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (w *wrapGrpcCodec) Marshal(v interface{}) ([]byte, error) {
 | 
			
		||||
	switch m := v.(type) {
 | 
			
		||||
	case *codec.Frame:
 | 
			
		||||
func (w *wrapGrpcCodec) Marshal(v interface{}, opts ...codec.Option) ([]byte, error) {
 | 
			
		||||
	if m, ok := v.(*codec.Frame); ok {
 | 
			
		||||
		return m.Data, nil
 | 
			
		||||
	}
 | 
			
		||||
	return w.Codec.Marshal(v)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (w wrapGrpcCodec) Unmarshal(d []byte, v interface{}) error {
 | 
			
		||||
func (w *wrapGrpcCodec) Unmarshal(d []byte, v interface{}, opts ...codec.Option) error {
 | 
			
		||||
	if d == nil || v == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	switch m := v.(type) {
 | 
			
		||||
	case *codec.Frame:
 | 
			
		||||
	if m, ok := v.(*codec.Frame); ok {
 | 
			
		||||
		m.Data = d
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	return w.Codec.Unmarshal(d, v)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
type grpcCodec struct {
 | 
			
		||||
	grpc.ServerStream
 | 
			
		||||
	// headers
 | 
			
		||||
	id       string
 | 
			
		||||
	target   string
 | 
			
		||||
	method   string
 | 
			
		||||
	endpoint string
 | 
			
		||||
 | 
			
		||||
	c encoding.Codec
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
func (g *wrapGrpcCodec) ReadHeader(conn io.Reader, m *codec.Message, mt codec.MessageType) error {
 | 
			
		||||
	/*
 | 
			
		||||
		if m == nil {
 | 
			
		||||
			m = codec.NewMessage(codec.Request)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if md, ok := metadata.FromIncomingContext(g.ServerStream.Context()); ok {
 | 
			
		||||
			if m.Header == nil {
 | 
			
		||||
				m.Header = meta.New(len(md))
 | 
			
		||||
			}
 | 
			
		||||
			for k, v := range md {
 | 
			
		||||
				m.Header[k] = strings.Join(v, ",")
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		m.Id = g.id
 | 
			
		||||
		m.Target = g.target
 | 
			
		||||
		m.Method = g.method
 | 
			
		||||
		m.Endpoint = g.endpoint
 | 
			
		||||
	*/
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *wrapGrpcCodec) ReadBody(conn io.Reader, v interface{}) error {
 | 
			
		||||
	// caller has requested a frame
 | 
			
		||||
	switch m := v.(type) {
 | 
			
		||||
	case *codec.Frame:
 | 
			
		||||
		_, err := conn.Read(m.Data)
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	return codec.ErrInvalidMessage
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *wrapGrpcCodec) Write(conn io.Writer, m *codec.Message, v interface{}) error {
 | 
			
		||||
	// if we don't have a body
 | 
			
		||||
	if v != nil {
 | 
			
		||||
		b, err := g.Marshal(v)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		m.Body = b
 | 
			
		||||
	}
 | 
			
		||||
	// write the body using the framing codec
 | 
			
		||||
	_, err := conn.Write(m.Body)
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										67
									
								
								codec_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								codec_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,67 @@
 | 
			
		||||
package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"testing"
 | 
			
		||||
 | 
			
		||||
	"go.unistack.org/micro/v4/codec"
 | 
			
		||||
	gmetadata "google.golang.org/grpc/metadata"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type mockStream struct {
 | 
			
		||||
	msg any
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m mockStream) Header() (gmetadata.MD, error) {
 | 
			
		||||
	return nil, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m mockStream) Trailer() gmetadata.MD {
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m mockStream) CloseSend() error {
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m mockStream) Context() context.Context {
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m *mockStream) SendMsg(msg any) error {
 | 
			
		||||
	m.msg = msg
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m *mockStream) RecvMsg(msg any) error {
 | 
			
		||||
 | 
			
		||||
	c := msg.(*codec.Frame)
 | 
			
		||||
	c.Data = m.msg.(*codec.Frame).Data
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func Test_ReadWrap(t *testing.T) {
 | 
			
		||||
 | 
			
		||||
	wp := wrapStream{
 | 
			
		||||
		&mockStream{},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	write, err := wp.Write([]byte("test_data"))
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatal(err)
 | 
			
		||||
	}
 | 
			
		||||
	if write != 9 {
 | 
			
		||||
		t.Error("uncorrected number wrote bytes")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	b := make([]byte, write)
 | 
			
		||||
	read, err := wp.Read(b)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatal(err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if read != 9 || string(b) != "test_data" {
 | 
			
		||||
		t.Error("uncorrected number wrote bytes or data")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										35
									
								
								error.go
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								error.go
									
									
									
									
									
								
							@@ -1,7 +1,7 @@
 | 
			
		||||
package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"github.com/unistack-org/micro/v3/errors"
 | 
			
		||||
	"go.unistack.org/micro/v4/errors"
 | 
			
		||||
	"google.golang.org/grpc/status"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
@@ -9,31 +9,46 @@ func microError(err error) error {
 | 
			
		||||
	// no error
 | 
			
		||||
 | 
			
		||||
	if err == nil {
 | 
			
		||||
		// nothing to do
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if verr, ok := err.(*errors.Error); ok {
 | 
			
		||||
		// micro error
 | 
			
		||||
		return verr
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// grpc error
 | 
			
		||||
	s, ok := status.FromError(err)
 | 
			
		||||
	if !ok {
 | 
			
		||||
		// can't get status detals from grpc error, return base error
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// return first error from details
 | 
			
		||||
	if details := s.Details(); len(details) > 0 {
 | 
			
		||||
		if verr, ok := details[0].(error); ok {
 | 
			
		||||
			return microError(verr)
 | 
			
		||||
	details := s.Details()
 | 
			
		||||
	switch len(details) {
 | 
			
		||||
	case 0:
 | 
			
		||||
		if verr := errors.Parse(s.Message()); verr.Code > 0 {
 | 
			
		||||
			// return micro error
 | 
			
		||||
			return verr
 | 
			
		||||
		}
 | 
			
		||||
		// return base error as it not micro error
 | 
			
		||||
		return err
 | 
			
		||||
	case 1:
 | 
			
		||||
		if verr, ok := details[0].(*errors.Error); ok {
 | 
			
		||||
			// return nested micro error
 | 
			
		||||
			return verr
 | 
			
		||||
		}
 | 
			
		||||
		// return base error as it not holds micro error
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// try to decode micro *errors.Error
 | 
			
		||||
	if e := errors.Parse(s.Message()); e.Code > 0 {
 | 
			
		||||
		return e // actually a micro error
 | 
			
		||||
	// attached messages in details more then 1, try to fallback to micro error
 | 
			
		||||
	if verr := errors.Parse(s.Message()); verr.Code > 0 {
 | 
			
		||||
		// return micro error
 | 
			
		||||
		return verr
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// fallback
 | 
			
		||||
	return errors.InternalServerError("go.micro.client", s.Message())
 | 
			
		||||
	// not micro error return base error
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										27
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								go.mod
									
									
									
									
									
								
							@@ -1,13 +1,24 @@
 | 
			
		||||
module github.com/unistack-org/micro-client-grpc/v3
 | 
			
		||||
module go.unistack.org/micro-client-grpc/v4
 | 
			
		||||
 | 
			
		||||
go 1.15
 | 
			
		||||
go 1.23.0
 | 
			
		||||
 | 
			
		||||
toolchain go1.23.3
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	github.com/google/go-cmp v0.5.1 // indirect
 | 
			
		||||
	github.com/unistack-org/micro/v3 v3.2.3
 | 
			
		||||
	golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 // indirect
 | 
			
		||||
	google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d // indirect
 | 
			
		||||
	google.golang.org/grpc v1.31.1
 | 
			
		||||
	go.unistack.org/micro/v4 v4.1.2
 | 
			
		||||
	google.golang.org/grpc v1.70.0
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
//replace github.com/unistack-org/micro/v3 => ../../micro
 | 
			
		||||
require (
 | 
			
		||||
	github.com/ash3in/uuidv8 v1.2.0 // indirect
 | 
			
		||||
	github.com/google/uuid v1.6.0 // indirect
 | 
			
		||||
	github.com/matoous/go-nanoid v1.5.1 // indirect
 | 
			
		||||
	github.com/spf13/cast v1.7.1 // indirect
 | 
			
		||||
	go.unistack.org/micro-proto/v4 v4.1.0 // indirect
 | 
			
		||||
	golang.org/x/net v0.35.0 // indirect
 | 
			
		||||
	golang.org/x/sys v0.30.0 // indirect
 | 
			
		||||
	golang.org/x/text v0.22.0 // indirect
 | 
			
		||||
	google.golang.org/genproto/googleapis/rpc v0.0.0-20250224174004-546df14abb99 // indirect
 | 
			
		||||
	google.golang.org/protobuf v1.36.5 // indirect
 | 
			
		||||
	gopkg.in/yaml.v3 v3.0.1 // indirect
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										188
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										188
									
								
								go.sum
									
									
									
									
									
								
							@@ -1,137 +1,57 @@
 | 
			
		||||
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
 | 
			
		||||
github.com/BurntSushi/toml v0.3.1 h1:WXkYYl6Yr3qBf1K79EBnL4mak0OimBfB0XUf9Vl28OQ=
 | 
			
		||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
 | 
			
		||||
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
 | 
			
		||||
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
 | 
			
		||||
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
 | 
			
		||||
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
 | 
			
		||||
github.com/ef-ds/deque v1.0.4/go.mod h1:gXDnTC3yqvBcHbq2lcExjtAcVrOnJCbMcZXmuj8Z4tg=
 | 
			
		||||
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
 | 
			
		||||
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
 | 
			
		||||
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
 | 
			
		||||
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
 | 
			
		||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
 | 
			
		||||
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
 | 
			
		||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
 | 
			
		||||
github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
 | 
			
		||||
github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
 | 
			
		||||
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
 | 
			
		||||
github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
 | 
			
		||||
github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
 | 
			
		||||
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
 | 
			
		||||
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
 | 
			
		||||
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
 | 
			
		||||
github.com/golang/protobuf v1.4.3 h1:JjCZWpVbqXDqFVmTfYWEVTMIYrL/NPdPSCHPJ0T/raM=
 | 
			
		||||
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
 | 
			
		||||
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
 | 
			
		||||
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
 | 
			
		||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
 | 
			
		||||
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
 | 
			
		||||
github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
 | 
			
		||||
github.com/google/go-cmp v0.5.1 h1:JFrFEBb2xKufg6XkJsJr+WbKb4FQlURi5RUcBveYu9k=
 | 
			
		||||
github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
 | 
			
		||||
github.com/google/uuid v1.1.5 h1:kxhtnfFVi+rYdOALN0B3k9UT86zVJKfBimRaciULW4I=
 | 
			
		||||
github.com/google/uuid v1.1.5/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
 | 
			
		||||
github.com/imdario/mergo v0.3.11 h1:3tnifQM4i+fbajXKBHXWEH+KvNHqojZ778UH75j3bGA=
 | 
			
		||||
github.com/imdario/mergo v0.3.11/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA=
 | 
			
		||||
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
 | 
			
		||||
github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
 | 
			
		||||
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
 | 
			
		||||
github.com/DATA-DOG/go-sqlmock v1.5.2 h1:OcvFkGmslmlZibjAjaHm3L//6LiuBgolP7OputlJIzU=
 | 
			
		||||
github.com/DATA-DOG/go-sqlmock v1.5.2/go.mod h1:88MAG/4G7SMwSE3CeA0ZKzrT5CiOU3OJ+JlNzwDqpNU=
 | 
			
		||||
github.com/ash3in/uuidv8 v1.2.0 h1:2oogGdtCPwaVtyvPPGin4TfZLtOGE5F+W++E880G6SI=
 | 
			
		||||
github.com/ash3in/uuidv8 v1.2.0/go.mod h1:BnU0wJBxnzdEKmVg4xckBkD+VZuecTFTUP3M0dWgyY4=
 | 
			
		||||
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
 | 
			
		||||
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
 | 
			
		||||
github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY=
 | 
			
		||||
github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
 | 
			
		||||
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
 | 
			
		||||
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
 | 
			
		||||
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
 | 
			
		||||
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
 | 
			
		||||
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
 | 
			
		||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
 | 
			
		||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
 | 
			
		||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
 | 
			
		||||
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
 | 
			
		||||
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
 | 
			
		||||
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
 | 
			
		||||
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
 | 
			
		||||
github.com/miekg/dns v1.1.35/go.mod h1:KNUDUusw/aVsxyTYZM1oqvCicbwhgbNgztCETuNZ7xM=
 | 
			
		||||
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs=
 | 
			
		||||
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno=
 | 
			
		||||
github.com/oxtoacart/bpool v0.0.0-20190530202638-03653db5a59c h1:rp5dCmg/yLR3mgFuSOe4oEnDDmGLROTvMragMUXpTQw=
 | 
			
		||||
github.com/oxtoacart/bpool v0.0.0-20190530202638-03653db5a59c/go.mod h1:X07ZCGwUbLaax7L0S3Tw4hpejzu63ZrrQiUe6W0hcy0=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
			
		||||
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
 | 
			
		||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
 | 
			
		||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
 | 
			
		||||
github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY=
 | 
			
		||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
 | 
			
		||||
github.com/unistack-org/micro/v3 v3.2.3 h1:gzEQwXBp3aoGDKgNJOiMp5sSadBOK81IhR9FCet3PcA=
 | 
			
		||||
github.com/unistack-org/micro/v3 v3.2.3/go.mod h1:J8XxJj4Pqa3Ee0a4biRRtut7UwTlfBq8QRe+s4PKGS0=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
 | 
			
		||||
golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I=
 | 
			
		||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
 | 
			
		||||
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
 | 
			
		||||
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
 | 
			
		||||
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
 | 
			
		||||
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
 | 
			
		||||
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
 | 
			
		||||
golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
 | 
			
		||||
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
 | 
			
		||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
 | 
			
		||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
 | 
			
		||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
 | 
			
		||||
golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
 | 
			
		||||
golang.org/x/net v0.0.0-20201224014010-6772e930b67b h1:iFwSg7t5GZmB/Q5TjiEAsdoLDrdJRC1RiF2WhuV29Qw=
 | 
			
		||||
golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
 | 
			
		||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
 | 
			
		||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
 | 
			
		||||
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
 | 
			
		||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
 | 
			
		||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68 h1:nxC68pudNYkKU6jWhgrqdreuFiOQWj1Fs7T3VrH4Pjw=
 | 
			
		||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
 | 
			
		||||
golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw=
 | 
			
		||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
 | 
			
		||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
 | 
			
		||||
golang.org/x/text v0.3.3 h1:cokOdA+Jmi5PJGXLlLllQSgYigAEfHXJAERHVMaCc2k=
 | 
			
		||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
 | 
			
		||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
 | 
			
		||||
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
 | 
			
		||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
 | 
			
		||||
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
 | 
			
		||||
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
 | 
			
		||||
golang.org/x/tools v0.0.0-20191216052735-49a3e744a425/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
 | 
			
		||||
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013 h1:+kGHl1aib/qcwaRi1CbqBZ1rk19r85MNUf8HaBghugY=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d h1:92D1fum1bJLKSdr11OJ+54YeCMCGYIygTA7R/YZxH5M=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
 | 
			
		||||
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
 | 
			
		||||
google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
 | 
			
		||||
google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY=
 | 
			
		||||
google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
 | 
			
		||||
google.golang.org/grpc v1.31.1 h1:SfXqXS5hkufcdZ/mHtYCh53P2b+92WQq/DZcKLgsFRs=
 | 
			
		||||
google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak=
 | 
			
		||||
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
 | 
			
		||||
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
 | 
			
		||||
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
 | 
			
		||||
google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
 | 
			
		||||
google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
 | 
			
		||||
google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
 | 
			
		||||
google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
 | 
			
		||||
google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
 | 
			
		||||
google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4=
 | 
			
		||||
google.golang.org/protobuf v1.25.0 h1:Ejskq+SyPohKW+1uil0JJMtmHCgJPJ/qWTxr8qp+R4c=
 | 
			
		||||
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
 | 
			
		||||
github.com/matoous/go-nanoid v1.5.1 h1:aCjdvTyO9LLnTIi0fgdXhOPPvOHjpXN6Ik9DaNjIct4=
 | 
			
		||||
github.com/matoous/go-nanoid v1.5.1/go.mod h1:zyD2a71IubI24efhpvkJz+ZwfwagzgSO6UNiFsZKN7U=
 | 
			
		||||
github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR38lUII=
 | 
			
		||||
github.com/rogpeppe/go-internal v1.13.1/go.mod h1:uMEvuHeurkdAXX61udpOXGD/AzZDWNMNyH2VO9fmH0o=
 | 
			
		||||
github.com/spf13/cast v1.7.1 h1:cuNEagBQEHWN1FnbGEjCXL2szYEXqfJPbP2HNUaca9Y=
 | 
			
		||||
github.com/spf13/cast v1.7.1/go.mod h1:ancEpBxwJDODSW/UG4rDrAqiKolqNNh2DX3mk86cAdo=
 | 
			
		||||
go.opentelemetry.io/otel v1.32.0 h1:WnBN+Xjcteh0zdk01SVqV55d/m62NJLJdIyb4y/WO5U=
 | 
			
		||||
go.opentelemetry.io/otel v1.32.0/go.mod h1:00DCVSB0RQcnzlwyTfqtxSm+DRr9hpYrHjNGiBHVQIg=
 | 
			
		||||
go.opentelemetry.io/otel/metric v1.32.0 h1:xV2umtmNcThh2/a/aCP+h64Xx5wsj8qqnkYZktzNa0M=
 | 
			
		||||
go.opentelemetry.io/otel/metric v1.32.0/go.mod h1:jH7CIbbK6SH2V2wE16W05BHCtIDzauciCRLoc/SyMv8=
 | 
			
		||||
go.opentelemetry.io/otel/sdk v1.32.0 h1:RNxepc9vK59A8XsgZQouW8ue8Gkb4jpWtJm9ge5lEG4=
 | 
			
		||||
go.opentelemetry.io/otel/sdk v1.32.0/go.mod h1:LqgegDBjKMmb2GC6/PrTnteJG39I8/vJCAP9LlJXEjU=
 | 
			
		||||
go.opentelemetry.io/otel/sdk/metric v1.32.0 h1:rZvFnvmvawYb0alrYkjraqJq0Z4ZUJAiyYCU9snn1CU=
 | 
			
		||||
go.opentelemetry.io/otel/sdk/metric v1.32.0/go.mod h1:PWeZlq0zt9YkYAp3gjKZ0eicRYvOh1Gd+X99x6GHpCQ=
 | 
			
		||||
go.opentelemetry.io/otel/trace v1.32.0 h1:WIC9mYrXf8TmY/EXuULKc8hR17vE+Hjv2cssQDe03fM=
 | 
			
		||||
go.opentelemetry.io/otel/trace v1.32.0/go.mod h1:+i4rkvCraA+tG6AzwloGaCtkx53Fa+L+V8e9a7YvhT8=
 | 
			
		||||
go.unistack.org/micro-proto/v4 v4.1.0 h1:qPwL2n/oqh9RE3RTTDgt28XK3QzV597VugQPaw9lKUk=
 | 
			
		||||
go.unistack.org/micro-proto/v4 v4.1.0/go.mod h1:ArmK7o+uFvxSY3dbJhKBBX4Pm1rhWdLEFf3LxBrMtec=
 | 
			
		||||
go.unistack.org/micro/v4 v4.1.2 h1:9SOlPYyPNNFpg1A7BsvhDyQm3gysLH1AhWbDCp1hyoY=
 | 
			
		||||
go.unistack.org/micro/v4 v4.1.2/go.mod h1:lr3oYED8Ay1vjK68QqRw30QOtdk/ffpZqMFDasOUhKw=
 | 
			
		||||
golang.org/x/net v0.35.0 h1:T5GQRQb2y08kTAByq9L4/bz8cipCdA8FbRTXewonqY8=
 | 
			
		||||
golang.org/x/net v0.35.0/go.mod h1:EglIi67kWsHKlRzzVMUD93VMSWGFOMSZgxFjparz1Qk=
 | 
			
		||||
golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc=
 | 
			
		||||
golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
 | 
			
		||||
golang.org/x/text v0.22.0 h1:bofq7m3/HAFvbF51jz3Q9wLg3jkvSPuiZu/pD1XwgtM=
 | 
			
		||||
golang.org/x/text v0.22.0/go.mod h1:YRoo4H8PVmsu+E3Ou7cqLVH8oXWIHVoX0jqUWALQhfY=
 | 
			
		||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250224174004-546df14abb99 h1:ZSlhAUqC4r8TPzqLXQ0m3upBNZeF+Y8jQ3c4CR3Ujms=
 | 
			
		||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250224174004-546df14abb99/go.mod h1:LuRYeWDFV6WOn90g357N17oMCaxpgCnbi/44qJvDn2I=
 | 
			
		||||
google.golang.org/grpc v1.70.0 h1:pWFv03aZoHzlRKHWicjsZytKAiYCtNS0dHbXnIdq7jQ=
 | 
			
		||||
google.golang.org/grpc v1.70.0/go.mod h1:ofIJqVKDXx/JiXrwr2IG4/zwdH9txy3IlF40RmcJSQw=
 | 
			
		||||
google.golang.org/protobuf v1.36.5 h1:tPhr+woSbjfYvY6/GPufUoYizxw1cF/yFoxJ2fmpwlM=
 | 
			
		||||
google.golang.org/protobuf v1.36.5/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE=
 | 
			
		||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
 | 
			
		||||
gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU=
 | 
			
		||||
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
			
		||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
 | 
			
		||||
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										478
									
								
								grpc.go
									
									
									
									
									
								
							
							
						
						
									
										478
									
								
								grpc.go
									
									
									
									
									
								
							@@ -1,4 +1,4 @@
 | 
			
		||||
// Package grpc provides a gRPC client
 | 
			
		||||
// Package grpc provides a gRPC client for micro framework
 | 
			
		||||
package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
@@ -7,44 +7,52 @@ import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"net"
 | 
			
		||||
	"reflect"
 | 
			
		||||
	"strconv"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"sync"
 | 
			
		||||
	"sync/atomic"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
	"github.com/unistack-org/micro/v3/broker"
 | 
			
		||||
	"github.com/unistack-org/micro/v3/client"
 | 
			
		||||
	"github.com/unistack-org/micro/v3/codec"
 | 
			
		||||
	"github.com/unistack-org/micro/v3/errors"
 | 
			
		||||
	"github.com/unistack-org/micro/v3/metadata"
 | 
			
		||||
	"go.unistack.org/micro/v4/client"
 | 
			
		||||
	"go.unistack.org/micro/v4/codec"
 | 
			
		||||
	"go.unistack.org/micro/v4/errors"
 | 
			
		||||
	"go.unistack.org/micro/v4/metadata"
 | 
			
		||||
	"go.unistack.org/micro/v4/options"
 | 
			
		||||
	"go.unistack.org/micro/v4/selector"
 | 
			
		||||
	"go.unistack.org/micro/v4/semconv"
 | 
			
		||||
	"go.unistack.org/micro/v4/tracer"
 | 
			
		||||
	"google.golang.org/grpc"
 | 
			
		||||
	"google.golang.org/grpc/codes"
 | 
			
		||||
	"google.golang.org/grpc/credentials"
 | 
			
		||||
	"google.golang.org/grpc/credentials/insecure"
 | 
			
		||||
	"google.golang.org/grpc/encoding"
 | 
			
		||||
	gmetadata "google.golang.org/grpc/metadata"
 | 
			
		||||
	"google.golang.org/grpc/status"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
const (
 | 
			
		||||
	DefaultContentType = "application/grpc"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type grpcClient struct {
 | 
			
		||||
	opts   client.Options
 | 
			
		||||
	codecs map[string]encoding.Codec
 | 
			
		||||
	pool   *pool
 | 
			
		||||
	once   atomic.Value
 | 
			
		||||
	funcCall   client.FuncCall
 | 
			
		||||
	funcStream client.FuncStream
 | 
			
		||||
	pool       *ConnPool
 | 
			
		||||
	opts       client.Options
 | 
			
		||||
	sync.RWMutex
 | 
			
		||||
	init bool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// secure returns the dial option for whether its a secure or insecure connection
 | 
			
		||||
func (g *grpcClient) secure(addr string) grpc.DialOption {
 | 
			
		||||
	// first we check if theres'a  tls config
 | 
			
		||||
	if g.opts.Context != nil {
 | 
			
		||||
		if v := g.opts.Context.Value(tlsAuth{}); v != nil {
 | 
			
		||||
			tls := v.(*tls.Config)
 | 
			
		||||
			creds := credentials.NewTLS(tls)
 | 
			
		||||
			// return tls config if it exists
 | 
			
		||||
			return grpc.WithTransportCredentials(creds)
 | 
			
		||||
		}
 | 
			
		||||
	if g.opts.TLSConfig != nil {
 | 
			
		||||
		creds := credentials.NewTLS(g.opts.TLSConfig)
 | 
			
		||||
		// return tls config if it exists
 | 
			
		||||
		return grpc.WithTransportCredentials(creds)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// default config
 | 
			
		||||
	tlsConfig := &tls.Config{}
 | 
			
		||||
	tlsConfig := &tls.Config{MinVersion: tls.VersionTLS12}
 | 
			
		||||
	defaultCreds := grpc.WithTransportCredentials(credentials.NewTLS(tlsConfig))
 | 
			
		||||
 | 
			
		||||
	// check if the address is prepended with https
 | 
			
		||||
@@ -62,36 +70,37 @@ func (g *grpcClient) secure(addr string) grpc.DialOption {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// other fallback to insecure
 | 
			
		||||
	return grpc.WithInsecure()
 | 
			
		||||
	return grpc.WithTransportCredentials(insecure.NewCredentials())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) call(ctx context.Context, addr string, req client.Request, rsp interface{}, opts client.CallOptions) error {
 | 
			
		||||
	var header map[string]string
 | 
			
		||||
	var header map[string][]string
 | 
			
		||||
 | 
			
		||||
	if md, ok := metadata.FromContext(ctx); ok {
 | 
			
		||||
		header = make(map[string]string, len(md))
 | 
			
		||||
		for k, v := range md {
 | 
			
		||||
			header[strings.ToLower(k)] = v
 | 
			
		||||
		}
 | 
			
		||||
	if md, ok := metadata.FromOutgoingContext(ctx); ok {
 | 
			
		||||
		header = metadata.Copy(md)
 | 
			
		||||
	} else {
 | 
			
		||||
		header = make(map[string]string, 2)
 | 
			
		||||
		header = make(map[string][]string, 2)
 | 
			
		||||
	}
 | 
			
		||||
	if opts.RequestMetadata != nil {
 | 
			
		||||
		for k, v := range opts.RequestMetadata {
 | 
			
		||||
			header[k] = v
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// set timeout in nanoseconds
 | 
			
		||||
	header["timeout"] = fmt.Sprintf("%d", opts.RequestTimeout)
 | 
			
		||||
	// set the content type for the request
 | 
			
		||||
	header["x-content-type"] = req.ContentType()
 | 
			
		||||
	header["grpc-timeout"] = append(header["grpc-timeout"], fmt.Sprintf("%dn", opts.RequestTimeout))
 | 
			
		||||
	header["timeout"] = append(header["timeout"], fmt.Sprintf("%dn", opts.RequestTimeout))
 | 
			
		||||
	header["content-type"] = append(header["content-type"], req.ContentType())
 | 
			
		||||
 | 
			
		||||
	md := gmetadata.New(header)
 | 
			
		||||
	ctx = gmetadata.NewOutgoingContext(ctx, md)
 | 
			
		||||
	ctx = gmetadata.NewOutgoingContext(ctx, header)
 | 
			
		||||
 | 
			
		||||
	cf, err := g.newGRPCCodec(req.ContentType())
 | 
			
		||||
	cf, err := g.newCodec(req.ContentType())
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "%+v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	maxRecvMsgSize := g.maxRecvMsgSizeValue()
 | 
			
		||||
	maxSendMsgSize := g.maxSendMsgSizeValue()
 | 
			
		||||
	cfgService := g.serviceConfig()
 | 
			
		||||
 | 
			
		||||
	var grr error
 | 
			
		||||
 | 
			
		||||
@@ -110,30 +119,50 @@ func (g *grpcClient) call(ctx context.Context, addr string, req client.Request,
 | 
			
		||||
			grpc.MaxCallRecvMsgSize(maxRecvMsgSize),
 | 
			
		||||
			grpc.MaxCallSendMsgSize(maxSendMsgSize),
 | 
			
		||||
		),
 | 
			
		||||
		grpc.WithDefaultServiceConfig(cfgService),
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if opts := g.getGrpcDialOptions(); opts != nil {
 | 
			
		||||
	if opts := g.getGrpcDialOptions(g.opts.Context); opts != nil {
 | 
			
		||||
		grpcDialOptions = append(grpcDialOptions, opts...)
 | 
			
		||||
	}
 | 
			
		||||
	if opts := g.getGrpcDialOptions(opts.Context); opts != nil {
 | 
			
		||||
		grpcDialOptions = append(grpcDialOptions, opts...)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cc, err := g.pool.getConn(dialCtx, addr, grpcDialOptions...)
 | 
			
		||||
	contextDialer := g.opts.ContextDialer
 | 
			
		||||
	if opts.ContextDialer != nil {
 | 
			
		||||
		contextDialer = opts.ContextDialer
 | 
			
		||||
	}
 | 
			
		||||
	if contextDialer != nil {
 | 
			
		||||
		grpcDialOptions = append(grpcDialOptions, grpc.WithContextDialer(contextDialer))
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cc, err := g.pool.Get(dialCtx, addr, grpcDialOptions...)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", fmt.Sprintf("Error sending request: %v", err))
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "Error sending request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
	defer func() {
 | 
			
		||||
		// defer execution of release
 | 
			
		||||
		g.pool.release(addr, cc, grr)
 | 
			
		||||
		g.pool.Put(cc, grr)
 | 
			
		||||
	}()
 | 
			
		||||
 | 
			
		||||
	ch := make(chan error, 1)
 | 
			
		||||
	var gmd gmetadata.MD
 | 
			
		||||
 | 
			
		||||
	grpcCallOptions := []grpc.CallOption{
 | 
			
		||||
		grpc.CallContentSubtype((&wrapMicroCodec{cf}).Name()),
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if opts := g.getGrpcCallOptions(opts.Context); opts != nil {
 | 
			
		||||
		grpcCallOptions = append(grpcCallOptions, opts...)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if opts.ResponseMetadata != nil {
 | 
			
		||||
		gmd = gmetadata.MD{}
 | 
			
		||||
		grpcCallOptions = append(grpcCallOptions, grpc.Header(&gmd))
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	go func() {
 | 
			
		||||
		grpcCallOptions := []grpc.CallOption{
 | 
			
		||||
			grpc.ForceCodec(&wrapMicroCodec{cf}),
 | 
			
		||||
			grpc.CallContentSubtype((&wrapMicroCodec{cf}).Name())}
 | 
			
		||||
		if opts := g.getGrpcCallOptions(); opts != nil {
 | 
			
		||||
			grpcCallOptions = append(grpcCallOptions, opts...)
 | 
			
		||||
		}
 | 
			
		||||
		err := cc.Invoke(ctx, methodToGRPC(req.Service(), req.Endpoint()), req.Body(), rsp, grpcCallOptions...)
 | 
			
		||||
		ch <- microError(err)
 | 
			
		||||
	}()
 | 
			
		||||
@@ -145,34 +174,38 @@ func (g *grpcClient) call(ctx context.Context, addr string, req client.Request,
 | 
			
		||||
		grr = errors.Timeout("go.micro.client", "%v", ctx.Err())
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if opts.ResponseMetadata != nil {
 | 
			
		||||
		*opts.ResponseMetadata = metadata.New(gmd.Len())
 | 
			
		||||
		for k, v := range gmd {
 | 
			
		||||
			opts.ResponseMetadata.Append(k, v...)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return grr
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) stream(ctx context.Context, addr string, req client.Request, rsp interface{}, opts client.CallOptions) error {
 | 
			
		||||
	var header map[string]string
 | 
			
		||||
	var header map[string][]string
 | 
			
		||||
 | 
			
		||||
	if md, ok := metadata.FromContext(ctx); ok {
 | 
			
		||||
		header = make(map[string]string, len(md))
 | 
			
		||||
		for k, v := range md {
 | 
			
		||||
			header[k] = v
 | 
			
		||||
		}
 | 
			
		||||
	if md, ok := metadata.FromOutgoingContext(ctx); ok {
 | 
			
		||||
		header = metadata.Copy(md)
 | 
			
		||||
	} else {
 | 
			
		||||
		header = make(map[string]string)
 | 
			
		||||
		header = make(map[string][]string)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// set timeout in nanoseconds
 | 
			
		||||
	if opts.StreamTimeout > time.Duration(0) {
 | 
			
		||||
		header["timeout"] = fmt.Sprintf("%d", opts.StreamTimeout)
 | 
			
		||||
		header["grpc-timeout"] = append(header["grpc-timeout"], fmt.Sprintf("%dn", opts.RequestTimeout))
 | 
			
		||||
		header["timeout"] = append(header["timeout"], fmt.Sprintf("%dn", opts.RequestTimeout))
 | 
			
		||||
	}
 | 
			
		||||
	// set the content type for the request
 | 
			
		||||
	header["x-content-type"] = req.ContentType()
 | 
			
		||||
	header["content-type"] = append(header["content-type"], req.ContentType())
 | 
			
		||||
 | 
			
		||||
	md := gmetadata.New(header)
 | 
			
		||||
	ctx = gmetadata.NewOutgoingContext(ctx, md)
 | 
			
		||||
	ctx = gmetadata.NewOutgoingContext(ctx, header)
 | 
			
		||||
 | 
			
		||||
	cf, err := g.newGRPCCodec(req.ContentType())
 | 
			
		||||
	cf, err := g.newCodec(req.ContentType())
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "%+v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var dialCtx context.Context
 | 
			
		||||
@@ -186,17 +219,34 @@ func (g *grpcClient) stream(ctx context.Context, addr string, req client.Request
 | 
			
		||||
 | 
			
		||||
	wc := &wrapMicroCodec{cf}
 | 
			
		||||
 | 
			
		||||
	maxRecvMsgSize := g.maxRecvMsgSizeValue()
 | 
			
		||||
	maxSendMsgSize := g.maxSendMsgSizeValue()
 | 
			
		||||
	cfgService := g.serviceConfig()
 | 
			
		||||
 | 
			
		||||
	grpcDialOptions := []grpc.DialOption{
 | 
			
		||||
		g.secure(addr),
 | 
			
		||||
		grpc.WithDefaultCallOptions(
 | 
			
		||||
			grpc.MaxCallRecvMsgSize(maxRecvMsgSize),
 | 
			
		||||
			grpc.MaxCallSendMsgSize(maxSendMsgSize),
 | 
			
		||||
		),
 | 
			
		||||
		grpc.WithDefaultServiceConfig(cfgService),
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if opts := g.getGrpcDialOptions(); opts != nil {
 | 
			
		||||
	if opts := g.getGrpcDialOptions(opts.Context); opts != nil {
 | 
			
		||||
		grpcDialOptions = append(grpcDialOptions, opts...)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cc, err := g.pool.getConn(dialCtx, addr, grpcDialOptions...)
 | 
			
		||||
	contextDialer := g.opts.ContextDialer
 | 
			
		||||
	if opts.ContextDialer != nil {
 | 
			
		||||
		contextDialer = opts.ContextDialer
 | 
			
		||||
	}
 | 
			
		||||
	if contextDialer != nil {
 | 
			
		||||
		grpcDialOptions = append(grpcDialOptions, grpc.WithContextDialer(contextDialer))
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cc, err := g.pool.Get(dialCtx, addr, grpcDialOptions...)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", fmt.Sprintf("Error sending request: %v", err))
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "Error sending request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	desc := &grpc.StreamDesc{
 | 
			
		||||
@@ -206,12 +256,17 @@ func (g *grpcClient) stream(ctx context.Context, addr string, req client.Request
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	grpcCallOptions := []grpc.CallOption{
 | 
			
		||||
		grpc.ForceCodec(wc),
 | 
			
		||||
		// grpc.ForceCodec(wc),
 | 
			
		||||
		grpc.CallContentSubtype(wc.Name()),
 | 
			
		||||
	}
 | 
			
		||||
	if opts := g.getGrpcCallOptions(); opts != nil {
 | 
			
		||||
	if opts := g.getGrpcCallOptions(opts.Context); opts != nil {
 | 
			
		||||
		grpcCallOptions = append(grpcCallOptions, opts...)
 | 
			
		||||
	}
 | 
			
		||||
	var gmd gmetadata.MD
 | 
			
		||||
	if opts.ResponseMetadata != nil {
 | 
			
		||||
		gmd = gmetadata.MD{}
 | 
			
		||||
		grpcCallOptions = append(grpcCallOptions, grpc.Header(&gmd))
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// create a new cancelling context
 | 
			
		||||
	newCtx, cancel := context.WithCancel(ctx)
 | 
			
		||||
@@ -222,9 +277,9 @@ func (g *grpcClient) stream(ctx context.Context, addr string, req client.Request
 | 
			
		||||
		// cancel the context
 | 
			
		||||
		cancel()
 | 
			
		||||
		// release the connection
 | 
			
		||||
		g.pool.release(addr, cc, err)
 | 
			
		||||
		g.pool.Put(cc, err)
 | 
			
		||||
		// now return the error
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", fmt.Sprintf("Error creating stream: %v", err))
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "Error creating stream: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// set request codec
 | 
			
		||||
@@ -244,19 +299,20 @@ func (g *grpcClient) stream(ctx context.Context, addr string, req client.Request
 | 
			
		||||
		},
 | 
			
		||||
		conn: cc,
 | 
			
		||||
		close: func(err error) {
 | 
			
		||||
			// cancel the context if an error occured
 | 
			
		||||
			// cancel the context if an error occurred
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				cancel()
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			// defer execution of release
 | 
			
		||||
			g.pool.release(addr, cc, err)
 | 
			
		||||
			g.pool.Put(cc, err)
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// set the stream as the response
 | 
			
		||||
	val := reflect.ValueOf(rsp).Elem()
 | 
			
		||||
	val.Set(reflect.ValueOf(stream).Elem())
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -304,17 +360,35 @@ func (g *grpcClient) maxSendMsgSizeValue() int {
 | 
			
		||||
	return v.(int)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) newGRPCCodec(ct string) (codec.Codec, error) {
 | 
			
		||||
func (g *grpcClient) newCodec(ct string) (codec.Codec, error) {
 | 
			
		||||
	g.RLock()
 | 
			
		||||
	defer g.RUnlock()
 | 
			
		||||
 | 
			
		||||
	if idx := strings.IndexRune(ct, ';'); idx >= 0 {
 | 
			
		||||
		ct = ct[:idx]
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if c, ok := g.opts.Codecs[ct]; ok {
 | 
			
		||||
		return c, nil
 | 
			
		||||
	}
 | 
			
		||||
	return nil, codec.ErrUnknownContentType
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) serviceConfig() string {
 | 
			
		||||
	if g.opts.Context == nil {
 | 
			
		||||
		return DefaultServiceConfig
 | 
			
		||||
	}
 | 
			
		||||
	v := g.opts.Context.Value(serviceConfigKey{})
 | 
			
		||||
	if v == nil {
 | 
			
		||||
		return DefaultServiceConfig
 | 
			
		||||
	}
 | 
			
		||||
	return v.(string)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) Init(opts ...client.Option) error {
 | 
			
		||||
	if len(opts) == 0 && g.init {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	size := g.opts.PoolSize
 | 
			
		||||
	ttl := g.opts.PoolTTL
 | 
			
		||||
 | 
			
		||||
@@ -330,6 +404,19 @@ func (g *grpcClient) Init(opts ...client.Option) error {
 | 
			
		||||
		g.pool.Unlock()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	g.funcCall = g.fnCall
 | 
			
		||||
	g.funcStream = g.fnStream
 | 
			
		||||
 | 
			
		||||
	g.opts.Hooks.EachPrev(func(hook options.Hook) {
 | 
			
		||||
		switch h := hook.(type) {
 | 
			
		||||
		case client.HookCall:
 | 
			
		||||
			g.funcCall = h(g.funcCall)
 | 
			
		||||
		case client.HookStream:
 | 
			
		||||
			g.funcStream = h(g.funcStream)
 | 
			
		||||
		}
 | 
			
		||||
	})
 | 
			
		||||
	g.init = true
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -337,22 +424,14 @@ func (g *grpcClient) Options() client.Options {
 | 
			
		||||
	return g.opts
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) NewMessage(topic string, msg interface{}, opts ...client.MessageOption) client.Message {
 | 
			
		||||
	return newGRPCEvent(topic, msg, g.opts.ContentType, opts...)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) NewRequest(service, method string, req interface{}, reqOpts ...client.RequestOption) client.Request {
 | 
			
		||||
	return newGRPCRequest(service, method, req, g.opts.ContentType, reqOpts...)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error {
 | 
			
		||||
	if req == nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "req is nil")
 | 
			
		||||
	} else if rsp == nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "rsp is nil")
 | 
			
		||||
	}
 | 
			
		||||
func (g *grpcClient) fnCall(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error {
 | 
			
		||||
	// make a copy of call opts
 | 
			
		||||
	callOpts := g.opts.CallOptions
 | 
			
		||||
 | 
			
		||||
	for _, opt := range opts {
 | 
			
		||||
		opt(&callOpts)
 | 
			
		||||
	}
 | 
			
		||||
@@ -381,11 +460,6 @@ func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface
 | 
			
		||||
	// make copy of call method
 | 
			
		||||
	gcall := g.call
 | 
			
		||||
 | 
			
		||||
	// wrap the call in reverse
 | 
			
		||||
	for i := len(callOpts.CallWrappers); i > 0; i-- {
 | 
			
		||||
		gcall = callOpts.CallWrappers[i-1](gcall)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// use the router passed as a call option, or fallback to the rpc clients router
 | 
			
		||||
	if callOpts.Router == nil {
 | 
			
		||||
		callOpts.Router = g.opts.Router
 | 
			
		||||
@@ -401,25 +475,13 @@ func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface
 | 
			
		||||
		callOpts.Address = []string{g.opts.Proxy}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// lookup the route to send the reques to
 | 
			
		||||
	// TODO apply any filtering here
 | 
			
		||||
	routes, err := g.opts.Lookup(ctx, req, callOpts)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
	}
 | 
			
		||||
	var next selector.Next
 | 
			
		||||
 | 
			
		||||
	// balance the list of nodes
 | 
			
		||||
	next, err := callOpts.Selector.Select(routes)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// return errors.New("go.micro.client", "request timeout", 408)
 | 
			
		||||
	call := func(i int) error {
 | 
			
		||||
		// call backoff first. Someone may want an initial start delay
 | 
			
		||||
		t, err := callOpts.Backoff(ctx, req, i)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
			return errors.InternalServerError("go.micro.client", "%+v", err)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// only sleep if greater than 0
 | 
			
		||||
@@ -427,6 +489,23 @@ func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface
 | 
			
		||||
			time.Sleep(t)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if next == nil {
 | 
			
		||||
			var routes []string
 | 
			
		||||
 | 
			
		||||
			// lookup the route to send the reques to
 | 
			
		||||
			// TODO apply any filtering here
 | 
			
		||||
			routes, err = g.opts.Lookup(ctx, req, callOpts)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				return errors.InternalServerError("go.micro.client", "%+v", err)
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			// balance the list of nodes
 | 
			
		||||
			next, err = callOpts.Selector.Select(routes)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				return err
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// get the next node
 | 
			
		||||
		node := next()
 | 
			
		||||
 | 
			
		||||
@@ -479,7 +558,38 @@ func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface
 | 
			
		||||
	return gerr
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) {
 | 
			
		||||
func (g *grpcClient) Call(ctx context.Context, req client.Request, rsp interface{}, opts ...client.CallOption) error {
 | 
			
		||||
	if req == nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "req is nil")
 | 
			
		||||
	} else if rsp == nil {
 | 
			
		||||
		return errors.InternalServerError("go.micro.client", "rsp is nil")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ts := time.Now()
 | 
			
		||||
	g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Inc()
 | 
			
		||||
	var sp tracer.Span
 | 
			
		||||
	ctx, sp = g.opts.Tracer.Start(ctx, req.Endpoint()+" rpc-client",
 | 
			
		||||
		tracer.WithSpanKind(tracer.SpanKindClient),
 | 
			
		||||
		tracer.WithSpanLabels("endpoint", req.Endpoint()),
 | 
			
		||||
	)
 | 
			
		||||
	err := g.funcCall(ctx, req, rsp, opts...)
 | 
			
		||||
	g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Dec()
 | 
			
		||||
	te := time.Since(ts)
 | 
			
		||||
	g.opts.Meter.Summary(semconv.ClientRequestLatencyMicroseconds, "endpoint", req.Endpoint()).Update(te.Seconds())
 | 
			
		||||
	g.opts.Meter.Histogram(semconv.ClientRequestDurationSeconds, "endpoint", req.Endpoint()).Update(te.Seconds())
 | 
			
		||||
 | 
			
		||||
	if me := errors.FromError(err); me == nil {
 | 
			
		||||
		sp.Finish()
 | 
			
		||||
		g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "success", "code", strconv.Itoa(int(200))).Inc()
 | 
			
		||||
	} else {
 | 
			
		||||
		sp.SetStatus(tracer.SpanStatusError, err.Error())
 | 
			
		||||
		g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "failure", "code", strconv.Itoa(int(me.Code))).Inc()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) fnStream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) {
 | 
			
		||||
	// make a copy of call opts
 | 
			
		||||
	callOpts := g.opts.CallOptions
 | 
			
		||||
	for _, opt := range opts {
 | 
			
		||||
@@ -498,11 +608,6 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli
 | 
			
		||||
	// 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)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// use the router passed as a call option, or fallback to the rpc clients router
 | 
			
		||||
	if callOpts.Router == nil {
 | 
			
		||||
		callOpts.Router = g.opts.Router
 | 
			
		||||
@@ -518,24 +623,13 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli
 | 
			
		||||
		callOpts.Address = []string{g.opts.Proxy}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// lookup the route to send the reques to
 | 
			
		||||
	// TODO: move to internal lookup func
 | 
			
		||||
	routes, err := g.opts.Lookup(ctx, req, callOpts)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// balance the list of nodes
 | 
			
		||||
	next, err := callOpts.Selector.Select(routes)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
	var next selector.Next
 | 
			
		||||
 | 
			
		||||
	call := func(i int) (client.Stream, error) {
 | 
			
		||||
		// call backoff first. Someone may want an initial start delay
 | 
			
		||||
		t, err := callOpts.Backoff(ctx, req, i)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			return nil, errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
			return nil, errors.InternalServerError("go.micro.client", "%+v", err)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// only sleep if greater than 0
 | 
			
		||||
@@ -543,6 +637,23 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli
 | 
			
		||||
			time.Sleep(t)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if next == nil {
 | 
			
		||||
			var routes []string
 | 
			
		||||
 | 
			
		||||
			// lookup the route to send the reques to
 | 
			
		||||
			// TODO apply any filtering here
 | 
			
		||||
			routes, err = g.opts.Lookup(ctx, req, callOpts)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				return nil, errors.InternalServerError("go.micro.client", "%+v", err)
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			// balance the list of nodes
 | 
			
		||||
			next, err = callOpts.Selector.Select(routes)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				return nil, err
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// get the next node
 | 
			
		||||
		node := next()
 | 
			
		||||
 | 
			
		||||
@@ -606,46 +717,29 @@ func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...cli
 | 
			
		||||
	return nil, grr
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) Publish(ctx context.Context, p client.Message, opts ...client.PublishOption) error {
 | 
			
		||||
	var body []byte
 | 
			
		||||
func (g *grpcClient) Stream(ctx context.Context, req client.Request, opts ...client.CallOption) (client.Stream, error) {
 | 
			
		||||
	ts := time.Now()
 | 
			
		||||
	g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Inc()
 | 
			
		||||
	var sp tracer.Span
 | 
			
		||||
	ctx, sp = g.opts.Tracer.Start(ctx, req.Endpoint()+" rpc-client",
 | 
			
		||||
		tracer.WithSpanKind(tracer.SpanKindClient),
 | 
			
		||||
		tracer.WithSpanLabels("endpoint", req.Endpoint()),
 | 
			
		||||
	)
 | 
			
		||||
	stream, err := g.funcStream(ctx, req, opts...)
 | 
			
		||||
	g.opts.Meter.Counter(semconv.ClientRequestInflight, "endpoint", req.Endpoint()).Dec()
 | 
			
		||||
	te := time.Since(ts)
 | 
			
		||||
	g.opts.Meter.Summary(semconv.ClientRequestLatencyMicroseconds, "endpoint", req.Endpoint()).Update(te.Seconds())
 | 
			
		||||
	g.opts.Meter.Histogram(semconv.ClientRequestDurationSeconds, "endpoint", req.Endpoint()).Update(te.Seconds())
 | 
			
		||||
 | 
			
		||||
	options := client.NewPublishOptions(opts...)
 | 
			
		||||
 | 
			
		||||
	md, ok := metadata.FromContext(ctx)
 | 
			
		||||
	if !ok {
 | 
			
		||||
		md = metadata.New(0)
 | 
			
		||||
	}
 | 
			
		||||
	md["Content-Type"] = p.ContentType()
 | 
			
		||||
	md["Micro-Topic"] = p.Topic()
 | 
			
		||||
 | 
			
		||||
	// passed in raw data
 | 
			
		||||
	if d, ok := p.Payload().(*codec.Frame); ok {
 | 
			
		||||
		body = d.Data
 | 
			
		||||
	if me := status.Convert(err); me == nil {
 | 
			
		||||
		sp.Finish()
 | 
			
		||||
		g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "success", "code", strconv.Itoa(int(codes.OK))).Inc()
 | 
			
		||||
	} 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 {
 | 
			
		||||
			return errors.InternalServerError("go.micro.client", err.Error())
 | 
			
		||||
		}
 | 
			
		||||
		body = b
 | 
			
		||||
		sp.SetStatus(tracer.SpanStatusError, err.Error())
 | 
			
		||||
		g.opts.Meter.Counter(semconv.ClientRequestTotal, "endpoint", req.Endpoint(), "status", "failure", "code", strconv.Itoa(int(me.Code()))).Inc()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	topic := p.Topic()
 | 
			
		||||
 | 
			
		||||
	// get the exchange
 | 
			
		||||
	if len(options.Exchange) > 0 {
 | 
			
		||||
		topic = options.Exchange
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return g.opts.Broker.Publish(ctx, topic, &broker.Message{
 | 
			
		||||
		Header: md,
 | 
			
		||||
		Body:   body,
 | 
			
		||||
	}, broker.PublishContext(options.Context))
 | 
			
		||||
	return stream, err
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) String() string {
 | 
			
		||||
@@ -656,41 +750,45 @@ func (g *grpcClient) Name() string {
 | 
			
		||||
	return g.opts.Name
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) getGrpcDialOptions() []grpc.DialOption {
 | 
			
		||||
	if g.opts.CallOptions.Context == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
func (g *grpcClient) getGrpcDialOptions(ctx context.Context) []grpc.DialOption {
 | 
			
		||||
	var opts []grpc.DialOption
 | 
			
		||||
 | 
			
		||||
	if g.opts.CallOptions.Context != nil {
 | 
			
		||||
		if v := g.opts.CallOptions.Context.Value(grpcDialOptions{}); v != nil {
 | 
			
		||||
			if vopts, ok := v.([]grpc.DialOption); ok {
 | 
			
		||||
				opts = append(opts, vopts...)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	v := g.opts.CallOptions.Context.Value(grpcDialOptions{})
 | 
			
		||||
 | 
			
		||||
	if v == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	opts, ok := v.([]grpc.DialOption)
 | 
			
		||||
 | 
			
		||||
	if !ok {
 | 
			
		||||
		return nil
 | 
			
		||||
	if ctx != nil {
 | 
			
		||||
		if v := ctx.Value(grpcDialOptions{}); v != nil {
 | 
			
		||||
			if vopts, ok := v.([]grpc.DialOption); ok {
 | 
			
		||||
				opts = append(opts, vopts...)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return opts
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcClient) getGrpcCallOptions() []grpc.CallOption {
 | 
			
		||||
	if g.opts.CallOptions.Context == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
func (g *grpcClient) getGrpcCallOptions(ctx context.Context) []grpc.CallOption {
 | 
			
		||||
	var opts []grpc.CallOption
 | 
			
		||||
 | 
			
		||||
	if g.opts.CallOptions.Context != nil {
 | 
			
		||||
		if v := g.opts.CallOptions.Context.Value(grpcCallOptions{}); v != nil {
 | 
			
		||||
			if vopts, ok := v.([]grpc.CallOption); ok {
 | 
			
		||||
				opts = append(opts, vopts...)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	v := g.opts.CallOptions.Context.Value(grpcCallOptions{})
 | 
			
		||||
 | 
			
		||||
	if v == nil {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	opts, ok := v.([]grpc.CallOption)
 | 
			
		||||
 | 
			
		||||
	if !ok {
 | 
			
		||||
		return nil
 | 
			
		||||
	if ctx != nil {
 | 
			
		||||
		if v := ctx.Value(grpcCallOptions{}); v != nil {
 | 
			
		||||
			if vopts, ok := v.([]grpc.CallOption); ok {
 | 
			
		||||
				opts = append(opts, vopts...)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return opts
 | 
			
		||||
@@ -699,25 +797,20 @@ func (g *grpcClient) getGrpcCallOptions() []grpc.CallOption {
 | 
			
		||||
func NewClient(opts ...client.Option) client.Client {
 | 
			
		||||
	options := client.NewOptions(opts...)
 | 
			
		||||
	// default content type for grpc
 | 
			
		||||
	options.ContentType = "application/grpc+proto"
 | 
			
		||||
	if options.ContentType == "" {
 | 
			
		||||
		options.ContentType = DefaultContentType
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	rc := &grpcClient{
 | 
			
		||||
	c := &grpcClient{
 | 
			
		||||
		opts: options,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	rc.pool = newPool(options.PoolSize, options.PoolTTL, rc.poolMaxIdle(), rc.poolMaxStreams())
 | 
			
		||||
	c.pool = NewConnPool(options.PoolSize, options.PoolTTL, c.poolMaxIdle(), c.poolMaxStreams())
 | 
			
		||||
 | 
			
		||||
	c := client.Client(rc)
 | 
			
		||||
 | 
			
		||||
	// wrap in reverse
 | 
			
		||||
	for i := len(options.Wrappers); i > 0; i-- {
 | 
			
		||||
		c = options.Wrappers[i-1](c)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if rc.opts.Context != nil {
 | 
			
		||||
		if codecs, ok := rc.opts.Context.Value(codecsKey{}).(map[string]encoding.Codec); ok && codecs != nil {
 | 
			
		||||
	if c.opts.Context != nil {
 | 
			
		||||
		if codecs, ok := c.opts.Context.Value(codecsKey{}).(map[string]encoding.Codec); ok && codecs != nil {
 | 
			
		||||
			for k, v := range codecs {
 | 
			
		||||
				rc.opts.Codecs[k] = &wrapGrpcCodec{v}
 | 
			
		||||
				c.opts.Codecs[k] = &wrapGrpcCodec{v}
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
@@ -726,5 +819,8 @@ func NewClient(opts ...client.Option) client.Client {
 | 
			
		||||
		encoding.RegisterCodec(&wrapMicroCodec{k})
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	c.funcCall = c.fnCall
 | 
			
		||||
	c.funcStream = c.fnStream
 | 
			
		||||
 | 
			
		||||
	return c
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										63
									
								
								grpc_pool.go
									
									
									
									
									
								
							
							
						
						
									
										63
									
								
								grpc_pool.go
									
									
									
									
									
								
							@@ -2,6 +2,7 @@ package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"sync"
 | 
			
		||||
	"time"
 | 
			
		||||
 | 
			
		||||
@@ -9,56 +10,47 @@ import (
 | 
			
		||||
	"google.golang.org/grpc/connectivity"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type pool struct {
 | 
			
		||||
	size int
 | 
			
		||||
	ttl  int64
 | 
			
		||||
 | 
			
		||||
	//  max streams on a *poolConn
 | 
			
		||||
type ConnPool struct {
 | 
			
		||||
	conns      map[string]*streamsPool
 | 
			
		||||
	size       int
 | 
			
		||||
	ttl        int64
 | 
			
		||||
	maxStreams int
 | 
			
		||||
	//  max idle conns
 | 
			
		||||
	maxIdle int
 | 
			
		||||
 | 
			
		||||
	maxIdle    int
 | 
			
		||||
	sync.Mutex
 | 
			
		||||
	conns map[string]*streamsPool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type streamsPool struct {
 | 
			
		||||
	//  head of list
 | 
			
		||||
	head *poolConn
 | 
			
		||||
	head *PoolConn
 | 
			
		||||
	//  busy conns list
 | 
			
		||||
	busy *poolConn
 | 
			
		||||
	busy *PoolConn
 | 
			
		||||
	//  the siza of list
 | 
			
		||||
	count int
 | 
			
		||||
	//  idle conn
 | 
			
		||||
	idle int
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type poolConn struct {
 | 
			
		||||
	//  grpc conn
 | 
			
		||||
type PoolConn struct {
 | 
			
		||||
	err error
 | 
			
		||||
	*grpc.ClientConn
 | 
			
		||||
	err  error
 | 
			
		||||
	addr string
 | 
			
		||||
 | 
			
		||||
	//  pool and streams pool
 | 
			
		||||
	pool    *pool
 | 
			
		||||
	next    *PoolConn
 | 
			
		||||
	pool    *ConnPool
 | 
			
		||||
	sp      *streamsPool
 | 
			
		||||
	pre     *PoolConn
 | 
			
		||||
	addr    string
 | 
			
		||||
	streams int
 | 
			
		||||
	created int64
 | 
			
		||||
 | 
			
		||||
	//  list
 | 
			
		||||
	pre  *poolConn
 | 
			
		||||
	next *poolConn
 | 
			
		||||
	in   bool
 | 
			
		||||
	in      bool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newPool(size int, ttl time.Duration, idle int, ms int) *pool {
 | 
			
		||||
func NewConnPool(size int, ttl time.Duration, idle int, ms int) *ConnPool {
 | 
			
		||||
	if ms <= 0 {
 | 
			
		||||
		ms = 1
 | 
			
		||||
	}
 | 
			
		||||
	if idle < 0 {
 | 
			
		||||
		idle = 0
 | 
			
		||||
	}
 | 
			
		||||
	return &pool{
 | 
			
		||||
	return &ConnPool{
 | 
			
		||||
		size:       size,
 | 
			
		||||
		ttl:        int64(ttl.Seconds()),
 | 
			
		||||
		maxStreams: ms,
 | 
			
		||||
@@ -67,12 +59,15 @@ func newPool(size int, ttl time.Duration, idle int, ms int) *pool {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (p *pool) getConn(ctx context.Context, addr string, opts ...grpc.DialOption) (*poolConn, error) {
 | 
			
		||||
func (p *ConnPool) Get(ctx context.Context, addr string, opts ...grpc.DialOption) (*PoolConn, error) {
 | 
			
		||||
	if strings.HasPrefix(addr, "http") {
 | 
			
		||||
		addr = addr[strings.Index(addr, ":")+3:]
 | 
			
		||||
	}
 | 
			
		||||
	now := time.Now().Unix()
 | 
			
		||||
	p.Lock()
 | 
			
		||||
	sp, ok := p.conns[addr]
 | 
			
		||||
	if !ok {
 | 
			
		||||
		sp = &streamsPool{head: &poolConn{}, busy: &poolConn{}, count: 0, idle: 0}
 | 
			
		||||
		sp = &streamsPool{head: &PoolConn{}, busy: &PoolConn{}, count: 0, idle: 0}
 | 
			
		||||
		p.conns[addr] = sp
 | 
			
		||||
	}
 | 
			
		||||
	//  while we have conns check streams and then return one
 | 
			
		||||
@@ -135,12 +130,12 @@ func (p *pool) getConn(ctx context.Context, addr string, opts ...grpc.DialOption
 | 
			
		||||
	}
 | 
			
		||||
	p.Unlock()
 | 
			
		||||
 | 
			
		||||
	//  create new conn
 | 
			
		||||
	// nolint (TODO need fix)  create new conn)
 | 
			
		||||
	cc, err := grpc.DialContext(ctx, addr, opts...)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
	conn = &poolConn{cc, nil, addr, p, sp, 1, time.Now().Unix(), nil, nil, false}
 | 
			
		||||
	conn = &PoolConn{ClientConn: cc, err: nil, addr: addr, pool: p, sp: sp, streams: 1, created: time.Now().Unix(), pre: nil, next: nil, in: false}
 | 
			
		||||
 | 
			
		||||
	//  add conn to streams pool
 | 
			
		||||
	p.Lock()
 | 
			
		||||
@@ -152,7 +147,7 @@ func (p *pool) getConn(ctx context.Context, addr string, opts ...grpc.DialOption
 | 
			
		||||
	return conn, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (p *pool) release(addr string, conn *poolConn, err error) {
 | 
			
		||||
func (p *ConnPool) Put(conn *PoolConn, err error) {
 | 
			
		||||
	p.Lock()
 | 
			
		||||
	p, sp, created := conn.pool, conn.sp, conn.created
 | 
			
		||||
	//  try to add conn
 | 
			
		||||
@@ -187,11 +182,11 @@ func (p *pool) release(addr string, conn *poolConn, err error) {
 | 
			
		||||
	p.Unlock()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (conn *poolConn) Close() {
 | 
			
		||||
	conn.pool.release(conn.addr, conn, conn.err)
 | 
			
		||||
func (conn *PoolConn) Close() {
 | 
			
		||||
	conn.pool.Put(conn, conn.err)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func removeConn(conn *poolConn) {
 | 
			
		||||
func removeConn(conn *PoolConn) {
 | 
			
		||||
	if conn.pre != nil {
 | 
			
		||||
		conn.pre.next = conn.next
 | 
			
		||||
	}
 | 
			
		||||
@@ -204,7 +199,7 @@ func removeConn(conn *poolConn) {
 | 
			
		||||
	conn.sp.count--
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func addConnAfter(conn *poolConn, after *poolConn) {
 | 
			
		||||
func addConnAfter(conn *PoolConn, after *PoolConn) {
 | 
			
		||||
	conn.next = after.next
 | 
			
		||||
	conn.pre = after
 | 
			
		||||
	if after.next != nil {
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										40
									
								
								message.go
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								message.go
									
									
									
									
									
								
							@@ -1,40 +0,0 @@
 | 
			
		||||
package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"github.com/unistack-org/micro/v3/client"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type grpcEvent struct {
 | 
			
		||||
	topic       string
 | 
			
		||||
	contentType string
 | 
			
		||||
	payload     interface{}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newGRPCEvent(topic string, payload interface{}, contentType string, opts ...client.MessageOption) client.Message {
 | 
			
		||||
	var options client.MessageOptions
 | 
			
		||||
	for _, o := range opts {
 | 
			
		||||
		o(&options)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if len(options.ContentType) > 0 {
 | 
			
		||||
		contentType = options.ContentType
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return &grpcEvent{
 | 
			
		||||
		payload:     payload,
 | 
			
		||||
		topic:       topic,
 | 
			
		||||
		contentType: contentType,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcEvent) ContentType() string {
 | 
			
		||||
	return g.contentType
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcEvent) Topic() string {
 | 
			
		||||
	return g.topic
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcEvent) Payload() interface{} {
 | 
			
		||||
	return g.payload
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										58
									
								
								options.go
									
									
									
									
									
								
							
							
						
						
									
										58
									
								
								options.go
									
									
									
									
									
								
							@@ -3,15 +3,14 @@ package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"crypto/tls"
 | 
			
		||||
 | 
			
		||||
	"github.com/unistack-org/micro/v3/client"
 | 
			
		||||
	"go.unistack.org/micro/v4/client"
 | 
			
		||||
	"google.golang.org/grpc"
 | 
			
		||||
	"google.golang.org/grpc/encoding"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var (
 | 
			
		||||
	// DefaultPoolMaxStreams maximum streams on a connectioin
 | 
			
		||||
	// DefaultPoolMaxStreams maximum streams on a connection
 | 
			
		||||
	// (20)
 | 
			
		||||
	DefaultPoolMaxStreams = 20
 | 
			
		||||
 | 
			
		||||
@@ -26,16 +25,12 @@ var (
 | 
			
		||||
	// DefaultMaxSendMsgSize maximum message that client can send
 | 
			
		||||
	// (4 MB).
 | 
			
		||||
	DefaultMaxSendMsgSize = 1024 * 1024 * 4
 | 
			
		||||
 | 
			
		||||
	// DefaultServiceConfig enable load balancing
 | 
			
		||||
	DefaultServiceConfig = `{"loadBalancingPolicy":"round_robin"}`
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type poolMaxStreams struct{}
 | 
			
		||||
type poolMaxIdle struct{}
 | 
			
		||||
type codecsKey struct{}
 | 
			
		||||
type tlsAuth struct{}
 | 
			
		||||
type maxRecvMsgSizeKey struct{}
 | 
			
		||||
type maxSendMsgSizeKey struct{}
 | 
			
		||||
type grpcDialOptions struct{}
 | 
			
		||||
type grpcCallOptions struct{}
 | 
			
		||||
 | 
			
		||||
// maximum streams on a connectioin
 | 
			
		||||
func PoolMaxStreams(n int) client.Option {
 | 
			
		||||
@@ -47,6 +42,8 @@ func PoolMaxStreams(n int) client.Option {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type poolMaxIdle struct{}
 | 
			
		||||
 | 
			
		||||
// maximum idle conns of a pool
 | 
			
		||||
func PoolMaxIdle(d int) client.Option {
 | 
			
		||||
	return func(o *client.Options) {
 | 
			
		||||
@@ -57,6 +54,8 @@ func PoolMaxIdle(d int) client.Option {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type codecsKey struct{}
 | 
			
		||||
 | 
			
		||||
// gRPC Codec to be used to encode/decode requests for a given content type
 | 
			
		||||
func Codec(contentType string, c encoding.Codec) client.Option {
 | 
			
		||||
	return func(o *client.Options) {
 | 
			
		||||
@@ -72,19 +71,9 @@ func Codec(contentType string, c encoding.Codec) client.Option {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// AuthTLS should be used to setup a secure authentication using TLS
 | 
			
		||||
func AuthTLS(t *tls.Config) client.Option {
 | 
			
		||||
	return func(o *client.Options) {
 | 
			
		||||
		if o.Context == nil {
 | 
			
		||||
			o.Context = context.Background()
 | 
			
		||||
		}
 | 
			
		||||
		o.Context = context.WithValue(o.Context, tlsAuth{}, t)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
type maxRecvMsgSizeKey struct{}
 | 
			
		||||
 | 
			
		||||
//
 | 
			
		||||
// MaxRecvMsgSize set the maximum size of message that client can receive.
 | 
			
		||||
//
 | 
			
		||||
func MaxRecvMsgSize(s int) client.Option {
 | 
			
		||||
	return func(o *client.Options) {
 | 
			
		||||
		if o.Context == nil {
 | 
			
		||||
@@ -94,9 +83,9 @@ func MaxRecvMsgSize(s int) client.Option {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//
 | 
			
		||||
type maxSendMsgSizeKey struct{}
 | 
			
		||||
 | 
			
		||||
// MaxSendMsgSize set the maximum size of message that client can send.
 | 
			
		||||
//
 | 
			
		||||
func MaxSendMsgSize(s int) client.Option {
 | 
			
		||||
	return func(o *client.Options) {
 | 
			
		||||
		if o.Context == nil {
 | 
			
		||||
@@ -106,11 +95,11 @@ func MaxSendMsgSize(s int) client.Option {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//
 | 
			
		||||
type grpcDialOptions struct{}
 | 
			
		||||
 | 
			
		||||
// DialOptions to be used to configure gRPC dial options
 | 
			
		||||
//
 | 
			
		||||
func DialOptions(opts ...grpc.DialOption) client.CallOption {
 | 
			
		||||
	return func(o *client.CallOptions) {
 | 
			
		||||
func DialOptions(opts ...grpc.DialOption) client.Option {
 | 
			
		||||
	return func(o *client.Options) {
 | 
			
		||||
		if o.Context == nil {
 | 
			
		||||
			o.Context = context.Background()
 | 
			
		||||
		}
 | 
			
		||||
@@ -118,9 +107,9 @@ func DialOptions(opts ...grpc.DialOption) client.CallOption {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//
 | 
			
		||||
type grpcCallOptions struct{}
 | 
			
		||||
 | 
			
		||||
// CallOptions to be used to configure gRPC call options
 | 
			
		||||
//
 | 
			
		||||
func CallOptions(opts ...grpc.CallOption) client.CallOption {
 | 
			
		||||
	return func(o *client.CallOptions) {
 | 
			
		||||
		if o.Context == nil {
 | 
			
		||||
@@ -129,3 +118,14 @@ func CallOptions(opts ...grpc.CallOption) client.CallOption {
 | 
			
		||||
		o.Context = context.WithValue(o.Context, grpcCallOptions{}, opts)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type serviceConfigKey struct{}
 | 
			
		||||
 | 
			
		||||
func ServiceConfig(str string) client.CallOption {
 | 
			
		||||
	return func(options *client.CallOptions) {
 | 
			
		||||
		if options.Context == nil {
 | 
			
		||||
			options.Context = context.Background()
 | 
			
		||||
		}
 | 
			
		||||
		options.Context = context.WithValue(options.Context, serviceConfigKey{}, str)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										21
									
								
								request.go
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								request.go
									
									
									
									
									
								
							@@ -4,17 +4,17 @@ import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/unistack-org/micro/v3/client"
 | 
			
		||||
	"github.com/unistack-org/micro/v3/codec"
 | 
			
		||||
	"go.unistack.org/micro/v4/client"
 | 
			
		||||
	"go.unistack.org/micro/v4/codec"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type grpcRequest struct {
 | 
			
		||||
	request     interface{}
 | 
			
		||||
	codec       codec.Codec
 | 
			
		||||
	service     string
 | 
			
		||||
	method      string
 | 
			
		||||
	contentType string
 | 
			
		||||
	request     interface{}
 | 
			
		||||
	opts        client.RequestOptions
 | 
			
		||||
	codec       codec.Codec
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// service Struct.Method /service.Struct/Method
 | 
			
		||||
@@ -38,15 +38,12 @@ func methodToGRPC(service, method string) string {
 | 
			
		||||
	return fmt.Sprintf("/%s.%s/%s", service, mParts[0], mParts[1])
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newGRPCRequest(service, method string, request interface{}, contentType string, reqOpts ...client.RequestOption) client.Request {
 | 
			
		||||
	var opts client.RequestOptions
 | 
			
		||||
	for _, o := range reqOpts {
 | 
			
		||||
		o(&opts)
 | 
			
		||||
	}
 | 
			
		||||
func newGRPCRequest(service, method string, request interface{}, contentType string, opts ...client.RequestOption) client.Request {
 | 
			
		||||
	options := client.NewRequestOptions(opts...)
 | 
			
		||||
 | 
			
		||||
	// set the content-type specified
 | 
			
		||||
	if len(opts.ContentType) > 0 {
 | 
			
		||||
		contentType = opts.ContentType
 | 
			
		||||
	if len(options.ContentType) > 0 {
 | 
			
		||||
		contentType = options.ContentType
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return &grpcRequest{
 | 
			
		||||
@@ -54,7 +51,7 @@ func newGRPCRequest(service, method string, request interface{}, contentType str
 | 
			
		||||
		method:      method,
 | 
			
		||||
		request:     request,
 | 
			
		||||
		contentType: contentType,
 | 
			
		||||
		opts:        opts,
 | 
			
		||||
		opts:        options,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										23
									
								
								response.go
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								response.go
									
									
									
									
									
								
							@@ -1,15 +1,13 @@
 | 
			
		||||
package grpc
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/unistack-org/micro/v3/codec"
 | 
			
		||||
	"github.com/unistack-org/micro/v3/metadata"
 | 
			
		||||
	"go.unistack.org/micro/v4/codec"
 | 
			
		||||
	"go.unistack.org/micro/v4/metadata"
 | 
			
		||||
	"google.golang.org/grpc"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type response struct {
 | 
			
		||||
	conn   *poolConn
 | 
			
		||||
	conn   *PoolConn
 | 
			
		||||
	stream grpc.ClientStream
 | 
			
		||||
	codec  codec.Codec
 | 
			
		||||
}
 | 
			
		||||
@@ -23,20 +21,21 @@ func (r *response) Codec() codec.Codec {
 | 
			
		||||
func (r *response) Header() metadata.Metadata {
 | 
			
		||||
	meta, err := r.stream.Header()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return metadata.New(0)
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
	md := metadata.New(len(meta))
 | 
			
		||||
	for k, v := range meta {
 | 
			
		||||
		md.Set(k, strings.Join(v, ","))
 | 
			
		||||
	}
 | 
			
		||||
	return md
 | 
			
		||||
 | 
			
		||||
	return metadata.Metadata(meta.Copy())
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Read the undecoded response
 | 
			
		||||
func (r *response) Read() ([]byte, error) {
 | 
			
		||||
	f := &codec.Frame{}
 | 
			
		||||
	if err := r.codec.ReadBody(&wrapStream{r.stream}, f); err != nil {
 | 
			
		||||
	wrap := &wrapStream{r.stream}
 | 
			
		||||
 | 
			
		||||
	_, err := wrap.Read(f.Data)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return f.Data, nil
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										59
									
								
								stream.go
									
									
									
									
									
								
							
							
						
						
									
										59
									
								
								stream.go
									
									
									
									
									
								
							@@ -5,23 +5,22 @@ import (
 | 
			
		||||
	"io"
 | 
			
		||||
	"sync"
 | 
			
		||||
 | 
			
		||||
	"github.com/unistack-org/micro/v3/client"
 | 
			
		||||
	"go.unistack.org/micro/v4/client"
 | 
			
		||||
	"go.unistack.org/micro/v4/tracer"
 | 
			
		||||
	"google.golang.org/grpc"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Implements the streamer interface
 | 
			
		||||
type grpcStream struct {
 | 
			
		||||
	// embed so we can access if need be
 | 
			
		||||
	grpc.ClientStream
 | 
			
		||||
 | 
			
		||||
	sync.RWMutex
 | 
			
		||||
	closed   bool
 | 
			
		||||
	context  context.Context
 | 
			
		||||
	err      error
 | 
			
		||||
	conn     *poolConn
 | 
			
		||||
	request  client.Request
 | 
			
		||||
	response client.Response
 | 
			
		||||
	context  context.Context
 | 
			
		||||
	close    func(err error)
 | 
			
		||||
	conn     *PoolConn
 | 
			
		||||
	sync.RWMutex
 | 
			
		||||
	closed bool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcStream) Context() context.Context {
 | 
			
		||||
@@ -44,6 +43,14 @@ func (g *grpcStream) Send(msg interface{}) error {
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcStream) SendMsg(msg interface{}) error {
 | 
			
		||||
	if err := g.ClientStream.SendMsg(msg); err != nil {
 | 
			
		||||
		g.setError(err)
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcStream) Recv(msg interface{}) (err error) {
 | 
			
		||||
	defer g.setError(err)
 | 
			
		||||
 | 
			
		||||
@@ -62,6 +69,24 @@ func (g *grpcStream) Recv(msg interface{}) (err error) {
 | 
			
		||||
	return
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcStream) RecvMsg(msg interface{}) (err error) {
 | 
			
		||||
	defer g.setError(err)
 | 
			
		||||
 | 
			
		||||
	if err = g.ClientStream.RecvMsg(msg); err != nil {
 | 
			
		||||
		// #202 - inconsistent gRPC stream behavior
 | 
			
		||||
		// the only way to tell if the stream is done is when we get a EOF on the Recv
 | 
			
		||||
		// here we should close the underlying gRPC ClientConn
 | 
			
		||||
		closeErr := g.Close()
 | 
			
		||||
		if err == io.EOF && closeErr != nil {
 | 
			
		||||
			err = closeErr
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcStream) Error() error {
 | 
			
		||||
	g.RLock()
 | 
			
		||||
	defer g.RUnlock()
 | 
			
		||||
@@ -87,6 +112,26 @@ func (g *grpcStream) Close() error {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if sp, ok := tracer.SpanFromContext(g.context); ok && sp != nil {
 | 
			
		||||
		if g.err != nil {
 | 
			
		||||
			sp.SetStatus(tracer.SpanStatusError, g.err.Error())
 | 
			
		||||
		}
 | 
			
		||||
		sp.Finish()
 | 
			
		||||
	}
 | 
			
		||||
	// close the connection
 | 
			
		||||
	g.closed = true
 | 
			
		||||
	g.close(g.err)
 | 
			
		||||
	return g.ClientStream.CloseSend()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (g *grpcStream) CloseSend() error {
 | 
			
		||||
	g.Lock()
 | 
			
		||||
	defer g.Unlock()
 | 
			
		||||
 | 
			
		||||
	if g.closed {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// close the connection
 | 
			
		||||
	g.closed = true
 | 
			
		||||
	g.close(g.err)
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user