2011-05-15 13:30:14 -04:00
|
|
|
/**
|
|
|
|
* Tests final members
|
|
|
|
*
|
|
|
|
* Copyright (C) 2010 Mike Gerwitz
|
|
|
|
*
|
|
|
|
* This file is part of ease.js.
|
|
|
|
*
|
|
|
|
* ease.js is free software: you can redistribute it and/or modify it under the
|
|
|
|
* terms of the GNU Lesser General Public License as published by the Free
|
|
|
|
* Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* @author Mike Gerwitz
|
|
|
|
* @package test
|
|
|
|
*/
|
|
|
|
|
2011-05-19 07:40:42 -04:00
|
|
|
var common = require( './common' ),
|
|
|
|
assert = require( 'assert' ),
|
2011-05-22 11:11:18 -04:00
|
|
|
builder = common.require( 'class_builder' ),
|
|
|
|
|
|
|
|
Class = common.require( 'class' )
|
|
|
|
FinalClass = common.require( 'class_final' )
|
2011-05-15 13:30:14 -04:00
|
|
|
;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Methods declared as final should not be able to be overridden by subtypes.
|
|
|
|
* Simple as that.
|
|
|
|
*/
|
|
|
|
( function testFinalMethodsCannotBeOverridenBySubtypes()
|
|
|
|
{
|
|
|
|
var Foo = builder.build(
|
|
|
|
{
|
|
|
|
'final public foo': function() {},
|
|
|
|
} );
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// attempt to override (should fail)
|
|
|
|
builder.build( Foo,
|
|
|
|
{
|
|
|
|
'public foo': function() {},
|
|
|
|
} );
|
|
|
|
}
|
|
|
|
catch ( e )
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
e.message.search( 'foo' ) !== -1,
|
|
|
|
"Final error message contains name of method"
|
|
|
|
);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.fail( 'Should not be able to override final methods' );
|
|
|
|
} )();
|
|
|
|
|
2011-05-18 20:51:02 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Due to our static implementation (late static binding), the use of 'final'
|
|
|
|
* with properties doesn't make much sense. We have to deal with different
|
|
|
|
* problems than languages like Java that truly bind members statically. Consult
|
|
|
|
* the documentation for rationale.
|
|
|
|
*
|
|
|
|
* See also const keyword.
|
|
|
|
*/
|
|
|
|
( function testFinalKeywordCannotBeUsedWithProperties()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// attempt to define a 'final' property (should fail)
|
|
|
|
builder.build(
|
|
|
|
{
|
|
|
|
'final public foo': 'bar',
|
|
|
|
} );
|
|
|
|
}
|
|
|
|
catch ( e )
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
e.message.search( 'foo' ) !== -1,
|
2011-05-22 11:11:18 -04:00
|
|
|
"Final property error message contains name of property"
|
2011-05-18 20:51:02 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.fail( "Should not be able to use final keyword with properties" );
|
|
|
|
} )();
|
|
|
|
|
2011-05-22 11:11:18 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The 'abstract' keyword's very point is to state that no definition is
|
|
|
|
* provided and that a subtype must provide one. Therefore, declaring something
|
|
|
|
* 'abstract final' is rather contradictory and should not be permitted.
|
|
|
|
*/
|
|
|
|
( function testFinalyKeywordCannotBeUsedWithAbstract()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// should fail
|
|
|
|
builder.build( { 'abstract final foo': [] } );
|
|
|
|
}
|
|
|
|
catch ( e )
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
e.message.search( 'foo' ) !== -1,
|
|
|
|
"Abstract final error message contains name of method"
|
|
|
|
);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.fail( "Should not be able to use final keyword with abstract" );
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Ensure that FinalClass properly forwards data to create a new Class.
|
|
|
|
*/
|
|
|
|
( function testFinalClassesAreValidClasses()
|
|
|
|
{
|
|
|
|
assert.ok( Class.isClass( FinalClass( {} ) ),
|
|
|
|
"Final classes should generate valid classes"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When a class is declared as final, it should prevent it from ever being
|
|
|
|
* extended. Ever.
|
|
|
|
*/
|
|
|
|
( function testFinalClassesCannotBeExtended()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// this should fail
|
|
|
|
FinalClass( 'Foo', {} ).extend( {} );
|
|
|
|
}
|
|
|
|
catch ( e )
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
e.message.search( 'Foo' ) !== -1,
|
|
|
|
"Final class error message should contain name of class"
|
|
|
|
);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.fail( "Should not be able to extend final classes" );
|
|
|
|
} )();
|
|
|
|
|
2011-05-22 21:35:29 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Ensure we're able to create final classes by extending existing classes.
|
|
|
|
*/
|
|
|
|
( function testCanCreateFinalSubtypes()
|
|
|
|
{
|
|
|
|
var Foo = builder.build( {} ),
|
|
|
|
FinalNamed = FinalClass( 'FinalNamed' ).extend( Foo, {} ),
|
|
|
|
FinalAnon = FinalClass.extend( Foo, {} )
|
|
|
|
;
|
|
|
|
|
|
|
|
// named
|
|
|
|
assert.throws( function()
|
|
|
|
{
|
|
|
|
FinalNamed.extend( {} );
|
|
|
|
}, Error, "Cannot extend final named subtype" );
|
|
|
|
|
|
|
|
// anonymous
|
|
|
|
assert.throws( function()
|
|
|
|
{
|
|
|
|
FinalAnon.extend( {} );
|
|
|
|
}, Error, "Cannot extend final anonymous subtype" );
|
|
|
|
} )();
|
|
|
|
|