From 57f1a1333d5bb0ae1b9dc48411ca986455c9e9be Mon Sep 17 00:00:00 2001 From: Joe Wang Date: Thu, 13 Jun 2024 16:29:15 -0400 Subject: [PATCH 1/4] feat: populate ``.repo-metadata.json` using the highest version --- .../generate_composed_library.py | 2 +- library_generation/model/gapic_config.py | 45 +++++++++++++++++++ library_generation/model/library_config.py | 7 ++- library_generation/utils/proto_path_utils.py | 9 ++++ 4 files changed, 60 insertions(+), 3 deletions(-) diff --git a/library_generation/generate_composed_library.py b/library_generation/generate_composed_library.py index 11c98e2dd9..0ea7eb26f7 100755 --- a/library_generation/generate_composed_library.py +++ b/library_generation/generate_composed_library.py @@ -66,7 +66,7 @@ def generate_composed_library( base_arguments = __construct_tooling_arg(config=config) owlbot_cli_source_folder = util.sh_util("mktemp -d") os.makedirs(f"{library_path}", exist_ok=True) - for gapic in library.gapic_configs: + for gapic in library.get_sorted_gapic_configs(): build_file_folder = Path(f"{output_folder}/{gapic.proto_path}").resolve() print(f"build_file_folder: {build_file_folder}") gapic_inputs = parse_build_file(build_file_folder, gapic.proto_path) diff --git a/library_generation/model/gapic_config.py b/library_generation/model/gapic_config.py index bec1645823..9c98038f4e 100644 --- a/library_generation/model/gapic_config.py +++ b/library_generation/model/gapic_config.py @@ -12,6 +12,10 @@ # 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. +from library_generation.utils.proto_path_utils import find_version_in_proto_path + +ALPHA_VERSION = "alpha" +BETA_VERSION = "beta" class GapicConfig: @@ -22,3 +26,44 @@ class GapicConfig: def __init__(self, proto_path: str): self.proto_path = proto_path + + def __eq__(self, other) -> bool: + if not isinstance(other, GapicConfig): + return False + return self.proto_path == other.proto_path + + def __lt__(self, other) -> bool: + self_version = find_version_in_proto_path(self.proto_path) + other_version = find_version_in_proto_path(other.proto_path) + # if only one config has a version in the proto_path, it is smaller + # than the other one. + if self_version and (not other_version): + return True + if (not self_version) and other_version: + return False + + self_dirs = self.proto_path.split("/") + other_dirs = other.proto_path.split("/") + # if both of the configs have no version in the proto_path, the one + # with lower depth is smaller. + if (not self_version) and (not other_version): + return len(self_dirs) < len(other_dirs) + + self_stable = GapicConfig.__is_stable(self_version) + other_stable = GapicConfig.__is_stable(other_version) + # if only config has a stable version in proto_path, it is smaller than + # the other one. + if self_stable and (not other_stable): + return True + if (not self_stable) and other_stable: + return False + # if two configs have different depth in proto_path, the one with + # lower depth is smaller. + if len(self_dirs) != len(other_dirs): + return len(self_dirs) < len(other_dirs) + # otherwise, the one with higher version is smaller. + return self_version > other_version + + @staticmethod + def __is_stable(version: str) -> bool: + return not (ALPHA_VERSION in version or BETA_VERSION in version) diff --git a/library_generation/model/library_config.py b/library_generation/model/library_config.py index 30d923da81..d8049ab391 100644 --- a/library_generation/model/library_config.py +++ b/library_generation/model/library_config.py @@ -14,7 +14,7 @@ # limitations under the License. from hashlib import sha1 -from typing import List, Optional +from typing import Optional from library_generation.model.gapic_config import GapicConfig @@ -29,7 +29,7 @@ def __init__( api_description: str, name_pretty: str, product_documentation: str, - gapic_configs: List[GapicConfig], + gapic_configs: list[GapicConfig], library_type: Optional[str] = None, release_level: Optional[str] = None, api_id: Optional[str] = None, @@ -80,6 +80,9 @@ def get_library_name(self) -> str: """ return self.library_name if self.library_name else self.api_shortname + def get_sorted_gapic_configs(self) -> list[GapicConfig]: + return sorted(self.gapic_configs) + def __eq__(self, other): return ( self.api_shortname == other.api_shortname diff --git a/library_generation/utils/proto_path_utils.py b/library_generation/utils/proto_path_utils.py index d2ae25f602..6d2d4f7022 100644 --- a/library_generation/utils/proto_path_utils.py +++ b/library_generation/utils/proto_path_utils.py @@ -13,6 +13,7 @@ # See the License for the specific language governing permissions and # limitations under the License. import re +from typing import Optional def remove_version_from(proto_path: str) -> str: @@ -45,3 +46,11 @@ def find_versioned_proto_path(proto_path: str) -> str: idx = proto_path.find(version) return proto_path[:idx] + version return proto_path + + +def find_version_in_proto_path(proto_path: str) -> Optional[str]: + version_regex = re.compile(r"^v[1-9].*") + for directory in proto_path.split("/"): + if version_regex.search(directory): + return directory + return None From bd77781b49096f0cd714cf1108d44918af1a425c Mon Sep 17 00:00:00 2001 From: Joe Wang Date: Thu, 13 Jun 2024 16:29:42 -0400 Subject: [PATCH 2/4] add unit tests --- .../test/model/library_config_unit_tests.py | 27 +++++++++++++++++++ .../test/utils/proto_path_utils_unit_tests.py | 9 +++++++ 2 files changed, 36 insertions(+) diff --git a/library_generation/test/model/library_config_unit_tests.py b/library_generation/test/model/library_config_unit_tests.py index 1c0b0d0690..35ba5be3e4 100644 --- a/library_generation/test/model/library_config_unit_tests.py +++ b/library_generation/test/model/library_config_unit_tests.py @@ -13,6 +13,7 @@ # limitations under the License. import unittest +from library_generation.model.gapic_config import GapicConfig from library_generation.model.library_config import LibraryConfig @@ -37,3 +38,29 @@ def test_get_library_returns_api_shortname(self): gapic_configs=list(), ) self.assertEqual("secret", library.get_library_name()) + + def test_get_sorted_gapic_configs_returns_correct_order(self): + v1beta1 = GapicConfig(proto_path="google/spanner/v1beta1") + v1 = GapicConfig(proto_path="google/spanner/v1") + v1alpha1 = GapicConfig(proto_path="google/spanner/v1alpha") + v2 = GapicConfig(proto_path="google/spanner/v2") + admin_v2 = GapicConfig(proto_path="google/spanner/admin/v2") + non_versioned = GapicConfig(proto_path="google/spanner/type") + library = LibraryConfig( + api_shortname="secret", + name_pretty="", + product_documentation="", + api_description="", + gapic_configs=[v1alpha1, v1, v2, admin_v2, non_versioned, v1beta1], + ) + self.assertEqual( + [ + v2, + v1, + admin_v2, + v1beta1, + v1alpha1, + non_versioned, + ], + library.get_sorted_gapic_configs(), + ) diff --git a/library_generation/test/utils/proto_path_utils_unit_tests.py b/library_generation/test/utils/proto_path_utils_unit_tests.py index 2e23c2b403..c8bf1e7847 100644 --- a/library_generation/test/utils/proto_path_utils_unit_tests.py +++ b/library_generation/test/utils/proto_path_utils_unit_tests.py @@ -18,6 +18,7 @@ from library_generation.utils.proto_path_utils import ( find_versioned_proto_path, remove_version_from, + find_version_in_proto_path, ) script_dir = os.path.dirname(os.path.realpath(__file__)) @@ -48,3 +49,11 @@ def test_remove_version_from_returns_non_versioned_path(self): def test_remove_version_from_returns_self(self): proto_path = "google/cloud/aiplatform" self.assertEqual("google/cloud/aiplatform", remove_version_from(proto_path)) + + def test_find_version_proto_path_returns_version(self): + proto_path = "google/cloud/aiplatform/v1beta" + self.assertEqual("v1beta", find_version_in_proto_path(proto_path)) + + def test_find_version_proto_path_returns_none(self): + proto_path = "google/cloud/aiplatform" + self.assertIsNone(find_version_in_proto_path(proto_path)) From b9611ff0dfafeec3c471eb3ae8bd6696daa30be3 Mon Sep 17 00:00:00 2001 From: Joe Wang Date: Tue, 18 Jun 2024 14:57:04 -0400 Subject: [PATCH 3/4] change comparsion alg --- library_generation/model/gapic_config.py | 78 +++++++++++++------ .../test/model/gapic_config_unit_tests.py | 66 ++++++++++++++++ library_generation/utils/proto_path_utils.py | 9 --- 3 files changed, 120 insertions(+), 33 deletions(-) create mode 100644 library_generation/test/model/gapic_config_unit_tests.py diff --git a/library_generation/model/gapic_config.py b/library_generation/model/gapic_config.py index 9c98038f4e..3bbba39454 100644 --- a/library_generation/model/gapic_config.py +++ b/library_generation/model/gapic_config.py @@ -12,7 +12,8 @@ # 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. -from library_generation.utils.proto_path_utils import find_version_in_proto_path +import re +from typing import Optional ALPHA_VERSION = "alpha" BETA_VERSION = "beta" @@ -26,6 +27,24 @@ class GapicConfig: def __init__(self, proto_path: str): self.proto_path = proto_path + self.version = self.__parse_version() + + def is_stable(self): + return ( + self.version + and (ALPHA_VERSION not in self.version) + and (BETA_VERSION not in self.version) + ) + + def get_version(self): + return self.version + + def __parse_version(self) -> Optional[str]: + version_regex = re.compile(r"^v[1-9]+(p[1-9]+)*(alpha|beta)?.*") + for directory in self.proto_path.split("/"): + if version_regex.search(directory): + return directory + return None def __eq__(self, other) -> bool: if not isinstance(other, GapicConfig): @@ -33,37 +52,48 @@ def __eq__(self, other) -> bool: return self.proto_path == other.proto_path def __lt__(self, other) -> bool: - self_version = find_version_in_proto_path(self.proto_path) - other_version = find_version_in_proto_path(other.proto_path) - # if only one config has a version in the proto_path, it is smaller - # than the other one. - if self_version and (not other_version): - return True - if (not self_version) and other_version: - return False + if not isinstance(other, GapicConfig): + raise ValueError( + f"Type {type(other)} can't be comparable " f"with GapicConfig." + ) + self_version = self.get_version() + other_version = other.get_version() self_dirs = self.proto_path.split("/") other_dirs = other.proto_path.split("/") - # if both of the configs have no version in the proto_path, the one - # with lower depth is smaller. + # Case 1: if both of the configs don't have a version in proto_path, + # the one with lower depth is smaller. if (not self_version) and (not other_version): return len(self_dirs) < len(other_dirs) - - self_stable = GapicConfig.__is_stable(self_version) - other_stable = GapicConfig.__is_stable(other_version) - # if only config has a stable version in proto_path, it is smaller than - # the other one. + # Case 2: if only one config has a version in proto_path, it is smaller + # than the other one. + if self_version and (not other_version): + return True + if (not self_version) and other_version: + return False + # Two configs both have a version in proto_path. + self_stable = self.is_stable() + other_stable = other.is_stable() + # Case 3, if only config has a stable version in proto_path, it is + # smaller than the other one. if self_stable and (not other_stable): return True if (not self_stable) and other_stable: return False - # if two configs have different depth in proto_path, the one with - # lower depth is smaller. + # Case 4, if two configs have a non-stable version in proto_path, + # the one with higher version is smaller. + # Note that using string comparison may lead unexpected result, + # e.g., v1beta10 is smaller than v1beta2. + # In reality, however, there's unlikely that a library has >10 beta + # versions. + if (not self_stable) and (not other_stable): + return self_version > other_version + # Two configs both have a stable version in proto_path. + # Case 5, if two configs have different depth in proto_path, the one + # with lower depth is smaller. if len(self_dirs) != len(other_dirs): return len(self_dirs) < len(other_dirs) - # otherwise, the one with higher version is smaller. - return self_version > other_version - - @staticmethod - def __is_stable(version: str) -> bool: - return not (ALPHA_VERSION in version or BETA_VERSION in version) + # Case 6, the config with higher stable version is smaller. + self_num = int(self_version[1:]) + other_num = int(other_version[1:]) + return self_num > other_num diff --git a/library_generation/test/model/gapic_config_unit_tests.py b/library_generation/test/model/gapic_config_unit_tests.py new file mode 100644 index 0000000000..35ba5be3e4 --- /dev/null +++ b/library_generation/test/model/gapic_config_unit_tests.py @@ -0,0 +1,66 @@ +# Copyright 2024 Google LLC +# +# 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 +# +# https://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 unittest + +from library_generation.model.gapic_config import GapicConfig +from library_generation.model.library_config import LibraryConfig + + +class LibraryConfigTest(unittest.TestCase): + def test_get_library_returns_library_name(self): + library = LibraryConfig( + api_shortname="secret", + name_pretty="", + product_documentation="", + api_description="", + gapic_configs=list(), + library_name="secretmanager", + ) + self.assertEqual("secretmanager", library.get_library_name()) + + def test_get_library_returns_api_shortname(self): + library = LibraryConfig( + api_shortname="secret", + name_pretty="", + product_documentation="", + api_description="", + gapic_configs=list(), + ) + self.assertEqual("secret", library.get_library_name()) + + def test_get_sorted_gapic_configs_returns_correct_order(self): + v1beta1 = GapicConfig(proto_path="google/spanner/v1beta1") + v1 = GapicConfig(proto_path="google/spanner/v1") + v1alpha1 = GapicConfig(proto_path="google/spanner/v1alpha") + v2 = GapicConfig(proto_path="google/spanner/v2") + admin_v2 = GapicConfig(proto_path="google/spanner/admin/v2") + non_versioned = GapicConfig(proto_path="google/spanner/type") + library = LibraryConfig( + api_shortname="secret", + name_pretty="", + product_documentation="", + api_description="", + gapic_configs=[v1alpha1, v1, v2, admin_v2, non_versioned, v1beta1], + ) + self.assertEqual( + [ + v2, + v1, + admin_v2, + v1beta1, + v1alpha1, + non_versioned, + ], + library.get_sorted_gapic_configs(), + ) diff --git a/library_generation/utils/proto_path_utils.py b/library_generation/utils/proto_path_utils.py index 6d2d4f7022..d2ae25f602 100644 --- a/library_generation/utils/proto_path_utils.py +++ b/library_generation/utils/proto_path_utils.py @@ -13,7 +13,6 @@ # See the License for the specific language governing permissions and # limitations under the License. import re -from typing import Optional def remove_version_from(proto_path: str) -> str: @@ -46,11 +45,3 @@ def find_versioned_proto_path(proto_path: str) -> str: idx = proto_path.find(version) return proto_path[:idx] + version return proto_path - - -def find_version_in_proto_path(proto_path: str) -> Optional[str]: - version_regex = re.compile(r"^v[1-9].*") - for directory in proto_path.split("/"): - if version_regex.search(directory): - return directory - return None From c2e7b1c63ca87139535dddb6a931871b22a65b4a Mon Sep 17 00:00:00 2001 From: Joe Wang Date: Tue, 18 Jun 2024 14:57:17 -0400 Subject: [PATCH 4/4] add unit tests --- .../test/model/gapic_config_unit_tests.py | 144 ++++++++++++------ .../test/utils/proto_path_utils_unit_tests.py | 9 -- 2 files changed, 100 insertions(+), 53 deletions(-) diff --git a/library_generation/test/model/gapic_config_unit_tests.py b/library_generation/test/model/gapic_config_unit_tests.py index 35ba5be3e4..64d8556648 100644 --- a/library_generation/test/model/gapic_config_unit_tests.py +++ b/library_generation/test/model/gapic_config_unit_tests.py @@ -14,53 +14,109 @@ import unittest from library_generation.model.gapic_config import GapicConfig -from library_generation.model.library_config import LibraryConfig -class LibraryConfigTest(unittest.TestCase): - def test_get_library_returns_library_name(self): - library = LibraryConfig( - api_shortname="secret", - name_pretty="", - product_documentation="", - api_description="", - gapic_configs=list(), - library_name="secretmanager", - ) - self.assertEqual("secretmanager", library.get_library_name()) +class GapicConfigTest(unittest.TestCase): + def test_get_version_returns_none(self): + self.assertIsNone(GapicConfig(proto_path="example/dir1/dir2").get_version()) - def test_get_library_returns_api_shortname(self): - library = LibraryConfig( - api_shortname="secret", - name_pretty="", - product_documentation="", - api_description="", - gapic_configs=list(), - ) - self.assertEqual("secret", library.get_library_name()) - - def test_get_sorted_gapic_configs_returns_correct_order(self): - v1beta1 = GapicConfig(proto_path="google/spanner/v1beta1") - v1 = GapicConfig(proto_path="google/spanner/v1") - v1alpha1 = GapicConfig(proto_path="google/spanner/v1alpha") - v2 = GapicConfig(proto_path="google/spanner/v2") - admin_v2 = GapicConfig(proto_path="google/spanner/admin/v2") - non_versioned = GapicConfig(proto_path="google/spanner/type") - library = LibraryConfig( - api_shortname="secret", - name_pretty="", - product_documentation="", - api_description="", - gapic_configs=[v1alpha1, v1, v2, admin_v2, non_versioned, v1beta1], + def test_get_version_returns_non_stable_version(self): + self.assertEqual( + "v2p2beta1", + GapicConfig(proto_path="example/dir1/dir2/v2p2beta1").get_version(), ) self.assertEqual( - [ - v2, - v1, - admin_v2, - v1beta1, - v1alpha1, - non_versioned, - ], - library.get_sorted_gapic_configs(), + "v2beta1", + GapicConfig(proto_path="example/dir1/dir2/v2beta1").get_version(), + ) + + def test_get_version_returns_stable_version(self): + self.assertEqual( + "v20", + GapicConfig(proto_path="example/dir1/dir2/v20").get_version(), + ) + + def test_is_stable_with_no_version_returns_false(self): + self.assertFalse( + GapicConfig(proto_path="example/dir1/dir2/non_version").is_stable(), + ) + + def test_is_stable_with_non_stable_version_returns_false(self): + self.assertFalse( + GapicConfig(proto_path="example/dir1/dir2/v20alpha").is_stable(), + ) + self.assertFalse( + GapicConfig(proto_path="example/dir1/dir2/v20beta2").is_stable(), + ) + + def test_is_stable_with_stable_version_returns_true(self): + self.assertTrue( + GapicConfig(proto_path="example/dir1/dir2/v30").is_stable(), + ) + + def test_compare_configs_without_a_version(self): + config_len_3 = GapicConfig(proto_path="example/dir1/dir2") + config_len_4 = GapicConfig(proto_path="example/dir1/dir2/dir3") + self.assertLess( + config_len_3, + config_len_4, + "config_len_3 should be smaller since it has a lower depth.", + ) + + def test_compare_configs_only_one_has_a_stable_version(self): + versioned_config = GapicConfig(proto_path="example/dir1/dir2/dir3/dir4/v1") + non_versioned_config = GapicConfig(proto_path="example/dir1/dir2/dir3") + self.assertLess( + versioned_config, + non_versioned_config, + "versioned_config should be smaller since it has a version.", + ) + + def test_compare_configs_only_one_has_a_non_stable_version(self): + non_stable_versioned_config = GapicConfig( + proto_path="example/dir1/dir2/dir3/dir4/v1beta" + ) + non_versioned_config = GapicConfig(proto_path="example/dir1/dir2/dir3") + self.assertLess( + non_stable_versioned_config, + non_versioned_config, + "non_stable_versioned_config should be smaller since it has a version.", + ) + + def test_compare_configs_one_has_non_stable_and_one_has_stable_version(self): + stable_versioned_config = GapicConfig( + proto_path="example/dir1/dir2/dir3/dir4/v1" + ) + non_stable_versioned_config = GapicConfig(proto_path="example/dir1/dir2/v2beta") + self.assertLess( + stable_versioned_config, + non_stable_versioned_config, + "stable_versioned_config should be smaller since it has a stable version.", + ) + + def test_compare_configs_two_have_non_stable_version(self): + v3p2beta = GapicConfig(proto_path="example/dir1/dir2/dir3/dir4/v3p2beta") + v2p4beta = GapicConfig(proto_path="example/dir1/dir2/v2p4beta") + self.assertLess( + v3p2beta, + v2p4beta, + "v3p2beta should be smaller since it has a higher version.", + ) + + def test_compare_configs_two_have_stable_version_different_depth(self): + v3 = GapicConfig(proto_path="example/dir1/dir2/v3") + v4 = GapicConfig(proto_path="example/dir1/dir2/dir3/dir4/v4") + self.assertLess( + v3, + v4, + "v3 should be smaller since it has lower depth", + ) + + def test_compare_configs_two_have_stable_version_same_depth(self): + v4 = GapicConfig(proto_path="example/dir1/dir2/v4") + v3 = GapicConfig(proto_path="example/dir1/dir2/v3") + self.assertLess( + v4, + v3, + "v4 should be smaller since it has a higher version", ) diff --git a/library_generation/test/utils/proto_path_utils_unit_tests.py b/library_generation/test/utils/proto_path_utils_unit_tests.py index c8bf1e7847..2e23c2b403 100644 --- a/library_generation/test/utils/proto_path_utils_unit_tests.py +++ b/library_generation/test/utils/proto_path_utils_unit_tests.py @@ -18,7 +18,6 @@ from library_generation.utils.proto_path_utils import ( find_versioned_proto_path, remove_version_from, - find_version_in_proto_path, ) script_dir = os.path.dirname(os.path.realpath(__file__)) @@ -49,11 +48,3 @@ def test_remove_version_from_returns_non_versioned_path(self): def test_remove_version_from_returns_self(self): proto_path = "google/cloud/aiplatform" self.assertEqual("google/cloud/aiplatform", remove_version_from(proto_path)) - - def test_find_version_proto_path_returns_version(self): - proto_path = "google/cloud/aiplatform/v1beta" - self.assertEqual("v1beta", find_version_in_proto_path(proto_path)) - - def test_find_version_proto_path_returns_none(self): - proto_path = "google/cloud/aiplatform" - self.assertIsNone(find_version_in_proto_path(proto_path))