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

Add Covariate Shift Conformal Regressor #151

Open
wants to merge 5 commits into
base: master
Choose a base branch
from

Conversation

RudrakshTuwani
Copy link

@RudrakshTuwani RudrakshTuwani commented Mar 30, 2022

Description

Add covariate shift conformal method (https://arxiv.org/abs/1904.06019) to MAPIE for regression.

Fixes #72

Type of change

Please remove options that are irrelevant.

  • New feature (non-breaking change which adds functionality)
  • This change requires a documentation update

How Has This Been Tested?

Reproducing the results of the original paper.

Checklist

  • I have read the contributing guidelines
  • I have updated the HISTORY.rst and AUTHORS.rst files
  • Linting passes successfully : make lint
  • Typing passes successfully : make type-check
  • Unit tests pass successfully : make tests
  • Coverage is 100% : make coverage
  • Documentation builds successfully : make doc

@RudrakshTuwani
Copy link
Author

Hey @gmartinonQM, please let me know how to prioritize tasks and would really appreciate feedback. Thanks! :)

@gmartinonQM
Copy link
Collaborator

Thanks for the PR @RudrakshTuwani , for sure. Maybe we will take several days to discuss it internally. Stay tuned !

@vtaquet
Copy link
Member

vtaquet commented May 6, 2022

Hi @RudrakshTuwani , thank you very much for the PR ! The implementation and the notebook look really great. I am sorry for the delay of our feedback, we have been quite busy during the last past weeks. Before giving you a more thorough feedback by next week, just a few general comments :

  • please move the class MapieCovShiftRegressor in a dedicated module (like covariate_shift_regression.py)
  • please move your notebook in the notebooks/regression folder. An alternative is to create a notebook-like example from a .py script like in this example.

@vtaquet vtaquet self-requested a review May 6, 2022 19:26
@RudrakshTuwani
Copy link
Author

Hello @vtaquet , thank you for getting back to me! No worries about the delay! :)

@RudrakshTuwani
Copy link
Author

Couldn't push to @vtaquet 's branch, so pulled the changes to the fork and re-opened PR. Still a WIP!

Copy link
Collaborator

@thibaultcordier thibaultcordier left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Your contribution is relevant and appreciated for the evolution of MAPIE for conformal prediction with covariate shift. Moreover, it provides the basis for future developments towards conditional conformal prediction. In this perspective, I give you a list of suggestions to improve your code.

  • I suggest you implement a new WeightedConformalScore class (in a new file weighted_conformity_scores.py), which inherits from ConformalScore with all your current changes.
  • Thus the CovariateShiftConformityScore class will inherit from WeightedConformalScore and will be the only class that benefits from your changes without impacting the others.
  • In MapieRegressor class, I suggest to adapt the if condition with a test like isinstance(self.conformity_score_function_, WeightedConformityScore). Is this possible in your opinion? Or am I missing something?

I remain available if you have any questions and thank you in advance for your feedback.

@@ -30,6 +32,7 @@ def __init__(
- get_estimation_distribution and
- get_signed_conformity_scores
by default True.
compute_score_weights : TODO
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Need to define what is compute_score_weights


@abstractmethod
def get_signed_conformity_scores(
self, y: ArrayLike, y_pred: ArrayLike,
self, y: ArrayLike, y_pred: ArrayLike, conformity_scores: ArrayLike
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is a mistake: conformity_scores should not be an argument to get_signed_conformity_scores.

self,
y: NDArray,
y_pred: NDArray,
conformity_scores: NDArray,
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is no reason to change the style (for consistency).

self, y: ArrayLike, y_pred: ArrayLike,
self,
y: ArrayLike,
y_pred: ArrayLike,
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is no reason to change the style (for consistency).

self, y: ArrayLike, y_pred: ArrayLike,
self,
y: ArrayLike,
y_pred: ArrayLike,
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is no reason to change the style (for consistency).

@@ -700,25 +724,41 @@ def predict(
)
)

if self.conformity_score_function_.compute_score_weights:
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I understand the motivation to modify the ConformityScore class by adding a compute_score_weights attribute and a get_weights method.

I suggest to adapt the if condition with a test like isinstance(self.conformity_score_function_, WeightedConformityScore). Is this possible in your opinion? Or am I missing something?

@@ -484,6 +485,29 @@ def _pred_multi(self, X: ArrayLike) -> NDArray:
y_pred_multi = self._aggregate_with_mask(y_pred_multi, self.k_)
return y_pred_multi

def _quantile(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A description would be appreciated (even if the method name is explicit).

"""
Returns the indices that would sort the array.
In case of ties, breaks them randomly.
"""
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A detailed description would be appreciated (with Parameters, Returns, Raises and Examples sections if relevant).

"""
Calculates the weighted empirical quantile of `values`.
Converted to Python from https://github.com/ryantibs/conformal/
"""
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A detailed description would be appreciated (with Parameters, Returns, Raises and Examples sections if relevant).

mapie/utils.py Show resolved Hide resolved
@RudrakshTuwani
Copy link
Author

RudrakshTuwani commented Mar 14, 2023

Thanks for the feedback! I will try to address the comments over the weekend. :)

@Valentin-Laurent
Copy link
Collaborator

Hello @RudrakshTuwani, long time no see!
Do you still plan to work on that PR?
Best

@Valentin-Laurent Valentin-Laurent added the PR - Work in progress (contributor) A contributor is working on it. label Jan 13, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
PR - Work in progress (contributor) A contributor is working on it.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[ENHANCEMENT] Covariate shift conformal
6 participants