Add Python GCF Distribtest (#29303)
* Initial GCF distribtest * Tenatively hook up to CI * Try again * Allow dev0 artifacts * Fix invocation path * Update gcloud * Add a 3.8 artifact for presubmits * And 3.9 too * Put test files back to normal * Formatting/linting * Copyright * That copyright script doesn't work with shebangs * Review comments * Try to create latest-manylinux label * Accidentally a letter * Add Python 3.7 manylinux 2014 to presubmit * Revert CI config file used for test * Review comments * Yapf * Re-add presubmit wheel * Review commentspull/29350/head
parent
696e4928e6
commit
e53ea89bca
12 changed files with 309 additions and 0 deletions
@ -0,0 +1,19 @@ |
|||||||
|
# This file specifies files that are *not* uploaded to Google Cloud |
||||||
|
# using gcloud. It follows the same syntax as .gitignore, with the addition of |
||||||
|
# "#!include" directives (which insert the entries of the given .gitignore-style |
||||||
|
# file at that point). |
||||||
|
# |
||||||
|
# For more information, run: |
||||||
|
# $ gcloud topic gcloudignore |
||||||
|
# |
||||||
|
.gcloudignore |
||||||
|
# If you would like to upload your .git directory, .gitignore file or files |
||||||
|
# from your .gitignore file, remove the corresponding line |
||||||
|
# below: |
||||||
|
.git |
||||||
|
.gitignore |
||||||
|
|
||||||
|
*.sh |
||||||
|
*.md |
||||||
|
*.whl |
||||||
|
requirements.txt.base |
@ -0,0 +1 @@ |
|||||||
|
requirements.txt |
@ -0,0 +1,12 @@ |
|||||||
|
# Python Google Cloud Functions Distribtest |
||||||
|
|
||||||
|
This distribtest acts as a smoke test for usage of the `grpcio` Python wheel in |
||||||
|
the GCF environment. This test is dependent on two long-lived GCP resources: |
||||||
|
|
||||||
|
- `gcf-distribtest-topic` Pub Sub Topic with default configuration. |
||||||
|
- `grpc-gcf-distribtest` GCS Bucket. This bucket has 1 day TTL on all artifacts. |
||||||
|
|
||||||
|
|
||||||
|
All Functions _should_ be deleted by the test under normal circumstances. In |
||||||
|
case anything goes wrong with this process, a `cleanup.sh` script is provided to |
||||||
|
delete any dangling test functions. |
@ -0,0 +1,24 @@ |
|||||||
|
#!/bin/bash |
||||||
|
# Copyright 2022 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. |
||||||
|
|
||||||
|
cd "$(dirname "$0")" |
||||||
|
|
||||||
|
# Shellcheck cant find the included file. |
||||||
|
# shellcheck disable=SC1091 |
||||||
|
source common.sh |
||||||
|
|
||||||
|
set -euxo pipefail |
||||||
|
|
||||||
|
gcloud functions list | grep "${FUNCTION_PREFIX}" | awk '{print $1;}' | xargs -n1 gcloud functions delete |
@ -0,0 +1,18 @@ |
|||||||
|
#!/bin/bash |
||||||
|
# Copyright 2022 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. |
||||||
|
|
||||||
|
# Shellcheck can't track usage across files. |
||||||
|
# shellcheck disable=SC2034 |
||||||
|
FUNCTION_PREFIX="grpc-gcf-distribtest" |
@ -0,0 +1,32 @@ |
|||||||
|
# Copyright 2022 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. |
||||||
|
|
||||||
|
import functions_framework |
||||||
|
from google.cloud import pubsub_v1 |
||||||
|
|
||||||
|
ps_client = pubsub_v1.PublisherClient() |
||||||
|
_PROJECT_ID = "grpc-testing" |
||||||
|
_PUBSUB_TOPIC = 'gcf-distribtest-topic' |
||||||
|
|
||||||
|
|
||||||
|
@functions_framework.http |
||||||
|
def test_publish(request): |
||||||
|
topic_path = ps_client.topic_path(_PROJECT_ID, _PUBSUB_TOPIC) |
||||||
|
message = '{"function": "TEST"}' |
||||||
|
message_bytes = message.encode('utf-8') |
||||||
|
|
||||||
|
for _ in range(100): |
||||||
|
future = ps_client.publish(topic_path, data=message_bytes) |
||||||
|
|
||||||
|
return "ok", 200 |
@ -0,0 +1 @@ |
|||||||
|
google-cloud-pubsub>=0.0.dev0 |
@ -0,0 +1,52 @@ |
|||||||
|
#!/bin/bash |
||||||
|
# Copyright 2022 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. |
||||||
|
|
||||||
|
set -euxo pipefail |
||||||
|
|
||||||
|
cd "$(dirname "$0")" |
||||||
|
|
||||||
|
ARTIFACT_DIRECTORY="$1" |
||||||
|
|
||||||
|
BUCKET_NAME="grpc-gcf-distribtests" |
||||||
|
|
||||||
|
RUN_ID=$(uuidgen) |
||||||
|
|
||||||
|
FAILED_RUNTIMES="" |
||||||
|
|
||||||
|
# This is the only programmatic way to get access to the list of runtimes. |
||||||
|
# While hacky, it's better than the alternative -- manually upgrading a |
||||||
|
# hand-curated list every few months. |
||||||
|
RUNTIMES=$(gcloud functions deploy --help | grep -Eo "python[0-9]+" | sort | uniq) |
||||||
|
|
||||||
|
while read -r RUNTIME; do |
||||||
|
BARE_VERSION=${RUNTIME//python/} |
||||||
|
|
||||||
|
# We sort to get the latest manylinux version. |
||||||
|
ARTIFACT=$(find "${ARTIFACT_DIRECTORY}" -regex '.*grpcio-[0-9\.]+.+-cp'"${BARE_VERSION}"'-cp'"${BARE_VERSION}"'m?-manylinux.+x86_64\.whl' | sort -r | head -n 1) |
||||||
|
ARTIFACT_BASENAME=$(basename "${ARTIFACT}") |
||||||
|
|
||||||
|
# Upload artifact to GCS so GCF can access it. |
||||||
|
# A 1 day retention policy is active on this bucket. |
||||||
|
gsutil cp "${ARTIFACT}" "gs://${BUCKET_NAME}/${RUN_ID}/${ARTIFACT_BASENAME}" |
||||||
|
|
||||||
|
echo "Testing runtime ${RUNTIME} with artifact ${ARTIFACT_BASENAME}" |
||||||
|
./run_single.sh "${RUNTIME}" "https://storage.googleapis.com/${BUCKET_NAME}/${RUN_ID}/${ARTIFACT_BASENAME}" || FAILED_RUNTIMES="${FAILED_RUNTIMES} ${RUNTIME}" |
||||||
|
done<<<"${RUNTIMES}" |
||||||
|
|
||||||
|
if [ "$FAILED_RUNTIMES" != "" ] |
||||||
|
then |
||||||
|
echo "GCF Distribtest failed: Failing runtimes: ${FAILED_RUNTIMES}" |
||||||
|
exit 1 |
||||||
|
fi |
@ -0,0 +1,54 @@ |
|||||||
|
#!/bin/bash |
||||||
|
# Copyright 2022 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. |
||||||
|
|
||||||
|
cd "$(dirname "$0")" |
||||||
|
|
||||||
|
# Shellcheck cant find the included file. |
||||||
|
# shellcheck disable=SC1091 |
||||||
|
source common.sh |
||||||
|
|
||||||
|
set -euxo pipefail |
||||||
|
|
||||||
|
RUNTIME="$1" |
||||||
|
ARTIFACT_URL="$2" |
||||||
|
|
||||||
|
REQUEST_COUNT=20 |
||||||
|
LOG_QUIESCE_SECONDS=10 |
||||||
|
|
||||||
|
rm -f requirements.txt |
||||||
|
cp requirements.txt.base requirements.txt |
||||||
|
echo "${ARTIFACT_URL}" >>requirements.txt |
||||||
|
|
||||||
|
# Generate Function name. |
||||||
|
FUNCTION_NAME="${FUNCTION_PREFIX}-$(uuidgen)" |
||||||
|
|
||||||
|
function cleanup() { |
||||||
|
# Wait for logs to quiesce. |
||||||
|
sleep "${LOG_QUIESCE_SECONDS}" |
||||||
|
gcloud functions logs read "${FUNCTION_NAME}" || true |
||||||
|
(yes || true) | gcloud functions delete "${FUNCTION_NAME}" |
||||||
|
} |
||||||
|
|
||||||
|
trap cleanup SIGINT SIGTERM EXIT |
||||||
|
|
||||||
|
# Deploy |
||||||
|
DEPLOY_OUTPUT=$(gcloud functions deploy "${FUNCTION_NAME}" --entry-point test_publish --runtime "${RUNTIME}" --trigger-http --allow-unauthenticated) |
||||||
|
HTTP_URL=$(echo "${DEPLOY_OUTPUT}" | grep "url: " | awk '{print $2;}') |
||||||
|
|
||||||
|
# Send Requests |
||||||
|
for _ in $(seq 1 "${REQUEST_COUNT}"); do |
||||||
|
GCP_IDENTITY_TOKEN=$(gcloud auth print-identity-token 2>/dev/null); |
||||||
|
curl -v -H "Authorization: Bearer $GCP_IDENTITY_TOKEN" "${HTTP_URL}" |
||||||
|
done |
@ -0,0 +1,26 @@ |
|||||||
|
# Copyright 2021 The 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. |
||||||
|
|
||||||
|
# Config file for the internal CI (in protobuf text format) |
||||||
|
|
||||||
|
# Location of the continuous shell script in repository. |
||||||
|
build_file: "grpc/tools/internal_ci/linux/grpc_distribtests_gcp_python.sh" |
||||||
|
timeout_mins: 240 |
||||||
|
action { |
||||||
|
define_artifacts { |
||||||
|
regex: "**/*sponge_log.*" |
||||||
|
regex: "github/grpc/reports/**" |
||||||
|
regex: "github/grpc/artifacts/**" |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,66 @@ |
|||||||
|
#!/bin/bash |
||||||
|
# Copyright 2021 The 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. |
||||||
|
|
||||||
|
set -ex |
||||||
|
|
||||||
|
# avoid slow finalization after the script has exited. |
||||||
|
source $(dirname $0)/../../../tools/internal_ci/helper_scripts/move_src_tree_and_respawn_itself_rc |
||||||
|
|
||||||
|
# change to grpc repo root |
||||||
|
cd $(dirname $0)/../../.. |
||||||
|
|
||||||
|
source tools/internal_ci/helper_scripts/prepare_build_linux_rc |
||||||
|
|
||||||
|
# some distribtests use a pre-registered binfmt_misc hook |
||||||
|
# to automatically execute foreign binaries (such as aarch64) |
||||||
|
# under qemu emulator. |
||||||
|
source tools/internal_ci/helper_scripts/prepare_qemu_rc |
||||||
|
|
||||||
|
# configure ccache |
||||||
|
source tools/internal_ci/helper_scripts/prepare_ccache_rc |
||||||
|
|
||||||
|
# Build all python linux artifacts (this step actually builds all the binary wheels and source archives) |
||||||
|
tools/run_tests/task_runner.py -f artifact linux python latest-manylinux ${TASK_RUNNER_EXTRA_FILTERS} -j 12 -x build_artifacts/sponge_log.xml || FAILED="true" |
||||||
|
|
||||||
|
# the next step expects to find the artifacts from the previous step in the "input_artifacts" folder. |
||||||
|
rm -rf input_artifacts |
||||||
|
mkdir -p input_artifacts |
||||||
|
cp -r artifacts/* input_artifacts/ || true |
||||||
|
rm -rf artifacts_from_build_artifacts_step |
||||||
|
mv artifacts artifacts_from_build_artifacts_step || true |
||||||
|
|
||||||
|
# This step simply collects python artifacts from subdirectories of input_artifacts/ and copies them to artifacts/ |
||||||
|
tools/run_tests/task_runner.py -f package linux python -x build_packages/sponge_log.xml || FAILED="true" |
||||||
|
|
||||||
|
# the next step expects to find the artifacts from the previous step in the "input_artifacts" folder. |
||||||
|
# in addition to that, preserve the contents of "artifacts" directory since we want kokoro |
||||||
|
# to upload its contents as job output artifacts |
||||||
|
rm -rf input_artifacts |
||||||
|
mkdir -p input_artifacts |
||||||
|
cp -r artifacts/* input_artifacts/ || true |
||||||
|
|
||||||
|
INPUT_ARTIFACTS=$(realpath input_artifacts) |
||||||
|
|
||||||
|
(yes || true) | gcloud components update |
||||||
|
|
||||||
|
# TODO(rbellevi): Run in a docker image. |
||||||
|
test/distrib/gcf/python/run.sh "$INPUT_ARTIFACTS" |
||||||
|
|
||||||
|
tools/internal_ci/helper_scripts/store_artifacts_from_moved_src_tree.sh |
||||||
|
|
||||||
|
if [ "$FAILED" != "" ] |
||||||
|
then |
||||||
|
exit 1 |
||||||
|
fi |
Loading…
Reference in new issue