2014-06-08 07:17:48 +02:00
|
|
|
/* vi: set ts=2:
|
|
|
|
+-------------------+
|
|
|
|
| | Enno Rehling <enno@eressea.de>
|
|
|
|
| Eressea PBEM host | Christian Schlittchen <corwin@amber.kn-bremen.de>
|
|
|
|
| (c) 1998 - 2004 | Katja Zedel <katze@felidae.kn-bremen.de>
|
|
|
|
| | Henning Peters <faroul@beyond.kn-bremen.de>
|
|
|
|
+-------------------+
|
|
|
|
|
|
|
|
This program may not be used, modified or distributed
|
|
|
|
without prior permission by the authors of Eressea.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <platform.h>
|
|
|
|
#include <kernel/config.h>
|
|
|
|
#include "jsonconf.h"
|
|
|
|
|
|
|
|
/* kernel includes */
|
|
|
|
#include "building.h"
|
2014-06-16 20:30:23 +02:00
|
|
|
#include "direction.h"
|
|
|
|
#include "keyword.h"
|
2014-06-08 07:17:48 +02:00
|
|
|
#include "equipment.h"
|
|
|
|
#include "item.h"
|
2014-06-09 18:54:48 +02:00
|
|
|
#include "messages.h"
|
2014-06-08 07:17:48 +02:00
|
|
|
#include "race.h"
|
|
|
|
#include "region.h"
|
|
|
|
#include "resources.h"
|
|
|
|
#include "ship.h"
|
|
|
|
#include "terrain.h"
|
|
|
|
#include "skill.h"
|
|
|
|
#include "spell.h"
|
|
|
|
#include "spellbook.h"
|
|
|
|
#include "calendar.h"
|
|
|
|
|
|
|
|
/* util includes */
|
|
|
|
#include <util/attrib.h>
|
|
|
|
#include <util/bsdstring.h>
|
|
|
|
#include <util/crmessage.h>
|
|
|
|
#include <util/functions.h>
|
|
|
|
#include <util/language.h>
|
|
|
|
#include <util/log.h>
|
|
|
|
#include <util/message.h>
|
|
|
|
#include <util/nrmessage.h>
|
|
|
|
#include <util/xml.h>
|
|
|
|
|
|
|
|
/* external libraries */
|
|
|
|
#include <cJSON.h>
|
|
|
|
|
|
|
|
/* libc includes */
|
|
|
|
#include <assert.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <limits.h>
|
2014-06-13 17:04:06 +02:00
|
|
|
#include <stdlib.h>
|
2014-06-08 07:17:48 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
2014-06-13 17:04:06 +02:00
|
|
|
void json_construction(cJSON *json, construction **consp) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("building %s is not a json object: %d", json->string, json->type);
|
2014-06-13 17:04:06 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
construction * cons = (construction *)calloc(sizeof(construction), 1);
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
switch(child->type) {
|
|
|
|
case cJSON_Number:
|
|
|
|
if (strcmp(child->string, "maxsize")==0) {
|
|
|
|
cons->maxsize = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "reqsize")==0) {
|
|
|
|
cons->reqsize = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "minskill")==0) {
|
|
|
|
cons->minskill = child->valueint;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("building %s contains unknown attribute %s", json->string, child->string);
|
2014-06-13 17:04:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
*consp = cons;
|
|
|
|
}
|
|
|
|
|
2014-06-14 02:36:05 +02:00
|
|
|
void json_terrain(cJSON *json, terrain_type *ter) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("terrain %s is not a json object: %d", json->string, json->type);
|
2014-06-14 02:36:05 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("terrain %s contains unknown attribute %s", json->string, child->string);
|
2014-06-14 02:36:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void json_building(cJSON *json, building_type *bt) {
|
2014-06-13 07:14:07 +02:00
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("building %s is not a json object: %d", json->string, json->type);
|
2014-06-13 07:14:07 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
2014-06-14 02:36:05 +02:00
|
|
|
switch(child->type) {
|
|
|
|
case cJSON_Object:
|
|
|
|
if (strcmp(child->string, "construction")==0) {
|
|
|
|
json_construction(child, &bt->construction);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("building %s contains unknown attribute %s", json->string, child->string);
|
2014-06-14 02:36:05 +02:00
|
|
|
}
|
2014-06-13 07:14:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-13 17:04:06 +02:00
|
|
|
void json_ship(cJSON *json, ship_type *st) {
|
2014-06-13 07:14:07 +02:00
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("ship %s is not a json object: %d", json->string, json->type);
|
2014-06-13 07:14:07 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
2014-06-13 17:04:06 +02:00
|
|
|
switch(child->type) {
|
|
|
|
case cJSON_Object:
|
|
|
|
if (strcmp(child->string, "construction")==0) {
|
|
|
|
json_construction(child, &st->construction);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("ship %s contains unknown attribute %s", json->string, child->string);
|
2014-06-13 17:04:06 +02:00
|
|
|
}
|
2014-06-13 07:14:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-08 07:17:48 +02:00
|
|
|
void json_race(cJSON *json, race *rc) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("race %s is not a json object: %d", json->string, json->type);
|
2014-06-08 07:17:48 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
switch(child->type) {
|
|
|
|
case cJSON_String:
|
|
|
|
if (strcmp(child->string, "damage")==0) {
|
|
|
|
rc->def_damage = _strdup(child->valuestring);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case cJSON_Number:
|
|
|
|
if (strcmp(child->string, "magres")==0) {
|
|
|
|
rc->magres = (float)child->valuedouble;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "maxaura")==0) {
|
|
|
|
rc->maxaura = (float)child->valuedouble;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "regaura")==0) {
|
|
|
|
rc->regaura = (float)child->valuedouble;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "speed")==0) {
|
|
|
|
rc->speed = (float)child->valuedouble;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "recruitcost")==0) {
|
|
|
|
rc->recruitcost = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "maintenance")==0) {
|
|
|
|
rc->maintenance = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "weight")==0) {
|
|
|
|
rc->weight = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "capacity")==0) {
|
|
|
|
rc->capacity = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "hp")==0) {
|
|
|
|
rc->hitpoints = child->valueint;
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "ac")==0) {
|
|
|
|
rc->armor = child->valueint;
|
|
|
|
}
|
|
|
|
// TODO: studyspeed (orcs only)
|
|
|
|
break;
|
|
|
|
case cJSON_True: {
|
|
|
|
const char *flags[] = {
|
|
|
|
"playerrace", "killpeasants", "scarepeasants",
|
|
|
|
"cansteal", "moverandom", "cannotmove",
|
|
|
|
"learn", "fly", "swim", "walk", "nolearn",
|
|
|
|
"noteach", "horse", "desert",
|
|
|
|
"illusionary", "absorbpeasants", "noheal",
|
|
|
|
"noweapons", "shapeshift", "", "undead", "dragon",
|
|
|
|
"coastal", "", "cansail", 0
|
|
|
|
};
|
|
|
|
int i;
|
|
|
|
for(i=0;flags[i];++i) {
|
|
|
|
const char * flag = flags[i];
|
|
|
|
if (*flag && strcmp(child->string, flag)==0) {
|
|
|
|
rc->flags |= (1<<i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-14 02:36:05 +02:00
|
|
|
void json_terrains(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("terrains is not a json object: %d", json->type);
|
2014-06-14 02:36:05 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
2014-06-16 06:19:47 +02:00
|
|
|
json_terrain(child, get_or_create_terrain(child->string));
|
2014-06-14 02:36:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-13 07:14:07 +02:00
|
|
|
void json_buildings(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("buildings is not a json object: %d", json->type);
|
2014-06-13 07:14:07 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
json_building(child, bt_get_or_create(child->string));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void json_ships(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("ships is not a json object: %d", json->type);
|
2014-06-13 07:14:07 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
json_ship(child, st_get_or_create(child->string));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-16 07:17:08 +02:00
|
|
|
static void json_direction(cJSON *json, struct locale *lang) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("directions for locale `%s` not a json object: %d", locale_name(lang), json->type);
|
2014-06-16 07:17:08 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
direction_t dir = finddirection(child->string);
|
|
|
|
if (dir!=NODIRECTION) {
|
|
|
|
if (child->type==cJSON_String) {
|
|
|
|
init_direction(lang, dir, child->valuestring);
|
|
|
|
}
|
|
|
|
else if (child->type==cJSON_Array) {
|
|
|
|
cJSON *entry;
|
|
|
|
for (entry=child->child;entry;entry=entry->next) {
|
|
|
|
init_direction(lang, dir, entry->valuestring);
|
|
|
|
}
|
|
|
|
} else {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("invalid type %d for direction `%s`", child->type, child->string);
|
2014-06-16 07:17:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void json_directions(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("directions is not a json object: %d", json->type);
|
2014-06-16 07:17:08 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
struct locale * lang = get_or_create_locale(child->string);
|
|
|
|
json_direction(child, lang);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-16 20:30:23 +02:00
|
|
|
static void json_keyword(cJSON *json, struct locale *lang) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("keywords for locale `%s` not a json object: %d", locale_name(lang), json->type);
|
2014-06-16 20:30:23 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
keyword_t kwd = findkeyword(child->string);
|
|
|
|
if (kwd!=NOKEYWORD) {
|
|
|
|
if (child->type==cJSON_String) {
|
|
|
|
init_keyword(lang, kwd, child->valuestring);
|
|
|
|
}
|
|
|
|
else if (child->type==cJSON_Array) {
|
|
|
|
cJSON *entry;
|
|
|
|
for (entry=child->child;entry;entry=entry->next) {
|
|
|
|
init_keyword(lang, kwd, entry->valuestring);
|
|
|
|
}
|
|
|
|
} else {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("invalid type %d for keyword `%s`", child->type, child->string);
|
2014-06-16 20:30:23 +02:00
|
|
|
}
|
2014-06-17 04:12:55 +02:00
|
|
|
} else {
|
|
|
|
log_error_n("unknown keyword `%s` for locale `%s`", child->string, locale_name(lang));
|
2014-06-16 20:30:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void json_keywords(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("keywords is not a json object: %d", json->type);
|
2014-06-16 20:30:23 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
struct locale * lang = get_or_create_locale(child->string);
|
|
|
|
json_keyword(child, lang);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-08 07:17:48 +02:00
|
|
|
void json_races(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("races is not a json object: %d", json->type);
|
2014-06-08 07:17:48 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (child=json->child;child;child=child->next) {
|
2014-06-13 07:14:07 +02:00
|
|
|
json_race(child, rc_get_or_create(child->string));
|
2014-06-08 07:17:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void json_config(cJSON *json) {
|
|
|
|
cJSON *child;
|
|
|
|
if (json->type!=cJSON_Object) {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("config is not a json object: %d", json->type);
|
2014-06-08 07:17:48 +02:00
|
|
|
return;
|
|
|
|
}
|
2014-06-14 02:36:05 +02:00
|
|
|
for (child=json->child;child;child=child->next) {
|
|
|
|
if (strcmp(child->string, "races")==0) {
|
|
|
|
json_races(child);
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "ships")==0) {
|
|
|
|
json_ships(child);
|
|
|
|
}
|
2014-06-16 07:17:08 +02:00
|
|
|
else if (strcmp(child->string, "directions")==0) {
|
|
|
|
json_directions(child);
|
|
|
|
}
|
2014-06-16 20:30:23 +02:00
|
|
|
else if (strcmp(child->string, "keywords")==0) {
|
|
|
|
json_keywords(child);
|
|
|
|
}
|
2014-06-14 02:36:05 +02:00
|
|
|
else if (strcmp(child->string, "buildings")==0) {
|
|
|
|
json_buildings(child);
|
|
|
|
}
|
|
|
|
else if (strcmp(child->string, "terrains")==0) {
|
|
|
|
json_terrains(child);
|
|
|
|
} else {
|
2014-06-17 04:12:55 +02:00
|
|
|
log_error_n("config contains unknown attribute %s", child->string);
|
2014-06-14 02:36:05 +02:00
|
|
|
}
|
2014-06-13 07:14:07 +02:00
|
|
|
}
|
2014-06-08 07:17:48 +02:00
|
|
|
}
|
2014-06-09 18:54:48 +02:00
|
|
|
|