mirror of
				https://github.com/slawkens/myaac.git
				synced 2025-10-31 07:56:23 +01:00 
			
		
		
		
	Compare commits
	
		
			81 Commits
		
	
	
		
			feature/2f
			...
			develop
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 1379c93439 | ||
|   | 19b1cfdd34 | ||
|   | 26c5aa2e51 | ||
|   | bc4107bd16 | ||
|   | d24bde2c1d | ||
|   | a2f8759a52 | ||
|   | 4b8c3ffae2 | ||
|   | 97321c9e80 | ||
|   | 2580edadf8 | ||
|   | 8f47b36dc8 | ||
|   | 6cd38ee1ec | ||
|   | 9d92a11fb7 | ||
|   | 44110a9496 | ||
|   | 727f68a575 | ||
|   | 07fd034fe4 | ||
|   | 13ea68cc0c | ||
|   | 598cec2fe4 | ||
|   | 89fae38caa | ||
|   | 16849e7578 | ||
|   | 470555f268 | ||
|   | 7f60b3d31d | ||
|   | 12e40b2592 | ||
|   | 38902c30d1 | ||
|   | 3e61692780 | ||
|   | c88b08eb1e | ||
|   | 82d417b590 | ||
|   | b797908e49 | ||
|   | 90c8463797 | ||
|   | c91bb5d409 | ||
|   | fe821c5808 | ||
|   | 9acad15451 | ||
|   | 8c3cb0e06f | ||
|   | 2eae44e075 | ||
|   | 8272f1373c | ||
|   | 901df48d13 | ||
|   | 2da0024c68 | ||
|   | 0d8f68a48e | ||
|   | 0cb9d3a208 | ||
|   | d8b73f55a3 | ||
|   | 3bb272ebbb | ||
|   | 64acf70d38 | ||
|   | 97f9d3d6f6 | ||
|   | f54b1bdd2a | ||
|   | c898fe25ef | ||
|   | 73c07d470d | ||
|   | 56bd7ec5ed | ||
|   | 4c6277c124 | ||
|   | 228780f0ad | ||
|   | 4e9999cc0d | ||
|   | 8bc328d6fb | ||
|   | ac41b82579 | ||
|   | df7b6e29fb | ||
|   | e0cc19ad86 | ||
|   | 85e7005fd3 | ||
|   | 3c0cb53e17 | ||
|   | d0112d1a67 | ||
|   | ed9beaf2b6 | ||
|   | 5aa9bbf1c8 | ||
|   | a6032093b2 | ||
|   | e719725841 | ||
|   | bb3e90110d | ||
|   | 2f0758e351 | ||
|   | 6667c8c364 | ||
|   | c13a540878 | ||
|   | 869ec035d9 | ||
|   | 9d696d31d8 | ||
|   | 8cc4caf587 | ||
|   | e1d1c7d5db | ||
|   | 320733c2c1 | ||
|   | c1809a98d1 | ||
|   | 46ed541015 | ||
|   | 29207361b7 | ||
|   | 25013ae91b | ||
|   | 5d630ba9dd | ||
|   | feadf1314d | ||
|   | 08b8a716d4 | ||
|   | cc26b5c744 | ||
|   | cb6e9a6a88 | ||
|   | 4adb0758c5 | ||
|   | 7312383f73 | ||
|   | 3c1210fefa | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -4,7 +4,7 @@ Thumbs.db | ||||
|  | ||||
| # | ||||
| /.htaccess | ||||
| lua | ||||
| /lua | ||||
|  | ||||
| # composer | ||||
| composer.phar | ||||
|   | ||||
| @@ -1,5 +1,56 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## [1.8.4 - 27.10.2025] | ||||
|  | ||||
| ### Changed | ||||
| * Reimport myaac_ tables on every install, this fixes errors when one table is missing or is duplicated (https://github.com/slawkens/myaac/commit/2580edadf84779f09fd395c21f92019b2c762f83) | ||||
| * Use custom env init on migrate, migrate:run and migrate:to (https://github.com/slawkens/myaac/commit/13ea68cc0c9349380c8e4051d702a6c2c8256f44, https://github.com/slawkens/myaac/commit/07fd034fe4cb0ffdb88667b1e400f414d0c6d06f) | ||||
|  | ||||
| ### Fixed | ||||
| * Show if there is mysql error on import schema (https://github.com/slawkens/myaac/commit/44110a9496b4385e42c31b75de301037e711b6c3) | ||||
| * Fix the premium checks, introduced in v1.8.3 (https://github.com/slawkens/myaac/commit/9d92a11fb7cb6d7a1619d79c12faaa0b1c01f980) | ||||
|  | ||||
| ## [1.8.3 - 21.10.2025] | ||||
|  | ||||
| ### Added | ||||
| * Feature: resend email verify (https://github.com/slawkens/myaac/commit/fe821c58085483e70491dcf76376ad5b96de3fdd) | ||||
| * New config: hooks_debug (To view where hooks are located in .twig files) (https://github.com/slawkens/myaac/commit/8c3cb0e06f9709c1de3398b48221241e7cbdd310) | ||||
| * Functions: Add db->getColumnInfo(table, column) (https://github.com/slawkens/myaac/commit/c898fe25efff6793a01d11c26fc153cb23fcb858) | ||||
| * Plugins: Add option to use ?subtopic=x for plugins pages (https://github.com/slawkens/myaac/commit/97f9d3d6f6c28aef6d824973058d7133f56e09c4) | ||||
| * getTopPlayers() Function - Add lookmount & promotion (https://github.com/slawkens/myaac/commit/2da0024c68f1cedc38a16ebbc6f52ffa55e65f7a, https://github.com/slawkens/myaac/commit/901df48d134079d648a18f9d82b60182e818ac02) | ||||
| * New hooks for account/change-password (https://github.com/slawkens/myaac/commit/470555f2687809a0c12491bbb27597e64b8929c1) | ||||
|  | ||||
| ### Changed | ||||
| * Feature: show vip days in account management (https://github.com/slawkens/myaac/commit/c88b08eb1ec1f560cbfdaaa16b24e3a0f26da7b3, by @andreoam) | ||||
| * Allow links in error_box.html.twig (https://github.com/slawkens/myaac/commit/9acad15451071639acf7a7d4e81619b0a9742b12) | ||||
| * Canary - Comment code to update lastday in login.php (https://github.com/slawkens/myaac/commit/38902c30d114fdbce259467f5820f97037b393e9) | ||||
| * Cache::remember $ttl = -1 = infinite (https://github.com/slawkens/myaac/commit/64acf70d3854182d88aaf0b67f77cea2a254f179) | ||||
|  | ||||
| ### Fixed | ||||
| * Online - Allow for html code (example - img) in online_datacenter (https://github.com/slawkens/myaac/commit/3bb272ebbbd2eb7769d174b7082061d14a17bd44) | ||||
| * Guilds - Fix guild create with freePremium enabled (https://github.com/slawkens/myaac/commit/c91bb5d4097647dca2196d3dea87bc90c89181d2) | ||||
| * Canary - Fix premDays count (https://github.com/slawkens/myaac/commit/3e61692780d4add93b7b0e9f12f7a283bd8f4b7a) | ||||
| * Template Change: Ignore set last visit for AJAX pages - Fixes template change redirect (https://github.com/slawkens/myaac/commit/89fae38caa7e4f645957fcf1a9330a36358ac04f) | ||||
| * Admin Panel - Accounts: Fix lastip v6 (TFS master) (https://github.com/slawkens/myaac/commit/f54b1bdd2af4c16c64ddff0e87a6c96bc4cf9eeb) | ||||
| * Functions - Prevent injection in $db->hasColumn (https://github.com/slawkens/myaac/commit/56bd7ec5ed904666074492f2e4f13e4fce226bee) | ||||
| * Compat Config: Add missing config: email_lai_sec_interval (https://github.com/slawkens/myaac/commit/2eae44e0755e624a91be68b4d1ec26d01eb4d9a1) | ||||
|  | ||||
| ## [1.8.2 - 26.09.2025] | ||||
|  | ||||
| ### Added | ||||
| * Routes: Possibility to override routes with plugins pages, like characters.php - No need to define routes in plugin.json anymore (https://github.com/slawkens/myaac/commit/3f24f961b1cdeff5c60387e837ae454448bc5e1b) | ||||
|  | ||||
| ### Changed | ||||
| * Style: Better look for myaac-table (https://github.com/slawkens/myaac/commit/a6032093b21e5bb3f0e75d2704da87d6dea6469d, https://github.com/slawkens/myaac/commit/5aa9bbf1c8e580d973ec82ac012489f8e7bc437e) | ||||
|  | ||||
| ### Fixed | ||||
| * Install: Fix when config.local.php cannot be saved (https://github.com/slawkens/myaac/commit/4eab805d26d8c5562b29ed699769919d77dabced) | ||||
| * Create Account: Fix an exception when email cannot be sent (https://github.com/slawkens/myaac/commit/d0112d1a67e8b854b65ad131f0375b79305df8d3) | ||||
| * Login Page: Add missing csrf() - fix create account button (https://github.com/slawkens/myaac/commit/3c0cb53e17dd0b85394cfa0fdc9cf9ad8d4551df) | ||||
| * tibiacom template: Fix account lost menu (https://github.com/slawkens/myaac/commit/ed9beaf2b6ca069e304e569c52e5b9188b58f05c) | ||||
| * tibiacom template: Fix Menu div wrong tag/closing (#329) (https://github.com/slawkens/myaac/commit/85e7005fd3f0be51466151a3c122b96085fdfe68) | ||||
| * tibiacom template: Replace firstChild with firstElementChild (Thanks to @un000000) (https://github.com/slawkens/myaac/commit/df7b6e29fb8875da97f431468c81ee99116271d9) | ||||
|  | ||||
| ## [1.8.1 - 05.09.2025] | ||||
|  | ||||
| ### Added | ||||
|   | ||||
							
								
								
									
										4
									
								
								aac
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								aac
									
									
									
									
									
								
							| @@ -25,7 +25,9 @@ foreach ($commandsGlob as $item) { | ||||
| 	} | ||||
|  | ||||
| 	$commandPre = '\\MyAAC\Commands\\'; | ||||
| 	$application->add(new ($commandPre . $name)); | ||||
| 	if (!trait_exists($class = $commandPre . $name)) { | ||||
| 		$application->add(new $class); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| $pluginCommands = Plugins::getCommands(); | ||||
|   | ||||
| @@ -9,6 +9,7 @@ | ||||
|  */ | ||||
|  | ||||
| use MyAAC\Models\Account as AccountModel; | ||||
| use MyAAC\Models\AccountAction; | ||||
| use MyAAC\Models\Player; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
| @@ -481,9 +482,8 @@ else if (isset($_REQUEST['search'])) { | ||||
| 									</thead> | ||||
| 									<tbody> | ||||
| 										<?php | ||||
| 											$accountActions = \MyAAC\Models\AccountAction::where('account_id', $account->getId())->orderByDesc('date')->get(); | ||||
| 											$accountActions = AccountAction::where('account_id', $account->getId())->orderByDesc('date')->get(); | ||||
| 											foreach ($accountActions as $i => $log): | ||||
| 												$log->ip = ($log->ip != 0 ? long2ip($log->ip) : inet_ntop($log->ipv6)); | ||||
| 												?> | ||||
| 											<tr> | ||||
| 												<td><?php echo $i + 1; ?></td> | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
|  * @package   MyAAC | ||||
|  * @author    Slawkens <slawkens@gmail.com> | ||||
|  * @author    Lee | ||||
|  * @author    gpedro | ||||
|  * @copyright 2020 MyAAC | ||||
|  * @link      https://my-aac.org | ||||
|  */ | ||||
| @@ -19,9 +20,9 @@ $title = 'Mass Account Actions'; | ||||
| csrfProtect(); | ||||
|  | ||||
| $hasPointsColumn = $db->hasColumn('accounts', 'premium_points'); | ||||
| $freePremium = $config['lua']['freePremium']; | ||||
| $freePremium = getBoolean(configLua('freePremium')); | ||||
|  | ||||
| function admin_give_points($points) | ||||
| function admin_give_points($points): void | ||||
| { | ||||
| 	global $hasPointsColumn; | ||||
|  | ||||
| @@ -37,7 +38,7 @@ function admin_give_points($points) | ||||
| 	displayMessage($points . ' points added to all accounts.', true); | ||||
| } | ||||
|  | ||||
| function admin_give_coins($coins) | ||||
| function admin_give_coins($coins): void | ||||
| { | ||||
| 	if (!HAS_ACCOUNT_COINS) { | ||||
| 		displayMessage('Coins not supported.'); | ||||
| @@ -52,7 +53,7 @@ function admin_give_coins($coins) | ||||
| 	displayMessage($coins . ' coins added to all accounts.', true); | ||||
| } | ||||
|  | ||||
| function admin_give_premdays($days) | ||||
| function admin_give_premdays($days): void | ||||
| { | ||||
| 	global $db, $freePremium; | ||||
|  | ||||
| @@ -63,6 +64,7 @@ function admin_give_premdays($days) | ||||
|  | ||||
| 	$value = $days * 86400; | ||||
| 	$now = time(); | ||||
|  | ||||
| 	// othire | ||||
| 	if ($db->hasColumn('accounts', 'premend')) { | ||||
| 		// append premend | ||||
| @@ -70,14 +72,11 @@ function admin_give_premdays($days) | ||||
| 			// set premend | ||||
| 			if (Account::where('premend', '<=', $now)->update(['premend' => $now + $value])) { | ||||
| 				displayMessage($days . ' premium days added to all accounts.', true); | ||||
| 				return; | ||||
| 			} else { | ||||
| 				displayMessage('Failed to execute set query.'); | ||||
| 				return; | ||||
| 			} | ||||
| 		} else { | ||||
| 			displayMessage('Failed to execute append query.'); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		return; | ||||
| @@ -92,20 +91,14 @@ function admin_give_premdays($days) | ||||
| 				// set lastday | ||||
| 				if (Account::where('lastday', '<=', $now)->update(['lastday' => $now + $value])) { | ||||
| 					displayMessage($days . ' premium days added to all accounts.', true); | ||||
| 					return; | ||||
| 				} else { | ||||
| 					displayMessage('Failed to execute set query.'); | ||||
| 					return; | ||||
| 				} | ||||
|  | ||||
| 				return; | ||||
| 			} else { | ||||
| 				displayMessage('Failed to execute append query.'); | ||||
| 				return; | ||||
| 			} | ||||
| 		} else { | ||||
| 			displayMessage('Failed to execute set days query.'); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		return; | ||||
| @@ -118,14 +111,11 @@ function admin_give_premdays($days) | ||||
| 			// set premium_ends_at | ||||
| 			if (Account::where('premium_ends_at', '<=', $now)->update(['premium_ends_at' => $now + $value])) { | ||||
| 				displayMessage($days . ' premium days added to all accounts.', true); | ||||
| 				return; | ||||
| 			} else { | ||||
| 				displayMessage('Failed to execute set query.'); | ||||
| 				return; | ||||
| 			} | ||||
| 		} else { | ||||
| 			displayMessage('Failed to execute append query.'); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		return; | ||||
| @@ -170,7 +160,8 @@ else { | ||||
| 	)); | ||||
| } | ||||
|  | ||||
| function displayMessage($message, $success = false) { | ||||
| function displayMessage($message, $success = false): void | ||||
| { | ||||
| 	global $twig, $hasPointsColumn, $freePremium; | ||||
|  | ||||
| 	$success ? success($message): error($message); | ||||
|   | ||||
| @@ -669,12 +669,18 @@ else if (isset($_REQUEST['search'])) { | ||||
| 									<div class="col-12 col-sm-12 col-lg-6"> | ||||
| 										<label for="lastip" class="control-label">Last IP:</label> | ||||
| 										<input type="text" class="form-control" id="lastip" name="lastip" autocomplete="off" maxlength="10" value="<?php | ||||
| 										if (strlen($player->getLastIP()) > 11) { | ||||
| 										$lastIPColumnInfo = $db->getColumnInfo('players', 'lastip'); | ||||
| 										if ($lastIPColumnInfo && is_array($lastIPColumnInfo)) { | ||||
| 											if (str_contains($lastIPColumnInfo['type'], 'varbinary')) { | ||||
| 												echo inet_ntop($player->getLastIP()); | ||||
| 											} | ||||
| 											else { | ||||
| 												echo longToIp($player->getLastIP()); | ||||
| 											} | ||||
| 										} | ||||
| 										else { | ||||
| 											echo 'Error'; | ||||
| 										} | ||||
| 										?>" readonly/> | ||||
| 									</div> | ||||
| 								</div> | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| <?php | ||||
| define('MYAAC_ADMIN', true); | ||||
| const MYAAC_ADMIN = true; | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| require '../../common.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
| @@ -26,6 +26,7 @@ | ||||
| use MyAAC\DataLoader; | ||||
|  | ||||
| const MYAAC_ADMIN = true; | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| require '../../common.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
| @@ -3,6 +3,7 @@ | ||||
| use MyAAC\Settings; | ||||
|  | ||||
| const MYAAC_ADMIN = true; | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| require '../../common.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| <?php | ||||
| define('MYAAC_ADMIN', true); | ||||
| const MYAAC_ADMIN = true; | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| require '../../common.php'; | ||||
| require SYSTEM . 'init.php'; | ||||
|   | ||||
| @@ -1,5 +1,6 @@ | ||||
| <?php | ||||
| define('MYAAC_ADMIN', true); | ||||
| const MYAAC_ADMIN = true; | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| require '../../common.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
| @@ -26,8 +26,8 @@ | ||||
| if (version_compare(phpversion(), '8.1', '<')) die('PHP version 8.1 or higher is required.'); | ||||
|  | ||||
| const MYAAC = true; | ||||
| const MYAAC_VERSION = '1.8.2-dev'; | ||||
| const DATABASE_VERSION = 46; | ||||
| const MYAAC_VERSION = '2.0-dev'; | ||||
| const DATABASE_VERSION = 47; | ||||
| const TABLE_PREFIX = 'myaac_'; | ||||
| define('START_TIME', microtime(true)); | ||||
| define('MYAAC_OS', stripos(PHP_OS, 'WIN') === 0 ? 'WINDOWS' : (strtoupper(PHP_OS) === 'DARWIN' ? 'MAC' : 'LINUX')); | ||||
|   | ||||
| @@ -19,8 +19,7 @@ | ||||
|         "symfony/var-dumper": "^6.4", | ||||
|         "filp/whoops": "^2.15", | ||||
|         "maximebf/debugbar": "1.*", | ||||
|         "guzzlehttp/guzzle": "7.9.3", | ||||
|         "spomky-labs/otphp": "^11.3" | ||||
|         "guzzlehttp/guzzle": "7.9.3" | ||||
|     }, | ||||
|     "require-dev": { | ||||
|         "phpstan/phpstan": "^1.10" | ||||
|   | ||||
							
								
								
									
										151
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										151
									
								
								composer.lock
									
									
									
										generated
									
									
									
								
							| @@ -4,7 +4,7 @@ | ||||
|         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", | ||||
|         "This file is @generated automatically" | ||||
|     ], | ||||
|     "content-hash": "07419f6fe133f9bebc99557f3df843c8", | ||||
|     "content-hash": "5317e97a5025ebc2a977214bd3fa964c", | ||||
|     "packages": [ | ||||
|         { | ||||
|             "name": "brick/math", | ||||
| @@ -1556,73 +1556,6 @@ | ||||
|             }, | ||||
|             "time": "2018-02-13T20:26:39+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "paragonie/constant_time_encoding", | ||||
|             "version": "v3.0.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/paragonie/constant_time_encoding.git", | ||||
|                 "reference": "df1e7fde177501eee2037dd159cf04f5f301a512" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/paragonie/constant_time_encoding/zipball/df1e7fde177501eee2037dd159cf04f5f301a512", | ||||
|                 "reference": "df1e7fde177501eee2037dd159cf04f5f301a512", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "php": "^8" | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "phpunit/phpunit": "^9", | ||||
|                 "vimeo/psalm": "^4|^5" | ||||
|             }, | ||||
|             "type": "library", | ||||
|             "autoload": { | ||||
|                 "psr-4": { | ||||
|                     "ParagonIE\\ConstantTime\\": "src/" | ||||
|                 } | ||||
|             }, | ||||
|             "notification-url": "https://packagist.org/downloads/", | ||||
|             "license": [ | ||||
|                 "MIT" | ||||
|             ], | ||||
|             "authors": [ | ||||
|                 { | ||||
|                     "name": "Paragon Initiative Enterprises", | ||||
|                     "email": "security@paragonie.com", | ||||
|                     "homepage": "https://paragonie.com", | ||||
|                     "role": "Maintainer" | ||||
|                 }, | ||||
|                 { | ||||
|                     "name": "Steve 'Sc00bz' Thomas", | ||||
|                     "email": "steve@tobtu.com", | ||||
|                     "homepage": "https://www.tobtu.com", | ||||
|                     "role": "Original Developer" | ||||
|                 } | ||||
|             ], | ||||
|             "description": "Constant-time Implementations of RFC 4648 Encoding (Base-64, Base-32, Base-16)", | ||||
|             "keywords": [ | ||||
|                 "base16", | ||||
|                 "base32", | ||||
|                 "base32_decode", | ||||
|                 "base32_encode", | ||||
|                 "base64", | ||||
|                 "base64_decode", | ||||
|                 "base64_encode", | ||||
|                 "bin2hex", | ||||
|                 "encoding", | ||||
|                 "hex", | ||||
|                 "hex2bin", | ||||
|                 "rfc4648" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "email": "info@paragonie.com", | ||||
|                 "issues": "https://github.com/paragonie/constant_time_encoding/issues", | ||||
|                 "source": "https://github.com/paragonie/constant_time_encoding" | ||||
|             }, | ||||
|             "time": "2024-05-08T12:36:18+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "peppeocchi/php-cron-scheduler", | ||||
|             "version": "v4.0", | ||||
| @@ -2169,88 +2102,6 @@ | ||||
|             }, | ||||
|             "time": "2019-03-08T08:55:37+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "spomky-labs/otphp", | ||||
|             "version": "11.3.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/Spomky-Labs/otphp.git", | ||||
|                 "reference": "2d8ccb5fc992b9cc65ef321fa4f00fefdb3f4b33" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/Spomky-Labs/otphp/zipball/2d8ccb5fc992b9cc65ef321fa4f00fefdb3f4b33", | ||||
|                 "reference": "2d8ccb5fc992b9cc65ef321fa4f00fefdb3f4b33", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "ext-mbstring": "*", | ||||
|                 "paragonie/constant_time_encoding": "^2.0 || ^3.0", | ||||
|                 "php": ">=8.1", | ||||
|                 "psr/clock": "^1.0", | ||||
|                 "symfony/deprecation-contracts": "^3.2" | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "ekino/phpstan-banned-code": "^1.0", | ||||
|                 "infection/infection": "^0.26|^0.27|^0.28|^0.29", | ||||
|                 "php-parallel-lint/php-parallel-lint": "^1.3", | ||||
|                 "phpstan/phpstan": "^1.0", | ||||
|                 "phpstan/phpstan-deprecation-rules": "^1.0", | ||||
|                 "phpstan/phpstan-phpunit": "^1.0", | ||||
|                 "phpstan/phpstan-strict-rules": "^1.0", | ||||
|                 "phpunit/phpunit": "^9.5.26|^10.0|^11.0", | ||||
|                 "qossmic/deptrac-shim": "^1.0", | ||||
|                 "rector/rector": "^1.0", | ||||
|                 "symfony/phpunit-bridge": "^6.1|^7.0", | ||||
|                 "symplify/easy-coding-standard": "^12.0" | ||||
|             }, | ||||
|             "type": "library", | ||||
|             "autoload": { | ||||
|                 "psr-4": { | ||||
|                     "OTPHP\\": "src/" | ||||
|                 } | ||||
|             }, | ||||
|             "notification-url": "https://packagist.org/downloads/", | ||||
|             "license": [ | ||||
|                 "MIT" | ||||
|             ], | ||||
|             "authors": [ | ||||
|                 { | ||||
|                     "name": "Florent Morselli", | ||||
|                     "homepage": "https://github.com/Spomky" | ||||
|                 }, | ||||
|                 { | ||||
|                     "name": "All contributors", | ||||
|                     "homepage": "https://github.com/Spomky-Labs/otphp/contributors" | ||||
|                 } | ||||
|             ], | ||||
|             "description": "A PHP library for generating one time passwords according to RFC 4226 (HOTP Algorithm) and the RFC 6238 (TOTP Algorithm) and compatible with Google Authenticator", | ||||
|             "homepage": "https://github.com/Spomky-Labs/otphp", | ||||
|             "keywords": [ | ||||
|                 "FreeOTP", | ||||
|                 "RFC 4226", | ||||
|                 "RFC 6238", | ||||
|                 "google authenticator", | ||||
|                 "hotp", | ||||
|                 "otp", | ||||
|                 "totp" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "issues": "https://github.com/Spomky-Labs/otphp/issues", | ||||
|                 "source": "https://github.com/Spomky-Labs/otphp/tree/11.3.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
|                     "url": "https://github.com/Spomky", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://www.patreon.com/FlorentMorselli", | ||||
|                     "type": "patreon" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-06-12T11:22:32+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/console", | ||||
|             "version": "v6.4.17", | ||||
|   | ||||
							
								
								
									
										69
									
								
								install/includes/import_base_data.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								install/includes/import_base_data.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| <?php | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| use MyAAC\Models\Changelog; | ||||
| use MyAAC\Models\Config; | ||||
| use MyAAC\Models\ForumBoard; | ||||
| use MyAAC\Models\Gallery; | ||||
| use MyAAC\Models\NewsCategory; | ||||
|  | ||||
| if (Changelog::count() === 0) { | ||||
| 	Changelog::create([ | ||||
| 		'type' => 3, | ||||
| 		'where' => 2, | ||||
| 		'date' => time(), | ||||
| 		'body' => 'MyAAC installed. (:', | ||||
| 		'hide' => 0, | ||||
| 	]); | ||||
| } | ||||
|  | ||||
| if (Config::where('name', 'database_version')->count() === 0) { | ||||
| 	Config::create([ | ||||
| 		'name' => 'database_version', | ||||
| 		'value' => DATABASE_VERSION, | ||||
| 	]); | ||||
| } | ||||
|  | ||||
| if (ForumBoard::count() === 0) { | ||||
| 	$forumBoards = [ | ||||
| 		['name' => 'News', 'description' => 'News commenting', 'closed' => 1], | ||||
| 		['name' => 'Trade', 'description' => 'Trade offers.', 'closed' => 0], | ||||
| 		['name' => 'Quests', 'description' => 'Quest making.', 'closed' => 0], | ||||
| 		['name' => 'Pictures', 'description' => 'Your pictures.', 'closed' => 0], | ||||
| 		['name' => 'Bug Report', 'description' => 'Report bugs there.', 'closed' => 0], | ||||
| 	]; | ||||
|  | ||||
| 	$i = 0; | ||||
| 	foreach ($forumBoards as $forumBoard) { | ||||
| 		ForumBoard::create([ | ||||
| 			'name' => $forumBoard['name'], | ||||
| 			'description' => $forumBoard['description'], | ||||
| 			'ordering' => $i++, | ||||
| 			'closed' => $forumBoard['closed'], | ||||
| 		]); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| if (NewsCategory::count() === 0) { | ||||
| 	$newsCategoriesIcons = [ | ||||
| 		0, 1, 2, 3, 4 | ||||
| 	]; | ||||
|  | ||||
| 	foreach ($newsCategoriesIcons as $iconId) { | ||||
| 		NewsCategory::create([ | ||||
| 			'icon_id' => $iconId, | ||||
| 		]); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| if (Gallery::count() === 0) { | ||||
| 	Gallery::create([ | ||||
| 		'comment' => 'Demon', | ||||
| 		'image' => 'images/gallery/demon.jpg', | ||||
| 		'thumb' => 'images/gallery/demon_thumb.gif', | ||||
| 		'author' => 'MyAAC', | ||||
| 		'ordering' => 0, | ||||
| 	]); | ||||
| } | ||||
|  | ||||
| success($locale['step_database_success_import_data']); | ||||
| @@ -1,25 +1,23 @@ | ||||
| SET @myaac_database_version = 45; | ||||
|  | ||||
| CREATE TABLE `myaac_account_actions` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_account_actions` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`account_id` int NOT NULL, | ||||
| 	`ip` int unsigned NOT NULL DEFAULT 0, | ||||
| 	`ipv6` binary(16) NOT NULL DEFAULT 0, | ||||
| 	`ip` varchar(45) NOT NULL DEFAULT '', | ||||
| 	`date` int NOT NULL DEFAULT 0, | ||||
| 	`action` varchar(255) NOT NULL DEFAULT '', | ||||
| 	KEY (`account_id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_account_email_codes` | ||||
| ( | ||||
| 	`id` int(11) NOT NULL AUTO_INCREMENT, | ||||
| 	`account_id` int NOT NULL, | ||||
| 	`code` varchar(6) NOT NULL, | ||||
| 	`created_at` int NOT NULL, | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_admin_menu` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_account_emails_verify` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`account_id` int NOT NULL, | ||||
| 	`hash` varchar(32) NOT NULL, | ||||
| 	`sent_at` int NOT NULL DEFAULT 0, | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE IF NOT EXISTS `myaac_admin_menu` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`name` varchar(255) NOT NULL DEFAULT '', | ||||
| @@ -30,7 +28,7 @@ CREATE TABLE `myaac_admin_menu` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_changelog` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_changelog` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`body` varchar(500) NOT NULL DEFAULT '', | ||||
| @@ -42,9 +40,7 @@ CREATE TABLE `myaac_changelog` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| INSERT INTO `myaac_changelog` (`id`, `type`, `where`, `date`, `body`, `hide`) VALUES (1, 3, 2, UNIX_TIMESTAMP(), 'MyAAC installed. (:', 0); | ||||
|  | ||||
| CREATE TABLE `myaac_config` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_config` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`name` varchar(30) NOT NULL, | ||||
| @@ -53,9 +49,7 @@ CREATE TABLE `myaac_config` | ||||
| 	UNIQUE (`name`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| INSERT INTO `myaac_config` (`name`, `value`) VALUES ('database_version', @myaac_database_version); | ||||
|  | ||||
| CREATE TABLE `myaac_faq` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_faq` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`question` varchar(255) NOT NULL DEFAULT '', | ||||
| @@ -65,7 +59,7 @@ CREATE TABLE `myaac_faq` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_forum_boards` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_forum_boards` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`name` varchar(32) NOT NULL, | ||||
| @@ -77,13 +71,8 @@ CREATE TABLE `myaac_forum_boards` | ||||
| 	`hide` tinyint NOT NULL DEFAULT 0, | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
| INSERT INTO `myaac_forum_boards` (`id`, `name`, `description`, `ordering`, `closed`) VALUES (NULL, 'News', 'News commenting', 0, 1); | ||||
| INSERT INTO `myaac_forum_boards` (`id`, `name`, `description`, `ordering`) VALUES (NULL, 'Trade', 'Trade offers.', 1); | ||||
| INSERT INTO `myaac_forum_boards` (`id`, `name`, `description`, `ordering`) VALUES (NULL, 'Quests', 'Quest making.', 2); | ||||
| INSERT INTO `myaac_forum_boards` (`id`, `name`, `description`, `ordering`) VALUES (NULL, 'Pictures', 'Your pictures.', 3); | ||||
| INSERT INTO `myaac_forum_boards` (`id`, `name`, `description`, `ordering`) VALUES (NULL, 'Bug Report', 'Report bugs there.', 4); | ||||
|  | ||||
| CREATE TABLE `myaac_forum` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_forum` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`first_post` int NOT NULL DEFAULT 0, | ||||
| @@ -107,7 +96,7 @@ CREATE TABLE `myaac_forum` | ||||
| 	KEY `section` (`section`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_menu` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_menu` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`template` varchar(255) NOT NULL, | ||||
| @@ -121,7 +110,7 @@ CREATE TABLE `myaac_menu` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_monsters` ( | ||||
| CREATE TABLE IF NOT EXISTS `myaac_monsters` ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`hide` tinyint NOT NULL DEFAULT 0, | ||||
| 	`name` varchar(255) NOT NULL, | ||||
| @@ -154,7 +143,7 @@ CREATE TABLE `myaac_monsters` ( | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_news` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_news` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`title` varchar(100) NOT NULL, | ||||
| @@ -172,7 +161,7 @@ CREATE TABLE `myaac_news` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_news_categories` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_news_categories` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`name` varchar(50) NOT NULL DEFAULT "", | ||||
| @@ -182,13 +171,7 @@ CREATE TABLE `myaac_news_categories` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| INSERT INTO `myaac_news_categories` (`id`, `icon_id`) VALUES (NULL, 0); | ||||
| INSERT INTO `myaac_news_categories` (`id`, `icon_id`) VALUES (NULL, 1); | ||||
| INSERT INTO `myaac_news_categories` (`id`, `icon_id`) VALUES (NULL, 2); | ||||
| INSERT INTO `myaac_news_categories` (`id`, `icon_id`) VALUES (NULL, 3); | ||||
| INSERT INTO `myaac_news_categories` (`id`, `icon_id`) VALUES (NULL, 4); | ||||
|  | ||||
| CREATE TABLE `myaac_notepad` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_notepad` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`account_id` int NOT NULL, | ||||
| @@ -198,7 +181,7 @@ CREATE TABLE `myaac_notepad` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_pages` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_pages` | ||||
| ( | ||||
| 	`id` INT NOT NULL AUTO_INCREMENT, | ||||
| 	`name` varchar(30) NOT NULL, | ||||
| @@ -214,7 +197,7 @@ CREATE TABLE `myaac_pages` | ||||
| 	UNIQUE (`name`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_gallery` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_gallery` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`comment` varchar(255) NOT NULL DEFAULT '', | ||||
| @@ -226,9 +209,7 @@ CREATE TABLE `myaac_gallery` | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| INSERT INTO `myaac_gallery` (`id`, `ordering`, `comment`, `image`, `thumb`, `author`) VALUES (NULL, 1, 'Demon', 'images/gallery/demon.jpg', 'images/gallery/demon_thumb.gif', 'MyAAC'); | ||||
|  | ||||
| CREATE TABLE `myaac_settings` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_settings` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`name` varchar(255) NOT NULL DEFAULT '', | ||||
| @@ -238,7 +219,7 @@ CREATE TABLE `myaac_settings` | ||||
| 	KEY `key` (`key`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_spells` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_spells` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`spell` varchar(255) NOT NULL DEFAULT '', | ||||
| @@ -261,7 +242,7 @@ CREATE TABLE `myaac_spells` | ||||
| 	UNIQUE (`name`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_visitors` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_visitors` | ||||
| ( | ||||
| 	`ip` varchar(45) NOT NULL, | ||||
| 	`lastvisit` int NOT NULL DEFAULT 0, | ||||
| @@ -270,7 +251,7 @@ CREATE TABLE `myaac_visitors` | ||||
| 	UNIQUE (`ip`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
|  | ||||
| CREATE TABLE `myaac_weapons` | ||||
| CREATE TABLE IF NOT EXISTS `myaac_weapons` | ||||
| ( | ||||
| 	`id` int NOT NULL, | ||||
| 	`level` int NOT NULL DEFAULT 0, | ||||
|   | ||||
| @@ -30,27 +30,23 @@ if(!$error) { | ||||
| 	} | ||||
| } | ||||
|  | ||||
| if($db->hasTable(TABLE_PREFIX . 'account_actions')) { | ||||
| 	$locale['step_database_error_table_exist'] = str_replace('$TABLE$', TABLE_PREFIX . 'account_actions', $locale['step_database_error_table_exist']); | ||||
| 	warning($locale['step_database_error_table_exist']); | ||||
| } | ||||
| else { | ||||
| 	// import schema | ||||
| 	try { | ||||
| // import schema | ||||
| try { | ||||
| 	$locale['step_database_importing'] = str_replace('$DATABASE_NAME$', config('database_name'), $locale['step_database_importing']); | ||||
| 	success($locale['step_database_importing']); | ||||
|  | ||||
| 		$db->query(file_get_contents(BASE . 'install/includes/schema.sql')); | ||||
| 	$db->exec(file_get_contents(BASE . 'install/includes/schema.sql')); | ||||
|  | ||||
| 	$locale['step_database_success_schema'] = str_replace('$PREFIX$', TABLE_PREFIX, $locale['step_database_success_schema']); | ||||
| 	success($locale['step_database_success_schema']); | ||||
| 	} | ||||
| 	catch(PDOException $error_) { | ||||
| } | ||||
| catch(PDOException $error_) { | ||||
| 	error($locale['step_database_error_schema'] . ' ' . $error_); | ||||
| 	return; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| require BASE . 'install/includes/import_base_data.php'; | ||||
|  | ||||
| if(!$db->hasColumn('accounts', 'email')) { | ||||
| 	if(query("ALTER TABLE `accounts` ADD `email` varchar(255) NOT NULL DEFAULT '';")) | ||||
| 		success($locale['step_database_adding_field'] . ' accounts.email...'); | ||||
| @@ -102,18 +98,13 @@ if(!$db->hasColumn('accounts', 'web_flags')) { | ||||
| 		success($locale['step_database_adding_field'] . ' accounts.web_flags...'); | ||||
| } | ||||
|  | ||||
| if(!$db->hasColumn('accounts', 'email_hash')) { | ||||
| 	if(query("ALTER TABLE `accounts` ADD `email_hash` VARCHAR(32) NOT NULL DEFAULT '' AFTER `web_flags`;")) | ||||
| 		success($locale['step_database_adding_field'] . ' accounts.email_hash...'); | ||||
| } | ||||
|  | ||||
| if(!$db->hasColumn('accounts', 'email_verified')) { | ||||
| 	if(query("ALTER TABLE `accounts` ADD `email_verified` TINYINT(1) NOT NULL DEFAULT 0 AFTER `email_hash`;")) | ||||
| 	if(query("ALTER TABLE `accounts` ADD `email_verified` TINYINT(1) NOT NULL DEFAULT 0 AFTER `web_flags`;")) | ||||
| 		success($locale['step_database_adding_field'] . ' accounts.email_verified...'); | ||||
| } | ||||
|  | ||||
| if(!$db->hasColumn('accounts', 'email_new')) { | ||||
| 	if(query("ALTER TABLE `accounts` ADD `email_new` VARCHAR(255) NOT NULL DEFAULT '' AFTER `email_hash`;")) | ||||
| 	if(query("ALTER TABLE `accounts` ADD `email_new` VARCHAR(255) NOT NULL DEFAULT '' AFTER `email_verified`;")) | ||||
| 		success($locale['step_database_adding_field'] . ' accounts.email_new...'); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -220,6 +220,8 @@ switch ($action) { | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		/* | ||||
| 		 * not needed anymore? | ||||
| 		if (fieldExist('premdays', 'accounts') && fieldExist('lastday', 'accounts')) { | ||||
| 			$save = false; | ||||
| 			$timeNow = time(); | ||||
| @@ -256,6 +258,7 @@ switch ($action) { | ||||
| 				$account->save(); | ||||
| 			} | ||||
| 		} | ||||
| 		*/ | ||||
|  | ||||
| 		$worlds = [$world]; | ||||
| 		$playdata = compact('worlds', 'characters'); | ||||
|   | ||||
| @@ -81,6 +81,7 @@ $deprecatedConfig = [ | ||||
| 	'account_change_character_name_points' => 'account_change_character_name_price', | ||||
| 	'account_change_character_sex', | ||||
| 	'account_change_character_sex_points' => 'account_change_character_name_price', | ||||
| 	'email_lai_sec_interval' => 'mail_lost_account_interval', | ||||
| ]; | ||||
|  | ||||
| foreach ($deprecatedConfig as $key => $value) { | ||||
|   | ||||
| @@ -1142,10 +1142,18 @@ function getTopPlayers($limit = 5, $skill = 'level') { | ||||
| 			'looktype', 'lookhead', 'lookbody', 'looklegs', 'lookfeet' | ||||
| 		]; | ||||
|  | ||||
| 		if ($db->hasColumn('players', 'promotion')) { | ||||
| 			$columns[] = 'promotion'; | ||||
| 		} | ||||
|  | ||||
| 		if ($db->hasColumn('players', 'lookaddons')) { | ||||
| 			$columns[] = 'lookaddons'; | ||||
| 		} | ||||
|  | ||||
| 		if ($db->hasColumn('players', 'lookmount')) { | ||||
| 			$columns[] = 'lookmount'; | ||||
| 		} | ||||
|  | ||||
| 		return Player::query() | ||||
| 			->select($columns) | ||||
| 			->withOnlineStatus() | ||||
| @@ -1632,13 +1640,14 @@ function camelCaseToUnderscore($input) | ||||
| 	return ltrim(strtolower(preg_replace('/[A-Z]([A-Z](?![a-z]))*/', '_$0', $input)), '_'); | ||||
| } | ||||
|  | ||||
| function removeIfFirstSlash(&$text) { | ||||
| function removeIfFirstSlash(&$text): void | ||||
| { | ||||
| 	if(strpos($text, '/') === 0) { | ||||
| 		$text = str_replace_first('/', '', $text); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| function escapeHtml($html) { | ||||
| function escapeHtml($html): string { | ||||
| 	return htmlspecialchars($html); | ||||
| } | ||||
|  | ||||
| @@ -1652,7 +1661,7 @@ function getGuildNameById($id) | ||||
| 	return false; | ||||
| } | ||||
|  | ||||
| function getGuildLogoById($id) | ||||
| function getGuildLogoById($id): string | ||||
| { | ||||
| 	$logo = 'default.gif'; | ||||
|  | ||||
| @@ -1668,7 +1677,8 @@ function getGuildLogoById($id) | ||||
| 	return BASE_URL . GUILD_IMAGES_DIR . $logo; | ||||
| } | ||||
|  | ||||
| function displayErrorBoxWithBackButton($errors, $action = null) { | ||||
| function displayErrorBoxWithBackButton($errors, $action = null): void | ||||
| { | ||||
| 	global $twig; | ||||
| 	$twig->display('error_box.html.twig', ['errors' => $errors]); | ||||
| 	$twig->display('account.back_button.html.twig', [ | ||||
| @@ -1696,6 +1706,12 @@ function getAccountIdentityColumn(): string | ||||
| 	return 'id'; | ||||
| } | ||||
|  | ||||
| function isCanary(): bool | ||||
| { | ||||
| 	$vipSystemEnabled = configLua('vipSystemEnabled'); | ||||
| 	return isset($vipSystemEnabled); | ||||
| } | ||||
|  | ||||
| // validator functions | ||||
| require_once SYSTEM . 'compat/base.php'; | ||||
|  | ||||
|   | ||||
| @@ -12,6 +12,8 @@ | ||||
|  * @license http://www.gnu.org/licenses/lgpl-3.0.txt GNU Lesser General Public License, Version 3 | ||||
|  */ | ||||
|  | ||||
| use MyAAC\Models\AccountAction; | ||||
|  | ||||
| /** | ||||
|  * OTServ account abstraction. | ||||
|  * | ||||
| @@ -443,8 +445,9 @@ class OTS_Account extends OTS_Row_DAO implements IteratorAggregate, Countable | ||||
| 			throw new E_OTS_NotLoaded(); | ||||
| 		} | ||||
|  | ||||
| 		if(isset($this->data['premium_ends_at']) || isset($this->data['premend'])) { | ||||
| 			$col = isset($this->data['premium_ends_at']) ? 'premium_ends_at' : 'premend'; | ||||
| 		if(isset($this->data['premium_ends_at']) || isset($this->data['premend']) || | ||||
| 			(isCanary() && isset($this->data['lastday']))) { | ||||
| 				$col = (isset($this->data['premium_ends_at']) ? 'premium_ends_at' : (isset($this->data['lastday']) ? 'lastday' : 'premend')); | ||||
| 				$ret = ceil(($this->data[$col] - time()) / (24 * 60 * 60)); | ||||
| 				return max($ret, 0); | ||||
| 		} | ||||
| @@ -471,14 +474,16 @@ class OTS_Account extends OTS_Row_DAO implements IteratorAggregate, Countable | ||||
| 		return $this->data['lastday']; | ||||
| 	} | ||||
|  | ||||
|     public function isPremium() | ||||
| 	public function isPremium(): bool | ||||
| 	{ | ||||
| 		if(isset($this->data['premium_ends_at'])) { | ||||
| 			return $this->data['premium_ends_at'] > time(); | ||||
| 		if(isset($this->data['premium_ends_at']) || isset($this->data['premend']) || | ||||
| 			(isCanary() && isset($this->data['lastday']))) { | ||||
| 			$col = (isset($this->data['premium_ends_at']) ? 'premium_ends_at' : (isset($this->data['lastday']) ? 'lastday' : 'premend')); | ||||
| 			return $this->data[$col] > time(); | ||||
| 		} | ||||
|  | ||||
| 		if(isset($this->data['premend'])) { | ||||
| 			return $this->data['premend'] > time(); | ||||
| 		if($this->data['premdays'] == self::GRATIS_PREMIUM_DAYS){ | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		return ($this->data['premdays'] - (date("z", time()) + (365 * (date("Y", time()) - date("Y", $this->data['lastday']))) - date("z", $this->data['lastday'])) > 0); | ||||
| @@ -1004,26 +1009,16 @@ class OTS_Account extends OTS_Row_DAO implements IteratorAggregate, Countable | ||||
|  | ||||
| 	public function logAction($action) | ||||
| 	{ | ||||
| 		$ip = get_browser_real_ip(); | ||||
| 		if(!str_contains($ip, ":")) { | ||||
| 			$ipv6 = '0'; | ||||
| 		} | ||||
| 		else { | ||||
| 			$ipv6 = $ip; | ||||
| 			$ip = ''; | ||||
| 		AccountAction::create([ | ||||
| 			'account_id' => $this->getId(), | ||||
| 			'ip' => get_browser_real_ip(), | ||||
| 			'date' => time(), | ||||
| 			'action' => $action, | ||||
| 		]); | ||||
| 	} | ||||
|  | ||||
| 		return $this->db->exec('INSERT INTO `' . TABLE_PREFIX . 'account_actions` (`account_id`, `ip`, `ipv6`, `date`, `action`) VALUES (' . $this->db->quote($this->getId()).', ' . ($ip == '' ? '0' : $this->db->quote(ip2long($ip))) . ', (' . ($ipv6 == '0' ? $this->db->quote('') : $this->db->quote(inet_pton($ipv6))) . '), UNIX_TIMESTAMP(NOW()), ' . $this->db->quote($action).')'); | ||||
| 	} | ||||
|  | ||||
| 	public function getActionsLog($limit1, $limit2) | ||||
| 	{ | ||||
| 		$actions = array(); | ||||
|  | ||||
| 		foreach($this->db->query('SELECT `ip`, `ipv6`, `date`, `action` FROM `' . TABLE_PREFIX . 'account_actions` WHERE `account_id` = ' . $this->data['id'] . ' ORDER by `date` DESC LIMIT ' . $limit1 . ', ' . $limit2 . '')->fetchAll() as $a) | ||||
| 			$actions[] = array('ip' => $a['ip'], 'ipv6' => $a['ipv6'], 'date' => $a['date'], 'action' => $a['action']); | ||||
|  | ||||
| 		return $actions; | ||||
| 	public function getActionsLog($limit) { | ||||
| 		return AccountAction::where('account_id', $this->data['id'])->orderByDesc('date')->limit($limit)->get()->toArray(); | ||||
| 	} | ||||
| /** | ||||
|  * Returns players iterator. | ||||
|   | ||||
| @@ -26,10 +26,11 @@ use MyAAC\Cache\Cache; | ||||
|  */ | ||||
| class OTS_DB_MySQL extends OTS_Base_DB | ||||
| { | ||||
| 	private $has_table_cache = array(); | ||||
| 	private $has_column_cache = array(); | ||||
| 	private array $has_table_cache = []; | ||||
| 	private array $has_column_cache = []; | ||||
| 	private array $get_column_info_cache = []; | ||||
|  | ||||
| 	private $clearCacheAfter = false; | ||||
| 	private bool $clearCacheAfter = false; | ||||
| /** | ||||
|  * Creates database connection. | ||||
|  * | ||||
| @@ -119,6 +120,11 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 				if($cache->fetch('database_columns', $tmp) && $tmp) { | ||||
| 					$this->has_column_cache = unserialize($tmp); | ||||
| 				} | ||||
|  | ||||
| 				$tmp = null; | ||||
| 				if($cache->fetch('database_columns_info', $tmp) && $tmp) { | ||||
| 					$this->get_column_info_cache = unserialize($tmp); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| @@ -155,11 +161,13 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 			if ($this->clearCacheAfter) { | ||||
| 				$cache->delete('database_tables'); | ||||
| 				$cache->delete('database_columns'); | ||||
| 				$cache->delete('database_columns_info'); | ||||
| 				$cache->delete('database_checksum'); | ||||
| 			} | ||||
| 			else { | ||||
| 				$cache->set('database_tables', serialize($this->has_table_cache), 3600); | ||||
| 				$cache->set('database_columns', serialize($this->has_column_cache), 3600); | ||||
| 				$cache->set('database_columns_info', serialize($this->get_column_info_cache), 3600); | ||||
| 				$cache->set('database_checksum', serialize(sha1($config['database_host'] . '.' . $config['database_name'])), 3600); | ||||
| 			} | ||||
| 		} | ||||
| @@ -209,7 +217,8 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 		return $sql; | ||||
| 	} | ||||
|  | ||||
| 	public function hasTable($name) { | ||||
| 	public function hasTable($name): bool | ||||
| 	{ | ||||
| 		if(isset($this->has_table_cache[$name])) { | ||||
| 			return $this->has_table_cache[$name]; | ||||
| 		} | ||||
| @@ -217,12 +226,13 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 		return $this->hasTableInternal($name); | ||||
| 	} | ||||
|  | ||||
| 	private function hasTableInternal($name) { | ||||
| 		global $config; | ||||
| 		return ($this->has_table_cache[$name] = $this->query('SELECT `TABLE_NAME` FROM `information_schema`.`tables` WHERE `TABLE_SCHEMA` = ' . $this->quote($config['database_name']) . ' AND `TABLE_NAME` = ' . $this->quote($name) . ' LIMIT 1;')->rowCount() > 0); | ||||
| 	private function hasTableInternal($name): bool | ||||
| 	{ | ||||
| 		return ($this->has_table_cache[$name] = $this->query('SELECT `TABLE_NAME` FROM `information_schema`.`tables` WHERE `TABLE_SCHEMA` = ' . $this->quote(config('database_name')) . ' AND `TABLE_NAME` = ' . $this->quote($name) . ' LIMIT 1;')->rowCount() > 0); | ||||
| 	} | ||||
|  | ||||
| 	public function hasColumn($table, $column) { | ||||
| 	public function hasColumn($table, $column): bool | ||||
| 	{ | ||||
| 		if(isset($this->has_column_cache[$table . '.' . $column])) { | ||||
| 			return $this->has_column_cache[$table . '.' . $column]; | ||||
| 		} | ||||
| @@ -230,8 +240,8 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 		return $this->hasColumnInternal($table, $column); | ||||
| 	} | ||||
|  | ||||
| 	private function hasColumnInternal($table, $column) { | ||||
| 		return $this->hasTable($table) && ($this->has_column_cache[$table . '.' . $column] = count($this->query('SHOW COLUMNS FROM `' . $table . "` LIKE '" . $column . "'")->fetchAll()) > 0); | ||||
| 	private function hasColumnInternal($table, $column): bool { | ||||
| 		return $this->hasTable($table) && ($this->has_column_cache[$table . '.' . $column] = count($this->query('SHOW COLUMNS FROM `' . $table . "` LIKE " . $this->quote($column))->fetchAll()) > 0); | ||||
| 	} | ||||
|  | ||||
| 	public function hasTableAndColumns(string $table, array $columns = []): bool | ||||
| @@ -247,7 +257,53 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 		return true; | ||||
| 	} | ||||
|  | ||||
| 	public function revalidateCache() { | ||||
| 	public function getColumnInfo(string $table, string $column): bool|array | ||||
| 	{ | ||||
| 		if(isset($this->get_column_info_cache[$table . '.' . $column])) { | ||||
| 			return $this->get_column_info_cache[$table . '.' . $column]; | ||||
| 		} | ||||
|  | ||||
| 		return $this->getColumnInfoInternal($table, $column); | ||||
| 	} | ||||
|  | ||||
| 	private function getColumnInfoInternal(string $table, string $column): bool|array | ||||
| 	{ | ||||
| 		if (!$this->hasTable($table) || !$this->hasColumn($table, $column)) { | ||||
| 			return false; | ||||
| 		} | ||||
|  | ||||
| 		$formatResult = function ($result) { | ||||
| 			return [ | ||||
| 				'field' => $result['Field'], | ||||
| 				'type' => $result['Type'], | ||||
| 				'null' => strtolower($result['Null']), | ||||
| 				'default' => $result['Default'], | ||||
| 				'extra' => $result['Extra'], | ||||
| 			]; | ||||
| 		}; | ||||
|  | ||||
| 		$query = $this->query('SHOW COLUMNS FROM `' . $table . "` LIKE " . $this->quote($column)); | ||||
| 		$rowCount = $query->rowCount(); | ||||
| 		if ($rowCount > 1) { | ||||
| 			$tmp = []; | ||||
|  | ||||
| 			$results = $query->fetchAll(PDO::FETCH_ASSOC); | ||||
| 			foreach ($results as $result) { | ||||
| 				$tmp[] = $formatResult($result); | ||||
| 			} | ||||
|  | ||||
| 			return ($this->get_column_info_cache[$table . '.' . $column] = $tmp); | ||||
| 		} | ||||
| 		else if ($rowCount == 1) { | ||||
| 			$result = $query->fetch(PDO::FETCH_ASSOC); | ||||
| 			return ($this->get_column_info_cache[$table . '.' . $column] = $formatResult($result)); | ||||
| 		} | ||||
|  | ||||
| 		return []; | ||||
| 	} | ||||
|  | ||||
| 	public function revalidateCache(): void | ||||
| 	{ | ||||
| 		foreach($this->has_table_cache as $key => $value) { | ||||
| 			$this->hasTableInternal($key); | ||||
| 		} | ||||
| @@ -262,6 +318,21 @@ class OTS_DB_MySQL extends OTS_Base_DB | ||||
| 				$this->hasColumnInternal($explode[0], $explode[1]); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		foreach($this->get_column_info_cache as $key => $value) { | ||||
| 			$explode = explode('.', $key); | ||||
| 			if(!isset($this->has_table_cache[$explode[0]])) { // first check if table exist | ||||
| 				$this->hasTableInternal($explode[0]); | ||||
| 			} | ||||
|  | ||||
| 			if($this->has_table_cache[$explode[0]]) { | ||||
| 				$this->hasColumnInternal($explode[0], $explode[1]); | ||||
| 			} | ||||
|  | ||||
| 			if($this->has_table_cache[$explode[0]]) { | ||||
| 				$this->getColumnInfoInternal($explode[0], $explode[1]); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public function setClearCacheAfter($clearCache) | ||||
|   | ||||
| @@ -78,6 +78,7 @@ $locale['step_database_error_mysql_connect_3'] = 'MySQL ist nicht richtig konfig | ||||
| $locale['step_database_error_mysql_connect_4'] = 'MySQL-Server läuft nicht.'; | ||||
| $locale['step_database_error_schema'] = 'Fehler beim Importieren des Schemas:'; | ||||
| $locale['step_database_success_schema'] = '$PREFIX$ Tabellen wurden erfolgreich installiert.'; | ||||
| $locale['step_database_success_import_data'] = 'Import von Daten für Tabellen was erfolgreich.'; | ||||
| $locale['step_database_error_file'] = '$FILE$ konnte nicht geöffnet werden. Bitte kopieren Sie diesen Inhalt und fügen Sie ihn dort ein:'; | ||||
| $locale['step_database_adding_field'] = 'Folgendes Feld wurde hinzugefügt: '; | ||||
| $locale['step_database_modifying_field'] = 'Folgendes Feld wurde geändert: '; | ||||
|   | ||||
| @@ -83,6 +83,7 @@ $locale['step_database_error_mysql_connect_3'] = 'MySQL is not configured proper | ||||
| $locale['step_database_error_mysql_connect_4'] = 'MySQL server is not running.'; | ||||
| $locale['step_database_error_schema'] = 'Error while importing schema:'; | ||||
| $locale['step_database_success_schema'] = 'Successfully installed $PREFIX$ tables.'; | ||||
| $locale['step_database_success_import_data'] = 'Successfully imported base data for tables.'; | ||||
| $locale['step_database_error_file'] = '$FILE$ couldn\'t be opened. Please copy this content and paste there:'; | ||||
| $locale['step_database_adding_field'] = 'Adding field'; | ||||
| $locale['step_database_modifying_field'] = 'Modifying field'; | ||||
|   | ||||
| @@ -81,7 +81,8 @@ $locale['step_database_error_mysql_connect_2'] = 'Możliwe przyczyny:'; | ||||
| $locale['step_database_error_mysql_connect_3'] = 'MySQL nie jest poprawnie skonfigurowane w <i>config.lua</i>.'; | ||||
| $locale['step_database_error_mysql_connect_4'] = 'Serwer MySQL nie jest uruchomiony.'; | ||||
| $locale['step_database_error_schema'] = 'Błąd podczas importowania struktury bazy danych:'; | ||||
| $locale['step_database_success_schema'] = 'Pomyślnie zainstalowano tabele $PREFIX$.'; | ||||
| $locale['step_database_success_schema'] = 'Pomyślnie zaimportowano tabele $PREFIX$.'; | ||||
| $locale['step_database_success_import_data'] = 'Pomyślnie załadowano bazowe dane dla tabel.'; | ||||
| $locale['step_database_error_file'] = '$FILE$ nie mógł zostać otwarty. Proszę skopiować zawartość pola tekstowego i wkleić do tego pliku:'; | ||||
| $locale['step_database_adding_field'] = 'Dodawanie pola'; | ||||
| $locale['step_database_modifying_field'] = 'Modyfikacja pola'; | ||||
|   | ||||
| @@ -34,8 +34,10 @@ if($logged) { | ||||
| 	$twig->addGlobal('account_logged', $account_logged); | ||||
| } | ||||
|  | ||||
| setSession('last_visit', time()); | ||||
| if(defined('PAGE')) { | ||||
| if (!defined('IGNORE_SET_LAST_VISIT') || !IGNORE_SET_LAST_VISIT) { | ||||
| 	setSession('last_visit', time()); | ||||
| 	if(defined('PAGE')) { | ||||
| 		setSession('last_page', PAGE); | ||||
| 	} | ||||
| 	setSession('last_uri', $_SERVER['REQUEST_URI']); | ||||
| } | ||||
| setSession('last_uri', $_SERVER['REQUEST_URI']); | ||||
|   | ||||
| @@ -9,6 +9,8 @@ | ||||
|  */ | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| global $db; | ||||
|  | ||||
| // database migrations | ||||
| $tmp = ''; | ||||
| if(fetchDatabaseConfig('database_version', $tmp)) { // we got version | ||||
|   | ||||
| @@ -1,8 +0,0 @@ | ||||
| CREATE TABLE `myaac_account_email_codes` | ||||
| ( | ||||
| 	`id` int(11) NOT NULL AUTO_INCREMENT, | ||||
| 	`account_id` int NOT NULL, | ||||
| 	`code` varchar(6) NOT NULL, | ||||
| 	`created_at` int NOT NULL, | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
							
								
								
									
										8
									
								
								system/migrations/46-account_emails_verify.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								system/migrations/46-account_emails_verify.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| CREATE TABLE `myaac_account_emails_verify` | ||||
| ( | ||||
| 	`id` int NOT NULL AUTO_INCREMENT, | ||||
| 	`account_id` int NOT NULL, | ||||
| 	`hash` varchar(32) NOT NULL, | ||||
| 	`sent_at` int NOT NULL DEFAULT 0, | ||||
| 	PRIMARY KEY (`id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARACTER SET=utf8mb4; | ||||
| @@ -1,27 +1,24 @@ | ||||
| <?php | ||||
| // add the myaac_account_email_codes | ||||
|  | ||||
| /** | ||||
|  * @var OTS_DB_MySQL $db | ||||
|  */ | ||||
|  | ||||
| $up = function () use ($db) { | ||||
| 	if (!$db->hasColumn('accounts', '2fa_type')) { | ||||
| 		$db->addColumn('accounts', '2fa_type', "tinyint NOT NULL DEFAULT 0 AFTER `web_flags`"); | ||||
| 	if ($db->hasColumn('accounts', 'email_hash')) { | ||||
| 		$db->dropColumn('accounts', 'email_hash'); | ||||
| 	} | ||||
|  | ||||
| 	// add myaac_account_email_codes table | ||||
| 	if (!$db->hasTable(TABLE_PREFIX . 'account_email_codes')) { | ||||
| 		$db->exec(file_get_contents(__DIR__ . '/46-account_email_codes.sql')); | ||||
| 	if (!$db->hasTable(TABLE_PREFIX . 'account_emails_verify')) { | ||||
| 		$db->query(file_get_contents(__DIR__ . '/46-account_emails_verify.sql')); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| $down = function () use ($db) { | ||||
| 	if ($db->hasColumn('accounts', '2fa_type')) { | ||||
| 		$db->dropColumn('accounts', '2fa_type'); | ||||
| 	if (!$db->hasColumn('accounts', 'email_hash')) { | ||||
| 		$db->addColumn('accounts', 'email_hash', "varchar(32) NOT NULL DEFAULT ''"); | ||||
| 	} | ||||
|  | ||||
| 	//if ($db->hasTable(TABLE_PREFIX . 'account_email_codes')) { | ||||
| 	//	$db->dropTable(TABLE_PREFIX . 'account_email_codes'); | ||||
| 	//} | ||||
| 	if ($db->hasTable(TABLE_PREFIX . 'account_emails_verify')) { | ||||
| 		$db->dropTable(TABLE_PREFIX . 'account_emails_verify'); | ||||
| 	} | ||||
| }; | ||||
|   | ||||
							
								
								
									
										27
									
								
								system/migrations/47.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								system/migrations/47.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| <?php | ||||
| /** | ||||
|  * @var OTS_DB_MySQL $db | ||||
|  */ | ||||
|  | ||||
| // 2025-02-27 | ||||
| // remove ipv6, change to ip (for both ipv4 + ipv6) as VARCHAR(45) | ||||
| $up = function () use ($db) { | ||||
| 	$db->query("ALTER TABLE `myaac_account_actions` DROP KEY `account_id`;"); | ||||
| 	$db->query("ALTER TABLE  `myaac_account_actions` ADD COLUMN `id` INT(11) NOT NULL AUTO_INCREMENT FIRST, ADD PRIMARY KEY (`id`);"); | ||||
|  | ||||
| 	$db->modifyColumn(TABLE_PREFIX . 'account_actions', 'ip', "VARCHAR(45) NOT NULL DEFAULT ''"); | ||||
| 	$db->query("UPDATE `" . TABLE_PREFIX . "account_actions` SET `ip` = INET_NTOA(`ip`) WHERE `ip` != '0';"); | ||||
| 	$db->query("UPDATE `" . TABLE_PREFIX . "account_actions` SET `ip` = INET6_NTOA(`ipv6`) WHERE `ip` = '0';"); | ||||
| 	$db->dropColumn(TABLE_PREFIX . 'account_actions', 'ipv6'); | ||||
| }; | ||||
|  | ||||
| $down = function () use ($db) { | ||||
| 	$db->query("ALTER TABLE `" . TABLE_PREFIX . "account_actions` DROP `id`;"); | ||||
| 	$db->query("ALTER TABLE  `" . TABLE_PREFIX . "account_actions` ADD KEY (`account_id`);"); | ||||
|  | ||||
| 	$db->addColumn(TABLE_PREFIX . 'account_actions', 'ipv6', "BINARY(16) NOT NULL DEFAULT 0x00000000000000000000000000000000 AFTER ip"); | ||||
| 	$db->query("UPDATE `" . TABLE_PREFIX . "account_actions` SET `ipv6` = INET6_ATON(ip) WHERE NOT IS_IPV4(`ip`);"); | ||||
| 	$db->query("UPDATE `" . TABLE_PREFIX . "account_actions` SET `ip` = INET_ATON(`ip`) WHERE IS_IPV4(`ip`);"); | ||||
| 	$db->query("UPDATE `" . TABLE_PREFIX . "account_actions` SET `ip` = 0 WHERE `ipv6` != 0x00000000000000000000000000000000;"); | ||||
| 	$db->modifyColumn(TABLE_PREFIX . 'account_actions', 'ip', "INT(11) UNSIGNED NOT NULL DEFAULT 0;"); | ||||
| }; | ||||
| @@ -1,124 +0,0 @@ | ||||
| <?php | ||||
| /** | ||||
|  * 2-factor authentication | ||||
|  * | ||||
|  * @package   MyAAC | ||||
|  * @author    Slawkens <slawkens@gmail.com> | ||||
|  * @copyright 2019 MyAAC | ||||
|  * @link      https://my-aac.org | ||||
|  */ | ||||
|  | ||||
| use MyAAC\TwoFactorAuth\TwoFactorAuth; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| $title = 'Two Factor Authentication'; | ||||
| require __DIR__ . '/base.php'; | ||||
|  | ||||
| csrfProtect(); | ||||
|  | ||||
| /** | ||||
|  * @var OTS_Account $account_logged | ||||
|  */ | ||||
| $step = $_REQUEST['step'] ?? ''; | ||||
| $code = $_REQUEST['auth-code'] ?? ''; | ||||
|  | ||||
| if ((!setting('core.mail_enabled')) && ACTION == 'email-code') { | ||||
| 	$twig->display('error_box.html.twig',  ['errors' => ['Account two-factor e-mail authentication disabled.']]); | ||||
| 	return; | ||||
| } | ||||
|  | ||||
| if (!isset($account_logged) || !$account_logged->isLoaded()) { | ||||
| 	$current_session = getSession('account'); | ||||
| 	if($current_session) { | ||||
| 		$account_logged = new OTS_Account(); | ||||
| 		$account_logged->load($current_session); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| $twoFactorAuth = TwoFactorAuth::getInstance($account_logged); | ||||
| $twig->addGlobal('account_logged', $account_logged); | ||||
|  | ||||
| if (ACTION == 'email-code') { | ||||
| 	if ($step == 'resend') { | ||||
| 		if ($twoFactorAuth->hasRecentEmailCode(15 * 60)) { | ||||
| 			$errors = ['Sorry, one email per 15 minutes']; | ||||
| 		} | ||||
| 		else { | ||||
| 			$twoFactorAuth->resendEmailCode(); | ||||
| 		} | ||||
|  | ||||
| 		if (!empty($errors)) { | ||||
| 			$twig->display('error_box.html.twig',  ['errors' => $errors]); | ||||
| 		} | ||||
|  | ||||
| 		$twig->display('account.2fa.email.login.html.twig'); | ||||
| 	} | ||||
| 	else if ($step == 'activate') { | ||||
| 		if (!$twoFactorAuth->hasRecentEmailCode(15 * 60)) { | ||||
| 			$twoFactorAuth->resendEmailCode(); | ||||
| 		} | ||||
|  | ||||
| 		if (isset($_POST['save'])) { | ||||
| 			if (!empty($code)) { | ||||
| 				$twoFactorAuth->setAuthGateway(TwoFactorAuth::TYPE_EMAIL); | ||||
| 				if ($twoFactorAuth->getAuthGateway()->verifyCode($code)) { | ||||
| 					$serverName = configLua('serverName'); | ||||
|  | ||||
| 					$twoFactorAuth->enable(TwoFactorAuth::TYPE_EMAIL); | ||||
| 					$twoFactorAuth->deleteOldCodes(); | ||||
|  | ||||
| 					$twig->display('success.html.twig', [ | ||||
| 						'title' => 'Email Code Authentication Activated', | ||||
| 						'description' => sprintf('You have successfully activated <b>email code authentication</b> for your account. This means an <b>email code</b> will be sent to the email address assigned to your account whenever you try to log in to the %s client or the %s website. In order to log in, you will need to enter the <b>most recent email code</b> you have received.', $serverName, $serverName) | ||||
| 					]); | ||||
|  | ||||
| 					return; | ||||
| 				} | ||||
| 				else { | ||||
| 					$errors[] = 'Invalid email code!'; | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (!empty($errors)) { | ||||
| 			$twig->display('error_box.html.twig', ['errors' => $errors]); | ||||
| 		} | ||||
|  | ||||
| 		$twig->display('account.2fa.email_code.html.twig', ['wrongCode' => count($errors) > 0]); | ||||
| 	} | ||||
| 	else if ($step == 'deactivate') { | ||||
| 		//if (!$twoFactorAuth->hasRecentEmailCode(15 * 60)) { | ||||
| 		//	$twoFactorAuth->resendEmailCode(); | ||||
| 		//} | ||||
|  | ||||
| 		/*if (isset($_POST['save'])) { | ||||
| 			if (!empty($code)) { | ||||
| 				if ($twoFactorAuth->getAuthGateway()->verifyCode($code)) { | ||||
| */ | ||||
| 					$twoFactorAuth->disable(); | ||||
| 					$twoFactorAuth->deleteOldCodes(); | ||||
|  | ||||
| 					$twig->display('success.html.twig', | ||||
| 						[ | ||||
| 							'title' => 'Email Code Authentication Deactivated', | ||||
| 							'description' => 'You have successfully <b>deactivated</b> the <b>Email Code Authentication</b> for your account.' | ||||
| 						] | ||||
| 					); | ||||
| 					/* | ||||
| 				} | ||||
| 				else { | ||||
| 					$errors[] = 'Invalid email code!'; | ||||
| 				} | ||||
| 			} | ||||
| 		}*/ | ||||
|  | ||||
| 		/* | ||||
| 		if (!empty($errors)) { | ||||
| 			$twig->display('error_box.html.twig', ['errors' => $errors]); | ||||
| 		} | ||||
|  | ||||
| 		$twig->display('account.2fa.email.deactivate.html.twig', ['wrongCode' => count($errors) > 0]); | ||||
| 		*/ | ||||
| 	} | ||||
| } | ||||
| @@ -17,10 +17,6 @@ if(!$logged) | ||||
| 	if(!empty($errors)) | ||||
| 		$twig->display('error_box.html.twig', array('errors' => $errors)); | ||||
|  | ||||
| 	if (defined('HIDE_LOGIN_BOX') && HIDE_LOGIN_BOX) { | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	$twig->display('account.login.html.twig', array( | ||||
| 		'redirect' => $_REQUEST['redirect'] ?? null, | ||||
| 		'account' => USE_ACCOUNT_NAME ? 'Name' : 'Number', | ||||
|   | ||||
| @@ -19,18 +19,17 @@ if(!$logged) { | ||||
|  | ||||
| csrfProtect(); | ||||
|  | ||||
| $new_password = $_POST['newpassword'] ?? NULL; | ||||
| $new_password_confirm = $_POST['newpassword_confirm'] ?? NULL; | ||||
| $old_password = $_POST['oldpassword'] ?? NULL; | ||||
| $new_password = $_POST['new_password'] ?? null; | ||||
| $new_password_confirm = $_POST['new_password_confirm'] ?? null; | ||||
| $old_password = $_POST['old_password'] ?? null; | ||||
| if(empty($new_password) && empty($new_password_confirm) && empty($old_password)) { | ||||
| 	$twig->display('account.change-password.html.twig'); | ||||
| } | ||||
| else | ||||
| { | ||||
| else { | ||||
| 	if(empty($new_password) || empty($new_password_confirm) || empty($old_password)){ | ||||
| 		$errors[] = 'Please fill in form.'; | ||||
| 	} | ||||
| 	$password_strlen = strlen($new_password); | ||||
|  | ||||
| 	if($new_password != $new_password_confirm) { | ||||
| 		$errors[] = 'The new passwords do not match!'; | ||||
| 	} | ||||
| @@ -41,10 +40,13 @@ else | ||||
| 		} | ||||
|  | ||||
| 		/** @var OTS_Account $account_logged */ | ||||
| 		$old_password = encrypt((USE_ACCOUNT_SALT ? $account_logged->getCustomField('salt') : '') . $old_password); | ||||
| 		if($old_password != $account_logged->getPassword()) { | ||||
| 		$old_password_hashed = encrypt((USE_ACCOUNT_SALT ? $account_logged->getCustomField('salt') : '') . $old_password); | ||||
| 		if($old_password_hashed != $account_logged->getPassword()) { | ||||
| 			$errors[] = 'Current password is incorrect!'; | ||||
| 		} | ||||
| 		else if ($old_password == $new_password) { | ||||
| 			$errors[] = 'The old password is same as the new password!'; | ||||
| 		} | ||||
|  | ||||
| 		$hooks->trigger(HOOK_ACCOUNT_CHANGE_PASSWORD_POST); | ||||
| 	} | ||||
|   | ||||
| @@ -9,6 +9,7 @@ | ||||
|  */ | ||||
|  | ||||
| use MyAAC\Models\Account; | ||||
| use MyAAC\Models\AccountEmailVerify; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| @@ -20,16 +21,20 @@ if(empty($hash)) { | ||||
| 	return; | ||||
| } | ||||
|  | ||||
| if(!Account::where('email_hash', $hash)->exists()) { | ||||
| 	note("Your email couldn't be verified. Please contact staff to do it manually."); | ||||
| // by default link is valid for 30 days | ||||
| $accountEmailVerify = AccountEmailVerify::where('hash', $hash)->where('sent_at', '>', time() - 30 * 24 * 60 * 60)->first(); | ||||
| if(!$accountEmailVerify) { | ||||
| 	note("Wrong link or link has expired."); | ||||
| } | ||||
| else | ||||
| { | ||||
| 	$accountModel = Account::where('email_hash', $hash)->where('email_verified', 0)->first(); | ||||
| 	$accountModel = Account::where('id', $accountEmailVerify->account_id)->where('email_verified', 0)->first(); | ||||
| 	if ($accountModel) { | ||||
| 		$accountModel->email_verified = 1; | ||||
| 		$accountModel->save(); | ||||
|  | ||||
| 		AccountEmailVerify::where('account_id', $accountModel->id)->delete(); | ||||
|  | ||||
| 		success('You have now verified your e-mail, this will increase the security of your account. Thank you for doing this. You can now <a href=' . getLink('account/manage') . '>log in</a>.'); | ||||
|  | ||||
| 		$account = new OTS_Account(); | ||||
| @@ -39,6 +44,6 @@ else | ||||
| 		} | ||||
| 	} | ||||
| 	else { | ||||
| 		error('Link has expired.'); | ||||
| 		error('Your account is already verified.'); | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -10,6 +10,7 @@ | ||||
|  */ | ||||
|  | ||||
| use MyAAC\CreateCharacter; | ||||
| use MyAAC\Models\AccountEmailVerify; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
| $title = 'Create Account'; | ||||
| @@ -244,7 +245,12 @@ if($save) | ||||
| 		if(setting('core.mail_enabled') && setting('core.account_mail_verify')) | ||||
| 		{ | ||||
| 			$hash = md5(generateRandomString(16, true, true) . $email); | ||||
| 			$new_account->setCustomField('email_hash', $hash); | ||||
|  | ||||
| 			AccountEmailVerify::create([ | ||||
| 				'account_id' => $new_account->getId(), | ||||
| 				'hash' => $hash, | ||||
| 				'sent_at' => time(), | ||||
| 			]); | ||||
|  | ||||
| 			$verify_url = getLink('account/confirm-email/' . $hash); | ||||
| 			$body_html = $twig->render('mail.account.verify.html.twig', array( | ||||
| @@ -268,8 +274,10 @@ if($save) | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				error('An error occorred while sending email! Account not created. Try again. For Admin: More info can be found in system/logs/mailer-error.log'); | ||||
| 				error('An error occurred while sending email! Account not created. Try again. For Admin: More info can be found in system/logs/mailer-error.log'); | ||||
| 				$new_account->delete(); | ||||
|  | ||||
| 				return; | ||||
| 			} | ||||
| 		} | ||||
| 		else | ||||
|   | ||||
| @@ -10,7 +10,6 @@ | ||||
|  */ | ||||
|  | ||||
| use MyAAC\RateLimit; | ||||
| use MyAAC\TwoFactorAuth\TwoFactorAuth; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| @@ -49,16 +48,12 @@ if(!empty($login_account) && !empty($login_password)) | ||||
| 	) | ||||
| 	{ | ||||
| 		if (setting('core.account_mail_verify') && (int)$account_logged->getCustomField('email_verified') !== 1) { | ||||
| 			$errors[] = 'Your account is not verified. Please verify your email address. If the message is not coming check the SPAM folder in your E-Mail client.'; | ||||
| 			$link = getLink('account/resend-email-verify'); | ||||
| 			$errors[] = 'Your account is not verified. Please verify your email address. If the message is not coming check the SPAM folder in your E-Mail client.<br/>' . | ||||
| 				'You can resend the Email here: <a href="' . $link . '">' . $link . '</a>'; | ||||
| 		} else { | ||||
| 			setSession('account', $account_logged->getId()); | ||||
|  | ||||
| 			$twoFactorAuth = TwoFactorAuth::getInstance($account_logged); | ||||
| 			if (!$twoFactorAuth->process($login_account, $login_password, $remember_me, $_POST['auth-code'] ?? '')) { | ||||
| 				return; | ||||
| 			} | ||||
|  | ||||
| 			session_regenerate_id(); | ||||
| 			setSession('account', $account_logged->getId()); | ||||
| 			setSession('password', encrypt((USE_ACCOUNT_SALT ? $account_logged->getCustomField('salt') : '') . $login_password)); | ||||
| 			if($remember_me) { | ||||
| 				setSession('remember_me', true); | ||||
|   | ||||
| @@ -8,9 +8,6 @@ | ||||
|  * @copyright 2019 MyAAC | ||||
|  * @link      https://my-aac.org | ||||
|  */ | ||||
|  | ||||
| use MyAAC\TwoFactorAuth\TwoFactorAuth; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| $title = 'Account Management'; | ||||
| @@ -41,15 +38,24 @@ csrfProtect(); | ||||
|  | ||||
| $groups = new OTS_Groups_List(); | ||||
|  | ||||
| $freePremium = isset($config['lua']['freePremium']) && getBoolean($config['lua']['freePremium']) || $account_logged->getPremDays() == OTS_Account::GRATIS_PREMIUM_DAYS; | ||||
| $dayOrDays = $account_logged->getPremDays() == 1 ? 'day' : 'days'; | ||||
| /** | ||||
|  * @var OTS_Account $account_logged | ||||
|  */ | ||||
| if(!$account_logged->isPremium()) | ||||
| $premDays = $account_logged->getPremDays(); | ||||
|  | ||||
| $freePremium = isset($config['lua']['freePremium']) && getBoolean($config['lua']['freePremium']) || $premDays == OTS_Account::GRATIS_PREMIUM_DAYS; | ||||
| $dayOrDays = ($premDays == 1 ? 'day' : 'days'); | ||||
|  | ||||
| $vipSystemEnabled = isset($config['lua']['vipSystemEnabled']) && getBoolean($config['lua']['vipSystemEnabled']); | ||||
| $premiumLabel = $vipSystemEnabled ? 'VIP' : 'Premium Account'; | ||||
|  | ||||
| if ($freePremium && !$vipSystemEnabled) { | ||||
| 	$account_status = '<b><span style="color: green">Gratis Premium Account</span></b>'; | ||||
| } else if(!$account_logged->isPremium()) { | ||||
| 	$account_status = '<b><span style="color: red">Free Account</span></b>'; | ||||
| else | ||||
| 	$account_status = '<b><span style="color: green">' . ($freePremium ? 'Gratis Premium Account' : 'Premium Account, ' . $account_logged->getPremDays() . ' '.$dayOrDays.' left') . '</span></b>'; | ||||
| } else { | ||||
| 	$account_status = '<b><span style="color: green">' . $premiumLabel . ', ' . $premDays . ' '.$dayOrDays.' left</span></b>'; | ||||
| } | ||||
|  | ||||
| $recovery_key = $account_logged->getCustomField('key'); | ||||
| if(empty($recovery_key)) | ||||
| @@ -90,12 +96,8 @@ if($email_new_time > 1) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| $actions = array(); | ||||
| foreach($account_logged->getActionsLog(0, 1000) as $action) { | ||||
| 	$actions[] = array('action' => $action['action'], 'date' => $action['date'], 'ip' => $action['ip'] != 0 ? long2ip($action['ip']) : inet_ntop($action['ipv6'])); | ||||
| } | ||||
| $actions = $account_logged->getActionsLog(1000); | ||||
|  | ||||
| $players = array(); | ||||
| /** @var OTS_Players_List $account_players */ | ||||
| $account_players = $account_logged->getPlayersList(); | ||||
| $account_players->orderBy('id'); | ||||
| @@ -114,8 +116,6 @@ $twig->display('account.management.html.twig', array( | ||||
| 	'account_registered' => $account_registered, | ||||
| 	'account_rlname' => $account_rlname, | ||||
| 	'account_location' => $account_location, | ||||
| 	'twoFactorViews' => TwoFactorAuth::getInstance($account_logged)->getAccountManageViews(), | ||||
|  | ||||
| 	'actions' => $actions, | ||||
| 	'players' => $account_players, | ||||
| 	'players' => $account_players | ||||
| )); | ||||
|   | ||||
							
								
								
									
										94
									
								
								system/pages/account/resend-email-verify.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								system/pages/account/resend-email-verify.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | ||||
| <?php | ||||
|  | ||||
| use MyAAC\Models\AccountEmailVerify; | ||||
|  | ||||
| defined('MYAAC') or die('Direct access not allowed!'); | ||||
|  | ||||
| $title = 'Resend Email'; | ||||
|  | ||||
| $errorWithBackButton = function ($msg) use ($twig) { | ||||
| 	$errors = [$msg]; | ||||
|  | ||||
| 	$twig->display('error_box.html.twig', ['errors' => $errors]); | ||||
| 	$twig->display('account.back_button.html.twig', [ | ||||
| 		'action' => getLink('account/resend-email-verify'), | ||||
| 	]); | ||||
| }; | ||||
|  | ||||
| if (!setting('core.mail_enabled') || !setting('core.account_mail_verify')) { | ||||
| 	$errorWithBackButton('Resending email is not possible on this server.'); | ||||
| 	return; | ||||
| } | ||||
|  | ||||
| $showForm = true; | ||||
|  | ||||
| if (isset($_POST['submit']) && $_POST['submit'] == '1') { | ||||
| 	$email = $_REQUEST['email']; | ||||
|  | ||||
| 	if (empty($email) || !filter_var($email, FILTER_VALIDATE_EMAIL)) { | ||||
| 		$errorWithBackButton('Please enter valid Email.'); | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	$account = new OTS_Account(); | ||||
| 	$account->findByEMail($email); | ||||
| 	if ($account->isLoaded()) { | ||||
| 		if ($account->getCustomField('email_verified') == '1') { | ||||
| 			$errorWithBackButton('This account is already verified! You can <a href=' . getLink('account/manage') . '>log in</a> on the website.'); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		$accountEmailVerify = AccountEmailVerify::where('account_id', $account->getId())->orderBy('sent_at', 'DESC')->first(); | ||||
| 		if ($accountEmailVerify && time() - $accountEmailVerify->sent_at < 60) { | ||||
| 			$errorWithBackButton('Only one Email per minute is allowed. Please try again later.'); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		$tmp_account = $email; | ||||
| 		if (!config('account_login_by_email')) { | ||||
| 			$tmp_account = (USE_ACCOUNT_NAME ? $account->getName() : $account->getId()); | ||||
| 		} | ||||
|  | ||||
| 		$hash = md5(generateRandomString(16, true, true) . $email); | ||||
|  | ||||
| 		AccountEmailVerify::create([ | ||||
| 			'account_id' => $account->getId(), | ||||
| 			'hash' => $hash, | ||||
| 			'sent_at' => time(), | ||||
| 		]); | ||||
|  | ||||
| 		$verify_url = getLink('account/confirm-email/' . $hash); | ||||
| 		$body_html = $twig->render('mail.account.resend-email-verify.html.twig', array( | ||||
| 			'account' => $tmp_account, | ||||
| 			'verify_url' => generateLink($verify_url, $verify_url, true) | ||||
| 		)); | ||||
|  | ||||
| 		if (_mail($account->getEMail(), configLua('serverName') . ' - Verify Account', $body_html)) { | ||||
| 			$message = "If account with this email exists - you will become an email with verification link."; | ||||
| 			$showForm = false; | ||||
| 		} else { | ||||
| 			$message = "<p class='error'>An error occurred while sending email (<b>{$email}</b> )! Try again later. For Admin: More info can be found in system/logs/mailer-error.log</p>"; | ||||
| 		} | ||||
| 	} | ||||
| 	else { | ||||
| 		$message = "<br />If account with this email exists - you will become an email with verification link."; | ||||
| 		$showForm = false; | ||||
| 	} | ||||
|  | ||||
| 	$twig->display('success.html.twig', array( | ||||
| 		'title' => 'Verify Email Sent', | ||||
| 		'description' => $message, | ||||
| 	)); | ||||
| } | ||||
|  | ||||
| //show errors if not empty | ||||
| if (!empty($errors)) { | ||||
| 	$twig->display('error_box.html.twig', ['errors' => $errors]); | ||||
| 	$twig->display('account.back_button.html.twig', [ | ||||
| 		'action' => getLink('account/resend-email-verify'), | ||||
| 	]); | ||||
| } | ||||
|  | ||||
| if ($showForm) { | ||||
| 	$twig->display('account.resend-email-verify.html.twig'); | ||||
| } | ||||
| @@ -21,6 +21,9 @@ if(!$logged) { | ||||
| 	$errors[] = 'You are not logged in. You can\'t create guild.'; | ||||
| } | ||||
|  | ||||
| $configLuaFreePremium = configLua('freePremium'); | ||||
| $freePremium = (isset($configLuaFreePremium) && getBoolean($configLuaFreePremium)) || ($logged && $account_logged->getPremDays() == OTS_Account::GRATIS_PREMIUM_DAYS); | ||||
|  | ||||
| $array_of_player_nig = array(); | ||||
| if(empty($errors)) | ||||
| { | ||||
| @@ -31,7 +34,7 @@ if(empty($errors)) | ||||
| 		if(!$player_rank->isLoaded()) | ||||
| 		{ | ||||
| 			if($player->getLevel() >= setting('core.guild_need_level')) { | ||||
| 				if(!setting('core.guild_need_premium') || $account_logged->isPremium()) { | ||||
| 				if(!setting('core.guild_need_premium') || $account_logged->isPremium() || $freePremium) { | ||||
| 					$array_of_player_nig[] = $player->getName(); | ||||
| 				} | ||||
| 			} | ||||
| @@ -95,7 +98,7 @@ if($todo == 'save') | ||||
| 		if($player->getLevel() < setting('core.guild_need_level')) { | ||||
| 			$errors[] = 'Character <b>'.$name.'</b> has too low level. To create guild you need character with level <b>' . setting('core.guild_need_level') . '</b>.'; | ||||
| 		} | ||||
| 		if(setting('core.guild_need_premium') && !$account_logged->isPremium()) { | ||||
| 		if(setting('core.guild_need_premium') && !$account_logged->isPremium() && !$freePremium) { | ||||
| 			$errors[] = 'Character <b>'.$name.'</b> is on FREE account. To create guild you need PREMIUM account.'; | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
| @@ -88,8 +88,10 @@ if($logged && $account_logged && $account_logged->isLoaded()) { | ||||
| /** | ||||
|  * Routes loading | ||||
|  */ | ||||
| $routesFinal = []; | ||||
| $dispatcher = FastRoute\cachedDispatcher(function (FastRoute\RouteCollector $r) { | ||||
| 	$routesFinal = []; | ||||
| 	global $cache, $routesFinal; | ||||
|  | ||||
| 	foreach(getDatabasePages() as $page) { | ||||
| 		$routesFinal[] = ['*', $page, '__database__/' . $page, 100]; | ||||
| 	} | ||||
| @@ -165,7 +167,7 @@ $dispatcher = FastRoute\cachedDispatcher(function (FastRoute\RouteCollector $r) | ||||
| 	echo '</pre>'; | ||||
| 	die; | ||||
| */ | ||||
| 	foreach ($routesFinal as $route) { | ||||
| 	foreach ($routesFinal as &$route) { | ||||
| 		if ($route[0] === '*') { | ||||
| 			$route[0] = ['GET', 'POST', 'PUT', 'DELETE', 'PATCH', 'HEAD']; | ||||
| 		} | ||||
| @@ -198,6 +200,10 @@ $dispatcher = FastRoute\cachedDispatcher(function (FastRoute\RouteCollector $r) | ||||
| 			log_append('router.log', $warning); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if ($cache->enabled()) { | ||||
| 		$cache->set('routes_final', serialize($routesFinal), 10 * 365 * 24 * 60 * 60); // 10 years / infinite | ||||
| 	} | ||||
| }, | ||||
| 	[ | ||||
| 		'cacheFile' => CACHE . 'route.cache', | ||||
| @@ -212,7 +218,7 @@ $found = true; | ||||
|  | ||||
| // old support for pages like /?subtopic=accountmanagement | ||||
| $page = $_REQUEST['p'] ?? ($_REQUEST['subtopic'] ?? ''); | ||||
| if(!empty($page) && preg_match('/^[A-z0-9\-]+$/', $page)) { | ||||
| if(!empty($page) && preg_match('/^[A-z0-9\/\-]+$/', $page)) { | ||||
| 	if (isset($_REQUEST['p'])) { // some plugins may require this | ||||
| 		$_REQUEST['subtopic'] = $_REQUEST['p']; | ||||
| 	} | ||||
| @@ -221,10 +227,27 @@ if(!empty($page) && preg_match('/^[A-z0-9\-]+$/', $page)) { | ||||
| 		require SYSTEM . 'compat/pages.php'; | ||||
| 	} | ||||
|  | ||||
| 	$foundRoute = false; | ||||
|  | ||||
| 	$tmp = null; | ||||
| 	if ($cache->enabled() && $cache->fetch('routes_final', $tmp)) { | ||||
| 		$routesFinal = unserialize($tmp); | ||||
| 	} | ||||
|  | ||||
| 	foreach ($routesFinal as $route) { | ||||
| 		if ($page === $route[1]) { | ||||
| 			$file = $route[2]; | ||||
| 			$foundRoute = true; | ||||
| 			break; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if (!$foundRoute) { | ||||
| 		$file = loadPageFromFileSystem($page, $found); | ||||
| 		if(!$found) { | ||||
| 			$file = false; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| else { | ||||
| 	$routeInfo = $dispatcher->dispatch($httpMethod, $uri); | ||||
|   | ||||
| @@ -115,6 +115,11 @@ class Cache | ||||
| 			return unserialize($value); | ||||
| 		} | ||||
|  | ||||
| 		// -1 for infinite cache | ||||
| 		if ($ttl == -1) { | ||||
| 			$ttl = 10 * 365 * 24 * 60 * 60; // 10 years should be enough | ||||
| 		} | ||||
|  | ||||
| 		$value = $callback(); | ||||
| 		$cache->set($key, serialize($value), $ttl); | ||||
| 		return $value; | ||||
|   | ||||
							
								
								
									
										33
									
								
								system/src/Commands/Env.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								system/src/Commands/Env.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\Commands; | ||||
|  | ||||
| use POT; | ||||
|  | ||||
| trait Env | ||||
| { | ||||
| 	protected function init(): void | ||||
| 	{ | ||||
| 		global $config; | ||||
| 		if (!isset($config['installed']) || !$config['installed']) { | ||||
| 			throw new \RuntimeException('MyAAC has not been installed yet or there was error during installation. Please install again.'); | ||||
| 		} | ||||
|  | ||||
| 		if(empty($config['server_path'])) { | ||||
| 			throw new \RuntimeException('Server Path has been not set. Go to config.php and set it.'); | ||||
| 		} | ||||
|  | ||||
| 		// take care of trailing slash at the end | ||||
| 		if($config['server_path'][strlen($config['server_path']) - 1] !== '/') | ||||
| 			$config['server_path'] .= '/'; | ||||
|  | ||||
| 		$config['lua'] = load_config_lua($config['server_path'] . 'config.lua'); | ||||
|  | ||||
| 		// POT | ||||
| 		require_once SYSTEM . 'libs/pot/OTS.php'; | ||||
| 		$ots = POT::getInstance(); | ||||
| 		$eloquentConnection = null; | ||||
|  | ||||
| 		require_once SYSTEM . 'database.php'; | ||||
| 	} | ||||
| } | ||||
| @@ -9,6 +9,8 @@ use Symfony\Component\Console\Style\SymfonyStyle; | ||||
|  | ||||
| class MigrateCommand extends Command | ||||
| { | ||||
| 	use Env; | ||||
|  | ||||
| 	protected function configure(): void | ||||
| 	{ | ||||
| 		$this->setName('migrate') | ||||
| @@ -17,9 +19,19 @@ class MigrateCommand extends Command | ||||
|  | ||||
| 	protected function execute(InputInterface $input, OutputInterface $output): int | ||||
| 	{ | ||||
| 		require SYSTEM . 'init.php'; | ||||
| 		$this->init(); | ||||
|  | ||||
| 		$io = new SymfonyStyle($input, $output); | ||||
|  | ||||
| 		$tmp = ''; | ||||
| 		if(fetchDatabaseConfig('database_version', $tmp)) { // we got version | ||||
| 			$tmp = (int)$tmp; | ||||
| 			if ($tmp >= DATABASE_VERSION) { | ||||
| 				$io->success('Already on latest version.'); | ||||
| 				return Command::SUCCESS; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		require SYSTEM . 'migrate.php'; | ||||
|  | ||||
| 		$io->success('Migrated to latest version (' . DATABASE_VERSION . ')'); | ||||
|   | ||||
| @@ -10,6 +10,8 @@ use Symfony\Component\Console\Style\SymfonyStyle; | ||||
|  | ||||
| class MigrateRunCommand extends Command | ||||
| { | ||||
| 	use Env; | ||||
|  | ||||
| 	protected function configure(): void | ||||
| 	{ | ||||
| 		$this->setName('migrate:run') | ||||
| @@ -23,12 +25,12 @@ class MigrateRunCommand extends Command | ||||
|  | ||||
| 	protected function execute(InputInterface $input, OutputInterface $output): int | ||||
| 	{ | ||||
| 		require SYSTEM . 'init.php'; | ||||
|  | ||||
| 		$io = new SymfonyStyle($input, $output); | ||||
|  | ||||
| 		$ids = $input->getArgument('id'); | ||||
|  | ||||
| 		$this->init(); | ||||
|  | ||||
| 		// pre-check | ||||
| 		// in case one of the migrations doesn't exist - we won't execute any of them | ||||
| 		foreach ($ids as $id) { | ||||
|   | ||||
| @@ -11,6 +11,8 @@ use Symfony\Component\Console\Style\SymfonyStyle; | ||||
|  | ||||
| class MigrateToCommand extends Command | ||||
| { | ||||
| 	use Env; | ||||
|  | ||||
| 	protected function configure(): void | ||||
| 	{ | ||||
| 		$this->setName('migrate:to') | ||||
| @@ -32,7 +34,7 @@ class MigrateToCommand extends Command | ||||
| 			return Command::FAILURE; | ||||
| 		} | ||||
|  | ||||
| 		$this->initEnv(); | ||||
| 		$this->init(); | ||||
|  | ||||
| 		$currentVersion = Config::where('name', 'database_version')->first()->value; | ||||
| 		if ($currentVersion > $versionDest) { | ||||
| @@ -80,29 +82,4 @@ class MigrateToCommand extends Command | ||||
|  | ||||
| 		updateDatabaseConfig('database_version', ($_up ? $id : $id - 1)); | ||||
| 	} | ||||
|  | ||||
| 	private function initEnv() | ||||
| 	{ | ||||
| 		global $config; | ||||
| 		if (!isset($config['installed']) || !$config['installed']) { | ||||
| 			throw new \RuntimeException('MyAAC has not been installed yet or there was error during installation. Please install again.'); | ||||
| 		} | ||||
|  | ||||
| 		if(empty($config['server_path'])) { | ||||
| 			throw new \RuntimeException('Server Path has been not set. Go to config.php and set it.'); | ||||
| 		} | ||||
|  | ||||
| 		// take care of trailing slash at the end | ||||
| 		if($config['server_path'][strlen($config['server_path']) - 1] !== '/') | ||||
| 			$config['server_path'] .= '/'; | ||||
|  | ||||
| 		$config['lua'] = load_config_lua($config['server_path'] . 'config.lua'); | ||||
|  | ||||
| 		// POT | ||||
| 		require_once SYSTEM . 'libs/pot/OTS.php'; | ||||
| 		$ots = POT::getInstance(); | ||||
| 		$eloquentConnection = null; | ||||
|  | ||||
| 		require_once SYSTEM . 'database.php'; | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -76,10 +76,11 @@ class Items | ||||
|  | ||||
| 	public static function get($id) { | ||||
| 		self::load(); | ||||
| 		return isset(self::$items[$id]) ? self::$items[$id] : []; | ||||
| 		return self::$items[$id] ?? []; | ||||
| 	} | ||||
|  | ||||
| 	public static function getDescription($id, $count = 1) { | ||||
| 	public static function getDescription($id, $count = 1): string | ||||
| 	{ | ||||
| 		$item = self::get($id); | ||||
|  | ||||
| 		$attr = $item['attributes']; | ||||
| @@ -112,15 +113,15 @@ class Items | ||||
| 			$s .= 'an item of type ' . $item['id']; | ||||
|  | ||||
| 		if(isset($attr['type']) && strtolower($attr['type']) == 'rune') { | ||||
| 			$item = Spell::where('item_id', $id)->first(); | ||||
| 			if($item) { | ||||
| 				if($item->level > 0 && $item->maglevel > 0) { | ||||
| 					$s .= '. ' . ($count > 1 ? "They" : "It") . ' can only be used by '; | ||||
| 			$spell = Spell::where('item_id', $id)->first(); | ||||
| 			if($spell) { | ||||
| 				if($spell->level > 0 && $spell->maglevel > 0) { | ||||
| 					$s .= '. ' . ($count > 1 ? 'They' : 'It') . ' can only be used by '; | ||||
| 				} | ||||
|  | ||||
| 				$configVocations = config('vocations'); | ||||
| 				if(!empty(trim($item->vocations))) { | ||||
| 					$vocations = json_decode($item->vocations); | ||||
| 				if(!empty(trim($spell->vocations))) { | ||||
| 					$vocations = json_decode($spell->vocations); | ||||
| 					if(count($vocations) > 0) { | ||||
| 						foreach($vocations as $voc => $show) { | ||||
| 							$vocations[$configVocations[$voc]] = $show; | ||||
| @@ -133,8 +134,39 @@ class Items | ||||
|  | ||||
| 				$s .= ' with'; | ||||
|  | ||||
| 				if ($spell->level > 0) { | ||||
| 					$s .= ' level ' . $spell->level; | ||||
| 				} | ||||
|  | ||||
| 				if ($spell->maglevel > 0) { | ||||
| 					if ($spell->level > 0) { | ||||
| 						$s .= ' and'; | ||||
| 					} | ||||
|  | ||||
| 					$s .= ' magic level ' . $spell->maglevel; | ||||
| 				} | ||||
|  | ||||
| 				$s .= ' or higher'; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (!empty($item['weaponType'])) { | ||||
| 			if ($item['weaponType'] == 'distance' && isset($item['ammoType'])) { | ||||
| 				$s .= ' (Range:' . $item['range']; | ||||
| 			} | ||||
|  | ||||
| 			if (isset($item['attack']) && $item['attack'] != 0) { | ||||
| 				$s .= ', Atk ' . ($item['attack'] > 0 ? '+' . $item['attack'] : '-' . $item['attack']); | ||||
| 			} | ||||
|  | ||||
| 			if (isset($item['hitChance']) && $item['hitChance'] != -1) { | ||||
| 				$s .= ', Hit% ' . ($item['hitChance'] > 0 ? '+' . $item['hitChance'] : '-' . $item['hitChance']); | ||||
| 			} | ||||
| 			elseif ($item['weaponType'] != 'ammo') { | ||||
| 				 | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return $s; | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -5,11 +5,15 @@ namespace MyAAC\Models; | ||||
| use Illuminate\Database\Eloquent\Model; | ||||
|  | ||||
| /** | ||||
|  * @property integer $premium_ends_at | ||||
|  * @property integer $premend | ||||
|  * @property integer $lastday | ||||
|  * @property integer $premdays | ||||
|  */ | ||||
| class Account extends Model { | ||||
|  | ||||
| 	const GRATIS_PREMIUM_DAYS = 65535; | ||||
|  | ||||
| 	protected $table = 'accounts'; | ||||
|  | ||||
| 	public $timestamps = false; | ||||
| @@ -33,32 +37,35 @@ class Account extends Model { | ||||
|  | ||||
| 	public function getPremiumDaysAttribute() | ||||
| 	{ | ||||
| 		if(isset($this->premium_ends_at) || isset($this->premend)) { | ||||
| 			$col = isset($this->premium_ends_at) ? 'premium_ends_at' : 'premend'; | ||||
| 			$ret = ceil(($this->{$col}- time()) / (24 * 60 * 60)); | ||||
| 			return $ret > 0 ? $ret : 0; | ||||
| 		if(isset($this->premium_ends_at) || isset($this->premend) || | ||||
| 			(isCanary() && isset($this->lastday))) { | ||||
| 				$col = (isset($this->premium_ends_at) ? 'premium_ends_at' : (isset($this->lastday) ? 'lastday' : 'premend')); | ||||
| 				$ret = ceil(($this->{$col} - time()) / (24 * 60 * 60)); | ||||
| 				return max($ret, 0); | ||||
| 		} | ||||
|  | ||||
| 		if($this->premdays == 0) { | ||||
| 			return 0; | ||||
| 		} | ||||
|  | ||||
| 		if($this->premdays == 65535){ | ||||
| 			return 65535; | ||||
| 		if($this->premdays == self::GRATIS_PREMIUM_DAYS){ | ||||
| 			return self::GRATIS_PREMIUM_DAYS; | ||||
| 		} | ||||
|  | ||||
| 		$ret = ceil($this->premdays - ((int)date("z", time()) + (365 * (date("Y", time()) - date("Y", $this->lastday))) - date("z", $this->lastday))); | ||||
| 		return max($ret, 0); | ||||
| 	} | ||||
|  | ||||
| 	public function getIsPremiumAttribute() | ||||
| 	public function getIsPremiumAttribute(): bool | ||||
| 	{ | ||||
| 		if(isset($this->premium_ends_at)) { | ||||
| 			return $this->premium_ends_at > time(); | ||||
| 		if(isset($this->premium_ends_at) || isset($this->premend) || | ||||
| 			(isCanary() && isset($this->lastday))) { | ||||
| 			$col = (isset($this->premium_ends_at) ? 'premium_ends_at' : (isset($this->lastday) ? 'lastday' : 'premend')); | ||||
| 			return $this->{$col} > time(); | ||||
| 		} | ||||
|  | ||||
| 		if(isset($this->premend)) { | ||||
| 			return $this->premend > time(); | ||||
| 		if($this->premdays == self::GRATIS_PREMIUM_DAYS){ | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		return ($this->premdays - (date("z", time()) + (365 * (date("Y", time()) - date("Y", $this->lastday))) - date("z", $this->lastday)) > 0); | ||||
|   | ||||
| @@ -9,6 +9,6 @@ class AccountAction extends Model { | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = ['account_id', 'ip', 'ipv6', 'date', 'action']; | ||||
| 	protected $fillable = ['account_id', 'ip', 'date', 'action']; | ||||
|  | ||||
| } | ||||
|   | ||||
| @@ -1,14 +0,0 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\Models; | ||||
| use Illuminate\Database\Eloquent\Model; | ||||
|  | ||||
| class AccountEMailCode extends Model { | ||||
|  | ||||
| 	protected $table = TABLE_PREFIX . 'account_email_codes'; | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = ['account_id', 'code', 'created_at']; | ||||
|  | ||||
| } | ||||
							
								
								
									
										15
									
								
								system/src/Models/AccountEmailVerify.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								system/src/Models/AccountEmailVerify.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\Models; | ||||
| use Illuminate\Database\Eloquent\Model; | ||||
|  | ||||
| class AccountEmailVerify extends Model | ||||
| { | ||||
|  | ||||
| 	protected $table = TABLE_PREFIX . 'account_emails_verify'; | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = ['account_id', 'hash', 'sent_at']; | ||||
|  | ||||
| } | ||||
| @@ -18,7 +18,16 @@ class Changelog extends Model { | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = [ | ||||
| 		'body', 'type', 'where', | ||||
| 		'date', 'player_id', 'hide', | ||||
| 	]; | ||||
|  | ||||
| 	public function scopeIsPublic($query) { | ||||
| 		$query->where('hide', '!=', 1); | ||||
| 	} | ||||
|  | ||||
| 	public function player() { | ||||
| 		return $this->belongsTo(Player::class); | ||||
| 	} | ||||
| } | ||||
|   | ||||
							
								
								
									
										16
									
								
								system/src/Models/ForumBoard.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								system/src/Models/ForumBoard.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\Models; | ||||
| use Illuminate\Database\Eloquent\Model; | ||||
|  | ||||
| class ForumBoard extends Model { | ||||
|  | ||||
| 	protected $table = TABLE_PREFIX . 'forum_boards'; | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = [ | ||||
| 		'name', 'description', 'ordering', | ||||
| 		'guild', 'access', 'closed', 'hide', | ||||
| 	]; | ||||
| } | ||||
| @@ -10,4 +10,9 @@ class Gallery extends Model { | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = [ | ||||
| 		'comment', 'image', 'thumb', | ||||
| 		'author', 'ordering', 'hide', | ||||
| 	]; | ||||
|  | ||||
| } | ||||
|   | ||||
							
								
								
									
										15
									
								
								system/src/Models/NewsCategory.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								system/src/Models/NewsCategory.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\Models; | ||||
| use Illuminate\Database\Eloquent\Model; | ||||
|  | ||||
| class NewsCategory extends Model { | ||||
|  | ||||
| 	protected $table = TABLE_PREFIX . 'news_categories'; | ||||
|  | ||||
| 	public $timestamps = false; | ||||
|  | ||||
| 	protected $fillable = [ | ||||
| 		'name', 'description', 'icon_id', 'hide' | ||||
| 	]; | ||||
| } | ||||
| @@ -1,13 +0,0 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\TwoFactorAuth\Gateway; | ||||
|  | ||||
| use MyAAC\TwoFactorAuth\Interface\AuthGatewayInterface; | ||||
|  | ||||
| class AppAuthGateway extends BaseAuthGateway implements AuthGatewayInterface | ||||
| { | ||||
| 	public function verifyCode(string $code): bool | ||||
| 	{ | ||||
| 		return true; | ||||
| 	} | ||||
| } | ||||
| @@ -1,12 +0,0 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\TwoFactorAuth\Gateway; | ||||
|  | ||||
| class BaseAuthGateway | ||||
| { | ||||
| 	protected \OTS_Account $account; | ||||
|  | ||||
| 	public function __construct(\OTS_Account $account) { | ||||
| 		$this->account = $account; | ||||
| 	} | ||||
| } | ||||
| @@ -1,16 +0,0 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\TwoFactorAuth\Gateway; | ||||
|  | ||||
| use MyAAC\Models\AccountEMailCode; | ||||
| use MyAAC\TwoFactorAuth\Interface\AuthGatewayInterface; | ||||
| use MyAAC\TwoFactorAuth\TwoFactorAuth; | ||||
|  | ||||
| class EmailAuthGateway extends BaseAuthGateway implements AuthGatewayInterface | ||||
| { | ||||
| 	public function verifyCode(string $code): bool | ||||
| 	{ | ||||
| 		return AccountEMailCode::where('account_id', '=', $this->account->getId())->where('code', $code)->where('created_at', '>', time() - TwoFactorAuth::EMAIL_CODE_VALID_UNTIL)->first() !== null; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -1,9 +0,0 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\TwoFactorAuth\Interface; | ||||
|  | ||||
| interface AuthGatewayInterface | ||||
| { | ||||
| 	public function __construct(\OTS_Account $account); | ||||
| 	public function verifyCode(string $code): bool; | ||||
| } | ||||
| @@ -1,183 +0,0 @@ | ||||
| <?php | ||||
|  | ||||
| namespace MyAAC\TwoFactorAuth; | ||||
|  | ||||
| use MyAAC\Models\AccountEMailCode; | ||||
| use MyAAC\TwoFactorAuth\Gateway\AppAuthGateway; | ||||
| use MyAAC\TwoFactorAuth\Gateway\EmailAuthGateway; | ||||
|  | ||||
| class TwoFactorAuth | ||||
| { | ||||
| 	const TYPE_NONE = 0; | ||||
| 	const TYPE_EMAIL = 1; | ||||
| 	const TYPE_APP = 2; | ||||
| 	// maybe later | ||||
| 	//const TYPE_SMS = 3; | ||||
|  | ||||
| 	const EMAIL_CODE_VALID_UNTIL = 24 * 60 * 60; | ||||
|  | ||||
| 	private static self $instance; | ||||
|  | ||||
| 	private \OTS_Account $account; | ||||
| 	private int $authType; | ||||
| 	private EmailAuthGateway|AppAuthGateway $authGateway; | ||||
|  | ||||
| 	public function __construct(\OTS_Account|int $account) { | ||||
| 		if (is_int($account)) { | ||||
| 			$this->account = new \OTS_Account(); | ||||
| 			$this->account->load($account); | ||||
| 		} | ||||
| 		else { | ||||
| 			$this->account = $account; | ||||
| 		} | ||||
|  | ||||
| 		$this->authType = (int)$this->account->getCustomField('2fa_type'); | ||||
| 		$this->setAuthGateway($this->authType); | ||||
| 	} | ||||
|  | ||||
| 	public static function getInstance($account = null): self | ||||
| 	{ | ||||
| 		if (!isset(self::$instance)) { | ||||
| 			self::$instance = new self($account); | ||||
| 		} | ||||
|  | ||||
| 		return self::$instance; | ||||
| 	} | ||||
|  | ||||
| 	public function process($login_account, $login_password, $remember_me, $code): bool | ||||
| 	{ | ||||
| 		global $twig; | ||||
|  | ||||
| 		if (!$this->isActive()) { | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		if (empty($code)) { | ||||
| 			if ($this->authType == self::TYPE_EMAIL) { | ||||
| 				if (!$this->hasRecentEmailCode(15 * 60)) { | ||||
| 					$this->resendEmailCode(); | ||||
| 					//success('Resent email.'); | ||||
| 				} | ||||
|  | ||||
| 				define('HIDE_LOGIN_BOX', true); | ||||
| 				$twig->display('account.2fa.email.login.html.twig', [ | ||||
| 					'account_login' => $login_account, | ||||
| 					'password_login' => $login_password, | ||||
| 					'remember_me' => $remember_me, | ||||
| 				]); | ||||
| 			} | ||||
| 			else { | ||||
| 				echo 'Two Factor App Auth'; | ||||
| 			} | ||||
|  | ||||
| 			return false; | ||||
| 		} | ||||
|  | ||||
| 		if ($this->getAuthGateway()->verifyCode($code)) { | ||||
| 			if ($this->authType === self::TYPE_EMAIL) { | ||||
| 				$this->deleteOldCodes(); | ||||
| 			} | ||||
|  | ||||
| 			return true; | ||||
| 		} | ||||
|  | ||||
| 		if (setting('core.mail_enabled')) { | ||||
| 			$mailBody = $twig->render('mail.account.2fa.email-code.wrong-attempt.html.twig'); | ||||
|  | ||||
| 			if (!_mail($this->account->getEMail(), configLua('serverName') . ' - Failed Two-Factor Authentication Attempt', $mailBody)) { | ||||
| 				error('An error occurred while sending email. For Admin: More info can be found in system/logs/mailer-error.log'); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		define('HIDE_LOGIN_BOX', true); | ||||
|  | ||||
| 		$errors[] = 'Invalid email code!'; | ||||
| 		$twig->display('error_box.html.twig', ['errors' => $errors]); | ||||
|  | ||||
| 		$twig->display('account.2fa.email.login.html.twig', | ||||
| 			[ | ||||
| 				'account_login' => $login_account, | ||||
| 				'password_login' => $login_password, | ||||
| 				'remember_me' => $remember_me, | ||||
|  | ||||
| 				'wrongCode' => true, | ||||
| 			]); | ||||
|  | ||||
| 		return false; | ||||
| 	} | ||||
|  | ||||
| 	public function setAuthGateway(int $authType): void | ||||
| 	{ | ||||
| 		if ($authType === self::TYPE_EMAIL) { | ||||
| 			$this->authGateway = new EmailAuthGateway($this->account); | ||||
| 		} | ||||
| 		else if ($authType === self::TYPE_APP) { | ||||
| 			$this->authGateway = new AppAuthGateway($this->account); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	public function getAccountManageViews(): array | ||||
| 	{ | ||||
| 		$twoFactorView = 'account.2fa.protected.html.twig'; | ||||
| 		if ($this->authType == self::TYPE_EMAIL) { | ||||
| 			$twoFactorView2 = 'account.2fa.email.activated.html.twig'; | ||||
| 		} | ||||
| 		elseif ($this->authType == self::TYPE_APP) { | ||||
| 			$twoFactorView2 = 'account.2fa.app.activated.html.twig'; | ||||
| 		} | ||||
| 		else { | ||||
| 			$twoFactorView = 'account.2fa.connect.html.twig'; | ||||
| 			$twoFactorView2 = 'account.2fa.email.activate.html.twig'; | ||||
| 		} | ||||
|  | ||||
| 		return [$twoFactorView, $twoFactorView2]; | ||||
| 	} | ||||
|  | ||||
| 	public function enable(int $type): void { | ||||
| 		$this->account->setCustomField('2fa_type', $type); | ||||
| 	} | ||||
|  | ||||
| 	public function disable(): void { | ||||
| 		$this->account->setCustomField('2fa_type', self::TYPE_NONE); | ||||
| 	} | ||||
|  | ||||
| 	public function isActive(): bool { | ||||
| 		return $this->authType != self::TYPE_NONE; | ||||
| 	} | ||||
|  | ||||
| 	public function getAuthType(): int { | ||||
| 		return $this->authType; | ||||
| 	} | ||||
|  | ||||
| 	public function getAuthGateway(): AppAuthGateway|EmailAuthGateway  { | ||||
| 		return $this->authGateway; | ||||
| 	} | ||||
|  | ||||
| 	public function hasRecentEmailCode($since = self::EMAIL_CODE_VALID_UNTIL): bool { | ||||
| 		return AccountEMailCode::where('account_id', '=', $this->account->getId())->where('created_at', '>', time() - $since)->first() !== null; | ||||
| 	} | ||||
|  | ||||
| 	public function deleteOldCodes(): void { | ||||
| 		AccountEMailCode::where('account_id', '=', $this->account->getId())->delete(); | ||||
| 	} | ||||
|  | ||||
| 	public function resendEmailCode(): void | ||||
| 	{ | ||||
| 		global $twig; | ||||
|  | ||||
| 		$newCode = generateRandomString(6, true, false, true); | ||||
| 		AccountEMailCode::create([ | ||||
| 			'account_id' => $this->account->getId(), | ||||
| 			'code' => $newCode, | ||||
| 			'created_at' => time(), | ||||
| 		]); | ||||
|  | ||||
| 		$mailBody = $twig->render('mail.account.2fa.email-code.html.twig', [ | ||||
| 			'code' => $newCode, | ||||
| 		]); | ||||
|  | ||||
| 		if (!_mail($this->account->getEMail(), configLua('serverName') . ' - Requested Authentication Email Code', $mailBody)) { | ||||
| 			error('An error occurred while sending email. For Admin: More info can be found in system/logs/mailer-error.log'); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -28,6 +28,8 @@ define('HOOK_CHARACTERS_AFTER_CHARACTERS', ++$i); | ||||
| define('HOOK_LOGIN', ++$i); | ||||
| define('HOOK_LOGIN_ATTEMPT', ++$i); | ||||
| define('HOOK_LOGOUT', ++$i); | ||||
| define('HOOK_ACCOUNT_CHANGE_PASSWORD_AFTER_OLD_PASSWORD', ++$i); | ||||
| define('HOOK_ACCOUNT_CHANGE_PASSWORD_AFTER_NEW_PASSWORD', ++$i); | ||||
| define('HOOK_ACCOUNT_CHANGE_PASSWORD_POST', ++$i); | ||||
| define('HOOK_ACCOUNT_CREATE_BEFORE_FORM', ++$i); | ||||
| define('HOOK_ACCOUNT_CREATE_BEFORE_BOXES', ++$i); | ||||
|   | ||||
| @@ -1,36 +0,0 @@ | ||||
| <tr> | ||||
| 	<td> | ||||
| 		<div class="TableShadowContainerRightTop"> | ||||
| 			<div class="TableShadowRightTop" style="background-image:url({{ template_path }}/images/global/content/table-shadow-rt.gif);"></div> | ||||
| 		</div> | ||||
| 		<div class="TableContentAndRightShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-rm.gif);"> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody><tr> | ||||
| 						<td class="LabelV"><b>Connect your {{ config.lua.serverName }} account to an authenticator app!</b> | ||||
| 							<div style="float: right; font-size: 1px;"> | ||||
| 								<form action="{{ getLink('account/2fa') }}?action=email-code" method="post" style="margin: 0px; padding: 0px;"> | ||||
| 									{{ csrf() }} | ||||
| 									{% set button_name = 'Request' %} | ||||
| 									{% include('buttons.base.html.twig') %} | ||||
| 								</form> | ||||
| 							</div> | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					<tr> | ||||
| 						<td> | ||||
| 							<p>As a first step to connect an <b>authenticator app</b> to your account, click on "Request"! An email with a confirmation key will be sent to the email address assigned to your account.</p> | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="TableShadowContainer"> | ||||
| 			<div class="TableBottomShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-bm.gif);"> | ||||
| 				<div class="TableBottomLeftShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-bl.gif);"></div> | ||||
| 				<div class="TableBottomRightShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-br.gif);"></div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</td> | ||||
| </tr> | ||||
| @@ -1,37 +0,0 @@ | ||||
| <tr> | ||||
| 	<td> | ||||
| 		<div class="TableShadowContainerRightTop"> | ||||
| 			<div class="TableShadowRightTop" style="background-image:url({{ template_path }}/images/global/content/table-shadow-rt.gif);"></div> | ||||
| 		</div> | ||||
| 		<div class="TableContentAndRightShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-rm.gif);"> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody> | ||||
| 					<tr> | ||||
| 						<td class="LabelV"><b>Activate email code authentication for your account!</b> | ||||
| 							<div style="float: right; font-size: 1px;"> | ||||
| 								<form action="{{ getLink('account/2fa') }}?action=email-code&step=activate" method="post" style="margin: 0; padding: 0;"> | ||||
| 									{{ csrf() }} | ||||
| 									{% set button_name = 'Request' %} | ||||
| 									{% include('buttons.base.html.twig') %} | ||||
| 								</form> | ||||
| 							</div> | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					<tr> | ||||
| 						<td> | ||||
| 							<p>As a first step to activate <b>email code authentication</b> for your account, click on "Request"! An <b>email code</b> will be sent to the email address assigned to your account. You will be asked to enter this <b>email code</b> on the next page within 24 hours.</p> | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 		<div class="TableShadowContainer"> | ||||
| 			<div class="TableBottomShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-bm.gif);"> | ||||
| 				<div class="TableBottomLeftShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-bl.gif);"></div> | ||||
| 				<div class="TableBottomRightShadow" style="background-image:url({{ template_path }}/images/global/content/table-shadow-br.gif);"></div> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</td> | ||||
| </tr> | ||||
| @@ -1,26 +0,0 @@ | ||||
| <tr> | ||||
| 	<td> | ||||
| 		<div class="TableContentContainer "> | ||||
| 			<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 				<tbody> | ||||
| 				<tr> | ||||
| 					<td> | ||||
| 						<div style="float: right; width: 135px;"> | ||||
| 							<form action="{{ getLink('account/2fa') }}?action=email-code" method="post" style="padding:0;margin:0;"> | ||||
| 								{{ csrf() }} | ||||
| 								<input type="hidden" name="step" value="deactivate"> | ||||
| 								{% set button_name = 'Deactivate' %} | ||||
| 								{{ include('buttons.base.html.twig') }} | ||||
| 							</form> | ||||
| 						</div> | ||||
| 						<b>Two-Factor Email Code Authentication <span style="color: green">Activated</span>!</b> | ||||
| 						<p>To deactivate <b>email code authentication</b>, click on the "Deactivate" button.</p> | ||||
| 						<!--p>You will have to confirm the deactivation by entering an <b>email code</b> which will be sent | ||||
| 							to the email address assigned to your account.</p--> | ||||
| 					</td> | ||||
| 				</tr> | ||||
| 				</tbody> | ||||
| 			</table> | ||||
| 		</div> | ||||
| 	</td> | ||||
| </tr> | ||||
| @@ -1,109 +0,0 @@ | ||||
| {% set title = 'Deactivate Email Code Authentication' %} | ||||
| {% set content %} | ||||
| 	<table style="width:100%;"> | ||||
| 		<tbody> | ||||
| 		<tr> | ||||
| 			<td> | ||||
| 				<div class="TableContentContainer "> | ||||
| 					<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 						<tbody> | ||||
| 						<tr> | ||||
| 							<td>To deactivate <b>two-factor email code authentication</b> for your account, enter the | ||||
| 								received <b>email code</b> below. Note, however, that <b>email code authentication</b> | ||||
| 								is an important security feature which helps to prevent any unauthorised access to your | ||||
| 								Tibia account. | ||||
| 							</td> | ||||
| 						</tr> | ||||
| 						</tbody> | ||||
| 					</table> | ||||
| 				</div> | ||||
| 			</td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td> | ||||
| 				<div class="TableContentContainer"> | ||||
| 					<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 						<tbody> | ||||
| 						<tr> | ||||
| 							<td> | ||||
| 								<div style="float: right;"> | ||||
| 									<form | ||||
| 										action="{{ getLink('account/2fa') }}?action=email-code&step=resend" | ||||
| 										method="post" | ||||
| 										style="padding:0;margin:0;" | ||||
| 									> | ||||
| 										{{ csrf() }} | ||||
|  | ||||
| 										{% set button_name = 'Resend Email Code' %} | ||||
| 										{{ include('buttons.base.html.twig') }} | ||||
| 									</form> | ||||
| 								</div> | ||||
| 								An <b>email code</b> has already been sent to the email address assigned to your | ||||
| 								account. | ||||
| 								Please check your email account's spam/junk filter and make sure that your mailbox is | ||||
| 								not | ||||
| 								full.<br>In case you need a new email code, you can request one by clicking on "Resend | ||||
| 								Email | ||||
| 								Code". | ||||
| 							</td> | ||||
| 						</tr> | ||||
| 						</tbody> | ||||
| 					</table> | ||||
| 				</div> | ||||
| 			</td> | ||||
| 		</tr> | ||||
| 		<tr> | ||||
| 			<td> | ||||
| 				<div class="TableContentContainer"> | ||||
| 					<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 						<tbody> | ||||
| 						<tr> | ||||
| 							<td>To complete the deactivation of <b>email code authentication</b>, please enter the <b>email | ||||
| 									code</b> you received at the email address assigned to your account. | ||||
| 								<div style="margin-top: 15px; margin-bottom: 15px;"> | ||||
| 									<div class="LabelV150 {{ wrongCode ? 'red' : '' }}" style="float:left;"><label | ||||
| 											for="email-code">Email Code:</label></div> | ||||
| 									<input form="form-code" id="auth-code" name="email-code" maxlength="15" | ||||
| 										   autocomplete="off"> | ||||
| 									{% if wrongCode %} | ||||
| 										<br/> | ||||
| 										<div class="LabelV150" style="float:left;">  </div> | ||||
| 										<div class="FormFieldError">Invalid email code!</div> | ||||
| 									{% endif %} | ||||
| 								</div> | ||||
| 							</td> | ||||
| 						</tr> | ||||
| 						</tbody> | ||||
| 					</table> | ||||
| 				</div> | ||||
| 			</td> | ||||
| 		</tr> | ||||
| 		</tbody> | ||||
| 	</table> | ||||
| {% endset %} | ||||
| {% include 'tables.headline.html.twig' %} | ||||
| <table style="width: 100%;"> | ||||
| 	<tbody> | ||||
| 	<tr align="center" valign="top"> | ||||
| 		<td> | ||||
| 			<form id="form-code" method="post" action="{{ getLink('account/2fa') }}?action=email-code"> | ||||
| 				{{ csrf() }} | ||||
|  | ||||
| 				<input type="hidden" name="step" value="deactivate"> | ||||
| 				<input type="hidden" name="save" value="1"> | ||||
|  | ||||
| 				{% set button_name = 'Continue' %} | ||||
| 				{% set button_color = 'green' %} | ||||
| 				{{ include('buttons.submit.html.twig') }} | ||||
| 			</form> | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<form action="{{ getLink('account/manage') }}" method="post" style="padding:0;margin:0;"> | ||||
| 				{{ csrf() }} | ||||
| 				{% set button_color = 'blue' %} | ||||
| 				{{ include('buttons.back.html.twig') }} | ||||
| 			</form> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	</tbody> | ||||
| </table> | ||||
| @@ -1,92 +0,0 @@ | ||||
| {% set title = 'Enter Email Code' %} | ||||
| {% set content %} | ||||
| <table style="width:100%;"> | ||||
| 	<tbody> | ||||
| 	<tr> | ||||
| 		<td> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody> | ||||
| 					<tr> | ||||
| 						<td> | ||||
| 							<div style="float: right;"> | ||||
| 								<form | ||||
| 										action="{{ getLink('account/2fa') }}?action=email-code&step=resend" | ||||
| 										method="post" | ||||
| 										style="padding:0;margin:0;" | ||||
| 								> | ||||
| 									{{ csrf() }} | ||||
|  | ||||
| 									{% set button_name = 'Resend Email Code' %} | ||||
| 									{{ include('buttons.base.html.twig') }} | ||||
| 								</form> | ||||
| 							</div> | ||||
| 							An <b>email code</b> has already been sent to the email address assigned to your account. | ||||
| 							Please check your email account's spam/junk filter and make sure that your mailbox is not | ||||
| 							full.<br>In case you need a new email code, you can request one by clicking on "Resend Email | ||||
| 							Code". | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	<tr> | ||||
| 		<td> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody> | ||||
| 					<tr> | ||||
| 						<td><b>Email code authentication is activated for your account.</b><br><br>Please enter the <b>most | ||||
| 								recent email code</b> you have received in order to log in.<br> | ||||
| 							<div style="margin-top: 15px; margin-bottom: 15px;"> | ||||
| 								<div class="LabelV150 {{ wrongCode ? 'red' : '' }}" style="float:left;"><label for="email-code">Email Code:</label></div> | ||||
| 								<input form="form-code" id="auth-code" name="auth-code" maxlength="15" autocomplete="off"> | ||||
| 								{% if wrongCode %} | ||||
| 									<br/> | ||||
| 									<div class="LabelV150" style="float:left;">  </div> | ||||
| 									<div class="FormFieldError">Invalid email code!</div> | ||||
| 								{% endif %} | ||||
| 							</div> | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	</tbody> | ||||
| </table> | ||||
| {% endset %} | ||||
| {% include 'tables.headline.html.twig' %} | ||||
| <table style="width: 100%;"> | ||||
| 	<tbody> | ||||
| 	<tr align="center" valign="top"> | ||||
| 		<td> | ||||
| 			<form id="form-code" method="post" action="{{ getLink('account/manage') }}"> | ||||
| 				{{ csrf() }} | ||||
|  | ||||
| 				<input type="hidden" name="account_login" value="{{ account_login ?? '' }}" /> | ||||
| 				<input type="hidden" name="password_login" value="{{ password_login ?? '' }}" /> | ||||
| 				{% if remember_me %} | ||||
| 					<input type="hidden" name="remember_me" value="true" /> | ||||
| 				{% endif %} | ||||
|  | ||||
| 				<input type="hidden" name="step" value="verify"> | ||||
| 				{% set button_name = 'Continue' %} | ||||
| 				{% set button_color = 'green' %} | ||||
| 				{{ include('buttons.submit.html.twig') }} | ||||
| 			</form> | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<form action="{{ getLink('account/manage') }}" method="post" style="padding:0;margin:0;"> | ||||
| 				{{ csrf() }} | ||||
|  | ||||
| 				{% set button_color = 'blue' %} | ||||
| 				{{ include('buttons.back.html.twig') }} | ||||
| 			</form> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	</tbody> | ||||
| </table> | ||||
| @@ -1,110 +0,0 @@ | ||||
| {% set title = 'Activate Email Code Authentication' %} | ||||
|  | ||||
| {% set content %} | ||||
| <table style="width:100%;"> | ||||
| 	<tbody> | ||||
| 	<tr> | ||||
| 		<td> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody> | ||||
| 					<tr> | ||||
| 						<td>Enter the email code below to activate <b>two-factor email code authentication</b>. Note | ||||
| 							that this code is only valid for 24 hours.<br><br> | ||||
| 							<div class="AttentionSign"><img src="{{ template_path }}/images/global/content/attentionsign.gif"></div> | ||||
| 							<b>Note:</b> Once you have email code authentication activated, an <b>email code</b> will be | ||||
| 							sent to the email address assigned to your account whenever you try to log in to the Tibia | ||||
| 							client or the {{ config.lua.serverName }} website. In order to log in, you will need to enter the <b>most recent | ||||
| 								email code</b> you have received. | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	<tr> | ||||
| 		<td> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody> | ||||
| 					<tr> | ||||
| 						<td> | ||||
| 							<div style="float: right;"> | ||||
| 								<form action="{{ getLink('account/2fa') }}?action=email-code" | ||||
| 									method="post" style="padding:0;margin:0;"> | ||||
| 									{{ csrf() }} | ||||
|  | ||||
| 									{% if account_logged is defined %} | ||||
| 										<input type="hidden" name="account_logged" value="{{ account_logged.getId() }}"> | ||||
| 									{% endif %} | ||||
| 									<input type="hidden" name="step" value="resend"> | ||||
|  | ||||
| 									{% set button_name = 'Resend Email Code' %} | ||||
| 									{% include('buttons.base.html.twig') %} | ||||
| 								</form> | ||||
| 							</div> | ||||
| 							An <b>email code</b> has already been sent to the email address assigned to your account. | ||||
| 							Please check your email account's spam/junk filter and make sure that your mailbox is not | ||||
| 							full.<br>In case you need a new email code, you can request one by clicking on "Resend Email | ||||
| 							Code". | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	<tr> | ||||
| 		<td> | ||||
| 			<div class="TableContentContainer"> | ||||
| 				<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 					<tbody> | ||||
| 					<tr> | ||||
| 						<td>To complete the activation of email code authentication for your Tibia account, please enter | ||||
| 							the email code you received at the email address assigned to your account. | ||||
| 							<div style="margin-top: 15px; margin-bottom: 15px;"> | ||||
| 								<div class="LabelV150 {{ wrongCode ? 'red' : '' }}" style="float:left;">Email Code:</div> | ||||
| 								<input form="confirmActivateForm" name="auth-code" maxlength="6"> | ||||
| 								{% if wrongCode %} | ||||
| 									<br/> | ||||
| 									<div class="LabelV150" style="float:left;">  </div> | ||||
| 									<div class="FormFieldError">Invalid email code!</div> | ||||
| 								{% endif %} | ||||
| 							</div> | ||||
| 						</td> | ||||
| 					</tr> | ||||
| 					</tbody> | ||||
| 				</table> | ||||
| 			</div> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	</tbody> | ||||
| </table> | ||||
| {% endset %} | ||||
| {% include 'tables.headline.html.twig' %} | ||||
| <br/> | ||||
| <table style="width: 100%;"> | ||||
| 	<tbody> | ||||
| 	<tr align="center" valign="top"> | ||||
| 		<td> | ||||
| 			<form id="confirmActivateForm" action="{{ getLink('account/2fa') }}?action=email-code" method="post" style="padding:0;margin:0;"> | ||||
| 				{{ csrf() }} | ||||
|  | ||||
| 				<input type="hidden" name="step" value="activate"> | ||||
| 				<input type="hidden" name="save" value="1"> | ||||
|  | ||||
| 				{% set button_color = 'green' %} | ||||
| 				{{ include('buttons.submit.html.twig') }} | ||||
| 			</form> | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<form action="{{ getLink('account/manage') }}" method="post" style="padding:0;margin:0;"> | ||||
| 				{{ csrf() }} | ||||
| 				{% set button_color = 'blue' %} | ||||
| 				{{ include('buttons.back.html.twig') }} | ||||
| 			</form> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| 	</tbody> | ||||
| </table> | ||||
| @@ -1,12 +0,0 @@ | ||||
| {% set title = 'Two-Factor Authentication' %} | ||||
|  | ||||
| {% set content %} | ||||
| <table style="width:100%;"> | ||||
| 	<tbody> | ||||
| 	{{ include(twoFactorViews[0]) }} | ||||
| 	{{ include(twoFactorViews[1]) }} | ||||
| 	</tbody> | ||||
| </table> | ||||
| {% endset %} | ||||
| {% include('tables.headline.html.twig') %} | ||||
| <br/> | ||||
| @@ -1,18 +0,0 @@ | ||||
| <tr> | ||||
| 	<td> | ||||
| 		<div class="TableContentContainer "> | ||||
| 			<table class="TableContent" width="100%" style="border:1px solid #faf0d7;"> | ||||
| 				<tbody> | ||||
| 				<tr> | ||||
| 					<td> | ||||
| 						<div class="InTableRightButtonContainer"></div> | ||||
| 						<b>Two-Factor Authenticator App</b> | ||||
| 						<p>Your account is currently protected by email code authentication. If you prefer to use a <b>two-factor | ||||
| 								authentication app</b>, you have to "Deactivate" email code authentication first.</p> | ||||
| 					</td> | ||||
| 				</tr> | ||||
| 				</tbody> | ||||
| 			</table> | ||||
| 		</div> | ||||
| 	</td> | ||||
| </tr> | ||||
| @@ -9,23 +9,29 @@ Please enter your current password and a new password. For your security, please | ||||
| 			<span>Current Password:</span> | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<input form="form" type="password" name="oldpassword" size="30" maxlength="29"> | ||||
| 			<input form="form" type="password" id="old_password" name="old_password" size="30" maxlength="29"> | ||||
| 		</td> | ||||
| 	</tr> | ||||
|  | ||||
| 	{{ hook('HOOK_ACCOUNT_CHANGE_PASSWORD_AFTER_OLD_PASSWORD') }} | ||||
|  | ||||
| 	<tr> | ||||
| 		<td class="LabelV"> | ||||
| 			<span>New Password:</span> | ||||
| 		</td> | ||||
| 		<td style="width:90%;"> | ||||
| 			<input form="form" type="password" name="newpassword" size="30" maxlength="29"> | ||||
| 			<input form="form" type="password" id="new_password" name="new_password" size="30" maxlength="29"> | ||||
| 		</td> | ||||
| 	</tr> | ||||
|  | ||||
| 	{{ hook('HOOK_ACCOUNT_CHANGE_PASSWORD_AFTER_NEW_PASSWORD') }} | ||||
|  | ||||
| 	<tr> | ||||
| 		<td class="LabelV"> | ||||
| 			<span>New Password Again:</span> | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<input form="form" type="password" name="newpassword_confirm" size="30" maxlength="29"> | ||||
| 			<input form="form" type="password" id="new_password_confirm" name="new_password_confirm" size="30" maxlength="29"> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| </table> | ||||
|   | ||||
| @@ -147,9 +147,6 @@ | ||||
| 			{% include('buttons.base.html.twig') %} | ||||
| 		</form> | ||||
| 		<br/> | ||||
|  | ||||
| 		{{ include('account.2fa.main.html.twig') }} | ||||
|  | ||||
| 		{{ hook('HOOK_ACCOUNT_MANAGE_BEFORE_ACCOUNT_LOGS') }} | ||||
| 		<a name="Account+Logs" ></a> | ||||
| 		<h2>Account Logs</h2> | ||||
|   | ||||
							
								
								
									
										45
									
								
								system/templates/account.resend-email-verify.html.twig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								system/templates/account.resend-email-verify.html.twig
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| Please enter your account Email address.<br/><br/> | ||||
| {% set title = 'Resend Email' %} | ||||
| {% set background = config('darkborder') %} | ||||
| {% set content %} | ||||
| 	<table style="width:100%;"> | ||||
| 		<tr> | ||||
| 			<td class="LabelV" > | ||||
| 				<span><label for="email">Email Address:</label></span> | ||||
| 			</td> | ||||
| 			<td style="width:90%;"> | ||||
| 				<input type="email" form="form" id="email" name="email" size="30" maxlength="50" autofocus/> | ||||
| 			</td> | ||||
| 		</tr> | ||||
| 	</table> | ||||
| {% endset %} | ||||
| {% include 'tables.headline.html.twig' %} | ||||
| <br/> | ||||
| <table style="width:100%;"> | ||||
| 	<tr align="center"> | ||||
| 		<td> | ||||
| 			<table border="0" cellspacing="0" cellpadding="0"> | ||||
| 				<tr> | ||||
| 					<td style="border:0;"> | ||||
| 						<form id="form" action="{{ getLink('account/resend-email-verify') }}" method="post"> | ||||
| 							{{ csrf() }} | ||||
| 							<input type="hidden" name="submit" value="1"/> | ||||
| 							{{ include('buttons.submit.html.twig') }} | ||||
| 						</form> | ||||
| 					</td> | ||||
| 				<tr> | ||||
| 			</table> | ||||
| 		</td> | ||||
| 		<td> | ||||
| 			<table border="0" cellspacing="0" cellpadding="0"> | ||||
| 				<tr> | ||||
| 					<td style="border:0;"> | ||||
| 						<form action="{{ getLink('news') }}" method="post"> | ||||
| 							{{ include('buttons.back.html.twig') }} | ||||
| 						</form> | ||||
| 					</td> | ||||
| 				</tr> | ||||
| 			</table> | ||||
| 		</td> | ||||
| 	</tr> | ||||
| </table> | ||||
| @@ -9,7 +9,7 @@ | ||||
| <table border="0" cellpadding="0" cellspacing="0" width="100%"><tr> | ||||
| 	<td><img src="{{ template_path }}/images/general/blank.gif" width="10" height="1" border="0"></td> | ||||
| 	<td> | ||||
|         {{ hook(constant('HOOK_CHARACTERS_BEFORE_INFORMATIONS')) }} | ||||
|         {{ hook('HOOK_CHARACTERS_BEFORE_INFORMATIONS') }} | ||||
| 		{% if canEdit %} | ||||
| 			<a href="{{ constant('ADMIN_URL') }}?p=players&id={{ player.getId() }}" title="Edit in Admin Panel" target="_blank"> | ||||
| 				<img src="images/edit.png"/>Edit | ||||
| @@ -153,11 +153,11 @@ | ||||
| 				<td>{% if account.isPremium() %}Premium Account{% else %}Free Account{% endif %}</td> | ||||
| 			</tr> | ||||
| 		</table> | ||||
| 		{{ hook(constant('HOOK_CHARACTERS_AFTER_INFORMATIONS')) }} | ||||
| 		{{ hook('HOOK_CHARACTERS_AFTER_INFORMATIONS') }} | ||||
| 		<br/> | ||||
| 		<table border="0" width="100%"> | ||||
| 			<tr> | ||||
| 				{{ hook(constant('HOOK_CHARACTERS_BEFORE_SKILLS')) }} | ||||
| 				{{ hook('HOOK_CHARACTERS_BEFORE_SKILLS') }} | ||||
|  | ||||
| 				{% if config.characters.skills %} | ||||
| 				<!-- SKILLS --> | ||||
| @@ -179,7 +179,7 @@ | ||||
| 				<!-- SKILLS_END --> | ||||
| 				{% endif %} | ||||
|  | ||||
| 				{{ hook(constant('HOOK_CHARACTERS_AFTER_SKILLS')) }} | ||||
| 				{{ hook('HOOK_CHARACTERS_AFTER_SKILLS') }} | ||||
|  | ||||
| 				{% if quests_enabled %} | ||||
| 				<!-- QUESTS --> | ||||
| @@ -201,7 +201,7 @@ | ||||
| 				<!-- QUESTS_END --> | ||||
| 				{% endif %} | ||||
|  | ||||
| 				{{ hook(constant('HOOK_CHARACTERS_AFTER_QUESTS')) }} | ||||
| 				{{ hook('HOOK_CHARACTERS_AFTER_QUESTS') }} | ||||
|  | ||||
| 				{% if config.characters.equipment %} | ||||
| 				<!-- EQUIPMENT --> | ||||
| @@ -239,11 +239,11 @@ | ||||
| 				<!-- EQUIPMENT_END --> | ||||
| 				{% endif %} | ||||
|  | ||||
| 				{{ hook(constant('HOOK_CHARACTERS_AFTER_EQUIPMENT')) }} | ||||
| 				{{ hook('HOOK_CHARACTERS_AFTER_EQUIPMENT') }} | ||||
| 			</tr> | ||||
| 		</table> | ||||
|  | ||||
| 		{{ hook(constant('HOOK_CHARACTERS_BEFORE_DEATHS')) }} | ||||
| 		{{ hook('HOOK_CHARACTERS_BEFORE_DEATHS') }} | ||||
|  | ||||
| 		{% if deaths|length > 0 %} | ||||
| 		<!-- DEATHS --> | ||||
| @@ -283,7 +283,7 @@ | ||||
| 		<!-- FRAGS_END --> | ||||
| 		{% endif %} | ||||
|  | ||||
| 		{{ hook(constant('HOOK_CHARACTERS_BEFORE_SIGNATURE')) }} | ||||
| 		{{ hook('HOOK_CHARACTERS_BEFORE_SIGNATURE') }} | ||||
|  | ||||
| 		{% if setting('core.signature_enabled') %} | ||||
| 		<!-- SIGNATURE --> | ||||
| @@ -327,7 +327,7 @@ | ||||
| 		</table> | ||||
| 		<!-- SIGNATURE_END --> | ||||
| 		{% endif %} | ||||
| 		{{ hook(constant('HOOK_CHARACTERS_AFTER_SIGNATURE')) }} | ||||
| 		{{ hook('HOOK_CHARACTERS_AFTER_SIGNATURE') }} | ||||
| 		{% if not player.isHidden() %} | ||||
| 		{% set rows = 0 %} | ||||
| 		<!-- ACCOUNT_INFORMATION --> | ||||
| @@ -377,7 +377,7 @@ | ||||
| 			</tr> | ||||
| 		</table> | ||||
| 		<!-- ACCOUNT_INFORMATION_END --> | ||||
| 		{{ hook(constant('HOOK_CHARACTERS_AFTER_ACCOUNT')) }} | ||||
| 		{{ hook('HOOK_CHARACTERS_AFTER_ACCOUNT') }} | ||||
| 		<!-- CHARACTERS_LIST --> | ||||
| 		<br/><br/> | ||||
| 		<table border="0" cellspacing="1" cellpadding="4" width="100%"> | ||||
| @@ -421,7 +421,7 @@ | ||||
| 		</table> | ||||
| 		<!-- CHARACTERS_LIST_END --> | ||||
| 		{% endif %} | ||||
| 		{{ hook(constant('HOOK_CHARACTERS_AFTER_CHARACTERS')) }} | ||||
| 		{{ hook('HOOK_CHARACTERS_AFTER_CHARACTERS') }} | ||||
| 		{% if canEdit %} | ||||
| 			<a href="{{ constant('ADMIN_URL') }}?p=players&id={{ player.getId() }}" title="Edit in Admin Panel" target="_blank"> | ||||
| 				<img src="images/edit.png"/>Edit | ||||
|   | ||||
| @@ -9,7 +9,7 @@ | ||||
| 			<div class="AttentionSign" style="background-image:url({{ template_path }}/images/content/attentionsign.gif);"></div> | ||||
| 			<b>The Following Errors Have Occurred:</b><br/> | ||||
| 			{% for error in errors %} | ||||
| 			<li>{{ error|striptags('<b>')|raw }}</li> | ||||
| 			<li>{{ error|striptags('<b><a>')|raw }}</li> | ||||
| 			{% endfor %} | ||||
| 		</div> | ||||
| 		<div class="BoxFrameHorizontal" style="background-image:url({{ template_path }}/images/content/box-frame-horizontal.gif);"></div> | ||||
|   | ||||
| @@ -1,9 +0,0 @@ | ||||
| Dear {{ config.lua.serverName}} player, | ||||
| <br/><br/> | ||||
| Your account is protected by email code authentication, and you requested a new email code: | ||||
| <br/><br/> | ||||
| <p>{{ code }}</p> | ||||
| <br/> | ||||
| Note that the code is only valid for 24 hours. | ||||
| <br/><br/> | ||||
| Kind Regards, | ||||
| @@ -1,5 +0,0 @@ | ||||
| Dear {{ config.lua.serverName}} player,<br/> | ||||
| <br/> | ||||
| A <strong>wrong two-factor authentication code</strong> was entered for your {{ config.lua.serverName}} account. If you simply mistyped the code, please try again.<br/> | ||||
| <br/> | ||||
| However, if this was <strong>not you</strong>, someone else may be trying to access your account. Since they already know your password, we strongly recommend that you <strong>change your password immediately</strong>. | ||||
| @@ -0,0 +1,7 @@ | ||||
| Hello {{ account }}!<br/> | ||||
| <br/> | ||||
| You requested to resend the verify Email on {{ config.lua.serverName }}!<br/> | ||||
| <br/> | ||||
|  | ||||
| To verify your email address please click the link below:<br/> | ||||
| {{ verify_url|raw }} | ||||
| @@ -101,7 +101,7 @@ | ||||
|  | ||||
| 	<tr> | ||||
| 		<td class="LabelV150"><b>Location Datacenter:</b></td> | ||||
| 		<td>{{ setting('core.online_datacenter') }} <small>(Server date & time: - {{ "now"|date("d/m/Y H:i:s") }})</small></td> | ||||
| 		<td>{{ setting('core.online_datacenter')|raw }} <small>(Server date & time: - {{ "now"|date("d/m/Y H:i:s") }})</small></td> | ||||
| 	</tr> | ||||
| 	<tr> | ||||
| 		<td class="LabelV150"><b>PvP Type:</b></td> | ||||
|   | ||||
| @@ -1,6 +1,9 @@ | ||||
| <style> | ||||
| 	.myaac-table tbody tr:nth-child(even) {background: {{ config.lightborder }}} | ||||
| 	.myaac-table tbody tr:nth-child(odd) {background:  {{ config.darkborder }}} | ||||
| 	.myaac-table thead td {background: {{ config.vdarkborder }}; color: #ffffff !important;} | ||||
| 	.myaac-table tfoot td {background: {{ config.vdarkborder }}; color: #ffffff !important;} | ||||
| 	.myaac-table {border-spacing: 1px;} | ||||
| 	.myaac-table > tbody > tr:nth-child(even) {background: {{ config.lightborder }}} | ||||
| 	.myaac-table > tbody > tr:nth-child(odd) {background:  {{ config.darkborder }}} | ||||
| 	.myaac-table > tbody > tr > td {padding: 4px; } | ||||
| 	.myaac-table > thead > tr > td {padding: 4px; background: {{ config.vdarkborder }}; color: #ffffff !important;} | ||||
| 	.myaac-table > thead > tr > th {padding: 4px; background: {{ config.vdarkborder }}; color: #ffffff !important;} | ||||
| 	.myaac-table > tfoot > tr > td {padding: 4px; background: {{ config.vdarkborder }}; color: #ffffff !important;} | ||||
| </style> | ||||
|   | ||||
| @@ -101,6 +101,10 @@ $twig->addFunction($function); | ||||
| $function = new TwigFunction('hook', function ($context, $hook, array $params = []) { | ||||
| 	global $hooks; | ||||
|  | ||||
| 	if (config('hooks_debug')) { | ||||
| 		note($hook); | ||||
| 	} | ||||
|  | ||||
| 	if(is_string($hook)) { | ||||
| 		if (defined($hook)) { | ||||
| 			$hook = constant($hook); | ||||
|   | ||||
| @@ -130,6 +130,7 @@ | ||||
| 													<div style="float: right; margin-top: 20px;" > | ||||
| 														{% apply spaceless %} | ||||
| 														<form class="MediumButtonForm" action="{{ getLink('account/create') }}" method="post" > | ||||
| 															{{ csrf() }} | ||||
| 															<div class="MediumButtonBackground" style="background-image:url({{ template_path }}/images/global/buttons/mediumbutton.gif)" onMouseOver="MouseOverBigButton(this);" onMouseOut="MouseOutBigButton(this);"> | ||||
| 																<div class="MediumButtonOver" style="background-image:url({{ template_path }}/images/global/buttons/mediumbutton-over.gif)" onMouseOver="MouseOverBigButton(this);" onMouseOut="MouseOutBigButton(this);"></div> | ||||
| 																<input class="MediumButtonText" type="image" name="Create Account" alt="Create Account" src="{{ template_path }}/images/global/buttons/mediumbutton_createaccount.png" /> | ||||
|   | ||||
| @@ -290,9 +290,6 @@ | ||||
| {% endset %} | ||||
| {% include 'tables.headline.html.twig' %} | ||||
| <br/> | ||||
|  | ||||
| {{ include('account.2fa.main.html.twig') }} | ||||
|  | ||||
| {{ hook('HOOK_ACCOUNT_MANAGE_BEFORE_ACCOUNT_LOGS') }} | ||||
| <a name="Account+Logs" ></a> | ||||
| <div class="TopButtonContainer"> | ||||
|   | ||||
| @@ -943,14 +943,6 @@ img { | ||||
|   font-size: 8pt; | ||||
|   color: red; | ||||
| } | ||||
| .AttentionSign img { | ||||
| 	float: left; | ||||
| 	top: 3px; | ||||
| 	left: 8px; | ||||
| 	width: 15px; | ||||
| 	height: 13px; | ||||
| 	margin-right: 5px; | ||||
| } | ||||
| .SmallBox { | ||||
|   position: relative; | ||||
|   font-size: 1px; | ||||
|   | ||||
| @@ -27,17 +27,12 @@ if(isset($config['boxes'])) | ||||
| 		var loginStatus="<?php echo ($logged ? 'true' : 'false'); ?>"; | ||||
| 		<?php | ||||
| 			if(PAGE !== 'news') { | ||||
| 				if(isset($_REQUEST['subtopic'])) { | ||||
| 					$tmp = escapeHtml($_REQUEST['subtopic']); | ||||
| 					if($tmp === 'accountmanagement') { | ||||
| 						$tmp = 'accountmanage'; | ||||
| 					} | ||||
| 				} | ||||
| 				else { | ||||
| 					$tmp = str_replace('/', '_', PAGE); | ||||
| 				$tmp = str_replace('/', '_', isset($_REQUEST['subtopic']) ? escapeHtml($_REQUEST['subtopic']) :  PAGE); | ||||
| 				$exp = explode('/', PAGE); | ||||
| 				if(PAGE !== 'account/create' && PAGE !== 'account/lost' && isset($exp[1])) { | ||||
| 						if ($exp[0] === 'account') { | ||||
| 					if ($exp[0] === 'account' && $exp[1] === 'lost') { | ||||
| 						$tmp = 'account_lost'; | ||||
| 					} elseif ($exp[0] === 'account') { | ||||
| 						$tmp = 'account_manage'; | ||||
| 					} else if ($exp[0] === 'news' && $exp[1] === 'archive') { | ||||
| 						$tmp = 'news_archive'; | ||||
| @@ -47,7 +42,6 @@ if(isset($config['boxes'])) | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 			} | ||||
| 			else { | ||||
| 				$tmp = 'news'; | ||||
| 			} | ||||
| @@ -90,24 +84,24 @@ if(isset($config['boxes'])) | ||||
| 		// mouse-over and click events of the loginbox | ||||
| 		function MouseOverLoginBoxText(source) | ||||
| 		{ | ||||
| 		  source.lastChild.style.visibility = "visible"; | ||||
| 		  source.firstChild.style.visibility = "hidden"; | ||||
| 		  source.lastElementChild.style.visibility = "visible"; | ||||
| 		  source.firstElementChild.style.visibility = "hidden"; | ||||
| 		} | ||||
| 		function MouseOutLoginBoxText(source) | ||||
| 		{ | ||||
| 		  source.firstChild.style.visibility = "visible"; | ||||
| 		  source.lastChild.style.visibility = "hidden"; | ||||
| 		  source.firstElementChild.style.visibility = "visible"; | ||||
| 		  source.lastElementChild.style.visibility = "hidden"; | ||||
| 		} | ||||
| 		function LoginButtonAction() | ||||
| 		{ | ||||
| 		  if(loginStatus == "false") { | ||||
| 		  if(loginStatus === "false") { | ||||
| 			window.location = "<?php echo getLink('account/manage'); ?>"; | ||||
| 		  } else { | ||||
| 			window.location = "<?php echo getLink('account/manage'); ?>"; | ||||
| 		  } | ||||
| 		} | ||||
| 		function LoginstatusTextAction(source) { | ||||
| 		  if(loginStatus == "false") { | ||||
| 		  if(loginStatus === "false") { | ||||
| 			window.location = "<?php echo getLink('account/create'); ?>"; | ||||
| 		  } else { | ||||
| 			window.location = "<?php echo getLink('account/logout'); ?>"; | ||||
| @@ -226,11 +220,11 @@ if(isset($config['boxes'])) | ||||
| 		// mouse-over effects of menubuttons and submenuitems | ||||
| 		function MouseOverMenuItem(source) | ||||
| 		{ | ||||
| 		  source.firstChild.style.visibility = "visible"; | ||||
| 		  source.firstElementChild.style.visibility = "visible"; | ||||
| 		} | ||||
| 		function MouseOutMenuItem(source) | ||||
| 		{ | ||||
| 		  source.firstChild.style.visibility = "hidden"; | ||||
| 		  source.firstElementChild.style.visibility = "hidden"; | ||||
| 		} | ||||
| 		function MouseOverSubmenuItem(source) | ||||
| 		{ | ||||
| @@ -336,7 +330,7 @@ if(isset($config['boxes'])) | ||||
|     <div id="LoginBottom" class="Loginstatus" style="background-image:url(<?php echo $template_path; ?>/images/general/box-bottom.gif)" ></div> | ||||
|   </div> | ||||
|  | ||||
| <div-- id='Menu'> | ||||
| <div id='Menu'> | ||||
| <div id='MenuTop' style='background-image:url(<?php echo $template_path; ?>/images/general/box-top.gif);'></div> | ||||
|  | ||||
| <?php | ||||
| @@ -401,6 +395,7 @@ foreach($config['menu_categories'] as $id => $cat) { | ||||
| 	<?php | ||||
| 	} | ||||
| 	?> | ||||
| </div> | ||||
| 		<script type="text/javascript"> | ||||
| 			InitializePage(); | ||||
|         </script> | ||||
|   | ||||
| @@ -1,11 +1,11 @@ | ||||
| function MouseOverBigButton(source) { | ||||
| 	if (source?.firstChild?.style) { | ||||
| 		source.firstChild.style.visibility = "visible"; | ||||
| 	if (source?.firstElementChild?.style) { | ||||
| 		source.firstElementChild.style.visibility = "visible"; | ||||
| 	} | ||||
| } | ||||
| function MouseOutBigButton(source) { | ||||
| 	if (source?.firstChild?.style) { | ||||
| 		source.firstChild.style.visibility = "hidden"; | ||||
| 	if (source?.firstElementChild?.style) { | ||||
| 		source.firstElementChild.style.visibility = "hidden"; | ||||
| 	} | ||||
| } | ||||
| function BigButtonAction(path) { | ||||
|   | ||||
| @@ -9,6 +9,8 @@ | ||||
|  * @link      https://my-aac.org | ||||
|  */ | ||||
|  | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| // we need some functions | ||||
| require '../common.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
| @@ -1,4 +1,7 @@ | ||||
| <?php | ||||
|  | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| require '../common.php'; | ||||
| require SYSTEM . 'init.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
| @@ -12,6 +12,8 @@ | ||||
| use MyAAC\CreateCharacter; | ||||
| use MyAAC\Models\Account; | ||||
|  | ||||
| const IGNORE_SET_LAST_VISIT = true; | ||||
|  | ||||
| // we need some functions | ||||
| require '../common.php'; | ||||
| require SYSTEM . 'functions.php'; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user