People seem to be pushing for a consistent method for checking the &quot;x-ness&quot; of objects (that is, interfaces that the object implements).<br><br>So I present an idea for a simple and straightforward type that provides a way to construct descriptions of object structures, and I&#39;d like some help expanding it into a useful extension to Python&#39;s standard set of utilities.
<br><br>Its a basic constructor that produces callable interface-checking predicates (which can be use in things such as list comprehensions, filter, if statements, or even a new syntax for function signatures that allows for automatic interface-checking). These predicates check that an object matches the behavior described by the constructor . Since I can&#39;t think of a name for this constructor, and because I&#39;ve never liked the term &quot;interface&quot;, I&#39;ll just call it &quot;can&quot;.
<br><br>Can takes an arbitrary number of keyword arguments and produces a callable object. The keys represent object attributes, while the values are behavior-checking predicates like the ones produced by can. Since the can constructor produces an object that can in turn be used in other can constructors, using previously defined interfaces in new constructions is fairly straight-forward
<br><br>&nbsp;&nbsp;&nbsp;&nbsp;callable =&nbsp;&nbsp;can(__call__ = object)#Returns an object that describes objects with a&nbsp;&nbsp;__call__ attribute<br>&nbsp;&nbsp;&nbsp;&nbsp;readable =&nbsp;&nbsp;can(read&nbsp;&nbsp;= callable) #&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;...with a callable read attribute
<br>&nbsp;&nbsp;&nbsp;&nbsp;writable =&nbsp;&nbsp;can(write = callable) #&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;...with a callable write attribute<br><br><br>&nbsp;&nbsp;&nbsp;&nbsp;#a join operator can be used to combine can objects...<br>&nbsp;&nbsp;&nbsp;&nbsp;#...for now I&#39;ll just use &quot;and&quot; and &quot;or&quot; to represent them.
<br>&nbsp;&nbsp;&nbsp;&nbsp;<br>&nbsp;&nbsp;&nbsp;&nbsp;isfilelike = readable and writable #returns an object that matches any type that is described<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; #by both readable and writable<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
<br>&nbsp;&nbsp;&nbsp;&nbsp;IOable&nbsp;&nbsp;&nbsp;&nbsp; = readable or writable&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; #any type that is readable or writable<br><br>objects that are constructed with can, when called, return True or False based on whether or not the passed object matches the behavior described.
<br><br>&nbsp;&nbsp;&nbsp;&nbsp;callable(hash)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;#returns True - as it would in the current version of Python.<br><br><br>Here&#39;s some more nifty examples:<br><br>&nbsp;&nbsp;&nbsp;&nbsp;iterable = can(__iter__=callable) or can(next=callable)<br>&nbsp;&nbsp;&nbsp;&nbsp;hashable = can(__hash__=callable)
<br><br>&nbsp;&nbsp;&nbsp;&nbsp;completefile = isfilelike and iterable and can(fileno=callable, close=callable)<br><br>&nbsp;&nbsp;&nbsp;&nbsp;def outlines(f, seq):<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&quot;&quot;&quot;Outputs a sequence of lines to a file-like object&quot;&quot;&quot;
<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;assert isfilelike(f), &quot;%r is not a file-like object.&quot; % f<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;assert isiterable(seq), &quot;%r must be iterable&quot;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; % seq<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;f.write(&quot;\n&quot;.join(seq))<br><br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;#a trivial example... you&#39;d get similar error messages from Python builtins even without the assertions.
<br><br><br>As it stands, I don&#39;t think that this deserves to be in Python - but I think the basic premise could be used as a foundation for&nbsp; better things.<br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<br><br>-- <br>&quot;What&#39;s money? A man is a success if he gets up in the morning and goes to bed at night and in between does what he wants to do.&quot; ~ Bob Dylan