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

feat: Multivariate threshold using Mahalanobis distance #234

Merged
merged 10 commits into from
Jul 26, 2023
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
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
2 changes: 1 addition & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ jobs:
- uses: actions/checkout@v3

- name: Install poetry
run: pipx install poetry==1.4.2
run: pipx install poetry==1.5.1

- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/coverage.yml
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ jobs:
- uses: actions/checkout@v3

- name: Install poetry
run: pipx install poetry==1.4.2
run: pipx install poetry==1.5.1

- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/pypi.yml
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ jobs:
- uses: actions/checkout@v3

- name: Install poetry
run: pipx install poetry
run: pipx install poetry==1.5.1

- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
Expand Down
8 changes: 7 additions & 1 deletion numalogic/config/factory.py
Original file line number Diff line number Diff line change
Expand Up @@ -63,12 +63,18 @@ class PostprocessFactory(_ObjectFactory):
class ThresholdFactory(_ObjectFactory):
"""Factory class to create threshold instances."""

from numalogic.models.threshold import StdDevThreshold, StaticThreshold, SigmoidThreshold
from numalogic.models.threshold import (
StdDevThreshold,
MahalanobisThreshold,
StaticThreshold,
SigmoidThreshold,
)

_CLS_MAP: ClassVar[dict] = {
"StdDevThreshold": StdDevThreshold,
"StaticThreshold": StaticThreshold,
"SigmoidThreshold": SigmoidThreshold,
"MahalanobisThreshold": MahalanobisThreshold,
}


Expand Down
3 changes: 2 additions & 1 deletion numalogic/models/threshold/__init__.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
from numalogic.models.threshold._std import StdDevThreshold
from numalogic.models.threshold._mahalanobis import MahalanobisThreshold
from numalogic.models.threshold._static import StaticThreshold, SigmoidThreshold

__all__ = ["StdDevThreshold", "StaticThreshold", "SigmoidThreshold"]
__all__ = ["StdDevThreshold", "StaticThreshold", "SigmoidThreshold", "MahalanobisThreshold"]
169 changes: 169 additions & 0 deletions numalogic/models/threshold/_mahalanobis.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,169 @@
# Copyright 2022 The Numaproj Authors.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

from typing import Final

import numpy as np
import numpy.typing as npt

from numalogic.base import BaseThresholdModel
from typing_extensions import Self

from numalogic.tools.exceptions import ModelInitializationError

_INLIER: Final[int] = 0
_OUTLIER: Final[int] = 1


class MahalanobisThreshold(BaseThresholdModel):
"""
Multivariate threshold estimator using Mahalanobis distance.

The maximum outlier probability is used to calculate the k value
using Chebyshev's inequality. This basically means that the
probability of values lying outside the interval
(mean - k * std, mean + k * std) is not more than max_outlier_prob.
A value of 0.1 means that 90% of the values lie within the interval and
10% of the values lie outside the interval.

The threshold is calculated as the
mean of Mahalanobis distance plus k times the standard deviation
of Mahalanobis distance.

Args:
----
max_outlier_prob: maximum outlier probability (default: 0.1)

Raises
------
ValueError: if max_outlier_prob is not in range (0, 1)

>>> import numpy as np
>>> from numalogic.models.threshold import MahalanobisThreshold
>>> rng = np.random.default_rng(42)
...
>>> x_train = rng.normal(size=(100, 15))
>>> x_test = rng.normal(size=(30, 15))
...
>>> clf = MahalanobisThreshold()
>>> clf.fit(x_train)
...
>>> y_test = clf.predict(x_test)
>>> test_scores = clf.score_samples(x_test)
"""

def __init__(self, max_outlier_prob: float = 0.1):
if not 0.0 < max_outlier_prob < 1.0:
raise ValueError("max_outlier_prob should be in range (0, 1)")
self._k = self._chebyshev_k(max_outlier_prob)
self._distr_mean = None
self._cov_inv = None
self._md_thresh = None
self._is_fitted = False

@property
def threshold(self):
"""Returns the threshold value."""
return self._md_thresh

@property
def std_factor(self):
"""Returns the k value calculated using Chebyshev's inequality."""
return self._k

@staticmethod
def _chebyshev_k(p: float) -> float:
"""Calculate the k value using Chebyshev's inequality."""
return np.reciprocal(np.sqrt(p))

def fit(self, x: npt.NDArray[float]) -> Self:
"""
Fit the estimator on the training set.

Args:
----
x: training data

Returns
-------
self
"""
self._distr_mean = np.mean(x, axis=0)
cov = np.cov(x, rowvar=False)
s0nicboOm marked this conversation as resolved.
Show resolved Hide resolved
self._cov_inv = np.linalg.inv(cov)
ab93 marked this conversation as resolved.
Show resolved Hide resolved
mahal_dist = self.mahalanobis(x)
self._md_thresh = np.mean(mahal_dist) + self._k * np.std(mahal_dist)
ab93 marked this conversation as resolved.
Show resolved Hide resolved
self._is_fitted = True
ab93 marked this conversation as resolved.
Show resolved Hide resolved
return self

def mahalanobis(self, x: npt.NDArray[float]) -> npt.NDArray[float]:
"""
Calculate the Mahalanobis distance.

Args:
----
x: input data

Returns
-------
Mahalanobis distance vector
"""
x_distance = x - self._distr_mean
mahal_grid = x_distance @ self._cov_inv @ x_distance.T
Copy link
Member Author

Choose a reason for hiding this comment

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

x_distance is a matrix of shape (n_obs, n_features). _cov_inv is (n_features, n_features), mahal_grid is (n_obs, n_obs). Therefore, it works on a matrix, not a number.

Copy link

Choose a reason for hiding this comment

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

Got it. x is expected to be array of vectors, not just one vector.

On a related note, is there a means to include dimension in the param type? Presently, type like npt.NDArray[float] doesn't carry dimension info.

ab93 marked this conversation as resolved.
Show resolved Hide resolved
return np.sqrt(np.diagonal(mahal_grid))

def predict(self, x: npt.NDArray[float]) -> npt.NDArray[int]:
"""
Returns an integer array of same shape as input.
0 denotes inlier, 1 denotes outlier.

Args:
----
x: input data

Returns
-------
Integer Array of 0s and 1s

Raises
------
RuntimeError: if the model is not fitted yet
"""
if not self._is_fitted:
raise ModelInitializationError("Model not fitted yet.")
md = self.mahalanobis(x)
y_hat = np.zeros_like(x)
y_hat[md < self._md_thresh] = _INLIER
ab93 marked this conversation as resolved.
Show resolved Hide resolved
y_hat[md >= self._md_thresh] = _OUTLIER
return y_hat

def score_samples(self, x: npt.NDArray[float]) -> npt.NDArray[float]:
"""
Returns the outlier score for each sample.

Score is calculated as the ratio of Mahalanobis distance to threshold.
A score greater than 1.0 means that the sample is an outlier.

Args:
----
x: input data

Returns
-------
Outlier score for each sample

Raises
------
RuntimeError: if the model is not fitted yet
"""
if not self._is_fitted:
raise ModelInitializationError("Model not fitted yet.")
return self.mahalanobis(x) / self._md_thresh
3 changes: 2 additions & 1 deletion numalogic/registry/dynamodb_registry.py
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,8 @@ class DynamoDBRegistry(ArtifactManager):
role: AWS role with access to DynamoDB table
models_to_retain: number of models to retain in the DB (default = 2)
cache_registry: ArtifactCache instance, must have an expiration set for the
model to be refreshed
model to be refreshed.

Examples
--------
>>> from numalogic.models.autoencoder.variants import VanillaAE
Expand Down
Loading