How do you get all classes defined in a module but not imported?

26,718

Solution 1

Inspect the __module__ attribute of the class to find out which module it was defined in.

Solution 2

I apologize for answering such an old question, but I didn't feel comfortable using the inspect module for this solution. I read somewhere that is wasn't safe to use in production.

Initialize all the classes in a module into nameless objects in a list

See Antonis Christofides comment to answer 1.

I got the answer for testing if an object is a class from How to check whether a variable is a class or not?

So this is my inspect-free solution

def classesinmodule(module):
    md = module.__dict__
    return [
        md[c] for c in md if (
            isinstance(md[c], type) and md[c].__module__ == module.__name__
        )
    ]

classesinmodule(modulename)

Solution 3

You may also want to consider using the "Python class browser" module in the standard library: http://docs.python.org/library/pyclbr.html

Since it doesn't actually execute the module in question (it does naive source inspection instead) there are some specific techniques it doesn't quite understand correctly, but for all "normal" class definitions, it will describe them accurately.

Solution 4

I used the below:

# Predicate to make sure the classes only come from the module in question
def pred(c):
    return inspect.isclass(c) and c.__module__ == pred.__module__
# fetch all members of module __name__ matching 'pred'
classes = inspect.getmembers(sys.modules[__name__], pred)

I didn't want to type the current module name in

Solution 5

from pyclbr import readmodule

clsmembers = readmodule(__name__).items()
Share:
26,718

Related videos on Youtube

Davy8
Author by

Davy8

Come work with me http://nerdery.com/workwithme/dv

Updated on January 19, 2020

Comments

  • Davy8
    Davy8 over 4 years

    I've already seen the following question but it doesn't quite get me where I want: How can I get a list of all classes within current module in Python?

    In particular, I do not want classes that are imported, e.g. if I had the following module:

    from my.namespace import MyBaseClass
    from somewhere.else import SomeOtherClass
    
    class NewClass(MyBaseClass):
        pass
    
    class AnotherClass(MyBaseClass):
        pass
    
    class YetAnotherClass(MyBaseClass):
        pass
    

    If I use clsmembers = inspect.getmembers(sys.modules[__name__], inspect.isclass) like the accepted answer in the linked question suggests, it would return MyBaseClass and SomeOtherClass in addition to the 3 defined in this module.

    How can I get only NewClass, AnotherClass and YetAnotherClass?

    • zxt
      zxt about 13 years
      Unless I understand wrong, do the comments to the accepted answer in that question not answer your question? (I did not try it to see if it works though)
    • Davy8
      Davy8 about 13 years
      @zxt d'oh glazed over the comments. You're right, it does answer my question.
  • Blair
    Blair about 13 years
    A list comprehension along the lines of [m for m in clsmembers if m[1].__module__ == 'mymodule'] should do the trick. You could also supply a lambda function as the predicate to the inspect.getmembers() call, as shown in the question you referred to.
  • Mr_and_Mrs_D
    Mr_and_Mrs_D almost 8 years
    Always iterate over items if you need both: v for c,v in md.iteritems() if ( isinstance(v, type) and v.__module__ == module.__name__ )
  • BDL
    BDL over 7 years
    Although this code might solve the problem, it always good to add an explanation to it.