diff --git a/kloppy/domain/models/event.py b/kloppy/domain/models/event.py index 2b541900..dc1a1d08 100644 --- a/kloppy/domain/models/event.py +++ b/kloppy/domain/models/event.py @@ -186,6 +186,7 @@ class EventType(Enum): PLAYER_ON (EventType): PLAYER_OFF (EventType): RECOVERY (EventType): + MISCONTROL (EventType): BALL_OUT (EventType): FOUL_COMMITTED (EventType): GOALKEEPER (EventType): @@ -205,6 +206,7 @@ class EventType(Enum): PLAYER_ON = "PLAYER_ON" PLAYER_OFF = "PLAYER_OFF" RECOVERY = "RECOVERY" + MISCONTROL = "MISCONTROL" BALL_OUT = "BALL_OUT" FOUL_COMMITTED = "FOUL_COMMITTED" GOALKEEPER = "GOALKEEPER" @@ -911,6 +913,20 @@ class BallOutEvent(Event): event_name: str = "ball_out" +@dataclass(repr=False) +@docstring_inherit_attributes(Event) +class MiscontrolEvent(Event): + """ + MiscontrolEvent + Attributes: + event_type (EventType): `EventType.MISCONTROL` (See [`EventType`][kloppy.domain.models.event.EventType]) + event_name (str): "miscontrol" + """ + + event_type: EventType = EventType.MISCONTROL + event_name: str = "miscontrol" + + @dataclass(repr=False) @docstring_inherit_attributes(Event) class FoulCommittedEvent(Event): @@ -1038,6 +1054,7 @@ def generic_record_converter(event: Event): "FormationChangeEvent", "EventDataset", "RecoveryEvent", + "MiscontrolEvent", "FoulCommittedEvent", "BallOutEvent", "SetPieceType", diff --git a/kloppy/domain/services/event_factory.py b/kloppy/domain/services/event_factory.py index bbbda6f9a..d8a7f0e9 100644 --- a/kloppy/domain/services/event_factory.py +++ b/kloppy/domain/services/event_factory.py @@ -9,6 +9,7 @@ GenericEvent, TakeOnEvent, RecoveryEvent, + MiscontrolEvent, CarryEvent, DuelEvent, ClearanceEvent, @@ -79,6 +80,9 @@ def build_generic(self, **kwargs) -> GenericEvent: def build_recovery(self, **kwargs) -> RecoveryEvent: return create_event(RecoveryEvent, **kwargs) + def build_miscontrol(self, **kwargs) -> MiscontrolEvent: + return create_event(MiscontrolEvent, **kwargs) + def build_take_on(self, **kwargs) -> TakeOnEvent: return create_event(TakeOnEvent, **kwargs) diff --git a/kloppy/infra/serializers/event/opta/deserializer.py b/kloppy/infra/serializers/event/opta/deserializer.py index b11f4cb4..db1a0975 100644 --- a/kloppy/infra/serializers/event/opta/deserializer.py +++ b/kloppy/infra/serializers/event/opta/deserializer.py @@ -74,6 +74,7 @@ EVENT_TYPE_CARD = 17 EVENT_TYPE_RECOVERY = 49 EVENT_TYPE_FORMATION_CHANGE = 40 +EVENT_TYPE_BALL_TOUCH = 61 EVENT_TYPE_SAVE = 10 EVENT_TYPE_CLAIM = 11 @@ -100,6 +101,7 @@ EVENT_TYPE_SHOT_SAVED, EVENT_TYPE_SHOT_GOAL, EVENT_TYPE_RECOVERY, + EVENT_TYPE_BALL_TOUCH, ) EVENT_QUALIFIER_GOAL_KICK = 124 @@ -765,6 +767,10 @@ def deserialize(self, inputs: OptaInputs) -> EventDataset: ) event = self.event_factory.build_goalkeeper_event( **goalkeeper_event_kwargs, + elif (type_id == EVENT_TYPE_BALL_TOUCH) & (outcome == 0): + event = self.event_factory.build_miscontrol( + result=None, + qualifiers=None, **generic_event_kwargs, ) elif (type_id == EVENT_TYPE_FOUL_COMMITTED) and ( diff --git a/kloppy/infra/serializers/event/statsbomb/deserializer.py b/kloppy/infra/serializers/event/statsbomb/deserializer.py index 806f7b67..4bf88632 100644 --- a/kloppy/infra/serializers/event/statsbomb/deserializer.py +++ b/kloppy/infra/serializers/event/statsbomb/deserializer.py @@ -55,6 +55,7 @@ SB_EVENT_TYPE_GOALKEEPER_EVENT = 23 SB_EVENT_TYPE_PASS = 30 SB_EVENT_TYPE_50_50 = 33 +SB_EVENT_TYPE_MISCONTROL = 38 SB_EVENT_TYPE_CARRY = 43 SB_EVENT_TYPE_HALF_START = 18 @@ -856,6 +857,13 @@ def deserialize(self, inputs: StatsBombInputs) -> EventDataset: **generic_event_kwargs, ) new_events.append(clearance_event) + elif event_type == SB_EVENT_TYPE_MISCONTROL: + miscontrol_event = self.event_factory.build_miscontrol( + result=None, + qualifiers=None, + **generic_event_kwargs, + ) + new_events.append(miscontrol_event) # For dribble and carry the definitions # are flipped between StatsBomb and kloppy elif event_type == SB_EVENT_TYPE_DRIBBLE: diff --git a/kloppy/infra/serializers/event/wyscout/deserializer_v2.py b/kloppy/infra/serializers/event/wyscout/deserializer_v2.py index 89a96c5c..26d58d20 100644 --- a/kloppy/infra/serializers/event/wyscout/deserializer_v2.py +++ b/kloppy/infra/serializers/event/wyscout/deserializer_v2.py @@ -443,6 +443,18 @@ def deserialize(self, inputs: WyscoutInputs) -> EventDataset: **clearance_event_args, **generic_event_args, ) + elif ( + raw_event["subEventId"] + == wyscout_events.OTHERS_ON_BALL.TOUCH + ) & (_has_tag(raw_event, wyscout_tags.MISSED_BALL)): + miscontrol_event_args = { + "result": None, + "qualifiers": _generic_qualifiers(raw_event), + } + event = self.event_factory.build_miscontrol( + **miscontrol_event_args, + **generic_event_args, + ) else: recovery_event_args = _parse_recovery(raw_event) event = self.event_factory.build_recovery( diff --git a/kloppy/infra/serializers/event/wyscout/deserializer_v3.py b/kloppy/infra/serializers/event/wyscout/deserializer_v3.py index 7a8ba9ee..36c86966 100644 --- a/kloppy/infra/serializers/event/wyscout/deserializer_v3.py +++ b/kloppy/infra/serializers/event/wyscout/deserializer_v3.py @@ -71,13 +71,6 @@ def _parse_team(raw_events, wyId: str, ground: Ground) -> Team: return team -def _has_tag(raw_event, tag_id) -> bool: - for tag in raw_event["tags"]: - if tag["id"] == tag_id: - return True - return False - - def _generic_qualifiers(raw_event: Dict) -> List[Qualifier]: qualifiers: List[Qualifier] = [] diff --git a/kloppy/tests/files/opta_f24.xml b/kloppy/tests/files/opta_f24.xml index bd27aa7e..50ac1b17 100644 --- a/kloppy/tests/files/opta_f24.xml +++ b/kloppy/tests/files/opta_f24.xml @@ -286,6 +286,9 @@ + + + diff --git a/kloppy/tests/test_adapter.py b/kloppy/tests/test_adapter.py index f9ae70a5..772f254c 100644 --- a/kloppy/tests/test_adapter.py +++ b/kloppy/tests/test_adapter.py @@ -57,4 +57,4 @@ def read_to_stream(self, url: str, output: BinaryIO): # Asserts borrowed from `test_opta.py` assert dataset.metadata.provider == Provider.OPTA assert dataset.dataset_type == DatasetType.EVENT - assert len(dataset.events) == 28 + assert len(dataset.events) == 29 diff --git a/kloppy/tests/test_opta.py b/kloppy/tests/test_opta.py index ecba4750..4301e716 100644 --- a/kloppy/tests/test_opta.py +++ b/kloppy/tests/test_opta.py @@ -42,7 +42,7 @@ def test_correct_deserialization(self, f7_data: str, f24_data: str): ) assert dataset.metadata.provider == Provider.OPTA assert dataset.dataset_type == DatasetType.EVENT - assert len(dataset.events) == 28 + assert len(dataset.events) == 29 assert len(dataset.metadata.periods) == 2 assert ( dataset.events[10].ball_owning_team == dataset.metadata.teams[1] @@ -115,6 +115,9 @@ def test_correct_deserialization(self, f7_data: str, f24_data: str): assert dataset.events[18].result.value == "OWN_GOAL" # 2318697001 # Check OFFSIDE pass has end_coordinates assert dataset.events[20].receiver_coordinates.x == 89.3 # 2360555167 + assert ( + dataset.events[23].event_type == EventType.MISCONTROL + ) # 250913217 # Check goalkeeper qualifiers assert ( diff --git a/kloppy/tests/test_statsbomb.py b/kloppy/tests/test_statsbomb.py index 7b056c5b..745d1fec 100644 --- a/kloppy/tests/test_statsbomb.py +++ b/kloppy/tests/test_statsbomb.py @@ -162,6 +162,7 @@ def test_correct_deserialization( == DuelType.GROUND ) assert dataset.events[272].event_type == EventType.CLEARANCE + assert dataset.events[68].event_type == EventType.MISCONTROL assert ( dataset.events[761].get_qualifier_value(GoalkeeperQualifier) diff --git a/kloppy/tests/test_wyscout.py b/kloppy/tests/test_wyscout.py index 5be6a5b9..93e757e5 100644 --- a/kloppy/tests/test_wyscout.py +++ b/kloppy/tests/test_wyscout.py @@ -68,6 +68,9 @@ def test_correct_v2_deserialization(self, event_v2_data: Path): ) assert dataset.records[2].coordinates == Point(29.0, 6.0) assert dataset.events[137].event_type == EventType.CLEARANCE + assert dataset.events[11].event_type == EventType.MISCONTROL + assert dataset.events[136].event_type == EventType.CLEARANCE + assert ( dataset.events[39].get_qualifier_value(DuelQualifier) == DuelType.GROUND