mirror of
https://github.com/mumble-voip/mumble.git
synced 2025-10-26 11:19:16 +00:00
248 lines
4.5 KiB
C++
248 lines
4.5 KiB
C++
// Copyright 2005-2018 The Mumble Developers. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license
|
|
// that can be found in the LICENSE file at the root of the
|
|
// Mumble source tree or at <https://www.mumble.info/LICENSE>.
|
|
|
|
#include "murmur_pch.h"
|
|
|
|
#include "Group.h"
|
|
|
|
#include "Channel.h"
|
|
#include "User.h"
|
|
#ifdef MURMUR
|
|
#include "ServerUser.h"
|
|
#endif
|
|
|
|
Group::Group(Channel *assoc, const QString &name) {
|
|
c = assoc;
|
|
bInherit = true;
|
|
bInheritable = true;
|
|
qsName = name;
|
|
if (c)
|
|
c->qhGroups[name] = this;
|
|
}
|
|
|
|
#ifdef MURMUR
|
|
|
|
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);
|
|
if (g) {
|
|
if ((p != c) && ! g->bInheritable)
|
|
break;
|
|
s.push(g);
|
|
if (! g->bInherit)
|
|
break;
|
|
}
|
|
p = p->cParent;
|
|
}
|
|
|
|
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 = p->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, ServerUser *pl) {
|
|
Channel *p;
|
|
Channel *c;
|
|
Group *g;
|
|
|
|
bool m = false;
|
|
bool invert = false;
|
|
bool token = false;
|
|
bool hash = false;
|
|
c = curChan;
|
|
|
|
while (true) {
|
|
if (name.isEmpty())
|
|
return false;
|
|
|
|
if (name.startsWith(QChar::fromLatin1('!'))) {
|
|
invert = true;
|
|
name = name.remove(0,1);
|
|
continue;
|
|
}
|
|
|
|
if (name.startsWith(QChar::fromLatin1('~'))) {
|
|
c = aclChan;
|
|
name = name.remove(0,1);
|
|
continue;
|
|
}
|
|
|
|
if (name.startsWith(QChar::fromLatin1('#'))) {
|
|
token = true;
|
|
name = name.remove(0,1);
|
|
continue;
|
|
}
|
|
if (name.startsWith(QChar::fromLatin1('$'))) {
|
|
hash = true;
|
|
name = name.remove(0,1);
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if (token)
|
|
m = pl->qslAccessTokens.contains(name, Qt::CaseInsensitive);
|
|
else if (hash)
|
|
m = pl->qsHash == name;
|
|
else if (name == QLatin1String("none"))
|
|
m = false;
|
|
else if (name == QLatin1String("all"))
|
|
m = true;
|
|
else if (name == QLatin1String("auth"))
|
|
m = (pl->iId >= 0);
|
|
else if (name == QLatin1String("strong"))
|
|
m = pl->bVerified;
|
|
else if (name == QLatin1String("in"))
|
|
m = (pl->cChannel == c);
|
|
else if (name == QLatin1String("out"))
|
|
m = !(pl->cChannel == c);
|
|
else if (name == QLatin1String("sub")
|
|
|| name.startsWith(QLatin1String("sub,"))) {
|
|
|
|
name = name.remove(0,4);
|
|
int mindesc = 1;
|
|
int maxdesc = 1000;
|
|
int minpath = 0;
|
|
QStringList args = name.split(QLatin1String(","));
|
|
switch (args.count()) {
|
|
default:
|
|
case 3:
|
|
maxdesc = args[2].isEmpty() ? maxdesc : args[2].toInt();
|
|
case 2:
|
|
mindesc = args[1].isEmpty() ? mindesc : args[1].toInt();
|
|
case 1:
|
|
minpath = args[0].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) || g->qsTemporary.contains(pl->iId) || g->qsTemporary.contains(- static_cast<int>(pl->uiSession)))
|
|
m = true;
|
|
if (g->qsRemove.contains(pl->iId))
|
|
m = false;
|
|
}
|
|
}
|
|
return invert ? !m : m;
|
|
}
|
|
|
|
#endif
|