dobrograd-13-06-2022/garrysmod/addons/admin-sg/lua/modules/sv_mysql.lua

746 lines
23 KiB
Lua
Raw Normal View History

2023-11-16 15:01:19 +05:00
--[[
mysql - 1.0.0
A simple MySQL wrapper for Garry's Mod.
Alexander Grist-Hucker
http://www.alexgrist.com
--]]
--- ## Serverside
-- Handles SQL module selection, SQL queries.
-- @module serverguard.mysql
serverguard.mysql = serverguard.mysql or {};
local QueueTable = {};
local Module = "mysqloo";
local type = type;
local tostring = tostring;
local table = table;
--[[
Phrases
--]]
local MODULE_NOT_EXIST = "[mysql] The %s module does not exist!\n";
--- Query class object that holds information about the current query being built.
-- @type QUERY_CLASS
local QUERY_CLASS = {};
QUERY_CLASS.__index = QUERY_CLASS;
--- **This is called internally, use the corresponding methods.**
-- Creates a new query object.
-- @string tableName The name of the table to manipulate.
-- @string queryType The type of query to start building. (i.e CREATE, UPDATE, INSERT, etc.)
function QUERY_CLASS:New(tableName, queryType)
local newObject = setmetatable({}, QUERY_CLASS);
newObject.queryType = queryType;
newObject.tableName = tableName;
newObject.selectList = {};
newObject.insertList = {};
newObject.updateList = {};
newObject.createList = {};
newObject.whereList = {};
newObject.orderByList = {};
return newObject;
end;
--- Escapes the given string.
-- @string text The string to escape.
-- @treturn string The escaped string.
function QUERY_CLASS:Escape(text)
return serverguard.mysql:Escape(tostring(text));
end;
function QUERY_CLASS:ForTable(tableName)
self.tableName = tableName;
end;
--- Adds a WHERE clause to the SQL query. Analagous to WhereEqual.
-- @string key The key to compare.
-- @string value The value to compare to.
function QUERY_CLASS:Where(key, value)
self:WhereEqual(key, value);
end;
--- Adds a "WHERE `column` = value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereEqual(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` = "..self:Escape(value);
end;
--- Adds a "WHERE `column` != value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereNotEqual(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` != "..self:Escape(value);
end;
--- Adds a "WHERE `column` LIKE value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereLike(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` LIKE "..self:Escape(value);
end;
--- Adds a "WHERE `column` NOT LIKE value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereNotLike(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` NOT LIKE "..self:Escape(value);
end;
--- Adds a "WHERE `column` > value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereGT(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` > "..self:Escape(value);
end;
--- Adds a "WHERE `column` < value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereLT(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` < "..self:Escape(value);
end;
--- Adds a "WHERE `column` >= value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereGTE(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` >= "..self:Escape(value);
end;
--- Adds a "WHERE `column` <= value" clause to the query.
-- You can use multiple of these statements to compare more than one column.
-- @string key The column to compare.
-- @string value The value to compare with.
function QUERY_CLASS:WhereLTE(key, value)
self.whereList[#self.whereList + 1] = "`"..key.."` <= "..self:Escape(value);
end;
--- Adds a "ORDER BY `column` DESC" clause to the query.
-- You can use multiple of these statements to sort by more than one column.
-- @string key The column to sort.
function QUERY_CLASS:OrderByDesc(key)
self.orderByList[#self.orderByList + 1] = "`"..key.."` DESC";
end;
--- Adds a "ORDER BY `column` ASC" clause to the query.
-- You can use multiple of these statements to sort by more than one column.
-- @string key The column to sort.
function QUERY_CLASS:OrderByAsc(key)
self.orderByList[#self.orderByList + 1] = "`"..key.."` ASC";
end;
--- Sets the function to be called when the query finishes.
-- @func queryCallback The callback function to execute.
function QUERY_CLASS:Callback(queryCallback)
self.callback = queryCallback;
end;
--- Adds a "SELECT `column` FROM" clause to the query.
-- You can use multiple of these statements to select more than one column.
-- @string fieldName The column to select.
function QUERY_CLASS:Select(fieldName)
self.selectList[#self.selectList + 1] = "`"..fieldName.."`";
end;
--- Adds a "INSERT INTO `column` VALUES(value)" clause to the query.
-- @string key The column to insert into.
-- @string value The value to insert.
function QUERY_CLASS:Insert(key, value)
self.insertList[#self.insertList + 1] = {"`"..key.."`", self:Escape(value)};
end;
--- Adds a "UPDATE `column` SET key=value" clause to the query.
-- @string key The column to update.
-- @string value The value to update to.
function QUERY_CLASS:Update(key, value)
self.updateList[#self.updateList + 1] = {"`"..key.."`", self:Escape(value)};
end;
--- Populates the table create list with the given keys and values if the query is a CREATE query.
-- @string key The column to create.
-- @string value The value to set it to.
function QUERY_CLASS:Create(key, value)
self.createList[#self.createList + 1] = {"`"..key.."`", value};
end;
--- Sets the primary key for the table.
-- @string key The primary key.
function QUERY_CLASS:PrimaryKey(key)
self.primaryKey = "`"..key.."`";
end;
--- Adds a "LIMIT value" clause to the query.
-- @number value The value to limit the query to.
function QUERY_CLASS:Limit(value)
self.limit = value;
end;
--- Adds a "OFFSET value" clause to the query.
-- @number value The amount to offset by.
function QUERY_CLASS:Offset(value)
self.offset = value;
end;
local function BuildSelectQuery(queryObj)
local queryString = {"SELECT"};
if (type(queryObj.selectList) != "table" or #queryObj.selectList == 0) then
queryString[#queryString + 1] = " *";
else
queryString[#queryString + 1] = " "..table.concat(queryObj.selectList, ", ");
end;
if (type(queryObj.tableName) == "string") then
queryString[#queryString + 1] = " FROM `"..queryObj.tableName.."` ";
else
ErrorNoHalt("[mysql] No table name specified!\n");
return;
end;
if (type(queryObj.whereList) == "table" and #queryObj.whereList > 0) then
queryString[#queryString + 1] = " WHERE ";
queryString[#queryString + 1] = table.concat(queryObj.whereList, " AND ");
end;
if (type(queryObj.orderByList) == "table" and #queryObj.orderByList > 0) then
queryString[#queryString + 1] = " ORDER BY ";
queryString[#queryString + 1] = table.concat(queryObj.orderByList, ", ");
end;
if (type(queryObj.limit) == "number") then
queryString[#queryString + 1] = " LIMIT ";
queryString[#queryString + 1] = queryObj.limit;
end;
return table.concat(queryString);
end;
local function BuildInsertQuery(queryObj)
local queryString = {"INSERT INTO"};
local keyList = {};
local valueList = {};
if (type(queryObj.tableName) == "string") then
queryString[#queryString + 1] = " `"..queryObj.tableName.."`";
else
ErrorNoHalt("[mysql] No table name specified!\n");
return;
end;
for i = 1, #queryObj.insertList do
keyList[#keyList + 1] = queryObj.insertList[i][1];
valueList[#valueList + 1] = queryObj.insertList[i][2];
end;
if (#keyList == 0) then
return;
end;
queryString[#queryString + 1] = " ("..table.concat(keyList, ", ")..")";
queryString[#queryString + 1] = " VALUES ("..table.concat(valueList, ", ")..")";
return table.concat(queryString);
end;
local function BuildUpdateQuery(queryObj)
local queryString = {"UPDATE"};
if (type(queryObj.tableName) == "string") then
queryString[#queryString + 1] = " `"..queryObj.tableName.."`";
else
ErrorNoHalt("[mysql] No table name specified!\n");
return;
end;
if (type(queryObj.updateList) == "table" and #queryObj.updateList > 0) then
local updateList = {};
queryString[#queryString + 1] = " SET";
for i = 1, #queryObj.updateList do
updateList[#updateList + 1] = queryObj.updateList[i][1].." = "..queryObj.updateList[i][2];
end;
queryString[#queryString + 1] = " "..table.concat(updateList, ", ");
end;
if (type(queryObj.whereList) == "table" and #queryObj.whereList > 0) then
queryString[#queryString + 1] = " WHERE ";
queryString[#queryString + 1] = table.concat(queryObj.whereList, " AND ");
end;
if (type(queryObj.offset) == "number") then
queryString[#queryString + 1] = " OFFSET ";
queryString[#queryString + 1] = queryObj.offset;
end;
return table.concat(queryString);
end;
local function BuildDeleteQuery(queryObj)
local queryString = {"DELETE FROM"}
if (type(queryObj.tableName) == "string") then
queryString[#queryString + 1] = " `"..queryObj.tableName.."`";
else
ErrorNoHalt("[mysql] No table name specified!\n");
return;
end;
if (type(queryObj.whereList) == "table" and #queryObj.whereList > 0) then
queryString[#queryString + 1] = " WHERE ";
queryString[#queryString + 1] = table.concat(queryObj.whereList, " AND ");
end;
if (type(queryObj.limit) == "number") then
queryString[#queryString + 1] = " LIMIT ";
queryString[#queryString + 1] = queryObj.limit;
end;
return table.concat(queryString);
end;
local function BuildDropQuery(queryObj)
local queryString = {"DROP TABLE"}
if (type(queryObj.tableName) == "string") then
queryString[#queryString + 1] = " `"..queryObj.tableName.."`";
else
ErrorNoHalt("[mysql] No table name specified!\n");
return;
end;
return table.concat(queryString);
end;
local function BuildCreateQuery(queryObj)
local queryString = {"CREATE TABLE IF NOT EXISTS"};
if (type(queryObj.tableName) == "string") then
queryString[#queryString + 1] = " `"..queryObj.tableName.."`";
else
ErrorNoHalt("[mysql] No table name specified!\n");
return;
end;
queryString[#queryString + 1] = " (";
if (type(queryObj.createList) == "table" and #queryObj.createList > 0) then
local createList = {};
for i = 1, #queryObj.createList do
if (Module == "sqlite") then
createList[#createList + 1] = queryObj.createList[i][1].." "..string.gsub(string.gsub(string.gsub(queryObj.createList[i][2], "AUTO_INCREMENT", ""), "AUTOINCREMENT", ""), "INT ", "INTEGER ");
else
createList[#createList + 1] = queryObj.createList[i][1].." "..queryObj.createList[i][2];
end;
end;
queryString[#queryString + 1] = " "..table.concat(createList, ", ").."," ;
end;
if (type(queryObj.primaryKey) == "string") then
queryString[#queryString + 1] = " PRIMARY KEY";
queryString[#queryString + 1] = " ("..queryObj.primaryKey..")";
end;
queryString[#queryString + 1] = " )";
return table.concat(queryString);
end;
--- Executes the query on the database.
-- @bool[opt] bQueueQuery Whether or not to queue the query. Defaults to false.
function QUERY_CLASS:Execute(bQueueQuery)
local queryString = nil;
local queryType = string.lower(self.queryType);
if (queryType == "select") then
queryString = BuildSelectQuery(self);
elseif (queryType == "insert") then
queryString = BuildInsertQuery(self);
elseif (queryType == "update") then
queryString = BuildUpdateQuery(self);
elseif (queryType == "delete") then
queryString = BuildDeleteQuery(self);
elseif (queryType == "drop") then
queryString = BuildDropQuery(self);
elseif (queryType == "create") then
queryString = BuildCreateQuery(self);
end;
if (type(queryString) == "string") then
if (!bQueueQuery) then
return serverguard.mysql:RawQuery(queryString, self.callback);
else
return serverguard.mysql:Queue(queryString, self.callback);
end;
end;
end;
--- Creates a new SELECT query.
-- @string tableName The name of the table.
-- @treturn QUERY_CLASS The query object.
function serverguard.mysql:Select(tableName)
return QUERY_CLASS:New(tableName, "SELECT");
end;
--- Creates a new INSERT INTO query.
-- @string tableName The name of the table.
-- @treturn QUERY_CLASS The query object.
function serverguard.mysql:Insert(tableName)
return QUERY_CLASS:New(tableName, "INSERT");
end;
--- Creates a new UPDATE query.
-- @string tableName The name of the table.
-- @treturn QUERY_CLASS The query object.
function serverguard.mysql:Update(tableName)
return QUERY_CLASS:New(tableName, "UPDATE");
end;
--- Creates a new DELETE query.
-- @string tableName The name of the table.
-- @treturn QUERY_CLASS The query object.
function serverguard.mysql:Delete(tableName)
return QUERY_CLASS:New(tableName, "DELETE");
end;
--- Creates a new DROP query.
-- @string tableName The name of the table.
-- @treturn QUERY_CLASS The query object.
function serverguard.mysql:Drop(tableName)
return QUERY_CLASS:New(tableName, "DROP");
end;
--- Creates a new CREATE TABLE query.
-- @string tableName The name of the table.
-- @treturn QUERY_CLASS The query object.
function serverguard.mysql:Create(tableName)
return QUERY_CLASS:New(tableName, "CREATE");
end;
--- Connects to the SQL database with the given information.
-- @string host The host name.
-- @string username The username.
-- @string password The password.
-- @string database The name of the database.
-- @number[opt] port The port to use. Defaults to 3306.
-- @string[opt] socket The unix socket to use. Defaults to none.
-- @number flags The query flags to use. Usually these aren't needed.
function serverguard.mysql:Connect(host, username, password, database, port, socket, flags)
if (!port) then
port = 3306;
end;
if (Module == "tmysql4") then
if (type(tmysql) != "table") then
require("tmysql4");
end;
if (tmysql) then
local errorText = nil;
self.connection, errorText = tmysql.initialize(host, username, password, database, port, socket, flags);
if (!self.connection) then
self:OnConnectionFailed(errorText);
else
self:OnConnected();
end;
else
ErrorNoHalt(string.format(MODULE_NOT_EXIST, Module));
end;
elseif (Module == "mysqloo") then
if (type(mysqloo) != "table") then
require("mysqloo");
end;
if (mysqloo) then
self.connection = mysqloo.connect(host, username, password, database, port, socket or "", (flags or 0));
self.connection.onConnected = function(database)
serverguard.mysql:OnConnected();
end;
self.connection.onConnectionFailed = function(database, errorText)
serverguard.mysql:OnConnectionFailed(errorText);
end;
self.connection:connect();
else
ErrorNoHalt(string.format(MODULE_NOT_EXIST, Module));
end;
elseif (Module == "sqlite") then
serverguard.mysql:OnConnected();
end;
end;
--- Sends a query to the database without any preprocessing. You should not use this unless
-- you know what you're doing.
-- @string query The query to run.
-- @func callback The function to call when the query finishes.
-- @number flags The flags to set on the query. Usually this isn't needed.
function serverguard.mysql:RawQuery(query, callback, flags, ...)
if (!self.connection and Module != "sqlite") then
self:Queue(query);
end;
if (Module == "tmysql4") then
local queryFlag = flags or QUERY_FLAG_ASSOC;
self.connection:Query(query, function(result)
local queryStatus = result[1]["status"];
if (queryStatus) then
if (type(callback) == "function") then
local bStatus, value = pcall(callback, result[1]["data"], queryStatus, result[1]["lastid"]);
if (!bStatus) then
ErrorNoHalt(string.format("[mysql] MySQL Callback Error!\n%s\n", value));
end;
end;
else
ErrorNoHalt(string.format("[mysql] MySQL Query Error!\nQuery: %s\n%s\n", query, result[1]["error"]));
end;
end, queryFlag, ...);
elseif (Module == "mysqloo") then
local queryObj = self.connection:query(query);
queryObj:setOption(mysqloo.OPTION_NAMED_FIELDS);
queryObj.onSuccess = function(queryObj, result)
if (callback) then
local bStatus, value = pcall(callback, result, self.connection:status(), queryObj:lastInsert());
if (!bStatus) then
ErrorNoHalt(string.format("[mysql] MySQL Callback Error!\n%s\n", value));
end;
end;
end;
queryObj.onError = function(queryObj, errorText, ...)
if (self.connection:status() != mysqloo.DATABASE_CONNECTED) then
self:Queue(query, callback);
self.connection:connect();
return;
end;
ErrorNoHalt(string.format("[mysql] MySQL Query Error!\nQuery: %s\n%s\n", query, errorText));
end;
queryObj:start();
elseif (Module == "sqlite") then
local result = sql.Query(query);
if (result == false) then
ErrorNoHalt(string.format("[mysql] SQL Query Error!\nQuery: %s\n%s\n", query, sql.LastError()));
else
if (callback) then
local bStatus, value = pcall(callback, result);
if (!bStatus) then
ErrorNoHalt(string.format("[mysql] SQL Callback Error!\n%s\n", value));
end;
end;
end;
else
ErrorNoHalt(string.format("[mysql] Unsupported module \"%s\"!\n", Module));
end;
end;
--- Adds a query to the queue to be executed once previously pending ones have finished.
-- @string queryString The query to run.
-- @func callback The function to run when the query executes.
function serverguard.mysql:Queue(queryString, callback)
if (type(queryString) == "string") then
QueueTable[#QueueTable + 1] = {queryString, callback};
end;
end;
--- Escapes the string to be used in a query.
-- @string text The string to escape.
-- @bool[opt] bNoQuotes Whether or not to NOT wrap with quotes. Defaults to false.
function serverguard.mysql:Escape(text, bNoQuotes)
if (self.connection) then
if (Module == "tmysql4") then
if (bNoQuotes) then
return self.connection:Escape(text);
else
return string.format("\"%s\"", self.connection:Escape(text));
end;
elseif (Module == "mysqloo") then
if (bNoQuotes) then
return self.connection:escape(text);
else
return string.format("\"%s\"", self.connection:escape(text));
end;
end;
else
return sql.SQLStr(text, bNoQuotes);
end;
end;
--- Disconnects from the SQL database.
function serverguard.mysql:Disconnect()
if (self.connection) then
if (Module == "tmysql4") then
return self.connection:Disconnect();
end;
end;
end;
-- Function that's ran for checking the query queue.
function serverguard.mysql:Think()
if (#QueueTable > 0) then
if (type(QueueTable[1]) == "table") then
local queueObj = QueueTable[1];
local queryString = queueObj[1];
local callback = queueObj[2];
if (type(queryString) == "string") then
self:RawQuery(queryString, callback);
end;
table.remove(QueueTable, 1);
end;
end;
end;
--- A function to set the SQL module to be used. This should not be set unless
-- you know what you're doing!
function serverguard.mysql:SetModule(moduleName)
Module = moduleName;
end;
-- Called when the database connects sucessfully.
function serverguard.mysql:OnConnected()
serverguard.PrintConsole("[mysql] Connected to the database!\n");
self.UpgradeCount = 0;
self.FinishedUpgrades = 0
hook.Call("serverguard.mysql.CreateTables", nil);
hook.Call("serverguard.mysql.UpgradeSchemas", nil, function()
self.FinishedUpgrades = self.FinishedUpgrades + 1
if (self.UpgradeCount == self.FinishedUpgrades) then
hook.Call("serverguard.mysql.OnConnected", nil);
end
end);
end;
-- Called when the database connection fails.
function serverguard.mysql:OnConnectionFailed(errorText)
ErrorNoHalt("[mysql] Unable to connect to the database!\n"..errorText.."\n");
hook.Call("serverguard.mysql.DatabaseConnectionFailed", nil, errorText);
end;
include 'sg_mysql.lua'
hook.Add("serverguard.Initialize", "serverguard.mysql.Initialize", function()
local config = SERVERGUARD.MySQL
if (config and config.enabled == 1) then
if (config.module != Module) then
Module = config.module;
end;
serverguard.mysql:Connect(config.host, config.username, config.password, config.database, config.port, config.unixsocket);
return;
end;
serverguard.mysql:Connect();
end);
hook.Add("serverguard.mysql.CreateTables", "serverguard.mysql.CreateTables", function()
local USERS_TABLE_QUERY = serverguard.mysql:Create("serverguard_users");
USERS_TABLE_QUERY:Create("id", "INTEGER NOT NULL AUTO_INCREMENT");
USERS_TABLE_QUERY:Create("steam_id", "VARCHAR(25) NOT NULL");
USERS_TABLE_QUERY:Create("rank", "VARCHAR(255) DEFAULT \"\" NOT NULL");
USERS_TABLE_QUERY:Create("name", "VARCHAR(255) NOT NULL");
USERS_TABLE_QUERY:Create("last_played", "INT(11) NOT NULL");
USERS_TABLE_QUERY:Create("data", "TEXT");
USERS_TABLE_QUERY:PrimaryKey("id");
USERS_TABLE_QUERY:Execute();
local BANS_TABLE_QUERY = serverguard.mysql:Create("serverguard_bans");
BANS_TABLE_QUERY:Create("id", "INTEGER NOT NULL AUTO_INCREMENT");
BANS_TABLE_QUERY:Create("steam_id", "VARCHAR(25) NOT NULL");
BANS_TABLE_QUERY:Create("community_id", "TEXT NOT NULL");
BANS_TABLE_QUERY:Create("player", "VARCHAR(255) NOT NULL");
BANS_TABLE_QUERY:Create("reason", "TEXT NOT NULL");
BANS_TABLE_QUERY:Create("start_time", "INT(11) NOT NULL");
BANS_TABLE_QUERY:Create("end_time", "INT(11) NOT NULL");
BANS_TABLE_QUERY:Create("admin", "VARCHAR(255) NOT NULL");
BANS_TABLE_QUERY:Create("ip_address", "VARCHAR(15) NOT NULL");
BANS_TABLE_QUERY:PrimaryKey("id");
BANS_TABLE_QUERY:Execute();
local SCHEMA_TABLE_QUERY = serverguard.mysql:Create("serverguard_schema");
SCHEMA_TABLE_QUERY:Create("id", "VARCHAR(31) NOT NULL");
SCHEMA_TABLE_QUERY:Create("version", "INT(11) NOT NULL DEFAULT '0'");
SCHEMA_TABLE_QUERY:PrimaryKey("id");
SCHEMA_TABLE_QUERY:Execute()
end);
-- Upgrades Schemas to latest
-- input: database name, table of strings for each version, callback when done
function serverguard.mysql:UpgradeSchema(id, schemas, callback)
self.UpgradeCount = self.UpgradeCount + 1
self:Queue("SELECT `version` from `serverguard_schema` where `id` = "..self:Escape(id)..";",
function(query)
local version = tonumber(query and query[1] and query[1].version or 0)
if (version == #schemas) then
callback(false)
else
local complete_query = "";
for i = 1, #schemas do
table.insert(schemas[i], "REPLACE INTO `serverguard_schema` (`id`, `version`) VALUES ("..serverguard.mysql:Escape(id)..", '"..i.."');");
end
local function UpgradeRecursion(upgrade, version, index)
if (not upgrade[version]) then
return callback(true)
end
local final_version_query = index == #upgrade[version];
local next_version = (final_version_query and 1 or 0) + version;
local next_index = final_version_query and 1 or index + 1
if (not upgrade[next_version]) then
return callback(true)
end
serverguard.mysql:Queue(upgrade[next_version][next_index], function()
UpgradeRecursion(upgrade, next_version, next_index);
end);
end
UpgradeRecursion(schemas, version + 1, 0)
end
end
);
end
timer.Create("serverguard.mysql.Think", 1, 0, function()
serverguard.mysql:Think();
end);