Skip to content

Latest commit

 

History

History
729 lines (563 loc) · 27.5 KB

README.md

File metadata and controls

729 lines (563 loc) · 27.5 KB

XLIB

A collection of snippets and tools for Garry's Mod development. (Short for Xavier's Library)

GLDoc documentation is currently underway. View ./docs/ It is currently not hosted anywhere but will be put on GH pages once more documentation is written. Feel free to submit any pull requests to improve the documentation!

Credential Store

Preamble: Most Garry's Mod scripts configurations hardcode credentials. This is less than ideal for source controlled projects.*

This aims to solve this issue by allowing you to store all of your credentials in a schemaless GarrysMod/garrysmod/CREDENTIAL_STORE Valve VDF file in the root of your Garry's Mod installation. In turn, this allows you to maintain separate configuration files for each environment (development, staging, production).

The VDF file is exposed as the global CREDENTIALS table in Lua.

Configuration

CREDENTIAL_STORE.txt

credentials
{
    // Here are the fields XLib uses
    environment "production"  // Defaults to production. Setting to development makes IsTestServer() return true
    extended "1"  // Enables parts of XLib that are meant to be for "internal use" AKA not guaranteed to not cause any conflicts outside of our environments.

    // DEPRECATED - USING `environment` WILL SET THIS VALUE
    // development_mode "1" // Makes IsTestServer() return true
    // production "0" // No particular use, for now. Slightly redundant, but you can use `CREDENTIALS.production` as a predicate in your code.

    // Example config for use with GDBC, optional
    mysql
    {
        sample
        {
            host "localhost"
            db   "cats"
            user "cat_lover"
            pass "CatLover1546"
            port "3306"
        }
    }

    // You can also have any custom values here to configure your custom scripts.
    example_script {
        url "https://test-endpoint.com/"
        apikey "some_api_key"
    }
}

Lua Usage

require "credentialstore"

if CREDENTIALS.environment == "development" then  -- or simply IsTestServer() if using XLIB Extended
    print("Server is running in development mode!")
end

if not CREDENTIALS.mysql.sample then
    print("MySQL credentials for `sample` database not found, resorting to sqlite.")
else
    print("Login info for `sample`", SPrintTable(CREDENTIALS.mysql.sample))
end

GDBC

"Garry's Mod Database Connector" - Spawned by the disdain of hardcoding MySQL queries everywhere. Originally featured its own libmysql <-> Lua bindings but was retrofitted for use with MysqlOO for ease of maintainability.

Some of the main features include:

  • Unique chaining control flow avoids callback hell
  • Prepared Statements (Actual prepared statements OR falls back to formatting the SQL query and sending it if connect.usePreparedStatememts=false)
  • MySQL Connection Pool (Set connect.threads= >1)
  • Database Versioning (Through a basic key-value config table added to each schema)
  • Schema Migrations (Players are not allowed to join until all migrations are complete!)

Declarative Syntax

For connection info, queries and migrations

schema "sample"
{
    connect
    {
        host = CREDENTIALS.mysql.sample.host,
        user = CREDENTIALS.mysql.sample.user,
        pass = CREDENTIALS.mysql.sample.pass,
        database = CREDENTIALS.mysql.sample.db,
        port = CREDENTIALS.mysql.sample.port,

        --[=====================================================================[
            Whether this database connection should use prepared statements.

            Before you enable this, ensure that all of your queries are
            compatible for use within prepared statements.

            https://mariadb.com/kb/en/prepare-statement/#permitted-statements/

            If you would like to benefit from prepared statements while still
            maintaining the ability to use statements not yet implemented,
            use :queryraw() along with :prepare():

            DB.schema_name
                :queryraw(DB.schema_name.table_name.query_name:prepare(...))
                :exec()

        ]=====================================================================]
        usePreparedStatements = false,

        --[=====================================================================[
            CAREFUL! Only queries contained within the same GDBC Query Sequence
            are guaranteed to be executed in order with strong consistency.

            In single-connection mode, initiating two query sequences one after
            the other will guarantee that the first sequence will finish before
            the second

            IF YOU'RE UNSURE WHAT THIS MEANS, LEAVE THREADS=1 TO AVOID ISSUES.
            I'm serious! Blindly enabling this feature without having designed
            your database interaction logic with this in mind could lead to
            bugs that will be nearly impossible to diagnose.

            Note:
            By default, GDBC already initiates a second connection that can be
            specifically targeted by calling :low() on the sequence.
            I recommend you call :low() on non-critical, write-intensive spammy
            queries (such as statistics collection or logging) and reevaluating
            performance before resorting to connection pools.
        ]=====================================================================]

        -- For the reasons listed above, I recommend leaving this parameter out
        -- entirely if you plan on distributing your script to avoid other
        -- people from being tempted to blindly modify its value.
        -- threads = 1,  -- Set > 1 to enable connection pool. READ ABOVE!
    };

    table "players"
    {
        get = [[SELECT id FROM players WHERE steamid=?]];
        steamid = [[SELECT id, steamid from players WHERE id=?]];
        insert = [[INSERT INTO players (steamid) VALUES (?)]];
    };

    table "player_info"
    {
        get = [[SELECT * FROM player_info WHERE player_id=?]];

        setLuck = [[UPDATE player_info SET luck=?]];
        setHappiness = [[UPDATE player_info SET happiness=?]];
        setRPName = [[UPDATE player_info SET rpname=?]];
        setSteamName = [[UPDATE player_info SET steamname=?]];

        getNames = [[
            SELECT steamname, rpname FROM player_info WHERE player_id=?
        ]];

        getNamesSteamID = [[
            SELECT player_info.rpname, player_info.steamname
            FROM player_info
            INNER JOIN players
            ON player_info.player_id = players.id
            WHERE players.steamid=?
        ]];
    };

    migration (1)
        [[
            -- -----------------------------------------------------
            -- Table `players`
            -- -----------------------------------------------------
            CREATE TABLE IF NOT EXISTS `players` (
              `id` INT UNSIGNED NOT NULL AUTO_INCREMENT,
              `steamid` VARCHAR(25) NOT NULL,
              PRIMARY KEY (`id`),
              UNIQUE INDEX `steamid_UNIQUE` (`steamid` ASC),
              INDEX `steamid_INDEX` (`steamid` ASC))
            ENGINE = InnoDB;

            -- -----------------------------------------------------
            -- Table `player_info`
            -- -----------------------------------------------------
            CREATE TABLE IF NOT EXISTS `player_info` (
              `player_id` INT UNSIGNED NOT NULL,
              `steamname` VARCHAR(45) NULL,
              `rpname` VARCHAR(45) NULL,
              `happiness` INT UNSIGNED NULL DEFAULT 75,
              `luck` INT NULL DEFAULT 100,
              PRIMARY KEY (`player_id`),
              UNIQUE INDEX `player_id_UNIQUE` (`player_id` ASC),
              CONSTRAINT `fk_player_info.player_id:players.id`
                FOREIGN KEY (`player_id`)
                REFERENCES `players` (`id`)
                ON DELETE NO ACTION
                ON UPDATE NO ACTION)
            ENGINE = InnoDB;
        ]];

    migration (2)
        [[
            -- -----------------------------------------------------
            -- Steam updated their maximum display name length.
            -- Update `player_info`.`steamname` accordingly.
            -- -----------------------------------------------------
            ALTER TABLE `player_info` MODIFY `steamname` VARCHAR(64);
        ]];

    -- Support for function-based migration for additional logic
    migration (3) (function(db, callback)
        db()
            :queryraw([[
                SELECT EXISTS (
                    SELECT * FROM `INFORMATION_SCHEMA`.`SESSION_STATUS`
                    WHERE VARIABLE_NAME = 'FEATURE_SYSTEM_VERSIONING'
                ) AS support;
            ]])
                :result(function(q, row)
                    -- This particular check could be done with pure SQL
                    -- But this is for demonstration purposes
                    if row.support == 1 then
                        return "alter"
                    end
                end)

            :queryraw "alter" [[
                ALTER TABLE `player_inventory` ADD SYSTEM VERSIONING;
            ]]

            :success(function(q)
                callback(true)
            end)

            :fail(function(q, err, sql, traceback)
                callback(false, "Migration SQL error occurred: "..err)
            end)

            :exec()
    end);
}

Control Flow

require "gdbc"

SAMPLEDB = {}
function SAMPLEDB:InitPlayer(ply)
    local steamid = ply:SteamID()
    local player_id

    -- Initiate a query chain
    DB.sample()
        -- Add a sequential query to the chain
        -- If your query returns more than one row, use queryall instead
        :query(DB.sample.players.get(steamid))
            :result(function(q, row)
                -- Return next_action, vararg parameters.
                -- If next_action is nil, execute the next sequential query or operation
                -- The remaining returned objects are passed to the sequence or the query object.
                return "on_id", row['id']
            end)

            :empty(function()
                -- The entry does not exist in the database yet, create it!
                return "insert", steamid
            end)

        -- Add a named query to the chain
        -- You can pass arguments to the query object if they are knonwn in advance
        :query "insert" (DB.sample.players.insert(steamid))
            :empty(function(q, last_insert)
                return "on_id", last_insert
            end)

        -- Store id into a local variable for future use
        :procedure "on_id" (function(q, id)
            player_id = id
            return "get_info", id
        end)

        -- Add another named query to the chain
        -- This time we don't know the player_id, so it is the caller's job to pass it in its return statement
        :query "get_info" (DB.sample.player_info.get)
            :result(function(q, row)
                -- We have the data, proceed!
                return "done", row
            end)

            :empty(function()
                -- The entry does not exist in the database yet, create it!
                return "default_info"
            end)

        :procedure "default_info" (function(q)
            if not IsValid(ply) then
                return false  -- Returning false will abort the query chain
            end

            -- Asynchronously populate `sample`.`player_info` with player_id and move on
            local info = SAMPLEDB:SaveDefaultInfo(player_id)

            return "done", info
        end)

        :procedure "done" (function(q, info)
            if not IsValid(ply) then
                return  -- The player managed to disconnect during the query chain
            end

            ply.SampleInfo = info
            ply.SAMPLEDB_ID = player_id
            hook.Run("SAMPLEDB:PlayerReady", ply)
        end)

        -- Enable logging the SQL queries sent from this particular chain only
        :log()

        -- Execute this query sequence on the single-threaded low-priority queue
        :low()

        -- Used to delay flushing of write-intensive queries to the latest
        -- dataset based on unique_id after max of time_in_seconds
        :throttle("unique_id", time_in_seconds or 1)

        -- Execute the chain. This will start with the first sequential query.
        :exec()
end

function SAMPLEDB:SaveDefaultInfo(player_id)
    local default_info = {
        steamname = "UNKNOWN",
        rpname = "John Doe",
        luck = 100,
        happiness = 100,
    }

    -- TODO: Save default_info asynchronously here with a different query chain

    return default_info
end

hook.Add("PlayerInitialSpawn", "SampleAddonHook", function(ply)
    SAMPLEDB:InitPlayer(ply)
end)

hook.Add("SAMPLEDB:PlayerReady", "SampleAddonHook", function(ply)
    -- Player's `sample` database unique ID is now ready for use by scripts that require it!
    print("Player", ply:Nick(), "successfully loaded from database with ID:", ply.SAMPLEDB_ID,
          "Luck:", ply.SampleInfo.luck)
end)

--[[

    THESE UPCOMING TWO HOOKS ARE REQUIRED TO GET GDBC WORKING!
    "GDBC:InitSchemas", "GDBC:Ready"
    Read the sample code below for more information.

    The rest of this file (above) is only
    for documentation purposes and should
    be replaced with your own implementation.

]]

-- Workaround for GM.Think to get called even with no players on the server
-- Required for MySQLOO Query callbacks to be processed.\
-- Enable this if you want need to load information from the database
-- before allowing player connections.
RunConsoleCommand("sv_hibernate_think", "1")

-- GDBC exposes two hooks, GDBC:InitSchemas and GDBC:Ready
hook.Add("GDBC:Ready", "SAMPLEDB:Ready", function()
    -- The database has connected and migrations have all been successfully executed.
    -- Execute any server initialization queries here
end)

hook.Add("GDBC:InitSchemas", "SAMPLEDB:InitSchemas", function()
    -- GDBC is ready to load schemas and start executing migrations.
    -- GDBC.LoadSchema("path/to/schema.lua")

    -- Note that GDBC.LoadSchema uses CompileFile internally.
    -- This means that file paths are relative to the Lua mount path, NOT the current file!
    -- This also means you should properly namespace these files to avoid conflicts.

    -- e.g file in garrysmod/addons/my_cool_addon/lua/cool_addon/database/schema.lua
    GDBC.LoadSchema("cool_addon/database/schema.lua")

    -- e.g file in garrysmod/gamemodes/gm_name/gamemode/database/schema.lua
    GDBC.LoadSchema((GM or GAMEMODE).FolderName .. "/gamemode/database/schema.lua")
end)

XLoader

"Xavier's Loader" - Predictable automatic recursive file includer

Some of the main features include:

  • Automatically include() and AddCSLuaFile() files in the specified directory
  • Auto-refresh aware Upstream bug
  • Guaranteed predictable load order

The include order is as follows:

  1. All sh_*.lua1 files for the current directory, sorted alphanumerically
  2. All sv_*.lua or cl_*.lua files for the current directory, sorted alphanumerically
  3. Recurse steps 1, 2, 3 in subfolders. Subfolders of subfolders take priority.

Conditional Networking

Sometimes you might need to dynamically enable/disable shared (or purely clientside) scripts based on a specific condition on server startup.

Introducing BlockCSLuaFile()! This function can be called within any sh_* file to mark the current file to be treated as a server-only file. This means that clients will not execute this file. It won't even be networked!

You can also leverage this for clientside-only files (click to expand)
if SERVER then
    -- If for whatever reason you don't want this file 
    if not testing_something then
        BlockCSLuaFile()
    end
return

end
-- Run clientside-only code that needs to execute strictly immediately on script evaluation here

When should you use this? The answer is: not very often. You should only resort to this when you need dynamic control over scripts that should run immediately on Lua state initialization.

Example uses of conditional networking:

Usage

require "xloader"
xloader("sample_addon", function(f) include(f) end)

-- Breakdown:
-- First argument is the directory relative to the Lua mount path
-- Second argument is required boilerplate to make it autorefresh-aware. View linked bug report above.

XLIB.Test

This is a basic testing suite including a few nifty tools including string comparison and diff output

Assertions support diff output string and nested table comparisons! It uses XLIB.Compare and XLIB.CompareString under the hood.

  • String comparisons will output pretty diff info on mismatch
  • Table comparisons will print out the first key path to differ between the two tables

A test group is one call to XLIB.Test. Each test group can have multiple assertions.

  • Players will be prevented from joining the server by a CheckPassword hook until all test groups succeed
  • Note that a test group is marked as "complete" upon the first assertion. If the first assertion is truthy and subsequent failures from a timer or callback are falsy, the group will be considered as passing until a falsy assertion is made. BE CAREFUL as poorly structuring your tests could allow players to join a broken server. e.g assert(true) timer.simple(5, function() assert(false) end) is very bad.
  • As a rule of thumb, you should separate your synchronous and asynchronous tests into different groups, and only have one asynchronous assertion per test group.

Usage:

XLIB.Test("API is reachable", function(assert, LOG, ERR)
    -- LOG(...) -> Pretty-prints a log message to the test data
    -- ERR(...) -> Pretty-prints an error message to the test data (DOESN'T FAIL THE TEST!)

    -- assert(reason, result, [wanted])
    -- If `result` is a boolean and compareTo is nil, `wanted` becomes the predicate to the assertion
    -- If `wanted` is provided, XLIB.Compare or XLIB.CompareString will be used.

    LOG("Running HTTP fetch")
    http.Fetch("http://ifconfig.co/", function(body)
        -- If this is truthy, the test group is marked as successful
        assert("Visible server external IP is correct", body, "123.123.123.123")

        -- DON'T DESIGN LIKE THIS! But be aware that the functionality does work.
        -- As a rule of thumb, use ONE test group per asynchronous test
        timer.Simple(60, functin()
            -- Subsequent assertions will also be logged and lock the server if it fails.
            assert("I decided to break everything", false)
        end)


    end, function(err)
        ERR("HTTP Failed with reason", err)
        assert("HTTP Failed", false)
    end)
end)

XLIB.Timer

This is an easy-to-use timing tool for basic profiling purposes. XLIB.Time(identifier) The identifier does not need to be unique. It will be used as a prefix for printing results. It allows you to log different "stages" during the profiling process and will time each step.

If you want precise, multi-run profiling, check out FProfiler instead. The main goal of this small module is to isolate lenghty processes once you have already figured out the bottleneck.

    clock:Start()  -- Starts the timer. Optional, call :Log() to start as well. Returns clock object.
    clock:Log(event)  -- Logs an event at this point in time
    clock("evt")      -- Alias to clock:Log() allows for `clock "Something"`
    clock:Print(...)  -- Queues the print to be printed with prefix at the end of execution.
                      -- It will print instantly if the clock is not started

    clock:Finish()    -- Finishes the timer and prints out the results
-- If you intend to re-use this clock, you can store it locally outside this scope
-- local clock = XLIB.Time("pairs vs ipairs vs for")
-- then call start in your event loop or something
-- clock:Start()

-- Alternatively, :Start() will return the clock object as well for one seamless operation
local clock = XLIB.Time("pairs vs ipairs vs for"):Start()

local b = 0
local str = string.rep("TEST", 10^3)
local t = {}
for i=1, 1000 do
    table.Add(t, {string.byte(str, 1, str:len())})
end

-- Clock:Print(...) will defer printing (with timer prefix) to the end of the
-- timing process to keep the output linear and to reduce I/O bottlenecks
clock:Print("Testing with", #t, "iterations")

-- Call :Log() after an action to add a memo at this particular timestamp
-- Calling :Log() will also call :Start() if it has not yet been started.
clock:Log("Build test data")

b = 0
for k, v in pairs(t) do
    b = b + v
end
clock:Log("pairs")

b = 0
for k, v in ipairs(t) do
    b = b + v
end
clock:Log("ipairs")

b = 0
for i=1, #t do
    b = b + t[i]
end
clock "for"  -- Using alternative, faster to write for quick iterations

b = 0
local l = #t
for i=1, l do
    b = b + t[i]
end
clock "for w/ cached len"

clock:Finish()

Output


[pairs vs ipairs vs for]  Testing with 4000000 iterations
[pairs vs ipairs vs for]  Build test data took 0.879236  - T+0.879236
[pairs vs ipairs vs for]  pairs took 0.023838  - T+0.903073
[pairs vs ipairs vs for]  ipairs took 0.004902  - T+0.907975
[pairs vs ipairs vs for]  for took 0.005205  - T+0.91318
[pairs vs ipairs vs for]  for w/ cached len took 0.003529  - T+0.916708
[pairs vs ipairs vs for]  Finished in  0.91671

NetWrapper

This library packages my fork of netwrapper.

NetWrapper is a lightweight, bandwidth-focused wrapper around the existing net library. You can view its documentation here as well as examples here

XLIB Extended

This is a part of XLIB disabled by default for use in packaged applications. To enable, simply set extended "1" in your CREDENTIAL_STORE

The current features include:

DevCommand

DevCommand(cmdname, callback(ply, cmd, args, arg_str), realm=SERVER)

This requires a function IsDeveloper (not provided by XLIB at this time) to be present on the Player metatable. Functionally similar to concommand.Add, this takes care of the boilerplate of doing concommand authentication.

XLIB Extended also ships with a few DevCommands:

Command Realm
lua Runs on the server
luacl Runs on your client
luash Runs on both client and server
luapl <PID> Runs on specified player's client where PID is UserID from status command
luaplall Runs client-side for all connected players
luashpl Runs server-side and on specified player's client where PID is UserID from status command
luashall Runs server-side and client-side for all connected players
manybots Spawns 10 bots
nobots Kicks all bost from the server
reloadmap Runs changelevel to the current map
testserver.toggle Temporarily makes IsTestServer() return false in a development environment
xlib_teststatus Prints ran tests as well as their output
xlib_cleartests Clears failed test status and unlocks the server
mapmanager.reload Reloads mapmanager settings without reloading the map
credentialstore.reload Reloads CREDENTIAL*STORE.txt without reloading the map
gdbc_log Temporarily enables GDBC logging
gdbc_reload Reloads all schema files (InitSchemas hook)
unblock_cvars Remove all FCVAR* restrictions on all cvars (requires cvarsx module). Can be ran on client and server
xlib_startuperrors Prints lua errors that occured during server start up
xlib_clearstartuperrors Clears startup lua errors and unlocks server
xlib_testhttp DEPRECATED - Enables logging for functions making HTTP calls before initialization and reloads the map
rmpanel Removes the VGUI panel under the cursor (useful if a panel becomes stuck after failing initialization)

These evaluate the passed string on either the server or client without needing RCON access or sv_allowcslua=1 respectively.

When executing the server-side lua command, any output from print, Msg, MsgC, PrintTable will be redirected to the client's console. This is also true for any syntax or runtime errors.

When using any command that runs on a client other than yours, the output will be piped back to the console of the initial command's runner in pink labeled with the client's name and Steam ID.

It also provides some useful shorthand globals to help you avoid the low character count restriction:

Name Object
me (SERVER) Player running the command
me (CLIENT) LocalPlayer()
metr me:GetEyeTrace()
metrent me:GetEyeTrace().Entity
wep me:GetActiveWeapon
veh me:GetVehicle()
plys player.GetAll()
dir(x) prints f.keys or metatable methods

gmod-sentry

This library packages gmod-sentry with CREDENTIAL_STORE support for convenience.

To enable, simply add and customize the following to your GarrysMod/garrysmod/CREDENTIAL_STORE file

sentry
{
    dsn "https://[email protected]/dsn_example"
    // auto "false"

    // The following are all optional, this is passed directly to sentry.Setup
    options {
        server_name "Sandbox-1"
        environment "Production"
        // release ""

        // tags {
        //     foo "bar"
        // }

        // no_detour "hook.Call net.Receive"  // space-separated
    }
}

View gmod-sentry Setup Options for more documentation on the sentry options.

Alternatively, if you want to have dynamic control over the sentry.Setup call, set auto "false" and use the following code:

require "credentialstore"
require "xlib_sentry"

if CREDENTIALS.sentry then
    local options = CREDENTIALS.sentry.options or {}

    options.release = (GM or GAMEMODE).Version  -- Or anything else
    sentry.Setup(CREDENTIALS.sentry.dsn, options)
end

Contributing

If you would like to contribute, feel free to create a pull request. As this is a personal project, I can't guarantee that every request will be merged. However, if it benefits the project, I will gladly consider it.

I'm not enforcing any strict code guidelines as this project is already all over the place but please try to match the project's dominant style.

If you would like to enable autorefresh support while modifying XLib files themselves, create an empty file named XLIB_AUTOLOAD inside the garrysmod directory.