mirror of
https://github.com/mumble-voip/mumble.git
synced 2025-10-26 11:19:16 +00:00
git-svn-id: https://mumble.svn.sourceforge.net/svnroot/mumble/trunk@270 05730e5d-ab1b-0410-a4ac-84af385074fa
241 lines
5.0 KiB
C++
241 lines
5.0 KiB
C++
/* Copyright (C) 2005, Thorvald Natvig <thorvald@natvig.com>
|
|
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions
|
|
are met:
|
|
|
|
- Redistributions of source code must retain the above copyright notice,
|
|
this list of conditions and the following disclaimer.
|
|
- Redistributions in binary form must reproduce the above copyright notice,
|
|
this list of conditions and the following disclaimer in the documentation
|
|
and/or other materials provided with the distribution.
|
|
- Neither the name of the Mumble Developers nor the names of its
|
|
contributors may be used to endorse or promote products derived from this
|
|
software without specific prior written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR
|
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <QStack>
|
|
#include <QStringList>
|
|
|
|
#include "Group.h"
|
|
#include "Channel.h"
|
|
#include "Player.h"
|
|
|
|
Group::Group(Channel *assoc, QString name) {
|
|
c = assoc;
|
|
bInherit = true;
|
|
bInheritable = true;
|
|
qsName = name;
|
|
c->qhGroups[name] = this;
|
|
}
|
|
|
|
QSet<int> Group::members() {
|
|
QStack<Group *> s;
|
|
QSet<int> m;
|
|
Channel *p;
|
|
Group *g;
|
|
int i;
|
|
|
|
p = c;
|
|
while (p) {
|
|
g = p->qhGroups.value(qsName);
|
|
p = p->cParent;
|
|
|
|
if (g) {
|
|
if ((p != c) && ! g->bInheritable)
|
|
break;
|
|
s.push(g);
|
|
if (! g->bInherit)
|
|
break;
|
|
}
|
|
}
|
|
|
|
while (! s.isEmpty()) {
|
|
g = s.pop();
|
|
foreach(i, g->qsAdd)
|
|
m.insert(i);
|
|
foreach(i, g->qsRemove)
|
|
m.remove(i);
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
Group *Group::getGroup(Channel *chan, QString name) {
|
|
Channel *p = chan;
|
|
while (p) {
|
|
Group *g = chan->qhGroups.value(name);
|
|
if (g) {
|
|
if (chan == p)
|
|
return g;
|
|
else if (g->bInheritable)
|
|
return g;
|
|
else
|
|
return NULL;
|
|
}
|
|
p = p->cParent;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
QSet<QString> Group::groupNames(Channel *chan) {
|
|
QStack<Channel *> s;
|
|
QSet<QString> m;
|
|
Channel *c = chan;
|
|
Group *g;
|
|
|
|
while (c) {
|
|
s.push(c);
|
|
c = c->cParent;
|
|
}
|
|
|
|
while (! s.isEmpty()) {
|
|
c = s.pop();
|
|
foreach(g, c->qhGroups) {
|
|
if ((chan != c) && (! g->bInheritable))
|
|
m.remove(g->qsName);
|
|
else
|
|
m.insert(g->qsName);
|
|
}
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
#define RET_FALSE (invert ? true : false)
|
|
#define RET_TRUE (invert ? false : true)
|
|
|
|
bool Group::isMember(Channel *curChan, Channel *aclChan, QString name, Player *pl) {
|
|
Channel *p;
|
|
Channel *c;
|
|
Group *g;
|
|
|
|
bool m = false;
|
|
bool invert = false;
|
|
c = curChan;
|
|
|
|
if (name.isEmpty())
|
|
return false;
|
|
|
|
|
|
if (name.startsWith('~')) {
|
|
invert = true;
|
|
name = name.remove(0,1);
|
|
}
|
|
|
|
if (name.startsWith('~')) {
|
|
c = aclChan;
|
|
name = name.remove(0,1);
|
|
}
|
|
|
|
if (name == "none")
|
|
m = false;
|
|
else if (name == "all")
|
|
m = true;
|
|
else if (name == "auth")
|
|
m = (pl->iId >= 0);
|
|
else if (name == "in")
|
|
m = (pl->cChannel == c);
|
|
else if (name == "out")
|
|
m = !(pl->cChannel == c);
|
|
else if (name.startsWith("sub")) {
|
|
name = name.remove(0,4);
|
|
int mindesc = 1;
|
|
int maxdesc = 1000;
|
|
int minpath = 0;
|
|
QStringList args = name.split(",");
|
|
switch(args.count()) {
|
|
default:
|
|
case 3:
|
|
maxdesc = args[1].isEmpty() ? maxdesc : args[2].toInt();
|
|
case 2:
|
|
mindesc = args[0].isEmpty() ? mindesc : args[1].toInt();
|
|
case 1:
|
|
minpath = args[2].isEmpty() ? minpath : args[0].toInt();
|
|
case 0:
|
|
break;
|
|
}
|
|
|
|
Channel *home = pl->cChannel;
|
|
QList<Channel *> playerChain;
|
|
QList<Channel *> groupChain;
|
|
|
|
p = home;
|
|
while (p) {
|
|
playerChain.prepend(p);
|
|
p = p->cParent;
|
|
}
|
|
|
|
p = curChan;
|
|
while (p) {
|
|
groupChain.prepend(p);
|
|
p = p->cParent;
|
|
}
|
|
|
|
int cofs = groupChain.indexOf(c);
|
|
Q_ASSERT( cofs != -1);
|
|
|
|
cofs += minpath;
|
|
|
|
if (cofs >= groupChain.count()) {
|
|
return RET_FALSE;
|
|
} else if (cofs < 0) {
|
|
cofs = 0;
|
|
}
|
|
|
|
Channel *needed = groupChain[cofs];
|
|
if (playerChain.indexOf(needed) == -1) {
|
|
return RET_FALSE;
|
|
}
|
|
|
|
int mindepth = cofs + mindesc;
|
|
int maxdepth = cofs + maxdesc;
|
|
|
|
int pdepth = playerChain.count() - 1;
|
|
|
|
m = (pdepth >= mindepth) && (pdepth <= maxdepth);
|
|
} else {
|
|
|
|
QStack<Group *> s;
|
|
|
|
p = c;
|
|
|
|
while (p) {
|
|
g = p->qhGroups.value(name);
|
|
|
|
if (g) {
|
|
if ((p != c) && ! g->bInheritable)
|
|
break;
|
|
s.push(g);
|
|
if (! g->bInherit)
|
|
break;
|
|
}
|
|
|
|
p = p->cParent;
|
|
}
|
|
|
|
while (! s.isEmpty()) {
|
|
g = s.pop();
|
|
if (g->qsAdd.contains(pl->iId))
|
|
m = true;
|
|
if (g->qsRemove.contains(pl->iId))
|
|
m = false;
|
|
}
|
|
}
|
|
return invert ? !m : m;
|
|
}
|