Added Asynchronous tasks, usage sending

This commit is contained in:
Shoghi Cervantes 2014-04-02 20:20:14 +02:00
parent 11356b68ba
commit 0f42cd9243
4 changed files with 207 additions and 30 deletions

View File

@ -71,6 +71,7 @@ use pocketmine\plugin\PluginManager;
use pocketmine\pmf\LevelFormat;
use pocketmine\recipes\Crafting;
use pocketmine\scheduler\CallbackTask;
use pocketmine\scheduler\SendUsageTask;
use pocketmine\scheduler\ServerScheduler;
use pocketmine\scheduler\TickScheduler;
use pocketmine\tile\Sign;
@ -126,6 +127,9 @@ class Server{
/** @var int */
private $maxPlayers;
/** @var RCON */
private $rcon;
/**
* Counts the ticks since the server start
*
@ -144,6 +148,8 @@ class Server{
private $dataPath;
private $pluginPath;
private $lastSendUsage = null;
/** @var QueryHandler */
private $queryHandler;
@ -1249,14 +1255,7 @@ class Server{
$this->properties->save();
//TODO
/*if($this->getProperty("send-usage", true) !== false){
$this->server->schedule(6000, array($this, "sendUsage"), array(), true); //Send the info after 5 minutes have passed
$this->sendUsage();
}
if(!defined("NO_THREADS") and $this->getProperty("enable-rcon") === true){
$this->rcon = new RCON($this->getProperty("rcon.password", ""), $this->getProperty("rcon.port", $this->getProperty("server-port")), ($ip = $this->getProperty("server-ip")) != "" ? $ip : "0.0.0.0", $this->getProperty("rcon.threads", 1), $this->getProperty("rcon.clients-per-thread", 50));
}*/
$this->scheduler->scheduleDelayedRepeatingTask(new CallbackTask("pocketmine\\utils\\Cache::cleanup"), 20 * 45, 20 * 45);
if($this->getConfigBoolean("auto-save", true) === true){
$this->scheduler->scheduleDelayedRepeatingTask(new CallbackTask(array($this, "doAutoSave")), 18000, 18000);
@ -1371,6 +1370,11 @@ class Server{
}
if($this->getConfigBoolean("send-usage", true) !== false){
$this->scheduler->scheduleDelayedRepeatingTask(new CallbackTask(array($this, "sendUsage")), 6000, 6000);
$this->sendUsage();
}
if($this->getConfigBoolean("upnp-forwarding", false) == true){
console("[INFO] [UPnP] Trying to port forward...");
@ -1389,7 +1393,7 @@ class Server{
}
*/
console("[INFO] Default game type: " . self::getGamemodeString($this->getGamemode())); //TODO: string name
//$this->trigger("server.start", microtime(true));
console('[INFO] Done (' . round(microtime(true) - \pocketmine\START_TIME, 3) . 's)! For help, type "help" or "?"');
if(Utils::getOS() === "win"){ //Workaround less usleep() waste
$this->tickProcessorWindows();
@ -1397,6 +1401,15 @@ class Server{
$this->tickProcessor();
}
if($this->rcon instanceof RCON){
$this->rcon->stop();
}
if($this->getConfigBoolean("upnp-forwarding", false) === true){
console("[INFO] [UPnP] Removing port forward...");
UPnP::RemovePortForward($this->getPort());
}
$this->pluginManager->disablePlugins();
foreach($this->players as $player){
@ -1570,31 +1583,33 @@ class Server{
}
public function sendUsage(){
//TODO
/*console("[DEBUG] Sending usage data...", true, true, 2);
if($this->lastSendUsage instanceof SendUsageTask){
if(!$this->lastSendUsage->isFinished()){ //do not call multiple times
return;
}
}
$plist = "";
foreach(Server::getInstance()->getPluginManager()->getPlugins() as $p){
foreach($this->getPluginManager()->getPlugins() as $p){
$d = $p->getDescription();
$plist .= str_replace(array(";", ":"), "", $d->getName()) . ":" . str_replace(array(";", ":"), "", $d->getVersion()) . ";";
}
$this->asyncOperation(ASYNC_CURL_POST, array(
"url" => "http://stats.pocketmine.net/usage.php",
"data" => array(
"serverid" => $this->server->serverID,
"port" => $this->server->port,
"os" => Utils::getOS(),
"memory_total" => $this->getProperty("memory-limit"),
"memory_usage" => memory_get_usage(true),
"php_version" => PHP_VERSION,
"version" => VERSION,
"mc_version" => MINECRAFT_VERSION,
"protocol" => Info::CURRENT_PROTOCOL,
"online" => count($this->players),
"max" => $this->server->maxClients,
"plugins" => $plist,
),
), null);*/
$this->lastSendUsage = new SendUsageTask("http://stats.pocketmine.net/usage.php", array(
"serverid" => Utils::readLong(substr(Utils::getUniqueID(true, $this->getIp() .":". $this->getPort()), 0, 8)),
"port" => $this->getPort(),
"os" => Utils::getOS(),
"memory_total" => $this->getConfigString("memory-limit"),
"memory_usage" => memory_get_usage(),
"php_version" => \pocketmine\PHP_VERSION,
"version" => \pocketmine\VERSION,
"mc_version" => \pocketmine\MINECRAFT_VERSION,
"protocol" => network\protocol\Info::CURRENT_PROTOCOL,
"online" => count($this->players),
"max" => $this->getMaxPlayers(),
"plugins" => $plist,
));
$this->scheduler->scheduleAsyncTask($this->lastSendUsage);
}
public function titleTick(){

View File

@ -0,0 +1,91 @@
<?php
/*
*
* ____ _ _ __ __ _ __ __ ____
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* @author PocketMine Team
*
*
*/
namespace pocketmine\scheduler;
/**
* Class used to run async tasks in other threads.
*
* WARNING: Do not call PocketMine-MP API methods from other Threads!!
*/
abstract class AsyncTask extends \Threaded{
private $complete;
private $finished;
private $result;
public function run(){
$this->synchronized(function(){
$this->result = null;
$this->onRun(\Thread::getCurrentThread());
$this->finished = true;
$this->complete = $this->result === null ? true : false;
});
}
/**
* @return bool
*/
public function isCompleted(){
return $this->complete === true;
}
/**
* @return mixed
*/
public function getResult(){
return $this->synchronized(function(){
$this->finished = true;
return @unserialize($this->result);
});
}
/**
* @return bool
*/
public function hasResult(){
return $this->result !== null;
}
/**
* @param mixed $result
*/
public function setResult($result){
$this->result = @serialize($result);
}
/**
* @return bool
*/
public function isFinished(){
return $this->finished === true;
}
/**
* Actions to execute when run
*
* @param \Thread $thread
*
* @return void
*/
public abstract function onRun(\Thread $thread);
}

View File

@ -0,0 +1,40 @@
<?php
/*
*
* ____ _ _ __ __ _ __ __ ____
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* @author PocketMine Team
*
*
*/
namespace pocketmine\scheduler;
use pocketmine\utils\Utils;
class SendUsageTask extends AsyncTask{
public $endpoint;
public $data;
public function __construct($endpoint, array $data){
$this->endpoint = $endpoint;
$this->data = serialize($data);
}
public function onRun(\Thread $thread){
Utils::postURL($this->endpoint, unserialize($this->data));
}
}

View File

@ -28,7 +28,7 @@ use pocketmine\plugin\Plugin;
use pocketmine\utils\ReversePriorityQueue;
class ServerScheduler{
protected static $WORKERS = 3;
/**
* @var ReversePriorityQueue<Task>
*/
@ -39,6 +39,11 @@ class ServerScheduler{
*/
protected $tasks = array();
/** @var \Pool */
protected $asyncPool;
protected $asyncTasks = 0;
/** @var int */
private $ids = 1;
@ -47,6 +52,7 @@ class ServerScheduler{
public function __construct(){
$this->queue = new ReversePriorityQueue();
$this->asyncPool = new \Pool(self::$WORKERS, "Worker");
}
/**
@ -58,6 +64,19 @@ class ServerScheduler{
return $this->addTask($task, -1, -1);
}
/**
* Submits a asynchronous task to the Pool
* If the AsyncTask sets a result, you have to get it so it can be deleted
*
* @param AsyncTask $task
*
* @return void
*/
public function scheduleAsyncTask(AsyncTask $task){
$this->asyncPool->submit($task);
++$this->asyncTasks;
}
/**
* @param Task $task
* @param int $delay
@ -116,6 +135,8 @@ class ServerScheduler{
$task->cancel();
}
$this->tasks = array();
$this->asyncPool->shutdown();
$this->asyncTasks = 0;
}
/**
@ -182,6 +203,16 @@ class ServerScheduler{
unset($this->tasks[$task->getTaskId()]);
}
}
if($this->asyncTasks > 0){ //Garbage collector
$this->asyncPool->collect(function(AsyncTask $task){
if($task->isCompleted() or ($task->isFinished() and !$task->hasResult())){
--$this->asyncTasks;
return true;
}
return false;
});
}
}
private function isReady($currentTicks){