| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623 |
- /**--------------------------------------------------------------------------**\
- ===================================
- y_users - Registration functions.
- ===================================
- Description:
- Provides access to a user system for registering and saving users.
- Legal:
- Version: MPL 1.1
-
- The contents of this file are subject to the Mozilla Public License Version
- 1.1 (the "License"); you may not use this file except in compliance with
- the License. You may obtain a copy of the License at
- http://www.mozilla.org/MPL/
-
- Software distributed under the License is distributed on an "AS IS" basis,
- WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
- for the specific language governing rights and limitations under the
- License.
-
- The Original Code is the YSI malloc include.
-
- The Initial Developer of the Original Code is Alex "Y_Less" Cole.
- Portions created by the Initial Developer are Copyright (C) 2011
- the Initial Developer. All Rights Reserved.
-
- Contributors:
- ZeeX, koolk, JoeBullet/Google63, g_aSlice/Slice
-
- Thanks:
- JoeBullet/Google63 - Handy arbitrary ASM jump code using SCTRL.
- ZeeX - Very productive conversations.
- koolk - IsPlayerinAreaEx code.
- TheAlpha - Danish translation.
- breadfish - German translation.
- Fireburn - Dutch translation.
- yom - French translation.
- 50p - Polish translation.
- Zamaroht - Spanish translation.
- Dracoblue, sintax, mabako, Xtreme, other coders - Producing other modes
- for me to strive to better.
- Pixels^ - Running XScripters where the idea was born.
- Matite - Pestering me to release it and using it.
-
- Very special thanks to:
- Thiadmer - PAWN, whose limits continue to amaze me!
- Kye/Kalcor - SA:MP.
- SA:MP Team past, present and future - SA:MP.
-
- Version:
- 2.2
- Changelog:
- 02/02/13:
- Added bits and join timestamp to preload data.
- Added backwards compatible salting to the hash.
- 15/11/11:
- Added comments.
- Added language to the top-level player data (needed for login).
- 11/11/11:
- First version.
- Functions:
- Public
- -
- Core:
- -
- Stock:
- -
- Static:
- -
- Inline:
- -
- API:
- -
- Callbacks:
- -
- Definitions:
- -
- Enums:
- -
- Macros:
- -
- Tags:
- -
- Variables:
- Global:
- -
- Static:
- -
- Commands:
- -
- Compile options:
- -
- Operators:
- -
- </remarks>
- \**--------------------------------------------------------------------------**/
- loadtext core[ysi_players];
- enum E_USER_PRELOAD
- {
- E_USER_PRELOAD_YID,
- Language:E_USER_PRELOAD_LANG,
- E_USER_PRELOAD_PASS[MAX_PASSWORD_LENGTH + 1],
- E_USER_PRELOAD_BITS,
- E_USER_PRELOAD_DATE
- }
- static stock
- YSI_g_sPlayerIndexFile,
- INI:YSI_g_sPlayerWriteFile = INI_NO_FILE,
- YSI_g_sPlayerYID[MAX_PLAYERS] = {-2, ...},
- //YSI_g_sCallbacks,
- YSI_g_sPreloadData[MAX_PLAYERS][E_USER_PRELOAD];
- //#define Player_GetIndexFile(%0,%1) ((YSI_g_sPlayerIndexFile=%1[0]),%0[sizeof(%0)-5]=(('a'<=(YSI_g_sPlayerIndexFile|0x20)<='z')?(YSI_g_sPlayerIndexFile|0x20):(('0'<=YSI_g_sPlayerIndexFile<='9')?('0'):('_'))))
- #define Player_GetIndexFile(%0,%1) ((YSI_g_sPlayerIndexFile=%1),%0[USER_FILE_LENGTH+4]=(('a'<=(YSI_g_sPlayerIndexFile|0x20)<='z')?(YSI_g_sPlayerIndexFile|0x20):(('0'<=YSI_g_sPlayerIndexFile<='9')?('0'):('_'))))
- //foreign Player_DoLogout(playerid, uid);
- foreign Player_TryRegister(playerid, string:password[]);
- foreign Player_TryLogin(playerid, string:password[]);
- foreign Player_TryGroup(playerid, string:other[], string:password[]);
- // No longer global because the YID is shared in all scripts.
- //foreign Player_GetYID(playerid);
- foreign Player_ChangePassword(playerid, string:password[]);
- foreign Player_ForceGroup(playerid, string:other[]);
- foreign Player_ForceLogin(playerid);
- /**--------------------------------------------------------------------------**\
- <summary>Player_Reload</summary>
- <param name="playerid">Player who changed name.</param>
- <returns>
- -
- </returns>
- <remarks>
- Reload a player's basic data when they change name.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_Reload(playerid)
- {
- new
- name[MAX_PLAYER_NAME];
- GetPlayerName(playerid, name, sizeof (name));
- Player_Preload(name, YSI_g_sPreloadData[playerid]);
- }
- stock Player_SetPreload(playerid, data[E_USER_PRELOAD])
- {
- if (0 <= playerid < MAX_PLAYERS)
- {
- YSI_g_sPreloadData[playerid] = data;
- }
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_Preload</summary>
- <param name="playerid">Player who is logging in.</param>
- <returns>
- -
- </returns>
- <remarks>
- Loads a player's data to an array.
- </remarks>
- \**--------------------------------------------------------------------------**/
- #if _YSI_PLUGINS_MYSQL == 7
- public _Player_Preload(ret[E_USER_PRELOAD])
- {
- // Need to use some clever AMX hacking to pass the "ret" array...
- }
- #endif
- stock Player_Preload(string:name[], ret[E_USER_PRELOAD])
- {
- // First, find the player's file. This should be the ONLY place where the
- // password is to be loaded.
- P:4("Player_Preload called: %s", name);
- ret[E_USER_PRELOAD_YID] = -2;
- ret[E_USER_PRELOAD_PASS] = '\0';
- ret[E_USER_PRELOAD_LANG] = NO_LANGUAGE;
- ret[E_USER_PRELOAD_BITS] = 0;
- ret[E_USER_PRELOAD_DATE] = 0;
- #if _YSI_PLUGINS_MYSQL == 7
- // Right then, lets get coding...
- new
- query[110];
- mysql_format(YSI_g_sMySQL, query, "SELECT `uid`, `language`, `hash` FROM `y_users_register` WHERE `name` = '%e' LIMIT 0,1", name);
- mysql_function_query(YSI_g_sMySQL, query, true, "_Player_Preload", "ii", );
- #else
- new
- namelen = strlen(name),
- filename[] = USER_FILE_PATH "ind_X.YSI",
- File:fIndex;
- Player_GetIndexFile(filename, name[0]);
- fIndex = fopen(filename, io_read);
- if (fIndex)
- {
- P:5("Player_Preload: fIndex OK");
- new
- line[INDEX_DATA_LINE_LENGTH],
- len;
- while ((len = fread(fIndex, line)))
- {
- P:6("Player_Preload: while");
- //new
- // len;
- //len = strlen(line);
- // Check if the line is the right length (could be one of three
- // lengths depending on newlines). Skip blanks.
- if (len < INDEX_DATA_LINE_LENGTH - 3)
- {
- continue;
- }
- P:6("Player_Preload: Not len");
- // Check the name on the line.
- if (!strcmp(line[MAX_INDEX_LENGTH + 1], name, false, namelen) && line[MAX_INDEX_LENGTH + 1 + namelen] == ' ')
- {
- P:6("Player_Preload: checked name");
- // Found the section on this one player.
- //P:6("Player_Preload: check pass: %s ?= %s", hash, line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1]);
- // Save the loaded data.
- line[MAX_INDEX_LENGTH] = '\0';
- //printf("line: %s", line);
- ret[E_USER_PRELOAD_YID] = strval(line);
- //printf("%d %d %d", ret[E_USER_PRELOAD_YID], strval(line), strval("00000022"));
- //printf("%d", strval(line));
- line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1 + MAX_PASSWORD_LENGTH + 1 + 2] = '\0';
- ret[E_USER_PRELOAD_LANG] = Langs_GetLanguage(line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1 + MAX_PASSWORD_LENGTH + 1]);
- strcat(ret[E_USER_PRELOAD_PASS], line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1], MAX_PASSWORD_LENGTH + 1);
- // Load the 32 extra "bits".
- ret[E_USER_PRELOAD_BITS] = hexstr(line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1 + MAX_PASSWORD_LENGTH + 1 + 2 + 1]);
- // Load the user registration date (if they have one).
- ret[E_USER_PRELOAD_DATE] = hexstr(line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1 + MAX_PASSWORD_LENGTH + 1 + 2 + 1 + 8 + 1]);
- P:6("Player_Preload: %s %d %d %x %x", ret[E_USER_PRELOAD_PASS], ret[E_USER_PRELOAD_YID], _:ret[E_USER_PRELOAD_LANG], ret[E_USER_PRELOAD_BITS], ret[E_USER_PRELOAD_DATE]);
- fclose(fIndex);
- return 1;
- }
- }
- fclose(fIndex);
- }
- else if (fexist(filename))
- {
- P:E("Error reading index %c.", filename[0]);
- return -1;
- }
- ret[E_USER_PRELOAD_YID] = -1;
- #endif
- return 0;
- }
- stock Language:Player_GetPreloadLanguage(playerid)
- {
- return Language:YSI_g_sPreloadData[playerid][E_USER_PRELOAD_LANG];
- }
- stock Player_GetPreloadBits(playerid)
- {
- return YSI_g_sPreloadData[playerid][E_USER_PRELOAD_BITS];
- }
- static remotefunc void:_Player_SetPreloadBits(playerid, bits)
- {
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_BITS] = bits;
- }
- stock Player_SetPreloadBits(playerid, bits)
- {
- broadcastfunc _Player_SetPreloadBits(playerid, bits);
- Player_RewritePreload(playerid);
- }
- stock Player_GetPreloadDate(playerid)
- {
- if (0 <= YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] < 1234567890)
- {
- return -1;
- }
- return YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE];
- }
- stock Player_IsRegistered(playerid)
- {
- return YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID] != -1;
- }
- hook OnScriptInit()
- {
- switch (ftouch(USER_FILE_PATH "index.YSI"))
- {
- case -1:
- {
- P:E(USER_FILE_PATH "index.YSI does not exist and couldn't be created.");
- }
- case 1:
- {
- if (!Player_CreateNewID())
- {
- P:E(USER_FILE_PATH "index.YSI does not exist and couldn't be created.");
- }
- }
- }
- // NOT using "ALS" for chaining.
- /*if (funcidx("OnPlayerLogin") != -1)
- {
- YSI_g_sCallbacks |= 1;
- }
- if (funcidx("OnPlayerLogout") != -1)
- {
- YSI_g_sCallbacks |= 2;
- }*/
- }
- public OnGameModeInit()
- {
- if (YSI_FILTERSCRIPT)
- {
- //if (YSI_g_sCallbacks & 16)
- //{
- //Users_OnGameModeInit();
- #if defined Users_OnGameModeInit
- Users_OnGameModeInit();
- #endif
- //}
- }
- else
- {
- #if defined Users_OnGameModeInit
- Users_OnGameModeInit();
- #endif
- //Users_OnGameModeInit();
- Users_DoUpgrade0();
- Users_DoUpgrade1();
- }
- return 1;
- }
- #if defined _ALS_OnGameModeInit
- #undef OnGameModeInit
- #else
- #define _ALS_OnGameModeInit
- #endif
- #if defined Users_OnGameModeInit
- forward Users_OnGameModeInit();
- #endif
- #define OnGameModeInit(%0) Users_OnGameModeInit(%0)
- public OnFilterScriptInit()
- {
- //if (funcidx("Users_OnGameModeInit") != -1)
- //{
- // YSI_g_sCallbacks |= 16;
- //}
- // DO ALL (MOST) OTHER INITS FIRST. ENSURE WE COME LATER.
- //Users_OnFilterScriptInit();
- #if defined Users_OnFilterScriptInit
- Users_OnFilterScriptInit();
- #endif
- Users_DoUpgrade0();
- Users_DoUpgrade1();
- return 1;
- }
- #if defined _ALS_OnFilterScriptInit
- #undef OnFilterScriptInit
- #else
- #define _ALS_OnFilterScriptInit
- #endif
- #if defined Users_OnFilterScriptInit
- forward Users_OnFilterScriptInit();
- #endif
- #define OnFilterScriptInit(%0) Users_OnFilterScriptInit(%0)
- static stock Users_DoUpgrade0()
- {
- // Only ever do this upgrade once.
- switch (ftouch(USER_FILE_PATH "y_users_v2_1.YSI"))
- {
- case -1:
- {
- P:E("Could not upgrade user files.");
- }
- case 1:
- {
- #if !defined YSI_DO_USER_UPGRADE
- P:E("y_users needs to upgrade user files. Please back up the old ones and recompile/rerun once with YSI_DO_USER_UPGRADE.");
- fremove(USER_FILE_PATH "y_users_v2_1.YSI");
- #else
- // Do the upgrade to add languages to all files (could take a
- // little while, but needs doing).
- P:I("Please wait, upgrading user files.");
- new
- ch,
- filename[] = USER_FILE_PATH "ind_X.YSI",
- File:f,
- File:g,
- line[INDEX_DATA_LINE_LENGTH],
- //Language:def = Langs_GetLanguageAt(0),
- code[3];
- strcpy(code, Langs_GetCode(Langs_GetLanguageAt(0)));
- for (new i = -2; i != 26; ++i)
- {
- if (i == -2)
- {
- ch = '_';
- }
- else if (i == -1)
- {
- ch = '0';
- }
- else
- {
- ch = 'a' + i;
- }
- Player_GetIndexFile(filename, ch);
- f = fopen(filename, io_read);
- if (!f)
- {
- if (fexist(filename))
- {
- P:E("Upgrade %c failed.", filename[USER_FILE_LENGTH + 4]);
- }
- continue;
- }
- g = ftemp();
- if (!g)
- {
- fclose(f);
- P:E("Upgrade %c failed.", filename[USER_FILE_LENGTH + 4]);
- continue;
- }
- while ((ch = fread(f, line)))
- {
- //ch = strlen(line);
- if (ch > 3)
- {
- if (line[ch - 2] < ' ')
- {
- // "/r/n" or "/n/r".
- // Copy the existing line ending.
- line[ch + 1] = line[ch - 2];
- line[ch + 2] = line[ch - 1];
-
- line[ch - 2] = ' ';
- line[ch - 1] = code[0];
- line[ch ] = code[1];
-
- line[ch + 3] = '\0';
- //line[ch - 1] = '\0';
- fwrite(g, line);
- }
- else
- {
- // "/n" or "/r".
- // Copy the existing line ending.
- line[ch + 2] = line[ch - 1];
-
- line[ch - 1] = ' ';
- line[ch ] = code[0];
- line[ch + 1] = code[1];
-
- line[ch + 2] = '\0';
- //line[ch - 1] = '\0';
- fwrite(g, line);
- }
- }
- }
- fseek(g);
- fclose(f);
- fremove(filename);
- f = fopen(filename, io_write);
- if (!f)
- {
- fclose(g);
- P:E("Upgrade %c failed.", filename[USER_FILE_LENGTH + 4]);
- continue;
- }
- while (fread(g, line))
- {
- fwrite(f, line);
- }
- fclose(f);
- fclose(g);
- }
- f = fopen(USER_FILE_PATH "index.YSI", io_append);
- if (f)
- {
- fwrite(f, " ");
- fclose(f);
- }
- else
- {
- P:E("Upgrade index failed.");
- }
- P:I("Upgrade complete.");
- #endif
- }
- }
- }
- static stock Users_DoUpgrade1()
- {
- // Only ever do this upgrade once.
- switch (ftouch(USER_FILE_PATH "y_users_v2_2.YSI"))
- {
- // case 0: Already exists.
- case -1:
- {
- P:E("Could not upgrade user files.");
- }
- case 1:
- {
- #if !defined YSI_DO_USER_UPGRADE
- P:E("y_users needs to upgrade user files. Please back up the old ones and recompile/rerun once with YSI_DO_USER_UPGRADE.");
- fremove(USER_FILE_PATH "y_users_v2_2.YSI");
- #else
- // Do the upgrade to add languages to all files (could take a
- // little while, but needs doing).
- P:I("Please wait, upgrading user files.");
- new
- ch,
- hash[MAX_PASSWORD_LENGTH + MAX_INDEX_LENGTH + 1],
- filename[] = USER_FILE_PATH "ind_X.YSI",
- File:f,
- File:g,
- line[INDEX_DATA_LINE_LENGTH],
- //Language:def = Langs_GetLanguageAt(0),
- code[3];
- strcpy(code, Langs_GetCode(Langs_GetLanguageAt(0)));
- for (new i = -2; i != 26; ++i)
- {
- if (i == -2)
- {
- ch = '_';
- }
- else if (i == -1)
- {
- ch = '0';
- }
- else
- {
- ch = 'a' + i;
- }
- Player_GetIndexFile(filename, ch);
- f = fopen(filename, io_read);
- if (!f)
- {
- if (fexist(filename))
- {
- P:E("Upgrade %c failed.", filename[USER_FILE_LENGTH + 4]);
- }
- continue;
- }
- g = ftemp();
- if (!g)
- {
- fclose(f);
- P:E("Upgrade %c failed.", filename[USER_FILE_LENGTH + 4]);
- continue;
- }
- while (fread(f, line))
- {
- ch = strlen(line);
- if (ch > 3)
- {
- // Get the current hash.
- strcpy(hash, line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1], MAX_PASSWORD_LENGTH + 1);
- //hash[MAX_PASSWORD_LENGTH] = '\0';
- // Append their ID (not QUITE their join time, but
- // best we can do retrospectively).
- //line[MAX_INDEX_LENGTH] = '\0';
- strcat(hash, line, sizeof (hash));
- //line[MAX_INDEX_LENGTH] = ' ';
- //hash[MAX_PASSWORD_LENGTH + MAX_INDEX_LENGTH] = '\0';
- //printf("HASH 1: %s", hash);
- Player_HashPass(hash, hash);
- //printf("HASH 2: %s", hash);
- // Copy the correct line ending.
- if (line[ch - 2] < ' ')
- {
- format(line, sizeof (line), "%.*s %s %.2s 00000000 %.8s%s", MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME, line, hash, line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1 + MAX_PASSWORD_LENGTH + 1], line[0], line[ch - 2]);
- //format(line[ch - 2], 21, " 00000000 %.*s%s", MAX_INDEX_LENGTH, line[0], line[ch - 2]);
- }
- else
- {
- format(line, sizeof (line), "%.*s %s %.2s 00000000 %.8s%s", MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME, line, hash, line[MAX_INDEX_LENGTH + 1 + MAX_PLAYER_NAME + 1 + MAX_PASSWORD_LENGTH + 1], line[0], line[ch - 1]);
- //format(line[ch - 1], 21, " 00000000 %.*s%s", MAX_INDEX_LENGTH, line[0], line[ch - 1]);
- }
- fwrite(g, line);
- }
- }
- fseek(g);
- fclose(f);
- fremove(filename);
- f = fopen(filename, io_write);
- if (!f)
- {
- fclose(g);
- P:E("Upgrade %c failed.", filename[USER_FILE_LENGTH + 4]);
- continue;
- }
- while (fread(g, line))
- {
- fwrite(f, line);
- }
- fclose(f);
- fclose(g);
- }
- f = fopen(USER_FILE_PATH "index.YSI", io_append);
- if (f)
- {
- fwrite(f, " ");
- fclose(f);
- }
- else
- {
- P:E("Upgrade index failed.");
- }
- P:I("Upgrade complete.");
- #endif
- }
- }
- }
- /*hook OnScriptExit()
- {
- _Player_CloseLogout();
- }*/
- hook OnPlayerConnect(playerid)
- {
- P:1("Users_OnPlayerConnect called: %d", playerid);
- // -2 means unknown.
- YSI_g_sPlayerYID[playerid] = -2;
- broadcastfunc _Player_IsLoggedIn(playerid);
- if (existproperty(8, YSIM_LOG_IN))
- {
- new
- uid = getproperty(8, YSIM_LOG_IN);
- P:5("Users_OnPlayerConnect: Exists %d", uid);
- if (uid == -1)
- {
- new
- name[MAX_PLAYER_NAME];
- GetPlayerName(playerid, name, sizeof (name));
- Player_Preload(name, YSI_g_sPreloadData[playerid]);
- }
- else
- {
- // This DOES NOT use "broadcastfunc" as it's local only.
- Player_DoLogin(playerid, uid);
- }
- deleteproperty(8, YSIM_LOG_IN);
- }
- else
- {
- P:5("Users_OnPlayerConnect: Doesn't exist");
- new
- name[MAX_PLAYER_NAME];
- GetPlayerName(playerid, name, sizeof (name));
- Player_Preload(name, YSI_g_sPreloadData[playerid]);
- P:5("Users_OnPlayerConnect: Done Preload");
- // Can do checking in here to see if they just rejoined.
- }
- YSI_g_sPlayerYID[playerid] = -1;
- }
- hook OnPlayerDisconnect(playerid, reason)
- {
- P:2("Users_OnPlayerDisconnect called: %d %d %d", playerid, reason, YSI_g_sPlayerYID[playerid]);
- if (YSI_g_sPlayerYID[playerid] >= 0)
- {
- // DO NOT broadcastfunc this in case it's just because of one script being
- // unloaded, not the player actually leaving (and thus do everything in
- // different scripts separately).
- Player_DoLogout(playerid, YSI_g_sPlayerYID[playerid]);
- }
- YSI_g_sPlayerYID[playerid] = -2;
- }
- static remotefunc void:_Player_IsLoggedIn(playerid)
- {
- P:4("_Player_IsLoggedIn called: %d %d", playerid, YSI_g_sPlayerYID[playerid]);
- if (YSI_g_sPlayerYID[playerid] != -2)
- {
- setproperty(8, YSIM_LOG_IN, YSI_g_sPlayerYID[playerid]);
- }
- }
- stock bool:Player_IsLoggedIn(playerid)
- {
- // -2 should never be an issue, but if it is...
- return YSI_g_sPlayerYID[playerid] >= 0;
- }
- stock Player_GetYID(playerid)
- {
- return YSI_g_sPlayerYID[playerid];
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_TryLogin</summary>
- <param name="playerid">Player who is logging in.</param>
- <param name="password[]">Password they entered.</param>
- <param name="f">Show the failed to login message?</param>
- <returns>
- -
- </returns>
- <remarks>
- Tries to log in a player - hashes and checks their password and if it's
- right calls the core login code. It doesn't matter WHICH script does this
- as they ALL get called and ALL track the login status of a player.
- </remarks>
- \**--------------------------------------------------------------------------**/
- //stock Player_TryLogin(playerid, string:password[], f = 0)
- //{
- //return _Player_TryLogin(playerid, password, f);
- //}
- global Player_TryLogin(playerid, string:password[])
- {
- P:2("Player_TryLogin start");
- if (Player_IsLoggedIn(playerid))
- {
- // They are already logged in.
- Text_Send(playerid, $YSI_LOGIN_ALREADY);
- return 1;
- }
- new
- hash[MAX_PASSWORD_LENGTH + 8 + 1];
- Player_HashPass(password, hash);
- format(hash[MAX_PASSWORD_LENGTH], sizeof (hash) - MAX_PASSWORD_LENGTH, "%04x%04x", YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] >>> 16, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] & 0xFFFF);
- //printf("HASH 1: %s", hash);
- Player_HashPass(hash, hash);
- //printf("HASH 2: %s", hash);
- //Player_HashPass(hash, hash);
- switch (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- }
- case -1:
- {
- Text_Send(playerid, $YSI_LOGIN_NOTF);
- }
- default:
- {
- // Match the password.
- if (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS][0] && !strcmp(YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS], hash, false, MAX_PASSWORD_LENGTH))
- {
- // Wipe the password from memory.
- memset(YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS], MAX_PASSWORD_LENGTH, 0);
- // Extract the uid and call in to the login code.
- Langs_SetPlayerLanguage(playerid, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_LANG]);
- broadcastfunc Player_DoLogin(playerid, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID]);
- Text_Send(playerid, $YSI_LOGIN_LOGIN);
- return 1;
- }
- else
- {
- Text_Send(playerid, $YSI_LOGIN_WRONG);
- }
- }
- }
- return 0;
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_ForceLogin</summary>
- <param name="playerid">Player who is logging in.</param>
- <returns>
- -
- </returns>
- <remarks>
- Like "Player_TryLogin" but doesn't take a password so always works.
- </remarks>
- \**--------------------------------------------------------------------------**/
- global Player_ForceLogin(playerid)
- {
- P:2("Player_TryLogin start");
- if (Player_IsLoggedIn(playerid))
- {
- // They are already logged in.
- Text_Send(playerid, $YSI_LOGIN_ALREADY);
- return 1;
- }
- switch (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- }
- case -1:
- {
- Text_Send(playerid, $YSI_LOGIN_NOTF);
- }
- default:
- {
- // Extract the uid and call in to the login code.
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS] = '\0';
- Langs_SetPlayerLanguage(playerid, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_LANG]);
- broadcastfunc Player_DoLogin(playerid, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID]);
- Text_Send(playerid, $YSI_LOGIN_LOGIN);
- return 1;
- }
- }
- return 0;
- }
- /*static*/ remotefunc void:Player_DoLogin(playerid, uid)
- {
- // Called when a player logs in - either locally (new script) or globally
- // (actually only just logged in).
- YSI_g_sPlayerYID[playerid] = uid;
- // Load any "uvar" variables.
- // Call the hooks version of this.
- new
- filename[64];
- format(filename, sizeof (filename), USER_FILE_PATH "%0" #MAX_INDEX_LENGTH "d.INI", uid);
- // INI_ParseFile will ONLY load the data for THIS mode, as well as data
- // which is mode independent (though there should be none).
- INI_ParseFile(filename, "@yU_%s", .bExtra = true, .extra = playerid, .bLocal = true, .bFilter = false, .filter = #MODE_NAME);
- //Hooks_OnPlayerLogin(playerid, uid);
- //if (YSI_g_sCallbacks & 1)
- //{
- //CallLocalFunction("OnPlayerLogin", "ii", playerid, uid);
- call OnPlayerLogin(playerid, uid);
- //}
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_RemoveEntry</summary>
- <param name="name[]">Item to remove.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for removing data.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_RemoveEntry(name[])
- {
- INI_RemoveEntry(YSI_g_sPlayerWriteFile, name);
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_WriteString</summary>
- <param name="name[]">Data name.</param>
- <param name="data[]">Data.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for strings.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_WriteString(name[], data[])
- {
- INI_WriteString(YSI_g_sPlayerWriteFile, name, data);
- }
- stock Player_WriteArray(const name[], data[], len)
- {
- //printf("name = %s", name);
- //printf("data = %d, %d, %d", data[0], data[1], data[2]);
- //printf("len = %d", len);
- INI_WriteArray(YSI_g_sPlayerWriteFile, name, data, len);
- return 1;
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_WriteInt</summary>
- <param name="name[]">Data name.</param>
- <param name="data">Integer data.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for integers.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_WriteInt(name[], data)
- {
- INI_WriteInt(YSI_g_sPlayerWriteFile, name, data);
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_WriteHex</summary>
- <param name="name[]">Data name.</param>
- <param name="data">Hex data.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for integers to be written as hex values.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_WriteHex(name[], data)
- {
- INI_WriteHex(YSI_g_sPlayerWriteFile, name, data);
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_WriteBin</summary>
- <param name="name[]">Data name.</param>
- <param name="data">Binary data.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for integers to be written as binary values.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_WriteBin(name[], data)
- {
- INI_WriteBin(YSI_g_sPlayerWriteFile, name, data);
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_WriteBool</summary>
- <param name="name[]">Data name.</param>
- <param name="data">Boolean data.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for booleans.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_WriteBool(name[], bool:data)
- {
- INI_WriteBool(YSI_g_sPlayerWriteFile, name, data);
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_WriteFloat</summary>
- <param name="name[]">Data name.</param>
- <param name="Float:data">Float data.</param>
- <param name="accuracy">number of decimal places to write.</param>
- <returns>
- -
- </returns>
- <remarks>
- Wrapper for Player_AddToBuffer for floats. Uses custom code instead of
- format() as it's actually faster for something simple like this.
- </remarks>
- \**--------------------------------------------------------------------------**/
- stock Player_WriteFloat(name[], Float:data, accuracy = 6)
- {
- INI_WriteFloat(YSI_g_sPlayerWriteFile, name, data, accuracy);
- }
- stock Player_SetTag(tag[])
- {
- // Make sure we ALWAYS store mode tags with a special prefix.
- new
- tag2[MAX_INI_TAG] = "@@" #MODE_NAME "-";
- strcat(tag2, tag);
- //printf("tag = %s, %s", tag, tag2);
- INI_SetTag(YSI_g_sPlayerWriteFile, tag2);
- }
- stock Player_DeleteTag(tag[])
- {
- INI_DeleteTag(YSI_g_sPlayerWriteFile, tag);
- }
- //global Player_DoLogout(playerid, uid)
- static stock Player_DoLogout(playerid, uid)
- {
- /*INI:YSI_g_sOneLogoutFile = INI_NO_FILE
- YSI_g_sOneLogoutPerson = INVALID_PLAYER_ID*/
- new
- filename[64];
- format(filename, sizeof (filename), USER_FILE_PATH "%0" #MAX_INDEX_LENGTH "d.INI", uid);
- YSI_g_sPlayerWriteFile = INI_Open(filename);
- if (YSI_g_sPlayerWriteFile != INI_NO_FILE)
- {
- //Hooks_OnPlayerLogout(playerid, uid);
- //if (YSI_g_sCallbacks & 2)
- //{
- call OnPlayerLogout(playerid, uid);
- //}
- INI_Close(YSI_g_sPlayerWriteFile);
- }
- // Do ALL the saving from ONE place. Detect one or many scripts ending.
- /*if (YSI_g_sOneLogoutFile == INI_NO_FILE)
- {
- new
- filename[64];
- format(filename, sizeof (filename), USER_FILE_PATH "%0" #MAX_INDEX_LENGTH "d.INI", uid);
- YSI_g_sOneLogoutFile = INI_Open(filename);
- YSI_g_sOneLogoutPerson = playerid;
- }
- else if (YSI_g_sOneLogoutPerson != playerid)
- {
- INI_Close(YSI_g_sOneLogoutFile);
- }
- // Will be called AFTER all "OnPlayerDisconnect" callbacks are called when
- // one player leaves, and if more leave it's still used for the last player
- // in the list.
- //defer _Player_CloseLogout();
- _Player_CloseLogout();
- // By now "YSI_g_sOneLogoutFile" contains a handle to the user's file,
- // either having just been opened, or invoked in a previous call to this
- // function from another script still running but loosing the player.
- // This could in theory be used from some force-logout script.
- YSI_g_sPlayerYID[playerid] = -1;*/
- return 1;
- }
- //static timer _Player_CloseLogout[0]()
- /*static stock _Player_CloseLogout()
- {
- if (YSI_g_sOneLogoutFile != INI_NO_FILE)
- {
- INI_Close(YSI_g_sOneLogoutFile);
- YSI_g_sOneLogoutFile = INI_NO_FILE;
- YSI_g_sOneLogoutPerson = INVALID_PLAYER_ID;
- }
- }*/
- /**--------------------------------------------------------------------------**\
- <summary>Player_HashPass</summary>
- <param name="pass[]">Data to hash.</param>
- <returns>
- -
- </returns>
- <remarks>
- Based on my Dad's hash system but slightly modifed. Updated for reverse
- compatability with other login systems. Needs more code for Whirlpool.
- </remarks>
- \**--------------------------------------------------------------------------**/
- static stock Player_HashPass(pass[], target[])
- {
- #if defined PP_ADLER32
- new
- s1 = 1,
- s2 = 0,
- i,
- You_REALLY_shouldnt_use_Adler32;
- while (pass[i])
- {
- s1 = (s1 + pass[i++]) % 65521;
- s2 = (s2 + s1) % 65521;
- }
- //new
- // target[MAX_PASSWORD_LENGTH + 1];
- format(target, sizeof (target), "%" #MAX_PASSWORD_LENGTH "d", (s2 << 16) + s1);
- //return target;
- #elseif defined PP_MD5 && defined MD5_Hash
- new
- You_REALLY_shouldnt_use_MD5;
- strcpy(target, MD5_Hash(pass, strlen(pass)));
- #elseif defined PP_SHA1
- #error SHA1 unsupported.
- #elseif defined PP_YSI
- static
- charset[] = "A,UbRgdnS#|rT_%5+ZvEK¬NF<9¦IH[(C)2O07 Y-Less]$Qw^?/om4;@'8k£Pp.c{&l\\3zay>DfxV:WXjuG6*!1\"i~=Mh`JB}qt",
- css = 99;
- new
- //target[MAX_PASSWORD_LENGTH + 1],
- j,
- sum = j,
- tmp = 0,
- i,
- mod;
- j = strlen(pass);
- for (i = 0; i < MAX_PASSWORD_LENGTH || i < j; i++)
- {
- mod = i % MAX_PASSWORD_LENGTH;
- tmp = (i >= j) ? charset[(7 * i) % css] : pass[i];
- sum = (sum + chrfind(tmp, charset) + 1) % css;
- target[mod] = charset[(sum + target[mod]) % css];
- }
- target[MAX_PASSWORD_LENGTH] = '\0';
- //return target;
- #elseif defined WP_Hash
- WP_Hash(target, MAX_PASSWORD_LENGTH + 1, pass);
- #else
- #error Whirlpool (or other) hash not found.
- #endif
- }
- // Hooray for bizare bugs! I think this is because the function above is
- // secretly a macro with "if/else" and a block statement, not a real function.
- stock Player_SomeWeirdBugFix()
- {
- }
- /*stock Anything0()
- {
- }*/
- /*stock Anything1()
- {
- }
- stock Anything2()
- {
- }
- stock Anything3()
- {
- }*/
- /*stock Anything4()
- {
- }
- stock Anything5()
- {
- }
- stock Anything6()
- {
- }
- stock Anything7()
- {
- }*/
- /**--------------------------------------------------------------------------**\
- <summary>Player_TryRegister</summary>
- <param name="playerid">Player who is registering.</param>
- <param name="string:password[]">The password they entered.</param>
- <returns>
- -
- </returns>
- <remarks>
- Register the player with the given password if there is no-one else with the
- name already. Or log them in if the username and password match an existing
- account. Note that there is no "Player_ForceRegister" as it would do the
- same thing with no less parameters (a password MUST be given to write in the
- file).
- </remarks>
- \**--------------------------------------------------------------------------**/
- //#endinput
- global Player_TryRegister(playerid, string:password[])
- {
- P:2("Player_TryRegister called");
- if (Player_IsLoggedIn(playerid))
- {
- // They are already logged in.
- Text_Send(playerid, $YSI_LOGIN_ALREADY);
- return 1;
- }
- new
- hash[MAX_PASSWORD_LENGTH + 1 + 8];
- Player_HashPass(password, hash);
- switch (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- }
- case -1:
- {
- }
- default:
- {
- // Get the salt.
- format(hash[MAX_PASSWORD_LENGTH], sizeof (hash) - MAX_PASSWORD_LENGTH, "%04x%04x", YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] >>> 16, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] & 0xFFFF);
- Player_HashPass(hash, hash);
- // Match the password.
- if (!strcmp(YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS], hash, false, MAX_PASSWORD_LENGTH) && YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS][0])
- {
- // Extract the uid and call in to the login code.
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS] = '\0';
- Langs_SetPlayerLanguage(playerid, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_LANG]);
- broadcastfunc Player_DoLogin(playerid, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID]);
- Text_Send(playerid, $YSI_LOGIN_LOGIN);
- return 1;
- }
- else
- {
- //Text_Send(playerid, $YSI_LOGIN_WRONG);
- Text_Send(playerid, $YSI_REG_TAKEN);
- return 0;
- }
- }
- }
- new
- name[MAX_PLAYER_NAME + 1];
- GetPlayerName(playerid, name, sizeof (name));
- //format(name, sizeof (name), "%" #MAX_PLAYER_NAME "s", name);
- new
- filename[64] = USER_FILE_PATH "ind_X.YSI",
- File:fIndex;//,
- //hash[MAX_PASSWORD_LENGTH + 1];
- //Player_HashPass(password, hash);
- Player_GetIndexFile(filename, name[0]);
- fIndex = fopen(filename, io_read);
- P:5("Player_TryRegister: fIndex");
- new
- line[INDEX_DATA_LINE_LENGTH];
- if ((fIndex = fopen(filename, io_append)))
- {
- P:5("Player_TryRegister: Write index.");
- // Write the new user to the index file.
- new
- uid = Player_GetNewID();
- if (uid == -1)
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- return 0;
- }
- new
- d = gettime(),
- date[9];
- format(date, sizeof (date), "%04x%04x", d >>> 16, d & 0xFFFF);
- strcat(hash, date);
- Player_HashPass(hash, hash);
- format(line, sizeof (line), "%0" #MAX_INDEX_LENGTH "d %" #MAX_PLAYER_NAME "s %" #MAX_PASSWORD_LENGTH "s %02s 00000000 %s" INI_NEW_LINE, uid, name, hash, Langs_GetCode(Langs_GetPlayerLanguage(playerid)), date);
- fwrite(fIndex, line);
- fclose(fIndex);
- format(filename, sizeof (filename), USER_FILE_PATH "%0" #MAX_INDEX_LENGTH "d.INI", uid);
- new
- INI:x = INI_Open(filename);
- if (x == INI_NO_FILE)
- {
- Text_Send(playerid, $YSI_LOGIN_NOLOAD);
- }
- else
- {
- INI_SetTag(x, "ysi_names");
- INI_WriteString(x, name, "name");
- INI_Close(x);
- }
- // Call in all scripts.
- broadcastfunc Player_DoLogin(playerid, uid);
- Text_Send(playerid, $YSI_LOGIN_LOGIN);
- return 1;
- }
- else
- {
- Text_Send(playerid, $YSI_ADDU_INDER2);
- return 0;
- }
- }
- global Player_ChangePassword(playerid, string:password[])
- {
- #pragma unused password
- new
- uid = Player_GetYID(playerid);
- if (uid < 0)
- {
- return 0;
- }
- // Create the new password.
- new
- hash[MAX_PASSWORD_LENGTH + 8 + 1];
- Player_HashPass(password, hash);
- format(hash[MAX_PASSWORD_LENGTH], sizeof (hash) - MAX_PASSWORD_LENGTH, "%04x%04x", YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] >>> 16, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] & 0xFFFF);
- Player_HashPass(hash, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS]);
- // Save it.
- Player_RewritePreload(playerid);
- // Wipe it.
- memset(YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS], MAX_PASSWORD_LENGTH, 0);
- return 1;
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_RewritePreload</summary>
- <param name="playerid">Player whose data should be saved.</param>
- <returns>
- -
- </returns>
- <remarks>
- When a player's preload data is modifed (new bit data or changed password),
- it needs to be written back out to file.
- </remarks>
- \**--------------------------------------------------------------------------**/
- static stock Player_RewritePreload(playerid)
- {
- if (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID] < 0)
- {
- return 0;
- }
- // First, find the player's file. This should be the ONLY place where the
- // password is to be loaded.
- //ret[E_USER_PRELOAD_YID] = -2;
- //ret[E_USER_PRELOAD_PASS] = '\0';
- //ret[E_USER_PRELOAD_LANG] = NO_LANGUAGE;
- //ret[E_USER_PRELOAD_BITS] = 0;
- //ret[E_USER_PRELOAD_DATE] = 0;
- new
- name[MAX_PLAYER_NAME + 1];
- GetPlayerName(playerid, name, sizeof (name));
- P:4("Player_RewritePreload called: %s", name);
- #if _YSI_PLUGINS_MYSQL == 7
- // Right then, lets get coding...
- new
- query[110];
- mysql_format(YSI_g_sMySQL, query, "SELECT `uid`, `language`, `hash` FROM `y_users_register` WHERE `name` = '%e' LIMIT 0,1", name);
- mysql_function_query(YSI_g_sMySQL, query, true, "_Player_Preload", "ii", );
- #else
- new
- namelen = strlen(name),
- filename[] = USER_FILE_PATH "ind_X.YSI",
- File:fIndex;
- Player_GetIndexFile(filename, name[0]);
- fIndex = fopen(filename, io_readwrite);
- if (fIndex)
- {
- P:5("Player_RewritePreload: fIndex OK");
- new
- line[INDEX_DATA_LINE_LENGTH],
- len;
- while ((len = fread(fIndex, line)))
- {
- P:6("Player_RewritePreload: while");
- //new
- // len;
- //len = strlen(line);
- // Check if the line is the right length (could be one of three
- // lengths depending on newlines). Skip blanks.
- if (len < INDEX_DATA_LINE_LENGTH - 3)
- {
- continue;
- }
- P:6("Player_RewritePreload: Not len");
- // Check the name on the line.
- if (!strcmp(line[MAX_INDEX_LENGTH + 1], name, false, namelen) && line[MAX_INDEX_LENGTH + 1 + namelen] == ' ')
- {
- P:6("Player_RewritePreload: checked name");
- fseek(fIndex, -len, seek_current);
- //format(line, sizeof (line),
- format(line, sizeof (line),
- "%0" #MAX_INDEX_LENGTH "d %" #MAX_PLAYER_NAME "s %" #MAX_PASSWORD_LENGTH "s %02s %04x%04x %04x%04x" INI_NEW_LINE,
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID],
- name,
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_PASS],
- Langs_GetCode(Langs_GetPlayerLanguage(playerid)),
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_BITS] >>> 16, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_BITS] & 0xFFFF,
- YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] >>> 16, YSI_g_sPreloadData[playerid][E_USER_PRELOAD_DATE] & 0xFFFF);
- P:5("Player_RewritePreload: Writing: %s", line);
- fwrite(fIndex, line);
- fclose(fIndex);
- return 1;
- }
- }
- fclose(fIndex);
- }
- else if (fexist(filename))
- {
- P:E("Error reading index %c.", filename[0]);
- return 0;
- }
- //ret[E_USER_PRELOAD_YID] = -1;
- #endif
- return 0;
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_TryGroup</summary>
- <param name="playerid">Player who is joining a group.</param>
- <param name="string:other[]">A player name already in the group.</param>
- <param name="string:password[]">The password of the group.</param>
- <returns>
- -
- </returns>
- <remarks>
- Links a player with an existing player such that they share all stats.
- </remarks>
- \**--------------------------------------------------------------------------**/
- //#endinput
- global Player_TryGroup(playerid, string:other[], string:password[])
- {
- P:2("Player_TryGroup called");
- if (Player_IsLoggedIn(playerid))
- {
- // They are already logged in.
- Text_Send(playerid, $YSI_LOGIN_ALREADY);
- return 1;
- }
- new
- hash[MAX_PASSWORD_LENGTH + 1];
- Player_HashPass(password, hash);
- // Check if the user is not registered already.
- switch (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- return 0;
- }
- case -1:
- {
- //Text_Send(playerid, $YSI_LOGIN_NOTF);
- }
- default:
- {
- Text_Send(playerid, $YSI_REG_TAKEN);
- return 0;
- }
- }
- // Check if the new data matches the old.
- new
- ret[E_USER_PRELOAD];
- Player_Preload(other, ret);
- switch (ret[E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- }
- case -1:
- {
- Text_Send(playerid, $YSI_LOGIN_NOTF);
- }
- default:
- {
- // Match the password.
- if (!strcmp(ret[E_USER_PRELOAD_PASS], hash, false, MAX_PASSWORD_LENGTH) && ret[E_USER_PRELOAD_PASS][0])
- {
- new
- name[MAX_PLAYER_NAME + 1];
- GetPlayerName(playerid, name, sizeof (name));
- new
- filename[64] = USER_FILE_PATH "ind_X.YSI";
- Player_GetIndexFile(filename, name[0]);
- new
- File:fIndex = fopen(filename, io_append);
- if (!fIndex)
- {
- Text_Send(playerid, $YSI_ADDU_INDERR2);
- return 0;
- }
- P:5("Player_TryGroup: Write index.");
- new
- uid = ret[E_USER_PRELOAD_YID],
- line[INDEX_DATA_LINE_LENGTH];
- // Use the loaded ID.
- format(line, sizeof (line), "%0" #MAX_INDEX_LENGTH "d %" #MAX_PLAYER_NAME "s %" #MAX_PASSWORD_LENGTH "s %02s" INI_NEW_LINE, uid, name, hash, Langs_GetCode(ret[E_USER_PRELOAD_LANG]));
- fwrite(fIndex, line);
- fclose(fIndex);
- format(filename, sizeof (filename), USER_FILE_PATH "%0" #MAX_INDEX_LENGTH "d.INI", uid);
- new
- INI:x = INI_Open(filename);
- if (x == INI_NO_FILE)
- {
- Text_Send(playerid, $YSI_LOGIN_NOLOAD);
- }
- else
- {
- // Add this name to the list of known names.
- INI_SetTag(x, "ysi_names");
- INI_WriteString(x, name, "name");
- INI_Close(x);
- }
- // Call in all scripts.
- Langs_SetPlayerLanguage(playerid, ret[E_USER_PRELOAD_LANG]);
- broadcastfunc Player_DoLogin(playerid, uid);
- Text_Send(playerid, $YSI_LOGIN_LOGIN);
- return 1;
- }
- else
- {
- Text_Send(playerid, $YSI_LOGIN_WRONG);
- }
- }
- }
- return 0;
- }
- /**--------------------------------------------------------------------------**\
- <summary>Player_ForceGroup</summary>
- <param name="playerid">Player who is joining a group.</param>
- <param name="string:other[]">A player name already in the group.</param>
- <returns>
- -
- </returns>
- <remarks>
- Like "Player_TryGroup", but doesn't take a password and instead just uses
- the password of the old player (hashed).
- </remarks>
- \**--------------------------------------------------------------------------**/
- //#endinput
- global Player_ForceGroup(playerid, string:other[])
- {
- P:2("Player_ForceGroup called");
- if (Player_IsLoggedIn(playerid))
- {
- // They are already logged in.
- Text_Send(playerid, $YSI_LOGIN_ALREADY);
- return 1;
- }
- // Check if the user is not registered already.
- switch (YSI_g_sPreloadData[playerid][E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- return 0;
- }
- case -1:
- {
- //Text_Send(playerid, $YSI_LOGIN_NOTF);
- }
- default:
- {
- Text_Send(playerid, $YSI_REG_TAKEN);
- return 0;
- }
- }
- // Check if the new data matches the old.
- new
- ret[E_USER_PRELOAD];
- Player_Preload(other, ret);
- switch (ret[E_USER_PRELOAD_YID])
- {
- case -2:
- {
- Text_Send(playerid, $YSI_LOGIN_INDERR);
- }
- case -1:
- {
- Text_Send(playerid, $YSI_LOGIN_NOTF);
- }
- default:
- {
- new
- name[MAX_PLAYER_NAME + 1];
- GetPlayerName(playerid, name, sizeof (name));
- new
- filename[64] = USER_FILE_PATH "ind_X.YSI";
- Player_GetIndexFile(filename, name[0]);
- new
- File:fIndex = fopen(filename, io_append);
- if (!fIndex)
- {
- Text_Send(playerid, $YSI_ADDU_INDERR2);
- return 0;
- }
- P:5("Player_TryGroup: Write index.");
- new
- uid = ret[E_USER_PRELOAD_YID],
- line[INDEX_DATA_LINE_LENGTH];
- // Use the loaded ID.
- format(line, sizeof (line), "%0" #MAX_INDEX_LENGTH "d %" #MAX_PLAYER_NAME "s %" #MAX_PASSWORD_LENGTH "s %02s" INI_NEW_LINE, uid, name, ret[E_USER_PRELOAD_PASS], Langs_GetCode(ret[E_USER_PRELOAD_LANG]));
- fwrite(fIndex, line);
- fclose(fIndex);
- format(filename, sizeof (filename), USER_FILE_PATH "%0" #MAX_INDEX_LENGTH "d.INI", uid);
- new
- INI:x = INI_Open(filename);
- if (x == INI_NO_FILE)
- {
- Text_Send(playerid, $YSI_LOGIN_NOLOAD);
- }
- else
- {
- // Add this name to the list of known names.
- INI_SetTag(x, "ysi_names");
- INI_WriteString(x, name, "name");
- INI_Close(x);
- }
- // Call in all scripts.
- Langs_SetPlayerLanguage(playerid, ret[E_USER_PRELOAD_LANG]);
- broadcastfunc Player_DoLogin(playerid, uid);
- Text_Send(playerid, $YSI_LOGIN_LOGIN);
- return 1;
- }
- }
- return 0;
- }
- static stock Player_GetNewID()
- {
- new
- File:fHnd = fopen(USER_FILE_PATH "index.YSI", io_readwrite),
- num[MAX_INDEX_LENGTH + 9],
- uid = -1;
- if (fHnd)
- {
- fread(fHnd, num);
- num[strfind(num, " ")] = '\0';
- uid = strval(num) + 1;
- valstr(num, uid);
- fseek(fHnd, 0, seek_start);
- fwrite(fHnd, num);
- fwrite(fHnd, " ");
- fclose(fHnd);
- }
- return uid;
- }
- static stock Player_CreateNewID()
- {
- new
- File:fHnd = fopen(USER_FILE_PATH "index.YSI", io_write);
- if (fHnd)
- {
- fwrite(fHnd, "-1 ");
- fclose(fHnd);
- return 1;
- }
- return 0;
- }
|