gomod: Update containers/psgo to 1.9.0
This solves an issue with a non-existing module: ``` $ go mod tidy [...] github.com/mitchellh/osext@v0.0.0-20151018003038-5e2d6d41470f: invalid version ``` Signed-off-by: Christophe Fergeau <cfergeau@redhat.com>
This commit is contained in:
		
							parent
							
								
									6b592bd4e4
								
							
						
					
					
						commit
						6ca6184ed6
					
				
							
								
								
									
										18
									
								
								go.mod
								
								
								
								
							
							
						
						
									
										18
									
								
								go.mod
								
								
								
								
							|  | @ -17,7 +17,7 @@ require ( | |||
| 	github.com/containers/image/v5 v5.29.2-0.20240130233108-e66a1ade2efc | ||||
| 	github.com/containers/libhvee v0.6.1-0.20240205152934-3a16bce3e4be | ||||
| 	github.com/containers/ocicrypt v1.1.9 | ||||
| 	github.com/containers/psgo v1.8.0 | ||||
| 	github.com/containers/psgo v1.9.0 | ||||
| 	github.com/containers/storage v1.52.1-0.20240202181245-1419a5980565 | ||||
| 	github.com/coreos/go-systemd/v22 v22.5.1-0.20231103132048-7d375ecc2b09 | ||||
| 	github.com/coreos/stream-metadata-go v0.4.4 | ||||
|  | @ -92,7 +92,7 @@ require ( | |||
| 	github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect | ||||
| 	github.com/chenzhuoyu/iasm v0.9.0 // indirect | ||||
| 	github.com/chzyer/readline v1.5.1 // indirect | ||||
| 	github.com/containerd/cgroups/v3 v3.0.2 // indirect | ||||
| 	github.com/containerd/cgroups/v3 v3.0.3 // indirect | ||||
| 	github.com/containerd/containerd v1.7.13 // indirect | ||||
| 	github.com/containerd/log v0.1.0 // indirect | ||||
| 	github.com/containerd/stargz-snapshotter/estargz v0.15.1 // indirect | ||||
|  | @ -150,6 +150,7 @@ require ( | |||
| 	github.com/klauspost/cpuid/v2 v2.2.5 // indirect | ||||
| 	github.com/klauspost/pgzip v1.2.6 // indirect | ||||
| 	github.com/kr/fs v0.1.0 // indirect | ||||
| 	github.com/kr/pretty v0.3.1 // indirect | ||||
| 	github.com/leodido/go-urn v1.2.4 // indirect | ||||
| 	github.com/letsencrypt/boulder v0.0.0-20230907030200-6d76a0f91e1e // indirect | ||||
| 	github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0 // indirect | ||||
|  | @ -171,7 +172,7 @@ require ( | |||
| 	github.com/oklog/ulid v1.3.1 // indirect | ||||
| 	github.com/opentracing/opentracing-go v1.2.0 // indirect | ||||
| 	github.com/ostreedev/ostree-go v0.0.0-20210805093236-719684c64e4f // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.1.0 // indirect | ||||
| 	github.com/pelletier/go-toml/v2 v2.1.1 // indirect | ||||
| 	github.com/pkg/errors v0.9.1 // indirect | ||||
| 	github.com/pkg/sftp v1.13.6 // indirect | ||||
| 	github.com/pmezard/go-difflib v1.0.0 // indirect | ||||
|  | @ -202,9 +203,10 @@ require ( | |||
| 	go.mozilla.org/pkcs7 v0.0.0-20210826202110-33d05740a352 // indirect | ||||
| 	go.opencensus.io v0.24.0 // indirect | ||||
| 	go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.45.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.19.0 // indirect | ||||
| 	go.opentelemetry.io/otel/metric v1.19.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.19.0 // indirect | ||||
| 	go.opentelemetry.io/otel v1.21.0 // indirect | ||||
| 	go.opentelemetry.io/otel/metric v1.21.0 // indirect | ||||
| 	go.opentelemetry.io/otel/sdk v1.21.0 // indirect | ||||
| 	go.opentelemetry.io/otel/trace v1.21.0 // indirect | ||||
| 	golang.org/x/arch v0.5.0 // indirect | ||||
| 	golang.org/x/crypto v0.18.0 // indirect | ||||
| 	golang.org/x/mod v0.14.0 // indirect | ||||
|  | @ -212,8 +214,8 @@ require ( | |||
| 	golang.org/x/time v0.3.0 // indirect | ||||
| 	golang.org/x/tools v0.16.1 // indirect | ||||
| 	google.golang.org/appengine v1.6.8 // indirect | ||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20230920204549-e6e6cdab5c13 // indirect | ||||
| 	google.golang.org/grpc v1.59.0 // indirect | ||||
| 	google.golang.org/genproto/googleapis/rpc v0.0.0-20231212172506-995d672761c0 // indirect | ||||
| 	google.golang.org/grpc v1.60.1 // indirect | ||||
| 	gopkg.in/go-jose/go-jose.v2 v2.6.1 // indirect | ||||
| 	gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 // indirect | ||||
| 	gopkg.in/yaml.v2 v2.4.0 // indirect | ||||
|  |  | |||
|  | @ -1,4 +1,4 @@ | |||
| [](https://godoc.org/github.com/containers/psgo) [](https://travis-ci.org/containers/psgo) | ||||
| [](https://godoc.org/github.com/containers/psgo) | ||||
| 
 | ||||
| # psgo | ||||
| A ps(1) AIX-format compatible golang library extended with various descriptors useful for displaying container-related data. | ||||
|  |  | |||
|  | @ -156,6 +156,9 @@ type Status struct { | |||
| 	// provided only if the kernel was built with the CONFIG_SECCOMP kernel
 | ||||
| 	// configu- ration option enabled.
 | ||||
| 	Seccomp string | ||||
| 	// SeccompFilters: Amount of filters attached to the process.
 | ||||
| 	// (since Linux 5.9)
 | ||||
| 	SeccompFilters string | ||||
| 	// Cpus_allowed:  Mask  of  CPUs  on  which  this process may run
 | ||||
| 	// (since Linux 2.6.24, see cpuset(7)).
 | ||||
| 	CpusAllowed string | ||||
|  | @ -379,6 +382,8 @@ func parseStatus(pid string, lines []string) (*Status, error) { | |||
| 			s.NoNewPrivs = fields[1] | ||||
| 		case "Seccomp:": | ||||
| 			s.Seccomp = fields[1] | ||||
| 		case "Seccomp_filters:": | ||||
| 			s.SeccompFilters = fields[1] | ||||
| 		case "Cpus_allowed:": | ||||
| 			s.CpusAllowed = fields[1] | ||||
| 		case "Cpus_allowed_list:": | ||||
|  |  | |||
|  | @ -3,4 +3,5 @@ fuzz/ | |||
| cmd/tomll/tomll | ||||
| cmd/tomljson/tomljson | ||||
| cmd/tomltestgen/tomltestgen | ||||
| dist | ||||
| dist | ||||
| tests/ | ||||
|  |  | |||
|  | @ -179,12 +179,12 @@ Execution time speedup compared to other Go TOML libraries: | |||
|         <tr><th>Benchmark</th><th>go-toml v1</th><th>BurntSushi/toml</th></tr> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|         <tr><td>Marshal/HugoFrontMatter-2</td><td>1.9x</td><td>1.9x</td></tr> | ||||
|         <tr><td>Marshal/ReferenceFile/map-2</td><td>1.7x</td><td>1.8x</td></tr> | ||||
|         <tr><td>Marshal/ReferenceFile/struct-2</td><td>2.2x</td><td>2.5x</td></tr> | ||||
|         <tr><td>Unmarshal/HugoFrontMatter-2</td><td>2.9x</td><td>2.9x</td></tr> | ||||
|         <tr><td>Unmarshal/ReferenceFile/map-2</td><td>2.6x</td><td>2.9x</td></tr> | ||||
|         <tr><td>Unmarshal/ReferenceFile/struct-2</td><td>4.4x</td><td>5.3x</td></tr> | ||||
|         <tr><td>Marshal/HugoFrontMatter-2</td><td>1.9x</td><td>2.2x</td></tr> | ||||
|         <tr><td>Marshal/ReferenceFile/map-2</td><td>1.7x</td><td>2.1x</td></tr> | ||||
|         <tr><td>Marshal/ReferenceFile/struct-2</td><td>2.2x</td><td>3.0x</td></tr> | ||||
|         <tr><td>Unmarshal/HugoFrontMatter-2</td><td>2.9x</td><td>2.7x</td></tr> | ||||
|         <tr><td>Unmarshal/ReferenceFile/map-2</td><td>2.6x</td><td>2.7x</td></tr> | ||||
|         <tr><td>Unmarshal/ReferenceFile/struct-2</td><td>4.6x</td><td>5.1x</td></tr> | ||||
|      </tbody> | ||||
| </table> | ||||
| <details><summary>See more</summary> | ||||
|  | @ -197,17 +197,17 @@ provided for completeness.</p> | |||
|         <tr><th>Benchmark</th><th>go-toml v1</th><th>BurntSushi/toml</th></tr> | ||||
|     </thead> | ||||
|     <tbody> | ||||
|         <tr><td>Marshal/SimpleDocument/map-2</td><td>1.8x</td><td>2.9x</td></tr> | ||||
|         <tr><td>Marshal/SimpleDocument/struct-2</td><td>2.7x</td><td>4.2x</td></tr> | ||||
|         <tr><td>Unmarshal/SimpleDocument/map-2</td><td>4.5x</td><td>3.1x</td></tr> | ||||
|         <tr><td>Unmarshal/SimpleDocument/struct-2</td><td>6.2x</td><td>3.9x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/example-2</td><td>3.1x</td><td>3.5x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/code-2</td><td>2.3x</td><td>3.1x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/twitter-2</td><td>2.5x</td><td>2.6x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/citm_catalog-2</td><td>2.1x</td><td>2.2x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/canada-2</td><td>1.6x</td><td>1.3x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/config-2</td><td>4.3x</td><td>3.2x</td></tr> | ||||
|         <tr><td>[Geo mean]</td><td>2.7x</td><td>2.8x</td></tr> | ||||
|         <tr><td>Marshal/SimpleDocument/map-2</td><td>1.8x</td><td>2.7x</td></tr> | ||||
|         <tr><td>Marshal/SimpleDocument/struct-2</td><td>2.7x</td><td>3.8x</td></tr> | ||||
|         <tr><td>Unmarshal/SimpleDocument/map-2</td><td>3.8x</td><td>3.0x</td></tr> | ||||
|         <tr><td>Unmarshal/SimpleDocument/struct-2</td><td>5.6x</td><td>4.1x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/example-2</td><td>3.0x</td><td>3.2x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/code-2</td><td>2.3x</td><td>2.9x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/twitter-2</td><td>2.6x</td><td>2.7x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/citm_catalog-2</td><td>2.2x</td><td>2.3x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/canada-2</td><td>1.8x</td><td>1.5x</td></tr> | ||||
|         <tr><td>UnmarshalDataset/config-2</td><td>4.1x</td><td>2.9x</td></tr> | ||||
|         <tr><td>geomean</td><td>2.7x</td><td>2.8x</td></tr> | ||||
|      </tbody> | ||||
| </table> | ||||
| <p>This table can be generated with <code>./ci.sh benchmark -a -html</code>.</p> | ||||
|  |  | |||
|  | @ -2,9 +2,6 @@ | |||
| 
 | ||||
| ## Supported Versions | ||||
| 
 | ||||
| Use this section to tell people about which versions of your project are | ||||
| currently being supported with security updates. | ||||
| 
 | ||||
| | Version    | Supported          | | ||||
| | ---------- | ------------------ | | ||||
| | Latest 2.x | :white_check_mark: | | ||||
|  |  | |||
|  | @ -77,7 +77,7 @@ cover() { | |||
| 
 | ||||
|     pushd "$dir" | ||||
|     go test -covermode=atomic  -coverpkg=./... -coverprofile=coverage.out.tmp ./... | ||||
|     cat coverage.out.tmp | grep -v fuzz | grep -v testsuite | grep -v tomltestgen | grep -v gotoml-test-decoder > coverage.out | ||||
|     grep -Ev '(fuzz|testsuite|tomltestgen|gotoml-test-decoder|gotoml-test-encoder)' coverage.out.tmp > coverage.out | ||||
|     go tool cover -func=coverage.out | ||||
|     echo "Coverage profile for ${branch}: ${dir}/coverage.out" >&2 | ||||
|     popd | ||||
|  | @ -152,7 +152,7 @@ bench() { | |||
|     fi | ||||
| 
 | ||||
|     export GOMAXPROCS=2 | ||||
|     nice -n -19 taskset --cpu-list 0,1 go test '-bench=^Benchmark(Un)?[mM]arshal' -count=5 -run=Nothing ./... | tee "${out}" | ||||
|     go test '-bench=^Benchmark(Un)?[mM]arshal' -count=10 -run=Nothing ./... | tee "${out}" | ||||
|     popd | ||||
| 
 | ||||
|     if [ "${branch}" != "HEAD" ]; then | ||||
|  | @ -161,10 +161,12 @@ bench() { | |||
| } | ||||
| 
 | ||||
| fmktemp() { | ||||
|     if mktemp --version|grep GNU >/dev/null; then | ||||
|         mktemp --suffix=-$1; | ||||
|     if mktemp --version &> /dev/null; then | ||||
| 	# GNU | ||||
|         mktemp --suffix=-$1 | ||||
|     else | ||||
|         mktemp -t $1; | ||||
| 	# BSD | ||||
| 	mktemp -t $1 | ||||
|     fi | ||||
| } | ||||
| 
 | ||||
|  | @ -184,12 +186,14 @@ with open(sys.argv[1]) as f: | |||
|             lines.append(line.split(',')) | ||||
| 
 | ||||
| results = [] | ||||
| for line in reversed(lines[1:]): | ||||
| for line in reversed(lines[2:]): | ||||
|     if len(line) < 8 or line[0] == "": | ||||
|         continue | ||||
|     v2 = float(line[1]) | ||||
|     results.append([ | ||||
|         line[0].replace("-32", ""), | ||||
|         "%.1fx" % (float(line[3])/v2),  # v1 | ||||
|         "%.1fx" % (float(line[5])/v2),  # bs | ||||
|         "%.1fx" % (float(line[7])/v2),  # bs | ||||
|     ]) | ||||
| # move geomean to the end | ||||
| results.append(results[0]) | ||||
|  | @ -260,10 +264,10 @@ benchmark() { | |||
| 
 | ||||
|         if [ "$1" = "-html" ]; then | ||||
|             tmpcsv=`fmktemp csv` | ||||
|             benchstat -csv -geomean go-toml-v2.txt go-toml-v1.txt bs-toml.txt > $tmpcsv | ||||
|             benchstat -format csv go-toml-v2.txt go-toml-v1.txt bs-toml.txt > $tmpcsv | ||||
|             benchstathtml $tmpcsv | ||||
|         else | ||||
|             benchstat -geomean go-toml-v2.txt go-toml-v1.txt bs-toml.txt | ||||
|             benchstat go-toml-v2.txt go-toml-v1.txt bs-toml.txt | ||||
|         fi | ||||
| 
 | ||||
|         rm -f go-toml-v2.txt go-toml-v1.txt bs-toml.txt | ||||
|  |  | |||
|  | @ -1097,9 +1097,9 @@ func (d *decoder) handleKeyValuePart(key unstable.Iterator, value *unstable.Node | |||
| 
 | ||||
| 		f := fieldByIndex(v, path) | ||||
| 
 | ||||
| 		if !f.CanSet() { | ||||
| 			// If the field is not settable, need to take a slower path and make a copy of
 | ||||
| 			// the struct itself to a new location.
 | ||||
| 		if !f.CanAddr() { | ||||
| 			// If the field is not addressable, need to take a slower path and
 | ||||
| 			// make a copy of the struct itself to a new location.
 | ||||
| 			nvp := reflect.New(v.Type()) | ||||
| 			nvp.Elem().Set(v) | ||||
| 			v = nvp.Elem() | ||||
|  |  | |||
|  | @ -14,12 +14,9 @@ go.work.sum | |||
| gen/ | ||||
| 
 | ||||
| /example/dice/dice | ||||
| /example/fib/fib | ||||
| /example/fib/traces.txt | ||||
| /example/jaeger/jaeger | ||||
| /example/namedtracer/namedtracer | ||||
| /example/otel-collector/otel-collector | ||||
| /example/opencensus/opencensus | ||||
| /example/passthrough/passthrough | ||||
| /example/prometheus/prometheus | ||||
| /example/zipkin/zipkin | ||||
| /example/otel-collector/otel-collector | ||||
|  |  | |||
|  | @ -12,8 +12,9 @@ linters: | |||
|     - depguard | ||||
|     - errcheck | ||||
|     - godot | ||||
|     - gofmt | ||||
|     - gofumpt | ||||
|     - goimports | ||||
|     - gosec | ||||
|     - gosimple | ||||
|     - govet | ||||
|     - ineffassign | ||||
|  | @ -53,6 +54,20 @@ issues: | |||
|       text: "calls to (.+) only in main[(][)] or init[(][)] functions" | ||||
|       linters: | ||||
|         - revive | ||||
|     # It's okay to not run gosec in a test. | ||||
|     - path: _test\.go | ||||
|       linters: | ||||
|         - gosec | ||||
|     # Igonoring gosec G404: Use of weak random number generator (math/rand instead of crypto/rand) | ||||
|     # as we commonly use it in tests and examples. | ||||
|     - text: "G404:" | ||||
|       linters: | ||||
|         - gosec | ||||
|     # Igonoring gosec G402: TLS MinVersion too low | ||||
|     # as the https://pkg.go.dev/crypto/tls#Config handles MinVersion default well. | ||||
|     - text: "G402: TLS MinVersion too low." | ||||
|       linters: | ||||
|         - gosec | ||||
|   include: | ||||
|     # revive exported should have comment or be unexported. | ||||
|     - EXC0012 | ||||
|  |  | |||
|  | @ -8,6 +8,85 @@ This project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm | |||
| 
 | ||||
| ## [Unreleased] | ||||
| 
 | ||||
| ## [1.21.0/0.44.0] 2023-11-16 | ||||
| 
 | ||||
| ### Removed | ||||
| 
 | ||||
| - Remove the deprecated `go.opentelemetry.io/otel/bridge/opencensus.NewTracer`. (#4706) | ||||
| - Remove the deprecated `go.opentelemetry.io/otel/exporters/otlp/otlpmetric` module. (#4707) | ||||
| - Remove the deprecated `go.opentelemetry.io/otel/example/view` module. (#4708) | ||||
| - Remove the deprecated `go.opentelemetry.io/otel/example/fib` module. (#4723) | ||||
| 
 | ||||
| ### Fixed | ||||
| 
 | ||||
| - Do not parse non-protobuf responses in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp`. (#4719) | ||||
| - Do not parse non-protobuf responses in `go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp`. (#4719) | ||||
| 
 | ||||
| ## [1.20.0/0.43.0] 2023-11-10 | ||||
| 
 | ||||
| This release brings a breaking change for custom trace API implementations. Some interfaces (`TracerProvider`, `Tracer`, `Span`) now embed the `go.opentelemetry.io/otel/trace/embedded` types. Implementors need to update their implementations based on what they want the default behavior to be. See the "API Implementations" section of the [trace API] package documentation for more information about how to accomplish this. | ||||
| 
 | ||||
| ### Added | ||||
| 
 | ||||
| - Add `go.opentelemetry.io/otel/bridge/opencensus.InstallTraceBridge`, which installs the OpenCensus trace bridge, and replaces `opencensus.NewTracer`. (#4567) | ||||
| - Add scope version to trace and metric bridges in `go.opentelemetry.io/otel/bridge/opencensus`. (#4584) | ||||
| - Add the `go.opentelemetry.io/otel/trace/embedded` package to be embedded in the exported trace API interfaces. (#4620) | ||||
| - Add the `go.opentelemetry.io/otel/trace/noop` package as a default no-op implementation of the trace API. (#4620) | ||||
| - Add context propagation in `go.opentelemetry.io/otel/example/dice`. (#4644) | ||||
| - Add view configuration to `go.opentelemetry.io/otel/example/prometheus`. (#4649) | ||||
| - Add `go.opentelemetry.io/otel/metric.WithExplicitBucketBoundaries`, which allows defining default explicit bucket boundaries when creating histogram instruments. (#4603) | ||||
| - Add `Version` function in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc`. (#4660) | ||||
| - Add `Version` function in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp`. (#4660) | ||||
| - Add Summary, SummaryDataPoint, and QuantileValue to `go.opentelemetry.io/sdk/metric/metricdata`. (#4622) | ||||
| - `go.opentelemetry.io/otel/bridge/opencensus.NewMetricProducer` now supports exemplars from OpenCensus. (#4585) | ||||
| - Add support for `WithExplicitBucketBoundaries` in `go.opentelemetry.io/otel/sdk/metric`. (#4605) | ||||
| - Add support for Summary metrics in `go.opentelemetry.io/otel/bridge/opencensus`. (#4668) | ||||
| 
 | ||||
| ### Deprecated | ||||
| 
 | ||||
| - Deprecate `go.opentelemetry.io/otel/bridge/opencensus.NewTracer` in favor of `opencensus.InstallTraceBridge`. (#4567) | ||||
| - Deprecate `go.opentelemetry.io/otel/example/fib` package is in favor of `go.opentelemetry.io/otel/example/dice`. (#4618) | ||||
| - Deprecate `go.opentelemetry.io/otel/trace.NewNoopTracerProvider`. | ||||
|   Use the added `NewTracerProvider` function in `go.opentelemetry.io/otel/trace/noop` instead. (#4620) | ||||
| - Deprecate `go.opentelemetry.io/otel/example/view` package in favor of `go.opentelemetry.io/otel/example/prometheus`. (#4649) | ||||
| - Deprecate `go.opentelemetry.io/otel/exporters/otlp/otlpmetric`. (#4693) | ||||
| 
 | ||||
| ### Changed | ||||
| 
 | ||||
| - `go.opentelemetry.io/otel/bridge/opencensus.NewMetricProducer` returns a `*MetricProducer` struct instead of the metric.Producer interface. (#4583) | ||||
| - The `TracerProvider` in `go.opentelemetry.io/otel/trace` now embeds the `go.opentelemetry.io/otel/trace/embedded.TracerProvider` type. | ||||
|   This extends the `TracerProvider` interface and is is a breaking change for any existing implementation. | ||||
|   Implementors need to update their implementations based on what they want the default behavior of the interface to be. | ||||
|   See the "API Implementations" section of the `go.opentelemetry.io/otel/trace` package documentation for more information about how to accomplish this. (#4620) | ||||
| - The `Tracer` in `go.opentelemetry.io/otel/trace` now embeds the `go.opentelemetry.io/otel/trace/embedded.Tracer` type. | ||||
|   This extends the `Tracer` interface and is is a breaking change for any existing implementation. | ||||
|   Implementors need to update their implementations based on what they want the default behavior of the interface to be. | ||||
|   See the "API Implementations" section of the `go.opentelemetry.io/otel/trace` package documentation for more information about how to accomplish this. (#4620) | ||||
| - The `Span` in `go.opentelemetry.io/otel/trace` now embeds the `go.opentelemetry.io/otel/trace/embedded.Span` type. | ||||
|   This extends the `Span` interface and is is a breaking change for any existing implementation. | ||||
|   Implementors need to update their implementations based on what they want the default behavior of the interface to be. | ||||
|   See the "API Implementations" section of the `go.opentelemetry.io/otel/trace` package documentation for more information about how to accomplish this. (#4620) | ||||
| - `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc` does no longer depend on `go.opentelemetry.io/otel/exporters/otlp/otlpmetric`. (#4660) | ||||
| - `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp` does no longer depend on `go.opentelemetry.io/otel/exporters/otlp/otlpmetric`. (#4660) | ||||
| - Retry for `502 Bad Gateway` and `504 Gateway Timeout` HTTP statuses in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp`. (#4670) | ||||
| - Retry for `502 Bad Gateway` and `504 Gateway Timeout` HTTP statuses in `go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp`. (#4670) | ||||
| - Retry for `RESOURCE_EXHAUSTED` only if RetryInfo is returned in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc`. (#4669) | ||||
| - Retry for `RESOURCE_EXHAUSTED` only if RetryInfo is returned in `go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc`. (#4669) | ||||
| - Retry temporary HTTP request failures in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp`. (#4679) | ||||
| - Retry temporary HTTP request failures in `go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp`. (#4679) | ||||
| 
 | ||||
| ### Fixed | ||||
| 
 | ||||
| - Fix improper parsing of characters such us `+`, `/` by `Parse` in `go.opentelemetry.io/otel/baggage` as they were rendered as a whitespace. (#4667) | ||||
| - Fix improper parsing of characters such us `+`, `/` passed via `OTEL_RESOURCE_ATTRIBUTES` in `go.opentelemetry.io/otel/sdk/resource` as they were rendered as a whitespace. (#4699) | ||||
| - Fix improper parsing of characters such us `+`, `/` passed via `OTEL_EXPORTER_OTLP_HEADERS` and `OTEL_EXPORTER_OTLP_METRICS_HEADERS` in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc` as they were rendered as a whitespace. (#4699) | ||||
| - Fix improper parsing of characters such us `+`, `/` passed via `OTEL_EXPORTER_OTLP_HEADERS` and `OTEL_EXPORTER_OTLP_METRICS_HEADERS` in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp` as they were rendered as a whitespace. (#4699) | ||||
| - Fix improper parsing of characters such us `+`, `/` passed via `OTEL_EXPORTER_OTLP_HEADERS` and `OTEL_EXPORTER_OTLP_TRACES_HEADERS` in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlptracegrpc` as they were rendered as a whitespace. (#4699) | ||||
| - Fix improper parsing of characters such us `+`, `/` passed via `OTEL_EXPORTER_OTLP_HEADERS` and `OTEL_EXPORTER_OTLP_TRACES_HEADERS` in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlptracehttp` as they were rendered as a whitespace. (#4699) | ||||
| - In `go.opentelemetry.op/otel/exporters/prometheus`, the exporter no longer `Collect`s metrics after `Shutdown` is invoked. (#4648) | ||||
| - Fix documentation for `WithCompressor` in `go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc`. (#4695) | ||||
| - Fix documentation for `WithCompressor` in `go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc`. (#4695) | ||||
| 
 | ||||
| ## [1.19.0/0.42.0/0.0.7] 2023-09-28 | ||||
| 
 | ||||
| This release contains the first stable release of the OpenTelemetry Go [metric SDK]. | ||||
|  | @ -2656,7 +2735,9 @@ It contains api and sdk for trace and meter. | |||
| - CircleCI build CI manifest files. | ||||
| - CODEOWNERS file to track owners of this project. | ||||
| 
 | ||||
| [Unreleased]: https://github.com/open-telemetry/opentelemetry-go/compare/v1.19.0...HEAD | ||||
| [Unreleased]: https://github.com/open-telemetry/opentelemetry-go/compare/v1.21.0...HEAD | ||||
| [1.21.0/0.44.0]: https://github.com/open-telemetry/opentelemetry-go/releases/tag/v1.21.0 | ||||
| [1.20.0/0.43.0]: https://github.com/open-telemetry/opentelemetry-go/releases/tag/v1.20.0 | ||||
| [1.19.0/0.42.0/0.0.7]: https://github.com/open-telemetry/opentelemetry-go/releases/tag/v1.19.0 | ||||
| [1.19.0-rc.1/0.42.0-rc.1]: https://github.com/open-telemetry/opentelemetry-go/releases/tag/v1.19.0-rc.1 | ||||
| [1.18.0/0.41.0/0.0.6]: https://github.com/open-telemetry/opentelemetry-go/releases/tag/v1.18.0 | ||||
|  | @ -2731,7 +2812,7 @@ It contains api and sdk for trace and meter. | |||
| [Go 1.20]: https://go.dev/doc/go1.20 | ||||
| [Go 1.19]: https://go.dev/doc/go1.19 | ||||
| [Go 1.18]: https://go.dev/doc/go1.18 | ||||
| [Go 1.19]: https://go.dev/doc/go1.19 | ||||
| 
 | ||||
| [metric API]:https://pkg.go.dev/go.opentelemetry.io/otel/metric | ||||
| [metric SDK]:https://pkg.go.dev/go.opentelemetry.io/otel/sdk/metric | ||||
| [trace API]:https://pkg.go.dev/go.opentelemetry.io/otel/trace | ||||
|  |  | |||
|  | @ -90,6 +90,10 @@ git push <YOUR_FORK> <YOUR_BRANCH_NAME> | |||
| Open a pull request against the main `opentelemetry-go` repo. Be sure to add the pull | ||||
| request ID to the entry you added to `CHANGELOG.md`. | ||||
| 
 | ||||
| Avoid rebasing and force-pushing to your branch to facilitate reviewing the pull request. | ||||
| Rewriting Git history makes it difficult to keep track of iterations during code review. | ||||
| All pull requests are squashed to a single commit upon merge to `main`. | ||||
| 
 | ||||
| ### How to Receive Comments | ||||
| 
 | ||||
| * If the PR is not ready for review, please put `[WIP]` in the title, | ||||
|  |  | |||
|  | @ -77,6 +77,9 @@ $(GOTMPL): PACKAGE=go.opentelemetry.io/build-tools/gotmpl | |||
| GORELEASE = $(TOOLS)/gorelease | ||||
| $(GORELEASE): PACKAGE=golang.org/x/exp/cmd/gorelease | ||||
| 
 | ||||
| GOVULNCHECK = $(TOOLS)/govulncheck | ||||
| $(TOOLS)/govulncheck: PACKAGE=golang.org/x/vuln/cmd/govulncheck | ||||
| 
 | ||||
| .PHONY: tools | ||||
| tools: $(CROSSLINK) $(DBOTCONF) $(GOLANGCI_LINT) $(MISSPELL) $(GOCOVMERGE) $(STRINGER) $(PORTO) $(GOJQ) $(SEMCONVGEN) $(MULTIMOD) $(SEMCONVKIT) $(GOTMPL) $(GORELEASE) | ||||
| 
 | ||||
|  | @ -189,6 +192,18 @@ test-coverage: | $(GOCOVMERGE) | |||
| 	done; \
 | ||||
| 	$(GOCOVMERGE) $$(find . -name coverage.out) > coverage.txt | ||||
| 
 | ||||
| # Adding a directory will include all benchmarks in that direcotry if a filter is not specified.
 | ||||
| BENCHMARK_TARGETS := sdk/trace | ||||
| .PHONY: benchmark | ||||
| benchmark: $(BENCHMARK_TARGETS:%=benchmark/%) | ||||
| BENCHMARK_FILTER = . | ||||
| # You can override the filter for a particular directory by adding a rule here.
 | ||||
| benchmark/sdk/trace: BENCHMARK_FILTER = SpanWithAttributes_8/AlwaysSample | ||||
| benchmark/%: | ||||
| 	@echo "$(GO) test -timeout $(TIMEOUT)s -run=xxxxxMatchNothingxxxxx -bench=$(BENCHMARK_FILTER) $*..." \
 | ||||
| 		&& cd $* \
 | ||||
| 		$(foreach filter, $(BENCHMARK_FILTER), && $(GO) test -timeout $(TIMEOUT)s -run=xxxxxMatchNothingxxxxx -bench=$(filter)) | ||||
| 
 | ||||
| .PHONY: golangci-lint golangci-lint-fix | ||||
| golangci-lint-fix: ARGS=--fix | ||||
| golangci-lint-fix: golangci-lint | ||||
|  | @ -216,7 +231,7 @@ go-mod-tidy/%: | crosslink | |||
| lint-modules: go-mod-tidy | ||||
| 
 | ||||
| .PHONY: lint | ||||
| lint: misspell lint-modules golangci-lint | ||||
| lint: misspell lint-modules golangci-lint govulncheck | ||||
| 
 | ||||
| .PHONY: vanity-import-check | ||||
| vanity-import-check: | $(PORTO) | ||||
|  | @ -226,6 +241,14 @@ vanity-import-check: | $(PORTO) | |||
| misspell: | $(MISSPELL) | ||||
| 	@$(MISSPELL) -w $(ALL_DOCS) | ||||
| 
 | ||||
| .PHONY: govulncheck | ||||
| govulncheck: $(OTEL_GO_MOD_DIRS:%=govulncheck/%) | ||||
| govulncheck/%: DIR=$* | ||||
| govulncheck/%: | $(GOVULNCHECK) | ||||
| 	@echo "govulncheck ./... in $(DIR)" \
 | ||||
| 		&& cd $(DIR) \
 | ||||
| 		&& $(GOVULNCHECK) ./... | ||||
| 
 | ||||
| .PHONY: codespell | ||||
| codespell: | $(CODESPELL) | ||||
| 	@$(DOCKERPY) $(CODESPELL) | ||||
|  | @ -289,3 +312,7 @@ COMMIT ?= "HEAD" | |||
| add-tags: | $(MULTIMOD) | ||||
| 	@[ "${MODSET}" ] || ( echo ">> env var MODSET is not set"; exit 1 ) | ||||
| 	$(MULTIMOD) verify && $(MULTIMOD) tag -m ${MODSET} -c ${COMMIT} | ||||
| 
 | ||||
| .PHONY: lint-markdown | ||||
| lint-markdown:  | ||||
| 	docker run -v "$(CURDIR):$(WORKDIR)" docker://avtodev/markdown-lint:v1 -c $(WORKDIR)/.markdownlint.yaml $(WORKDIR)/**/*.md | ||||
|  |  | |||
|  | @ -11,16 +11,13 @@ It provides a set of APIs to directly measure performance and behavior of your s | |||
| 
 | ||||
| ## Project Status | ||||
| 
 | ||||
| | Signal  | Status     | Project               | | ||||
| |---------|------------|-----------------------| | ||||
| | Traces  | Stable     | N/A                   | | ||||
| | Metrics | Mixed [1]  | [Go: Metric SDK (GA)] | | ||||
| | Logs    | Frozen [2] | N/A                   | | ||||
| | Signal  | Status     | | ||||
| |---------|------------| | ||||
| | Traces  | Stable     | | ||||
| | Metrics | Stable     | | ||||
| | Logs    | Design [1] | | ||||
| 
 | ||||
| [Go: Metric SDK (GA)]: https://github.com/orgs/open-telemetry/projects/34 | ||||
| 
 | ||||
| - [1]: [Metrics API](https://pkg.go.dev/go.opentelemetry.io/otel/metric) is Stable. [Metrics SDK](https://pkg.go.dev/go.opentelemetry.io/otel/sdk/metric) is Beta. | ||||
| - [2]: The Logs signal development is halted for this project while we stabilize the Metrics SDK. | ||||
| - [1]: Currently the logs signal development is in a design phase ([#4696](https://github.com/open-telemetry/opentelemetry-go/issues/4696)). | ||||
|    No Logs Pull Requests are currently being accepted. | ||||
| 
 | ||||
| Progress and status specific to this repository is tracked in our | ||||
|  |  | |||
|  | @ -254,7 +254,7 @@ func NewMember(key, value string, props ...Property) (Member, error) { | |||
| 	if err := m.validate(); err != nil { | ||||
| 		return newInvalidMember(), err | ||||
| 	} | ||||
| 	decodedValue, err := url.QueryUnescape(value) | ||||
| 	decodedValue, err := url.PathUnescape(value) | ||||
| 	if err != nil { | ||||
| 		return newInvalidMember(), fmt.Errorf("%w: %q", errInvalidValue, value) | ||||
| 	} | ||||
|  | @ -301,7 +301,7 @@ func parseMember(member string) (Member, error) { | |||
| 	// when converting the header into a data structure."
 | ||||
| 	key = strings.TrimSpace(k) | ||||
| 	var err error | ||||
| 	value, err = url.QueryUnescape(strings.TrimSpace(v)) | ||||
| 	value, err = url.PathUnescape(strings.TrimSpace(v)) | ||||
| 	if err != nil { | ||||
| 		return newInvalidMember(), fmt.Errorf("%w: %q", err, value) | ||||
| 	} | ||||
|  |  | |||
|  | @ -34,11 +34,13 @@ type afCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Float64ObservableCounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Float64ObservableCounter
 | ||||
| 	delegate atomic.Value // metric.Float64ObservableCounter
 | ||||
| } | ||||
| 
 | ||||
| var _ unwrapper = (*afCounter)(nil) | ||||
| var _ metric.Float64ObservableCounter = (*afCounter)(nil) | ||||
| var ( | ||||
| 	_ unwrapper                       = (*afCounter)(nil) | ||||
| 	_ metric.Float64ObservableCounter = (*afCounter)(nil) | ||||
| ) | ||||
| 
 | ||||
| func (i *afCounter) setDelegate(m metric.Meter) { | ||||
| 	ctr, err := m.Float64ObservableCounter(i.name, i.opts...) | ||||
|  | @ -63,11 +65,13 @@ type afUpDownCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Float64ObservableUpDownCounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Float64ObservableUpDownCounter
 | ||||
| 	delegate atomic.Value // metric.Float64ObservableUpDownCounter
 | ||||
| } | ||||
| 
 | ||||
| var _ unwrapper = (*afUpDownCounter)(nil) | ||||
| var _ metric.Float64ObservableUpDownCounter = (*afUpDownCounter)(nil) | ||||
| var ( | ||||
| 	_ unwrapper                             = (*afUpDownCounter)(nil) | ||||
| 	_ metric.Float64ObservableUpDownCounter = (*afUpDownCounter)(nil) | ||||
| ) | ||||
| 
 | ||||
| func (i *afUpDownCounter) setDelegate(m metric.Meter) { | ||||
| 	ctr, err := m.Float64ObservableUpDownCounter(i.name, i.opts...) | ||||
|  | @ -92,11 +96,13 @@ type afGauge struct { | |||
| 	name string | ||||
| 	opts []metric.Float64ObservableGaugeOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Float64ObservableGauge
 | ||||
| 	delegate atomic.Value // metric.Float64ObservableGauge
 | ||||
| } | ||||
| 
 | ||||
| var _ unwrapper = (*afGauge)(nil) | ||||
| var _ metric.Float64ObservableGauge = (*afGauge)(nil) | ||||
| var ( | ||||
| 	_ unwrapper                     = (*afGauge)(nil) | ||||
| 	_ metric.Float64ObservableGauge = (*afGauge)(nil) | ||||
| ) | ||||
| 
 | ||||
| func (i *afGauge) setDelegate(m metric.Meter) { | ||||
| 	ctr, err := m.Float64ObservableGauge(i.name, i.opts...) | ||||
|  | @ -121,11 +127,13 @@ type aiCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Int64ObservableCounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Int64ObservableCounter
 | ||||
| 	delegate atomic.Value // metric.Int64ObservableCounter
 | ||||
| } | ||||
| 
 | ||||
| var _ unwrapper = (*aiCounter)(nil) | ||||
| var _ metric.Int64ObservableCounter = (*aiCounter)(nil) | ||||
| var ( | ||||
| 	_ unwrapper                     = (*aiCounter)(nil) | ||||
| 	_ metric.Int64ObservableCounter = (*aiCounter)(nil) | ||||
| ) | ||||
| 
 | ||||
| func (i *aiCounter) setDelegate(m metric.Meter) { | ||||
| 	ctr, err := m.Int64ObservableCounter(i.name, i.opts...) | ||||
|  | @ -150,11 +158,13 @@ type aiUpDownCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Int64ObservableUpDownCounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Int64ObservableUpDownCounter
 | ||||
| 	delegate atomic.Value // metric.Int64ObservableUpDownCounter
 | ||||
| } | ||||
| 
 | ||||
| var _ unwrapper = (*aiUpDownCounter)(nil) | ||||
| var _ metric.Int64ObservableUpDownCounter = (*aiUpDownCounter)(nil) | ||||
| var ( | ||||
| 	_ unwrapper                           = (*aiUpDownCounter)(nil) | ||||
| 	_ metric.Int64ObservableUpDownCounter = (*aiUpDownCounter)(nil) | ||||
| ) | ||||
| 
 | ||||
| func (i *aiUpDownCounter) setDelegate(m metric.Meter) { | ||||
| 	ctr, err := m.Int64ObservableUpDownCounter(i.name, i.opts...) | ||||
|  | @ -179,11 +189,13 @@ type aiGauge struct { | |||
| 	name string | ||||
| 	opts []metric.Int64ObservableGaugeOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Int64ObservableGauge
 | ||||
| 	delegate atomic.Value // metric.Int64ObservableGauge
 | ||||
| } | ||||
| 
 | ||||
| var _ unwrapper = (*aiGauge)(nil) | ||||
| var _ metric.Int64ObservableGauge = (*aiGauge)(nil) | ||||
| var ( | ||||
| 	_ unwrapper                   = (*aiGauge)(nil) | ||||
| 	_ metric.Int64ObservableGauge = (*aiGauge)(nil) | ||||
| ) | ||||
| 
 | ||||
| func (i *aiGauge) setDelegate(m metric.Meter) { | ||||
| 	ctr, err := m.Int64ObservableGauge(i.name, i.opts...) | ||||
|  | @ -208,7 +220,7 @@ type sfCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Float64CounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Float64Counter
 | ||||
| 	delegate atomic.Value // metric.Float64Counter
 | ||||
| } | ||||
| 
 | ||||
| var _ metric.Float64Counter = (*sfCounter)(nil) | ||||
|  | @ -234,7 +246,7 @@ type sfUpDownCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Float64UpDownCounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Float64UpDownCounter
 | ||||
| 	delegate atomic.Value // metric.Float64UpDownCounter
 | ||||
| } | ||||
| 
 | ||||
| var _ metric.Float64UpDownCounter = (*sfUpDownCounter)(nil) | ||||
|  | @ -260,7 +272,7 @@ type sfHistogram struct { | |||
| 	name string | ||||
| 	opts []metric.Float64HistogramOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Float64Histogram
 | ||||
| 	delegate atomic.Value // metric.Float64Histogram
 | ||||
| } | ||||
| 
 | ||||
| var _ metric.Float64Histogram = (*sfHistogram)(nil) | ||||
|  | @ -286,7 +298,7 @@ type siCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Int64CounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Int64Counter
 | ||||
| 	delegate atomic.Value // metric.Int64Counter
 | ||||
| } | ||||
| 
 | ||||
| var _ metric.Int64Counter = (*siCounter)(nil) | ||||
|  | @ -312,7 +324,7 @@ type siUpDownCounter struct { | |||
| 	name string | ||||
| 	opts []metric.Int64UpDownCounterOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Int64UpDownCounter
 | ||||
| 	delegate atomic.Value // metric.Int64UpDownCounter
 | ||||
| } | ||||
| 
 | ||||
| var _ metric.Int64UpDownCounter = (*siUpDownCounter)(nil) | ||||
|  | @ -338,7 +350,7 @@ type siHistogram struct { | |||
| 	name string | ||||
| 	opts []metric.Int64HistogramOption | ||||
| 
 | ||||
| 	delegate atomic.Value //metric.Int64Histogram
 | ||||
| 	delegate atomic.Value // metric.Int64Histogram
 | ||||
| } | ||||
| 
 | ||||
| var _ metric.Int64Histogram = (*siHistogram)(nil) | ||||
|  |  | |||
|  | @ -39,6 +39,7 @@ import ( | |||
| 	"go.opentelemetry.io/otel/attribute" | ||||
| 	"go.opentelemetry.io/otel/codes" | ||||
| 	"go.opentelemetry.io/otel/trace" | ||||
| 	"go.opentelemetry.io/otel/trace/embedded" | ||||
| ) | ||||
| 
 | ||||
| // tracerProvider is a placeholder for a configured SDK TracerProvider.
 | ||||
|  | @ -46,6 +47,8 @@ import ( | |||
| // All TracerProvider functionality is forwarded to a delegate once
 | ||||
| // configured.
 | ||||
| type tracerProvider struct { | ||||
| 	embedded.TracerProvider | ||||
| 
 | ||||
| 	mtx      sync.Mutex | ||||
| 	tracers  map[il]*tracer | ||||
| 	delegate trace.TracerProvider | ||||
|  | @ -119,6 +122,8 @@ type il struct { | |||
| // All Tracer functionality is forwarded to a delegate once configured.
 | ||||
| // Otherwise, all functionality is forwarded to a NoopTracer.
 | ||||
| type tracer struct { | ||||
| 	embedded.Tracer | ||||
| 
 | ||||
| 	name     string | ||||
| 	opts     []trace.TracerOption | ||||
| 	provider *tracerProvider | ||||
|  | @ -156,6 +161,8 @@ func (t *tracer) Start(ctx context.Context, name string, opts ...trace.SpanStart | |||
| // SpanContext. It performs no operations other than to return the wrapped
 | ||||
| // SpanContext.
 | ||||
| type nonRecordingSpan struct { | ||||
| 	embedded.Span | ||||
| 
 | ||||
| 	sc     trace.SpanContext | ||||
| 	tracer *tracer | ||||
| } | ||||
|  |  | |||
|  | @ -149,7 +149,7 @@ of [go.opentelemetry.io/otel/metric]. | |||
| 
 | ||||
| Finally, an author can embed another implementation in theirs. The embedded | ||||
| implementation will be used for methods not defined by the author. For example, | ||||
| an author who want to default to silently dropping the call can use | ||||
| an author who wants to default to silently dropping the call can use | ||||
| [go.opentelemetry.io/otel/metric/noop]: | ||||
| 
 | ||||
| 	import "go.opentelemetry.io/otel/metric/noop" | ||||
|  |  | |||
|  | @ -39,6 +39,12 @@ type InstrumentOption interface { | |||
| 	Float64ObservableGaugeOption | ||||
| } | ||||
| 
 | ||||
| // HistogramOption applies options to histogram instruments.
 | ||||
| type HistogramOption interface { | ||||
| 	Int64HistogramOption | ||||
| 	Float64HistogramOption | ||||
| } | ||||
| 
 | ||||
| type descOpt string | ||||
| 
 | ||||
| func (o descOpt) applyFloat64Counter(c Float64CounterConfig) Float64CounterConfig { | ||||
|  | @ -171,6 +177,23 @@ func (o unitOpt) applyInt64ObservableGauge(c Int64ObservableGaugeConfig) Int64Ob | |||
| // The unit u should be defined using the appropriate [UCUM](https://ucum.org) case-sensitive code.
 | ||||
| func WithUnit(u string) InstrumentOption { return unitOpt(u) } | ||||
| 
 | ||||
| // WithExplicitBucketBoundaries sets the instrument explicit bucket boundaries.
 | ||||
| //
 | ||||
| // This option is considered "advisory", and may be ignored by API implementations.
 | ||||
| func WithExplicitBucketBoundaries(bounds ...float64) HistogramOption { return bucketOpt(bounds) } | ||||
| 
 | ||||
| type bucketOpt []float64 | ||||
| 
 | ||||
| func (o bucketOpt) applyFloat64Histogram(c Float64HistogramConfig) Float64HistogramConfig { | ||||
| 	c.explicitBucketBoundaries = o | ||||
| 	return c | ||||
| } | ||||
| 
 | ||||
| func (o bucketOpt) applyInt64Histogram(c Int64HistogramConfig) Int64HistogramConfig { | ||||
| 	c.explicitBucketBoundaries = o | ||||
| 	return c | ||||
| } | ||||
| 
 | ||||
| // AddOption applies options to an addition measurement. See
 | ||||
| // [MeasurementOption] for other options that can be used as an AddOption.
 | ||||
| type AddOption interface { | ||||
|  |  | |||
|  | @ -147,8 +147,9 @@ type Float64Histogram interface { | |||
| // Float64HistogramConfig contains options for synchronous counter instruments
 | ||||
| // that record int64 values.
 | ||||
| type Float64HistogramConfig struct { | ||||
| 	description string | ||||
| 	unit        string | ||||
| 	description              string | ||||
| 	unit                     string | ||||
| 	explicitBucketBoundaries []float64 | ||||
| } | ||||
| 
 | ||||
| // NewFloat64HistogramConfig returns a new [Float64HistogramConfig] with all
 | ||||
|  | @ -171,6 +172,11 @@ func (c Float64HistogramConfig) Unit() string { | |||
| 	return c.unit | ||||
| } | ||||
| 
 | ||||
| // ExplicitBucketBoundaries returns the configured explicit bucket boundaries.
 | ||||
| func (c Float64HistogramConfig) ExplicitBucketBoundaries() []float64 { | ||||
| 	return c.explicitBucketBoundaries | ||||
| } | ||||
| 
 | ||||
| // Float64HistogramOption applies options to a [Float64HistogramConfig]. See
 | ||||
| // [InstrumentOption] for other options that can be used as a
 | ||||
| // Float64HistogramOption.
 | ||||
|  |  | |||
|  | @ -147,8 +147,9 @@ type Int64Histogram interface { | |||
| // Int64HistogramConfig contains options for synchronous counter instruments
 | ||||
| // that record int64 values.
 | ||||
| type Int64HistogramConfig struct { | ||||
| 	description string | ||||
| 	unit        string | ||||
| 	description              string | ||||
| 	unit                     string | ||||
| 	explicitBucketBoundaries []float64 | ||||
| } | ||||
| 
 | ||||
| // NewInt64HistogramConfig returns a new [Int64HistogramConfig] with all opts
 | ||||
|  | @ -171,6 +172,11 @@ func (c Int64HistogramConfig) Unit() string { | |||
| 	return c.unit | ||||
| } | ||||
| 
 | ||||
| // ExplicitBucketBoundaries returns the configured explicit bucket boundaries.
 | ||||
| func (c Int64HistogramConfig) ExplicitBucketBoundaries() []float64 { | ||||
| 	return c.explicitBucketBoundaries | ||||
| } | ||||
| 
 | ||||
| // Int64HistogramOption applies options to a [Int64HistogramConfig]. See
 | ||||
| // [InstrumentOption] for other options that can be used as an
 | ||||
| // Int64HistogramOption.
 | ||||
|  |  | |||
|  | @ -40,8 +40,10 @@ const ( | |||
| // their proprietary information.
 | ||||
| type TraceContext struct{} | ||||
| 
 | ||||
| var _ TextMapPropagator = TraceContext{} | ||||
| var traceCtxRegExp = regexp.MustCompile("^(?P<version>[0-9a-f]{2})-(?P<traceID>[a-f0-9]{32})-(?P<spanID>[a-f0-9]{16})-(?P<traceFlags>[a-f0-9]{2})(?:-.*)?$") | ||||
| var ( | ||||
| 	_              TextMapPropagator = TraceContext{} | ||||
| 	traceCtxRegExp                   = regexp.MustCompile("^(?P<version>[0-9a-f]{2})-(?P<traceID>[a-f0-9]{32})-(?P<spanID>[a-f0-9]{16})-(?P<traceFlags>[a-f0-9]{2})(?:-.*)?$") | ||||
| ) | ||||
| 
 | ||||
| // Inject set tracecontext from the Context into the carrier.
 | ||||
| func (tc TraceContext) Inject(ctx context.Context, carrier TextMapCarrier) { | ||||
|  |  | |||
|  | @ -1 +1 @@ | |||
| codespell==2.2.5 | ||||
| codespell==2.2.6 | ||||
|  |  | |||
|  | @ -268,6 +268,7 @@ func (o stackTraceOption) applyEvent(c EventConfig) EventConfig { | |||
| 	c.stackTrace = bool(o) | ||||
| 	return c | ||||
| } | ||||
| 
 | ||||
| func (o stackTraceOption) applySpan(c SpanConfig) SpanConfig { | ||||
| 	c.stackTrace = bool(o) | ||||
| 	return c | ||||
|  |  | |||
|  | @ -62,5 +62,69 @@ a default. | |||
| 		defer span.End() | ||||
| 		// ...
 | ||||
| 	} | ||||
| 
 | ||||
| # API Implementations | ||||
| 
 | ||||
| This package does not conform to the standard Go versioning policy; all of its | ||||
| interfaces may have methods added to them without a package major version bump. | ||||
| This non-standard API evolution could surprise an uninformed implementation | ||||
| author. They could unknowingly build their implementation in a way that would | ||||
| result in a runtime panic for their users that update to the new API. | ||||
| 
 | ||||
| The API is designed to help inform an instrumentation author about this | ||||
| non-standard API evolution. It requires them to choose a default behavior for | ||||
| unimplemented interface methods. There are three behavior choices they can | ||||
| make: | ||||
| 
 | ||||
|   - Compilation failure | ||||
|   - Panic | ||||
|   - Default to another implementation | ||||
| 
 | ||||
| All interfaces in this API embed a corresponding interface from | ||||
| [go.opentelemetry.io/otel/trace/embedded]. If an author wants the default | ||||
| behavior of their implementations to be a compilation failure, signaling to | ||||
| their users they need to update to the latest version of that implementation, | ||||
| they need to embed the corresponding interface from | ||||
| [go.opentelemetry.io/otel/trace/embedded] in their implementation. For | ||||
| example, | ||||
| 
 | ||||
| 	import "go.opentelemetry.io/otel/trace/embedded" | ||||
| 
 | ||||
| 	type TracerProvider struct { | ||||
| 		embedded.TracerProvider | ||||
| 		// ...
 | ||||
| 	} | ||||
| 
 | ||||
| If an author wants the default behavior of their implementations to panic, they | ||||
| can embed the API interface directly. | ||||
| 
 | ||||
| 	import "go.opentelemetry.io/otel/trace" | ||||
| 
 | ||||
| 	type TracerProvider struct { | ||||
| 		trace.TracerProvider | ||||
| 		// ...
 | ||||
| 	} | ||||
| 
 | ||||
| This option is not recommended. It will lead to publishing packages that | ||||
| contain runtime panics when users update to newer versions of | ||||
| [go.opentelemetry.io/otel/trace], which may be done with a trasitive | ||||
| dependency. | ||||
| 
 | ||||
| Finally, an author can embed another implementation in theirs. The embedded | ||||
| implementation will be used for methods not defined by the author. For example, | ||||
| an author who wants to default to silently dropping the call can use | ||||
| [go.opentelemetry.io/otel/trace/noop]: | ||||
| 
 | ||||
| 	import "go.opentelemetry.io/otel/trace/noop" | ||||
| 
 | ||||
| 	type TracerProvider struct { | ||||
| 		noop.TracerProvider | ||||
| 		// ...
 | ||||
| 	} | ||||
| 
 | ||||
| It is strongly recommended that authors only embed | ||||
| [go.opentelemetry.io/otel/trace/noop] if they choose this default behavior. | ||||
| That implementation is the only one OpenTelemetry authors can guarantee will | ||||
| fully implement all the API interfaces when a user updates their API. | ||||
| */ | ||||
| package trace // import "go.opentelemetry.io/otel/trace"
 | ||||
|  |  | |||
|  | @ -0,0 +1,56 @@ | |||
| // Copyright The OpenTelemetry Authors
 | ||||
| //
 | ||||
| // Licensed under the Apache License, Version 2.0 (the "License");
 | ||||
| // you may not use this file except in compliance with the License.
 | ||||
| // You may obtain a copy of the License at
 | ||||
| //
 | ||||
| //     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
| //
 | ||||
| // Unless required by applicable law or agreed to in writing, software
 | ||||
| // distributed under the License is distributed on an "AS IS" BASIS,
 | ||||
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | ||||
| // See the License for the specific language governing permissions and
 | ||||
| // limitations under the License.
 | ||||
| 
 | ||||
| // Package embedded provides interfaces embedded within the [OpenTelemetry
 | ||||
| // trace API].
 | ||||
| //
 | ||||
| // Implementers of the [OpenTelemetry trace API] can embed the relevant type
 | ||||
| // from this package into their implementation directly. Doing so will result
 | ||||
| // in a compilation error for users when the [OpenTelemetry trace API] is
 | ||||
| // extended (which is something that can happen without a major version bump of
 | ||||
| // the API package).
 | ||||
| //
 | ||||
| // [OpenTelemetry trace API]: https://pkg.go.dev/go.opentelemetry.io/otel/trace
 | ||||
| package embedded // import "go.opentelemetry.io/otel/trace/embedded"
 | ||||
| 
 | ||||
| // TracerProvider is embedded in
 | ||||
| // [go.opentelemetry.io/otel/trace.TracerProvider].
 | ||||
| //
 | ||||
| // Embed this interface in your implementation of the
 | ||||
| // [go.opentelemetry.io/otel/trace.TracerProvider] if you want users to
 | ||||
| // experience a compilation error, signaling they need to update to your latest
 | ||||
| // implementation, when the [go.opentelemetry.io/otel/trace.TracerProvider]
 | ||||
| // interface is extended (which is something that can happen without a major
 | ||||
| // version bump of the API package).
 | ||||
| type TracerProvider interface{ tracerProvider() } | ||||
| 
 | ||||
| // Tracer is embedded in [go.opentelemetry.io/otel/trace.Tracer].
 | ||||
| //
 | ||||
| // Embed this interface in your implementation of the
 | ||||
| // [go.opentelemetry.io/otel/trace.Tracer] if you want users to experience a
 | ||||
| // compilation error, signaling they need to update to your latest
 | ||||
| // implementation, when the [go.opentelemetry.io/otel/trace.Tracer] interface
 | ||||
| // is extended (which is something that can happen without a major version bump
 | ||||
| // of the API package).
 | ||||
| type Tracer interface{ tracer() } | ||||
| 
 | ||||
| // Span is embedded in [go.opentelemetry.io/otel/trace.Span].
 | ||||
| //
 | ||||
| // Embed this interface in your implementation of the
 | ||||
| // [go.opentelemetry.io/otel/trace.Span] if you want users to experience a
 | ||||
| // compilation error, signaling they need to update to your latest
 | ||||
| // implementation, when the [go.opentelemetry.io/otel/trace.Span] interface is
 | ||||
| // extended (which is something that can happen without a major version bump of
 | ||||
| // the API package).
 | ||||
| type Span interface{ span() } | ||||
|  | @ -19,16 +19,20 @@ import ( | |||
| 
 | ||||
| 	"go.opentelemetry.io/otel/attribute" | ||||
| 	"go.opentelemetry.io/otel/codes" | ||||
| 	"go.opentelemetry.io/otel/trace/embedded" | ||||
| ) | ||||
| 
 | ||||
| // NewNoopTracerProvider returns an implementation of TracerProvider that
 | ||||
| // performs no operations. The Tracer and Spans created from the returned
 | ||||
| // TracerProvider also perform no operations.
 | ||||
| //
 | ||||
| // Deprecated: Use [go.opentelemetry.io/otel/trace/noop.NewTracerProvider]
 | ||||
| // instead.
 | ||||
| func NewNoopTracerProvider() TracerProvider { | ||||
| 	return noopTracerProvider{} | ||||
| } | ||||
| 
 | ||||
| type noopTracerProvider struct{} | ||||
| type noopTracerProvider struct{ embedded.TracerProvider } | ||||
| 
 | ||||
| var _ TracerProvider = noopTracerProvider{} | ||||
| 
 | ||||
|  | @ -38,7 +42,7 @@ func (p noopTracerProvider) Tracer(string, ...TracerOption) Tracer { | |||
| } | ||||
| 
 | ||||
| // noopTracer is an implementation of Tracer that performs no operations.
 | ||||
| type noopTracer struct{} | ||||
| type noopTracer struct{ embedded.Tracer } | ||||
| 
 | ||||
| var _ Tracer = noopTracer{} | ||||
| 
 | ||||
|  | @ -54,7 +58,7 @@ func (t noopTracer) Start(ctx context.Context, name string, _ ...SpanStartOption | |||
| } | ||||
| 
 | ||||
| // noopSpan is an implementation of Span that performs no operations.
 | ||||
| type noopSpan struct{} | ||||
| type noopSpan struct{ embedded.Span } | ||||
| 
 | ||||
| var _ Span = noopSpan{} | ||||
| 
 | ||||
|  |  | |||
|  | @ -22,6 +22,7 @@ import ( | |||
| 
 | ||||
| 	"go.opentelemetry.io/otel/attribute" | ||||
| 	"go.opentelemetry.io/otel/codes" | ||||
| 	"go.opentelemetry.io/otel/trace/embedded" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
|  | @ -48,8 +49,10 @@ func (e errorConst) Error() string { | |||
| // nolint:revive // revive complains about stutter of `trace.TraceID`.
 | ||||
| type TraceID [16]byte | ||||
| 
 | ||||
| var nilTraceID TraceID | ||||
| var _ json.Marshaler = nilTraceID | ||||
| var ( | ||||
| 	nilTraceID TraceID | ||||
| 	_          json.Marshaler = nilTraceID | ||||
| ) | ||||
| 
 | ||||
| // IsValid checks whether the trace TraceID is valid. A valid trace ID does
 | ||||
| // not consist of zeros only.
 | ||||
|  | @ -71,8 +74,10 @@ func (t TraceID) String() string { | |||
| // SpanID is a unique identity of a span in a trace.
 | ||||
| type SpanID [8]byte | ||||
| 
 | ||||
| var nilSpanID SpanID | ||||
| var _ json.Marshaler = nilSpanID | ||||
| var ( | ||||
| 	nilSpanID SpanID | ||||
| 	_         json.Marshaler = nilSpanID | ||||
| ) | ||||
| 
 | ||||
| // IsValid checks whether the SpanID is valid. A valid SpanID does not consist
 | ||||
| // of zeros only.
 | ||||
|  | @ -338,8 +343,15 @@ func (sc SpanContext) MarshalJSON() ([]byte, error) { | |||
| // create a Span and it is then up to the operation the Span represents to
 | ||||
| // properly end the Span when the operation itself ends.
 | ||||
| //
 | ||||
| // Warning: methods may be added to this interface in minor releases.
 | ||||
| // Warning: Methods may be added to this interface in minor releases. See
 | ||||
| // package documentation on API implementation for information on how to set
 | ||||
| // default behavior for unimplemented methods.
 | ||||
| type Span interface { | ||||
| 	// Users of the interface can ignore this. This embedded type is only used
 | ||||
| 	// by implementations of this interface. See the "API Implementations"
 | ||||
| 	// section of the package documentation for more information.
 | ||||
| 	embedded.Span | ||||
| 
 | ||||
| 	// End completes the Span. The Span is considered complete and ready to be
 | ||||
| 	// delivered through the rest of the telemetry pipeline after this method
 | ||||
| 	// is called. Therefore, updates to the Span are not allowed after this
 | ||||
|  | @ -486,8 +498,15 @@ func (sk SpanKind) String() string { | |||
| 
 | ||||
| // Tracer is the creator of Spans.
 | ||||
| //
 | ||||
| // Warning: methods may be added to this interface in minor releases.
 | ||||
| // Warning: Methods may be added to this interface in minor releases. See
 | ||||
| // package documentation on API implementation for information on how to set
 | ||||
| // default behavior for unimplemented methods.
 | ||||
| type Tracer interface { | ||||
| 	// Users of the interface can ignore this. This embedded type is only used
 | ||||
| 	// by implementations of this interface. See the "API Implementations"
 | ||||
| 	// section of the package documentation for more information.
 | ||||
| 	embedded.Tracer | ||||
| 
 | ||||
| 	// Start creates a span and a context.Context containing the newly-created span.
 | ||||
| 	//
 | ||||
| 	// If the context.Context provided in `ctx` contains a Span then the newly-created
 | ||||
|  | @ -518,8 +537,15 @@ type Tracer interface { | |||
| // at runtime from its users or it can simply use the globally registered one
 | ||||
| // (see https://pkg.go.dev/go.opentelemetry.io/otel#GetTracerProvider).
 | ||||
| //
 | ||||
| // Warning: methods may be added to this interface in minor releases.
 | ||||
| // Warning: Methods may be added to this interface in minor releases. See
 | ||||
| // package documentation on API implementation for information on how to set
 | ||||
| // default behavior for unimplemented methods.
 | ||||
| type TracerProvider interface { | ||||
| 	// Users of the interface can ignore this. This embedded type is only used
 | ||||
| 	// by implementations of this interface. See the "API Implementations"
 | ||||
| 	// section of the package documentation for more information.
 | ||||
| 	embedded.TracerProvider | ||||
| 
 | ||||
| 	// Tracer returns a unique Tracer scoped to be used by instrumentation code
 | ||||
| 	// to trace computational workflows. The scope and identity of that
 | ||||
| 	// instrumentation code is uniquely defined by the name and options passed.
 | ||||
|  |  | |||
|  | @ -28,9 +28,9 @@ const ( | |||
| 
 | ||||
| 	// based on the W3C Trace Context specification, see
 | ||||
| 	// https://www.w3.org/TR/trace-context-1/#tracestate-header
 | ||||
| 	noTenantKeyFormat   = `[a-z][_0-9a-z\-\*\/]{0,255}` | ||||
| 	withTenantKeyFormat = `[a-z0-9][_0-9a-z\-\*\/]{0,240}@[a-z][_0-9a-z\-\*\/]{0,13}` | ||||
| 	valueFormat         = `[\x20-\x2b\x2d-\x3c\x3e-\x7e]{0,255}[\x21-\x2b\x2d-\x3c\x3e-\x7e]` | ||||
| 	noTenantKeyFormat   = `[a-z][_0-9a-z\-\*\/]*` | ||||
| 	withTenantKeyFormat = `[a-z0-9][_0-9a-z\-\*\/]*@[a-z][_0-9a-z\-\*\/]*` | ||||
| 	valueFormat         = `[\x20-\x2b\x2d-\x3c\x3e-\x7e]*[\x21-\x2b\x2d-\x3c\x3e-\x7e]` | ||||
| 
 | ||||
| 	errInvalidKey    errorConst = "invalid tracestate key" | ||||
| 	errInvalidValue  errorConst = "invalid tracestate value" | ||||
|  | @ -40,9 +40,10 @@ const ( | |||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	keyRe    = regexp.MustCompile(`^((` + noTenantKeyFormat + `)|(` + withTenantKeyFormat + `))$`) | ||||
| 	valueRe  = regexp.MustCompile(`^(` + valueFormat + `)$`) | ||||
| 	memberRe = regexp.MustCompile(`^\s*((` + noTenantKeyFormat + `)|(` + withTenantKeyFormat + `))=(` + valueFormat + `)\s*$`) | ||||
| 	noTenantKeyRe   = regexp.MustCompile(`^` + noTenantKeyFormat + `$`) | ||||
| 	withTenantKeyRe = regexp.MustCompile(`^` + withTenantKeyFormat + `$`) | ||||
| 	valueRe         = regexp.MustCompile(`^` + valueFormat + `$`) | ||||
| 	memberRe        = regexp.MustCompile(`^\s*((?:` + noTenantKeyFormat + `)|(?:` + withTenantKeyFormat + `))=(` + valueFormat + `)\s*$`) | ||||
| ) | ||||
| 
 | ||||
| type member struct { | ||||
|  | @ -51,10 +52,19 @@ type member struct { | |||
| } | ||||
| 
 | ||||
| func newMember(key, value string) (member, error) { | ||||
| 	if !keyRe.MatchString(key) { | ||||
| 	if len(key) > 256 { | ||||
| 		return member{}, fmt.Errorf("%w: %s", errInvalidKey, key) | ||||
| 	} | ||||
| 	if !valueRe.MatchString(value) { | ||||
| 	if !noTenantKeyRe.MatchString(key) { | ||||
| 		if !withTenantKeyRe.MatchString(key) { | ||||
| 			return member{}, fmt.Errorf("%w: %s", errInvalidKey, key) | ||||
| 		} | ||||
| 		atIndex := strings.LastIndex(key, "@") | ||||
| 		if atIndex > 241 || len(key)-1-atIndex > 14 { | ||||
| 			return member{}, fmt.Errorf("%w: %s", errInvalidKey, key) | ||||
| 		} | ||||
| 	} | ||||
| 	if len(value) > 256 || !valueRe.MatchString(value) { | ||||
| 		return member{}, fmt.Errorf("%w: %s", errInvalidValue, value) | ||||
| 	} | ||||
| 	return member{Key: key, Value: value}, nil | ||||
|  | @ -62,14 +72,14 @@ func newMember(key, value string) (member, error) { | |||
| 
 | ||||
| func parseMember(m string) (member, error) { | ||||
| 	matches := memberRe.FindStringSubmatch(m) | ||||
| 	if len(matches) != 5 { | ||||
| 	if len(matches) != 3 { | ||||
| 		return member{}, fmt.Errorf("%w: %s", errInvalidMember, m) | ||||
| 	} | ||||
| 
 | ||||
| 	return member{ | ||||
| 		Key:   matches[1], | ||||
| 		Value: matches[4], | ||||
| 	}, nil | ||||
| 	result, e := newMember(matches[1], matches[2]) | ||||
| 	if e != nil { | ||||
| 		return member{}, fmt.Errorf("%w: %s", errInvalidMember, m) | ||||
| 	} | ||||
| 	return result, nil | ||||
| } | ||||
| 
 | ||||
| // String encodes member into a string compliant with the W3C Trace Context
 | ||||
|  |  | |||
|  | @ -16,5 +16,5 @@ package otel // import "go.opentelemetry.io/otel" | |||
| 
 | ||||
| // Version is the current release version of OpenTelemetry in use.
 | ||||
| func Version() string { | ||||
| 	return "1.19.0" | ||||
| 	return "1.21.0" | ||||
| } | ||||
|  |  | |||
|  | @ -14,13 +14,12 @@ | |||
| 
 | ||||
| module-sets: | ||||
|   stable-v1: | ||||
|     version: v1.19.0 | ||||
|     version: v1.21.0 | ||||
|     modules: | ||||
|       - go.opentelemetry.io/otel | ||||
|       - go.opentelemetry.io/otel/bridge/opentracing | ||||
|       - go.opentelemetry.io/otel/bridge/opentracing/test | ||||
|       - go.opentelemetry.io/otel/example/dice | ||||
|       - go.opentelemetry.io/otel/example/fib | ||||
|       - go.opentelemetry.io/otel/example/namedtracer | ||||
|       - go.opentelemetry.io/otel/example/otel-collector | ||||
|       - go.opentelemetry.io/otel/example/passthrough | ||||
|  | @ -35,14 +34,12 @@ module-sets: | |||
|       - go.opentelemetry.io/otel/sdk/metric | ||||
|       - go.opentelemetry.io/otel/trace | ||||
|   experimental-metrics: | ||||
|     version: v0.42.0 | ||||
|     version: v0.44.0 | ||||
|     modules: | ||||
|       - go.opentelemetry.io/otel/bridge/opencensus | ||||
|       - go.opentelemetry.io/otel/bridge/opencensus/test | ||||
|       - go.opentelemetry.io/otel/example/opencensus | ||||
|       - go.opentelemetry.io/otel/example/prometheus | ||||
|       - go.opentelemetry.io/otel/example/view | ||||
|       - go.opentelemetry.io/otel/exporters/otlp/otlpmetric | ||||
|       - go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc | ||||
|       - go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp | ||||
|       - go.opentelemetry.io/otel/exporters/prometheus | ||||
|  |  | |||
|  | @ -32,21 +32,13 @@ import ( | |||
| 	"google.golang.org/grpc/resolver" | ||||
| ) | ||||
| 
 | ||||
| type ccbMode int | ||||
| 
 | ||||
| const ( | ||||
| 	ccbModeActive = iota | ||||
| 	ccbModeIdle | ||||
| 	ccbModeClosed | ||||
| 	ccbModeExitingIdle | ||||
| ) | ||||
| 
 | ||||
| // ccBalancerWrapper sits between the ClientConn and the Balancer.
 | ||||
| //
 | ||||
| // ccBalancerWrapper implements methods corresponding to the ones on the
 | ||||
| // balancer.Balancer interface. The ClientConn is free to call these methods
 | ||||
| // concurrently and the ccBalancerWrapper ensures that calls from the ClientConn
 | ||||
| // to the Balancer happen synchronously and in order.
 | ||||
| // to the Balancer happen in order by performing them in the serializer, without
 | ||||
| // any mutexes held.
 | ||||
| //
 | ||||
| // ccBalancerWrapper also implements the balancer.ClientConn interface and is
 | ||||
| // passed to the Balancer implementations. It invokes unexported methods on the
 | ||||
|  | @ -57,87 +49,75 @@ const ( | |||
| type ccBalancerWrapper struct { | ||||
| 	// The following fields are initialized when the wrapper is created and are
 | ||||
| 	// read-only afterwards, and therefore can be accessed without a mutex.
 | ||||
| 	cc   *ClientConn | ||||
| 	opts balancer.BuildOptions | ||||
| 	cc               *ClientConn | ||||
| 	opts             balancer.BuildOptions | ||||
| 	serializer       *grpcsync.CallbackSerializer | ||||
| 	serializerCancel context.CancelFunc | ||||
| 
 | ||||
| 	// Outgoing (gRPC --> balancer) calls are guaranteed to execute in a
 | ||||
| 	// mutually exclusive manner as they are scheduled in the serializer. Fields
 | ||||
| 	// accessed *only* in these serializer callbacks, can therefore be accessed
 | ||||
| 	// without a mutex.
 | ||||
| 	balancer        *gracefulswitch.Balancer | ||||
| 	// The following fields are only accessed within the serializer or during
 | ||||
| 	// initialization.
 | ||||
| 	curBalancerName string | ||||
| 	balancer        *gracefulswitch.Balancer | ||||
| 
 | ||||
| 	// mu guards access to the below fields. Access to the serializer and its
 | ||||
| 	// cancel function needs to be mutex protected because they are overwritten
 | ||||
| 	// when the wrapper exits idle mode.
 | ||||
| 	mu               sync.Mutex | ||||
| 	serializer       *grpcsync.CallbackSerializer // To serialize all outoing calls.
 | ||||
| 	serializerCancel context.CancelFunc           // To close the seralizer at close/enterIdle time.
 | ||||
| 	mode             ccbMode                      // Tracks the current mode of the wrapper.
 | ||||
| 	// The following field is protected by mu.  Caller must take cc.mu before
 | ||||
| 	// taking mu.
 | ||||
| 	mu     sync.Mutex | ||||
| 	closed bool | ||||
| } | ||||
| 
 | ||||
| // newCCBalancerWrapper creates a new balancer wrapper. The underlying balancer
 | ||||
| // is not created until the switchTo() method is invoked.
 | ||||
| func newCCBalancerWrapper(cc *ClientConn, bopts balancer.BuildOptions) *ccBalancerWrapper { | ||||
| 	ctx, cancel := context.WithCancel(context.Background()) | ||||
| // newCCBalancerWrapper creates a new balancer wrapper in idle state. The
 | ||||
| // underlying balancer is not created until the switchTo() method is invoked.
 | ||||
| func newCCBalancerWrapper(cc *ClientConn) *ccBalancerWrapper { | ||||
| 	ctx, cancel := context.WithCancel(cc.ctx) | ||||
| 	ccb := &ccBalancerWrapper{ | ||||
| 		cc:               cc, | ||||
| 		opts:             bopts, | ||||
| 		cc: cc, | ||||
| 		opts: balancer.BuildOptions{ | ||||
| 			DialCreds:        cc.dopts.copts.TransportCredentials, | ||||
| 			CredsBundle:      cc.dopts.copts.CredsBundle, | ||||
| 			Dialer:           cc.dopts.copts.Dialer, | ||||
| 			Authority:        cc.authority, | ||||
| 			CustomUserAgent:  cc.dopts.copts.UserAgent, | ||||
| 			ChannelzParentID: cc.channelzID, | ||||
| 			Target:           cc.parsedTarget, | ||||
| 		}, | ||||
| 		serializer:       grpcsync.NewCallbackSerializer(ctx), | ||||
| 		serializerCancel: cancel, | ||||
| 	} | ||||
| 	ccb.balancer = gracefulswitch.NewBalancer(ccb, bopts) | ||||
| 	ccb.balancer = gracefulswitch.NewBalancer(ccb, ccb.opts) | ||||
| 	return ccb | ||||
| } | ||||
| 
 | ||||
| // updateClientConnState is invoked by grpc to push a ClientConnState update to
 | ||||
| // the underlying balancer.
 | ||||
| // the underlying balancer.  This is always executed from the serializer, so
 | ||||
| // it is safe to call into the balancer here.
 | ||||
| func (ccb *ccBalancerWrapper) updateClientConnState(ccs *balancer.ClientConnState) error { | ||||
| 	ccb.mu.Lock() | ||||
| 	errCh := make(chan error, 1) | ||||
| 	// Here and everywhere else where Schedule() is called, it is done with the
 | ||||
| 	// lock held. But the lock guards only the scheduling part. The actual
 | ||||
| 	// callback is called asynchronously without the lock being held.
 | ||||
| 	ok := ccb.serializer.Schedule(func(_ context.Context) { | ||||
| 		errCh <- ccb.balancer.UpdateClientConnState(*ccs) | ||||
| 	errCh := make(chan error) | ||||
| 	ok := ccb.serializer.Schedule(func(ctx context.Context) { | ||||
| 		defer close(errCh) | ||||
| 		if ctx.Err() != nil || ccb.balancer == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		err := ccb.balancer.UpdateClientConnState(*ccs) | ||||
| 		if logger.V(2) && err != nil { | ||||
| 			logger.Infof("error from balancer.UpdateClientConnState: %v", err) | ||||
| 		} | ||||
| 		errCh <- err | ||||
| 	}) | ||||
| 	if !ok { | ||||
| 		// If we are unable to schedule a function with the serializer, it
 | ||||
| 		// indicates that it has been closed. A serializer is only closed when
 | ||||
| 		// the wrapper is closed or is in idle.
 | ||||
| 		ccb.mu.Unlock() | ||||
| 		return fmt.Errorf("grpc: cannot send state update to a closed or idle balancer") | ||||
| 		return nil | ||||
| 	} | ||||
| 	ccb.mu.Unlock() | ||||
| 
 | ||||
| 	// We get here only if the above call to Schedule succeeds, in which case it
 | ||||
| 	// is guaranteed that the scheduled function will run. Therefore it is safe
 | ||||
| 	// to block on this channel.
 | ||||
| 	err := <-errCh | ||||
| 	if logger.V(2) && err != nil { | ||||
| 		logger.Infof("error from balancer.UpdateClientConnState: %v", err) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
| 
 | ||||
| // updateSubConnState is invoked by grpc to push a subConn state update to the
 | ||||
| // underlying balancer.
 | ||||
| func (ccb *ccBalancerWrapper) updateSubConnState(sc balancer.SubConn, s connectivity.State, err error) { | ||||
| 	ccb.mu.Lock() | ||||
| 	ccb.serializer.Schedule(func(_ context.Context) { | ||||
| 		// Even though it is optional for balancers, gracefulswitch ensures
 | ||||
| 		// opts.StateListener is set, so this cannot ever be nil.
 | ||||
| 		sc.(*acBalancerWrapper).stateListener(balancer.SubConnState{ConnectivityState: s, ConnectionError: err}) | ||||
| 	}) | ||||
| 	ccb.mu.Unlock() | ||||
| 	return <-errCh | ||||
| } | ||||
| 
 | ||||
| // resolverError is invoked by grpc to push a resolver error to the underlying
 | ||||
| // balancer.  The call to the balancer is executed from the serializer.
 | ||||
| func (ccb *ccBalancerWrapper) resolverError(err error) { | ||||
| 	ccb.mu.Lock() | ||||
| 	ccb.serializer.Schedule(func(_ context.Context) { | ||||
| 	ccb.serializer.Schedule(func(ctx context.Context) { | ||||
| 		if ctx.Err() != nil || ccb.balancer == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		ccb.balancer.ResolverError(err) | ||||
| 	}) | ||||
| 	ccb.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| // switchTo is invoked by grpc to instruct the balancer wrapper to switch to the
 | ||||
|  | @ -151,8 +131,10 @@ func (ccb *ccBalancerWrapper) resolverError(err error) { | |||
| // the ccBalancerWrapper keeps track of the current LB policy name, and skips
 | ||||
| // the graceful balancer switching process if the name does not change.
 | ||||
| func (ccb *ccBalancerWrapper) switchTo(name string) { | ||||
| 	ccb.mu.Lock() | ||||
| 	ccb.serializer.Schedule(func(_ context.Context) { | ||||
| 	ccb.serializer.Schedule(func(ctx context.Context) { | ||||
| 		if ctx.Err() != nil || ccb.balancer == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		// TODO: Other languages use case-sensitive balancer registries. We should
 | ||||
| 		// switch as well. See: https://github.com/grpc/grpc-go/issues/5288.
 | ||||
| 		if strings.EqualFold(ccb.curBalancerName, name) { | ||||
|  | @ -160,7 +142,6 @@ func (ccb *ccBalancerWrapper) switchTo(name string) { | |||
| 		} | ||||
| 		ccb.buildLoadBalancingPolicy(name) | ||||
| 	}) | ||||
| 	ccb.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| // buildLoadBalancingPolicy performs the following:
 | ||||
|  | @ -187,115 +168,49 @@ func (ccb *ccBalancerWrapper) buildLoadBalancingPolicy(name string) { | |||
| 	ccb.curBalancerName = builder.Name() | ||||
| } | ||||
| 
 | ||||
| // close initiates async shutdown of the wrapper.  cc.mu must be held when
 | ||||
| // calling this function.  To determine the wrapper has finished shutting down,
 | ||||
| // the channel should block on ccb.serializer.Done() without cc.mu held.
 | ||||
| func (ccb *ccBalancerWrapper) close() { | ||||
| 	channelz.Info(logger, ccb.cc.channelzID, "ccBalancerWrapper: closing") | ||||
| 	ccb.closeBalancer(ccbModeClosed) | ||||
| } | ||||
| 
 | ||||
| // enterIdleMode is invoked by grpc when the channel enters idle mode upon
 | ||||
| // expiry of idle_timeout. This call blocks until the balancer is closed.
 | ||||
| func (ccb *ccBalancerWrapper) enterIdleMode() { | ||||
| 	channelz.Info(logger, ccb.cc.channelzID, "ccBalancerWrapper: entering idle mode") | ||||
| 	ccb.closeBalancer(ccbModeIdle) | ||||
| } | ||||
| 
 | ||||
| // closeBalancer is invoked when the channel is being closed or when it enters
 | ||||
| // idle mode upon expiry of idle_timeout.
 | ||||
| func (ccb *ccBalancerWrapper) closeBalancer(m ccbMode) { | ||||
| 	ccb.mu.Lock() | ||||
| 	if ccb.mode == ccbModeClosed || ccb.mode == ccbModeIdle { | ||||
| 		ccb.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	ccb.mode = m | ||||
| 	done := ccb.serializer.Done() | ||||
| 	b := ccb.balancer | ||||
| 	ok := ccb.serializer.Schedule(func(_ context.Context) { | ||||
| 		// Close the serializer to ensure that no more calls from gRPC are sent
 | ||||
| 		// to the balancer.
 | ||||
| 		ccb.serializerCancel() | ||||
| 		// Empty the current balancer name because we don't have a balancer
 | ||||
| 		// anymore and also so that we act on the next call to switchTo by
 | ||||
| 		// creating a new balancer specified by the new resolver.
 | ||||
| 		ccb.curBalancerName = "" | ||||
| 	}) | ||||
| 	if !ok { | ||||
| 		ccb.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 	ccb.closed = true | ||||
| 	ccb.mu.Unlock() | ||||
| 
 | ||||
| 	// Give enqueued callbacks a chance to finish before closing the balancer.
 | ||||
| 	<-done | ||||
| 	b.Close() | ||||
| } | ||||
| 
 | ||||
| // exitIdleMode is invoked by grpc when the channel exits idle mode either
 | ||||
| // because of an RPC or because of an invocation of the Connect() API. This
 | ||||
| // recreates the balancer that was closed previously when entering idle mode.
 | ||||
| //
 | ||||
| // If the channel is not in idle mode, we know for a fact that we are here as a
 | ||||
| // result of the user calling the Connect() method on the ClientConn. In this
 | ||||
| // case, we can simply forward the call to the underlying balancer, instructing
 | ||||
| // it to reconnect to the backends.
 | ||||
| func (ccb *ccBalancerWrapper) exitIdleMode() { | ||||
| 	ccb.mu.Lock() | ||||
| 	if ccb.mode == ccbModeClosed { | ||||
| 		// Request to exit idle is a no-op when wrapper is already closed.
 | ||||
| 		ccb.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if ccb.mode == ccbModeIdle { | ||||
| 		// Recreate the serializer which was closed when we entered idle.
 | ||||
| 		ctx, cancel := context.WithCancel(context.Background()) | ||||
| 		ccb.serializer = grpcsync.NewCallbackSerializer(ctx) | ||||
| 		ccb.serializerCancel = cancel | ||||
| 	} | ||||
| 
 | ||||
| 	// The ClientConn guarantees that mutual exclusion between close() and
 | ||||
| 	// exitIdleMode(), and since we just created a new serializer, we can be
 | ||||
| 	// sure that the below function will be scheduled.
 | ||||
| 	done := make(chan struct{}) | ||||
| 	ccb.serializer.Schedule(func(_ context.Context) { | ||||
| 		defer close(done) | ||||
| 
 | ||||
| 		ccb.mu.Lock() | ||||
| 		defer ccb.mu.Unlock() | ||||
| 
 | ||||
| 		if ccb.mode != ccbModeIdle { | ||||
| 			ccb.balancer.ExitIdle() | ||||
| 	channelz.Info(logger, ccb.cc.channelzID, "ccBalancerWrapper: closing") | ||||
| 	ccb.serializer.Schedule(func(context.Context) { | ||||
| 		if ccb.balancer == nil { | ||||
| 			return | ||||
| 		} | ||||
| 
 | ||||
| 		// Gracefulswitch balancer does not support a switchTo operation after
 | ||||
| 		// being closed. Hence we need to create a new one here.
 | ||||
| 		ccb.balancer = gracefulswitch.NewBalancer(ccb, ccb.opts) | ||||
| 		ccb.mode = ccbModeActive | ||||
| 		channelz.Info(logger, ccb.cc.channelzID, "ccBalancerWrapper: exiting idle mode") | ||||
| 
 | ||||
| 		ccb.balancer.Close() | ||||
| 		ccb.balancer = nil | ||||
| 	}) | ||||
| 	ccb.mu.Unlock() | ||||
| 
 | ||||
| 	<-done | ||||
| 	ccb.serializerCancel() | ||||
| } | ||||
| 
 | ||||
| func (ccb *ccBalancerWrapper) isIdleOrClosed() bool { | ||||
| 	ccb.mu.Lock() | ||||
| 	defer ccb.mu.Unlock() | ||||
| 	return ccb.mode == ccbModeIdle || ccb.mode == ccbModeClosed | ||||
| // exitIdle invokes the balancer's exitIdle method in the serializer.
 | ||||
| func (ccb *ccBalancerWrapper) exitIdle() { | ||||
| 	ccb.serializer.Schedule(func(ctx context.Context) { | ||||
| 		if ctx.Err() != nil || ccb.balancer == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		ccb.balancer.ExitIdle() | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (ccb *ccBalancerWrapper) NewSubConn(addrs []resolver.Address, opts balancer.NewSubConnOptions) (balancer.SubConn, error) { | ||||
| 	if ccb.isIdleOrClosed() { | ||||
| 		return nil, fmt.Errorf("grpc: cannot create SubConn when balancer is closed or idle") | ||||
| 	ccb.cc.mu.Lock() | ||||
| 	defer ccb.cc.mu.Unlock() | ||||
| 
 | ||||
| 	ccb.mu.Lock() | ||||
| 	if ccb.closed { | ||||
| 		ccb.mu.Unlock() | ||||
| 		return nil, fmt.Errorf("balancer is being closed; no new SubConns allowed") | ||||
| 	} | ||||
| 	ccb.mu.Unlock() | ||||
| 
 | ||||
| 	if len(addrs) == 0 { | ||||
| 		return nil, fmt.Errorf("grpc: cannot create SubConn with empty address list") | ||||
| 	} | ||||
| 	ac, err := ccb.cc.newAddrConn(addrs, opts) | ||||
| 	ac, err := ccb.cc.newAddrConnLocked(addrs, opts) | ||||
| 	if err != nil { | ||||
| 		channelz.Warningf(logger, ccb.cc.channelzID, "acBalancerWrapper: NewSubConn: failed to newAddrConn: %v", err) | ||||
| 		return nil, err | ||||
|  | @ -316,10 +231,6 @@ func (ccb *ccBalancerWrapper) RemoveSubConn(sc balancer.SubConn) { | |||
| } | ||||
| 
 | ||||
| func (ccb *ccBalancerWrapper) UpdateAddresses(sc balancer.SubConn, addrs []resolver.Address) { | ||||
| 	if ccb.isIdleOrClosed() { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	acbw, ok := sc.(*acBalancerWrapper) | ||||
| 	if !ok { | ||||
| 		return | ||||
|  | @ -328,25 +239,39 @@ func (ccb *ccBalancerWrapper) UpdateAddresses(sc balancer.SubConn, addrs []resol | |||
| } | ||||
| 
 | ||||
| func (ccb *ccBalancerWrapper) UpdateState(s balancer.State) { | ||||
| 	if ccb.isIdleOrClosed() { | ||||
| 	ccb.cc.mu.Lock() | ||||
| 	defer ccb.cc.mu.Unlock() | ||||
| 
 | ||||
| 	ccb.mu.Lock() | ||||
| 	if ccb.closed { | ||||
| 		ccb.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	ccb.mu.Unlock() | ||||
| 	// Update picker before updating state.  Even though the ordering here does
 | ||||
| 	// not matter, it can lead to multiple calls of Pick in the common start-up
 | ||||
| 	// case where we wait for ready and then perform an RPC.  If the picker is
 | ||||
| 	// updated later, we could call the "connecting" picker when the state is
 | ||||
| 	// updated, and then call the "ready" picker after the picker gets updated.
 | ||||
| 	ccb.cc.blockingpicker.updatePicker(s.Picker) | ||||
| 
 | ||||
| 	// Note that there is no need to check if the balancer wrapper was closed,
 | ||||
| 	// as we know the graceful switch LB policy will not call cc if it has been
 | ||||
| 	// closed.
 | ||||
| 	ccb.cc.pickerWrapper.updatePicker(s.Picker) | ||||
| 	ccb.cc.csMgr.updateState(s.ConnectivityState) | ||||
| } | ||||
| 
 | ||||
| func (ccb *ccBalancerWrapper) ResolveNow(o resolver.ResolveNowOptions) { | ||||
| 	if ccb.isIdleOrClosed() { | ||||
| 	ccb.cc.mu.RLock() | ||||
| 	defer ccb.cc.mu.RUnlock() | ||||
| 
 | ||||
| 	ccb.mu.Lock() | ||||
| 	if ccb.closed { | ||||
| 		ccb.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	ccb.cc.resolveNow(o) | ||||
| 	ccb.mu.Unlock() | ||||
| 	ccb.cc.resolveNowLocked(o) | ||||
| } | ||||
| 
 | ||||
| func (ccb *ccBalancerWrapper) Target() string { | ||||
|  | @ -364,6 +289,20 @@ type acBalancerWrapper struct { | |||
| 	producers map[balancer.ProducerBuilder]*refCountedProducer | ||||
| } | ||||
| 
 | ||||
| // updateState is invoked by grpc to push a subConn state update to the
 | ||||
| // underlying balancer.
 | ||||
| func (acbw *acBalancerWrapper) updateState(s connectivity.State, err error) { | ||||
| 	acbw.ccb.serializer.Schedule(func(ctx context.Context) { | ||||
| 		if ctx.Err() != nil || acbw.ccb.balancer == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		// Even though it is optional for balancers, gracefulswitch ensures
 | ||||
| 		// opts.StateListener is set, so this cannot ever be nil.
 | ||||
| 		// TODO: delete this comment when UpdateSubConnState is removed.
 | ||||
| 		acbw.stateListener(balancer.SubConnState{ConnectivityState: s, ConnectionError: err}) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (acbw *acBalancerWrapper) String() string { | ||||
| 	return fmt.Sprintf("SubConn(id:%d)", acbw.ac.channelzID.Int()) | ||||
| } | ||||
|  | @ -377,20 +316,7 @@ func (acbw *acBalancerWrapper) Connect() { | |||
| } | ||||
| 
 | ||||
| func (acbw *acBalancerWrapper) Shutdown() { | ||||
| 	ccb := acbw.ccb | ||||
| 	if ccb.isIdleOrClosed() { | ||||
| 		// It it safe to ignore this call when the balancer is closed or in idle
 | ||||
| 		// because the ClientConn takes care of closing the connections.
 | ||||
| 		//
 | ||||
| 		// Not returning early from here when the balancer is closed or in idle
 | ||||
| 		// leads to a deadlock though, because of the following sequence of
 | ||||
| 		// calls when holding cc.mu:
 | ||||
| 		// cc.exitIdleMode --> ccb.enterIdleMode --> gsw.Close -->
 | ||||
| 		// ccb.RemoveAddrConn --> cc.removeAddrConn
 | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	ccb.cc.removeAddrConn(acbw.ac, errConnDrain) | ||||
| 	acbw.ccb.cc.removeAddrConn(acbw.ac, errConnDrain) | ||||
| } | ||||
| 
 | ||||
| // NewStream begins a streaming RPC on the addrConn.  If the addrConn is not
 | ||||
|  | @ -33,9 +33,7 @@ import ( | |||
| 	"google.golang.org/grpc/balancer/base" | ||||
| 	"google.golang.org/grpc/codes" | ||||
| 	"google.golang.org/grpc/connectivity" | ||||
| 	"google.golang.org/grpc/credentials" | ||||
| 	"google.golang.org/grpc/internal" | ||||
| 	"google.golang.org/grpc/internal/backoff" | ||||
| 	"google.golang.org/grpc/internal/channelz" | ||||
| 	"google.golang.org/grpc/internal/grpcsync" | ||||
| 	"google.golang.org/grpc/internal/idle" | ||||
|  | @ -48,9 +46,9 @@ import ( | |||
| 	"google.golang.org/grpc/status" | ||||
| 
 | ||||
| 	_ "google.golang.org/grpc/balancer/roundrobin"           // To register roundrobin.
 | ||||
| 	_ "google.golang.org/grpc/internal/resolver/dns"         // To register dns resolver.
 | ||||
| 	_ "google.golang.org/grpc/internal/resolver/passthrough" // To register passthrough resolver.
 | ||||
| 	_ "google.golang.org/grpc/internal/resolver/unix"        // To register unix resolver.
 | ||||
| 	_ "google.golang.org/grpc/resolver/dns"                  // To register dns resolver.
 | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
|  | @ -119,23 +117,8 @@ func (dcs *defaultConfigSelector) SelectConfig(rpcInfo iresolver.RPCInfo) (*ires | |||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| // DialContext creates a client connection to the given target. By default, it's
 | ||||
| // a non-blocking dial (the function won't wait for connections to be
 | ||||
| // established, and connecting happens in the background). To make it a blocking
 | ||||
| // dial, use WithBlock() dial option.
 | ||||
| //
 | ||||
| // In the non-blocking case, the ctx does not act against the connection. It
 | ||||
| // only controls the setup steps.
 | ||||
| //
 | ||||
| // In the blocking case, ctx can be used to cancel or expire the pending
 | ||||
| // connection. Once this function returns, the cancellation and expiration of
 | ||||
| // ctx will be noop. Users should call ClientConn.Close to terminate all the
 | ||||
| // pending operations after this function returns.
 | ||||
| //
 | ||||
| // The target name syntax is defined in
 | ||||
| // https://github.com/grpc/grpc/blob/master/doc/naming.md.
 | ||||
| // e.g. to use dns resolver, a "dns:///" prefix should be applied to the target.
 | ||||
| func DialContext(ctx context.Context, target string, opts ...DialOption) (conn *ClientConn, err error) { | ||||
| // newClient returns a new client in idle mode.
 | ||||
| func newClient(target string, opts ...DialOption) (conn *ClientConn, err error) { | ||||
| 	cc := &ClientConn{ | ||||
| 		target: target, | ||||
| 		conns:  make(map[*addrConn]struct{}), | ||||
|  | @ -143,23 +126,11 @@ func DialContext(ctx context.Context, target string, opts ...DialOption) (conn * | |||
| 		czData: new(channelzData), | ||||
| 	} | ||||
| 
 | ||||
| 	// We start the channel off in idle mode, but kick it out of idle at the end
 | ||||
| 	// of this method, instead of waiting for the first RPC. Other gRPC
 | ||||
| 	// implementations do wait for the first RPC to kick the channel out of
 | ||||
| 	// idle. But doing so would be a major behavior change for our users who are
 | ||||
| 	// used to seeing the channel active after Dial.
 | ||||
| 	//
 | ||||
| 	// Taking this approach of kicking it out of idle at the end of this method
 | ||||
| 	// allows us to share the code between channel creation and exiting idle
 | ||||
| 	// mode. This will also make it easy for us to switch to starting the
 | ||||
| 	// channel off in idle, if at all we ever get to do that.
 | ||||
| 	cc.idlenessState = ccIdlenessStateIdle | ||||
| 
 | ||||
| 	cc.retryThrottler.Store((*retryThrottler)(nil)) | ||||
| 	cc.safeConfigSelector.UpdateConfigSelector(&defaultConfigSelector{nil}) | ||||
| 	cc.ctx, cc.cancel = context.WithCancel(context.Background()) | ||||
| 	cc.exitIdleCond = sync.NewCond(&cc.mu) | ||||
| 
 | ||||
| 	// Apply dial options.
 | ||||
| 	disableGlobalOpts := false | ||||
| 	for _, opt := range opts { | ||||
| 		if _, ok := opt.(*disableGlobalDialOptions); ok { | ||||
|  | @ -177,21 +148,9 @@ func DialContext(ctx context.Context, target string, opts ...DialOption) (conn * | |||
| 	for _, opt := range opts { | ||||
| 		opt.apply(&cc.dopts) | ||||
| 	} | ||||
| 
 | ||||
| 	chainUnaryClientInterceptors(cc) | ||||
| 	chainStreamClientInterceptors(cc) | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		if err != nil { | ||||
| 			cc.Close() | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	// Register ClientConn with channelz.
 | ||||
| 	cc.channelzRegistration(target) | ||||
| 
 | ||||
| 	cc.csMgr = newConnectivityStateManager(cc.ctx, cc.channelzID) | ||||
| 
 | ||||
| 	if err := cc.validateTransportCredentials(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | @ -205,10 +164,80 @@ func DialContext(ctx context.Context, target string, opts ...DialOption) (conn * | |||
| 	} | ||||
| 	cc.mkp = cc.dopts.copts.KeepaliveParams | ||||
| 
 | ||||
| 	if cc.dopts.copts.UserAgent != "" { | ||||
| 		cc.dopts.copts.UserAgent += " " + grpcUA | ||||
| 	} else { | ||||
| 		cc.dopts.copts.UserAgent = grpcUA | ||||
| 	// Register ClientConn with channelz.
 | ||||
| 	cc.channelzRegistration(target) | ||||
| 
 | ||||
| 	// TODO: Ideally it should be impossible to error from this function after
 | ||||
| 	// channelz registration.  This will require removing some channelz logs
 | ||||
| 	// from the following functions that can error.  Errors can be returned to
 | ||||
| 	// the user, and successful logs can be emitted here, after the checks have
 | ||||
| 	// passed and channelz is subsequently registered.
 | ||||
| 
 | ||||
| 	// Determine the resolver to use.
 | ||||
| 	if err := cc.parseTargetAndFindResolver(); err != nil { | ||||
| 		channelz.RemoveEntry(cc.channelzID) | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err = cc.determineAuthority(); err != nil { | ||||
| 		channelz.RemoveEntry(cc.channelzID) | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	cc.csMgr = newConnectivityStateManager(cc.ctx, cc.channelzID) | ||||
| 	cc.pickerWrapper = newPickerWrapper(cc.dopts.copts.StatsHandlers) | ||||
| 
 | ||||
| 	cc.initIdleStateLocked() // Safe to call without the lock, since nothing else has a reference to cc.
 | ||||
| 	cc.idlenessMgr = idle.NewManager((*idler)(cc), cc.dopts.idleTimeout) | ||||
| 	return cc, nil | ||||
| } | ||||
| 
 | ||||
| // DialContext creates a client connection to the given target. By default, it's
 | ||||
| // a non-blocking dial (the function won't wait for connections to be
 | ||||
| // established, and connecting happens in the background). To make it a blocking
 | ||||
| // dial, use WithBlock() dial option.
 | ||||
| //
 | ||||
| // In the non-blocking case, the ctx does not act against the connection. It
 | ||||
| // only controls the setup steps.
 | ||||
| //
 | ||||
| // In the blocking case, ctx can be used to cancel or expire the pending
 | ||||
| // connection. Once this function returns, the cancellation and expiration of
 | ||||
| // ctx will be noop. Users should call ClientConn.Close to terminate all the
 | ||||
| // pending operations after this function returns.
 | ||||
| //
 | ||||
| // The target name syntax is defined in
 | ||||
| // https://github.com/grpc/grpc/blob/master/doc/naming.md.
 | ||||
| // e.g. to use dns resolver, a "dns:///" prefix should be applied to the target.
 | ||||
| func DialContext(ctx context.Context, target string, opts ...DialOption) (conn *ClientConn, err error) { | ||||
| 	cc, err := newClient(target, opts...) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	// We start the channel off in idle mode, but kick it out of idle now,
 | ||||
| 	// instead of waiting for the first RPC. Other gRPC implementations do wait
 | ||||
| 	// for the first RPC to kick the channel out of idle. But doing so would be
 | ||||
| 	// a major behavior change for our users who are used to seeing the channel
 | ||||
| 	// active after Dial.
 | ||||
| 	//
 | ||||
| 	// Taking this approach of kicking it out of idle at the end of this method
 | ||||
| 	// allows us to share the code between channel creation and exiting idle
 | ||||
| 	// mode. This will also make it easy for us to switch to starting the
 | ||||
| 	// channel off in idle, i.e. by making newClient exported.
 | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		if err != nil { | ||||
| 			cc.Close() | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	// This creates the name resolver, load balancer, etc.
 | ||||
| 	if err := cc.idlenessMgr.ExitIdleMode(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	// Return now for non-blocking dials.
 | ||||
| 	if !cc.dopts.block { | ||||
| 		return cc, nil | ||||
| 	} | ||||
| 
 | ||||
| 	if cc.dopts.timeout > 0 { | ||||
|  | @ -231,49 +260,6 @@ func DialContext(ctx context.Context, target string, opts ...DialOption) (conn * | |||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	if cc.dopts.bs == nil { | ||||
| 		cc.dopts.bs = backoff.DefaultExponential | ||||
| 	} | ||||
| 
 | ||||
| 	// Determine the resolver to use.
 | ||||
| 	if err := cc.parseTargetAndFindResolver(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if err = cc.determineAuthority(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	if cc.dopts.scChan != nil { | ||||
| 		// Blocking wait for the initial service config.
 | ||||
| 		select { | ||||
| 		case sc, ok := <-cc.dopts.scChan: | ||||
| 			if ok { | ||||
| 				cc.sc = &sc | ||||
| 				cc.safeConfigSelector.UpdateConfigSelector(&defaultConfigSelector{&sc}) | ||||
| 			} | ||||
| 		case <-ctx.Done(): | ||||
| 			return nil, ctx.Err() | ||||
| 		} | ||||
| 	} | ||||
| 	if cc.dopts.scChan != nil { | ||||
| 		go cc.scWatcher() | ||||
| 	} | ||||
| 
 | ||||
| 	// This creates the name resolver, load balancer, blocking picker etc.
 | ||||
| 	if err := cc.exitIdleMode(); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	// Configure idleness support with configured idle timeout or default idle
 | ||||
| 	// timeout duration. Idleness can be explicitly disabled by the user, by
 | ||||
| 	// setting the dial option to 0.
 | ||||
| 	cc.idlenessMgr = idle.NewManager(idle.ManagerOptions{Enforcer: (*idler)(cc), Timeout: cc.dopts.idleTimeout, Logger: logger}) | ||||
| 
 | ||||
| 	// Return early for non-blocking dials.
 | ||||
| 	if !cc.dopts.block { | ||||
| 		return cc, nil | ||||
| 	} | ||||
| 
 | ||||
| 	// A blocking dial blocks until the clientConn is ready.
 | ||||
| 	for { | ||||
| 		s := cc.GetState() | ||||
|  | @ -320,8 +306,8 @@ func (cc *ClientConn) addTraceEvent(msg string) { | |||
| 
 | ||||
| type idler ClientConn | ||||
| 
 | ||||
| func (i *idler) EnterIdleMode() error { | ||||
| 	return (*ClientConn)(i).enterIdleMode() | ||||
| func (i *idler) EnterIdleMode() { | ||||
| 	(*ClientConn)(i).enterIdleMode() | ||||
| } | ||||
| 
 | ||||
| func (i *idler) ExitIdleMode() error { | ||||
|  | @ -329,117 +315,71 @@ func (i *idler) ExitIdleMode() error { | |||
| } | ||||
| 
 | ||||
| // exitIdleMode moves the channel out of idle mode by recreating the name
 | ||||
| // resolver and load balancer.
 | ||||
| func (cc *ClientConn) exitIdleMode() error { | ||||
| // resolver and load balancer.  This should never be called directly; use
 | ||||
| // cc.idlenessMgr.ExitIdleMode instead.
 | ||||
| func (cc *ClientConn) exitIdleMode() (err error) { | ||||
| 	cc.mu.Lock() | ||||
| 	if cc.conns == nil { | ||||
| 		cc.mu.Unlock() | ||||
| 		return errConnClosing | ||||
| 	} | ||||
| 	if cc.idlenessState != ccIdlenessStateIdle { | ||||
| 		channelz.Infof(logger, cc.channelzID, "ClientConn asked to exit idle mode, current mode is %v", cc.idlenessState) | ||||
| 		cc.mu.Unlock() | ||||
| 		return nil | ||||
| 	} | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		// When Close() and exitIdleMode() race against each other, one of the
 | ||||
| 		// following two can happen:
 | ||||
| 		// - Close() wins the race and runs first. exitIdleMode() runs after, and
 | ||||
| 		//   sees that the ClientConn is already closed and hence returns early.
 | ||||
| 		// - exitIdleMode() wins the race and runs first and recreates the balancer
 | ||||
| 		//   and releases the lock before recreating the resolver. If Close() runs
 | ||||
| 		//   in this window, it will wait for exitIdleMode to complete.
 | ||||
| 		//
 | ||||
| 		// We achieve this synchronization using the below condition variable.
 | ||||
| 		cc.mu.Lock() | ||||
| 		cc.idlenessState = ccIdlenessStateActive | ||||
| 		cc.exitIdleCond.Signal() | ||||
| 		cc.mu.Unlock() | ||||
| 	}() | ||||
| 
 | ||||
| 	cc.idlenessState = ccIdlenessStateExitingIdle | ||||
| 	exitedIdle := false | ||||
| 	if cc.blockingpicker == nil { | ||||
| 		cc.blockingpicker = newPickerWrapper(cc.dopts.copts.StatsHandlers) | ||||
| 	} else { | ||||
| 		cc.blockingpicker.exitIdleMode() | ||||
| 		exitedIdle = true | ||||
| 	} | ||||
| 
 | ||||
| 	var credsClone credentials.TransportCredentials | ||||
| 	if creds := cc.dopts.copts.TransportCredentials; creds != nil { | ||||
| 		credsClone = creds.Clone() | ||||
| 	} | ||||
| 	if cc.balancerWrapper == nil { | ||||
| 		cc.balancerWrapper = newCCBalancerWrapper(cc, balancer.BuildOptions{ | ||||
| 			DialCreds:        credsClone, | ||||
| 			CredsBundle:      cc.dopts.copts.CredsBundle, | ||||
| 			Dialer:           cc.dopts.copts.Dialer, | ||||
| 			Authority:        cc.authority, | ||||
| 			CustomUserAgent:  cc.dopts.copts.UserAgent, | ||||
| 			ChannelzParentID: cc.channelzID, | ||||
| 			Target:           cc.parsedTarget, | ||||
| 		}) | ||||
| 	} else { | ||||
| 		cc.balancerWrapper.exitIdleMode() | ||||
| 	} | ||||
| 	cc.firstResolveEvent = grpcsync.NewEvent() | ||||
| 	cc.mu.Unlock() | ||||
| 
 | ||||
| 	// This needs to be called without cc.mu because this builds a new resolver
 | ||||
| 	// which might update state or report error inline which needs to be handled
 | ||||
| 	// by cc.updateResolverState() which also grabs cc.mu.
 | ||||
| 	if err := cc.initResolverWrapper(credsClone); err != nil { | ||||
| 	// which might update state or report error inline, which would then need to
 | ||||
| 	// acquire cc.mu.
 | ||||
| 	if err := cc.resolverWrapper.start(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	if exitedIdle { | ||||
| 		cc.addTraceEvent("exiting idle mode") | ||||
| 	} | ||||
| 	cc.addTraceEvent("exiting idle mode") | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // enterIdleMode puts the channel in idle mode, and as part of it shuts down the
 | ||||
| // name resolver, load balancer and any subchannels.
 | ||||
| func (cc *ClientConn) enterIdleMode() error { | ||||
| 	cc.mu.Lock() | ||||
| 	defer cc.mu.Unlock() | ||||
| 
 | ||||
| 	if cc.conns == nil { | ||||
| 		return ErrClientConnClosing | ||||
| 	} | ||||
| 	if cc.idlenessState != ccIdlenessStateActive { | ||||
| 		channelz.Warningf(logger, cc.channelzID, "ClientConn asked to enter idle mode, current mode is %v", cc.idlenessState) | ||||
| 		return nil | ||||
| 	} | ||||
| 
 | ||||
| // initIdleStateLocked initializes common state to how it should be while idle.
 | ||||
| func (cc *ClientConn) initIdleStateLocked() { | ||||
| 	cc.resolverWrapper = newCCResolverWrapper(cc) | ||||
| 	cc.balancerWrapper = newCCBalancerWrapper(cc) | ||||
| 	cc.firstResolveEvent = grpcsync.NewEvent() | ||||
| 	// cc.conns == nil is a proxy for the ClientConn being closed. So, instead
 | ||||
| 	// of setting it to nil here, we recreate the map. This also means that we
 | ||||
| 	// don't have to do this when exiting idle mode.
 | ||||
| 	conns := cc.conns | ||||
| 	cc.conns = make(map[*addrConn]struct{}) | ||||
| } | ||||
| 
 | ||||
| 	// TODO: Currently, we close the resolver wrapper upon entering idle mode
 | ||||
| 	// and create a new one upon exiting idle mode. This means that the
 | ||||
| 	// `cc.resolverWrapper` field would be overwritten everytime we exit idle
 | ||||
| 	// mode. While this means that we need to hold `cc.mu` when accessing
 | ||||
| 	// `cc.resolverWrapper`, it makes the code simpler in the wrapper. We should
 | ||||
| 	// try to do the same for the balancer and picker wrappers too.
 | ||||
| 	cc.resolverWrapper.close() | ||||
| 	cc.blockingpicker.enterIdleMode() | ||||
| 	cc.balancerWrapper.enterIdleMode() | ||||
| // enterIdleMode puts the channel in idle mode, and as part of it shuts down the
 | ||||
| // name resolver, load balancer, and any subchannels.  This should never be
 | ||||
| // called directly; use cc.idlenessMgr.EnterIdleMode instead.
 | ||||
| func (cc *ClientConn) enterIdleMode() { | ||||
| 	cc.mu.Lock() | ||||
| 
 | ||||
| 	if cc.conns == nil { | ||||
| 		cc.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	conns := cc.conns | ||||
| 
 | ||||
| 	rWrapper := cc.resolverWrapper | ||||
| 	rWrapper.close() | ||||
| 	cc.pickerWrapper.reset() | ||||
| 	bWrapper := cc.balancerWrapper | ||||
| 	bWrapper.close() | ||||
| 	cc.csMgr.updateState(connectivity.Idle) | ||||
| 	cc.idlenessState = ccIdlenessStateIdle | ||||
| 	cc.addTraceEvent("entering idle mode") | ||||
| 
 | ||||
| 	go func() { | ||||
| 		for ac := range conns { | ||||
| 			ac.tearDown(errConnIdling) | ||||
| 		} | ||||
| 	}() | ||||
| 	cc.initIdleStateLocked() | ||||
| 
 | ||||
| 	return nil | ||||
| 	cc.mu.Unlock() | ||||
| 
 | ||||
| 	// Block until the name resolver and LB policy are closed.
 | ||||
| 	<-rWrapper.serializer.Done() | ||||
| 	<-bWrapper.serializer.Done() | ||||
| 
 | ||||
| 	// Close all subchannels after the LB policy is closed.
 | ||||
| 	for ac := range conns { | ||||
| 		ac.tearDown(errConnIdling) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // validateTransportCredentials performs a series of checks on the configured
 | ||||
|  | @ -649,66 +589,35 @@ type ClientConn struct { | |||
| 	dopts           dialOptions          // Default and user specified dial options.
 | ||||
| 	channelzID      *channelz.Identifier // Channelz identifier for the channel.
 | ||||
| 	resolverBuilder resolver.Builder     // See parseTargetAndFindResolver().
 | ||||
| 	balancerWrapper *ccBalancerWrapper   // Uses gracefulswitch.balancer underneath.
 | ||||
| 	idlenessMgr     idle.Manager | ||||
| 	idlenessMgr     *idle.Manager | ||||
| 
 | ||||
| 	// The following provide their own synchronization, and therefore don't
 | ||||
| 	// require cc.mu to be held to access them.
 | ||||
| 	csMgr              *connectivityStateManager | ||||
| 	blockingpicker     *pickerWrapper | ||||
| 	pickerWrapper      *pickerWrapper | ||||
| 	safeConfigSelector iresolver.SafeConfigSelector | ||||
| 	czData             *channelzData | ||||
| 	retryThrottler     atomic.Value // Updated from service config.
 | ||||
| 
 | ||||
| 	// firstResolveEvent is used to track whether the name resolver sent us at
 | ||||
| 	// least one update. RPCs block on this event.
 | ||||
| 	firstResolveEvent *grpcsync.Event | ||||
| 
 | ||||
| 	// mu protects the following fields.
 | ||||
| 	// TODO: split mu so the same mutex isn't used for everything.
 | ||||
| 	mu              sync.RWMutex | ||||
| 	resolverWrapper *ccResolverWrapper         // Initialized in Dial; cleared in Close.
 | ||||
| 	resolverWrapper *ccResolverWrapper         // Always recreated whenever entering idle to simplify Close.
 | ||||
| 	balancerWrapper *ccBalancerWrapper         // Always recreated whenever entering idle to simplify Close.
 | ||||
| 	sc              *ServiceConfig             // Latest service config received from the resolver.
 | ||||
| 	conns           map[*addrConn]struct{}     // Set to nil on close.
 | ||||
| 	mkp             keepalive.ClientParameters // May be updated upon receipt of a GoAway.
 | ||||
| 	idlenessState   ccIdlenessState            // Tracks idleness state of the channel.
 | ||||
| 	exitIdleCond    *sync.Cond                 // Signalled when channel exits idle.
 | ||||
| 	// firstResolveEvent is used to track whether the name resolver sent us at
 | ||||
| 	// least one update. RPCs block on this event.  May be accessed without mu
 | ||||
| 	// if we know we cannot be asked to enter idle mode while accessing it (e.g.
 | ||||
| 	// when the idle manager has already been closed, or if we are already
 | ||||
| 	// entering idle mode).
 | ||||
| 	firstResolveEvent *grpcsync.Event | ||||
| 
 | ||||
| 	lceMu               sync.Mutex // protects lastConnectionError
 | ||||
| 	lastConnectionError error | ||||
| } | ||||
| 
 | ||||
| // ccIdlenessState tracks the idleness state of the channel.
 | ||||
| //
 | ||||
| // Channels start off in `active` and move to `idle` after a period of
 | ||||
| // inactivity. When moving back to `active` upon an incoming RPC, they
 | ||||
| // transition through `exiting_idle`. This state is useful for synchronization
 | ||||
| // with Close().
 | ||||
| //
 | ||||
| // This state tracking is mostly for self-protection. The idlenessManager is
 | ||||
| // expected to keep track of the state as well, and is expected not to call into
 | ||||
| // the ClientConn unnecessarily.
 | ||||
| type ccIdlenessState int8 | ||||
| 
 | ||||
| const ( | ||||
| 	ccIdlenessStateActive ccIdlenessState = iota | ||||
| 	ccIdlenessStateIdle | ||||
| 	ccIdlenessStateExitingIdle | ||||
| ) | ||||
| 
 | ||||
| func (s ccIdlenessState) String() string { | ||||
| 	switch s { | ||||
| 	case ccIdlenessStateActive: | ||||
| 		return "active" | ||||
| 	case ccIdlenessStateIdle: | ||||
| 		return "idle" | ||||
| 	case ccIdlenessStateExitingIdle: | ||||
| 		return "exitingIdle" | ||||
| 	default: | ||||
| 		return "unknown" | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // WaitForStateChange waits until the connectivity.State of ClientConn changes from sourceState or
 | ||||
| // ctx expires. A true value is returned in former case and false in latter.
 | ||||
| //
 | ||||
|  | @ -748,29 +657,15 @@ func (cc *ClientConn) GetState() connectivity.State { | |||
| // Notice: This API is EXPERIMENTAL and may be changed or removed in a later
 | ||||
| // release.
 | ||||
| func (cc *ClientConn) Connect() { | ||||
| 	cc.exitIdleMode() | ||||
| 	if err := cc.idlenessMgr.ExitIdleMode(); err != nil { | ||||
| 		cc.addTraceEvent(err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	// If the ClientConn was not in idle mode, we need to call ExitIdle on the
 | ||||
| 	// LB policy so that connections can be created.
 | ||||
| 	cc.balancerWrapper.exitIdleMode() | ||||
| } | ||||
| 
 | ||||
| func (cc *ClientConn) scWatcher() { | ||||
| 	for { | ||||
| 		select { | ||||
| 		case sc, ok := <-cc.dopts.scChan: | ||||
| 			if !ok { | ||||
| 				return | ||||
| 			} | ||||
| 			cc.mu.Lock() | ||||
| 			// TODO: load balance policy runtime change is ignored.
 | ||||
| 			// We may revisit this decision in the future.
 | ||||
| 			cc.sc = &sc | ||||
| 			cc.safeConfigSelector.UpdateConfigSelector(&defaultConfigSelector{&sc}) | ||||
| 			cc.mu.Unlock() | ||||
| 		case <-cc.ctx.Done(): | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	cc.mu.Lock() | ||||
| 	cc.balancerWrapper.exitIdle() | ||||
| 	cc.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| // waitForResolvedAddrs blocks until the resolver has provided addresses or the
 | ||||
|  | @ -804,11 +699,11 @@ func init() { | |||
| 	internal.SubscribeToConnectivityStateChanges = func(cc *ClientConn, s grpcsync.Subscriber) func() { | ||||
| 		return cc.csMgr.pubSub.Subscribe(s) | ||||
| 	} | ||||
| 	internal.EnterIdleModeForTesting = func(cc *ClientConn) error { | ||||
| 		return cc.enterIdleMode() | ||||
| 	internal.EnterIdleModeForTesting = func(cc *ClientConn) { | ||||
| 		cc.idlenessMgr.EnterIdleModeForTesting() | ||||
| 	} | ||||
| 	internal.ExitIdleModeForTesting = func(cc *ClientConn) error { | ||||
| 		return cc.exitIdleMode() | ||||
| 		return cc.idlenessMgr.ExitIdleMode() | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | @ -824,9 +719,8 @@ func (cc *ClientConn) maybeApplyDefaultServiceConfig(addrs []resolver.Address) { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (cc *ClientConn) updateResolverState(s resolver.State, err error) error { | ||||
| func (cc *ClientConn) updateResolverStateAndUnlock(s resolver.State, err error) error { | ||||
| 	defer cc.firstResolveEvent.Fire() | ||||
| 	cc.mu.Lock() | ||||
| 	// Check if the ClientConn is already closed. Some fields (e.g.
 | ||||
| 	// balancerWrapper) are set to nil when closing the ClientConn, and could
 | ||||
| 	// cause nil pointer panic if we don't have this check.
 | ||||
|  | @ -872,7 +766,7 @@ func (cc *ClientConn) updateResolverState(s resolver.State, err error) error { | |||
| 			if cc.sc == nil { | ||||
| 				// Apply the failing LB only if we haven't received valid service config
 | ||||
| 				// from the name resolver in the past.
 | ||||
| 				cc.applyFailingLB(s.ServiceConfig) | ||||
| 				cc.applyFailingLBLocked(s.ServiceConfig) | ||||
| 				cc.mu.Unlock() | ||||
| 				return ret | ||||
| 			} | ||||
|  | @ -894,15 +788,13 @@ func (cc *ClientConn) updateResolverState(s resolver.State, err error) error { | |||
| 	return ret | ||||
| } | ||||
| 
 | ||||
| // applyFailingLB is akin to configuring an LB policy on the channel which
 | ||||
| // applyFailingLBLocked is akin to configuring an LB policy on the channel which
 | ||||
| // always fails RPCs. Here, an actual LB policy is not configured, but an always
 | ||||
| // erroring picker is configured, which returns errors with information about
 | ||||
| // what was invalid in the received service config. A config selector with no
 | ||||
| // service config is configured, and the connectivity state of the channel is
 | ||||
| // set to TransientFailure.
 | ||||
| //
 | ||||
| // Caller must hold cc.mu.
 | ||||
| func (cc *ClientConn) applyFailingLB(sc *serviceconfig.ParseResult) { | ||||
| func (cc *ClientConn) applyFailingLBLocked(sc *serviceconfig.ParseResult) { | ||||
| 	var err error | ||||
| 	if sc.Err != nil { | ||||
| 		err = status.Errorf(codes.Unavailable, "error parsing service config: %v", sc.Err) | ||||
|  | @ -910,14 +802,10 @@ func (cc *ClientConn) applyFailingLB(sc *serviceconfig.ParseResult) { | |||
| 		err = status.Errorf(codes.Unavailable, "illegal service config type: %T", sc.Config) | ||||
| 	} | ||||
| 	cc.safeConfigSelector.UpdateConfigSelector(&defaultConfigSelector{nil}) | ||||
| 	cc.blockingpicker.updatePicker(base.NewErrPicker(err)) | ||||
| 	cc.pickerWrapper.updatePicker(base.NewErrPicker(err)) | ||||
| 	cc.csMgr.updateState(connectivity.TransientFailure) | ||||
| } | ||||
| 
 | ||||
| func (cc *ClientConn) handleSubConnStateChange(sc balancer.SubConn, s connectivity.State, err error) { | ||||
| 	cc.balancerWrapper.updateSubConnState(sc, s, err) | ||||
| } | ||||
| 
 | ||||
| // Makes a copy of the input addresses slice and clears out the balancer
 | ||||
| // attributes field. Addresses are passed during subconn creation and address
 | ||||
| // update operations. In both cases, we will clear the balancer attributes by
 | ||||
|  | @ -932,10 +820,14 @@ func copyAddressesWithoutBalancerAttributes(in []resolver.Address) []resolver.Ad | |||
| 	return out | ||||
| } | ||||
| 
 | ||||
| // newAddrConn creates an addrConn for addrs and adds it to cc.conns.
 | ||||
| // newAddrConnLocked creates an addrConn for addrs and adds it to cc.conns.
 | ||||
| //
 | ||||
| // Caller needs to make sure len(addrs) > 0.
 | ||||
| func (cc *ClientConn) newAddrConn(addrs []resolver.Address, opts balancer.NewSubConnOptions) (*addrConn, error) { | ||||
| func (cc *ClientConn) newAddrConnLocked(addrs []resolver.Address, opts balancer.NewSubConnOptions) (*addrConn, error) { | ||||
| 	if cc.conns == nil { | ||||
| 		return nil, ErrClientConnClosing | ||||
| 	} | ||||
| 
 | ||||
| 	ac := &addrConn{ | ||||
| 		state:        connectivity.Idle, | ||||
| 		cc:           cc, | ||||
|  | @ -947,12 +839,6 @@ func (cc *ClientConn) newAddrConn(addrs []resolver.Address, opts balancer.NewSub | |||
| 		stateChan:    make(chan struct{}), | ||||
| 	} | ||||
| 	ac.ctx, ac.cancel = context.WithCancel(cc.ctx) | ||||
| 	// Track ac in cc. This needs to be done before any getTransport(...) is called.
 | ||||
| 	cc.mu.Lock() | ||||
| 	defer cc.mu.Unlock() | ||||
| 	if cc.conns == nil { | ||||
| 		return nil, ErrClientConnClosing | ||||
| 	} | ||||
| 
 | ||||
| 	var err error | ||||
| 	ac.channelzID, err = channelz.RegisterSubChannel(ac, cc.channelzID, "") | ||||
|  | @ -968,6 +854,7 @@ func (cc *ClientConn) newAddrConn(addrs []resolver.Address, opts balancer.NewSub | |||
| 		}, | ||||
| 	}) | ||||
| 
 | ||||
| 	// Track ac in cc. This needs to be done before any getTransport(...) is called.
 | ||||
| 	cc.conns[ac] = struct{}{} | ||||
| 	return ac, nil | ||||
| } | ||||
|  | @ -1174,7 +1061,7 @@ func (cc *ClientConn) healthCheckConfig() *healthCheckConfig { | |||
| } | ||||
| 
 | ||||
| func (cc *ClientConn) getTransport(ctx context.Context, failfast bool, method string) (transport.ClientTransport, balancer.PickResult, error) { | ||||
| 	return cc.blockingpicker.pick(ctx, failfast, balancer.PickInfo{ | ||||
| 	return cc.pickerWrapper.pick(ctx, failfast, balancer.PickInfo{ | ||||
| 		Ctx:            ctx, | ||||
| 		FullMethodName: method, | ||||
| 	}) | ||||
|  | @ -1216,12 +1103,12 @@ func (cc *ClientConn) applyServiceConfigAndBalancer(sc *ServiceConfig, configSel | |||
| 
 | ||||
| func (cc *ClientConn) resolveNow(o resolver.ResolveNowOptions) { | ||||
| 	cc.mu.RLock() | ||||
| 	r := cc.resolverWrapper | ||||
| 	cc.resolverWrapper.resolveNow(o) | ||||
| 	cc.mu.RUnlock() | ||||
| 	if r == nil { | ||||
| 		return | ||||
| 	} | ||||
| 	go r.resolveNow(o) | ||||
| } | ||||
| 
 | ||||
| func (cc *ClientConn) resolveNowLocked(o resolver.ResolveNowOptions) { | ||||
| 	cc.resolverWrapper.resolveNow(o) | ||||
| } | ||||
| 
 | ||||
| // ResetConnectBackoff wakes up all subchannels in transient failure and causes
 | ||||
|  | @ -1253,40 +1140,32 @@ func (cc *ClientConn) Close() error { | |||
| 		<-cc.csMgr.pubSub.Done() | ||||
| 	}() | ||||
| 
 | ||||
| 	// Prevent calls to enter/exit idle immediately, and ensure we are not
 | ||||
| 	// currently entering/exiting idle mode.
 | ||||
| 	cc.idlenessMgr.Close() | ||||
| 
 | ||||
| 	cc.mu.Lock() | ||||
| 	if cc.conns == nil { | ||||
| 		cc.mu.Unlock() | ||||
| 		return ErrClientConnClosing | ||||
| 	} | ||||
| 
 | ||||
| 	for cc.idlenessState == ccIdlenessStateExitingIdle { | ||||
| 		cc.exitIdleCond.Wait() | ||||
| 	} | ||||
| 
 | ||||
| 	conns := cc.conns | ||||
| 	cc.conns = nil | ||||
| 	cc.csMgr.updateState(connectivity.Shutdown) | ||||
| 
 | ||||
| 	pWrapper := cc.blockingpicker | ||||
| 	rWrapper := cc.resolverWrapper | ||||
| 	bWrapper := cc.balancerWrapper | ||||
| 	idlenessMgr := cc.idlenessMgr | ||||
| 	// We can safely unlock and continue to access all fields now as
 | ||||
| 	// cc.conns==nil, preventing any further operations on cc.
 | ||||
| 	cc.mu.Unlock() | ||||
| 
 | ||||
| 	cc.resolverWrapper.close() | ||||
| 	// The order of closing matters here since the balancer wrapper assumes the
 | ||||
| 	// picker is closed before it is closed.
 | ||||
| 	if pWrapper != nil { | ||||
| 		pWrapper.close() | ||||
| 	} | ||||
| 	if bWrapper != nil { | ||||
| 		bWrapper.close() | ||||
| 	} | ||||
| 	if rWrapper != nil { | ||||
| 		rWrapper.close() | ||||
| 	} | ||||
| 	if idlenessMgr != nil { | ||||
| 		idlenessMgr.Close() | ||||
| 	} | ||||
| 	cc.pickerWrapper.close() | ||||
| 	cc.balancerWrapper.close() | ||||
| 
 | ||||
| 	<-cc.resolverWrapper.serializer.Done() | ||||
| 	<-cc.balancerWrapper.serializer.Done() | ||||
| 
 | ||||
| 	for ac := range conns { | ||||
| 		ac.tearDown(ErrClientConnClosing) | ||||
|  | @ -1307,7 +1186,7 @@ type addrConn struct { | |||
| 
 | ||||
| 	cc     *ClientConn | ||||
| 	dopts  dialOptions | ||||
| 	acbw   balancer.SubConn | ||||
| 	acbw   *acBalancerWrapper | ||||
| 	scopts balancer.NewSubConnOptions | ||||
| 
 | ||||
| 	// transport is set when there's a viable transport (note: ac state may not be READY as LB channel
 | ||||
|  | @ -1345,7 +1224,7 @@ func (ac *addrConn) updateConnectivityState(s connectivity.State, lastErr error) | |||
| 	} else { | ||||
| 		channelz.Infof(logger, ac.channelzID, "Subchannel Connectivity change to %v, last error: %s", s, lastErr) | ||||
| 	} | ||||
| 	ac.cc.handleSubConnStateChange(ac.acbw, s, lastErr) | ||||
| 	ac.acbw.updateState(s, lastErr) | ||||
| } | ||||
| 
 | ||||
| // adjustParams updates parameters used to create transports upon
 | ||||
|  | @ -1849,7 +1728,7 @@ func (cc *ClientConn) parseTargetAndFindResolver() error { | |||
| 	if err != nil { | ||||
| 		channelz.Infof(logger, cc.channelzID, "dial target %q parse failed: %v", cc.target, err) | ||||
| 	} else { | ||||
| 		channelz.Infof(logger, cc.channelzID, "parsed dial target is: %+v", parsedTarget) | ||||
| 		channelz.Infof(logger, cc.channelzID, "parsed dial target is: %#v", parsedTarget) | ||||
| 		rb = cc.getResolver(parsedTarget.URL.Scheme) | ||||
| 		if rb != nil { | ||||
| 			cc.parsedTarget = parsedTarget | ||||
|  | @ -2007,32 +1886,3 @@ func (cc *ClientConn) determineAuthority() error { | |||
| 	channelz.Infof(logger, cc.channelzID, "Channel authority set to %q", cc.authority) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // initResolverWrapper creates a ccResolverWrapper, which builds the name
 | ||||
| // resolver. This method grabs the lock to assign the newly built resolver
 | ||||
| // wrapper to the cc.resolverWrapper field.
 | ||||
| func (cc *ClientConn) initResolverWrapper(creds credentials.TransportCredentials) error { | ||||
| 	rw, err := newCCResolverWrapper(cc, ccResolverWrapperOpts{ | ||||
| 		target:  cc.parsedTarget, | ||||
| 		builder: cc.resolverBuilder, | ||||
| 		bOpts: resolver.BuildOptions{ | ||||
| 			DisableServiceConfig: cc.dopts.disableServiceConfig, | ||||
| 			DialCreds:            creds, | ||||
| 			CredsBundle:          cc.dopts.copts.CredsBundle, | ||||
| 			Dialer:               cc.dopts.copts.Dialer, | ||||
| 		}, | ||||
| 		channelzID: cc.channelzID, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("failed to build resolver: %v", err) | ||||
| 	} | ||||
| 	// Resolver implementations may report state update or error inline when
 | ||||
| 	// built (or right after), and this is handled in cc.updateResolverState.
 | ||||
| 	// Also, an error from the resolver might lead to a re-resolution request
 | ||||
| 	// from the balancer, which is handled in resolveNow() where
 | ||||
| 	// `cc.resolverWrapper` is accessed. Hence, we need to hold the lock here.
 | ||||
| 	cc.mu.Lock() | ||||
| 	cc.resolverWrapper = rw | ||||
| 	cc.mu.Unlock() | ||||
| 	return nil | ||||
| } | ||||
|  |  | |||
|  | @ -25,7 +25,13 @@ import ( | |||
| 	"strconv" | ||||
| ) | ||||
| 
 | ||||
| // A Code is an unsigned 32-bit error code as defined in the gRPC spec.
 | ||||
| // A Code is a status code defined according to the [gRPC documentation].
 | ||||
| //
 | ||||
| // Only the codes defined as consts in this package are valid codes. Do not use
 | ||||
| // other code values.  Behavior of other codes is implementation-specific and
 | ||||
| // interoperability between implementations is not guaranteed.
 | ||||
| //
 | ||||
| // [gRPC documentation]: https://github.com/grpc/grpc/blob/master/doc/statuscodes.md
 | ||||
| type Code uint32 | ||||
| 
 | ||||
| const ( | ||||
|  |  | |||
|  | @ -44,10 +44,25 @@ func (t TLSInfo) AuthType() string { | |||
| 	return "tls" | ||||
| } | ||||
| 
 | ||||
| // cipherSuiteLookup returns the string version of a TLS cipher suite ID.
 | ||||
| func cipherSuiteLookup(cipherSuiteID uint16) string { | ||||
| 	for _, s := range tls.CipherSuites() { | ||||
| 		if s.ID == cipherSuiteID { | ||||
| 			return s.Name | ||||
| 		} | ||||
| 	} | ||||
| 	for _, s := range tls.InsecureCipherSuites() { | ||||
| 		if s.ID == cipherSuiteID { | ||||
| 			return s.Name | ||||
| 		} | ||||
| 	} | ||||
| 	return fmt.Sprintf("unknown ID: %v", cipherSuiteID) | ||||
| } | ||||
| 
 | ||||
| // GetSecurityValue returns security info requested by channelz.
 | ||||
| func (t TLSInfo) GetSecurityValue() ChannelzSecurityValue { | ||||
| 	v := &TLSChannelzSecurityValue{ | ||||
| 		StandardName: cipherSuiteLookup[t.State.CipherSuite], | ||||
| 		StandardName: cipherSuiteLookup(t.State.CipherSuite), | ||||
| 	} | ||||
| 	// Currently there's no way to get LocalCertificate info from tls package.
 | ||||
| 	if len(t.State.PeerCertificates) > 0 { | ||||
|  | @ -138,10 +153,39 @@ func (c *tlsCreds) OverrideServerName(serverNameOverride string) error { | |||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // The following cipher suites are forbidden for use with HTTP/2 by
 | ||||
| // https://datatracker.ietf.org/doc/html/rfc7540#appendix-A
 | ||||
| var tls12ForbiddenCipherSuites = map[uint16]struct{}{ | ||||
| 	tls.TLS_RSA_WITH_AES_128_CBC_SHA:         {}, | ||||
| 	tls.TLS_RSA_WITH_AES_256_CBC_SHA:         {}, | ||||
| 	tls.TLS_RSA_WITH_AES_128_GCM_SHA256:      {}, | ||||
| 	tls.TLS_RSA_WITH_AES_256_GCM_SHA384:      {}, | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA: {}, | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA: {}, | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA:   {}, | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA:   {}, | ||||
| } | ||||
| 
 | ||||
| // NewTLS uses c to construct a TransportCredentials based on TLS.
 | ||||
| func NewTLS(c *tls.Config) TransportCredentials { | ||||
| 	tc := &tlsCreds{credinternal.CloneTLSConfig(c)} | ||||
| 	tc.config.NextProtos = credinternal.AppendH2ToNextProtos(tc.config.NextProtos) | ||||
| 	// If the user did not configure a MinVersion and did not configure a
 | ||||
| 	// MaxVersion < 1.2, use MinVersion=1.2, which is required by
 | ||||
| 	// https://datatracker.ietf.org/doc/html/rfc7540#section-9.2
 | ||||
| 	if tc.config.MinVersion == 0 && (tc.config.MaxVersion == 0 || tc.config.MaxVersion >= tls.VersionTLS12) { | ||||
| 		tc.config.MinVersion = tls.VersionTLS12 | ||||
| 	} | ||||
| 	// If the user did not configure CipherSuites, use all "secure" cipher
 | ||||
| 	// suites reported by the TLS package, but remove some explicitly forbidden
 | ||||
| 	// by https://datatracker.ietf.org/doc/html/rfc7540#appendix-A
 | ||||
| 	if tc.config.CipherSuites == nil { | ||||
| 		for _, cs := range tls.CipherSuites() { | ||||
| 			if _, ok := tls12ForbiddenCipherSuites[cs.ID]; !ok { | ||||
| 				tc.config.CipherSuites = append(tc.config.CipherSuites, cs.ID) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return tc | ||||
| } | ||||
| 
 | ||||
|  | @ -205,32 +249,3 @@ type TLSChannelzSecurityValue struct { | |||
| 	LocalCertificate  []byte | ||||
| 	RemoteCertificate []byte | ||||
| } | ||||
| 
 | ||||
| var cipherSuiteLookup = map[uint16]string{ | ||||
| 	tls.TLS_RSA_WITH_RC4_128_SHA:                "TLS_RSA_WITH_RC4_128_SHA", | ||||
| 	tls.TLS_RSA_WITH_3DES_EDE_CBC_SHA:           "TLS_RSA_WITH_3DES_EDE_CBC_SHA", | ||||
| 	tls.TLS_RSA_WITH_AES_128_CBC_SHA:            "TLS_RSA_WITH_AES_128_CBC_SHA", | ||||
| 	tls.TLS_RSA_WITH_AES_256_CBC_SHA:            "TLS_RSA_WITH_AES_256_CBC_SHA", | ||||
| 	tls.TLS_RSA_WITH_AES_128_GCM_SHA256:         "TLS_RSA_WITH_AES_128_GCM_SHA256", | ||||
| 	tls.TLS_RSA_WITH_AES_256_GCM_SHA384:         "TLS_RSA_WITH_AES_256_GCM_SHA384", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_RC4_128_SHA:        "TLS_ECDHE_ECDSA_WITH_RC4_128_SHA", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA:    "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA:    "TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_RC4_128_SHA:          "TLS_ECDHE_RSA_WITH_RC4_128_SHA", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA:     "TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA:      "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA:      "TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256:   "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256: "TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384:   "TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384: "TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384", | ||||
| 	tls.TLS_FALLBACK_SCSV:                       "TLS_FALLBACK_SCSV", | ||||
| 	tls.TLS_RSA_WITH_AES_128_CBC_SHA256:         "TLS_RSA_WITH_AES_128_CBC_SHA256", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256: "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256:   "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256", | ||||
| 	tls.TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305:    "TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305", | ||||
| 	tls.TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305:  "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305", | ||||
| 	tls.TLS_AES_128_GCM_SHA256:                  "TLS_AES_128_GCM_SHA256", | ||||
| 	tls.TLS_AES_256_GCM_SHA384:                  "TLS_AES_256_GCM_SHA384", | ||||
| 	tls.TLS_CHACHA20_POLY1305_SHA256:            "TLS_CHACHA20_POLY1305_SHA256", | ||||
| } | ||||
|  |  | |||
|  | @ -46,6 +46,7 @@ func init() { | |||
| 	internal.WithBinaryLogger = withBinaryLogger | ||||
| 	internal.JoinDialOptions = newJoinDialOption | ||||
| 	internal.DisableGlobalDialOptions = newDisableGlobalDialOptions | ||||
| 	internal.WithRecvBufferPool = withRecvBufferPool | ||||
| } | ||||
| 
 | ||||
| // dialOptions configure a Dial call. dialOptions are set by the DialOption
 | ||||
|  | @ -63,7 +64,6 @@ type dialOptions struct { | |||
| 	block                       bool | ||||
| 	returnLastError             bool | ||||
| 	timeout                     time.Duration | ||||
| 	scChan                      <-chan ServiceConfig | ||||
| 	authority                   string | ||||
| 	binaryLogger                binarylog.Logger | ||||
| 	copts                       transport.ConnectOptions | ||||
|  | @ -250,19 +250,6 @@ func WithDecompressor(dc Decompressor) DialOption { | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // WithServiceConfig returns a DialOption which has a channel to read the
 | ||||
| // service configuration.
 | ||||
| //
 | ||||
| // Deprecated: service config should be received through name resolver or via
 | ||||
| // WithDefaultServiceConfig, as specified at
 | ||||
| // https://github.com/grpc/grpc/blob/master/doc/service_config.md.  Will be
 | ||||
| // removed in a future 1.x release.
 | ||||
| func WithServiceConfig(c <-chan ServiceConfig) DialOption { | ||||
| 	return newFuncDialOption(func(o *dialOptions) { | ||||
| 		o.scChan = c | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // WithConnectParams configures the ClientConn to use the provided ConnectParams
 | ||||
| // for creating and maintaining connections to servers.
 | ||||
| //
 | ||||
|  | @ -413,6 +400,17 @@ func WithTimeout(d time.Duration) DialOption { | |||
| // connections. If FailOnNonTempDialError() is set to true, and an error is
 | ||||
| // returned by f, gRPC checks the error's Temporary() method to decide if it
 | ||||
| // should try to reconnect to the network address.
 | ||||
| //
 | ||||
| // Note: All supported releases of Go (as of December 2023) override the OS
 | ||||
| // defaults for TCP keepalive time and interval to 15s. To enable TCP keepalive
 | ||||
| // with OS defaults for keepalive time and interval, use a net.Dialer that sets
 | ||||
| // the KeepAlive field to a negative value, and sets the SO_KEEPALIVE socket
 | ||||
| // option to true from the Control field. For a concrete example of how to do
 | ||||
| // this, see internal.NetDialerWithTCPKeepalive().
 | ||||
| //
 | ||||
| // For more information, please see [issue 23459] in the Go github repo.
 | ||||
| //
 | ||||
| // [issue 23459]: https://github.com/golang/go/issues/23459
 | ||||
| func WithContextDialer(f func(context.Context, string) (net.Conn, error)) DialOption { | ||||
| 	return newFuncDialOption(func(o *dialOptions) { | ||||
| 		o.copts.Dialer = f | ||||
|  | @ -487,7 +485,7 @@ func FailOnNonTempDialError(f bool) DialOption { | |||
| // the RPCs.
 | ||||
| func WithUserAgent(s string) DialOption { | ||||
| 	return newFuncDialOption(func(o *dialOptions) { | ||||
| 		o.copts.UserAgent = s | ||||
| 		o.copts.UserAgent = s + " " + grpcUA | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
|  | @ -637,14 +635,16 @@ func withHealthCheckFunc(f internal.HealthChecker) DialOption { | |||
| 
 | ||||
| func defaultDialOptions() dialOptions { | ||||
| 	return dialOptions{ | ||||
| 		healthCheckFunc: internal.HealthCheckFunc, | ||||
| 		copts: transport.ConnectOptions{ | ||||
| 			WriteBufferSize: defaultWriteBufSize, | ||||
| 			ReadBufferSize:  defaultReadBufSize, | ||||
| 			WriteBufferSize: defaultWriteBufSize, | ||||
| 			UseProxy:        true, | ||||
| 			UserAgent:       grpcUA, | ||||
| 		}, | ||||
| 		recvBufferPool: nopBufferPool{}, | ||||
| 		idleTimeout:    30 * time.Minute, | ||||
| 		bs:              internalbackoff.DefaultExponential, | ||||
| 		healthCheckFunc: internal.HealthCheckFunc, | ||||
| 		idleTimeout:     30 * time.Minute, | ||||
| 		recvBufferPool:  nopBufferPool{}, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | @ -705,11 +705,13 @@ func WithIdleTimeout(d time.Duration) DialOption { | |||
| // options are used: WithStatsHandler, EnableTracing, or binary logging. In such
 | ||||
| // cases, the shared buffer pool will be ignored.
 | ||||
| //
 | ||||
| // # Experimental
 | ||||
| //
 | ||||
| // Notice: This API is EXPERIMENTAL and may be changed or removed in a
 | ||||
| // later release.
 | ||||
| // Deprecated: use experimental.WithRecvBufferPool instead.  Will be deleted in
 | ||||
| // v1.60.0 or later.
 | ||||
| func WithRecvBufferPool(bufferPool SharedBufferPool) DialOption { | ||||
| 	return withRecvBufferPool(bufferPool) | ||||
| } | ||||
| 
 | ||||
| func withRecvBufferPool(bufferPool SharedBufferPool) DialOption { | ||||
| 	return newFuncDialOption(func(o *dialOptions) { | ||||
| 		o.recvBufferPool = bufferPool | ||||
| 	}) | ||||
|  |  | |||
|  | @ -18,7 +18,10 @@ | |||
| // Package buffer provides an implementation of an unbounded buffer.
 | ||||
| package buffer | ||||
| 
 | ||||
| import "sync" | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"sync" | ||||
| ) | ||||
| 
 | ||||
| // Unbounded is an implementation of an unbounded buffer which does not use
 | ||||
| // extra goroutines. This is typically used for passing updates from one entity
 | ||||
|  | @ -36,6 +39,7 @@ import "sync" | |||
| type Unbounded struct { | ||||
| 	c       chan any | ||||
| 	closed  bool | ||||
| 	closing bool | ||||
| 	mu      sync.Mutex | ||||
| 	backlog []any | ||||
| } | ||||
|  | @ -45,32 +49,32 @@ func NewUnbounded() *Unbounded { | |||
| 	return &Unbounded{c: make(chan any, 1)} | ||||
| } | ||||
| 
 | ||||
| var errBufferClosed = errors.New("Put called on closed buffer.Unbounded") | ||||
| 
 | ||||
| // Put adds t to the unbounded buffer.
 | ||||
| func (b *Unbounded) Put(t any) { | ||||
| func (b *Unbounded) Put(t any) error { | ||||
| 	b.mu.Lock() | ||||
| 	defer b.mu.Unlock() | ||||
| 	if b.closed { | ||||
| 		return | ||||
| 	if b.closing { | ||||
| 		return errBufferClosed | ||||
| 	} | ||||
| 	if len(b.backlog) == 0 { | ||||
| 		select { | ||||
| 		case b.c <- t: | ||||
| 			return | ||||
| 			return nil | ||||
| 		default: | ||||
| 		} | ||||
| 	} | ||||
| 	b.backlog = append(b.backlog, t) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // Load sends the earliest buffered data, if any, onto the read channel
 | ||||
| // returned by Get(). Users are expected to call this every time they read a
 | ||||
| // Load sends the earliest buffered data, if any, onto the read channel returned
 | ||||
| // by Get(). Users are expected to call this every time they successfully read a
 | ||||
| // value from the read channel.
 | ||||
| func (b *Unbounded) Load() { | ||||
| 	b.mu.Lock() | ||||
| 	defer b.mu.Unlock() | ||||
| 	if b.closed { | ||||
| 		return | ||||
| 	} | ||||
| 	if len(b.backlog) > 0 { | ||||
| 		select { | ||||
| 		case b.c <- b.backlog[0]: | ||||
|  | @ -78,6 +82,8 @@ func (b *Unbounded) Load() { | |||
| 			b.backlog = b.backlog[1:] | ||||
| 		default: | ||||
| 		} | ||||
| 	} else if b.closing && !b.closed { | ||||
| 		close(b.c) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | @ -88,18 +94,23 @@ func (b *Unbounded) Load() { | |||
| // send the next buffered value onto the channel if there is any.
 | ||||
| //
 | ||||
| // If the unbounded buffer is closed, the read channel returned by this method
 | ||||
| // is closed.
 | ||||
| // is closed after all data is drained.
 | ||||
| func (b *Unbounded) Get() <-chan any { | ||||
| 	return b.c | ||||
| } | ||||
| 
 | ||||
| // Close closes the unbounded buffer.
 | ||||
| // Close closes the unbounded buffer. No subsequent data may be Put(), and the
 | ||||
| // channel returned from Get() will be closed after all the data is read and
 | ||||
| // Load() is called for the final time.
 | ||||
| func (b *Unbounded) Close() { | ||||
| 	b.mu.Lock() | ||||
| 	defer b.mu.Unlock() | ||||
| 	if b.closed { | ||||
| 	if b.closing { | ||||
| 		return | ||||
| 	} | ||||
| 	b.closed = true | ||||
| 	close(b.c) | ||||
| 	b.closing = true | ||||
| 	if len(b.backlog) == 0 { | ||||
| 		b.closed = true | ||||
| 		close(b.c) | ||||
| 	} | ||||
| } | ||||
|  |  | |||
|  | @ -31,6 +31,7 @@ import ( | |||
| 	"time" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/grpclog" | ||||
| 	"google.golang.org/grpc/internal" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
|  | @ -58,6 +59,12 @@ func TurnOn() { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| func init() { | ||||
| 	internal.ChannelzTurnOffForTesting = func() { | ||||
| 		atomic.StoreInt32(&curState, 0) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // IsOn returns whether channelz data collection is on.
 | ||||
| func IsOn() bool { | ||||
| 	return atomic.LoadInt32(&curState) == 1 | ||||
|  |  | |||
|  | @ -36,9 +36,6 @@ var ( | |||
| 	// "GRPC_RING_HASH_CAP".  This does not override the default bounds
 | ||||
| 	// checking which NACKs configs specifying ring sizes > 8*1024*1024 (~8M).
 | ||||
| 	RingHashCap = uint64FromEnv("GRPC_RING_HASH_CAP", 4096, 1, 8*1024*1024) | ||||
| 	// PickFirstLBConfig is set if we should support configuration of the
 | ||||
| 	// pick_first LB policy.
 | ||||
| 	PickFirstLBConfig = boolFromEnv("GRPC_EXPERIMENTAL_PICKFIRST_LB_CONFIG", true) | ||||
| 	// LeastRequestLB is set if we should support the least_request_experimental
 | ||||
| 	// LB policy, which can be enabled by setting the environment variable
 | ||||
| 	// "GRPC_EXPERIMENTAL_ENABLE_LEAST_REQUEST" to "true".
 | ||||
|  |  | |||
|  | @ -50,46 +50,7 @@ var ( | |||
| 	//
 | ||||
| 	// When both bootstrap FileName and FileContent are set, FileName is used.
 | ||||
| 	XDSBootstrapFileContent = os.Getenv(XDSBootstrapFileContentEnv) | ||||
| 	// XDSRingHash indicates whether ring hash support is enabled, which can be
 | ||||
| 	// disabled by setting the environment variable
 | ||||
| 	// "GRPC_XDS_EXPERIMENTAL_ENABLE_RING_HASH" to "false".
 | ||||
| 	XDSRingHash = boolFromEnv("GRPC_XDS_EXPERIMENTAL_ENABLE_RING_HASH", true) | ||||
| 	// XDSClientSideSecurity is used to control processing of security
 | ||||
| 	// configuration on the client-side.
 | ||||
| 	//
 | ||||
| 	// Note that there is no env var protection for the server-side because we
 | ||||
| 	// have a brand new API on the server-side and users explicitly need to use
 | ||||
| 	// the new API to get security integration on the server.
 | ||||
| 	XDSClientSideSecurity = boolFromEnv("GRPC_XDS_EXPERIMENTAL_SECURITY_SUPPORT", true) | ||||
| 	// XDSAggregateAndDNS indicates whether processing of aggregated cluster and
 | ||||
| 	// DNS cluster is enabled, which can be disabled by setting the environment
 | ||||
| 	// variable "GRPC_XDS_EXPERIMENTAL_ENABLE_AGGREGATE_AND_LOGICAL_DNS_CLUSTER"
 | ||||
| 	// to "false".
 | ||||
| 	XDSAggregateAndDNS = boolFromEnv("GRPC_XDS_EXPERIMENTAL_ENABLE_AGGREGATE_AND_LOGICAL_DNS_CLUSTER", true) | ||||
| 
 | ||||
| 	// XDSRBAC indicates whether xDS configured RBAC HTTP Filter is enabled,
 | ||||
| 	// which can be disabled by setting the environment variable
 | ||||
| 	// "GRPC_XDS_EXPERIMENTAL_RBAC" to "false".
 | ||||
| 	XDSRBAC = boolFromEnv("GRPC_XDS_EXPERIMENTAL_RBAC", true) | ||||
| 	// XDSOutlierDetection indicates whether outlier detection support is
 | ||||
| 	// enabled, which can be disabled by setting the environment variable
 | ||||
| 	// "GRPC_EXPERIMENTAL_ENABLE_OUTLIER_DETECTION" to "false".
 | ||||
| 	XDSOutlierDetection = boolFromEnv("GRPC_EXPERIMENTAL_ENABLE_OUTLIER_DETECTION", true) | ||||
| 	// XDSFederation indicates whether federation support is enabled, which can
 | ||||
| 	// be enabled by setting the environment variable
 | ||||
| 	// "GRPC_EXPERIMENTAL_XDS_FEDERATION" to "true".
 | ||||
| 	XDSFederation = boolFromEnv("GRPC_EXPERIMENTAL_XDS_FEDERATION", true) | ||||
| 
 | ||||
| 	// XDSRLS indicates whether processing of Cluster Specifier plugins and
 | ||||
| 	// support for the RLS CLuster Specifier is enabled, which can be disabled by
 | ||||
| 	// setting the environment variable "GRPC_EXPERIMENTAL_XDS_RLS_LB" to
 | ||||
| 	// "false".
 | ||||
| 	XDSRLS = boolFromEnv("GRPC_EXPERIMENTAL_XDS_RLS_LB", true) | ||||
| 
 | ||||
| 	// C2PResolverTestOnlyTrafficDirectorURI is the TD URI for testing.
 | ||||
| 	C2PResolverTestOnlyTrafficDirectorURI = os.Getenv("GRPC_TEST_ONLY_GOOGLE_C2P_RESOLVER_TRAFFIC_DIRECTOR_URI") | ||||
| 	// XDSCustomLBPolicy indicates whether Custom LB Policies are enabled, which
 | ||||
| 	// can be disabled by setting the environment variable
 | ||||
| 	// "GRPC_EXPERIMENTAL_XDS_CUSTOM_LB_CONFIG" to "false".
 | ||||
| 	XDSCustomLBPolicy = boolFromEnv("GRPC_EXPERIMENTAL_XDS_CUSTOM_LB_CONFIG", true) | ||||
| ) | ||||
|  |  | |||
|  | @ -0,0 +1,28 @@ | |||
| /* | ||||
|  * Copyright 2023 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| package internal | ||||
| 
 | ||||
| var ( | ||||
| 	// WithRecvBufferPool is implemented by the grpc package and returns a dial
 | ||||
| 	// option to configure a shared buffer pool for a grpc.ClientConn.
 | ||||
| 	WithRecvBufferPool any // func (grpc.SharedBufferPool) grpc.DialOption
 | ||||
| 
 | ||||
| 	// RecvBufferPool is implemented by the grpc package and returns a server
 | ||||
| 	// option to configure a shared buffer pool for a grpc.Server.
 | ||||
| 	RecvBufferPool any // func (grpc.SharedBufferPool) grpc.ServerOption
 | ||||
| ) | ||||
|  | @ -20,7 +20,6 @@ package grpcsync | |||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"sync" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/internal/buffer" | ||||
| ) | ||||
|  | @ -38,8 +37,6 @@ type CallbackSerializer struct { | |||
| 	done chan struct{} | ||||
| 
 | ||||
| 	callbacks *buffer.Unbounded | ||||
| 	closedMu  sync.Mutex | ||||
| 	closed    bool | ||||
| } | ||||
| 
 | ||||
| // NewCallbackSerializer returns a new CallbackSerializer instance. The provided
 | ||||
|  | @ -65,56 +62,34 @@ func NewCallbackSerializer(ctx context.Context) *CallbackSerializer { | |||
| // callbacks to be executed by the serializer. It is not possible to add
 | ||||
| // callbacks once the context passed to NewCallbackSerializer is cancelled.
 | ||||
| func (cs *CallbackSerializer) Schedule(f func(ctx context.Context)) bool { | ||||
| 	cs.closedMu.Lock() | ||||
| 	defer cs.closedMu.Unlock() | ||||
| 
 | ||||
| 	if cs.closed { | ||||
| 		return false | ||||
| 	} | ||||
| 	cs.callbacks.Put(f) | ||||
| 	return true | ||||
| 	return cs.callbacks.Put(f) == nil | ||||
| } | ||||
| 
 | ||||
| func (cs *CallbackSerializer) run(ctx context.Context) { | ||||
| 	var backlog []func(context.Context) | ||||
| 
 | ||||
| 	defer close(cs.done) | ||||
| 
 | ||||
| 	// TODO: when Go 1.21 is the oldest supported version, this loop and Close
 | ||||
| 	// can be replaced with:
 | ||||
| 	//
 | ||||
| 	// context.AfterFunc(ctx, cs.callbacks.Close)
 | ||||
| 	for ctx.Err() == nil { | ||||
| 		select { | ||||
| 		case <-ctx.Done(): | ||||
| 			// Do nothing here. Next iteration of the for loop will not happen,
 | ||||
| 			// since ctx.Err() would be non-nil.
 | ||||
| 		case callback, ok := <-cs.callbacks.Get(): | ||||
| 			if !ok { | ||||
| 				return | ||||
| 			} | ||||
| 		case cb := <-cs.callbacks.Get(): | ||||
| 			cs.callbacks.Load() | ||||
| 			callback.(func(ctx context.Context))(ctx) | ||||
| 			cb.(func(context.Context))(ctx) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// Fetch pending callbacks if any, and execute them before returning from
 | ||||
| 	// this method and closing cs.done.
 | ||||
| 	cs.closedMu.Lock() | ||||
| 	cs.closed = true | ||||
| 	backlog = cs.fetchPendingCallbacks() | ||||
| 	// Close the buffer to prevent new callbacks from being added.
 | ||||
| 	cs.callbacks.Close() | ||||
| 	cs.closedMu.Unlock() | ||||
| 	for _, b := range backlog { | ||||
| 		b(ctx) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (cs *CallbackSerializer) fetchPendingCallbacks() []func(context.Context) { | ||||
| 	var backlog []func(context.Context) | ||||
| 	for { | ||||
| 		select { | ||||
| 		case b := <-cs.callbacks.Get(): | ||||
| 			backlog = append(backlog, b.(func(context.Context))) | ||||
| 			cs.callbacks.Load() | ||||
| 		default: | ||||
| 			return backlog | ||||
| 		} | ||||
| 	// Run all pending callbacks.
 | ||||
| 	for cb := range cs.callbacks.Get() { | ||||
| 		cs.callbacks.Load() | ||||
| 		cb.(func(context.Context))(ctx) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -26,8 +26,6 @@ import ( | |||
| 	"sync" | ||||
| 	"sync/atomic" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/grpclog" | ||||
| ) | ||||
| 
 | ||||
| // For overriding in unit tests.
 | ||||
|  | @ -39,27 +37,12 @@ var timeAfterFunc = func(d time.Duration, f func()) *time.Timer { | |||
| // and exit from idle mode.
 | ||||
| type Enforcer interface { | ||||
| 	ExitIdleMode() error | ||||
| 	EnterIdleMode() error | ||||
| 	EnterIdleMode() | ||||
| } | ||||
| 
 | ||||
| // Manager defines the functionality required to track RPC activity on a
 | ||||
| // channel.
 | ||||
| type Manager interface { | ||||
| 	OnCallBegin() error | ||||
| 	OnCallEnd() | ||||
| 	Close() | ||||
| } | ||||
| 
 | ||||
| type noopManager struct{} | ||||
| 
 | ||||
| func (noopManager) OnCallBegin() error { return nil } | ||||
| func (noopManager) OnCallEnd()         {} | ||||
| func (noopManager) Close()             {} | ||||
| 
 | ||||
| // manager implements the Manager interface. It uses atomic operations to
 | ||||
| // synchronize access to shared state and a mutex to guarantee mutual exclusion
 | ||||
| // in a critical section.
 | ||||
| type manager struct { | ||||
| // Manager implements idleness detection and calls the configured Enforcer to
 | ||||
| // enter/exit idle mode when appropriate.  Must be created by NewManager.
 | ||||
| type Manager struct { | ||||
| 	// State accessed atomically.
 | ||||
| 	lastCallEndTime           int64 // Unix timestamp in nanos; time when the most recent RPC completed.
 | ||||
| 	activeCallsCount          int32 // Count of active RPCs; -math.MaxInt32 means channel is idle or is trying to get there.
 | ||||
|  | @ -69,8 +52,7 @@ type manager struct { | |||
| 	// Can be accessed without atomics or mutex since these are set at creation
 | ||||
| 	// time and read-only after that.
 | ||||
| 	enforcer Enforcer // Functionality provided by grpc.ClientConn.
 | ||||
| 	timeout  int64    // Idle timeout duration nanos stored as an int64.
 | ||||
| 	logger   grpclog.LoggerV2 | ||||
| 	timeout  time.Duration | ||||
| 
 | ||||
| 	// idleMu is used to guarantee mutual exclusion in two scenarios:
 | ||||
| 	// - Opposing intentions:
 | ||||
|  | @ -88,57 +70,48 @@ type manager struct { | |||
| 	timer        *time.Timer | ||||
| } | ||||
| 
 | ||||
| // ManagerOptions is a collection of options used by
 | ||||
| // NewManager.
 | ||||
| type ManagerOptions struct { | ||||
| 	Enforcer Enforcer | ||||
| 	Timeout  time.Duration | ||||
| 	Logger   grpclog.LoggerV2 | ||||
| } | ||||
| 
 | ||||
| // NewManager creates a new idleness manager implementation for the
 | ||||
| // given idle timeout.
 | ||||
| func NewManager(opts ManagerOptions) Manager { | ||||
| 	if opts.Timeout == 0 { | ||||
| 		return noopManager{} | ||||
| // given idle timeout.  It begins in idle mode.
 | ||||
| func NewManager(enforcer Enforcer, timeout time.Duration) *Manager { | ||||
| 	return &Manager{ | ||||
| 		enforcer:         enforcer, | ||||
| 		timeout:          timeout, | ||||
| 		actuallyIdle:     true, | ||||
| 		activeCallsCount: -math.MaxInt32, | ||||
| 	} | ||||
| 
 | ||||
| 	m := &manager{ | ||||
| 		enforcer: opts.Enforcer, | ||||
| 		timeout:  int64(opts.Timeout), | ||||
| 		logger:   opts.Logger, | ||||
| 	} | ||||
| 	m.timer = timeAfterFunc(opts.Timeout, m.handleIdleTimeout) | ||||
| 	return m | ||||
| } | ||||
| 
 | ||||
| // resetIdleTimer resets the idle timer to the given duration. This method
 | ||||
| // should only be called from the timer callback.
 | ||||
| func (m *manager) resetIdleTimer(d time.Duration) { | ||||
| 	m.idleMu.Lock() | ||||
| 	defer m.idleMu.Unlock() | ||||
| 
 | ||||
| 	if m.timer == nil { | ||||
| 		// Only close sets timer to nil. We are done.
 | ||||
| // resetIdleTimerLocked resets the idle timer to the given duration.  Called
 | ||||
| // when exiting idle mode or when the timer fires and we need to reset it.
 | ||||
| func (m *Manager) resetIdleTimerLocked(d time.Duration) { | ||||
| 	if m.isClosed() || m.timeout == 0 || m.actuallyIdle { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// It is safe to ignore the return value from Reset() because this method is
 | ||||
| 	// only ever called from the timer callback, which means the timer has
 | ||||
| 	// already fired.
 | ||||
| 	m.timer.Reset(d) | ||||
| 	// only ever called from the timer callback or when exiting idle mode.
 | ||||
| 	if m.timer != nil { | ||||
| 		m.timer.Stop() | ||||
| 	} | ||||
| 	m.timer = timeAfterFunc(d, m.handleIdleTimeout) | ||||
| } | ||||
| 
 | ||||
| func (m *Manager) resetIdleTimer(d time.Duration) { | ||||
| 	m.idleMu.Lock() | ||||
| 	defer m.idleMu.Unlock() | ||||
| 	m.resetIdleTimerLocked(d) | ||||
| } | ||||
| 
 | ||||
| // handleIdleTimeout is the timer callback that is invoked upon expiry of the
 | ||||
| // configured idle timeout. The channel is considered inactive if there are no
 | ||||
| // ongoing calls and no RPC activity since the last time the timer fired.
 | ||||
| func (m *manager) handleIdleTimeout() { | ||||
| func (m *Manager) handleIdleTimeout() { | ||||
| 	if m.isClosed() { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	if atomic.LoadInt32(&m.activeCallsCount) > 0 { | ||||
| 		m.resetIdleTimer(time.Duration(m.timeout)) | ||||
| 		m.resetIdleTimer(m.timeout) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
|  | @ -148,24 +121,12 @@ func (m *manager) handleIdleTimeout() { | |||
| 		// Set the timer to fire after a duration of idle timeout, calculated
 | ||||
| 		// from the time the most recent RPC completed.
 | ||||
| 		atomic.StoreInt32(&m.activeSinceLastTimerCheck, 0) | ||||
| 		m.resetIdleTimer(time.Duration(atomic.LoadInt64(&m.lastCallEndTime) + m.timeout - time.Now().UnixNano())) | ||||
| 		m.resetIdleTimer(time.Duration(atomic.LoadInt64(&m.lastCallEndTime)-time.Now().UnixNano()) + m.timeout) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// This CAS operation is extremely likely to succeed given that there has
 | ||||
| 	// been no activity since the last time we were here.  Setting the
 | ||||
| 	// activeCallsCount to -math.MaxInt32 indicates to OnCallBegin() that the
 | ||||
| 	// channel is either in idle mode or is trying to get there.
 | ||||
| 	if !atomic.CompareAndSwapInt32(&m.activeCallsCount, 0, -math.MaxInt32) { | ||||
| 		// This CAS operation can fail if an RPC started after we checked for
 | ||||
| 		// activity at the top of this method, or one was ongoing from before
 | ||||
| 		// the last time we were here. In both case, reset the timer and return.
 | ||||
| 		m.resetIdleTimer(time.Duration(m.timeout)) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// Now that we've set the active calls count to -math.MaxInt32, it's time to
 | ||||
| 	// actually move to idle mode.
 | ||||
| 	// Now that we've checked that there has been no activity, attempt to enter
 | ||||
| 	// idle mode, which is very likely to succeed.
 | ||||
| 	if m.tryEnterIdleMode() { | ||||
| 		// Successfully entered idle mode. No timer needed until we exit idle.
 | ||||
| 		return | ||||
|  | @ -174,8 +135,7 @@ func (m *manager) handleIdleTimeout() { | |||
| 	// Failed to enter idle mode due to a concurrent RPC that kept the channel
 | ||||
| 	// active, or because of an error from the channel. Undo the attempt to
 | ||||
| 	// enter idle, and reset the timer to try again later.
 | ||||
| 	atomic.AddInt32(&m.activeCallsCount, math.MaxInt32) | ||||
| 	m.resetIdleTimer(time.Duration(m.timeout)) | ||||
| 	m.resetIdleTimer(m.timeout) | ||||
| } | ||||
| 
 | ||||
| // tryEnterIdleMode instructs the channel to enter idle mode. But before
 | ||||
|  | @ -185,36 +145,49 @@ func (m *manager) handleIdleTimeout() { | |||
| // Return value indicates whether or not the channel moved to idle mode.
 | ||||
| //
 | ||||
| // Holds idleMu which ensures mutual exclusion with exitIdleMode.
 | ||||
| func (m *manager) tryEnterIdleMode() bool { | ||||
| func (m *Manager) tryEnterIdleMode() bool { | ||||
| 	// Setting the activeCallsCount to -math.MaxInt32 indicates to OnCallBegin()
 | ||||
| 	// that the channel is either in idle mode or is trying to get there.
 | ||||
| 	if !atomic.CompareAndSwapInt32(&m.activeCallsCount, 0, -math.MaxInt32) { | ||||
| 		// This CAS operation can fail if an RPC started after we checked for
 | ||||
| 		// activity in the timer handler, or one was ongoing from before the
 | ||||
| 		// last time the timer fired, or if a test is attempting to enter idle
 | ||||
| 		// mode without checking.  In all cases, abort going into idle mode.
 | ||||
| 		return false | ||||
| 	} | ||||
| 	// N.B. if we fail to enter idle mode after this, we must re-add
 | ||||
| 	// math.MaxInt32 to m.activeCallsCount.
 | ||||
| 
 | ||||
| 	m.idleMu.Lock() | ||||
| 	defer m.idleMu.Unlock() | ||||
| 
 | ||||
| 	if atomic.LoadInt32(&m.activeCallsCount) != -math.MaxInt32 { | ||||
| 		// We raced and lost to a new RPC. Very rare, but stop entering idle.
 | ||||
| 		atomic.AddInt32(&m.activeCallsCount, math.MaxInt32) | ||||
| 		return false | ||||
| 	} | ||||
| 	if atomic.LoadInt32(&m.activeSinceLastTimerCheck) == 1 { | ||||
| 		// An very short RPC could have come in (and also finished) after we
 | ||||
| 		// A very short RPC could have come in (and also finished) after we
 | ||||
| 		// checked for calls count and activity in handleIdleTimeout(), but
 | ||||
| 		// before the CAS operation. So, we need to check for activity again.
 | ||||
| 		atomic.AddInt32(&m.activeCallsCount, math.MaxInt32) | ||||
| 		return false | ||||
| 	} | ||||
| 
 | ||||
| 	// No new RPCs have come in since we last set the active calls count value
 | ||||
| 	// -math.MaxInt32 in the timer callback. And since we have the lock, it is
 | ||||
| 	// safe to enter idle mode now.
 | ||||
| 	if err := m.enforcer.EnterIdleMode(); err != nil { | ||||
| 		m.logger.Errorf("Failed to enter idle mode: %v", err) | ||||
| 		return false | ||||
| 	} | ||||
| 
 | ||||
| 	// Successfully entered idle mode.
 | ||||
| 	// No new RPCs have come in since we set the active calls count value to
 | ||||
| 	// -math.MaxInt32. And since we have the lock, it is safe to enter idle mode
 | ||||
| 	// unconditionally now.
 | ||||
| 	m.enforcer.EnterIdleMode() | ||||
| 	m.actuallyIdle = true | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func (m *Manager) EnterIdleModeForTesting() { | ||||
| 	m.tryEnterIdleMode() | ||||
| } | ||||
| 
 | ||||
| // OnCallBegin is invoked at the start of every RPC.
 | ||||
| func (m *manager) OnCallBegin() error { | ||||
| func (m *Manager) OnCallBegin() error { | ||||
| 	if m.isClosed() { | ||||
| 		return nil | ||||
| 	} | ||||
|  | @ -227,7 +200,7 @@ func (m *manager) OnCallBegin() error { | |||
| 
 | ||||
| 	// Channel is either in idle mode or is in the process of moving to idle
 | ||||
| 	// mode. Attempt to exit idle mode to allow this RPC.
 | ||||
| 	if err := m.exitIdleMode(); err != nil { | ||||
| 	if err := m.ExitIdleMode(); err != nil { | ||||
| 		// Undo the increment to calls count, and return an error causing the
 | ||||
| 		// RPC to fail.
 | ||||
| 		atomic.AddInt32(&m.activeCallsCount, -1) | ||||
|  | @ -238,28 +211,30 @@ func (m *manager) OnCallBegin() error { | |||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // exitIdleMode instructs the channel to exit idle mode.
 | ||||
| //
 | ||||
| // Holds idleMu which ensures mutual exclusion with tryEnterIdleMode.
 | ||||
| func (m *manager) exitIdleMode() error { | ||||
| // ExitIdleMode instructs m to call the enforcer's ExitIdleMode and update m's
 | ||||
| // internal state.
 | ||||
| func (m *Manager) ExitIdleMode() error { | ||||
| 	// Holds idleMu which ensures mutual exclusion with tryEnterIdleMode.
 | ||||
| 	m.idleMu.Lock() | ||||
| 	defer m.idleMu.Unlock() | ||||
| 
 | ||||
| 	if !m.actuallyIdle { | ||||
| 		// This can happen in two scenarios:
 | ||||
| 	if m.isClosed() || !m.actuallyIdle { | ||||
| 		// This can happen in three scenarios:
 | ||||
| 		// - handleIdleTimeout() set the calls count to -math.MaxInt32 and called
 | ||||
| 		//   tryEnterIdleMode(). But before the latter could grab the lock, an RPC
 | ||||
| 		//   came in and OnCallBegin() noticed that the calls count is negative.
 | ||||
| 		// - Channel is in idle mode, and multiple new RPCs come in at the same
 | ||||
| 		//   time, all of them notice a negative calls count in OnCallBegin and get
 | ||||
| 		//   here. The first one to get the lock would got the channel to exit idle.
 | ||||
| 		// - Channel is not in idle mode, and the user calls Connect which calls
 | ||||
| 		//   m.ExitIdleMode.
 | ||||
| 		//
 | ||||
| 		// Either way, nothing to do here.
 | ||||
| 		// In any case, there is nothing to do here.
 | ||||
| 		return nil | ||||
| 	} | ||||
| 
 | ||||
| 	if err := m.enforcer.ExitIdleMode(); err != nil { | ||||
| 		return fmt.Errorf("channel failed to exit idle mode: %v", err) | ||||
| 		return fmt.Errorf("failed to exit idle mode: %w", err) | ||||
| 	} | ||||
| 
 | ||||
| 	// Undo the idle entry process. This also respects any new RPC attempts.
 | ||||
|  | @ -267,12 +242,12 @@ func (m *manager) exitIdleMode() error { | |||
| 	m.actuallyIdle = false | ||||
| 
 | ||||
| 	// Start a new timer to fire after the configured idle timeout.
 | ||||
| 	m.timer = timeAfterFunc(time.Duration(m.timeout), m.handleIdleTimeout) | ||||
| 	m.resetIdleTimerLocked(m.timeout) | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // OnCallEnd is invoked at the end of every RPC.
 | ||||
| func (m *manager) OnCallEnd() { | ||||
| func (m *Manager) OnCallEnd() { | ||||
| 	if m.isClosed() { | ||||
| 		return | ||||
| 	} | ||||
|  | @ -287,15 +262,17 @@ func (m *manager) OnCallEnd() { | |||
| 	atomic.AddInt32(&m.activeCallsCount, -1) | ||||
| } | ||||
| 
 | ||||
| func (m *manager) isClosed() bool { | ||||
| func (m *Manager) isClosed() bool { | ||||
| 	return atomic.LoadInt32(&m.closed) == 1 | ||||
| } | ||||
| 
 | ||||
| func (m *manager) Close() { | ||||
| func (m *Manager) Close() { | ||||
| 	atomic.StoreInt32(&m.closed, 1) | ||||
| 
 | ||||
| 	m.idleMu.Lock() | ||||
| 	m.timer.Stop() | ||||
| 	m.timer = nil | ||||
| 	if m.timer != nil { | ||||
| 		m.timer.Stop() | ||||
| 		m.timer = nil | ||||
| 	} | ||||
| 	m.idleMu.Unlock() | ||||
| } | ||||
|  |  | |||
|  | @ -73,6 +73,11 @@ var ( | |||
| 	// xDS-enabled server invokes this method on a grpc.Server when a particular
 | ||||
| 	// listener moves to "not-serving" mode.
 | ||||
| 	DrainServerTransports any // func(*grpc.Server, string)
 | ||||
| 	// IsRegisteredMethod returns whether the passed in method is registered as
 | ||||
| 	// a method on the server.
 | ||||
| 	IsRegisteredMethod any // func(*grpc.Server, string) bool
 | ||||
| 	// ServerFromContext returns the server from the context.
 | ||||
| 	ServerFromContext any // func(context.Context) *grpc.Server
 | ||||
| 	// AddGlobalServerOptions adds an array of ServerOption that will be
 | ||||
| 	// effective globally for newly created servers. The priority will be: 1.
 | ||||
| 	// user-provided; 2. this method; 3. default values.
 | ||||
|  | @ -177,10 +182,12 @@ var ( | |||
| 	GRPCResolverSchemeExtraMetadata string = "xds" | ||||
| 
 | ||||
| 	// EnterIdleModeForTesting gets the ClientConn to enter IDLE mode.
 | ||||
| 	EnterIdleModeForTesting any // func(*grpc.ClientConn) error
 | ||||
| 	EnterIdleModeForTesting any // func(*grpc.ClientConn)
 | ||||
| 
 | ||||
| 	// ExitIdleModeForTesting gets the ClientConn to exit IDLE mode.
 | ||||
| 	ExitIdleModeForTesting any // func(*grpc.ClientConn) error
 | ||||
| 
 | ||||
| 	ChannelzTurnOffForTesting func() | ||||
| ) | ||||
| 
 | ||||
| // HealthChecker defines the signature of the client-side LB channel health checking function.
 | ||||
|  |  | |||
|  | @ -23,7 +23,6 @@ package dns | |||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"net" | ||||
| 	"os" | ||||
|  | @ -37,6 +36,7 @@ import ( | |||
| 	"google.golang.org/grpc/internal/backoff" | ||||
| 	"google.golang.org/grpc/internal/envconfig" | ||||
| 	"google.golang.org/grpc/internal/grpcrand" | ||||
| 	"google.golang.org/grpc/internal/resolver/dns/internal" | ||||
| 	"google.golang.org/grpc/resolver" | ||||
| 	"google.golang.org/grpc/serviceconfig" | ||||
| ) | ||||
|  | @ -47,15 +47,11 @@ var EnableSRVLookups = false | |||
| 
 | ||||
| var logger = grpclog.Component("dns") | ||||
| 
 | ||||
| // Globals to stub out in tests. TODO: Perhaps these two can be combined into a
 | ||||
| // single variable for testing the resolver?
 | ||||
| var ( | ||||
| 	newTimer           = time.NewTimer | ||||
| 	newTimerDNSResRate = time.NewTimer | ||||
| ) | ||||
| 
 | ||||
| func init() { | ||||
| 	resolver.Register(NewBuilder()) | ||||
| 	internal.TimeAfterFunc = time.After | ||||
| 	internal.NewNetResolver = newNetResolver | ||||
| 	internal.AddressDialer = addressDialer | ||||
| } | ||||
| 
 | ||||
| const ( | ||||
|  | @ -70,23 +66,6 @@ const ( | |||
| 	txtAttribute = "grpc_config=" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	errMissingAddr = errors.New("dns resolver: missing address") | ||||
| 
 | ||||
| 	// Addresses ending with a colon that is supposed to be the separator
 | ||||
| 	// between host and port is not allowed.  E.g. "::" is a valid address as
 | ||||
| 	// it is an IPv6 address (host only) and "[::]:" is invalid as it ends with
 | ||||
| 	// a colon as the host and port separator
 | ||||
| 	errEndsWithColon = errors.New("dns resolver: missing port after port-separator colon") | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	defaultResolver netResolver = net.DefaultResolver | ||||
| 	// To prevent excessive re-resolution, we enforce a rate limit on DNS
 | ||||
| 	// resolution requests.
 | ||||
| 	minDNSResRate = 30 * time.Second | ||||
| ) | ||||
| 
 | ||||
| var addressDialer = func(address string) func(context.Context, string, string) (net.Conn, error) { | ||||
| 	return func(ctx context.Context, network, _ string) (net.Conn, error) { | ||||
| 		var dialer net.Dialer | ||||
|  | @ -94,7 +73,11 @@ var addressDialer = func(address string) func(context.Context, string, string) ( | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| var newNetResolver = func(authority string) (netResolver, error) { | ||||
| var newNetResolver = func(authority string) (internal.NetResolver, error) { | ||||
| 	if authority == "" { | ||||
| 		return net.DefaultResolver, nil | ||||
| 	} | ||||
| 
 | ||||
| 	host, port, err := parseTarget(authority, defaultDNSSvrPort) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | @ -104,7 +87,7 @@ var newNetResolver = func(authority string) (netResolver, error) { | |||
| 
 | ||||
| 	return &net.Resolver{ | ||||
| 		PreferGo: true, | ||||
| 		Dial:     addressDialer(authorityWithPort), | ||||
| 		Dial:     internal.AddressDialer(authorityWithPort), | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
|  | @ -142,13 +125,9 @@ func (b *dnsBuilder) Build(target resolver.Target, cc resolver.ClientConn, opts | |||
| 		disableServiceConfig: opts.DisableServiceConfig, | ||||
| 	} | ||||
| 
 | ||||
| 	if target.URL.Host == "" { | ||||
| 		d.resolver = defaultResolver | ||||
| 	} else { | ||||
| 		d.resolver, err = newNetResolver(target.URL.Host) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	d.resolver, err = internal.NewNetResolver(target.URL.Host) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	d.wg.Add(1) | ||||
|  | @ -161,12 +140,6 @@ func (b *dnsBuilder) Scheme() string { | |||
| 	return "dns" | ||||
| } | ||||
| 
 | ||||
| type netResolver interface { | ||||
| 	LookupHost(ctx context.Context, host string) (addrs []string, err error) | ||||
| 	LookupSRV(ctx context.Context, service, proto, name string) (cname string, addrs []*net.SRV, err error) | ||||
| 	LookupTXT(ctx context.Context, name string) (txts []string, err error) | ||||
| } | ||||
| 
 | ||||
| // deadResolver is a resolver that does nothing.
 | ||||
| type deadResolver struct{} | ||||
| 
 | ||||
|  | @ -178,7 +151,7 @@ func (deadResolver) Close() {} | |||
| type dnsResolver struct { | ||||
| 	host     string | ||||
| 	port     string | ||||
| 	resolver netResolver | ||||
| 	resolver internal.NetResolver | ||||
| 	ctx      context.Context | ||||
| 	cancel   context.CancelFunc | ||||
| 	cc       resolver.ClientConn | ||||
|  | @ -223,29 +196,27 @@ func (d *dnsResolver) watcher() { | |||
| 			err = d.cc.UpdateState(*state) | ||||
| 		} | ||||
| 
 | ||||
| 		var timer *time.Timer | ||||
| 		var waitTime time.Duration | ||||
| 		if err == nil { | ||||
| 			// Success resolving, wait for the next ResolveNow. However, also wait 30
 | ||||
| 			// seconds at the very least to prevent constantly re-resolving.
 | ||||
| 			backoffIndex = 1 | ||||
| 			timer = newTimerDNSResRate(minDNSResRate) | ||||
| 			waitTime = internal.MinResolutionRate | ||||
| 			select { | ||||
| 			case <-d.ctx.Done(): | ||||
| 				timer.Stop() | ||||
| 				return | ||||
| 			case <-d.rn: | ||||
| 			} | ||||
| 		} else { | ||||
| 			// Poll on an error found in DNS Resolver or an error received from
 | ||||
| 			// ClientConn.
 | ||||
| 			timer = newTimer(backoff.DefaultExponential.Backoff(backoffIndex)) | ||||
| 			waitTime = backoff.DefaultExponential.Backoff(backoffIndex) | ||||
| 			backoffIndex++ | ||||
| 		} | ||||
| 		select { | ||||
| 		case <-d.ctx.Done(): | ||||
| 			timer.Stop() | ||||
| 			return | ||||
| 		case <-timer.C: | ||||
| 		case <-internal.TimeAfterFunc(waitTime): | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | @ -387,7 +358,7 @@ func formatIP(addr string) (addrIP string, ok bool) { | |||
| // target: ":80" defaultPort: "443" returns host: "localhost", port: "80"
 | ||||
| func parseTarget(target, defaultPort string) (host, port string, err error) { | ||||
| 	if target == "" { | ||||
| 		return "", "", errMissingAddr | ||||
| 		return "", "", internal.ErrMissingAddr | ||||
| 	} | ||||
| 	if ip := net.ParseIP(target); ip != nil { | ||||
| 		// target is an IPv4 or IPv6(without brackets) address
 | ||||
|  | @ -397,7 +368,7 @@ func parseTarget(target, defaultPort string) (host, port string, err error) { | |||
| 		if port == "" { | ||||
| 			// If the port field is empty (target ends with colon), e.g. "[::1]:",
 | ||||
| 			// this is an error.
 | ||||
| 			return "", "", errEndsWithColon | ||||
| 			return "", "", internal.ErrEndsWithColon | ||||
| 		} | ||||
| 		// target has port, i.e ipv4-host:port, [ipv6-host]:port, host-name:port
 | ||||
| 		if host == "" { | ||||
|  |  | |||
							
								
								
									
										70
									
								
								vendor/google.golang.org/grpc/internal/resolver/dns/internal/internal.go
								
								
									generated
								
								
									vendored
								
								
									Normal file
								
							
							
						
						
									
										70
									
								
								vendor/google.golang.org/grpc/internal/resolver/dns/internal/internal.go
								
								
									generated
								
								
									vendored
								
								
									Normal file
								
							|  | @ -0,0 +1,70 @@ | |||
| /* | ||||
|  * | ||||
|  * Copyright 2023 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| // Package internal contains functionality internal to the dns resolver package.
 | ||||
| package internal | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"net" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // NetResolver groups the methods on net.Resolver that are used by the DNS
 | ||||
| // resolver implementation. This allows the default net.Resolver instance to be
 | ||||
| // overidden from tests.
 | ||||
| type NetResolver interface { | ||||
| 	LookupHost(ctx context.Context, host string) (addrs []string, err error) | ||||
| 	LookupSRV(ctx context.Context, service, proto, name string) (cname string, addrs []*net.SRV, err error) | ||||
| 	LookupTXT(ctx context.Context, name string) (txts []string, err error) | ||||
| } | ||||
| 
 | ||||
| var ( | ||||
| 	// ErrMissingAddr is the error returned when building a DNS resolver when
 | ||||
| 	// the provided target name is empty.
 | ||||
| 	ErrMissingAddr = errors.New("dns resolver: missing address") | ||||
| 
 | ||||
| 	// ErrEndsWithColon is the error returned when building a DNS resolver when
 | ||||
| 	// the provided target name ends with a colon that is supposed to be the
 | ||||
| 	// separator between host and port.  E.g. "::" is a valid address as it is
 | ||||
| 	// an IPv6 address (host only) and "[::]:" is invalid as it ends with a
 | ||||
| 	// colon as the host and port separator
 | ||||
| 	ErrEndsWithColon = errors.New("dns resolver: missing port after port-separator colon") | ||||
| ) | ||||
| 
 | ||||
| // The following vars are overridden from tests.
 | ||||
| var ( | ||||
| 	// MinResolutionRate is the minimum rate at which re-resolutions are
 | ||||
| 	// allowed. This helps to prevent excessive re-resolution.
 | ||||
| 	MinResolutionRate = 30 * time.Second | ||||
| 
 | ||||
| 	// TimeAfterFunc is used by the DNS resolver to wait for the given duration
 | ||||
| 	// to elapse. In non-test code, this is implemented by time.After.  In test
 | ||||
| 	// code, this can be used to control the amount of time the resolver is
 | ||||
| 	// blocked waiting for the duration to elapse.
 | ||||
| 	TimeAfterFunc func(time.Duration) <-chan time.Time | ||||
| 
 | ||||
| 	// NewNetResolver returns the net.Resolver instance for the given target.
 | ||||
| 	NewNetResolver func(string) (NetResolver, error) | ||||
| 
 | ||||
| 	// AddressDialer is the dialer used to dial the DNS server. It accepts the
 | ||||
| 	// Host portion of the URL corresponding to the user's dial target and
 | ||||
| 	// returns a dial function.
 | ||||
| 	AddressDialer func(address string) func(context.Context, string, string) (net.Conn, error) | ||||
| ) | ||||
|  | @ -0,0 +1,29 @@ | |||
| //go:build !unix
 | ||||
| 
 | ||||
| /* | ||||
|  * Copyright 2023 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| package internal | ||||
| 
 | ||||
| import ( | ||||
| 	"net" | ||||
| ) | ||||
| 
 | ||||
| // NetDialerWithTCPKeepalive returns a vanilla net.Dialer on non-unix platforms.
 | ||||
| func NetDialerWithTCPKeepalive() *net.Dialer { | ||||
| 	return &net.Dialer{} | ||||
| } | ||||
|  | @ -0,0 +1,54 @@ | |||
| //go:build unix
 | ||||
| 
 | ||||
| /* | ||||
|  * Copyright 2023 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| package internal | ||||
| 
 | ||||
| import ( | ||||
| 	"net" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"golang.org/x/sys/unix" | ||||
| ) | ||||
| 
 | ||||
| // NetDialerWithTCPKeepalive returns a net.Dialer that enables TCP keepalives on
 | ||||
| // the underlying connection with OS default values for keepalive parameters.
 | ||||
| //
 | ||||
| // TODO: Once https://github.com/golang/go/issues/62254 lands, and the
 | ||||
| // appropriate Go version becomes less than our least supported Go version, we
 | ||||
| // should look into using the new API to make things more straightforward.
 | ||||
| func NetDialerWithTCPKeepalive() *net.Dialer { | ||||
| 	return &net.Dialer{ | ||||
| 		// Setting a negative value here prevents the Go stdlib from overriding
 | ||||
| 		// the values of TCP keepalive time and interval. It also prevents the
 | ||||
| 		// Go stdlib from enabling TCP keepalives by default.
 | ||||
| 		KeepAlive: time.Duration(-1), | ||||
| 		// This method is called after the underlying network socket is created,
 | ||||
| 		// but before dialing the socket (or calling its connect() method). The
 | ||||
| 		// combination of unconditionally enabling TCP keepalives here, and
 | ||||
| 		// disabling the overriding of TCP keepalive parameters by setting the
 | ||||
| 		// KeepAlive field to a negative value above, results in OS defaults for
 | ||||
| 		// the TCP keealive interval and time parameters.
 | ||||
| 		Control: func(_, _ string, c syscall.RawConn) error { | ||||
| 			return c.Control(func(fd uintptr) { | ||||
| 				unix.SetsockoptInt(int(fd), unix.SOL_SOCKET, unix.SO_KEEPALIVE, 1) | ||||
| 			}) | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
|  | @ -75,11 +75,25 @@ func NewServerHandlerTransport(w http.ResponseWriter, r *http.Request, stats []s | |||
| 		return nil, errors.New(msg) | ||||
| 	} | ||||
| 
 | ||||
| 	var localAddr net.Addr | ||||
| 	if la := r.Context().Value(http.LocalAddrContextKey); la != nil { | ||||
| 		localAddr, _ = la.(net.Addr) | ||||
| 	} | ||||
| 	var authInfo credentials.AuthInfo | ||||
| 	if r.TLS != nil { | ||||
| 		authInfo = credentials.TLSInfo{State: *r.TLS, CommonAuthInfo: credentials.CommonAuthInfo{SecurityLevel: credentials.PrivacyAndIntegrity}} | ||||
| 	} | ||||
| 	p := peer.Peer{ | ||||
| 		Addr:      strAddr(r.RemoteAddr), | ||||
| 		LocalAddr: localAddr, | ||||
| 		AuthInfo:  authInfo, | ||||
| 	} | ||||
| 	st := &serverHandlerTransport{ | ||||
| 		rw:             w, | ||||
| 		req:            r, | ||||
| 		closedCh:       make(chan struct{}), | ||||
| 		writes:         make(chan func()), | ||||
| 		peer:           p, | ||||
| 		contentType:    contentType, | ||||
| 		contentSubtype: contentSubtype, | ||||
| 		stats:          stats, | ||||
|  | @ -134,6 +148,8 @@ type serverHandlerTransport struct { | |||
| 
 | ||||
| 	headerMD metadata.MD | ||||
| 
 | ||||
| 	peer peer.Peer | ||||
| 
 | ||||
| 	closeOnce sync.Once | ||||
| 	closedCh  chan struct{} // closed on Close
 | ||||
| 
 | ||||
|  | @ -165,7 +181,13 @@ func (ht *serverHandlerTransport) Close(err error) { | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (ht *serverHandlerTransport) RemoteAddr() net.Addr { return strAddr(ht.req.RemoteAddr) } | ||||
| func (ht *serverHandlerTransport) Peer() *peer.Peer { | ||||
| 	return &peer.Peer{ | ||||
| 		Addr:      ht.peer.Addr, | ||||
| 		LocalAddr: ht.peer.LocalAddr, | ||||
| 		AuthInfo:  ht.peer.AuthInfo, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // strAddr is a net.Addr backed by either a TCP "ip:port" string, or
 | ||||
| // the empty string if unknown.
 | ||||
|  | @ -347,10 +369,8 @@ func (ht *serverHandlerTransport) WriteHeader(s *Stream, md metadata.MD) error { | |||
| 	return err | ||||
| } | ||||
| 
 | ||||
| func (ht *serverHandlerTransport) HandleStreams(startStream func(*Stream)) { | ||||
| func (ht *serverHandlerTransport) HandleStreams(ctx context.Context, startStream func(*Stream)) { | ||||
| 	// With this transport type there will be exactly 1 stream: this HTTP request.
 | ||||
| 
 | ||||
| 	ctx := ht.req.Context() | ||||
| 	var cancel context.CancelFunc | ||||
| 	if ht.timeoutSet { | ||||
| 		ctx, cancel = context.WithTimeout(ctx, ht.timeout) | ||||
|  | @ -370,34 +390,19 @@ func (ht *serverHandlerTransport) HandleStreams(startStream func(*Stream)) { | |||
| 		ht.Close(errors.New("request is done processing")) | ||||
| 	}() | ||||
| 
 | ||||
| 	req := ht.req | ||||
| 
 | ||||
| 	s := &Stream{ | ||||
| 		id:             0, // irrelevant
 | ||||
| 		requestRead:    func(int) {}, | ||||
| 		cancel:         cancel, | ||||
| 		buf:            newRecvBuffer(), | ||||
| 		st:             ht, | ||||
| 		method:         req.URL.Path, | ||||
| 		recvCompress:   req.Header.Get("grpc-encoding"), | ||||
| 		contentSubtype: ht.contentSubtype, | ||||
| 	} | ||||
| 	pr := &peer.Peer{ | ||||
| 		Addr: ht.RemoteAddr(), | ||||
| 	} | ||||
| 	if req.TLS != nil { | ||||
| 		pr.AuthInfo = credentials.TLSInfo{State: *req.TLS, CommonAuthInfo: credentials.CommonAuthInfo{SecurityLevel: credentials.PrivacyAndIntegrity}} | ||||
| 	} | ||||
| 	ctx = metadata.NewIncomingContext(ctx, ht.headerMD) | ||||
| 	s.ctx = peer.NewContext(ctx, pr) | ||||
| 	for _, sh := range ht.stats { | ||||
| 		s.ctx = sh.TagRPC(s.ctx, &stats.RPCTagInfo{FullMethodName: s.method}) | ||||
| 		inHeader := &stats.InHeader{ | ||||
| 			FullMethod:  s.method, | ||||
| 			RemoteAddr:  ht.RemoteAddr(), | ||||
| 			Compression: s.recvCompress, | ||||
| 		} | ||||
| 		sh.HandleRPC(s.ctx, inHeader) | ||||
| 	req := ht.req | ||||
| 	s := &Stream{ | ||||
| 		id:               0, // irrelevant
 | ||||
| 		ctx:              ctx, | ||||
| 		requestRead:      func(int) {}, | ||||
| 		cancel:           cancel, | ||||
| 		buf:              newRecvBuffer(), | ||||
| 		st:               ht, | ||||
| 		method:           req.URL.Path, | ||||
| 		recvCompress:     req.Header.Get("grpc-encoding"), | ||||
| 		contentSubtype:   ht.contentSubtype, | ||||
| 		headerWireLength: 0, // won't have access to header wire length until golang/go#18997.
 | ||||
| 	} | ||||
| 	s.trReader = &transportReader{ | ||||
| 		reader:        &recvBufferReader{ctx: s.ctx, ctxDone: s.ctx.Done(), recv: s.buf, freeBuffer: func(*bytes.Buffer) {}}, | ||||
|  |  | |||
|  | @ -36,6 +36,7 @@ import ( | |||
| 	"golang.org/x/net/http2/hpack" | ||||
| 	"google.golang.org/grpc/codes" | ||||
| 	"google.golang.org/grpc/credentials" | ||||
| 	"google.golang.org/grpc/internal" | ||||
| 	"google.golang.org/grpc/internal/channelz" | ||||
| 	icredentials "google.golang.org/grpc/internal/credentials" | ||||
| 	"google.golang.org/grpc/internal/grpclog" | ||||
|  | @ -43,7 +44,7 @@ import ( | |||
| 	"google.golang.org/grpc/internal/grpcutil" | ||||
| 	imetadata "google.golang.org/grpc/internal/metadata" | ||||
| 	istatus "google.golang.org/grpc/internal/status" | ||||
| 	"google.golang.org/grpc/internal/syscall" | ||||
| 	isyscall "google.golang.org/grpc/internal/syscall" | ||||
| 	"google.golang.org/grpc/internal/transport/networktype" | ||||
| 	"google.golang.org/grpc/keepalive" | ||||
| 	"google.golang.org/grpc/metadata" | ||||
|  | @ -176,7 +177,7 @@ func dial(ctx context.Context, fn func(context.Context, string) (net.Conn, error | |||
| 	if networkType == "tcp" && useProxy { | ||||
| 		return proxyDial(ctx, address, grpcUA) | ||||
| 	} | ||||
| 	return (&net.Dialer{}).DialContext(ctx, networkType, address) | ||||
| 	return internal.NetDialerWithTCPKeepalive().DialContext(ctx, networkType, address) | ||||
| } | ||||
| 
 | ||||
| func isTemporary(err error) bool { | ||||
|  | @ -262,7 +263,7 @@ func newHTTP2Client(connectCtx, ctx context.Context, addr resolver.Address, opts | |||
| 	} | ||||
| 	keepaliveEnabled := false | ||||
| 	if kp.Time != infinity { | ||||
| 		if err = syscall.SetTCPUserTimeout(conn, kp.Timeout); err != nil { | ||||
| 		if err = isyscall.SetTCPUserTimeout(conn, kp.Timeout); err != nil { | ||||
| 			return nil, connectionErrorf(false, err, "transport: failed to set TCP_USER_TIMEOUT: %v", err) | ||||
| 		} | ||||
| 		keepaliveEnabled = true | ||||
|  | @ -493,8 +494,9 @@ func (t *http2Client) newStream(ctx context.Context, callHdr *CallHdr) *Stream { | |||
| 
 | ||||
| func (t *http2Client) getPeer() *peer.Peer { | ||||
| 	return &peer.Peer{ | ||||
| 		Addr:     t.remoteAddr, | ||||
| 		AuthInfo: t.authInfo, // Can be nil
 | ||||
| 		Addr:      t.remoteAddr, | ||||
| 		AuthInfo:  t.authInfo, // Can be nil
 | ||||
| 		LocalAddr: t.localAddr, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -68,18 +68,15 @@ var serverConnectionCounter uint64 | |||
| 
 | ||||
| // http2Server implements the ServerTransport interface with HTTP2.
 | ||||
| type http2Server struct { | ||||
| 	lastRead    int64 // Keep this field 64-bit aligned. Accessed atomically.
 | ||||
| 	ctx         context.Context | ||||
| 	done        chan struct{} | ||||
| 	conn        net.Conn | ||||
| 	loopy       *loopyWriter | ||||
| 	readerDone  chan struct{} // sync point to enable testing.
 | ||||
| 	writerDone  chan struct{} // sync point to enable testing.
 | ||||
| 	remoteAddr  net.Addr | ||||
| 	localAddr   net.Addr | ||||
| 	authInfo    credentials.AuthInfo // auth info about the connection
 | ||||
| 	inTapHandle tap.ServerInHandle | ||||
| 	framer      *framer | ||||
| 	lastRead        int64 // Keep this field 64-bit aligned. Accessed atomically.
 | ||||
| 	done            chan struct{} | ||||
| 	conn            net.Conn | ||||
| 	loopy           *loopyWriter | ||||
| 	readerDone      chan struct{} // sync point to enable testing.
 | ||||
| 	loopyWriterDone chan struct{} | ||||
| 	peer            peer.Peer | ||||
| 	inTapHandle     tap.ServerInHandle | ||||
| 	framer          *framer | ||||
| 	// The max number of concurrent streams.
 | ||||
| 	maxStreams uint32 | ||||
| 	// controlBuf delivers all the control related tasks (e.g., window
 | ||||
|  | @ -243,16 +240,18 @@ func NewServerTransport(conn net.Conn, config *ServerConfig) (_ ServerTransport, | |||
| 	} | ||||
| 
 | ||||
| 	done := make(chan struct{}) | ||||
| 	peer := peer.Peer{ | ||||
| 		Addr:      conn.RemoteAddr(), | ||||
| 		LocalAddr: conn.LocalAddr(), | ||||
| 		AuthInfo:  authInfo, | ||||
| 	} | ||||
| 	t := &http2Server{ | ||||
| 		ctx:               setConnection(context.Background(), rawConn), | ||||
| 		done:              done, | ||||
| 		conn:              conn, | ||||
| 		remoteAddr:        conn.RemoteAddr(), | ||||
| 		localAddr:         conn.LocalAddr(), | ||||
| 		authInfo:          authInfo, | ||||
| 		peer:              peer, | ||||
| 		framer:            framer, | ||||
| 		readerDone:        make(chan struct{}), | ||||
| 		writerDone:        make(chan struct{}), | ||||
| 		loopyWriterDone:   make(chan struct{}), | ||||
| 		maxStreams:        config.MaxStreams, | ||||
| 		inTapHandle:       config.InTapHandle, | ||||
| 		fc:                &trInFlow{limit: uint32(icwz)}, | ||||
|  | @ -267,8 +266,6 @@ func NewServerTransport(conn net.Conn, config *ServerConfig) (_ ServerTransport, | |||
| 		bufferPool:        newBufferPool(), | ||||
| 	} | ||||
| 	t.logger = prefixLoggerForServerTransport(t) | ||||
| 	// Add peer information to the http2server context.
 | ||||
| 	t.ctx = peer.NewContext(t.ctx, t.getPeer()) | ||||
| 
 | ||||
| 	t.controlBuf = newControlBuffer(t.done) | ||||
| 	if dynamicWindow { | ||||
|  | @ -277,15 +274,7 @@ func NewServerTransport(conn net.Conn, config *ServerConfig) (_ ServerTransport, | |||
| 			updateFlowControl: t.updateFlowControl, | ||||
| 		} | ||||
| 	} | ||||
| 	for _, sh := range t.stats { | ||||
| 		t.ctx = sh.TagConn(t.ctx, &stats.ConnTagInfo{ | ||||
| 			RemoteAddr: t.remoteAddr, | ||||
| 			LocalAddr:  t.localAddr, | ||||
| 		}) | ||||
| 		connBegin := &stats.ConnBegin{} | ||||
| 		sh.HandleConn(t.ctx, connBegin) | ||||
| 	} | ||||
| 	t.channelzID, err = channelz.RegisterNormalSocket(t, config.ChannelzParentID, fmt.Sprintf("%s -> %s", t.remoteAddr, t.localAddr)) | ||||
| 	t.channelzID, err = channelz.RegisterNormalSocket(t, config.ChannelzParentID, fmt.Sprintf("%s -> %s", t.peer.Addr, t.peer.LocalAddr)) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | @ -334,7 +323,7 @@ func NewServerTransport(conn net.Conn, config *ServerConfig) (_ ServerTransport, | |||
| 		t.loopy = newLoopyWriter(serverSide, t.framer, t.controlBuf, t.bdpEst, t.conn, t.logger) | ||||
| 		t.loopy.ssGoAwayHandler = t.outgoingGoAwayHandler | ||||
| 		t.loopy.run() | ||||
| 		close(t.writerDone) | ||||
| 		close(t.loopyWriterDone) | ||||
| 	}() | ||||
| 	go t.keepalive() | ||||
| 	return t, nil | ||||
|  | @ -342,7 +331,7 @@ func NewServerTransport(conn net.Conn, config *ServerConfig) (_ ServerTransport, | |||
| 
 | ||||
| // operateHeaders takes action on the decoded headers. Returns an error if fatal
 | ||||
| // error encountered and transport needs to close, otherwise returns nil.
 | ||||
| func (t *http2Server) operateHeaders(frame *http2.MetaHeadersFrame, handle func(*Stream)) error { | ||||
| func (t *http2Server) operateHeaders(ctx context.Context, frame *http2.MetaHeadersFrame, handle func(*Stream)) error { | ||||
| 	// Acquire max stream ID lock for entire duration
 | ||||
| 	t.maxStreamMu.Lock() | ||||
| 	defer t.maxStreamMu.Unlock() | ||||
|  | @ -369,10 +358,11 @@ func (t *http2Server) operateHeaders(frame *http2.MetaHeadersFrame, handle func( | |||
| 
 | ||||
| 	buf := newRecvBuffer() | ||||
| 	s := &Stream{ | ||||
| 		id:  streamID, | ||||
| 		st:  t, | ||||
| 		buf: buf, | ||||
| 		fc:  &inFlow{limit: uint32(t.initialWindowSize)}, | ||||
| 		id:               streamID, | ||||
| 		st:               t, | ||||
| 		buf:              buf, | ||||
| 		fc:               &inFlow{limit: uint32(t.initialWindowSize)}, | ||||
| 		headerWireLength: int(frame.Header().Length), | ||||
| 	} | ||||
| 	var ( | ||||
| 		// if false, content-type was missing or invalid
 | ||||
|  | @ -511,9 +501,9 @@ func (t *http2Server) operateHeaders(frame *http2.MetaHeadersFrame, handle func( | |||
| 		s.state = streamReadDone | ||||
| 	} | ||||
| 	if timeoutSet { | ||||
| 		s.ctx, s.cancel = context.WithTimeout(t.ctx, timeout) | ||||
| 		s.ctx, s.cancel = context.WithTimeout(ctx, timeout) | ||||
| 	} else { | ||||
| 		s.ctx, s.cancel = context.WithCancel(t.ctx) | ||||
| 		s.ctx, s.cancel = context.WithCancel(ctx) | ||||
| 	} | ||||
| 
 | ||||
| 	// Attach the received metadata to the context.
 | ||||
|  | @ -592,18 +582,6 @@ func (t *http2Server) operateHeaders(frame *http2.MetaHeadersFrame, handle func( | |||
| 	s.requestRead = func(n int) { | ||||
| 		t.adjustWindow(s, uint32(n)) | ||||
| 	} | ||||
| 	for _, sh := range t.stats { | ||||
| 		s.ctx = sh.TagRPC(s.ctx, &stats.RPCTagInfo{FullMethodName: s.method}) | ||||
| 		inHeader := &stats.InHeader{ | ||||
| 			FullMethod:  s.method, | ||||
| 			RemoteAddr:  t.remoteAddr, | ||||
| 			LocalAddr:   t.localAddr, | ||||
| 			Compression: s.recvCompress, | ||||
| 			WireLength:  int(frame.Header().Length), | ||||
| 			Header:      mdata.Copy(), | ||||
| 		} | ||||
| 		sh.HandleRPC(s.ctx, inHeader) | ||||
| 	} | ||||
| 	s.ctxDone = s.ctx.Done() | ||||
| 	s.wq = newWriteQuota(defaultWriteQuota, s.ctxDone) | ||||
| 	s.trReader = &transportReader{ | ||||
|  | @ -629,8 +607,11 @@ func (t *http2Server) operateHeaders(frame *http2.MetaHeadersFrame, handle func( | |||
| // HandleStreams receives incoming streams using the given handler. This is
 | ||||
| // typically run in a separate goroutine.
 | ||||
| // traceCtx attaches trace to ctx and returns the new context.
 | ||||
| func (t *http2Server) HandleStreams(handle func(*Stream)) { | ||||
| 	defer close(t.readerDone) | ||||
| func (t *http2Server) HandleStreams(ctx context.Context, handle func(*Stream)) { | ||||
| 	defer func() { | ||||
| 		<-t.loopyWriterDone | ||||
| 		close(t.readerDone) | ||||
| 	}() | ||||
| 	for { | ||||
| 		t.controlBuf.throttle() | ||||
| 		frame, err := t.framer.fr.ReadFrame() | ||||
|  | @ -664,7 +645,7 @@ func (t *http2Server) HandleStreams(handle func(*Stream)) { | |||
| 		} | ||||
| 		switch frame := frame.(type) { | ||||
| 		case *http2.MetaHeadersFrame: | ||||
| 			if err := t.operateHeaders(frame, handle); err != nil { | ||||
| 			if err := t.operateHeaders(ctx, frame, handle); err != nil { | ||||
| 				t.Close(err) | ||||
| 				break | ||||
| 			} | ||||
|  | @ -1242,10 +1223,6 @@ func (t *http2Server) Close(err error) { | |||
| 	for _, s := range streams { | ||||
| 		s.cancel() | ||||
| 	} | ||||
| 	for _, sh := range t.stats { | ||||
| 		connEnd := &stats.ConnEnd{} | ||||
| 		sh.HandleConn(t.ctx, connEnd) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // deleteStream deletes the stream s from transport's active streams.
 | ||||
|  | @ -1311,10 +1288,6 @@ func (t *http2Server) closeStream(s *Stream, rst bool, rstCode http2.ErrCode, eo | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| func (t *http2Server) RemoteAddr() net.Addr { | ||||
| 	return t.remoteAddr | ||||
| } | ||||
| 
 | ||||
| func (t *http2Server) Drain(debugData string) { | ||||
| 	t.mu.Lock() | ||||
| 	defer t.mu.Unlock() | ||||
|  | @ -1397,11 +1370,11 @@ func (t *http2Server) ChannelzMetric() *channelz.SocketInternalMetric { | |||
| 		LastMessageReceivedTimestamp:     time.Unix(0, atomic.LoadInt64(&t.czData.lastMsgRecvTime)), | ||||
| 		LocalFlowControlWindow:           int64(t.fc.getSize()), | ||||
| 		SocketOptions:                    channelz.GetSocketOption(t.conn), | ||||
| 		LocalAddr:                        t.localAddr, | ||||
| 		RemoteAddr:                       t.remoteAddr, | ||||
| 		LocalAddr:                        t.peer.LocalAddr, | ||||
| 		RemoteAddr:                       t.peer.Addr, | ||||
| 		// RemoteName :
 | ||||
| 	} | ||||
| 	if au, ok := t.authInfo.(credentials.ChannelzSecurityInfo); ok { | ||||
| 	if au, ok := t.peer.AuthInfo.(credentials.ChannelzSecurityInfo); ok { | ||||
| 		s.Security = au.GetSecurityValue() | ||||
| 	} | ||||
| 	s.RemoteFlowControlWindow = t.getOutFlowWindow() | ||||
|  | @ -1433,10 +1406,12 @@ func (t *http2Server) getOutFlowWindow() int64 { | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (t *http2Server) getPeer() *peer.Peer { | ||||
| // Peer returns the peer of the transport.
 | ||||
| func (t *http2Server) Peer() *peer.Peer { | ||||
| 	return &peer.Peer{ | ||||
| 		Addr:     t.remoteAddr, | ||||
| 		AuthInfo: t.authInfo, // Can be nil
 | ||||
| 		Addr:      t.peer.Addr, | ||||
| 		LocalAddr: t.peer.LocalAddr, | ||||
| 		AuthInfo:  t.peer.AuthInfo, // Can be nil
 | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
|  | @ -1461,6 +1436,6 @@ func GetConnection(ctx context.Context) net.Conn { | |||
| // SetConnection adds the connection to the context to be able to get
 | ||||
| // information about the destination ip and port for an incoming RPC. This also
 | ||||
| // allows any unary or streaming interceptors to see the connection.
 | ||||
| func setConnection(ctx context.Context, conn net.Conn) context.Context { | ||||
| func SetConnection(ctx context.Context, conn net.Conn) context.Context { | ||||
| 	return context.WithValue(ctx, connectionKey{}, conn) | ||||
| } | ||||
|  |  | |||
|  | @ -28,6 +28,8 @@ import ( | |||
| 	"net/http" | ||||
| 	"net/http/httputil" | ||||
| 	"net/url" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/internal" | ||||
| ) | ||||
| 
 | ||||
| const proxyAuthHeaderKey = "Proxy-Authorization" | ||||
|  | @ -112,7 +114,7 @@ func doHTTPConnectHandshake(ctx context.Context, conn net.Conn, backendAddr stri | |||
| // proxyDial dials, connecting to a proxy first if necessary. Checks if a proxy
 | ||||
| // is necessary, dials, does the HTTP CONNECT handshake, and returns the
 | ||||
| // connection.
 | ||||
| func proxyDial(ctx context.Context, addr string, grpcUA string) (conn net.Conn, err error) { | ||||
| func proxyDial(ctx context.Context, addr string, grpcUA string) (net.Conn, error) { | ||||
| 	newAddr := addr | ||||
| 	proxyURL, err := mapAddress(addr) | ||||
| 	if err != nil { | ||||
|  | @ -122,15 +124,15 @@ func proxyDial(ctx context.Context, addr string, grpcUA string) (conn net.Conn, | |||
| 		newAddr = proxyURL.Host | ||||
| 	} | ||||
| 
 | ||||
| 	conn, err = (&net.Dialer{}).DialContext(ctx, "tcp", newAddr) | ||||
| 	conn, err := internal.NetDialerWithTCPKeepalive().DialContext(ctx, "tcp", newAddr) | ||||
| 	if err != nil { | ||||
| 		return | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	if proxyURL != nil { | ||||
| 	if proxyURL == nil { | ||||
| 		// proxy is disabled if proxyURL is nil.
 | ||||
| 		conn, err = doHTTPConnectHandshake(ctx, conn, addr, proxyURL, grpcUA) | ||||
| 		return conn, err | ||||
| 	} | ||||
| 	return | ||||
| 	return doHTTPConnectHandshake(ctx, conn, addr, proxyURL, grpcUA) | ||||
| } | ||||
| 
 | ||||
| func sendHTTPRequest(ctx context.Context, req *http.Request, conn net.Conn) error { | ||||
|  |  | |||
|  | @ -37,6 +37,7 @@ import ( | |||
| 	"google.golang.org/grpc/internal/channelz" | ||||
| 	"google.golang.org/grpc/keepalive" | ||||
| 	"google.golang.org/grpc/metadata" | ||||
| 	"google.golang.org/grpc/peer" | ||||
| 	"google.golang.org/grpc/resolver" | ||||
| 	"google.golang.org/grpc/stats" | ||||
| 	"google.golang.org/grpc/status" | ||||
|  | @ -265,7 +266,8 @@ type Stream struct { | |||
| 	// headerValid indicates whether a valid header was received.  Only
 | ||||
| 	// meaningful after headerChan is closed (always call waitOnHeader() before
 | ||||
| 	// reading its value).  Not valid on server side.
 | ||||
| 	headerValid bool | ||||
| 	headerValid      bool | ||||
| 	headerWireLength int // Only set on server side.
 | ||||
| 
 | ||||
| 	// hdrMu protects header and trailer metadata on the server-side.
 | ||||
| 	hdrMu sync.Mutex | ||||
|  | @ -425,6 +427,12 @@ func (s *Stream) Context() context.Context { | |||
| 	return s.ctx | ||||
| } | ||||
| 
 | ||||
| // SetContext sets the context of the stream. This will be deleted once the
 | ||||
| // stats handler callouts all move to gRPC layer.
 | ||||
| func (s *Stream) SetContext(ctx context.Context) { | ||||
| 	s.ctx = ctx | ||||
| } | ||||
| 
 | ||||
| // Method returns the method for the stream.
 | ||||
| func (s *Stream) Method() string { | ||||
| 	return s.method | ||||
|  | @ -437,6 +445,12 @@ func (s *Stream) Status() *status.Status { | |||
| 	return s.status | ||||
| } | ||||
| 
 | ||||
| // HeaderWireLength returns the size of the headers of the stream as received
 | ||||
| // from the wire. Valid only on the server.
 | ||||
| func (s *Stream) HeaderWireLength() int { | ||||
| 	return s.headerWireLength | ||||
| } | ||||
| 
 | ||||
| // SetHeader sets the header metadata. This can be called multiple times.
 | ||||
| // Server side only.
 | ||||
| // This should not be called in parallel to other data writes.
 | ||||
|  | @ -698,7 +712,7 @@ type ClientTransport interface { | |||
| // Write methods for a given Stream will be called serially.
 | ||||
| type ServerTransport interface { | ||||
| 	// HandleStreams receives incoming streams using the given handler.
 | ||||
| 	HandleStreams(func(*Stream)) | ||||
| 	HandleStreams(context.Context, func(*Stream)) | ||||
| 
 | ||||
| 	// WriteHeader sends the header metadata for the given stream.
 | ||||
| 	// WriteHeader may not be called on all streams.
 | ||||
|  | @ -717,8 +731,8 @@ type ServerTransport interface { | |||
| 	// handlers will be terminated asynchronously.
 | ||||
| 	Close(err error) | ||||
| 
 | ||||
| 	// RemoteAddr returns the remote network address.
 | ||||
| 	RemoteAddr() net.Addr | ||||
| 	// Peer returns the peer of the server transport.
 | ||||
| 	Peer() *peer.Peer | ||||
| 
 | ||||
| 	// Drain notifies the client this ServerTransport stops accepting new RPCs.
 | ||||
| 	Drain(debugData string) | ||||
|  |  | |||
|  | @ -153,14 +153,16 @@ func Join(mds ...MD) MD { | |||
| type mdIncomingKey struct{} | ||||
| type mdOutgoingKey struct{} | ||||
| 
 | ||||
| // NewIncomingContext creates a new context with incoming md attached.
 | ||||
| // NewIncomingContext creates a new context with incoming md attached. md must
 | ||||
| // not be modified after calling this function.
 | ||||
| func NewIncomingContext(ctx context.Context, md MD) context.Context { | ||||
| 	return context.WithValue(ctx, mdIncomingKey{}, md) | ||||
| } | ||||
| 
 | ||||
| // NewOutgoingContext creates a new context with outgoing md attached. If used
 | ||||
| // in conjunction with AppendToOutgoingContext, NewOutgoingContext will
 | ||||
| // overwrite any previously-appended metadata.
 | ||||
| // overwrite any previously-appended metadata. md must not be modified after
 | ||||
| // calling this function.
 | ||||
| func NewOutgoingContext(ctx context.Context, md MD) context.Context { | ||||
| 	return context.WithValue(ctx, mdOutgoingKey{}, rawMD{md: md}) | ||||
| } | ||||
|  | @ -203,7 +205,8 @@ func FromIncomingContext(ctx context.Context) (MD, bool) { | |||
| } | ||||
| 
 | ||||
| // ValueFromIncomingContext returns the metadata value corresponding to the metadata
 | ||||
| // key from the incoming metadata if it exists. Key must be lower-case.
 | ||||
| // key from the incoming metadata if it exists. Keys are matched in a case insensitive
 | ||||
| // manner.
 | ||||
| //
 | ||||
| // # Experimental
 | ||||
| //
 | ||||
|  | @ -219,17 +222,16 @@ func ValueFromIncomingContext(ctx context.Context, key string) []string { | |||
| 		return copyOf(v) | ||||
| 	} | ||||
| 	for k, v := range md { | ||||
| 		// We need to manually convert all keys to lower case, because MD is a
 | ||||
| 		// map, and there's no guarantee that the MD attached to the context is
 | ||||
| 		// created using our helper functions.
 | ||||
| 		if strings.ToLower(k) == key { | ||||
| 		// Case insenitive comparison: MD is a map, and there's no guarantee
 | ||||
| 		// that the MD attached to the context is created using our helper
 | ||||
| 		// functions.
 | ||||
| 		if strings.EqualFold(k, key) { | ||||
| 			return copyOf(v) | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // the returned slice must not be modified in place
 | ||||
| func copyOf(v []string) []string { | ||||
| 	vals := make([]string, len(v)) | ||||
| 	copy(vals, v) | ||||
|  |  | |||
|  | @ -32,6 +32,8 @@ import ( | |||
| type Peer struct { | ||||
| 	// Addr is the peer address.
 | ||||
| 	Addr net.Addr | ||||
| 	// LocalAddr is the local address.
 | ||||
| 	LocalAddr net.Addr | ||||
| 	// AuthInfo is the authentication information of the transport.
 | ||||
| 	// It is nil if there is no transport security being used.
 | ||||
| 	AuthInfo credentials.AuthInfo | ||||
|  |  | |||
|  | @ -37,7 +37,6 @@ import ( | |||
| type pickerWrapper struct { | ||||
| 	mu            sync.Mutex | ||||
| 	done          bool | ||||
| 	idle          bool | ||||
| 	blockingCh    chan struct{} | ||||
| 	picker        balancer.Picker | ||||
| 	statsHandlers []stats.Handler // to record blocking picker calls
 | ||||
|  | @ -53,11 +52,7 @@ func newPickerWrapper(statsHandlers []stats.Handler) *pickerWrapper { | |||
| // updatePicker is called by UpdateBalancerState. It unblocks all blocked pick.
 | ||||
| func (pw *pickerWrapper) updatePicker(p balancer.Picker) { | ||||
| 	pw.mu.Lock() | ||||
| 	if pw.done || pw.idle { | ||||
| 		// There is a small window where a picker update from the LB policy can
 | ||||
| 		// race with the channel going to idle mode. If the picker is idle here,
 | ||||
| 		// it is because the channel asked it to do so, and therefore it is sage
 | ||||
| 		// to ignore the update from the LB policy.
 | ||||
| 	if pw.done { | ||||
| 		pw.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
|  | @ -210,23 +205,15 @@ func (pw *pickerWrapper) close() { | |||
| 	close(pw.blockingCh) | ||||
| } | ||||
| 
 | ||||
| func (pw *pickerWrapper) enterIdleMode() { | ||||
| 	pw.mu.Lock() | ||||
| 	defer pw.mu.Unlock() | ||||
| 	if pw.done { | ||||
| 		return | ||||
| 	} | ||||
| 	pw.idle = true | ||||
| } | ||||
| 
 | ||||
| func (pw *pickerWrapper) exitIdleMode() { | ||||
| // reset clears the pickerWrapper and prepares it for being used again when idle
 | ||||
| // mode is exited.
 | ||||
| func (pw *pickerWrapper) reset() { | ||||
| 	pw.mu.Lock() | ||||
| 	defer pw.mu.Unlock() | ||||
| 	if pw.done { | ||||
| 		return | ||||
| 	} | ||||
| 	pw.blockingCh = make(chan struct{}) | ||||
| 	pw.idle = false | ||||
| } | ||||
| 
 | ||||
| // dropError is a wrapper error that indicates the LB policy wishes to drop the
 | ||||
|  |  | |||
|  | @ -25,7 +25,6 @@ import ( | |||
| 
 | ||||
| 	"google.golang.org/grpc/balancer" | ||||
| 	"google.golang.org/grpc/connectivity" | ||||
| 	"google.golang.org/grpc/internal/envconfig" | ||||
| 	internalgrpclog "google.golang.org/grpc/internal/grpclog" | ||||
| 	"google.golang.org/grpc/internal/grpcrand" | ||||
| 	"google.golang.org/grpc/internal/pretty" | ||||
|  | @ -65,19 +64,6 @@ type pfConfig struct { | |||
| } | ||||
| 
 | ||||
| func (*pickfirstBuilder) ParseConfig(js json.RawMessage) (serviceconfig.LoadBalancingConfig, error) { | ||||
| 	if !envconfig.PickFirstLBConfig { | ||||
| 		// Prior to supporting loadbalancing configuration, the pick_first LB
 | ||||
| 		// policy did not implement the balancer.ConfigParser interface. This
 | ||||
| 		// meant that if a non-empty configuration was passed to it, the service
 | ||||
| 		// config unmarshaling code would throw a warning log, but would
 | ||||
| 		// continue using the pick_first LB policy. The code below ensures the
 | ||||
| 		// same behavior is retained if the env var is not set.
 | ||||
| 		if string(js) != "{}" { | ||||
| 			logger.Warningf("Ignoring non-empty balancer configuration %q for the pick_first LB policy", string(js)) | ||||
| 		} | ||||
| 		return nil, nil | ||||
| 	} | ||||
| 
 | ||||
| 	var cfg pfConfig | ||||
| 	if err := json.Unmarshal(js, &cfg); err != nil { | ||||
| 		return nil, fmt.Errorf("pickfirst: unable to unmarshal LB policy config: %s, error: %v", string(js), err) | ||||
|  |  | |||
|  | @ -0,0 +1,36 @@ | |||
| /* | ||||
|  * | ||||
|  * Copyright 2018 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| // Package dns implements a dns resolver to be installed as the default resolver
 | ||||
| // in grpc.
 | ||||
| //
 | ||||
| // Deprecated: this package is imported by grpc and should not need to be
 | ||||
| // imported directly by users.
 | ||||
| package dns | ||||
| 
 | ||||
| import ( | ||||
| 	"google.golang.org/grpc/internal/resolver/dns" | ||||
| 	"google.golang.org/grpc/resolver" | ||||
| ) | ||||
| 
 | ||||
| // NewBuilder creates a dnsBuilder which is used to factory DNS resolvers.
 | ||||
| //
 | ||||
| // Deprecated: import grpc and use resolver.Get("dns") instead.
 | ||||
| func NewBuilder() resolver.Builder { | ||||
| 	return dns.NewBuilder() | ||||
| } | ||||
|  | @ -136,3 +136,116 @@ func (a *AddressMap) Values() []any { | |||
| 	} | ||||
| 	return ret | ||||
| } | ||||
| 
 | ||||
| type endpointNode struct { | ||||
| 	addrs map[string]struct{} | ||||
| } | ||||
| 
 | ||||
| // Equal returns whether the unordered set of addrs are the same between the
 | ||||
| // endpoint nodes.
 | ||||
| func (en *endpointNode) Equal(en2 *endpointNode) bool { | ||||
| 	if len(en.addrs) != len(en2.addrs) { | ||||
| 		return false | ||||
| 	} | ||||
| 	for addr := range en.addrs { | ||||
| 		if _, ok := en2.addrs[addr]; !ok { | ||||
| 			return false | ||||
| 		} | ||||
| 	} | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| func toEndpointNode(endpoint Endpoint) endpointNode { | ||||
| 	en := make(map[string]struct{}) | ||||
| 	for _, addr := range endpoint.Addresses { | ||||
| 		en[addr.Addr] = struct{}{} | ||||
| 	} | ||||
| 	return endpointNode{ | ||||
| 		addrs: en, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // EndpointMap is a map of endpoints to arbitrary values keyed on only the
 | ||||
| // unordered set of address strings within an endpoint. This map is not thread
 | ||||
| // safe, thus it is unsafe to access concurrently. Must be created via
 | ||||
| // NewEndpointMap; do not construct directly.
 | ||||
| type EndpointMap struct { | ||||
| 	endpoints map[*endpointNode]any | ||||
| } | ||||
| 
 | ||||
| // NewEndpointMap creates a new EndpointMap.
 | ||||
| func NewEndpointMap() *EndpointMap { | ||||
| 	return &EndpointMap{ | ||||
| 		endpoints: make(map[*endpointNode]any), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Get returns the value for the address in the map, if present.
 | ||||
| func (em *EndpointMap) Get(e Endpoint) (value any, ok bool) { | ||||
| 	en := toEndpointNode(e) | ||||
| 	if endpoint := em.find(en); endpoint != nil { | ||||
| 		return em.endpoints[endpoint], true | ||||
| 	} | ||||
| 	return nil, false | ||||
| } | ||||
| 
 | ||||
| // Set updates or adds the value to the address in the map.
 | ||||
| func (em *EndpointMap) Set(e Endpoint, value any) { | ||||
| 	en := toEndpointNode(e) | ||||
| 	if endpoint := em.find(en); endpoint != nil { | ||||
| 		em.endpoints[endpoint] = value | ||||
| 		return | ||||
| 	} | ||||
| 	em.endpoints[&en] = value | ||||
| } | ||||
| 
 | ||||
| // Len returns the number of entries in the map.
 | ||||
| func (em *EndpointMap) Len() int { | ||||
| 	return len(em.endpoints) | ||||
| } | ||||
| 
 | ||||
| // Keys returns a slice of all current map keys, as endpoints specifying the
 | ||||
| // addresses present in the endpoint keys, in which uniqueness is determined by
 | ||||
| // the unordered set of addresses. Thus, endpoint information returned is not
 | ||||
| // the full endpoint data (drops duplicated addresses and attributes) but can be
 | ||||
| // used for EndpointMap accesses.
 | ||||
| func (em *EndpointMap) Keys() []Endpoint { | ||||
| 	ret := make([]Endpoint, 0, len(em.endpoints)) | ||||
| 	for en := range em.endpoints { | ||||
| 		var endpoint Endpoint | ||||
| 		for addr := range en.addrs { | ||||
| 			endpoint.Addresses = append(endpoint.Addresses, Address{Addr: addr}) | ||||
| 		} | ||||
| 		ret = append(ret, endpoint) | ||||
| 	} | ||||
| 	return ret | ||||
| } | ||||
| 
 | ||||
| // Values returns a slice of all current map values.
 | ||||
| func (em *EndpointMap) Values() []any { | ||||
| 	ret := make([]any, 0, len(em.endpoints)) | ||||
| 	for _, val := range em.endpoints { | ||||
| 		ret = append(ret, val) | ||||
| 	} | ||||
| 	return ret | ||||
| } | ||||
| 
 | ||||
| // find returns a pointer to the endpoint node in em if the endpoint node is
 | ||||
| // already present. If not found, nil is returned. The comparisons are done on
 | ||||
| // the unordered set of addresses within an endpoint.
 | ||||
| func (em EndpointMap) find(e endpointNode) *endpointNode { | ||||
| 	for endpoint := range em.endpoints { | ||||
| 		if e.Equal(endpoint) { | ||||
| 			return endpoint | ||||
| 		} | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // Delete removes the specified endpoint from the map.
 | ||||
| func (em *EndpointMap) Delete(e Endpoint) { | ||||
| 	en := toEndpointNode(e) | ||||
| 	if entry := em.find(en); entry != nil { | ||||
| 		delete(em.endpoints, entry) | ||||
| 	} | ||||
| } | ||||
|  |  | |||
|  | @ -240,11 +240,6 @@ type ClientConn interface { | |||
| 	//
 | ||||
| 	// Deprecated: Use UpdateState instead.
 | ||||
| 	NewAddress(addresses []Address) | ||||
| 	// NewServiceConfig is called by resolver to notify ClientConn a new
 | ||||
| 	// service config. The service config should be provided as a json string.
 | ||||
| 	//
 | ||||
| 	// Deprecated: Use UpdateState instead.
 | ||||
| 	NewServiceConfig(serviceConfig string) | ||||
| 	// ParseServiceConfig parses the provided service config and returns an
 | ||||
| 	// object that provides the parsed config.
 | ||||
| 	ParseServiceConfig(serviceConfigJSON string) *serviceconfig.ParseResult | ||||
|  | @ -286,6 +281,11 @@ func (t Target) Endpoint() string { | |||
| 	return strings.TrimPrefix(endpoint, "/") | ||||
| } | ||||
| 
 | ||||
| // String returns a string representation of Target.
 | ||||
| func (t Target) String() string { | ||||
| 	return t.URL.String() | ||||
| } | ||||
| 
 | ||||
| // Builder creates a resolver that will be used to watch name resolution updates.
 | ||||
| type Builder interface { | ||||
| 	// Build creates a new resolver for the given target.
 | ||||
|  |  | |||
|  | @ -1,247 +0,0 @@ | |||
| /* | ||||
|  * | ||||
|  * Copyright 2017 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| package grpc | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/balancer" | ||||
| 	"google.golang.org/grpc/internal/channelz" | ||||
| 	"google.golang.org/grpc/internal/grpcsync" | ||||
| 	"google.golang.org/grpc/internal/pretty" | ||||
| 	"google.golang.org/grpc/resolver" | ||||
| 	"google.golang.org/grpc/serviceconfig" | ||||
| ) | ||||
| 
 | ||||
| // resolverStateUpdater wraps the single method used by ccResolverWrapper to
 | ||||
| // report a state update from the actual resolver implementation.
 | ||||
| type resolverStateUpdater interface { | ||||
| 	updateResolverState(s resolver.State, err error) error | ||||
| } | ||||
| 
 | ||||
| // ccResolverWrapper is a wrapper on top of cc for resolvers.
 | ||||
| // It implements resolver.ClientConn interface.
 | ||||
| type ccResolverWrapper struct { | ||||
| 	// The following fields are initialized when the wrapper is created and are
 | ||||
| 	// read-only afterwards, and therefore can be accessed without a mutex.
 | ||||
| 	cc                  resolverStateUpdater | ||||
| 	channelzID          *channelz.Identifier | ||||
| 	ignoreServiceConfig bool | ||||
| 	opts                ccResolverWrapperOpts | ||||
| 	serializer          *grpcsync.CallbackSerializer // To serialize all incoming calls.
 | ||||
| 	serializerCancel    context.CancelFunc           // To close the serializer, accessed only from close().
 | ||||
| 
 | ||||
| 	// All incoming (resolver --> gRPC) calls are guaranteed to execute in a
 | ||||
| 	// mutually exclusive manner as they are scheduled on the serializer.
 | ||||
| 	// Fields accessed *only* in these serializer callbacks, can therefore be
 | ||||
| 	// accessed without a mutex.
 | ||||
| 	curState resolver.State | ||||
| 
 | ||||
| 	// mu guards access to the below fields.
 | ||||
| 	mu       sync.Mutex | ||||
| 	closed   bool | ||||
| 	resolver resolver.Resolver // Accessed only from outgoing calls.
 | ||||
| } | ||||
| 
 | ||||
| // ccResolverWrapperOpts wraps the arguments to be passed when creating a new
 | ||||
| // ccResolverWrapper.
 | ||||
| type ccResolverWrapperOpts struct { | ||||
| 	target     resolver.Target       // User specified dial target to resolve.
 | ||||
| 	builder    resolver.Builder      // Resolver builder to use.
 | ||||
| 	bOpts      resolver.BuildOptions // Resolver build options to use.
 | ||||
| 	channelzID *channelz.Identifier  // Channelz identifier for the channel.
 | ||||
| } | ||||
| 
 | ||||
| // newCCResolverWrapper uses the resolver.Builder to build a Resolver and
 | ||||
| // returns a ccResolverWrapper object which wraps the newly built resolver.
 | ||||
| func newCCResolverWrapper(cc resolverStateUpdater, opts ccResolverWrapperOpts) (*ccResolverWrapper, error) { | ||||
| 	ctx, cancel := context.WithCancel(context.Background()) | ||||
| 	ccr := &ccResolverWrapper{ | ||||
| 		cc:                  cc, | ||||
| 		channelzID:          opts.channelzID, | ||||
| 		ignoreServiceConfig: opts.bOpts.DisableServiceConfig, | ||||
| 		opts:                opts, | ||||
| 		serializer:          grpcsync.NewCallbackSerializer(ctx), | ||||
| 		serializerCancel:    cancel, | ||||
| 	} | ||||
| 
 | ||||
| 	// Cannot hold the lock at build time because the resolver can send an
 | ||||
| 	// update or error inline and these incoming calls grab the lock to schedule
 | ||||
| 	// a callback in the serializer.
 | ||||
| 	r, err := opts.builder.Build(opts.target, ccr, opts.bOpts) | ||||
| 	if err != nil { | ||||
| 		cancel() | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	// Any error reported by the resolver at build time that leads to a
 | ||||
| 	// re-resolution request from the balancer is dropped by grpc until we
 | ||||
| 	// return from this function. So, we don't have to handle pending resolveNow
 | ||||
| 	// requests here.
 | ||||
| 	ccr.mu.Lock() | ||||
| 	ccr.resolver = r | ||||
| 	ccr.mu.Unlock() | ||||
| 
 | ||||
| 	return ccr, nil | ||||
| } | ||||
| 
 | ||||
| func (ccr *ccResolverWrapper) resolveNow(o resolver.ResolveNowOptions) { | ||||
| 	ccr.mu.Lock() | ||||
| 	defer ccr.mu.Unlock() | ||||
| 
 | ||||
| 	// ccr.resolver field is set only after the call to Build() returns. But in
 | ||||
| 	// the process of building, the resolver may send an error update which when
 | ||||
| 	// propagated to the balancer may result in a re-resolution request.
 | ||||
| 	if ccr.closed || ccr.resolver == nil { | ||||
| 		return | ||||
| 	} | ||||
| 	ccr.resolver.ResolveNow(o) | ||||
| } | ||||
| 
 | ||||
| func (ccr *ccResolverWrapper) close() { | ||||
| 	ccr.mu.Lock() | ||||
| 	if ccr.closed { | ||||
| 		ccr.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	channelz.Info(logger, ccr.channelzID, "Closing the name resolver") | ||||
| 
 | ||||
| 	// Close the serializer to ensure that no more calls from the resolver are
 | ||||
| 	// handled, before actually closing the resolver.
 | ||||
| 	ccr.serializerCancel() | ||||
| 	ccr.closed = true | ||||
| 	r := ccr.resolver | ||||
| 	ccr.mu.Unlock() | ||||
| 
 | ||||
| 	// Give enqueued callbacks a chance to finish.
 | ||||
| 	<-ccr.serializer.Done() | ||||
| 
 | ||||
| 	// Spawn a goroutine to close the resolver (since it may block trying to
 | ||||
| 	// cleanup all allocated resources) and return early.
 | ||||
| 	go r.Close() | ||||
| } | ||||
| 
 | ||||
| // serializerScheduleLocked is a convenience method to schedule a function to be
 | ||||
| // run on the serializer while holding ccr.mu.
 | ||||
| func (ccr *ccResolverWrapper) serializerScheduleLocked(f func(context.Context)) { | ||||
| 	ccr.mu.Lock() | ||||
| 	ccr.serializer.Schedule(f) | ||||
| 	ccr.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| // UpdateState is called by resolver implementations to report new state to gRPC
 | ||||
| // which includes addresses and service config.
 | ||||
| func (ccr *ccResolverWrapper) UpdateState(s resolver.State) error { | ||||
| 	errCh := make(chan error, 1) | ||||
| 	if s.Endpoints == nil { | ||||
| 		s.Endpoints = make([]resolver.Endpoint, 0, len(s.Addresses)) | ||||
| 		for _, a := range s.Addresses { | ||||
| 			ep := resolver.Endpoint{Addresses: []resolver.Address{a}, Attributes: a.BalancerAttributes} | ||||
| 			ep.Addresses[0].BalancerAttributes = nil | ||||
| 			s.Endpoints = append(s.Endpoints, ep) | ||||
| 		} | ||||
| 	} | ||||
| 	ok := ccr.serializer.Schedule(func(context.Context) { | ||||
| 		ccr.addChannelzTraceEvent(s) | ||||
| 		ccr.curState = s | ||||
| 		if err := ccr.cc.updateResolverState(ccr.curState, nil); err == balancer.ErrBadResolverState { | ||||
| 			errCh <- balancer.ErrBadResolverState | ||||
| 			return | ||||
| 		} | ||||
| 		errCh <- nil | ||||
| 	}) | ||||
| 	if !ok { | ||||
| 		// The only time when Schedule() fail to add the callback to the
 | ||||
| 		// serializer is when the serializer is closed, and this happens only
 | ||||
| 		// when the resolver wrapper is closed.
 | ||||
| 		return nil | ||||
| 	} | ||||
| 	return <-errCh | ||||
| } | ||||
| 
 | ||||
| // ReportError is called by resolver implementations to report errors
 | ||||
| // encountered during name resolution to gRPC.
 | ||||
| func (ccr *ccResolverWrapper) ReportError(err error) { | ||||
| 	ccr.serializerScheduleLocked(func(_ context.Context) { | ||||
| 		channelz.Warningf(logger, ccr.channelzID, "ccResolverWrapper: reporting error to cc: %v", err) | ||||
| 		ccr.cc.updateResolverState(resolver.State{}, err) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // NewAddress is called by the resolver implementation to send addresses to
 | ||||
| // gRPC.
 | ||||
| func (ccr *ccResolverWrapper) NewAddress(addrs []resolver.Address) { | ||||
| 	ccr.serializerScheduleLocked(func(_ context.Context) { | ||||
| 		ccr.addChannelzTraceEvent(resolver.State{Addresses: addrs, ServiceConfig: ccr.curState.ServiceConfig}) | ||||
| 		ccr.curState.Addresses = addrs | ||||
| 		ccr.cc.updateResolverState(ccr.curState, nil) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // NewServiceConfig is called by the resolver implementation to send service
 | ||||
| // configs to gRPC.
 | ||||
| func (ccr *ccResolverWrapper) NewServiceConfig(sc string) { | ||||
| 	ccr.serializerScheduleLocked(func(_ context.Context) { | ||||
| 		channelz.Infof(logger, ccr.channelzID, "ccResolverWrapper: got new service config: %s", sc) | ||||
| 		if ccr.ignoreServiceConfig { | ||||
| 			channelz.Info(logger, ccr.channelzID, "Service config lookups disabled; ignoring config") | ||||
| 			return | ||||
| 		} | ||||
| 		scpr := parseServiceConfig(sc) | ||||
| 		if scpr.Err != nil { | ||||
| 			channelz.Warningf(logger, ccr.channelzID, "ccResolverWrapper: error parsing service config: %v", scpr.Err) | ||||
| 			return | ||||
| 		} | ||||
| 		ccr.addChannelzTraceEvent(resolver.State{Addresses: ccr.curState.Addresses, ServiceConfig: scpr}) | ||||
| 		ccr.curState.ServiceConfig = scpr | ||||
| 		ccr.cc.updateResolverState(ccr.curState, nil) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // ParseServiceConfig is called by resolver implementations to parse a JSON
 | ||||
| // representation of the service config.
 | ||||
| func (ccr *ccResolverWrapper) ParseServiceConfig(scJSON string) *serviceconfig.ParseResult { | ||||
| 	return parseServiceConfig(scJSON) | ||||
| } | ||||
| 
 | ||||
| // addChannelzTraceEvent adds a channelz trace event containing the new
 | ||||
| // state received from resolver implementations.
 | ||||
| func (ccr *ccResolverWrapper) addChannelzTraceEvent(s resolver.State) { | ||||
| 	var updates []string | ||||
| 	var oldSC, newSC *ServiceConfig | ||||
| 	var oldOK, newOK bool | ||||
| 	if ccr.curState.ServiceConfig != nil { | ||||
| 		oldSC, oldOK = ccr.curState.ServiceConfig.Config.(*ServiceConfig) | ||||
| 	} | ||||
| 	if s.ServiceConfig != nil { | ||||
| 		newSC, newOK = s.ServiceConfig.Config.(*ServiceConfig) | ||||
| 	} | ||||
| 	if oldOK != newOK || (oldOK && newOK && oldSC.rawJSONString != newSC.rawJSONString) { | ||||
| 		updates = append(updates, "service config updated") | ||||
| 	} | ||||
| 	if len(ccr.curState.Addresses) > 0 && len(s.Addresses) == 0 { | ||||
| 		updates = append(updates, "resolver returned an empty address list") | ||||
| 	} else if len(ccr.curState.Addresses) == 0 && len(s.Addresses) > 0 { | ||||
| 		updates = append(updates, "resolver returned new addresses") | ||||
| 	} | ||||
| 	channelz.Infof(logger, ccr.channelzID, "Resolver state updated: %s (%v)", pretty.ToJSON(s), strings.Join(updates, "; ")) | ||||
| } | ||||
|  | @ -0,0 +1,197 @@ | |||
| /* | ||||
|  * | ||||
|  * Copyright 2017 gRPC authors. | ||||
|  * | ||||
|  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||
|  * you may not use this file except in compliance with the License. | ||||
|  * You may obtain a copy of the License at | ||||
|  * | ||||
|  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||||
|  * | ||||
|  * Unless required by applicable law or agreed to in writing, software | ||||
|  * distributed under the License is distributed on an "AS IS" BASIS, | ||||
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  * | ||||
|  */ | ||||
| 
 | ||||
| package grpc | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/internal/channelz" | ||||
| 	"google.golang.org/grpc/internal/grpcsync" | ||||
| 	"google.golang.org/grpc/internal/pretty" | ||||
| 	"google.golang.org/grpc/resolver" | ||||
| 	"google.golang.org/grpc/serviceconfig" | ||||
| ) | ||||
| 
 | ||||
| // ccResolverWrapper is a wrapper on top of cc for resolvers.
 | ||||
| // It implements resolver.ClientConn interface.
 | ||||
| type ccResolverWrapper struct { | ||||
| 	// The following fields are initialized when the wrapper is created and are
 | ||||
| 	// read-only afterwards, and therefore can be accessed without a mutex.
 | ||||
| 	cc                  *ClientConn | ||||
| 	ignoreServiceConfig bool | ||||
| 	serializer          *grpcsync.CallbackSerializer | ||||
| 	serializerCancel    context.CancelFunc | ||||
| 
 | ||||
| 	resolver resolver.Resolver // only accessed within the serializer
 | ||||
| 
 | ||||
| 	// The following fields are protected by mu.  Caller must take cc.mu before
 | ||||
| 	// taking mu.
 | ||||
| 	mu       sync.Mutex | ||||
| 	curState resolver.State | ||||
| 	closed   bool | ||||
| } | ||||
| 
 | ||||
| // newCCResolverWrapper initializes the ccResolverWrapper.  It can only be used
 | ||||
| // after calling start, which builds the resolver.
 | ||||
| func newCCResolverWrapper(cc *ClientConn) *ccResolverWrapper { | ||||
| 	ctx, cancel := context.WithCancel(cc.ctx) | ||||
| 	return &ccResolverWrapper{ | ||||
| 		cc:                  cc, | ||||
| 		ignoreServiceConfig: cc.dopts.disableServiceConfig, | ||||
| 		serializer:          grpcsync.NewCallbackSerializer(ctx), | ||||
| 		serializerCancel:    cancel, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // start builds the name resolver using the resolver.Builder in cc and returns
 | ||||
| // any error encountered.  It must always be the first operation performed on
 | ||||
| // any newly created ccResolverWrapper, except that close may be called instead.
 | ||||
| func (ccr *ccResolverWrapper) start() error { | ||||
| 	errCh := make(chan error) | ||||
| 	ccr.serializer.Schedule(func(ctx context.Context) { | ||||
| 		if ctx.Err() != nil { | ||||
| 			return | ||||
| 		} | ||||
| 		opts := resolver.BuildOptions{ | ||||
| 			DisableServiceConfig: ccr.cc.dopts.disableServiceConfig, | ||||
| 			DialCreds:            ccr.cc.dopts.copts.TransportCredentials, | ||||
| 			CredsBundle:          ccr.cc.dopts.copts.CredsBundle, | ||||
| 			Dialer:               ccr.cc.dopts.copts.Dialer, | ||||
| 		} | ||||
| 		var err error | ||||
| 		ccr.resolver, err = ccr.cc.resolverBuilder.Build(ccr.cc.parsedTarget, ccr, opts) | ||||
| 		errCh <- err | ||||
| 	}) | ||||
| 	return <-errCh | ||||
| } | ||||
| 
 | ||||
| func (ccr *ccResolverWrapper) resolveNow(o resolver.ResolveNowOptions) { | ||||
| 	ccr.serializer.Schedule(func(ctx context.Context) { | ||||
| 		if ctx.Err() != nil || ccr.resolver == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		ccr.resolver.ResolveNow(o) | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // close initiates async shutdown of the wrapper.  To determine the wrapper has
 | ||||
| // finished shutting down, the channel should block on ccr.serializer.Done()
 | ||||
| // without cc.mu held.
 | ||||
| func (ccr *ccResolverWrapper) close() { | ||||
| 	channelz.Info(logger, ccr.cc.channelzID, "Closing the name resolver") | ||||
| 	ccr.mu.Lock() | ||||
| 	ccr.closed = true | ||||
| 	ccr.mu.Unlock() | ||||
| 
 | ||||
| 	ccr.serializer.Schedule(func(context.Context) { | ||||
| 		if ccr.resolver == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		ccr.resolver.Close() | ||||
| 		ccr.resolver = nil | ||||
| 	}) | ||||
| 	ccr.serializerCancel() | ||||
| } | ||||
| 
 | ||||
| // UpdateState is called by resolver implementations to report new state to gRPC
 | ||||
| // which includes addresses and service config.
 | ||||
| func (ccr *ccResolverWrapper) UpdateState(s resolver.State) error { | ||||
| 	ccr.cc.mu.Lock() | ||||
| 	ccr.mu.Lock() | ||||
| 	if ccr.closed { | ||||
| 		ccr.mu.Unlock() | ||||
| 		ccr.cc.mu.Unlock() | ||||
| 		return nil | ||||
| 	} | ||||
| 	if s.Endpoints == nil { | ||||
| 		s.Endpoints = make([]resolver.Endpoint, 0, len(s.Addresses)) | ||||
| 		for _, a := range s.Addresses { | ||||
| 			ep := resolver.Endpoint{Addresses: []resolver.Address{a}, Attributes: a.BalancerAttributes} | ||||
| 			ep.Addresses[0].BalancerAttributes = nil | ||||
| 			s.Endpoints = append(s.Endpoints, ep) | ||||
| 		} | ||||
| 	} | ||||
| 	ccr.addChannelzTraceEvent(s) | ||||
| 	ccr.curState = s | ||||
| 	ccr.mu.Unlock() | ||||
| 	return ccr.cc.updateResolverStateAndUnlock(s, nil) | ||||
| } | ||||
| 
 | ||||
| // ReportError is called by resolver implementations to report errors
 | ||||
| // encountered during name resolution to gRPC.
 | ||||
| func (ccr *ccResolverWrapper) ReportError(err error) { | ||||
| 	ccr.cc.mu.Lock() | ||||
| 	ccr.mu.Lock() | ||||
| 	if ccr.closed { | ||||
| 		ccr.mu.Unlock() | ||||
| 		ccr.cc.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 	ccr.mu.Unlock() | ||||
| 	channelz.Warningf(logger, ccr.cc.channelzID, "ccResolverWrapper: reporting error to cc: %v", err) | ||||
| 	ccr.cc.updateResolverStateAndUnlock(resolver.State{}, err) | ||||
| } | ||||
| 
 | ||||
| // NewAddress is called by the resolver implementation to send addresses to
 | ||||
| // gRPC.
 | ||||
| func (ccr *ccResolverWrapper) NewAddress(addrs []resolver.Address) { | ||||
| 	ccr.cc.mu.Lock() | ||||
| 	ccr.mu.Lock() | ||||
| 	if ccr.closed { | ||||
| 		ccr.mu.Unlock() | ||||
| 		ccr.cc.mu.Unlock() | ||||
| 		return | ||||
| 	} | ||||
| 	s := resolver.State{Addresses: addrs, ServiceConfig: ccr.curState.ServiceConfig} | ||||
| 	ccr.addChannelzTraceEvent(s) | ||||
| 	ccr.curState = s | ||||
| 	ccr.mu.Unlock() | ||||
| 	ccr.cc.updateResolverStateAndUnlock(s, nil) | ||||
| } | ||||
| 
 | ||||
| // ParseServiceConfig is called by resolver implementations to parse a JSON
 | ||||
| // representation of the service config.
 | ||||
| func (ccr *ccResolverWrapper) ParseServiceConfig(scJSON string) *serviceconfig.ParseResult { | ||||
| 	return parseServiceConfig(scJSON) | ||||
| } | ||||
| 
 | ||||
| // addChannelzTraceEvent adds a channelz trace event containing the new
 | ||||
| // state received from resolver implementations.
 | ||||
| func (ccr *ccResolverWrapper) addChannelzTraceEvent(s resolver.State) { | ||||
| 	var updates []string | ||||
| 	var oldSC, newSC *ServiceConfig | ||||
| 	var oldOK, newOK bool | ||||
| 	if ccr.curState.ServiceConfig != nil { | ||||
| 		oldSC, oldOK = ccr.curState.ServiceConfig.Config.(*ServiceConfig) | ||||
| 	} | ||||
| 	if s.ServiceConfig != nil { | ||||
| 		newSC, newOK = s.ServiceConfig.Config.(*ServiceConfig) | ||||
| 	} | ||||
| 	if oldOK != newOK || (oldOK && newOK && oldSC.rawJSONString != newSC.rawJSONString) { | ||||
| 		updates = append(updates, "service config updated") | ||||
| 	} | ||||
| 	if len(ccr.curState.Addresses) > 0 && len(s.Addresses) == 0 { | ||||
| 		updates = append(updates, "resolver returned an empty address list") | ||||
| 	} else if len(ccr.curState.Addresses) == 0 && len(s.Addresses) > 0 { | ||||
| 		updates = append(updates, "resolver returned new addresses") | ||||
| 	} | ||||
| 	channelz.Infof(logger, ccr.cc.channelzID, "Resolver state updated: %s (%v)", pretty.ToJSON(s), strings.Join(updates, "; ")) | ||||
| } | ||||
|  | @ -70,6 +70,10 @@ func init() { | |||
| 	internal.GetServerCredentials = func(srv *Server) credentials.TransportCredentials { | ||||
| 		return srv.opts.creds | ||||
| 	} | ||||
| 	internal.IsRegisteredMethod = func(srv *Server, method string) bool { | ||||
| 		return srv.isRegisteredMethod(method) | ||||
| 	} | ||||
| 	internal.ServerFromContext = serverFromContext | ||||
| 	internal.DrainServerTransports = func(srv *Server, addr string) { | ||||
| 		srv.drainServerTransports(addr) | ||||
| 	} | ||||
|  | @ -81,6 +85,7 @@ func init() { | |||
| 	} | ||||
| 	internal.BinaryLogger = binaryLogger | ||||
| 	internal.JoinServerOptions = newJoinServerOption | ||||
| 	internal.RecvBufferPool = recvBufferPool | ||||
| } | ||||
| 
 | ||||
| var statusOK = status.New(codes.OK, "") | ||||
|  | @ -139,7 +144,8 @@ type Server struct { | |||
| 	channelzID *channelz.Identifier | ||||
| 	czData     *channelzData | ||||
| 
 | ||||
| 	serverWorkerChannel chan func() | ||||
| 	serverWorkerChannel      chan func() | ||||
| 	serverWorkerChannelClose func() | ||||
| } | ||||
| 
 | ||||
| type serverOptions struct { | ||||
|  | @ -578,11 +584,13 @@ func NumStreamWorkers(numServerWorkers uint32) ServerOption { | |||
| // options are used: StatsHandler, EnableTracing, or binary logging. In such
 | ||||
| // cases, the shared buffer pool will be ignored.
 | ||||
| //
 | ||||
| // # Experimental
 | ||||
| //
 | ||||
| // Notice: This API is EXPERIMENTAL and may be changed or removed in a
 | ||||
| // later release.
 | ||||
| // Deprecated: use experimental.WithRecvBufferPool instead.  Will be deleted in
 | ||||
| // v1.60.0 or later.
 | ||||
| func RecvBufferPool(bufferPool SharedBufferPool) ServerOption { | ||||
| 	return recvBufferPool(bufferPool) | ||||
| } | ||||
| 
 | ||||
| func recvBufferPool(bufferPool SharedBufferPool) ServerOption { | ||||
| 	return newFuncServerOption(func(o *serverOptions) { | ||||
| 		o.recvBufferPool = bufferPool | ||||
| 	}) | ||||
|  | @ -616,15 +624,14 @@ func (s *Server) serverWorker() { | |||
| // connections to reduce the time spent overall on runtime.morestack.
 | ||||
| func (s *Server) initServerWorkers() { | ||||
| 	s.serverWorkerChannel = make(chan func()) | ||||
| 	s.serverWorkerChannelClose = grpcsync.OnceFunc(func() { | ||||
| 		close(s.serverWorkerChannel) | ||||
| 	}) | ||||
| 	for i := uint32(0); i < s.opts.numServerWorkers; i++ { | ||||
| 		go s.serverWorker() | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (s *Server) stopServerWorkers() { | ||||
| 	close(s.serverWorkerChannel) | ||||
| } | ||||
| 
 | ||||
| // NewServer creates a gRPC server which has no service registered and has not
 | ||||
| // started to accept requests yet.
 | ||||
| func NewServer(opt ...ServerOption) *Server { | ||||
|  | @ -806,6 +813,18 @@ func (l *listenSocket) Close() error { | |||
| // Serve returns when lis.Accept fails with fatal errors.  lis will be closed when
 | ||||
| // this method returns.
 | ||||
| // Serve will return a non-nil error unless Stop or GracefulStop is called.
 | ||||
| //
 | ||||
| // Note: All supported releases of Go (as of December 2023) override the OS
 | ||||
| // defaults for TCP keepalive time and interval to 15s. To enable TCP keepalive
 | ||||
| // with OS defaults for keepalive time and interval, callers need to do the
 | ||||
| // following two things:
 | ||||
| //   - pass a net.Listener created by calling the Listen method on a
 | ||||
| //     net.ListenConfig with the `KeepAlive` field set to a negative value. This
 | ||||
| //     will result in the Go standard library not overriding OS defaults for TCP
 | ||||
| //     keepalive interval and time. But this will also result in the Go standard
 | ||||
| //     library not enabling TCP keepalives by default.
 | ||||
| //   - override the Accept method on the passed in net.Listener and set the
 | ||||
| //     SO_KEEPALIVE socket option to enable TCP keepalives, with OS defaults.
 | ||||
| func (s *Server) Serve(lis net.Listener) error { | ||||
| 	s.mu.Lock() | ||||
| 	s.printf("serving") | ||||
|  | @ -917,7 +936,7 @@ func (s *Server) handleRawConn(lisAddr string, rawConn net.Conn) { | |||
| 		return | ||||
| 	} | ||||
| 	go func() { | ||||
| 		s.serveStreams(st) | ||||
| 		s.serveStreams(context.Background(), st, rawConn) | ||||
| 		s.removeConn(lisAddr, st) | ||||
| 	}() | ||||
| } | ||||
|  | @ -971,18 +990,29 @@ func (s *Server) newHTTP2Transport(c net.Conn) transport.ServerTransport { | |||
| 	return st | ||||
| } | ||||
| 
 | ||||
| func (s *Server) serveStreams(st transport.ServerTransport) { | ||||
| 	defer st.Close(errors.New("finished serving streams for the server transport")) | ||||
| 	var wg sync.WaitGroup | ||||
| func (s *Server) serveStreams(ctx context.Context, st transport.ServerTransport, rawConn net.Conn) { | ||||
| 	ctx = transport.SetConnection(ctx, rawConn) | ||||
| 	ctx = peer.NewContext(ctx, st.Peer()) | ||||
| 	for _, sh := range s.opts.statsHandlers { | ||||
| 		ctx = sh.TagConn(ctx, &stats.ConnTagInfo{ | ||||
| 			RemoteAddr: st.Peer().Addr, | ||||
| 			LocalAddr:  st.Peer().LocalAddr, | ||||
| 		}) | ||||
| 		sh.HandleConn(ctx, &stats.ConnBegin{}) | ||||
| 	} | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		st.Close(errors.New("finished serving streams for the server transport")) | ||||
| 		for _, sh := range s.opts.statsHandlers { | ||||
| 			sh.HandleConn(ctx, &stats.ConnEnd{}) | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	streamQuota := newHandlerQuota(s.opts.maxConcurrentStreams) | ||||
| 	st.HandleStreams(func(stream *transport.Stream) { | ||||
| 		wg.Add(1) | ||||
| 
 | ||||
| 	st.HandleStreams(ctx, func(stream *transport.Stream) { | ||||
| 		streamQuota.acquire() | ||||
| 		f := func() { | ||||
| 			defer streamQuota.release() | ||||
| 			defer wg.Done() | ||||
| 			s.handleStream(st, stream) | ||||
| 		} | ||||
| 
 | ||||
|  | @ -996,7 +1026,6 @@ func (s *Server) serveStreams(st transport.ServerTransport) { | |||
| 		} | ||||
| 		go f() | ||||
| 	}) | ||||
| 	wg.Wait() | ||||
| } | ||||
| 
 | ||||
| var _ http.Handler = (*Server)(nil) | ||||
|  | @ -1040,7 +1069,7 @@ func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) { | |||
| 		return | ||||
| 	} | ||||
| 	defer s.removeConn(listenerAddressForServeHTTP, st) | ||||
| 	s.serveStreams(st) | ||||
| 	s.serveStreams(r.Context(), st, nil) | ||||
| } | ||||
| 
 | ||||
| func (s *Server) addConn(addr string, st transport.ServerTransport) bool { | ||||
|  | @ -1689,6 +1718,7 @@ func (s *Server) processStreamingRPC(ctx context.Context, t transport.ServerTran | |||
| 
 | ||||
| func (s *Server) handleStream(t transport.ServerTransport, stream *transport.Stream) { | ||||
| 	ctx := stream.Context() | ||||
| 	ctx = contextWithServer(ctx, s) | ||||
| 	var ti *traceInfo | ||||
| 	if EnableTracing { | ||||
| 		tr := trace.New("grpc.Recv."+methodFamily(stream.Method()), stream.Method()) | ||||
|  | @ -1697,7 +1727,7 @@ func (s *Server) handleStream(t transport.ServerTransport, stream *transport.Str | |||
| 			tr: tr, | ||||
| 			firstLine: firstLine{ | ||||
| 				client:     false, | ||||
| 				remoteAddr: t.RemoteAddr(), | ||||
| 				remoteAddr: t.Peer().Addr, | ||||
| 			}, | ||||
| 		} | ||||
| 		if dl, ok := ctx.Deadline(); ok { | ||||
|  | @ -1731,6 +1761,22 @@ func (s *Server) handleStream(t transport.ServerTransport, stream *transport.Str | |||
| 	service := sm[:pos] | ||||
| 	method := sm[pos+1:] | ||||
| 
 | ||||
| 	md, _ := metadata.FromIncomingContext(ctx) | ||||
| 	for _, sh := range s.opts.statsHandlers { | ||||
| 		ctx = sh.TagRPC(ctx, &stats.RPCTagInfo{FullMethodName: stream.Method()}) | ||||
| 		sh.HandleRPC(ctx, &stats.InHeader{ | ||||
| 			FullMethod:  stream.Method(), | ||||
| 			RemoteAddr:  t.Peer().Addr, | ||||
| 			LocalAddr:   t.Peer().LocalAddr, | ||||
| 			Compression: stream.RecvCompress(), | ||||
| 			WireLength:  stream.HeaderWireLength(), | ||||
| 			Header:      md, | ||||
| 		}) | ||||
| 	} | ||||
| 	// To have calls in stream callouts work. Will delete once all stats handler
 | ||||
| 	// calls come from the gRPC layer.
 | ||||
| 	stream.SetContext(ctx) | ||||
| 
 | ||||
| 	srv, knownService := s.services[service] | ||||
| 	if knownService { | ||||
| 		if md, ok := srv.methods[method]; ok { | ||||
|  | @ -1820,62 +1866,68 @@ func ServerTransportStreamFromContext(ctx context.Context) ServerTransportStream | |||
| // pending RPCs on the client side will get notified by connection
 | ||||
| // errors.
 | ||||
| func (s *Server) Stop() { | ||||
| 	s.quit.Fire() | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		s.serveWG.Wait() | ||||
| 		s.done.Fire() | ||||
| 	}() | ||||
| 
 | ||||
| 	s.channelzRemoveOnce.Do(func() { channelz.RemoveEntry(s.channelzID) }) | ||||
| 
 | ||||
| 	s.mu.Lock() | ||||
| 	listeners := s.lis | ||||
| 	s.lis = nil | ||||
| 	conns := s.conns | ||||
| 	s.conns = nil | ||||
| 	// interrupt GracefulStop if Stop and GracefulStop are called concurrently.
 | ||||
| 	s.cv.Broadcast() | ||||
| 	s.mu.Unlock() | ||||
| 
 | ||||
| 	for lis := range listeners { | ||||
| 		lis.Close() | ||||
| 	} | ||||
| 	for _, cs := range conns { | ||||
| 		for st := range cs { | ||||
| 			st.Close(errors.New("Server.Stop called")) | ||||
| 		} | ||||
| 	} | ||||
| 	if s.opts.numServerWorkers > 0 { | ||||
| 		s.stopServerWorkers() | ||||
| 	} | ||||
| 
 | ||||
| 	s.mu.Lock() | ||||
| 	if s.events != nil { | ||||
| 		s.events.Finish() | ||||
| 		s.events = nil | ||||
| 	} | ||||
| 	s.mu.Unlock() | ||||
| 	s.stop(false) | ||||
| } | ||||
| 
 | ||||
| // GracefulStop stops the gRPC server gracefully. It stops the server from
 | ||||
| // accepting new connections and RPCs and blocks until all the pending RPCs are
 | ||||
| // finished.
 | ||||
| func (s *Server) GracefulStop() { | ||||
| 	s.stop(true) | ||||
| } | ||||
| 
 | ||||
| func (s *Server) stop(graceful bool) { | ||||
| 	s.quit.Fire() | ||||
| 	defer s.done.Fire() | ||||
| 
 | ||||
| 	s.channelzRemoveOnce.Do(func() { channelz.RemoveEntry(s.channelzID) }) | ||||
| 
 | ||||
| 	s.mu.Lock() | ||||
| 	if s.conns == nil { | ||||
| 		s.mu.Unlock() | ||||
| 		return | ||||
| 	s.closeListenersLocked() | ||||
| 	// Wait for serving threads to be ready to exit.  Only then can we be sure no
 | ||||
| 	// new conns will be created.
 | ||||
| 	s.mu.Unlock() | ||||
| 	s.serveWG.Wait() | ||||
| 
 | ||||
| 	s.mu.Lock() | ||||
| 	defer s.mu.Unlock() | ||||
| 
 | ||||
| 	if graceful { | ||||
| 		s.drainAllServerTransportsLocked() | ||||
| 	} else { | ||||
| 		s.closeServerTransportsLocked() | ||||
| 	} | ||||
| 
 | ||||
| 	for lis := range s.lis { | ||||
| 		lis.Close() | ||||
| 	for len(s.conns) != 0 { | ||||
| 		s.cv.Wait() | ||||
| 	} | ||||
| 	s.lis = nil | ||||
| 	s.conns = nil | ||||
| 
 | ||||
| 	if s.opts.numServerWorkers > 0 { | ||||
| 		// Closing the channel (only once, via grpcsync.OnceFunc) after all the
 | ||||
| 		// connections have been closed above ensures that there are no
 | ||||
| 		// goroutines executing the callback passed to st.HandleStreams (where
 | ||||
| 		// the channel is written to).
 | ||||
| 		s.serverWorkerChannelClose() | ||||
| 	} | ||||
| 
 | ||||
| 	if s.events != nil { | ||||
| 		s.events.Finish() | ||||
| 		s.events = nil | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // s.mu must be held by the caller.
 | ||||
| func (s *Server) closeServerTransportsLocked() { | ||||
| 	for _, conns := range s.conns { | ||||
| 		for st := range conns { | ||||
| 			st.Close(errors.New("Server.Stop called")) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // s.mu must be held by the caller.
 | ||||
| func (s *Server) drainAllServerTransportsLocked() { | ||||
| 	if !s.drain { | ||||
| 		for _, conns := range s.conns { | ||||
| 			for st := range conns { | ||||
|  | @ -1884,22 +1936,14 @@ func (s *Server) GracefulStop() { | |||
| 		} | ||||
| 		s.drain = true | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| 	// Wait for serving threads to be ready to exit.  Only then can we be sure no
 | ||||
| 	// new conns will be created.
 | ||||
| 	s.mu.Unlock() | ||||
| 	s.serveWG.Wait() | ||||
| 	s.mu.Lock() | ||||
| 
 | ||||
| 	for len(s.conns) != 0 { | ||||
| 		s.cv.Wait() | ||||
| // s.mu must be held by the caller.
 | ||||
| func (s *Server) closeListenersLocked() { | ||||
| 	for lis := range s.lis { | ||||
| 		lis.Close() | ||||
| 	} | ||||
| 	s.conns = nil | ||||
| 	if s.events != nil { | ||||
| 		s.events.Finish() | ||||
| 		s.events = nil | ||||
| 	} | ||||
| 	s.mu.Unlock() | ||||
| 	s.lis = nil | ||||
| } | ||||
| 
 | ||||
| // contentSubtype must be lowercase
 | ||||
|  | @ -1913,11 +1957,50 @@ func (s *Server) getCodec(contentSubtype string) baseCodec { | |||
| 	} | ||||
| 	codec := encoding.GetCodec(contentSubtype) | ||||
| 	if codec == nil { | ||||
| 		logger.Warningf("Unsupported codec %q. Defaulting to %q for now. This will start to fail in future releases.", contentSubtype, proto.Name) | ||||
| 		return encoding.GetCodec(proto.Name) | ||||
| 	} | ||||
| 	return codec | ||||
| } | ||||
| 
 | ||||
| type serverKey struct{} | ||||
| 
 | ||||
| // serverFromContext gets the Server from the context.
 | ||||
| func serverFromContext(ctx context.Context) *Server { | ||||
| 	s, _ := ctx.Value(serverKey{}).(*Server) | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| // contextWithServer sets the Server in the context.
 | ||||
| func contextWithServer(ctx context.Context, server *Server) context.Context { | ||||
| 	return context.WithValue(ctx, serverKey{}, server) | ||||
| } | ||||
| 
 | ||||
| // isRegisteredMethod returns whether the passed in method is registered as a
 | ||||
| // method on the server. /service/method and service/method will match if the
 | ||||
| // service and method are registered on the server.
 | ||||
| func (s *Server) isRegisteredMethod(serviceMethod string) bool { | ||||
| 	if serviceMethod != "" && serviceMethod[0] == '/' { | ||||
| 		serviceMethod = serviceMethod[1:] | ||||
| 	} | ||||
| 	pos := strings.LastIndex(serviceMethod, "/") | ||||
| 	if pos == -1 { // Invalid method name syntax.
 | ||||
| 		return false | ||||
| 	} | ||||
| 	service := serviceMethod[:pos] | ||||
| 	method := serviceMethod[pos+1:] | ||||
| 	srv, knownService := s.services[service] | ||||
| 	if knownService { | ||||
| 		if _, ok := srv.methods[method]; ok { | ||||
| 			return true | ||||
| 		} | ||||
| 		if _, ok := srv.streams[method]; ok { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
| 
 | ||||
| // SetHeader sets the header metadata to be sent from the server to the client.
 | ||||
| // The context provided must be the context passed to the server's handler.
 | ||||
| //
 | ||||
|  |  | |||
|  | @ -19,4 +19,4 @@ | |||
| package grpc | ||||
| 
 | ||||
| // Version is the current grpc version.
 | ||||
| const Version = "1.59.0" | ||||
| const Version = "1.60.1" | ||||
|  |  | |||
|  | @ -35,7 +35,6 @@ if [[ "$1" = "-install" ]]; then | |||
|   # Install the pinned versions as defined in module tools. | ||||
|   pushd ./test/tools | ||||
|   go install \ | ||||
|     golang.org/x/lint/golint \ | ||||
|     golang.org/x/tools/cmd/goimports \ | ||||
|     honnef.co/go/tools/cmd/staticcheck \ | ||||
|     github.com/client9/misspell/cmd/misspell | ||||
|  | @ -77,12 +76,16 @@ fi | |||
| not grep 'func Test[^(]' *_test.go | ||||
| not grep 'func Test[^(]' test/*.go | ||||
| 
 | ||||
| # - Check for typos in test function names | ||||
| git grep 'func (s) ' -- "*_test.go" | not grep -v 'func (s) Test' | ||||
| git grep 'func [A-Z]' -- "*_test.go" | not grep -v 'func Test\|Benchmark\|Example' | ||||
| 
 | ||||
| # - Do not import x/net/context. | ||||
| not git grep -l 'x/net/context' -- "*.go" | ||||
| 
 | ||||
| # - Do not import math/rand for real library code.  Use internal/grpcrand for | ||||
| #   thread safety. | ||||
| git grep -l '"math/rand"' -- "*.go" 2>&1 | not grep -v '^examples\|^stress\|grpcrand\|^benchmark\|wrr_test' | ||||
| git grep -l '"math/rand"' -- "*.go" 2>&1 | not grep -v '^examples\|^interop/stress\|grpcrand\|^benchmark\|wrr_test' | ||||
| 
 | ||||
| # - Do not use "interface{}"; use "any" instead. | ||||
| git grep -l 'interface{}' -- "*.go" 2>&1 | not grep -v '\.pb\.go\|protoc-gen-go-grpc' | ||||
|  | @ -94,15 +97,14 @@ git grep -l -e 'grpclog.I' --or -e 'grpclog.W' --or -e 'grpclog.E' --or -e 'grpc | |||
| not git grep "\(import \|^\s*\)\"github.com/golang/protobuf/ptypes/" -- "*.go" | ||||
| 
 | ||||
| # - Ensure all usages of grpc_testing package are renamed when importing. | ||||
| not git grep "\(import \|^\s*\)\"google.golang.org/grpc/interop/grpc_testing" -- "*.go"  | ||||
| not git grep "\(import \|^\s*\)\"google.golang.org/grpc/interop/grpc_testing" -- "*.go" | ||||
| 
 | ||||
| # - Ensure all xds proto imports are renamed to *pb or *grpc. | ||||
| git grep '"github.com/envoyproxy/go-control-plane/envoy' -- '*.go' ':(exclude)*.pb.go' | not grep -v 'pb "\|grpc "' | ||||
| 
 | ||||
| misspell -error . | ||||
| 
 | ||||
| # - gofmt, goimports, golint (with exceptions for generated code), go vet, | ||||
| # go mod tidy. | ||||
| # - gofmt, goimports, go vet, go mod tidy. | ||||
| # Perform these checks on each module inside gRPC. | ||||
| for MOD_FILE in $(find . -name 'go.mod'); do | ||||
|   MOD_DIR=$(dirname ${MOD_FILE}) | ||||
|  | @ -110,7 +112,6 @@ for MOD_FILE in $(find . -name 'go.mod'); do | |||
|   go vet -all ./... | fail_on_output | ||||
|   gofmt -s -d -l . 2>&1 | fail_on_output | ||||
|   goimports -l . 2>&1 | not grep -vE "\.pb\.go" | ||||
|   golint ./... 2>&1 | not grep -vE "/grpc_testing_not_regenerate/.*\.pb\.go:" | ||||
| 
 | ||||
|   go mod tidy -compat=1.19 | ||||
|   git status --porcelain 2>&1 | fail_on_output || \ | ||||
|  | @ -119,94 +120,73 @@ for MOD_FILE in $(find . -name 'go.mod'); do | |||
| done | ||||
| 
 | ||||
| # - Collection of static analysis checks | ||||
| # | ||||
| # TODO(dfawley): don't use deprecated functions in examples or first-party | ||||
| # plugins. | ||||
| # TODO(dfawley): enable ST1019 (duplicate imports) but allow for protobufs. | ||||
| SC_OUT="$(mktemp)" | ||||
| staticcheck -go 1.19 -checks 'inherit,-ST1015,-ST1019,-SA1019' ./... > "${SC_OUT}" || true | ||||
| # Error if anything other than deprecation warnings are printed. | ||||
| not grep -v "is deprecated:.*SA1019" "${SC_OUT}" | ||||
| # Only ignore the following deprecated types/fields/functions. | ||||
| not grep -Fv '.CredsBundle | ||||
| .HeaderMap | ||||
| .Metadata is deprecated: use Attributes | ||||
| .NewAddress | ||||
| .NewServiceConfig | ||||
| .Type is deprecated: use Attributes | ||||
| BuildVersion is deprecated | ||||
| balancer.ErrTransientFailure | ||||
| balancer.Picker | ||||
| extDesc.Filename is deprecated | ||||
| github.com/golang/protobuf/jsonpb is deprecated | ||||
| grpc.CallCustomCodec | ||||
| grpc.Code | ||||
| grpc.Compressor | ||||
| grpc.CustomCodec | ||||
| grpc.Decompressor | ||||
| grpc.MaxMsgSize | ||||
| grpc.MethodConfig | ||||
| grpc.NewGZIPCompressor | ||||
| grpc.NewGZIPDecompressor | ||||
| grpc.RPCCompressor | ||||
| grpc.RPCDecompressor | ||||
| grpc.ServiceConfig | ||||
| grpc.WithCompressor | ||||
| grpc.WithDecompressor | ||||
| grpc.WithDialer | ||||
| grpc.WithMaxMsgSize | ||||
| grpc.WithServiceConfig | ||||
| grpc.WithTimeout | ||||
| http.CloseNotifier | ||||
| info.SecurityVersion | ||||
| proto is deprecated | ||||
| proto.InternalMessageInfo is deprecated | ||||
| proto.EnumName is deprecated | ||||
| proto.ErrInternalBadWireType is deprecated | ||||
| proto.FileDescriptor is deprecated | ||||
| proto.Marshaler is deprecated | ||||
| proto.MessageType is deprecated | ||||
| proto.RegisterEnum is deprecated | ||||
| proto.RegisterFile is deprecated | ||||
| proto.RegisterType is deprecated | ||||
| proto.RegisterExtension is deprecated | ||||
| proto.RegisteredExtension is deprecated | ||||
| proto.RegisteredExtensions is deprecated | ||||
| proto.RegisterMapType is deprecated | ||||
| proto.Unmarshaler is deprecated | ||||
| staticcheck -go 1.19 -checks 'all' ./... > "${SC_OUT}" || true | ||||
| 
 | ||||
| # Error for anything other than checks that need exclusions. | ||||
| grep -v "(ST1000)" "${SC_OUT}" | grep -v "(SA1019)" | grep -v "(ST1003)" | not grep -v "(ST1019)\|\(other import of\)" | ||||
| 
 | ||||
| # Exclude underscore checks for generated code. | ||||
| grep "(ST1003)" "${SC_OUT}" | not grep -v '\(.pb.go:\)\|\(code_string_test.go:\)' | ||||
| 
 | ||||
| # Error for duplicate imports not including grpc protos. | ||||
| grep "(ST1019)\|\(other import of\)" "${SC_OUT}" | not grep -Fv 'XXXXX PleaseIgnoreUnused | ||||
| channelz/grpc_channelz_v1" | ||||
| go-control-plane/envoy | ||||
| grpclb/grpc_lb_v1" | ||||
| health/grpc_health_v1" | ||||
| interop/grpc_testing" | ||||
| orca/v3" | ||||
| proto/grpc_gcp" | ||||
| proto/grpc_lookup_v1" | ||||
| reflection/grpc_reflection_v1" | ||||
| reflection/grpc_reflection_v1alpha" | ||||
| XXXXX PleaseIgnoreUnused' | ||||
| 
 | ||||
| # Error for any package comments not in generated code. | ||||
| grep "(ST1000)" "${SC_OUT}" | not grep -v "\.pb\.go:" | ||||
| 
 | ||||
| # Only ignore the following deprecated types/fields/functions and exclude | ||||
| # generated code. | ||||
| grep "(SA1019)" "${SC_OUT}" | not grep -Fv 'XXXXX PleaseIgnoreUnused | ||||
| XXXXX Protobuf related deprecation errors: | ||||
| "github.com/golang/protobuf | ||||
| .pb.go: | ||||
| : ptypes. | ||||
| proto.RegisterType | ||||
| XXXXX gRPC internal usage deprecation errors: | ||||
| "google.golang.org/grpc | ||||
| : grpc. | ||||
| : v1alpha. | ||||
| : v1alphareflectionpb. | ||||
| BalancerAttributes is deprecated: | ||||
| CredsBundle is deprecated: | ||||
| Metadata is deprecated: use Attributes instead. | ||||
| NewSubConn is deprecated: | ||||
| OverrideServerName is deprecated: | ||||
| RemoveSubConn is deprecated: | ||||
| SecurityVersion is deprecated: | ||||
| Target is deprecated: Use the Target field in the BuildOptions instead. | ||||
| xxx_messageInfo_ | ||||
| ' "${SC_OUT}" | ||||
| 
 | ||||
| # - special golint on package comments. | ||||
| lint_package_comment_per_package() { | ||||
|   # Number of files in this go package. | ||||
|   fileCount=$(go list -f '{{len .GoFiles}}' $1) | ||||
|   if [ ${fileCount} -eq 0 ]; then | ||||
|     return 0 | ||||
|   fi | ||||
|   # Number of package errors generated by golint. | ||||
|   lintPackageCommentErrorsCount=$(golint --min_confidence 0 $1 | grep -c "should have a package comment") | ||||
|   # golint complains about every file that's missing the package comment. If the | ||||
|   # number of files for this package is greater than the number of errors, there's | ||||
|   # at least one file with package comment, good. Otherwise, fail. | ||||
|   if [ ${fileCount} -le ${lintPackageCommentErrorsCount} ]; then | ||||
|     echo "Package $1 (with ${fileCount} files) is missing package comment" | ||||
|     return 1 | ||||
|   fi | ||||
| } | ||||
| lint_package_comment() { | ||||
|   set +ex | ||||
| 
 | ||||
|   count=0 | ||||
|   for i in $(go list ./...); do | ||||
|     lint_package_comment_per_package "$i" | ||||
|     ((count += $?)) | ||||
|   done | ||||
| 
 | ||||
|   set -ex | ||||
|   return $count | ||||
| } | ||||
| lint_package_comment | ||||
| UpdateAddresses is deprecated: | ||||
| UpdateSubConnState is deprecated: | ||||
| balancer.ErrTransientFailure is deprecated: | ||||
| grpc/reflection/v1alpha/reflection.proto | ||||
| XXXXX xDS deprecated fields we support | ||||
| .ExactMatch | ||||
| .PrefixMatch | ||||
| .SafeRegexMatch | ||||
| .SuffixMatch | ||||
| GetContainsMatch | ||||
| GetExactMatch | ||||
| GetMatchSubjectAltNames | ||||
| GetPrefixMatch | ||||
| GetSafeRegexMatch | ||||
| GetSuffixMatch | ||||
| GetTlsCertificateCertificateProviderInstance | ||||
| GetValidationContextCertificateProviderInstance | ||||
| XXXXX TODO: Remove the below deprecation usages: | ||||
| CloseNotifier | ||||
| Roots.Subjects | ||||
| XXXXX PleaseIgnoreUnused' | ||||
| 
 | ||||
| echo SUCCESS | ||||
|  |  | |||
|  | @ -104,7 +104,7 @@ github.com/chenzhuoyu/iasm/x86_64 | |||
| # github.com/chzyer/readline v1.5.1 | ||||
| ## explicit; go 1.15 | ||||
| github.com/chzyer/readline | ||||
| # github.com/containerd/cgroups/v3 v3.0.2 | ||||
| # github.com/containerd/cgroups/v3 v3.0.3 | ||||
| ## explicit; go 1.18 | ||||
| github.com/containerd/cgroups/v3/cgroup1/stats | ||||
| # github.com/containerd/containerd v1.7.13 | ||||
|  | @ -340,8 +340,8 @@ github.com/containers/ocicrypt/keywrap/pkcs7 | |||
| github.com/containers/ocicrypt/spec | ||||
| github.com/containers/ocicrypt/utils | ||||
| github.com/containers/ocicrypt/utils/keyprovider | ||||
| # github.com/containers/psgo v1.8.0 | ||||
| ## explicit; go 1.14 | ||||
| # github.com/containers/psgo v1.9.0 | ||||
| ## explicit; go 1.18 | ||||
| github.com/containers/psgo | ||||
| github.com/containers/psgo/internal/capabilities | ||||
| github.com/containers/psgo/internal/cgroups | ||||
|  | @ -731,6 +731,8 @@ github.com/klauspost/pgzip | |||
| # github.com/kr/fs v0.1.0 | ||||
| ## explicit | ||||
| github.com/kr/fs | ||||
| # github.com/kr/pretty v0.3.1 | ||||
| ## explicit; go 1.12 | ||||
| # github.com/leodido/go-urn v1.2.4 | ||||
| ## explicit; go 1.16 | ||||
| github.com/leodido/go-urn | ||||
|  | @ -913,7 +915,7 @@ github.com/opentracing/opentracing-go/log | |||
| ## explicit | ||||
| github.com/ostreedev/ostree-go/pkg/glibobject | ||||
| github.com/ostreedev/ostree-go/pkg/otbuiltin | ||||
| # github.com/pelletier/go-toml/v2 v2.1.0 | ||||
| # github.com/pelletier/go-toml/v2 v2.1.1 | ||||
| ## explicit; go 1.16 | ||||
| github.com/pelletier/go-toml/v2 | ||||
| github.com/pelletier/go-toml/v2/internal/characters | ||||
|  | @ -1111,7 +1113,7 @@ go.opencensus.io/trace/tracestate | |||
| ## explicit; go 1.19 | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp/internal/semconvutil | ||||
| # go.opentelemetry.io/otel v1.19.0 | ||||
| # go.opentelemetry.io/otel v1.21.0 | ||||
| ## explicit; go 1.20 | ||||
| go.opentelemetry.io/otel | ||||
| go.opentelemetry.io/otel/attribute | ||||
|  | @ -1125,13 +1127,16 @@ go.opentelemetry.io/otel/propagation | |||
| go.opentelemetry.io/otel/semconv/internal | ||||
| go.opentelemetry.io/otel/semconv/v1.12.0 | ||||
| go.opentelemetry.io/otel/semconv/v1.17.0 | ||||
| # go.opentelemetry.io/otel/metric v1.19.0 | ||||
| # go.opentelemetry.io/otel/metric v1.21.0 | ||||
| ## explicit; go 1.20 | ||||
| go.opentelemetry.io/otel/metric | ||||
| go.opentelemetry.io/otel/metric/embedded | ||||
| # go.opentelemetry.io/otel/trace v1.19.0 | ||||
| # go.opentelemetry.io/otel/sdk v1.21.0 | ||||
| ## explicit; go 1.20 | ||||
| # go.opentelemetry.io/otel/trace v1.21.0 | ||||
| ## explicit; go 1.20 | ||||
| go.opentelemetry.io/otel/trace | ||||
| go.opentelemetry.io/otel/trace/embedded | ||||
| # golang.org/x/arch v0.5.0 | ||||
| ## explicit; go 1.17 | ||||
| golang.org/x/arch/x86/x86asm | ||||
|  | @ -1265,10 +1270,10 @@ google.golang.org/appengine/internal/log | |||
| google.golang.org/appengine/internal/remote_api | ||||
| google.golang.org/appengine/internal/urlfetch | ||||
| google.golang.org/appengine/urlfetch | ||||
| # google.golang.org/genproto/googleapis/rpc v0.0.0-20230920204549-e6e6cdab5c13 | ||||
| # google.golang.org/genproto/googleapis/rpc v0.0.0-20231212172506-995d672761c0 | ||||
| ## explicit; go 1.19 | ||||
| google.golang.org/genproto/googleapis/rpc/status | ||||
| # google.golang.org/grpc v1.59.0 | ||||
| # google.golang.org/grpc v1.60.1 | ||||
| ## explicit; go 1.19 | ||||
| google.golang.org/grpc | ||||
| google.golang.org/grpc/attributes | ||||
|  | @ -1304,6 +1309,7 @@ google.golang.org/grpc/internal/metadata | |||
| google.golang.org/grpc/internal/pretty | ||||
| google.golang.org/grpc/internal/resolver | ||||
| google.golang.org/grpc/internal/resolver/dns | ||||
| google.golang.org/grpc/internal/resolver/dns/internal | ||||
| google.golang.org/grpc/internal/resolver/passthrough | ||||
| google.golang.org/grpc/internal/resolver/unix | ||||
| google.golang.org/grpc/internal/serviceconfig | ||||
|  | @ -1315,6 +1321,7 @@ google.golang.org/grpc/keepalive | |||
| google.golang.org/grpc/metadata | ||||
| google.golang.org/grpc/peer | ||||
| google.golang.org/grpc/resolver | ||||
| google.golang.org/grpc/resolver/dns | ||||
| google.golang.org/grpc/serviceconfig | ||||
| google.golang.org/grpc/stats | ||||
| google.golang.org/grpc/status | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue