2019-11-08 17:27:16 +01:00
import pytest
from generic . multimethod import has_multimethods , multimethod
def test_multimethod ( ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-08 17:27:16 +01:00
@multimethod ( int )
def foo ( self , x ) :
return x + 1
2019-11-08 17:38:51 +01:00
@foo.register ( str ) # type: ignore[no-redef]
2019-11-08 17:27:16 +01:00
def foo ( self , x ) :
2022-11-19 22:09:49 -05:00
return f " { x } 1 "
2019-11-08 17:27:16 +01:00
assert Dummy ( ) . foo ( 1 ) == 2
assert Dummy ( ) . foo ( " 1 " ) == " 11 "
with pytest . raises ( TypeError ) :
Dummy ( ) . foo ( [ ] )
2019-11-09 19:27:15 +01:00
def test_multimethod_with_two_arguments ( ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-09 19:27:15 +01:00
@multimethod ( int , int )
def foo ( self , x , y ) :
return x * y
@foo.register ( str , int ) # type: ignore[no-redef]
def foo ( self , s , x ) :
return s * x
assert Dummy ( ) . foo ( 1 , 1 ) == 1
assert Dummy ( ) . foo ( " 1 " , 2 ) == " 11 "
with pytest . raises ( TypeError ) :
Dummy ( ) . foo ( [ ] )
def test_multimethod_otherwise_clause ( ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-09 19:27:15 +01:00
@multimethod ( int )
def foo ( self , x ) :
return x + 1
@foo.otherwise # type: ignore[no-redef]
def foo ( self , x ) :
return type ( x )
assert Dummy ( ) . foo ( 1 ) == 2
assert Dummy ( ) . foo ( " " ) is str
assert Dummy ( ) . foo ( [ ] ) is list
def test_multimethod_otherwise_clausewith_two_arguments ( ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-09 19:27:15 +01:00
@multimethod ( int , int )
def foo ( self , x , y ) :
return x * y
@foo.otherwise # type: ignore[no-redef]
def foo ( self , s , x ) :
return f " { s } { x } "
assert Dummy ( ) . foo ( 1 , 2 ) == 2
assert Dummy ( ) . foo ( " a " , [ ] ) == " a [] "
2019-11-08 17:27:16 +01:00
def test_inheritance ( ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-08 17:27:16 +01:00
@multimethod ( int )
def foo ( self , x ) :
return x + 1
2019-11-08 17:38:51 +01:00
@foo.register ( float ) # type: ignore[no-redef]
2019-11-08 17:27:16 +01:00
def foo ( self , x ) :
return x + 1.5
@has_multimethods
class DummySub ( Dummy ) :
@Dummy.foo.register ( str )
def foo ( self , x ) :
2022-11-19 22:09:49 -05:00
return f " { x } 1 "
2019-11-08 17:27:16 +01:00
2019-11-08 17:38:51 +01:00
@foo.register ( tuple ) # type: ignore[no-redef]
2019-11-08 17:27:16 +01:00
def foo ( self , x ) :
return x + ( 1 , )
2019-11-08 17:38:51 +01:00
@Dummy.foo.register ( bool ) # type: ignore[no-redef]
2019-11-08 17:27:16 +01:00
def foo ( self , x ) :
return not x
assert Dummy ( ) . foo ( 1 ) == 2
assert Dummy ( ) . foo ( 1.5 ) == 3.0
with pytest . raises ( TypeError ) :
Dummy ( ) . foo ( " 1 " )
assert DummySub ( ) . foo ( 1 ) == 2
assert DummySub ( ) . foo ( 1.5 ) == 3.0
assert DummySub ( ) . foo ( " 1 " ) == " 11 "
assert DummySub ( ) . foo ( ( 1 , 2 ) ) == ( 1 , 2 , 1 )
2020-08-28 22:13:46 -04:00
assert DummySub ( ) . foo ( True ) is False
2019-11-08 17:27:16 +01:00
with pytest . raises ( TypeError ) :
DummySub ( ) . foo ( [ ] )
def test_override_in_same_class_not_allowed ( ) :
with pytest . raises ( ValueError ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-08 17:27:16 +01:00
@multimethod ( str , str )
def foo ( self , x , y ) :
return x + y
2019-11-08 17:38:51 +01:00
@foo.register ( str , str ) # type: ignore[no-redef]
2019-11-08 17:27:16 +01:00
def foo ( self , x , y ) :
return y + x
def test_inheritance_override ( ) :
@has_multimethods
2022-10-10 18:54:50 -04:00
class Dummy :
2019-11-08 17:27:16 +01:00
@multimethod ( int )
def foo ( self , x ) :
return x + 1
@has_multimethods
class DummySub ( Dummy ) :
@Dummy.foo.register ( int )
def foo ( self , x ) :
return x + 3
assert Dummy ( ) . foo ( 1 ) == 2
assert DummySub ( ) . foo ( 1 ) == 4