Source

rope_py3k / rope / refactor / localtofield.py

import rope.base.codeanalyze
from rope.refactor.rename import RenameRefactoring


class ConvertLocalToFieldRefactoring(object):

    def __init__(self, project, resource, offset):
        self.project = project
        self.pycore = project.pycore
        self.resource = resource
        self.offset = offset

    def get_changes(self):
        name = rope.base.codeanalyze.get_name_at(self.resource, self.offset)
        pyname = rope.base.codeanalyze.get_pyname_at(self.pycore, self.resource, self.offset)
        if not self._is_a_method_local(pyname):
            raise rope.base.exceptions.RefactoringError(
                'Convert local variable to field should be performed on \n'
                'the a local variable of a method.')

        pymodule, lineno = pyname.get_definition_location()
        function_scope = pymodule.get_scope().get_inner_scope_for_line(lineno)
        class_scope = function_scope.parent
        if name in class_scope.pyobject.get_attributes():
            raise rope.base.exceptions.RefactoringError(
                'The field %s already exists' % name)

        new_name = self._get_field_name(function_scope.pyobject, name)
        changes = RenameRefactoring(self.project, self.resource, self.offset).\
                  get_changes(new_name, in_file=True)
        return changes

    def _get_field_name(self, pyfunction, name):
        self_name = pyfunction.parameters[0]
        new_name = self_name + '.' + name
        return new_name

    def _is_a_method_local(self, pyname):
        pymodule, lineno = pyname.get_definition_location()
        holding_scope = pymodule.get_scope().get_inner_scope_for_line(lineno)
        parent = holding_scope.parent
        return isinstance(pyname, rope.base.pynames.AssignedName) and \
               pyname in holding_scope.get_names().values() and \
               holding_scope.get_kind() == 'Function' and \
               parent is not None and parent.get_kind() == 'Class'