From ec176434998f68b1aae78be58803bce8c664da37 Mon Sep 17 00:00:00 2001 From: Suwen Ge Date: Thu, 22 Feb 2024 16:01:39 -0800 Subject: [PATCH 1/3] [Issue 7] Update import torchvision.models as models --- .gitignore | 1 + .../codemod/torchvision_models.py | 22 ++++++++++++++++ .../codemod/torchvision_models.py.out | 22 ++++++++++++++++ torchfix/torchfix.py | 7 +++++- .../visitors/deprecated_symbols/__init__.py | 25 +++++++++++++++++++ 5 files changed, 76 insertions(+), 1 deletion(-) create mode 100644 tests/fixtures/deprecated_symbols/codemod/torchvision_models.py create mode 100644 tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out diff --git a/.gitignore b/.gitignore index b775a05..396cbb4 100644 --- a/.gitignore +++ b/.gitignore @@ -2,3 +2,4 @@ .pytest_cache/ __pycache__/ *.egg-info/* +torchfix/visitors/deprecated_symbols/__init__.py diff --git a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py new file mode 100644 index 0000000..d14921f --- /dev/null +++ b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py @@ -0,0 +1,22 @@ +import torch +import torchvision.models as models + +import torch.autograd.profiler as profiler + +for with_cuda in [False, True]: + model = models.test1()#resnet18() + inputs = torch.randn(5, 3, 224, 224) + sort_key = "self_cpu_memory_usage" + if with_cuda and torch.cuda.is_available(): + model = model.cuda() + inputs = inputs.cuda() + sort_key = "self_cuda_memory_usage" + print("Profiling CUDA Resnet model") + else: + print("Profiling CPU Resnet model") + + with profiler.profile(profile_memory=True, record_shapes=True) as prof: + with profiler.record_function("root"): + model(inputs) + + print(prof.key_averages(group_by_input_shape=True).table(sort_by=sort_key, row_limit=-1)) diff --git a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out new file mode 100644 index 0000000..290b4e6 --- /dev/null +++ b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out @@ -0,0 +1,22 @@ +import torch +from torchvision import models + +import torch.autograd.profiler as profiler + +for with_cuda in [False, True]: + model = models.test1()#resnet18() + inputs = torch.randn(5, 3, 224, 224) + sort_key = "self_cpu_memory_usage" + if with_cuda and torch.cuda.is_available(): + model = model.cuda() + inputs = inputs.cuda() + sort_key = "self_cuda_memory_usage" + print("Profiling CUDA Resnet model") + else: + print("Profiling CPU Resnet model") + + with profiler.profile(profile_memory=True, record_shapes=True) as prof: + with profiler.record_function("root"): + model(inputs) + + print(prof.key_averages(group_by_input_shape=True).table(sort_by=sort_key, row_limit=-1)) diff --git a/torchfix/torchfix.py b/torchfix/torchfix.py index a26097a..20eda7a 100644 --- a/torchfix/torchfix.py +++ b/torchfix/torchfix.py @@ -9,6 +9,7 @@ from .visitors.deprecated_symbols import ( TorchDeprecatedSymbolsVisitor, _UpdateFunctorchImports, + _UpdateTorchvisionModelsImports, ) from .visitors.internal import TorchScopedLibraryVisitor @@ -229,7 +230,11 @@ def transform_module_impl(self, module: cst.Module) -> cst.Module: update_functorch_imports_visitor = _UpdateFunctorchImports() new_module = new_module.visit(update_functorch_imports_visitor) - if fixes_count == 0 and not update_functorch_imports_visitor.changed: + update_torchvision_models_visitor = _UpdateTorchvisionModelsImports() + new_module = new_module.visit(update_torchvision_models_visitor) + + if fixes_count == 0 and not update_functorch_imports_visitor.changed \ + and not update_torchvision_models_visitor.changed: raise codemod.SkipFile("No changes") return new_module diff --git a/torchfix/visitors/deprecated_symbols/__init__.py b/torchfix/visitors/deprecated_symbols/__init__.py index 93a9082..532f7f0 100644 --- a/torchfix/visitors/deprecated_symbols/__init__.py +++ b/torchfix/visitors/deprecated_symbols/__init__.py @@ -117,3 +117,28 @@ def leave_ImportFrom( self.changed = True return updated_node.with_changes(module=cst.parse_expression("torch.func")) return updated_node + +# TODO: refactor/generalize this. +class _UpdateTorchvisionModelsImports(cst.CSTTransformer): + + def __init__(self): + self.changed = False + + def leave_Import( + self, node: cst.Import, updated_node: cst.Import + ) -> cst.CSTNode: + if len(updated_node.names) == 1: + alias = updated_node.names[0] + if isinstance(alias.name, cst.Attribute) and \ + alias.name.value.value == 'torchvision' and \ + alias.name.attr.value == 'models' and \ + alias.asname and alias.asname.name.value == 'models': + + self.changed = True + new_import = cst.ImportFrom( + module=cst.Name(value='torchvision'), + names=[cst.ImportAlias(name=cst.Name(value='models'))] + ) + return new_import + + return updated_node From 1d57713dafbb85684370a8cef178be68531025f7 Mon Sep 17 00:00:00 2001 From: Suwen Ge Date: Sun, 3 Mar 2024 10:42:45 -0800 Subject: [PATCH 2/3] Move torchvision.models visitor to vision dir --- .../codemod/torchvision_models.py | 22 ---------- .../codemod/torchvision_models.py.out | 22 ---------- .../fixtures/vision/checker/models_import.py | 5 +++ .../fixtures/vision/checker/models_import.txt | 1 + torchfix/torchfix.py | 8 ++-- .../visitors/deprecated_symbols/__init__.py | 25 ------------ torchfix/visitors/vision/__init__.py | 1 + torchfix/visitors/vision/models_import.py | 40 +++++++++++++++++++ 8 files changed, 50 insertions(+), 74 deletions(-) delete mode 100644 tests/fixtures/deprecated_symbols/codemod/torchvision_models.py delete mode 100644 tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out create mode 100644 tests/fixtures/vision/checker/models_import.py create mode 100644 tests/fixtures/vision/checker/models_import.txt create mode 100644 torchfix/visitors/vision/models_import.py diff --git a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py deleted file mode 100644 index d14921f..0000000 --- a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py +++ /dev/null @@ -1,22 +0,0 @@ -import torch -import torchvision.models as models - -import torch.autograd.profiler as profiler - -for with_cuda in [False, True]: - model = models.test1()#resnet18() - inputs = torch.randn(5, 3, 224, 224) - sort_key = "self_cpu_memory_usage" - if with_cuda and torch.cuda.is_available(): - model = model.cuda() - inputs = inputs.cuda() - sort_key = "self_cuda_memory_usage" - print("Profiling CUDA Resnet model") - else: - print("Profiling CPU Resnet model") - - with profiler.profile(profile_memory=True, record_shapes=True) as prof: - with profiler.record_function("root"): - model(inputs) - - print(prof.key_averages(group_by_input_shape=True).table(sort_by=sort_key, row_limit=-1)) diff --git a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out deleted file mode 100644 index 290b4e6..0000000 --- a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out +++ /dev/null @@ -1,22 +0,0 @@ -import torch -from torchvision import models - -import torch.autograd.profiler as profiler - -for with_cuda in [False, True]: - model = models.test1()#resnet18() - inputs = torch.randn(5, 3, 224, 224) - sort_key = "self_cpu_memory_usage" - if with_cuda and torch.cuda.is_available(): - model = model.cuda() - inputs = inputs.cuda() - sort_key = "self_cuda_memory_usage" - print("Profiling CUDA Resnet model") - else: - print("Profiling CPU Resnet model") - - with profiler.profile(profile_memory=True, record_shapes=True) as prof: - with profiler.record_function("root"): - model(inputs) - - print(prof.key_averages(group_by_input_shape=True).table(sort_by=sort_key, row_limit=-1)) diff --git a/tests/fixtures/vision/checker/models_import.py b/tests/fixtures/vision/checker/models_import.py new file mode 100644 index 0000000..8eae98e --- /dev/null +++ b/tests/fixtures/vision/checker/models_import.py @@ -0,0 +1,5 @@ +import torchvision.models as models +import torchvision.models as cnn +from torchvision.models import resnet50, resnet101 +import torchvision.models +from torchvision.models import * diff --git a/tests/fixtures/vision/checker/models_import.txt b/tests/fixtures/vision/checker/models_import.txt new file mode 100644 index 0000000..29bbc7b --- /dev/null +++ b/tests/fixtures/vision/checker/models_import.txt @@ -0,0 +1 @@ +1:1: TOR203 Consider replacing 'import torchvision.models as models' with 'from torchvision import models'. diff --git a/torchfix/torchfix.py b/torchfix/torchfix.py index 20eda7a..490caf4 100644 --- a/torchfix/torchfix.py +++ b/torchfix/torchfix.py @@ -9,7 +9,6 @@ from .visitors.deprecated_symbols import ( TorchDeprecatedSymbolsVisitor, _UpdateFunctorchImports, - _UpdateTorchvisionModelsImports, ) from .visitors.internal import TorchScopedLibraryVisitor @@ -20,6 +19,7 @@ from .visitors.vision import ( TorchVisionDeprecatedPretrainedVisitor, TorchVisionDeprecatedToTensorVisitor, + TorchVisionModelsImportVisitor, ) from .visitors.security import TorchUnsafeLoadVisitor @@ -36,6 +36,7 @@ TorchSynchronizedDataLoaderVisitor, TorchVisionDeprecatedPretrainedVisitor, TorchVisionDeprecatedToTensorVisitor, + TorchVisionModelsImportVisitor, TorchUnsafeLoadVisitor, TorchReentrantCheckpointVisitor, ] @@ -230,11 +231,8 @@ def transform_module_impl(self, module: cst.Module) -> cst.Module: update_functorch_imports_visitor = _UpdateFunctorchImports() new_module = new_module.visit(update_functorch_imports_visitor) - update_torchvision_models_visitor = _UpdateTorchvisionModelsImports() - new_module = new_module.visit(update_torchvision_models_visitor) - if fixes_count == 0 and not update_functorch_imports_visitor.changed \ - and not update_torchvision_models_visitor.changed: + if fixes_count == 0 and not update_functorch_imports_visitor.changed: raise codemod.SkipFile("No changes") return new_module diff --git a/torchfix/visitors/deprecated_symbols/__init__.py b/torchfix/visitors/deprecated_symbols/__init__.py index 532f7f0..93a9082 100644 --- a/torchfix/visitors/deprecated_symbols/__init__.py +++ b/torchfix/visitors/deprecated_symbols/__init__.py @@ -117,28 +117,3 @@ def leave_ImportFrom( self.changed = True return updated_node.with_changes(module=cst.parse_expression("torch.func")) return updated_node - -# TODO: refactor/generalize this. -class _UpdateTorchvisionModelsImports(cst.CSTTransformer): - - def __init__(self): - self.changed = False - - def leave_Import( - self, node: cst.Import, updated_node: cst.Import - ) -> cst.CSTNode: - if len(updated_node.names) == 1: - alias = updated_node.names[0] - if isinstance(alias.name, cst.Attribute) and \ - alias.name.value.value == 'torchvision' and \ - alias.name.attr.value == 'models' and \ - alias.asname and alias.asname.name.value == 'models': - - self.changed = True - new_import = cst.ImportFrom( - module=cst.Name(value='torchvision'), - names=[cst.ImportAlias(name=cst.Name(value='models'))] - ) - return new_import - - return updated_node diff --git a/torchfix/visitors/vision/__init__.py b/torchfix/visitors/vision/__init__.py index 7adcc19..9bc944e 100644 --- a/torchfix/visitors/vision/__init__.py +++ b/torchfix/visitors/vision/__init__.py @@ -1,2 +1,3 @@ from .pretrained import TorchVisionDeprecatedPretrainedVisitor # noqa: F401 from .to_tensor import TorchVisionDeprecatedToTensorVisitor # noqa: F401 +from .models_import import TorchVisionModelsImportVisitor # noqa: F401 diff --git a/torchfix/visitors/vision/models_import.py b/torchfix/visitors/vision/models_import.py new file mode 100644 index 0000000..2f810a3 --- /dev/null +++ b/torchfix/visitors/vision/models_import.py @@ -0,0 +1,40 @@ +import libcst as cst + +from ...common import LintViolation, TorchVisitor + + +class TorchVisionModelsImportVisitor(TorchVisitor): + ERROR_CODE = "TOR203" + + def visit_Import(self, node: cst.Import) -> None: + for imported_item in node.names: + if isinstance(imported_item.name, cst.Attribute): + if ( + isinstance(imported_item.name.value, cst.Name) + and imported_item.name.value.value == "torchvision" + and imported_item.name.attr.value == "models" + and imported_item.asname is not None + and imported_item.asname.name.value == "models" + ): + print(imported_item.asname.name.value) + position = self.get_metadata( + cst.metadata.WhitespaceInclusivePositionProvider, node + ) + # print(position) + replacement = cst.ImportFrom( + module=cst.Name("torchvision"), + names=[cst.ImportAlias(name=cst.Name("models"))], + ) + self.violations.append( + LintViolation( + error_code=self.ERROR_CODE, + message=( + "Consider replacing 'import torchvision.models as" + " models' with 'from torchvision import models'. " + ), + line=position.start.line, + column=position.start.column, + node=node, + replacement=replacement + ) + ) From 05aaad23576a000dd9f697c3221f28458414c696 Mon Sep 17 00:00:00 2001 From: Suwen Ge Date: Sun, 3 Mar 2024 10:53:24 -0800 Subject: [PATCH 3/3] Move torchvision.models visitor to vision dir --- .../codemod/torchvision_models.py | 22 ---------- .../codemod/torchvision_models.py.out | 22 ---------- .../fixtures/vision/checker/models_import.py | 5 +++ .../fixtures/vision/checker/models_import.txt | 1 + torchfix/torchfix.py | 8 ++-- .../visitors/deprecated_symbols/__init__.py | 25 ------------ torchfix/visitors/vision/__init__.py | 1 + torchfix/visitors/vision/models_import.py | 40 +++++++++++++++++++ 8 files changed, 50 insertions(+), 74 deletions(-) delete mode 100644 tests/fixtures/deprecated_symbols/codemod/torchvision_models.py delete mode 100644 tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out create mode 100644 tests/fixtures/vision/checker/models_import.py create mode 100644 tests/fixtures/vision/checker/models_import.txt create mode 100644 torchfix/visitors/vision/models_import.py diff --git a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py deleted file mode 100644 index d14921f..0000000 --- a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py +++ /dev/null @@ -1,22 +0,0 @@ -import torch -import torchvision.models as models - -import torch.autograd.profiler as profiler - -for with_cuda in [False, True]: - model = models.test1()#resnet18() - inputs = torch.randn(5, 3, 224, 224) - sort_key = "self_cpu_memory_usage" - if with_cuda and torch.cuda.is_available(): - model = model.cuda() - inputs = inputs.cuda() - sort_key = "self_cuda_memory_usage" - print("Profiling CUDA Resnet model") - else: - print("Profiling CPU Resnet model") - - with profiler.profile(profile_memory=True, record_shapes=True) as prof: - with profiler.record_function("root"): - model(inputs) - - print(prof.key_averages(group_by_input_shape=True).table(sort_by=sort_key, row_limit=-1)) diff --git a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out b/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out deleted file mode 100644 index 290b4e6..0000000 --- a/tests/fixtures/deprecated_symbols/codemod/torchvision_models.py.out +++ /dev/null @@ -1,22 +0,0 @@ -import torch -from torchvision import models - -import torch.autograd.profiler as profiler - -for with_cuda in [False, True]: - model = models.test1()#resnet18() - inputs = torch.randn(5, 3, 224, 224) - sort_key = "self_cpu_memory_usage" - if with_cuda and torch.cuda.is_available(): - model = model.cuda() - inputs = inputs.cuda() - sort_key = "self_cuda_memory_usage" - print("Profiling CUDA Resnet model") - else: - print("Profiling CPU Resnet model") - - with profiler.profile(profile_memory=True, record_shapes=True) as prof: - with profiler.record_function("root"): - model(inputs) - - print(prof.key_averages(group_by_input_shape=True).table(sort_by=sort_key, row_limit=-1)) diff --git a/tests/fixtures/vision/checker/models_import.py b/tests/fixtures/vision/checker/models_import.py new file mode 100644 index 0000000..8eae98e --- /dev/null +++ b/tests/fixtures/vision/checker/models_import.py @@ -0,0 +1,5 @@ +import torchvision.models as models +import torchvision.models as cnn +from torchvision.models import resnet50, resnet101 +import torchvision.models +from torchvision.models import * diff --git a/tests/fixtures/vision/checker/models_import.txt b/tests/fixtures/vision/checker/models_import.txt new file mode 100644 index 0000000..864cf35 --- /dev/null +++ b/tests/fixtures/vision/checker/models_import.txt @@ -0,0 +1 @@ +1:1 TOR203 Consider replacing 'import torchvision.models as models' with 'from torchvision import models'. diff --git a/torchfix/torchfix.py b/torchfix/torchfix.py index 20eda7a..490caf4 100644 --- a/torchfix/torchfix.py +++ b/torchfix/torchfix.py @@ -9,7 +9,6 @@ from .visitors.deprecated_symbols import ( TorchDeprecatedSymbolsVisitor, _UpdateFunctorchImports, - _UpdateTorchvisionModelsImports, ) from .visitors.internal import TorchScopedLibraryVisitor @@ -20,6 +19,7 @@ from .visitors.vision import ( TorchVisionDeprecatedPretrainedVisitor, TorchVisionDeprecatedToTensorVisitor, + TorchVisionModelsImportVisitor, ) from .visitors.security import TorchUnsafeLoadVisitor @@ -36,6 +36,7 @@ TorchSynchronizedDataLoaderVisitor, TorchVisionDeprecatedPretrainedVisitor, TorchVisionDeprecatedToTensorVisitor, + TorchVisionModelsImportVisitor, TorchUnsafeLoadVisitor, TorchReentrantCheckpointVisitor, ] @@ -230,11 +231,8 @@ def transform_module_impl(self, module: cst.Module) -> cst.Module: update_functorch_imports_visitor = _UpdateFunctorchImports() new_module = new_module.visit(update_functorch_imports_visitor) - update_torchvision_models_visitor = _UpdateTorchvisionModelsImports() - new_module = new_module.visit(update_torchvision_models_visitor) - if fixes_count == 0 and not update_functorch_imports_visitor.changed \ - and not update_torchvision_models_visitor.changed: + if fixes_count == 0 and not update_functorch_imports_visitor.changed: raise codemod.SkipFile("No changes") return new_module diff --git a/torchfix/visitors/deprecated_symbols/__init__.py b/torchfix/visitors/deprecated_symbols/__init__.py index 532f7f0..93a9082 100644 --- a/torchfix/visitors/deprecated_symbols/__init__.py +++ b/torchfix/visitors/deprecated_symbols/__init__.py @@ -117,28 +117,3 @@ def leave_ImportFrom( self.changed = True return updated_node.with_changes(module=cst.parse_expression("torch.func")) return updated_node - -# TODO: refactor/generalize this. -class _UpdateTorchvisionModelsImports(cst.CSTTransformer): - - def __init__(self): - self.changed = False - - def leave_Import( - self, node: cst.Import, updated_node: cst.Import - ) -> cst.CSTNode: - if len(updated_node.names) == 1: - alias = updated_node.names[0] - if isinstance(alias.name, cst.Attribute) and \ - alias.name.value.value == 'torchvision' and \ - alias.name.attr.value == 'models' and \ - alias.asname and alias.asname.name.value == 'models': - - self.changed = True - new_import = cst.ImportFrom( - module=cst.Name(value='torchvision'), - names=[cst.ImportAlias(name=cst.Name(value='models'))] - ) - return new_import - - return updated_node diff --git a/torchfix/visitors/vision/__init__.py b/torchfix/visitors/vision/__init__.py index 7adcc19..9bc944e 100644 --- a/torchfix/visitors/vision/__init__.py +++ b/torchfix/visitors/vision/__init__.py @@ -1,2 +1,3 @@ from .pretrained import TorchVisionDeprecatedPretrainedVisitor # noqa: F401 from .to_tensor import TorchVisionDeprecatedToTensorVisitor # noqa: F401 +from .models_import import TorchVisionModelsImportVisitor # noqa: F401 diff --git a/torchfix/visitors/vision/models_import.py b/torchfix/visitors/vision/models_import.py new file mode 100644 index 0000000..9274120 --- /dev/null +++ b/torchfix/visitors/vision/models_import.py @@ -0,0 +1,40 @@ +import libcst as cst + +from ...common import LintViolation, TorchVisitor + + +class TorchVisionModelsImportVisitor(TorchVisitor): + ERROR_CODE = "TOR203" + + def visit_Import(self, node: cst.Import) -> None: + for imported_item in node.names: + if isinstance(imported_item.name, cst.Attribute): + if ( + isinstance(imported_item.name.value, cst.Name) + and imported_item.name.value.value == "torchvision" + and imported_item.name.attr.value == "models" + and imported_item.asname is not None + and imported_item.asname.name.value == "models" + ): + print(imported_item.asname.name.value) + position = self.get_metadata( + cst.metadata.WhitespaceInclusivePositionProvider, node + ) + # print(position) + replacement = cst.ImportFrom( + module=cst.Name("torchvision"), + names=[cst.ImportAlias(name=cst.Name("models"))], + ) + self.violations.append( + LintViolation( + error_code=self.ERROR_CODE, + message=( + "Consider replacing 'import torchvision.models as" + " models' with 'from torchvision import models'." + ), + line=position.start.line, + column=position.start.column, + node=node, + replacement=replacement + ) + )