main.cpp 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638
  1. /*
  2. Space Construct v0.6-dev
  3. Beanzilla@21:4/110 on FsxNet
  4. Beanzilla@637:1/110 on HappyNet
  5. */
  6. #define VERSION_MAJOR 0
  7. #define VERSION_MINOR 6
  8. #ifndef VERSION_TYPE
  9. #define VERSION_TYPE "dev"
  10. #endif
  11. #define PATH_MAX 256
  12. #define PATH_SEP "/"
  13. //#include <MagiDoor.h> // Sorry Apam... as much as I would like to keep using your stuff it's just not fully functional
  14. #include <OpenDoor.h> // Now using odoors for c++
  15. #include <sqlite3.h>
  16. #if defined(_MSC_VER) || defined(WIN32)
  17. #define snprintf _snprintf
  18. #define strcasecmp _stricmp
  19. #include <winsock2.h>
  20. #ifndef _MSC_VER
  21. #define _MSC_VER 1
  22. #endif
  23. #else
  24. #include <arpa/inet.h>
  25. #endif
  26. // Standard C
  27. #include <time.h>
  28. #include <stdio.h>
  29. #include <string.h>
  30. #include <stdlib.h>
  31. #include <stdarg.h>
  32. #include <ctype.h>
  33. #include <sys/types.h>
  34. #include <sys/stat.h>
  35. #include <unistd.h> // stat
  36. #include <string>
  37. // sc.log ODoors has logging so why not use it!
  38. int inuse = 0; // Are any other copies of us running? (We are a single user door!)
  39. int debug = 0; // Are we in debug mode?
  40. // Please change it for your system!
  41. int allowDev = 1; // Allow "Beanzilla" to not need the password.
  42. std::string sysop_pass = "spaceISbig"; // Case sensitive!
  43. // LOGGING with file output
  44. #include "zf_log.h"
  45. FILE *g_log_file;
  46. static void file_output_callback(const zf_log_message *msg, void *arg) {
  47. (void)arg;
  48. *msg->p = '\n';
  49. fwrite(msg->buf, msg->p - msg->buf + 1, 1, g_log_file);
  50. fflush(g_log_file);
  51. }
  52. static void file_output_close(void) { fclose(g_log_file); }
  53. static int file_output_open(const char *const log_path) {
  54. g_log_file = fopen(log_path, "a");
  55. if (!g_log_file) {
  56. ZF_LOGW("Failed to open log file %s", log_path);
  57. return 0;
  58. }
  59. atexit(file_output_close);
  60. zf_log_set_output_v(ZF_LOG_PUT_STD, 0, file_output_callback);
  61. return 1;
  62. }
  63. void log_flush(void) { fflush(g_log_file); }
  64. // END LOGGING
  65. // Randrange
  66. int randrange(int min, int max){
  67. return min + rand() / (RAND_MAX / (max - min + 1) + 1);
  68. }
  69. // Moved to User class
  70. #include "user.h"
  71. // DateStamp Structure to provide date difference
  72. typedef struct dateTamp {
  73. int year; // YYYY, or number of years difference
  74. int month; // MM, or number of months difference
  75. int day; // DD, or number of days difference.
  76. int age; // Taking down to aproximate day count.
  77. } dT; // YYYYMMDD it in int form
  78. int dateStamp() {
  79. // YYYYMMDD \o/ In a INT so we can store it and compare against it!
  80. struct tm *time_now;
  81. time_t timen;
  82. int result = 0;
  83. timen = time(NULL);
  84. time_now = localtime(&timen);
  85. result += ((time_now->tm_year + 1900) * 10000);
  86. result += (time_now->tm_mon * 100) + 100;
  87. result += time_now->tm_mday;
  88. return result;
  89. }
  90. int compareDate(int dt) {
  91. // Returns integer of difference, from now.
  92. struct tm *time_now;
  93. time_t timen;
  94. int now = 0;
  95. timen = time(NULL);
  96. time_now = localtime(&timen);
  97. now += ((time_now->tm_year + 1900) * 10000);
  98. now += (time_now->tm_mon * 100) + 100;
  99. now += time_now->tm_mday;
  100. return (now - dt); // 10000 = 1 Year, 100 = 1 Month, 1 = 1 Day
  101. }
  102. dT fromDate(int diff) {
  103. // Reverse process so we can compare exact days or get the structure back out too.
  104. dT result;
  105. result.year = diff / 10000;
  106. result.month = (diff / 100) % 100;
  107. result.day = diff % 100;
  108. result.age = 0;
  109. // Process Aprox day count
  110. result.age += (365 * result.year);
  111. result.age += (30 * result.month);
  112. result.age += result.day;
  113. return result;
  114. }
  115. int db_test() {
  116. // Displays all users in the database
  117. sqlite3 *db;
  118. sqlite3_stmt *stmt;
  119. char sqlbuffer[256];
  120. char strbuffer[256];
  121. int sizeofdb = 0;
  122. int rc = sqlite3_open("spaceconstruct.db3", &db);
  123. if(rc) { // Did we do a successful open?
  124. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  125. sqlite3_close(db);
  126. od_exit(-1, FALSE);
  127. }
  128. sqlite3_busy_timeout(db, 5000);
  129. // DB open
  130. strcpy(sqlbuffer, "SELECT * FROM user;");
  131. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer), &stmt, NULL);
  132. while(sqlite3_step(stmt) == SQLITE_ROW) {
  133. ZF_LOGV("%s=%d %s=%s %s=%s %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d %s=%d",
  134. sqlite3_column_name(stmt, 0), // int, uid
  135. sqlite3_column_int(stmt, 0),
  136. sqlite3_column_name(stmt, 1), // text, nick
  137. sqlite3_column_text(stmt, 1),
  138. sqlite3_column_name(stmt, 2), // text, real
  139. sqlite3_column_text(stmt, 2),
  140. sqlite3_column_name(stmt, 3), // int, experiece
  141. sqlite3_column_int(stmt, 3),
  142. sqlite3_column_name(stmt, 4), // int, metal
  143. sqlite3_column_int(stmt, 4),
  144. sqlite3_column_name(stmt, 5), // int, fuel
  145. sqlite3_column_int(stmt, 5),
  146. sqlite3_column_name(stmt, 6), // int, gun
  147. sqlite3_column_int(stmt, 6),
  148. sqlite3_column_name(stmt, 7), // int, armor
  149. sqlite3_column_int(stmt, 7),
  150. sqlite3_column_name(stmt, 8), // int, shield
  151. sqlite3_column_int(stmt, 8),
  152. sqlite3_column_name(stmt, 9), // int, armorpoints
  153. sqlite3_column_int(stmt, 9),
  154. sqlite3_column_name(stmt, 10), // int, shieldpoints
  155. sqlite3_column_int(stmt, 10),
  156. sqlite3_column_name(stmt, 11), // int, hitpoints
  157. sqlite3_column_int(stmt, 11),
  158. sqlite3_column_name(stmt, 12), // int, shieldsup
  159. sqlite3_column_int(stmt, 12),
  160. sqlite3_column_name(stmt, 13), // int, laston
  161. sqlite3_column_int(stmt, 13)
  162. );
  163. sizeofdb += 1;
  164. } // Clean up database
  165. ZF_LOGV("There are %d users", sizeofdb);
  166. sqlite3_finalize(stmt);
  167. sqlite3_close(db);
  168. return sizeofdb;
  169. }
  170. int check_lock() {
  171. // Checks stats of lock, (1 = Lock is in effect, 0 = No Lock established)
  172. struct stat s;
  173. if (stat("lock.flg", &s) == 0) {
  174. return 1;
  175. } else {
  176. return 0;
  177. }
  178. }
  179. int grab_lock() {
  180. // Attempt to grab lock, (0 = Already gotten, 1 = Success, -1 = Error)
  181. int valid = check_lock();
  182. FILE *fhandle;
  183. if (valid == 0) {
  184. fhandle = fopen("lock.flg", "w");
  185. if(!fhandle) {
  186. ZF_LOGF("Unable to make lock.flg!");
  187. ZF_LOGF("Something went wrong perhaps we don't have permissions?");
  188. return -1;
  189. }
  190. fprintf(fhandle, "I am in use already!\n");
  191. fclose(fhandle);
  192. ZF_LOGV("Lock Established");
  193. return 1;
  194. } else {
  195. ZF_LOGW("Lock already established!");
  196. return 0;
  197. }
  198. }
  199. void rel_lock() {
  200. // Attempt to release lock
  201. int valid = check_lock();
  202. if (valid == 1) {
  203. if (unlink("lock.flg") != 0) {
  204. ZF_LOGF("Unable to release lock.flg!");
  205. ZF_LOGF("Something went wrong! Players might not be able to play now!");
  206. }
  207. } else {
  208. ZF_LOGW("Lock already released!");
  209. }
  210. }
  211. void log_drop() {
  212. // Spits out info from Drop File:
  213. ZF_LOGV("Name=%s Alias=%s TimeLeft=%d SecLevel=%d Location=%s Node=%d Sysop=%s",
  214. od_control.user_name,
  215. od_control.user_handle,
  216. od_control.user_timelimit,
  217. od_control.user_security,
  218. od_control.user_location,
  219. od_control.od_node,
  220. od_control.sysop_name
  221. );
  222. }
  223. void paws() {
  224. // Aaah, DRY
  225. od_printf("`white`Press any key to continue...");
  226. od_get_key(true);
  227. od_printf("\r\n");
  228. }
  229. int yesNo() {
  230. char ch;
  231. int done = 0;
  232. od_printf("`bright black`(`bright white`Y`bright black`)`white`es `bright black`(`bright red`N`bright black`)`red`o`white`");
  233. ch = od_get_answer("YyNn\r");
  234. od_printf("\r\n");
  235. switch(tolower(ch)) {
  236. case 'y':
  237. done = 1;
  238. break;
  239. case '\r':
  240. case 'n':
  241. break;
  242. }
  243. return done;
  244. }
  245. int locate_player(char name[]) {
  246. // returns user id for given real name and 0 for no record found
  247. sqlite3 *db;
  248. sqlite3_stmt *stmt;
  249. char sqlbuffer[256];
  250. int result = 0;
  251. int rc = sqlite3_open("spaceconstruct.db3", &db);
  252. if(rc) { // Did we do a successful open?
  253. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  254. sqlite3_close(db);
  255. od_exit(-1, FALSE);
  256. }
  257. sqlite3_busy_timeout(db, 5000);
  258. //od_printf("realname = '%s'\r\n", name);
  259. // Locating user with given name
  260. strcpy(sqlbuffer, "SELECT * from user where real=? COLLATE NOCASE;");
  261. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  262. sqlite3_bind_text(stmt, 1, name, strlen(name), SQLITE_STATIC);
  263. rc = sqlite3_step(stmt);
  264. if (rc == SQLITE_ROW) {
  265. // Return user id
  266. result = sqlite3_column_int(stmt, 0);
  267. } else {
  268. // User not found
  269. ZF_LOGW("Unable to locate user %s", name);
  270. result = 0;
  271. } // Clean Up, return results
  272. sqlite3_finalize(stmt);
  273. sqlite3_close(db);
  274. return result;
  275. }
  276. void check_database() {
  277. // If the table users does not exist make it.
  278. sqlite3 *db;
  279. sqlite3_stmt *stmt;
  280. char sqlbuffer[256];
  281. int rc = sqlite3_open("spaceconstruct.db3", &db);
  282. if(rc) {
  283. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  284. sqlite3_close(db);
  285. od_exit(-1, FALSE);
  286. }
  287. sqlite3_busy_timeout(db, 5000);
  288. strcpy(sqlbuffer, "SELECT COUNT(*) FROM user;");
  289. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  290. rc = sqlite3_step(stmt);
  291. if(rc == SQLITE_ROW) {
  292. // Good
  293. } else {
  294. // Bad, Create table
  295. char *errmsg;
  296. rc = sqlite3_exec(db, "CREATE TABLE IF NOT EXISTS \"user\" (`uid` INTEGER PRIMARY KEY AUTOINCREMENT,`nick` TEXT,`real` TEXT,`experience` INTEGER,`metal` INTEGER,`fuel` INTEGER,`gun` INTEGER,`armor` INTEGER,`shield` INTEGER,`armorpoints` INTEGER,`shieldpoints` INTEGER,`hitpoints` INTEGER,`shieldsup` INTEGER,`laston` INTEGER);", NULL, NULL, &errmsg);
  297. ZF_LOGW("Users table did not exist created");
  298. }
  299. sqlite3_finalize(stmt);
  300. sqlite3_close(db);
  301. }
  302. User load_player(int uuid) {
  303. // Returns a player Structure from database
  304. sqlite3 *db;
  305. sqlite3_stmt *stmt;
  306. char sqlbuffer[256];
  307. //User *result;
  308. int rc = sqlite3_open("spaceconstruct.db3", &db);
  309. if(rc) { // Did we do a successful open?
  310. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  311. sqlite3_close(db);
  312. od_exit(-1, FALSE);
  313. }
  314. sqlite3_busy_timeout(db, 5000);
  315. strcpy(sqlbuffer, "SELECT * FROM user WHERE uid=?;");
  316. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  317. sqlite3_bind_int(stmt, 1, uuid);
  318. rc = sqlite3_step(stmt);
  319. if (rc == SQLITE_ROW) {
  320. std::string val1( (const char *)sqlite3_column_text(stmt, 1) );
  321. std::string val2( (const char *)sqlite3_column_text(stmt, 2) );
  322. User result(sqlite3_column_int(stmt, 0), val1, val2, sqlite3_column_int(stmt, 3), sqlite3_column_int(stmt, 13), sqlite3_column_int(stmt, 7), sqlite3_column_int(stmt, 9), sqlite3_column_int(stmt, 8), sqlite3_column_int(stmt, 10), sqlite3_column_int(stmt, 12), sqlite3_column_int(stmt, 11), sqlite3_column_int(stmt, 6), sqlite3_column_int(stmt, 4), sqlite3_column_int(stmt, 5));
  323. sqlite3_finalize(stmt);
  324. sqlite3_close(db);
  325. return result;
  326. } else {
  327. sqlite3_finalize(stmt);
  328. sqlite3_close(db);
  329. User result;
  330. return result;
  331. }
  332. sqlite3_finalize(stmt);
  333. sqlite3_close(db);
  334. }
  335. User highest_player() {
  336. // Used to find if someone may have fought the Unknown One and won the game already!
  337. int done = 0;
  338. User test;
  339. int high = 0;
  340. int high_exp = 0;
  341. int id = 1;
  342. while(!done) {
  343. test = load_player(id);
  344. if(test.get_uid() != 0) {
  345. if(test.get_experience() > high_exp) {
  346. // Update highest player id and experience
  347. high_exp = test.get_experience();
  348. high = id;
  349. }
  350. } else {
  351. done = 1;
  352. }
  353. id += 1;
  354. } // Ok highest experiencing player found!
  355. test = load_player(high);
  356. return test;
  357. }
  358. void update_player(User data) {
  359. sqlite3 *db;
  360. sqlite3_stmt *stmt;
  361. char sqlbuffer[1024];
  362. int rc = sqlite3_open("spaceconstruct.db3", &db);
  363. if(rc) { // Did we do a successful open?
  364. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  365. sqlite3_close(db);
  366. od_exit(-1, FALSE);
  367. }
  368. sqlite3_busy_timeout(db, 5000);
  369. // Bad, don't do this... opens to SQL injection!
  370. //snprintf(sqlbuffer, 1024, "UPDATE user SET nick = '%s', experience = %d, metal = %d, fuel = %d, guns = %d, armors = %d, shields = %d, armorpoints = %d, shieldpoints = %d, hitpoints = %d WHERE uid=%d;",
  371. // data.nick, data.experience, data.metal, data.fuel, data.guns, data.armors, data.shields, data.armorpoints, data.shieldpoints, data.hitpoints, data.uid);
  372. strcpy(sqlbuffer, "UPDATE user SET nick=?, experience=?, metal=?, fuel=?, gun=?, armor=?, shield=?, armorpoints=?, shieldpoints=?, hitpoints=?, shieldsup=?, laston=? WHERE uid=?;");
  373. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  374. // Bind All data values
  375. sqlite3_bind_text(stmt, 1, data.get_nick().c_str(), data.get_nick().length(), SQLITE_STATIC);
  376. sqlite3_bind_int(stmt, 2, data.get_experience());
  377. sqlite3_bind_int(stmt, 3, data.get_metal());
  378. sqlite3_bind_int(stmt, 4, data.get_fuel());
  379. sqlite3_bind_int(stmt, 5, data.get_gun());
  380. sqlite3_bind_int(stmt, 6, data.get_armor());
  381. sqlite3_bind_int(stmt, 7, data.get_shield());
  382. sqlite3_bind_int(stmt, 8, data.get_armorpoints());
  383. sqlite3_bind_int(stmt, 9, data.get_shieldpoints());
  384. sqlite3_bind_int(stmt, 10, data.get_hitpoints());
  385. sqlite3_bind_int(stmt, 11, data.get_shieldsup());
  386. sqlite3_bind_int(stmt, 12, data.get_laston());
  387. sqlite3_bind_int(stmt, 13, data.get_uid());
  388. // Execute
  389. rc = sqlite3_step(stmt);
  390. if(rc != SQLITE_DONE) {
  391. ZF_LOGF("Failed updating player %s (%d)", sqlite3_errmsg(db), rc);
  392. sqlite3_finalize(stmt);
  393. sqlite3_close(db);
  394. od_exit(-1, FALSE);
  395. }
  396. sqlite3_finalize(stmt);
  397. sqlite3_close(db);
  398. }
  399. int create_player(User data) {
  400. sqlite3 *db;
  401. sqlite3_stmt *stmt;
  402. char sqlbuffer[1024];
  403. int rc = sqlite3_open("spaceconstruct.db3", &db);
  404. if(rc) { // Did we do a successful open?
  405. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  406. sqlite3_close(db);
  407. od_exit(-1, FALSE);
  408. }
  409. sqlite3_busy_timeout(db, 5000);
  410. strcpy(sqlbuffer, "INSERT INTO user (nick, real, experience, metal, fuel, gun, armor, shield, armorpoints, shieldpoints, hitpoints, shieldsup, laston) VALUES(?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?);");
  411. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  412. sqlite3_bind_text(stmt, 1, data.get_nick().c_str(), data.get_nick().length(), SQLITE_STATIC);
  413. sqlite3_bind_text(stmt, 2, data.get_real().c_str(), data.get_real().length(), SQLITE_STATIC);
  414. sqlite3_bind_int(stmt, 3, data.get_experience());
  415. sqlite3_bind_int(stmt, 4, data.get_metal());
  416. sqlite3_bind_int(stmt, 5, data.get_fuel());
  417. sqlite3_bind_int(stmt, 6, data.get_gun());
  418. sqlite3_bind_int(stmt, 7, data.get_armor());
  419. sqlite3_bind_int(stmt, 8, data.get_shield());
  420. sqlite3_bind_int(stmt, 9, data.get_armorpoints());
  421. sqlite3_bind_int(stmt, 10, data.get_shieldpoints());
  422. sqlite3_bind_int(stmt, 11, data.get_hitpoints());
  423. sqlite3_bind_int(stmt, 12, data.get_shieldsup());
  424. sqlite3_bind_int(stmt, 13, data.get_laston());
  425. rc = sqlite3_step(stmt);
  426. if(rc != SQLITE_DONE) {
  427. ZF_LOGF("Failed inserting player %s (%d)", sqlite3_errmsg(db), rc);
  428. sqlite3_finalize(stmt);
  429. sqlite3_close(db);
  430. od_exit(-1, FALSE);
  431. }
  432. sqlite3_finalize(stmt);
  433. sqlite3_close(db);
  434. return 1; // Good
  435. }
  436. void delete_player(int uuid) {
  437. sqlite3 *db;
  438. sqlite3_stmt *stmt;
  439. char sqlbuffer[256];
  440. int rc = sqlite3_open("spaceconstruct.db3", &db);
  441. if(rc) { // Did we do a successful open?
  442. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  443. sqlite3_close(db);
  444. od_exit(-1, FALSE);
  445. }
  446. sqlite3_busy_timeout(db, 5000);
  447. strcpy(sqlbuffer, "DELETE FROM user WHERE uid=?;");
  448. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  449. sqlite3_bind_int(stmt, 1, uuid);
  450. rc = sqlite3_step(stmt);
  451. if (rc != SQLITE_DONE){
  452. ZF_LOGF("Unable to locate user with id=%d got %s (%d)", uuid, sqlite3_errmsg(db), rc);
  453. sqlite3_finalize(stmt);
  454. sqlite3_close(db);
  455. od_exit(-1, FALSE);
  456. }
  457. sqlite3_finalize(stmt);
  458. sqlite3_close(db);
  459. }
  460. void delete_players() {
  461. sqlite3 *db;
  462. sqlite3_stmt *stmt;
  463. char sqlbuffer[256];
  464. int rc = sqlite3_open("spaceconstruct.db3", &db);
  465. if(rc) { // Did we do a successful open?
  466. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  467. sqlite3_close(db);
  468. od_exit(-1, FALSE);
  469. }
  470. sqlite3_busy_timeout(db, 5000);
  471. strcpy(sqlbuffer, "DELETE FROM user;");
  472. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  473. rc = sqlite3_step(stmt);
  474. if (rc != SQLITE_DONE){
  475. ZF_LOGF("Delete_players 1/2, Unexpected error %s (%d)", sqlite3_errmsg(db), rc);
  476. sqlite3_finalize(stmt);
  477. sqlite3_close(db);
  478. od_exit(-1, FALSE);
  479. }
  480. strcpy(sqlbuffer, "UPDATE sqlite_sequence SET seq = 0 WHERE name='user';");
  481. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  482. rc = sqlite3_step(stmt);
  483. if (rc != SQLITE_DONE){
  484. ZF_LOGF("Delete_players 2/2, Unexpected error %s (%d)", sqlite3_errmsg(db), rc);
  485. sqlite3_finalize(stmt);
  486. sqlite3_close(db);
  487. od_exit(-1, FALSE);
  488. }
  489. sqlite3_finalize(stmt);
  490. sqlite3_close(db);
  491. }
  492. int display_all_players() {
  493. sqlite3 *db;
  494. sqlite3_stmt *stmt;
  495. char sqlbuffer[1024];
  496. int rc = sqlite3_open("spaceconstruct.db3", &db);
  497. if(rc) { // Did we do a successful open?
  498. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  499. sqlite3_close(db);
  500. od_exit(-1, FALSE);
  501. }
  502. sqlite3_busy_timeout(db, 5000);
  503. strcpy(sqlbuffer, "SELECT * from user ORDER BY experience DESC, nick DESC;");
  504. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  505. int total = 0;
  506. od_printf("`bright green`___ The Players ___\r\n");
  507. while(sqlite3_step(stmt) == SQLITE_ROW) {
  508. od_printf("`bright white` %s (%d)\r\n", sqlite3_column_text(stmt, 1), sqlite3_column_int(stmt, 3));
  509. total += 1;
  510. }
  511. if(total > 1) {
  512. od_printf("`green`There are a total of `bright green`%d`green` players\r\n", total);
  513. } else if(total == 0) {
  514. od_printf("`green`There are a total of `bright green`%d`green` players\r\n", total);
  515. } else {
  516. od_printf("`green`There are a total of `bright green`%d`green` player\r\n", total);
  517. }
  518. sqlite3_finalize(stmt);
  519. sqlite3_close(db);
  520. return total;
  521. }
  522. int display_all_opponents(int us) {
  523. sqlite3 *db;
  524. sqlite3_stmt *stmt;
  525. char sqlbuffer[1024];
  526. int rc = sqlite3_open("spaceconstruct.db3", &db);
  527. if(rc) { // Did we do a successful open?
  528. ZF_LOGF("Failed opening database %s (%d)", sqlite3_errmsg(db), rc);
  529. sqlite3_close(db);
  530. od_exit(-1, FALSE);
  531. }
  532. sqlite3_busy_timeout(db, 5000);
  533. strcpy(sqlbuffer, "SELECT * from user ORDER BY experience DESC, nick DESC;");
  534. sqlite3_prepare_v2(db, sqlbuffer, strlen(sqlbuffer) + 1, &stmt, NULL);
  535. int total = 1;
  536. od_printf("`bright green`___ Target List ___\r\n");
  537. while(sqlite3_step(stmt) == SQLITE_ROW) {
  538. if(total != us) {
  539. od_printf("`bright black`(`bright white`%d`bright black`) `bright white`%s (%d)\r\n", total, sqlite3_column_text(stmt, 1), sqlite3_column_int(stmt, 3));
  540. }
  541. total += 1;
  542. }
  543. total -= 1;
  544. sqlite3_finalize(stmt);
  545. sqlite3_close(db);
  546. return total;
  547. }
  548. User build_menu(User my) {
  549. int done = 0;
  550. char ch;
  551. if(my.get_metal() != 0) {
  552. while(!done) {
  553. od_clr_scr();
  554. od_send_file("ansis/sc_build.ans");
  555. // Paint over sc_build.ans file
  556. od_set_cursor(3, 15);
  557. od_printf("%5d", my.calcGun());
  558. od_set_cursor(4, 15);
  559. od_printf("%5d", my.calcArmor());
  560. od_set_cursor(5, 15);
  561. od_printf("%5d", my.calcShield());
  562. od_set_cursor(9, 1);
  563. // Resuming normal operations
  564. od_printf("`bright yellow`You have %8d metal\r\n", my.get_metal());
  565. ch = od_get_answer("GgAaSsCcRr\r");
  566. switch(tolower(ch)) {
  567. case 'g':
  568. if(my.get_metal() >= my.calcGun()) {
  569. my.rmMetal(my.calcGun());
  570. my.addGun(1);
  571. od_printf("`bright green`Added a new gun!\r\n`white`");
  572. paws();
  573. } else {
  574. od_printf("`bright red`You don't have the %d metal for this!\r\n`white`", my.calcGun());
  575. paws();
  576. }
  577. break;
  578. case 'a':
  579. if(my.get_metal() >= my.calcArmor()) {
  580. my.rmMetal(my.calcArmor());
  581. my.addArmor(1);
  582. od_printf("`bright green`Added a new armor!\r\n`white`");
  583. paws();
  584. } else {
  585. od_printf("`bright red`You don't have the %d metal for this!\r\n`white`", my.calcArmor());
  586. paws();
  587. }
  588. break;
  589. case 's':
  590. if(my.get_metal() >= my.calcShield()) {
  591. my.rmMetal(my.calcShield());
  592. my.addShield(1);
  593. od_printf("`bright green`Added a new shield!\r\n`white`");
  594. paws();
  595. } else {
  596. od_printf("`bright red`You don't have the %d metal for this!\r\n`white`", my.calcShield());
  597. paws();
  598. }
  599. break;
  600. case '\r':
  601. case 'c':
  602. done = 1;
  603. break;
  604. case 'r':
  605. if(my.get_experience() >= 25) {
  606. if(my.get_metal() >= 25) {
  607. my.rmEXP(25);
  608. my.rmMetal(25);
  609. my.addFuel(5);
  610. od_printf("`bright green`Made 5 fuel using 25 experience and 25 metal!\r\n");
  611. paws();
  612. } else {
  613. od_printf("`bright red`You need 25 experience and 25 metal to make fuel!\r\n");
  614. paws();
  615. }
  616. } else {
  617. od_printf("`bright red`You need 25 experience and 25 metal to make fuel!\r\n");
  618. paws();
  619. }
  620. break;
  621. }
  622. }
  623. update_player(my);
  624. return my;
  625. } else {
  626. return my;
  627. }
  628. }
  629. User pvp_menu(User my, User targ) {
  630. int done = 0;
  631. int playerWon = 0;
  632. char ch;
  633. // We need to set the targ's hit points and such to max and enable us to "restore" them
  634. int temp_hp = targ.get_hitpoints();
  635. int temp_ap = targ.get_armorpoints();
  636. int temp_sp = targ.get_shieldpoints();
  637. int temp_su = targ.get_shieldsup();
  638. // Set Offline to max values
  639. int targ_dmg = (targ.get_gun() + 1); // Add bonus damage for offline
  640. targ.set_hitpoints(targ.calcHP());
  641. targ.set_armorpoints(targ.calcAP());
  642. targ.set_shieldpoints(targ.calcSP());
  643. targ.set_shieldsup(0);
  644. // Prep Combat values
  645. int fire = 0; // Fire Guns
  646. int flee = 0; // Run away
  647. int regen = 0; // Shields
  648. int test = 0;
  649. int test2 = 0;
  650. while(!done) {
  651. // Process Shields up
  652. if(temp_su != 0) {
  653. temp_su -= 1;
  654. }
  655. if(my.get_shieldsup() != 0) {
  656. my.rmSU(1);
  657. }
  658. // Process Crew repairing hitpoints
  659. if(targ.get_hitpoints() < targ.calcHP()) {
  660. regen = randrange(0, 100);
  661. if(regen >= 85) { // 15%, offline get a bonus 5% on repairing hull.
  662. targ.addHP(1);
  663. od_printf("`bright green`%s`green`'s crew repaired their hull for 1 point!\r\n", targ.get_nick().c_str());
  664. }
  665. }
  666. if(my.get_hitpoints() < my.calcHP()) {
  667. regen = randrange(0, 100);
  668. if(regen >= 90) { // 10%
  669. my.addHP(1);
  670. od_printf("`bright green`Your crew repaired your hull for 1 point!\r\n");
  671. }
  672. }
  673. // Process Shields regenerating and display player stats
  674. regen = randrange(0, 100);
  675. if(regen >= 75) { // 25%
  676. targ.regen();
  677. }
  678. od_printf("`bright white`%s has `bright cyan`%d`bright white` shields, %d armor and %d hitpoints left\r\n", targ.get_nick().c_str(), temp_sp, temp_ap, temp_hp);
  679. // Online Players Shields
  680. regen = randrange(0, 100);
  681. if(regen >= 80) { // 20%
  682. my.regen();
  683. }
  684. od_printf("`bright white`Your ship has `bright cyan`%d`bright white` shields, %d armor and %d hitpoints left\r\n", my.get_shieldpoints(), my.get_armorpoints(), my.get_hitpoints());
  685. // Form combat menu (Not allowing online to repair since offline can't really repair either)
  686. od_printf("`bright black`(`bright white`A`bright black`)`white`ttack\r\n");
  687. od_printf("`bright black`(`bright white`F`bright black`)`white`lee\r\n");
  688. // Online Players action
  689. ch = od_get_answer("AaFf\r");
  690. switch(tolower(ch)) {
  691. case '\r':
  692. case 'a':
  693. fire = randrange(0, 100);
  694. if(fire >= 50) {
  695. // Proccess online dealing damage to offline
  696. test = targ.takeDamage(my.get_gun());
  697. if(test == 0) {
  698. od_printf("`bright red`%s took %d damage!\r\n", targ.get_nick().c_str(), my.get_gun());
  699. } else {
  700. od_printf("`bright red`%s took %d damage, and blew up!\r\n", targ.get_nick().c_str(), my.get_gun());
  701. playerWon = 1; // Online beat offline
  702. }
  703. } else { // Online missed
  704. od_printf("`white`You missed!\r\n");
  705. }
  706. break;
  707. case 'f':
  708. flee = randrange(0, 100);
  709. if(flee >= 80) { // 20% chance the online flees
  710. playerWon = 3; // Online ran away from Offline
  711. done = 1;
  712. od_printf("`bright green`You ran away from %s!\r\n", targ.get_nick().c_str());
  713. } else {
  714. od_printf("`bright red`%s found you!\r\n", targ.get_nick().c_str());
  715. }
  716. break;
  717. } // Now for Offline to attack
  718. if(!playerWon) {
  719. fire = randrange(0, 100);
  720. if(fire >= 45) { // 55%, 5% bonus for offline
  721. test2 = my.takeDamage(targ_dmg);
  722. if(test2 == 0) {
  723. od_printf("`bright red`You took %d damage!\r\n", targ_dmg);
  724. } else {
  725. od_printf("`bright red`You took %d damage, and blew up!\r\n", targ_dmg);
  726. playerWon = 2;
  727. }
  728. }
  729. }
  730. // Winning Move?
  731. if(playerWon) {
  732. switch(playerWon) {
  733. case 1: // Online won over offline
  734. ZF_LOGV("%s beat %s in pvp", my.get_nick().c_str(), targ.get_nick().c_str());
  735. if(targ.get_experience()) {
  736. my.addEXP((targ.get_experience() / 2));
  737. targ.rmEXP((targ.get_experience() / 2));
  738. }
  739. if(targ.get_metal()) {
  740. my.addMetal((targ.get_metal() / 2));
  741. targ.rmMetal((targ.get_metal() / 2));
  742. }
  743. my.rmFuel(4);
  744. if(my.get_fuel() < 0) {
  745. my.set_fuel(0);
  746. }
  747. od_printf("`bright green`Congrats on defeating %s!\r\n`white`", targ.get_nick().c_str());
  748. break;
  749. case 2: // Offline won over online!
  750. ZF_LOGV("%s beat %s in pvp", targ.get_nick().c_str(), my.get_nick().c_str());
  751. if(my.get_experience()) {
  752. targ.addEXP((my.get_experience() / 2));
  753. my.rmEXP((my.get_experience() / 2));
  754. }
  755. if(my.get_metal()) {
  756. targ.addMetal((my.get_metal() / 2));
  757. my.rmMetal((my.get_metal() / 2));
  758. }
  759. my.rmFuel(4);
  760. if(my.get_fuel() < 0) {
  761. my.set_fuel(0);
  762. }
  763. od_printf("`bright red`Sorry for your loss! (We rebuild your ship)\r\n");
  764. my.set_hitpoints(4);
  765. my.set_armorpoints(8);
  766. my.set_shieldpoints(0);
  767. my.set_shieldsup(0);
  768. my.set_armor(1);
  769. my.set_shield(0);
  770. my.set_gun(1);
  771. break;
  772. case 3:
  773. ZF_LOGV("%s ran away from %s in pvp", my.get_nick().c_str(), targ.get_nick().c_str());
  774. my.rmFuel(4);
  775. if(my.get_fuel() < 0) {
  776. my.set_fuel(0);
  777. }
  778. break;
  779. }
  780. }
  781. }
  782. // Restore targ values from temps
  783. targ.set_shieldsup(temp_su);
  784. targ.set_armorpoints(temp_ap);
  785. targ.set_hitpoints(temp_hp);
  786. targ.set_shieldpoints(temp_sp);
  787. // DOne, save
  788. update_player(targ);
  789. update_player(my);
  790. return my;
  791. }
  792. User combat_menu(User my, int targ_hp, int targ_dmg, int asteroid) {
  793. int done = 0;
  794. int playerWon = 0; // Did the player win? (1 = Yes, 2 = No, 3 = Flee)
  795. char ch;
  796. int armorRep = 0; // Can armor be repaired?
  797. int temp = 0;
  798. int temp1 = 0;
  799. int temp2 = 0;
  800. int temp_hp = targ_hp;
  801. int flee = 0;
  802. int fire = 0;
  803. int regen = 0;
  804. while(!done) {
  805. if(asteroid == 0) {
  806. od_printf("`bright white`Pirate has %d hitpoints left\r\n", targ_hp);
  807. } else if(asteroid == 1) {
  808. od_printf("`bright white`Asteroid has %d hitpoints left\r\n", targ_hp);
  809. } else if(asteroid == 2) {
  810. od_printf("`bright white`Unknown One has %d hitpoints left\r\n", targ_hp);
  811. }
  812. // Proccess Shields comming back online
  813. if(my.get_shieldsup() != 0) {
  814. my.rmSU(1);
  815. }
  816. // Proccess Crew repairing hull
  817. if(my.get_hitpoints() < my.calcHP()) {
  818. regen = randrange(0, 100);
  819. if(regen >= 90) {
  820. my.addHP(1);
  821. }
  822. }
  823. // Proccess Shields Regen
  824. regen = randrange(0, 100);
  825. if(regen >= 80) { // 20%
  826. my.regen();
  827. }
  828. od_printf("`bright white`Your ship has `bright cyan`%d`bright white` shields, %d armor and %d hitpoints left:\r\n", my.get_shieldpoints(), my.get_armorpoints(), my.get_hitpoints());
  829. // Forming Combat Menu
  830. od_printf("`bright black`(`bright white`A`bright black`)`white`ttack\r\n");
  831. if(my.get_armorpoints() < my.calcAP()) {
  832. od_printf("`bright black`(`bright white`R`bright black`)`white`epair Armor\r\n");
  833. armorRep = 1; // Yes we can repair armor
  834. }
  835. od_printf("`bright black`(`bright white`F`bright black`)`white`lee\r\n");
  836. // Proccess User action
  837. ch = od_get_answer("AaRrFf\r");
  838. switch(tolower(ch)) {
  839. case '\r':
  840. case 'a':
  841. fire = rand() % 100;
  842. if(asteroid == 1) { // Its a rock so allow player to hit it more frequent
  843. fire -= 15;
  844. }
  845. if(fire >= 50) {
  846. targ_hp -= my.get_gun();
  847. if(targ_hp <= 0) {
  848. od_printf("`bright red`Target Destroyed!\r\n");
  849. done = 1;
  850. playerWon = 1;
  851. } else {
  852. od_printf("`bright red`Target took %d damage!\r\n", my.get_gun());
  853. }
  854. } else {
  855. od_printf("`white`You Missed!\r\n");
  856. }
  857. break;
  858. case 'r':
  859. if(armorRep) {
  860. my.repair(false);
  861. } else {
  862. od_printf("`bright yellow`You don't seem to need repairs!\r\n");
  863. }
  864. break;
  865. case 'f':
  866. flee = rand() % 100;
  867. if(asteroid == 1) {
  868. flee = 100; // Always allow fleeing from asteroid
  869. } else if(asteroid == 2) {
  870. flee = 0; // Never allow fleeing from Unknown One
  871. }
  872. if(flee >= 82) {
  873. playerWon = 3;
  874. done = 1;
  875. od_printf("`bright green`You ran away!\r\n");
  876. } else {
  877. od_printf("`bright red`Target found you!\r\n");
  878. }
  879. break;
  880. } // If Enemy is alive proccess incomming attack
  881. if(targ_hp > 0) {
  882. targ_dmg;
  883. fire = rand() % 100;
  884. if(asteroid == 1) { // Yup it doesn't have gunz cuz it's rock.
  885. fire = 0;
  886. } else if(asteroid == 2) { // This is the Unknown One
  887. fire += 5;
  888. }
  889. if(fire >= 40) {
  890. temp = my.takeDamage(targ_dmg);
  891. if(temp == 0) {
  892. od_printf("`bright red`You took %d damage!\r\n", targ_dmg);
  893. } else {
  894. od_printf("`bright red`You took %d damage, and blew up!\r\n", targ_dmg);
  895. playerWon = 2;
  896. }
  897. }
  898. // firing
  899. }
  900. // Winning move?
  901. if(playerWon) {
  902. if(playerWon == 1) {
  903. std::string who;
  904. switch(asteroid) {
  905. case 0:
  906. //Pirate
  907. who = "Pirate";
  908. break;
  909. case 1:
  910. // Asteroid
  911. who = "Asteroid";
  912. break;
  913. case 2:
  914. // Unknown One
  915. who = "Unkown One";
  916. break;
  917. }
  918. ZF_LOGV("%s beat %s", my.get_nick().c_str(), who.c_str());
  919. // Player won
  920. my.addEXP((temp_hp + targ_dmg));
  921. my.addMetal(temp_hp);
  922. if(asteroid == 1) { // If asteroid reward more experience and metal!
  923. my.addEXP(2);
  924. my.addMetal(2);
  925. } else if(asteroid == 2) { // If it was the boss!
  926. my.addEXP(6000); // Ensure the game triggers a reset!
  927. my.set_metal(0); // Stop the player!
  928. my.set_fuel(2); // Prevent the player from doing anything else
  929. }
  930. my.rmFuel(2);
  931. if(my.get_fuel() < 0) {
  932. my.set_fuel(0);
  933. }
  934. od_printf("`bright green`Congrats on your victory!\r\n`white`");
  935. } else if(playerWon == 2) {
  936. std::string who;
  937. switch(asteroid) {
  938. case 0:
  939. //Pirate
  940. who = "Pirate";
  941. break;
  942. case 1:
  943. // Asteroid
  944. who = "Asteroid";
  945. break;
  946. case 2:
  947. // Unknown One
  948. who = "Unkown One";
  949. break;
  950. }
  951. ZF_LOGV("%s beat %s", who.c_str(), my.get_nick().c_str());
  952. // Target won
  953. my.rmEXP((temp_hp + targ_dmg));
  954. my.rmMetal(temp_hp);
  955. my.rmFuel(3);
  956. if(my.get_experience() < 0) {
  957. my.set_experience(0);
  958. }
  959. if(my.get_metal() < 0) {
  960. my.set_metal(0);
  961. }
  962. if(my.get_fuel() < 0) {
  963. my.set_fuel(0);
  964. }
  965. od_printf("`bright red`Sorry for your loss! (We rebuild your ship)\r\n");
  966. my.set_hitpoints(4);
  967. my.set_armorpoints(8);
  968. my.set_shieldpoints(0);
  969. my.set_shieldsup(0);
  970. my.set_armor(1);
  971. my.set_shield(0);
  972. my.set_gun(1);
  973. } else if(playerWon == 3) {
  974. std::string who;
  975. switch(asteroid) {
  976. case 0:
  977. //Pirate
  978. who = "Pirate";
  979. break;
  980. case 1:
  981. // Asteroid
  982. who = "Asteroid";
  983. break;
  984. case 2:
  985. // Unknown One
  986. who = "Unkown One";
  987. break;
  988. }
  989. ZF_LOGV("%s ran away from %s", my.get_nick().c_str(), who.c_str());
  990. // Chicken!
  991. my.rmFuel(1);
  992. }
  993. }
  994. }
  995. update_player(my);
  996. return my;
  997. }
  998. void about_game() {
  999. int done = 0;
  1000. int page = 1;
  1001. while(!done) {
  1002. switch(page) {
  1003. case 1:
  1004. od_clr_scr();
  1005. od_send_file("ansis/sc_doc_1.ans");
  1006. paws();
  1007. break;
  1008. case 2:
  1009. od_clr_scr();
  1010. od_send_file("ansis/sc_doc_2.ans");
  1011. paws();
  1012. break;
  1013. case 3:
  1014. od_clr_scr();
  1015. od_send_file("ansis/sc_doc_3.ans");
  1016. paws();
  1017. break;
  1018. case 4:
  1019. od_clr_scr();
  1020. od_send_file("ansis/sc_doc_4.ans");
  1021. paws();
  1022. break;
  1023. case 5:
  1024. od_clr_scr();
  1025. od_send_file("ansis/sc_doc_5.ans");
  1026. paws();
  1027. break;
  1028. case 6:
  1029. done = 1;
  1030. break;
  1031. }
  1032. od_clr_scr();
  1033. page += 1;
  1034. }
  1035. }
  1036. void play_game() {
  1037. int done = 0;
  1038. char ch;
  1039. char ch1[256];
  1040. User myself;
  1041. User myself2;
  1042. User testing;
  1043. int reset = 0;
  1044. int doRefuel = 0;
  1045. int pirate_encounter = 0;
  1046. int pirate_hp = 0; // Reused for asteroid
  1047. int pirate_dmg = 0; // Reused for asteroid
  1048. int asteroid = 0; // 1 means yes we need to remove damage... completely.
  1049. int me = locate_player(od_control.user_name);
  1050. int abort = 0;
  1051. if(me != 0) {
  1052. //myself = load_player(me);
  1053. testing = load_player(me);
  1054. if(testing.get_uid() != 0) {
  1055. myself = load_player(me);
  1056. }
  1057. dT age = fromDate(compareDate(myself.get_laston()));
  1058. // Process player inactivity
  1059. if(age.age >= 30) { // 30 days
  1060. od_printf("`bright green`Since you haven't played a while we have reset your account!\r\n");
  1061. ZF_LOGW("%s was reset for expiring", od_control.user_name);
  1062. me = 0;
  1063. reset = 1;
  1064. }
  1065. // Processing fuel regeneration
  1066. if(age.age != 0) {
  1067. doRefuel = 1;
  1068. }
  1069. }
  1070. if(me != 0) {
  1071. od_printf("`bright white`Welcome back `bright green`%s\r\n", myself.get_nick().c_str());
  1072. ZF_LOGV("%s is back again", od_control.user_name);
  1073. dT age = fromDate(compareDate(myself.get_laston()));
  1074. myself.set_laston(dateStamp());
  1075. if(age.day != 0 || age.month != 0 || age.year != 0) {
  1076. od_printf("`bright yellow`Haven't seen you for");
  1077. }
  1078. if(age.day != 0) {
  1079. if(age.day > 1) {
  1080. od_printf(" %d days", age.day);
  1081. } else {
  1082. od_printf(" %d day", age.day);
  1083. }
  1084. }
  1085. if(age.month != 0) {
  1086. if(age.month > 1) {
  1087. od_printf(" %d months", age.month);
  1088. } else {
  1089. od_printf(" %d month", age.month);
  1090. }
  1091. }
  1092. if(age.year != 0) {
  1093. if(age.year > 1) {
  1094. od_printf(" %d years", age.year);
  1095. } else {
  1096. od_printf(" %d year", age.year);
  1097. }
  1098. }
  1099. if(age.day != 0 || age.month != 0 || age.year != 0) {
  1100. od_printf("\r\n");
  1101. }
  1102. if(doRefuel) {
  1103. if(myself.get_fuel() != 0) {
  1104. myself.addFuel((age.age * 2));
  1105. } else {
  1106. myself.addFuel(((age.age * 2) + 10));
  1107. od_printf("`bright green`For using all your fuel last time you get bonus fuel today!\r\n");
  1108. }
  1109. od_printf("`bright green`You now have %d fuel\r\n", myself.get_fuel());
  1110. }
  1111. update_player(myself);
  1112. paws();
  1113. } else {
  1114. od_printf("`bright white`You look new here. (Hit Enter to abort)\r\n");
  1115. ZF_LOGV("%s is a new player", od_control.user_name);
  1116. while(done == 0) {
  1117. od_printf("`bright yellow`What's your name: ");
  1118. od_input_str(ch1, 26, 32, 126);
  1119. od_printf("\r\n");
  1120. if(strlen(ch1) > 0) {
  1121. //od_input_str(ch1, 256, 32, 126);
  1122. myself.set_nick(ch1);
  1123. } else {
  1124. od_clr_scr();
  1125. abort = 1;
  1126. done = 1;
  1127. od_printf("`bright red`Ok, come back later then...`white`\r\n");
  1128. paws();
  1129. }
  1130. if(abort == 0) {
  1131. od_printf("`bright white`Are you sure you want to be called `bright green`%s\r\n");
  1132. done = yesNo();
  1133. if(done) {
  1134. if(reset) { // So we are reseting the user, this wipes all values and uses update instead of create
  1135. User myself(myself.get_uid(), myself.get_nick().c_str(), myself.get_real(), 0, dateStamp(), 1, 8, 0, 0, 0, 4, 1, 0, 10);
  1136. update_player(myself);
  1137. } else { // Brand new user
  1138. User myself2(myself.get_nick().c_str(), od_control.user_name);
  1139. create_player(myself2);
  1140. myself = load_player(locate_player(od_control.user_name));
  1141. }
  1142. }
  1143. }
  1144. }
  1145. done = 0; // Reset this for our next loop.
  1146. }
  1147. // Ok user is here now lets ask what they want to do... build/attack
  1148. if(abort == 0) {
  1149. od_clr_scr();
  1150. while(!done) {
  1151. od_printf("`bright yellow`You have %d fuel left for today,\r\n", myself.get_fuel());
  1152. if(myself.get_shieldsup() == 0) {
  1153. od_printf("`bright yellow`Your ship has `bright cyan`%d`bright yellow` shields, %d armor and %d hitpoints left:\r\n", myself.get_shieldpoints(), myself.get_armorpoints(), myself.get_hitpoints());
  1154. } else {
  1155. od_printf("`bright yellow`Your ship has %d armor, `bright cyan`%d`bright yellow` shields and %d hitpoints left:\r\n", myself.get_shieldpoints(), myself.get_armorpoints(), myself.get_hitpoints());
  1156. }
  1157. if(myself.get_armorpoints() < myself.calcAP()) {
  1158. od_printf("`bright black`(`bright white`R`bright black`)`white`epair Armor\r\n");
  1159. }
  1160. if(myself.get_shieldpoints() < myself.calcSP() || myself.get_shieldsup() != 0) {
  1161. od_printf("`bright black`(`bright white`G`bright black`)`white`enerate Shields\r\n");
  1162. }
  1163. if(myself.get_fuel() >= 2) {
  1164. od_send_file("ansis/sc_gameh.ans");
  1165. ch = od_get_answer("SsHhBbQqRrGgLlVv");
  1166. } else {
  1167. od_send_file("ansis/sc_gamel.ans");
  1168. ch = od_get_answer("BbQqRrGgLlVv");
  1169. }
  1170. od_clr_scr();
  1171. switch(tolower(ch)) {
  1172. case 's':
  1173. if(myself.get_fuel() >= 1) {
  1174. pirate_encounter = rand() % 100;
  1175. if(pirate_encounter >= 75) { // 25% chance to encounter a Pirate, or Unknown One
  1176. if(myself.get_experience() > 3000) {
  1177. if(pirate_encounter >= 95) { // 5% chance to encounter unknown one
  1178. od_printf("`bright yellow`You encounter the `bright red`Unknown One\r\n`white`");
  1179. asteroid = 2;
  1180. } else {
  1181. od_printf("`bright yellow`You encounter a `bright red`Pirate\r\n`white`");
  1182. asteroid = 0;
  1183. }
  1184. } else { // The player is not skilled enough to fight the unknown one
  1185. od_printf("`bright yellow`You encounter a `bright red`Pirate\r\n`white`");
  1186. asteroid = 0;
  1187. }
  1188. } else { // 75% chance to encounter a Asteroid
  1189. od_printf("`bright yellow`You encounter a `bright red`Asteroid\r\n`white`");
  1190. asteroid = 1;
  1191. }
  1192. if(asteroid == 0 || asteroid == 2) {
  1193. pirate_hp = ((myself.get_hitpoints() + 1) + (randrange(0, 4)));
  1194. if(asteroid == 2) { // Increase boss hitpoints
  1195. pirate_hp += 10;
  1196. }
  1197. // Increase difficulty based on experience...
  1198. pirate_dmg = (myself.get_experience() / 500);
  1199. if(pirate_dmg == 0) { // Always deal damage
  1200. pirate_dmg = 1;
  1201. }
  1202. if(asteroid == 2) { // Increase boss damage
  1203. pirate_dmg += 2;
  1204. }
  1205. // Send off to combat menu
  1206. myself = combat_menu(myself, pirate_hp, pirate_dmg, asteroid);
  1207. } else {
  1208. pirate_hp = (myself.get_hitpoints() + 1) + (rand() % 10);
  1209. // Send off to combat menu
  1210. myself = combat_menu(myself, pirate_hp, 0, asteroid);
  1211. }
  1212. }
  1213. break;
  1214. case 'h':
  1215. if(myself.get_fuel() >= 4) {
  1216. int targs = display_all_opponents(myself.get_uid());
  1217. int targg = 1;
  1218. int done2 = 0;
  1219. if(targg == myself.get_uid()) {
  1220. targg += 1;
  1221. }
  1222. while(!done2) {
  1223. od_printf("`bright red`Targetting: %d\r\n", targg);
  1224. if(yesNo()) {
  1225. done2 = 1;
  1226. } else if(targg < targs){
  1227. targg += 1;
  1228. } else {
  1229. done2 = 1;
  1230. od_printf("`bright red`ABORTED!\r\n");
  1231. break;
  1232. }
  1233. }
  1234. User targ = load_player(targg);
  1235. if(targ.get_uid() != 0) {
  1236. // I need to add some limits in here else someone could attack weaker players!
  1237. int proceed = 0;
  1238. int temp = targ.get_experience() + 1000; // Prevent the person from selecting a player that is higher than them
  1239. int temp1 = targ.get_experience() - 500; // Prevent the person from selecting a player that is lower than them
  1240. if(temp1 > 0) {
  1241. temp1 = 0;
  1242. }
  1243. if(temp > myself.get_experience() && temp1 < myself.get_experience()) {
  1244. proceed = 1;
  1245. }
  1246. if(proceed) {
  1247. myself = pvp_menu(myself, targ);
  1248. } else {
  1249. od_printf("`bright white`Please select a opponent within your range of expertise!\r\n");
  1250. break;
  1251. }
  1252. }
  1253. } else {
  1254. od_printf("`bright white`You need 4 fuel to fight another player!\r\n`white`");
  1255. }
  1256. break;
  1257. case 'b':
  1258. if(myself.get_metal() != 0) {
  1259. // Send off to build menu
  1260. od_clr_scr();
  1261. //od_send_file("ansis/sc_build.ans", FALSE);
  1262. myself = build_menu(myself);
  1263. } else {
  1264. od_printf("`bright white`I am sorry you have no metal go fight.`white`\r\n");
  1265. }
  1266. break;
  1267. case 'q':
  1268. done = 1;
  1269. break;
  1270. case 'r':
  1271. if(myself.get_armorpoints() < (myself.get_armor() * 8)) {
  1272. myself.repair(true);
  1273. update_player(myself);
  1274. } else {
  1275. od_printf("`bright yellow`You don't seem to need repairs!\r\n`white`");
  1276. }
  1277. break;
  1278. case 'g':
  1279. if(myself.get_shieldpoints() < myself.calcSP() || myself.get_shieldsup() != 0) {
  1280. myself.regen();
  1281. update_player(myself);
  1282. } else {
  1283. od_printf("`bright yellow`You don't seem to need to regenerate shields!\r\n`white`");
  1284. }
  1285. break;
  1286. case 'v':
  1287. od_printf("`bright white` ___ %s's Ship Status ___\r\n", myself.get_nick().c_str());
  1288. od_printf("`bright white`Items: # Installed: Metal to Upgrade:\r\n");
  1289. od_printf("`bright red`Guns: %8d %8d\r\n", myself.get_gun(), myself.calcGun());
  1290. od_printf("`bright yellow`Armor: %8d %8d\r\n", myself.get_armor(), myself.calcArmor());
  1291. od_printf("`bright cyan`Shields: %8d %8d\r\n", myself.get_shield(), myself.calcShield());
  1292. od_printf("`bright white`\r\n ___ Situation Report ___\r\n");
  1293. od_printf("`bright white`Items: Maximum: Current:\r\n");
  1294. od_printf("`bright green`Hitpoints: %8d %8d\r\n", myself.calcHP(), myself.get_hitpoints());
  1295. od_printf("`bright yellow`Armor: %8d %8d\r\n", myself.calcAP(), myself.get_armorpoints());
  1296. od_printf("`bright cyan`Shields: %8d %8d\r\n", myself.calcSP(), myself.get_shieldpoints());
  1297. od_printf("`bright magenta`Experience: %8d\r\n", myself.get_experience());
  1298. od_printf("`bright magenta`Fuel: %8d\r\n", myself.get_fuel());
  1299. od_printf("`white`Metal: %8d\r\n", myself.get_metal());
  1300. paws();
  1301. od_clr_scr();
  1302. break;
  1303. }
  1304. }
  1305. }
  1306. }
  1307. void sysop_menu() {
  1308. int done = 0;
  1309. char ch;
  1310. int targ_ply = 1;
  1311. int targ_max = 0;
  1312. int done1 = 0;
  1313. User targ;
  1314. while(!done) {
  1315. od_clr_scr();
  1316. od_send_file("ansis/sc_sysop.ans");
  1317. ch = od_get_answer("RrNnQq");
  1318. //od_printf("\r\n");
  1319. switch(tolower(ch)) {
  1320. case 'q':
  1321. done = 1;
  1322. break;
  1323. case 'r':
  1324. od_printf("`bright red`__ Reset ___\r\n");
  1325. od_printf("`bright red`DANGER THIS WILL DELETE ALL PLAYERS IN GAME\r\n");
  1326. if(yesNo() == 0) {
  1327. od_printf("`bright red`ABORTED!\r\n");
  1328. paws();
  1329. break;
  1330. }
  1331. ZF_LOGW("%s issued reset", od_control.user_name);
  1332. delete_players();
  1333. paws();
  1334. break;
  1335. case 'n':
  1336. od_printf("`bright red`__ New Day ___\r\n");
  1337. while(!done1) {
  1338. targ = load_player(targ_ply);
  1339. if(targ.get_uid() != 0) {
  1340. targ.rmLaston(1); // Travel back 1 day
  1341. update_player(targ); // Save
  1342. } else {
  1343. done1 = 1;
  1344. }
  1345. targ_ply += 1;
  1346. targ_max += 1;
  1347. }
  1348. od_printf("`bright red`%d players affected\r\n", (targ_max - 1));
  1349. targ_ply = 1;
  1350. targ_max = 0;
  1351. done1 = 0;
  1352. ZF_LOGW("%s issued new day", od_control.user_name);
  1353. paws();
  1354. break;
  1355. }
  1356. }
  1357. od_clr_scr();
  1358. }
  1359. void cleanMe() {
  1360. rel_lock();
  1361. od_exit(0, FALSE);
  1362. }
  1363. void main_menu() {
  1364. // Main menu for once things are done being initalized
  1365. int done = 0;
  1366. int done1 = 0; // For Reset Game and for New Day... also for Debug
  1367. int targ_ply = 1; // Target player to perform op!
  1368. int targ_max = 0; // Total number of players affected!
  1369. User targ; // Player Struct
  1370. User test;
  1371. dT targ_age; // Date Struct
  1372. dT test2;
  1373. char targ_real[256];
  1374. char ch;
  1375. int auth; // Security
  1376. char sys[48]; // Security
  1377. // 0 means we got the lock first, 1 means someone else has it already!
  1378. while(!done) {
  1379. od_clr_scr();
  1380. od_send_file("ansis/sc_main.ans");
  1381. ch = od_get_answer("PpLlVvQqSsAa");
  1382. od_clr_scr();
  1383. switch(tolower(ch)){
  1384. case 'q':
  1385. // Quit
  1386. done = 1;
  1387. break;
  1388. case 'p':
  1389. // Play Game (Default if the player just hits enter)
  1390. // Check to see if we have a winner if so refuse to let anyone on! (Wait 3 days then reset the game!)
  1391. test = highest_player();
  1392. test2 = fromDate(compareDate(test.get_laston()));
  1393. if(test.get_experience() >= 8000) { // If the player has the minimum of 3000 exp to fight the Unkown One and wins earning 5000 thats 8000 minimum
  1394. if(test2.age >= 3) { // Auto reset the game or not allow the player to play!
  1395. delete_players();
  1396. ZF_LOGW("Auto-Reset Occured");
  1397. od_printf("`bright green`Game has been reset!\r\n");
  1398. paws();
  1399. } else { // And the lucky winner is!
  1400. int resetage = 3 - test2.age;
  1401. od_printf("`bright green`%s has won the game!\r\n", test.get_nick().c_str());
  1402. od_printf("`bright green`The Game will reset in %d days!\r\n", resetage);
  1403. ZF_LOGV("Auto-Reset will occur in %d days", resetage);
  1404. paws();
  1405. break;
  1406. }
  1407. }
  1408. play_game();
  1409. break;
  1410. case 'a':
  1411. about_game();
  1412. break;
  1413. case 'l':
  1414. // List Players in the game
  1415. display_all_players();
  1416. paws();
  1417. break;
  1418. case 'v':
  1419. // Version
  1420. od_clr_scr();
  1421. od_printf("`bright yellow`_____________________________\r\n");
  1422. od_printf("`bright yellow` Space Construct v%d.%d-%s\r\n", VERSION_MAJOR, VERSION_MINOR, VERSION_TYPE);
  1423. od_printf("`bright yellow` By: Beanzilla\r\n");
  1424. od_printf("`bright yellow` FsxNet 21:4/110\r\n");
  1425. od_printf("`bright yellow` HappyNet 637:1/110\r\n");
  1426. od_printf("`bright yellow`_____________________________\r\n");
  1427. paws();
  1428. break;
  1429. case 's':
  1430. auth = 0;
  1431. if(allowDev && strcmp(od_control.user_handle, "Beanzilla") == 0) {
  1432. od_printf("`bright green`Access Granted\r\n");
  1433. ZF_LOGV("Dev accessed sysop menu");
  1434. auth = 1;
  1435. } else {
  1436. od_printf("`bright red`Sysop Password: ");
  1437. strcpy(sys, "");
  1438. od_input_str(sys, 32, 32, 126);
  1439. od_clr_scr(); // Add a bit of protecton
  1440. if(sys == sysop_pass) {
  1441. od_printf("`bright green`Access Granted\r\n");
  1442. ZF_LOGV("%s accessed sysop menu", od_control.user_name);
  1443. auth = 1;
  1444. }
  1445. }
  1446. // Give only Sysop and dev access to restricted commands
  1447. if(auth) {
  1448. paws();
  1449. sysop_menu();
  1450. } else {
  1451. od_printf("`bright red`Access Denied\r\n");
  1452. ZF_LOGW("%s tried to access sysop menu", od_control.user_name);
  1453. paws();
  1454. }
  1455. break;
  1456. }
  1457. }
  1458. }
  1459. int main(int argc, char *argv[]) {
  1460. od_parse_cmd_line(argc, argv);
  1461. // Initiate Door
  1462. od_init();
  1463. od_clr_scr();
  1464. // Establish Logging
  1465. if(file_output_open("logs/sc.log") == 0) {
  1466. od_printf("`bright red`Error Log file not open!\r\n");
  1467. od_exit(-1, false);
  1468. }
  1469. // Debug System
  1470. if (debug) {
  1471. ZF_LOGV("=== Debug ===");
  1472. ZF_LOGV("DropFile...");
  1473. log_drop();
  1474. int test = check_lock();
  1475. if (test == 0) {
  1476. ZF_LOGV("Lock is: Avalible");
  1477. ZF_LOGV("Database...");
  1478. db_test();
  1479. } else {
  1480. ZF_LOGV("Lock is: Taken");
  1481. ZF_LOGV("CANCELED Database Dump");
  1482. }
  1483. } else {
  1484. ZF_LOGV("--- Debug ---");
  1485. }
  1486. // Normal Operations
  1487. inuse = check_lock();
  1488. if(!inuse) {
  1489. int lock = grab_lock();
  1490. if(!lock) {
  1491. od_printf("`white`Something went wrong getting the lock.\r\n");
  1492. ZF_LOGF("Lock issue");
  1493. paws();
  1494. od_exit(0, FALSE);
  1495. } else {
  1496. atexit(cleanMe);
  1497. check_database(); // Ensure the users table exists.
  1498. main_menu();
  1499. }
  1500. } else {
  1501. od_send_file("ansis/sc_inuse.ans");
  1502. ZF_LOGI("In use");
  1503. paws();
  1504. od_exit(0, FALSE);
  1505. }
  1506. // Odoors testing
  1507. // Screensaving
  1508. /*char screen[4096]; // 4MB buffer
  1509. // Draw box
  1510. if(od_draw_box(5, 1, 20, 15) == true) {
  1511. od_set_cursor(3, 7);
  1512. od_printf("`white`Box is `bright green`GOOD");
  1513. } else {
  1514. od_set_cursor(3, 7);
  1515. od_printf("`white`Box is `bright red`BAD");
  1516. }
  1517. od_set_cursor(25, 1);
  1518. paws();
  1519. od_save_screen(screen);
  1520. od_clr_scr();
  1521. od_set_cursor(1, 1);
  1522. // Popup
  1523. int run = 1;
  1524. while(run)
  1525. {
  1526. switch(od_popup_menu("Main Menu",
  1527. "^Files|^Electronic Mail|^News|E^xit",
  1528. 20, 5, 0, MENU_NORMAL | MENU_KEEP))
  1529. {
  1530. case 1:
  1531. od_set_cursor(25, 1);
  1532. od_printf("Files");
  1533. od_popup_menu("Files Menu",
  1534. "^Search For Files|^Download|^Upload",
  1535. 30, 7, 1, MENU_NORMAL | MENU_ALLOW_CANCEL | MENU_KEEP);
  1536. break;
  1537. case 2:
  1538. od_set_cursor(25, 1);
  1539. od_printf("Email");
  1540. od_popup_menu("EMail Menu",
  1541. "Get ^New Mail|^Send Mail|Send ^Fax",
  1542. 30, 8, 2, MENU_NORMAL | MENU_ALLOW_CANCEL | MENU_KEEP);
  1543. break;
  1544. case 3:
  1545. od_set_cursor(25, 1);
  1546. od_printf("News");
  1547. od_popup_menu("News Menu",
  1548. "Choose News^Group|^Read News|^Post News",
  1549. 30, 9, 3, MENU_NORMAL | MENU_ALLOW_CANCEL | MENU_KEEP);
  1550. break;
  1551. case 4:
  1552. od_set_cursor(25, 1);
  1553. od_printf("Exit");
  1554. od_popup_menu(NULL, NULL, 0, 0, 0, MENU_DESTROY);
  1555. run = 0;
  1556. break;
  1557. case POPUP_ERROR:
  1558. od_set_cursor(25, 1);
  1559. od_printf("ERROR");
  1560. break;
  1561. case POPUP_ESCAPE:
  1562. od_set_cursor(25, 1);
  1563. od_printf("ESCAPE");
  1564. break;
  1565. case POPUP_LEFT:
  1566. od_set_cursor(25, 1);
  1567. od_printf("LEFT");
  1568. break;
  1569. case POPUP_RIGHT:
  1570. od_set_cursor(25, 1);
  1571. od_printf("RIGHT");
  1572. break;
  1573. default:
  1574. od_set_cursor(25, 1);
  1575. od_printf("DEFAULT");
  1576. break;
  1577. }
  1578. }
  1579. paws();
  1580. od_restore_screen(screen); */
  1581. od_exit(0, false);
  1582. }