Skip to content

Commit e65368c

Browse files
committedAug 23, 2011
Fix group membership by scratch variables for Visitor. Fixes bug #12195. Kudos to trex for the base patch. Also fix the same problem in the IP based group membership.
1 parent 8ab2dcc commit e65368c

File tree

4 files changed

+71
-31
lines changed

4 files changed

+71
-31
lines changed
 

‎docs/changelog/7.x.x.txt

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
7.10.23
22
- fixed #12225: Stock asset, multiple instances on a page
33
- fixed #12229: Indexed thingy data has gateway url prepended to it
4+
- fixed #12195: Visitor group by scratch membership shared among all Visitors (Dale Trexel)
45

56
7.10.22
67
- rfe #12223: Add date type to content profiling (metadata)

‎lib/WebGUI/Group.pm

+24-9
Original file line numberDiff line numberDiff line change
@@ -1091,22 +1091,29 @@ Membership will always be false if no IpFilter has been set
10911091
10921092
id of the user to check for membership
10931093
1094+
=head3 sessionId
1095+
1096+
id of the session to check for user data. If no sessionId is passed in, then the
1097+
group's session will be used to find one.
1098+
1099+
10941100
=cut
10951101

10961102
sub hasIpUser {
10971103
my $self = shift;
1098-
my $userId = shift;
10991104
my $session = $self->session;
1105+
my $userId = shift;
1106+
my $userSessionId = shift || $session->getId;
11001107

11011108
my $IpFilter = $self->ipFilter();
11021109
return 0 unless ($IpFilter && $userId);
11031110

11041111
$IpFilter =~ s/\s//g;
11051112
my @filters = split /;/, $IpFilter;
11061113

1107-
my @ips = $session->db->buildArray(
1108-
q{ select lastIP from userSession where expires > ? and userId = ? }
1109-
,[ time(), $userId ]
1114+
my @ips = $session->db->buildArray(
1115+
q{ select lastIP from userSession where expires > ? and userId = ? and sessionId=?}
1116+
,[ time(), $userId, $userSessionId, ]
11101117
);
11111118

11121119
foreach my $ip (@ips) {
@@ -1207,7 +1214,7 @@ sub hasLDAPUser {
12071214

12081215
#-------------------------------------------------------------------
12091216

1210-
=head2 hasScratchUser ( userId )
1217+
=head2 hasScratchUser ( userId, [ $sessionId ] )
12111218
12121219
Determine if the user passed in is a member of this group via session scratch
12131220
variable settings and this group's scratchFilter.
@@ -1218,12 +1225,18 @@ If no scratchFilter has been set for this group, membership will always be false
12181225
12191226
id of the user to check for membership
12201227
1228+
=head3 sessionId
1229+
1230+
id of the session for the user being checked for membership. If no sessionId is passed in, then the
1231+
group's session will be used to find one.
1232+
12211233
=cut
12221234

12231235
sub hasScratchUser {
12241236
my $self = shift;
1225-
my $userId = shift;
12261237
my $session = $self->session;
1238+
my $userId = shift;
1239+
my $userSessionId = shift || $self->session;
12271240

12281241
my $scratchFilter = $self->scratchFilter();
12291242
return 0 unless ($scratchFilter && $userId);
@@ -1232,7 +1245,7 @@ sub hasScratchUser {
12321245
my @filters = split /;/, $scratchFilter;
12331246

12341247
my @scratchClauses = ();
1235-
my @scratchPlaceholders = ( $userId, time() );
1248+
my @scratchPlaceholders = ( $userSessionId, $userId, time() );
12361249
foreach my $filter (@filters) {
12371250
my ($name, $value) = split /=/, $filter;
12381251
push @scratchClauses, "(s.name=? AND s.value=?)";
@@ -1246,6 +1259,7 @@ sub hasScratchUser {
12461259
from
12471260
userSession u, userSessionScratch s
12481261
where
1262+
u.sessionId = ? AND
12491263
u.sessionId=s.sessionId AND
12501264
u.userId = ? AND
12511265
u.expires > ? AND
@@ -1273,6 +1287,7 @@ sub hasUser {
12731287
my $self = shift;
12741288
my $session = $self->session;
12751289
my $user = shift || WebGUI::User->new($session,3); #Check the admin account if no user is passed in
1290+
my $uSessionId = $user->session->getId;
12761291
my $gid = $self->getId;
12771292
my $db = $session->db;
12781293

@@ -1359,9 +1374,9 @@ sub hasUser {
13591374
my $groupToCheck = __PACKAGE__->new($session,$groupIdInGroup);
13601375
### Check the 'has' method for each of the 'other' group methods available for this user
13611376
### perform checks in a least -> most expensive manner. If we find the user, stow the cache and return true
1362-
if( $groupToCheck->hasIpUser($uid)
1377+
if( $groupToCheck->hasIpUser($uid, $uSessionId)
13631378
|| $groupToCheck->hasKarmaUser($uid)
1364-
|| $groupToCheck->hasScratchUser($uid)
1379+
|| $groupToCheck->hasScratchUser($uid, $uSessionId)
13651380
|| $groupToCheck->hasDatabaseUser($uid)
13661381
|| $groupToCheck->hasLDAPUser($uid)
13671382
) {

‎t/Group.t

+12-10
Original file line numberDiff line numberDiff line change
@@ -640,15 +640,17 @@ WebGUI::Test->addToCleanup(@sessionBank);
640640

641641
#isInGroup test
642642
foreach my $scratchTest (@scratchTests) {
643-
is($scratchTest->{user}->isInGroup($gS->getId), $scratchTest->{expect}, $scratchTest->{comment});
643+
is($scratchTest->{user}->isInGroup($gS->getId), $scratchTest->{expect}, $scratchTest->{comment});
644644
}
645645

646646
WebGUI::Cache->new($session, $gS->getId)->delete(); ##Delete cached key for testing
647647
$session->stow->delete("isInGroup");
648648

649649
#hasScratchUser test
650-
foreach my $scratchTest (@scratchTests) {
651-
is($gS->hasScratchUser($scratchTest->{user}->getId), $scratchTest->{expect}, $scratchTest->{comment}." - hasScratchUser");
650+
foreach my $idx (0..$#scratchTests) {
651+
my $scratchTest = $scratchTests[$idx];
652+
my $sessionId = $sessionBank[$idx]->getId;
653+
is($gS->hasScratchUser($scratchTest->{user}->getId, $sessionId), $scratchTest->{expect}, $scratchTest->{comment}." - hasScratchUser");
652654
}
653655

654656

@@ -666,7 +668,7 @@ cmp_bag(
666668

667669
{ ##Add scope to force cleanup
668670

669-
note "Checking for user Visitor session leak";
671+
note "Checking for user Visitor session leak with scratch";
670672

671673
my $remoteSession = WebGUI::Test->newSession;
672674
$remoteSession->user({userId => 1});
@@ -681,13 +683,12 @@ cmp_bag(
681683
my $localSession = WebGUI::Test->newSession;
682684
WebGUI::Test->addToCleanup($localScratchGroup, $remoteSession, $localSession);
683685
$localSession->user({userId => 1});
684-
$remoteSession->scratch->set('local','ok');
686+
$localSession->scratch->set('local','ok');
685687
$localScratchGroup->clearCaches;
686688

687689
ok $localSession->user->isInGroup($localScratchGroup->getId), 'Local Visitor is in the scratch group';
688690

689691
$remoteSession->stow->delete('isInGroup');
690-
$localScratchGroup->clearCaches;
691692
ok !$remoteSession->user->isInGroup($localScratchGroup->getId), 'Remote Visitor is not in the scratch group, even though a different Visitor passed';
692693

693694
}
@@ -710,8 +711,9 @@ foreach my $idx (0..$#ipTests) {
710711
##Name this user for convenience
711712
$tcps[$idx]->username("tcp$idx");
712713

713-
##Assign this user to this test to be fetched later
714-
$ipTests[$idx]->{user} = $tcps[$idx];
714+
##Assign this user and session to this test to be fetched later
715+
$ipTests[$idx]->{user} = $tcps[$idx];
716+
$ipTests[$idx]->{session} = $sessionBank[$idx];
715717
}
716718
WebGUI::Test->addToCleanup(@tcps);
717719
WebGUI::Test->addToCleanup(@sessionBank);
@@ -734,7 +736,7 @@ cmp_bag(
734736
);
735737

736738
is_deeply(
737-
[ (map { $gI->hasIpUser($_->{user}->getId) } @ipTests) ],
739+
[ (map { $gI->hasIpUser($_->{user}->getId, $_->{session}->getId) } @ipTests) ],
738740
[ (map { $_->{expect} } @ipTests) ],
739741
'hasIpUsers for group with IP filter'
740742
);
@@ -745,7 +747,7 @@ foreach my $ipTest (@ipTests) {
745747

746748
{ ##Add scope to force cleanup
747749

748-
note "Checking for user Visitor session leak";
750+
note "Checking for user Visitor session leak via IP address";
749751

750752
$ENV{REMOTE_ADDR} = '191.168.1.1';
751753
my $remoteSession = WebGUI::Test->newSession;

‎t/Group/group_scratch.t

+34-12
Original file line numberDiff line numberDiff line change
@@ -20,27 +20,49 @@ use WebGUI::Group;
2020

2121
#----------------------------------------------------------------------------
2222
# Init
23-
my $session = WebGUI::Test->session;
24-
23+
my $session1 = WebGUI::Test->session;
24+
my $session2 = WebGUI::Session->open(WebGUI::Test::root, WebGUI::Test::file);
2525

2626
#----------------------------------------------------------------------------
2727
# Tests
28+
### Updates by DRT test that group membership is restricted by user session
29+
### ...specifically for Visitors to have separate scratch group memberships
2830

29-
plan tests => 5; # Increment this number for each test you create
31+
plan tests => 14; # Increment this number for each test you create
3032

31-
my $group = WebGUI::Group->new($session, 'new');
33+
my $group = WebGUI::Group->new($session1, 'new');
3234
WebGUI::Test->addToCleanup($group);
3335
$group->scratchFilter("itchy=test_value");
3436
is( $group->scratchFilter(), "itchy=test_value",'Group->scratchFilter is properly set and retrieved');
3537
$group->groupCacheTimeout(0);
3638
is( $group->groupCacheTimeout(), 0, 'set groupCacheTimeout to 0');
3739

38-
$session->user({userId => 1});
39-
ok( !$session->user->isInGroup($group->getId), 'Visitor is NOT in the group BEFORE scratch value is set');
40-
$session->scratch->set('itchy', 'test_value');
41-
is ($group->hasScratchUser($session->user->getId), 1, 'Group->hasScratchUser correctly returns 1 immediately after scratch is set');
40+
$session1->user({userId => 1});
41+
$session2->user({userId => 1});
42+
43+
### Test group membership before scratch is set
44+
### NOTE: test hasScratchUser first, because isInGroup sets stow & cache
45+
is ($group->hasScratchUser($session1->user->getId,$session1->user->session->getId), 0, 'Group->hasScratchUser correctly returns 0 for Visitor 1 before scratch is set');
46+
is ($group->hasScratchUser($session2->user->getId,$session2->user->session->getId), 0, 'Group->hasScratchUser correctly returns 0 for Visitor 2 before scratch is set');
47+
ok( !$session1->user->isInGroup($group->getId), 'user1->isInGroup correctly returns 0 before scratch is set');
48+
ok( !$session2->user->isInGroup($group->getId), 'user2->isInGroup correctly returns 0 before scratch is set');
49+
50+
### Test group membership after scratch is set
51+
### Clear stow, which is volatile, to simulate new page view
52+
$session1->stow->deleteAll;
53+
$session2->stow->deleteAll;
54+
$session1->scratch->set('itchy', 'test_value');
55+
is ($group->hasScratchUser($session1->user->getId,$session1->user->session->getId), 1, 'Group->hasScratchUser correctly returns 1 for Visitor 1 after scratch for Visitor 1 is set');
56+
is ($group->hasScratchUser($session2->user->getId,$session2->user->session->getId), 0, 'Group->hasScratchUser correctly returns 0 for Visitor 2 after scratch for Visitor 1 is set');
57+
ok( $session1->user->isInGroup($group->getId), 'user1->isInGroup correctly returns 1 after scratch for Visitor 1 is set');
58+
ok( !$session2->user->isInGroup($group->getId), 'user2->isInGroup correctly returns 0 after scratch for Visitor 1 is set');
4259

43-
##Simulate another page view by clearing stow, which is volatile
44-
$session->stow->deleteAll;
45-
$session->scratch->delete('itchy');
46-
is ($group->hasScratchUser($session->user->getId), 0, 'after clearing scratch, user is not in the group any longer');
60+
### Test group membership after scratch is deleted
61+
### Clear stow, which is volatile, to simulate new page view
62+
$session1->stow->deleteAll;
63+
$session2->stow->deleteAll;
64+
$session1->scratch->delete('itchy');
65+
is ($group->hasScratchUser($session1->user->getId,$session1->user->session->getId), 0, 'Group->hasScratchUser for Visitor 1 correctly returns 0 after clearing scratch for Visitor 1');
66+
is ($group->hasScratchUser($session2->user->getId,$session2->user->session->getId), 0, 'Group->hasScratchUser for Visitor 2 correctly returns 0 after clearing scratch for Visitor 1');
67+
ok( !$session1->user->isInGroup($group->getId), 'user1->isInGroup correctly returns 0 after scratch for Visitor 1 is deleted');
68+
ok( !$session2->user->isInGroup($group->getId), 'user2->isInGroup correctly returns 0 after scratch for Visitor 1 is deleted');

0 commit comments

Comments
 (0)
Please sign in to comment.