print_signatures.py 6.2 KB
Newer Older
Y
yuyang18 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
# Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""
Print all signature of a python module in alphabet order.

Usage:
18
    ./print_signature  "paddle.fluid" > signature.txt
Y
yuyang18 已提交
19
"""
M
minqiyang 已提交
20 21
from __future__ import print_function

Y
yuyang18 已提交
22 23 24 25 26
import importlib
import inspect
import collections
import sys
import pydoc
27
import hashlib
28
import platform
Z
Zeng Jinle 已提交
29
import functools
Y
yuyang18 已提交
30 31 32

member_dict = collections.OrderedDict()

Z
Zeng Jinle 已提交
33 34
visited_modules = set()

Y
yuyang18 已提交
35

36
def md5(doc):
37 38 39 40 41 42 43 44 45 46 47 48 49 50 51
    try:
        hashinst = hashlib.md5()
        if platform.python_version()[0] == "2":
            hashinst.update(str(doc))
        else:
            hashinst.update(str(doc).encode('utf-8'))
        md5sum = hashinst.hexdigest()
    except UnicodeDecodeError as e:
        md5sum = None
        print(
            "Error({}) occurred when `md5({})`, discard it.".format(
                str(e), doc),
            file=sys.stderr)

    return md5sum
52 53


Z
Zeng Jinle 已提交
54 55 56 57 58 59 60
def get_functools_partial_spec(func):
    func_str = func.func.__name__
    args = func.args
    keywords = func.keywords
    return '{}(args={}, keywords={})'.format(func_str, args, keywords)


61
def format_spec(spec):
Z
Zeng Jinle 已提交
62 63 64
    args = spec.args
    varargs = spec.varargs
    keywords = spec.keywords
65 66 67 68 69 70
    defaults = spec.defaults
    if defaults is not None:
        defaults = list(defaults)
        for idx, item in enumerate(defaults):
            if not isinstance(item, functools.partial):
                continue
Z
Zeng Jinle 已提交
71

72 73 74
            defaults[idx] = get_functools_partial_spec(item)

        defaults = tuple(defaults)
Z
Zeng Jinle 已提交
75 76

    return 'ArgSpec(args={}, varargs={}, keywords={}, defaults={})'.format(
77
        args, varargs, keywords, defaults)
Z
Zeng Jinle 已提交
78 79


80
def queue_dict(member, cur_name):
81 82 83 84 85 86 87 88 89 90 91 92 93
    if cur_name != 'paddle':
        try:
            eval(cur_name)
        except (AttributeError, NameError, SyntaxError) as e:
            print(
                "Error({}) occurred when `eval({})`, discard it.".format(
                    str(e), cur_name),
                file=sys.stderr)
            return

    if (inspect.isclass(member) or inspect.isfunction(member) or
            inspect.ismethod(member)) and hasattr(
                member, '__module__') and hasattr(member, '__name__'):
T
tianshuo78520a 已提交
94
        args = member.__module__ + "." + member.__name__
95 96 97 98 99 100 101 102
        try:
            eval(args)
        except (AttributeError, NameError, SyntaxError) as e:
            print(
                "Error({}) occurred when `eval({})`, discard it for {}.".format(
                    str(e), args, cur_name),
                file=sys.stderr)
            return
T
tianshuo78520a 已提交
103 104
    else:
        try:
105 106
            args = inspect.getargspec(member)
            has_type_error = False
T
tianshuo78520a 已提交
107 108 109 110 111
        except TypeError:  # special for PyBind method
            args = "  ".join([
                line.strip() for line in pydoc.render_doc(member).split('\n')
                if "->" in line
            ])
112 113 114 115 116
            has_type_error = True

        if not has_type_error:
            args = format_spec(args)

117
    doc_md5 = md5(member.__doc__)
Z
Zeng Jinle 已提交
118
    member_dict[cur_name] = "({}, ('document', '{}'))".format(args, doc_md5)
119 120


121 122 123 124 125
def visit_member(parent_name, member, member_name=None):
    if member_name:
        cur_name = ".".join([parent_name, member_name])
    else:
        cur_name = ".".join([parent_name, member.__name__])
X
fix py3  
Xin Pan 已提交
126
    if inspect.isclass(member):
127
        queue_dict(member, cur_name)
Y
yuyang18 已提交
128
        for name, value in inspect.getmembers(member):
129
            if hasattr(value, '__name__') and not name.startswith("_"):
Y
yuyang18 已提交
130
                visit_member(cur_name, value)
Z
zhangchunle 已提交
131 132
    elif inspect.ismethoddescriptor(member):
        return
133 134
    elif inspect.isbuiltin(member):
        return
Y
yuyang18 已提交
135
    elif callable(member):
136
        queue_dict(member, cur_name)
X
fix py3  
Xin Pan 已提交
137 138
    elif inspect.isgetsetdescriptor(member):
        return
Y
yuyang18 已提交
139 140 141 142 143
    else:
        raise RuntimeError("Unsupported generate signature of member, type {0}".
                           format(str(type(member))))


Z
Zeng Jinle 已提交
144
def is_primitive(instance):
145
    int_types = (int, long) if platform.python_version()[0] == "2" else (int, )
Z
Zeng Jinle 已提交
146 147 148 149 150 151 152 153 154 155 156 157 158
    pritimitive_types = int_types + (float, str)
    if isinstance(instance, pritimitive_types):
        return True
    elif isinstance(instance, (list, tuple, set)):
        for obj in instance:
            if not is_primitive(obj):
                return False

        return True
    else:
        return False


Y
yuyang18 已提交
159
def visit_all_module(mod):
Z
Zeng Jinle 已提交
160 161 162 163
    mod_name = mod.__name__
    if mod_name != 'paddle' and not mod_name.startswith('paddle.'):
        return

164 165 166
    if mod_name.startswith('paddle.fluid.core'):
        return

Z
Zeng Jinle 已提交
167 168 169 170
    if mod in visited_modules:
        return

    visited_modules.add(mod)
171 172 173 174 175 176 177 178
    if hasattr(mod, "__all__"):
        member_names = (name for name in mod.__all__
                        if not name.startswith("_"))
    elif mod_name == 'paddle':
        member_names = dir(mod)
    else:
        return
    for member_name in member_names:
Y
yuyang18 已提交
179 180 181
        instance = getattr(mod, member_name, None)
        if instance is None:
            continue
Z
Zeng Jinle 已提交
182 183 184 185 186 187 188

        if is_primitive(instance):
            continue

        if not hasattr(instance, "__name__"):
            continue

Y
yuyang18 已提交
189 190 191
        if inspect.ismodule(instance):
            visit_all_module(instance)
        else:
192
            if member_name != instance.__name__:
193
                print(
194
                    "Found alias API, alias name is: {}, original name is: {}".
195 196
                    format(member_name, instance.__name__),
                    file=sys.stderr)
197 198 199
                visit_member(mod.__name__, instance, member_name)
            else:
                visit_member(mod.__name__, instance)
Y
yuyang18 已提交
200 201


202
if __name__ == '__main__':
203
    import paddle
204 205 206
    modules = sys.argv[1].split(",")
    for m in modules:
        visit_all_module(importlib.import_module(m))
Y
yuyang18 已提交
207

208 209
    for name in member_dict:
        print(name, member_dict[name])