Compare commits
No commits in common. "506f1a44d1e63b7a6e2fff2a1de617d6614ea8cc" and "6835d6832c2869f365c2260776f7c966b889e556" have entirely different histories.
506f1a44d1
...
6835d6832c
|
@ -19,7 +19,7 @@ The Update has not been merged into `master` yet, but it will be. In the meantim
|
|||
- We're no longer using MySQL as the database! This was done for ease of install and operation; SQLite is plenty performant for Dragon Knight and makes it trivial to spin up new instances. The database is contained in `server/database/` as the file `dragon.db`. WAL mode is enabled, so you may see a couple extra files but this is expected.
|
||||
- `lib.php` renamed to `server/library.php`
|
||||
- The installer has been totally rewritten using the new database wrapper and a handful of new library functions.
|
||||
- Classes have been totally reworked! Prior, they were hard-coded into the game's overall settings. This made them highly inflexible and allowed only three classes which all needed their own level rows to define. This sucked! Now, classes are their own rows in the `classes` table, with starting stats and stat growth per-level. A class spells table now exists to allow admins to easily define what spells are awarded at what level per class.
|
||||
- Classes have been totally reworked! Prior, they were hard-coded into the game's overall settings. This made them highly inflexible and allowed only three classes which all needed their own level rows to define. This sucked! Now, classes are their own rows in the `classes` table, with starting stats and stat growth per-level. They also now have a special syntax in the `spells` field to detail at what level what spells the player gets.
|
||||
- The help pages have been moved to the new structure and have been renamed to "Guide".
|
||||
- Items now use a new special syntax similar to class spells to have multiple attributes; no more limits!
|
||||
|
||||
|
|
|
@ -1,80 +0,0 @@
|
|||
<?php
|
||||
|
||||
/*
|
||||
Security, and especially authentication, is not a simple matter.
|
||||
There's a lot to learn here.
|
||||
*/
|
||||
|
||||
class Auth
|
||||
{
|
||||
// name of the remember me cookie
|
||||
private const COOKIE_NAME = 'dragon-of-memory';
|
||||
|
||||
// id of the player
|
||||
public static int $id = 0;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->good();
|
||||
}
|
||||
|
||||
public function login(string $identifier, string $password, bool $remember = false): bool
|
||||
{
|
||||
// delete the old session
|
||||
if (isset($_SESSION['player_id'])) $this->logout();
|
||||
|
||||
// get the player by their username
|
||||
$id = Player::validateCredentials($identifier, $password);
|
||||
if ($id === false) return false;
|
||||
|
||||
// set the session
|
||||
$_SESSION['player_id'] = $id;
|
||||
self::$id = $id;
|
||||
|
||||
// set the remember me cookie
|
||||
if ($remember) $this->remember($id);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private function remember(int $id): array|false
|
||||
{
|
||||
$data = ['player_id' => $id, 'token' => token()];
|
||||
|
||||
Session::createOrUpdate($data);
|
||||
setcookie(self::COOKIE_NAME, implode('::', $data), strtotime('+30 days'), '/', '', true, true);
|
||||
|
||||
return $data;
|
||||
}
|
||||
|
||||
private function logout(): void
|
||||
{
|
||||
if (isset($_SESSION['player_id'])) unset($_SESSION['player_id']);
|
||||
if (isset($_COOKIE[self::COOKIE_NAME])) setcookie(self::COOKIE_NAME, '', time() - 86400, '/', '', true, true);
|
||||
}
|
||||
|
||||
public function good(): bool
|
||||
{
|
||||
// if our player_id session still exists, carry on
|
||||
if (isset($_SESSION['player_id'])) {
|
||||
self::$id = $_SESSION['player_id'];
|
||||
return true;
|
||||
}
|
||||
|
||||
// if a remember me cookie exists, try to validate it
|
||||
if (isset($_COOKIE[self::COOKIE_NAME])) {
|
||||
$cookie = explode('::', $_COOKIE[self::COOKIE_NAME]); // player_id::token
|
||||
|
||||
// try to validate the token
|
||||
if (!Session::validate($cookie[0], $cookie[1])) return false; // the token is invalid
|
||||
|
||||
// token is valid, refresh cookie and assign session
|
||||
$this->remember($cookie[0]);
|
||||
$_SESSION['player_id'] = $cookie[0];
|
||||
self::$id = $cookie[0];
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
|
@ -9,21 +9,12 @@ class App
|
|||
public static Database $db;
|
||||
private static string $dbPath;
|
||||
public static Request $req;
|
||||
public static Auth $auth;
|
||||
public static array $s = []; // game settings
|
||||
|
||||
public function __construct(string $dbPath)
|
||||
{
|
||||
self::$req = new Request(); // the current request
|
||||
self::$db = new Database($dbPath); // the database
|
||||
self::$dbPath = $dbPath; // the database path
|
||||
|
||||
// load game settings
|
||||
$s = self::$db->q('SELECT * FROM settings WHERE id = 1;');
|
||||
self::$s = $s ? $s->fetch() : [];
|
||||
|
||||
// init authentication
|
||||
self::$auth = new Auth();
|
||||
}
|
||||
|
||||
public static function performDatabaseReset(): void
|
||||
|
@ -33,9 +24,4 @@ class App
|
|||
self::$db = new Database(self::$dbPath);
|
||||
}
|
||||
}
|
||||
|
||||
public static function auth(): bool
|
||||
{
|
||||
return self::$auth->good();
|
||||
}
|
||||
}
|
|
@ -6,23 +6,7 @@ error_reporting(E_ALL | E_STRICT);
|
|||
|
||||
define('START', microtime(true)); // start the timer for this execution
|
||||
|
||||
// adjust session settings
|
||||
ini_set('session.gc_maxlifetime', 604800); // 1 week in seconds
|
||||
ini_set('session.cookie_lifetime', 604800); // 1 week in seconds
|
||||
|
||||
// ensure secure session handling
|
||||
ini_set('session.use_strict_mode', 1);
|
||||
ini_set('session.cookie_httponly', 1);
|
||||
ini_set('session.cookie_secure', 1); // only if using HTTPS
|
||||
|
||||
// start the session
|
||||
session_start();
|
||||
|
||||
// regenerate session ID to prevent session fixation
|
||||
if (!isset($_SESSION['initiated'])) {
|
||||
session_regenerate_id(true);
|
||||
$_SESSION['initiated'] = true;
|
||||
}
|
||||
session_start(); // initialize the session engine
|
||||
|
||||
// @todo move these to a settings config somewhere
|
||||
const VERSION = '1.1.11';
|
||||
|
@ -39,21 +23,17 @@ const MAP = [
|
|||
// 'Class' => 'path/to/class.php',
|
||||
|
||||
// server-level classes
|
||||
'App' => SERVER.'/app/App.php',
|
||||
'Database' => SERVER.'/app/Database.php',
|
||||
'Request' => SERVER.'/app/Request.php',
|
||||
'Auth' => SERVER.'/app/Auth.php',
|
||||
'App' => SERVER.'/app/app.php',
|
||||
'Database' => SERVER.'/app/database.php',
|
||||
'Request' => SERVER.'/app/request.php',
|
||||
|
||||
// modules
|
||||
'HomeModule' => SERVER.'/modules/HomeModule.php',
|
||||
'InstallModule' => SERVER.'/modules/InstallModule.php',
|
||||
'GateModule' => SERVER.'/modules/GateModule.php',
|
||||
|
||||
// models
|
||||
'Classes' => SERVER.'/models/Classes.php',
|
||||
'Player' => SERVER.'/models/Player.php',
|
||||
'Spell' => SERVER.'/models/Spell.php',
|
||||
'Session' => SERVER.'/models/Session.php',
|
||||
];
|
||||
|
||||
// autoloader
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
This folder serves as the home for the game's database; `dragon.db` by default. This is a WAL- and foreign key-enabled SQLite database wrapped in a very thin class based on the PDO wrapper in PHP. In production, the `dragon.db` file will be created if it doesn't exist, and the installer should be used to populate the database. This file does not exist in the repo.
|
||||
|
||||
### Packs
|
||||
New to Dragon Knight is the ability to upload "data packs" to the game! Using this feature, it is possible to upload `.zip` files that contain `.csv` files (spreadsheets) of data for the game. These spreadsheets must have a 1:1 structure to what's expected in Dragon Knight. This allows an admin to populate the game data quickly and easily with data they either make or get from someone else.
|
||||
New to Dragon Knight is the ability to upload "data packs" to the game! Using this feature, it is possible to upload and store `.zip` files that contain `.csv` files (spreadsheets) of data for the game. These spreadsheets must have a 1:1 structure to what's expected in Dragon Knight. This allows an admin to populate the game data quickly and easily with data they either make or get from someone else.
|
||||
|
||||
The `Default` data pack is the default data used when doing a **Complete** install of Dragon Knight. You can edit this before running the installer to change the default data. You can also use it as a template for your own data packs!
|
||||
|
||||
|
|
|
@ -1,6 +0,0 @@
|
|||
Level,"Class ID","Spell ID"
|
||||
1,1,6
|
||||
1,1,18
|
||||
1,3,1
|
||||
1,3,15
|
||||
1,3,18
|
|
|
@ -1,4 +1,4 @@
|
|||
Name,"Start HP","Start MP","Start STR","Start ATK","Start DEF","Start DEX","Growth HP","Growth MP","Growth STR","Growth ATK","Growth DEF","Growth DEX"
|
||||
Mage,10,10,5,5,5,5,3,5,1,3,1,3
|
||||
Warrior,20,0,10,5,10,5,6,2,3,1,3,1
|
||||
Paladin,15,5,5,5,10,10,4,4,2,2,2,2
|
||||
Name,"Start HP","Start MP","Start STR","Start ATK","Start DEF","Start DEX","Growth HP","Growth MP","Growth STR","Growth ATK","Growth DEF","Growth DEX",Spells
|
||||
Mage,10,10,5,5,5,5,3,5,1,3,1,3,"1:6,18"
|
||||
Warrior,20,0,10,5,10,5,6,2,3,1,3,1,
|
||||
Paladin,15,5,5,5,10,10,4,4,2,2,2,2,"1:1,15,18"
|
||||
|
|
|
|
@ -1,42 +1,32 @@
|
|||
<?php // library.php :: Common functions used throughout the program.
|
||||
|
||||
// Diff two times in seconds.
|
||||
function stopwatch(float $start, int $roundTo = 3): float
|
||||
{
|
||||
return round(microtime(true) - $start, $roundTo);
|
||||
}
|
||||
|
||||
// Redirect to another page.
|
||||
function redirect(string $url): void
|
||||
{
|
||||
header("Location: $url");
|
||||
exit;
|
||||
}
|
||||
|
||||
// Redirect to the install page if not installed.
|
||||
// Otherwise, redirect to the main page if requesting the install page.
|
||||
function installRedirect(string $route)
|
||||
{
|
||||
if (!INSTALLED && $route != 'install') redirect('/install');
|
||||
if (INSTALLED && $route == 'install') redirect('/');
|
||||
}
|
||||
|
||||
// Get the path to a template.
|
||||
function template(string $name): string
|
||||
{
|
||||
return SERVER."/templates/$name.php";
|
||||
}
|
||||
|
||||
// Checks if all required fields are set.
|
||||
function required(array $keys): bool
|
||||
{
|
||||
foreach ($keys as $key) if (!isset($_POST[$key]) || empty($_POST[$key])) return false;
|
||||
return true;
|
||||
}
|
||||
|
||||
// Renders a template. Pass data to it - uses an output buffer to have PHP process the template instead of using
|
||||
// a template engine. If you're including partials in the page, call `render('partial', $data)`, as $data will still
|
||||
// be available.
|
||||
/**
|
||||
* Renders a template. Pass data to it - uses an output buffer to have PHP process the template instead of using
|
||||
* a template engine. If you're including partials in the page, call `render('partial', $data)`, as $data will still
|
||||
* be available.
|
||||
*/
|
||||
function render(string $baseView, array $data = []): string
|
||||
{
|
||||
ob_start();
|
||||
|
@ -45,7 +35,20 @@ function render(string $baseView, array $data = []): string
|
|||
return ob_get_clean();
|
||||
}
|
||||
|
||||
// Dump and die.
|
||||
/**
|
||||
* Checks if all required fields are set.
|
||||
*/
|
||||
function required(array $keys): bool
|
||||
{
|
||||
foreach ($keys as $key) {
|
||||
if (!isset($_POST[$key]) || empty($_POST[$key])) return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Dump and die. Useful for debugging.
|
||||
*/
|
||||
function dd(mixed $var, bool $r = false)
|
||||
{
|
||||
echo '<pre>';
|
||||
|
@ -54,32 +57,11 @@ function dd(mixed $var, bool $r = false)
|
|||
exit;
|
||||
}
|
||||
|
||||
// Calculate the EXP to level up, given a level.
|
||||
function expToLevel(int $level): int
|
||||
{
|
||||
// constants for the quadratic formula
|
||||
$a = 5; // adjust this value to change the curve's steepness
|
||||
$b = 20; // adjust this value to change the initial XP increase
|
||||
$baseOffset = 0; // starting point, no initial offset
|
||||
function getmicrotime() { // Used for timing script operations.
|
||||
|
||||
if ($level == 1) return 0; // level 1 does not require any XP
|
||||
list($usec, $sec) = explode(" ",microtime());
|
||||
return ((float)$usec + (float)$sec);
|
||||
|
||||
// calculate the dynamic offset based on the level
|
||||
$additionalOffset = floor(($level - 1) / 20) * 100;
|
||||
|
||||
// total offset
|
||||
$c = $baseOffset + $additionalOffset;
|
||||
|
||||
// calculate the experience required for the given level
|
||||
$experience = $a * pow($level - 1, 2) + $b * ($level - 1) + $c;
|
||||
|
||||
return $experience;
|
||||
}
|
||||
|
||||
// Generate a 32 byte cryptographically secure random hex string.
|
||||
function token(int $length = 32): string
|
||||
{
|
||||
return bin2hex(random_bytes($length));
|
||||
}
|
||||
|
||||
function prettydate($uglydate) { // Change the MySQL date format (YYYY-MM-DD) into something friendlier.
|
||||
|
|
|
@ -7,23 +7,4 @@ class Classes
|
|||
$res = App::$db->q("SELECT * FROM classes");
|
||||
return $res->fetchAll() ?: false;
|
||||
}
|
||||
|
||||
public static function get(int $id): array|false
|
||||
{
|
||||
$res = App::$db->do("SELECT * FROM classes WHERE id = ?", [$id]);
|
||||
return $res->fetch() ?: false;
|
||||
}
|
||||
|
||||
public static function spells(int $id): array|false
|
||||
{
|
||||
$res = App::$db->do("SELECT level, spell_id FROM class_spells WHERE class_id = ?", [$id]);
|
||||
return $res->fetchAll() ?: false;
|
||||
}
|
||||
|
||||
public static function spellsAtLevel(int $id, int $level): array|false
|
||||
{
|
||||
// get all spells for the class under or equal to the level
|
||||
$ids = App::$db->do("SELECT spell_id FROM class_spells WHERE class_id = ? AND level <= ?", [$id, $level]);
|
||||
return $ids->fetchAll(PDO::FETCH_COLUMN) ?: false;
|
||||
}
|
||||
}
|
|
@ -4,52 +4,10 @@ class Player
|
|||
{
|
||||
public static function create(array $data): int
|
||||
{
|
||||
// get the player's class
|
||||
$class = Classes::get($data['class_id'] ?? 1);
|
||||
if ($class == false) die('Player::create: Invalid class selected. ' . print_r($data, true));
|
||||
|
||||
// get player level
|
||||
$l = $data['level'] ?? 1;
|
||||
|
||||
// calculate player stats
|
||||
$data['hp'] = $data['max_hp'] = $data['max_hp'] ?? ($class['start_hp'] + ($class['growth_hp'] * ($l - 1)));
|
||||
$data['mp'] = $data['max_mp'] = $data['max_mp'] ?? ($class['start_mp'] + ($class['growth_mp'] * ($l - 1)));
|
||||
$data['tp'] = $data['max_tp'] = $data['max_tp'] ?? (5 + (App::$s['tp_growth'] * ($l - 1)));
|
||||
$data['str'] = $data['str'] ?? ($class['start_str'] + ($class['growth_str'] * ($l - 1)));
|
||||
$data['atk'] = $data['atk'] ?? ($class['start_atk'] + ($class['growth_atk'] * ($l - 1)));
|
||||
$data['def'] = $data['def'] ?? ($class['start_def'] + ($class['growth_def'] * ($l - 1)));
|
||||
$data['dex'] = $data['dex'] ?? ($class['start_dex'] + ($class['growth_dex'] * ($l - 1)));
|
||||
$data['stat_points'] = $data['stat_points'] ?? (App::$s['stat_point_gain'] * ($l - 1));
|
||||
$data['exp2l'] = $data['exp2l'] ?? expToLevel($l + 1);
|
||||
|
||||
// award spells
|
||||
if (!isset($data['spells']) || empty($data['spells'])) {
|
||||
$spells = Classes::spellsAtLevel($class['id'], $l);
|
||||
$data['spells'] = implode(',', $spells);
|
||||
}
|
||||
|
||||
// compress data and generate placeholers
|
||||
$keys = implode(', ', array_keys($data));
|
||||
$placeholders = implode(', ', array_fill(0, count($data), '?'));
|
||||
|
||||
// insert into db
|
||||
App::$db->do("INSERT INTO 'players' ($keys) VALUES ($placeholders);", array_values($data));
|
||||
return App::$db->lastInsertID();
|
||||
}
|
||||
|
||||
public static function validateCredentials(string $identifier, string $password, bool $fetch = false): int|false
|
||||
{
|
||||
// get the player from their username or email
|
||||
$player = App::$db->do("SELECT " . ($fetch ? '*' : 'id, password') . " FROM players WHERE username = :i OR email = :i LIMIT 1;", ['i' => $identifier]);
|
||||
if ($player == false) return false;
|
||||
$player = $player->fetch();
|
||||
|
||||
// check password, return the player data if good
|
||||
if (password_verify($password, $player['password'])) {
|
||||
unset($player['password']);
|
||||
return $fetch ? $player : $player['id'];
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
}
|
|
@ -1,32 +0,0 @@
|
|||
<?php
|
||||
|
||||
class Session
|
||||
{
|
||||
public static function createOrUpdate(array $data): void
|
||||
{
|
||||
App::$db->do("INSERT OR REPLACE INTO sessions (player_id, token, expires) VALUES (?, ?, DATETIME(CURRENT_TIMESTAMP, '+30 days'));", $data);
|
||||
}
|
||||
|
||||
public static function get(int $id): array|false
|
||||
{
|
||||
$session = App::$db->do("SELECT * FROM sessions WHERE player_id = ? LIMIT 1;", [$id]);
|
||||
return $session->fetch() ?: false;
|
||||
}
|
||||
|
||||
public static function delete(int $id): void
|
||||
{
|
||||
App::$db->do("DELETE FROM sessions WHERE player_id = ?;", [$id]);
|
||||
}
|
||||
|
||||
public static function validate(int $id, string $token): bool
|
||||
{
|
||||
$session = App::$db->do("SELECT * FROM sessions WHERE player_id = ? AND token = ? LIMIT 1;", [$id, $token]);
|
||||
if ($session === false) return false;
|
||||
$session = $session->fetch();
|
||||
|
||||
// if the current time is after the expires column, the token is invalid
|
||||
if (strtotime($session['expires']) < time()) return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
|
@ -1,27 +0,0 @@
|
|||
<?php
|
||||
|
||||
class Spell
|
||||
{
|
||||
public static function all(): array|false
|
||||
{
|
||||
$spells = App::$db->do("SELECT * FROM spells");
|
||||
return $spells->fetchAll() ?: false;
|
||||
}
|
||||
|
||||
public static function get(int $id): array|false
|
||||
{
|
||||
$spell = App::$db->do("SELECT * FROM spells WHERE id = ?", [$id]);
|
||||
return $spell->fetch() ?: false;
|
||||
}
|
||||
|
||||
public static function getFromList(string|array $list): array
|
||||
{
|
||||
if (is_string($list)) $list = explode(',', $list);
|
||||
$spells = [];
|
||||
foreach ($list as $id) {
|
||||
$spell = self::get($id);
|
||||
if ($spell !== false) $spells[] = $spell;
|
||||
}
|
||||
return $spells;
|
||||
}
|
||||
}
|
|
@ -1,6 +0,0 @@
|
|||
<?php
|
||||
|
||||
class GateModule
|
||||
{
|
||||
|
||||
}
|
|
@ -4,12 +4,7 @@ class HomeModule
|
|||
{
|
||||
public static function home()
|
||||
{
|
||||
if (App::auth()) {
|
||||
echo 'You are already logged in!<br>';
|
||||
} else {
|
||||
echo 'You are not logged in!<br>';
|
||||
}
|
||||
|
||||
echo 'Welcome to the home module!';
|
||||
echo 'Your request is: ' . App::$req->uri(0);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -34,7 +34,7 @@ class InstallModule
|
|||
|
||||
// @Settings
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'settings' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'game_name' TEXT DEFAULT 'Dragon Knight',
|
||||
'game_version' TEXT DEFAULT '1.0',
|
||||
'game_dev' TEXT DEFAULT 'Sharkk',
|
||||
|
@ -47,9 +47,7 @@ class InstallModule
|
|||
'verify_email' INT DEFAULT 1,
|
||||
'show_news' INT DEFAULT 1,
|
||||
'show_online' INT DEFAULT 1,
|
||||
'show_babble' INT DEFAULT 1,
|
||||
'tp_growth' INT DEFAULT 1,
|
||||
'stat_point_gain' INT DEFAULT 5
|
||||
'show_babble' INT DEFAULT 1
|
||||
);");
|
||||
|
||||
// insert default settings
|
||||
|
@ -57,7 +55,7 @@ class InstallModule
|
|||
|
||||
// @Classes
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'classes' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'name' TEXT DEFAULT '',
|
||||
'start_hp' INT DEFAULT 0,
|
||||
'start_mp' INT DEFAULT 0,
|
||||
|
@ -70,7 +68,8 @@ class InstallModule
|
|||
'growth_str' INT DEFAULT 0,
|
||||
'growth_atk' INT DEFAULT 0,
|
||||
'growth_dex' INT DEFAULT 0,
|
||||
'growth_def' INT DEFAULT 0
|
||||
'growth_def' INT DEFAULT 0,
|
||||
'spells' TEXT DEFAULT ''
|
||||
);");
|
||||
|
||||
if ($complete) {
|
||||
|
@ -83,7 +82,7 @@ class InstallModule
|
|||
|
||||
// @Babble
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'babble' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'author' INTEGER NOT NULL,
|
||||
'babble' TEXT NOT NULL,
|
||||
'posted' DATETIME DEFAULT CURRENT_TIMESTAMP
|
||||
|
@ -91,7 +90,7 @@ class InstallModule
|
|||
|
||||
// @Drops
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'drops' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'name' TEXT NOT NULL,
|
||||
'level' INTEGER DEFAULT 1,
|
||||
'type' INTEGER DEFAULT 1,
|
||||
|
@ -103,7 +102,7 @@ class InstallModule
|
|||
|
||||
// @Forum
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'forum' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'posted' DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||
'new_post' DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||
'author' INTEGER NOT NULL,
|
||||
|
@ -119,7 +118,7 @@ class InstallModule
|
|||
|
||||
// @Items
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'items' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'type' INTEGER DEFAULT 1,
|
||||
'name' TEXT NOT NULL,
|
||||
'cost' INTEGER DEFAULT 0,
|
||||
|
@ -132,7 +131,7 @@ class InstallModule
|
|||
|
||||
// @Monsters
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'monsters' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'name' TEXT NOT NULL,
|
||||
'level' INTEGER DEFAULT 1,
|
||||
'hp' INTEGER DEFAULT 1,
|
||||
|
@ -149,7 +148,7 @@ class InstallModule
|
|||
|
||||
// @News
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'news' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'author' INTEGER DEFAULT 1,
|
||||
'title' TEXT DEFAULT '',
|
||||
'content' TEXT DEFAULT '',
|
||||
|
@ -158,7 +157,7 @@ class InstallModule
|
|||
|
||||
// @Spells
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'spells' (
|
||||
'id' INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
'name' TEXT NOT NULL,
|
||||
'type' INTEGER DEFAULT 1,
|
||||
'mp' INTEGER DEFAULT 0,
|
||||
|
@ -169,16 +168,6 @@ class InstallModule
|
|||
// add default spells if complete install
|
||||
if ($complete) App::$db->insertFromCSV('spells', "$defaults/spells.csv");
|
||||
|
||||
// @LearnedSpells
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'class_spells' (
|
||||
'level' INTEGER NOT NULL,
|
||||
'class_id' INTEGER NOT NULL,
|
||||
'spell_id' INTEGER NOT NULL
|
||||
);");
|
||||
|
||||
// add default class spells if complete install
|
||||
if ($complete) App::$db->insertFromCSV('class_spells', "$defaults/class_spells.csv");
|
||||
|
||||
// @Towns
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'towns' (
|
||||
'id' INTEGER PRIMARY KEY,
|
||||
|
@ -209,15 +198,14 @@ class InstallModule
|
|||
'class_id' INTEGER DEFAULT 1,
|
||||
'level' INTEGER DEFAULT 1,
|
||||
'exp' INTEGER DEFAULT 0,
|
||||
'exp2l' INTEGER DEFAULT 0,
|
||||
'gold' INTEGER DEFAULT 0,
|
||||
'stat_points' INTEGER DEFAULT 0,
|
||||
'hp' INTEGER DEFAULT 0,
|
||||
'max_hp' INTEGER DEFAULT 0,
|
||||
'mp' INTEGER DEFAULT 0,
|
||||
'max_mp' INTEGER DEFAULT 0,
|
||||
'tp' INTEGER DEFAULT 5,
|
||||
'max_tp' INTEGER DEFAULT 5,
|
||||
'tp' INTEGER DEFAULT 0,
|
||||
'max_tp' INTEGER DEFAULT 0,
|
||||
'str' INTEGER DEFAULT 0,
|
||||
'atk' INTEGER DEFAULT 0,
|
||||
'dex' INTEGER DEFAULT 0,
|
||||
|
@ -244,14 +232,7 @@ class InstallModule
|
|||
'p_maxmp' INTEGER DEFAULT 0,
|
||||
'm_hp' INTEGER DEFAULT 0,
|
||||
'm_maxhp' INTEGER DEFAULT 0,
|
||||
'effects' TEXT DEFAULT ''
|
||||
);");
|
||||
|
||||
// @Sessions
|
||||
App::$db->q("CREATE TABLE IF NOT EXISTS 'sessions' (
|
||||
'player_id' INTEGER NOT NULL UNIQUE,
|
||||
'token' TEXT NOT NULL,
|
||||
'expires' DATETIME NOT NULL
|
||||
'condi' TEXT DEFAULT ''
|
||||
);");
|
||||
|
||||
echo render('install/layout', ['title' => 'Database Setup', 'step' => 'second', 'complete' => $complete, 'start' => $istart]);
|
||||
|
@ -274,8 +255,8 @@ class InstallModule
|
|||
}
|
||||
}
|
||||
|
||||
// Make sure the class selection is present
|
||||
$class = $_POST['class'] ?? 1;
|
||||
// Make sure the class selection is valid
|
||||
$class = isset($_POST['class']) && in_array($_POST['class'], [1, 2, 3]) ? $_POST['class'] : 1;
|
||||
|
||||
// If we have any errors, bail to the form and let the user know
|
||||
if (!empty($errors)) {
|
||||
|
@ -283,6 +264,9 @@ class InstallModule
|
|||
exit;
|
||||
}
|
||||
|
||||
// Create the .installed file in the server folder
|
||||
file_put_contents(SERVER.'/.installed', 'Installed on '.date('Y-m-d H:i:s'));
|
||||
|
||||
// Create the admin account
|
||||
Player::create([
|
||||
'username' => trim($_POST['username']),
|
||||
|
@ -290,19 +274,11 @@ class InstallModule
|
|||
'email' => trim($_POST['email']),
|
||||
'class_id' => $class,
|
||||
'verified' => 1,
|
||||
'role' => 5,
|
||||
'level' => $_POST['level'] ?? 1
|
||||
'role' => 5
|
||||
]);
|
||||
|
||||
// Create the .installed file in the server folder
|
||||
file_put_contents(SERVER.'/.installed', 'Installed on '.date('Y-m-d H:i:s'));
|
||||
|
||||
// login the admin
|
||||
App::$auth->login($_POST['username'], $_POST['password']);
|
||||
|
||||
// Render the finished page!
|
||||
echo render('install/layout', ['title' => 'Finished!', 'step' => 'done', 'name' => $_POST['username'], 'complete' => $_POST['complete'] ?? false]);
|
||||
|
||||
}
|
||||
|
||||
private static function fourOhFour()
|
||||
|
|
|
@ -1,12 +1,12 @@
|
|||
<p class="mb-1">
|
||||
Congratulations, <?= $name ?>! Your installation is complete. Dragon Knight is ready to go.
|
||||
All that's left is to start playing. <?php if (!$complete): ?>Once you've logged in,
|
||||
All that's left is to log in and start playing. <?php if (!$complete): ?>Once you've logged in,
|
||||
you can create some classes and assign your character one. By default you are a useless Adventurer.
|
||||
😜<?php endif; ?>
|
||||
</p>
|
||||
|
||||
<p class="mb-1">
|
||||
<a href="/gate/login">Click here to begin your adventure.</a>
|
||||
<a href="/gate/login">Click here to log in.</a>
|
||||
</p>
|
||||
|
||||
<p>
|
||||
|
|
|
@ -27,10 +27,5 @@
|
|||
</div>
|
||||
<?php endif; ?>
|
||||
|
||||
<div class="form-group">
|
||||
<label for="level">Level</label>
|
||||
<input type="number" name="level" id="level" min="1" placeholder="1" value="1" required>
|
||||
</div>
|
||||
|
||||
<button type="submit" name="submit">Submit</button>
|
||||
</form>
|
||||
|
|
Loading…
Reference in New Issue
Block a user