Skip to content

Commit

Permalink
WIP
Browse files Browse the repository at this point in the history
  • Loading branch information
theOehrly committed Oct 21, 2023
1 parent 0fe2c54 commit 6458a70
Show file tree
Hide file tree
Showing 5 changed files with 224 additions and 145 deletions.
88 changes: 11 additions & 77 deletions fastf1/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,14 +44,15 @@
from functools import cached_property
import warnings
import typing
from typing import Optional, List, Literal, Iterable, Union, Tuple, Any
from typing import Optional, List, Literal, Iterable, Union, Tuple, Any, Type

import numpy as np
import pandas as pd

import fastf1
from fastf1 import _api as api
from fastf1 import ergast
from fastf1.internals.pandas_base import BaseDataFrame, BaseSeries
from fastf1.mvapi import get_circuit_info, CircuitInfo
from fastf1.logger import get_logger, soft_exceptions
from fastf1.utils import to_timedelta
Expand Down Expand Up @@ -2271,7 +2272,7 @@ def _calculate_t0_date(self, *tel_data_sets: dict):
self._t0_date = date_offset.round('ms')


class Laps(pd.DataFrame):
class Laps(BaseDataFrame):
"""Object for accessing lap (timing) data of multiple laps.
Args:
Expand Down Expand Up @@ -2425,8 +2426,7 @@ class Laps(pd.DataFrame):
}

_metadata = ['session']
_internal_names = pd.DataFrame._internal_names \
+ ['base_class_view', 'telemetry']
_internal_names = BaseDataFrame._internal_names + ['telemetry']
_internal_names_set = set(_internal_names)

QUICKLAP_THRESHOLD = 1.07
Expand Down Expand Up @@ -2465,29 +2465,9 @@ def __init__(self,
self.session = session

@property
def _constructor(self):
return Laps

@property
def _constructor_sliced(self):
def _constructor_sliced_horizontal(self) -> Type["Lap"]:
return Lap

def _constructor_sliced_from_mgr(self, mgr, axes):
if axes[0] is self.index:
# horizontal slice
ser = pd.Series._from_mgr(mgr, axes)
ser._name = None # caller is responsible for setting real name
return ser

# vertical slice
return super()._constructor_sliced_from_mgr(mgr, axes)

@property
def base_class_view(self):
"""For a nicer debugging experience; can now view as
dataframe in various IDEs"""
return pd.DataFrame(self)

@cached_property
def telemetry(self) -> Telemetry:
"""Telemetry data for all laps in `self`
Expand Down Expand Up @@ -3086,26 +3066,16 @@ def iterlaps(self, require: Optional[Iterable] = None) \
yield index, lap


class Lap(pd.Series):
class Lap(BaseSeries):
"""Object for accessing lap (timing) data of a single lap.
This class wraps :class:`pandas.Series`. It provides extra functionality for accessing a lap's associated
telemetry data.
"""
_metadata = ['session']
_internal_names = pd.Series._internal_names + ['telemetry']
_internal_names = BaseSeries._internal_names + ['telemetry']
_internal_names_set = set(_internal_names)

def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)

@property
def _constructor(self):
def _new(*args, **kwargs):
return Lap(*args, **kwargs).__finalize__(self)

return _new

@cached_property
def telemetry(self) -> Telemetry:
"""Telemetry data for this lap
Expand Down Expand Up @@ -3255,7 +3225,7 @@ def get_weather_data(self) -> pd.Series:
return pd.Series(index=self.session.weather_data.columns)


class SessionResults(pd.DataFrame):
class SessionResults(BaseDataFrame):
"""This class provides driver and result information for all drivers that
participated in a session.
Expand Down Expand Up @@ -3391,9 +3361,6 @@ class SessionResults(pd.DataFrame):
'Points': 'float64'
}

_internal_names = pd.DataFrame._internal_names + ['base_class_view']
_internal_names_set = set(_internal_names)

def __init__(self, *args, force_default_cols: bool = False, **kwargs):
if force_default_cols:
kwargs['columns'] = list(self._COL_TYPES.keys())
Expand All @@ -3412,35 +3379,12 @@ def __init__(self, *args, force_default_cols: bool = False, **kwargs):

self[col] = self[col].astype(_type)

def __repr__(self):
return self.base_class_view.__repr__()

@property
def _constructor(self):
return SessionResults

@property
def _constructor_sliced(self):
def _constructor_sliced_horizontal(self) -> Type["DriverResult"]:
return DriverResult

def _constructor_sliced_from_mgr(self, mgr, axes):
if axes[0] is self.index:
# horizontal slice
ser = pd.Series._from_mgr(mgr, axes)
ser._name = None # caller is responsible for setting real name
return ser

# vertical slice
return super()._constructor_sliced_from_mgr(mgr, axes)

@property
def base_class_view(self):
"""For a nicer debugging experience; can view DataFrame through
this property in various IDEs"""
return pd.DataFrame(self)


class DriverResult(pd.Series):
class DriverResult(BaseSeries):
"""This class provides driver and result information for a single driver.
This class subclasses a :class:`pandas.Series` and the usual methods
Expand All @@ -3459,19 +3403,9 @@ class DriverResult(pd.Series):
.. versionadded:: 2.2
"""

_internal_names = pd.DataFrame._internal_names + ['dnf']
_internal_names = BaseSeries._internal_names + ['dnf']
_internal_names_set = set(_internal_names)

def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)

@property
def _constructor(self):
def _new(*args, **kwargs):
return DriverResult(*args, **kwargs).__finalize__(self)

return _new

@property
def dnf(self) -> bool:
"""True if driver did not finish"""
Expand Down
47 changes: 13 additions & 34 deletions fastf1/ergast/interface.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,10 @@

from fastf1.req import Cache
import fastf1.ergast.structure as API
from fastf1.internals.pandas_base import BaseDataFrame, BaseSeries
from fastf1.version import __version__


import pandas as pd


BASE_URL = 'https://ergast.com/api/f1'
HEADERS = {'User-Agent': f'FastF1/{__version__}'}

Expand Down Expand Up @@ -101,7 +99,7 @@ def get_prev_result_page(self) -> Union['ErgastSimpleResponse',
)


class ErgastResultFrame(pd.DataFrame):
class ErgastResultFrame(BaseDataFrame):
"""
Wraps a Pandas ``DataFrame``. Additionally, this class can be
initialized from Ergast response data with automatic flattening and type
Expand All @@ -117,7 +115,7 @@ class ErgastResultFrame(pd.DataFrame):
auto_cast: Determines if values are automatically cast to the most
appropriate data type from their original string representation
"""
_internal_names = pd.DataFrame._internal_names + ['base_class_view']
_internal_names = BaseDataFrame._internal_names + ['base_class_view']
_internal_names_set = set(_internal_names)

def __init__(self, data=None, *,
Expand Down Expand Up @@ -164,43 +162,17 @@ def _flatten_element(cls, nested: dict, category: dict, cast: bool):
return nested, flat

@property
def _constructor(self):
return ErgastResultFrame

@property
def _constructor_sliced(self):
def _constructor_sliced_horizontal(self):
return ErgastResultSeries

def _constructor_sliced_from_mgr(self, mgr, axes):
if axes[0] is self.index:
# horizontal slice
ser = pd.Series._from_mgr(mgr, axes)
ser._name = None # caller is responsible for setting real name
return ser

# vertical slice
return super()._constructor_sliced_from_mgr(mgr, axes)

@property
def base_class_view(self):
"""For a nicer debugging experience; can view DataFrame through
this property in various IDEs"""
return pd.DataFrame(self)


class ErgastResultSeries(pd.Series):
class ErgastResultSeries(BaseSeries):
"""
Wraps a Pandas ``Series``.
Currently, no extra functionality is implemented.
"""

def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)

@property
def _constructor(self):
return ErgastResultSeries
pass


class ErgastRawResponse(ErgastResponseMixin, list):
Expand Down Expand Up @@ -269,6 +241,13 @@ class ErgastSimpleResponse(ErgastResponseMixin, ErgastResultFrame):
+ ErgastResponseMixin._internal_names
_internal_names_set = set(_internal_names)

@property
def _constructor(self):
# drop from ErgastSimpleResponse to ErgastResultFrame, removing the
# ErgastResponseMixin because a slice of the data is no longer a full
# response and pagination, ... is therefore not supported any more
return ErgastResultFrame


class ErgastMultiResponse(ErgastResponseMixin):
"""
Expand Down
39 changes: 5 additions & 34 deletions fastf1/events.py
Original file line number Diff line number Diff line change
Expand Up @@ -163,7 +163,7 @@
import datetime
import json
import warnings
from typing import Literal, Union, Optional
from typing import Callable, Literal, Union, Optional

import dateutil.parser

Expand All @@ -180,6 +180,7 @@
import fastf1._api
from fastf1.core import Session
import fastf1.ergast
from fastf1.internals.pandas_base import BaseSeries, BaseDataFrame
from fastf1.logger import get_logger, soft_exceptions
from fastf1.req import Cache
from fastf1.utils import recursive_dict_get, to_datetime, to_timedelta
Expand Down Expand Up @@ -733,7 +734,7 @@ def _get_schedule_from_ergast(year) -> "EventSchedule":
return schedule


class EventSchedule(pd.DataFrame):
class EventSchedule(BaseDataFrame):
"""This class implements a per-season event schedule.
For detailed information about the information that is available for each
Expand Down Expand Up @@ -782,9 +783,6 @@ class EventSchedule(pd.DataFrame):

_metadata = ['year']

_internal_names = pd.DataFrame._internal_names + ['base_class_view']
_internal_names_set = set(_internal_names)

def __init__(self, *args, year: int = 0,
force_default_cols: bool = False, **kwargs):
if force_default_cols:
Expand All @@ -805,33 +803,10 @@ def __init__(self, *args, year: int = 0,
self[col] = _type()
self[col] = self[col].astype(_type)

def __repr__(self):
return self.base_class_view.__repr__()

@property
def _constructor(self):
return EventSchedule

@property
def _constructor_sliced(self):
def _constructor_sliced_horizontal(self) -> Callable[..., "Event"]:
return Event

def _constructor_sliced_from_mgr(self, mgr, axes):
if axes[0] is self.index:
# horizontal slice
ser = pd.Series._from_mgr(mgr, axes)
ser._name = None # caller is responsible for setting real name
return ser

# vertical slice
return super()._constructor_sliced_from_mgr(mgr, axes)

@property
def base_class_view(self):
"""For a nicer debugging experience; can view DataFrame through
this property in various IDEs"""
return pd.DataFrame(self)

def is_testing(self):
"""Return `True` or `False`, depending on whether each event is a
testing event."""
Expand Down Expand Up @@ -935,7 +910,7 @@ def get_event_by_name(
return self._fuzzy_event_search(name)


class Event(pd.Series):
class Event(BaseSeries):
"""This class represents a single event (race weekend or testing event).
Each event consists of one or multiple sessions, depending on the type
Expand All @@ -956,10 +931,6 @@ def __init__(self, *args, year: int = None, **kwargs):
super().__init__(*args, **kwargs)
self.year = year

@property
def _constructor(self):
return Event

def is_testing(self) -> bool:
"""Return `True` or `False`, depending on whether this event is a
testing event."""
Expand Down
Loading

0 comments on commit 6458a70

Please sign in to comment.