[#25] Moved getter/setter validation tests into new test case
Much more elegant a test case now.closure/master
parent
ad0343fb9b
commit
93021f3dbc
|
@ -210,7 +210,7 @@ exports.buildGetter = function( members, meta, name, value, keywords, base )
|
|||
var prev_data = scanMembers( members, name, base );
|
||||
|
||||
this._validate.validateGetterSetter(
|
||||
name, members, keywords, base, prev_data
|
||||
name, keywords, prev_data
|
||||
);
|
||||
|
||||
Object.defineProperty(
|
||||
|
@ -250,7 +250,7 @@ exports.buildSetter = function( members, meta, name, value, keywords, base )
|
|||
var prev_data = scanMembers( members, name, base );
|
||||
|
||||
this._validate.validateGetterSetter(
|
||||
name, members, keywords, base, prev_data
|
||||
name, keywords, prev_data
|
||||
);
|
||||
|
||||
this._validate.validateGetterSetter( name, members, keywords, base );
|
||||
|
|
|
@ -237,16 +237,12 @@ exports.prototype.validateProperty = function(
|
|||
* If a problem is found, an exception will be thrown.
|
||||
*
|
||||
* @param {string} name getter/setter name
|
||||
*
|
||||
* @param {{public: Object, protected: Object, private: Object}} members
|
||||
*
|
||||
* @param {Object.<string,boolean>} keywords parsed keywords
|
||||
* @param {Object} base optional base to parse
|
||||
*
|
||||
* @return {undefined}
|
||||
*/
|
||||
exports.prototype.validateGetterSetter = function(
|
||||
name, members, keywords, base, prev_data
|
||||
name, keywords, prev_data
|
||||
)
|
||||
{
|
||||
var prev = ( prev_data ) ? prev_data.member : null,
|
||||
|
|
|
@ -0,0 +1,79 @@
|
|||
/**
|
||||
* Tests member builder validation rules for getters/setters
|
||||
*
|
||||
* These tests can be run in a pre-ES5 environment since they do not deal with
|
||||
* actual getters/setters; they deal only with the data associated with them.
|
||||
*
|
||||
* 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
|
||||
*/
|
||||
|
||||
var shared = require( __dirname + '/inc-common' );
|
||||
|
||||
require( 'common' ).testCase(
|
||||
{
|
||||
caseSetUp: function()
|
||||
{
|
||||
this.quickFailureTest = shared.quickFailureTest;
|
||||
},
|
||||
|
||||
|
||||
setUp: function()
|
||||
{
|
||||
this.sut = this.require( 'MemberBuilderValidator' )();
|
||||
},
|
||||
|
||||
|
||||
/**
|
||||
* Getters/setters should not be able to override methods, for the obvious
|
||||
* reason that they are two different types and operate entirely
|
||||
* differently. Go figure.
|
||||
*/
|
||||
'Cannot override method with getter or setter': function()
|
||||
{
|
||||
var name = 'foo',
|
||||
_self = this;
|
||||
|
||||
// getters and setters share the same call, so we don't need two
|
||||
// separate tests
|
||||
this.quickFailureTest( name, 'method', function()
|
||||
{
|
||||
_self.sut.validateGetterSetter(
|
||||
name, {}, { member: function() {} }
|
||||
);
|
||||
} );
|
||||
},
|
||||
|
||||
|
||||
'Cannot override property with getter or setter': function()
|
||||
{
|
||||
var name = 'foo',
|
||||
_self = this;
|
||||
|
||||
// getters and setters share the same call, so we don't need two
|
||||
// separate tests
|
||||
this.quickFailureTest( name, 'method', function()
|
||||
{
|
||||
_self.sut.validateGetterSetter(
|
||||
name, {}, { member: 'foo' }
|
||||
);
|
||||
} );
|
||||
},
|
||||
} );
|
|
@ -1,192 +0,0 @@
|
|||
/**
|
||||
* Tests getter/setter builder
|
||||
*
|
||||
* 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
|
||||
*/
|
||||
|
||||
var common = require( './common' ),
|
||||
assert = require( 'assert' ),
|
||||
util = common.require( 'util' ),
|
||||
|
||||
builder = common.require( 'MemberBuilder' )(),
|
||||
|
||||
buildGetter = function()
|
||||
{
|
||||
builder.buildGetter.apply( builder, arguments );
|
||||
},
|
||||
|
||||
buildSetter = function()
|
||||
{
|
||||
builder.buildSetter.apply( builder, arguments );
|
||||
},
|
||||
|
||||
// member visibility types are quoted because they are reserved keywords
|
||||
members = {},
|
||||
meta = {},
|
||||
|
||||
// stub values
|
||||
name = 'foo',
|
||||
value = function() {}
|
||||
;
|
||||
|
||||
|
||||
// no need to test getters/setters in browsers that do not support them
|
||||
if ( util.definePropertyFallback() )
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
function setUp()
|
||||
{
|
||||
// clear out the members for a fresh start
|
||||
members = { 'public': {}, 'protected': {}, 'private': {} };
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Partially applied function to quickly build getter from common test data
|
||||
*/
|
||||
function buildGetterSetterQuick( keywords, val, preserve_prior, use )
|
||||
{
|
||||
preserve_prior = !!preserve_prior;
|
||||
use = ( use === undefined ) ? 0 : +use;
|
||||
|
||||
keywords = keywords || {};
|
||||
val = val || value;
|
||||
|
||||
if ( !preserve_prior )
|
||||
{
|
||||
setUp();
|
||||
}
|
||||
|
||||
if ( use == 0 || use == 1 )
|
||||
{
|
||||
buildGetter( members, meta, name, val, keywords );
|
||||
}
|
||||
if ( use == 0 || use == 2 )
|
||||
{
|
||||
buildSetter( members, meta, name, val, keywords );
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
function testEach( test )
|
||||
{
|
||||
test( 'getter', function( keywords, val, preserve )
|
||||
{
|
||||
buildGetterSetterQuick.call( this, keywords, val, preserve, 1 );
|
||||
} );
|
||||
|
||||
test( 'setter', function( keywords, val, preserve )
|
||||
{
|
||||
buildGetterSetterQuick.call( this, keywords, val, preserve, 2 );
|
||||
} );
|
||||
}
|
||||
|
||||
|
||||
( function testThrowsTypeErrorIfMultipleVisibilityKeywordsAreGiven()
|
||||
{
|
||||
assert.throws( function()
|
||||
{
|
||||
buildGetterSetterQuick( {
|
||||
'public': true,
|
||||
'protected': true,
|
||||
} );
|
||||
}, TypeError, "Cannot specify multiple visibility keywords (0)" );
|
||||
|
||||
assert.throws( function()
|
||||
{
|
||||
buildGetterSetterQuick( {
|
||||
'public': true,
|
||||
'private': true,
|
||||
} );
|
||||
}, TypeError, "Cannot specify multiple visibility keywords (1)" );
|
||||
|
||||
assert.throws( function()
|
||||
{
|
||||
buildGetterSetterQuick( {
|
||||
'protected': true,
|
||||
'private': true,
|
||||
} );
|
||||
}, TypeError, "Cannot specify multiple visibility keywords (2)" );
|
||||
|
||||
} )();
|
||||
|
||||
|
||||
/**
|
||||
* Getters/setters should not be able to override methods, for the obvious
|
||||
* reason that they are two different types and operate entirely differently. Go
|
||||
* figure.
|
||||
*/
|
||||
testEach( function testCannotOverrideMethodWithGetterOrSetter( type, build )
|
||||
{
|
||||
setUp();
|
||||
|
||||
// method
|
||||
members[ 'public' ][ name ] = function() {};
|
||||
|
||||
try
|
||||
{
|
||||
// attempt to override method with getter/setter (should fail)
|
||||
build( { 'public': true }, null, true );
|
||||
}
|
||||
catch ( e )
|
||||
{
|
||||
assert.ok( e.message.search( name ) !== -1,
|
||||
"Method override error message should contain " + type + " name"
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
assert.fail( type + " should not be able to override methods");
|
||||
} );
|
||||
|
||||
|
||||
/**
|
||||
* Getters/setters should not be able to override properties. While, at first,
|
||||
* this concept may seem odd, keep in mind that the parent would likely not
|
||||
* expect a subtype to be able to override property assignments. This could open
|
||||
* up holes to exploit the parent class.
|
||||
*/
|
||||
testEach( function testCannotOverridePropertiesWithGetterOrSetter( type, build )
|
||||
{
|
||||
setUp();
|
||||
|
||||
// declare a property
|
||||
members[ 'public' ][ name ] = 'foo';
|
||||
|
||||
try
|
||||
{
|
||||
// attempt to override property with getter/setter (should fail)
|
||||
build( { 'public': true }, null, true );
|
||||
}
|
||||
catch ( e )
|
||||
{
|
||||
assert.ok( e.message.search( name ) !== -1,
|
||||
"Property override error message should contain getter/setter name"
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
assert.fail( type + " should not be able to override properties" );
|
||||
} );
|
||||
|
Loading…
Reference in New Issue