server/src/common/modules/gmcmd.c

543 lines
14 KiB
C
Raw Normal View History

2001-04-26 19:41:06 +02:00
/* vi: set ts=2:
+-------------------+ Christian Schlittchen <corwin@amber.kn-bremen.de>
| | Enno Rehling <enno@eressea-pbem.de>
| Eressea PBEM host | Katja Zedel <katze@felidae.kn-bremen.de>
| (c) 1998 - 2001 | Henning Peters <faroul@beyond.kn-bremen.de>
| | Ingo Wilken <Ingo.Wilken@informatik.uni-oldenburg.de>
+-------------------+ Stefan Reich <reich@halbling.de>
This program may not be used, modified or distributed
without prior permission by the authors of Eressea.
2001-05-11 22:19:22 +02:00
*/
2001-01-28 11:10:22 +01:00
#include <config.h>
#include <eressea.h>
#include "gmcmd.h"
2001-04-26 19:41:06 +02:00
#include "command.h"
2001-01-28 11:10:22 +01:00
2001-05-11 22:19:22 +02:00
/* misc includes */
2001-01-28 11:10:22 +01:00
#include <items/demonseye.h>
#include <attributes/key.h>
2001-05-11 22:19:22 +02:00
#include <triggers/gate.h>
2001-01-28 11:10:22 +01:00
/* kernel includes */
2001-05-11 22:19:22 +02:00
#include <building.h>
2001-01-28 11:10:22 +01:00
#include <faction.h>
#include <item.h>
#include <message.h>
#include <plane.h>
#include <region.h>
#include <terrain.h>
#include <unit.h>
2001-01-28 11:10:22 +01:00
/* util includes */
2001-05-11 22:19:22 +02:00
#include <attrib.h>
#include <base36.h>
2001-05-11 22:19:22 +02:00
#include <event.h>
2001-01-28 11:10:22 +01:00
#include <umlaut.h>
/* libc includes */
#include <stdlib.h>
#include <string.h>
/**
** at_permissions
**/
static void
write_permissions(const attrib * a, FILE * F)
{
a_write(F, (attrib*)a->data.v);
}
static int
read_permissions(attrib * a, FILE * F)
{
attrib ** p_a = (attrib**)&a->data.v;
a_read(F, p_a);
return 1;
}
struct attrib_type at_permissions = {
2001-01-28 11:10:22 +01:00
"GM:permissions",
NULL, NULL, NULL,
write_permissions, read_permissions,
ATF_UNIQUE
};
attrib *
make_atpermissions(void)
{
return a_new(&at_permissions);
}
2001-01-28 11:10:22 +01:00
/**
** GM: CREATE <number> <itemtype>
**/
static void
write_gmcreate(const attrib * a, FILE * F)
{
const item_type * itype = (const item_type *)a->data.v;
assert(itype);
2001-01-28 11:10:22 +01:00
fprintf(F, "%s", resourcename(itype->rtype, 0));
}
static int
read_gmcreate(attrib * a, FILE * F)
{
char zText[32];
const item_type ** p_itype = (const item_type **)&a->data.v;
fscanf(F, "%s", zText);
*p_itype = it_find(zText);
assert(*p_itype);
2001-01-28 11:10:22 +01:00
return 1;
}
/* at_gmcreate specifies that the owner can create items of a particular type */
attrib_type at_gmcreate = {
2001-01-28 11:10:22 +01:00
"GM:create",
NULL, NULL, NULL,
write_gmcreate, read_gmcreate
};
attrib *
2001-01-28 11:10:22 +01:00
make_atgmcreate(const struct item_type * itype)
{
attrib * a = a_new(&at_gmcreate);
a->data.v = (void*)itype;
return a;
}
static void
2001-04-26 19:41:06 +02:00
gm_create(const char * str, void * data, const char * cmd)
2001-01-28 11:10:22 +01:00
{
2001-04-26 19:41:06 +02:00
unit * u = (unit*)data;
2001-01-28 11:10:22 +01:00
int i;
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (permissions) permissions = (attrib*)permissions->data.v;
if (!permissions) return;
2001-01-28 11:10:22 +01:00
i = atoi(igetstrtoken(str));
if (i>0) {
char * iname = getstrtoken();
const item_type * itype = finditemtype(iname, u->faction->locale);
if (itype==NULL) {
mistake(u, cmd, "Unbekannter Gegenstand.\n", 0);
} else {
attrib * a = a_find(permissions, &at_gmcreate);
while (a && a->data.v!=(void*)itype) a=a->nexttype;
if (a) i_change(&u->items, itype, i);
else mistake(u, cmd, "Diesen Gegenstand darf deine Partei nicht machen.\n", 0);
}
2001-01-28 11:10:22 +01:00
}
}
static boolean
has_permission(const attrib * permissions, unsigned int key)
{
return (find_key((attrib*)permissions->data.v, key) ||
find_key((attrib*)permissions->data.v, atoi36("master")));
}
2001-05-11 22:19:22 +02:00
/**
** GM: GATE <id> <x> <y>
** requires: permission-key "gmgate"
**/
static void
gm_gate(const char * str, void * data, const char * cmd)
{
unit * u = (unit*)data;
const struct plane * p = rplane(u->region);
int id = atoi36(igetstrtoken(str));
int x = rel_to_abs(p, u->faction, atoi(getstrtoken()), 0);
int y = rel_to_abs(p, u->faction, atoi(getstrtoken()), 1);
region * r = findregion(x, y);
building * b = findbuilding(id);
if (b==NULL || r==NULL || p!=rplane(b->region) || p!=rplane(r)) {
mistake(u, cmd, "Dieses Geb<65>ude kann die Einheit nicht umwandeln.\n", 0);
return;
} else {
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (permissions && has_permission(permissions, atoi36("gmgate"))) {
2001-05-11 22:19:22 +02:00
add_trigger(&u->attribs, "timer", trigger_gate(b, r));
}
}
}
2001-01-28 11:10:22 +01:00
/**
** GM: TERRAFORM <terrain> <x> <y>
** requires: permission-key "gmterf"
2001-01-28 11:10:22 +01:00
**/
static void
2001-04-26 19:41:06 +02:00
gm_terraform(const char * str, void * data, const char * cmd)
2001-01-28 11:10:22 +01:00
{
2001-04-26 19:41:06 +02:00
unit * u = (unit*)data;
const struct plane * p = rplane(u->region);
int x = rel_to_abs(p, u->faction, atoi(igetstrtoken(str)), 0);
int y = rel_to_abs(p, u->faction, atoi(getstrtoken()), 1);
const char * c = getstrtoken();
region * r = findregion(x, y);
2001-01-28 11:10:22 +01:00
terrain_t t;
if (r==NULL || p!=rplane(r)) {
2001-05-11 22:19:22 +02:00
mistake(u, cmd, "Diese Region kann die Einheit nicht umwandeln.\n", 0);
return;
} else {
2001-01-28 11:10:22 +01:00
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (!permissions || !has_permission(permissions, atoi36("gmterf"))) return;
2001-01-28 11:10:22 +01:00
}
for (t=0;t!=MAXTERRAINS;++t) {
if (!strcasecmp(locale_string(u->faction->locale, terrain[t].name), c)) break;
}
if (t!=MAXTERRAINS) terraform(r, t);
}
/**
** GM: TELEPORT <unit> <x> <y>
** requires: permission-key "gmtele"
**/
static void
2001-04-26 19:41:06 +02:00
gm_teleport(const char * str, void * data, const char * cmd)
{
2001-04-26 19:41:06 +02:00
unit * u = (unit*)data;
const struct plane * p = rplane(u->region);
unit * to = findunit(atoi36(igetstrtoken(str)));
int x = rel_to_abs(p, u->faction, atoi(getstrtoken()), 0);
int y = rel_to_abs(p, u->faction, atoi(getstrtoken()), 1);
region * r = findregion(x, y);
if (r==NULL || p!=rplane(r)) {
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "In diese Region kann die Einheit nicht teleportieren.\n", 0);
} else if (to==NULL) {
mistake(u, cmd, "Die Einheit wurde nicht gefunden.\n", 0);
} else if (rplane(to->region)!=rplane(r) && !ucontact(to, u)) {
mistake(u, cmd, "Die Einheit hat uns nicht kontaktiert.\n", 0);
} else {
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (!permissions || !has_permission(permissions, atoi36("gmtele"))) {
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Unzureichende Rechte f<>r diesen Befehl.\n", 0);
}
else move_unit(to, r, NULL);
}
}
/**
** GM: MESSAGE <x> <y> <string>
** requires: permission-key "gmmsgr"
**/
static void
gm_messageregion(const char * str, void * data, const char * cmd)
{
unit * u = (unit*)data;
const struct plane * p = rplane(u->region);
int x = rel_to_abs(p, u->faction, atoi(igetstrtoken(str)), 0);
int y = rel_to_abs(p, u->faction, atoi(getstrtoken()), 1);
const char * msg = getstrtoken();
region * r = findregion(x, y);
if (r==NULL || p!=rplane(r)) {
mistake(u, cmd, "In diese Region kann keine Nachricht gesandt werden.\n", 0);
} else {
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (!permissions || !has_permission(permissions, atoi36("gmmsgr"))) {
mistake(u, cmd, "Unzureichende Rechte f<>r diesen Befehl.\n", 0);
}
else {
add_message(&r->msgs, msg_message("msg_event", "string", msg));
}
}
}
/**
** GM: GIVE <unit> <int> <itemtype>
** requires: permission-key "gmgive"
**/
static void
2001-04-26 19:41:06 +02:00
gm_give(const char * str, void * data, const char * cmd)
{
2001-04-26 19:41:06 +02:00
unit * u = (unit*)data;
unit * to = findunit(atoi36(igetstrtoken(str)));
int num = atoi(getstrtoken());
const item_type * itype = finditemtype(getstrtoken(), u->faction->locale);
if (to==NULL || rplane(to->region) != rplane(u->region)) {
/* unknown or in another plane */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Die Einheit wurde nicht gefunden.\n", 0);
} else if (itype==NULL || i_get(u->items, itype)==0) {
/* unknown or not enough */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "So einen Gegenstand hat die Einheit nicht.\n", 0);
} else {
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (!permissions || !has_permission(permissions, atoi36("gmgive"))) {
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Unzureichende Rechte f<>r diesen Befehl.\n", 0);
}
else {
int i = i_get(u->items, itype);
if (i<num) num=i;
if (num) {
i_change(&u->items, itype, -num);
i_change(&to->items, itype, num);
}
}
}
}
/**
** GM: TAKE <unit> <int> <itemtype>
** requires: permission-key "gmtake"
**/
static void
2001-04-26 19:41:06 +02:00
gm_take(const char * str, void * data, const char * cmd)
{
2001-04-26 19:41:06 +02:00
unit * u = (unit*)data;
unit * to = findunit(atoi36(igetstrtoken(str)));
int num = atoi(getstrtoken());
const item_type * itype = finditemtype(getstrtoken(), u->faction->locale);
if (to==NULL || rplane(to->region) != rplane(u->region)) {
/* unknown or in another plane */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Die Einheit wurde nicht gefunden.\n", 0);
} else if (itype==NULL || i_get(to->items, itype)==0) {
/* unknown or not enough */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "So einen Gegenstand hat die Einheit nicht.\n", 0);
} else {
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (!permissions || !has_permission(permissions, atoi36("gmtake"))) {
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Unzureichende Rechte f<>r diesen Befehl.\n", 0);
}
else {
int i = i_get(to->items, itype);
if (i<num) num=i;
if (num) {
i_change(&to->items, itype, -num);
i_change(&u->items, itype, num);
}
}
}
}
/**
** GM: SKILL <unit> <skill> <tage>
** requires: permission-key "gmskil"
**/
static void
2001-04-26 19:41:06 +02:00
gm_skill(const char * str, void * data, const char * cmd)
{
2001-04-26 19:41:06 +02:00
unit * u = (unit*)data;
unit * to = findunit(atoi36(igetstrtoken(str)));
skill_t skill = findskill(getstrtoken(), u->faction->locale);
int num = atoi(getstrtoken());
if (to==NULL || rplane(to->region) != rplane(u->region)) {
/* unknown or in another plane */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Die Einheit wurde nicht gefunden.\n", 0);
} else if (skill==NOSKILL || skill==SK_MAGIC || skill==SK_ALCHEMY) {
/* unknown or not enough */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Dieses Talent ist unbekannt, oder kann nicht erh<72>ht werden.\n", 0);
} else if (num<0 || num>5000) {
/* sanity check failed */
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Der gew<65>hlte Wert ist nicht zugelassen.\n", 0);
} else {
/* checking permissions */
attrib * permissions = a_find(u->faction->attribs, &at_permissions);
if (!permissions || !has_permission(permissions, atoi36("gmskil"))) {
2001-04-26 19:41:06 +02:00
mistake(u, cmd, "Unzureichende Rechte f<>r diesen Befehl.\n", 0);
}
else {
2001-04-26 19:41:06 +02:00
set_skill(to, skill, num*to->number);
}
}
2001-01-28 11:10:22 +01:00
}
2001-04-26 19:41:06 +02:00
static struct command * g_cmds;
static tnode g_keys;
2001-01-30 21:02:06 +01:00
static void
2001-04-26 19:41:06 +02:00
gm_command(const char * str, void * data, const char * cmd)
2001-01-28 11:10:22 +01:00
{
2001-04-26 19:41:06 +02:00
do_command(&g_keys, data, str);
2001-01-28 11:10:22 +01:00
}
void
init_gmcmd(void)
{
at_register(&at_gmcreate);
at_register(&at_permissions);
2001-04-26 19:41:06 +02:00
add_command(&g_keys, &g_cmds, "gm", &gm_command);
add_command(&g_keys, &g_cmds, "terraform", &gm_terraform);
add_command(&g_keys, &g_cmds, "create", &gm_create);
2001-05-11 22:19:22 +02:00
add_command(&g_keys, &g_cmds, "gate", &gm_gate);
2001-04-26 19:41:06 +02:00
add_command(&g_keys, &g_cmds, "give", &gm_give);
add_command(&g_keys, &g_cmds, "take", &gm_take);
add_command(&g_keys, &g_cmds, "teleport", &gm_teleport);
add_command(&g_keys, &g_cmds, "skill", &gm_skill);
add_command(&g_keys, &g_cmds, "message", &gm_messageregion);
2001-01-28 11:10:22 +01:00
}
2001-01-30 21:02:06 +01:00
/*
* execute gm-commands for all units in the game
*/
void
gmcommands(void)
{
region ** rp = &regions;
while (*rp) {
region * r = *rp;
unit **up = &r->units;
while (*up) {
unit * u = *up;
strlist * order;
for (order = u->orders; order; order = order->next)
if (igetkeyword(order->s, u->faction->locale) == K_GM) {
2001-04-26 19:41:06 +02:00
do_command(&g_keys, u, order->s);
}
2001-01-30 21:02:06 +01:00
if (u==*up) up = &u->next;
}
if (*rp==r) rp = &r->next;
}
}
#define EXTENSION 10000
faction *
gm_addquest(const char * email, const char * name, int radius, unsigned int flags)
{
plane * p;
attrib * a;
unit * u;
region * center;
boolean invalid = false;
int minx, miny, maxx, maxy, cx, cy;
int x, y, i;
faction * f = calloc(1, sizeof(faction));
/* GM faction */
a_add(&f->attribs, make_key(atoi36("quest")));
f->banner = strdup("Questenpartei");
f->passw = strdup(itoa36(rand()));
f->email = strdup(email);
f->name = strdup("Questenpartei");
f->race = RC_TEMPLATE;
f->age = 0;
f->lastorders = turn;
f->alive = true;
f->locale = find_locale("de");
f->options = want(O_COMPRESS) | want(O_REPORT) | want(O_COMPUTER) | want(O_ADRESSEN);
{
faction * xist;
int i = atoi36("gm00")-1;
do {
xist = findfaction(++i);
} while (xist);
f->no = i;
addlist(&factions, f);
}
/* GM playfield */
do {
minx = (rand() % (2*EXTENSION)) - EXTENSION;
miny = (rand() % (2*EXTENSION)) - EXTENSION;
for (x=0;!invalid && x<=radius*2;++x) {
for (y=0;!invalid && y<=radius*2;++y) {
region * r = findregion(minx+x, miny+y);
if (r) invalid = true;
}
}
} while (invalid);
maxx = minx+2*radius; cx = minx+radius;
maxy = miny+2*radius; cy = miny+radius;
p = create_new_plane(rand(), name, minx, maxx, miny, maxy, flags);
center = new_region(cx, cy);
for (x=0;x<=2*radius;++x) {
int y;
for (y=0;y<=2*radius;++y) {
region * r = findregion(minx+x, miny+y);
if (!r) r = new_region(minx+x, miny+y);
freset(r, RF_ENCOUNTER);
r->planep = p;
if (distance(r, center)==radius) {
terraform(r, T_FIREWALL);
} else if (r==center) {
terraform(r, T_PLAIN);
} else {
terraform(r, T_OCEAN);
}
}
}
/* generic permissions */
a = a_add(&f->attribs, a_new(&at_permissions));
a_add((attrib**)&a->data.v, make_key(atoi36("gmterf")));
2001-05-11 22:19:22 +02:00
a_add((attrib**)&a->data.v, make_key(atoi36("gmgate")));
a_add((attrib**)&a->data.v, make_key(atoi36("gmtele")));
a_add((attrib**)&a->data.v, make_key(atoi36("gmgive")));
a_add((attrib**)&a->data.v, make_key(atoi36("gmskil")));
a_add((attrib**)&a->data.v, make_key(atoi36("gmtake")));
a_add((attrib**)&a->data.v, make_key(atoi36("gmmsgr")));
a_add((attrib**)&a->data.v, make_atgmcreate(resource2item(r_silver)));
for (i=0;i<=I_INCENSE;++i) {
a_add((attrib**)&a->data.v, make_atgmcreate(olditemtype[i]));
}
for (i=I_GREATSWORD;i!=I_KEKS;++i) {
a_add((attrib**)&a->data.v, make_atgmcreate(olditemtype[i]));
}
/* one initial unit */
u = createunit(center, f, 1, RC_TEMPLATE);
u->irace = RC_GNOME;
u->number = 1;
set_string(&u->name, "Questenmeister");
u->irace = RC_GOBLIN;
return f;
}
#ifdef TEST_GM_COMMANDS
void
setup_gm_faction(void)
{
int i = atoi36("gms")-1;
faction * f = factions;
unit * newunit;
region * r = regions;
attrib * a;
do {
f = findfaction(++i);
} while (f);
f = (faction *) calloc(1, sizeof(faction));
f->no = i;
set_string(&f->email, "gms@eressea-pbem.de");
set_string(&f->passw, "geheim");
set_string(&f->name, "GMs");
f->alive = 1;
f->options |= (1 << O_REPORT);
f->options |= (1 << O_COMPUTER);
addlist(&factions, f);
a = a_add(&f->attribs, a_new(&at_permissions));
a_add((attrib**)&a->data.v, make_atgmcreate(&it_demonseye));
a_add((attrib**)&a->data.v, make_key(atoi36("gmtf")));
while (r && !r->land) r=r->next;
newunit = createunit(r, f, 1, RC_DAEMON);
set_string(&newunit->name, "Flamdring, Gott des Feuers");
set_money(newunit, 100);
fset(newunit, FL_ISNEW);
}
#endif