Merge pull request #13276 from rhatdan/containers-common
Add containers-common spec and command to podman
This commit is contained in:
		
						commit
						bc0e084f2d
					
				|  | @ -0,0 +1,56 @@ | |||
| #!/usr/bin/bash -e | ||||
| # | ||||
| # Script used for downloading man pages and config files from | ||||
| # github.com/containers libraries (storage, common, image) | ||||
| # | ||||
| # Must be run from directory containing input specfile | ||||
| # | ||||
| 
 | ||||
| die() { | ||||
|     echo "$(basename $0): $*" >&2 | ||||
|     exit 1 | ||||
| } | ||||
| 
 | ||||
| branchversion() { | ||||
|     gomod=$(git rev-parse --show-toplevel)/go.mod | ||||
|     v=$(awk -v X=github.com/containers/$1 '$1 ~ X { print $2 }' <$gomod) | ||||
|     hash=$(expr "$v" : "v.*-[0-9.]\+-\([0-9a-f]\+\)") | ||||
|     if [[ -n "$hash" ]]; then | ||||
|         v="$hash" | ||||
|     fi | ||||
|     echo "$v" | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| SPECFILE=containers-common.spec | ||||
| if [[ ! -e $SPECFILE.in ]]; then | ||||
|     die "Please run me from the same directory as $SPECFILE.in" | ||||
| fi | ||||
| 
 | ||||
| declare -A moduleversion | ||||
| for module in common image storage; do | ||||
|     v=$(branchversion $module) | ||||
|     if [[ -z "$v" ]]; then | ||||
|         die "Could not find version for module '$v'" | ||||
|     fi | ||||
|     moduleversion[$module]=$v | ||||
| done | ||||
| 
 | ||||
| builddir=containers-common-${moduleversion[common]} | ||||
| mkdir -p $builddir | ||||
| 
 | ||||
| sed -e "s/COMMON_BRANCH/${moduleversion[common]}/g" \ | ||||
|     -e "s/IMAGE_BRANCH/${moduleversion[image]}/g"  \ | ||||
|     -e "s/STORAGE_BRANCH/${moduleversion[storage]}/g"  \ | ||||
|     <$SPECFILE.in >$builddir/$SPECFILE | ||||
| 
 | ||||
| cd $builddir | ||||
| spectool -fg $SPECFILE | ||||
| 
 | ||||
| if [[ ! -e storage.conf ]]; then | ||||
|     die "spectool did not pull storage.conf" | ||||
| fi | ||||
| 
 | ||||
| echo "Changing storage.conf..." | ||||
| sed -i -e 's/^driver.*=.*/driver = "overlay"/' -e 's/^mountopt.*=.*/mountopt = "nodev,metacopy=on"/' \ | ||||
|         storage.conf | ||||
|  | @ -0,0 +1,175 @@ | |||
| # Below definitions are used to deliver config files from a particular branch | ||||
| # of c/image, c/common, c/storage vendored in all of Buildah, Podman and Skopeo. | ||||
| # These vendored components must have the same version. If it is not the case, | ||||
| # pick the oldest version on c/image, c/common, c/storage vendored in | ||||
| # Buildah/Podman/Skopeo. | ||||
| %global skopeo_branch main | ||||
| %global image_branch  IMAGE_BRANCH | ||||
| %global common_branch COMMON_BRANCH | ||||
| %global storage_branch STORAGE_BRANCH | ||||
| %global shortnames_branch main | ||||
| 
 | ||||
| %global github_containers https://raw.githubusercontent.com/containers | ||||
| 
 | ||||
| Epoch: 4 | ||||
| Name: containers-common | ||||
| Version: COMMON_BRANCH | ||||
| Release: %autorelease | ||||
| Summary: Common configuration and documentation for containers | ||||
| License: ASL 2.0 | ||||
| BuildArch: noarch | ||||
| BuildRequires: go-md2man | ||||
| Provides: skopeo-containers = %{epoch}:%{version}-%{release} | ||||
| Requires: (container-selinux >= 2:2.162.1 if selinux-policy) | ||||
| Requires: oci-runtime | ||||
| Requires: container-network-stack | ||||
| Recommends: netavark | ||||
| Recommends: fuse-overlayfs | ||||
| Recommends: slirp4netns | ||||
| Source1: %{github_containers}/common/%{common_branch}/docs/containers.conf.5.md | ||||
| Source2: %{github_containers}/common/%{common_branch}/pkg/config/containers.conf | ||||
| Source3: %{github_containers}/common/%{common_branch}/pkg/seccomp/seccomp.json | ||||
| Source4: %{github_containers}/common/%{common_branch}/pkg/subscriptions/mounts.conf | ||||
| Source5: %{github_containers}/image/%{image_branch}/docs/containers-auth.json.5.md | ||||
| Source6: %{github_containers}/image/%{image_branch}/docs/containers-certs.d.5.md | ||||
| Source7: %{github_containers}/image/%{image_branch}/docs/containers-policy.json.5.md | ||||
| Source8: %{github_containers}/image/%{image_branch}/docs/containers-registries.conf.5.md | ||||
| Source9: %{github_containers}/image/%{image_branch}/docs/containers-registries.conf.d.5.md | ||||
| Source10: %{github_containers}/image/%{image_branch}/docs/containers-registries.d.5.md | ||||
| Source11: %{github_containers}/image/%{image_branch}/docs/containers-signature.5.md | ||||
| Source12: %{github_containers}/image/%{image_branch}/docs/containers-transports.5.md | ||||
| Source13: %{github_containers}/image/%{image_branch}/registries.conf | ||||
| Source14: %{github_containers}/common/%{common_branch}/docs/containers-mounts.conf.5.md | ||||
| Source15: %{github_containers}/shortnames/%{shortnames_branch}/shortnames.conf | ||||
| Source16: %{github_containers}/skopeo/%{skopeo_branch}/default.yaml | ||||
| Source17: %{github_containers}/skopeo/%{skopeo_branch}/default-policy.json | ||||
| Source18: %{github_containers}/storage/%{storage_branch}/docs/containers-storage.conf.5.md | ||||
| Source19: %{github_containers}/storage/%{storage_branch}/storage.conf | ||||
| Source20: RPM-GPG-KEY-redhat-release | ||||
| Source21: registry.access.redhat.com.yaml | ||||
| Source22: registry.redhat.io.yaml | ||||
| Source23: %{github_containers}/common/%{common_branch}/docs/Containerfile.5.md | ||||
| Source24: %{github_containers}/common/%{common_branch}/docs/containerignore.5.md | ||||
| Source25: %{github_containers}/common/%{common_branch}/docs/links/.containerignore.5 | ||||
| 
 | ||||
| %description | ||||
| This package contains common configuration files and documentation for container | ||||
| tools ecosystem, such as Podman, Buildah and Skopeo. | ||||
| 
 | ||||
| It is required because the most of configuration files and docs come from projects | ||||
| which are vendored into Podman, Buildah, Skopeo, etc. but they are not packaged | ||||
| separately. | ||||
| 
 | ||||
| %prep | ||||
| cp %{SOURCE1} . | ||||
| cp %{SOURCE2} . | ||||
| cp %{SOURCE3} . | ||||
| cp %{SOURCE4} . | ||||
| cp %{SOURCE5} . | ||||
| cp %{SOURCE6} . | ||||
| cp %{SOURCE7} . | ||||
| cp %{SOURCE8} . | ||||
| cp %{SOURCE9} . | ||||
| cp %{SOURCE10} . | ||||
| cp %{SOURCE11} . | ||||
| cp %{SOURCE12} . | ||||
| cp %{SOURCE13} . | ||||
| cp %{SOURCE14} . | ||||
| cp %{SOURCE15} 000-shortnames.conf | ||||
| cp %{SOURCE16} . | ||||
| cp %{SOURCE17} policy.json | ||||
| cp %{SOURCE18} . | ||||
| cp %{SOURCE19} . | ||||
| cp %{SOURCE20} . | ||||
| cp %{SOURCE21} . | ||||
| cp %{SOURCE22} . | ||||
| cp %{SOURCE23} . | ||||
| cp %{SOURCE24} . | ||||
| cp %{SOURCE25} . | ||||
| 
 | ||||
| %build | ||||
| mkdir -p man5 | ||||
| for FILE in $(ls *.5.md); do | ||||
|     go-md2man -in $FILE -out man5/$(basename $FILE .md) | ||||
| done | ||||
| 
 | ||||
| cp man5/containerignore.5 man5/.containerignore.5 | ||||
| 
 | ||||
| %install | ||||
| # install config and policy files for registries | ||||
| install -dp %{buildroot}%{_sysconfdir}/containers/{certs.d,oci/hooks.d} | ||||
| install -dp %{buildroot}%{_sharedstatedir}/containers/sigstore | ||||
| install -Dp -m0644 default.yaml -t %{buildroot}%{_sysconfdir}/containers/registries.d | ||||
| install -Dp -m0644 storage.conf -t %{buildroot}%{_datadir}/containers | ||||
| install -Dp -m0644 registries.conf -t %{buildroot}%{_sysconfdir}/containers | ||||
| install -Dp -m0644 000-shortnames.conf -t %{buildroot}%{_sysconfdir}/containers/registries.conf.d | ||||
| install -Dp -m0644 policy.json -t %{buildroot}%{_sysconfdir}/containers | ||||
| install -Dp -m0644 RPM-GPG-KEY-redhat-release -t %{buildroot}%{_sysconfdir}/pki/rpm-gpg | ||||
| install -Dp -m0644 registry.access.redhat.com.yaml -t %{buildroot}%{_sysconfdir}/containers/registries.d | ||||
| install -Dp -m0644 registry.redhat.io.yaml -t %{buildroot}%{_sysconfdir}/containers/registries.d | ||||
| 
 | ||||
| # install manpages | ||||
| for FILE in $(ls -a man5 | grep 5); do | ||||
|     install -Dp -m0644 man5/$FILE -t %{buildroot}%{_mandir}/man5 | ||||
| done | ||||
| 
 | ||||
| # install config files for mounts, containers and seccomp | ||||
| install -m0644 mounts.conf %{buildroot}%{_datadir}/containers/mounts.conf | ||||
| install -m0644 seccomp.json %{buildroot}%{_datadir}/containers/seccomp.json | ||||
| install -m0644 containers.conf %{buildroot}%{_datadir}/containers/containers.conf | ||||
| 
 | ||||
| # install secrets patch directory | ||||
| install -d -p -m 755 %{buildroot}/%{_datadir}/rhel/secrets | ||||
| # rhbz#1110876 - update symlinks for subscription management | ||||
| ln -s %{_sysconfdir}/pki/entitlement %{buildroot}%{_datadir}/rhel/secrets/etc-pki-entitlement | ||||
| ln -s %{_sysconfdir}/rhsm %{buildroot}%{_datadir}/rhel/secrets/rhsm | ||||
| ln -s %{_sysconfdir}/yum.repos.d/redhat.repo %{buildroot}%{_datadir}/rhel/secrets/redhat.repo | ||||
| 
 | ||||
| %post | ||||
| if [[ ! -f %{_mandir}/man5/Dockerfile.5.gz ]]; then | ||||
|     echo .so %{_mandir}/man5/Containerfile.5 > %{_mandir}/man5/Dockerfile.5 | ||||
|     gzip %{_mandir}/man5/Dockerfile.5 | ||||
| fi | ||||
| if [[ ! -f %{_mandir}/man5/dockerignore.5.gz ]]; then | ||||
|     echo .so %{_mandir}/man5/containerignore.5 > %{_mandir}/man5/dockerignore.5 | ||||
|     gzip %{_mandir}/man5/dockerignore.5 | ||||
| fi | ||||
| if [[ ! -f %{_mandir}/man5/.dockerignore.5.gz ]]; then | ||||
|     echo .so %{_mandir}/man5/containerignore.5 > %{_mandir}/man5/.dockerignore.5 | ||||
|     gzip %{_mandir}/man5/.dockerignore.5 | ||||
| fi | ||||
| 
 | ||||
| %files | ||||
| %dir %{_sysconfdir}/containers | ||||
| %dir %{_sysconfdir}/containers/certs.d | ||||
| %dir %{_sysconfdir}/containers/oci | ||||
| %dir %{_sysconfdir}/containers/oci/hooks.d | ||||
| %dir %{_sysconfdir}/containers/registries.conf.d | ||||
| %dir %{_sysconfdir}/containers/registries.d | ||||
| %config(noreplace) %{_sysconfdir}/containers/policy.json | ||||
| %config(noreplace) %{_sysconfdir}/containers/registries.conf | ||||
| %config(noreplace) %{_sysconfdir}/containers/registries.conf.d/000-shortnames.conf | ||||
| %{_sysconfdir}/pki/rpm-gpg/RPM-GPG-KEY-redhat-release | ||||
| %config(noreplace) %{_sysconfdir}/containers/registries.d/default.yaml | ||||
| %{_sysconfdir}/containers/registries.d/registry.redhat.io.yaml | ||||
| %{_sysconfdir}/containers/registries.d/registry.access.redhat.com.yaml | ||||
| %ghost %{_sysconfdir}/containers/storage.conf | ||||
| %ghost %{_sysconfdir}/containers/containers.conf | ||||
| %dir %{_sharedstatedir}/containers/sigstore | ||||
| %ghost %{_mandir}/man5/Dockerfile.5.gz | ||||
| %ghost %{_mandir}/man5/dockerignore.5.gz | ||||
| %ghost %{_mandir}/man5/.dockerignore.5.gz | ||||
| %{_mandir}/man5/Containerfile.5.gz | ||||
| %{_mandir}/man5/containerignore.5.gz | ||||
| %{_mandir}/man5/.containerignore.5.gz | ||||
| %{_mandir}/man5/containers*.5.gz | ||||
| %dir %{_datadir}/containers | ||||
| %{_datadir}/containers/storage.conf | ||||
| %{_datadir}/containers/containers.conf | ||||
| %{_datadir}/containers/mounts.conf | ||||
| %{_datadir}/containers/seccomp.json | ||||
| %dir %{_datadir}/rhel/secrets | ||||
| %{_datadir}/rhel/secrets/* | ||||
| 
 | ||||
| %changelog | ||||
| %autochangelog | ||||
							
								
								
									
										6
									
								
								go.mod
								
								
								
								
							
							
						
						
									
										6
									
								
								go.mod
								
								
								
								
							|  | @ -12,7 +12,7 @@ require ( | |||
| 	github.com/containernetworking/cni v1.0.1 | ||||
| 	github.com/containernetworking/plugins v1.0.1 | ||||
| 	github.com/containers/buildah v1.24.2 | ||||
| 	github.com/containers/common v0.47.4 | ||||
| 	github.com/containers/common v0.47.5-0.20220222185251-194ee74231c3 | ||||
| 	github.com/containers/conmon v2.0.20+incompatible | ||||
| 	github.com/containers/image/v5 v5.19.1 | ||||
| 	github.com/containers/ocicrypt v1.1.2 | ||||
|  | @ -70,8 +70,8 @@ require ( | |||
| 	golang.org/x/text v0.3.7 | ||||
| 	google.golang.org/protobuf v1.27.1 | ||||
| 	gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b | ||||
| 	k8s.io/api v0.22.4 | ||||
| 	k8s.io/apimachinery v0.22.4 | ||||
| 	k8s.io/api v0.22.5 | ||||
| 	k8s.io/apimachinery v0.22.5 | ||||
| ) | ||||
| 
 | ||||
| replace github.com/onsi/gomega => github.com/onsi/gomega v1.16.0 | ||||
|  |  | |||
							
								
								
									
										148
									
								
								go.sum
								
								
								
								
							
							
						
						
									
										148
									
								
								go.sum
								
								
								
								
							|  | @ -1,5 +1,6 @@ | |||
| 4d63.com/gochecknoglobals v0.1.0/go.mod h1:wfdC5ZjKSPr7CybKEcgJhUOgeAQW1+7WcyK8OvUilfo= | ||||
| bazil.org/fuse v0.0.0-20160811212531-371fbbdaa898/go.mod h1:Xbm+BRKSBEpa4q4hTSxohYNQpsxXPbPry4JJWOB3LB8= | ||||
| bazil.org/fuse v0.0.0-20200407214033-5883e5a4b512/go.mod h1:FbcW6z/2VytnFDhZfumh8Ss8zxHE6qpMP5sHTRe0EaM= | ||||
| bitbucket.org/creachadair/shell v0.0.6/go.mod h1:8Qqi/cYk7vPnsOePHroKXDJYmb5x7ENhtiFtfZq8K+M= | ||||
| cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
| cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||
|  | @ -57,21 +58,26 @@ contrib.go.opencensus.io/exporter/stackdriver v0.13.4/go.mod h1:aXENhDJ1Y4lIg4EU | |||
| dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= | ||||
| github.com/14rcole/gopopulate v0.0.0-20180821133914-b175b219e774 h1:SCbEWT58NSt7d2mcFdvxC9uyrdcTfvBbPLThhkDmXzg= | ||||
| github.com/14rcole/gopopulate v0.0.0-20180821133914-b175b219e774/go.mod h1:6/0dYRLLXyJjbkIPeeGyoJ/eKOSI0eU6eTlCBYibgd0= | ||||
| github.com/AdaLogics/go-fuzz-headers v0.0.0-20210715213245-6c3934b029d8/go.mod h1:CzsSbkDixRphAF5hS6wbMKq0eI6ccJRb7/A0M6JBnwg= | ||||
| github.com/Antonboom/errname v0.1.5/go.mod h1:DugbBstvPFQbv/5uLcRRzfrNqKE9tVdVCqWCLp6Cifo= | ||||
| github.com/Antonboom/nilnil v0.1.0/go.mod h1:PhHLvRPSghY5Y7mX4TW+BHZQYo1A8flE5H20D3IPZBo= | ||||
| github.com/Azure/azure-sdk-for-go v16.2.1+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc= | ||||
| github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8= | ||||
| github.com/Azure/go-ansiterm v0.0.0-20210608223527-2377c96fe795/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8= | ||||
| github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 h1:UQHMgLO+TxOElx5B5HZ4hJQsoJ/PvUvKRhJHDQXO8P8= | ||||
| github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E= | ||||
| github.com/Azure/go-autorest v10.8.1+incompatible/go.mod h1:r+4oMnoxhatjLLJ6zxSWATqVooLgysK6ZNox3g/xq24= | ||||
| github.com/Azure/go-autorest v14.2.0+incompatible/go.mod h1:r+4oMnoxhatjLLJ6zxSWATqVooLgysK6ZNox3g/xq24= | ||||
| github.com/Azure/go-autorest/autorest v0.11.1/go.mod h1:JFgpikqFJ/MleTTxwepExTKnFUKKszPS8UavbQYUMuw= | ||||
| github.com/Azure/go-autorest/autorest v0.11.18/go.mod h1:dSiJPy22c3u0OtOKDNttNgqpNFY/GeWa7GH/Pz56QRA= | ||||
| github.com/Azure/go-autorest/autorest/adal v0.9.0/go.mod h1:/c022QCutn2P7uY+/oQWWNcK9YU+MH96NgK+jErpbcg= | ||||
| github.com/Azure/go-autorest/autorest/adal v0.9.5/go.mod h1:B7KF7jKIeC9Mct5spmyCB/A8CG/sEz1vwIRGv/bbw7A= | ||||
| github.com/Azure/go-autorest/autorest/adal v0.9.13/go.mod h1:W/MM4U6nLxnIskrw4UwWzlHfGjwUS50aOsc/I3yuU8M= | ||||
| github.com/Azure/go-autorest/autorest/date v0.3.0/go.mod h1:BI0uouVdmngYNUzGWeSYnokU+TrmwEsOqdt8Y6sso74= | ||||
| github.com/Azure/go-autorest/autorest/mocks v0.4.0/go.mod h1:LTp+uSrOhSkaKrUy935gNZuuIPPVsHlr9DSOxSayd+k= | ||||
| github.com/Azure/go-autorest/autorest/mocks v0.4.1/go.mod h1:LTp+uSrOhSkaKrUy935gNZuuIPPVsHlr9DSOxSayd+k= | ||||
| github.com/Azure/go-autorest/logger v0.2.0/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8= | ||||
| github.com/Azure/go-autorest/logger v0.2.1/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8= | ||||
| github.com/Azure/go-autorest/tracing v0.6.0/go.mod h1:+vhtPC754Xsa23ID7GlGsrdKBpUA79WCAKPPZVC2DeU= | ||||
| github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= | ||||
| github.com/BurntSushi/toml v0.4.1/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ= | ||||
|  | @ -112,6 +118,7 @@ github.com/Microsoft/hcsshim v0.9.2/go.mod h1:7pLA8lDk46WKDWlVsENo92gC0XFa8rbKfy | |||
| github.com/Microsoft/hcsshim/test v0.0.0-20201218223536-d3e5debf77da/go.mod h1:5hlzMzRKMLyo42nCZ9oml8AdTlq/0cvIaBv6tK1RehU= | ||||
| github.com/Microsoft/hcsshim/test v0.0.0-20210227013316-43a75bb4edd3/go.mod h1:mw7qgWloBUl75W/gVH3cQszUg1+gUITj7D6NY7ywVnY= | ||||
| github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ= | ||||
| github.com/NYTimes/gziphandler v1.1.1/go.mod h1:n/CVRwUEOgIxrgPvAQhUUr9oeUtvrhMomdKFjzJNB0c= | ||||
| github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= | ||||
| github.com/OpenPeeDeeP/depguard v1.0.1/go.mod h1:xsIw86fROiiwelg+jB2uM9PiKihMMmUx/1V+TNhjQvM= | ||||
| github.com/ProtonMail/go-crypto v0.0.0-20210428141323-04723f9f07d7/go.mod h1:z4/9nQmJSSwwds7ejkxaJwO37dru3geImFUdJlaLzQo= | ||||
|  | @ -132,6 +139,7 @@ github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuy | |||
| github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= | ||||
| github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= | ||||
| github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= | ||||
| github.com/alexflint/go-filemutex v0.0.0-20171022225611-72bdc8eae2ae/go.mod h1:CgnQgUtFrFz9mxFNtED3jI5tLDjKlOM+oUF/sTk6ps0= | ||||
| github.com/alexflint/go-filemutex v1.1.0/go.mod h1:7P4iRhttt/nUvUOrYIhcpMzv2G6CY9UnI16Z+UJqRyk= | ||||
| github.com/alexkohler/prealloc v1.0.0/go.mod h1:VetnK3dIgFBBKmg0YnD9F9x6Icjd+9cvfHR56wJVlKE= | ||||
|  | @ -155,6 +163,7 @@ github.com/aws/aws-sdk-go v1.15.11/go.mod h1:mFuSZ37Z9YOHbQEwBWztmVzqXrEkub65tZo | |||
| github.com/aws/aws-sdk-go v1.23.20/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= | ||||
| github.com/aws/aws-sdk-go v1.25.37/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= | ||||
| github.com/aws/aws-sdk-go v1.36.30/go.mod h1:hcU610XS61/+aQV88ixoOzUoG7v3b31pl2zKMmprdro= | ||||
| github.com/benbjohnson/clock v1.0.3/go.mod h1:bGMdMPoPVvcYyt1gHDf4J2KE153Yf9BuiUKYMaxlTDM= | ||||
| github.com/beorn7/perks v0.0.0-20160804104726-4c0e84591b9a/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= | ||||
| github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= | ||||
| github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= | ||||
|  | @ -163,6 +172,7 @@ github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6r | |||
| github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= | ||||
| github.com/bitly/go-simplejson v0.5.0/go.mod h1:cXHtHw4XUPsvGaxgjIAn8PhEWG9NfngEKAMDJEczWVA= | ||||
| github.com/bits-and-blooms/bitset v1.2.0/go.mod h1:gIdJ4wp64HaoK2YrL1Q5/N7Y16edYb8uY+O0FJTyyDA= | ||||
| github.com/bketelsen/crypt v0.0.3-0.20200106085610-5cbc8cc4026c/go.mod h1:MKsuJmJgSg28kpZDP6UIiPt0e0Oz0kqKNGyRaWEPv84= | ||||
| github.com/bketelsen/crypt v0.0.4/go.mod h1:aI6NrJ0pMGgvZKL1iVgXLnfIFJtfV+bKCoqOes/6LfM= | ||||
| github.com/bkielbasa/cyclop v1.2.0/go.mod h1:qOI0yy6A7dYC4Zgsa72Ppm9kONl0RoIlPbzot9mhmeI= | ||||
| github.com/blang/semver v3.1.0+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk= | ||||
|  | @ -184,8 +194,11 @@ github.com/bugsnag/osext v0.0.0-20130617224835-0dd3f918b21b/go.mod h1:obH5gd0Bsq | |||
| github.com/bugsnag/panicwrap v0.0.0-20151223152923-e2c28503fcd0/go.mod h1:D/8v3kj0zr8ZAKg1AQ6crr+5VwKN5eIywRkfhyM/+dE= | ||||
| github.com/butuzov/ireturn v0.1.1/go.mod h1:Wh6Zl3IMtTpaIKbmwzqi6olnM9ptYQxxVacMsOEFPoc= | ||||
| github.com/cenkalti/backoff/v4 v4.1.1/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= | ||||
| github.com/cenkalti/backoff/v4 v4.1.2/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/certifi/gocertifi v0.0.0-20191021191039-0944d244cd40/go.mod h1:sGbDF6GwGcLpkNXPUTkMRoywsNa/ol15pxFe6ERfguA= | ||||
| github.com/certifi/gocertifi v0.0.0-20200922220541-2c3bb06c6054/go.mod h1:sGbDF6GwGcLpkNXPUTkMRoywsNa/ol15pxFe6ERfguA= | ||||
| github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
|  | @ -226,6 +239,9 @@ github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWH | |||
| github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/cncf/xds/go v0.0.0-20211130200136-a8f946100490/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= | ||||
| github.com/cockroachdb/datadriven v0.0.0-20190809214429-80d97fb3cbaa/go.mod h1:zn76sxSg3SzpJ0PPJaLDCu+Bu0Lg3sKTORVIj19EIF8= | ||||
| github.com/cockroachdb/datadriven v0.0.0-20200714090401-bf6692d28da5/go.mod h1:h6jFvWxBdQXxjopDMZyH2UVceIRfR84bdzbkoKrsWNo= | ||||
| github.com/cockroachdb/errors v1.2.4/go.mod h1:rQD95gz6FARkaKkQXUksEje/d9a6wBJoCr5oaCLELYA= | ||||
| github.com/cockroachdb/logtags v0.0.0-20190617123548-eb05cc24525f/go.mod h1:i/u985jwjWRlyHXQbwatDASoW0RMlZ/3i9yJHE2xLkI= | ||||
| github.com/container-orchestrated-devices/container-device-interface v0.0.0-20220111162300-46367ec063fd h1:Pzh64A349jzW89R73gwkxWoPvpkd8rz2X+KLUaMmBRY= | ||||
| github.com/container-orchestrated-devices/container-device-interface v0.0.0-20220111162300-46367ec063fd/go.mod h1:YqXyXS/oVW3ix0IHVXitKBq3RZoCF8ccm5RPmRBraUI= | ||||
| github.com/containerd/aufs v0.0.0-20200908144142-dab0cbea06f4/go.mod h1:nukgQABAEopAHvB6j7cnP5zJ+/3aVcE7hCYqvIwAHyE= | ||||
|  | @ -241,8 +257,9 @@ github.com/containerd/cgroups v0.0.0-20200531161412-0dbf7f05ba59/go.mod h1:pA0z1 | |||
| github.com/containerd/cgroups v0.0.0-20200710171044-318312a37340/go.mod h1:s5q4SojHctfxANBDvMeIaIovkq29IP48TKAxnhYRxvo= | ||||
| github.com/containerd/cgroups v0.0.0-20200824123100-0b889c03f102/go.mod h1:s5q4SojHctfxANBDvMeIaIovkq29IP48TKAxnhYRxvo= | ||||
| github.com/containerd/cgroups v0.0.0-20210114181951-8a68de567b68/go.mod h1:ZJeTFisyysqgcCdecO57Dj79RfL0LNeGiFUqLYQRYLE= | ||||
| github.com/containerd/cgroups v1.0.1 h1:iJnMvco9XGvKUvNQkv88bE4uJXxRQH18efbKo9w5vHQ= | ||||
| github.com/containerd/cgroups v1.0.1/go.mod h1:0SJrPIenamHDcZhEcJMNBB85rHcUsw4f25ZfBiPYRkU= | ||||
| github.com/containerd/cgroups v1.0.3 h1:ADZftAkglvCiD44c77s5YmMqaP2pzVCFZvBmAlBdAP4= | ||||
| github.com/containerd/cgroups v1.0.3/go.mod h1:/ofk34relqNjSGyqPrmEULrO4Sc8LJhvJmWbUCUKqj8= | ||||
| github.com/containerd/console v0.0.0-20180822173158-c12b1e7919c1/go.mod h1:Tj/on1eG8kiEhd0+fhSDzsPAFESxzBBvdyEgyryXffw= | ||||
| github.com/containerd/console v0.0.0-20181022165439-0650fd9eeb50/go.mod h1:Tj/on1eG8kiEhd0+fhSDzsPAFESxzBBvdyEgyryXffw= | ||||
| github.com/containerd/console v0.0.0-20191206165004-02ecf6a7291e/go.mod h1:8Pf4gM6VEbTNRIT26AyyU7hxdQU3MvAvxVI0sc00XBE= | ||||
|  | @ -264,16 +281,19 @@ github.com/containerd/containerd v1.5.0-beta.4/go.mod h1:GmdgZd2zA2GYIBZ0w09Zvgq | |||
| github.com/containerd/containerd v1.5.0-rc.0/go.mod h1:V/IXoMqNGgBlabz3tHD2TWDoTJseu1FGOKuoA4nNb2s= | ||||
| github.com/containerd/containerd v1.5.1/go.mod h1:0DOxVqwDy2iZvrZp2JUx/E+hS0UNTVn7dJnIOwtYR4g= | ||||
| github.com/containerd/containerd v1.5.7/go.mod h1:gyvv6+ugqY25TiXxcZC3L5yOeYgEw0QMhscqVp1AR9c= | ||||
| github.com/containerd/containerd v1.5.9 h1:rs6Xg1gtIxaeyG+Smsb/0xaSDu1VgFhOCKBXxMxbsF4= | ||||
| github.com/containerd/containerd v1.5.8/go.mod h1:YdFSv5bTFLpG2HIYmfqDpSYYTDX+mc5qtSuYx1YUb/s= | ||||
| github.com/containerd/containerd v1.5.9/go.mod h1:fvQqCfadDGga5HZyn3j4+dx56qj2I9YwBrlSdalvJYQ= | ||||
| github.com/containerd/containerd v1.6.0 h1:CLa12ZcV0d2ZTRKq1ssioeJpTnPJBMyndpEKA+UtzJg= | ||||
| github.com/containerd/containerd v1.6.0/go.mod h1:1nJz5xCZPusx6jJU8Frfct988y0NpumIq9ODB0kLtoE= | ||||
| github.com/containerd/continuity v0.0.0-20190426062206-aaeac12a7ffc/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y= | ||||
| github.com/containerd/continuity v0.0.0-20190815185530-f2a389ac0a02/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y= | ||||
| github.com/containerd/continuity v0.0.0-20191127005431-f65d91d395eb/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y= | ||||
| github.com/containerd/continuity v0.0.0-20200710164510-efbc4488d8fe/go.mod h1:cECdGN1O8G9bgKTlLhuPJimka6Xb/Gg7vYzCTNVxhvo= | ||||
| github.com/containerd/continuity v0.0.0-20201208142359-180525291bb7/go.mod h1:kR3BEg7bDFaEddKm54WSmrol1fKWDU1nKYkgrcgZT7Y= | ||||
| github.com/containerd/continuity v0.0.0-20210208174643-50096c924a4e/go.mod h1:EXlVlkqNba9rJe3j7w3Xa924itAMLgZH4UD/Q4PExuQ= | ||||
| github.com/containerd/continuity v0.1.0 h1:UFRRY5JemiAhPZrr/uE0n8fMTLcZsUvySPr1+D7pgr8= | ||||
| github.com/containerd/continuity v0.1.0/go.mod h1:ICJu0PwR54nI0yPEnJ6jcS+J7CZAUXrLh8lPo2knzsM= | ||||
| github.com/containerd/continuity v0.2.2 h1:QSqfxcn8c+12slxwu00AtzXrsami0MJb/MQs9lOLHLA= | ||||
| github.com/containerd/continuity v0.2.2/go.mod h1:pWygW9u7LtS1o4N/Tn0FoCFDIXZ7rxcMX7HX1Dmibvk= | ||||
| github.com/containerd/fifo v0.0.0-20180307165137-3d5202aec260/go.mod h1:ODA38xgv3Kuk8dQz2ZQXpnv/UZZUHUCL7pnLehbXgQI= | ||||
| github.com/containerd/fifo v0.0.0-20190226154929-a9fb20d87448/go.mod h1:ODA38xgv3Kuk8dQz2ZQXpnv/UZZUHUCL7pnLehbXgQI= | ||||
| github.com/containerd/fifo v0.0.0-20200410184934-f15a3290365b/go.mod h1:jPQ2IAeZRCYxpS/Cm1495vGFww6ecHmMk1YJH2Q5ln0= | ||||
|  | @ -282,6 +302,8 @@ github.com/containerd/fifo v0.0.0-20210316144830-115abcc95a1d/go.mod h1:ocF/ME1S | |||
| github.com/containerd/fifo v1.0.0/go.mod h1:ocF/ME1SX5b1AOlWi9r677YJmCPSwwWnQ9O123vzpE4= | ||||
| github.com/containerd/go-cni v1.0.1/go.mod h1:+vUpYxKvAF72G9i1WoDOiPGRtQpqsNW/ZHtSlv++smU= | ||||
| github.com/containerd/go-cni v1.0.2/go.mod h1:nrNABBHzu0ZwCug9Ije8hL2xBCYh/pjfMb1aZGrrohk= | ||||
| github.com/containerd/go-cni v1.1.0/go.mod h1:Rflh2EJ/++BA2/vY5ao3K6WJRR/bZKsX123aPk+kUtA= | ||||
| github.com/containerd/go-cni v1.1.3/go.mod h1:Rflh2EJ/++BA2/vY5ao3K6WJRR/bZKsX123aPk+kUtA= | ||||
| github.com/containerd/go-runc v0.0.0-20180907222934-5a6d9f37cfa3/go.mod h1:IV7qH3hrUgRmyYrtgEeGWJfWbgcHL9CSRruz2Vqcph0= | ||||
| github.com/containerd/go-runc v0.0.0-20190911050354-e029b79d8cda/go.mod h1:IV7qH3hrUgRmyYrtgEeGWJfWbgcHL9CSRruz2Vqcph0= | ||||
| github.com/containerd/go-runc v0.0.0-20200220073739-7016d3ce2328/go.mod h1:PpyHrqVs8FTi9vpyHwPwiNEGaACDxT/N/pLcvMSRA9g= | ||||
|  | @ -291,6 +313,7 @@ github.com/containerd/imgcrypt v1.0.1/go.mod h1:mdd8cEPW7TPgNG4FpuP3sGBiQ7Yi/zak | |||
| github.com/containerd/imgcrypt v1.0.4-0.20210301171431-0ae5c75f59ba/go.mod h1:6TNsg0ctmizkrOgXRNQjAPFWpMYRWuiB6dSF4Pfa5SA= | ||||
| github.com/containerd/imgcrypt v1.1.1-0.20210312161619-7ed62a527887/go.mod h1:5AZJNI6sLHJljKuI9IHnw1pWqo/F0nGDOuR9zgTs7ow= | ||||
| github.com/containerd/imgcrypt v1.1.1/go.mod h1:xpLnwiQmEUJPvQoAapeb2SNCxz7Xr6PJrXQb0Dpc4ms= | ||||
| github.com/containerd/imgcrypt v1.1.3/go.mod h1:/TPA1GIDXMzbj01yd8pIbQiLdQxed5ue1wb8bP7PQu4= | ||||
| github.com/containerd/nri v0.0.0-20201007170849-eb1350a75164/go.mod h1:+2wGSDGFYfE5+So4M5syatU0N0f0LbWpuqyMi4/BE8c= | ||||
| github.com/containerd/nri v0.0.0-20210316161719-dbaa18c31c14/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY= | ||||
| github.com/containerd/nri v0.1.0/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY= | ||||
|  | @ -325,8 +348,9 @@ github.com/containernetworking/plugins v1.0.1 h1:wwCfYbTCj5FC0EJgyzyjTXmqysOiJE9 | |||
| github.com/containernetworking/plugins v1.0.1/go.mod h1:QHCfGpaTwYTbbH+nZXKVTxNBDZcxSOplJT5ico8/FLE= | ||||
| github.com/containers/buildah v1.24.2 h1:B+3Drs0oDnaDogZXT75Ktim+P0Yo0O1Da6kuAWX8ID4= | ||||
| github.com/containers/buildah v1.24.2/go.mod h1:iA7OJjcSuRhd9+YqHvvnmwUokNnwerXQOA2kt8ARDVw= | ||||
| github.com/containers/common v0.47.4 h1:kS202Z/bTQIM/pwyuJ+lF8143Uli6AB9Q9OVR0xa9CM= | ||||
| github.com/containers/common v0.47.4/go.mod h1:HgX0mFXyB0Tbe2REEIp9x9CxET6iSzmHfwR6S/t2LZc= | ||||
| github.com/containers/common v0.47.5-0.20220222185251-194ee74231c3 h1:1JwpCh5a+VW7nNCwjlZGvT6TNvyxel+mhZLRMIqO6Mw= | ||||
| github.com/containers/common v0.47.5-0.20220222185251-194ee74231c3/go.mod h1:QJmjMDNLbFmMYUQ66R7AvKOC3s5FNI7L+jxIRqQbik0= | ||||
| github.com/containers/conmon v2.0.20+incompatible h1:YbCVSFSCqFjjVwHTPINGdMX1F6JXHGTUje2ZYobNrkg= | ||||
| github.com/containers/conmon v2.0.20+incompatible/go.mod h1:hgwZ2mtuDrppv78a/cOBNiCm6O0UMWGx1mu7P00nu5I= | ||||
| github.com/containers/image/v5 v5.19.1 h1:g4/+XIuh1kRoRn2MfLDhfHhkNOIO9JtqhSyo55tjpfE= | ||||
|  | @ -347,6 +371,7 @@ github.com/containers/storage v1.38.3-0.20220214113600-26c561f9a645 h1:2zzCKvfSq | |||
| github.com/containers/storage v1.38.3-0.20220214113600-26c561f9a645/go.mod h1:ugJrnJvpb6LrUshIYF/9g9YP9D4VTCYP+wShSiuqxN4= | ||||
| github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk= | ||||
| github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= | ||||
| github.com/coreos/etcd v3.3.13+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= | ||||
| github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= | ||||
| github.com/coreos/go-iptables v0.4.5/go.mod h1:/mVI274lEDI2ns62jHCDnCyBF9Iwsmekav8Dbxlm1MU= | ||||
| github.com/coreos/go-iptables v0.5.0/go.mod h1:/mVI274lEDI2ns62jHCDnCyBF9Iwsmekav8Dbxlm1MU= | ||||
|  | @ -467,6 +492,7 @@ github.com/felixge/httpsnoop v1.0.1 h1:lvB5Jl89CsZtGIWuTcDM1E/vkVs49/Ml7JJe07l8S | |||
| github.com/felixge/httpsnoop v1.0.1/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U= | ||||
| github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= | ||||
| github.com/form3tech-oss/jwt-go v3.2.2+incompatible/go.mod h1:pbq4aXjuKjdthFRnoDwaVPLA+WlJuPGy+QneDUgJi2k= | ||||
| github.com/form3tech-oss/jwt-go v3.2.3+incompatible/go.mod h1:pbq4aXjuKjdthFRnoDwaVPLA+WlJuPGy+QneDUgJi2k= | ||||
| github.com/frankban/quicktest v1.11.3/go.mod h1:wRf/ReqHper53s+kmmSZizM8NamnL3IM0I9ntUbOk+k= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | ||||
|  | @ -479,6 +505,7 @@ github.com/fullsailor/pkcs7 v0.0.0-20190404230743-d7302db945fa/go.mod h1:KnogPXt | |||
| github.com/fullstorydev/grpcurl v1.6.0/go.mod h1:ZQ+ayqbKMJNhzLmbpCiurTVlaK2M/3nqZCxaQ2Ze/sM= | ||||
| github.com/fzipp/gocyclo v0.3.1/go.mod h1:DJHO6AUmbdqj2ET4Z9iArSuwWgYDRryYt2wASxc7x3E= | ||||
| github.com/garyburd/redigo v0.0.0-20150301180006-535138d7bcd7/go.mod h1:NR3MbYisc3/PwhQ00EMzDiPmrwpPxAn5GI05/YaO1SY= | ||||
| github.com/getsentry/raven-go v0.2.0/go.mod h1:KungGk8q33+aIAZUIVWZDr2OfAEBsO49PX4NzFV5kcQ= | ||||
| github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
| github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk= | ||||
| github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= | ||||
|  | @ -495,25 +522,35 @@ github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2 | |||
| github.com/go-ini/ini v1.25.4/go.mod h1:ByCAeIL28uOIIG0E3PJtZPDL8WnHpFKFOtgjp+3Ies8= | ||||
| github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= | ||||
| github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= | ||||
| github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= | ||||
| github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= | ||||
| github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= | ||||
| github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas= | ||||
| github.com/go-logr/logr v0.2.0/go.mod h1:z6/tIYblkpsD+a4lm/fGIIU9mZ+XfAiaFtq7xTgseGU= | ||||
| github.com/go-logr/logr v0.4.0 h1:K7/B1jt6fIBQVd4Owv2MqGQClcgf0R266+7C/QjRcLc= | ||||
| github.com/go-logr/logr v0.4.0/go.mod h1:z6/tIYblkpsD+a4lm/fGIIU9mZ+XfAiaFtq7xTgseGU= | ||||
| github.com/go-logr/logr v1.2.0/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= | ||||
| github.com/go-logr/logr v1.2.1/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= | ||||
| github.com/go-logr/logr v1.2.2 h1:ahHml/yUpnlb96Rp8HCvtYVPY8ZYpxq3g7UYchIYwbs= | ||||
| github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= | ||||
| github.com/go-logr/stdr v1.2.0/go.mod h1:YkVgnZu1ZjjL7xTxrfm/LLZBfkhTqSR1ydtm6jTKKwI= | ||||
| github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= | ||||
| github.com/go-ole/go-ole v1.2.5/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= | ||||
| github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= | ||||
| github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0= | ||||
| github.com/go-openapi/jsonpointer v0.19.2/go.mod h1:3akKfEdA7DF1sugOqz1dVQHBcuDBPKZGEoHC/NkiQRg= | ||||
| github.com/go-openapi/jsonpointer v0.19.3/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= | ||||
| github.com/go-openapi/jsonpointer v0.19.5/go.mod h1:Pl9vOtqEWErmShwVjC8pYs9cog34VGT37dQOVbmoatg= | ||||
| github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg= | ||||
| github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwohSTlpa0o73RUL1owJc= | ||||
| github.com/go-openapi/jsonreference v0.19.3/go.mod h1:rjx6GuL8TTa9VaixXglHmQmIL98+wF9xc8zWvFonSJ8= | ||||
| github.com/go-openapi/jsonreference v0.19.5/go.mod h1:RdybgQwPxbL4UEjuAruzK1x3nE69AqPYEJeo/TWfEeg= | ||||
| github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc= | ||||
| github.com/go-openapi/spec v0.19.3/go.mod h1:FpwSN1ksY1eteniUU7X0N/BgJ7a4WvBFVA8Lj9mJglo= | ||||
| github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I= | ||||
| github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= | ||||
| github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk= | ||||
| github.com/go-openapi/swag v0.19.14/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ= | ||||
| github.com/go-redis/redis v6.15.8+incompatible/go.mod h1:NAIEuMOZ/fxfXJIrKDQDz8wamY7mA7PouImQ2Jvg6kA= | ||||
| github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= | ||||
| github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= | ||||
|  | @ -600,6 +637,7 @@ github.com/golangci/revgrep v0.0.0-20210930125155-c22e5001d4f2/go.mod h1:LK+zW4M | |||
| github.com/golangci/unconvert v0.0.0-20180507085042-28b1c447d1f4/go.mod h1:Izgrg8RkN3rCIMLGE9CyYmU9pY2Jer6DgANEnZ/L/cQ= | ||||
| github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= | ||||
| github.com/google/btree v1.0.1/go.mod h1:xXMiIv4Fb/0kKde4SpL7qlzvu5cMJDRkFDxJfI9uaxA= | ||||
| github.com/google/certificate-transparency-go v1.0.21/go.mod h1:QeJfpSbVSfYc7RgB3gJFj9cbuQMMchQxrWXz8Ruopmg= | ||||
| github.com/google/certificate-transparency-go v1.1.1/go.mod h1:FDKqPvSXawb2ecErVRrD+nfy23RCzyl7eqVCEmlT1Zs= | ||||
| github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= | ||||
|  | @ -620,8 +658,9 @@ github.com/google/go-containerregistry v0.5.1/go.mod h1:Ct15B4yir3PLOP5jsy0GNeYV | |||
| github.com/google/go-intervals v0.0.2 h1:FGrVEiUnTRKR8yE04qzXYaJMtnIYqobR5QbblK3ixcM= | ||||
| github.com/google/go-intervals v0.0.2/go.mod h1:MkaR3LNRfeKLPmqgJYs4E66z5InYjmCjbbr4TQlcT6Y= | ||||
| github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/gofuzz v1.1.0 h1:Hsa8mG0dQ46ij8Sl2AYJDUv1oA9/d6Vk+3LG99Oe02g= | ||||
| github.com/google/gofuzz v1.1.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0= | ||||
| github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= | ||||
| github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= | ||||
| github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= | ||||
| github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= | ||||
|  | @ -695,6 +734,7 @@ github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79/go.mod h1:Fecb | |||
| github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= | ||||
| github.com/grpc-ecosystem/go-grpc-middleware v1.0.1-0.20190118093823-f849b5445de4/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= | ||||
| github.com/grpc-ecosystem/go-grpc-middleware v1.2.2/go.mod h1:EaizFBKfUKtMIF5iaDEhniwNedqGo9FuLFzppDr3uwI= | ||||
| github.com/grpc-ecosystem/go-grpc-middleware v1.3.0/go.mod h1:z0ButlSOZa5vEBq9m2m2hlwIgKw+rp3sdCBRoJY+30Y= | ||||
| github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= | ||||
| github.com/grpc-ecosystem/grpc-gateway v1.9.5/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= | ||||
|  | @ -706,8 +746,9 @@ github.com/hashicorp/consul/api v1.11.0/go.mod h1:XjsvQN+RJGWI2TWy1/kqaE16HrR2J/ | |||
| github.com/hashicorp/consul/sdk v0.1.1/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= | ||||
| github.com/hashicorp/consul/sdk v0.8.0/go.mod h1:GBvyrGALthsZObzUGsfgHZQDXjg4lOjagTIwIR1vPms= | ||||
| github.com/hashicorp/errwrap v0.0.0-20141028054710-7554cd9344ce/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||
| github.com/hashicorp/errwrap v1.0.0 h1:hLrqtEDnRye3+sgx6z4qVLNuviH3MR5aQ0ykNJa/UYA= | ||||
| github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||
| github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I= | ||||
| github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= | ||||
| github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= | ||||
|  | @ -762,6 +803,7 @@ github.com/imdario/mergo v0.3.12/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH | |||
| github.com/inconshreveable/mousetrap v1.0.0 h1:Z8tu5sraLXCXIcARxBp/8cbvlwVa7Z1NHg9XEKhtSvM= | ||||
| github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= | ||||
| github.com/insomniacslk/dhcp v0.0.0-20211026125128-ad197bcd36fd/go.mod h1:h+MxyHxRg9NH3terB1nfRIUaQEcI0XOVkdR9LNBlp8E= | ||||
| github.com/intel/goresctrl v0.2.0/go.mod h1:+CZdzouYFn5EsxgqAQTEzMfwKwuc0fVdMrT9FCCAVRQ= | ||||
| github.com/ishidawataru/sctp v0.0.0-20210226210310-f2269e66cdee h1:PAXLXk1heNZ5yokbMBpVLZQxo43wCZxRwl00mX+dd44= | ||||
| github.com/ishidawataru/sctp v0.0.0-20210226210310-f2269e66cdee/go.mod h1:co9pwDoBCm1kGxawmb4sPq0cSIOOWNPT4KnHotMP1Zg= | ||||
| github.com/j-keck/arping v0.0.0-20160618110441-2cf9dc699c56/go.mod h1:ymszkNOg6tORTn+6F6j+Jc8TOr5osrynvN6ivFWZ2GA= | ||||
|  | @ -783,7 +825,10 @@ github.com/jmoiron/sqlx v1.2.0/go.mod h1:1FEQNm3xlJgrMD+FBdI9+xvCksHtbpVBBw5dYhB | |||
| github.com/joefitzgerald/rainbow-reporter v0.1.0/go.mod h1:481CNgqmVHQZzdIbN52CupLJyoVwB10FQ/IQlF1pdL8= | ||||
| github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= | ||||
| github.com/jonboulle/clockwork v0.2.0/go.mod h1:Pkfl5aHPm1nk2H9h0bjmnJD/BcgbGXUBGnn1kMkgxc8= | ||||
| github.com/jonboulle/clockwork v0.2.2/go.mod h1:Pkfl5aHPm1nk2H9h0bjmnJD/BcgbGXUBGnn1kMkgxc8= | ||||
| github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= | ||||
| github.com/josharian/txtarfs v0.0.0-20210218200122-0702f000015a/go.mod h1:izVPOvVRsHiKkeGCT6tYBNWyDVuzj9wAaBb5R9qamfw= | ||||
| github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= | ||||
| github.com/jsimonetti/rtnetlink v0.0.0-20190606172950-9527aa82566a/go.mod h1:Oz+70psSo5OFh8DBl0Zv2ACw7Esh6pPUphlvZG9x7uw= | ||||
| github.com/jsimonetti/rtnetlink v0.0.0-20200117123717-f846d4f6c1f4/go.mod h1:WGuG/smIU4J/54PblvSbh+xvCZmpJnFgr3ds6Z55XMQ= | ||||
| github.com/jsimonetti/rtnetlink v0.0.0-20201009170750-9c6f07d100c1/go.mod h1:hqoO/u39cqLeBLebZ8fWdE96O7FxrAsRYhnVOdgHxok= | ||||
|  | @ -800,6 +845,7 @@ github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/X | |||
| github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= | ||||
| github.com/juju/ratelimit v1.0.1/go.mod h1:qapgC/Gy+xNh9UxzV13HGGl/6UXNN+ct+vwSgWNm/qk= | ||||
| github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= | ||||
| github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= | ||||
| github.com/julz/importas v0.0.0-20210419104244-841f0c0fe66d/go.mod h1:oSFU2R4XK/P7kNBrnL/FEQlDGN1/6WoxXEjSSXO0DV0= | ||||
| github.com/k0kubun/colorstring v0.0.0-20150214042306-9440f1994b88/go.mod h1:3w7q1U84EfirKl04SVQ/s7nPm1ZPhiXd34z40TNz36k= | ||||
| github.com/kevinburke/ssh_config v0.0.0-20201106050909-4977a11b4351/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM= | ||||
|  | @ -850,11 +896,13 @@ github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0/go.mod h1:zJYVVT2 | |||
| github.com/lyft/protoc-gen-star v0.5.3/go.mod h1:V0xaHgaf5oCCqmcxYcWiDfTiKsZsRc87/1qhoTACD8w= | ||||
| github.com/magefile/mage v1.11.0/go.mod h1:z5UZb/iS3GoOSn0JgWuiw7dxlurVYTu+/jHXqQg881A= | ||||
| github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= | ||||
| github.com/magiconair/properties v1.8.1/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= | ||||
| github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= | ||||
| github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= | ||||
| github.com/mailru/easyjson v0.0.0-20190614124828-94de47d64c63/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= | ||||
| github.com/mailru/easyjson v0.0.0-20190626092158-b2ccc519800e/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc= | ||||
| github.com/mailru/easyjson v0.7.0/go.mod h1:KAzv3t3aY1NaHWoQz1+4F1ccyAH66Jk7yos7ldAVICs= | ||||
| github.com/mailru/easyjson v0.7.6/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= | ||||
| github.com/manifoldco/promptui v0.9.0 h1:3V4HzJk1TtXW1MTZMP7mdlwbBpIinw3HztaIlYthEiA= | ||||
| github.com/manifoldco/promptui v0.9.0/go.mod h1:ka04sppxSGFAtxX0qhlYQjISsg9mR4GWtQEhdbn6Pgg= | ||||
| github.com/maratori/testpackage v1.0.1/go.mod h1:ddKdw+XG0Phzhx8BFDTKgpWP4i7MpApTE5fXSKAqwDU= | ||||
|  | @ -941,8 +989,11 @@ github.com/moby/sys/mountinfo v0.4.1/go.mod h1:rEr8tzG/lsIZHBtN/JjGG+LMYx9eXgW2J | |||
| github.com/moby/sys/mountinfo v0.5.0/go.mod h1:3bMD3Rg+zkqx8MRYPi7Pyb0Ie97QEBmdxbhnCLlSvSU= | ||||
| github.com/moby/sys/mountinfo v0.6.0 h1:gUDhXQx58YNrpHlK4nSL+7y2pxFZkUcXqzFDKWdC0Oo= | ||||
| github.com/moby/sys/mountinfo v0.6.0/go.mod h1:3bMD3Rg+zkqx8MRYPi7Pyb0Ie97QEBmdxbhnCLlSvSU= | ||||
| github.com/moby/sys/signal v0.6.0/go.mod h1:GQ6ObYZfqacOwTtlXvcmh9A26dVRul/hbOZn88Kg8Tg= | ||||
| github.com/moby/sys/symlink v0.1.0/go.mod h1:GGDODQmbFOjFsXvfLVn3+ZRxkch54RkSiGqsZeMYowQ= | ||||
| github.com/moby/sys/symlink v0.2.0/go.mod h1:7uZVF2dqJjG/NsClqul95CqKOBRQyYSNnJ6BMgR/gFs= | ||||
| github.com/moby/term v0.0.0-20200312100748-672ec06f55cd/go.mod h1:DdlQx2hp0Ss5/fLikoLlEeIYiATotOjgB//nb973jeo= | ||||
| github.com/moby/term v0.0.0-20210610120745-9d4ed1856297/go.mod h1:vgPCkQMyxTZ7IDy8SXRufE172gr8+K/JE/7hHFxHW3A= | ||||
| github.com/moby/term v0.0.0-20210619224110-3f7ff695adc6 h1:dcztxKSvZ4Id8iPpHERQBbIJfabdt4wUm5qy3wOL2Zc= | ||||
| github.com/moby/term v0.0.0-20210619224110-3f7ff695adc6/go.mod h1:E2VnQOmVuvZB6UYnnDB0qG5Nq/1tD9acaOpo6xmt0Kw= | ||||
| github.com/moby/vpnkit v0.5.0/go.mod h1:KyjUrL9cb6ZSNNAUwZfqRjhwwgJ3BJN+kXh0t43WTUQ= | ||||
|  | @ -964,6 +1015,7 @@ github.com/mrunalp/fileutils v0.5.0/go.mod h1:M1WthSahJixYnrXQl/DFQuteStB1weuxD2 | |||
| github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= | ||||
| github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= | ||||
| github.com/mwitkow/go-proto-validators v0.0.0-20180403085117-0950a7990007/go.mod h1:m2XC9Qq0AlmmVksL6FktJCdTYyLk7V3fKyp0sl1yWQo= | ||||
| github.com/mwitkow/go-proto-validators v0.2.0/go.mod h1:ZfA1hW+UH/2ZHOWvQ3HnQaU0DtnpXu850MZiy+YUgcc= | ||||
| github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw= | ||||
|  | @ -1005,6 +1057,7 @@ github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8 | |||
| github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM= | ||||
| github.com/opencontainers/image-spec v1.0.0/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= | ||||
| github.com/opencontainers/image-spec v1.0.1/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= | ||||
| github.com/opencontainers/image-spec v1.0.2-0.20211117181255-693428a734f5/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= | ||||
| github.com/opencontainers/image-spec v1.0.2/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0= | ||||
| github.com/opencontainers/image-spec v1.0.3-0.20211202193544-a5463b7f9c84/go.mod h1:Qnt1q4cjDNQI9bT832ziho5Iw2BhK8o1KwLOwW56VP4= | ||||
| github.com/opencontainers/image-spec v1.0.3-0.20220114050600-8b9d41f48198 h1:+czc/J8SlhPKLOtVLMQc+xDCFBT73ZStMsRhSsUhsSg= | ||||
|  | @ -1079,8 +1132,9 @@ github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDf | |||
| github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= | ||||
| github.com/prometheus/client_golang v1.1.0/go.mod h1:I1FGZT9+L76gKKOs5djB6ezCbFQP1xR9D75/vuwEF3g= | ||||
| github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= | ||||
| github.com/prometheus/client_golang v1.7.1 h1:NTGy1Ja9pByO+xAeH/qiWnLrKtr3hJPNjaVUwnjpdpA= | ||||
| github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= | ||||
| github.com/prometheus/client_golang v1.11.0 h1:HNkLOAEQMIDv/K+04rukrLx6ch7msSRwf3/SASFAGtQ= | ||||
| github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= | ||||
| github.com/prometheus/client_model v0.0.0-20171117100541-99fa1f4be8e5/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= | ||||
| github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= | ||||
|  | @ -1093,8 +1147,10 @@ github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y8 | |||
| github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= | ||||
| github.com/prometheus/common v0.6.0/go.mod h1:eBmuwkDJBwy6iBfxCBob6t6dR6ENT/y+J+Zk0j9GMYc= | ||||
| github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= | ||||
| github.com/prometheus/common v0.10.0 h1:RyRA7RzGXQZiW+tGMr7sxa85G1z0yOpM1qq5c8lNawc= | ||||
| github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= | ||||
| github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= | ||||
| github.com/prometheus/common v0.30.0 h1:JEkYlQnpzrzQFxi6gnukFPdQ+ac82oRhzMcIduJu/Ug= | ||||
| github.com/prometheus/common v0.30.0/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= | ||||
| github.com/prometheus/procfs v0.0.0-20180125133057-cb4147076ac7/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= | ||||
| github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= | ||||
| github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= | ||||
|  | @ -1105,8 +1161,9 @@ github.com/prometheus/procfs v0.0.5/go.mod h1:4A/X28fw3Fc593LaREMrKMqOKvUAntwMDa | |||
| github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= | ||||
| github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= | ||||
| github.com/prometheus/procfs v0.2.0/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= | ||||
| github.com/prometheus/procfs v0.6.0 h1:mxy4L2jP6qMonqmq+aTtOx1ifVWUgG/TAmntgbh3xv4= | ||||
| github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | ||||
| github.com/prometheus/procfs v0.7.3 h1:4jVXhlkAyzOScmCkXBTOLRLTz8EeU+eyjrwB/EPq0VU= | ||||
| github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= | ||||
| github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU= | ||||
| github.com/pseudomuto/protoc-gen-doc v1.3.2/go.mod h1:y5+P6n3iGrbKG+9O04V5ld71in3v/bX88wUwgt+U8EA= | ||||
| github.com/pseudomuto/protokit v0.2.0/go.mod h1:2PdH30hxVHsup8KpBTOXTBeMVhJZVio3Q8ViKSAXT0Q= | ||||
|  | @ -1173,6 +1230,7 @@ github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1 | |||
| github.com/smartystreets/goconvey v0.0.0-20190330032615-68dc04aab96a/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= | ||||
| github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= | ||||
| github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM= | ||||
| github.com/soheilhy/cmux v0.1.5/go.mod h1:T7TcVDs9LWfQgPlPsdngu6I6QIoyIFZDDC6sNE1GqG0= | ||||
| github.com/sonatard/noctx v0.0.1/go.mod h1:9D2D/EoULe8Yy2joDHJj7bv3sZoq9AaSb8B4lqBjiZI= | ||||
| github.com/songgao/water v0.0.0-20200317203138-2b4b6d7c09d8/go.mod h1:P5HUIBuIWKbyjl083/loAegFkfbFNx5i2qEP4CNbm7E= | ||||
| github.com/sourcegraph/go-diff v0.6.1/go.mod h1:iBszgVvyxdc8SFZ7gm69go2KDdt3ag071iBaWPF6cjs= | ||||
|  | @ -1188,6 +1246,7 @@ github.com/spf13/cobra v0.0.2-0.20171109065643-2da4a54c5cee/go.mod h1:1l0Ry5zgKv | |||
| github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= | ||||
| github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU= | ||||
| github.com/spf13/cobra v1.0.0/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE= | ||||
| github.com/spf13/cobra v1.1.3/go.mod h1:pGADOWyqRD/YMrPZigI/zbliZ2wVD/23d+is3pSWzOo= | ||||
| github.com/spf13/cobra v1.2.1/go.mod h1:ExllRjgxM/piMAM+3tAZvg8fsklGAf3tPfi+i8t68Nk= | ||||
| github.com/spf13/cobra v1.3.0 h1:R7cSvGu+Vv+qX0gW5R/85dx2kmmJT5z5NM8ifdYjdn0= | ||||
| github.com/spf13/cobra v1.3.0/go.mod h1:BrRVncBjOJa/eUcVVm9CE+oC6as8k+VYr4NY7WCi9V4= | ||||
|  | @ -1201,6 +1260,7 @@ github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= | |||
| github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= | ||||
| github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s= | ||||
| github.com/spf13/viper v1.4.0/go.mod h1:PTJ7Z/lr49W6bUbkmS1V3by4uWynFiR9p7+dSq/yZzE= | ||||
| github.com/spf13/viper v1.7.0/go.mod h1:8WkrPz2fc9jxqZNCJI/76HCieCp4Q8HaLFoCha5qpdg= | ||||
| github.com/spf13/viper v1.8.1/go.mod h1:o0Pch8wJ9BVSWGQMbra6iw0oQ5oktSIBaujf1rJH9Ns= | ||||
| github.com/spf13/viper v1.9.0/go.mod h1:+i6ajR7OX2XaiBkrcZJFK21htRk7eDeLg7+O6bhUPP4= | ||||
| github.com/spf13/viper v1.10.0/go.mod h1:SoyBPwAtKDzypXNDFKN5kzH7ppppbGZtls1UpIy5AsM= | ||||
|  | @ -1245,10 +1305,12 @@ github.com/tklauser/numcpus v0.3.0/go.mod h1:yFGUr7TUHQRAhyqBcEg0Ge34zDBAsIvJJcy | |||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20200427203606-3cfed13b9966/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| github.com/tmc/grpc-websocket-proxy v0.0.0-20201229170055-e5319fda7802/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= | ||||
| github.com/tomarrell/wrapcheck/v2 v2.4.0/go.mod h1:68bQ/eJg55BROaRTbMjC7vuhL2OgfoG8bLp9ZyoBfyY= | ||||
| github.com/tomasen/realip v0.0.0-20180522021738-f0c99a92ddce/go.mod h1:o8v6yHRoik09Xen7gje4m9ERNah1d1PPsVq1VEx9vE4= | ||||
| github.com/tommy-muehle/go-mnd/v2 v2.4.0/go.mod h1:WsUAkMJMYww6l/ufffCD3m+P7LEvr8TnZn9lwVDlgzw= | ||||
| github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= | ||||
| github.com/tv42/httpunix v0.0.0-20191220191345-2ba4b9c3382c/go.mod h1:hzIxponao9Kjc7aWznkXaL4U4TWaDSs8zcsY4Ka08nM= | ||||
| github.com/u-root/uio v0.0.0-20210528114334-82958018845c/go.mod h1:LpEX5FO/cB+WF4TYGY1V5qktpaZLkKkSegbr0V4eYXA= | ||||
| github.com/uber/jaeger-client-go v2.30.0+incompatible h1:D6wyKGCecFaSRUpo8lCVbaOOb6ThwMmTEbhRwtKR97o= | ||||
| github.com/uber/jaeger-client-go v2.30.0+incompatible/go.mod h1:WVhlPFC8FDjOFMMWRy2pZqQJSXxYSwNYOkTr/Z6d3Kk= | ||||
|  | @ -1330,6 +1392,10 @@ go.etcd.io/etcd/client/pkg/v3 v3.5.0/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3 | |||
| go.etcd.io/etcd/client/pkg/v3 v3.5.1/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= | ||||
| go.etcd.io/etcd/client/v2 v2.305.0/go.mod h1:h9puh54ZTgAKtEbut2oe9P4L/oqKCVB6xsXlzd7alYQ= | ||||
| go.etcd.io/etcd/client/v2 v2.305.1/go.mod h1:pMEacxZW7o8pg4CrFE7pquyCJJzZvkvdD2RibOCCCGs= | ||||
| go.etcd.io/etcd/client/v3 v3.5.0/go.mod h1:AIKXXVX/DQXtfTEqBryiLTUXwON+GuvO6Z7lLS/oTh0= | ||||
| go.etcd.io/etcd/pkg/v3 v3.5.0/go.mod h1:UzJGatBQ1lXChBkQF0AuAtkRQMYnHubxAEYIrC3MSsE= | ||||
| go.etcd.io/etcd/raft/v3 v3.5.0/go.mod h1:UFOHSIvO/nKwd4lhkwabrTD3cqW5yVyYYf/KlD00Szc= | ||||
| go.etcd.io/etcd/server/v3 v3.5.0/go.mod h1:3Ah5ruV+M+7RZr0+Y/5mNLwC+eQlni+mQmOVdCRJoS4= | ||||
| go.mozilla.org/mozlog v0.0.0-20170222151521-4bb13139d403/go.mod h1:jHoPAGnDrCy6kaI2tAze5Prf0Nr0w/oNkROt2lw3n3o= | ||||
| go.mozilla.org/pkcs7 v0.0.0-20200128120323-432b2356ecb1 h1:A/5uWzF44DlIgdm/PQFwfMkW0JX+cIcQi/SwLAmZP5M= | ||||
| go.mozilla.org/pkcs7 v0.0.0-20200128120323-432b2356ecb1/go.mod h1:SNgMg+EgDFwmvSmLRTNKC5fegJjB7v23qTQ0XLGUNHk= | ||||
|  | @ -1341,11 +1407,33 @@ go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= | |||
| go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= | ||||
| go.opencensus.io v0.23.0 h1:gqCw0LfLxScz8irSi8exQc7fyQ0fKQU/qnC/X8+V/1M= | ||||
| go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= | ||||
| go.opentelemetry.io/contrib v0.20.0/go.mod h1:G/EtFaa6qaN7+LxqfIAT3GiZa7Wv5DTBUzl5H4LY0Kc= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.20.0/go.mod h1:oVGt1LRbBOBq1A5BQLlUg9UaU/54aiHw8cgjV3aWZ/E= | ||||
| go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.28.0/go.mod h1:vEhqr0m4eTc+DWxfsXoXue2GBgV2uUwVznkGIHW/e5w= | ||||
| go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.20.0/go.mod h1:2AboqHi0CiIZU0qwhtUfCYD1GeUzvvIXWNkhDt7ZMG4= | ||||
| go.opentelemetry.io/otel v0.20.0/go.mod h1:Y3ugLH2oa81t5QO+Lty+zXf8zC9L26ax4Nzoxm/dooo= | ||||
| go.opentelemetry.io/otel v1.3.0/go.mod h1:PWIKzi6JCp7sM0k9yZ43VX+T345uNbAkDKwHVjb2PTs= | ||||
| go.opentelemetry.io/otel/exporters/otlp v0.20.0/go.mod h1:YIieizyaN77rtLJra0buKiNBOm9XQfkPEKBeuhoMwAM= | ||||
| go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.3.0/go.mod h1:VpP4/RMn8bv8gNo9uK7/IMY4mtWLELsS+JIP0inH0h4= | ||||
| go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.3.0/go.mod h1:hO1KLR7jcKaDDKDkvI9dP/FIhpmna5lkqPUQdEjFAM8= | ||||
| go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.3.0/go.mod h1:keUU7UfnwWTWpJ+FWnyqmogPa82nuU5VUANFq49hlMY= | ||||
| go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.3.0/go.mod h1:QNX1aly8ehqqX1LEa6YniTU7VY9I6R3X/oPxhGdTceE= | ||||
| go.opentelemetry.io/otel/metric v0.20.0/go.mod h1:598I5tYlH1vzBjn+BTuhzTCSb/9debfNp6R3s7Pr1eU= | ||||
| go.opentelemetry.io/otel/oteltest v0.20.0/go.mod h1:L7bgKf9ZB7qCwT9Up7i9/pn0PWIa9FqQ2IQ8LoxiGnw= | ||||
| go.opentelemetry.io/otel/sdk v0.20.0/go.mod h1:g/IcepuwNsoiX5Byy2nNV0ySUF1em498m7hBWC279Yc= | ||||
| go.opentelemetry.io/otel/sdk v1.3.0/go.mod h1:rIo4suHNhQwBIPg9axF8V9CA72Wz2mKF1teNrup8yzs= | ||||
| go.opentelemetry.io/otel/sdk/export/metric v0.20.0/go.mod h1:h7RBNMsDJ5pmI1zExLi+bJK+Dr8NQCh0qGhm1KDnNlE= | ||||
| go.opentelemetry.io/otel/sdk/metric v0.20.0/go.mod h1:knxiS8Xd4E/N+ZqKmUPf3gTTZ4/0TjTXukfxjzSTpHE= | ||||
| go.opentelemetry.io/otel/trace v0.20.0/go.mod h1:6GjCW8zgDjwGHGa6GkyeB8+/5vjT16gUEi0Nf1iBdgw= | ||||
| go.opentelemetry.io/otel/trace v1.3.0/go.mod h1:c/VDhno8888bvQYmbYLqe41/Ldmr/KKunbvWM4/fEjk= | ||||
| go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= | ||||
| go.opentelemetry.io/proto/otlp v0.11.0/go.mod h1:QpEjXPrNQzrFDZgoTo49dgHR9RYRSrg3NAKnUGl9YpQ= | ||||
| go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= | ||||
| go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= | ||||
| go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= | ||||
| go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= | ||||
| go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A= | ||||
| go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= | ||||
| go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= | ||||
| go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= | ||||
| go.uber.org/multierr v1.4.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= | ||||
|  | @ -1373,6 +1461,7 @@ golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPh | |||
| golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20201002170205-7f63de1d35b0/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= | ||||
| golang.org/x/crypto v0.0.0-20210220033148-5ea612d1eb83/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I= | ||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/crypto v0.0.0-20210513164829-c07d793c2f9a/go.mod h1:P+XmwS30IXTQdn5tA2iutPOUgjI07+tq3H3K9MVA1s8= | ||||
|  | @ -1466,6 +1555,7 @@ golang.org/x/net v0.0.0-20201010224723-4f7140c49acb/go.mod h1:sp8m0HH+o8qH0wwXwY | |||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
|  | @ -1478,12 +1568,15 @@ golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT | |||
| golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210510120150-4163338589ed/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210520170846-37e1c6afe023/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210825183410-e898025ed96a/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20210929193557-e81a3d93ecf6/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20211105192438-b53810dc28af/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20211209124913-491a49abca63/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20211216030914-fe4d6282115f/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20220114011407-0dd24b26b47d h1:1n1fc535VhN8SYtD4cDUyNlfpAF2ROMM9+11equK3hs= | ||||
| golang.org/x/net v0.0.0-20220114011407-0dd24b26b47d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||
|  | @ -1583,9 +1676,11 @@ golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7w | |||
| golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200728102440-3e129f6d46b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200817155316-9781c653f443/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200831180312-196b9ba8737a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200909081042-eff7692f9009/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200916030750-2334cc1a136f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
|  | @ -1613,11 +1708,13 @@ golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7w | |||
| golang.org/x/sys v0.0.0-20210331175145-43e1dd70ce54/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210426230700-d19ff857e887/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210502180810-71e4cd670f79/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210525143221-35b2ab0089ea/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
|  | @ -1628,6 +1725,8 @@ golang.org/x/sys v0.0.0-20210816074244-15123e1e1f71/go.mod h1:oPkhp1MJrh7nUepCBc | |||
| golang.org/x/sys v0.0.0-20210816183151-1e6c022a8912/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210820121016-41cdb8703e55/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210831042530-f4d43177bf5e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210903071746-97244b99971b/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210906170528-6f6e22806c34/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210915083310-ed5796bab164/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
|  | @ -1646,6 +1745,7 @@ golang.org/x/sys v0.0.0-20220114195835-da31bd327af9 h1:XfKQ4OlFl8okEOr5UvAqFRVj8 | |||
| golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210220032956-6a3ed077a48d/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b h1:9zKuko04nR4gjZ4+DNjHqRlAJqbJETHwiNKDqTfOjfE= | ||||
| golang.org/x/term v0.0.0-20210615171337-6886f2dfbf5b/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
|  | @ -1663,8 +1763,10 @@ golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxb | |||
| golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20200416051211-89c76fbcd5d1/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e h1:EHBhcS0mlXEAVwNyO2dLfjToGsyY4j24pTs2ScHnX7s= | ||||
| golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/time v0.0.0-20210723032227-1f47c861a9ac h1:7zkz7BUtwNFFqcowJ+RIgu2MaV/MapERkDIy+mwPyjs= | ||||
| golang.org/x/time v0.0.0-20210723032227-1f47c861a9ac/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= | ||||
| golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
|  | @ -1931,8 +2033,9 @@ google.golang.org/grpc v1.39.0/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnD | |||
| google.golang.org/grpc v1.39.1/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= | ||||
| google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= | ||||
| google.golang.org/grpc v1.40.1/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= | ||||
| google.golang.org/grpc v1.42.0 h1:XT2/MFpuPFsEX2fWh3YQtHkZ+WYZFQRfaUgLZYj/p6A= | ||||
| google.golang.org/grpc v1.42.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= | ||||
| google.golang.org/grpc v1.43.0 h1:Eeu7bZtDZ2DpRCsLhUlcrLnvYaMK1Gz86a+hMVvELmM= | ||||
| google.golang.org/grpc v1.43.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= | ||||
| google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
|  | @ -1966,6 +2069,7 @@ gopkg.in/gcfg.v1 v1.2.3/go.mod h1:yesOnuUOFQAhST5vPY4nbZsb/huCgGGXlipJsBn0b3o= | |||
| gopkg.in/gemnasium/logrus-airbrake-hook.v2 v2.1.2/go.mod h1:Xk6kEKp8OKb+X14hQBKWaSkCsqBpgog8nAV2xsGOxlo= | ||||
| gopkg.in/inf.v0 v0.9.1 h1:73M5CoZyi3ZLMOyDlQh031Cx6N9NDJ2Vvfl76EDAgDc= | ||||
| gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw= | ||||
| gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/ini.v1 v1.62.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/ini.v1 v1.63.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
| gopkg.in/ini.v1 v1.66.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= | ||||
|  | @ -2009,27 +2113,32 @@ honnef.co/go/tools v0.2.1/go.mod h1:lPVVZ2BS5TfnjLyizF7o7hv7j9/L+8cZY2hLyjP9cGY= | |||
| k8s.io/api v0.20.1/go.mod h1:KqwcCVogGxQY3nBlRpwt+wpAMF/KjaCc7RpywacvqUo= | ||||
| k8s.io/api v0.20.4/go.mod h1:++lNL1AJMkDymriNniQsWRkMDzRaX2Y/POTUi8yvqYQ= | ||||
| k8s.io/api v0.20.6/go.mod h1:X9e8Qag6JV/bL5G6bU8sdVRltWKmdHsFUGS3eVndqE8= | ||||
| k8s.io/api v0.22.4 h1:UvyHW0ezB2oIgHAxlYoo6UJQObYXU7awuNarwoHEOjw= | ||||
| k8s.io/api v0.22.4/go.mod h1:Rgs+9gIGYC5laXQSZZ9JqT5NevNgoGiOdVWi1BAB3qk= | ||||
| k8s.io/api v0.22.5 h1:xk7C+rMjF/EGELiD560jdmwzrB788mfcHiNbMQLIVI8= | ||||
| k8s.io/api v0.22.5/go.mod h1:mEhXyLaSD1qTOf40rRiKXkc+2iCem09rWLlFwhCEiAs= | ||||
| k8s.io/apimachinery v0.20.1/go.mod h1:WlLqWAHZGg07AeltaI0MV5uk1Omp8xaN0JGLY6gkRpU= | ||||
| k8s.io/apimachinery v0.20.4/go.mod h1:WlLqWAHZGg07AeltaI0MV5uk1Omp8xaN0JGLY6gkRpU= | ||||
| k8s.io/apimachinery v0.20.6/go.mod h1:ejZXtW1Ra6V1O5H8xPBGz+T3+4gfkTCeExAHKU57MAc= | ||||
| k8s.io/apimachinery v0.22.4 h1:9uwcvPpukBw/Ri0EUmWz+49cnFtaoiyEhQTK+xOe7Ck= | ||||
| k8s.io/apimachinery v0.22.4/go.mod h1:yU6oA6Gnax9RrxGzVvPFFJ+mpnW6PBSqp0sx0I0HHW0= | ||||
| k8s.io/apimachinery v0.22.1/go.mod h1:O3oNtNadZdeOMxHFVxOreoznohCpy0z6mocxbZr7oJ0= | ||||
| k8s.io/apimachinery v0.22.5 h1:cIPwldOYm1Slq9VLBRPtEYpyhjIm1C6aAMAoENuvN9s= | ||||
| k8s.io/apimachinery v0.22.5/go.mod h1:xziclGKwuuJ2RM5/rSFQSYAj0zdbci3DH8kj+WvyN0U= | ||||
| k8s.io/apiserver v0.20.1/go.mod h1:ro5QHeQkgMS7ZGpvf4tSMx6bBOgPfE+f52KwvXfScaU= | ||||
| k8s.io/apiserver v0.20.4/go.mod h1:Mc80thBKOyy7tbvFtB4kJv1kbdD0eIH8k8vianJcbFM= | ||||
| k8s.io/apiserver v0.20.6/go.mod h1:QIJXNt6i6JB+0YQRNcS0hdRHJlMhflFmsBDeSgT1r8Q= | ||||
| k8s.io/apiserver v0.22.5/go.mod h1:s2WbtgZAkTKt679sYtSudEQrTGWUSQAPe6MupLnlmaQ= | ||||
| k8s.io/client-go v0.20.1/go.mod h1:/zcHdt1TeWSd5HoUe6elJmHSQ6uLLgp4bIJHVEuy+/Y= | ||||
| k8s.io/client-go v0.20.4/go.mod h1:LiMv25ND1gLUdBeYxBIwKpkSC5IsozMMmOOeSJboP+k= | ||||
| k8s.io/client-go v0.20.6/go.mod h1:nNQMnOvEUEsOzRRFIIkdmYOjAZrC8bgq0ExboWSU1I0= | ||||
| k8s.io/client-go v0.22.5/go.mod h1:cs6yf/61q2T1SdQL5Rdcjg9J1ElXSwbjSrW2vFImM4Y= | ||||
| k8s.io/code-generator v0.19.7/go.mod h1:lwEq3YnLYb/7uVXLorOJfxg+cUu2oihFhHZ0n9NIla0= | ||||
| k8s.io/component-base v0.20.1/go.mod h1:guxkoJnNoh8LNrbtiQOlyp2Y2XFCZQmrcg2n/DeYNLk= | ||||
| k8s.io/component-base v0.20.4/go.mod h1:t4p9EdiagbVCJKrQ1RsA5/V4rFQNDfRlevJajlGwgjI= | ||||
| k8s.io/component-base v0.20.6/go.mod h1:6f1MPBAeI+mvuts3sIdtpjljHWBQ2cIy38oBIWMYnrM= | ||||
| k8s.io/component-base v0.22.5/go.mod h1:VK3I+TjuF9eaa+Ln67dKxhGar5ynVbwnGrUiNF4MqCI= | ||||
| k8s.io/cri-api v0.17.3/go.mod h1:X1sbHmuXhwaHs9xxYffLqJogVsnI+f6cPRcgPel7ywM= | ||||
| k8s.io/cri-api v0.20.1/go.mod h1:2JRbKt+BFLTjtrILYVqQK5jqhI+XNdF6UiGMgczeBCI= | ||||
| k8s.io/cri-api v0.20.4/go.mod h1:2JRbKt+BFLTjtrILYVqQK5jqhI+XNdF6UiGMgczeBCI= | ||||
| k8s.io/cri-api v0.20.6/go.mod h1:ew44AjNXwyn1s0U4xCKGodU7J1HzBeZ1MpGrpa5r8Yc= | ||||
| k8s.io/cri-api v0.23.1/go.mod h1:REJE3PSU0h/LOV1APBrupxrEJqnoxZC8KWzkBUHwrK4= | ||||
| k8s.io/gengo v0.0.0-20200413195148-3a45101e95ac/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0= | ||||
| k8s.io/gengo v0.0.0-20200428234225-8167cfdcfc14/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0= | ||||
| k8s.io/gengo v0.0.0-20201113003025-83324d819ded/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E= | ||||
|  | @ -2038,13 +2147,17 @@ k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I= | |||
| k8s.io/klog/v2 v2.0.0/go.mod h1:PBfzABfn139FHAV07az/IF9Wp1bkk3vpT2XSJ76fSDE= | ||||
| k8s.io/klog/v2 v2.2.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y= | ||||
| k8s.io/klog/v2 v2.4.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y= | ||||
| k8s.io/klog/v2 v2.9.0 h1:D7HV+n1V57XeZ0m6tdRkfknthUaM06VFbWldOFh8kzM= | ||||
| k8s.io/klog/v2 v2.9.0/go.mod h1:hy9LJ/NvuK+iVyP4Ehqva4HxZG/oXyIS3n3Jmire4Ec= | ||||
| k8s.io/klog/v2 v2.30.0 h1:bUO6drIvCIsvZ/XFgfxoGFQU/a4Qkh0iAlvUR7vlHJw= | ||||
| k8s.io/klog/v2 v2.30.0/go.mod h1:y1WjHnz7Dj687irZUWR/WLkLc5N1YHtjLdmgWjndZn0= | ||||
| k8s.io/kube-openapi v0.0.0-20200805222855-6aeccd4b50c6/go.mod h1:UuqjUnNftUyPE5H64/qeyjQoUZhGpeFDVdxjTeEVN2o= | ||||
| k8s.io/kube-openapi v0.0.0-20201113171705-d219536bb9fd/go.mod h1:WOJ3KddDSol4tAGcJo0Tvi+dK12EcqSLqcWsryKMpfM= | ||||
| k8s.io/kube-openapi v0.0.0-20210421082810-95288971da7e/go.mod h1:vHXdDvt9+2spS2Rx9ql3I8tycm3H9FDfdUoIuKCefvw= | ||||
| k8s.io/kube-openapi v0.0.0-20211109043538-20434351676c/go.mod h1:vHXdDvt9+2spS2Rx9ql3I8tycm3H9FDfdUoIuKCefvw= | ||||
| k8s.io/kubernetes v1.13.0/go.mod h1:ocZa8+6APFNC2tX1DZASIbocyYT5jHzqFVsY5aoB7Jk= | ||||
| k8s.io/utils v0.0.0-20201110183641-67b214c5f920/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA= | ||||
| k8s.io/utils v0.0.0-20210819203725-bdf08cb9a70a/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA= | ||||
| k8s.io/utils v0.0.0-20210930125809-cb0fa318a74b/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA= | ||||
| mvdan.cc/gofumpt v0.1.1/go.mod h1:yXG1r1WqZVKWbVRtBWKWX9+CxGYfA51nSomhM0woR48= | ||||
| mvdan.cc/interfacer v0.0.0-20180901003855-c20040233aed/go.mod h1:Xkxe497xwlCKkIaQYRfC7CSLworTXY9RMqwhhCm+8Nc= | ||||
| mvdan.cc/lint v0.0.0-20170908181259-adc824a0674b/go.mod h1:2odslEg/xrtNQqCYg2/jCoyKnw3vv5biOc3JnIcYfL4= | ||||
|  | @ -2054,6 +2167,7 @@ rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= | |||
| rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= | ||||
| sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.0.14/go.mod h1:LEScyzhFmoF5pso/YSeBstl57mOzx9xlU9n85RGrDQg= | ||||
| sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.0.15/go.mod h1:LEScyzhFmoF5pso/YSeBstl57mOzx9xlU9n85RGrDQg= | ||||
| sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.0.22/go.mod h1:LEScyzhFmoF5pso/YSeBstl57mOzx9xlU9n85RGrDQg= | ||||
| sigs.k8s.io/structured-merge-diff/v4 v4.0.1/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw= | ||||
| sigs.k8s.io/structured-merge-diff/v4 v4.0.2/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw= | ||||
| sigs.k8s.io/structured-merge-diff/v4 v4.0.3/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw= | ||||
|  |  | |||
|  | @ -17,7 +17,7 @@ | |||
| // Package errdefs defines the common errors used throughout containerd
 | ||||
| // packages.
 | ||||
| //
 | ||||
| // Use with errors.Wrap and error.Wrapf to add context to an error.
 | ||||
| // Use with fmt.Errorf to add context to an error.
 | ||||
| //
 | ||||
| // To detect an error class, use the IsXXX functions to tell whether an error
 | ||||
| // is of a certain type.
 | ||||
|  | @ -28,8 +28,7 @@ package errdefs | |||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 
 | ||||
| 	"github.com/pkg/errors" | ||||
| 	"errors" | ||||
| ) | ||||
| 
 | ||||
| // Definitions of common error types used throughout containerd. All containerd
 | ||||
|  |  | |||
|  | @ -18,9 +18,9 @@ package errdefs | |||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	"github.com/pkg/errors" | ||||
| 	"google.golang.org/grpc/codes" | ||||
| 	"google.golang.org/grpc/status" | ||||
| ) | ||||
|  | @ -68,9 +68,9 @@ func ToGRPC(err error) error { | |||
| // ToGRPCf maps the error to grpc error codes, assembling the formatting string
 | ||||
| // and combining it with the target error string.
 | ||||
| //
 | ||||
| // This is equivalent to errors.ToGRPC(errors.Wrapf(err, format, args...))
 | ||||
| // This is equivalent to errdefs.ToGRPC(fmt.Errorf("%s: %w", fmt.Sprintf(format, args...), err))
 | ||||
| func ToGRPCf(err error, format string, args ...interface{}) error { | ||||
| 	return ToGRPC(errors.Wrapf(err, format, args...)) | ||||
| 	return ToGRPC(fmt.Errorf("%s: %w", fmt.Sprintf(format, args...), err)) | ||||
| } | ||||
| 
 | ||||
| // FromGRPC returns the underlying error from a grpc service based on the grpc error code
 | ||||
|  | @ -104,9 +104,9 @@ func FromGRPC(err error) error { | |||
| 
 | ||||
| 	msg := rebaseMessage(cls, err) | ||||
| 	if msg != "" { | ||||
| 		err = errors.Wrap(cls, msg) | ||||
| 		err = fmt.Errorf("%s: %w", msg, cls) | ||||
| 	} else { | ||||
| 		err = errors.WithStack(cls) | ||||
| 		err = cls | ||||
| 	} | ||||
| 
 | ||||
| 	return err | ||||
|  |  | |||
|  | @ -52,7 +52,8 @@ const ( | |||
| // WithLogger returns a new context with the provided logger. Use in
 | ||||
| // combination with logger.WithField(s) for great effect.
 | ||||
| func WithLogger(ctx context.Context, logger *logrus.Entry) context.Context { | ||||
| 	return context.WithValue(ctx, loggerKey{}, logger) | ||||
| 	e := logger.WithContext(ctx) | ||||
| 	return context.WithValue(ctx, loggerKey{}, e) | ||||
| } | ||||
| 
 | ||||
| // GetLogger retrieves the current logger from the context. If no logger is
 | ||||
|  | @ -61,7 +62,7 @@ func GetLogger(ctx context.Context) *logrus.Entry { | |||
| 	logger := ctx.Value(loggerKey{}) | ||||
| 
 | ||||
| 	if logger == nil { | ||||
| 		return L | ||||
| 		return L.WithContext(ctx) | ||||
| 	} | ||||
| 
 | ||||
| 	return logger.(*logrus.Entry) | ||||
|  |  | |||
|  | @ -1,3 +1,4 @@ | |||
| //go:build !linux
 | ||||
| // +build !linux
 | ||||
| 
 | ||||
| /* | ||||
|  |  | |||
|  | @ -38,12 +38,22 @@ func platformVector(platform specs.Platform) []specs.Platform { | |||
| 
 | ||||
| 	switch platform.Architecture { | ||||
| 	case "amd64": | ||||
| 		if amd64Version, err := strconv.Atoi(strings.TrimPrefix(platform.Variant, "v")); err == nil && amd64Version > 1 { | ||||
| 			for amd64Version--; amd64Version >= 1; amd64Version-- { | ||||
| 				vector = append(vector, specs.Platform{ | ||||
| 					Architecture: platform.Architecture, | ||||
| 					OS:           platform.OS, | ||||
| 					OSVersion:    platform.OSVersion, | ||||
| 					OSFeatures:   platform.OSFeatures, | ||||
| 					Variant:      "v" + strconv.Itoa(amd64Version), | ||||
| 				}) | ||||
| 			} | ||||
| 		} | ||||
| 		vector = append(vector, specs.Platform{ | ||||
| 			Architecture: "386", | ||||
| 			OS:           platform.OS, | ||||
| 			OSVersion:    platform.OSVersion, | ||||
| 			OSFeatures:   platform.OSFeatures, | ||||
| 			Variant:      platform.Variant, | ||||
| 		}) | ||||
| 	case "arm": | ||||
| 		if armVersion, err := strconv.Atoi(strings.TrimPrefix(platform.Variant, "v")); err == nil && armVersion > 5 { | ||||
|  |  | |||
|  | @ -18,6 +18,7 @@ package platforms | |||
| 
 | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
|  | @ -25,7 +26,6 @@ import ( | |||
| 
 | ||||
| 	"github.com/containerd/containerd/errdefs" | ||||
| 	"github.com/containerd/containerd/log" | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
| 
 | ||||
| // Present the ARM instruction set architecture, eg: v7, v8
 | ||||
|  | @ -48,7 +48,7 @@ func cpuVariant() string { | |||
| // by ourselves. We can just parse these information from /proc/cpuinfo
 | ||||
| func getCPUInfo(pattern string) (info string, err error) { | ||||
| 	if !isLinuxOS(runtime.GOOS) { | ||||
| 		return "", errors.Wrapf(errdefs.ErrNotImplemented, "getCPUInfo for OS %s", runtime.GOOS) | ||||
| 		return "", fmt.Errorf("getCPUInfo for OS %s: %w", runtime.GOOS, errdefs.ErrNotImplemented) | ||||
| 	} | ||||
| 
 | ||||
| 	cpuinfo, err := os.Open("/proc/cpuinfo") | ||||
|  | @ -75,7 +75,7 @@ func getCPUInfo(pattern string) (info string, err error) { | |||
| 		return "", err | ||||
| 	} | ||||
| 
 | ||||
| 	return "", errors.Wrapf(errdefs.ErrNotFound, "getCPUInfo for pattern: %s", pattern) | ||||
| 	return "", fmt.Errorf("getCPUInfo for pattern: %s: %w", pattern, errdefs.ErrNotFound) | ||||
| } | ||||
| 
 | ||||
| func getCPUVariant() string { | ||||
|  |  | |||
|  | @ -38,7 +38,7 @@ func isLinuxOS(os string) bool { | |||
| // The OS value should be normalized before calling this function.
 | ||||
| func isKnownOS(os string) bool { | ||||
| 	switch os { | ||||
| 	case "aix", "android", "darwin", "dragonfly", "freebsd", "hurd", "illumos", "js", "linux", "nacl", "netbsd", "openbsd", "plan9", "solaris", "windows", "zos": | ||||
| 	case "aix", "android", "darwin", "dragonfly", "freebsd", "hurd", "illumos", "ios", "js", "linux", "nacl", "netbsd", "openbsd", "plan9", "solaris", "windows", "zos": | ||||
| 		return true | ||||
| 	} | ||||
| 	return false | ||||
|  | @ -60,7 +60,7 @@ func isArmArch(arch string) bool { | |||
| // The arch value should be normalized before being passed to this function.
 | ||||
| func isKnownArch(arch string) bool { | ||||
| 	switch arch { | ||||
| 	case "386", "amd64", "amd64p32", "arm", "armbe", "arm64", "arm64be", "ppc64", "ppc64le", "mips", "mipsle", "mips64", "mips64le", "mips64p32", "mips64p32le", "ppc", "riscv", "riscv64", "s390", "s390x", "sparc", "sparc64", "wasm": | ||||
| 	case "386", "amd64", "amd64p32", "arm", "armbe", "arm64", "arm64be", "ppc64", "ppc64le", "loong64", "mips", "mipsle", "mips64", "mips64le", "mips64p32", "mips64p32le", "ppc", "riscv", "riscv64", "s390", "s390x", "sparc", "sparc64", "wasm": | ||||
| 		return true | ||||
| 	} | ||||
| 	return false | ||||
|  | @ -86,9 +86,11 @@ func normalizeArch(arch, variant string) (string, string) { | |||
| 	case "i386": | ||||
| 		arch = "386" | ||||
| 		variant = "" | ||||
| 	case "x86_64", "x86-64": | ||||
| 	case "x86_64", "x86-64", "amd64": | ||||
| 		arch = "amd64" | ||||
| 		variant = "" | ||||
| 		if variant == "v1" { | ||||
| 			variant = "" | ||||
| 		} | ||||
| 	case "aarch64", "arm64": | ||||
| 		arch = "arm64" | ||||
| 		switch variant { | ||||
|  |  | |||
|  | @ -16,27 +16,11 @@ | |||
| 
 | ||||
| package platforms | ||||
| 
 | ||||
| import ( | ||||
| 	"runtime" | ||||
| 
 | ||||
| 	specs "github.com/opencontainers/image-spec/specs-go/v1" | ||||
| ) | ||||
| 
 | ||||
| // DefaultString returns the default string specifier for the platform.
 | ||||
| func DefaultString() string { | ||||
| 	return Format(DefaultSpec()) | ||||
| } | ||||
| 
 | ||||
| // DefaultSpec returns the current platform's default platform specification.
 | ||||
| func DefaultSpec() specs.Platform { | ||||
| 	return specs.Platform{ | ||||
| 		OS:           runtime.GOOS, | ||||
| 		Architecture: runtime.GOARCH, | ||||
| 		// The Variant field will be empty if arch != ARM.
 | ||||
| 		Variant: cpuVariant(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // DefaultStrict returns strict form of Default.
 | ||||
| func DefaultStrict() MatchComparer { | ||||
| 	return OnlyStrict(DefaultSpec()) | ||||
|  |  | |||
							
								
								
									
										45
									
								
								vendor/github.com/containerd/containerd/platforms/defaults_darwin.go
								
								
									generated
								
								
									vendored
								
								
									Normal file
								
							
							
						
						
									
										45
									
								
								vendor/github.com/containerd/containerd/platforms/defaults_darwin.go
								
								
									generated
								
								
									vendored
								
								
									Normal file
								
							|  | @ -0,0 +1,45 @@ | |||
| //go:build darwin
 | ||||
| // +build darwin
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd 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 platforms | ||||
| 
 | ||||
| import ( | ||||
| 	"runtime" | ||||
| 
 | ||||
| 	specs "github.com/opencontainers/image-spec/specs-go/v1" | ||||
| ) | ||||
| 
 | ||||
| // DefaultSpec returns the current platform's default platform specification.
 | ||||
| func DefaultSpec() specs.Platform { | ||||
| 	return specs.Platform{ | ||||
| 		OS:           runtime.GOOS, | ||||
| 		Architecture: runtime.GOARCH, | ||||
| 		// The Variant field will be empty if arch != ARM.
 | ||||
| 		Variant: cpuVariant(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Default returns the default matcher for the platform.
 | ||||
| func Default() MatchComparer { | ||||
| 	return Ordered(DefaultSpec(), specs.Platform{ | ||||
| 		// darwin runtime also supports Linux binary via runu/LKL
 | ||||
| 		OS:           "linux", | ||||
| 		Architecture: runtime.GOARCH, | ||||
| 	}) | ||||
| } | ||||
|  | @ -1,4 +1,5 @@ | |||
| // +build !windows
 | ||||
| //go:build !windows && !darwin
 | ||||
| // +build !windows,!darwin
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd Authors. | ||||
|  | @ -18,6 +19,22 @@ | |||
| 
 | ||||
| package platforms | ||||
| 
 | ||||
| import ( | ||||
| 	"runtime" | ||||
| 
 | ||||
| 	specs "github.com/opencontainers/image-spec/specs-go/v1" | ||||
| ) | ||||
| 
 | ||||
| // DefaultSpec returns the current platform's default platform specification.
 | ||||
| func DefaultSpec() specs.Platform { | ||||
| 	return specs.Platform{ | ||||
| 		OS:           runtime.GOOS, | ||||
| 		Architecture: runtime.GOARCH, | ||||
| 		// The Variant field will be empty if arch != ARM.
 | ||||
| 		Variant: cpuVariant(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Default returns the default matcher for the platform.
 | ||||
| func Default() MatchComparer { | ||||
| 	return Only(DefaultSpec()) | ||||
|  |  | |||
|  | @ -1,5 +1,3 @@ | |||
| // +build windows
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd Authors. | ||||
| 
 | ||||
|  | @ -29,6 +27,18 @@ import ( | |||
| 	"golang.org/x/sys/windows" | ||||
| ) | ||||
| 
 | ||||
| // DefaultSpec returns the current platform's default platform specification.
 | ||||
| func DefaultSpec() specs.Platform { | ||||
| 	major, minor, build := windows.RtlGetNtVersionNumbers() | ||||
| 	return specs.Platform{ | ||||
| 		OS:           runtime.GOOS, | ||||
| 		Architecture: runtime.GOARCH, | ||||
| 		OSVersion:    fmt.Sprintf("%d.%d.%d", major, minor, build), | ||||
| 		// The Variant field will be empty if arch != ARM.
 | ||||
| 		Variant: cpuVariant(), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| type matchComparer struct { | ||||
| 	defaults        Matcher | ||||
| 	osVersionPrefix string | ||||
|  |  | |||
|  | @ -107,6 +107,8 @@ | |||
| package platforms | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"path" | ||||
| 	"regexp" | ||||
| 	"runtime" | ||||
| 	"strconv" | ||||
|  | @ -114,7 +116,6 @@ import ( | |||
| 
 | ||||
| 	"github.com/containerd/containerd/errdefs" | ||||
| 	specs "github.com/opencontainers/image-spec/specs-go/v1" | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
|  | @ -166,14 +167,14 @@ func (m *matcher) String() string { | |||
| func Parse(specifier string) (specs.Platform, error) { | ||||
| 	if strings.Contains(specifier, "*") { | ||||
| 		// TODO(stevvooe): need to work out exact wildcard handling
 | ||||
| 		return specs.Platform{}, errors.Wrapf(errdefs.ErrInvalidArgument, "%q: wildcards not yet supported", specifier) | ||||
| 		return specs.Platform{}, fmt.Errorf("%q: wildcards not yet supported: %w", specifier, errdefs.ErrInvalidArgument) | ||||
| 	} | ||||
| 
 | ||||
| 	parts := strings.Split(specifier, "/") | ||||
| 
 | ||||
| 	for _, part := range parts { | ||||
| 		if !specifierRe.MatchString(part) { | ||||
| 			return specs.Platform{}, errors.Wrapf(errdefs.ErrInvalidArgument, "%q is an invalid component of %q: platform specifier component must match %q", part, specifier, specifierRe.String()) | ||||
| 			return specs.Platform{}, fmt.Errorf("%q is an invalid component of %q: platform specifier component must match %q: %w", part, specifier, specifierRe.String(), errdefs.ErrInvalidArgument) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
|  | @ -205,7 +206,7 @@ func Parse(specifier string) (specs.Platform, error) { | |||
| 			return p, nil | ||||
| 		} | ||||
| 
 | ||||
| 		return specs.Platform{}, errors.Wrapf(errdefs.ErrInvalidArgument, "%q: unknown operating system or architecture", specifier) | ||||
| 		return specs.Platform{}, fmt.Errorf("%q: unknown operating system or architecture: %w", specifier, errdefs.ErrInvalidArgument) | ||||
| 	case 2: | ||||
| 		// In this case, we treat as a regular os/arch pair. We don't care
 | ||||
| 		// about whether or not we know of the platform.
 | ||||
|  | @ -227,7 +228,7 @@ func Parse(specifier string) (specs.Platform, error) { | |||
| 		return p, nil | ||||
| 	} | ||||
| 
 | ||||
| 	return specs.Platform{}, errors.Wrapf(errdefs.ErrInvalidArgument, "%q: cannot parse platform specifier", specifier) | ||||
| 	return specs.Platform{}, fmt.Errorf("%q: cannot parse platform specifier: %w", specifier, errdefs.ErrInvalidArgument) | ||||
| } | ||||
| 
 | ||||
| // MustParse is like Parses but panics if the specifier cannot be parsed.
 | ||||
|  | @ -246,20 +247,7 @@ func Format(platform specs.Platform) string { | |||
| 		return "unknown" | ||||
| 	} | ||||
| 
 | ||||
| 	return joinNotEmpty(platform.OS, platform.Architecture, platform.Variant) | ||||
| } | ||||
| 
 | ||||
| func joinNotEmpty(s ...string) string { | ||||
| 	var ss []string | ||||
| 	for _, s := range s { | ||||
| 		if s == "" { | ||||
| 			continue | ||||
| 		} | ||||
| 
 | ||||
| 		ss = append(ss, s) | ||||
| 	} | ||||
| 
 | ||||
| 	return strings.Join(ss, "/") | ||||
| 	return path.Join(platform.OS, platform.Architecture, platform.Variant) | ||||
| } | ||||
| 
 | ||||
| // Normalize validates and translate the platform to the canonical value.
 | ||||
|  | @ -269,10 +257,5 @@ func joinNotEmpty(s ...string) string { | |||
| func Normalize(platform specs.Platform) specs.Platform { | ||||
| 	platform.OS = normalizeOS(platform.OS) | ||||
| 	platform.Architecture, platform.Variant = normalizeArch(platform.Architecture, platform.Variant) | ||||
| 
 | ||||
| 	// these fields are deprecated, remove them
 | ||||
| 	platform.OSFeatures = nil | ||||
| 	platform.OSVersion = "" | ||||
| 
 | ||||
| 	return platform | ||||
| } | ||||
|  |  | |||
|  | @ -1,3 +1,4 @@ | |||
| //go:build linux
 | ||||
| // +build linux
 | ||||
| 
 | ||||
| /* | ||||
|  |  | |||
|  | @ -1,3 +1,4 @@ | |||
| //go:build !windows && !darwin
 | ||||
| // +build !windows,!darwin
 | ||||
| 
 | ||||
| /* | ||||
|  | @ -19,14 +20,14 @@ | |||
| package sys | ||||
| 
 | ||||
| import ( | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| ) | ||||
| 
 | ||||
| // GetOpenFds returns the number of open fds for the process provided by pid
 | ||||
| func GetOpenFds(pid int) (int, error) { | ||||
| 	dirs, err := ioutil.ReadDir(filepath.Join("/proc", strconv.Itoa(pid), "fd")) | ||||
| 	dirs, err := os.ReadDir(filepath.Join("/proc", strconv.Itoa(pid), "fd")) | ||||
| 	if err != nil { | ||||
| 		return -1, err | ||||
| 	} | ||||
|  |  | |||
|  | @ -1,3 +1,4 @@ | |||
| //go:build !windows
 | ||||
| // +build !windows
 | ||||
| 
 | ||||
| /* | ||||
|  |  | |||
|  | @ -1,5 +1,3 @@ | |||
| // +build windows
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd Authors. | ||||
| 
 | ||||
|  | @ -19,6 +17,7 @@ | |||
| package sys | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"regexp" | ||||
|  | @ -29,7 +28,6 @@ import ( | |||
| 	"unsafe" | ||||
| 
 | ||||
| 	"github.com/Microsoft/hcsshim" | ||||
| 	"github.com/pkg/errors" | ||||
| 	"golang.org/x/sys/windows" | ||||
| ) | ||||
| 
 | ||||
|  | @ -270,7 +268,7 @@ func ForceRemoveAll(path string) error { | |||
| 	snapshotDir := filepath.Join(path, snapshotPlugin, "snapshots") | ||||
| 	if stat, err := os.Stat(snapshotDir); err == nil && stat.IsDir() { | ||||
| 		if err := cleanupWCOWLayers(snapshotDir); err != nil { | ||||
| 			return errors.Wrapf(err, "failed to cleanup WCOW layers in %s", snapshotDir) | ||||
| 			return fmt.Errorf("failed to cleanup WCOW layers in %s: %w", snapshotDir, err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
|  | @ -280,12 +278,22 @@ func ForceRemoveAll(path string) error { | |||
| func cleanupWCOWLayers(root string) error { | ||||
| 	// See snapshots/windows/windows.go getSnapshotDir()
 | ||||
| 	var layerNums []int | ||||
| 	var rmLayerNums []int | ||||
| 	if err := filepath.Walk(root, func(path string, info os.FileInfo, err error) error { | ||||
| 		if path != root && info.IsDir() { | ||||
| 			if layerNum, err := strconv.Atoi(filepath.Base(path)); err == nil { | ||||
| 				layerNums = append(layerNums, layerNum) | ||||
| 			name := filepath.Base(path) | ||||
| 			if strings.HasPrefix(name, "rm-") { | ||||
| 				layerNum, err := strconv.Atoi(strings.TrimPrefix(name, "rm-")) | ||||
| 				if err != nil { | ||||
| 					return err | ||||
| 				} | ||||
| 				rmLayerNums = append(rmLayerNums, layerNum) | ||||
| 			} else { | ||||
| 				return err | ||||
| 				layerNum, err := strconv.Atoi(name) | ||||
| 				if err != nil { | ||||
| 					return err | ||||
| 				} | ||||
| 				layerNums = append(layerNums, layerNum) | ||||
| 			} | ||||
| 			return filepath.SkipDir | ||||
| 		} | ||||
|  | @ -295,8 +303,14 @@ func cleanupWCOWLayers(root string) error { | |||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	sort.Sort(sort.Reverse(sort.IntSlice(layerNums))) | ||||
| 	sort.Sort(sort.Reverse(sort.IntSlice(rmLayerNums))) | ||||
| 	for _, rmLayerNum := range rmLayerNums { | ||||
| 		if err := cleanupWCOWLayer(filepath.Join(root, "rm-"+strconv.Itoa(rmLayerNum))); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	sort.Sort(sort.Reverse(sort.IntSlice(layerNums))) | ||||
| 	for _, layerNum := range layerNums { | ||||
| 		if err := cleanupWCOWLayer(filepath.Join(root, strconv.Itoa(layerNum))); err != nil { | ||||
| 			return err | ||||
|  | @ -311,19 +325,20 @@ func cleanupWCOWLayer(layerPath string) error { | |||
| 		HomeDir: filepath.Dir(layerPath), | ||||
| 	} | ||||
| 
 | ||||
| 	// ERROR_DEV_NOT_EXIST is returned if the layer is not currently prepared.
 | ||||
| 	// ERROR_DEV_NOT_EXIST is returned if the layer is not currently prepared or activated.
 | ||||
| 	// ERROR_FLT_INSTANCE_NOT_FOUND is returned if the layer is currently activated but not prepared.
 | ||||
| 	if err := hcsshim.UnprepareLayer(info, filepath.Base(layerPath)); err != nil { | ||||
| 		if hcserror, ok := err.(*hcsshim.HcsError); !ok || hcserror.Err != windows.ERROR_DEV_NOT_EXIST { | ||||
| 			return errors.Wrapf(err, "failed to unprepare %s", layerPath) | ||||
| 		if hcserror, ok := err.(*hcsshim.HcsError); !ok || (hcserror.Err != windows.ERROR_DEV_NOT_EXIST && hcserror.Err != syscall.Errno(windows.ERROR_FLT_INSTANCE_NOT_FOUND)) { | ||||
| 			return fmt.Errorf("failed to unprepare %s: %w", layerPath, err) | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if err := hcsshim.DeactivateLayer(info, filepath.Base(layerPath)); err != nil { | ||||
| 		return errors.Wrapf(err, "failed to deactivate %s", layerPath) | ||||
| 		return fmt.Errorf("failed to deactivate %s: %w", layerPath, err) | ||||
| 	} | ||||
| 
 | ||||
| 	if err := hcsshim.DestroyLayer(info, filepath.Base(layerPath)); err != nil { | ||||
| 		return errors.Wrapf(err, "failed to destroy %s", layerPath) | ||||
| 		return fmt.Errorf("failed to destroy %s: %w", layerPath, err) | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
|  |  | |||
|  | @ -1,145 +0,0 @@ | |||
| /* | ||||
|    Copyright The containerd 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 sys | ||||
| 
 | ||||
| import ( | ||||
| 	"runtime" | ||||
| 	"syscall" | ||||
| 	"unsafe" | ||||
| 
 | ||||
| 	"github.com/containerd/containerd/log" | ||||
| 	"github.com/pkg/errors" | ||||
| 	"golang.org/x/sys/unix" | ||||
| ) | ||||
| 
 | ||||
| // FMountat performs mount from the provided directory.
 | ||||
| func FMountat(dirfd uintptr, source, target, fstype string, flags uintptr, data string) error { | ||||
| 	var ( | ||||
| 		sourceP, targetP, fstypeP, dataP *byte | ||||
| 		pid                              uintptr | ||||
| 		err                              error | ||||
| 		errno, status                    syscall.Errno | ||||
| 	) | ||||
| 
 | ||||
| 	sourceP, err = syscall.BytePtrFromString(source) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	targetP, err = syscall.BytePtrFromString(target) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	fstypeP, err = syscall.BytePtrFromString(fstype) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	if data != "" { | ||||
| 		dataP, err = syscall.BytePtrFromString(data) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	runtime.LockOSThread() | ||||
| 	defer runtime.UnlockOSThread() | ||||
| 
 | ||||
| 	var pipefds [2]int | ||||
| 	if err := syscall.Pipe2(pipefds[:], syscall.O_CLOEXEC); err != nil { | ||||
| 		return errors.Wrap(err, "failed to open pipe") | ||||
| 	} | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		// close both ends of the pipe in a deferred function, since open file
 | ||||
| 		// descriptor table is shared with child
 | ||||
| 		syscall.Close(pipefds[0]) | ||||
| 		syscall.Close(pipefds[1]) | ||||
| 	}() | ||||
| 
 | ||||
| 	pid, errno = forkAndMountat(dirfd, | ||||
| 		uintptr(unsafe.Pointer(sourceP)), | ||||
| 		uintptr(unsafe.Pointer(targetP)), | ||||
| 		uintptr(unsafe.Pointer(fstypeP)), | ||||
| 		flags, | ||||
| 		uintptr(unsafe.Pointer(dataP)), | ||||
| 		pipefds[1], | ||||
| 	) | ||||
| 
 | ||||
| 	if errno != 0 { | ||||
| 		return errors.Wrap(errno, "failed to fork thread") | ||||
| 	} | ||||
| 
 | ||||
| 	defer func() { | ||||
| 		_, err := unix.Wait4(int(pid), nil, 0, nil) | ||||
| 		for err == syscall.EINTR { | ||||
| 			_, err = unix.Wait4(int(pid), nil, 0, nil) | ||||
| 		} | ||||
| 
 | ||||
| 		if err != nil { | ||||
| 			log.L.WithError(err).Debugf("failed to find pid=%d process", pid) | ||||
| 		} | ||||
| 	}() | ||||
| 
 | ||||
| 	_, _, errno = syscall.RawSyscall(syscall.SYS_READ, | ||||
| 		uintptr(pipefds[0]), | ||||
| 		uintptr(unsafe.Pointer(&status)), | ||||
| 		unsafe.Sizeof(status)) | ||||
| 	if errno != 0 { | ||||
| 		return errors.Wrap(errno, "failed to read pipe") | ||||
| 	} | ||||
| 
 | ||||
| 	if status != 0 { | ||||
| 		return errors.Wrap(status, "failed to mount") | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // forkAndMountat will fork thread, change working dir and mount.
 | ||||
| //
 | ||||
| // precondition: the runtime OS thread must be locked.
 | ||||
| func forkAndMountat(dirfd uintptr, source, target, fstype, flags, data uintptr, pipefd int) (pid uintptr, errno syscall.Errno) { | ||||
| 
 | ||||
| 	// block signal during clone
 | ||||
| 	beforeFork() | ||||
| 
 | ||||
| 	// the cloned thread shares the open file descriptor, but the thread
 | ||||
| 	// never be reused by runtime.
 | ||||
| 	pid, _, errno = syscall.RawSyscall6(syscall.SYS_CLONE, uintptr(syscall.SIGCHLD)|syscall.CLONE_FILES, 0, 0, 0, 0, 0) | ||||
| 	if errno != 0 || pid != 0 { | ||||
| 		// restore all signals
 | ||||
| 		afterFork() | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	// restore all signals
 | ||||
| 	afterForkInChild() | ||||
| 
 | ||||
| 	// change working dir
 | ||||
| 	_, _, errno = syscall.RawSyscall(syscall.SYS_FCHDIR, dirfd, 0, 0) | ||||
| 	if errno != 0 { | ||||
| 		goto childerr | ||||
| 	} | ||||
| 	_, _, errno = syscall.RawSyscall6(syscall.SYS_MOUNT, source, target, fstype, flags, data, 0) | ||||
| 
 | ||||
| childerr: | ||||
| 	_, _, errno = syscall.RawSyscall(syscall.SYS_WRITE, uintptr(pipefd), uintptr(unsafe.Pointer(&errno)), unsafe.Sizeof(errno)) | ||||
| 	syscall.RawSyscall(syscall.SYS_EXIT, uintptr(errno), 0, 0) | ||||
| 	panic("unreachable") | ||||
| } | ||||
|  | @ -18,7 +18,6 @@ package sys | |||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
|  | @ -69,7 +68,7 @@ func SetOOMScore(pid, score int) error { | |||
| // no oom score is set, or a sore is set to 0.
 | ||||
| func GetOOMScoreAdj(pid int) (int, error) { | ||||
| 	path := fmt.Sprintf("/proc/%d/oom_score_adj", pid) | ||||
| 	data, err := ioutil.ReadFile(path) | ||||
| 	data, err := os.ReadFile(path) | ||||
| 	if err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  |  | |||
|  | @ -1,3 +1,4 @@ | |||
| //go:build !linux
 | ||||
| // +build !linux
 | ||||
| 
 | ||||
| /* | ||||
|  |  | |||
|  | @ -1,3 +1,4 @@ | |||
| //go:build !windows
 | ||||
| // +build !windows
 | ||||
| 
 | ||||
| /* | ||||
|  | @ -19,11 +20,11 @@ | |||
| package sys | ||||
| 
 | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"net" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 
 | ||||
| 	"github.com/pkg/errors" | ||||
| 	"golang.org/x/sys/unix" | ||||
| ) | ||||
| 
 | ||||
|  | @ -31,7 +32,7 @@ import ( | |||
| func CreateUnixSocket(path string) (net.Listener, error) { | ||||
| 	// BSDs have a 104 limit
 | ||||
| 	if len(path) > 104 { | ||||
| 		return nil, errors.Errorf("%q: unix socket path too long (> 104)", path) | ||||
| 		return nil, fmt.Errorf("%q: unix socket path too long (> 104)", path) | ||||
| 	} | ||||
| 	if err := os.MkdirAll(filepath.Dir(path), 0660); err != nil { | ||||
| 		return nil, err | ||||
|  |  | |||
|  | @ -1,5 +1,3 @@ | |||
| // +build windows
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd Authors. | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,44 +0,0 @@ | |||
| // +build darwin freebsd netbsd
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd 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 sys | ||||
| 
 | ||||
| import ( | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // StatAtime returns the access time from a stat struct
 | ||||
| func StatAtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Atimespec | ||||
| } | ||||
| 
 | ||||
| // StatCtime returns the created time from a stat struct
 | ||||
| func StatCtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Ctimespec | ||||
| } | ||||
| 
 | ||||
| // StatMtime returns the modified time from a stat struct
 | ||||
| func StatMtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Mtimespec | ||||
| } | ||||
| 
 | ||||
| // StatATimeAsTime returns the access time as a time.Time
 | ||||
| func StatATimeAsTime(st *syscall.Stat_t) time.Time { | ||||
| 	return time.Unix(int64(st.Atimespec.Sec), int64(st.Atimespec.Nsec)) // nolint: unconvert
 | ||||
| } | ||||
|  | @ -1,45 +0,0 @@ | |||
| // +build openbsd
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd 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 sys | ||||
| 
 | ||||
| import ( | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // StatAtime returns the Atim
 | ||||
| func StatAtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Atim | ||||
| } | ||||
| 
 | ||||
| // StatCtime returns the Ctim
 | ||||
| func StatCtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Ctim | ||||
| } | ||||
| 
 | ||||
| // StatMtime returns the Mtim
 | ||||
| func StatMtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Mtim | ||||
| } | ||||
| 
 | ||||
| // StatATimeAsTime returns st.Atim as a time.Time
 | ||||
| func StatATimeAsTime(st *syscall.Stat_t) time.Time { | ||||
| 	// The int64 conversions ensure the line compiles for 32-bit systems as well.
 | ||||
| 	return time.Unix(int64(st.Atim.Sec), int64(st.Atim.Nsec)) // nolint: unconvert
 | ||||
| } | ||||
|  | @ -1,44 +0,0 @@ | |||
| // +build linux solaris
 | ||||
| 
 | ||||
| /* | ||||
|    Copyright The containerd 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 sys | ||||
| 
 | ||||
| import ( | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| ) | ||||
| 
 | ||||
| // StatAtime returns the Atim
 | ||||
| func StatAtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Atim | ||||
| } | ||||
| 
 | ||||
| // StatCtime returns the Ctim
 | ||||
| func StatCtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Ctim | ||||
| } | ||||
| 
 | ||||
| // StatMtime returns the Mtim
 | ||||
| func StatMtime(st *syscall.Stat_t) syscall.Timespec { | ||||
| 	return st.Mtim | ||||
| } | ||||
| 
 | ||||
| // StatATimeAsTime returns st.Atim as a time.Time
 | ||||
| func StatATimeAsTime(st *syscall.Stat_t) time.Time { | ||||
| 	return time.Unix(int64(st.Atim.Sec), int64(st.Atim.Nsec)) // nolint: unconvert
 | ||||
| } | ||||
|  | @ -1,30 +0,0 @@ | |||
| /* | ||||
|    Copyright The containerd 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 sys | ||||
| 
 | ||||
| import ( | ||||
| 	_ "unsafe" // required for go:linkname.
 | ||||
| ) | ||||
| 
 | ||||
| //go:linkname beforeFork syscall.runtime_BeforeFork
 | ||||
| func beforeFork() | ||||
| 
 | ||||
| //go:linkname afterFork syscall.runtime_AfterFork
 | ||||
| func afterFork() | ||||
| 
 | ||||
| //go:linkname afterForkInChild syscall.runtime_AfterForkInChild
 | ||||
| func afterForkInChild() | ||||
|  | @ -1,15 +0,0 @@ | |||
| /* | ||||
|    Copyright The containerd 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. | ||||
| */ | ||||
|  | @ -69,7 +69,7 @@ func (n *cniNetwork) networkCreate(newNetwork *types.Network, defaultNet bool) ( | |||
| 
 | ||||
| 	switch newNetwork.Driver { | ||||
| 	case types.BridgeNetworkDriver: | ||||
| 		err = internalutil.CreateBridge(n, newNetwork, usedNetworks) | ||||
| 		err = internalutil.CreateBridge(n, newNetwork, usedNetworks, n.defaultsubnetPools) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  |  | |||
|  | @ -13,6 +13,7 @@ import ( | |||
| 
 | ||||
| 	"github.com/containernetworking/cni/libcni" | ||||
| 	"github.com/containers/common/libnetwork/types" | ||||
| 	"github.com/containers/common/pkg/config" | ||||
| 	"github.com/containers/storage/pkg/lockfile" | ||||
| 	"github.com/pkg/errors" | ||||
| 	"github.com/sirupsen/logrus" | ||||
|  | @ -31,6 +32,9 @@ type cniNetwork struct { | |||
| 	// defaultSubnet is the default subnet for the default network.
 | ||||
| 	defaultSubnet types.IPNet | ||||
| 
 | ||||
| 	// defaultsubnetPools contains the subnets which must be used to allocate a free subnet by network create
 | ||||
| 	defaultsubnetPools []config.SubnetPool | ||||
| 
 | ||||
| 	// isMachine describes whenever podman runs in a podman machine environment.
 | ||||
| 	isMachine bool | ||||
| 
 | ||||
|  | @ -62,6 +66,9 @@ type InitConfig struct { | |||
| 	// DefaultSubnet is the default subnet for the default network.
 | ||||
| 	DefaultSubnet string | ||||
| 
 | ||||
| 	// DefaultsubnetPools contains the subnets which must be used to allocate a free subnet by network create
 | ||||
| 	DefaultsubnetPools []config.SubnetPool | ||||
| 
 | ||||
| 	// IsMachine describes whenever podman runs in a podman machine environment.
 | ||||
| 	IsMachine bool | ||||
| } | ||||
|  | @ -89,15 +96,21 @@ func NewCNINetworkInterface(conf *InitConfig) (types.ContainerNetwork, error) { | |||
| 		return nil, errors.Wrap(err, "failed to parse default subnet") | ||||
| 	} | ||||
| 
 | ||||
| 	defaultSubnetPools := conf.DefaultsubnetPools | ||||
| 	if defaultSubnetPools == nil { | ||||
| 		defaultSubnetPools = config.DefaultSubnetPools | ||||
| 	} | ||||
| 
 | ||||
| 	cni := libcni.NewCNIConfig(conf.CNIPluginDirs, &cniExec{}) | ||||
| 	n := &cniNetwork{ | ||||
| 		cniConfigDir:   conf.CNIConfigDir, | ||||
| 		cniPluginDirs:  conf.CNIPluginDirs, | ||||
| 		cniConf:        cni, | ||||
| 		defaultNetwork: defaultNetworkName, | ||||
| 		defaultSubnet:  defaultNet, | ||||
| 		isMachine:      conf.IsMachine, | ||||
| 		lock:           lock, | ||||
| 		cniConfigDir:       conf.CNIConfigDir, | ||||
| 		cniPluginDirs:      conf.CNIPluginDirs, | ||||
| 		cniConf:            cni, | ||||
| 		defaultNetwork:     defaultNetworkName, | ||||
| 		defaultSubnet:      defaultNet, | ||||
| 		defaultsubnetPools: defaultSubnetPools, | ||||
| 		isMachine:          conf.IsMachine, | ||||
| 		lock:               lock, | ||||
| 	} | ||||
| 
 | ||||
| 	return n, nil | ||||
|  |  | |||
|  | @ -5,11 +5,12 @@ import ( | |||
| 
 | ||||
| 	"github.com/containers/common/libnetwork/types" | ||||
| 	"github.com/containers/common/libnetwork/util" | ||||
| 	"github.com/containers/common/pkg/config" | ||||
| 	pkgutil "github.com/containers/common/pkg/util" | ||||
| 	"github.com/pkg/errors" | ||||
| ) | ||||
| 
 | ||||
| func CreateBridge(n NetUtil, network *types.Network, usedNetworks []*net.IPNet) error { | ||||
| func CreateBridge(n NetUtil, network *types.Network, usedNetworks []*net.IPNet, subnetPools []config.SubnetPool) error { | ||||
| 	if network.NetworkInterface != "" { | ||||
| 		bridges := GetBridgeInterfaceNames(n) | ||||
| 		if pkgutil.StringInSlice(network.NetworkInterface, bridges) { | ||||
|  | @ -28,7 +29,7 @@ func CreateBridge(n NetUtil, network *types.Network, usedNetworks []*net.IPNet) | |||
| 
 | ||||
| 	if network.IPAMOptions["driver"] != types.DHCPIPAMDriver { | ||||
| 		if len(network.Subnets) == 0 { | ||||
| 			freeSubnet, err := GetFreeIPv4NetworkSubnet(usedNetworks) | ||||
| 			freeSubnet, err := GetFreeIPv4NetworkSubnet(usedNetworks, subnetPools) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
|  | @ -48,7 +49,7 @@ func CreateBridge(n NetUtil, network *types.Network, usedNetworks []*net.IPNet) | |||
| 				} | ||||
| 			} | ||||
| 			if !ipv4 { | ||||
| 				freeSubnet, err := GetFreeIPv4NetworkSubnet(usedNetworks) | ||||
| 				freeSubnet, err := GetFreeIPv4NetworkSubnet(usedNetworks, subnetPools) | ||||
| 				if err != nil { | ||||
| 					return err | ||||
| 				} | ||||
|  |  | |||
|  | @ -11,11 +11,15 @@ func incByte(subnet *net.IPNet, idx int, shift uint) error { | |||
| 	if idx < 0 { | ||||
| 		return errors.New("no more subnets left") | ||||
| 	} | ||||
| 	if subnet.IP[idx] == 255 { | ||||
| 		subnet.IP[idx] = 0 | ||||
| 		return incByte(subnet, idx-1, 0) | ||||
| 
 | ||||
| 	var val byte = 1 << shift | ||||
| 	// if overflow we have to inc the previous byte
 | ||||
| 	if uint(subnet.IP[idx])+uint(val) > 255 { | ||||
| 		if err := incByte(subnet, idx-1, 0); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
| 	subnet.IP[idx] += 1 << shift | ||||
| 	subnet.IP[idx] += val | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
|  | @ -31,10 +35,7 @@ func NextSubnet(subnet *net.IPNet) (*net.IPNet, error) { | |||
| 	} | ||||
| 	zeroes := uint(bits - ones) | ||||
| 	shift := zeroes % 8 | ||||
| 	idx := ones/8 - 1 | ||||
| 	if idx < 0 { | ||||
| 		idx = 0 | ||||
| 	} | ||||
| 	idx := (ones - 1) / 8 | ||||
| 	if err := incByte(newSubnet, idx, shift); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  |  | |||
|  | @ -6,6 +6,7 @@ import ( | |||
| 	"net" | ||||
| 
 | ||||
| 	"github.com/containers/common/libnetwork/types" | ||||
| 	"github.com/containers/common/pkg/config" | ||||
| 	"github.com/containers/common/pkg/util" | ||||
| 	"github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | @ -79,28 +80,36 @@ func GetUsedSubnets(n NetUtil) ([]*net.IPNet, error) { | |||
| } | ||||
| 
 | ||||
| // GetFreeIPv4NetworkSubnet returns a unused ipv4 subnet
 | ||||
| func GetFreeIPv4NetworkSubnet(usedNetworks []*net.IPNet) (*types.Subnet, error) { | ||||
| 	// the default podman network is 10.88.0.0/16
 | ||||
| 	// start locking for free /24 networks
 | ||||
| 	network := &net.IPNet{ | ||||
| 		IP:   net.IP{10, 89, 0, 0}, | ||||
| 		Mask: net.IPMask{255, 255, 255, 0}, | ||||
| func GetFreeIPv4NetworkSubnet(usedNetworks []*net.IPNet, subnetPools []config.SubnetPool) (*types.Subnet, error) { | ||||
| 	var err error | ||||
| 	for _, pool := range subnetPools { | ||||
| 		// make sure to copy the netip to prevent overwriting the subnet pool
 | ||||
| 		netIP := make(net.IP, net.IPv4len) | ||||
| 		copy(netIP, pool.Base.IP) | ||||
| 		network := &net.IPNet{ | ||||
| 			IP:   netIP, | ||||
| 			Mask: net.CIDRMask(pool.Size, 32), | ||||
| 		} | ||||
| 		for pool.Base.Contains(network.IP) { | ||||
| 			if !NetworkIntersectsWithNetworks(network, usedNetworks) { | ||||
| 				logrus.Debugf("found free ipv4 network subnet %s", network.String()) | ||||
| 				return &types.Subnet{ | ||||
| 					Subnet: types.IPNet{IPNet: *network}, | ||||
| 				}, nil | ||||
| 			} | ||||
| 			network, err = NextSubnet(network) | ||||
| 			if err != nil { | ||||
| 				// when error go to next pool, we return the error only when all pools are done
 | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// TODO: make sure to not use public subnets
 | ||||
| 	for { | ||||
| 		if intersectsConfig := NetworkIntersectsWithNetworks(network, usedNetworks); !intersectsConfig { | ||||
| 			logrus.Debugf("found free ipv4 network subnet %s", network.String()) | ||||
| 			return &types.Subnet{ | ||||
| 				Subnet: types.IPNet{IPNet: *network}, | ||||
| 			}, nil | ||||
| 		} | ||||
| 		var err error | ||||
| 		network, err = NextSubnet(network) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return nil, errors.New("could not find free subnet from subnet pools") | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| // GetFreeIPv6NetworkSubnet returns a unused ipv6 subnet
 | ||||
|  |  | |||
|  | @ -83,7 +83,7 @@ func (n *netavarkNetwork) networkCreate(newNetwork *types.Network, defaultNet bo | |||
| 
 | ||||
| 	switch newNetwork.Driver { | ||||
| 	case types.BridgeNetworkDriver: | ||||
| 		err = internalutil.CreateBridge(n, newNetwork, usedNetworks) | ||||
| 		err = internalutil.CreateBridge(n, newNetwork, usedNetworks, n.defaultsubnetPools) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  |  | |||
|  | @ -12,6 +12,7 @@ import ( | |||
| 
 | ||||
| 	"github.com/containers/common/libnetwork/internal/util" | ||||
| 	"github.com/containers/common/libnetwork/types" | ||||
| 	"github.com/containers/common/pkg/config" | ||||
| 	"github.com/containers/storage/pkg/lockfile" | ||||
| 	"github.com/containers/storage/pkg/unshare" | ||||
| 	"github.com/pkg/errors" | ||||
|  | @ -38,6 +39,9 @@ type netavarkNetwork struct { | |||
| 	// defaultSubnet is the default subnet for the default network.
 | ||||
| 	defaultSubnet types.IPNet | ||||
| 
 | ||||
| 	// defaultsubnetPools contains the subnets which must be used to allocate a free subnet by network create
 | ||||
| 	defaultsubnetPools []config.SubnetPool | ||||
| 
 | ||||
| 	// ipamDBPath is the path to the ip allocation bolt db
 | ||||
| 	ipamDBPath string | ||||
| 
 | ||||
|  | @ -72,6 +76,9 @@ type InitConfig struct { | |||
| 	// DefaultSubnet is the default subnet for the default network.
 | ||||
| 	DefaultSubnet string | ||||
| 
 | ||||
| 	// DefaultsubnetPools contains the subnets which must be used to allocate a free subnet by network create
 | ||||
| 	DefaultsubnetPools []config.SubnetPool | ||||
| 
 | ||||
| 	// Syslog describes whenever the netavark debbug output should be log to the syslog as well.
 | ||||
| 	// This will use logrus to do so, make sure logrus is set up to log to the syslog.
 | ||||
| 	Syslog bool | ||||
|  | @ -108,17 +115,23 @@ func NewNetworkInterface(conf *InitConfig) (types.ContainerNetwork, error) { | |||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	defaultSubnetPools := conf.DefaultsubnetPools | ||||
| 	if defaultSubnetPools == nil { | ||||
| 		defaultSubnetPools = config.DefaultSubnetPools | ||||
| 	} | ||||
| 
 | ||||
| 	n := &netavarkNetwork{ | ||||
| 		networkConfigDir: conf.NetworkConfigDir, | ||||
| 		networkRunDir:    conf.NetworkRunDir, | ||||
| 		netavarkBinary:   conf.NetavarkBinary, | ||||
| 		aardvarkBinary:   conf.AardvarkBinary, | ||||
| 		networkRootless:  unshare.IsRootless(), | ||||
| 		ipamDBPath:       filepath.Join(conf.NetworkRunDir, "ipam.db"), | ||||
| 		defaultNetwork:   defaultNetworkName, | ||||
| 		defaultSubnet:    defaultNet, | ||||
| 		lock:             lock, | ||||
| 		syslog:           conf.Syslog, | ||||
| 		networkConfigDir:   conf.NetworkConfigDir, | ||||
| 		networkRunDir:      conf.NetworkRunDir, | ||||
| 		netavarkBinary:     conf.NetavarkBinary, | ||||
| 		aardvarkBinary:     conf.AardvarkBinary, | ||||
| 		networkRootless:    unshare.IsRootless(), | ||||
| 		ipamDBPath:         filepath.Join(conf.NetworkRunDir, "ipam.db"), | ||||
| 		defaultNetwork:     defaultNetworkName, | ||||
| 		defaultSubnet:      defaultNet, | ||||
| 		defaultsubnetPools: defaultSubnetPools, | ||||
| 		lock:               lock, | ||||
| 		syslog:             conf.Syslog, | ||||
| 	} | ||||
| 
 | ||||
| 	return n, nil | ||||
|  |  | |||
|  | @ -82,13 +82,14 @@ func NetworkBackend(store storage.Store, conf *config.Config, syslog bool) (type | |||
| 		} | ||||
| 
 | ||||
| 		netInt, err := netavark.NewNetworkInterface(&netavark.InitConfig{ | ||||
| 			NetworkConfigDir: confDir, | ||||
| 			NetworkRunDir:    runDir, | ||||
| 			NetavarkBinary:   netavarkBin, | ||||
| 			AardvarkBinary:   aardvarkBin, | ||||
| 			DefaultNetwork:   conf.Network.DefaultNetwork, | ||||
| 			DefaultSubnet:    conf.Network.DefaultSubnet, | ||||
| 			Syslog:           syslog, | ||||
| 			NetworkConfigDir:   confDir, | ||||
| 			NetworkRunDir:      runDir, | ||||
| 			NetavarkBinary:     netavarkBin, | ||||
| 			AardvarkBinary:     aardvarkBin, | ||||
| 			DefaultNetwork:     conf.Network.DefaultNetwork, | ||||
| 			DefaultSubnet:      conf.Network.DefaultSubnet, | ||||
| 			DefaultsubnetPools: conf.Network.DefaultSubnetPools, | ||||
| 			Syslog:             syslog, | ||||
| 		}) | ||||
| 		return types.Netavark, netInt, err | ||||
| 	case types.CNI: | ||||
|  | @ -171,11 +172,12 @@ func getCniInterface(conf *config.Config) (types.ContainerNetwork, error) { | |||
| 		} | ||||
| 	} | ||||
| 	return cni.NewCNINetworkInterface(&cni.InitConfig{ | ||||
| 		CNIConfigDir:   confDir, | ||||
| 		CNIPluginDirs:  conf.Network.CNIPluginDirs, | ||||
| 		DefaultNetwork: conf.Network.DefaultNetwork, | ||||
| 		DefaultSubnet:  conf.Network.DefaultSubnet, | ||||
| 		IsMachine:      conf.Engine.MachineEnabled, | ||||
| 		CNIConfigDir:       confDir, | ||||
| 		CNIPluginDirs:      conf.Network.CNIPluginDirs, | ||||
| 		DefaultNetwork:     conf.Network.DefaultNetwork, | ||||
| 		DefaultSubnet:      conf.Network.DefaultSubnet, | ||||
| 		DefaultsubnetPools: conf.Network.DefaultSubnetPools, | ||||
| 		IsMachine:          conf.Engine.MachineEnabled, | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -10,6 +10,7 @@ import ( | |||
| 	"sync" | ||||
| 
 | ||||
| 	"github.com/BurntSushi/toml" | ||||
| 	"github.com/containers/common/libnetwork/types" | ||||
| 	"github.com/containers/common/pkg/capabilities" | ||||
| 	"github.com/containers/storage/pkg/unshare" | ||||
| 	units "github.com/docker/go-units" | ||||
|  | @ -486,20 +487,36 @@ type NetworkConfig struct { | |||
| 	// CNIPluginDirs is where CNI plugin binaries are stored.
 | ||||
| 	CNIPluginDirs []string `toml:"cni_plugin_dirs,omitempty"` | ||||
| 
 | ||||
| 	// DefaultNetwork is the network name of the default CNI network
 | ||||
| 	// DefaultNetwork is the network name of the default network
 | ||||
| 	// to attach pods to.
 | ||||
| 	DefaultNetwork string `toml:"default_network,omitempty"` | ||||
| 
 | ||||
| 	// DefaultSubnet is the subnet to be used for the default CNI network.
 | ||||
| 	// DefaultSubnet is the subnet to be used for the default network.
 | ||||
| 	// If a network with the name given in DefaultNetwork is not present
 | ||||
| 	// then a new network using this subnet will be created.
 | ||||
| 	// Must be a valid IPv4 CIDR block.
 | ||||
| 	DefaultSubnet string `toml:"default_subnet,omitempty"` | ||||
| 
 | ||||
| 	// NetworkConfigDir is where CNI network configuration files are stored.
 | ||||
| 	// DefaultSubnetPools is a list of subnets and size which are used to
 | ||||
| 	// allocate subnets automatically for podman network create.
 | ||||
| 	// It will iterate through the list and will pick the first free subnet
 | ||||
| 	// with the given size. This is only used for ipv4 subnets, ipv6 subnets
 | ||||
| 	// are always assigned randomly.
 | ||||
| 	DefaultSubnetPools []SubnetPool `toml:"default_subnet_pools,omitempty"` | ||||
| 
 | ||||
| 	// NetworkConfigDir is where network configuration files are stored.
 | ||||
| 	NetworkConfigDir string `toml:"network_config_dir,omitempty"` | ||||
| } | ||||
| 
 | ||||
| type SubnetPool struct { | ||||
| 	// Base is a bigger subnet which will be used to allocate a subnet with
 | ||||
| 	// the given size.
 | ||||
| 	Base *types.IPNet `toml:"base,omitempty"` | ||||
| 	// Size is the CIDR for the new subnet. It must be equal or small
 | ||||
| 	// than the CIDR from the base subnet.
 | ||||
| 	Size int `toml:"size,omitempty"` | ||||
| } | ||||
| 
 | ||||
| // SecretConfig represents the "secret" TOML config table
 | ||||
| type SecretConfig struct { | ||||
| 	// Driver specifies the secret driver to use.
 | ||||
|  | @ -830,6 +847,21 @@ func (c *ContainersConfig) Validate() error { | |||
| // execution checks. It returns an `error` on validation failure, otherwise
 | ||||
| // `nil`.
 | ||||
| func (c *NetworkConfig) Validate() error { | ||||
| 	if &c.DefaultSubnetPools != &DefaultSubnetPools { | ||||
| 		for _, pool := range c.DefaultSubnetPools { | ||||
| 			if pool.Base.IP.To4() == nil { | ||||
| 				return errors.Errorf("invalid subnet pool ip %q", pool.Base.IP) | ||||
| 			} | ||||
| 			ones, _ := pool.Base.IPNet.Mask.Size() | ||||
| 			if ones > pool.Size { | ||||
| 				return errors.Errorf("invalid subnet pool, size is bigger than subnet %q", &pool.Base.IPNet) | ||||
| 			} | ||||
| 			if pool.Size > 32 { | ||||
| 				return errors.New("invalid subnet pool size, must be between 0-32") | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if stringsEq(c.CNIPluginDirs, DefaultCNIPluginDirs) { | ||||
| 		return nil | ||||
| 	} | ||||
|  |  | |||
|  | @ -284,6 +284,20 @@ default_sysctls = [ | |||
| # | ||||
| #default_subnet = "10.88.0.0/16" | ||||
| 
 | ||||
| # DefaultSubnetPools is a list of subnets and size which are used to | ||||
| # allocate subnets automatically for podman network create. | ||||
| # It will iterate through the list and will pick the first free subnet | ||||
| # with the given size. This is only used for ipv4 subnets, ipv6 subnets | ||||
| # are always assigned randomly. | ||||
| # | ||||
| #default_subnet_pools = [ | ||||
| #  {"base" = "10.89.0.0/16", "size" = 24}, | ||||
| #  {"base" = "10.90.0.0/15", "size" = 24}, | ||||
| #  {"base" = "10.92.0.0/14", "size" = 24}, | ||||
| #  {"base" = "10.96.0.0/11", "size" = 24}, | ||||
| #  {"base" = "10.128.0.0/9", "size" = 24}, | ||||
| #] | ||||
| 
 | ||||
| # Path to the directory where network configuration files are located. | ||||
| # For the CNI backend the default is "/etc/cni/net.d" as root | ||||
| # and "$HOME/.config/cni/net.d" as rootless. | ||||
|  |  | |||
|  | @ -3,12 +3,14 @@ package config | |||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"net" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"path/filepath" | ||||
| 	"regexp" | ||||
| 	"strconv" | ||||
| 
 | ||||
| 	nettypes "github.com/containers/common/libnetwork/types" | ||||
| 	"github.com/containers/common/pkg/apparmor" | ||||
| 	"github.com/containers/common/pkg/cgroupv2" | ||||
| 	"github.com/containers/common/pkg/util" | ||||
|  | @ -85,8 +87,26 @@ var ( | |||
| 		"/usr/lib/cni", | ||||
| 		"/opt/cni/bin", | ||||
| 	} | ||||
| 	DefaultSubnetPools = []SubnetPool{ | ||||
| 		// 10.89.0.0/24-10.255.255.0/24
 | ||||
| 		parseSubnetPool("10.89.0.0/16", 24), | ||||
| 		parseSubnetPool("10.90.0.0/15", 24), | ||||
| 		parseSubnetPool("10.92.0.0/14", 24), | ||||
| 		parseSubnetPool("10.96.0.0/11", 24), | ||||
| 		parseSubnetPool("10.128.0.0/9", 24), | ||||
| 	} | ||||
| ) | ||||
| 
 | ||||
| // nolint:unparam
 | ||||
| func parseSubnetPool(subnet string, size int) SubnetPool { | ||||
| 	_, n, _ := net.ParseCIDR(subnet) | ||||
| 	return SubnetPool{ | ||||
| 		Base: &nettypes.IPNet{IPNet: *n}, | ||||
| 		Size: size, | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| const ( | ||||
| 	// _etcDir is the sysconfdir where podman should look for system config files.
 | ||||
| 	// It can be overridden at build time.
 | ||||
|  | @ -111,7 +131,7 @@ const ( | |||
| 	// DefaultSignaturePolicyPath is the default value for the
 | ||||
| 	// policy.json file.
 | ||||
| 	DefaultSignaturePolicyPath = "/etc/containers/policy.json" | ||||
| 	// DefaultSubnet is the subnet that will be used for the default CNI
 | ||||
| 	// DefaultSubnet is the subnet that will be used for the default
 | ||||
| 	// network.
 | ||||
| 	DefaultSubnet = "10.88.0.0/16" | ||||
| 	// DefaultRootlessSignaturePolicyPath is the location within
 | ||||
|  | @ -195,9 +215,10 @@ func DefaultConfig() (*Config, error) { | |||
| 			UserNSSize: DefaultUserNSSize, | ||||
| 		}, | ||||
| 		Network: NetworkConfig{ | ||||
| 			DefaultNetwork: "podman", | ||||
| 			DefaultSubnet:  DefaultSubnet, | ||||
| 			CNIPluginDirs:  DefaultCNIPluginDirs, | ||||
| 			DefaultNetwork:     "podman", | ||||
| 			DefaultSubnet:      DefaultSubnet, | ||||
| 			DefaultSubnetPools: DefaultSubnetPools, | ||||
| 			CNIPluginDirs:      DefaultCNIPluginDirs, | ||||
| 		}, | ||||
| 		Engine:  *defaultEngineConfig, | ||||
| 		Secrets: defaultSecretConfig(), | ||||
|  | @ -385,15 +406,14 @@ func probeConmon(conmonBinary string) error { | |||
| 	cmd := exec.Command(conmonBinary, "--version") | ||||
| 	var out bytes.Buffer | ||||
| 	cmd.Stdout = &out | ||||
| 	err := cmd.Run() | ||||
| 	if err != nil { | ||||
| 	if err := cmd.Run(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	r := regexp.MustCompile(`^conmon version (?P<Major>\d+).(?P<Minor>\d+).(?P<Patch>\d+)`) | ||||
| 
 | ||||
| 	matches := r.FindStringSubmatch(out.String()) | ||||
| 	if len(matches) != 4 { | ||||
| 		return errors.Wrap(err, _conmonVersionFormatErr) | ||||
| 		return errors.New(_conmonVersionFormatErr) | ||||
| 	} | ||||
| 	major, err := strconv.Atoi(matches[1]) | ||||
| 	if err != nil { | ||||
|  |  | |||
|  | @ -34,6 +34,10 @@ func ValidateVolumeOpts(options []string) ([]string, error) { | |||
| 			finalOpts = append(finalOpts, opt) | ||||
| 			continue | ||||
| 		} | ||||
| 		if strings.HasPrefix(opt, "idmap") { | ||||
| 			finalOpts = append(finalOpts, opt) | ||||
| 			continue | ||||
| 		} | ||||
| 
 | ||||
| 		switch opt { | ||||
| 		case "noexec", "exec": | ||||
|  | @ -84,7 +88,6 @@ func ValidateVolumeOpts(options []string) ([]string, error) { | |||
| 			// are intended to be always safe to use, even not on OS
 | ||||
| 			// X).
 | ||||
| 			continue | ||||
| 		case "idmap": | ||||
| 		default: | ||||
| 			return nil, errors.Errorf("invalid option type %q", opt) | ||||
| 		} | ||||
|  |  | |||
|  | @ -1,4 +1,4 @@ | |||
| package version | ||||
| 
 | ||||
| // Version is the version of the build.
 | ||||
| const Version = "0.47.4" | ||||
| const Version = "0.47.4+dev" | ||||
|  |  | |||
|  | @ -0,0 +1,29 @@ | |||
| run: | ||||
|   timeout: 1m | ||||
|   tests: true | ||||
| 
 | ||||
| linters: | ||||
|   disable-all: true | ||||
|   enable: | ||||
|     - asciicheck | ||||
|     - deadcode | ||||
|     - errcheck | ||||
|     - forcetypeassert | ||||
|     - gocritic | ||||
|     - gofmt | ||||
|     - goimports | ||||
|     - gosimple | ||||
|     - govet | ||||
|     - ineffassign | ||||
|     - misspell | ||||
|     - revive | ||||
|     - staticcheck | ||||
|     - structcheck | ||||
|     - typecheck | ||||
|     - unused | ||||
|     - varcheck | ||||
| 
 | ||||
| issues: | ||||
|   exclude-use-default: false | ||||
|   max-issues-per-linter: 0 | ||||
|   max-same-issues: 10 | ||||
|  | @ -0,0 +1,6 @@ | |||
| # CHANGELOG | ||||
| 
 | ||||
| ## v1.0.0-rc1 | ||||
| 
 | ||||
| This is the first logged release.  Major changes (including breaking changes) | ||||
| have occurred since earlier tags. | ||||
|  | @ -0,0 +1,17 @@ | |||
| # Contributing | ||||
| 
 | ||||
| Logr is open to pull-requests, provided they fit within the intended scope of | ||||
| the project.  Specifically, this library aims to be VERY small and minimalist, | ||||
| with no external dependencies. | ||||
| 
 | ||||
| ## Compatibility | ||||
| 
 | ||||
| This project intends to follow [semantic versioning](http://semver.org) and | ||||
| is very strict about compatibility.  Any proposed changes MUST follow those | ||||
| rules. | ||||
| 
 | ||||
| ## Performance | ||||
| 
 | ||||
| As a logging library, logr must be as light-weight as possible.  Any proposed | ||||
| code change must include results of running the [benchmark](./benchmark) | ||||
| before and after the change. | ||||
|  | @ -1,112 +1,182 @@ | |||
| # A more minimal logging API for Go | ||||
| # A minimal logging API for Go | ||||
| 
 | ||||
| [](https://pkg.go.dev/github.com/go-logr/logr) | ||||
| 
 | ||||
| logr offers an(other) opinion on how Go programs and libraries can do logging | ||||
| without becoming coupled to a particular logging implementation.  This is not | ||||
| an implementation of logging - it is an API.  In fact it is two APIs with two | ||||
| different sets of users. | ||||
| 
 | ||||
| The `Logger` type is intended for application and library authors.  It provides | ||||
| a relatively small API which can be used everywhere you want to emit logs.  It | ||||
| defers the actual act of writing logs (to files, to stdout, or whatever) to the | ||||
| `LogSink` interface. | ||||
| 
 | ||||
| The `LogSink` interface is intended for logging library implementers.  It is a | ||||
| pure interface which can be implemented by logging frameworks to provide the actual logging | ||||
| functionality. | ||||
| 
 | ||||
| This decoupling allows application and library developers to write code in | ||||
| terms of `logr.Logger` (which has very low dependency fan-out) while the | ||||
| implementation of logging is managed "up stack" (e.g. in or near `main()`.) | ||||
| Application developers can then switch out implementations as necessary. | ||||
| 
 | ||||
| Many people assert that libraries should not be logging, and as such efforts | ||||
| like this are pointless.  Those people are welcome to convince the authors of | ||||
| the tens-of-thousands of libraries that *DO* write logs that they are all | ||||
| wrong.  In the meantime, logr takes a more practical approach. | ||||
| 
 | ||||
| ## Typical usage | ||||
| 
 | ||||
| Somewhere, early in an application's life, it will make a decision about which | ||||
| logging library (implementation) it actually wants to use.  Something like: | ||||
| 
 | ||||
| ``` | ||||
|     func main() { | ||||
|         // ... other setup code ... | ||||
| 
 | ||||
|         // Create the "root" logger.  We have chosen the "logimpl" implementation, | ||||
|         // which takes some initial parameters and returns a logr.Logger. | ||||
|         logger := logimpl.New(param1, param2) | ||||
| 
 | ||||
|         // ... other setup code ... | ||||
| ``` | ||||
| 
 | ||||
| Most apps will call into other libraries, create structures to govern the flow, | ||||
| etc.  The `logr.Logger` object can be passed to these other libraries, stored | ||||
| in structs, or even used as a package-global variable, if needed.  For example: | ||||
| 
 | ||||
| ``` | ||||
|     app := createTheAppObject(logger) | ||||
|     app.Run() | ||||
| ``` | ||||
| 
 | ||||
| Outside of this early setup, no other packages need to know about the choice of | ||||
| implementation.  They write logs in terms of the `logr.Logger` that they | ||||
| received: | ||||
| 
 | ||||
| ``` | ||||
|     type appObject struct { | ||||
|         // ... other fields ... | ||||
|         logger logr.Logger | ||||
|         // ... other fields ... | ||||
|     } | ||||
| 
 | ||||
|     func (app *appObject) Run() { | ||||
|         app.logger.Info("starting up", "timestamp", time.Now()) | ||||
| 
 | ||||
|         // ... app code ... | ||||
| ``` | ||||
| 
 | ||||
| ## Background | ||||
| 
 | ||||
| If the Go standard library had defined an interface for logging, this project | ||||
| probably would not be needed.  Alas, here we are. | ||||
| 
 | ||||
| ### Inspiration | ||||
| 
 | ||||
| Before you consider this package, please read [this blog post by the | ||||
| inimitable Dave Cheney][warning-makes-no-sense].  I really appreciate what | ||||
| he has to say, and it largely aligns with my own experiences.  Too many | ||||
| choices of levels means inconsistent logs. | ||||
| inimitable Dave Cheney][warning-makes-no-sense].  We really appreciate what | ||||
| he has to say, and it largely aligns with our own experiences. | ||||
| 
 | ||||
| This package offers a purely abstract interface, based on these ideas but with | ||||
| a few twists.  Code can depend on just this interface and have the actual | ||||
| logging implementation be injected from callers.  Ideally only `main()` knows | ||||
| what logging implementation is being used. | ||||
| 
 | ||||
| # Differences from Dave's ideas | ||||
| ### Differences from Dave's ideas | ||||
| 
 | ||||
| The main differences are: | ||||
| 
 | ||||
| 1) Dave basically proposes doing away with the notion of a logging API in favor | ||||
| of `fmt.Printf()`.  I disagree, especially when you consider things like output | ||||
| locations, timestamps, file and line decorations, and structured logging.  I | ||||
| restrict the API to just 2 types of logs: info and error. | ||||
| 1. Dave basically proposes doing away with the notion of a logging API in favor | ||||
| of `fmt.Printf()`.  We disagree, especially when you consider things like output | ||||
| locations, timestamps, file and line decorations, and structured logging.  This | ||||
| package restricts the logging API to just 2 types of logs: info and error. | ||||
| 
 | ||||
| Info logs are things you want to tell the user which are not errors.  Error | ||||
| logs are, well, errors.  If your code receives an `error` from a subordinate | ||||
| function call and is logging that `error` *and not returning it*, use error | ||||
| logs. | ||||
| 
 | ||||
| 2) Verbosity-levels on info logs.  This gives developers a chance to indicate | ||||
| 2. Verbosity-levels on info logs.  This gives developers a chance to indicate | ||||
| arbitrary grades of importance for info logs, without assigning names with | ||||
| semantic meaning such as "warning", "trace", and "debug".  Superficially this | ||||
| semantic meaning such as "warning", "trace", and "debug."  Superficially this | ||||
| may feel very similar, but the primary difference is the lack of semantics. | ||||
| Because verbosity is a numerical value, it's safe to assume that an app running | ||||
| with higher verbosity means more (and less important) logs will be generated. | ||||
| 
 | ||||
| This is a BETA grade API. | ||||
| ## Implementations (non-exhaustive) | ||||
| 
 | ||||
| There are implementations for the following logging libraries: | ||||
| 
 | ||||
| - **a function** (can bridge to non-structured libraries): [funcr](https://github.com/go-logr/logr/tree/master/funcr) | ||||
| - **github.com/google/glog**: [glogr](https://github.com/go-logr/glogr) | ||||
| - **k8s.io/klog**: [klogr](https://git.k8s.io/klog/klogr) | ||||
| - **k8s.io/klog** (for Kubernetes): [klogr](https://git.k8s.io/klog/klogr) | ||||
| - **go.uber.org/zap**: [zapr](https://github.com/go-logr/zapr) | ||||
| - **log** (the Go standard library logger): | ||||
|   [stdr](https://github.com/go-logr/stdr) | ||||
| - **log** (the Go standard library logger): [stdr](https://github.com/go-logr/stdr) | ||||
| - **github.com/sirupsen/logrus**: [logrusr](https://github.com/bombsimon/logrusr) | ||||
| - **github.com/wojas/genericr**: [genericr](https://github.com/wojas/genericr) (makes it easy to implement your own backend) | ||||
| - **logfmt** (Heroku style [logging](https://www.brandur.org/logfmt)): [logfmtr](https://github.com/iand/logfmtr) | ||||
| - **github.com/rs/zerolog**: [zerologr](https://github.com/go-logr/zerologr) | ||||
| 
 | ||||
| # FAQ | ||||
| ## FAQ | ||||
| 
 | ||||
| ## Conceptual | ||||
| ### Conceptual | ||||
| 
 | ||||
| ## Why structured logging? | ||||
| #### Why structured logging? | ||||
| 
 | ||||
| - **Structured logs are more easily queriable**: Since you've got | ||||
| - **Structured logs are more easily queryable**: Since you've got | ||||
|   key-value pairs, it's much easier to query your structured logs for | ||||
|   particular values by filtering on the contents of a particular key -- | ||||
|   think searching request logs for error codes, Kubernetes reconcilers for | ||||
|   the name and namespace of the reconciled object, etc | ||||
|   the name and namespace of the reconciled object, etc. | ||||
| 
 | ||||
| - **Structured logging makes it easier to have cross-referencable logs**: | ||||
| - **Structured logging makes it easier to have cross-referenceable logs**: | ||||
|   Similarly to searchability, if you maintain conventions around your | ||||
|   keys, it becomes easy to gather all log lines related to a particular | ||||
|   concept. | ||||
|   | ||||
| 
 | ||||
| - **Structured logs allow better dimensions of filtering**: if you have | ||||
|   structure to your logs, you've got more precise control over how much | ||||
|   information is logged -- you might choose in a particular configuration | ||||
|   to log certain keys but not others, only log lines where a certain key | ||||
|   matches a certain value, etc, instead of just having v-levels and names | ||||
|   matches a certain value, etc., instead of just having v-levels and names | ||||
|   to key off of. | ||||
| 
 | ||||
| - **Structured logs better represent structured data**: sometimes, the | ||||
|   data that you want to log is inherently structured (think tuple-link | ||||
|   objects).  Structured logs allow you to preserve that structure when | ||||
|   objects.)  Structured logs allow you to preserve that structure when | ||||
|   outputting. | ||||
| 
 | ||||
| ## Why V-levels? | ||||
| #### Why V-levels? | ||||
| 
 | ||||
| **V-levels give operators an easy way to control the chattiness of log | ||||
| operations**.  V-levels provide a way for a given package to distinguish | ||||
| the relative importance or verbosity of a given log message.  Then, if | ||||
| a particular logger or package is logging too many messages, the user | ||||
| of the package can simply change the v-levels for that library.  | ||||
| of the package can simply change the v-levels for that library. | ||||
| 
 | ||||
| ## Why not more named levels, like Warning? | ||||
| #### Why not named levels, like Info/Warning/Error? | ||||
| 
 | ||||
| Read [Dave Cheney's post][warning-makes-no-sense].  Then read [Differences | ||||
| from Dave's ideas](#differences-from-daves-ideas). | ||||
| 
 | ||||
| ## Why not allow format strings, too? | ||||
| #### Why not allow format strings, too? | ||||
| 
 | ||||
| **Format strings negate many of the benefits of structured logs**: | ||||
| 
 | ||||
| - They're not easily searchable without resorting to fuzzy searching, | ||||
|   regular expressions, etc | ||||
|   regular expressions, etc. | ||||
| 
 | ||||
| - They don't store structured data well, since contents are flattened into | ||||
|   a string | ||||
|   a string. | ||||
| 
 | ||||
| - They're not cross-referencable | ||||
| - They're not cross-referenceable. | ||||
| 
 | ||||
| - They don't compress easily, since the message is not constant | ||||
| - They don't compress easily, since the message is not constant. | ||||
| 
 | ||||
| (unless you turn positional parameters into key-value pairs with numerical | ||||
| (Unless you turn positional parameters into key-value pairs with numerical | ||||
| keys, at which point you've gotten key-value logging with meaningless | ||||
| keys) | ||||
| keys.) | ||||
| 
 | ||||
| ## Practical | ||||
| ### Practical | ||||
| 
 | ||||
| ## Why key-value pairs, and not a map? | ||||
| #### Why key-value pairs, and not a map? | ||||
| 
 | ||||
| Key-value pairs are *much* easier to optimize, especially around | ||||
| allocations.  Zap (a structured logger that inspired logr's interface) has | ||||
|  | @ -117,26 +187,26 @@ While the interface ends up being a little less obvious, you get | |||
| potentially better performance, plus avoid making users type | ||||
| `map[string]string{}` every time they want to log. | ||||
| 
 | ||||
| ## What if my V-levels differ between libraries? | ||||
| #### What if my V-levels differ between libraries? | ||||
| 
 | ||||
| That's fine.  Control your V-levels on a per-logger basis, and use the | ||||
| `WithName` function to pass different loggers to different libraries. | ||||
| `WithName` method to pass different loggers to different libraries. | ||||
| 
 | ||||
| Generally, you should take care to ensure that you have relatively | ||||
| consistent V-levels within a given logger, however, as this makes deciding | ||||
| on what verbosity of logs to request easier. | ||||
| 
 | ||||
| ## But I *really* want to use a format string! | ||||
| #### But I really want to use a format string! | ||||
| 
 | ||||
| That's not actually a question.  Assuming your question is "how do | ||||
| I convert my mental model of logging with format strings to logging with | ||||
| constant messages": | ||||
| 
 | ||||
| 1. figure out what the error actually is, as you'd write in a TL;DR style, | ||||
|    and use that as a message | ||||
| 1. Figure out what the error actually is, as you'd write in a TL;DR style, | ||||
|    and use that as a message. | ||||
| 
 | ||||
| 2. For every place you'd write a format specifier, look to the word before | ||||
|    it, and add that as a key value pair | ||||
|    it, and add that as a key value pair. | ||||
| 
 | ||||
| For instance, consider the following examples (all taken from spots in the | ||||
| Kubernetes codebase): | ||||
|  | @ -150,34 +220,59 @@ Kubernetes codebase): | |||
|   response when requesting url", "attempt", retries, "after | ||||
|   seconds", seconds, "url", url)` | ||||
| 
 | ||||
| If you *really* must use a format string, place it as a key value, and | ||||
| call `fmt.Sprintf` yourself -- for instance, `log.Printf("unable to | ||||
| If you *really* must use a format string, use it in a key's value, and | ||||
| call `fmt.Sprintf` yourself.  For instance: `log.Printf("unable to | ||||
| reflect over type %T")` becomes `logger.Info("unable to reflect over | ||||
| type", "type", fmt.Sprintf("%T"))`.  In general though, the cases where | ||||
| this is necessary should be few and far between. | ||||
| 
 | ||||
| ## How do I choose my V-levels? | ||||
| #### How do I choose my V-levels? | ||||
| 
 | ||||
| This is basically the only hard constraint: increase V-levels to denote | ||||
| more verbose or more debug-y logs. | ||||
| 
 | ||||
| Otherwise, you can start out with `0` as "you always want to see this", | ||||
| `1` as "common logging that you might *possibly* want to turn off", and | ||||
| `10` as "I would like to performance-test your log collection stack". | ||||
| `10` as "I would like to performance-test your log collection stack." | ||||
| 
 | ||||
| Then gradually choose levels in between as you need them, working your way | ||||
| down from 10 (for debug and trace style logs) and up from 1 (for chattier | ||||
| info-type logs). | ||||
| info-type logs.) | ||||
| 
 | ||||
| ## How do I choose my keys | ||||
| #### How do I choose my keys? | ||||
| 
 | ||||
| - make your keys human-readable | ||||
| - constant keys are generally a good idea | ||||
| - be consistent across your codebase | ||||
| - keys should naturally match parts of the message string | ||||
| Keys are fairly flexible, and can hold more or less any string | ||||
| value. For best compatibility with implementations and consistency | ||||
| with existing code in other projects, there are a few conventions you | ||||
| should consider. | ||||
| 
 | ||||
| - Make your keys human-readable. | ||||
| - Constant keys are generally a good idea. | ||||
| - Be consistent across your codebase. | ||||
| - Keys should naturally match parts of the message string. | ||||
| - Use lower case for simple keys and | ||||
|   [lowerCamelCase](https://en.wiktionary.org/wiki/lowerCamelCase) for | ||||
|   more complex ones. Kubernetes is one example of a project that has | ||||
|   [adopted that | ||||
|   convention](https://github.com/kubernetes/community/blob/HEAD/contributors/devel/sig-instrumentation/migration-to-structured-logging.md#name-arguments). | ||||
| 
 | ||||
| While key names are mostly unrestricted (and spaces are acceptable), | ||||
| it's generally a good idea to stick to printable ascii characters, or at | ||||
| least match the general character set of your log lines. | ||||
| 
 | ||||
| #### Why should keys be constant values? | ||||
| 
 | ||||
| The point of structured logging is to make later log processing easier.  Your | ||||
| keys are, effectively, the schema of each log message.  If you use different | ||||
| keys across instances of the same log line, you will make your structured logs | ||||
| much harder to use.  `Sprintf()` is for values, not for keys! | ||||
| 
 | ||||
| #### Why is this not a pure interface? | ||||
| 
 | ||||
| The Logger type is implemented as a struct in order to allow the Go compiler to | ||||
| optimize things like high-V `Info` logs that are not triggered.  Not all of | ||||
| these implementations are implemented yet, but this structure was suggested as | ||||
| a way to ensure they *can* be implemented.  All of the real work is behind the | ||||
| `LogSink` interface. | ||||
| 
 | ||||
| [warning-makes-no-sense]: http://dave.cheney.net/2015/11/05/lets-talk-about-logging | ||||
|  |  | |||
|  | @ -16,36 +16,39 @@ limitations under the License. | |||
| 
 | ||||
| package logr | ||||
| 
 | ||||
| // Discard returns a valid Logger that discards all messages logged to it.
 | ||||
| // It can be used whenever the caller is not interested in the logs.
 | ||||
| // Discard returns a Logger that discards all messages logged to it.  It can be
 | ||||
| // used whenever the caller is not interested in the logs.  Logger instances
 | ||||
| // produced by this function always compare as equal.
 | ||||
| func Discard() Logger { | ||||
| 	return DiscardLogger{} | ||||
| 	return Logger{ | ||||
| 		level: 0, | ||||
| 		sink:  discardLogSink{}, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // DiscardLogger is a Logger that discards all messages.
 | ||||
| type DiscardLogger struct{} | ||||
| // discardLogSink is a LogSink that discards all messages.
 | ||||
| type discardLogSink struct{} | ||||
| 
 | ||||
| func (l DiscardLogger) Enabled() bool { | ||||
| // Verify that it actually implements the interface
 | ||||
| var _ LogSink = discardLogSink{} | ||||
| 
 | ||||
| func (l discardLogSink) Init(RuntimeInfo) { | ||||
| } | ||||
| 
 | ||||
| func (l discardLogSink) Enabled(int) bool { | ||||
| 	return false | ||||
| } | ||||
| 
 | ||||
| func (l DiscardLogger) Info(msg string, keysAndValues ...interface{}) { | ||||
| func (l discardLogSink) Info(int, string, ...interface{}) { | ||||
| } | ||||
| 
 | ||||
| func (l DiscardLogger) Error(err error, msg string, keysAndValues ...interface{}) { | ||||
| func (l discardLogSink) Error(error, string, ...interface{}) { | ||||
| } | ||||
| 
 | ||||
| func (l DiscardLogger) V(level int) Logger { | ||||
| func (l discardLogSink) WithValues(...interface{}) LogSink { | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| func (l DiscardLogger) WithValues(keysAndValues ...interface{}) Logger { | ||||
| func (l discardLogSink) WithName(string) LogSink { | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| func (l DiscardLogger) WithName(name string) Logger { | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| // Verify that it actually implements the interface
 | ||||
| var _ Logger = DiscardLogger{} | ||||
|  |  | |||
|  | @ -1,3 +1,3 @@ | |||
| module github.com/go-logr/logr | ||||
| 
 | ||||
| go 1.14 | ||||
| go 1.16 | ||||
|  |  | |||
|  | @ -16,83 +16,104 @@ limitations under the License. | |||
| 
 | ||||
| // This design derives from Dave Cheney's blog:
 | ||||
| //     http://dave.cheney.net/2015/11/05/lets-talk-about-logging
 | ||||
| //
 | ||||
| // This is a BETA grade API.  Until there is a significant 2nd implementation,
 | ||||
| // I don't really know how it will change.
 | ||||
| 
 | ||||
| // Package logr defines abstract interfaces for logging.  Packages can depend on
 | ||||
| // these interfaces and callers can implement logging in whatever way is
 | ||||
| // appropriate.
 | ||||
| // Package logr defines a general-purpose logging API and abstract interfaces
 | ||||
| // to back that API.  Packages in the Go ecosystem can depend on this package,
 | ||||
| // while callers can implement logging with whatever backend is appropriate.
 | ||||
| //
 | ||||
| // Usage
 | ||||
| //
 | ||||
| // Logging is done using a Logger.  Loggers can have name prefixes and named
 | ||||
| // values attached, so that all log messages logged with that Logger have some
 | ||||
| // base context associated.
 | ||||
| // Logging is done using a Logger instance.  Logger is a concrete type with
 | ||||
| // methods, which defers the actual logging to a LogSink interface.  The main
 | ||||
| // methods of Logger are Info() and Error().  Arguments to Info() and Error()
 | ||||
| // are key/value pairs rather than printf-style formatted strings, emphasizing
 | ||||
| // "structured logging".
 | ||||
| //
 | ||||
| // The term "key" is used to refer to the name associated with a particular
 | ||||
| // value, to disambiguate it from the general Logger name.
 | ||||
| // With Go's standard log package, we might write:
 | ||||
| //   log.Printf("setting target value %s", targetValue)
 | ||||
| //
 | ||||
| // For instance, suppose we're trying to reconcile the state of an object, and
 | ||||
| // we want to log that we've made some decision.
 | ||||
| // With logr's structured logging, we'd write:
 | ||||
| //   logger.Info("setting target", "value", targetValue)
 | ||||
| //
 | ||||
| // With the traditional log package, we might write:
 | ||||
| // Errors are much the same.  Instead of:
 | ||||
| //   log.Printf("failed to open the pod bay door for user %s: %v", user, err)
 | ||||
| //
 | ||||
| // We'd write:
 | ||||
| //   logger.Error(err, "failed to open the pod bay door", "user", user)
 | ||||
| //
 | ||||
| // Info() and Error() are very similar, but they are separate methods so that
 | ||||
| // LogSink implementations can choose to do things like attach additional
 | ||||
| // information (such as stack traces) on calls to Error(). Error() messages are
 | ||||
| // always logged, regardless of the current verbosity.  If there is no error
 | ||||
| // instance available, passing nil is valid.
 | ||||
| //
 | ||||
| // Verbosity
 | ||||
| //
 | ||||
| // Often we want to log information only when the application in "verbose
 | ||||
| // mode".  To write log lines that are more verbose, Logger has a V() method.
 | ||||
| // The higher the V-level of a log line, the less critical it is considered.
 | ||||
| // Log-lines with V-levels that are not enabled (as per the LogSink) will not
 | ||||
| // be written.  Level V(0) is the default, and logger.V(0).Info() has the same
 | ||||
| // meaning as logger.Info().  Negative V-levels have the same meaning as V(0).
 | ||||
| // Error messages do not have a verbosity level and are always logged.
 | ||||
| //
 | ||||
| // Where we might have written:
 | ||||
| //   if flVerbose >= 2 {
 | ||||
| //       log.Printf("an unusual thing happened")
 | ||||
| //   }
 | ||||
| //
 | ||||
| // We can write:
 | ||||
| //   logger.V(2).Info("an unusual thing happened")
 | ||||
| //
 | ||||
| // Logger Names
 | ||||
| //
 | ||||
| // Logger instances can have name strings so that all messages logged through
 | ||||
| // that instance have additional context.  For example, you might want to add
 | ||||
| // a subsystem name:
 | ||||
| //
 | ||||
| //   logger.WithName("compactor").Info("started", "time", time.Now())
 | ||||
| //
 | ||||
| // The WithName() method returns a new Logger, which can be passed to
 | ||||
| // constructors or other functions for further use.  Repeated use of WithName()
 | ||||
| // will accumulate name "segments".  These name segments will be joined in some
 | ||||
| // way by the LogSink implementation.  It is strongly recommended that name
 | ||||
| // segments contain simple identifiers (letters, digits, and hyphen), and do
 | ||||
| // not contain characters that could muddle the log output or confuse the
 | ||||
| // joining operation (e.g. whitespace, commas, periods, slashes, brackets,
 | ||||
| // quotes, etc).
 | ||||
| //
 | ||||
| // Saved Values
 | ||||
| //
 | ||||
| // Logger instances can store any number of key/value pairs, which will be
 | ||||
| // logged alongside all messages logged through that instance.  For example,
 | ||||
| // you might want to create a Logger instance per managed object:
 | ||||
| //
 | ||||
| // With the standard log package, we might write:
 | ||||
| //   log.Printf("decided to set field foo to value %q for object %s/%s",
 | ||||
| //       targetValue, object.Namespace, object.Name)
 | ||||
| //
 | ||||
| // With logr's structured logging, we'd write:
 | ||||
| //   // elsewhere in the file, set up the logger to log with the prefix of
 | ||||
| //   // "reconcilers", and the named value target-type=Foo, for extra context.
 | ||||
| //   log := mainLogger.WithName("reconcilers").WithValues("target-type", "Foo")
 | ||||
| // With logr we'd write:
 | ||||
| //   // Elsewhere: set up the logger to log the object name.
 | ||||
| //   obj.logger = mainLogger.WithValues(
 | ||||
| //       "name", obj.name, "namespace", obj.namespace)
 | ||||
| //
 | ||||
| //   // later on...
 | ||||
| //   log.Info("setting foo on object", "value", targetValue, "object", object)
 | ||||
| //   obj.logger.Info("setting foo", "value", targetValue)
 | ||||
| //
 | ||||
| // Depending on our logging implementation, we could then make logging decisions
 | ||||
| // based on field values (like only logging such events for objects in a certain
 | ||||
| // namespace), or copy the structured information into a structured log store.
 | ||||
| // Best Practices
 | ||||
| //
 | ||||
| // For logging errors, Logger has a method called Error.  Suppose we wanted to
 | ||||
| // log an error while reconciling.  With the traditional log package, we might
 | ||||
| // write:
 | ||||
| //   log.Errorf("unable to reconcile object %s/%s: %v", object.Namespace, object.Name, err)
 | ||||
| //
 | ||||
| // With logr, we'd instead write:
 | ||||
| //   // assuming the above setup for log
 | ||||
| //   log.Error(err, "unable to reconcile object", "object", object)
 | ||||
| //
 | ||||
| // This functions similarly to:
 | ||||
| //   log.Info("unable to reconcile object", "error", err, "object", object)
 | ||||
| //
 | ||||
| // However, it ensures that a standard key for the error value ("error") is used
 | ||||
| // across all error logging.  Furthermore, certain implementations may choose to
 | ||||
| // attach additional information (such as stack traces) on calls to Error, so
 | ||||
| // it's preferred to use Error to log errors.
 | ||||
| //
 | ||||
| // Parts of a log line
 | ||||
| //
 | ||||
| // Each log message from a Logger has four types of context:
 | ||||
| // logger name, log verbosity, log message, and the named values.
 | ||||
| //
 | ||||
| // The Logger name consists of a series of name "segments" added by successive
 | ||||
| // calls to WithName.  These name segments will be joined in some way by the
 | ||||
| // underlying implementation.  It is strongly recommended that name segments
 | ||||
| // contain simple identifiers (letters, digits, and hyphen), and do not contain
 | ||||
| // characters that could muddle the log output or confuse the joining operation
 | ||||
| // (e.g.  whitespace, commas, periods, slashes, brackets, quotes, etc).
 | ||||
| //
 | ||||
| // Log verbosity represents how little a log matters.  Level zero, the default,
 | ||||
| // matters most.  Increasing levels matter less and less.  Try to avoid lots of
 | ||||
| // different verbosity levels, and instead provide useful keys, logger names,
 | ||||
| // and log messages for users to filter on.  It's illegal to pass a log level
 | ||||
| // below zero.
 | ||||
| // Logger has very few hard rules, with the goal that LogSink implementations
 | ||||
| // might have a lot of freedom to differentiate.  There are, however, some
 | ||||
| // things to consider.
 | ||||
| //
 | ||||
| // The log message consists of a constant message attached to the log line.
 | ||||
| // This should generally be a simple description of what's occurring, and should
 | ||||
| // never be a format string.
 | ||||
| // never be a format string.  Variable information can then be attached using
 | ||||
| // named values.
 | ||||
| //
 | ||||
| // Variable information can then be attached using named values (key/value
 | ||||
| // pairs).  Keys are arbitrary strings, while values may be any Go value.
 | ||||
| // Keys are arbitrary strings, but should generally be constant values.  Values
 | ||||
| // may be any Go value, but how the value is formatted is determined by the
 | ||||
| // LogSink implementation.
 | ||||
| //
 | ||||
| // Key Naming Conventions
 | ||||
| //
 | ||||
|  | @ -102,6 +123,7 @@ limitations under the License. | |||
| //   * be constant (not dependent on input data)
 | ||||
| //   * contain only printable characters
 | ||||
| //   * not contain whitespace or punctuation
 | ||||
| //   * use lower case for simple keys and lowerCamelCase for more complex ones
 | ||||
| //
 | ||||
| // These guidelines help ensure that log data is processed properly regardless
 | ||||
| // of the log implementation.  For example, log implementations will try to
 | ||||
|  | @ -110,21 +132,22 @@ limitations under the License. | |||
| // While users are generally free to use key names of their choice, it's
 | ||||
| // generally best to avoid using the following keys, as they're frequently used
 | ||||
| // by implementations:
 | ||||
| //
 | ||||
| //   * `"caller"`: the calling information (file/line) of a particular log line.
 | ||||
| //   * `"error"`: the underlying error value in the `Error` method.
 | ||||
| //   * `"level"`: the log level.
 | ||||
| //   * `"logger"`: the name of the associated logger.
 | ||||
| //   * `"msg"`: the log message.
 | ||||
| //   * `"stacktrace"`: the stack trace associated with a particular log line or
 | ||||
| //                     error (often from the `Error` message).
 | ||||
| //   * `"ts"`: the timestamp for a log line.
 | ||||
| //   * "caller": the calling information (file/line) of a particular log line
 | ||||
| //   * "error": the underlying error value in the `Error` method
 | ||||
| //   * "level": the log level
 | ||||
| //   * "logger": the name of the associated logger
 | ||||
| //   * "msg": the log message
 | ||||
| //   * "stacktrace": the stack trace associated with a particular log line or
 | ||||
| //                   error (often from the `Error` message)
 | ||||
| //   * "ts": the timestamp for a log line
 | ||||
| //
 | ||||
| // Implementations are encouraged to make use of these keys to represent the
 | ||||
| // above concepts, when necessary (for example, in a pure-JSON output form, it
 | ||||
| // would be necessary to represent at least message and timestamp as ordinary
 | ||||
| // named values).
 | ||||
| //
 | ||||
| // Break Glass
 | ||||
| //
 | ||||
| // Implementations may choose to give callers access to the underlying
 | ||||
| // logging implementation.  The recommended pattern for this is:
 | ||||
| //   // Underlier exposes access to the underlying logging implementation.
 | ||||
|  | @ -134,81 +157,222 @@ limitations under the License. | |||
| //   type Underlier interface {
 | ||||
| //       GetUnderlying() <underlying-type>
 | ||||
| //   }
 | ||||
| //
 | ||||
| // Logger grants access to the sink to enable type assertions like this:
 | ||||
| //   func DoSomethingWithImpl(log logr.Logger) {
 | ||||
| //       if underlier, ok := log.GetSink()(impl.Underlier) {
 | ||||
| //          implLogger := underlier.GetUnderlying()
 | ||||
| //          ...
 | ||||
| //       }
 | ||||
| //   }
 | ||||
| //
 | ||||
| // Custom `With*` functions can be implemented by copying the complete
 | ||||
| // Logger struct and replacing the sink in the copy:
 | ||||
| //   // WithFooBar changes the foobar parameter in the log sink and returns a
 | ||||
| //   // new logger with that modified sink.  It does nothing for loggers where
 | ||||
| //   // the sink doesn't support that parameter.
 | ||||
| //   func WithFoobar(log logr.Logger, foobar int) logr.Logger {
 | ||||
| //      if foobarLogSink, ok := log.GetSink()(FoobarSink); ok {
 | ||||
| //         log = log.WithSink(foobarLogSink.WithFooBar(foobar))
 | ||||
| //      }
 | ||||
| //      return log
 | ||||
| //   }
 | ||||
| //
 | ||||
| // Don't use New to construct a new Logger with a LogSink retrieved from an
 | ||||
| // existing Logger. Source code attribution might not work correctly and
 | ||||
| // unexported fields in Logger get lost.
 | ||||
| //
 | ||||
| // Beware that the same LogSink instance may be shared by different logger
 | ||||
| // instances. Calling functions that modify the LogSink will affect all of
 | ||||
| // those.
 | ||||
| package logr | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| ) | ||||
| 
 | ||||
| // TODO: consider adding back in format strings if they're really needed
 | ||||
| // TODO: consider other bits of zap/zapcore functionality like ObjectMarshaller (for arbitrary objects)
 | ||||
| // TODO: consider other bits of glog functionality like Flush, OutputStats
 | ||||
| 
 | ||||
| // Logger represents the ability to log messages, both errors and not.
 | ||||
| type Logger interface { | ||||
| 	// Enabled tests whether this Logger is enabled.  For example, commandline
 | ||||
| 	// flags might be used to set the logging verbosity and disable some info
 | ||||
| 	// logs.
 | ||||
| 	Enabled() bool | ||||
| 
 | ||||
| 	// Info logs a non-error message with the given key/value pairs as context.
 | ||||
| 	//
 | ||||
| 	// The msg argument should be used to add some constant description to
 | ||||
| 	// the log line.  The key/value pairs can then be used to add additional
 | ||||
| 	// variable information.  The key/value pairs should alternate string
 | ||||
| 	// keys and arbitrary values.
 | ||||
| 	Info(msg string, keysAndValues ...interface{}) | ||||
| 
 | ||||
| 	// Error logs an error, with the given message and key/value pairs as context.
 | ||||
| 	// It functions similarly to calling Info with the "error" named value, but may
 | ||||
| 	// have unique behavior, and should be preferred for logging errors (see the
 | ||||
| 	// package documentations for more information).
 | ||||
| 	//
 | ||||
| 	// The msg field should be used to add context to any underlying error,
 | ||||
| 	// while the err field should be used to attach the actual error that
 | ||||
| 	// triggered this log line, if present.
 | ||||
| 	Error(err error, msg string, keysAndValues ...interface{}) | ||||
| 
 | ||||
| 	// V returns an Logger value for a specific verbosity level, relative to
 | ||||
| 	// this Logger.  In other words, V values are additive.  V higher verbosity
 | ||||
| 	// level means a log message is less important.  It's illegal to pass a log
 | ||||
| 	// level less than zero.
 | ||||
| 	V(level int) Logger | ||||
| 
 | ||||
| 	// WithValues adds some key-value pairs of context to a logger.
 | ||||
| 	// See Info for documentation on how key/value pairs work.
 | ||||
| 	WithValues(keysAndValues ...interface{}) Logger | ||||
| 
 | ||||
| 	// WithName adds a new element to the logger's name.
 | ||||
| 	// Successive calls with WithName continue to append
 | ||||
| 	// suffixes to the logger's name.  It's strongly recommended
 | ||||
| 	// that name segments contain only letters, digits, and hyphens
 | ||||
| 	// (see the package documentation for more information).
 | ||||
| 	WithName(name string) Logger | ||||
| // New returns a new Logger instance.  This is primarily used by libraries
 | ||||
| // implementing LogSink, rather than end users.
 | ||||
| func New(sink LogSink) Logger { | ||||
| 	logger := Logger{} | ||||
| 	logger.setSink(sink) | ||||
| 	sink.Init(runtimeInfo) | ||||
| 	return logger | ||||
| } | ||||
| 
 | ||||
| // InfoLogger provides compatibility with code that relies on the v0.1.0
 | ||||
| // interface.
 | ||||
| //
 | ||||
| // Deprecated: InfoLogger is an artifact of early versions of this API.  New
 | ||||
| // users should never use it and existing users should use Logger instead. This
 | ||||
| // will be removed in a future release.
 | ||||
| type InfoLogger = Logger | ||||
| // setSink stores the sink and updates any related fields. It mutates the
 | ||||
| // logger and thus is only safe to use for loggers that are not currently being
 | ||||
| // used concurrently.
 | ||||
| func (l *Logger) setSink(sink LogSink) { | ||||
| 	l.sink = sink | ||||
| } | ||||
| 
 | ||||
| // GetSink returns the stored sink.
 | ||||
| func (l Logger) GetSink() LogSink { | ||||
| 	return l.sink | ||||
| } | ||||
| 
 | ||||
| // WithSink returns a copy of the logger with the new sink.
 | ||||
| func (l Logger) WithSink(sink LogSink) Logger { | ||||
| 	l.setSink(sink) | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| // Logger is an interface to an abstract logging implementation.  This is a
 | ||||
| // concrete type for performance reasons, but all the real work is passed on to
 | ||||
| // a LogSink.  Implementations of LogSink should provide their own constructors
 | ||||
| // that return Logger, not LogSink.
 | ||||
| //
 | ||||
| // The underlying sink can be accessed through GetSink and be modified through
 | ||||
| // WithSink. This enables the implementation of custom extensions (see "Break
 | ||||
| // Glass" in the package documentation). Normally the sink should be used only
 | ||||
| // indirectly.
 | ||||
| type Logger struct { | ||||
| 	sink  LogSink | ||||
| 	level int | ||||
| } | ||||
| 
 | ||||
| // Enabled tests whether this Logger is enabled.  For example, commandline
 | ||||
| // flags might be used to set the logging verbosity and disable some info logs.
 | ||||
| func (l Logger) Enabled() bool { | ||||
| 	return l.sink.Enabled(l.level) | ||||
| } | ||||
| 
 | ||||
| // Info logs a non-error message with the given key/value pairs as context.
 | ||||
| //
 | ||||
| // The msg argument should be used to add some constant description to the log
 | ||||
| // line.  The key/value pairs can then be used to add additional variable
 | ||||
| // information.  The key/value pairs must alternate string keys and arbitrary
 | ||||
| // values.
 | ||||
| func (l Logger) Info(msg string, keysAndValues ...interface{}) { | ||||
| 	if l.Enabled() { | ||||
| 		if withHelper, ok := l.sink.(CallStackHelperLogSink); ok { | ||||
| 			withHelper.GetCallStackHelper()() | ||||
| 		} | ||||
| 		l.sink.Info(l.level, msg, keysAndValues...) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Error logs an error, with the given message and key/value pairs as context.
 | ||||
| // It functions similarly to Info, but may have unique behavior, and should be
 | ||||
| // preferred for logging errors (see the package documentations for more
 | ||||
| // information). The log message will always be emitted, regardless of
 | ||||
| // verbosity level.
 | ||||
| //
 | ||||
| // The msg argument should be used to add context to any underlying error,
 | ||||
| // while the err argument should be used to attach the actual error that
 | ||||
| // triggered this log line, if present. The err parameter is optional
 | ||||
| // and nil may be passed instead of an error instance.
 | ||||
| func (l Logger) Error(err error, msg string, keysAndValues ...interface{}) { | ||||
| 	if withHelper, ok := l.sink.(CallStackHelperLogSink); ok { | ||||
| 		withHelper.GetCallStackHelper()() | ||||
| 	} | ||||
| 	l.sink.Error(err, msg, keysAndValues...) | ||||
| } | ||||
| 
 | ||||
| // V returns a new Logger instance for a specific verbosity level, relative to
 | ||||
| // this Logger.  In other words, V-levels are additive.  A higher verbosity
 | ||||
| // level means a log message is less important.  Negative V-levels are treated
 | ||||
| // as 0.
 | ||||
| func (l Logger) V(level int) Logger { | ||||
| 	if level < 0 { | ||||
| 		level = 0 | ||||
| 	} | ||||
| 	l.level += level | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| // WithValues returns a new Logger instance with additional key/value pairs.
 | ||||
| // See Info for documentation on how key/value pairs work.
 | ||||
| func (l Logger) WithValues(keysAndValues ...interface{}) Logger { | ||||
| 	l.setSink(l.sink.WithValues(keysAndValues...)) | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| // WithName returns a new Logger instance with the specified name element added
 | ||||
| // to the Logger's name.  Successive calls with WithName append additional
 | ||||
| // suffixes to the Logger's name.  It's strongly recommended that name segments
 | ||||
| // contain only letters, digits, and hyphens (see the package documentation for
 | ||||
| // more information).
 | ||||
| func (l Logger) WithName(name string) Logger { | ||||
| 	l.setSink(l.sink.WithName(name)) | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| // WithCallDepth returns a Logger instance that offsets the call stack by the
 | ||||
| // specified number of frames when logging call site information, if possible.
 | ||||
| // This is useful for users who have helper functions between the "real" call
 | ||||
| // site and the actual calls to Logger methods.  If depth is 0 the attribution
 | ||||
| // should be to the direct caller of this function.  If depth is 1 the
 | ||||
| // attribution should skip 1 call frame, and so on.  Successive calls to this
 | ||||
| // are additive.
 | ||||
| //
 | ||||
| // If the underlying log implementation supports a WithCallDepth(int) method,
 | ||||
| // it will be called and the result returned.  If the implementation does not
 | ||||
| // support CallDepthLogSink, the original Logger will be returned.
 | ||||
| //
 | ||||
| // To skip one level, WithCallStackHelper() should be used instead of
 | ||||
| // WithCallDepth(1) because it works with implementions that support the
 | ||||
| // CallDepthLogSink and/or CallStackHelperLogSink interfaces.
 | ||||
| func (l Logger) WithCallDepth(depth int) Logger { | ||||
| 	if withCallDepth, ok := l.sink.(CallDepthLogSink); ok { | ||||
| 		l.setSink(withCallDepth.WithCallDepth(depth)) | ||||
| 	} | ||||
| 	return l | ||||
| } | ||||
| 
 | ||||
| // WithCallStackHelper returns a new Logger instance that skips the direct
 | ||||
| // caller when logging call site information, if possible.  This is useful for
 | ||||
| // users who have helper functions between the "real" call site and the actual
 | ||||
| // calls to Logger methods and want to support loggers which depend on marking
 | ||||
| // each individual helper function, like loggers based on testing.T.
 | ||||
| //
 | ||||
| // In addition to using that new logger instance, callers also must call the
 | ||||
| // returned function.
 | ||||
| //
 | ||||
| // If the underlying log implementation supports a WithCallDepth(int) method,
 | ||||
| // WithCallDepth(1) will be called to produce a new logger. If it supports a
 | ||||
| // WithCallStackHelper() method, that will be also called. If the
 | ||||
| // implementation does not support either of these, the original Logger will be
 | ||||
| // returned.
 | ||||
| func (l Logger) WithCallStackHelper() (func(), Logger) { | ||||
| 	var helper func() | ||||
| 	if withCallDepth, ok := l.sink.(CallDepthLogSink); ok { | ||||
| 		l.setSink(withCallDepth.WithCallDepth(1)) | ||||
| 	} | ||||
| 	if withHelper, ok := l.sink.(CallStackHelperLogSink); ok { | ||||
| 		helper = withHelper.GetCallStackHelper() | ||||
| 	} else { | ||||
| 		helper = func() {} | ||||
| 	} | ||||
| 	return helper, l | ||||
| } | ||||
| 
 | ||||
| // contextKey is how we find Loggers in a context.Context.
 | ||||
| type contextKey struct{} | ||||
| 
 | ||||
| // FromContext returns a Logger constructed from ctx or nil if no
 | ||||
| // logger details are found.
 | ||||
| func FromContext(ctx context.Context) Logger { | ||||
| // FromContext returns a Logger from ctx or an error if no Logger is found.
 | ||||
| func FromContext(ctx context.Context) (Logger, error) { | ||||
| 	if v, ok := ctx.Value(contextKey{}).(Logger); ok { | ||||
| 		return v | ||||
| 		return v, nil | ||||
| 	} | ||||
| 
 | ||||
| 	return nil | ||||
| 	return Logger{}, notFoundError{} | ||||
| } | ||||
| 
 | ||||
| // FromContextOrDiscard returns a Logger constructed from ctx or a Logger
 | ||||
| // that discards all messages if no logger details are found.
 | ||||
| // notFoundError exists to carry an IsNotFound method.
 | ||||
| type notFoundError struct{} | ||||
| 
 | ||||
| func (notFoundError) Error() string { | ||||
| 	return "no logr.Logger was present" | ||||
| } | ||||
| 
 | ||||
| func (notFoundError) IsNotFound() bool { | ||||
| 	return true | ||||
| } | ||||
| 
 | ||||
| // FromContextOrDiscard returns a Logger from ctx.  If no Logger is found, this
 | ||||
| // returns a Logger that discards all log messages.
 | ||||
| func FromContextOrDiscard(ctx context.Context) Logger { | ||||
| 	if v, ok := ctx.Value(contextKey{}).(Logger); ok { | ||||
| 		return v | ||||
|  | @ -217,12 +381,59 @@ func FromContextOrDiscard(ctx context.Context) Logger { | |||
| 	return Discard() | ||||
| } | ||||
| 
 | ||||
| // NewContext returns a new context derived from ctx that embeds the Logger.
 | ||||
| func NewContext(ctx context.Context, l Logger) context.Context { | ||||
| 	return context.WithValue(ctx, contextKey{}, l) | ||||
| // NewContext returns a new Context, derived from ctx, which carries the
 | ||||
| // provided Logger.
 | ||||
| func NewContext(ctx context.Context, logger Logger) context.Context { | ||||
| 	return context.WithValue(ctx, contextKey{}, logger) | ||||
| } | ||||
| 
 | ||||
| // CallDepthLogger represents a Logger that knows how to climb the call stack
 | ||||
| // RuntimeInfo holds information that the logr "core" library knows which
 | ||||
| // LogSinks might want to know.
 | ||||
| type RuntimeInfo struct { | ||||
| 	// CallDepth is the number of call frames the logr library adds between the
 | ||||
| 	// end-user and the LogSink.  LogSink implementations which choose to print
 | ||||
| 	// the original logging site (e.g. file & line) should climb this many
 | ||||
| 	// additional frames to find it.
 | ||||
| 	CallDepth int | ||||
| } | ||||
| 
 | ||||
| // runtimeInfo is a static global.  It must not be changed at run time.
 | ||||
| var runtimeInfo = RuntimeInfo{ | ||||
| 	CallDepth: 1, | ||||
| } | ||||
| 
 | ||||
| // LogSink represents a logging implementation.  End-users will generally not
 | ||||
| // interact with this type.
 | ||||
| type LogSink interface { | ||||
| 	// Init receives optional information about the logr library for LogSink
 | ||||
| 	// implementations that need it.
 | ||||
| 	Init(info RuntimeInfo) | ||||
| 
 | ||||
| 	// Enabled tests whether this LogSink is enabled at the specified V-level.
 | ||||
| 	// For example, commandline flags might be used to set the logging
 | ||||
| 	// verbosity and disable some info logs.
 | ||||
| 	Enabled(level int) bool | ||||
| 
 | ||||
| 	// Info logs a non-error message with the given key/value pairs as context.
 | ||||
| 	// The level argument is provided for optional logging.  This method will
 | ||||
| 	// only be called when Enabled(level) is true. See Logger.Info for more
 | ||||
| 	// details.
 | ||||
| 	Info(level int, msg string, keysAndValues ...interface{}) | ||||
| 
 | ||||
| 	// Error logs an error, with the given message and key/value pairs as
 | ||||
| 	// context.  See Logger.Error for more details.
 | ||||
| 	Error(err error, msg string, keysAndValues ...interface{}) | ||||
| 
 | ||||
| 	// WithValues returns a new LogSink with additional key/value pairs.  See
 | ||||
| 	// Logger.WithValues for more details.
 | ||||
| 	WithValues(keysAndValues ...interface{}) LogSink | ||||
| 
 | ||||
| 	// WithName returns a new LogSink with the specified name appended.  See
 | ||||
| 	// Logger.WithName for more details.
 | ||||
| 	WithName(name string) LogSink | ||||
| } | ||||
| 
 | ||||
| // CallDepthLogSink represents a Logger that knows how to climb the call stack
 | ||||
| // to identify the original call site and can offset the depth by a specified
 | ||||
| // number of frames.  This is useful for users who have helper functions
 | ||||
| // between the "real" call site and the actual calls to Logger methods.
 | ||||
|  | @ -232,35 +443,59 @@ func NewContext(ctx context.Context, l Logger) context.Context { | |||
| //
 | ||||
| // This is an optional interface and implementations are not required to
 | ||||
| // support it.
 | ||||
| type CallDepthLogger interface { | ||||
| 	Logger | ||||
| 
 | ||||
| 	// WithCallDepth returns a Logger that will offset the call stack by the
 | ||||
| 	// specified number of frames when logging call site information.  If depth
 | ||||
| 	// is 0 the attribution should be to the direct caller of this method.  If
 | ||||
| 	// depth is 1 the attribution should skip 1 call frame, and so on.
 | ||||
| type CallDepthLogSink interface { | ||||
| 	// WithCallDepth returns a LogSink that will offset the call
 | ||||
| 	// stack by the specified number of frames when logging call
 | ||||
| 	// site information.
 | ||||
| 	//
 | ||||
| 	// If depth is 0, the LogSink should skip exactly the number
 | ||||
| 	// of call frames defined in RuntimeInfo.CallDepth when Info
 | ||||
| 	// or Error are called, i.e. the attribution should be to the
 | ||||
| 	// direct caller of Logger.Info or Logger.Error.
 | ||||
| 	//
 | ||||
| 	// If depth is 1 the attribution should skip 1 call frame, and so on.
 | ||||
| 	// Successive calls to this are additive.
 | ||||
| 	WithCallDepth(depth int) Logger | ||||
| 	WithCallDepth(depth int) LogSink | ||||
| } | ||||
| 
 | ||||
| // WithCallDepth returns a Logger that will offset the call stack by the
 | ||||
| // specified number of frames when logging call site information, if possible.
 | ||||
| // This is useful for users who have helper functions between the "real" call
 | ||||
| // site and the actual calls to Logger methods.  If depth is 0 the attribution
 | ||||
| // should be to the direct caller of this function.  If depth is 1 the
 | ||||
| // attribution should skip 1 call frame, and so on.  Successive calls to this
 | ||||
| // are additive.
 | ||||
| // CallStackHelperLogSink represents a Logger that knows how to climb
 | ||||
| // the call stack to identify the original call site and can skip
 | ||||
| // intermediate helper functions if they mark themselves as
 | ||||
| // helper. Go's testing package uses that approach.
 | ||||
| //
 | ||||
| // If the underlying log implementation supports the CallDepthLogger interface,
 | ||||
| // the WithCallDepth method will be called and the result returned.  If the
 | ||||
| // implementation does not support CallDepthLogger, the original Logger will be
 | ||||
| // returned.
 | ||||
| // This is useful for users who have helper functions between the
 | ||||
| // "real" call site and the actual calls to Logger methods.
 | ||||
| // Implementations that log information about the call site (such as
 | ||||
| // file, function, or line) would otherwise log information about the
 | ||||
| // intermediate helper functions.
 | ||||
| //
 | ||||
| // Callers which care about whether this was supported or not should test for
 | ||||
| // CallDepthLogger support themselves.
 | ||||
| func WithCallDepth(logger Logger, depth int) Logger { | ||||
| 	if decorator, ok := logger.(CallDepthLogger); ok { | ||||
| 		return decorator.WithCallDepth(depth) | ||||
| 	} | ||||
| 	return logger | ||||
| // This is an optional interface and implementations are not required
 | ||||
| // to support it. Implementations that choose to support this must not
 | ||||
| // simply implement it as WithCallDepth(1), because
 | ||||
| // Logger.WithCallStackHelper will call both methods if they are
 | ||||
| // present. This should only be implemented for LogSinks that actually
 | ||||
| // need it, as with testing.T.
 | ||||
| type CallStackHelperLogSink interface { | ||||
| 	// GetCallStackHelper returns a function that must be called
 | ||||
| 	// to mark the direct caller as helper function when logging
 | ||||
| 	// call site information.
 | ||||
| 	GetCallStackHelper() func() | ||||
| } | ||||
| 
 | ||||
| // Marshaler is an optional interface that logged values may choose to
 | ||||
| // implement. Loggers with structured output, such as JSON, should
 | ||||
| // log the object return by the MarshalLog method instead of the
 | ||||
| // original value.
 | ||||
| type Marshaler interface { | ||||
| 	// MarshalLog can be used to:
 | ||||
| 	//   - ensure that structs are not logged as strings when the original
 | ||||
| 	//     value has a String method: return a different type without a
 | ||||
| 	//     String method
 | ||||
| 	//   - select which fields of a complex type should get logged:
 | ||||
| 	//     return a simpler struct with fewer fields
 | ||||
| 	//   - log unexported fields: return a different struct
 | ||||
| 	//     with exported fields
 | ||||
| 	//
 | ||||
| 	// It may return any value of any type.
 | ||||
| 	MarshalLog() interface{} | ||||
| } | ||||
|  |  | |||
|  | @ -1,13 +1,10 @@ | |||
| language: go | ||||
| 
 | ||||
| go: | ||||
|   - 1.4 | ||||
|   - 1.3 | ||||
|   - 1.2 | ||||
|   - tip | ||||
| 
 | ||||
| install: | ||||
|   - if ! go get code.google.com/p/go.tools/cmd/cover; then go get golang.org/x/tools/cmd/cover; fi | ||||
|   - 1.11.x | ||||
|   - 1.12.x | ||||
|   - 1.13.x | ||||
|   - master | ||||
| 
 | ||||
| script: | ||||
|   - go test -cover | ||||
|  |  | |||
|  | @ -1,7 +1,7 @@ | |||
| # How to contribute # | ||||
| 
 | ||||
| We'd love to accept your patches and contributions to this project.  There are | ||||
| a just a few small guidelines you need to follow. | ||||
| just a few small guidelines you need to follow. | ||||
| 
 | ||||
| 
 | ||||
| ## Contributor License Agreement ## | ||||
|  |  | |||
|  | @ -68,4 +68,22 @@ f.Fuzz(&myObject) // Type will correspond to whether A or B info is set. | |||
| 
 | ||||
| See more examples in ```example_test.go```. | ||||
| 
 | ||||
| You can use this library for easier [go-fuzz](https://github.com/dvyukov/go-fuzz)ing. | ||||
| go-fuzz provides the user a byte-slice, which should be converted to different inputs | ||||
| for the tested function. This library can help convert the byte slice. Consider for | ||||
| example a fuzz test for a the function `mypackage.MyFunc` that takes an int arguments: | ||||
| ```go | ||||
| // +build gofuzz | ||||
| package mypackage | ||||
| 
 | ||||
| import fuzz "github.com/google/gofuzz" | ||||
| 
 | ||||
| func Fuzz(data []byte) int { | ||||
|         var i int | ||||
|         fuzz.NewFromGoFuzz(data).Fuzz(&i) | ||||
|         MyFunc(i) | ||||
|         return 0 | ||||
| } | ||||
| ``` | ||||
| 
 | ||||
| Happy testing! | ||||
|  |  | |||
|  | @ -0,0 +1,81 @@ | |||
| /* | ||||
| Copyright 2014 Google Inc. All rights reserved. | ||||
| 
 | ||||
| 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 bytesource provides a rand.Source64 that is determined by a slice of bytes.
 | ||||
| package bytesource | ||||
| 
 | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/binary" | ||||
| 	"io" | ||||
| 	"math/rand" | ||||
| ) | ||||
| 
 | ||||
| // ByteSource implements rand.Source64 determined by a slice of bytes. The random numbers are
 | ||||
| // generated from each 8 bytes in the slice, until the last bytes are consumed, from which a
 | ||||
| // fallback pseudo random source is created in case more random numbers are required.
 | ||||
| // It also exposes a `bytes.Reader` API, which lets callers consume the bytes directly.
 | ||||
| type ByteSource struct { | ||||
| 	*bytes.Reader | ||||
| 	fallback rand.Source | ||||
| } | ||||
| 
 | ||||
| // New returns a new ByteSource from a given slice of bytes.
 | ||||
| func New(input []byte) *ByteSource { | ||||
| 	s := &ByteSource{ | ||||
| 		Reader:   bytes.NewReader(input), | ||||
| 		fallback: rand.NewSource(0), | ||||
| 	} | ||||
| 	if len(input) > 0 { | ||||
| 		s.fallback = rand.NewSource(int64(s.consumeUint64())) | ||||
| 	} | ||||
| 	return s | ||||
| } | ||||
| 
 | ||||
| func (s *ByteSource) Uint64() uint64 { | ||||
| 	// Return from input if it was not exhausted.
 | ||||
| 	if s.Len() > 0 { | ||||
| 		return s.consumeUint64() | ||||
| 	} | ||||
| 
 | ||||
| 	// Input was exhausted, return random number from fallback (in this case fallback should not be
 | ||||
| 	// nil). Try first having a Uint64 output (Should work in current rand implementation),
 | ||||
| 	// otherwise return a conversion of Int63.
 | ||||
| 	if s64, ok := s.fallback.(rand.Source64); ok { | ||||
| 		return s64.Uint64() | ||||
| 	} | ||||
| 	return uint64(s.fallback.Int63()) | ||||
| } | ||||
| 
 | ||||
| func (s *ByteSource) Int63() int64 { | ||||
| 	return int64(s.Uint64() >> 1) | ||||
| } | ||||
| 
 | ||||
| func (s *ByteSource) Seed(seed int64) { | ||||
| 	s.fallback = rand.NewSource(seed) | ||||
| 	s.Reader = bytes.NewReader(nil) | ||||
| } | ||||
| 
 | ||||
| // consumeUint64 reads 8 bytes from the input and convert them to a uint64. It assumes that the the
 | ||||
| // bytes reader is not empty.
 | ||||
| func (s *ByteSource) consumeUint64() uint64 { | ||||
| 	var bytes [8]byte | ||||
| 	_, err := s.Read(bytes[:]) | ||||
| 	if err != nil && err != io.EOF { | ||||
| 		panic("failed reading source") // Should not happen.
 | ||||
| 	} | ||||
| 	return binary.BigEndian.Uint64(bytes[:]) | ||||
| } | ||||
|  | @ -22,6 +22,9 @@ import ( | |||
| 	"reflect" | ||||
| 	"regexp" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/google/gofuzz/bytesource" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| // fuzzFuncMap is a map from a type to a fuzzFunc that handles that type.
 | ||||
|  | @ -61,6 +64,34 @@ func NewWithSeed(seed int64) *Fuzzer { | |||
| 	return f | ||||
| } | ||||
| 
 | ||||
| // NewFromGoFuzz is a helper function that enables using gofuzz (this
 | ||||
| // project) with go-fuzz (https://github.com/dvyukov/go-fuzz) for continuous
 | ||||
| // fuzzing. Essentially, it enables translating the fuzzing bytes from
 | ||||
| // go-fuzz to any Go object using this library.
 | ||||
| //
 | ||||
| // This implementation promises a constant translation from a given slice of
 | ||||
| // bytes to the fuzzed objects. This promise will remain over future
 | ||||
| // versions of Go and of this library.
 | ||||
| //
 | ||||
| // Note: the returned Fuzzer should not be shared between multiple goroutines,
 | ||||
| // as its deterministic output will no longer be available.
 | ||||
| //
 | ||||
| // Example: use go-fuzz to test the function `MyFunc(int)` in the package
 | ||||
| // `mypackage`. Add the file: "mypacakge_fuzz.go" with the content:
 | ||||
| //
 | ||||
| // // +build gofuzz
 | ||||
| // package mypacakge
 | ||||
| // import fuzz "github.com/google/gofuzz"
 | ||||
| // func Fuzz(data []byte) int {
 | ||||
| // 	var i int
 | ||||
| // 	fuzz.NewFromGoFuzz(data).Fuzz(&i)
 | ||||
| // 	MyFunc(i)
 | ||||
| // 	return 0
 | ||||
| // }
 | ||||
| func NewFromGoFuzz(data []byte) *Fuzzer { | ||||
| 	return New().RandSource(bytesource.New(data)) | ||||
| } | ||||
| 
 | ||||
| // Funcs adds each entry in fuzzFuncs as a custom fuzzing function.
 | ||||
| //
 | ||||
| // Each entry in fuzzFuncs must be a function taking two parameters.
 | ||||
|  | @ -141,7 +172,7 @@ func (f *Fuzzer) genElementCount() int { | |||
| } | ||||
| 
 | ||||
| func (f *Fuzzer) genShouldFill() bool { | ||||
| 	return f.r.Float64() > f.nilChance | ||||
| 	return f.r.Float64() >= f.nilChance | ||||
| } | ||||
| 
 | ||||
| // MaxDepth sets the maximum number of recursive fuzz calls that will be made
 | ||||
|  | @ -240,6 +271,7 @@ func (fc *fuzzerContext) doFuzz(v reflect.Value, flags uint64) { | |||
| 		fn(v, fc.fuzzer.r) | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	switch v.Kind() { | ||||
| 	case reflect.Map: | ||||
| 		if fc.fuzzer.genShouldFill() { | ||||
|  | @ -450,10 +482,10 @@ var fillFuncMap = map[reflect.Kind]func(reflect.Value, *rand.Rand){ | |||
| 		v.SetFloat(r.Float64()) | ||||
| 	}, | ||||
| 	reflect.Complex64: func(v reflect.Value, r *rand.Rand) { | ||||
| 		panic("unimplemented") | ||||
| 		v.SetComplex(complex128(complex(r.Float32(), r.Float32()))) | ||||
| 	}, | ||||
| 	reflect.Complex128: func(v reflect.Value, r *rand.Rand) { | ||||
| 		panic("unimplemented") | ||||
| 		v.SetComplex(complex(r.Float64(), r.Float64())) | ||||
| 	}, | ||||
| 	reflect.String: func(v reflect.Value, r *rand.Rand) { | ||||
| 		v.SetString(randString(r)) | ||||
|  | @ -465,38 +497,105 @@ var fillFuncMap = map[reflect.Kind]func(reflect.Value, *rand.Rand){ | |||
| 
 | ||||
| // randBool returns true or false randomly.
 | ||||
| func randBool(r *rand.Rand) bool { | ||||
| 	if r.Int()&1 == 1 { | ||||
| 		return true | ||||
| 	} | ||||
| 	return false | ||||
| 	return r.Int31()&(1<<30) == 0 | ||||
| } | ||||
| 
 | ||||
| type charRange struct { | ||||
| 	first, last rune | ||||
| type int63nPicker interface { | ||||
| 	Int63n(int64) int64 | ||||
| } | ||||
| 
 | ||||
| // UnicodeRange describes a sequential range of unicode characters.
 | ||||
| // Last must be numerically greater than First.
 | ||||
| type UnicodeRange struct { | ||||
| 	First, Last rune | ||||
| } | ||||
| 
 | ||||
| // UnicodeRanges describes an arbitrary number of sequential ranges of unicode characters.
 | ||||
| // To be useful, each range must have at least one character (First <= Last) and
 | ||||
| // there must be at least one range.
 | ||||
| type UnicodeRanges []UnicodeRange | ||||
| 
 | ||||
| // choose returns a random unicode character from the given range, using the
 | ||||
| // given randomness source.
 | ||||
| func (r *charRange) choose(rand *rand.Rand) rune { | ||||
| 	count := int64(r.last - r.first) | ||||
| 	return r.first + rune(rand.Int63n(count)) | ||||
| func (ur UnicodeRange) choose(r int63nPicker) rune { | ||||
| 	count := int64(ur.Last - ur.First + 1) | ||||
| 	return ur.First + rune(r.Int63n(count)) | ||||
| } | ||||
| 
 | ||||
| var unicodeRanges = []charRange{ | ||||
| // CustomStringFuzzFunc constructs a FuzzFunc which produces random strings.
 | ||||
| // Each character is selected from the range ur. If there are no characters
 | ||||
| // in the range (cr.Last < cr.First), this will panic.
 | ||||
| func (ur UnicodeRange) CustomStringFuzzFunc() func(s *string, c Continue) { | ||||
| 	ur.check() | ||||
| 	return func(s *string, c Continue) { | ||||
| 		*s = ur.randString(c.Rand) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // check is a function that used to check whether the first of ur(UnicodeRange)
 | ||||
| // is greater than the last one.
 | ||||
| func (ur UnicodeRange) check() { | ||||
| 	if ur.Last < ur.First { | ||||
| 		panic("The last encoding must be greater than the first one.") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // randString of UnicodeRange makes a random string up to 20 characters long.
 | ||||
| // Each character is selected form ur(UnicodeRange).
 | ||||
| func (ur UnicodeRange) randString(r *rand.Rand) string { | ||||
| 	n := r.Intn(20) | ||||
| 	sb := strings.Builder{} | ||||
| 	sb.Grow(n) | ||||
| 	for i := 0; i < n; i++ { | ||||
| 		sb.WriteRune(ur.choose(r)) | ||||
| 	} | ||||
| 	return sb.String() | ||||
| } | ||||
| 
 | ||||
| // defaultUnicodeRanges sets a default unicode range when user do not set
 | ||||
| // CustomStringFuzzFunc() but wants fuzz string.
 | ||||
| var defaultUnicodeRanges = UnicodeRanges{ | ||||
| 	{' ', '~'},           // ASCII characters
 | ||||
| 	{'\u00a0', '\u02af'}, // Multi-byte encoded characters
 | ||||
| 	{'\u4e00', '\u9fff'}, // Common CJK (even longer encodings)
 | ||||
| } | ||||
| 
 | ||||
| // CustomStringFuzzFunc constructs a FuzzFunc which produces random strings.
 | ||||
| // Each character is selected from one of the ranges of ur(UnicodeRanges).
 | ||||
| // Each range has an equal probability of being chosen. If there are no ranges,
 | ||||
| // or a selected range has no characters (.Last < .First), this will panic.
 | ||||
| // Do not modify any of the ranges in ur after calling this function.
 | ||||
| func (ur UnicodeRanges) CustomStringFuzzFunc() func(s *string, c Continue) { | ||||
| 	// Check unicode ranges slice is empty.
 | ||||
| 	if len(ur) == 0 { | ||||
| 		panic("UnicodeRanges is empty.") | ||||
| 	} | ||||
| 	// if not empty, each range should be checked.
 | ||||
| 	for i := range ur { | ||||
| 		ur[i].check() | ||||
| 	} | ||||
| 	return func(s *string, c Continue) { | ||||
| 		*s = ur.randString(c.Rand) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // randString of UnicodeRanges makes a random string up to 20 characters long.
 | ||||
| // Each character is selected form one of the ranges of ur(UnicodeRanges),
 | ||||
| // and each range has an equal probability of being chosen.
 | ||||
| func (ur UnicodeRanges) randString(r *rand.Rand) string { | ||||
| 	n := r.Intn(20) | ||||
| 	sb := strings.Builder{} | ||||
| 	sb.Grow(n) | ||||
| 	for i := 0; i < n; i++ { | ||||
| 		sb.WriteRune(ur[r.Intn(len(ur))].choose(r)) | ||||
| 	} | ||||
| 	return sb.String() | ||||
| } | ||||
| 
 | ||||
| // randString makes a random string up to 20 characters long. The returned string
 | ||||
| // may include a variety of (valid) UTF-8 encodings.
 | ||||
| func randString(r *rand.Rand) string { | ||||
| 	n := r.Intn(20) | ||||
| 	runes := make([]rune, n) | ||||
| 	for i := range runes { | ||||
| 		runes[i] = unicodeRanges[r.Intn(len(unicodeRanges))].choose(r) | ||||
| 	} | ||||
| 	return string(runes) | ||||
| 	return defaultUnicodeRanges.randString(r) | ||||
| } | ||||
| 
 | ||||
| // randUint64 makes random 64 bit numbers.
 | ||||
|  |  | |||
|  | @ -44,6 +44,8 @@ func Wrap(outer, inner error) error { | |||
| //
 | ||||
| // format is the format of the error message. The string '{{err}}' will
 | ||||
| // be replaced with the original error message.
 | ||||
| //
 | ||||
| // Deprecated: Use fmt.Errorf()
 | ||||
| func Wrapf(format string, err error) error { | ||||
| 	outerMsg := "<nil>" | ||||
| 	if err != nil { | ||||
|  | @ -148,6 +150,9 @@ func Walk(err error, cb WalkFunc) { | |||
| 		for _, err := range e.WrappedErrors() { | ||||
| 			Walk(err, cb) | ||||
| 		} | ||||
| 	case interface{ Unwrap() error }: | ||||
| 		cb(err) | ||||
| 		Walk(e.Unwrap(), cb) | ||||
| 	default: | ||||
| 		cb(err) | ||||
| 	} | ||||
|  | @ -167,3 +172,7 @@ func (w *wrappedError) Error() string { | |||
| func (w *wrappedError) WrappedErrors() []error { | ||||
| 	return []error{w.Outer, w.Inner} | ||||
| } | ||||
| 
 | ||||
| func (w *wrappedError) Unwrap() error { | ||||
| 	return w.Inner | ||||
| } | ||||
|  |  | |||
|  | @ -1,29 +0,0 @@ | |||
| // Copyright 2019 The Prometheus 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.
 | ||||
| 
 | ||||
| // +build go1.12
 | ||||
| 
 | ||||
| package prometheus | ||||
| 
 | ||||
| import "runtime/debug" | ||||
| 
 | ||||
| // readBuildInfo is a wrapper around debug.ReadBuildInfo for Go 1.12+.
 | ||||
| func readBuildInfo() (path, version, sum string) { | ||||
| 	path, version, sum = "unknown", "unknown", "unknown" | ||||
| 	if bi, ok := debug.ReadBuildInfo(); ok { | ||||
| 		path = bi.Main.Path | ||||
| 		version = bi.Main.Version | ||||
| 		sum = bi.Main.Sum | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | @ -163,7 +163,7 @@ func (c *counter) updateExemplar(v float64, l Labels) { | |||
| // (e.g. number of HTTP requests, partitioned by response code and
 | ||||
| // method). Create instances with NewCounterVec.
 | ||||
| type CounterVec struct { | ||||
| 	*metricVec | ||||
| 	*MetricVec | ||||
| } | ||||
| 
 | ||||
| // NewCounterVec creates a new CounterVec based on the provided CounterOpts and
 | ||||
|  | @ -176,11 +176,11 @@ func NewCounterVec(opts CounterOpts, labelNames []string) *CounterVec { | |||
| 		opts.ConstLabels, | ||||
| 	) | ||||
| 	return &CounterVec{ | ||||
| 		metricVec: newMetricVec(desc, func(lvs ...string) Metric { | ||||
| 		MetricVec: NewMetricVec(desc, func(lvs ...string) Metric { | ||||
| 			if len(lvs) != len(desc.variableLabels) { | ||||
| 				panic(makeInconsistentCardinalityError(desc.fqName, desc.variableLabels, lvs)) | ||||
| 			} | ||||
| 			result := &counter{desc: desc, labelPairs: makeLabelPairs(desc, lvs), now: time.Now} | ||||
| 			result := &counter{desc: desc, labelPairs: MakeLabelPairs(desc, lvs), now: time.Now} | ||||
| 			result.init(result) // Init self-collection.
 | ||||
| 			return result | ||||
| 		}), | ||||
|  | @ -188,7 +188,7 @@ func NewCounterVec(opts CounterOpts, labelNames []string) *CounterVec { | |||
| } | ||||
| 
 | ||||
| // GetMetricWithLabelValues returns the Counter for the given slice of label
 | ||||
| // values (same order as the VariableLabels in Desc). If that combination of
 | ||||
| // values (same order as the variable labels in Desc). If that combination of
 | ||||
| // label values is accessed for the first time, a new Counter is created.
 | ||||
| //
 | ||||
| // It is possible to call this method without using the returned Counter to only
 | ||||
|  | @ -202,7 +202,7 @@ func NewCounterVec(opts CounterOpts, labelNames []string) *CounterVec { | |||
| // Counter with the same label values is created later.
 | ||||
| //
 | ||||
| // An error is returned if the number of label values is not the same as the
 | ||||
| // number of VariableLabels in Desc (minus any curried labels).
 | ||||
| // number of variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // Note that for more than one label value, this method is prone to mistakes
 | ||||
| // caused by an incorrect order of arguments. Consider GetMetricWith(Labels) as
 | ||||
|  | @ -211,7 +211,7 @@ func NewCounterVec(opts CounterOpts, labelNames []string) *CounterVec { | |||
| // with a performance overhead (for creating and processing the Labels map).
 | ||||
| // See also the GaugeVec example.
 | ||||
| func (v *CounterVec) GetMetricWithLabelValues(lvs ...string) (Counter, error) { | ||||
| 	metric, err := v.metricVec.getMetricWithLabelValues(lvs...) | ||||
| 	metric, err := v.MetricVec.GetMetricWithLabelValues(lvs...) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Counter), err | ||||
| 	} | ||||
|  | @ -219,19 +219,19 @@ func (v *CounterVec) GetMetricWithLabelValues(lvs ...string) (Counter, error) { | |||
| } | ||||
| 
 | ||||
| // GetMetricWith returns the Counter for the given Labels map (the label names
 | ||||
| // must match those of the VariableLabels in Desc). If that label map is
 | ||||
| // must match those of the variable labels in Desc). If that label map is
 | ||||
| // accessed for the first time, a new Counter is created. Implications of
 | ||||
| // creating a Counter without using it and keeping the Counter for later use are
 | ||||
| // the same as for GetMetricWithLabelValues.
 | ||||
| //
 | ||||
| // An error is returned if the number and names of the Labels are inconsistent
 | ||||
| // with those of the VariableLabels in Desc (minus any curried labels).
 | ||||
| // with those of the variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // This method is used for the same purpose as
 | ||||
| // GetMetricWithLabelValues(...string). See there for pros and cons of the two
 | ||||
| // methods.
 | ||||
| func (v *CounterVec) GetMetricWith(labels Labels) (Counter, error) { | ||||
| 	metric, err := v.metricVec.getMetricWith(labels) | ||||
| 	metric, err := v.MetricVec.GetMetricWith(labels) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Counter), err | ||||
| 	} | ||||
|  | @ -275,7 +275,7 @@ func (v *CounterVec) With(labels Labels) Counter { | |||
| // registered with a given registry (usually the uncurried version). The Reset
 | ||||
| // method deletes all metrics, even if called on a curried vector.
 | ||||
| func (v *CounterVec) CurryWith(labels Labels) (*CounterVec, error) { | ||||
| 	vec, err := v.curryWith(labels) | ||||
| 	vec, err := v.MetricVec.CurryWith(labels) | ||||
| 	if vec != nil { | ||||
| 		return &CounterVec{vec}, err | ||||
| 	} | ||||
|  |  | |||
|  | @ -20,7 +20,7 @@ import ( | |||
| 	"strings" | ||||
| 
 | ||||
| 	"github.com/cespare/xxhash/v2" | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/prometheus/common/model" | ||||
| 
 | ||||
|  | @ -51,7 +51,7 @@ type Desc struct { | |||
| 	// constLabelPairs contains precalculated DTO label pairs based on
 | ||||
| 	// the constant labels.
 | ||||
| 	constLabelPairs []*dto.LabelPair | ||||
| 	// VariableLabels contains names of labels for which the metric
 | ||||
| 	// variableLabels contains names of labels for which the metric
 | ||||
| 	// maintains variable values.
 | ||||
| 	variableLabels []string | ||||
| 	// id is a hash of the values of the ConstLabels and fqName. This
 | ||||
|  |  | |||
|  | @ -22,43 +22,10 @@ type expvarCollector struct { | |||
| 	exports map[string]*Desc | ||||
| } | ||||
| 
 | ||||
| // NewExpvarCollector returns a newly allocated expvar Collector that still has
 | ||||
| // to be registered with a Prometheus registry.
 | ||||
| // NewExpvarCollector is the obsolete version of collectors.NewExpvarCollector.
 | ||||
| // See there for documentation.
 | ||||
| //
 | ||||
| // An expvar Collector collects metrics from the expvar interface. It provides a
 | ||||
| // quick way to expose numeric values that are already exported via expvar as
 | ||||
| // Prometheus metrics. Note that the data models of expvar and Prometheus are
 | ||||
| // fundamentally different, and that the expvar Collector is inherently slower
 | ||||
| // than native Prometheus metrics. Thus, the expvar Collector is probably great
 | ||||
| // for experiments and prototying, but you should seriously consider a more
 | ||||
| // direct implementation of Prometheus metrics for monitoring production
 | ||||
| // systems.
 | ||||
| //
 | ||||
| // The exports map has the following meaning:
 | ||||
| //
 | ||||
| // The keys in the map correspond to expvar keys, i.e. for every expvar key you
 | ||||
| // want to export as Prometheus metric, you need an entry in the exports
 | ||||
| // map. The descriptor mapped to each key describes how to export the expvar
 | ||||
| // value. It defines the name and the help string of the Prometheus metric
 | ||||
| // proxying the expvar value. The type will always be Untyped.
 | ||||
| //
 | ||||
| // For descriptors without variable labels, the expvar value must be a number or
 | ||||
| // a bool. The number is then directly exported as the Prometheus sample
 | ||||
| // value. (For a bool, 'false' translates to 0 and 'true' to 1). Expvar values
 | ||||
| // that are not numbers or bools are silently ignored.
 | ||||
| //
 | ||||
| // If the descriptor has one variable label, the expvar value must be an expvar
 | ||||
| // map. The keys in the expvar map become the various values of the one
 | ||||
| // Prometheus label. The values in the expvar map must be numbers or bools again
 | ||||
| // as above.
 | ||||
| //
 | ||||
| // For descriptors with more than one variable label, the expvar must be a
 | ||||
| // nested expvar map, i.e. where the values of the topmost map are maps again
 | ||||
| // etc. until a depth is reached that corresponds to the number of labels. The
 | ||||
| // leaves of that structure must be numbers or bools as above to serve as the
 | ||||
| // sample values.
 | ||||
| //
 | ||||
| // Anything that does not fit into the scheme above is silently ignored.
 | ||||
| // Deprecated: Use collectors.NewExpvarCollector instead.
 | ||||
| func NewExpvarCollector(exports map[string]*Desc) Collector { | ||||
| 	return &expvarCollector{ | ||||
| 		exports: exports, | ||||
|  |  | |||
|  | @ -132,7 +132,7 @@ func (g *gauge) Write(out *dto.Metric) error { | |||
| // (e.g. number of operations queued, partitioned by user and operation
 | ||||
| // type). Create instances with NewGaugeVec.
 | ||||
| type GaugeVec struct { | ||||
| 	*metricVec | ||||
| 	*MetricVec | ||||
| } | ||||
| 
 | ||||
| // NewGaugeVec creates a new GaugeVec based on the provided GaugeOpts and
 | ||||
|  | @ -145,11 +145,11 @@ func NewGaugeVec(opts GaugeOpts, labelNames []string) *GaugeVec { | |||
| 		opts.ConstLabels, | ||||
| 	) | ||||
| 	return &GaugeVec{ | ||||
| 		metricVec: newMetricVec(desc, func(lvs ...string) Metric { | ||||
| 		MetricVec: NewMetricVec(desc, func(lvs ...string) Metric { | ||||
| 			if len(lvs) != len(desc.variableLabels) { | ||||
| 				panic(makeInconsistentCardinalityError(desc.fqName, desc.variableLabels, lvs)) | ||||
| 			} | ||||
| 			result := &gauge{desc: desc, labelPairs: makeLabelPairs(desc, lvs)} | ||||
| 			result := &gauge{desc: desc, labelPairs: MakeLabelPairs(desc, lvs)} | ||||
| 			result.init(result) // Init self-collection.
 | ||||
| 			return result | ||||
| 		}), | ||||
|  | @ -157,7 +157,7 @@ func NewGaugeVec(opts GaugeOpts, labelNames []string) *GaugeVec { | |||
| } | ||||
| 
 | ||||
| // GetMetricWithLabelValues returns the Gauge for the given slice of label
 | ||||
| // values (same order as the VariableLabels in Desc). If that combination of
 | ||||
| // values (same order as the variable labels in Desc). If that combination of
 | ||||
| // label values is accessed for the first time, a new Gauge is created.
 | ||||
| //
 | ||||
| // It is possible to call this method without using the returned Gauge to only
 | ||||
|  | @ -172,7 +172,7 @@ func NewGaugeVec(opts GaugeOpts, labelNames []string) *GaugeVec { | |||
| // example.
 | ||||
| //
 | ||||
| // An error is returned if the number of label values is not the same as the
 | ||||
| // number of VariableLabels in Desc (minus any curried labels).
 | ||||
| // number of variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // Note that for more than one label value, this method is prone to mistakes
 | ||||
| // caused by an incorrect order of arguments. Consider GetMetricWith(Labels) as
 | ||||
|  | @ -180,7 +180,7 @@ func NewGaugeVec(opts GaugeOpts, labelNames []string) *GaugeVec { | |||
| // latter has a much more readable (albeit more verbose) syntax, but it comes
 | ||||
| // with a performance overhead (for creating and processing the Labels map).
 | ||||
| func (v *GaugeVec) GetMetricWithLabelValues(lvs ...string) (Gauge, error) { | ||||
| 	metric, err := v.metricVec.getMetricWithLabelValues(lvs...) | ||||
| 	metric, err := v.MetricVec.GetMetricWithLabelValues(lvs...) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Gauge), err | ||||
| 	} | ||||
|  | @ -188,19 +188,19 @@ func (v *GaugeVec) GetMetricWithLabelValues(lvs ...string) (Gauge, error) { | |||
| } | ||||
| 
 | ||||
| // GetMetricWith returns the Gauge for the given Labels map (the label names
 | ||||
| // must match those of the VariableLabels in Desc). If that label map is
 | ||||
| // must match those of the variable labels in Desc). If that label map is
 | ||||
| // accessed for the first time, a new Gauge is created. Implications of
 | ||||
| // creating a Gauge without using it and keeping the Gauge for later use are
 | ||||
| // the same as for GetMetricWithLabelValues.
 | ||||
| //
 | ||||
| // An error is returned if the number and names of the Labels are inconsistent
 | ||||
| // with those of the VariableLabels in Desc (minus any curried labels).
 | ||||
| // with those of the variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // This method is used for the same purpose as
 | ||||
| // GetMetricWithLabelValues(...string). See there for pros and cons of the two
 | ||||
| // methods.
 | ||||
| func (v *GaugeVec) GetMetricWith(labels Labels) (Gauge, error) { | ||||
| 	metric, err := v.metricVec.getMetricWith(labels) | ||||
| 	metric, err := v.MetricVec.GetMetricWith(labels) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Gauge), err | ||||
| 	} | ||||
|  | @ -244,7 +244,7 @@ func (v *GaugeVec) With(labels Labels) Gauge { | |||
| // registered with a given registry (usually the uncurried version). The Reset
 | ||||
| // method deletes all metrics, even if called on a curried vector.
 | ||||
| func (v *GaugeVec) CurryWith(labels Labels) (*GaugeVec, error) { | ||||
| 	vec, err := v.curryWith(labels) | ||||
| 	vec, err := v.MetricVec.CurryWith(labels) | ||||
| 	if vec != nil { | ||||
| 		return &GaugeVec{vec}, err | ||||
| 	} | ||||
|  |  | |||
|  | @ -36,31 +36,10 @@ type goCollector struct { | |||
| 	msMaxAge        time.Duration           // Maximum allowed age of old memstats.
 | ||||
| } | ||||
| 
 | ||||
| // NewGoCollector returns a collector that exports metrics about the current Go
 | ||||
| // process. This includes memory stats. To collect those, runtime.ReadMemStats
 | ||||
| // is called. This requires to “stop the world”, which usually only happens for
 | ||||
| // garbage collection (GC). Take the following implications into account when
 | ||||
| // deciding whether to use the Go collector:
 | ||||
| // NewGoCollector is the obsolete version of collectors.NewGoCollector.
 | ||||
| // See there for documentation.
 | ||||
| //
 | ||||
| // 1. The performance impact of stopping the world is the more relevant the more
 | ||||
| // frequently metrics are collected. However, with Go1.9 or later the
 | ||||
| // stop-the-world time per metrics collection is very short (~25µs) so that the
 | ||||
| // performance impact will only matter in rare cases. However, with older Go
 | ||||
| // versions, the stop-the-world duration depends on the heap size and can be
 | ||||
| // quite significant (~1.7 ms/GiB as per
 | ||||
| // https://go-review.googlesource.com/c/go/+/34937).
 | ||||
| //
 | ||||
| // 2. During an ongoing GC, nothing else can stop the world. Therefore, if the
 | ||||
| // metrics collection happens to coincide with GC, it will only complete after
 | ||||
| // GC has finished. Usually, GC is fast enough to not cause problems. However,
 | ||||
| // with a very large heap, GC might take multiple seconds, which is enough to
 | ||||
| // cause scrape timeouts in common setups. To avoid this problem, the Go
 | ||||
| // collector will use the memstats from a previous collection if
 | ||||
| // runtime.ReadMemStats takes more than 1s. However, if there are no previously
 | ||||
| // collected memstats, or their collection is more than 5m ago, the collection
 | ||||
| // will block until runtime.ReadMemStats succeeds. (The problem might be solved
 | ||||
| // in Go1.13, see https://github.com/golang/go/issues/19812 for the related Go
 | ||||
| // issue.)
 | ||||
| // Deprecated: Use collectors.NewGoCollector instead.
 | ||||
| func NewGoCollector() Collector { | ||||
| 	return &goCollector{ | ||||
| 		goroutinesDesc: NewDesc( | ||||
|  | @ -365,25 +344,17 @@ type memStatsMetrics []struct { | |||
| 	valType ValueType | ||||
| } | ||||
| 
 | ||||
| // NewBuildInfoCollector returns a collector collecting a single metric
 | ||||
| // "go_build_info" with the constant value 1 and three labels "path", "version",
 | ||||
| // and "checksum". Their label values contain the main module path, version, and
 | ||||
| // checksum, respectively. The labels will only have meaningful values if the
 | ||||
| // binary is built with Go module support and from source code retrieved from
 | ||||
| // the source repository (rather than the local file system). This is usually
 | ||||
| // accomplished by building from outside of GOPATH, specifying the full address
 | ||||
| // of the main package, e.g. "GO111MODULE=on go run
 | ||||
| // github.com/prometheus/client_golang/examples/random". If built without Go
 | ||||
| // module support, all label values will be "unknown". If built with Go module
 | ||||
| // support but using the source code from the local file system, the "path" will
 | ||||
| // be set appropriately, but "checksum" will be empty and "version" will be
 | ||||
| // "(devel)".
 | ||||
| // NewBuildInfoCollector is the obsolete version of collectors.NewBuildInfoCollector.
 | ||||
| // See there for documentation.
 | ||||
| //
 | ||||
| // This collector uses only the build information for the main module. See
 | ||||
| // https://github.com/povilasv/prommod for an example of a collector for the
 | ||||
| // module dependencies.
 | ||||
| // Deprecated: Use collectors.NewBuildInfoCollector instead.
 | ||||
| func NewBuildInfoCollector() Collector { | ||||
| 	path, version, sum := readBuildInfo() | ||||
| 	path, version, sum := "unknown", "unknown", "unknown" | ||||
| 	if bi, ok := debug.ReadBuildInfo(); ok { | ||||
| 		path = bi.Main.Path | ||||
| 		version = bi.Main.Version | ||||
| 		sum = bi.Main.Sum | ||||
| 	} | ||||
| 	c := &selfCollector{MustNewConstMetric( | ||||
| 		NewDesc( | ||||
| 			"go_build_info", | ||||
|  |  | |||
|  | @ -22,7 +22,7 @@ import ( | |||
| 	"sync/atomic" | ||||
| 	"time" | ||||
| 
 | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 
 | ||||
| 	dto "github.com/prometheus/client_model/go" | ||||
|  | @ -47,7 +47,12 @@ type Histogram interface { | |||
| 	Metric | ||||
| 	Collector | ||||
| 
 | ||||
| 	// Observe adds a single observation to the histogram.
 | ||||
| 	// Observe adds a single observation to the histogram. Observations are
 | ||||
| 	// usually positive or zero. Negative observations are accepted but
 | ||||
| 	// prevent current versions of Prometheus from properly detecting
 | ||||
| 	// counter resets in the sum of observations. See
 | ||||
| 	// https://prometheus.io/docs/practices/histograms/#count-and-sum-of-observations
 | ||||
| 	// for details.
 | ||||
| 	Observe(float64) | ||||
| } | ||||
| 
 | ||||
|  | @ -192,7 +197,7 @@ func newHistogram(desc *Desc, opts HistogramOpts, labelValues ...string) Histogr | |||
| 	h := &histogram{ | ||||
| 		desc:        desc, | ||||
| 		upperBounds: opts.Buckets, | ||||
| 		labelPairs:  makeLabelPairs(desc, labelValues), | ||||
| 		labelPairs:  MakeLabelPairs(desc, labelValues), | ||||
| 		counts:      [2]*histogramCounts{{}, {}}, | ||||
| 		now:         time.Now, | ||||
| 	} | ||||
|  | @ -409,7 +414,7 @@ func (h *histogram) updateExemplar(v float64, bucket int, l Labels) { | |||
| // (e.g. HTTP request latencies, partitioned by status code and method). Create
 | ||||
| // instances with NewHistogramVec.
 | ||||
| type HistogramVec struct { | ||||
| 	*metricVec | ||||
| 	*MetricVec | ||||
| } | ||||
| 
 | ||||
| // NewHistogramVec creates a new HistogramVec based on the provided HistogramOpts and
 | ||||
|  | @ -422,14 +427,14 @@ func NewHistogramVec(opts HistogramOpts, labelNames []string) *HistogramVec { | |||
| 		opts.ConstLabels, | ||||
| 	) | ||||
| 	return &HistogramVec{ | ||||
| 		metricVec: newMetricVec(desc, func(lvs ...string) Metric { | ||||
| 		MetricVec: NewMetricVec(desc, func(lvs ...string) Metric { | ||||
| 			return newHistogram(desc, opts, lvs...) | ||||
| 		}), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // GetMetricWithLabelValues returns the Histogram for the given slice of label
 | ||||
| // values (same order as the VariableLabels in Desc). If that combination of
 | ||||
| // values (same order as the variable labels in Desc). If that combination of
 | ||||
| // label values is accessed for the first time, a new Histogram is created.
 | ||||
| //
 | ||||
| // It is possible to call this method without using the returned Histogram to only
 | ||||
|  | @ -444,7 +449,7 @@ func NewHistogramVec(opts HistogramOpts, labelNames []string) *HistogramVec { | |||
| // example.
 | ||||
| //
 | ||||
| // An error is returned if the number of label values is not the same as the
 | ||||
| // number of VariableLabels in Desc (minus any curried labels).
 | ||||
| // number of variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // Note that for more than one label value, this method is prone to mistakes
 | ||||
| // caused by an incorrect order of arguments. Consider GetMetricWith(Labels) as
 | ||||
|  | @ -453,7 +458,7 @@ func NewHistogramVec(opts HistogramOpts, labelNames []string) *HistogramVec { | |||
| // with a performance overhead (for creating and processing the Labels map).
 | ||||
| // See also the GaugeVec example.
 | ||||
| func (v *HistogramVec) GetMetricWithLabelValues(lvs ...string) (Observer, error) { | ||||
| 	metric, err := v.metricVec.getMetricWithLabelValues(lvs...) | ||||
| 	metric, err := v.MetricVec.GetMetricWithLabelValues(lvs...) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Observer), err | ||||
| 	} | ||||
|  | @ -461,19 +466,19 @@ func (v *HistogramVec) GetMetricWithLabelValues(lvs ...string) (Observer, error) | |||
| } | ||||
| 
 | ||||
| // GetMetricWith returns the Histogram for the given Labels map (the label names
 | ||||
| // must match those of the VariableLabels in Desc). If that label map is
 | ||||
| // must match those of the variable labels in Desc). If that label map is
 | ||||
| // accessed for the first time, a new Histogram is created. Implications of
 | ||||
| // creating a Histogram without using it and keeping the Histogram for later use
 | ||||
| // are the same as for GetMetricWithLabelValues.
 | ||||
| //
 | ||||
| // An error is returned if the number and names of the Labels are inconsistent
 | ||||
| // with those of the VariableLabels in Desc (minus any curried labels).
 | ||||
| // with those of the variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // This method is used for the same purpose as
 | ||||
| // GetMetricWithLabelValues(...string). See there for pros and cons of the two
 | ||||
| // methods.
 | ||||
| func (v *HistogramVec) GetMetricWith(labels Labels) (Observer, error) { | ||||
| 	metric, err := v.metricVec.getMetricWith(labels) | ||||
| 	metric, err := v.MetricVec.GetMetricWith(labels) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Observer), err | ||||
| 	} | ||||
|  | @ -517,7 +522,7 @@ func (v *HistogramVec) With(labels Labels) Observer { | |||
| // registered with a given registry (usually the uncurried version). The Reset
 | ||||
| // method deletes all metrics, even if called on a curried vector.
 | ||||
| func (v *HistogramVec) CurryWith(labels Labels) (ObserverVec, error) { | ||||
| 	vec, err := v.curryWith(labels) | ||||
| 	vec, err := v.MetricVec.CurryWith(labels) | ||||
| 	if vec != nil { | ||||
| 		return &HistogramVec{vec}, err | ||||
| 	} | ||||
|  | @ -602,7 +607,7 @@ func NewConstHistogram( | |||
| 		count:      count, | ||||
| 		sum:        sum, | ||||
| 		buckets:    buckets, | ||||
| 		labelPairs: makeLabelPairs(desc, labelValues), | ||||
| 		labelPairs: MakeLabelPairs(desc, labelValues), | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -17,7 +17,7 @@ import ( | |||
| 	"strings" | ||||
| 	"time" | ||||
| 
 | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/prometheus/common/model" | ||||
| 
 | ||||
|  | @ -58,7 +58,7 @@ type Metric interface { | |||
| } | ||||
| 
 | ||||
| // Opts bundles the options for creating most Metric types. Each metric
 | ||||
| // implementation XXX has its own XXXOpts type, but in most cases, it is just be
 | ||||
| // implementation XXX has its own XXXOpts type, but in most cases, it is just
 | ||||
| // an alias of this type (which might change when the requirement arises.)
 | ||||
| //
 | ||||
| // It is mandatory to set Name to a non-empty string. All other fields are
 | ||||
|  | @ -89,7 +89,7 @@ type Opts struct { | |||
| 	// better covered by target labels set by the scraping Prometheus
 | ||||
| 	// server, or by one specific metric (e.g. a build_info or a
 | ||||
| 	// machine_role metric). See also
 | ||||
| 	// https://prometheus.io/docs/instrumenting/writing_exporters/#target-labels,-not-static-scraped-labels
 | ||||
| 	// https://prometheus.io/docs/instrumenting/writing_exporters/#target-labels-not-static-scraped-labels
 | ||||
| 	ConstLabels Labels | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -15,7 +15,11 @@ package prometheus | |||
| 
 | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| type processCollector struct { | ||||
|  | @ -50,16 +54,10 @@ type ProcessCollectorOpts struct { | |||
| 	ReportErrors bool | ||||
| } | ||||
| 
 | ||||
| // NewProcessCollector returns a collector which exports the current state of
 | ||||
| // process metrics including CPU, memory and file descriptor usage as well as
 | ||||
| // the process start time. The detailed behavior is defined by the provided
 | ||||
| // ProcessCollectorOpts. The zero value of ProcessCollectorOpts creates a
 | ||||
| // collector for the current process with an empty namespace string and no error
 | ||||
| // reporting.
 | ||||
| // NewProcessCollector is the obsolete version of collectors.NewProcessCollector.
 | ||||
| // See there for documentation.
 | ||||
| //
 | ||||
| // The collector only works on operating systems with a Linux-style proc
 | ||||
| // filesystem and on Microsoft Windows. On other operating systems, it will not
 | ||||
| // collect any metrics.
 | ||||
| // Deprecated: Use collectors.NewProcessCollector instead.
 | ||||
| func NewProcessCollector(opts ProcessCollectorOpts) Collector { | ||||
| 	ns := "" | ||||
| 	if len(opts.Namespace) > 0 { | ||||
|  | @ -149,3 +147,20 @@ func (c *processCollector) reportError(ch chan<- Metric, desc *Desc, err error) | |||
| 	} | ||||
| 	ch <- NewInvalidMetric(desc, err) | ||||
| } | ||||
| 
 | ||||
| // NewPidFileFn returns a function that retrieves a pid from the specified file.
 | ||||
| // It is meant to be used for the PidFn field in ProcessCollectorOpts.
 | ||||
| func NewPidFileFn(pidFilePath string) func() (int, error) { | ||||
| 	return func() (int, error) { | ||||
| 		content, err := ioutil.ReadFile(pidFilePath) | ||||
| 		if err != nil { | ||||
| 			return 0, fmt.Errorf("can't read pid file %q: %+v", pidFilePath, err) | ||||
| 		} | ||||
| 		pid, err := strconv.Atoi(strings.TrimSpace(string(content))) | ||||
| 		if err != nil { | ||||
| 			return 0, fmt.Errorf("can't parse pid file %q: %+v", pidFilePath, err) | ||||
| 		} | ||||
| 
 | ||||
| 		return pid, nil | ||||
| 	} | ||||
| } | ||||
|  |  | |||
|  | @ -83,8 +83,7 @@ type readerFromDelegator struct{ *responseWriterDelegator } | |||
| type pusherDelegator struct{ *responseWriterDelegator } | ||||
| 
 | ||||
| func (d closeNotifierDelegator) CloseNotify() <-chan bool { | ||||
| 	//lint:ignore SA1019 http.CloseNotifier is deprecated but we don't want to
 | ||||
| 	//remove support from client_golang yet.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. http.CloseNotifier is deprecated but we keep it here to not break existing users.
 | ||||
| 	return d.ResponseWriter.(http.CloseNotifier).CloseNotify() | ||||
| } | ||||
| func (d flusherDelegator) Flush() { | ||||
|  | @ -348,8 +347,7 @@ func newDelegator(w http.ResponseWriter, observeWriteHeaderFunc func(int)) deleg | |||
| 	} | ||||
| 
 | ||||
| 	id := 0 | ||||
| 	//lint:ignore SA1019 http.CloseNotifier is deprecated but we don't want to
 | ||||
| 	//remove support from client_golang yet.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. http.CloseNotifier is deprecated but we keep it here to not break existing users.
 | ||||
| 	if _, ok := w.(http.CloseNotifier); ok { | ||||
| 		id += closeNotifier | ||||
| 	} | ||||
|  |  | |||
|  | @ -99,7 +99,7 @@ func HandlerFor(reg prometheus.Gatherer, opts HandlerOpts) http.Handler { | |||
| 		inFlightSem = make(chan struct{}, opts.MaxRequestsInFlight) | ||||
| 	} | ||||
| 	if opts.Registry != nil { | ||||
| 		// Initialize all possibilites that can occur below.
 | ||||
| 		// Initialize all possibilities that can occur below.
 | ||||
| 		errCnt.WithLabelValues("gathering") | ||||
| 		errCnt.WithLabelValues("encoding") | ||||
| 		if err := opts.Registry.Register(errCnt); err != nil { | ||||
|  | @ -303,8 +303,12 @@ type Logger interface { | |||
| // HandlerOpts specifies options how to serve metrics via an http.Handler. The
 | ||||
| // zero value of HandlerOpts is a reasonable default.
 | ||||
| type HandlerOpts struct { | ||||
| 	// ErrorLog specifies an optional logger for errors collecting and
 | ||||
| 	// serving metrics. If nil, errors are not logged at all.
 | ||||
| 	// ErrorLog specifies an optional Logger for errors collecting and
 | ||||
| 	// serving metrics. If nil, errors are not logged at all. Note that the
 | ||||
| 	// type of a reported error is often prometheus.MultiError, which
 | ||||
| 	// formats into a multi-line error string. If you want to avoid the
 | ||||
| 	// latter, create a Logger implementation that detects a
 | ||||
| 	// prometheus.MultiError and formats the contained errors into one line.
 | ||||
| 	ErrorLog Logger | ||||
| 	// ErrorHandling defines how errors are handled. Note that errors are
 | ||||
| 	// logged regardless of the configured ErrorHandling provided ErrorLog
 | ||||
|  |  | |||
							
								
								
									
										91
									
								
								vendor/github.com/prometheus/client_golang/prometheus/promhttp/instrument_server.go
								
								
									generated
								
								
									vendored
								
								
							
							
						
						
									
										91
									
								
								vendor/github.com/prometheus/client_golang/prometheus/promhttp/instrument_server.go
								
								
									generated
								
								
									vendored
								
								
							|  | @ -43,14 +43,14 @@ func InstrumentHandlerInFlight(g prometheus.Gauge, next http.Handler) http.Handl | |||
| 
 | ||||
| // InstrumentHandlerDuration is a middleware that wraps the provided
 | ||||
| // http.Handler to observe the request duration with the provided ObserverVec.
 | ||||
| // The ObserverVec must have zero, one, or two non-const non-curried labels. For
 | ||||
| // those, the only allowed label names are "code" and "method". The function
 | ||||
| // panics otherwise. The Observe method of the Observer in the ObserverVec is
 | ||||
| // called with the request duration in seconds. Partitioning happens by HTTP
 | ||||
| // status code and/or HTTP method if the respective instance label names are
 | ||||
| // present in the ObserverVec. For unpartitioned observations, use an
 | ||||
| // ObserverVec with zero labels. Note that partitioning of Histograms is
 | ||||
| // expensive and should be used judiciously.
 | ||||
| // The ObserverVec must have valid metric and label names and must have zero,
 | ||||
| // one, or two non-const non-curried labels. For those, the only allowed label
 | ||||
| // names are "code" and "method". The function panics otherwise. The Observe
 | ||||
| // method of the Observer in the ObserverVec is called with the request duration
 | ||||
| // in seconds. Partitioning happens by HTTP status code and/or HTTP method if
 | ||||
| // the respective instance label names are present in the ObserverVec. For
 | ||||
| // unpartitioned observations, use an ObserverVec with zero labels. Note that
 | ||||
| // partitioning of Histograms is expensive and should be used judiciously.
 | ||||
| //
 | ||||
| // If the wrapped Handler does not set a status code, a status code of 200 is assumed.
 | ||||
| //
 | ||||
|  | @ -79,12 +79,13 @@ func InstrumentHandlerDuration(obs prometheus.ObserverVec, next http.Handler) ht | |||
| } | ||||
| 
 | ||||
| // InstrumentHandlerCounter is a middleware that wraps the provided http.Handler
 | ||||
| // to observe the request result with the provided CounterVec.  The CounterVec
 | ||||
| // must have zero, one, or two non-const non-curried labels. For those, the only
 | ||||
| // allowed label names are "code" and "method". The function panics
 | ||||
| // otherwise. Partitioning of the CounterVec happens by HTTP status code and/or
 | ||||
| // HTTP method if the respective instance label names are present in the
 | ||||
| // CounterVec. For unpartitioned counting, use a CounterVec with zero labels.
 | ||||
| // to observe the request result with the provided CounterVec. The CounterVec
 | ||||
| // must have valid metric and label names and must have zero, one, or two
 | ||||
| // non-const non-curried labels. For those, the only allowed label names are
 | ||||
| // "code" and "method". The function panics otherwise. Partitioning of the
 | ||||
| // CounterVec happens by HTTP status code and/or HTTP method if the respective
 | ||||
| // instance label names are present in the CounterVec. For unpartitioned
 | ||||
| // counting, use a CounterVec with zero labels.
 | ||||
| //
 | ||||
| // If the wrapped Handler does not set a status code, a status code of 200 is assumed.
 | ||||
| //
 | ||||
|  | @ -110,14 +111,15 @@ func InstrumentHandlerCounter(counter *prometheus.CounterVec, next http.Handler) | |||
| 
 | ||||
| // InstrumentHandlerTimeToWriteHeader is a middleware that wraps the provided
 | ||||
| // http.Handler to observe with the provided ObserverVec the request duration
 | ||||
| // until the response headers are written. The ObserverVec must have zero, one,
 | ||||
| // or two non-const non-curried labels. For those, the only allowed label names
 | ||||
| // are "code" and "method". The function panics otherwise. The Observe method of
 | ||||
| // the Observer in the ObserverVec is called with the request duration in
 | ||||
| // seconds. Partitioning happens by HTTP status code and/or HTTP method if the
 | ||||
| // respective instance label names are present in the ObserverVec. For
 | ||||
| // unpartitioned observations, use an ObserverVec with zero labels. Note that
 | ||||
| // partitioning of Histograms is expensive and should be used judiciously.
 | ||||
| // until the response headers are written. The ObserverVec must have valid
 | ||||
| // metric and label names and must have zero, one, or two non-const non-curried
 | ||||
| // labels. For those, the only allowed label names are "code" and "method". The
 | ||||
| // function panics otherwise. The Observe method of the Observer in the
 | ||||
| // ObserverVec is called with the request duration in seconds. Partitioning
 | ||||
| // happens by HTTP status code and/or HTTP method if the respective instance
 | ||||
| // label names are present in the ObserverVec. For unpartitioned observations,
 | ||||
| // use an ObserverVec with zero labels. Note that partitioning of Histograms is
 | ||||
| // expensive and should be used judiciously.
 | ||||
| //
 | ||||
| // If the wrapped Handler panics before calling WriteHeader, no value is
 | ||||
| // reported.
 | ||||
|  | @ -139,15 +141,15 @@ func InstrumentHandlerTimeToWriteHeader(obs prometheus.ObserverVec, next http.Ha | |||
| } | ||||
| 
 | ||||
| // InstrumentHandlerRequestSize is a middleware that wraps the provided
 | ||||
| // http.Handler to observe the request size with the provided ObserverVec.  The
 | ||||
| // ObserverVec must have zero, one, or two non-const non-curried labels. For
 | ||||
| // those, the only allowed label names are "code" and "method". The function
 | ||||
| // panics otherwise. The Observe method of the Observer in the ObserverVec is
 | ||||
| // called with the request size in bytes. Partitioning happens by HTTP status
 | ||||
| // code and/or HTTP method if the respective instance label names are present in
 | ||||
| // the ObserverVec. For unpartitioned observations, use an ObserverVec with zero
 | ||||
| // labels. Note that partitioning of Histograms is expensive and should be used
 | ||||
| // judiciously.
 | ||||
| // http.Handler to observe the request size with the provided ObserverVec. The
 | ||||
| // ObserverVec must have valid metric and label names and must have zero, one,
 | ||||
| // or two non-const non-curried labels. For those, the only allowed label names
 | ||||
| // are "code" and "method". The function panics otherwise. The Observe method of
 | ||||
| // the Observer in the ObserverVec is called with the request size in
 | ||||
| // bytes. Partitioning happens by HTTP status code and/or HTTP method if the
 | ||||
| // respective instance label names are present in the ObserverVec. For
 | ||||
| // unpartitioned observations, use an ObserverVec with zero labels. Note that
 | ||||
| // partitioning of Histograms is expensive and should be used judiciously.
 | ||||
| //
 | ||||
| // If the wrapped Handler does not set a status code, a status code of 200 is assumed.
 | ||||
| //
 | ||||
|  | @ -174,15 +176,15 @@ func InstrumentHandlerRequestSize(obs prometheus.ObserverVec, next http.Handler) | |||
| } | ||||
| 
 | ||||
| // InstrumentHandlerResponseSize is a middleware that wraps the provided
 | ||||
| // http.Handler to observe the response size with the provided ObserverVec.  The
 | ||||
| // ObserverVec must have zero, one, or two non-const non-curried labels. For
 | ||||
| // those, the only allowed label names are "code" and "method". The function
 | ||||
| // panics otherwise. The Observe method of the Observer in the ObserverVec is
 | ||||
| // called with the response size in bytes. Partitioning happens by HTTP status
 | ||||
| // code and/or HTTP method if the respective instance label names are present in
 | ||||
| // the ObserverVec. For unpartitioned observations, use an ObserverVec with zero
 | ||||
| // labels. Note that partitioning of Histograms is expensive and should be used
 | ||||
| // judiciously.
 | ||||
| // http.Handler to observe the response size with the provided ObserverVec. The
 | ||||
| // ObserverVec must have valid metric and label names and must have zero, one,
 | ||||
| // or two non-const non-curried labels. For those, the only allowed label names
 | ||||
| // are "code" and "method". The function panics otherwise. The Observe method of
 | ||||
| // the Observer in the ObserverVec is called with the response size in
 | ||||
| // bytes. Partitioning happens by HTTP status code and/or HTTP method if the
 | ||||
| // respective instance label names are present in the ObserverVec. For
 | ||||
| // unpartitioned observations, use an ObserverVec with zero labels. Note that
 | ||||
| // partitioning of Histograms is expensive and should be used judiciously.
 | ||||
| //
 | ||||
| // If the wrapped Handler does not set a status code, a status code of 200 is assumed.
 | ||||
| //
 | ||||
|  | @ -198,6 +200,11 @@ func InstrumentHandlerResponseSize(obs prometheus.ObserverVec, next http.Handler | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // checkLabels returns whether the provided Collector has a non-const,
 | ||||
| // non-curried label named "code" and/or "method". It panics if the provided
 | ||||
| // Collector does not have a Desc or has more than one Desc or its Desc is
 | ||||
| // invalid. It also panics if the Collector has any non-const, non-curried
 | ||||
| // labels that are not named "code" or "method".
 | ||||
| func checkLabels(c prometheus.Collector) (code bool, method bool) { | ||||
| 	// TODO(beorn7): Remove this hacky way to check for instance labels
 | ||||
| 	// once Descriptors can have their dimensionality queried.
 | ||||
|  | @ -225,6 +232,10 @@ func checkLabels(c prometheus.Collector) (code bool, method bool) { | |||
| 
 | ||||
| 	close(descc) | ||||
| 
 | ||||
| 	// Make sure the Collector has a valid Desc by registering it with a
 | ||||
| 	// temporary registry.
 | ||||
| 	prometheus.NewRegistry().MustRegister(c) | ||||
| 
 | ||||
| 	// Create a ConstMetric with the Desc. Since we don't know how many
 | ||||
| 	// variable labels there are, try for as long as it needs.
 | ||||
| 	for err := errors.New("dummy"); err != nil; lvs = append(lvs, magicString) { | ||||
|  |  | |||
|  | @ -26,7 +26,7 @@ import ( | |||
| 	"unicode/utf8" | ||||
| 
 | ||||
| 	"github.com/cespare/xxhash/v2" | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/prometheus/common/expfmt" | ||||
| 
 | ||||
|  | @ -215,6 +215,8 @@ func (err AlreadyRegisteredError) Error() string { | |||
| // by a Gatherer to report multiple errors during MetricFamily gathering.
 | ||||
| type MultiError []error | ||||
| 
 | ||||
| // Error formats the contained errors as a bullet point list, preceded by the
 | ||||
| // total number of errors. Note that this results in a multi-line string.
 | ||||
| func (errs MultiError) Error() string { | ||||
| 	if len(errs) == 0 { | ||||
| 		return "" | ||||
|  |  | |||
|  | @ -23,7 +23,7 @@ import ( | |||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/beorn7/perks/quantile" | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 
 | ||||
| 	dto "github.com/prometheus/client_model/go" | ||||
|  | @ -55,7 +55,12 @@ type Summary interface { | |||
| 	Metric | ||||
| 	Collector | ||||
| 
 | ||||
| 	// Observe adds a single observation to the summary.
 | ||||
| 	// Observe adds a single observation to the summary. Observations are
 | ||||
| 	// usually positive or zero. Negative observations are accepted but
 | ||||
| 	// prevent current versions of Prometheus from properly detecting
 | ||||
| 	// counter resets in the sum of observations. See
 | ||||
| 	// https://prometheus.io/docs/practices/histograms/#count-and-sum-of-observations
 | ||||
| 	// for details.
 | ||||
| 	Observe(float64) | ||||
| } | ||||
| 
 | ||||
|  | @ -110,7 +115,7 @@ type SummaryOpts struct { | |||
| 	// better covered by target labels set by the scraping Prometheus
 | ||||
| 	// server, or by one specific metric (e.g. a build_info or a
 | ||||
| 	// machine_role metric). See also
 | ||||
| 	// https://prometheus.io/docs/instrumenting/writing_exporters/#target-labels,-not-static-scraped-labels
 | ||||
| 	// https://prometheus.io/docs/instrumenting/writing_exporters/#target-labels-not-static-scraped-labels
 | ||||
| 	ConstLabels Labels | ||||
| 
 | ||||
| 	// Objectives defines the quantile rank estimates with their respective
 | ||||
|  | @ -121,7 +126,9 @@ type SummaryOpts struct { | |||
| 	Objectives map[float64]float64 | ||||
| 
 | ||||
| 	// MaxAge defines the duration for which an observation stays relevant
 | ||||
| 	// for the summary. Must be positive. The default value is DefMaxAge.
 | ||||
| 	// for the summary. Only applies to pre-calculated quantiles, does not
 | ||||
| 	// apply to _sum and _count. Must be positive. The default value is
 | ||||
| 	// DefMaxAge.
 | ||||
| 	MaxAge time.Duration | ||||
| 
 | ||||
| 	// AgeBuckets is the number of buckets used to exclude observations that
 | ||||
|  | @ -208,7 +215,7 @@ func newSummary(desc *Desc, opts SummaryOpts, labelValues ...string) Summary { | |||
| 		// Use the lock-free implementation of a Summary without objectives.
 | ||||
| 		s := &noObjectivesSummary{ | ||||
| 			desc:       desc, | ||||
| 			labelPairs: makeLabelPairs(desc, labelValues), | ||||
| 			labelPairs: MakeLabelPairs(desc, labelValues), | ||||
| 			counts:     [2]*summaryCounts{{}, {}}, | ||||
| 		} | ||||
| 		s.init(s) // Init self-collection.
 | ||||
|  | @ -221,7 +228,7 @@ func newSummary(desc *Desc, opts SummaryOpts, labelValues ...string) Summary { | |||
| 		objectives:       opts.Objectives, | ||||
| 		sortedObjectives: make([]float64, 0, len(opts.Objectives)), | ||||
| 
 | ||||
| 		labelPairs: makeLabelPairs(desc, labelValues), | ||||
| 		labelPairs: MakeLabelPairs(desc, labelValues), | ||||
| 
 | ||||
| 		hotBuf:         make([]float64, 0, opts.BufCap), | ||||
| 		coldBuf:        make([]float64, 0, opts.BufCap), | ||||
|  | @ -513,7 +520,7 @@ func (s quantSort) Less(i, j int) bool { | |||
| // (e.g. HTTP request latencies, partitioned by status code and method). Create
 | ||||
| // instances with NewSummaryVec.
 | ||||
| type SummaryVec struct { | ||||
| 	*metricVec | ||||
| 	*MetricVec | ||||
| } | ||||
| 
 | ||||
| // NewSummaryVec creates a new SummaryVec based on the provided SummaryOpts and
 | ||||
|  | @ -535,14 +542,14 @@ func NewSummaryVec(opts SummaryOpts, labelNames []string) *SummaryVec { | |||
| 		opts.ConstLabels, | ||||
| 	) | ||||
| 	return &SummaryVec{ | ||||
| 		metricVec: newMetricVec(desc, func(lvs ...string) Metric { | ||||
| 		MetricVec: NewMetricVec(desc, func(lvs ...string) Metric { | ||||
| 			return newSummary(desc, opts, lvs...) | ||||
| 		}), | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // GetMetricWithLabelValues returns the Summary for the given slice of label
 | ||||
| // values (same order as the VariableLabels in Desc). If that combination of
 | ||||
| // values (same order as the variable labels in Desc). If that combination of
 | ||||
| // label values is accessed for the first time, a new Summary is created.
 | ||||
| //
 | ||||
| // It is possible to call this method without using the returned Summary to only
 | ||||
|  | @ -557,7 +564,7 @@ func NewSummaryVec(opts SummaryOpts, labelNames []string) *SummaryVec { | |||
| // example.
 | ||||
| //
 | ||||
| // An error is returned if the number of label values is not the same as the
 | ||||
| // number of VariableLabels in Desc (minus any curried labels).
 | ||||
| // number of variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // Note that for more than one label value, this method is prone to mistakes
 | ||||
| // caused by an incorrect order of arguments. Consider GetMetricWith(Labels) as
 | ||||
|  | @ -566,7 +573,7 @@ func NewSummaryVec(opts SummaryOpts, labelNames []string) *SummaryVec { | |||
| // with a performance overhead (for creating and processing the Labels map).
 | ||||
| // See also the GaugeVec example.
 | ||||
| func (v *SummaryVec) GetMetricWithLabelValues(lvs ...string) (Observer, error) { | ||||
| 	metric, err := v.metricVec.getMetricWithLabelValues(lvs...) | ||||
| 	metric, err := v.MetricVec.GetMetricWithLabelValues(lvs...) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Observer), err | ||||
| 	} | ||||
|  | @ -574,19 +581,19 @@ func (v *SummaryVec) GetMetricWithLabelValues(lvs ...string) (Observer, error) { | |||
| } | ||||
| 
 | ||||
| // GetMetricWith returns the Summary for the given Labels map (the label names
 | ||||
| // must match those of the VariableLabels in Desc). If that label map is
 | ||||
| // must match those of the variable labels in Desc). If that label map is
 | ||||
| // accessed for the first time, a new Summary is created. Implications of
 | ||||
| // creating a Summary without using it and keeping the Summary for later use are
 | ||||
| // the same as for GetMetricWithLabelValues.
 | ||||
| //
 | ||||
| // An error is returned if the number and names of the Labels are inconsistent
 | ||||
| // with those of the VariableLabels in Desc (minus any curried labels).
 | ||||
| // with those of the variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // This method is used for the same purpose as
 | ||||
| // GetMetricWithLabelValues(...string). See there for pros and cons of the two
 | ||||
| // methods.
 | ||||
| func (v *SummaryVec) GetMetricWith(labels Labels) (Observer, error) { | ||||
| 	metric, err := v.metricVec.getMetricWith(labels) | ||||
| 	metric, err := v.MetricVec.GetMetricWith(labels) | ||||
| 	if metric != nil { | ||||
| 		return metric.(Observer), err | ||||
| 	} | ||||
|  | @ -630,7 +637,7 @@ func (v *SummaryVec) With(labels Labels) Observer { | |||
| // registered with a given registry (usually the uncurried version). The Reset
 | ||||
| // method deletes all metrics, even if called on a curried vector.
 | ||||
| func (v *SummaryVec) CurryWith(labels Labels) (ObserverVec, error) { | ||||
| 	vec, err := v.curryWith(labels) | ||||
| 	vec, err := v.MetricVec.CurryWith(labels) | ||||
| 	if vec != nil { | ||||
| 		return &SummaryVec{vec}, err | ||||
| 	} | ||||
|  | @ -716,7 +723,7 @@ func NewConstSummary( | |||
| 		count:      count, | ||||
| 		sum:        sum, | ||||
| 		quantiles:  quantiles, | ||||
| 		labelPairs: makeLabelPairs(desc, labelValues), | ||||
| 		labelPairs: MakeLabelPairs(desc, labelValues), | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -19,7 +19,7 @@ import ( | |||
| 	"time" | ||||
| 	"unicode/utf8" | ||||
| 
 | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/golang/protobuf/ptypes" | ||||
| 
 | ||||
|  | @ -63,7 +63,7 @@ func newValueFunc(desc *Desc, valueType ValueType, function func() float64) *val | |||
| 		desc:       desc, | ||||
| 		valType:    valueType, | ||||
| 		function:   function, | ||||
| 		labelPairs: makeLabelPairs(desc, nil), | ||||
| 		labelPairs: MakeLabelPairs(desc, nil), | ||||
| 	} | ||||
| 	result.init(result) | ||||
| 	return result | ||||
|  | @ -95,7 +95,7 @@ func NewConstMetric(desc *Desc, valueType ValueType, value float64, labelValues | |||
| 		desc:       desc, | ||||
| 		valType:    valueType, | ||||
| 		val:        value, | ||||
| 		labelPairs: makeLabelPairs(desc, labelValues), | ||||
| 		labelPairs: MakeLabelPairs(desc, labelValues), | ||||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
|  | @ -145,7 +145,14 @@ func populateMetric( | |||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func makeLabelPairs(desc *Desc, labelValues []string) []*dto.LabelPair { | ||||
| // MakeLabelPairs is a helper function to create protobuf LabelPairs from the
 | ||||
| // variable and constant labels in the provided Desc. The values for the
 | ||||
| // variable labels are defined by the labelValues slice, which must be in the
 | ||||
| // same order as the corresponding variable labels in the Desc.
 | ||||
| //
 | ||||
| // This function is only needed for custom Metric implementations. See MetricVec
 | ||||
| // example.
 | ||||
| func MakeLabelPairs(desc *Desc, labelValues []string) []*dto.LabelPair { | ||||
| 	totalLen := len(desc.variableLabels) + len(desc.constLabelPairs) | ||||
| 	if totalLen == 0 { | ||||
| 		// Super fast path.
 | ||||
|  |  | |||
|  | @ -20,12 +20,20 @@ import ( | |||
| 	"github.com/prometheus/common/model" | ||||
| ) | ||||
| 
 | ||||
| // metricVec is a Collector to bundle metrics of the same name that differ in
 | ||||
| // their label values. metricVec is not used directly (and therefore
 | ||||
| // unexported). It is used as a building block for implementations of vectors of
 | ||||
| // a given metric type, like GaugeVec, CounterVec, SummaryVec, and HistogramVec.
 | ||||
| // It also handles label currying.
 | ||||
| type metricVec struct { | ||||
| // MetricVec is a Collector to bundle metrics of the same name that differ in
 | ||||
| // their label values. MetricVec is not used directly but as a building block
 | ||||
| // for implementations of vectors of a given metric type, like GaugeVec,
 | ||||
| // CounterVec, SummaryVec, and HistogramVec. It is exported so that it can be
 | ||||
| // used for custom Metric implementations.
 | ||||
| //
 | ||||
| // To create a FooVec for custom Metric Foo, embed a pointer to MetricVec in
 | ||||
| // FooVec and initialize it with NewMetricVec. Implement wrappers for
 | ||||
| // GetMetricWithLabelValues and GetMetricWith that return (Foo, error) rather
 | ||||
| // than (Metric, error). Similarly, create a wrapper for CurryWith that returns
 | ||||
| // (*FooVec, error) rather than (*MetricVec, error). It is recommended to also
 | ||||
| // add the convenience methods WithLabelValues, With, and MustCurryWith, which
 | ||||
| // panic instead of returning errors. See also the MetricVec example.
 | ||||
| type MetricVec struct { | ||||
| 	*metricMap | ||||
| 
 | ||||
| 	curry []curriedLabelValue | ||||
|  | @ -35,9 +43,9 @@ type metricVec struct { | |||
| 	hashAddByte func(h uint64, b byte) uint64 | ||||
| } | ||||
| 
 | ||||
| // newMetricVec returns an initialized metricVec.
 | ||||
| func newMetricVec(desc *Desc, newMetric func(lvs ...string) Metric) *metricVec { | ||||
| 	return &metricVec{ | ||||
| // NewMetricVec returns an initialized metricVec.
 | ||||
| func NewMetricVec(desc *Desc, newMetric func(lvs ...string) Metric) *MetricVec { | ||||
| 	return &MetricVec{ | ||||
| 		metricMap: &metricMap{ | ||||
| 			metrics:   map[uint64][]metricWithLabelValues{}, | ||||
| 			desc:      desc, | ||||
|  | @ -63,7 +71,7 @@ func newMetricVec(desc *Desc, newMetric func(lvs ...string) Metric) *metricVec { | |||
| // latter has a much more readable (albeit more verbose) syntax, but it comes
 | ||||
| // with a performance overhead (for creating and processing the Labels map).
 | ||||
| // See also the CounterVec example.
 | ||||
| func (m *metricVec) DeleteLabelValues(lvs ...string) bool { | ||||
| func (m *MetricVec) DeleteLabelValues(lvs ...string) bool { | ||||
| 	h, err := m.hashLabelValues(lvs) | ||||
| 	if err != nil { | ||||
| 		return false | ||||
|  | @ -82,7 +90,7 @@ func (m *metricVec) DeleteLabelValues(lvs ...string) bool { | |||
| //
 | ||||
| // This method is used for the same purpose as DeleteLabelValues(...string). See
 | ||||
| // there for pros and cons of the two methods.
 | ||||
| func (m *metricVec) Delete(labels Labels) bool { | ||||
| func (m *MetricVec) Delete(labels Labels) bool { | ||||
| 	h, err := m.hashLabels(labels) | ||||
| 	if err != nil { | ||||
| 		return false | ||||
|  | @ -95,15 +103,32 @@ func (m *metricVec) Delete(labels Labels) bool { | |||
| // show up in GoDoc.
 | ||||
| 
 | ||||
| // Describe implements Collector.
 | ||||
| func (m *metricVec) Describe(ch chan<- *Desc) { m.metricMap.Describe(ch) } | ||||
| func (m *MetricVec) Describe(ch chan<- *Desc) { m.metricMap.Describe(ch) } | ||||
| 
 | ||||
| // Collect implements Collector.
 | ||||
| func (m *metricVec) Collect(ch chan<- Metric) { m.metricMap.Collect(ch) } | ||||
| func (m *MetricVec) Collect(ch chan<- Metric) { m.metricMap.Collect(ch) } | ||||
| 
 | ||||
| // Reset deletes all metrics in this vector.
 | ||||
| func (m *metricVec) Reset() { m.metricMap.Reset() } | ||||
| func (m *MetricVec) Reset() { m.metricMap.Reset() } | ||||
| 
 | ||||
| func (m *metricVec) curryWith(labels Labels) (*metricVec, error) { | ||||
| // CurryWith returns a vector curried with the provided labels, i.e. the
 | ||||
| // returned vector has those labels pre-set for all labeled operations performed
 | ||||
| // on it. The cardinality of the curried vector is reduced accordingly. The
 | ||||
| // order of the remaining labels stays the same (just with the curried labels
 | ||||
| // taken out of the sequence – which is relevant for the
 | ||||
| // (GetMetric)WithLabelValues methods). It is possible to curry a curried
 | ||||
| // vector, but only with labels not yet used for currying before.
 | ||||
| //
 | ||||
| // The metrics contained in the MetricVec are shared between the curried and
 | ||||
| // uncurried vectors. They are just accessed differently. Curried and uncurried
 | ||||
| // vectors behave identically in terms of collection. Only one must be
 | ||||
| // registered with a given registry (usually the uncurried version). The Reset
 | ||||
| // method deletes all metrics, even if called on a curried vector.
 | ||||
| //
 | ||||
| // Note that CurryWith is usually not called directly but through a wrapper
 | ||||
| // around MetricVec, implementing a vector for a specific Metric
 | ||||
| // implementation, for example GaugeVec.
 | ||||
| func (m *MetricVec) CurryWith(labels Labels) (*MetricVec, error) { | ||||
| 	var ( | ||||
| 		newCurry []curriedLabelValue | ||||
| 		oldCurry = m.curry | ||||
|  | @ -128,7 +153,7 @@ func (m *metricVec) curryWith(labels Labels) (*metricVec, error) { | |||
| 		return nil, fmt.Errorf("%d unknown label(s) found during currying", l) | ||||
| 	} | ||||
| 
 | ||||
| 	return &metricVec{ | ||||
| 	return &MetricVec{ | ||||
| 		metricMap:   m.metricMap, | ||||
| 		curry:       newCurry, | ||||
| 		hashAdd:     m.hashAdd, | ||||
|  | @ -136,7 +161,34 @@ func (m *metricVec) curryWith(labels Labels) (*metricVec, error) { | |||
| 	}, nil | ||||
| } | ||||
| 
 | ||||
| func (m *metricVec) getMetricWithLabelValues(lvs ...string) (Metric, error) { | ||||
| // GetMetricWithLabelValues returns the Metric for the given slice of label
 | ||||
| // values (same order as the variable labels in Desc). If that combination of
 | ||||
| // label values is accessed for the first time, a new Metric is created (by
 | ||||
| // calling the newMetric function provided during construction of the
 | ||||
| // MetricVec).
 | ||||
| //
 | ||||
| // It is possible to call this method without using the returned Metric to only
 | ||||
| // create the new Metric but leave it in its initial state.
 | ||||
| //
 | ||||
| // Keeping the Metric for later use is possible (and should be considered if
 | ||||
| // performance is critical), but keep in mind that Reset, DeleteLabelValues and
 | ||||
| // Delete can be used to delete the Metric from the MetricVec. In that case, the
 | ||||
| // Metric will still exist, but it will not be exported anymore, even if a
 | ||||
| // Metric with the same label values is created later.
 | ||||
| //
 | ||||
| // An error is returned if the number of label values is not the same as the
 | ||||
| // number of variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // Note that for more than one label value, this method is prone to mistakes
 | ||||
| // caused by an incorrect order of arguments. Consider GetMetricWith(Labels) as
 | ||||
| // an alternative to avoid that type of mistake. For higher label numbers, the
 | ||||
| // latter has a much more readable (albeit more verbose) syntax, but it comes
 | ||||
| // with a performance overhead (for creating and processing the Labels map).
 | ||||
| //
 | ||||
| // Note that GetMetricWithLabelValues is usually not called directly but through
 | ||||
| // a wrapper around MetricVec, implementing a vector for a specific Metric
 | ||||
| // implementation, for example GaugeVec.
 | ||||
| func (m *MetricVec) GetMetricWithLabelValues(lvs ...string) (Metric, error) { | ||||
| 	h, err := m.hashLabelValues(lvs) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | @ -145,7 +197,23 @@ func (m *metricVec) getMetricWithLabelValues(lvs ...string) (Metric, error) { | |||
| 	return m.metricMap.getOrCreateMetricWithLabelValues(h, lvs, m.curry), nil | ||||
| } | ||||
| 
 | ||||
| func (m *metricVec) getMetricWith(labels Labels) (Metric, error) { | ||||
| // GetMetricWith returns the Metric for the given Labels map (the label names
 | ||||
| // must match those of the variable labels in Desc). If that label map is
 | ||||
| // accessed for the first time, a new Metric is created. Implications of
 | ||||
| // creating a Metric without using it and keeping the Metric for later use
 | ||||
| // are the same as for GetMetricWithLabelValues.
 | ||||
| //
 | ||||
| // An error is returned if the number and names of the Labels are inconsistent
 | ||||
| // with those of the variable labels in Desc (minus any curried labels).
 | ||||
| //
 | ||||
| // This method is used for the same purpose as
 | ||||
| // GetMetricWithLabelValues(...string). See there for pros and cons of the two
 | ||||
| // methods.
 | ||||
| //
 | ||||
| // Note that GetMetricWith is usually not called directly but through a wrapper
 | ||||
| // around MetricVec, implementing a vector for a specific Metric implementation,
 | ||||
| // for example GaugeVec.
 | ||||
| func (m *MetricVec) GetMetricWith(labels Labels) (Metric, error) { | ||||
| 	h, err := m.hashLabels(labels) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | @ -154,7 +222,7 @@ func (m *metricVec) getMetricWith(labels Labels) (Metric, error) { | |||
| 	return m.metricMap.getOrCreateMetricWithLabels(h, labels, m.curry), nil | ||||
| } | ||||
| 
 | ||||
| func (m *metricVec) hashLabelValues(vals []string) (uint64, error) { | ||||
| func (m *MetricVec) hashLabelValues(vals []string) (uint64, error) { | ||||
| 	if err := validateLabelValues(vals, len(m.desc.variableLabels)-len(m.curry)); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | @ -177,7 +245,7 @@ func (m *metricVec) hashLabelValues(vals []string) (uint64, error) { | |||
| 	return h, nil | ||||
| } | ||||
| 
 | ||||
| func (m *metricVec) hashLabels(labels Labels) (uint64, error) { | ||||
| func (m *MetricVec) hashLabels(labels Labels) (uint64, error) { | ||||
| 	if err := validateValuesInLabels(labels, len(m.desc.variableLabels)-len(m.curry)); err != nil { | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | @ -276,7 +344,9 @@ func (m *metricMap) deleteByHashWithLabelValues( | |||
| 	} | ||||
| 
 | ||||
| 	if len(metrics) > 1 { | ||||
| 		old := metrics | ||||
| 		m.metrics[h] = append(metrics[:i], metrics[i+1:]...) | ||||
| 		old[len(old)-1] = metricWithLabelValues{} | ||||
| 	} else { | ||||
| 		delete(m.metrics, h) | ||||
| 	} | ||||
|  | @ -302,7 +372,9 @@ func (m *metricMap) deleteByHashWithLabels( | |||
| 	} | ||||
| 
 | ||||
| 	if len(metrics) > 1 { | ||||
| 		old := metrics | ||||
| 		m.metrics[h] = append(metrics[:i], metrics[i+1:]...) | ||||
| 		old[len(old)-1] = metricWithLabelValues{} | ||||
| 	} else { | ||||
| 		delete(m.metrics, h) | ||||
| 	} | ||||
|  |  | |||
|  | @ -17,7 +17,7 @@ import ( | |||
| 	"fmt" | ||||
| 	"sort" | ||||
| 
 | ||||
| 	//lint:ignore SA1019 Need to keep deprecated package for compatibility.
 | ||||
| 	//nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 
 | ||||
| 	dto "github.com/prometheus/client_model/go" | ||||
|  | @ -32,7 +32,9 @@ import ( | |||
| // in a no-op Registerer.
 | ||||
| //
 | ||||
| // WrapRegistererWith provides a way to add fixed labels to a subset of
 | ||||
| // Collectors. It should not be used to add fixed labels to all metrics exposed.
 | ||||
| // Collectors. It should not be used to add fixed labels to all metrics
 | ||||
| // exposed. See also
 | ||||
| // https://prometheus.io/docs/instrumenting/writing_exporters/#target-labels-not-static-scraped-labels
 | ||||
| //
 | ||||
| // Conflicts between Collectors registered through the original Registerer with
 | ||||
| // Collectors registered through the wrapping Registerer will still be
 | ||||
|  |  | |||
|  | @ -164,7 +164,7 @@ func (sd *SampleDecoder) Decode(s *model.Vector) error { | |||
| } | ||||
| 
 | ||||
| // ExtractSamples builds a slice of samples from the provided metric
 | ||||
| // families. If an error occurrs during sample extraction, it continues to
 | ||||
| // families. If an error occurs during sample extraction, it continues to
 | ||||
| // extract from the remaining metric families. The returned error is the last
 | ||||
| // error that has occurred.
 | ||||
| func ExtractSamples(o *DecodeOptions, fams ...*dto.MetricFamily) (model.Vector, error) { | ||||
|  |  | |||
|  | @ -18,7 +18,7 @@ import ( | |||
| 	"io" | ||||
| 	"net/http" | ||||
| 
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/golang/protobuf/proto" //nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/matttproud/golang_protobuf_extensions/pbutil" | ||||
| 	"github.com/prometheus/common/internal/bitbucket.org/ww/goautoneg" | ||||
| 
 | ||||
|  |  | |||
|  | @ -24,7 +24,7 @@ import ( | |||
| 
 | ||||
| 	dto "github.com/prometheus/client_model/go" | ||||
| 
 | ||||
| 	"github.com/golang/protobuf/proto" | ||||
| 	"github.com/golang/protobuf/proto" //nolint:staticcheck // Ignore SA1019. Need to keep deprecated package for compatibility.
 | ||||
| 	"github.com/prometheus/common/model" | ||||
| ) | ||||
| 
 | ||||
|  | @ -299,6 +299,17 @@ func (p *TextParser) startLabelName() stateFn { | |||
| 		p.parseError(fmt.Sprintf("expected '=' after label name, found %q", p.currentByte)) | ||||
| 		return nil | ||||
| 	} | ||||
| 	// Check for duplicate label names.
 | ||||
| 	labels := make(map[string]struct{}) | ||||
| 	for _, l := range p.currentMetric.Label { | ||||
| 		lName := l.GetName() | ||||
| 		if _, exists := labels[lName]; !exists { | ||||
| 			labels[lName] = struct{}{} | ||||
| 		} else { | ||||
| 			p.parseError(fmt.Sprintf("duplicate label names for metric %q", p.currentMF.GetName())) | ||||
| 			return nil | ||||
| 		} | ||||
| 	} | ||||
| 	return p.startLabelValue | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -20,7 +20,7 @@ const ( | |||
| 	prime64  = 1099511628211 | ||||
| ) | ||||
| 
 | ||||
| // hashNew initializies a new fnv64a hash value.
 | ||||
| // hashNew initializes a new fnv64a hash value.
 | ||||
| func hashNew() uint64 { | ||||
| 	return offset64 | ||||
| } | ||||
|  |  | |||
|  | @ -45,6 +45,14 @@ const ( | |||
| 	// scrape a target.
 | ||||
| 	MetricsPathLabel = "__metrics_path__" | ||||
| 
 | ||||
| 	// ScrapeIntervalLabel is the name of the label that holds the scrape interval
 | ||||
| 	// used to scrape a target.
 | ||||
| 	ScrapeIntervalLabel = "__scrape_interval__" | ||||
| 
 | ||||
| 	// ScrapeTimeoutLabel is the name of the label that holds the scrape
 | ||||
| 	// timeout used to scrape a target.
 | ||||
| 	ScrapeTimeoutLabel = "__scrape_timeout__" | ||||
| 
 | ||||
| 	// ReservedLabelPrefix is a prefix which is not legal in user-supplied
 | ||||
| 	// label names.
 | ||||
| 	ReservedLabelPrefix = "__" | ||||
|  |  | |||
|  | @ -14,6 +14,8 @@ | |||
| package model | ||||
| 
 | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
| 	"regexp" | ||||
|  | @ -181,77 +183,118 @@ func (d *Duration) Type() string { | |||
| 	return "duration" | ||||
| } | ||||
| 
 | ||||
| var durationRE = regexp.MustCompile("^([0-9]+)(y|w|d|h|m|s|ms)$") | ||||
| var durationRE = regexp.MustCompile("^(([0-9]+)y)?(([0-9]+)w)?(([0-9]+)d)?(([0-9]+)h)?(([0-9]+)m)?(([0-9]+)s)?(([0-9]+)ms)?$") | ||||
| 
 | ||||
| // ParseDuration parses a string into a time.Duration, assuming that a year
 | ||||
| // always has 365d, a week always has 7d, and a day always has 24h.
 | ||||
| func ParseDuration(durationStr string) (Duration, error) { | ||||
| 	// Allow 0 without a unit.
 | ||||
| 	if durationStr == "0" { | ||||
| 	switch durationStr { | ||||
| 	case "0": | ||||
| 		// Allow 0 without a unit.
 | ||||
| 		return 0, nil | ||||
| 	case "": | ||||
| 		return 0, fmt.Errorf("empty duration string") | ||||
| 	} | ||||
| 	matches := durationRE.FindStringSubmatch(durationStr) | ||||
| 	if len(matches) != 3 { | ||||
| 	if matches == nil { | ||||
| 		return 0, fmt.Errorf("not a valid duration string: %q", durationStr) | ||||
| 	} | ||||
| 	var ( | ||||
| 		n, _ = strconv.Atoi(matches[1]) | ||||
| 		dur  = time.Duration(n) * time.Millisecond | ||||
| 	) | ||||
| 	switch unit := matches[2]; unit { | ||||
| 	case "y": | ||||
| 		dur *= 1000 * 60 * 60 * 24 * 365 | ||||
| 	case "w": | ||||
| 		dur *= 1000 * 60 * 60 * 24 * 7 | ||||
| 	case "d": | ||||
| 		dur *= 1000 * 60 * 60 * 24 | ||||
| 	case "h": | ||||
| 		dur *= 1000 * 60 * 60 | ||||
| 	case "m": | ||||
| 		dur *= 1000 * 60 | ||||
| 	case "s": | ||||
| 		dur *= 1000 | ||||
| 	case "ms": | ||||
| 		// Value already correct
 | ||||
| 	default: | ||||
| 		return 0, fmt.Errorf("invalid time unit in duration string: %q", unit) | ||||
| 	var dur time.Duration | ||||
| 
 | ||||
| 	// Parse the match at pos `pos` in the regex and use `mult` to turn that
 | ||||
| 	// into ms, then add that value to the total parsed duration.
 | ||||
| 	var overflowErr error | ||||
| 	m := func(pos int, mult time.Duration) { | ||||
| 		if matches[pos] == "" { | ||||
| 			return | ||||
| 		} | ||||
| 		n, _ := strconv.Atoi(matches[pos]) | ||||
| 
 | ||||
| 		// Check if the provided duration overflows time.Duration (> ~ 290years).
 | ||||
| 		if n > int((1<<63-1)/mult/time.Millisecond) { | ||||
| 			overflowErr = errors.New("duration out of range") | ||||
| 		} | ||||
| 		d := time.Duration(n) * time.Millisecond | ||||
| 		dur += d * mult | ||||
| 
 | ||||
| 		if dur < 0 { | ||||
| 			overflowErr = errors.New("duration out of range") | ||||
| 		} | ||||
| 	} | ||||
| 	return Duration(dur), nil | ||||
| 
 | ||||
| 	m(2, 1000*60*60*24*365) // y
 | ||||
| 	m(4, 1000*60*60*24*7)   // w
 | ||||
| 	m(6, 1000*60*60*24)     // d
 | ||||
| 	m(8, 1000*60*60)        // h
 | ||||
| 	m(10, 1000*60)          // m
 | ||||
| 	m(12, 1000)             // s
 | ||||
| 	m(14, 1)                // ms
 | ||||
| 
 | ||||
| 	return Duration(dur), overflowErr | ||||
| } | ||||
| 
 | ||||
| func (d Duration) String() string { | ||||
| 	var ( | ||||
| 		ms   = int64(time.Duration(d) / time.Millisecond) | ||||
| 		unit = "ms" | ||||
| 		ms = int64(time.Duration(d) / time.Millisecond) | ||||
| 		r  = "" | ||||
| 	) | ||||
| 	if ms == 0 { | ||||
| 		return "0s" | ||||
| 	} | ||||
| 	factors := map[string]int64{ | ||||
| 		"y":  1000 * 60 * 60 * 24 * 365, | ||||
| 		"w":  1000 * 60 * 60 * 24 * 7, | ||||
| 		"d":  1000 * 60 * 60 * 24, | ||||
| 		"h":  1000 * 60 * 60, | ||||
| 		"m":  1000 * 60, | ||||
| 		"s":  1000, | ||||
| 		"ms": 1, | ||||
| 
 | ||||
| 	f := func(unit string, mult int64, exact bool) { | ||||
| 		if exact && ms%mult != 0 { | ||||
| 			return | ||||
| 		} | ||||
| 		if v := ms / mult; v > 0 { | ||||
| 			r += fmt.Sprintf("%d%s", v, unit) | ||||
| 			ms -= v * mult | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	switch int64(0) { | ||||
| 	case ms % factors["y"]: | ||||
| 		unit = "y" | ||||
| 	case ms % factors["w"]: | ||||
| 		unit = "w" | ||||
| 	case ms % factors["d"]: | ||||
| 		unit = "d" | ||||
| 	case ms % factors["h"]: | ||||
| 		unit = "h" | ||||
| 	case ms % factors["m"]: | ||||
| 		unit = "m" | ||||
| 	case ms % factors["s"]: | ||||
| 		unit = "s" | ||||
| 	// Only format years and weeks if the remainder is zero, as it is often
 | ||||
| 	// easier to read 90d than 12w6d.
 | ||||
| 	f("y", 1000*60*60*24*365, true) | ||||
| 	f("w", 1000*60*60*24*7, true) | ||||
| 
 | ||||
| 	f("d", 1000*60*60*24, false) | ||||
| 	f("h", 1000*60*60, false) | ||||
| 	f("m", 1000*60, false) | ||||
| 	f("s", 1000, false) | ||||
| 	f("ms", 1, false) | ||||
| 
 | ||||
| 	return r | ||||
| } | ||||
| 
 | ||||
| // MarshalJSON implements the json.Marshaler interface.
 | ||||
| func (d Duration) MarshalJSON() ([]byte, error) { | ||||
| 	return json.Marshal(d.String()) | ||||
| } | ||||
| 
 | ||||
| // UnmarshalJSON implements the json.Unmarshaler interface.
 | ||||
| func (d *Duration) UnmarshalJSON(bytes []byte) error { | ||||
| 	var s string | ||||
| 	if err := json.Unmarshal(bytes, &s); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return fmt.Sprintf("%v%v", ms/factors[unit], unit) | ||||
| 	dur, err := ParseDuration(s) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	*d = dur | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // MarshalText implements the encoding.TextMarshaler interface.
 | ||||
| func (d *Duration) MarshalText() ([]byte, error) { | ||||
| 	return []byte(d.String()), nil | ||||
| } | ||||
| 
 | ||||
| // UnmarshalText implements the encoding.TextUnmarshaler interface.
 | ||||
| func (d *Duration) UnmarshalText(text []byte) error { | ||||
| 	var err error | ||||
| 	*d, err = ParseDuration(string(text)) | ||||
| 	return err | ||||
| } | ||||
| 
 | ||||
| // MarshalYAML implements the yaml.Marshaler interface.
 | ||||
|  |  | |||
|  | @ -18,6 +18,8 @@ include Makefile.common | |||
| 	./ttar -C $(dir $*) -x -f $*.ttar | ||||
| 	touch $@ | ||||
| 
 | ||||
| fixtures: fixtures/.unpacked | ||||
| 
 | ||||
| update_fixtures: | ||||
| 	rm -vf fixtures/.unpacked | ||||
| 	./ttar -c -f fixtures.ttar fixtures/ | ||||
|  |  | |||
|  | @ -78,12 +78,12 @@ ifneq ($(shell which gotestsum),) | |||
| endif | ||||
| endif | ||||
| 
 | ||||
| PROMU_VERSION ?= 0.7.0 | ||||
| PROMU_VERSION ?= 0.12.0 | ||||
| PROMU_URL     := https://github.com/prometheus/promu/releases/download/v$(PROMU_VERSION)/promu-$(PROMU_VERSION).$(GO_BUILD_PLATFORM).tar.gz | ||||
| 
 | ||||
| GOLANGCI_LINT := | ||||
| GOLANGCI_LINT_OPTS ?= | ||||
| GOLANGCI_LINT_VERSION ?= v1.18.0 | ||||
| GOLANGCI_LINT_VERSION ?= v1.39.0 | ||||
| # golangci-lint only supports linux, darwin and windows platforms on i386/amd64.
 | ||||
| # windows isn't included here because of the path separator being different.
 | ||||
| ifeq ($(GOHOSTOS),$(filter $(GOHOSTOS),linux darwin)) | ||||
|  | @ -118,7 +118,7 @@ endif | |||
| %: common-% ; | ||||
| 
 | ||||
| .PHONY: common-all | ||||
| common-all: precheck style check_license lint unused build test | ||||
| common-all: precheck style check_license lint yamllint unused build test | ||||
| 
 | ||||
| .PHONY: common-style | ||||
| common-style: | ||||
|  | @ -198,6 +198,15 @@ else | |||
| endif | ||||
| endif | ||||
| 
 | ||||
| .PHONY: common-yamllint | ||||
| common-yamllint: | ||||
| 	@echo ">> running yamllint on all YAML files in the repository" | ||||
| ifeq (, $(shell which yamllint)) | ||||
| 	@echo "yamllint not installed so skipping" | ||||
| else | ||||
| 	yamllint . | ||||
| endif | ||||
| 
 | ||||
| # For backward-compatibility.
 | ||||
| .PHONY: common-staticcheck | ||||
| common-staticcheck: lint | ||||
|  |  | |||
|  | @ -6,8 +6,8 @@ metrics from the pseudo-filesystems /proc and /sys. | |||
| *WARNING*: This package is a work in progress. Its API may still break in | ||||
| backwards-incompatible ways without warnings. Use it at your own risk. | ||||
| 
 | ||||
| [](https://godoc.org/github.com/prometheus/procfs) | ||||
| [](https://travis-ci.org/prometheus/procfs) | ||||
| [](https://pkg.go.dev/github.com/prometheus/procfs) | ||||
| [](https://circleci.com/gh/prometheus/procfs/tree/master) | ||||
| [](https://goreportcard.com/report/github.com/prometheus/procfs) | ||||
| 
 | ||||
| ## Usage | ||||
|  |  | |||
|  | @ -1,4 +1,4 @@ | |||
| // Copyright 2019 The Prometheus Authors
 | ||||
| // Copyright 2021 The Prometheus 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
 | ||||
|  | @ -11,12 +11,20 @@ | |||
| // See the License for the specific language governing permissions and
 | ||||
| // limitations under the License.
 | ||||
| 
 | ||||
| // +build !go1.12
 | ||||
| package procfs | ||||
| 
 | ||||
| package prometheus | ||||
| import ( | ||||
| 	"strings" | ||||
| 
 | ||||
| // readBuildInfo is a wrapper around debug.ReadBuildInfo for Go versions before
 | ||||
| // 1.12. Remove this whole file once the minimum supported Go version is 1.12.
 | ||||
| func readBuildInfo() (path, version, sum string) { | ||||
| 	return "unknown", "unknown", "unknown" | ||||
| 	"github.com/prometheus/procfs/internal/util" | ||||
| ) | ||||
| 
 | ||||
| // CmdLine returns the command line of the kernel.
 | ||||
| func (fs FS) CmdLine() ([]string, error) { | ||||
| 	data, err := util.ReadFileNoStat(fs.proc.Path("cmdline")) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	return strings.Fields(string(data)), nil | ||||
| } | ||||
|  | @ -31,7 +31,7 @@ | |||
| //    		log.Fatalf("could not get process: %s", err)
 | ||||
| //    	}
 | ||||
| //
 | ||||
| //    	stat, err := p.NewStat()
 | ||||
| //    	stat, err := p.Stat()
 | ||||
| //    	if err != nil {
 | ||||
| //    		log.Fatalf("could not get process stat: %s", err)
 | ||||
| //    	}
 | ||||
|  |  | |||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -22,9 +22,12 @@ import ( | |||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	statusLineRE      = regexp.MustCompile(`(\d+) blocks .*\[(\d+)/(\d+)\] \[[U_]+\]`) | ||||
| 	recoveryLineRE    = regexp.MustCompile(`\((\d+)/\d+\)`) | ||||
| 	componentDeviceRE = regexp.MustCompile(`(.*)\[\d+\]`) | ||||
| 	statusLineRE         = regexp.MustCompile(`(\d+) blocks .*\[(\d+)/(\d+)\] \[([U_]+)\]`) | ||||
| 	recoveryLineBlocksRE = regexp.MustCompile(`\((\d+)/\d+\)`) | ||||
| 	recoveryLinePctRE    = regexp.MustCompile(`= (.+)%`) | ||||
| 	recoveryLineFinishRE = regexp.MustCompile(`finish=(.+)min`) | ||||
| 	recoveryLineSpeedRE  = regexp.MustCompile(`speed=(.+)[A-Z]`) | ||||
| 	componentDeviceRE    = regexp.MustCompile(`(.*)\[\d+\]`) | ||||
| ) | ||||
| 
 | ||||
| // MDStat holds info parsed from /proc/mdstat.
 | ||||
|  | @ -39,12 +42,20 @@ type MDStat struct { | |||
| 	DisksTotal int64 | ||||
| 	// Number of failed disks.
 | ||||
| 	DisksFailed int64 | ||||
| 	// Number of "down" disks. (the _ indicator in the status line)
 | ||||
| 	DisksDown int64 | ||||
| 	// Spare disks in the device.
 | ||||
| 	DisksSpare int64 | ||||
| 	// Number of blocks the device holds.
 | ||||
| 	BlocksTotal int64 | ||||
| 	// Number of blocks on the device that are in sync.
 | ||||
| 	BlocksSynced int64 | ||||
| 	// progress percentage of current sync
 | ||||
| 	BlocksSyncedPct float64 | ||||
| 	// estimated finishing time for current sync (in minutes)
 | ||||
| 	BlocksSyncedFinishTime float64 | ||||
| 	// current sync speed (in Kilobytes/sec)
 | ||||
| 	BlocksSyncedSpeed float64 | ||||
| 	// Name of md component devices
 | ||||
| 	Devices []string | ||||
| } | ||||
|  | @ -91,7 +102,7 @@ func parseMDStat(mdStatData []byte) ([]MDStat, error) { | |||
| 		// Failed disks have the suffix (F) & Spare disks have the suffix (S).
 | ||||
| 		fail := int64(strings.Count(line, "(F)")) | ||||
| 		spare := int64(strings.Count(line, "(S)")) | ||||
| 		active, total, size, err := evalStatusLine(lines[i], lines[i+1]) | ||||
| 		active, total, down, size, err := evalStatusLine(lines[i], lines[i+1]) | ||||
| 
 | ||||
| 		if err != nil { | ||||
| 			return nil, fmt.Errorf("error parsing md device lines: %w", err) | ||||
|  | @ -105,6 +116,9 @@ func parseMDStat(mdStatData []byte) ([]MDStat, error) { | |||
| 		// If device is syncing at the moment, get the number of currently
 | ||||
| 		// synced bytes, otherwise that number equals the size of the device.
 | ||||
| 		syncedBlocks := size | ||||
| 		speed := float64(0) | ||||
| 		finish := float64(0) | ||||
| 		pct := float64(0) | ||||
| 		recovering := strings.Contains(lines[syncLineIdx], "recovery") | ||||
| 		resyncing := strings.Contains(lines[syncLineIdx], "resync") | ||||
| 		checking := strings.Contains(lines[syncLineIdx], "check") | ||||
|  | @ -124,7 +138,7 @@ func parseMDStat(mdStatData []byte) ([]MDStat, error) { | |||
| 				strings.Contains(lines[syncLineIdx], "DELAYED") { | ||||
| 				syncedBlocks = 0 | ||||
| 			} else { | ||||
| 				syncedBlocks, err = evalRecoveryLine(lines[syncLineIdx]) | ||||
| 				syncedBlocks, pct, finish, speed, err = evalRecoveryLine(lines[syncLineIdx]) | ||||
| 				if err != nil { | ||||
| 					return nil, fmt.Errorf("error parsing sync line in md device %q: %w", mdName, err) | ||||
| 				} | ||||
|  | @ -132,69 +146,104 @@ func parseMDStat(mdStatData []byte) ([]MDStat, error) { | |||
| 		} | ||||
| 
 | ||||
| 		mdStats = append(mdStats, MDStat{ | ||||
| 			Name:          mdName, | ||||
| 			ActivityState: state, | ||||
| 			DisksActive:   active, | ||||
| 			DisksFailed:   fail, | ||||
| 			DisksSpare:    spare, | ||||
| 			DisksTotal:    total, | ||||
| 			BlocksTotal:   size, | ||||
| 			BlocksSynced:  syncedBlocks, | ||||
| 			Devices:       evalComponentDevices(deviceFields), | ||||
| 			Name:                   mdName, | ||||
| 			ActivityState:          state, | ||||
| 			DisksActive:            active, | ||||
| 			DisksFailed:            fail, | ||||
| 			DisksDown:              down, | ||||
| 			DisksSpare:             spare, | ||||
| 			DisksTotal:             total, | ||||
| 			BlocksTotal:            size, | ||||
| 			BlocksSynced:           syncedBlocks, | ||||
| 			BlocksSyncedPct:        pct, | ||||
| 			BlocksSyncedFinishTime: finish, | ||||
| 			BlocksSyncedSpeed:      speed, | ||||
| 			Devices:                evalComponentDevices(deviceFields), | ||||
| 		}) | ||||
| 	} | ||||
| 
 | ||||
| 	return mdStats, nil | ||||
| } | ||||
| 
 | ||||
| func evalStatusLine(deviceLine, statusLine string) (active, total, size int64, err error) { | ||||
| func evalStatusLine(deviceLine, statusLine string) (active, total, down, size int64, err error) { | ||||
| 
 | ||||
| 	sizeStr := strings.Fields(statusLine)[0] | ||||
| 	size, err = strconv.ParseInt(sizeStr, 10, 64) | ||||
| 	if err != nil { | ||||
| 		return 0, 0, 0, fmt.Errorf("unexpected statusLine %q: %w", statusLine, err) | ||||
| 		return 0, 0, 0, 0, fmt.Errorf("unexpected statusLine %q: %w", statusLine, err) | ||||
| 	} | ||||
| 
 | ||||
| 	if strings.Contains(deviceLine, "raid0") || strings.Contains(deviceLine, "linear") { | ||||
| 		// In the device deviceLine, only disks have a number associated with them in [].
 | ||||
| 		total = int64(strings.Count(deviceLine, "[")) | ||||
| 		return total, total, size, nil | ||||
| 		return total, total, 0, size, nil | ||||
| 	} | ||||
| 
 | ||||
| 	if strings.Contains(deviceLine, "inactive") { | ||||
| 		return 0, 0, size, nil | ||||
| 		return 0, 0, 0, size, nil | ||||
| 	} | ||||
| 
 | ||||
| 	matches := statusLineRE.FindStringSubmatch(statusLine) | ||||
| 	if len(matches) != 4 { | ||||
| 		return 0, 0, 0, fmt.Errorf("couldn't find all the substring matches: %s", statusLine) | ||||
| 	if len(matches) != 5 { | ||||
| 		return 0, 0, 0, 0, fmt.Errorf("couldn't find all the substring matches: %s", statusLine) | ||||
| 	} | ||||
| 
 | ||||
| 	total, err = strconv.ParseInt(matches[2], 10, 64) | ||||
| 	if err != nil { | ||||
| 		return 0, 0, 0, fmt.Errorf("unexpected statusLine %q: %w", statusLine, err) | ||||
| 		return 0, 0, 0, 0, fmt.Errorf("unexpected statusLine %q: %w", statusLine, err) | ||||
| 	} | ||||
| 
 | ||||
| 	active, err = strconv.ParseInt(matches[3], 10, 64) | ||||
| 	if err != nil { | ||||
| 		return 0, 0, 0, fmt.Errorf("unexpected statusLine %q: %w", statusLine, err) | ||||
| 		return 0, 0, 0, 0, fmt.Errorf("unexpected statusLine %q: %w", statusLine, err) | ||||
| 	} | ||||
| 	down = int64(strings.Count(matches[4], "_")) | ||||
| 
 | ||||
| 	return active, total, size, nil | ||||
| 	return active, total, down, size, nil | ||||
| } | ||||
| 
 | ||||
| func evalRecoveryLine(recoveryLine string) (syncedBlocks int64, err error) { | ||||
| 	matches := recoveryLineRE.FindStringSubmatch(recoveryLine) | ||||
| func evalRecoveryLine(recoveryLine string) (syncedBlocks int64, pct float64, finish float64, speed float64, err error) { | ||||
| 	matches := recoveryLineBlocksRE.FindStringSubmatch(recoveryLine) | ||||
| 	if len(matches) != 2 { | ||||
| 		return 0, fmt.Errorf("unexpected recoveryLine: %s", recoveryLine) | ||||
| 		return 0, 0, 0, 0, fmt.Errorf("unexpected recoveryLine: %s", recoveryLine) | ||||
| 	} | ||||
| 
 | ||||
| 	syncedBlocks, err = strconv.ParseInt(matches[1], 10, 64) | ||||
| 	if err != nil { | ||||
| 		return 0, fmt.Errorf("error parsing int from recoveryLine %q: %w", recoveryLine, err) | ||||
| 		return 0, 0, 0, 0, fmt.Errorf("error parsing int from recoveryLine %q: %w", recoveryLine, err) | ||||
| 	} | ||||
| 
 | ||||
| 	return syncedBlocks, nil | ||||
| 	// Get percentage complete
 | ||||
| 	matches = recoveryLinePctRE.FindStringSubmatch(recoveryLine) | ||||
| 	if len(matches) != 2 { | ||||
| 		return syncedBlocks, 0, 0, 0, fmt.Errorf("unexpected recoveryLine matching percentage: %s", recoveryLine) | ||||
| 	} | ||||
| 	pct, err = strconv.ParseFloat(strings.TrimSpace(matches[1]), 64) | ||||
| 	if err != nil { | ||||
| 		return syncedBlocks, 0, 0, 0, fmt.Errorf("error parsing float from recoveryLine %q: %w", recoveryLine, err) | ||||
| 	} | ||||
| 
 | ||||
| 	// Get time expected left to complete
 | ||||
| 	matches = recoveryLineFinishRE.FindStringSubmatch(recoveryLine) | ||||
| 	if len(matches) != 2 { | ||||
| 		return syncedBlocks, pct, 0, 0, fmt.Errorf("unexpected recoveryLine matching est. finish time: %s", recoveryLine) | ||||
| 	} | ||||
| 	finish, err = strconv.ParseFloat(matches[1], 64) | ||||
| 	if err != nil { | ||||
| 		return syncedBlocks, pct, 0, 0, fmt.Errorf("error parsing float from recoveryLine %q: %w", recoveryLine, err) | ||||
| 	} | ||||
| 
 | ||||
| 	// Get recovery speed
 | ||||
| 	matches = recoveryLineSpeedRE.FindStringSubmatch(recoveryLine) | ||||
| 	if len(matches) != 2 { | ||||
| 		return syncedBlocks, pct, finish, 0, fmt.Errorf("unexpected recoveryLine matching speed: %s", recoveryLine) | ||||
| 	} | ||||
| 	speed, err = strconv.ParseFloat(matches[1], 64) | ||||
| 	if err != nil { | ||||
| 		return syncedBlocks, pct, finish, 0, fmt.Errorf("error parsing float from recoveryLine %q: %w", recoveryLine, err) | ||||
| 	} | ||||
| 
 | ||||
| 	return syncedBlocks, pct, finish, speed, nil | ||||
| } | ||||
| 
 | ||||
| func evalComponentDevices(deviceFields []string) []string { | ||||
|  |  | |||
|  | @ -65,6 +65,7 @@ type ( | |||
| 		TxQueue   uint64 | ||||
| 		RxQueue   uint64 | ||||
| 		UID       uint64 | ||||
| 		Inode     uint64 | ||||
| 	} | ||||
| ) | ||||
| 
 | ||||
|  | @ -150,9 +151,9 @@ func parseIP(hexIP string) (net.IP, error) { | |||
| // parseNetIPSocketLine parses a single line, represented by a list of fields.
 | ||||
| func parseNetIPSocketLine(fields []string) (*netIPSocketLine, error) { | ||||
| 	line := &netIPSocketLine{} | ||||
| 	if len(fields) < 8 { | ||||
| 	if len(fields) < 10 { | ||||
| 		return nil, fmt.Errorf( | ||||
| 			"cannot parse net socket line as it has less then 8 columns %q", | ||||
| 			"cannot parse net socket line as it has less then 10 columns %q", | ||||
| 			strings.Join(fields, " "), | ||||
| 		) | ||||
| 	} | ||||
|  | @ -216,5 +217,10 @@ func parseNetIPSocketLine(fields []string) (*netIPSocketLine, error) { | |||
| 		return nil, fmt.Errorf("cannot parse uid value in socket line: %w", err) | ||||
| 	} | ||||
| 
 | ||||
| 	// inode
 | ||||
| 	if line.Inode, err = strconv.ParseUint(fields[9], 0, 64); err != nil { | ||||
| 		return nil, fmt.Errorf("cannot parse inode value in socket line: %w", err) | ||||
| 	} | ||||
| 
 | ||||
| 	return line, nil | ||||
| } | ||||
|  |  | |||
|  | @ -0,0 +1,68 @@ | |||
| // Copyright 2020 The Prometheus 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 procfs | ||||
| 
 | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| // NetStat contains statistics for all the counters from one file
 | ||||
| type NetStat struct { | ||||
| 	Filename string | ||||
| 	Stats    map[string][]uint64 | ||||
| } | ||||
| 
 | ||||
| // NetStat retrieves stats from /proc/net/stat/
 | ||||
| func (fs FS) NetStat() ([]NetStat, error) { | ||||
| 	statFiles, err := filepath.Glob(fs.proc.Path("net/stat/*")) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	var netStatsTotal []NetStat | ||||
| 
 | ||||
| 	for _, filePath := range statFiles { | ||||
| 		file, err := os.Open(filePath) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 
 | ||||
| 		netStatFile := NetStat{ | ||||
| 			Filename: filepath.Base(filePath), | ||||
| 			Stats:    make(map[string][]uint64), | ||||
| 		} | ||||
| 		scanner := bufio.NewScanner(file) | ||||
| 		scanner.Scan() | ||||
| 		// First string is always a header for stats
 | ||||
| 		var headers []string | ||||
| 		headers = append(headers, strings.Fields(scanner.Text())...) | ||||
| 
 | ||||
| 		// Other strings represent per-CPU counters
 | ||||
| 		for scanner.Scan() { | ||||
| 			for num, counter := range strings.Fields(scanner.Text()) { | ||||
| 				value, err := strconv.ParseUint(counter, 16, 32) | ||||
| 				if err != nil { | ||||
| 					return nil, err | ||||
| 				} | ||||
| 				netStatFile.Stats[headers[num]] = append(netStatFile.Stats[headers[num]], value) | ||||
| 			} | ||||
| 		} | ||||
| 		netStatsTotal = append(netStatsTotal, netStatFile) | ||||
| 	} | ||||
| 	return netStatsTotal, nil | ||||
| } | ||||
|  | @ -90,7 +90,7 @@ func parseCgroups(data []byte) ([]Cgroup, error) { | |||
| // control hierarchy running on this system. On every system (v1 and v2), all hierarchies contain all processes,
 | ||||
| // so the len of the returned struct is equal to the number of active hierarchies on this system
 | ||||
| func (p Proc) Cgroups() ([]Cgroup, error) { | ||||
| 	data, err := util.ReadFileNoStat(fmt.Sprintf("/proc/%d/cgroup", p.PID)) | ||||
| 	data, err := util.ReadFileNoStat(p.path("cgroup")) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  |  | |||
|  | @ -100,6 +100,15 @@ type ProcStat struct { | |||
| 	VSize uint | ||||
| 	// Resident set size in pages.
 | ||||
| 	RSS int | ||||
| 	// Soft limit in bytes on the rss of the process.
 | ||||
| 	RSSLimit uint64 | ||||
| 	// Real-time scheduling priority, a number in the range 1 to 99 for processes
 | ||||
| 	// scheduled under a real-time policy, or 0, for non-real-time processes.
 | ||||
| 	RTPriority uint | ||||
| 	// Scheduling policy.
 | ||||
| 	Policy uint | ||||
| 	// Aggregated block I/O delays, measured in clock ticks (centiseconds).
 | ||||
| 	DelayAcctBlkIOTicks uint64 | ||||
| 
 | ||||
| 	proc fs.FS | ||||
| } | ||||
|  | @ -119,7 +128,8 @@ func (p Proc) Stat() (ProcStat, error) { | |||
| 	} | ||||
| 
 | ||||
| 	var ( | ||||
| 		ignore int | ||||
| 		ignoreInt64  int64 | ||||
| 		ignoreUint64 uint64 | ||||
| 
 | ||||
| 		s = ProcStat{PID: p.PID, proc: p.fs} | ||||
| 		l = bytes.Index(data, []byte("(")) | ||||
|  | @ -151,10 +161,28 @@ func (p Proc) Stat() (ProcStat, error) { | |||
| 		&s.Priority, | ||||
| 		&s.Nice, | ||||
| 		&s.NumThreads, | ||||
| 		&ignore, | ||||
| 		&ignoreInt64, | ||||
| 		&s.Starttime, | ||||
| 		&s.VSize, | ||||
| 		&s.RSS, | ||||
| 		&s.RSSLimit, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreUint64, | ||||
| 		&ignoreInt64, | ||||
| 		&ignoreInt64, | ||||
| 		&s.RTPriority, | ||||
| 		&s.Policy, | ||||
| 		&s.DelayAcctBlkIOTicks, | ||||
| 	) | ||||
| 	if err != nil { | ||||
| 		return ProcStat{}, err | ||||
|  |  | |||
|  | @ -99,7 +99,6 @@ func parseZoneinfo(zoneinfoData []byte) ([]Zoneinfo, error) { | |||
| 				continue | ||||
| 			} | ||||
| 			if strings.HasPrefix(strings.TrimSpace(line), "per-node stats") { | ||||
| 				zoneinfoElement.Zone = "" | ||||
| 				continue | ||||
| 			} | ||||
| 			parts := strings.Fields(strings.TrimSpace(line)) | ||||
|  |  | |||
|  | @ -174,25 +174,32 @@ type ClientConn interface { | |||
| 
 | ||||
| // BuildOptions contains additional information for Build.
 | ||||
| type BuildOptions struct { | ||||
| 	// DialCreds is the transport credential the Balancer implementation can
 | ||||
| 	// use to dial to a remote load balancer server. The Balancer implementations
 | ||||
| 	// can ignore this if it does not need to talk to another party securely.
 | ||||
| 	// DialCreds is the transport credentials to use when communicating with a
 | ||||
| 	// remote load balancer server. Balancer implementations which do not
 | ||||
| 	// communicate with a remote load balancer server can ignore this field.
 | ||||
| 	DialCreds credentials.TransportCredentials | ||||
| 	// CredsBundle is the credentials bundle that the Balancer can use.
 | ||||
| 	// CredsBundle is the credentials bundle to use when communicating with a
 | ||||
| 	// remote load balancer server. Balancer implementations which do not
 | ||||
| 	// communicate with a remote load balancer server can ignore this field.
 | ||||
| 	CredsBundle credentials.Bundle | ||||
| 	// Dialer is the custom dialer the Balancer implementation can use to dial
 | ||||
| 	// to a remote load balancer server. The Balancer implementations
 | ||||
| 	// can ignore this if it doesn't need to talk to remote balancer.
 | ||||
| 	// Dialer is the custom dialer to use when communicating with a remote load
 | ||||
| 	// balancer server. Balancer implementations which do not communicate with a
 | ||||
| 	// remote load balancer server can ignore this field.
 | ||||
| 	Dialer func(context.Context, string) (net.Conn, error) | ||||
| 	// ChannelzParentID is the entity parent's channelz unique identification number.
 | ||||
| 	// Authority is the server name to use as part of the authentication
 | ||||
| 	// handshake when communicating with a remote load balancer server. Balancer
 | ||||
| 	// implementations which do not communicate with a remote load balancer
 | ||||
| 	// server can ignore this field.
 | ||||
| 	Authority string | ||||
| 	// ChannelzParentID is the parent ClientConn's channelz ID.
 | ||||
| 	ChannelzParentID int64 | ||||
| 	// CustomUserAgent is the custom user agent set on the parent ClientConn.
 | ||||
| 	// The balancer should set the same custom user agent if it creates a
 | ||||
| 	// ClientConn.
 | ||||
| 	CustomUserAgent string | ||||
| 	// Target contains the parsed address info of the dial target. It is the same resolver.Target as
 | ||||
| 	// passed to the resolver.
 | ||||
| 	// See the documentation for the resolver.Target type for details about what it contains.
 | ||||
| 	// Target contains the parsed address info of the dial target. It is the
 | ||||
| 	// same resolver.Target as passed to the resolver. See the documentation for
 | ||||
| 	// the resolver.Target type for details about what it contains.
 | ||||
| 	Target resolver.Target | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -83,13 +83,13 @@ var ( | |||
| 	// errTransportCredsAndBundle indicates that creds bundle is used together
 | ||||
| 	// with other individual Transport Credentials.
 | ||||
| 	errTransportCredsAndBundle = errors.New("grpc: credentials.Bundle may not be used with individual TransportCredentials") | ||||
| 	// errTransportCredentialsMissing indicates that users want to transmit security
 | ||||
| 	// information (e.g., OAuth2 token) which requires secure connection on an insecure
 | ||||
| 	// connection.
 | ||||
| 	// errNoTransportCredsInBundle indicated that the configured creds bundle
 | ||||
| 	// returned a transport credentials which was nil.
 | ||||
| 	errNoTransportCredsInBundle = errors.New("grpc: credentials.Bundle must return non-nil transport credentials") | ||||
| 	// errTransportCredentialsMissing indicates that users want to transmit
 | ||||
| 	// security information (e.g., OAuth2 token) which requires secure
 | ||||
| 	// connection on an insecure connection.
 | ||||
| 	errTransportCredentialsMissing = errors.New("grpc: the credentials require transport level security (use grpc.WithTransportCredentials() to set)") | ||||
| 	// errCredentialsConflict indicates that grpc.WithTransportCredentials()
 | ||||
| 	// and grpc.WithInsecure() are both called for a connection.
 | ||||
| 	errCredentialsConflict = errors.New("grpc: transport credentials are set for an insecure connection (grpc.WithTransportCredentials() and grpc.WithInsecure() are both called)") | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
|  | @ -177,17 +177,20 @@ func DialContext(ctx context.Context, target string, opts ...DialOption) (conn * | |||
| 		cc.csMgr.channelzID = cc.channelzID | ||||
| 	} | ||||
| 
 | ||||
| 	if !cc.dopts.insecure { | ||||
| 		if cc.dopts.copts.TransportCredentials == nil && cc.dopts.copts.CredsBundle == nil { | ||||
| 			return nil, errNoTransportSecurity | ||||
| 		} | ||||
| 		if cc.dopts.copts.TransportCredentials != nil && cc.dopts.copts.CredsBundle != nil { | ||||
| 			return nil, errTransportCredsAndBundle | ||||
| 		} | ||||
| 	} else { | ||||
| 		if cc.dopts.copts.TransportCredentials != nil || cc.dopts.copts.CredsBundle != nil { | ||||
| 			return nil, errCredentialsConflict | ||||
| 		} | ||||
| 	if cc.dopts.copts.TransportCredentials == nil && cc.dopts.copts.CredsBundle == nil { | ||||
| 		return nil, errNoTransportSecurity | ||||
| 	} | ||||
| 	if cc.dopts.copts.TransportCredentials != nil && cc.dopts.copts.CredsBundle != nil { | ||||
| 		return nil, errTransportCredsAndBundle | ||||
| 	} | ||||
| 	if cc.dopts.copts.CredsBundle != nil && cc.dopts.copts.CredsBundle.TransportCredentials() == nil { | ||||
| 		return nil, errNoTransportCredsInBundle | ||||
| 	} | ||||
| 	transportCreds := cc.dopts.copts.TransportCredentials | ||||
| 	if transportCreds == nil { | ||||
| 		transportCreds = cc.dopts.copts.CredsBundle.TransportCredentials() | ||||
| 	} | ||||
| 	if transportCreds.Info().SecurityProtocol == "insecure" { | ||||
| 		for _, cd := range cc.dopts.copts.PerRPCCredentials { | ||||
| 			if cd.RequireTransportSecurity() { | ||||
| 				return nil, errTransportCredentialsMissing | ||||
|  | @ -282,6 +285,7 @@ func DialContext(ctx context.Context, target string, opts ...DialOption) (conn * | |||
| 		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, | ||||
|  | @ -629,7 +633,10 @@ func (cc *ClientConn) updateResolverState(s resolver.State, err error) error { | |||
| 	} | ||||
| 
 | ||||
| 	var ret error | ||||
| 	if cc.dopts.disableServiceConfig || s.ServiceConfig == nil { | ||||
| 	if cc.dopts.disableServiceConfig { | ||||
| 		channelz.Infof(logger, cc.channelzID, "ignoring service config from resolver (%v) and applying the default because service config is disabled", s.ServiceConfig) | ||||
| 		cc.maybeApplyDefaultServiceConfig(s.Addresses) | ||||
| 	} else if s.ServiceConfig == nil { | ||||
| 		cc.maybeApplyDefaultServiceConfig(s.Addresses) | ||||
| 		// TODO: do we need to apply a failing LB policy if there is no
 | ||||
| 		// default, per the error handling design?
 | ||||
|  |  | |||
|  | @ -178,8 +178,18 @@ type TransportCredentials interface { | |||
| //
 | ||||
| // This API is experimental.
 | ||||
| type Bundle interface { | ||||
| 	// TransportCredentials returns the transport credentials from the Bundle.
 | ||||
| 	//
 | ||||
| 	// Implementations must return non-nil transport credentials. If transport
 | ||||
| 	// security is not needed by the Bundle, implementations may choose to
 | ||||
| 	// return insecure.NewCredentials().
 | ||||
| 	TransportCredentials() TransportCredentials | ||||
| 
 | ||||
| 	// PerRPCCredentials returns the per-RPC credentials from the Bundle.
 | ||||
| 	//
 | ||||
| 	// May be nil if per-RPC credentials are not needed.
 | ||||
| 	PerRPCCredentials() PerRPCCredentials | ||||
| 
 | ||||
| 	// NewWithMode should make a copy of Bundle, and switch mode. Modifying the
 | ||||
| 	// existing Bundle may cause races.
 | ||||
| 	//
 | ||||
|  |  | |||
|  | @ -0,0 +1,77 @@ | |||
| /* | ||||
|  * | ||||
|  * Copyright 2020 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 insecure provides an implementation of the
 | ||||
| // credentials.TransportCredentials interface which disables transport security.
 | ||||
| //
 | ||||
| // Experimental
 | ||||
| //
 | ||||
| // Notice: This package is EXPERIMENTAL and may be changed or removed in a
 | ||||
| // later release.
 | ||||
| package insecure | ||||
| 
 | ||||
| import ( | ||||
| 	"context" | ||||
| 	"net" | ||||
| 
 | ||||
| 	"google.golang.org/grpc/credentials" | ||||
| ) | ||||
| 
 | ||||
| // NewCredentials returns a credentials which disables transport security.
 | ||||
| //
 | ||||
| // Note that using this credentials with per-RPC credentials which require
 | ||||
| // transport security is incompatible and will cause grpc.Dial() to fail.
 | ||||
| func NewCredentials() credentials.TransportCredentials { | ||||
| 	return insecureTC{} | ||||
| } | ||||
| 
 | ||||
| // insecureTC implements the insecure transport credentials. The handshake
 | ||||
| // methods simply return the passed in net.Conn and set the security level to
 | ||||
| // NoSecurity.
 | ||||
| type insecureTC struct{} | ||||
| 
 | ||||
| func (insecureTC) ClientHandshake(ctx context.Context, _ string, conn net.Conn) (net.Conn, credentials.AuthInfo, error) { | ||||
| 	return conn, info{credentials.CommonAuthInfo{SecurityLevel: credentials.NoSecurity}}, nil | ||||
| } | ||||
| 
 | ||||
| func (insecureTC) ServerHandshake(conn net.Conn) (net.Conn, credentials.AuthInfo, error) { | ||||
| 	return conn, info{credentials.CommonAuthInfo{SecurityLevel: credentials.NoSecurity}}, nil | ||||
| } | ||||
| 
 | ||||
| func (insecureTC) Info() credentials.ProtocolInfo { | ||||
| 	return credentials.ProtocolInfo{SecurityProtocol: "insecure"} | ||||
| } | ||||
| 
 | ||||
| func (insecureTC) Clone() credentials.TransportCredentials { | ||||
| 	return insecureTC{} | ||||
| } | ||||
| 
 | ||||
| func (insecureTC) OverrideServerName(string) error { | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // info contains the auth information for an insecure connection.
 | ||||
| // It implements the AuthInfo interface.
 | ||||
| type info struct { | ||||
| 	credentials.CommonAuthInfo | ||||
| } | ||||
| 
 | ||||
| // AuthType returns the type of info as a string.
 | ||||
| func (info) AuthType() string { | ||||
| 	return "insecure" | ||||
| } | ||||
|  | @ -27,9 +27,9 @@ import ( | |||
| 	"google.golang.org/grpc/backoff" | ||||
| 	"google.golang.org/grpc/balancer" | ||||
| 	"google.golang.org/grpc/credentials" | ||||
| 	"google.golang.org/grpc/credentials/insecure" | ||||
| 	"google.golang.org/grpc/internal" | ||||
| 	internalbackoff "google.golang.org/grpc/internal/backoff" | ||||
| 	"google.golang.org/grpc/internal/envconfig" | ||||
| 	"google.golang.org/grpc/internal/transport" | ||||
| 	"google.golang.org/grpc/keepalive" | ||||
| 	"google.golang.org/grpc/resolver" | ||||
|  | @ -50,7 +50,6 @@ type dialOptions struct { | |||
| 	bs              internalbackoff.Strategy | ||||
| 	block           bool | ||||
| 	returnLastError bool | ||||
| 	insecure        bool | ||||
| 	timeout         time.Duration | ||||
| 	scChan          <-chan ServiceConfig | ||||
| 	authority       string | ||||
|  | @ -228,18 +227,14 @@ func WithServiceConfig(c <-chan ServiceConfig) DialOption { | |||
| 	}) | ||||
| } | ||||
| 
 | ||||
| // WithConnectParams configures the dialer to use the provided ConnectParams.
 | ||||
| // WithConnectParams configures the ClientConn to use the provided ConnectParams
 | ||||
| // for creating and maintaining connections to servers.
 | ||||
| //
 | ||||
| // The backoff configuration specified as part of the ConnectParams overrides
 | ||||
| // all defaults specified in
 | ||||
| // https://github.com/grpc/grpc/blob/master/doc/connection-backoff.md. Consider
 | ||||
| // using the backoff.DefaultConfig as a base, in cases where you want to
 | ||||
| // override only a subset of the backoff configuration.
 | ||||
| //
 | ||||
| // Experimental
 | ||||
| //
 | ||||
| // Notice: This API is EXPERIMENTAL and may be changed or removed in a
 | ||||
| // later release.
 | ||||
| func WithConnectParams(p ConnectParams) DialOption { | ||||
| 	return newFuncDialOption(func(o *dialOptions) { | ||||
| 		o.bs = internalbackoff.Exponential{Config: p.Backoff} | ||||
|  | @ -303,11 +298,17 @@ func WithReturnConnectionError() DialOption { | |||
| } | ||||
| 
 | ||||
| // WithInsecure returns a DialOption which disables transport security for this
 | ||||
| // ClientConn. Note that transport security is required unless WithInsecure is
 | ||||
| // set.
 | ||||
| // ClientConn. Under the hood, it uses insecure.NewCredentials().
 | ||||
| //
 | ||||
| // Note that using this DialOption with per-RPC credentials (through
 | ||||
| // WithCredentialsBundle or WithPerRPCCredentials) which require transport
 | ||||
| // security is incompatible and will cause grpc.Dial() to fail.
 | ||||
| //
 | ||||
| // Deprecated: use insecure.NewCredentials() instead.
 | ||||
| // Will be supported throughout 1.x.
 | ||||
| func WithInsecure() DialOption { | ||||
| 	return newFuncDialOption(func(o *dialOptions) { | ||||
| 		o.insecure = true | ||||
| 		o.copts.TransportCredentials = insecure.NewCredentials() | ||||
| 	}) | ||||
| } | ||||
| 
 | ||||
|  | @ -580,7 +581,6 @@ func withHealthCheckFunc(f internal.HealthChecker) DialOption { | |||
| 
 | ||||
| func defaultDialOptions() dialOptions { | ||||
| 	return dialOptions{ | ||||
| 		disableRetry:    !envconfig.Retry, | ||||
| 		healthCheckFunc: internal.HealthCheckFunc, | ||||
| 		copts: transport.ConnectOptions{ | ||||
| 			WriteBufferSize: defaultWriteBufSize, | ||||
|  |  | |||
|  | @ -204,9 +204,9 @@ func RegisterChannel(c Channel, pid int64, ref string) int64 { | |||
| 		trace:       &channelTrace{createdTime: time.Now(), events: make([]*TraceEvent, 0, getMaxTraceEntry())}, | ||||
| 	} | ||||
| 	if pid == 0 { | ||||
| 		db.get().addChannel(id, cn, true, pid, ref) | ||||
| 		db.get().addChannel(id, cn, true, pid) | ||||
| 	} else { | ||||
| 		db.get().addChannel(id, cn, false, pid, ref) | ||||
| 		db.get().addChannel(id, cn, false, pid) | ||||
| 	} | ||||
| 	return id | ||||
| } | ||||
|  | @ -228,7 +228,7 @@ func RegisterSubChannel(c Channel, pid int64, ref string) int64 { | |||
| 		pid:     pid, | ||||
| 		trace:   &channelTrace{createdTime: time.Now(), events: make([]*TraceEvent, 0, getMaxTraceEntry())}, | ||||
| 	} | ||||
| 	db.get().addSubChannel(id, sc, pid, ref) | ||||
| 	db.get().addSubChannel(id, sc, pid) | ||||
| 	return id | ||||
| } | ||||
| 
 | ||||
|  | @ -258,7 +258,7 @@ func RegisterListenSocket(s Socket, pid int64, ref string) int64 { | |||
| 	} | ||||
| 	id := idGen.genID() | ||||
| 	ls := &listenSocket{refName: ref, s: s, id: id, pid: pid} | ||||
| 	db.get().addListenSocket(id, ls, pid, ref) | ||||
| 	db.get().addListenSocket(id, ls, pid) | ||||
| 	return id | ||||
| } | ||||
| 
 | ||||
|  | @ -273,11 +273,11 @@ func RegisterNormalSocket(s Socket, pid int64, ref string) int64 { | |||
| 	} | ||||
| 	id := idGen.genID() | ||||
| 	ns := &normalSocket{refName: ref, s: s, id: id, pid: pid} | ||||
| 	db.get().addNormalSocket(id, ns, pid, ref) | ||||
| 	db.get().addNormalSocket(id, ns, pid) | ||||
| 	return id | ||||
| } | ||||
| 
 | ||||
| // RemoveEntry removes an entry with unique channelz trakcing id to be id from
 | ||||
| // RemoveEntry removes an entry with unique channelz tracking id to be id from
 | ||||
| // channelz database.
 | ||||
| func RemoveEntry(id int64) { | ||||
| 	db.get().removeEntry(id) | ||||
|  | @ -333,7 +333,7 @@ func (c *channelMap) addServer(id int64, s *server) { | |||
| 	c.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| func (c *channelMap) addChannel(id int64, cn *channel, isTopChannel bool, pid int64, ref string) { | ||||
| func (c *channelMap) addChannel(id int64, cn *channel, isTopChannel bool, pid int64) { | ||||
| 	c.mu.Lock() | ||||
| 	cn.cm = c | ||||
| 	cn.trace.cm = c | ||||
|  | @ -346,7 +346,7 @@ func (c *channelMap) addChannel(id int64, cn *channel, isTopChannel bool, pid in | |||
| 	c.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| func (c *channelMap) addSubChannel(id int64, sc *subChannel, pid int64, ref string) { | ||||
| func (c *channelMap) addSubChannel(id int64, sc *subChannel, pid int64) { | ||||
| 	c.mu.Lock() | ||||
| 	sc.cm = c | ||||
| 	sc.trace.cm = c | ||||
|  | @ -355,7 +355,7 @@ func (c *channelMap) addSubChannel(id int64, sc *subChannel, pid int64, ref stri | |||
| 	c.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| func (c *channelMap) addListenSocket(id int64, ls *listenSocket, pid int64, ref string) { | ||||
| func (c *channelMap) addListenSocket(id int64, ls *listenSocket, pid int64) { | ||||
| 	c.mu.Lock() | ||||
| 	ls.cm = c | ||||
| 	c.listenSockets[id] = ls | ||||
|  | @ -363,7 +363,7 @@ func (c *channelMap) addListenSocket(id int64, ls *listenSocket, pid int64, ref | |||
| 	c.mu.Unlock() | ||||
| } | ||||
| 
 | ||||
| func (c *channelMap) addNormalSocket(id int64, ns *normalSocket, pid int64, ref string) { | ||||
| func (c *channelMap) addNormalSocket(id int64, ns *normalSocket, pid int64) { | ||||
| 	c.mu.Lock() | ||||
| 	ns.cm = c | ||||
| 	c.normalSockets[id] = ns | ||||
|  |  | |||
|  | @ -22,20 +22,14 @@ package envconfig | |||
| import ( | ||||
| 	"os" | ||||
| 	"strings" | ||||
| 
 | ||||
| 	xdsenv "google.golang.org/grpc/internal/xds/env" | ||||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	prefix          = "GRPC_GO_" | ||||
| 	retryStr        = prefix + "RETRY" | ||||
| 	txtErrIgnoreStr = prefix + "IGNORE_TXT_ERRORS" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	// Retry is enabled unless explicitly disabled via "GRPC_GO_RETRY=off" or
 | ||||
| 	// if XDS retry support is explicitly disabled.
 | ||||
| 	Retry = !strings.EqualFold(os.Getenv(retryStr), "off") && xdsenv.RetrySupport | ||||
| 	// TXTErrIgnore is set if TXT errors should be ignored ("GRPC_GO_IGNORE_TXT_ERRORS" is not "false").
 | ||||
| 	TXTErrIgnore = !strings.EqualFold(os.Getenv(txtErrIgnoreStr), "false") | ||||
| ) | ||||
|  |  | |||
|  | @ -16,9 +16,7 @@ | |||
|  * | ||||
|  */ | ||||
| 
 | ||||
| // Package env acts a single source of definition for all environment variables
 | ||||
| // related to the xDS implementation in gRPC.
 | ||||
| package env | ||||
| package envconfig | ||||
| 
 | ||||
| import ( | ||||
| 	"os" | ||||
|  | @ -26,72 +24,67 @@ import ( | |||
| ) | ||||
| 
 | ||||
| const ( | ||||
| 	// BootstrapFileNameEnv is the env variable to set bootstrap file name.
 | ||||
| 	// XDSBootstrapFileNameEnv is the env variable to set bootstrap file name.
 | ||||
| 	// Do not use this and read from env directly. Its value is read and kept in
 | ||||
| 	// variable BootstrapFileName.
 | ||||
| 	//
 | ||||
| 	// When both bootstrap FileName and FileContent are set, FileName is used.
 | ||||
| 	BootstrapFileNameEnv = "GRPC_XDS_BOOTSTRAP" | ||||
| 	// BootstrapFileContentEnv is the env variable to set bootstrapp file
 | ||||
| 	XDSBootstrapFileNameEnv = "GRPC_XDS_BOOTSTRAP" | ||||
| 	// XDSBootstrapFileContentEnv is the env variable to set bootstrapp file
 | ||||
| 	// content. Do not use this and read from env directly. Its value is read
 | ||||
| 	// and kept in variable BootstrapFileName.
 | ||||
| 	//
 | ||||
| 	// When both bootstrap FileName and FileContent are set, FileName is used.
 | ||||
| 	BootstrapFileContentEnv = "GRPC_XDS_BOOTSTRAP_CONFIG" | ||||
| 	XDSBootstrapFileContentEnv = "GRPC_XDS_BOOTSTRAP_CONFIG" | ||||
| 
 | ||||
| 	ringHashSupportEnv           = "GRPC_XDS_EXPERIMENTAL_ENABLE_RING_HASH" | ||||
| 	clientSideSecuritySupportEnv = "GRPC_XDS_EXPERIMENTAL_SECURITY_SUPPORT" | ||||
| 	aggregateAndDNSSupportEnv    = "GRPC_XDS_EXPERIMENTAL_ENABLE_AGGREGATE_AND_LOGICAL_DNS_CLUSTER" | ||||
| 	retrySupportEnv              = "GRPC_XDS_EXPERIMENTAL_ENABLE_RETRY" | ||||
| 	rbacSupportEnv               = "GRPC_XDS_EXPERIMENTAL_RBAC" | ||||
| 	federationEnv                = "GRPC_EXPERIMENTAL_XDS_FEDERATION" | ||||
| 
 | ||||
| 	c2pResolverSupportEnv                    = "GRPC_EXPERIMENTAL_GOOGLE_C2P_RESOLVER" | ||||
| 	c2pResolverTestOnlyTrafficDirectorURIEnv = "GRPC_TEST_ONLY_GOOGLE_C2P_RESOLVER_TRAFFIC_DIRECTOR_URI" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	// BootstrapFileName holds the name of the file which contains xDS bootstrap
 | ||||
| 	// configuration. Users can specify the location of the bootstrap file by
 | ||||
| 	// setting the environment variable "GRPC_XDS_BOOTSTRAP".
 | ||||
| 	// XDSBootstrapFileName holds the name of the file which contains xDS
 | ||||
| 	// bootstrap configuration. Users can specify the location of the bootstrap
 | ||||
| 	// file by setting the environment variable "GRPC_XDS_BOOTSTRAP".
 | ||||
| 	//
 | ||||
| 	// When both bootstrap FileName and FileContent are set, FileName is used.
 | ||||
| 	BootstrapFileName = os.Getenv(BootstrapFileNameEnv) | ||||
| 	// BootstrapFileContent holds the content of the xDS bootstrap
 | ||||
| 	// configuration. Users can specify the bootstrap config by
 | ||||
| 	// setting the environment variable "GRPC_XDS_BOOTSTRAP_CONFIG".
 | ||||
| 	XDSBootstrapFileName = os.Getenv(XDSBootstrapFileNameEnv) | ||||
| 	// XDSBootstrapFileContent holds the content of the xDS bootstrap
 | ||||
| 	// configuration. Users can specify the bootstrap config by setting the
 | ||||
| 	// environment variable "GRPC_XDS_BOOTSTRAP_CONFIG".
 | ||||
| 	//
 | ||||
| 	// When both bootstrap FileName and FileContent are set, FileName is used.
 | ||||
| 	BootstrapFileContent = os.Getenv(BootstrapFileContentEnv) | ||||
| 	// RingHashSupport indicates whether ring hash support is enabled, which can
 | ||||
| 	// be disabled by setting the environment variable
 | ||||
| 	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".
 | ||||
| 	RingHashSupport = !strings.EqualFold(os.Getenv(ringHashSupportEnv), "false") | ||||
| 	// ClientSideSecuritySupport is used to control processing of security
 | ||||
| 	XDSRingHash = !strings.EqualFold(os.Getenv(ringHashSupportEnv), "false") | ||||
| 	// 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.
 | ||||
| 	ClientSideSecuritySupport = !strings.EqualFold(os.Getenv(clientSideSecuritySupportEnv), "false") | ||||
| 	// AggregateAndDNSSupportEnv indicates whether processing of aggregated
 | ||||
| 	// cluster and DNS cluster is enabled, which can be enabled by setting the
 | ||||
| 	XDSClientSideSecurity = !strings.EqualFold(os.Getenv(clientSideSecuritySupportEnv), "false") | ||||
| 	// XDSAggregateAndDNS indicates whether processing of aggregated cluster
 | ||||
| 	// and DNS cluster is enabled, which can be enabled by setting the
 | ||||
| 	// environment variable
 | ||||
| 	// "GRPC_XDS_EXPERIMENTAL_ENABLE_AGGREGATE_AND_LOGICAL_DNS_CLUSTER" to
 | ||||
| 	// "true".
 | ||||
| 	AggregateAndDNSSupportEnv = strings.EqualFold(os.Getenv(aggregateAndDNSSupportEnv), "true") | ||||
| 	XDSAggregateAndDNS = strings.EqualFold(os.Getenv(aggregateAndDNSSupportEnv), "true") | ||||
| 
 | ||||
| 	// RetrySupport indicates whether xDS retry is enabled.
 | ||||
| 	RetrySupport = !strings.EqualFold(os.Getenv(retrySupportEnv), "false") | ||||
| 
 | ||||
| 	// RBACSupport indicates whether xDS configured RBAC HTTP Filter is enabled,
 | ||||
| 	// 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".
 | ||||
| 	RBACSupport = !strings.EqualFold(os.Getenv(rbacSupportEnv), "false") | ||||
| 	XDSRBAC = !strings.EqualFold(os.Getenv(rbacSupportEnv), "false") | ||||
| 
 | ||||
| 	// XDSFederation indicates whether federation support is enabled.
 | ||||
| 	XDSFederation = strings.EqualFold(os.Getenv(federationEnv), "true") | ||||
| 
 | ||||
| 	// C2PResolverSupport indicates whether support for C2P resolver is enabled.
 | ||||
| 	// This can be enabled by setting the environment variable
 | ||||
| 	// "GRPC_EXPERIMENTAL_GOOGLE_C2P_RESOLVER" to "true".
 | ||||
| 	C2PResolverSupport = strings.EqualFold(os.Getenv(c2pResolverSupportEnv), "true") | ||||
| 	// C2PResolverTestOnlyTrafficDirectorURI is the TD URI for testing.
 | ||||
| 	C2PResolverTestOnlyTrafficDirectorURI = os.Getenv(c2pResolverTestOnlyTrafficDirectorURIEnv) | ||||
| ) | ||||
|  | @ -136,12 +136,10 @@ type inFlow struct { | |||
| 
 | ||||
| // newLimit updates the inflow window to a new value n.
 | ||||
| // It assumes that n is always greater than the old limit.
 | ||||
| func (f *inFlow) newLimit(n uint32) uint32 { | ||||
| func (f *inFlow) newLimit(n uint32) { | ||||
| 	f.mu.Lock() | ||||
| 	d := n - f.limit | ||||
| 	f.limit = n | ||||
| 	f.mu.Unlock() | ||||
| 	return d | ||||
| } | ||||
| 
 | ||||
| func (f *inFlow) maybeAdjust(n uint32) uint32 { | ||||
|  |  | |||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
		Reference in New Issue