Skip to content

Commit 9e60ab1

Browse files
committed
Add build environment validation checks
- Adds a warning if the Python buildpack has been run multiple times in the same build. - Adds a warning if an existing `.heroku/python/` directory is found in the app source. - Improves the error message shown if the buildpack is used on an unsupported stack. Previously the build would fail with a curl download error depending on the availability of assets on S3. This scenario can only happen outside of Heroku, or in an old buildpack is used with a brand new stack (eg for Heroku-26). The two warnings are the first step towards #1704 and #1710, and will be turned into errors in January 2025. Towards #1704 and #1710. GUS-W-17386432. GUS-W-17309709.
1 parent f225177 commit 9e60ab1

File tree

6 files changed

+197
-3
lines changed

6 files changed

+197
-3
lines changed

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,9 @@
22

33
## [Unreleased]
44

5+
- Added a warning if the Python buildpack has been run multiple times in the same build. In January 2025 this warning will be made an error. ([#1724](https://github.com/heroku/heroku-buildpack-python/pull/1724))
6+
- Added a warning if an existing `.heroku/python/` directory is found in the app source. In January 2025 this warning will be made an error. ([#1724](https://github.com/heroku/heroku-buildpack-python/pull/1724))
7+
- Improved the error message shown if the buildpack is used on an unsupported stack. ([#1724](https://github.com/heroku/heroku-buildpack-python/pull/1724))
58
- Fixed Dev Center links to reflect recent article URL changes. ([#1723](https://github.com/heroku/heroku-buildpack-python/pull/1723))
69
- Added metrics for the existence of a uv lockfile. ([#1725](https://github.com/heroku/heroku-buildpack-python/pull/1725))
710

bin/compile

+12-2
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@ BUILDPACK_DIR=$(cd "$(dirname "$(dirname "${BASH_SOURCE[0]}")")" && pwd)
2121
source "${BUILDPACK_DIR}/bin/utils"
2222
source "${BUILDPACK_DIR}/lib/utils.sh"
2323
source "${BUILDPACK_DIR}/lib/cache.sh"
24+
source "${BUILDPACK_DIR}/lib/checks.sh"
2425
source "${BUILDPACK_DIR}/lib/hooks.sh"
2526
source "${BUILDPACK_DIR}/lib/metadata.sh"
2627
source "${BUILDPACK_DIR}/lib/output.sh"
@@ -32,10 +33,15 @@ source "${BUILDPACK_DIR}/lib/poetry.sh"
3233

3334
compile_start_time=$(nowms)
3435

35-
# Initialise metadata store.
36+
# Initialise the buildpack metadata store.
37+
# This is used to track state across builds (for cache invalidation and messaging when build
38+
# configuration changes) and also so that `bin/report` can generate the build report.
3639
meta_init "${CACHE_DIR}" "python"
3740
meta_setup
3841

42+
checks::ensure_supported_stack "${STACK:?Required env var STACK is not set}"
43+
checks::warn_if_duplicate_python_buildpack "${BUILD_DIR}"
44+
3945
# Prepend proper path for old-school virtualenv hackery.
4046
# This may not be necessary.
4147
export PATH=:/usr/local/bin:$PATH
@@ -103,6 +109,10 @@ cd "$BUILD_DIR"
103109
# Runs a `bin/pre_compile` script if found in the app source, allowing build customisation.
104110
hooks::run_hook "pre_compile"
105111

112+
# This check must be after the pre_compile hook, so that we can check not only the original
113+
# app source, but also that the hook hasn't written to '.heroku/python/' either.
114+
checks::warn_if_existing_python_dir_present "${BUILD_DIR}"
115+
106116
package_manager="$(package_manager::determine_package_manager "${BUILD_DIR}")"
107117
meta_set "package_manager" "${package_manager}"
108118

@@ -135,7 +145,7 @@ python_major_version="${python_full_version%.*}"
135145
meta_set "python_version" "${python_full_version}"
136146
meta_set "python_version_major" "${python_major_version}"
137147

138-
cache::restore "${BUILD_DIR}" "${CACHE_DIR}" "${STACK:?}" "${cached_python_full_version}" "${python_full_version}" "${package_manager}"
148+
cache::restore "${BUILD_DIR}" "${CACHE_DIR}" "${STACK}" "${cached_python_full_version}" "${python_full_version}" "${package_manager}"
139149

140150
# The directory for the .profile.d scripts.
141151
mkdir -p "$(dirname "$PROFILE_PATH")"

bin/report

+2
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,8 @@ ALL_OTHER_FIELDS=(
8282
cache_save_duration
8383
dependencies_install_duration
8484
django_collectstatic_duration
85+
duplicate_python_buildpack
86+
existing_python_dir
8587
nltk_downloader_duration
8688
package_manager_install_duration
8789
pipenv_has_lockfile

lib/cache.sh

+1-1
Original file line numberDiff line numberDiff line change
@@ -138,7 +138,7 @@ function cache::restore() {
138138
mkdir -p "${build_dir}/.heroku"
139139

140140
# NB: For now this has to handle files already existing in build_dir since some apps accidentally
141-
# run the Python buildpack twice. TODO: Add an explicit check/error for duplicate buildpacks.
141+
# run the Python buildpack twice. TODO: Refactor this once duplicate buildpacks become an error.
142142
# TODO: Investigate why errors are ignored and ideally stop doing so.
143143
# TODO: Compare the performance of moving the directory vs copying files.
144144
cp -R "${cache_dir}/.heroku/python" "${build_dir}/.heroku/" &>/dev/null || true

lib/checks.sh

+116
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,116 @@
1+
#!/usr/bin/env bash
2+
3+
function checks::ensure_supported_stack() {
4+
local stack="${1}"
5+
6+
case "${stack}" in
7+
heroku-20 | heroku-22 | heroku-24)
8+
return 0
9+
;;
10+
cedar* | heroku-16 | heroku-18)
11+
# This error will only ever be seen on non-Heroku environments, since the
12+
# Heroku build system rejects builds using EOL stacks.
13+
output::error <<-EOF
14+
Error: The '${stack}' stack is no longer supported.
15+
16+
This buildpack no longer supports the '${stack}' stack since it has
17+
reached its end-of-life:
18+
https://devcenter.heroku.com/articles/stack#stack-support-details-for-apps-using-classic-buildpacks
19+
20+
Upgrade to a newer stack to continue using this buildpack.
21+
EOF
22+
meta_set "failure_reason" "stack::eol"
23+
exit 1
24+
;;
25+
*)
26+
output::error <<-EOF
27+
Error: The '${stack}' stack is not recognised.
28+
29+
This buildpack does not recognise or support the '${stack}' stack.
30+
31+
If '${stack}' is a valid stack, make sure that you are using the latest
32+
version of this buildpack and have not pinned to an older release:
33+
https://devcenter.heroku.com/articles/managing-buildpacks#view-your-buildpacks
34+
https://devcenter.heroku.com/articles/managing-buildpacks#classic-buildpacks-references
35+
EOF
36+
meta_set "failure_reason" "stack::unknown"
37+
exit 1
38+
;;
39+
esac
40+
}
41+
42+
# TODO: Turn this into an error in January 2025.
43+
function checks::warn_if_duplicate_python_buildpack() {
44+
local build_dir="${1}"
45+
46+
# The check for the `PYTHONHOME` env var prevents this warning triggering in the case
47+
# where the Python install was committed to the Git repo (which will be handled later).
48+
# (The env var can only have come from the `export` file of an earlier buildpack,
49+
# since app provided config vars haven't been exported to the environment here.)
50+
if [[ -f "${build_dir}/.heroku/python/bin/python" && -v PYTHONHOME ]]; then
51+
output::warning <<-EOF
52+
Warning: The Python buildpack has already been run this build.
53+
54+
An existing Python installation was found in the build directory
55+
from a buildpack run earlier in the build.
56+
57+
This normally means there are duplicate Python buildpacks set
58+
on your app, which is not supported, can cause errors and
59+
slow down builds.
60+
61+
Check the buildpacks set on your app and remove any duplicate
62+
Python buildpack entries:
63+
https://devcenter.heroku.com/articles/managing-buildpacks#view-your-buildpacks
64+
https://devcenter.heroku.com/articles/managing-buildpacks#remove-classic-buildpacks
65+
66+
If you have a use-case that requires duplicate buildpacks,
67+
please comment on:
68+
https://github.com/heroku/heroku-buildpack-python/issues/1704
69+
70+
In January 2025 this warning will be made an error.
71+
EOF
72+
meta_set "duplicate_python_buildpack" "true"
73+
# shellcheck disable=SC2034 # This is used below until we make this check an error.
74+
DUPLICATE_PYTHON_BUILDPACK=1
75+
fi
76+
}
77+
78+
# TODO: Turn this into an error in January 2025.
79+
function checks::warn_if_existing_python_dir_present() {
80+
local build_dir="${1}"
81+
82+
# Avoid warning twice in the case of duplicate buildpacks.
83+
# TODO: Remove this once `warn_if_duplicate_python_buildpack` becomes an error.
84+
if [[ -v DUPLICATE_PYTHON_BUILDPACK ]]; then
85+
return 0
86+
fi
87+
88+
# We use `-e` here to catch the case where `python` is a file rather than a directory.
89+
if [[ -e "${build_dir}/.heroku/python" ]]; then
90+
output::warning <<-EOF
91+
Warning: Existing '.heroku/python/' directory found.
92+
93+
Your app's source code contains an existing directory named
94+
'.heroku/python/', which is where the Python buildpack needs
95+
to install its files. This existing directory contains:
96+
97+
$(find .heroku/python/ -maxdepth 2 || true)
98+
99+
Writing to internal locations used by the Python buildpack
100+
is not supported and can cause unexpected errors.
101+
102+
If you have committed a '.heroku/python/' directory to your
103+
Git repo, you must delete it or use a different location.
104+
105+
Otherwise, check that an earlier buildpack or 'bin/pre_compile'
106+
hook has not created this directory.
107+
108+
If you have a use-case that requires writing to this location,
109+
please comment on:
110+
https://github.com/heroku/heroku-buildpack-python/issues/1704
111+
112+
In January 2025 this warning will be made an error.
113+
EOF
114+
meta_set "existing_python_dir" "true"
115+
fi
116+
}

spec/hatchet/checks_spec.rb

+63
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,76 @@
33
require_relative '../spec_helper'
44

55
RSpec.describe 'Buildpack validation checks' do
6+
context 'when there are duplicate Python buildpacks set on the app' do
7+
let(:buildpacks) { %i[default default] }
8+
let(:app) { Hatchet::Runner.new("spec/fixtures/python_#{DEFAULT_PYTHON_MAJOR_VERSION}", buildpacks:) }
9+
10+
it 'fails detection' do
11+
app.deploy do |app|
12+
expect(clean_output(app.output)).to include(<<~OUTPUT)
13+
remote: -----> Python app detected
14+
remote:
15+
remote: ! Warning: The Python buildpack has already been run this build.
16+
remote: !
17+
remote: ! An existing Python installation was found in the build directory
18+
remote: ! from a buildpack run earlier in the build.
19+
remote: !
20+
remote: ! This normally means there are duplicate Python buildpacks set
21+
remote: ! on your app, which is not supported, can cause errors and
22+
remote: ! slow down builds.
23+
remote: !
24+
remote: ! Check the buildpacks set on your app and remove any duplicate
25+
remote: ! Python buildpack entries:
26+
remote: ! https://devcenter.heroku.com/articles/managing-buildpacks#view-your-buildpacks
27+
remote: ! https://devcenter.heroku.com/articles/managing-buildpacks#remove-classic-buildpacks
28+
remote: !
29+
remote: ! If you have a use-case that requires duplicate buildpacks,
30+
remote: ! please comment on:
31+
remote: ! https://github.com/heroku/heroku-buildpack-python/issues/1704
32+
remote: !
33+
remote: ! In January 2025 this warning will be made an error.
34+
remote:
35+
remote: -----> Using Python #{DEFAULT_PYTHON_MAJOR_VERSION} specified in .python-version
36+
remote: -----> Restoring cache
37+
remote: -----> Using cached install of Python #{DEFAULT_PYTHON_FULL_VERSION}
38+
OUTPUT
39+
end
40+
end
41+
end
42+
643
context 'when the app source contains a broken Python install' do
744
let(:app) { Hatchet::Runner.new('spec/fixtures/python_in_app_source', allow_failure: true) }
845

946
it 'fails detection' do
1047
app.deploy do |app|
1148
expect(clean_output(app.output)).to include(<<~OUTPUT)
1249
remote: -----> Python app detected
50+
remote:
51+
remote: ! Warning: Existing '.heroku/python/' directory found.
52+
remote: !
53+
remote: ! Your app's source code contains an existing directory named
54+
remote: ! '.heroku/python/', which is where the Python buildpack needs
55+
remote: ! to install its files. This existing directory contains:
56+
remote: !
57+
remote: ! .heroku/python/
58+
remote: ! .heroku/python/bin
59+
remote: ! .heroku/python/bin/python
60+
remote: !
61+
remote: ! Writing to internal locations used by the Python buildpack
62+
remote: ! is not supported and can cause unexpected errors.
63+
remote: !
64+
remote: ! If you have committed a '.heroku/python/' directory to your
65+
remote: ! Git repo, you must delete it or use a different location.
66+
remote: !
67+
remote: ! Otherwise, check that an earlier buildpack or 'bin/pre_compile'
68+
remote: ! hook has not created this directory.
69+
remote: !
70+
remote: ! If you have a use-case that requires writing to this location,
71+
remote: ! please comment on:
72+
remote: ! https://github.com/heroku/heroku-buildpack-python/issues/1704
73+
remote: !
74+
remote: ! In January 2025 this warning will be made an error.
75+
remote:
1376
remote: -----> Using Python #{DEFAULT_PYTHON_MAJOR_VERSION} specified in .python-version
1477
remote: -----> Using cached install of Python #{DEFAULT_PYTHON_FULL_VERSION}
1578
remote:

0 commit comments

Comments
 (0)