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 SuperSimpleNet model #2428

Merged

Conversation

blaz-r
Copy link
Contributor

@blaz-r blaz-r commented Nov 21, 2024

📝 Description

  • This PR adds the SuperSimpleNet model implementation.
  • I also verified the performance on MVTec AD and the results are very close tho the ones reported in the paper.

✨ Changes

Select what type of change your PR is:

  • 🐞 Bug fix (non-breaking change which fixes an issue)
  • 🔨 Refactor (non-breaking change which refactors the code base)
  • 🚀 New feature (non-breaking change which adds functionality)
  • 💥 Breaking change (fix or feature that would cause existing functionality to not work as expected)
  • 📚 Documentation update
  • 🔒 Security update

✅ Checklist

Before you submit your pull request, please make sure you have completed the following steps:

  • 📋 I have summarized my changes in the CHANGELOG and followed the guidelines for my type of change (skip for minor changes, documentation updates, and test enhancements).
  • 📚 I have made the necessary updates to the documentation (if applicable).
  • 🧪 I have written tests that support my changes and prove that my fix is effective or my feature works (if applicable).

For more information about code review checklists, see the Code Review Checklist.

blaz-r and others added 19 commits November 21, 2024 09:53
Signed-off-by: Blaz Rolih <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: Blaz Rolih <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: Blaz Rolih <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
@blaz-r blaz-r force-pushed the feature/supersimplenet branch from 6acde98 to 3c65da6 Compare November 21, 2024 08:53
@samet-akcay
Copy link
Contributor

This is amazing @blaz-r! Thanks a lot! I'll review it shortly

Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
Signed-off-by: blaz.rolih <blaz.rolih@fri.uni-lj.si>
@blaz-r
Copy link
Contributor Author

blaz-r commented Nov 21, 2024

I have fixed the linter error and formated the files. The tests for SuperSimpleNet are still failing on onnx and openvino export locally. I'm not entirely sure how to fix this, but I'll take a lot when I get the time.

@blaz-r
Copy link
Contributor Author

blaz-r commented Nov 21, 2024

Tests are failing on imgaug, somethign with numpy version incompatiblity it seems.

@samet-akcay
Copy link
Contributor

samet-akcay commented Nov 22, 2024

@blaz-r, we will need to remove imgaug dependency to bump the numpy version.

directly predicts the anomaly map and score. The predicted anomaly map is upscaled to match the input image size
and refined with a Gaussian filter.

This implementation supports both unsupervised and supervised setting, but Anomalib currently supports only unsupervised learning.
Copy link
Contributor

Choose a reason for hiding this comment

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

What are the missing points in Anomalib to support supervised setting

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Right now I believe there are no standard supervised datasets. Another problem is the Folder dataset as it assumes that abnormal samples are always in test set:

samples.loc[(samples.label == DirType.NORMAL), "split"] = Split.TRAIN
samples.loc[(samples.label == DirType.ABNORMAL) | (samples.label == DirType.NORMAL_TEST), "split"] = Split.TEST

Another thing for full reproduction of SuperSimpleNet results is the fixed flipping augmentation and frequency sampling. This is however not necessary, but needed for best results. It's also not SuperSimpleNet specific, so might be worth considering if other supervised model will be supported.

Copy link
Contributor

Choose a reason for hiding this comment

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

yeah, we would like to diversify the model pool, and include more learning types than one-class models. Thanks for the feedback.
@abc-125, you might want to be aware of this discussion as you have recently worked on this stuff

Copy link
Contributor

Choose a reason for hiding this comment

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

Thank you for adding me, it would be great to have supervised models and datasets in Anomalib. Recently, I looked at how to add a supervised dataset, and it certainly would require changing some base structures, such as paths to folders (I guess we will need abnormal_train_dir and maybe renaming the rest to make it easier to understand, normal_train_dir, etc.):

normal_dir (str | Path | Sequence): Path to the directory containing normal images.
root (str | Path | None): Root folder of the dataset.
Defaults to ``None``.
abnormal_dir (str | Path | Sequence | None, optional): Path to the directory containing abnormal images.
Defaults to ``None``.
normal_test_dir (str | Path | Sequence | None, optional): Path to the directory containing
normal images for the test dataset.
Defaults to ``None``.

@samet-akcay
Copy link
Contributor

@blaz-r, I've created #2436 to remove imgaug stuff and write our own perlin noise generation stuff. Would it be an idea to try this in supersimplenet ?

@samet-akcay
Copy link
Contributor

@blaz-r can you pull the latest changes again? The tests should pass after that

@blaz-r
Copy link
Contributor Author

blaz-r commented Dec 2, 2024

Okay, I am currently at the ICPR. I will try to sort this out when I get some time.

@samet-akcay
Copy link
Contributor

Okay, I am currently at the ICPR. I will try to sort this out when I get some time.

ah nice, enjoy the conference!

…ature/supersimplenet

# Conflicts:
#	src/anomalib/models/components/feature_extractors/torchfx.py
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
@blaz-r
Copy link
Contributor Author

blaz-r commented Dec 23, 2024

This is now using the update perlin noise code and includes the latest v2 changes.

Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
@blaz-r
Copy link
Contributor Author

blaz-r commented Jan 3, 2025

Can you run the checks for this please @djdameln @samet-akcay

@samet-akcay samet-akcay self-requested a review January 3, 2025 22:48
Copy link

codecov bot commented Jan 3, 2025

Codecov Report

Attention: Patch coverage is 97.81659% with 5 lines in your changes missing coverage. Please review.

Project coverage is 78.93%. Comparing base (8a82dc7) to head (abc71ac).
Report is 3 commits behind head on release/v2.0.0.

Files with missing lines Patch % Lines
...nomalib/models/image/supersimplenet/torch_model.py 97.32% 3 Missing ⚠️
...lib/models/image/supersimplenet/lightning_model.py 95.91% 2 Missing ⚠️
Additional details and impacted files
@@                Coverage Diff                 @@
##           release/v2.0.0    #2428      +/-   ##
==================================================
+ Coverage           78.53%   78.93%   +0.39%     
==================================================
  Files                 303      311       +8     
  Lines               12934    13197     +263     
==================================================
+ Hits                10158    10417     +259     
- Misses               2776     2780       +4     
Flag Coverage Δ
integration_py3.10 ?

Flags with carried forward coverage won't be shown. Click here to find out more.

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

@samet-akcay
Copy link
Contributor

Can you run the checks for this please @djdameln @samet-akcay

@blaz-r, the tests seem to pass. I'll give another review and we could merge. Thanks for this great contribution!

Copy link
Contributor

@samet-akcay samet-akcay left a comment

Choose a reason for hiding this comment

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

I only have a single comment...

Anomalib's model naming convention is usually snake-case to camel case for model directory vs model class implementation. If this is followed, supersimplenet would be super_simple_net as the class name is SuperSimpleNet. Which one do you think is better in this case?

@blaz-r
Copy link
Contributor Author

blaz-r commented Jan 4, 2025

I think that supersimplenet would be better, since this is a single word. So I will rename the class to Supersimplenet.

Signed-off-by: blaz-r <blaz.rolih@gmail.com>
@blaz-r
Copy link
Contributor Author

blaz-r commented Jan 4, 2025

I have now renamed the class to follow the naming convention.

Copy link
Contributor

@djdameln djdameln left a comment

Choose a reason for hiding this comment

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

Very nice and clean implementation, thanks a lot Blaž!

I only have some minor comments, mainly some naming issues.

Comment on lines 159 to 177
optim = AdamW(
[
{
"params": self.model.adaptor.parameters(),
"lr": 0.0001,
},
{
"params": self.model.segdec.parameters(),
"lr": 0.0002,
"weight_decay": 0.00001,
},
],
)
sched = MultiStepLR(
optim,
milestones=[int(self.trainer.max_epochs * 0.8), int(self.trainer.max_epochs * 0.9)],
gamma=0.4,
)
return [optim], [sched]
Copy link
Contributor

Choose a reason for hiding this comment

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

We prefer to use full variables names, i.e. optimizer and scheduler

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I update this and some other names as well.

Comment on lines 179 to 186
@property
def learning_type(self) -> LearningType:
"""Return the learning type of the model.

Returns:
LearningType: Learning type of the model.
"""
return LearningType.ONE_CLASS
Copy link
Contributor

Choose a reason for hiding this comment

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

I guess this would change when we add support for the supervised mode to the overall Anomalib pipeline. Maybe you could add a comment about this, as it might confuse users. Just a brief mention in the docstring would be sufficient.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I added a comment clarifying this.

Comment on lines 130 to 139
def init_weights(m: nn.Module) -> None:
"""Init weight of the model.

Args:
m (nn.Module): torch module.
"""
if isinstance(m, nn.Linear | nn.Conv2d):
nn.init.xavier_normal_(m.weight)
elif isinstance(m, nn.BatchNorm1d | nn.BatchNorm2d):
nn.init.constant_(m.weight, 1)
Copy link
Contributor

Choose a reason for hiding this comment

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

Again, I would prefer a full variable name here for readability.

Suggested change
def init_weights(m: nn.Module) -> None:
"""Init weight of the model.
Args:
m (nn.Module): torch module.
"""
if isinstance(m, nn.Linear | nn.Conv2d):
nn.init.xavier_normal_(m.weight)
elif isinstance(m, nn.BatchNorm1d | nn.BatchNorm2d):
nn.init.constant_(m.weight, 1)
def init_weights(module: nn.Module) -> None:
"""Init weight of the model.
Args:
module (nn.Module): torch module.
"""
if isinstance(module, nn.Linear | nn.Conv2d):
nn.init.xavier_normal_(module.weight)
elif isinstance(module, nn.BatchNorm1d | nn.BatchNorm2d):
nn.init.constant_(module.weight, 1)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is now changed.

return sum(feature.shape[1] for feature in features.values())


class FeatureAdaptor(nn.Module):
Copy link
Contributor

Choose a reason for hiding this comment

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

I would prefer FeatureAdapter, since we try to use American English throughout the code base. but I guess it would cause a discrepancy with the official implementation in your own repo.

@samet-akcay @ashwinvaidya17 thoughts?

Copy link
Contributor

Choose a reason for hiding this comment

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

as someone based in the UK, not sure what to say here 😁

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I changed this to FeatureAdapter, I think it's clear enough when compared with the official code 😄

from anomalib.data.utils.generators import generate_perlin_noise


class SSNAnomalyGenerator(nn.Module):
Copy link
Contributor

Choose a reason for hiding this comment

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

Not sure about the naming here. Why not simply use AnomalyGenerator? You don't use the SSN prefix for the other submodules.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This is now just AnomalyGenerator.

blaz-r added 2 commits January 6, 2025 22:18
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
Signed-off-by: blaz-r <blaz.rolih@gmail.com>
@blaz-r
Copy link
Contributor Author

blaz-r commented Jan 6, 2025

Thank you for the review. The comments are now addressed. Let me know if anything else needs to be sorted.

@blaz-r
Copy link
Contributor Author

blaz-r commented Jan 7, 2025

Seems like all checks pass but one that is failing due to some dead links. I think this is unrelated to my PR, or did I break something in docs?

@samet-akcay
Copy link
Contributor

Seems like all checks pass but one that is failing due to some dead links. I think this is unrelated to my PR, or did I break something in docs?

no worries, it is related to some new documentation URL, which is not on the readthedocs yet

Copy link
Contributor

@djdameln djdameln left a comment

Choose a reason for hiding this comment

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

Thanks for the awesome work!

@samet-akcay samet-akcay merged commit a304221 into openvinotoolkit:release/v2.0.0 Jan 7, 2025
6 of 7 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants