1
0
Fork 0
easejs/test/MemberBuilder/VisibilityTest.js

298 lines
8.5 KiB
JavaScript

/**
* Tests visibility portion of member 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
*/
require( 'common' ).testCase(
{
caseSetUp: function()
{
var _self = this;
this.buildStubMethod = function( name, val, visibility )
{
var keywords = {};
// set visibility level using access modifier
keywords[ visibility ] = true;
this.sut.buildMethod( this.members, {}, name,
function() {
return val;
},
keywords,
function() {},
1,
{}
);
};
this.buildStubProp = function( name, val, visibility )
{
var keywords = {};
// set visibility level using access modifier
keywords[ visibility ] = true;
this.sut.buildProp( this.members, {}, name, val, keywords, {} );
};
this.assertOnlyIn = function( vis, name )
{
var found = false;
this.incAssertCount();
for ( level in this.members )
{
if ( typeof this.members[ level ][ name ] === 'undefined' )
{
continue;
}
// we found it; ensure it's in the expected visibility level
found = true;
if ( level !== vis )
{
this.fail( name + " should only be accessible in: " + vis );
}
}
found || this.fail(
"Did not find '" + name + "' in level: " + vis
);
};
this.basicVisPropTest = function( vis )
{
var name = vis + 'propname',
val = vis + 'val';
this.buildStubProp( name, val, vis );
this.assertEqual( this.members[ vis ][ name ][ 0 ], val );
this.assertOnlyIn( vis, name, this.members );
};
this.basicVisMethodTest = function( vis )
{
var name = vis + 'metohdname',
val = vis + 'val';
this.buildStubMethod( name, val, vis );
this.assertEqual(
this.members[ vis ][ name ](),
val
);
this.assertOnlyIn( vis, name, this.members );
};
this.multiVisFailureTest = function( test )
{
var multi = [
{ 'public': true, 'protected': true },
{ 'public': true, 'private': true },
{ 'protected': true, 'private': true },
],
name = 'foo'
;
// run the test for each combination of multiple access modifiers
for ( var i = 0, len = multi.length; i < len; i++ )
{
_self.incAssertCount();
try
{
test( name, multi[ i ] );
}
catch ( e )
{
// ensure we received the correct error
_self.assertOk(
( e.message.search( 'access modifier' ) > -1 ),
'Unexpected error for multiple access modifiers'
);
// ensure the error message contains the name of the member
_self.assertOk(
( e.message.search( name ) > -1 ),
'Multiple access modifier error message should ' +
'contain name of member'
);
return;
}
_self.fail(
'Should fail with multiple access modifiers: ' + i
);
}
};
},
setUp: function()
{
// stub factories used for testing
var stubFactory = this.require( 'MethodWrapperFactory' )(
function( func ) { return func; }
);
this.sut = this.require( 'MemberBuilder' )(
stubFactory, stubFactory
);
this.members = this.sut.initMembers();
},
/**
* The member object stores the members associated with each of the three
* levels of visibility that are denoted by access modifiers: public,
* protected and private. The initMembers() method is simply an abstraction.
*/
'Can create empty member object': function()
{
var members = this.sut.initMembers(),
test = [ 'public', 'protected', 'private' ];
// ensure each level of visibility exists in the new member object
// (aren't these for statements terribly repetitive? 0 <= i < len would
// be nice to be able to do.)
for ( var i = 0, len = test.length; i < len; i++ )
{
this.assertOk( ( typeof members[ test[ i ] ] !== 'undefined' ),
'Clean member object is missing visibility level: ' + test[ i ]
);
}
},
/**
* The initialization method gives us the option to use existing objects
* for each level of visibility rather than creating new, empty ones.
*/
'Can initialize member object with existing objects': function()
{
var pub = { foo: 'bar' },
prot = { bar: 'baz' },
priv = { baz: 'foo' },
members = this.sut.initMembers( pub, prot, priv ),
test = {
'public': pub,
'protected': prot,
'private': priv,
}
;
// ensure we can initialize the values of each visibility level
for ( vis in test )
{
this.assertStrictEqual( test[ vis ], members[ vis ],
"Visibility level '" + vis + "' cannot be initialized"
);
}
},
/**
* The various members should be copied only to the interface specified by
* their access modifiers (public, protected, or private).
*/
'Members are only accessible via their respective interfaces': function()
{
var _self = this,
tests = [ 'public', 'protected', 'private' ];
for ( i in tests )
{
_self.basicVisPropTest( tests[ i ] );
_self.basicVisMethodTest( tests[ i ] );
};
},
/**
* If no access modifier is provided, it should be assumed that the member
* is to be public. This also allows for more concise code should the
* developer with to omit unnecessary keywords.
*/
'Members will be declared public if access modifier is omitted': function()
{
var name_prop = 'prop', val_prop = 'foo',
name_method = 'method', val_method = function() {}
;
this.sut.buildProp( this.members, {}, name_prop, val_prop, {}, {} );
this.sut.buildMethod( this.members, {}, name_method, val_method,
{}, function() {}, 1, {}
);
this.assertStrictEqual(
this.members[ 'public' ][ name_prop ][ 0 ],
val_prop,
'Properties should be public by default'
);
this.assertStrictEqual(
this.members[ 'public' ][ name_method ],
val_method,
'Methods should be public by default'
);
},
'Only one access modifier may be used per property': function()
{
var _self = this;
this.multiVisFailureTest( function( name, keywords )
{
_self.sut.buildProp( _self.members, {}, name, 'baz', keywords, {} );
} );
},
'Only one access modifier may be used per method': function()
{
var _self = this;
this.multiVisFailureTest( function( name, keywords )
{
_self.sut.buildMethod(
_self.members, {}, name, function() {}, keywords, {}
);
} );
},
} );