Arno-Can Uestuensoez <acue.opensou...@gmail.com> added the comment:

Hey,
first of all thank you for the fast reply.

Your examples reinforce exactly my issue. I prefer the KISS principle in 
providing isolated use-cases for transparency without extras. Which is 
seperated reduced flat calls only.

The other point is, I develop and test shared code for  Python2.7(which will be 
out there for a while...) and Python3.5+, thus prefer shared syntax whenever 
possible. The "super()" interface has changed, using the shared syntax from 
2.7.16:

- Python2.7.16:  super(type[, object-or-type])
- Python3.7.4:   super([type[, object-or-type]])

The test of your code with Python 2.7.16 requires modification.


You are right with the case 4, this is the actual issue related to the MRO and 
the parameter passing. I did not ask for the number of calls of the common base 
classes, but the passing of the parameters. The call order/MRO is OK, but I 
doubt the passed parameter signatures. My basic assumption here is that the 
call routing by MRO should or even must not be intermixed with the actually 
passed call parameters.

So I added the output of your example for "mixin_tests2.py" for Python3.6.5, 
tested for Python3.7.0 too.

#--------------
(3.6.5) [acue@lap001 test-modified]$ python mixin_tests2.py 

==============================
C MRO: ['C', 'A_With_Arg', 'B_With_Arg', 'MyBase', 'object']
C:C True
A_With_Arg:C True
call super().__init__ with argument
B_With_Arg:C True
call super().__init__ with argument
MyBase received 1 arguments
TypeError('object.__init__() takes no parameters',)
==============================
C MRO: ['C', 'B_With_Arg', 'A_With_Arg', 'MyBase', 'object']
C:C True
B_With_Arg:C True
call super().__init__ with argument
A_With_Arg:C True
call super().__init__ with argument
MyBase received 1 arguments
TypeError('object.__init__() takes no parameters',)
==============================
C MRO: ['C', 'A_With_Arg', 'B_Without_Arg', 'MyBase', 'object']
C:C True
A_With_Arg:C True
call super().__init__ with argument
B_Without_Arg:C True
call super().__init__ without argument
MyBase received 0 arguments
success
==============================
C MRO: ['C', 'B_Without_Arg', 'A_With_Arg', 'MyBase', 'object']
C:C True
B_Without_Arg:C True
call super().__init__ without argument
A_With_Arg:C False
success
(3.6.5) [acue@lap001 test-modified]$ 

#-----------------------

Or reduced to the focus of the actual issue and depicted the actual passed 
parameters. See the following "Case 4 - same as mixin_C_is_B0A1" for the 
essential question.

==============================

Case 0 - same as mixin_C_is_A1B1

   A_With_Arg:C True
   B_With_Arg:C True

   TypeError('object.__init__() takes no parameters',)

  Resulting in:
     C.__init__(True)
            A_With_Arg.__init__(True)
                   B_With_Arg.__init__(True)
                      MyBase.__init__(True)
                 object.__init__(True)

Comment:
  The TypeError is expected and OK.

==============================

Case 1 - same as mixin_C_is_B1A1

  B_With_Arg:C True
  A_With_Arg:C True

  TypeError('object.__init__() takes no parameters',)

  Resulting in:
     C.__init__(True)
            B_With_Arg.__init__(True)
                   A_With_Arg.__init__(True)
                      MyBase.__init__(True)
                 object.__init__(True)

Comment:
  The TypeError is expected and OK.

==============================

Case 2 - same as mixin_C_is_A1B0

  A_With_Arg:C True
  B_Without_Arg:C True

  MyBase received 0 arguments
  success

Comment:
  The success is expected and OK.

  The actual call due to MRO is:

     C(A_With_Arg(), B_Without_Arg())

  Resulting in:
     C.__init__(True)
            A_With_Arg.__init__(True)
                   B_Without_Arg.__init__(True)
                      MyBase.__init__(False)
                 object.__init__()

  Remark: I assume, that the single call of "object.__init__()" is here 
intentional.
  
==============================

Case 4 - same as mixin_C_is_B0A1

  B_Without_Arg:C True
  A_With_Arg:C False

  success

Comment:
  The success is NOT expected and as far as I can see should be treated as an 
Error so NOK.

  The actual call is:

     C(B_Without_Arg(), A_With_Arg())

  Resulting in:
     Seemingly the behaviour is:

     C.__init__(True)
            B_Without_Arg.__init__(True)
                   A_With_Arg.__init__(False)

     This means the call of "super(B, self).__init__()" within 
     "B.__init__()" defines the call parameters of "A.__init__()". 
     Even though I expect the call order as routed by the MRO, 
     I did not expected the parameters of A to be passed from the
     sibling class B, but as defined by the base class C. Thus the
     following:

     C.__init__(True)
        B_Without_Arg()
               A_With_Arg.__init__(True)
              object.__init__(True)    # this is the behaviour I expect 
                                       # due to the MRO and the
                                       # parameters passed from the 
                                       # base class

           
  Remark: I assume, that the single call of "object.__init__()" is here 
intentional.

#-----------------------

The essential part I doubt is given by the case 4 / same as mixin_C_is_B0A1, 
where the call parameter of A is defined by the sibling  B, instead of by the 
base class C. The routing of the call order is OK.

My remark with "standalone call" depicts the fact, that instanciating e.g. only 
your class "A_With_Arg(True)" will raise in any case a TypeError, but using it 
in the context "C(A_With_Arg(), B_Without_Arg())" may not because the call of 
the "object" class follows the right-most mixin. But anyhow, this means 
different parameters by the "suppressed" left-side classes/alls would be 
silently suppressed.


Resulting from the case 4, I tend to see the aspect of the parameter passing as 
erroneous.

----------
Added file: https://bugs.python.org/file48624/mixin_C_is_B0A1.py

_______________________________________
Python tracker <rep...@bugs.python.org>
<https://bugs.python.org/issue38262>
_______________________________________
_______________________________________________
Python-bugs-list mailing list
Unsubscribe: 
https://mail.python.org/mailman/options/python-bugs-list/archive%40mail-archive.com

Reply via email to