[Tutorial] Account system with MySQL [R34][ORM]
#1

A quick and basic tutorial for anyone, about an account system with MySQL-R34, if you need to know more about ORM(Object Relational Mapping). Read the tutorial by AndreT.

EDIT: Added saving with ORM.
This account system is going to be working on dialogs, basic and simple. No extra fixes.

At first, we would need to declare our player enum. Where it would have a structure of everything we are planning to use.
pawn Code:
enum _pInfo
{
    ORM:pORM_ID,
    pPassword[128], // 128 character long password, you can change it for your own needs. Note this only save a password 127 length long.
    pUsername[MAX_PLAYER_NAME+1],
    pKills, // How many times a player killed
    pDeaths, // How many times a player was killed
    pMoney // How much money a player has, you can also set this a long:pMoney but it would need a very custom functioning system.
}
new PlayerInfo[MAX_PLAYERS][_pInfo]; // PlayerInfo[playerid][structure];
As we got that set in, it's time to insert our a_mysql.inc, simply by doings this at under a_samp.inc.
pawn Code:
#include <a_samp>
#include <a_mysql>
Alright, time to get to connect towards the mysql server. OnGameModeInit(); this should be executed:
pawn Code:
// Ontop, near new PlayerInfo[][];
new mysql_connectionHandle;

public OnGameModeInit()
{
    mysql_connectionHandle = mysql_connect("hostname_here", "username_here", "database_here", "password_here");
    // ^ Example: mysql_connect("127.0.0.1", "root", "gameserver_db", "admin");
   
    // Check if we connected!
    if(mysql_errno() == 2003)
    {
        print("[OnGameModeInit] Connection with mysql server couldn't be established, error: 2003");
        SendRconCommand("exit");
        return 1;
    }
    print("[OnGameModeInit] Established connection with mysql server");
}
If you can't connect towards the mysql server, it will return error 2003. (Read more about mysql errors here: click) And it will close the server.
If you could connect towards the mysql server, everything will be fine and you could continue.

Once a player connects, we would want to check if he is registered or not. At firstly, we would make a new thread to load us any variables that the player has in the database.
pawn Code:
public OnPlayerConnet(playerid)
{
    new playerQuery[128]; // Query string.
    GetPlayerName(playerid, playerQuery, sizeof(playerQuery)); // Using on playerQuery to not declare a new expression twice.
    mysql_format(mysql_connectionHandle, playerQuery, sizeof(playerQuery), "SELECT * FROM `accounts` WHERE `username` = '%e'", playerQuery); // mysql format.
    mysql_tquery(mysql_connectionHandle, playerQuery, "OnPlayerMysqlFinish", "d", playerid); // mysql tquery is a new mysql thread query, means it will run apart from the server and won't lag if a lot of data is being loaded.
}
Alright, so we got the player connection part handled, now we would want to actually react when it finishes to load.
pawn Code:
// I am lazy sometimes, so I just add this at the top of the script:
#define Skeleton::%1(%2) forward %1(%2); \
                        public %1(%2)
// This would be the dialog numbers, on top of the script as well.
#define DIALOG_ACCOUNT_LOGIN 1
// Why not zero? Cause showing dialog zero clears all dialogs from screen.
#define DIALOG_ACCOUNT_REGISTER 2

Skeleton::OnPlayerMysqlFinish(playerid)
{
    if(cache_num_rows() == 1)
    { // Registered
        new ORM:ormid = PlayerInfo[playerid][pORM_ID] = orm_create("accounts");
        // Notice! We're using "accounts" and ... FROM "accounts" .. ; This is mandatory cause it applies on the mysql database.

        orm_addvar_string(ormid, PlayerInfo[playerid][pPassword], 128, "Password"); // Loading string
        orm_addvar_string(ormid, PlayerInfo[playerid][pUsername], MAX_PLAYER_NAME+1, "Username"); // Loading string
        orm_addvar_int(ormid, PlayerInfo[playerid][pKills], "Kills"); // Loading int, "kills"
        orm_addvar_int(ormid, PlayerInfo[playerid][pDeaths], "Deaths"); // Loading int, "deaths"
        orm_addvar_int(ormid, PlayerInfo[playerid][pMoney], "Money"); // Loading int, "money"

        orm_setkey(ormid, "Username");
        orm_apply_cache(); // Apply everything we did above.
       
        ShowPlayerDialog(playerid, DIALOG_ACCOUNT_LOGIN, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below.", "Login", "Exit");
    }
    else if(!cache_num_rows())
    { // Needs to register.
        ShowPlayerDialog(playerid, DIALOG_ACCOUNT_REGISTER, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below\nNotice! You'll need to save it to login under this account.", "Register", "Exit");
    }
    else
    { // Too much registered under same name.
        // You can do here whatever you want.
    }
    return 1;
}
So we got that sorted out, let's proceed on towards the dialogs.
You'll need to use two dialogs, one for login, one for register.
Quick note here, I didn't use ORM on register, sorry!

pawn Code:
public OnDialogResponse(playerid, dialogid, response, listitem, inputtext[])
{
    if(dialogid == DIALOG_ACCOUNT_LOGIN)
    {
        if(!response) return Kick(playerid); // If he clicked "EXIT".
        if(response)
        {
             if(strlen(inputtext) == 0) return ShowPlayerDialog(playerid, DIALOG_ACCOUNT_LOGIN, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below.\nYou can't use an empty password.", "Login", "Exit");
             if(strcmp(inputtext, PlayerInfo[playerid][pPassword], false) == 0)
             { // If password was correct.
                 // Create your own spawn method!
             }
             else return ShowPlayerDialog(playerid, DIALOG_ACCOUNT_LOGIN, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below.\nPassword was incorrect!", "Login", "Exit");
        }
    }
    else if(dialogid == DIALOG_ACCOUNT_REGISTER)
    {
        if(!response) return Kick(playerid); // If he clicked "EXIT".
        if(response)
        {
            if(strlen(inputtext) == 0) return ShowPlayerDialog(playerid, DIALOG_ACCOUNT_REGISTER, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below\nNotice! You'll need to save it to login under this account.\nYou can't use an empty password!", "Register", "Exit");
            format(PlayerInfo[playerid][pPassword], 128, "%s", inputtext[0]);
            PlayerInfo[playerid][pKills] = PlayerInfo[playerid][pDeaths] = PlayerInfo[playerid][pMoney] = 0; // Zero-up our values.
            // You can add money with PlayerInfo[playerid][pMoney](server) & use GivePlayerMoney for visual view.
            new playerQuery[128]; // You can change the 128, try to stick towards the int*2 method. 64->128->256->512->etc.
            mysql_format(mysql_connectionHandle, playerQuery, sizeof playerQuery, "INSERT INTO accounts (pPassword, pKills, pDeaths, pMoney) VALUES('%s', '%d', '%d', '%d');", PlayerInfo[playerid][pPassword], PlayerInfo[playerid][pKills], PlayerInfo[playerid][pDeaths], PlayerInfo[playerid][pMoney]);
            mysql_query(mysql_connectionHandle, playerQuery); // NOTE! We are not using a threaded query.
            // Done! Create your own spawn method.
        }
    }
    return 1;
}
Let's add some functions towards the player class, shall we?
pawn Code:
public OnPlayerDeath(playerid, killerid, reason)
{
    if(killerid != INVALID_PLAYER_ID)
    {
        ++PlayerInfo[killerid][pKills];
        PlayerInfo[killerid][pMoney] += 100; GivePlayerMoney(killerid, 100);
        PlayerInfo[playerid][pMoney] -= 100; GivePlayerMoney(playerid, -100);
    }
    PlayerInfo[playerid][pDeaths] += 1;
    return 1;
}
Also, we should destroy ORM's by the end of each session, this should be a simply done when a player discconects:
pawn Code:
public OnPlayerDisconnect(playerid, reason)
{
    if(strlen(PlayerInfo[playerid][pUsername]) != 0)
        orm_update(PlayerInfo[playerid][pORM_ID]);
    if(_:PlayerInfo[playerid][pORM_ID] != 0)
        orm_destroy(PlayerInfo[playerid][pORM_ID]);

    // Let's also clear all values!
    for(new i = 0; i <_:pInfo; i++)
    {
        PlayerInfo[playerid][_pInfo:i] = 0;
    }
    return 1;
}
Alright, now you have completed the tutorial and your account system. A few tips before I let you off:
* Security! - Players can exploit the dialog method, use mysql_escape_string. -> https://sampwiki.blast.hk/wiki/MySQL/R33..._escape_string

This should be your final result:
pawn Code:
#include <a_samp>
#include <a_mysql>

#define Skeleton::%1(%2) forward %1(%2); \
                                    public %1(%2)

#define DIALOG_ACCOUNT_LOGIN 1
#define DIALOG_ACCOUNT_REGISTER 2

enum _pInfo
{
    ORM:pORM_ID,
    pPassword[128],
    pKills,
    pDeaths,
    pMoney
}

new PlayerInfo[MAX_PLAYERS][_pInfo];
new mysql_connectionHandle;

public OnGameModeInit()
{
    mysql_connectionHandle = mysql_connect("127.0.0.1", "root", "gameserver_db", "admin");
    if(mysql_errno() == 2003)
    {
        print("[OnGameModeInit] Connection with mysql server couldn't be established, error: 2003");
        SendRconCommand("exit");
        return 1;
    }
    print("[OnGameModeInit] Established connection with mysql server");
}

public OnPlayerConnet(playerid)
{
    new playerQuery[128];
    GetPlayerName(playerid, playerQuery, sizeof(playerQuery));
    mysql_format(mysql_connectionHandle, playerQuery, sizeof(playerQuery), "SELECT * FROM `accounts` WHERE `username` = '%e'", playerQuery);
    mysql_tquery(mysql_connectionHandle, playerQuery, "OnPlayerMysqlFinish", "d", playerid);
    return 1;
}

Skeleton::OnPlayerMysqlFinish(playerid)
{
    if(cache_num_rows() == 1)
    {
        new ORM:ormid = PlayerInfo[playerid][pORM_ID] = orm_create("accounts");

        orm_addvar_string(ormid, PlayerInfo[playerid][pPassword], 128, "Password");
        orm_addvar_string(ormid, PlayerInfo[playerid][pUsername], MAX_PLAYER_NAME+1, "Username");
        orm_addvar_int(ormid, PlayerInfo[playerid][pKills], "Kills");
        orm_addvar_int(ormid, PlayerInfo[playerid][pDeaths], "Deaths");
        orm_addvar_int(ormid, PlayerInfo[playerid][pMoney], "Money");

        orm_setkey(ormid, "Username");
        orm_apply_cache();
       
        ShowPlayerDialog(playerid, DIALOG_ACCOUNT_LOGIN, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below.", "Login", "Exit");
    }
    else if(!cache_num_rows())
    {
        ShowPlayerDialog(playerid, DIALOG_ACCOUNT_REGISTER, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below\nNotice! You'll need to save it to login under this account.", "Register", "Exit");
    }
    else return SendClientMessage(playerid, -1, "Duplicated account, contanct the administartor."), Kick(playerid);
    return 1;
}

public OnDialogResponse(playerid, dialogid, response, listitem, inputtext[])
{
    if(dialogid == DIALOG_ACCOUNT_LOGIN)
    {
        if(!response) return Kick(playerid);
        if(response)
        {
             if(strlen(inputtext) == 0) return ShowPlayerDialog(playerid, DIALOG_ACCOUNT_LOGIN, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below.\nYou can't use an empty password.", "Login", "Exit");
             if(strcmp(inputtext, PlayerInfo[playerid][pPassword], false) == 0) {}
             else return ShowPlayerDialog(playerid, DIALOG_ACCOUNT_LOGIN, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below.\nPassword was incorrect!", "Login", "Exit");
        }
    }
    else if(dialogid == DIALOG_ACCOUNT_REGISTER)
    {
        if(!response) return Kick(playerid);
        if(response)
        {
            if(strlen(inputtext) == 0) return ShowPlayerDialog(playerid, DIALOG_ACCOUNT_REGISTER, DIALOG_STYLE_PASSWORD, "Welcome to my server", "Please insert your password down below\nNotice! You'll need to save it to login under this account.\nYou can't use an empty password!", "Register", "Exit");
            format(PlayerInfo[playerid][pPassword], 128, "%s", inputtext[0]);
            PlayerInfo[playerid][pKills] = PlayerInfo[playerid][pDeaths] = PlayerInfo[playerid][pMoney] = 0;
            new playerQuery[128];
            mysql_format(mysql_connectionHandle, playerQuery, sizeof playerQuery, "INSERT INTO accounts (pPassword, pKills, pDeaths, pMoney) VALUES('%s', '%d', '%d', '%d');", PlayerInfo[playerid][pPassword], PlayerInfo[playerid][pKills], PlayerInfo[playerid][pDeaths], PlayerInfo[playerid][pMoney]);
            mysql_query(mysql_connectionHandle, playerQuery);
        }
    }
    return 1;
}

public OnPlayerDeath(playerid, killerid, reason)
{
    if(killerid != INVALID_PLAYER_ID)
    {
        ++PlayerInfo[killerid][pKills];
        PlayerInfo[killerid][pMoney] += 100; GivePlayerMoney(killerid, 100);
        PlayerInfo[playerid][pMoney] -= 100; GivePlayerMoney(playerid, -100);
    }
    PlayerInfo[playerid][pDeaths] += 1;
    return 1;
}

public OnPlayerDisconnect(playerid, reason)
{
    if(strlen(PlayerInfo[playerid][pORM_ID]) != 0)
        orm_update(PlayerInfo[playerid][pORM_ID]);
    if(_:PlayerInfo[playerid][pORM_ID] != 0)
        orm_destroy(PlayerInfo[playerid][pORM_ID]);

    for(new i = 0; i <_:pInfo; i++)
    {
        PlayerInfo[playerid][_pInfo:i] = 0;
    }
    return 1;
}
Reply


Messages In This Thread
Account system with MySQL [R34][ORM] - by BullseyeHawk - 07.02.2014, 10:41
Re: Account system with MySQL [R34][ORM] - by Konstantinos - 07.02.2014, 10:56
Re: Account system with MySQL [R34][ORM] - by BullseyeHawk - 07.02.2014, 13:59
Re: Account system with MySQL [R34][ORM] - by iZN - 07.02.2014, 14:23
Re: Account system with MySQL [R34][ORM] - by Konstantinos - 07.02.2014, 14:38
Re: Account system with MySQL [R34][ORM] - by BullseyeHawk - 07.02.2014, 15:07
Re: Account system with MySQL [R34][ORM] - by Dorito - 08.02.2014, 06:00
Re: Account system with MySQL [R34][ORM] - by Lordzy - 08.02.2014, 06:37
Re: Account system with MySQL [R34][ORM] - by yvoms - 08.11.2015, 15:25

Forum Jump:


Users browsing this thread: 1 Guest(s)