f31408fa34c73b111e985730cf9453bce32a9448
[gnome.gobject-introspection] / giscanner / transformer.py
1 # -*- Mode: Python -*-
2 # GObject-Introspection - a framework for introspecting GObject libraries
3 # Copyright (C) 2008  Johan Dahlin
4 #
5 # This library is free software; you can redistribute it and/or
6 # modify it under the terms of the GNU Lesser General Public
7 # License as published by the Free Software Foundation; either
8 # version 2 of the License, or (at your option) any later version.
9 #
10 # This library is distributed in the hope that it will be useful,
11 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13 # Lesser General Public License for more details.
14 #
15 # You should have received a copy of the GNU Lesser General Public
16 # License along with this library; if not, write to the
17 # Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 # Boston, MA 02111-1307, USA.
19 #
20
21 import os
22
23 from .ast import (Bitfield, Callback, Enum, Function, Namespace, Member,
24                   Parameter, Return, Struct, Field,
25                   Type, Array, Alias, Interface, Class, Node, Union,
26                   Varargs, Constant, type_name_from_ctype,
27                   type_names, TYPE_STRING, BASIC_GIR_TYPES)
28 from .config import DATADIR, GIR_DIR, GIR_SUFFIX
29 from .glibast import GLibBoxed
30 from .girparser import GIRParser
31 from .odict import odict
32 from .sourcescanner import (
33     SourceSymbol, ctype_name, CTYPE_POINTER,
34     CTYPE_BASIC_TYPE, CTYPE_UNION, CTYPE_ARRAY, CTYPE_TYPEDEF,
35     CTYPE_VOID, CTYPE_ENUM, CTYPE_FUNCTION, CTYPE_STRUCT,
36     CSYMBOL_TYPE_FUNCTION, CSYMBOL_TYPE_TYPEDEF, CSYMBOL_TYPE_STRUCT,
37     CSYMBOL_TYPE_ENUM, CSYMBOL_TYPE_UNION, CSYMBOL_TYPE_OBJECT,
38     CSYMBOL_TYPE_MEMBER, CSYMBOL_TYPE_ELLIPSIS, CSYMBOL_TYPE_CONST,
39     TYPE_QUALIFIER_CONST)
40 from .utils import to_underscores
41
42 _xdg_data_dirs = [x for x in os.environ.get('XDG_DATA_DIRS', '').split(':') \
43                       + [DATADIR, '/usr/share'] if x]
44
45
46 class SkipError(Exception):
47     pass
48
49
50 class Names(object):
51     names = property(lambda self: self._names)
52     aliases = property(lambda self: self._aliases)
53     type_names = property(lambda self: self._type_names)
54     ctypes = property(lambda self: self._ctypes)
55
56     def __init__(self):
57         super(Names, self).__init__()
58         self._names = odict() # Maps from GIName -> (namespace, node)
59         self._aliases = {} # Maps from GIName -> GIName
60         self._type_names = {} # Maps from GTName -> (namespace, node)
61         self._ctypes = {} # Maps from CType -> (namespace, node)
62
63
64 class Transformer(object):
65
66     def __init__(self, cachestore, namespace_name, namespace_version):
67         self._cachestore = cachestore
68         self.generator = None
69         self._namespace = Namespace(namespace_name, namespace_version)
70         self._names = Names()
71         self._pkg_config_packages = set()
72         self._typedefs_ns = {}
73         self._strip_prefix = ''
74         self._includes = set()
75         self._includepaths = []
76
77     def get_names(self):
78         return self._names
79
80     def get_includes(self):
81         return self._includes
82
83     def set_strip_prefix(self, strip_prefix):
84         self._strip_prefix = strip_prefix
85
86     def get_strip_prefix(self):
87         return self._strip_prefix
88
89     def get_pkgconfig_packages(self):
90         return self._pkg_config_packages
91
92     def set_source_ast(self, src_ast):
93         self.generator = src_ast
94
95     def parse(self):
96         nodes = []
97         for symbol in self.generator.get_symbols():
98             try:
99                 node = self._traverse_one(symbol)
100             except SkipError:
101                 continue
102             self._add_node(node)
103         return self._namespace
104
105     def set_include_paths(self, paths):
106         self._includepaths = list(paths)
107
108     def register_include(self, include):
109         if include in self._includes:
110             return
111         filename = self._find_include(include)
112         self._parse_include(filename)
113         self._includes.add(include)
114
115     # Private
116
117     def _find_include(self, include):
118         searchdirs = self._includepaths[:]
119         for path in _xdg_data_dirs:
120             searchdirs.append(os.path.join(path, GIR_SUFFIX))
121         searchdirs.append(GIR_DIR)
122
123         girname = '%s-%s.gir' % (include.name, include.version)
124         for d in searchdirs:
125             path = os.path.join(d, girname)
126             if os.path.exists(path):
127                 return path
128         raise ValueError("Couldn't find include %r (search path: %r)"\
129                          % (girname, searchdirs))
130
131     def _parse_include(self, filename):
132         parser = self._cachestore.load(filename)
133         if parser is None:
134             parser = GIRParser()
135             parser.set_include_parsing(True)
136             parser.parse(filename)
137             self._cachestore.store(filename, parser)
138
139         for include in parser.get_includes():
140             self.register_include(include)
141
142         for pkg in parser.get_pkgconfig_packages():
143             self._pkg_config_packages.add(pkg)
144         namespace = parser.get_namespace()
145         nsname = namespace.name
146         for node in namespace.nodes:
147             if isinstance(node, Alias):
148                 self._names.aliases[node.name] = (nsname, node)
149             elif isinstance(node, (GLibBoxed, Interface, Class)):
150                 self._names.type_names[node.type_name] = (nsname, node)
151             giname = '%s.%s' % (nsname, node.name)
152             self._names.names[giname] = (nsname, node)
153             if hasattr(node, 'ctype'):
154                 self._names.ctypes[node.ctype] = (nsname, node)
155             elif hasattr(node, 'symbol'):
156                 self._names.ctypes[node.symbol] = (nsname, node)
157
158     def _add_node(self, node):
159         if node is None:
160             return
161         if node.name.startswith('_'):
162             return
163         self._namespace.nodes.append(node)
164         self._names.names[node.name] = (None, node)
165
166     def _strip_namespace_func(self, name):
167         prefix = self._namespace.name.lower() + '_'
168         if name.lower().startswith(prefix):
169             name = name[len(prefix):]
170         else:
171             prefix = to_underscores(self._namespace.name).lower() + '_'
172             if name.lower().startswith(prefix):
173                 name = name[len(prefix):]
174         return self.remove_prefix(name, isfunction=True)
175
176     def remove_prefix(self, name, isfunction=False):
177         # when --strip-prefix=g:
178         #   GHashTable -> HashTable
179         #   g_hash_table_new -> hash_table_new
180         prefix = self._strip_prefix.lower()
181         if isfunction:
182             prefix += '_'
183         if len(name) > len(prefix) and name.lower().startswith(prefix):
184             name = name[len(prefix):]
185
186         while name.startswith('_'):
187             name = name[1:]
188         return name
189
190     def _traverse_one(self, symbol, stype=None):
191         assert isinstance(symbol, SourceSymbol), symbol
192
193         if stype is None:
194             stype = symbol.type
195         if stype == CSYMBOL_TYPE_FUNCTION:
196             return self._create_function(symbol)
197         elif stype == CSYMBOL_TYPE_TYPEDEF:
198             return self._create_typedef(symbol)
199         elif stype == CSYMBOL_TYPE_STRUCT:
200             return self._create_struct(symbol)
201         elif stype == CSYMBOL_TYPE_ENUM:
202             return self._create_enum(symbol)
203         elif stype == CSYMBOL_TYPE_OBJECT:
204             return self._create_object(symbol)
205         elif stype == CSYMBOL_TYPE_MEMBER:
206             return self._create_member(symbol)
207         elif stype == CSYMBOL_TYPE_UNION:
208             return self._create_union(symbol)
209         elif stype == CSYMBOL_TYPE_CONST:
210             return self._create_const(symbol)
211         else:
212             raise NotImplementedError(
213                 'Transformer: unhandled symbol: %r' % (symbol, ))
214
215     def _enum_common_prefix(self, symbol):
216         def common_prefix(a, b):
217             commonparts = []
218             for aword, bword in zip(a.split('_'), b.split('_')):
219                 if aword != bword:
220                     return '_'.join(commonparts) + '_'
221                 commonparts.append(aword)
222             return min(a, b)
223
224         # Nothing less than 2 has a common prefix
225         if len(list(symbol.base_type.child_list)) < 2:
226             return None
227         prefix = None
228         for child in symbol.base_type.child_list:
229             if prefix is None:
230                 prefix = child.ident
231             else:
232                 prefix = common_prefix(prefix, child.ident)
233                 if prefix == '':
234                     return None
235         return prefix
236
237     def _create_enum(self, symbol):
238         prefix = self._enum_common_prefix(symbol)
239         if prefix:
240             prefixlen = len(prefix)
241         else:
242             prefixlen = 0
243         members = []
244         for child in symbol.base_type.child_list:
245             if prefixlen > 0:
246                 name = child.ident[prefixlen:]
247             else:
248                 # Ok, the enum members don't have a consistent prefix
249                 # among them, so let's just remove the global namespace
250                 # prefix.
251                 name = self.remove_prefix(child.ident)
252             members.append(Member(name.lower(),
253                                   child.const_int,
254                                   child.ident))
255
256         enum_name = self.remove_prefix(symbol.ident)
257         if symbol.base_type.is_bitfield:
258             klass = Bitfield
259         else:
260             klass = Enum
261         node = klass(enum_name, symbol.ident, members)
262         self._names.type_names[symbol.ident] = (None, node)
263         return node
264
265     def _create_object(self, symbol):
266         return Member(symbol.ident, symbol.base_type.name,
267                       symbol.ident)
268
269     def _type_is_callback(self, type):
270         if isinstance(type, Callback):
271             return True
272         node = self._names.names.get(type.name)
273         if node and isinstance(node[1], Callback):
274             return True
275         return False
276
277     def _handle_closure(self, param, closure_idx, closure_param):
278         if (closure_param.type.name == 'any' and
279             closure_param.name.endswith('data')):
280             param.closure_name = closure_param.name
281             param.closure_index = closure_idx
282             return True
283         return False
284
285     def _handle_destroy(self, param, destroy_idx, destroy_param):
286         if (destroy_param.type.name == 'GLib.DestroyNotify' or
287             destroy_param.type.ctype == 'GDestroyNotify'):
288             param.destroy_name = destroy_param.name
289             param.destroy_index = destroy_idx
290             return True
291         return False
292
293     def _augment_callback_params(self, params):
294         for i, param in enumerate(params):
295             if not self._type_is_callback(param.type):
296                 continue
297
298             # set a default scope
299             if param.scope is None:
300                 param.scope = 'call'
301
302             # j is the index where we look for closure/destroy to
303             # group with the callback param
304             j = i + 1
305             if j == len(params):
306                 continue # no more args -> nothing to group
307             # look at the param directly following for either a
308             # closure or a destroy; only one of these will fire
309             had_closure = self._handle_closure(param, j, params[j])
310             had_destroy = self._handle_destroy(param, j, params[j])
311             j += 1
312             # are we out of params, or did we find neither?
313             if j == len(params) or (not had_closure and not had_destroy):
314                 continue
315             # we found either a closure or a destroy; check the
316             # parameter following for the other
317             if not had_closure:
318                 self._handle_closure(param, j, params[j])
319             if not had_destroy:
320                 self._handle_destroy(param, j, params[j])
321
322     def _create_function(self, symbol):
323         parameters = list(self._create_parameters(symbol.base_type))
324         return_ = self._create_return(symbol.base_type.base_type)
325         self._augment_callback_params(parameters)
326         name = self._strip_namespace_func(symbol.ident)
327         func = Function(name, return_, parameters, symbol.ident)
328         return func
329
330     def _create_source_type(self, source_type):
331         if source_type is None:
332             return 'None'
333         if source_type.type == CTYPE_VOID:
334             value = 'void'
335         elif source_type.type == CTYPE_BASIC_TYPE:
336             value = source_type.name
337         elif source_type.type == CTYPE_TYPEDEF:
338             value = source_type.name
339         elif source_type.type == CTYPE_ARRAY:
340             return self._create_source_type(source_type.base_type)
341         elif source_type.type == CTYPE_POINTER:
342             value = self._create_source_type(source_type.base_type) + '*'
343         else:
344             value = 'any'
345         return value
346
347     def _create_parameters(self, base_type):
348
349         # warn if we see annotations for unknown parameters
350         param_names = set(child.ident for child in base_type.child_list)
351         for child in base_type.child_list:
352             yield self._create_parameter(child)
353
354     def _create_member(self, symbol):
355         source_type = symbol.base_type
356         if (source_type.type == CTYPE_POINTER and
357             symbol.base_type.base_type.type == CTYPE_FUNCTION):
358             node = self._create_callback(symbol)
359         elif source_type.type == CTYPE_STRUCT and source_type.name is None:
360             node = self._create_struct(symbol, anonymous=True)
361         elif source_type.type == CTYPE_UNION and source_type.name is None:
362             node = self._create_union(symbol, anonymous=True)
363         else:
364             # Special handling for fields; we don't have annotations on them
365             # to apply later, yet.
366             if source_type.type == CTYPE_ARRAY:
367                 ctype = self._create_source_type(source_type)
368                 canonical_ctype = self._canonicalize_ctype(ctype)
369                 if canonical_ctype[-1] == '*':
370                     derefed_name = canonical_ctype[:-1]
371                 else:
372                     derefed_name = canonical_ctype
373                 derefed_name = self.resolve_param_type(derefed_name)
374                 ftype = Array(ctype, self.parse_ctype(derefed_name))
375                 child_list = list(symbol.base_type.child_list)
376                 ftype.zeroterminated = False
377                 if child_list:
378                     ftype.size = '%d' % (child_list[0].const_int, )
379             else:
380                 ftype = self._create_type(symbol.base_type,
381                                           is_param=False, is_retval=False)
382             ftype = self.resolve_param_type(ftype)
383             # Fields are assumed to be read-write
384             # (except for Objects, see also glibtransformer.py)
385             node = Field(symbol.ident, ftype, ftype.name,
386                          readable=True, writable=True, bits=symbol.const_int)
387         return node
388
389     def _create_typedef(self, symbol):
390         ctype = symbol.base_type.type
391         if (ctype == CTYPE_POINTER and
392             symbol.base_type.base_type.type == CTYPE_FUNCTION):
393             node = self._create_typedef_callback(symbol)
394         elif (ctype == CTYPE_POINTER and
395             symbol.base_type.base_type.type == CTYPE_STRUCT):
396             node = self._create_typedef_struct(symbol, disguised=True)
397         elif ctype == CTYPE_STRUCT:
398             node = self._create_typedef_struct(symbol)
399         elif ctype == CTYPE_UNION:
400             node = self._create_typedef_union(symbol)
401         elif ctype == CTYPE_ENUM:
402             return self._create_enum(symbol)
403         elif ctype in (CTYPE_TYPEDEF,
404                        CTYPE_POINTER,
405                        CTYPE_BASIC_TYPE,
406                        CTYPE_VOID):
407             name = self.remove_prefix(symbol.ident)
408             if symbol.base_type.name:
409                 target = self.remove_prefix(symbol.base_type.name)
410             else:
411                 target = 'none'
412             if name in type_names:
413                 return None
414             return Alias(name, target, ctype=symbol.ident)
415         else:
416             raise NotImplementedError(
417                 "symbol %r of type %s" % (symbol.ident, ctype_name(ctype)))
418         return node
419
420     def _canonicalize_ctype(self, ctype):
421         # First look up the ctype including any pointers;
422         # a few type names like 'char*' have their own aliases
423         # and we need pointer information for those.
424         firstpass = type_name_from_ctype(ctype)
425
426         # If we have a particular alias for this, skip deep
427         # canonicalization to prevent changing
428         # e.g. char* -> int8*
429         if firstpass != ctype:
430             return firstpass
431
432         # We're also done if the type is already a fundamental
433         # known type, or there are no pointers.
434         if ctype in type_names or not firstpass.endswith('*'):
435             return firstpass
436
437         # We have a pointer type.
438         # Strip the end pointer, canonicalize our base type
439         base = firstpass[:-1]
440         canonical_base = self._canonicalize_ctype(base)
441
442         # Append the pointer again
443         canonical = canonical_base + '*'
444
445         return canonical
446
447     def parse_ctype(self, ctype, is_member=False):
448         canonical = self._canonicalize_ctype(ctype)
449
450         # Remove all pointers - we require standard calling
451         # conventions.  For example, an 'int' is always passed by
452         # value (unless it's out or inout).
453         derefed_typename = canonical.replace('*', '')
454
455         # Preserve "pointerness" of struct/union members
456         if (is_member and canonical.endswith('*') and
457             derefed_typename in BASIC_GIR_TYPES):
458             return 'any'
459         else:
460             return derefed_typename
461
462     def _create_type(self, source_type, is_param, is_retval):
463         ctype = self._create_source_type(source_type)
464         if ctype.startswith('va_list'):
465             raise SkipError()
466         # FIXME: FILE* should not be skipped, it should be handled
467         #        properly instead
468         elif ctype == 'FILE*':
469             raise SkipError
470
471         is_member = not (is_param or is_retval)
472         # Here we handle basic type parsing; most of the heavy lifting
473         # and inference comes in annotationparser.py when we merge
474         # in annotation data.
475         derefed_name = self.parse_ctype(ctype, is_member)
476         rettype = Type(derefed_name, ctype)
477         rettype.canonical = self._canonicalize_ctype(ctype)
478         derefed_ctype = ctype.replace('*', '')
479         rettype.derefed_canonical = self._canonicalize_ctype(derefed_ctype)
480
481         canontype = type_name_from_ctype(ctype)
482         # Is it a const char * or a const gpointer?
483         if ((canontype == TYPE_STRING or source_type.type == CTYPE_POINTER) and
484             (source_type.base_type.type_qualifier & TYPE_QUALIFIER_CONST)):
485             rettype.is_const = True
486         return rettype
487
488     def _create_parameter(self, symbol):
489         if symbol.type == CSYMBOL_TYPE_ELLIPSIS:
490             ptype = Varargs()
491         else:
492             ptype = self._create_type(symbol.base_type,
493                                       is_param=True, is_retval=False)
494             ptype = self.resolve_param_type(ptype)
495         return Parameter(symbol.ident, ptype)
496
497     def _create_return(self, source_type):
498         rtype = self._create_type(source_type,
499                                   is_param=False, is_retval=True)
500         rtype = self.resolve_param_type(rtype)
501         return_ = Return(rtype)
502         return return_
503
504     def _create_const(self, symbol):
505         # Don't create constants for non-public things
506         # http://bugzilla.gnome.org/show_bug.cgi?id=572790
507         if (symbol.source_filename is None or
508             not symbol.source_filename.endswith('.h')):
509             return None
510         name = self.remove_prefix(symbol.ident)
511         if symbol.const_string is not None:
512             type_name = 'utf8'
513             value = symbol.const_string
514         elif symbol.const_int is not None:
515             type_name = 'int'
516             value = symbol.const_int
517         elif symbol.const_double is not None:
518             type_name = 'double'
519             value = symbol.const_double
520         else:
521             raise AssertionError()
522
523         const = Constant(name, type_name, value)
524         return const
525
526     def _create_typedef_struct(self, symbol, disguised=False):
527         name = self.remove_prefix(symbol.ident)
528         struct = Struct(name, symbol.ident, disguised)
529         self._typedefs_ns[symbol.ident] = struct
530         self._create_struct(symbol)
531         return struct
532
533     def _create_typedef_union(self, symbol):
534         name = self.remove_prefix(symbol.ident)
535         union = Union(name, symbol.ident)
536         self._typedefs_ns[symbol.ident] = union
537         self._create_union(symbol)
538         return union
539
540     def _create_typedef_callback(self, symbol):
541         callback = self._create_callback(symbol)
542         self._typedefs_ns[callback.name] = callback
543         return callback
544
545     def _create_compound(self, klass, symbol, anonymous):
546         if symbol.ident is None:
547             # the compound is an anonymous member of another union or a struct
548             assert anonymous
549             compound = klass(None, None)
550         else:
551             compound = self._typedefs_ns.get(symbol.ident, None)
552
553         if compound is None:
554             # This is a bit of a hack; really we should try
555             # to resolve through the typedefs to find the real
556             # name
557             if symbol.ident.startswith('_'):
558                 name = symbol.ident[1:]
559                 compound = self._typedefs_ns.get(name, None)
560             else:
561                 name = symbol.ident
562             if compound is None:
563                 name = self.remove_prefix(name)
564                 compound = klass(name, symbol.ident)
565
566         for child in symbol.base_type.child_list:
567             field = self._traverse_one(child)
568             if field:
569                 compound.fields.append(field)
570
571         return compound
572
573     def _create_struct(self, symbol, anonymous=False):
574         return self._create_compound(Struct, symbol, anonymous)
575
576     def _create_union(self, symbol, anonymous=False):
577         return self._create_compound(Union, symbol, anonymous)
578
579     def _create_callback(self, symbol):
580         parameters = list(self._create_parameters(symbol.base_type.base_type))
581         retval = self._create_return(symbol.base_type.base_type.base_type)
582
583         # Mark the 'user_data' arguments
584         for i, param in enumerate(parameters):
585             if (param.type.name == 'any' and
586                 param.name == 'user_data'):
587                 param.closure_index = i
588
589         if symbol.ident.find('_') > 0:
590             name = self.remove_prefix(symbol.ident, True)
591         else:
592             name = self.remove_prefix(symbol.ident)
593         callback = Callback(name, retval, parameters, symbol.ident)
594
595         return callback
596
597     def _typepair_to_str(self, item):
598         nsname, item = item
599         if nsname is None:
600             return item.name
601         return '%s.%s' % (nsname, item.name)
602
603     def _resolve_type_name_1(self, type_name, ctype, names):
604         # First look using the built-in names
605         if ctype:
606             try:
607                 return type_names[ctype]
608             except KeyError, e:
609                 pass
610         try:
611             return type_names[type_name]
612         except KeyError, e:
613             pass
614
615         if ctype:
616             ctype = ctype.replace('*', '')
617             resolved = names.ctypes.get(ctype)
618             if resolved:
619                 return self._typepair_to_str(resolved)
620         type_name = self.remove_prefix(type_name)
621         resolved = names.aliases.get(type_name)
622         if resolved:
623             return self._typepair_to_str(resolved)
624         resolved = names.names.get(type_name)
625         if resolved:
626             return self._typepair_to_str(resolved)
627         resolved = names.type_names.get(type_name)
628         if resolved:
629             return self._typepair_to_str(resolved)
630         raise KeyError("failed to find %r" % (type_name, ))
631
632     def resolve_type_name_full(self, type_name, ctype,
633                                names, allow_invalid=True):
634         try:
635             return self._resolve_type_name_1(type_name, ctype, names)
636         except KeyError, e:
637             try:
638                 return self._resolve_type_name_1(type_name, ctype, self._names)
639             except KeyError, e:
640                 if not allow_invalid:
641                     raise
642                 return type_name
643
644     def resolve_type_name(self, type_name, ctype=None):
645         try:
646             return self.resolve_type_name_full(type_name, ctype, self._names)
647         except KeyError, e:
648             return type_name
649
650     def gtypename_to_giname(self, gtname, names):
651         resolved = names.type_names.get(gtname)
652         if resolved:
653             return self._typepair_to_str(resolved)
654         resolved = self._names.type_names.get(gtname)
655         if resolved:
656             return self._typepair_to_str(resolved)
657         raise KeyError("Failed to resolve GType name: %r" % (gtname, ))
658
659     def ctype_of(self, obj):
660         if hasattr(obj, 'ctype'):
661             return obj.ctype
662         elif hasattr(obj, 'symbol'):
663             return obj.symbol
664         else:
665             return None
666
667     def resolve_param_type_full(self, ptype, names, **kwargs):
668         if isinstance(ptype, Node):
669             ptype.name = self.resolve_type_name_full(ptype.name,
670                                                      self.ctype_of(ptype),
671                                                      names, **kwargs)
672         elif isinstance(ptype, basestring):
673             return self.resolve_type_name_full(ptype, ptype, names, **kwargs)
674         else:
675             raise AssertionError("Unhandled param: %r" % (ptype, ))
676         return ptype
677
678     def resolve_param_type(self, ptype):
679         try:
680             return self.resolve_param_type_full(ptype, self._names)
681         except KeyError, e:
682             return ptype
683
684     def follow_aliases(self, type_name, names):
685         while True:
686             resolved = names.aliases.get(type_name)
687             if resolved:
688                 (ns, alias) = resolved
689                 type_name = alias.target
690             else:
691                 break
692         return type_name