Skip to content

Latest commit

 

History

History
341 lines (252 loc) · 21.8 KB

File metadata and controls

341 lines (252 loc) · 21.8 KB

Google Summer of Code 2024 - Sktime Project Banner


⭐️ About Project

Program Google Summer of Code, 2024
Organisation Sktime: A unified framework for ML with time series
Project Sktime integration with deep learning backends - pytorch and huggingface - Dashboard
Mentors Franz Király - Benedikt Heidrich - Anirban Ray
Project Length 350 hours (Large)

Overview

I worked with sktime as a Google Summer of Code student during the period late May to August 2024. This post is created to summarise the work I’ve done over this period as the work product submission required to be submitted at the end of GSoC.

Sktime is a library for time series analysis in Python. It provides a unified interface for multiple time series learning tasks like classification, regression, clustering, annotation, and forecasting.

My project was focused on implementing and interfacing deep learning models in sktime leveraging PyTorch and Hugging Face.

Topics

Data Science - AI - Time Series - Toolbox Frameworks - Machine Learning - Deep Learning

Technologies

Python - GitHub - Pytorch - Huggingface - Scikit-Learn

Outcomes

  • Enhanced skills in PyTorch and Hugging Face.
  • Improved Python coding practices, focusing on writing efficient, high-quality code.
  • Gained experience in test-driven development and designing optimal code solutions.
  • Acquired knowledge of machine learning and deep learning techniques for time-series data analysis.
  • Familiarized with time-series-related libraries and packages.
  • Gained insights into the life cycle, development, and maintenance of a Python package through hands-on experience with sktime.
  • Enhanced experience in open-source project contributions.
  • Strengthened Git and GitHub skills.
  • Improved communication with mentors and collaboration on complex design decisions.

Challenges

  • Initially, managing time was difficult, as it was my first experience working on a project of this scale. However, I quickly adapted by breaking down tasks and prioritizing them, which improved my efficiency.
  • Maintaining consistency with daily stand-ups and weekly mentoring sessions was challenging at first. Over time, I became more comfortable with agile practices, and I eventually led the weekly stand-ups, which enhanced my productivity and integration with the team.
  • Understanding and modifying a large, complex codebase was tough, but I tackled this by gradually familiarizing myself with key components, which made navigating and contributing to the codebase easier.
  • Implementing features required deeper knowledge of certain libraries than I initially had. I addressed this by dedicating extra time to learning these libraries, which allowed me to implement features effectively.
  • Designing solutions proved more challenging than implementing them. Recognizing this, I shifted my focus to developing efficient design strategies before execution, leading to more robust solutions.

🎯 Contributions

Pull Requests

These contributions primarily involve the implementation of new algorithms and the enhancement and fixing of existing ones.

Pull Request Status Title Related Issue
#6928 Draft [ENH] Global Forecast API for BaseDeepNetworkPyTorch based interfaces #6836
#6842 Open [ENH] Implements Autoregressive Wrapper #6802
#6571 Open [ENH] interface to TimesFM Forecaster #6408
#6791 Merged [ENH] Pytorch Classifier & de-novo implementation of Transformer #6786
#6712 Merged [ENH] Interface to TinyTimeMixer foundation model #6698
#6202 Merged [ENH] de-novo implementation of LTSFTransformer based on cure-lab research code base #4939
#6457 Merged [ENH] Extend HFTransformersForecaster for PEFT methods #6435
#6321 Merged [BUG] fixes failing test in neuralforecast auto freq, amid pandas freq deprecations
#6237 Merged [ENH] Update doc and behavior of freq="auto" in neuralforecast
#6367 Merged [MNT] final change cycle (0.30.0) for renaming cINNForecaster to CINNForecaster #6120
#6238 Merged [MNT] change cycle (0.29.0) for renaming cINNForecaster to CINNForecaster #6120

In addition to this, these PRs were submitted during the application review period.

Pull Request Status Title Related Issue
#6121 Merged [MNT] initialize change cycle (0.28.0) for renaming cINNForecaster to CINNForecaster #6120
#6039 Merged [ENH] NeuralForecastRNN should auto-detect freq
#6088 Merged [MNT] create build tool to check invalid backticks
#6023 Merged [DOC] Fix invalid use of single-grave in docstrings
#6116 Merged [ENH] Adds MSTL import statement in detrend #6085
#6059 Merged [ENH] Examples for YtoX transformer docstring

Walk Through

Here, I will walk through some of the major contributions, from the above pull requests, where I added estimators to sktime.

To see the working and inference of these estimators, please refer to CODE.ipynb.

Jupyter Logo

⚡ MVTSTransformerClassifier

This pull request introduces the MVTSTransformerClassifier, based on the paper "A Transformer-based Framework for Multivariate Time Series Representation Learning," applying it to classification and regression.

I implemented the BaseDeepClassifierPytorch class as a foundation for PyTorch-based classifiers. Then, I used the TSTransformerEncoderClassiregressor to build the PyTorch network. Finally, I created the MVTSTransformerClassifier class to integrate the network with the base class.

This process enhanced my understanding of transformer architecture.

The estimator can be loaded into sktime using the following code:

from sktime.classification.deep_learning import MVTSTransformerClassifier

model = MVTSTransformerClassifier(
    d_model=256,
    n_heads=4,
    num_layers=4,
    dim_feedforward=128,
    dropout=0.1,
    pos_encoding="fixed",
    activation="relu",
    norm="BatchNorm",
    freeze=False,
)

For more details on how the estimator works, please refer to CODE.ipynb.

⚡ TinyTimeMixer

TinyTimeMixer (TTM) is a compact, pre-trained model for time-series forecasting, developed and open-sourced by IBM Research.

In this PR, I integrated TTM into the sktime framework by forking the official code into the sktime/libs/granite_ttm directory, as the source package was not available on PyPI.

Next, I developed an interface for the estimator within the TinyTimeMixerForecaster class.

Throughout this implementation, I gained valuable experience in creating custom Hugging Face models and configurations, loading and modifying weights, altering architecture, and training newly initialized weights.

The estimator can now be loaded into sktime using the following code:

from sktime.forecasting.ttm import TinyTimeMixerForecaster

model = TinyTimeMixerForecaster(
    model_path="ibm/TTM",
    revision="main",
    validation_split=0.2,
    config=None,
    training_args=None,
    compute_metrics=None,
    callbacks=None,
    broadcasting=False,
    use_source_package=False,
)

For further details on how the estimator functions, please refer to CODE.ipynb.

⚡ LTSFTransformer

This pull request introduces the LTSFTransformer, an implementation based on the paper "Are Transformers Effective for Time Series Forecasting?" which explores the application of transformer architecture to time series forecasting.

To begin the implementation, I structured the transformer architecture in the sktime/networks/ltsf/layers directory, along with the PyTorch dataset class PytorchFormerDataset.

Next, I developed the LTSFTransformerNetwork interface class by leveraging the base PyTorch forecasting class, which connects to the network created in the previous step.

Throughout this implementation, I gained valuable insights into transformer architecture, particularly in applying various embeddings and encodings to temporal features in time series data.

The estimator can be loaded into sktime with the following code:

from sktime.forecasting.ltsf import LTSFTransformerForecaster

model = LTSFTransformerForecaster(
    seq_len=30,
    context_len=15,
    pred_len=15,
    num_epochs=50,
    batch_size=8,
    in_channels=1,
    individual=False,
    criterion=None,
    criterion_kwargs=None,
    optimizer=None,
    optimizer_kwargs=None,
    lr=0.002,
    position_encoding=True,
    temporal_encoding=True,
    temporal_encoding_type="embed",  # linear, embed, fixed-embed
    d_model=32,
    n_heads=1,
    d_ff=64,
    e_layers=1,
    d_layers=1,
    factor=1,
    dropout=0.1,
    activation="relu",
    freq="M",
)

For further details on how the estimator functions, please refer to CODE.ipynb.

⚡ TimesFM

TimesFM (Time Series Foundation Model) is a pre-trained model developed by Google Research, designed specifically for time-series forecasting.

While integrating this model into sktime, I encountered new libraries and packages. Due to dependency conflicts with the package available on PyPI, I forked the code to sktime/libs/timesfm.

I then created an interface for the model within the TimesFMForecaster class.

Throughout this implementation, I gained hands-on experience with foundation models and explored their capabilities.

This Pull Request is still in progress, but when merged, you can load the estimator into sktime using the following code:

from sktime.forecasting.timesfm_forecaster import TimesFMForecaster

forecaster = TimesFMForecaster(
    context_len=64,
    horizon_len=32,
)

For more details on how the estimator functions, please refer to CODE.ipynb.

⚡ PEFT for HFTransformersForecaster

  • Title: [ENH] Extend HFTransformersForecaster for PEFT Methods
  • Status: Merged
  • Pull Request: #6457
  • Related Issue: #6435

The HFTransformersForecaster in sktime allows users to load and fine-tune pre-trained models from Hugging Face. In this PR, I extended the HFTransformersForecaster to support Parameter-Efficient Fine-Tuning (PEFT) methods, enabling more efficient fine-tuning of large pre-trained models using customized configurations.

Through this implementation, I gained a deeper understanding of various PEFT techniques and how they can enhance the fine-tuning process for large-scale models.

You can now load the estimator in sktime with a PEFT configuration using the following code:

from sktime.forecasting.hf_transformers_forecaster import HFTransformersForecaster
from peft import LoraConfig

forecaster = HFTransformersForecaster(
    model_path="huggingface/autoformer-tourism-monthly",
    fit_strategy="peft",
    training_args={
        "num_train_epochs": 20,
        "output_dir": "test_output",
        "per_device_train_batch_size": 32,
    },
    config={
        "lags_sequence": [1, 2, 3],
        "context_length": 2,
        "prediction_length": 4,
        "use_cpu": True,
        "label_length": 2,
    },
    peft_config=LoraConfig(
        r=8,
        lora_alpha=32,
        target_modules=["q_proj", "v_proj"],
        lora_dropout=0.01,
    )
)

For more details on how the estimator works, please refer to CODE.ipynb.

⚡ AutoregressiveWrapper

  • Title: [ENH] Implement Autoregressive Wrapper
  • Status: Open
  • Pull Request: #6842
  • Related Issue: #6802

In sktime, some global forecasters require the forecasting horizon to be specified during the fitting process, limiting their ability to predict on different horizons afterward. This pull request introduces the AutoregressiveWrapper, which wraps around these forecasters, allowing them to forecast on varying horizons while fitting on a fixed horizon that is generated internally.

During this implementation, I deepened my understanding of pandas indexes, particularly in handling multi-indexes. By the end of the process, I was able to create efficient and reliable code.

This PR is still in progress, but once merged, you can load a forecaster and apply the AutoregressiveWrapper using the following code:

from sktime.forecasting.pytorchforecasting import PytorchForecastingNBeats
from sktime.forecasting.compose import AutoRegressiveWrapper

forecaster = PytorchForecastingNBeats(trainer_params={
    "max_epochs": 20,
})

wrapper = AutoRegressiveWrapper(
    forecaster=forecaster,
    horizon_length=5,
    aggregate_method=np.mean,
)

For more details on how the estimator works, please refer to CODE.ipynb.

⚡ Global Forecasting for PyTorch Models

  • Title: [ENH] Global Forecast API for BaseDeepNetworkPyTorch-Based Interfaces
  • Status: Draft
  • Pull Request: #6928
  • Related Issue: #6836

This PR enhances the BaseDeepNetworkPyTorch class to support global forecasting, enabling models like CINNForecaster and the LTSF family to operate as global forecasters.

Although still a work in progress, once merged, these models can be loaded and trained on hierarchical data, similar to other global forecasters in the sktime framework.

Future Work

Some relevant future work:

  • There is a list of foundation models, expected to be integrated in sktime - #6177
  • Some estimators are required to be extended for global forecasting interface - #6836
  • Enhancements are expected around the pytorch adapter for forecasting - #6641
  • Improvements are also planned with the global forecasting interface - #6997
  • Enabling PEFT for foundation models - #6968

⭐️ Acknowledgements

I had a great experience over the summer, and although the GSoC period is coming to an end, going forward I shall continue to remain a contributor to sktime. I'm incredibly thankful to both Google and sktime for giving me this opportunity, and to the welcoming community and amazing mentors at sktime for making this experience such a memorable one. There is no doubt that I am a better coder than I was 4 months ago, and I'm eagerly looking forward to learning more in the time to come.