...
1#!/usr/bin/env bash
2
3# Copyright 2017 The Kubernetes Authors.
4#
5# Licensed under the Apache License, Version 2.0 (the "License");
6# you may not use this file except in compliance with the License.
7# You may obtain a copy of the License at
8#
9# http://www.apache.org/licenses/LICENSE-2.0
10#
11# Unless required by applicable law or agreed to in writing, software
12# distributed under the License is distributed on an "AS IS" BASIS,
13# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14# See the License for the specific language governing permissions and
15# limitations under the License.
16
17set -o errexit
18set -o nounset
19set -o pipefail
20
21# Short-circuit if protoc.sh has already been sourced
22[[ $(type -t kube::protoc::loaded) == function ]] && return 0
23
24# The root of the build/dist directory
25KUBE_ROOT="$(cd "$(dirname "${BASH_SOURCE[0]}")/../.." && pwd -P)"
26source "${KUBE_ROOT}/hack/lib/init.sh"
27
28PROTOC_VERSION=23.4
29
30# Generates $1/api.pb.go from the protobuf file $1/api.proto
31# and formats it correctly
32# $1: Full path to the directory where the api.proto file is
33function kube::protoc::generate_proto() {
34 kube::golang::setup_env
35 GOPROXY=off go install k8s.io/code-generator/cmd/go-to-protobuf/protoc-gen-gogo
36
37 kube::protoc::check_protoc
38
39 local package=${1}
40 kube::protoc::protoc "${package}"
41 kube::protoc::format "${package}"
42}
43
44# Checks that the current protoc version matches the required version and
45# exit 1 if it's not the case
46function kube::protoc::check_protoc() {
47 if [[ -z "$(which protoc)" || "$(protoc --version)" != "libprotoc ${PROTOC_VERSION}"* ]]; then
48 echo "Generating protobuf requires protoc ${PROTOC_VERSION}."
49 echo "Run hack/install-protoc.sh or download and install the"
50 echo "platform-appropriate Protobuf package for your OS from"
51 echo "https://github.com/protocolbuffers/protobuf/releases"
52 return 1
53 fi
54}
55
56# Generates $1/api.pb.go from the protobuf file $1/api.proto
57# $1: Full path to the directory where the api.proto file is
58function kube::protoc::protoc() {
59 local package=${1}
60 gogopath=$(dirname "$(kube::util::find-binary "protoc-gen-gogo")")
61
62 (
63 cd "${package}"
64
65 # This invocation of --gogo_out produces its output in the current
66 # directory (despite gogo docs saying it would be source-relative, it
67 # isn't). The inputs to this function do not all have a common root, so
68 # this works best for all inputs.
69 PATH="${gogopath}:${PATH}" protoc \
70 --proto_path="$(pwd -P)" \
71 --proto_path="${KUBE_ROOT}/vendor" \
72 --proto_path="${KUBE_ROOT}/staging/src" \
73 --proto_path="${KUBE_ROOT}/third_party/protobuf" \
74 --gogo_out=paths=source_relative,plugins=grpc:. \
75 api.proto
76 )
77}
78
79# Formats $1/api.pb.go, adds the boilerplate comments and run gofmt on it
80# $1: Full path to the directory where the api.proto file is
81function kube::protoc::format() {
82 local package=${1}
83
84 # Update boilerplate for the generated file.
85 cat hack/boilerplate/boilerplate.generatego.txt "${package}/api.pb.go" > tmpfile && mv tmpfile "${package}/api.pb.go"
86
87 # Run gofmt to clean up the generated code.
88 kube::golang::setup_env
89 gofmt -s -w "${package}/api.pb.go"
90}
91
92# Compares the contents of $1 and $2
93# Echo's $3 in case of error and exits 1
94function kube::protoc::diff() {
95 local ret=0
96 diff -I "gzipped FileDescriptorProto" -I "0x" -Naupr "${1}" "${2}" || ret=$?
97 if [[ ${ret} -ne 0 ]]; then
98 echo "${3}"
99 exit 1
100 fi
101}
102
103function kube::protoc::install() {
104 # run in a subshell to isolate caller from directory changes
105 (
106 local os
107 local arch
108 local download_folder
109 local download_file
110
111 os=$(kube::util::host_os)
112 arch=$(kube::util::host_arch)
113 download_folder="protoc-v${PROTOC_VERSION}-${os}-${arch}"
114 download_file="${download_folder}.zip"
115
116 cd "${KUBE_ROOT}/third_party" || return 1
117 if [[ $(readlink protoc) != "${download_folder}" ]]; then
118 local url
119 if [[ ${os} == "darwin" ]]; then
120 # TODO: switch to universal binary when updating to 3.20+
121 url="https://github.com/protocolbuffers/protobuf/releases/download/v${PROTOC_VERSION}/protoc-${PROTOC_VERSION}-osx-x86_64.zip"
122 elif [[ ${os} == "linux" && ${arch} == "amd64" ]]; then
123 url="https://github.com/protocolbuffers/protobuf/releases/download/v${PROTOC_VERSION}/protoc-${PROTOC_VERSION}-linux-x86_64.zip"
124 elif [[ ${os} == "linux" && ${arch} == "arm64" ]]; then
125 url="https://github.com/protocolbuffers/protobuf/releases/download/v${PROTOC_VERSION}/protoc-${PROTOC_VERSION}-linux-aarch_64.zip"
126 else
127 kube::log::info "This install script does not support ${os}/${arch}"
128 return 1
129 fi
130 kube::util::download_file "${url}" "${download_file}"
131 unzip -o "${download_file}" -d "${download_folder}"
132 ln -fns "${download_folder}" protoc
133 mv protoc/bin/protoc protoc/protoc
134 chmod -R +rX protoc/protoc
135 rm -fr protoc/include
136 rm "${download_file}"
137 fi
138 kube::log::info "protoc v${PROTOC_VERSION} installed. To use:"
139 kube::log::info "export PATH=\"$(pwd)/protoc:\${PATH}\""
140 )
141}
142
143# Marker function to indicate protoc.sh has been fully sourced
144kube::protoc::loaded() {
145 return 0
146}
View as plain text