PageRenderTime 1787ms CodeModel.GetById 13ms RepoModel.GetById 1ms app.codeStats 0ms

/pymode/libs3/rope/base/oi/transform.py

https://gitlab.com/vim-IDE/python-mode
Python | 285 lines | 190 code | 44 blank | 51 comment | 24 complexity | b5dd139dc8d97195d868eee6f3226874 MD5 | raw file
  1. """Provides classes for persisting `PyObject`\s"""
  2. import os
  3. import re
  4. import rope.base.builtins
  5. from rope.base import exceptions
  6. class PyObjectToTextual(object):
  7. """For transforming `PyObject` to textual form
  8. This can be used for storing `PyObjects` in files. Use
  9. `TextualToPyObject` for converting back.
  10. """
  11. def __init__(self, project):
  12. self.project = project
  13. def transform(self, pyobject):
  14. """Transform a `PyObject` to textual form"""
  15. if pyobject is None:
  16. return ('none',)
  17. object_type = type(pyobject)
  18. try:
  19. method = getattr(self, object_type.__name__ + '_to_textual')
  20. return method(pyobject)
  21. except AttributeError:
  22. return ('unknown',)
  23. def __call__(self, pyobject):
  24. return self.transform(pyobject)
  25. def PyObject_to_textual(self, pyobject):
  26. if isinstance(pyobject.get_type(), rope.base.pyobjects.AbstractClass):
  27. result = self.transform(pyobject.get_type())
  28. if result[0] == 'defined':
  29. return ('instance', result)
  30. return result
  31. return ('unknown',)
  32. def PyFunction_to_textual(self, pyobject):
  33. return self._defined_to_textual(pyobject)
  34. def PyClass_to_textual(self, pyobject):
  35. return self._defined_to_textual(pyobject)
  36. def _defined_to_textual(self, pyobject):
  37. address = []
  38. while pyobject.parent is not None:
  39. address.insert(0, pyobject.get_name())
  40. pyobject = pyobject.parent
  41. return ('defined', self._get_pymodule_path(pyobject.get_module()),
  42. '.'.join(address))
  43. def PyModule_to_textual(self, pyobject):
  44. return ('defined', self._get_pymodule_path(pyobject))
  45. def PyPackage_to_textual(self, pyobject):
  46. return ('defined', self._get_pymodule_path(pyobject))
  47. def List_to_textual(self, pyobject):
  48. return ('builtin', 'list', self.transform(pyobject.holding))
  49. def Dict_to_textual(self, pyobject):
  50. return ('builtin', 'dict', self.transform(pyobject.keys),
  51. self.transform(pyobject.values))
  52. def Tuple_to_textual(self, pyobject):
  53. objects = [self.transform(holding)
  54. for holding in pyobject.get_holding_objects()]
  55. return tuple(['builtin', 'tuple'] + objects)
  56. def Set_to_textual(self, pyobject):
  57. return ('builtin', 'set', self.transform(pyobject.holding))
  58. def Iterator_to_textual(self, pyobject):
  59. return ('builtin', 'iter', self.transform(pyobject.holding))
  60. def Generator_to_textual(self, pyobject):
  61. return ('builtin', 'generator', self.transform(pyobject.holding))
  62. def Str_to_textual(self, pyobject):
  63. return ('builtin', 'str')
  64. def File_to_textual(self, pyobject):
  65. return ('builtin', 'file')
  66. def BuiltinFunction_to_textual(self, pyobject):
  67. return ('builtin', 'function', pyobject.get_name())
  68. def _get_pymodule_path(self, pymodule):
  69. return self.resource_to_path(pymodule.get_resource())
  70. def resource_to_path(self, resource):
  71. if resource.project == self.project:
  72. return resource.path
  73. else:
  74. return resource.real_path
  75. class TextualToPyObject(object):
  76. """For transforming textual form to `PyObject`"""
  77. def __init__(self, project, allow_in_project_absolutes=False):
  78. self.project = project
  79. def __call__(self, textual):
  80. return self.transform(textual)
  81. def transform(self, textual):
  82. """Transform an object from textual form to `PyObject`"""
  83. if textual is None:
  84. return None
  85. type = textual[0]
  86. try:
  87. method = getattr(self, type + '_to_pyobject')
  88. return method(textual)
  89. except AttributeError:
  90. return None
  91. def builtin_to_pyobject(self, textual):
  92. name = textual[1]
  93. method = getattr(self, 'builtin_%s_to_pyobject' % textual[1], None)
  94. if method is not None:
  95. return method(textual)
  96. def builtin_str_to_pyobject(self, textual):
  97. return rope.base.builtins.get_str()
  98. def builtin_list_to_pyobject(self, textual):
  99. holding = self.transform(textual[2])
  100. return rope.base.builtins.get_list(holding)
  101. def builtin_dict_to_pyobject(self, textual):
  102. keys = self.transform(textual[2])
  103. values = self.transform(textual[3])
  104. return rope.base.builtins.get_dict(keys, values)
  105. def builtin_tuple_to_pyobject(self, textual):
  106. objects = []
  107. for holding in textual[2:]:
  108. objects.append(self.transform(holding))
  109. return rope.base.builtins.get_tuple(*objects)
  110. def builtin_set_to_pyobject(self, textual):
  111. holding = self.transform(textual[2])
  112. return rope.base.builtins.get_set(holding)
  113. def builtin_iter_to_pyobject(self, textual):
  114. holding = self.transform(textual[2])
  115. return rope.base.builtins.get_iterator(holding)
  116. def builtin_generator_to_pyobject(self, textual):
  117. holding = self.transform(textual[2])
  118. return rope.base.builtins.get_generator(holding)
  119. def builtin_file_to_pyobject(self, textual):
  120. return rope.base.builtins.get_file()
  121. def builtin_function_to_pyobject(self, textual):
  122. if textual[2] in rope.base.builtins.builtins:
  123. return rope.base.builtins.builtins[textual[2]].get_object()
  124. def unknown_to_pyobject(self, textual):
  125. return None
  126. def none_to_pyobject(self, textual):
  127. return None
  128. def _module_to_pyobject(self, textual):
  129. path = textual[1]
  130. return self._get_pymodule(path)
  131. def _hierarchical_defined_to_pyobject(self, textual):
  132. path = textual[1]
  133. names = textual[2].split('.')
  134. pymodule = self._get_pymodule(path)
  135. pyobject = pymodule
  136. for name in names:
  137. if pyobject is None:
  138. return None
  139. if isinstance(pyobject, rope.base.pyobjects.PyDefinedObject):
  140. try:
  141. pyobject = pyobject.get_scope()[name].get_object()
  142. except exceptions.NameNotFoundError:
  143. return None
  144. else:
  145. return None
  146. return pyobject
  147. def defined_to_pyobject(self, textual):
  148. if len(textual) == 2 or textual[2] == '':
  149. return self._module_to_pyobject(textual)
  150. else:
  151. return self._hierarchical_defined_to_pyobject(textual)
  152. def instance_to_pyobject(self, textual):
  153. type = self.transform(textual[1])
  154. if type is not None:
  155. return rope.base.pyobjects.PyObject(type)
  156. def _get_pymodule(self, path):
  157. resource = self.path_to_resource(path)
  158. if resource is not None:
  159. return self.project.pycore.resource_to_pyobject(resource)
  160. def path_to_resource(self, path):
  161. try:
  162. root = self.project.address
  163. if not os.path.isabs(path):
  164. return self.project.get_resource(path)
  165. if path == root or path.startswith(root + os.sep):
  166. # INFO: This is a project file; should not be absolute
  167. return None
  168. import rope.base.project
  169. return rope.base.project.get_no_project().get_resource(path)
  170. except exceptions.ResourceNotFoundError:
  171. return None
  172. class DOITextualToPyObject(TextualToPyObject):
  173. """For transforming textual form to `PyObject`
  174. The textual form DOI uses is different from rope's standard
  175. textual form. The reason is that we cannot find the needed
  176. information by analyzing live objects. This class can be
  177. used to transform DOI textual form to `PyObject` and later
  178. we can convert it to standard textual form using
  179. `TextualToPyObject` class.
  180. """
  181. def _function_to_pyobject(self, textual):
  182. path = textual[1]
  183. lineno = int(textual[2])
  184. pymodule = self._get_pymodule(path)
  185. if pymodule is not None:
  186. scope = pymodule.get_scope()
  187. inner_scope = scope.get_inner_scope_for_line(lineno)
  188. return inner_scope.pyobject
  189. def _class_to_pyobject(self, textual):
  190. path, name = textual[1:]
  191. pymodule = self._get_pymodule(path)
  192. if pymodule is None:
  193. return None
  194. module_scope = pymodule.get_scope()
  195. suspected = None
  196. if name in module_scope.get_names():
  197. suspected = module_scope[name].get_object()
  198. if suspected is not None and \
  199. isinstance(suspected, rope.base.pyobjects.PyClass):
  200. return suspected
  201. else:
  202. lineno = self._find_occurrence(name, pymodule.get_resource().read())
  203. if lineno is not None:
  204. inner_scope = module_scope.get_inner_scope_for_line(lineno)
  205. return inner_scope.pyobject
  206. def defined_to_pyobject(self, textual):
  207. if len(textual) == 2:
  208. return self._module_to_pyobject(textual)
  209. else:
  210. if textual[2].isdigit():
  211. result = self._function_to_pyobject(textual)
  212. else:
  213. result = self._class_to_pyobject(textual)
  214. if not isinstance(result, rope.base.pyobjects.PyModule):
  215. return result
  216. def _find_occurrence(self, name, source):
  217. pattern = re.compile(r'^\s*class\s*' + name + r'\b')
  218. lines = source.split('\n')
  219. for i in range(len(lines)):
  220. if pattern.match(lines[i]):
  221. return i + 1
  222. def path_to_resource(self, path):
  223. import rope.base.libutils
  224. root = self.project.address
  225. relpath = rope.base.libutils.relative(root, path)
  226. if relpath is not None:
  227. path = relpath
  228. return super(DOITextualToPyObject, self).path_to_resource(path)