Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove default check for is_set #61

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
19 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -78,5 +78,5 @@ jobs:
run: |
source .venv/bin/activate
pip install .
osivalidator --data data/20210818T150542Z_sv_312_50_one_moving_object.osi -r rules
osivalidator --data data/20210818T150542Z_sv_312_50_one_moving_object.osi -r rules --parallel
osivalidator --data data/20240221T141700Z_sv_300_2112_10_one_moving_object.osi -r rules
osivalidator --data data/20240221T141700Z_sv_300_2112_10_one_moving_object.osi -r rules --parallel
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -80,5 +80,5 @@ $ source .venv/Scripts/activate
## Example command

```bash
$ osivalidator --data data/20210818T150542Z_sv_312_50_one_moving_object.osi --rules rules/
$ osivalidator --data data/20240221T141700Z_sv_300_2112_10_one_moving_object.osi --rules rules/
```
Binary file not shown.
Binary file not shown.
339 changes: 37 additions & 302 deletions doc/usage.adoc

Large diffs are not rendered by default.

55 changes: 38 additions & 17 deletions doc/writing-rules.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -2,19 +2,37 @@

== Folder structure

Currently the rules are contained in `+*.yml+` files in the folder
https://github.com/OpenSimulationInterface/osi-validation/tree/master/rules[rules].
The organization of the files in this folder follows the architecture of
OSI for consistency. In the future the rules will be ported directly
into the `+*.proto+` files of
Official rules for individual OSI fields are defined in the `+*.proto+` files of
https://github.com/OpenSimulationInterface/open-simulation-interface[OSI].
The first step to using and editing rules is to extract them from the proto files with `+rules2yml.py+`.
It has the following options:
[source,bash]
----
usage: python3 rules2yml.py [-h] [--dir DIR] [--full-osi]

Export the rules of *.proto files into the *.yml format so it can be used by
the validator.

options:
-h, --help show this help message and exit
--dir DIR, -d DIR Name of the directory where the yml rules will be stored.
--full-osi, -f Add is_set rule to all fields that do not contain it already.
----

The following example command will generate the yml rule files in a folder `rules` and add the `is_set` rule to every OSI field that does not have this rule anyways.
This will result in a rule base to check for a full interface.

[source,bash]
----
python3 rules2yml.py --dir rules --full-osi
----

The resulting yml rule files can be freely edited and are explained in the following in more detail.

== File structure

Below you can see an example of the
https://github.com/OpenSimulationInterface/osi-validation/blob/master/rules/osi_detectedlane.yml[osi_detectedlane.yml]
rule file for
https://github.com/OpenSimulationInterface/open-simulation-interface/blob/master/osi_detectedlane.proto[osi_detectedlane.proto]:
Below you can see an example of the osi_detectedlane.yml rule file for
https://github.com/OpenSimulationInterface/open-simulation-interface/blob/master/osi_detectedlane.proto[osi_detectedlane.proto] without the `--full-osi` option.

[source,YAML]
----
Expand Down Expand Up @@ -46,10 +64,11 @@ apply to that specific field. For example the probability of the message

The rules can either be with or without any parameters.

....
[source,YAML]
----
is_greater_than_or_equal_to: 0.0
is_equal: 1
....
----

In the case a rule has a parameter, it is written as a
[.title-ref]#mapping# ( a [.title-ref]#scalar# followed by a colon ":").
Expand All @@ -62,15 +81,16 @@ https://github.com/OpenSimulationInterface/osi-validation/blob/master/osivalidat
See also examples for available rules which can be defined in `+*.yml+`
files below:

[source,python]
[source,YAML]
----
is_set:
is_greater_than: 1
is_greater_than_or_equal_to: 1
is_less_than_or_equal_to: 10
is_less_than: 2
is_equal: 1
is_different: 2
is_globally_unique
is_globally_unique:
refers_to: Lane
is_iso_country_code:
first_element: {is_equal: 0.13, is_greater_than: 0.13}
Expand All @@ -80,12 +100,13 @@ check_if: [{is_equal: 2, is_greater_than: 3, target: this.y}, {do_check: {is_equ

== Severity

When an attribute does not comply with a rule, a Warning is throw. An
Error will be throw if an exclamation mark is written at the end of the
When an attribute does not comply with a rule, a warning is thrown. An
error will be thrown if an exclamation mark is written at the end of the
verb of a rule.

=== Example

....
[source,YAML]
----
is_greater_than!: 0
....
----
2 changes: 1 addition & 1 deletion open-simulation-interface
Submodule open-simulation-interface updated 69 files
+24 −0 .github/.pyspelling.yml
+390 −0 .github/spelling_custom_words_en_US.txt
+1 −1 .github/workflows/antora-generator.yml
+128 −36 .github/workflows/protobuf.yml
+17 −9 .github/workflows/release.yml
+2 −2 .gitignore
+3 −1 CMakeLists.txt
+1 −0 MANIFEST.in
+1 −1 VERSION
+4 −2 doc/architecture/architecture_overview.adoc
+0 −9 doc/architecture/feature_data.adoc
+29 −0 doc/architecture/formatting_script.adoc
+0 −56 doc/architecture/formatting_scripts.adoc
+1 −0 doc/architecture/reference_points_coordinate_systems.adoc
+4 −0 doc/architecture/sensor_data.adoc
+6 −5 doc/architecture/trace_file_formats.adoc
+2 −0 doc/architecture/trace_file_naming.adoc
+ doc/images/OSI_Planned_Route.png
+ doc/images/OSI_Route_Segment.png
+ doc/images/osi-streaming-principle.png
+1 −3 doc/open-simulation-interface_user_guide.adoc
+26 −12 doc/setup/installing_linux_python.adoc
+2 −2 doc/setup/installing_prerequisites.adoc
+30 −2 doc/setup/installing_windows_python.adoc
+0 −244 format/OSITrace.py
+0 −63 format/osi2read.py
+0 −75 format/txt2osi.py
+66 −0 osi3trace/osi2read.py
+169 −0 osi3trace/osi_trace.py
+165 −9 osi_common.proto
+18 −2 osi_detectedobject.proto
+2 −2 osi_environment.proto
+34 −12 osi_featuredata.proto
+60 −2 osi_groundtruth.proto
+8 −3 osi_hostvehicledata.proto
+6 −5 osi_lane.proto
+5 −1 osi_logicaldetectiondata.proto
+140 −6 osi_logicallane.proto
+8 −0 osi_motionrequest.proto
+70 −15 osi_object.proto
+7 −6 osi_referenceline.proto
+5 −1 osi_roadmarking.proto
+138 −0 osi_route.proto
+50 −0 osi_sensordata.proto
+16 −3 osi_sensorview.proto
+15 −2 osi_sensorviewconfiguration.proto
+9 −1 osi_streamingupdate.proto
+30 −14 osi_trafficcommand.proto
+12 −0 osi_trafficcommandupdate.proto
+2 −1 osi_trafficlight.proto
+12 −11 osi_trafficsign.proto
+9 −1 osi_trafficupdate.proto
+35 −0 pyproject.toml
+2 −0 requirements_tests.txt
+2 −1 rules.yml
+111 −78 setup.py
+107 −44 tests/test_comment_type.py
+26 −9 tests/test_doxygen_output.py
+18 −5 tests/test_invalid_comment.py
+76 −27 tests/test_invalid_enum.py
+87 −47 tests/test_invalid_html.py
+168 −59 tests/test_invalid_message.py
+8 −3 tests/test_invalid_punctuation.py
+7 −2 tests/test_invalid_tabs.py
+8 −3 tests/test_newline.py
+23 −4 tests/test_non_ascii.py
+20 −5 tests/test_osi_trace.py
+77 −22 tests/test_rules.py
+39 −10 tests/test_units.py
2 changes: 1 addition & 1 deletion osivalidator/osi_general_validator.py
Original file line number Diff line number Diff line change
Expand Up @@ -184,7 +184,7 @@ def process_message(message, timestep, data_type):
LOGGER.info(None, f"Analyze message of timestamp {timestamp}", False)

# Check common rules
getattr(rule_checker, "is_valid")(
getattr(rule_checker, "check_children")(
linked_proto_field.LinkedProtoField(message, name=data_type),
VALIDATION_RULES.get_rules().get_type(data_type),
)
Expand Down
17 changes: 4 additions & 13 deletions osivalidator/osi_rules_implementations.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,15 +24,7 @@ def add_default_rules_to_subfields(message, type_rules):
)

if descriptor.message_type:
field_rules.add_rule(osi_rules.Rule(verb="is_valid"))

is_set_severity = (
osi_rules.Severity.WARN
if field_rules.has_rule("is_optional")
else osi_rules.Severity.ERROR
)

field_rules.add_rule(osi_rules.Rule(verb="is_set", severity=is_set_severity))
field_rules.add_rule(osi_rules.Rule(verb="check_children"))


# DECORATORS
Expand Down Expand Up @@ -100,7 +92,7 @@ def wrapper(self, field, rule, **kwargs):


@rule_implementation
def is_valid(self, field, rule):
def check_children(self, field, rule):
"""Check if a field message is valid, that is all the inner rules of the
message in the field are complying.

Expand All @@ -122,20 +114,19 @@ def is_valid(self, field, rule):
else:
subfield_rules = rule.root.get_type(field.message_type)

result = True
# Add default rules for each subfield that can be validated (default)
add_default_rules_to_subfields(field, subfield_rules)

# loop over the fields in the rules
for subfield_rules in subfield_rules.fields.values():
for subfield_rule in subfield_rules.rules.values():
result = self.check_rule(field, subfield_rule) and result
self.check_rule(field, subfield_rule)

# Resolve ID and references
if not field.parent:
self.id_manager.resolve_unicity(self.timestamp)
self.id_manager.resolve_references(self.timestamp)
return result
return True


@rule_implementation
Expand Down
32 changes: 29 additions & 3 deletions rules2yml.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,11 +23,18 @@ def command_line_arguments():
required=False,
type=str,
)
parser.add_argument(
"--full-osi",
"-f",
help="Add is_set rule to all fields that do not contain it already.",
action="store_true",
required=False,
)

return parser.parse_args()


def gen_yml_rules(dir_name="rules"):
def gen_yml_rules(dir_name="rules", full_osi=False):
with open(r"open-simulation-interface/rules.yml") as file:
yaml = YAML(typ="safe")
rules_dict = yaml.load(file)
Expand Down Expand Up @@ -161,6 +168,15 @@ def gen_yml_rules(dir_name="rules"):
yml_file.write(
(2 * numMessage) * " " + f"{field}:\n"
)
# If option --full-osi is enabled:
# Check if is_set is already a rule for the current field, if not, add it.
if full_osi and not any(
"is_set" in rule for rule in rules
):
yml_file.write(
(2 * numMessage + 2) * " "
+ f"- is_set:\n"
)

if shiftCounter:
for rule in rules:
Expand Down Expand Up @@ -217,11 +233,21 @@ def gen_yml_rules(dir_name="rules"):
(2 * numMessage + 8) * " "
+ f"- {rule_list[2]}: {rule_list[3]}\n"
)
elif "is_globally_unique" in rule_list:
# Standalone rules
elif any(
list_item
in [
"is_globally_unique",
"is_set",
"is_iso_country_code",
]
for list_item in rule_list
):
yml_file.write(
(2 * numMessage + 2) * " "
+ f"-{rule}:\n"
)
# Values or parameters of rules
else:
yml_file.write(
(2 * numMessage + 2) * " "
Expand All @@ -235,7 +261,7 @@ def gen_yml_rules(dir_name="rules"):
def main():
# Handling of command line arguments
args = command_line_arguments()
gen_yml_rules(args.dir)
gen_yml_rules(args.dir, args.full_osi)


if __name__ == "__main__":
Expand Down
108 changes: 0 additions & 108 deletions tests/tests_rules/test_is_valid.py

This file was deleted.