Skip to content

Commit

Permalink
Add New Testcase Generators (#311)
Browse files Browse the repository at this point in the history
* Added generator decorator and tests for it

- generator decorator supports k8s schema name, field name, field type,
  paths, and priority configuration

* Rename PathIndex to PathSegment

Signed-off-by: Tyler Gu <[email protected]>

* Code style changes

Signed-off-by: Tyler Gu <[email protected]>

* Create test generator directory

Signed-off-by: Tyler Gu <[email protected]>

* Migrate test generator from known schemas

Signed-off-by: Tyler Gu <[email protected]>

* Add init for test_generators to make it a module

Signed-off-by: Tyler Gu <[email protected]>

* Rename tests to be consistent with previous names

Signed-off-by: Tyler Gu <[email protected]>

* Add attributes to TestCase class

Signed-off-by: Tyler Gu <[email protected]>

* Optimize and improve generator decorator

- Rename `field` to `property` in decorator
- Move test generator matching code to the TestGenerator data class
- Support multiple constraints for test generator matching
- Tests for multiple constraints

* Primitive test generators

- Copied value generator code to the function decorator pattern
- Moved gen() method declaration in test generator to BaseSchema
- Moved gen() method implementation in [Type]Schemas

* Fix import error

* Custom validate_call decorator for test generators

- Include unittest

* Fix primitive test generators

* Fix unittest

* Rename generator to test_generator

Signed-off-by: Tyler Gu <[email protected]>

* Skip the test_generator function for pytest

Signed-off-by: Tyler Gu <[email protected]>

* Prioritize Integer test gens over Number test gens

* Migrate test generator priority to int enum

* Export all test generators

* Add cli schema_match that annotates CRD yaml file

* Add manual for test generator decorator

* Integrate new test gen into input model

Signed-off-by: Tyler Gu <[email protected]>

* Fix custom mapping

Signed-off-by: Tyler Gu <[email protected]>

* Fix applying property attributes

Signed-off-by: Tyler Gu <[email protected]>

* Fix value with schema

Signed-off-by: Tyler Gu <[email protected]>

* Fix value with anyof schema

Signed-off-by: Tyler Gu <[email protected]>

* Add tests for test execution

Signed-off-by: Tyler Gu <[email protected]>

---------

Signed-off-by: Tyler Gu <[email protected]>
Co-authored-by: Tyler Gu <[email protected]>
  • Loading branch information
MarkintoshZ and tylergu authored Feb 14, 2024
1 parent d3d9944 commit 79263d5
Show file tree
Hide file tree
Showing 55 changed files with 3,652 additions and 1,001 deletions.
1 change: 1 addition & 0 deletions acto/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
DEFAULT_KUBERNETES_VERSION = "v1.27.0"
12 changes: 10 additions & 2 deletions acto/checker/impl/consistency.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
"""State checker"""

import copy
import json
import re
Expand All @@ -20,6 +21,7 @@
)
from acto.input import InputModel
from acto.input.get_matched_schemas import find_matched_schema
from acto.input.property_attribute import PropertyAttribute
from acto.k8s_util.k8sutil import canonicalize_quantity
from acto.result import ConsistencyOracleResult, InvalidInputResult
from acto.schema import ArraySchema, BaseSchema, ObjectSchema, extract_schema
Expand Down Expand Up @@ -372,7 +374,8 @@ def check(
)
if (
diff_type == "iterable_item_removed"
or corresponding_schema.patch
or corresponding_schema.attributes & PropertyAttribute.Patch
== PropertyAttribute.Patch
or input_diff.path[-1] == "ACTOKEY"
):
pass
Expand All @@ -395,7 +398,12 @@ def check(
)

should_compare = (
should_compare or corresponding_schema.mapped
should_compare
or (
corresponding_schema.attributes
& PropertyAttribute.Mapped
== PropertyAttribute.Mapped
)
) and must_produce_delta

# Policy: pass if any of the matched deltas is equivalent
Expand Down
3 changes: 2 additions & 1 deletion acto/checker/impl/tests/test_state.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,6 @@ def input_model_and_context_mapping() -> (
input_model = DeterministicInputModel(
crd=context["crd"]["body"],
seed_input=defaultdict(lambda: defaultdict(dict)),
used_fields=[],
example_dir=None,
num_workers=1,
num_cases=1,
Expand All @@ -57,6 +56,7 @@ def input_model_and_context_mapping() -> (


def checker_func(s: Snapshot, prev_s: Snapshot) -> Optional[OracleResult]:
"""Run the consistency checker and return the result."""
api_version = s.input_cr["apiVersion"]
checker = ConsistencyChecker(
trial_dir="",
Expand Down Expand Up @@ -107,6 +107,7 @@ def checker_func(s: Snapshot, prev_s: Snapshot) -> Optional[OracleResult]:
),
)
def test_consistency_checker(test_case_id, result_dict):
"""Test the consistency checker."""
snapshot = load_snapshot("state", test_case_id)
snapshot_prev = load_snapshot("state", test_case_id, load_prev=True)
oracle_result = checker_func(snapshot, snapshot_prev)
Expand Down
Empty file added acto/cli/__init__.py
Empty file.
10 changes: 7 additions & 3 deletions acto/cli/schema_match.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
from ruamel.yaml import YAML

from acto.input.k8s_schemas import K8sSchemaMatcher
from acto.schema.object import ObjectSchema
from acto.schema.schema import extract_schema


Expand All @@ -22,7 +23,7 @@ def main():
parser.add_argument(
"--k8s-version",
required=False,
default="1.29",
default="v1.29.0",
help="Kubernetes version to match the schema with",
)
parser.add_argument(
Expand All @@ -43,7 +44,7 @@ def main():

# match the schema with Kubernetes resource schemas
schema_matcher = K8sSchemaMatcher.from_version(args.k8s_version)
matches = schema_matcher.find_matched_schemas(root)
matches = schema_matcher.find_all_matched_schemas(root)

# output the breakdown of the matched schema information
df = pd.DataFrame(
Expand All @@ -53,6 +54,7 @@ def main():
"schema_path": "/".join(schema.path),
}
for schema, k8s_schema in matches
if k8s_schema.k8s_schema_name is not None
]
)

Expand All @@ -61,14 +63,16 @@ def main():

# annotate the yaml file with the matched schema information
for schema, k8s_schema in matches:
if k8s_schema.k8s_schema_name is None:
continue
comment = k8s_schema.k8s_schema_name
curr = schema_yaml
for segment in schema.path[:-1]:
if segment == "ITEM":
curr = curr["items"]
else:
curr = curr["properties"][segment]
if schema.path[-1] != "ITEM":
if schema.path[-1] != "ITEM" and isinstance(schema, ObjectSchema):
curr["properties"].yaml_add_eol_comment(comment, schema.path[-1])
else:
curr.yaml_add_eol_comment(comment, "items")
Expand Down
10 changes: 5 additions & 5 deletions acto/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
import random
import re
import string
from typing import Any, Tuple, TypeAlias, Union
from typing import Any, Sequence, Tuple, TypeAlias, Union

import deepdiff.model as deepdiff_model
import kubernetes
Expand All @@ -14,15 +14,15 @@

from acto.utils.thread_logger import get_thread_logger

PathIndex: TypeAlias = Union[str, int]
PathSegment: TypeAlias = Union[str, int]


class PropertyPath(pydantic.BaseModel):
"""Path of a field in a dict"""

path: list[PathIndex]
path: list[PathSegment]

def __init__(self, path: list[PathIndex]) -> None:
def __init__(self, path: Sequence[PathSegment]) -> None:
"""Override constructor to allow positional argument"""
super().__init__(path=path)

Expand All @@ -44,7 +44,7 @@ def __getitem__(self, item: int):
def __len__(self):
return len(self.path)

def __contains__(self, item: PathIndex):
def __contains__(self, item: PathSegment):
return item in self.path


Expand Down
65 changes: 3 additions & 62 deletions acto/engine.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,19 +16,15 @@
import jsonpatch
import yaml

from acto.acto_config import ACTO_CONFIG
from acto.checker.checker_set import CheckerSet
from acto.common import kubernetes_client, print_event
from acto.constant import CONST
from acto.deploy import Deploy
from acto.input import InputModel
from acto.input.input import DeterministicInputModel, OverSpecifiedField
from acto.input.known_schemas.base import K8sField
from acto.input.known_schemas.known_schema import find_all_matched_schemas_type
from acto.input.input import DeterministicInputModel
from acto.input.testcase import TestCase
from acto.input.testplan import TestGroup
from acto.input.value_with_schema import ValueWithSchema, attach_schema_to_value
from acto.input.valuegenerator import ArrayGenerator
from acto.kubectl_client import KubectlClient
from acto.kubernetes_engine import base, kind
from acto.lib.operator_config import OperatorConfig
Expand Down Expand Up @@ -825,72 +821,17 @@ def __init__(
if preload_images_ is not None:
self.context["preload_images"].update(preload_images_)

# Apply custom fields
if operator_config.analysis is not None:
used_fields = self.context["analysis_result"]["used_fields"]
else:
used_fields = None
self.input_model: DeterministicInputModel = input_model(
crd=self.context["crd"]["body"],
seed_input=self.seed,
used_fields=used_fields,
example_dir=operator_config.example_dir,
num_workers=num_workers,
num_cases=num_cases,
mount=mount,
kubernetes_version=operator_config.kubernetes_version,
custom_module_path=operator_config.custom_module,
)

applied_custom_k8s_fields = False

if operator_config.k8s_fields is not None:
module = importlib.import_module(operator_config.k8s_fields)
if hasattr(module, "BLACKBOX") and ACTO_CONFIG.mode == "blackbox":
applied_custom_k8s_fields = True
for k8s_field in module.BLACKBOX:
self.input_model.apply_k8s_schema(k8s_field)
elif hasattr(module, "WHITEBOX") and ACTO_CONFIG.mode == "whitebox":
applied_custom_k8s_fields = True
for k8s_field in module.WHITEBOX:
self.input_model.apply_k8s_schema(k8s_field)
if not applied_custom_k8s_fields:
# default to use the known_schema module to automatically find the mapping
# from CRD to K8s schema
logger.info(
"Using known_schema to find the mapping from CRD to K8s schema"
)
tuples = find_all_matched_schemas_type(self.input_model.root_schema)
for match_tuple in tuples:
logger.debug(
"Found matched schema: %s -> %s",
match_tuple[0].path,
match_tuple[1],
)
k8s_schema = K8sField(match_tuple[0].path, match_tuple[1])
self.input_model.apply_k8s_schema(k8s_schema)

if operator_config.custom_fields is not None:
if ACTO_CONFIG.mode == "blackbox":
pruned_list = []
module = importlib.import_module(operator_config.custom_fields)
for custom_field in module.custom_fields:
pruned_list.append(custom_field.path)
self.input_model.apply_custom_field(custom_field)
else:
pruned_list = []
module = importlib.import_module(operator_config.custom_fields)
for custom_field in module.custom_fields:
pruned_list.append(custom_field.path)
self.input_model.apply_custom_field(custom_field)
else:
pruned_list = []
tuples = find_all_matched_schemas_type(self.input_model.root_schema)
for match_tuple in tuples:
custom_field = OverSpecifiedField(
match_tuple[0].path,
array=isinstance(match_tuple[1], ArrayGenerator),
)
self.input_model.apply_custom_field(custom_field)

self.sequence_base = 20 if delta_from else 0

if operator_config.custom_oracle is not None:
Expand Down
Loading

0 comments on commit 79263d5

Please sign in to comment.