Skip to content

Commit 5a29227

Browse files
Googlerpiotrwrotniak
Googler
authored andcommitted
Add Kokoro job configuration for Windows
Change-Id: I418e50648fb3e39fbf0fdc3738e9ee4ac34101ce GitOrigin-RevId: b774261461ab2c8b3b28ea099b0c1bb0a93f94ab
1 parent 818b5b4 commit 5a29227

File tree

8 files changed

+117
-37
lines changed

8 files changed

+117
-37
lines changed

.kokoro/tests/run_tests.bat

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
@rem Copyright 2022 Google LLC
2+
@rem
3+
@rem Licensed under the Apache License, Version 2.0 (the "License");
4+
@rem you may not use this file except in compliance with the License.
5+
@rem You may obtain a copy of the License at
6+
@rem
7+
@rem http://www.apache.org/licenses/LICENSE-2.0
8+
@rem
9+
@rem Unless required by applicable law or agreed to in writing, software
10+
@rem distributed under the License is distributed on an "AS IS" BASIS,
11+
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
@rem See the License for the specific language governing permissions and
13+
@rem limitations under the License.
14+
15+
@echo "Install pyenv tool for installing multiple Pythons"
16+
call C:\Python37\python.exe -m pip install pyenv-win --target %USERPROFILE%\.pyenv
17+
set "PYENV=%USERPROFILE%\.pyenv\pyenv-win\"
18+
set "PYENV_ROOT=%USERPROFILE%\.pyenv\pyenv-win\"
19+
set "PYENV_HOME=%USERPROFILE%\.pyenv\pyenv-win\"
20+
set "PATH=%USERPROFILE%\.pyenv\pyenv-win\bin;%USERPROFILE%\.pyenv\pyenv-win\shims;%PATH%"
21+
22+
@echo "Install additional Python versions"
23+
call %USERPROFILE%\.pyenv\pyenv-win\bin\pyenv install --skip-existing --register 3.8.10 3.9.13 3.10.5
24+
call %USERPROFILE%\.pyenv\pyenv-win\bin\pyenv global 3.8.10 3.9.13 3.10.5
25+
26+
if defined KOKORO_BUILD_ID ( @rem export vars only for Kokoro job
27+
@rem Setup service account credentials
28+
set "GOOGLE_APPLICATION_CREDENTIALS=%KOKORO_GFILE_DIR%/kokoro/service-account-key.json"
29+
30+
@rem Setup project id
31+
set /p PROJECT_ID=<%KOKORO_GFILE_DIR%/kokoro/project-id.txt
32+
set COMPOSER_TESTS_PROJECT_ID=PROJECT_ID
33+
)
34+
35+
@rem install nox for testing
36+
call C:\Python37\python.exe -m pip install --require-hashes --upgrade --quiet -r .kokoro/tests/requirements.txt
37+
call C:\Python37\python.exe -m nox --version
38+
39+
@echo "******************** Running unit tests... ********************\n"
40+
call C:\Python37\python.exe -m nox -s "unit"
41+
@echo "******************** Unit tests complete. ********************\n"
42+
@echo "******************** Running E2E tests... ********************\n"
43+
@rem call C:\C:\Python37\python.exe\python.exe -m nox -s "e2e"
44+
@echo "******************** Tests complete. ********************\n"

.kokoro/tests/run_tests.sh

+4-2
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@
1212
# See the License for the specific language governing permissions and
1313
# limitations under the License.
1414

15-
set -e
15+
set -xe
1616

1717
# add user's pip binary path to PATH
1818
export PATH="${HOME}/.local/bin:${PATH}"
@@ -32,7 +32,9 @@ if [[ "$OSTYPE" == "darwin"* ]]; then # Mac OSX
3232
pyenv global 3.7.13
3333
else
3434
pyenv install --skip-existing 3.7.10
35-
pyenv global 3.7.10
35+
pyenv install --skip-existing 3.8.10
36+
pyenv install --skip-existing 3.9.5
37+
pyenv global 3.7.10 3.8.10 3.9.5
3638
fi
3739

3840
# install nox for testing

.kokoro/windows/kokoro_build.bat

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
@rem Copyright 2022 Google LLC
2+
3+
@rem Licensed under the Apache License, Version 2.0 (the "License");
4+
@rem you may not use this file except in compliance with the License.
5+
@rem You may obtain a copy of the License at
6+
7+
@rem http://www.apache.org/licenses/LICENSE-2.0
8+
9+
@rem Unless required by applicable law or agreed to in writing, software
10+
@rem distributed under the License is distributed on an "AS IS" BASIS,
11+
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
@rem See the License for the specific language governing permissions and
13+
@rem limitations under the License.
14+
15+
16+
cd "%KOKORO_ARTIFACTS_DIR%\git\composer-local-development"
17+
call .kokoro\tests\run_tests.bat

.kokoro/windows/presubmit.cfg

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
# -*- protobuffer -*-
2+
# proto-file: google3/devtools/kokoro/config/proto/build.proto
3+
# proto-message: BuildConfig
4+
5+
# Resources needed for E2E tests
6+
gfile_resources: "/bigstore/composer-local-dev-test-resources"
7+
8+
# Location of the bash script. Should have value <github_scm.name>/<path_from_repository_root>.
9+
# github_scm.name is specified in the job configuration (next section).
10+
build_file: "composer-local-development/.kokoro/windows/kokoro_build.bat"

noxfile.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@
1717
import nox
1818

1919
DEFAULT_PYTHON_VERSION = "3.9"
20-
TEST_PYTHON_VERSIONS = ["3.7", "3.8", "3.10"] # TODO Fix 3.9 on Kokoro
20+
TEST_PYTHON_VERSIONS = ["3.7", "3.8", "3.9", "3.10"]
2121

2222
CURRENT_DIRECTORY = Path(__file__).parent.absolute()
2323

tests/unit/test_environment.py

+18-18
Original file line numberDiff line numberDiff line change
@@ -412,8 +412,9 @@ def test_missing_variables_env(self):
412412
f"Environment variables file '{env_dir / 'variables.env'}' "
413413
f"not found."
414414
)
415-
with pytest.raises(errors.ComposerCliError, match=exp_error):
415+
with pytest.raises(errors.ComposerCliError) as err:
416416
environment.load_environment_variables(env_dir)
417+
assert str(err) == exp_error
417418

418419
def test_load_environment_variables_filter_blocked_env_vars(self):
419420
env_dir = (TEST_DATA_DIR / "blocked_env_vars").resolve()
@@ -465,10 +466,9 @@ def test_invalid_env_variables(self):
465466
exp_error = constants.INVALID_ENV_VARIABLES_FILE_ERROR.format(
466467
env_file_path=env_file_path, line="AIRFLOW_KEYVALUE"
467468
)
468-
with pytest.raises(
469-
errors.FailedToParseVariablesError, match=re.escape(exp_error)
470-
):
469+
with pytest.raises(errors.FailedToParseVariablesError) as err:
471470
environment.load_environment_variables(env_dir)
471+
assert str(err) == exp_error
472472

473473
@mock.patch("composer_local_dev.environment.docker.from_env")
474474
@mock.patch(
@@ -484,10 +484,10 @@ def test_missing_requirements(self, mocked_docker, mocked_tag, tmpdir):
484484
location="location",
485485
dags_path=str(env_dir),
486486
)
487-
with pytest.raises(
488-
errors.ComposerCliError, match=f"Missing '{requirement_file}' file."
489-
):
487+
exp_error = f"Missing '{requirement_file}' file."
488+
with pytest.raises(errors.ComposerCliError) as err:
490489
env.assert_requirements_exist()
490+
assert str(err) == exp_error
491491

492492
@mock.patch("composer_local_dev.environment.docker.from_env")
493493
def test_get_container(self, mocked_docker, default_env):
@@ -1059,17 +1059,19 @@ def test_invalid_config(self):
10591059
config_path=config_path, error=""
10601060
)
10611061
with pytest.raises(
1062-
errors.FailedToParseConfigError, match=exp_error
1063-
), working_directory(env_dir):
1062+
errors.FailedToParseConfigError
1063+
) as err, working_directory(env_dir):
10641064
environment.EnvironmentConfig(env_dir, None)
1065+
assert str(err) == exp_error
10651066

10661067
def test_missing_config(self):
10671068
env_dir = (TEST_DATA_DIR / "missing_composer").resolve()
10681069
exp_error = f"Configuration file '{env_dir / 'config.json'}' not found."
1069-
with pytest.raises(
1070-
errors.ComposerCliError, match=exp_error
1071-
), working_directory(env_dir):
1070+
with pytest.raises(errors.ComposerCliError) as err, working_directory(
1071+
env_dir
1072+
):
10721073
environment.EnvironmentConfig(env_dir, None)
1074+
assert str(err) == exp_error
10731075

10741076
@mock.patch(
10751077
"composer_local_dev.environment.EnvironmentConfig.load_configuration_from_file"
@@ -1091,10 +1093,9 @@ def test_param_invalid_int(self, mocked_load_conf, tmp_path, param, value):
10911093
exp_error = constants.INVALID_INT_VALUE_ERROR.format(
10921094
param_name=param, value=value
10931095
)
1094-
with pytest.raises(
1095-
errors.FailedToParseConfigParamIntError, match=exp_error
1096-
):
1096+
with pytest.raises(errors.FailedToParseConfigParamIntError) as err:
10971097
environment.EnvironmentConfig(tmp_path, None)
1098+
assert str(err) == exp_error
10981099

10991100
@mock.patch(
11001101
"composer_local_dev.environment.EnvironmentConfig.load_configuration_from_file"
@@ -1122,7 +1123,6 @@ def test_param_invalid_int_range(
11221123
exp_error = constants.INVALID_INT_RANGE_VALUE_ERROR.format(
11231124
param_name=param, value=value, allowed_range=allowed_range
11241125
)
1125-
with pytest.raises(
1126-
errors.FailedToParseConfigParamIntRangeError, match=exp_error
1127-
):
1126+
with pytest.raises(errors.FailedToParseConfigParamIntRangeError) as err:
11281127
environment.EnvironmentConfig(tmp_path, None)
1128+
assert str(err) == exp_error

tests/unit/test_files.py

+20-14
Original file line numberDiff line numberDiff line change
@@ -56,10 +56,11 @@ def test_no_composer_dir(self, env_name):
5656
)
5757
+ constants.ADD_DEBUG_ON_ERROR_INFO
5858
)
59-
with pytest.raises(
60-
errors.ComposerCliError, match=re.escape(exp_error)
61-
), working_directory(env_dir):
59+
with pytest.raises(errors.ComposerCliError) as err, working_directory(
60+
env_dir
61+
):
6262
files.resolve_environment_path(env_name)
63+
assert str(err) == exp_error
6364

6465
@pytest.mark.parametrize("env_name", [None, "test"])
6566
def test_no_envs(self, env_name):
@@ -71,10 +72,11 @@ def test_no_envs(self, env_name):
7172
)
7273
+ constants.ADD_DEBUG_ON_ERROR_INFO
7374
)
74-
with pytest.raises(
75-
errors.ComposerCliError, match=re.escape(exp_error)
76-
), working_directory(env_dir):
75+
with pytest.raises(errors.ComposerCliError) as err, working_directory(
76+
env_dir
77+
):
7778
files.resolve_environment_path(env_name)
79+
assert str(err) == exp_error
7880

7981
@pytest.mark.parametrize("env_name", ["one_env", "two_envs"])
8082
def test_not_exising_name(self, env_name):
@@ -87,10 +89,11 @@ def test_not_exising_name(self, env_name):
8789
)
8890
+ constants.ADD_DEBUG_ON_ERROR_INFO
8991
)
90-
with pytest.raises(
91-
errors.ComposerCliError, match=re.escape(exp_error)
92-
), working_directory(env_dir):
92+
with pytest.raises(errors.ComposerCliError) as err, working_directory(
93+
env_dir
94+
):
9395
files.resolve_environment_path(invalid_env)
96+
assert str(err) == exp_error
9497

9598
@pytest.mark.parametrize("env_name", [None, "example_env"])
9699
def test_one_env_existing_name(self, env_name):
@@ -113,10 +116,11 @@ def test_two_envs_no_default(self):
113116
error_msg = constants.ENVIRONMENT_NOT_SELECTED_ERROR.format(
114117
env_dir=composer_dir, env_names=env_names
115118
)
116-
with pytest.raises(
117-
errors.ComposerCliError, match=error_msg
118-
), working_directory(env_dir):
119+
with pytest.raises(errors.ComposerCliError) as err, working_directory(
120+
env_dir
121+
):
119122
files.resolve_environment_path(None)
123+
assert str(err) == error_msg
120124

121125

122126
class TestGetEnvironmentDirectories:
@@ -210,8 +214,9 @@ def test_existing_path(self, tmp_path):
210214
def test_missing_path(self):
211215
path = "i/dont/exist"
212216
error_msg = f"Dags path does not exist or is not a directory: {path}"
213-
with pytest.raises(errors.DAGPathNotExistError, match=error_msg):
217+
with pytest.raises(errors.DAGPathNotExistError) as err:
214218
files.assert_dag_path_exists(path)
219+
assert str(err) == error_msg
215220

216221
def test_path_is_file(self, tmp_path):
217222
file_path = tmp_path / "file.ext"
@@ -220,8 +225,9 @@ def test_path_is_file(self, tmp_path):
220225
error_msg = (
221226
f"Dags path does not exist or is not a directory: {file_path}"
222227
)
223-
with pytest.raises(errors.DAGPathNotExistError, match=error_msg):
228+
with pytest.raises(errors.DAGPathNotExistError) as err:
224229
files.assert_dag_path_exists(str(file_path))
230+
assert str(err) == error_msg
225231

226232

227233
def test_dos2unix_file(tmp_path):

tests/unit/test_utils.py

+3-2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
# limitations under the License.
1414

1515
import json
16+
import os
1617
import pathlib
1718
import re
1819
import subprocess
@@ -107,12 +108,12 @@ def test_resolve_path_on_windows(self, mocked_os_check, tmpdir):
107108
@mock.patch.object(pathlib.Path, "is_dir")
108109
@mock.patch.object(pathlib.Path, "expanduser")
109110
@mock.patch(
110-
"composer_local_dev.environment.utils.is_windows_os", return_value=True
111+
"composer_local_dev.environment.utils.is_windows_os", return_value=False
111112
)
112113
def test_resolve_path_not_on_windows(
113114
self, mocked_os_check, mocked_expand, mocked_is_dir
114115
):
115-
config_path = "path/to/config"
116+
config_path = os.pathsep.join(["path", "to", "config"])
116117
mocked_expand.return_value = pathlib.Path(config_path)
117118
mocked_is_dir.return_value = True
118119
actual_config_path = utils.resolve_gcloud_config_path()

0 commit comments

Comments
 (0)