ReadData()
function Recognise($targetstring) { return FALSE; }
// the actual ReadData
// returns an array of two values (in,out). -1,-1 if it couldn't get valid data
// configline is passed in, to allow for better error messages
// itemtype and itemname may be used as part of the target (e.g. for TSV source line)
// function ReadData($targetstring, $configline, $itemtype, $itemname, $map) { return (array(-1,-1)); }
function ReadData($targetstring, &$map, &$item)
{
return ([-1, -1]);
}
// pre-register a target + context, to allow a plugin to batch up queries to a slow database, or snmp for example
function Register($targetstring, &$map, &$item) {}
// called before ReadData, to allow plugins to DO the prefetch of targets known from Register
function Prefetch() {}
}
// template classes for the pre- and post-processor plugins
class WeatherMapPreProcessor
{
function run(&$map) { return FALSE; }
}
class WeatherMapPostProcessor
{
function run(&$map) { return FALSE; }
}
// ***********************************************
// Links, Nodes and the Map object inherit from this class ultimately.
// Just to make some common code common.
class WeatherMapBase
{
var $notes = [];
var $hints = [];
var $inherit_fieldlist;
function add_note($name, $value)
{
wm_debug("Adding note $name='$value' to " . $this -> name . "\n");
$this -> notes[$name] = $value;
}
function get_note($name)
{
if (isset($this -> notes[$name])) {
// debug("Found note $name in ".$this->name." with value of ".$this->notes[$name].".\n");
return ($this -> notes[$name]);
} else {
// debug("Looked for note $name in ".$this->name." which doesn't exist.\n");
return (NULL);
}
}
function add_hint($name, $value)
{
wm_debug("Adding hint $name='$value' to " . $this -> name . "\n");
$this -> hints[$name] = $value;
# warn("Adding hint $name to ".$this->my_type()."/".$this->name."\n");
}
function get_hint($name)
{
if (isset($this -> hints[$name])) {
// debug("Found hint $name in ".$this->name." with value of ".$this->hints[$name].".\n");
return ($this -> hints[$name]);
} else {
// debug("Looked for hint $name in ".$this->name." which doesn't exist.\n");
return (NULL);
}
}
}
class WeatherMapConfigItem
{
var $defined_in;
var $name;
var $value;
var $type;
}
// The 'things on the map' class. More common code (mainly variables, actually)
class WeatherMapItem extends WeatherMapBase
{
var $owner;
var $configline;
var $infourl;
var $overliburl;
var $overlibwidth, $overlibheight;
var $overlibcaption;
var $my_default;
var $defined_in;
var $config_override; # used by the editor to allow text-editing
function my_type() { return "ITEM"; }
}
class WeatherMap extends WeatherMapBase
{
var $nodes = []; // an array of WeatherMapNodes
var $links = []; // an array of WeatherMapLinks
var $texts = []; // an array containing all the extraneous text bits
var $used_images = []; // an array of image filenames referred to (used by editor)
var $seen_zlayers = [0 => [], 1000 => []]; // 0 is the background, 1000 is the legends, title, etc
var $config;
var $next_id;
var $min_ds_time;
var $max_ds_time;
var $background;
var $htmlstyle;
var $imap;
var $colours;
var $configfile;
var $imagefile, $imageuri;
var $rrdtool;
var $daemon;
var $daemon_args;
var $title, $titlefont;
var $kilo;
var $sizedebug, $widthmod, $debugging;
var $linkfont, $nodefont, $keyfont, $timefont;
// var $bg_r, $bg_g, $bg_b;
var $timex, $timey;
var $width, $height;
var $keyx, $keyy, $keyimage;
var $titlex, $titley;
var $keytext, $stamptext, $datestamp;
var $min_data_time, $max_data_time;
var $htmloutputfile, $imageoutputfile;
var $dataoutputfile;
var $htmlstylesheet;
var $defaultlink, $defaultnode;
var $need_size_precalc;
var $keystyle, $keysize;
var $rrdtool_check;
var $inherit_fieldlist;
var $mintimex, $maxtimex;
var $mintimey, $maxtimey;
var $minstamptext, $maxstamptext;
var $context;
var $cachefolder, $mapcache, $cachefile_version;
var $name;
var $black, $white, $grey, $selected;
var $datasourceclasses;
var $preprocessclasses;
var $postprocessclasses;
var $activedatasourceclasses;
var $thumb_width, $thumb_height;
var $has_includes;
var $has_overlibs;
var $node_template_tree;
var $link_template_tree;
var $dsinfocache = [];
var $plugins = [];
var $included_files = [];
var $usage_stats = [];
var $coverage = [];
var $colourtable = [];
var $warncount = 0;
function __construct()
{
$this -> inherit_fieldlist = ['width' => 800, 'height' => 600, 'kilo' => 1000, 'numscales' => ['DEFAULT' => 0], 'datasourceclasses' => [], 'preprocessclasses' => [], 'postprocessclasses' => [], 'included_files' => [], 'context' => '', 'dumpconfig' => FALSE, 'rrdtool_check' => '', 'background' => '', 'imageoutputfile' => '', 'imageuri' => '', 'htmloutputfile' => '', 'dataoutputfile' => '', 'htmlstylesheet' => '', 'labelstyle' => 'percent', // redundant?
'htmlstyle' => 'static', 'keystyle' => ['DEFAULT' => 'classic'], 'title' => 'Network Weathermap', 'keytext' => ['DEFAULT' => 'Traffic Load'], 'keyx' => ['DEFAULT' => -1], 'keyy' => ['DEFAULT' => -1], 'keyimage' => [], 'keysize' => ['DEFAULT' => 400], 'stamptext' => 'Created: %b %d %Y %H:%M:%S', 'keyfont' => 4, 'titlefont' => 2, 'timefont' => 2, 'timex' => 0, 'timey' => 0,
'mintimex' => -10000, 'mintimey' => -10000, 'maxtimex' => -10000, 'maxtimey' => -10000, 'minstamptext' => 'Oldest Data: %b %d %Y %H:%M:%S', 'maxstamptext' => 'Newest Data: %b %d %Y %H:%M:%S',
'thumb_width' => 0, 'thumb_height' => 0, 'titlex' => -1, 'titley' => -1, 'cachefolder' => 'cached', 'mapcache' => '', 'sizedebug' => FALSE, 'debugging' => FALSE, 'widthmod' => FALSE, 'has_includes' => FALSE, 'has_overlibs' => FALSE, 'name' => 'MAP'];
$this -> Reset();
}
function my_type() { return "MAP"; }
function Reset()
{
$this -> next_id = 100;
foreach (array_keys($this -> inherit_fieldlist) as $fld) {
$this -> $fld = $this -> inherit_fieldlist[$fld];
}
$this -> min_ds_time = NULL;
$this -> max_ds_time = NULL;
$this -> need_size_precalc = FALSE;
$this -> nodes = []; // an array of WeatherMapNodes
$this -> links = []; // an array of WeatherMapLinks
// these are the default defaults
// by putting them into a normal object, we can use the
// same code for writing out LINK DEFAULT as any other link.
wm_debug("Creating ':: DEFAULT ::' DEFAULT LINK\n");
// these two are used for default settings
$deflink = new WeatherMapLink;
$deflink -> name = ":: DEFAULT ::";
$deflink -> template = ":: DEFAULT ::";
$deflink -> Reset($this);
$this -> links[':: DEFAULT ::'] = &$deflink;
wm_debug("Creating ':: DEFAULT ::' DEFAULT NODE\n");
$defnode = new WeatherMapNode;
$defnode -> name = ":: DEFAULT ::";
$defnode -> template = ":: DEFAULT ::";
$defnode -> Reset($this);
$this -> nodes[':: DEFAULT ::'] = &$defnode;
$this -> node_template_tree = [];
$this -> link_template_tree = [];
$this -> node_template_tree['DEFAULT'] = [];
$this -> link_template_tree['DEFAULT'] = [];
// ************************************
// now create the DEFAULT link and node, based on those.
// these can be modified by the user, but their template (and therefore comparison in WriteConfig) is ':: DEFAULT ::'
wm_debug("Creating actual DEFAULT NODE from :: DEFAULT ::\n");
$defnode2 = new WeatherMapNode;
$defnode2 -> name = "DEFAULT";
$defnode2 -> template = ":: DEFAULT ::";
$defnode2 -> Reset($this);
$this -> nodes['DEFAULT'] = &$defnode2;
wm_debug("Creating actual DEFAULT LINK from :: DEFAULT ::\n");
$deflink2 = new WeatherMapLink;
$deflink2 -> name = "DEFAULT";
$deflink2 -> template = ":: DEFAULT ::";
$deflink2 -> Reset($this);
$this -> links['DEFAULT'] = &$deflink2;
// for now, make the old defaultlink and defaultnode work too.
// $this->defaultlink = $this->links['DEFAULT'];
// $this->defaultnode = $this->nodes['DEFAULT'];
assert('is_object($this->nodes[":: DEFAULT ::"])');
assert('is_object($this->links[":: DEFAULT ::"])');
assert('is_object($this->nodes["DEFAULT"])');
assert('is_object($this->links["DEFAULT"])');
// ************************************
$this -> imap = new HTML_ImageMap('weathermap');
$this -> colours = [];
wm_debug("Adding default map colour set.\n");
$defaults = ['KEYTEXT' => ['bottom' => -2, 'top' => -1, 'red1' => 0, 'green1' => 0, 'blue1' => 0, 'special' => 1], 'KEYOUTLINE' => ['bottom' => -2, 'top' => -1, 'red1' => 0, 'green1' => 0, 'blue1' => 0, 'special' => 1], 'KEYBG' => ['bottom' => -2, 'top' => -1, 'red1' => 255, 'green1' => 255, 'blue1' => 255, 'special' => 1], 'BG' => ['bottom' => -2, 'top' => -1, 'red1' => 255, 'green1' => 255, 'blue1' => 255, 'special' => 1], 'TITLE' => ['bottom' => -2, 'top' => -1, 'red1' => 0, 'green1' => 0, 'blue1' => 0, 'special' => 1], 'TIME' => ['bottom' => -2, 'top' => -1, 'red1' => 0, 'green1' => 0, 'blue1' => 0, 'special' => 1]];
foreach ($defaults as $key => $def) {
$this -> colours['DEFAULT'][$key] = $def;
}
$this -> configfile = '';
$this -> imagefile = '';
$this -> imageuri = '';
$this -> fonts = [];
// Adding these makes the editor's job a little easier, mainly
for ($i = 1; $i <= 5; $i++) {
$this -> fonts[$i] = new WMFont();
$this -> fonts[$i] -> type = "GD builtin";
$this -> fonts[$i] -> file = '';
$this -> fonts[$i] -> size = 0;
}
$this -> LoadPlugins('data', 'lib' . DIRECTORY_SEPARATOR . 'datasources');
$this -> LoadPlugins('pre', 'lib' . DIRECTORY_SEPARATOR . 'pre');
$this -> LoadPlugins('post', 'lib' . DIRECTORY_SEPARATOR . 'post');
wm_debug("WeatherMap class Reset() complete\n");
}
/**
* Create an array of all the nodes and links, mixed together.
* readData() makes several passes through this list.
*
* @return MapDataItem[]
*/
public function buildAllItemsList()
{
// TODO - this should probably be a static, or otherwise cached
$allItems = [];
foreach ([&$this -> nodes, &$this -> links] as $innerList) {
foreach ($innerList as $item) {
$allItems[] = $item;
}
}
return $allItems;
}
function myimagestring($image, $fontnumber, $x, $y, $string, $colour, $angle = 0)
{
// if it's supposed to be a special font, and it hasn't been defined, then fall through
if ($fontnumber > 5 && !isset($this -> fonts[$fontnumber])) {
wm_warn("Using a non-existent special font ($fontnumber) - falling back to internal GD fonts [WMWARN03]\n");
if ($angle != 0) {
wm_warn("Angled text doesn't work with non-FreeType fonts [WMWARN02]\n");
}
$fontnumber = 5;
}
if (($fontnumber > 0) && ($fontnumber < 6)) {
imagestring($image, $fontnumber, $x, $y - imagefontheight($fontnumber), $string, $colour);
if ($angle != 0) {
wm_warn("Angled text doesn't work with non-FreeType fonts [WMWARN02]\n");
}
} else {
// look up what font is defined for this slot number
if ($this -> fonts[$fontnumber] -> type == 'truetype') {
wimagettftext($image, $this -> fonts[$fontnumber] -> size, $angle, $x, $y, $colour, $this -> fonts[$fontnumber] -> file, $string);
}
if ($this -> fonts[$fontnumber] -> type == 'gd') {
imagestring($image, $this -> fonts[$fontnumber] -> gdnumber, $x, $y - imagefontheight($this -> fonts[$fontnumber] -> gdnumber), $string, $colour);
if ($angle != 0) {
wm_warn("Angled text doesn't work with non-FreeType fonts [WMWARN04]\n");
}
}
}
}
function myimagestringsize($fontnumber, $string)
{
$linecount = 1;
$lines = explode("\n", $string);
$linecount = sizeof($lines);
$maxlinelength = 0;
foreach ($lines as $line) {
$l = strlen($line);
if ($l > $maxlinelength) {
$maxlinelength = $l;
}
}
if (($fontnumber > 0) && ($fontnumber < 6)) {
return [imagefontwidth($fontnumber) * $maxlinelength, $linecount * imagefontheight($fontnumber)];
} else {
// look up what font is defined for this slot number
if (!isset($this -> fonts[$fontnumber])) {
wm_warn("Using a non-existent special font ($fontnumber) - falling back to internal GD fonts [WMWARN36]\n");
$fontnumber = 5;
return [imagefontwidth($fontnumber) * $maxlinelength, $linecount * imagefontheight($fontnumber)];
} else {
if ($this -> fonts[$fontnumber] -> type == 'truetype') {
$ysize = 0;
$xsize = 0;
foreach ($lines as $line) {
$bounds = imagettfbbox($this -> fonts[$fontnumber] -> size, 0, $this -> fonts[$fontnumber] -> file, $line);
$cx = $bounds[4] - $bounds[0];
$cy = $bounds[1] - $bounds[5];
if ($cx > $xsize) {
$xsize = $cx;
}
$ysize += ($cy * 1.2);
# warn("Adding $cy (x was $cx)\n");
}
#$bounds=imagettfbbox($this->fonts[$fontnumber]->size, 0, $this->fonts[$fontnumber]->file,
# $string);
# return (array($bounds[4] - $bounds[0], $bounds[1] - $bounds[5]));
# warn("Size of $string is $xsize x $ysize over $linecount lines\n");
return ([$xsize, $ysize]);
}
if ($this -> fonts[$fontnumber] -> type == 'gd') {
return [imagefontwidth($this -> fonts[$fontnumber] -> gdnumber) * $maxlinelength, $linecount * imagefontheight($this -> fonts[$fontnumber] -> gdnumber)];
}
}
}
}
function ProcessString($input, &$context, $include_notes = TRUE, $multiline = FALSE)
{
# debug("ProcessString: input is $input\n");
assert('is_scalar($input)');
$context_description = strtolower($context -> my_type());
if ($context_description != "map") {
$context_description .= ":" . $context -> name;
}
wm_debug("Trace: ProcessString($input, $context_description)\n");
if ($multiline == TRUE) {
$i = $input;
$input = str_replace("\\n", "\n", $i);
# if($i != $input) warn("$i into $input\n");
}
$output = $input;
# while( preg_match("/(\{[^}]+\})/",$input,$matches) )
while (preg_match('/(\{(?:node|map|link)[^}]+\})/', $input, $matches)) {
$value = "[UNKNOWN]";
$format = "";
$key = $matches[1];
wm_debug("ProcessString: working on " . $key . "\n");
if (preg_match('/\{(node|map|link):([^}]+)\}/', $key, $matches)) {
$type = $matches[1];
$args = $matches[2];
# debug("ProcessString: type is ".$type.", arguments are ".$args."\n");
if ($type == 'map') {
$the_item = $this;
if (preg_match("/map:([^:]+):*([^:]*)/", $args, $matches)) {
$args = $matches[1];
$format = $matches[2];
}
}
if (($type == 'link') || ($type == 'node')) {
if (preg_match("/([^:]+):([^:]+):*([^:]*)/", $args, $matches)) {
$itemname = $matches[1];
$args = $matches[2];
$format = $matches[3];
# debug("ProcessString: item is $itemname, and args are now $args\n");
$the_item = NULL;
if (($itemname == "this") && ($type == strtolower($context -> my_type()))) {
$the_item = $context;
} elseif (strtolower($context -> my_type()) == "link" && $type == 'node' && ($itemname == '_linkstart_' || $itemname == '_linkend_')) {
// this refers to the two nodes at either end of this link
if ($itemname == '_linkstart_') {
$the_item = $context -> a;
}
if ($itemname == '_linkend_') {
$the_item = $context -> b;
}
} elseif (($itemname == "parent") && ($type == strtolower($context -> my_type())) && ($type == 'node') && ($context -> relative_to != '')) {
$the_item = $this -> nodes[$context -> relative_to];
} else {
if (($type == 'link') && isset($this -> links[$itemname])) {
$the_item = $this -> links[$itemname];
}
if (($type == 'node') && isset($this -> nodes[$itemname])) {
$the_item = $this -> nodes[$itemname];
}
}
}
}
if (is_null($the_item)) {
wm_warn("ProcessString: $key refers to unknown item (context is $context_description) [WMWARN05]\n");
} else {
# warn($the_item->name.": ".var_dump($the_item->hints)."\n");
wm_debug("ProcessString: Found appropriate item: " . get_class($the_item) . " " . $the_item -> name . "\n");
# warn($the_item->name."/hints: ".var_dump($the_item->hints)."\n");
# warn($the_item->name."/notes: ".var_dump($the_item->notes)."\n");
// SET and notes have precedent over internal properties
// this is my laziness - it saves me having a list of reserved words
// which are currently used for internal props. You can just 'overwrite' any of them.
if (isset($the_item -> hints[$args])) {
$value = $the_item -> hints[$args];
wm_debug("ProcessString: used hint\n");
}
// for some things, we don't want to allow notes to be considered.
// mainly - TARGET (which can define command-lines), shouldn't be
// able to get data from uncontrolled sources (i.e. data sources rather than SET in config files).
elseif ($include_notes && isset($the_item -> notes[$args])) {
$value = $the_item -> notes[$args];
wm_debug("ProcessString: used note\n");
} elseif (isset($the_item -> $args)) {
$value = $the_item -> $args;
wm_debug("ProcessString: used internal property\n");
}
}
}
// format, and sanitise the value string here, before returning it
if ($value === NULL) {
$value = 'NULL';
}
wm_debug("ProcessString: replacing " . $key . " with $value\n");
# if($format != '') $value = sprintf($format,$value);
if ($format != '') {
# debug("Formatting with mysprintf($format,$value)\n");
$value = mysprintf($format, $value, $this -> kilo);
}
# debug("ProcessString: formatted to $value\n");
$input = str_replace($key, '', $input);
$output = str_replace($key, $value, $output);
}
#debug("ProcessString: output is $output\n");
return ($output);
}
function RandomData()
{
foreach ($this -> links as $link) {
$this -> links[$link -> name] -> bandwidth_in = rand(0, $link -> max_bandwidth_in);
$this -> links[$link -> name] -> bandwidth_out = rand(0, $link -> max_bandwidth_out);
}
}
function LoadPlugins($type = "data", $dir = "lib/datasources")
{
wm_debug("Beginning to load $type plugins from $dir\n");
if (!file_exists($dir)) {
$dir = dirname(__FILE__) . DIRECTORY_SEPARATOR . $dir;
wm_debug("Relative path didn't exist. Trying $dir\n");
}
# $this->datasourceclasses = array();
$dh = @opendir($dir);
if (!$dh) { // try to find it with the script, if the relative path fails
$srcdir = substr($_SERVER['argv'][0], 0, strrpos($_SERVER['argv'][0], DIRECTORY_SEPARATOR));
$dh = opendir($srcdir . DIRECTORY_SEPARATOR . $dir);
if ($dh) {
$dir = $srcdir . DIRECTORY_SEPARATOR . $dir;
}
}
if ($dh) {
while ($file = readdir($dh)) {
$realfile = $dir . DIRECTORY_SEPARATOR . $file;
if (is_file($realfile) && preg_match('/\.php$/', $realfile)) {
wm_debug("Loading $type Plugin class from $file\n");
include_once($realfile);
$class = preg_replace("/\.php$/", "", $file);
if ($type == 'data') {
$this -> datasourceclasses [$class] = $class;
$this -> activedatasourceclasses[$class] = 1;
}
if ($type == 'pre') {
$this -> preprocessclasses [$class] = $class;
}
if ($type == 'post') {
$this -> postprocessclasses [$class] = $class;
}
wm_debug("Loaded $type Plugin class $class from $file\n");
$this -> plugins[$type][$class] = new $class;
if (!isset($this -> plugins[$type][$class])) {
wm_debug("** Failed to create an object for plugin $type/$class\n");
} else {
wm_debug("Instantiated $class.\n");
}
} else {
wm_debug("Skipping $file\n");
}
}
} else {
wm_warn("Couldn't open $type Plugin directory ($dir). Things will probably go wrong. [WMWARN06]\n");
}
}
function DatasourceInit()
{
wm_debug("Running Init() for Data Source Plugins...\n");
foreach ($this -> datasourceclasses as $ds_class) {
// make an instance of the class
$dsplugins[$ds_class] = new $ds_class;
wm_debug("Running $ds_class" . "->Init()\n");
# $ret = call_user_func(array($ds_class, 'Init'), $this);
assert('isset($this->plugins["data"][$ds_class])');
$ret = $this -> plugins['data'][$ds_class] -> Init($this);
if (!$ret) {
wm_debug("Removing $ds_class from Data Source list, since Init() failed\n");
$this -> activedatasourceclasses[$ds_class] = 0;
# unset($this->datasourceclasses[$ds_class]);
}
}
wm_debug("Finished Initialising Plugins...\n");
}
function ProcessTargets()
{
wm_debug("Preprocessing targets\n");
$allitems = $this -> buildAllItemsList();
foreach ($allitems as $myobj) {
$type = $myobj -> my_type();
$name = $myobj -> name;
if (($type == 'LINK' && isset($myobj -> a)) || ($type == 'NODE' && !is_null($myobj -> x))) {
if (count($myobj -> targets) > 0) {
$tindex = 0;
foreach ($myobj -> targets as $target) {
wm_debug("ProcessTargets: New Target: $target[4]\n");
// processstring won't use notes (only hints) for this string
$targetstring = $this -> ProcessString($target[4], $myobj, FALSE, FALSE);
if ($target[4] != $targetstring) {
wm_debug("Targetstring is now $targetstring\n");
}
// if the targetstring starts with a -, then we're taking this value OFF the aggregate
$multiply = 1;
if (preg_match("/^-(.*)/", $targetstring, $matches)) {
$targetstring = $matches[1];
$multiply = -1 * $multiply;
}
// if the remaining targetstring starts with a number and a *-, then this is a scale factor
if (preg_match("/^(\d+\.?\d*)\*(.*)/", $targetstring, $matches)) {
$targetstring = $matches[2];
$multiply = $multiply * floatval($matches[1]);
}
$matched = FALSE;
$matched_by = '';
foreach ($this -> datasourceclasses as $ds_class) {
if (!$matched) {
// $recognised = call_user_func(array($ds_class, 'Recognise'), $targetstring);
$recognised = $this -> plugins['data'][$ds_class] -> Recognise($targetstring);
if ($recognised) {
$matched = TRUE;
$matched_by = $ds_class;
if ($this -> activedatasourceclasses[$ds_class]) {
$this -> plugins['data'][$ds_class] -> Register($targetstring, $this, $myobj);
if ($type == 'NODE') {
$this -> nodes[$name] -> targets[$tindex][1] = $multiply;
$this -> nodes[$name] -> targets[$tindex][0] = $targetstring;
$this -> nodes[$name] -> targets[$tindex][5] = $matched_by;
}
if ($type == 'LINK') {
$this -> links[$name] -> targets[$tindex][1] = $multiply;
$this -> links[$name] -> targets[$tindex][0] = $targetstring;
$this -> links[$name] -> targets[$tindex][5] = $matched_by;
}
} else {
wm_warn("ProcessTargets: $type $name, target: $targetstring on config line $target[3] of $target[2] was recognised as a valid TARGET by a plugin that is unable to run ($ds_class) [WMWARN07]\n");
}
}
}
}
if (!$matched) {
wm_warn("ProcessTargets: $type $name, target: $target[4] on config line $target[3] of $target[2] was not recognised as a valid TARGET [WMWARN08]\n");
}
$tindex++;
}
}
}
}
}
function ReadData()
{
$this -> DatasourceInit();
wm_debug("======================================\n");
wm_debug("ReadData: Updating link data for all links and nodes\n");
// we skip readdata completely in sizedebug mode
if ($this -> sizedebug == 0) {
$this -> ProcessTargets();
wm_debug("======================================\n");
wm_debug("Starting prefetch\n");
foreach ($this -> datasourceclasses as $ds_class) {
$this -> plugins['data'][$ds_class] -> Prefetch();
}
wm_debug("======================================\n");
wm_debug("Starting main collection loop\n");
$allitems = $this -> buildAllItemsList();
foreach ($allitems as $myobj) {
$type = $myobj -> my_type();
$total_in = 0;
$total_out = 0;
$name = $myobj -> name;
wm_debug("\n");
wm_debug("ReadData for $type $name: \n");
if (($type == 'LINK' && isset($myobj -> a)) || ($type == 'NODE' && !is_null($myobj -> x))) {
if (count($myobj -> targets) > 0) {
$tindex = 0;
foreach ($myobj -> targets as $target) {
wm_debug("ReadData: New Target: $target[4]\n");
//debug ( var_dump($target));
$targetstring = $target[0];
$multiply = (int)$target[1];
//exit();
$in = 0;
$out = 0;
$datatime = 0;
if ($target[4] != '') {
// processstring won't use notes (only hints) for this string
$targetstring = $this -> ProcessString($target[0], $myobj, FALSE, FALSE);
if ($target[0] != $targetstring) {
wm_debug("Targetstring is now $targetstring\n");
}
if ($multiply != 1) {
wm_debug("Will multiply result by $multiply\n");
}
if ($target[0] != "") {
$matched_by = $target[5];
[$in, $out, $datatime] = $this -> plugins['data'][$target[5]] -> ReadData($targetstring, $this, $myobj);
}
if (($in === NULL) && ($out === NULL)) {
$in = 0;
$out = 0;
wm_warn("ReadData: $type $name, target: $targetstring on config line $target[3] of $target[2] had no valid data, according to $matched_by\n");
} else {
if ($in === NULL) {
$in = 0;
}
if ($out === NULL) {
$out = 0;
}
}
if ($multiply != 1) {
wm_debug("Pre-multiply: $in $out\n");
$in = $multiply * $in;
$out = $multiply * $out;
wm_debug("Post-multiply: $in $out\n");
}
$total_in = $total_in + $in;
$total_out = $total_out + $out;
wm_debug("Aggregate so far: $total_in $total_out\n");
# keep a track of the range of dates for data sources (mainly for MRTG/textfile based DS)
if ($datatime > 0) {
if ($this -> max_data_time == NULL || $datatime > $this -> max_data_time) {
$this -> max_data_time = $datatime;
}
if ($this -> min_data_time == NULL || $datatime < $this -> min_data_time) {
$this -> min_data_time = $datatime;
}
wm_debug("DataTime MINMAX: " . $this -> min_data_time . " -> " . $this -> max_data_time . "\n");
}
}
$tindex++;
}
wm_debug("ReadData complete for $type $name: $total_in $total_out\n");
} else {
wm_debug("ReadData: No targets for $type $name\n");
}
} else {
wm_debug("ReadData: Skipping $type $name that looks like a template\n.");
}
# $this->links[$name]->bandwidth_in=$total_in;
# $this->links[$name]->bandwidth_out=$total_out;
$myobj -> bandwidth_in = $total_in;
$myobj -> bandwidth_out = $total_out;
if ($type == 'LINK' && $myobj -> duplex == 'half') {
// in a half duplex link, in and out share a common bandwidth pool, so percentages need to include both
wm_debug("Calculating percentage using half-duplex\n");
$myobj -> outpercent = (($total_in + $total_out) / ($myobj -> max_bandwidth_out)) * 100;
$myobj -> inpercent = (($total_out + $total_in) / ($myobj -> max_bandwidth_in)) * 100;
if ($myobj -> max_bandwidth_out != $myobj -> max_bandwidth_in) {
wm_warn("ReadData: $type $name: You're using asymmetric bandwidth AND half-duplex in the same link. That makes no sense. [WMWARN44]\n");
}
} else {
$myobj -> outpercent = (($total_out) / ($myobj -> max_bandwidth_out)) * 100;
$myobj -> inpercent = (($total_in) / ($myobj -> max_bandwidth_in)) * 100;
}
# print $myobj->name."=>".$myobj->inpercent."%/".$myobj->outpercent."\n";
$warn_in = TRUE;
$warn_out = TRUE;
if ($type == 'NODE' && $myobj -> scalevar == 'in') {
$warn_out = FALSE;
}
if ($type == 'NODE' && $myobj -> scalevar == 'out') {
$warn_in = FALSE;
}
if ($myobj -> scaletype == 'percent') {
[$incol, $inscalekey, $inscaletag] = $this -> NewColourFromPercent($myobj -> inpercent, $myobj -> usescale, $myobj -> name, TRUE, $warn_in);
[$outcol, $outscalekey, $outscaletag] = $this -> NewColourFromPercent($myobj -> outpercent, $myobj -> usescale, $myobj -> name, TRUE, $warn_out);
} else {
// use absolute values, if that's what is requested
[$incol, $inscalekey, $inscaletag] = $this -> NewColourFromPercent($myobj -> bandwidth_in, $myobj -> usescale, $myobj -> name, FALSE, $warn_in);
[$outcol, $outscalekey, $outscaletag] = $this -> NewColourFromPercent($myobj -> bandwidth_out, $myobj -> usescale, $myobj -> name, FALSE, $warn_out);
}
$myobj -> add_note("inscalekey", $inscalekey);
$myobj -> add_note("outscalekey", $outscalekey);
$myobj -> add_note("inscaletag", $inscaletag);
$myobj -> add_note("outscaletag", $outscaletag);
$myobj -> add_note("inscalecolor", $incol -> as_html());
$myobj -> add_note("outscalecolor", $outcol -> as_html());
$myobj -> colours[IN] = $incol;
$myobj -> colours[OUT] = $outcol;
### warn("TAGS (setting) |$inscaletag| |$outscaletag| \n");
wm_debug("ReadData: Setting $total_in,$total_out\n");
// unset($myobj);
}
// }
wm_debug("ReadData Completed.\n");
wm_debug("------------------------------\n");
}
}
// nodename is a vestigal parameter, from the days when nodes were just big labels
function DrawLabelRotated($im, $x, $y, $angle, $text, $font, $padding, $linkname, $textcolour, $bgcolour, $outlinecolour, &$map, $direction)
{
[$strwidth, $strheight] = $this -> myimagestringsize($font, $text);
if (abs($angle) > 90) {
$angle -= 180;
}
if ($angle < -180) {
$angle += 360;
}
$rangle = -deg2rad($angle);
$extra = 3;
$x1 = $x - ($strwidth / 2) - $padding - $extra;
$x2 = $x + ($strwidth / 2) + $padding + $extra;
$y1 = $y - ($strheight / 2) - $padding - $extra;
$y2 = $y + ($strheight / 2) + $padding + $extra;
// a box. the last point is the start point for the text.
$points = [$x1, $y1, $x1, $y2, $x2, $y2, $x2, $y1, $x - $strwidth / 2, $y + $strheight / 2 + 1];
$npoints = count($points) / 2;
rotateAboutPoint($points, $x, $y, $rangle);
if ($bgcolour != [-1, -1, -1]) {
$bgcol = myimagecolorallocate($im, $bgcolour[0], $bgcolour[1], $bgcolour[2]);
# imagefilledrectangle($im, $x1, $y1, $x2, $y2, $bgcol);
wimagefilledpolygon($im, $points, 4, $bgcol);
}
if ($outlinecolour != [-1, -1, -1]) {
$outlinecol = myimagecolorallocate($im, $outlinecolour[0], $outlinecolour[1], $outlinecolour[2]);
# imagerectangle($im, $x1, $y1, $x2, $y2, $outlinecol);
wimagepolygon($im, $points, 4, $outlinecol);
}
$textcol = myimagecolorallocate($im, $textcolour[0], $textcolour[1], $textcolour[2]);
$this -> myimagestring($im, $font, $points[8], $points[9], $text, $textcol, $angle);
$areaname = "LINK:L" . $map -> links[$linkname] -> id . ':' . ($direction + 2);
// the rectangle is about half the size in the HTML, and easier to optimise/detect in the browser
if ($angle == 0) {
$map -> imap -> addArea("Rectangle", $areaname, '', [$x1, $y1, $x2, $y2]);
wm_debug("Adding Rectangle imagemap for $areaname\n");
} else {
$map -> imap -> addArea("Polygon", $areaname, '', $points);
wm_debug("Adding Poly imagemap for $areaname\n");
}
}
function ColourFromPercent($image, $percent, $scalename = "DEFAULT", $name = "")
{
$col = NULL;
$tag = '';
$nowarn_clipping = intval($this -> get_hint("nowarn_clipping"));
$nowarn_scalemisses = intval($this -> get_hint("nowarn_scalemisses"));
$bt = debug_backtrace();
$function = (isset($bt[1]['function']) ? $bt[1]['function'] : '');
print "$function calls ColourFromPercent\n";
exit();
if (isset($this -> colours[$scalename])) {
$colours = $this -> colours[$scalename];
if ($percent > 100) {
if ($nowarn_clipping == 0) {
wm_warn("ColourFromPercent: Clipped $name $percent% to 100% [WMWARN33]\n");
}
$percent = 100;
}
foreach ($colours as $key => $colour) {
if (($percent >= $colour['bottom']) and ($percent <= $colour['top'])) {
if (isset($colour['tag'])) {
$tag = $colour['tag'];
}
// we get called early now, so might not need to actually allocate a colour
if (isset($image)) {
if (isset($colour['red2'])) {
if ($colour["bottom"] == $colour["top"]) {
$ratio = 0;
} else {
$ratio = ($percent - $colour["bottom"]) / ($colour["top"] - $colour["bottom"]);
}
$r = $colour["red1"] + ($colour["red2"] - $colour["red1"]) * $ratio;
$g = $colour["green1"] + ($colour["green2"] - $colour["green1"]) * $ratio;
$b = $colour["blue1"] + ($colour["blue2"] - $colour["blue1"]) * $ratio;
$col = myimagecolorallocate($image, $r, $g, $b);
} else {
$r = $colour["red1"];
$g = $colour["green1"];
$b = $colour["blue1"];
$col = myimagecolorallocate($image, $r, $g, $b);
# $col = $colour['gdref1'];
}
wm_debug("CFPC $name $tag $key $r $g $b\n");
}
### warn(">>TAGS CFPC $tag\n");
return ([$col, $key, $tag]);
}
}
} else {
if ($scalename != 'none') {
wm_warn("ColourFromPercent: Attempted to use non-existent scale: $scalename for $name [WMWARN09]\n");
} else {
return [$this -> white, '', ''];
}
}
// you'll only get grey for a COMPLETELY quiet link if there's no 0 in the SCALE lines
if ($percent == 0) {
return [$this -> grey, '', ''];
}
// and you'll only get white for a link with no colour assigned
if ($nowarn_scalemisses == 0) {
wm_warn("ColourFromPercent: Scale $scalename doesn't cover $percent% for $name [WMWARN29]\n");
}
return [$this -> white, '', ''];
}
function NewColourFromPercent($value, $scalename = "DEFAULT", $name = "", $is_percent = TRUE, $scale_warning = TRUE)
{
$col = new Colour(0, 0, 0);
$tag = '';
$matchsize = NULL;
$nowarn_clipping = intval($this -> get_hint("nowarn_clipping"));
$nowarn_scalemisses = (!$scale_warning) || intval($this -> get_hint("nowarn_scalemisses"));
if (isset($this -> colours[$scalename])) {
$colours = $this -> colours[$scalename];
if ($is_percent && $value > 100) {
if ($nowarn_clipping == 0) {
wm_warn("NewColourFromPercent: Clipped $value% to 100% for item $name [WMWARN33]\n");
}
$value = 100;
}
if ($is_percent && $value < 0) {
if ($nowarn_clipping == 0) {
wm_warn("NewColourFromPercent: Clipped $value% to 0% for item $name [WMWARN34]\n");
}
$value = 0;
}
foreach ($colours as $key => $colour) {
if ((!isset($colour['special']) || $colour['special'] == 0) and ($value >= $colour['bottom']) and ($value <= $colour['top'])) {
$range = $colour['top'] - $colour['bottom'];
if (isset($colour['red2'])) {
if ($colour["bottom"] == $colour["top"]) {
$ratio = 0;
} else {
$ratio = ($value - $colour["bottom"]) / ($colour["top"] - $colour["bottom"]);
}
$r = $colour["red1"] + ($colour["red2"] - $colour["red1"]) * $ratio;
$g = $colour["green1"] + ($colour["green2"] - $colour["green1"]) * $ratio;
$b = $colour["blue1"] + ($colour["blue2"] - $colour["blue1"]) * $ratio;
} else {
$r = $colour["red1"];
$g = $colour["green1"];
$b = $colour["blue1"];
# $col = new Colour($r, $g, $b);
# $col = $colour['gdref1'];
}
// change in behaviour - with multiple matching ranges for a value, the smallest range wins
if (is_null($matchsize) || ($range < $matchsize)) {
$col = new Colour($r, $g, $b);
$matchsize = $range;
}
if (isset($colour['tag'])) {
$tag = $colour['tag'];
}
#### warn(">>NCFPC TAGS $tag\n");
wm_debug("NCFPC $name $scalename $value '$tag' $key $r $g $b\n");
return ([$col, $key, $tag]);
}
}
} else {
if ($scalename != 'none') {
wm_warn("ColourFromPercent: Attempted to use non-existent scale: $scalename for item $name [WMWARN09]\n");
} else {
return [new Colour(255, 255, 255), '', ''];
}
}
// shouldn't really get down to here if there's a complete SCALE
// you'll only get grey for a COMPLETELY quiet link if there's no 0 in the SCALE lines
if ($value == 0) {
return [new Colour(192, 192, 192), '', ''];
}
if ($nowarn_scalemisses == 0) {
wm_warn("NewColourFromPercent: Scale $scalename doesn't include a line for $value" . ($is_percent ? "%" : "") . " while drawing item $name [WMWARN29]\n");
}
// and you'll only get white for a link with no colour assigned
return [new Colour(255, 255, 255), '', ''];
}
function coloursort($a, $b)
{
if ($a['bottom'] == $b['bottom']) {
if ($a['top'] < $b['top']) {
return -1;
}
if ($a['top'] > $b['top']) {
return 1;
}
return 0;
}
if ($a['bottom'] < $b['bottom']) {
return -1;
}
return 1;
}
function FindScaleExtent($scalename = "DEFAULT")
{
$max = -999999999999999999999;
$min = -$max;
if (isset($this -> colours[$scalename])) {
$colours = $this -> colours[$scalename];
foreach ($colours as $key => $colour) {
if (!$colour['special']) {
$min = min($colour['bottom'], $min);
$max = max($colour['top'], $max);
}
}
} else {
wm_warn("FindScaleExtent: non-existent SCALE $scalename [WMWARN43]\n");
}
return [$min, $max];
}
function DrawLegend_Horizontal($im, $scalename = "DEFAULT", $width = 400)
{
$title = $this -> keytext[$scalename];
$colours = $this -> colours[$scalename];
$nscales = $this -> numscales[$scalename];
wm_debug("Drawing $nscales colours into SCALE\n");
$font = $this -> keyfont;
# $x=$this->keyx[$scalename];
# $y=$this->keyy[$scalename];
$x = 0;
$y = 0;
# $width = 400;
$scalefactor = $width / 100;
[$tilewidth, $tileheight] = $this -> myimagestringsize($font, "100%");
$box_left = $x;
# $box_left = 0;
$scale_left = $box_left + 4 + $scalefactor / 2;
$box_right = $scale_left + $width + $tilewidth + 4 + $scalefactor / 2;
$scale_right = $scale_left + $width;
$box_top = $y;
# $box_top = 0;
$scale_top = $box_top + $tileheight + 6;
$scale_bottom = $scale_top + $tileheight * 1.5;
$box_bottom = $scale_bottom + $tileheight * 2 + 6;
$scale_im = imagecreatetruecolor($box_right + 1, $box_bottom + 1);
$scale_ref = 'gdref_legend_' . $scalename;
// Start with a transparent box, in case the fill or outline colour is 'none'
imageSaveAlpha($scale_im, TRUE);
$nothing = imagecolorallocatealpha($scale_im, 128, 0, 0, 127);
imagefill($scale_im, 0, 0, $nothing);
$this -> AllocateScaleColours($scale_im, $scale_ref);
if (!is_none($this -> colours['DEFAULT']['KEYBG'])) {
wimagefilledrectangle($scale_im, $box_left, $box_top, $box_right, $box_bottom, $this -> colours['DEFAULT']['KEYBG'][$scale_ref]);
}
if (!is_none($this -> colours['DEFAULT']['KEYOUTLINE'])) {
wimagerectangle($scale_im, $box_left, $box_top, $box_right, $box_bottom, $this -> colours['DEFAULT']['KEYOUTLINE'][$scale_ref]);
}
$this -> myimagestring($scale_im, $font, $scale_left, $scale_bottom + $tileheight * 2 + 2, $title, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
for ($p = 0; $p <= 100; $p++) {
$dx = $p * $scalefactor;
if (($p % 25) == 0) {
imageline($scale_im, $scale_left + $dx, $scale_top - $tileheight, $scale_left + $dx, $scale_bottom + $tileheight, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
$labelstring = sprintf("%d%%", $p);
$this -> myimagestring($scale_im, $font, $scale_left + $dx + 2, $scale_top - 2, $labelstring, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
}
[$col, $junk] = $this -> NewColourFromPercent($p, $scalename);
if ($col -> is_real()) {
$cc = $col -> gdallocate($scale_im);
wimagefilledrectangle($scale_im, $scale_left + $dx - $scalefactor / 2, $scale_top, $scale_left + $dx + $scalefactor / 2, $scale_bottom, $cc);
}
}
imagecopy($im, $scale_im, $this -> keyx[$scalename], $this -> keyy[$scalename], 0, 0, imagesx($scale_im), imagesy($scale_im));
$this -> keyimage[$scalename] = $scale_im;
$rx = $this -> keyx[$scalename];
$ry = $this -> keyy[$scalename];
$this -> imap -> addArea("Rectangle", "LEGEND:$scalename", '', [$rx + $box_left, $ry + $box_top, $rx + $box_right, $ry + $box_bottom]);
}
function DrawLegend_Vertical($im, $scalename = "DEFAULT", $height = 400, $inverted = FALSE)
{
$title = $this -> keytext[$scalename];
$colours = $this -> colours[$scalename];
$nscales = $this -> numscales[$scalename];
wm_debug("Drawing $nscales colours into SCALE\n");
$font = $this -> keyfont;
$x = $this -> keyx[$scalename];
$y = $this -> keyy[$scalename];
# $height = 400;
$scalefactor = $height / 100;
[$tilewidth, $tileheight] = $this -> myimagestringsize($font, "100%");
# $box_left = $x;
# $box_top = $y;
$box_left = 0;
$box_top = 0;
$scale_left = $box_left + $scalefactor * 2 + 4;
$scale_right = $scale_left + $tileheight * 2;
$box_right = $scale_right + $tilewidth + $scalefactor * 2 + 4;
[$titlewidth, $titleheight] = $this -> myimagestringsize($font, $title);
if (($box_left + $titlewidth + $scalefactor * 3) > $box_right) {
$box_right = $box_left + $scalefactor * 4 + $titlewidth;
}
$scale_top = $box_top + 4 + $scalefactor + $tileheight * 2;
$scale_bottom = $scale_top + $height;
$box_bottom = $scale_bottom + $scalefactor + $tileheight / 2 + 4;
$scale_im = imagecreatetruecolor($box_right + 1, $box_bottom + 1);
$scale_ref = 'gdref_legend_' . $scalename;
// Start with a transparent box, in case the fill or outline colour is 'none'
imageSaveAlpha($scale_im, TRUE);
$nothing = imagecolorallocatealpha($scale_im, 128, 0, 0, 127);
imagefill($scale_im, 0, 0, $nothing);
$this -> AllocateScaleColours($scale_im, $scale_ref);
if (!is_none($this -> colours['DEFAULT']['KEYBG'])) {
wimagefilledrectangle($scale_im, $box_left, $box_top, $box_right, $box_bottom, $this -> colours['DEFAULT']['KEYBG']['gdref1']);
}
if (!is_none($this -> colours['DEFAULT']['KEYOUTLINE'])) {
wimagerectangle($scale_im, $box_left, $box_top, $box_right, $box_bottom, $this -> colours['DEFAULT']['KEYOUTLINE']['gdref1']);
}
$this -> myimagestring($scale_im, $font, $scale_left - $scalefactor, $scale_top - $tileheight, $title, $this -> colours['DEFAULT']['KEYTEXT']['gdref1']);
$updown = 1;
if ($inverted) {
$updown = -1;
}
for ($p = 0; $p <= 100; $p++) {
if ($inverted) {
$dy = (100 - $p) * $scalefactor;
} else {
$dy = $p * $scalefactor;
}
if (($p % 25) == 0) {
imageline($scale_im, $scale_left - $scalefactor, $scale_top + $dy, $scale_right + $scalefactor, $scale_top + $dy, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
$labelstring = sprintf("%d%%", $p);
$this -> myimagestring($scale_im, $font, $scale_right + $scalefactor * 2, $scale_top + $dy + $tileheight / 2, $labelstring, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
}
[$col, $junk] = $this -> NewColourFromPercent($p, $scalename);
if ($col -> is_real()) {
$cc = $col -> gdallocate($scale_im);
wimagefilledrectangle($scale_im, $scale_left, $scale_top + $dy - $scalefactor / 2, $scale_right, $scale_top + $dy + $scalefactor / 2, $cc);
}
}
imagecopy($im, $scale_im, $this -> keyx[$scalename], $this -> keyy[$scalename], 0, 0, imagesx($scale_im), imagesy($scale_im));
$this -> keyimage[$scalename] = $scale_im;
$rx = $this -> keyx[$scalename];
$ry = $this -> keyy[$scalename];
$this -> imap -> addArea("Rectangle", "LEGEND:$scalename", '', [$rx + $box_left, $ry + $box_top, $rx + $box_right, $ry + $box_bottom]);
}
function DrawLegend_Classic($im, $scalename = "DEFAULT", $use_tags = FALSE)
{
$title = $this -> keytext[$scalename];
$colours = $this -> colours[$scalename];
usort($colours, ["Weathermap", "coloursort"]);
$nscales = $this -> numscales[$scalename];
wm_debug("Drawing $nscales colours into SCALE\n");
$hide_zero = intval($this -> get_hint("key_hidezero_" . $scalename));
$hide_percent = intval($this -> get_hint("key_hidepercent_" . $scalename));
// did we actually hide anything?
$hid_zero = FALSE;
if (($hide_zero == 1) && isset($colours['0_0'])) {
$nscales--;
$hid_zero = TRUE;
}
$font = $this -> keyfont;
$x = $this -> keyx[$scalename];
$y = $this -> keyy[$scalename];
[$tilewidth, $tileheight] = $this -> myimagestringsize($font, "MMMM");
$tileheight = $tileheight * 1.1;
$tilespacing = $tileheight + 2;
if (($this -> keyx[$scalename] >= 0) && ($this -> keyy[$scalename] >= 0)) {
# $minwidth = imagefontwidth($font) * strlen('XX 100%-100%')+10;
# $boxwidth = imagefontwidth($font) * strlen($title) + 10;
[$minwidth, $junk] = $this -> myimagestringsize($font, 'MMMM 100%-100%');
[$minminwidth, $junk] = $this -> myimagestringsize($font, 'MMMM ');
[$boxwidth, $junk] = $this -> myimagestringsize($font, $title);
if ($use_tags) {
$max_tag = 0;
foreach ($colours as $colour) {
if (isset($colour['tag'])) {
[$w, $junk] = $this -> myimagestringsize($font, $colour['tag']);
# print $colour['tag']." $w \n";
if ($w > $max_tag) {
$max_tag = $w;
}
}
}
// now we can tweak the widths, appropriately to allow for the tag strings
# print "$max_tag > $minwidth?\n";
if (($max_tag + $minminwidth) > $minwidth) {
$minwidth = $minminwidth + $max_tag;
}
# print "minwidth is now $minwidth\n";
}
$minwidth += 10;
$boxwidth += 10;
if ($boxwidth < $minwidth) {
$boxwidth = $minwidth;
}
$boxheight = $tilespacing * ($nscales + 1) + 10;
$boxx = $x;
$boxy = $y;
$boxx = 0;
$boxy = 0;
// allow for X11-style negative positioning
if ($boxx < 0) {
$boxx += $this -> width;
}
if ($boxy < 0) {
$boxy += $this -> height;
}
$scale_im = imagecreatetruecolor($boxwidth + 1, $boxheight + 1);
$scale_ref = 'gdref_legend_' . $scalename;
// Start with a transparent box, in case the fill or outline colour is 'none'
imageSaveAlpha($scale_im, TRUE);
$nothing = imagecolorallocatealpha($scale_im, 128, 0, 0, 127);
imagefill($scale_im, 0, 0, $nothing);
$this -> AllocateScaleColours($scale_im, $scale_ref);
if (!is_none($this -> colours['DEFAULT']['KEYBG'])) {
wimagefilledrectangle($scale_im, $boxx, $boxy, $boxx + $boxwidth, $boxy + $boxheight, $this -> colours['DEFAULT']['KEYBG'][$scale_ref]);
}
if (!is_none($this -> colours['DEFAULT']['KEYOUTLINE'])) {
wimagerectangle($scale_im, $boxx, $boxy, $boxx + $boxwidth, $boxy + $boxheight, $this -> colours['DEFAULT']['KEYOUTLINE'][$scale_ref]);
}
$this -> myimagestring($scale_im, $font, $boxx + 4, $boxy + 4 + $tileheight, $title, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
$i = 1;
foreach ($colours as $colour) {
if (!isset($colour['special']) || $colour['special'] == 0) // if ( 1==1 || $colour['bottom'] >= 0)
{
// pick a value in the middle...
$value = ($colour['bottom'] + $colour['top']) / 2;
wm_debug(sprintf("%f-%f (%f) %d %d %d\n", $colour['bottom'], $colour['top'], $value, $colour['red1'], $colour['green1'], $colour['blue1']));
# debug("$i: drawing\n");
if (($hide_zero == 0) || $colour['key'] != '0_0') {
$y = $boxy + $tilespacing * $i + 8;
$x = $boxx + 6;
$fudgefactor = 0;
if ($hid_zero && $colour['bottom'] == 0) {
// calculate a small offset that can be added, which will hide the zero-value in a
// gradient, but not make the scale incorrect. A quarter of a pixel should do it.
$fudgefactor = ($colour['top'] - $colour['bottom']) / ($tilewidth * 4);
# warn("FUDGING $fudgefactor\n");
}
// if it's a gradient, red2 is defined, and we need to sweep the values
if (isset($colour['red2'])) {
for ($n = 0; $n <= $tilewidth; $n++) {
$value
= $fudgefactor + $colour['bottom'] + ($n / $tilewidth) * ($colour['top'] - $colour['bottom']);
[$ccol, $junk] = $this -> NewColourFromPercent($value, $scalename, "", FALSE);
$col = $ccol -> gdallocate($scale_im);
wimagefilledrectangle($scale_im, $x + $n, $y, $x + $n, $y + $tileheight, $col);
}
} else {
// pick a value in the middle...
//$value = ($colour['bottom'] + $colour['top']) / 2;
[$ccol, $junk] = $this -> NewColourFromPercent($value, $scalename, "", FALSE);
$col = $ccol -> gdallocate($scale_im);
wimagefilledrectangle($scale_im, $x, $y, $x + $tilewidth, $y + $tileheight, $col);
}
if ($use_tags) {
$labelstring = "";
if (isset($colour['tag'])) {
$labelstring = $colour['tag'];
}
} else {
$labelstring = sprintf("%s-%s", $colour['bottom'], $colour['top']);
if ($hide_percent == 0) {
$labelstring .= "%";
}
}
$this -> myimagestring($scale_im, $font, $x + 4 + $tilewidth, $y + $tileheight, $labelstring, $this -> colours['DEFAULT']['KEYTEXT'][$scale_ref]);
$i++;
}
imagecopy($im, $scale_im, $this -> keyx[$scalename], $this -> keyy[$scalename], 0, 0, imagesx($scale_im), imagesy($scale_im));
$this -> keyimage[$scalename] = $scale_im;
}
}
$this -> imap -> addArea("Rectangle", "LEGEND:$scalename", '', [$this -> keyx[$scalename], $this -> keyy[$scalename], $this -> keyx[$scalename] + $boxwidth, $this -> keyy[$scalename] + $boxheight]);
# $this->imap->setProp("href","#","LEGEND");
# $this->imap->setProp("extrahtml","onclick=\"position_legend();\"","LEGEND");
}
}
function DrawTimestamp($im, $font, $colour, $which = "")
{
// add a timestamp to the corner, so we can tell if it's all being updated
# $datestring = "Created: ".date("M d Y H:i:s",time());
# $this->datestamp=strftime($this->stamptext, time());
switch ($which) {
case "MIN":
$stamp = strftime($this -> minstamptext, $this -> min_data_time);
$pos_x = $this -> mintimex;
$pos_y = $this -> mintimey;
break;
case "MAX":
$stamp = strftime($this -> maxstamptext, $this -> max_data_time);
$pos_x = $this -> maxtimex;
$pos_y = $this -> maxtimey;
break;
default:
$stamp = $this -> datestamp;
$pos_x = $this -> timex;
$pos_y = $this -> timey;
break;
}
[$boxwidth, $boxheight] = $this -> myimagestringsize($font, $stamp);
$x = $this -> width - $boxwidth;
$y = $boxheight;
if (($pos_x != 0) && ($pos_y != 0)) {
$x = $pos_x;
$y = $pos_y;
}
$this -> myimagestring($im, $font, $x, $y, $stamp, $colour);
$this -> imap -> addArea("Rectangle", $which . "TIMESTAMP", '', [$x, $y, $x + $boxwidth, $y - $boxheight]);
}
function DrawTitle($im, $font, $colour)
{
$string = $this -> ProcessString($this -> title, $this);
if ($this -> get_hint('screenshot_mode') == 1) {
$string = screenshotify($string);
}
[$boxwidth, $boxheight] = $this -> myimagestringsize($font, $string);
$x = 10;
$y = $this -> titley - $boxheight;
if (($this -> titlex >= 0) && ($this -> titley >= 0)) {
$x = $this -> titlex;
$y = $this -> titley;
}
$this -> myimagestring($im, $font, $x, $y, $string, $colour);
$this -> imap -> addArea("Rectangle", "TITLE", '', [$x, $y, $x + $boxwidth, $y - $boxheight]);
}
function ReadConfig($input, $is_include = FALSE)
{
global $weathermap_error_suppress;
$curnode = NULL;
$curlink = NULL;
$matches = 0;
$nodesseen = 0;
$linksseen = 0;
$scalesseen = 0;
$last_seen = "GLOBAL";
$filename = "";
$objectlinecount = 0;
// check if $input is more than one line. if it is, it's a text of a config file
// if it isn't, it's the filename
$lines = [];
/*
if( (strchr($input,"\n")!=FALSE) || (strchr($input,"\r")!=FALSE ) )
{
wm_debug("ReadConfig Detected that this is a config fragment.\n");
// strip out any Windows line-endings that have gotten in here
$input=str_replace("\r", "", $input);
$lines = explode("/n",$input);
$filename = "{text insert}";
}
else
{
wm_debug("ReadConfig Detected that this is a config filename.\n");
$filename = $input;
if($is_include){
wm_debug("ReadConfig Detected that this is an INCLUDED config filename.\n");
if($is_include && in_array($filename, $this->included_files))
{
wm_warn("Attempt to include '$filename' twice! Skipping it.\n");
return(FALSE);
}
else
{
$this->included_files[] = $filename;
$this->has_includes = TRUE;
}
}
$fd=fopen($filename, "r");
if ($fd)
{
while (!feof($fd))
{
$buffer=fgets($fd, 4096);
// strip out any Windows line-endings that have gotten in here
$buffer=str_replace("\r", "", $buffer);
$lines[] = $buffer;
}
fclose($fd);
}
}
*/
if (dbFetchCell("SELECT COUNT(*) FROM `weathermaps` WHERE `wmap_name` = ?", [$input]) > 0) {
$config = dbFetchCell("SELECT `wmap_conf` FROM `weathermaps` WHERE `wmap_name` = ?", [$input]);
$lines = explode(PHP_EOL, $config);
} else {
print_error("Invalid Weathermap Name");
return;
}
//r($lines);
$linecount = 0;
$objectlinecount = 0;
foreach ($lines as $buffer) {
$linematched = 0;
$linecount++;
if (preg_match("/^\s*#/", $buffer)) {
// this is a comment line
} else {
$buffer = trim($buffer);
// for any other config elements that are shared between nodes and links, they can use this
unset($curobj);
$curobj = NULL;
if ($last_seen == "LINK") {
$curobj = &$curlink;
}
if ($last_seen == "NODE") {
$curobj = &$curnode;
}
if ($last_seen == "GLOBAL") {
$curobj = &$this;
}
$objectlinecount++;
#if (preg_match("/^\s*(LINK|NODE)\s+([A-Za-z][A-Za-z0-9_\.\-\:]*)\s*$/i", $buffer, $matches))
if (preg_match("/^\s*(LINK|NODE)\s+(\S+)\s*$/i", $buffer, $matches)) {
$objectlinecount = 0;
if (1 == 1) {
$this -> ReadConfig_Commit($curobj);
} else {
// first, save the previous item, before starting work on the new one
if ($last_seen == "NODE") {
$this -> nodes[$curnode -> name] = $curnode;
if ($curnode -> template == 'DEFAULT') {
$this -> node_template_tree["DEFAULT"][] = $curnode -> name;
}
wm_debug("Saving Node: " . $curnode -> name . "\n");
}
if ($last_seen == "LINK") {
if (isset($curlink -> a) && isset($curlink -> b)) {
$this -> links[$curlink -> name] = $curlink;
wm_debug("Saving Link: " . $curlink -> name . "\n");
} else {
$this -> links[$curlink -> name] = $curlink;
wm_debug("Saving Template-Only Link: " . $curlink -> name . "\n");
}
if ($curlink -> template == 'DEFAULT') {
$this -> link_template_tree["DEFAULT"][] = $curlink -> name;
}
}
}
if ($matches[1] == 'LINK') {
if ($matches[2] == 'DEFAULT') {
if ($linksseen > 0) {
wm_warn("LINK DEFAULT is not the first LINK. Defaults will not apply to earlier LINKs. [WMWARN26]\n");
}
unset($curlink);
wm_debug("Loaded LINK DEFAULT\n");
$curlink = $this -> links['DEFAULT'];
} else {
unset($curlink);
if (isset($this -> links[$matches[2]])) {
wm_warn("Duplicate link name " . $matches[2] . " at line $linecount - only the last one defined is used. [WMWARN25]\n");
}
wm_debug("New LINK " . $matches[2] . "\n");
$curlink = new WeatherMapLink;
$curlink -> name = $matches[2];
$curlink -> Reset($this);
$linksseen++;
}
$last_seen = "LINK";
$curlink -> configline = $linecount;
$linematched++;
$curobj = &$curlink;
}
if ($matches[1] == 'NODE') {
if ($matches[2] == 'DEFAULT') {
if ($nodesseen > 0) {
wm_warn("NODE DEFAULT is not the first NODE. Defaults will not apply to earlier NODEs. [WMWARN27]\n");
}
unset($curnode);
wm_debug("Loaded NODE DEFAULT\n");
$curnode = $this -> nodes['DEFAULT'];
} else {
unset($curnode);
if (isset($this -> nodes[$matches[2]])) {
wm_warn("Duplicate node name " . $matches[2] . " at line $linecount - only the last one defined is used. [WMWARN24]\n");
}
$curnode = new WeatherMapNode;
$curnode -> name = $matches[2];
$curnode -> Reset($this);
$nodesseen++;
}
$curnode -> configline = $linecount;
$last_seen = "NODE";
$linematched++;
$curobj = &$curnode;
}
# record where we first heard about this object
$curobj -> defined_in = $filename;
}
// most of the config keywords just copy stuff into object properties.
// these are all dealt with from this one array. The special-cases
// follow on from that
$config_keywords = [['LINK', '/^\s*(MAXVALUE|BANDWIDTH)\s+(\d+\.?\d*[KMGT]?)\s+(\d+\.?\d*[KMGT]?)\s*$/i', ['max_bandwidth_in_cfg' => 2, 'max_bandwidth_out_cfg' => 3]], ['LINK', '/^\s*(MAXVALUE|BANDWIDTH)\s+(\d+\.?\d*[KMGT]?)\s*$/i', ['max_bandwidth_in_cfg' => 2, 'max_bandwidth_out_cfg' => 2]], ['NODE', '/^\s*(MAXVALUE)\s+(\d+\.?\d*[KMGT]?)\s+(\d+\.?\d*[KMGT]?)\s*$/i', ['max_bandwidth_in_cfg' => 2, 'max_bandwidth_out_cfg' => 3]], ['NODE', '/^\s*(MAXVALUE)\s+(\d+\.?\d*[KMGT]?)\s*$/i', ['max_bandwidth_in_cfg' => 2, 'max_bandwidth_out_cfg' => 2]], ['GLOBAL', '/^\s*BACKGROUND\s+(.*)\s*$/i', ['background' => 1]], ['GLOBAL', '/^\s*HTMLOUTPUTFILE\s+(.*)\s*$/i', ['htmloutputfile' => 1]], ['GLOBAL', '/^\s*HTMLSTYLESHEET\s+(.*)\s*$/i', ['htmlstylesheet' => 1]], ['GLOBAL', '/^\s*IMAGEOUTPUTFILE\s+(.*)\s*$/i', ['imageoutputfile' => 1]], ['GLOBAL', '/^\s*DATAOUTPUTFILE\s+(.*)\s*$/i', ['dataoutputfile' => 1]], ['GLOBAL', '/^\s*IMAGEURI\s+(.*)\s*$/i', ['imageuri' => 1]], ['GLOBAL', '/^\s*TITLE\s+(.*)\s*$/i', ['title' => 1]], ['GLOBAL', '/^\s*HTMLSTYLE\s+(static|overlib)\s*$/i', ['htmlstyle' => 1]], ['GLOBAL', '/^\s*KEYFONT\s+(\d+)\s*$/i', ['keyfont' => 1]], ['GLOBAL', '/^\s*TITLEFONT\s+(\d+)\s*$/i', ['titlefont' => 1]], ['GLOBAL', '/^\s*TIMEFONT\s+(\d+)\s*$/i', ['timefont' => 1]], ['GLOBAL', '/^\s*TITLEPOS\s+(-?\d+)\s+(-?\d+)\s*$/i', ['titlex' => 1, 'titley' => 2]], ['GLOBAL', '/^\s*TITLEPOS\s+(-?\d+)\s+(-?\d+)\s+(.*)\s*$/i', ['titlex' => 1, 'titley' => 2, 'title' => 3]], ['GLOBAL', '/^\s*TIMEPOS\s+(-?\d+)\s+(-?\d+)\s*$/i', ['timex' => 1, 'timey' => 2]], ['GLOBAL', '/^\s*TIMEPOS\s+(-?\d+)\s+(-?\d+)\s+(.*)\s*$/i', ['timex' => 1, 'timey' => 2, 'stamptext' => 3]], ['GLOBAL', '/^\s*MINTIMEPOS\s+(-?\d+)\s+(-?\d+)\s*$/i', ['mintimex' => 1, 'mintimey' => 2]], ['GLOBAL', '/^\s*MINTIMEPOS\s+(-?\d+)\s+(-?\d+)\s+(.*)\s*$/i', ['mintimex' => 1, 'mintimey' => 2, 'minstamptext' => 3]], ['GLOBAL', '/^\s*MAXTIMEPOS\s+(-?\d+)\s+(-?\d+)\s*$/i', ['maxtimex' => 1, 'maxtimey' => 2]], ['GLOBAL', '/^\s*MAXTIMEPOS\s+(-?\d+)\s+(-?\d+)\s+(.*)\s*$/i', ['maxtimex' => 1, 'maxtimey' => 2, 'maxstamptext' => 3]], ['NODE', "/^\s*LABEL\s*$/i", ['label' => '']], # special case for blank labels
['NODE', "/^\s*LABEL\s+(.*)\s*$/i", ['label' => 1]], ['(LINK|GLOBAL)', "/^\s*WIDTH\s+(\d+)\s*$/i", ['width' => 1]], ['(LINK|GLOBAL)', "/^\s*HEIGHT\s+(\d+)\s*$/i", ['height' => 1]], ['LINK', "/^\s*WIDTH\s+(\d+\.\d+)\s*$/i", ['width' => 1]], ['LINK', '/^\s*ARROWSTYLE\s+(classic|compact)\s*$/i', ['arrowstyle' => 1]], ['LINK', '/^\s*VIASTYLE\s+(curved|angled)\s*$/i', ['viastyle' => 1]], ['LINK', '/^\s*INCOMMENT\s+(.*)\s*$/i', ['comments[IN]' => 1]], ['LINK', '/^\s*OUTCOMMENT\s+(.*)\s*$/i', ['comments[OUT]' => 1]], ['LINK', '/^\s*BWFONT\s+(\d+)\s*$/i', ['bwfont' => 1]], ['LINK', '/^\s*COMMENTFONT\s+(\d+)\s*$/i', ['commentfont' => 1]], ['LINK', '/^\s*COMMENTSTYLE\s+(edge|center)\s*$/i', ['commentstyle' => 1]], ['LINK', '/^\s*DUPLEX\s+(full|half)\s*$/i', ['duplex' => 1]], ['LINK', '/^\s*BWSTYLE\s+(classic|angled)\s*$/i', ['labelboxstyle' => 1]], ['LINK', '/^\s*LINKSTYLE\s+(twoway|oneway)\s*$/i', ['linkstyle' => 1]], ['LINK', '/^\s*BWLABELPOS\s+(\d+)\s(\d+)\s*$/i', ['labeloffset_in' => 1, 'labeloffset_out' => 2]], ['LINK', '/^\s*COMMENTPOS\s+(\d+)\s(\d+)\s*$/i', ['commentoffset_in' => 1, 'commentoffset_out' => 2]], ['LINK', '/^\s*USESCALE\s+([A-Za-z][A-Za-z0-9_]*)\s*$/i', ['usescale' => 1]], ['LINK', '/^\s*USESCALE\s+([A-Za-z][A-Za-z0-9_]*)\s+(absolute|percent)\s*$/i', ['usescale' => 1, 'scaletype' => 2]],
['LINK', '/^\s*SPLITPOS\s+(\d+)\s*$/i', ['splitpos' => 1]],
['NODE', '/^\s*LABELOFFSET\s+([-+]?\d+)\s+([-+]?\d+)\s*$/i', ['labeloffsetx' => 1, 'labeloffsety' => 2]], ['NODE', '/^\s*LABELOFFSET\s+(C|NE|SE|NW|SW|N|S|E|W)\s*$/i', ['labeloffset' => 1]], ['NODE', '/^\s*LABELOFFSET\s+((C|NE|SE|NW|SW|N|S|E|W)\d+)\s*$/i', ['labeloffset' => 1]], ['NODE', '/^\s*LABELOFFSET\s+(-?\d+r\d+)\s*$/i', ['labeloffset' => 1]],
['NODE', '/^\s*LABELFONT\s+(\d+)\s*$/i', ['labelfont' => 1]], ['NODE', '/^\s*LABELANGLE\s+(0|90|180|270)\s*$/i', ['labelangle' => 1]], # array('(NODE|LINK)', '/^\s*TEMPLATE\s+(\S+)\s*$/i', array('template'=>1)),
['LINK', '/^\s*OUTBWFORMAT\s+(.*)\s*$/i', ['bwlabelformats[OUT]' => 1, 'labelstyle' => '--']], ['LINK', '/^\s*INBWFORMAT\s+(.*)\s*$/i', ['bwlabelformats[IN]' => 1, 'labelstyle' => '--']], # array('NODE','/^\s*ICON\s+none\s*$/i',array('iconfile'=>'')),
['NODE', '/^\s*ICON\s+(\S+)\s*$/i', ['iconfile' => 1, 'iconscalew' => '#0', 'iconscaleh' => '#0']], ['NODE', '/^\s*ICON\s+(\S+)\s*$/i', ['iconfile' => 1]], ['NODE', '/^\s*ICON\s+(\d+)\s+(\d+)\s+(inpie|outpie|box|rbox|round|gauge|nink)\s*$/i', ['iconfile' => 3, 'iconscalew' => 1, 'iconscaleh' => 2]], ['NODE', '/^\s*ICON\s+(\d+)\s+(\d+)\s+(\S+)\s*$/i', ['iconfile' => 3, 'iconscalew' => 1, 'iconscaleh' => 2]],
['NODE', '/^\s*NOTES\s+(.*)\s*$/i', ['notestext[IN]' => 1, 'notestext[OUT]' => 1]], ['LINK', '/^\s*NOTES\s+(.*)\s*$/i', ['notestext[IN]' => 1, 'notestext[OUT]' => 1]], ['LINK', '/^\s*INNOTES\s+(.*)\s*$/i', ['notestext[IN]' => 1]], ['LINK', '/^\s*OUTNOTES\s+(.*)\s*$/i', ['notestext[OUT]' => 1]],
['NODE', '/^\s*INFOURL\s+(.*)\s*$/i', ['infourl[IN]' => 1, 'infourl[OUT]' => 1]], ['LINK', '/^\s*INFOURL\s+(.*)\s*$/i', ['infourl[IN]' => 1, 'infourl[OUT]' => 1]], ['LINK', '/^\s*ININFOURL\s+(.*)\s*$/i', ['infourl[IN]' => 1]], ['LINK', '/^\s*OUTINFOURL\s+(.*)\s*$/i', ['infourl[OUT]' => 1]],
['NODE', '/^\s*OVERLIBCAPTION\s+(.*)\s*$/i', ['overlibcaption[IN]' => 1, 'overlibcaption[OUT]' => 1]], ['LINK', '/^\s*OVERLIBCAPTION\s+(.*)\s*$/i', ['overlibcaption[IN]' => 1, 'overlibcaption[OUT]' => 1]], ['LINK', '/^\s*INOVERLIBCAPTION\s+(.*)\s*$/i', ['overlibcaption[IN]' => 1]], ['LINK', '/^\s*OUTOVERLIBCAPTION\s+(.*)\s*$/i', ['overlibcaption[OUT]' => 1]],
['(NODE|LINK)', "/^\s*ZORDER\s+([-+]?\d+)\s*$/i", ['zorder' => 1]], ['(NODE|LINK)', "/^\s*OVERLIBWIDTH\s+(\d+)\s*$/i", ['overlibwidth' => 1]], ['(NODE|LINK)', "/^\s*OVERLIBHEIGHT\s+(\d+)\s*$/i", ['overlibheight' => 1]], ['NODE', "/^\s*POSITION\s+([-+]?\d+)\s+([-+]?\d+)\s*$/i", ['x' => 1, 'y' => 2]], ['NODE', "/^\s*POSITION\s+(\S+)\s+([-+]?\d+)\s+([-+]?\d+)\s*$/i", ['x' => 2, 'y' => 3, 'original_x' => 2, 'original_y' => 3, 'relative_to' => 1, 'relative_resolved' => FALSE]], ['NODE', "/^\s*POSITION\s+(\S+)\s+([-+]?\d+)r(\d+)\s*$/i", ['x' => 2, 'y' => 3, 'original_x' => 2, 'original_y' => 3, 'relative_to' => 1, 'polar' => TRUE, 'relative_resolved' => FALSE]]];
// alternative for use later where quoted strings are more useful
$args = wm_parse_string($buffer);
// this loop replaces a whole pile of duplicated ifs with something with consistent handling
foreach ($config_keywords as $keyword) {
if (preg_match("/" . $keyword[0] . "/", $last_seen)) {
$statskey = $last_seen . "-" . $keyword[1];
$statskey = str_replace(['/^\s*', '\s*$/i'], ['', ''], $statskey);
if (!isset($this -> usage_stats[$statskey])) {
$this -> usage_stats[$statskey] = 0;
}
if (preg_match($keyword[1], $buffer, $matches)) {
# print "CONFIG MATCHED: ".$keyword[1]."\n";
$this -> usage_stats[$statskey]++;
foreach ($keyword[2] as $key => $val) {
// so we can poke in numbers too, if the value starts with #
// then take the # off, and treat the rest as a number literal
if (preg_match("/^#(.*)/", $val, $m)) {
$val = $m[1];
} elseif (is_numeric($val)) {
// if it's a number, then it;s a match number,
// otherwise it's a literal to be put into a variable
$val = $matches[$val];
}
assert('is_object($curobj)');
if (preg_match('/^(.*)\[([^\]]+)\]$/', $key, $m)) {
$index = constant($m[2]);
$key = $m[1];
$curobj ->{$key}[$index] = $val;
} else {
$curobj -> $key = $val;
}
}
$linematched++;
# print "\n\n";
break;
}
}
}
if (preg_match("/^\s*NODES\s+(\S+)\s+(\S+)\s*$/i", $buffer, $matches)) {
if ($last_seen == 'LINK') {
$valid_nodes = 2;
foreach ([1, 2] as $i) {
$endoffset[$i] = 'C';
$nodenames[$i] = $matches[$i];
// percentage of compass - must be first
if (preg_match("/:(NE|SE|NW|SW|N|S|E|W|C)(\d+)$/i", $matches[$i], $submatches)) {
$endoffset[$i] = $submatches[1] . $submatches[2];
$nodenames[$i] = preg_replace("/:(NE|SE|NW|SW|N|S|E|W|C)\d+$/i", '', $matches[$i]);
$this -> need_size_precalc = TRUE;
}
if (preg_match("/:(NE|SE|NW|SW|N|S|E|W|C)$/i", $matches[$i], $submatches)) {
$endoffset[$i] = $submatches[1];
$nodenames[$i] = preg_replace("/:(NE|SE|NW|SW|N|S|E|W|C)$/i", '', $matches[$i]);
$this -> need_size_precalc = TRUE;
}
if (preg_match("/:(-?\d+r\d+)$/i", $matches[$i], $submatches)) {
$endoffset[$i] = $submatches[1];
$nodenames[$i] = preg_replace("/:(-?\d+r\d+)$/i", '', $matches[$i]);
$this -> need_size_precalc = TRUE;
}
if (preg_match("/:([-+]?\d+):([-+]?\d+)$/i", $matches[$i], $submatches)) {
$xoff = $submatches[1];
$yoff = $submatches[2];
$endoffset[$i] = $xoff . ":" . $yoff;
$nodenames[$i] = preg_replace("/:$xoff:$yoff$/i", '', $matches[$i]);
$this -> need_size_precalc = TRUE;
}
if (!array_key_exists($nodenames[$i], $this -> nodes)) {
wm_warn("Unknown node '" . $nodenames[$i] . "' on line $linecount of config\n");
$valid_nodes--;
}
}
// TODO - really, this should kill the whole link, and reset for the next one
if ($valid_nodes == 2) {
$curlink -> a = $this -> nodes[$nodenames[1]];
$curlink -> b = $this -> nodes[$nodenames[2]];
$curlink -> a_offset = $endoffset[1];
$curlink -> b_offset = $endoffset[2];
} else {
// this'll stop the current link being added
$last_seen = "broken";
}
$linematched++;
}
}
if ($last_seen == 'GLOBAL' && preg_match("/^\s*INCLUDE\s+(.*)\s*$/i", $buffer, $matches)) {
if (file_exists($matches[1])) {
wm_debug("Including '{$matches[1]}'\n");
$this -> ReadConfig($matches[1], TRUE);
$last_seen = "GLOBAL";
} else {
wm_warn("INCLUDE File '{$matches[1]}' not found!\n");
}
$linematched++;
}
if (($last_seen == 'NODE' || $last_seen == 'LINK') && preg_match("/^\s*TARGET\s+(.*)\s*$/i", $buffer, $matches)) {
$linematched++;
# $targets=preg_split('/\s+/', $matches[1], -1, PREG_SPLIT_NO_EMPTY);
$rawtargetlist = $matches[1] . " ";
if ($args[0] == 'TARGET') {
// wipe any existing targets, otherwise things in the DEFAULT accumulate with the new ones
$curobj -> targets = [];
array_shift($args); // take off the actual TARGET keyword
foreach ($args as $arg) {
// we store the original TARGET string, and line number, along with the breakdown, to make nicer error messages later
// array of 7 things:
// - only 0,1,2,3,4 are used at the moment (more used to be before DS plugins)
// 0 => final target string (filled in by ReadData)
// 1 => multiplier (filled in by ReadData)
// 2 => config filename where this line appears
// 3 => linenumber in that file
// 4 => the original target string
// 5 => the plugin to use to pull data
$newtarget = ['', '', $filename, $linecount, $arg, "", ""];
if ($curobj) {
wm_debug(" TARGET: $arg\n");
$curobj -> targets[] = $newtarget;
}
}
}
}
if ($last_seen == 'LINK' && preg_match("/^\s*BWLABEL\s+(bits|percent|unformatted|none)\s*$/i", $buffer, $matches)) {
$format_in = '';
$format_out = '';
$style = strtolower($matches[1]);
if ($style == 'percent') {
$format_in = FMT_PERC_IN;
$format_out = FMT_PERC_OUT;
}
if ($style == 'bits') {
$format_in = FMT_BITS_IN;
$format_out = FMT_BITS_OUT;
}
if ($style == 'unformatted') {
$format_in = FMT_UNFORM_IN;
$format_out = FMT_UNFORM_OUT;
}
$curobj -> labelstyle = $style;
$curobj -> bwlabelformats[IN] = $format_in;
$curobj -> bwlabelformats[OUT] = $format_out;
$linematched++;
}
if (preg_match("/^\s*SET\s+(\S+)\s+(.*)\s*$/i", $buffer, $matches)) {
$curobj -> add_hint($matches[1], trim($matches[2]));
if ($curobj -> my_type() == "map" && substr($matches[1], 0, 7) == 'nowarn_') {
$weathermap_error_suppress[$matches[1]] = 1;
}
$linematched++;
}
// allow setting a variable to ""
if (preg_match("/^\s*SET\s+(\S+)\s*$/i", $buffer, $matches)) {
$curobj -> add_hint($matches[1], '');
if ($curobj -> my_type() == "map" && substr($matches[1], 0, 7) == 'nowarn_') {
$weathermap_error_suppress[$matches[1]] = 1;
}
$linematched++;
}
if (preg_match("/^\s*(IN|OUT)?OVERLIBGRAPH\s+(.+)$/i", $buffer, $matches)) {
$this -> has_overlibs = TRUE;
if ($last_seen == 'NODE' && $matches[1] != '') {
wm_warn("IN/OUTOVERLIBGRAPH make no sense for a NODE! [WMWARN42]\n");
} elseif ($last_seen == 'LINK' || $last_seen == 'NODE') {
$urls = preg_split('/\s+/', $matches[2], -1, PREG_SPLIT_NO_EMPTY);
if ($matches[1] == 'IN') {
$index = IN;
}
if ($matches[1] == 'OUT') {
$index = OUT;
}
if ($matches[1] == '') {
$curobj -> overliburl[IN] = $urls;
$curobj -> overliburl[OUT] = $urls;
} else {
$curobj -> overliburl[$index] = $urls;
}
$linematched++;
}
}
// array('(NODE|LINK)', '/^\s*TEMPLATE\s+(\S+)\s*$/i', array('template'=>1)),
if (($last_seen == 'NODE' || $last_seen == 'LINK') && preg_match("/^\s*TEMPLATE\s+(\S+)\s*$/i", $buffer, $matches)) {
$tname = $matches[1];
if (($last_seen == 'NODE' && isset($this -> nodes[$tname])) || ($last_seen == 'LINK' && isset($this -> links[$tname]))) {
$curobj -> template = $matches[1];
wm_debug("Resetting to template $last_seen " . $curobj -> template . "\n");
$curobj -> Reset($this);
if ($objectlinecount > 1) {
wm_warn("line $linecount: TEMPLATE is not first line of object. Some data may be lost. [WMWARN39]\n");
}
// build up a list of templates - this will be useful later for the tree view
if ($last_seen == 'NODE') {
$this -> node_template_tree[$tname][] = $curobj -> name;
}
if ($last_seen == 'LINK') {
$this -> link_template_tree[$tname][] = $curobj -> name;
}
} else {
wm_warn("line $linecount: $last_seen TEMPLATE '$tname' doesn't exist! (if it does exist, check it's defined first) [WMWARN40]\n");
}
$linematched++;
}
if ($last_seen == 'LINK' && preg_match("/^\s*VIA\s+([-+]?\d+)\s+([-+]?\d+)\s*$/i", $buffer, $matches)) {
$curlink -> vialist[] = [$matches[1], $matches[2]];
$linematched++;
}
if ($last_seen == 'LINK' && preg_match("/^\s*VIA\s+(\S+)\s+([-+]?\d+)\s+([-+]?\d+)\s*$/i", $buffer, $matches)) {
$curlink -> vialist[] = [$matches[2], $matches[3], $matches[1]];
$linematched++;
}
if (($last_seen == 'NODE') && preg_match("/^\s*USE(ICON)?SCALE\s+([A-Za-z][A-Za-z0-9_]*)(\s+(in|out))?(\s+(absolute|percent))?\s*$/i", $buffer, $matches)) {
$svar = '';
$stype = 'percent';
if (isset($matches[3])) {
$svar = trim($matches[3]);
}
if (isset($matches[6])) {
$stype = strtolower(trim($matches[6]));
}
// opens the door for other scaley things...
switch ($matches[1]) {
case 'ICON':
$varname = 'iconscalevar';
$uvarname = 'useiconscale';
$tvarname = 'iconscaletype';
// if(!function_exists("imagefilter"))
// {
// warn("ICON SCALEs require imagefilter, which is not present in your PHP [WMWARN040]\n");
// }
break;
default:
$varname = 'scalevar';
$uvarname = 'usescale';
$tvarname = 'scaletype';
break;
}
if ($svar != '') {
$curnode -> $varname = $svar;
}
$curnode -> $tvarname = $stype;
$curnode -> $uvarname = $matches[2];
// warn("Set $varname and $uvarname\n");
// print ">> $stype $svar ".$matches[2]." ".$curnode->name." \n";
$linematched++;
}
// one REGEXP to rule them all:
// if(preg_match("/^\s*SCALE\s+([A-Za-z][A-Za-z0-9_]*\s+)?(\d+\.?\d*)\s+(\d+\.?\d*)\s+(\d+)\s+(\d+)\s+(\d+)(?:\s+(\d+)\s+(\d+)\s+(\d+))?\s*$/i",
// 0.95b if(preg_match("/^\s*SCALE\s+([A-Za-z][A-Za-z0-9_]*\s+)?(\d+\.?\d*)\s+(\d+\.?\d*)\s+(\d+)\s+(\d+)\s+(\d+)(?:\s+(\d+)\s+(\d+)\s+(\d+))?\s*(.*)$/i",
if (preg_match("/^\s*SCALE\s+([A-Za-z][A-Za-z0-9_]*\s+)?(\-?\d+\.?\d*[munKMGT]?)\s+(\-?\d+\.?\d*[munKMGT]?)\s+(?:(\d+)\s+(\d+)\s+(\d+)(?:\s+(\d+)\s+(\d+)\s+(\d+))?|(none))\s*(.*)$/i", $buffer, $matches)) {
// The default scale name is DEFAULT
if ($matches[1] == '') {
$matches[1] = 'DEFAULT';
} else {
$matches[1] = trim($matches[1]);
}
$key = $matches[2] . '_' . $matches[3];
$this -> colours[$matches[1]][$key]['key'] = $key;
$tag = $matches[11];
$this -> colours[$matches[1]][$key]['tag'] = $tag;
$this -> colours[$matches[1]][$key]['bottom'] = unwm_format_number($matches[2], $this -> kilo);
$this -> colours[$matches[1]][$key]['top'] = unwm_format_number($matches[3], $this -> kilo);
$this -> colours[$matches[1]][$key]['special'] = 0;
if (isset($matches[10]) && $matches[10] == 'none') {
$this -> colours[$matches[1]][$key]['red1'] = -1;
$this -> colours[$matches[1]][$key]['green1'] = -1;
$this -> colours[$matches[1]][$key]['blue1'] = -1;
} else {
$this -> colours[$matches[1]][$key]['red1'] = (int)($matches[4]);
$this -> colours[$matches[1]][$key]['green1'] = (int)($matches[5]);
$this -> colours[$matches[1]][$key]['blue1'] = (int)($matches[6]);
}
// this is the second colour, if there is one
if (isset($matches[7]) && $matches[7] != '') {
$this -> colours[$matches[1]][$key]['red2'] = (int)($matches[7]);
$this -> colours[$matches[1]][$key]['green2'] = (int)($matches[8]);
$this -> colours[$matches[1]][$key]['blue2'] = (int)($matches[9]);
}
if (!isset($this -> numscales[$matches[1]])) {
$this -> numscales[$matches[1]] = 1;
} else {
$this -> numscales[$matches[1]]++;
}
// we count if we've seen any default scale, otherwise, we have to add
// one at the end.
if ($matches[1] == 'DEFAULT') {
$scalesseen++;
}
$linematched++;
}
if (preg_match("/^\s*KEYPOS\s+([A-Za-z][A-Za-z0-9_]*\s+)?(-?\d+)\s+(-?\d+)(.*)/i", $buffer, $matches)) {
$whichkey = trim($matches[1]);
if ($whichkey == '') {
$whichkey = 'DEFAULT';
}
$this -> keyx[$whichkey] = $matches[2];
$this -> keyy[$whichkey] = $matches[3];
$extra = trim($matches[4]);
if ($extra != '') {
$this -> keytext[$whichkey] = $extra;
}
if (!isset($this -> keytext[$whichkey])) {
$this -> keytext[$whichkey] = "DEFAULT TITLE";
}
if (!isset($this -> keystyle[$whichkey])) {
$this -> keystyle[$whichkey] = "classic";
}
$linematched++;
}
// truetype font definition (actually, we don't really check if it's truetype) - filename + size
if (preg_match("/^\s*FONTDEFINE\s+(\d+)\s+(\S+)\s+(\d+)\s*$/i", $buffer, $matches)) {
if (function_exists("imagettfbbox")) {
// test if this font is valid, before adding it to the font table...
$bounds = @imagettfbbox($matches[3], 0, $matches[2], "Ignore me");
if (isset($bounds[0])) {
$this -> fonts[$matches[1]] = new WMFont();
$this -> fonts[$matches[1]] -> type = "truetype";
$this -> fonts[$matches[1]] -> file = $matches[2];
$this -> fonts[$matches[1]] -> size = $matches[3];
} else {
wm_warn("Failed to load ttf font " . $matches[2] . " - at config line $linecount [WMWARN30]");
}
} else {
wm_warn("imagettfbbox() is not a defined function. You don't seem to have FreeType compiled into your gd module. [WMWARN31]\n");
}
$linematched++;
}
// GD font definition (no size here)
if (preg_match("/^\s*FONTDEFINE\s+(\d+)\s+(\S+)\s*$/i", $buffer, $matches)) {
$newfont = imageloadfont($matches[2]);
if ($newfont) {
$this -> fonts[$matches[1]] = new WMFont();
$this -> fonts[$matches[1]] -> type = "gd";
$this -> fonts[$matches[1]] -> file = $matches[2];
$this -> fonts[$matches[1]] -> gdnumber = $newfont;
} else {
wm_warn("Failed to load GD font: " . $matches[2] . " ($newfont) at config line $linecount [WMWARN32]\n");
}
$linematched++;
}
if (preg_match("/^\s*KEYSTYLE\s+([A-Za-z][A-Za-z0-9_]+\s+)?(classic|horizontal|vertical|inverted|tags)\s?(\d+)?\s*$/i", $buffer, $matches)) {
$whichkey = trim($matches[1]);
if ($whichkey == '') {
$whichkey = 'DEFAULT';
}
$this -> keystyle[$whichkey] = strtolower($matches[2]);
if (isset($matches[3]) && $matches[3] != '') {
$this -> keysize[$whichkey] = $matches[3];
} else {
$this -> keysize[$whichkey] = $this -> keysize['DEFAULT'];
}
$linematched++;
}
if (preg_match("/^\s*KILO\s+(\d+)\s*$/i", $buffer, $matches)) {
$this -> kilo = $matches[1];
# $this->defaultlink->owner->kilo=$matches[1];
# $this->links['DEFAULT']=$matches[1];
$linematched++;
}
if (preg_match("/^\s*(TIME|TITLE|KEYBG|KEYTEXT|KEYOUTLINE|BG)COLOR\s+((\d+)\s+(\d+)\s+(\d+)|none)\s*$/i", $buffer, $matches)) {
$key = $matches[1];
$val = strtolower($matches[2]);
# "Found colour line for $key\n";
if (isset($matches[3])) // this is a regular colour setting thing
{
$this -> colours['DEFAULT'][$key]['red1'] = $matches[3];
$this -> colours['DEFAULT'][$key]['green1'] = $matches[4];
$this -> colours['DEFAULT'][$key]['blue1'] = $matches[5];
$this -> colours['DEFAULT'][$key]['bottom'] = -2;
$this -> colours['DEFAULT'][$key]['top'] = -1;
$this -> colours['DEFAULT'][$key]['special'] = 1;
$linematched++;
}
if ($val == 'none' && ($matches[1] == 'KEYBG' || $matches[1] == 'KEYOUTLINE')) {
$this -> colours['DEFAULT'][$key]['red1'] = -1;
$this -> colours['DEFAULT'][$key]['green1'] = -1;
$this -> colours['DEFAULT'][$key]['blue1'] = -1;
$this -> colours['DEFAULT'][$key]['bottom'] = -2;
$this -> colours['DEFAULT'][$key]['top'] = -1;
$this -> colours['DEFAULT'][$key]['special'] = 1;
$linematched++;
}
}
if (($last_seen == 'NODE') && (preg_match("/^\s*(AICONOUTLINE|AICONFILL|LABELFONT|LABELFONTSHADOW|LABELBG|LABELOUTLINE)COLOR\s+((\d+)\s+(\d+)\s+(\d+)|none|contrast|copy)\s*$/i", $buffer, $matches))) {
$key = $matches[1];
$field = strtolower($matches[1]) . 'colour';
$val = strtolower($matches[2]);
if (isset($matches[3])) // this is a regular colour setting thing
{
$curnode -> $field = [$matches[3], $matches[4], $matches[5]];
$linematched++;
}
if ($val == 'none' && ($matches[1] == 'LABELFONTSHADOW' || $matches[1] == 'LABELBG' || $matches[1] == 'LABELOUTLINE' || $matches[1] == 'AICONFILL' || $matches[1] == 'AICONOUTLINE')) {
$curnode -> $field = [-1, -1, -1];
$linematched++;
}
if ($val == 'contrast' && $matches[1] == 'LABELFONT') {
$curnode -> $field = [-3, -3, -3];
$linematched++;
}
if ($matches[2] == 'copy' && $matches[1] == 'AICONFILL') {
$curnode -> $field = [-2, -2, -2];
$linematched++;
}
}
if (($last_seen == 'LINK') && (preg_match("/^\s*(COMMENTFONT|BWBOX|BWFONT|BWOUTLINE|OUTLINE)COLOR\s+((\d+)\s+(\d+)\s+(\d+)|none|contrast|copy)\s*$/i", $buffer, $matches))) {
$key = $matches[1];
$field = strtolower($matches[1]) . 'colour';
$val = strtolower($matches[2]);
if (isset($matches[3])) // this is a regular colour setting thing
{
$curlink -> $field = [$matches[3], $matches[4], $matches[5]];
$linematched++;
}
if ($val == 'none' && ($key == 'BWBOX' || $key == 'BWOUTLINE' || $key == 'OUTLINE' || $key == 'KEYOUTLINE' || $key == 'KEYBG')) {
// print "***********************************\n";
$curlink -> $field = [-1, -1, -1];
$linematched++;
}
if ($val == 'contrast' && $key == 'COMMENTFONT') {
// print "***********************************\n";
$curlink -> $field = [-3, -3, -3];
$linematched++;
}
}
if ($last_seen == 'LINK' && preg_match("/^\s*ARROWSTYLE\s+(\d+)\s+(\d+)\s*$/i", $buffer, $matches)) {
$curlink -> arrowstyle = $matches[1] . ' ' . $matches[2];
$linematched++;
}
if ($linematched == 0 && trim($buffer) != '') {
wm_warn("Unrecognised config on line $linecount: $buffer\n");
}
if ($linematched > 1) {
wm_warn("Same line ($linecount) interpreted twice. This is a program error. Please report to Howie with your config!\nThe line was: $buffer");
}
} // if blankline
} // while
if (1 == 1) {
$this -> ReadConfig_Commit($curobj);
} else {
if ($last_seen == "NODE") {
$this -> nodes[$curnode -> name] = $curnode;
wm_debug("Saving Node: " . $curnode -> name . "\n");
if ($curnode -> template == 'DEFAULT') {
$this -> node_template_tree["DEFAULT"][] = $curnode -> name;
}
}
if ($last_seen == "LINK") {
if (isset($curlink -> a) && isset($curlink -> b)) {
$this -> links[$curlink -> name] = $curlink;
wm_debug("Saving Link: " . $curlink -> name . "\n");
if ($curlink -> template == 'DEFAULT') {
$this -> link_template_tree["DEFAULT"][] = $curlink -> name;
}
} else {
wm_warn("Dropping LINK " . $curlink -> name . " - it hasn't got 2 NODES!");
}
}
}
wm_debug("ReadConfig has finished reading the config ($linecount lines)\n");
wm_debug("------------------------------------------\n");
// load some default colouring, otherwise it all goes wrong
if ($scalesseen == 0) {
wm_debug("Adding default SCALE colour set (no SCALE lines seen).\n");
$defaults = ['0_0' => ['bottom' => 0, 'top' => 0, 'red1' => 192, 'green1' => 192, 'blue1' => 192, 'special' => 0], '0_1' => ['bottom' => 0, 'top' => 1, 'red1' => 255, 'green1' => 255, 'blue1' => 255, 'special' => 0], '1_10' => ['bottom' => 1, 'top' => 10, 'red1' => 140, 'green1' => 0, 'blue1' => 255, 'special' => 0], '10_25' => ['bottom' => 10, 'top' => 25, 'red1' => 32, 'green1' => 32, 'blue1' => 255, 'special' => 0], '25_40' => ['bottom' => 25, 'top' => 40, 'red1' => 0, 'green1' => 192, 'blue1' => 255, 'special' => 0], '40_55' => ['bottom' => 40, 'top' => 55, 'red1' => 0, 'green1' => 240, 'blue1' => 0, 'special' => 0], '55_70' => ['bottom' => 55, 'top' => 70, 'red1' => 240, 'green1' => 240, 'blue1' => 0, 'special' => 0], '70_85' => ['bottom' => 70, 'top' => 85, 'red1' => 255, 'green1' => 192, 'blue1' => 0, 'special' => 0], '85_100' => ['bottom' => 85, 'top' => 100, 'red1' => 255, 'green1' => 0, 'blue1' => 0, 'special' => 0]];
foreach ($defaults as $key => $def) {
$this -> colours['DEFAULT'][$key] = $def;
$this -> colours['DEFAULT'][$key]['key'] = $key;
$scalesseen++;
}
// we have a 0-0 line now, so we need to hide that.
$this -> add_hint("key_hidezero_DEFAULT", 1);
} else {
wm_debug("Already have $scalesseen scales, no defaults added.\n");
}
$this -> numscales['DEFAULT'] = $scalesseen;
$this -> configfile = "$filename";
if ($this -> has_overlibs && $this -> htmlstyle == 'static') {
wm_warn("OVERLIBGRAPH is used, but HTMLSTYLE is static. This is probably wrong. [WMWARN41]\n");
}
wm_debug("Building cache of z-layers and finalising bandwidth.\n");
// $allitems = array_merge($this->links, $this->nodes);
$allitems = [];
foreach ($this -> nodes as $node) {
$allitems[] = $node;
}
foreach ($this -> links as $link) {
$allitems[] = $link;
}
# foreach ($allitems as &$item)
foreach ($allitems as $ky => $vl) {
$item =& $allitems[$ky];
$z = $item -> zorder;
if (!isset($this -> seen_zlayers[$z]) || !is_array($this -> seen_zlayers[$z])) {
$this -> seen_zlayers[$z] = [];
}
array_push($this -> seen_zlayers[$z], $item);
// while we're looping through, let's set the real bandwidths
if ($item -> my_type() == "LINK") {
$this -> links[$item -> name] -> max_bandwidth_in = unwm_format_number($item -> max_bandwidth_in_cfg, $this -> kilo);
$this -> links[$item -> name] -> max_bandwidth_out = unwm_format_number($item -> max_bandwidth_out_cfg, $this -> kilo);
} elseif ($item -> my_type() == "NODE") {
$this -> nodes[$item -> name] -> max_bandwidth_in = unwm_format_number($item -> max_bandwidth_in_cfg, $this -> kilo);
$this -> nodes[$item -> name] -> max_bandwidth_out = unwm_format_number($item -> max_bandwidth_out_cfg, $this -> kilo);
} else {
wm_warn("Internal bug - found an item of type: " . $item -> my_type() . "\n");
}
// $item->max_bandwidth_in=unwm_format_number($item->max_bandwidth_in_cfg, $this->kilo);
// $item->max_bandwidth_out=unwm_format_number($item->max_bandwidth_out_cfg, $this->kilo);
wm_debug(sprintf(" Setting bandwidth on " . $item -> my_type() . " $item->name (%s -> %d bps, %s -> %d bps, KILO = %d)\n", $item -> max_bandwidth_in_cfg, $item -> max_bandwidth_in, $item -> max_bandwidth_out_cfg, $item -> max_bandwidth_out, $this -> kilo));
}
wm_debug("Found " . sizeof($this -> seen_zlayers) . " z-layers including builtins (0,100).\n");
// calculate any relative positions here - that way, nothing else
// really needs to know about them
wm_debug("Resolving relative positions for NODEs...\n");
// safety net for cyclic dependencies
$i = 100;
do {
$skipped = 0;
$set = 0;
foreach ($this -> nodes as $node) {
if (($node -> relative_to != '') && (!$node -> relative_resolved)) {
wm_debug("Resolving relative position for NODE " . $node -> name . " to " . $node -> relative_to . "\n");
if (array_key_exists($node -> relative_to, $this -> nodes)) {
// check if we are relative to another node which is in turn relative to something
// we need to resolve that one before we can resolve this one!
if (($this -> nodes[$node -> relative_to] -> relative_to != '') && (!$this -> nodes[$node -> relative_to] -> relative_resolved)) {
wm_debug("Skipping unresolved relative_to. Let's hope it's not a circular one\n");
$skipped++;
} else {
$rx = $this -> nodes[$node -> relative_to] -> x;
$ry = $this -> nodes[$node -> relative_to] -> y;
if ($node -> polar) {
// treat this one as a POLAR relative coordinate.
// - draw rings around a node!
$angle = $node -> x;
$distance = $node -> y;
$newpos_x = $rx + $distance * sin(deg2rad($angle));
$newpos_y = $ry - $distance * cos(deg2rad($angle));
wm_debug("->$newpos_x,$newpos_y\n");
$this -> nodes[$node -> name] -> x = $newpos_x;
$this -> nodes[$node -> name] -> y = $newpos_y;
$this -> nodes[$node -> name] -> relative_resolved = TRUE;
$set++;
} else {
// save the relative coords, so that WriteConfig can work
// resolve the relative stuff
$newpos_x = $rx + $this -> nodes[$node -> name] -> x;
$newpos_y = $ry + $this -> nodes[$node -> name] -> y;
wm_debug("->$newpos_x,$newpos_y\n");
$this -> nodes[$node -> name] -> x = $newpos_x;
$this -> nodes[$node -> name] -> y = $newpos_y;
$this -> nodes[$node -> name] -> relative_resolved = TRUE;
$set++;
}
}
} else {
wm_warn("NODE " . $node -> name . " has a relative position to an unknown node! [WMWARN10]\n");
}
}
}
wm_debug("Relative Positions Cycle $i - set $set and Skipped $skipped for unresolved dependencies\n");
$i--;
} while (($set > 0) && ($i != 0));
if ($skipped > 0) {
wm_warn("There are Circular dependencies in relative POSITION lines for $skipped nodes. [WMWARN11]\n");
}
wm_debug("-----------------------------------\n");
wm_debug("Running Pre-Processing Plugins...\n");
foreach ($this -> preprocessclasses as $pre_class) {
wm_debug("Running $pre_class" . "->run()\n");
$this -> plugins['pre'][$pre_class] -> run($this);
}
wm_debug("Finished Pre-Processing Plugins...\n");
return (TRUE);
}
function ReadConfig_Commit(&$curobj)
{
if (is_null($curobj)) {
return;
}
$last_seen = $curobj -> my_type();
// first, save the previous item, before starting work on the new one
if ($last_seen == "NODE") {
$this -> nodes[$curobj -> name] = $curobj;
wm_debug("Saving Node: " . $curobj -> name . "\n");
if ($curobj -> template == 'DEFAULT') {
$this -> node_template_tree["DEFAULT"][] = $curobj -> name;
}
}
if ($last_seen == "LINK") {
if (isset($curobj -> a) && isset($curobj -> b)) {
$this -> links[$curobj -> name] = $curobj;
wm_debug("Saving Link: " . $curobj -> name . "\n");
} else {
$this -> links[$curobj -> name] = $curobj;
wm_debug("Saving Template-Only Link: " . $curobj -> name . "\n");
}
if ($curobj -> template == 'DEFAULT') {
$this -> link_template_tree["DEFAULT"][] = $curobj -> name;
}
}
}
function WriteDataFile($filename)
{
if ($filename != "") {
$fd = fopen($filename, 'w');
# $output = '';
if ($fd) {
foreach ($this -> nodes as $node) {
if (!preg_match('/^::\s/', $node -> name) && sizeof($node -> targets) > 0) {
fputs($fd, sprintf("N_%s\t%f\t%f\r\n", $node -> name, $node -> bandwidth_in, $node -> bandwidth_out));
}
}
foreach ($this -> links as $link) {
if (!preg_match('/^::\s/', $link -> name) && sizeof($link -> targets) > 0) {
fputs($fd, sprintf("L_%s\t%f\t%f\r\n", $link -> name, $link -> bandwidth_in, $link -> bandwidth_out));
}
}
fclose($fd);
}
}
}
function WriteConfig($filename)
{
global $WEATHERMAP_VERSION;
$output = "";
//if ($fd) {
$output .= "# Automatically generated by php-weathermapp v$WEATHERMAP_VERSION\n\n";
if (count($this -> fonts) > 0) {
foreach ($this -> fonts as $fontnumber => $font) {
if ($font -> type == 'truetype') {
$output .= sprintf("FONTDEFINE %d %s %d\n", $fontnumber, $font -> file, $font -> size);
}
if ($font -> type == 'gd') {
$output .= sprintf("FONTDEFINE %d %s\n", $fontnumber, $font -> file);
}
}
$output .= "\n";
}
$basic_params = [['background', 'BACKGROUND', CONFIG_TYPE_LITERAL], ['width', 'WIDTH', CONFIG_TYPE_LITERAL], ['height', 'HEIGHT', CONFIG_TYPE_LITERAL], ['htmlstyle', 'HTMLSTYLE', CONFIG_TYPE_LITERAL], ['kilo', 'KILO', CONFIG_TYPE_LITERAL], ['keyfont', 'KEYFONT', CONFIG_TYPE_LITERAL], ['timefont', 'TIMEFONT', CONFIG_TYPE_LITERAL], ['titlefont', 'TITLEFONT', CONFIG_TYPE_LITERAL], ['title', 'TITLE', CONFIG_TYPE_LITERAL], ['htmloutputfile', 'HTMLOUTPUTFILE', CONFIG_TYPE_LITERAL], ['dataoutputfile', 'DATAOUTPUTFILE', CONFIG_TYPE_LITERAL], ['htmlstylesheet', 'HTMLSTYLESHEET', CONFIG_TYPE_LITERAL], ['imageuri', 'IMAGEURI', CONFIG_TYPE_LITERAL], ['imageoutputfile', 'IMAGEOUTPUTFILE', CONFIG_TYPE_LITERAL]];
foreach ($basic_params as $param) {
$field = $param[0];
$keyword = $param[1];
if ($this -> $field != $this -> inherit_fieldlist[$field]) {
if ($param[2] == CONFIG_TYPE_COLOR) {
$output .= "$keyword " . render_colour($this -> $field) . "\n";
}
if ($param[2] == CONFIG_TYPE_LITERAL) {
$output .= "$keyword " . $this -> $field . "\n";
}
}
}
if (($this -> timex != $this -> inherit_fieldlist['timex']) || ($this -> timey != $this -> inherit_fieldlist['timey']) || ($this -> stamptext != $this -> inherit_fieldlist['stamptext'])) {
$output .= "TIMEPOS " . $this -> timex . " " . $this -> timey . " " . $this -> stamptext . "\n";
}
if (($this -> mintimex != $this -> inherit_fieldlist['mintimex']) || ($this -> mintimey != $this -> inherit_fieldlist['mintimey']) || ($this -> minstamptext != $this -> inherit_fieldlist['minstamptext'])) {
$output .= "MINTIMEPOS " . $this -> mintimex . " " . $this -> mintimey . " " . $this -> minstamptext . "\n";
}
if (($this -> maxtimex != $this -> inherit_fieldlist['maxtimex']) || ($this -> maxtimey != $this -> inherit_fieldlist['maxtimey']) || ($this -> maxstamptext != $this -> inherit_fieldlist['maxstamptext'])) {
$output .= "MAXTIMEPOS " . $this -> maxtimex . " " . $this -> maxtimey . " " . $this -> maxstamptext . "\n";
}
if (($this -> titlex != $this -> inherit_fieldlist['titlex']) || ($this -> titley != $this -> inherit_fieldlist['titley'])) {
$output .= "TITLEPOS " . $this -> titlex . " " . $this -> titley . "\n";
}
$output .= "\n";
foreach ($this -> colours as $scalename => $colours) {
// not all keys will have keypos but if they do, then all three vars should be defined
if ((isset($this -> keyx[$scalename])) && (isset($this -> keyy[$scalename])) && (isset($this -> keytext[$scalename])) && (($this -> keytext[$scalename] != $this -> inherit_fieldlist['keytext']) || ($this -> keyx[$scalename] != $this -> inherit_fieldlist['keyx']) || ($this -> keyy[$scalename] != $this -> inherit_fieldlist['keyy']))) {
// sometimes a scale exists but without defaults. A proper scale object would sort this out...
if ($this -> keyx[$scalename] == '') {
$this -> keyx[$scalename] = -1;
}
if ($this -> keyy[$scalename] == '') {
$this -> keyy[$scalename] = -1;
}
$output .= "KEYPOS " . $scalename . " " . $this -> keyx[$scalename] . " " . $this -> keyy[$scalename] . " " . $this -> keytext[$scalename] . "\n";
}
if ((isset($this -> keystyle[$scalename])) && ($this -> keystyle[$scalename] != $this -> inherit_fieldlist['keystyle']['DEFAULT'])) {
$extra = '';
if ((isset($this -> keysize[$scalename])) && ($this -> keysize[$scalename] != $this -> inherit_fieldlist['keysize']['DEFAULT'])) {
$extra = " " . $this -> keysize[$scalename];
}
$output .= "KEYSTYLE " . $scalename . " " . $this -> keystyle[$scalename] . $extra . "\n";
}
$locale = localeconv();
$decimal_point = $locale['decimal_point'];
foreach ($colours as $k => $colour) {
if (!isset($colour['special']) || !$colour['special']) {
$top = rtrim(rtrim(sprintf("%f", $colour['top']), "0"), $decimal_point);
$bottom = rtrim(rtrim(sprintf("%f", $colour['bottom']), "0"), $decimal_point);
if ($bottom > 1000) {
$bottom = nice_bandwidth($colour['bottom'], $this -> kilo);
}
if ($top > 1000) {
$top = nice_bandwidth($colour['top'], $this -> kilo);
}
$tag = (isset($colour['tag']) ? $colour['tag'] : '');
if (($colour['red1'] == -1) && ($colour['green1'] == -1) && ($colour['blue1'] == -1)) {
$output .= sprintf("SCALE %s %-4s %-4s none %s\n", $scalename, $bottom, $top, $tag);
} elseif (!isset($colour['red2'])) {
$output .= sprintf("SCALE %s %-4s %-4s %3d %3d %3d %s\n", $scalename, $bottom, $top, $colour['red1'], $colour['green1'], $colour['blue1'], $tag);
} else {
$output .= sprintf("SCALE %s %-4s %-4s %3d %3d %3d %3d %3d %3d %s\n", $scalename, $bottom, $top, $colour['red1'], $colour['green1'], $colour['blue1'], $colour['red2'], $colour['green2'], $colour['blue2'], $tag);
}
} else {
if (($colour['red1'] == -1) && ($colour['green1'] == -1) && ($colour['blue1'] == -1)) {
$output .= sprintf("%sCOLOR none\n", $k);
} else {
$output .= sprintf("%sCOLOR %d %d %d\n", $k, $colour['red1'], $colour['green1'], $colour['blue1']);
}
}
}
$output .= "\n";
}
foreach ($this -> hints as $hintname => $hint) {
$output .= "SET $hintname $hint\n";
}
// this doesn't really work right, but let's try anyway
if ($this -> has_includes) {
$output .= "\n# Included files\n";
foreach ($this -> included_files as $ifile) {
$output .= "INCLUDE $ifile\n";
}
}
$output .= "\n# End of global section\n\n";
foreach (["template", "normal"] as $which) {
if ($which == "template") {
$output .= "\n# TEMPLATE-only NODEs:\n";
}
if ($which == "normal") {
$output .= "\n# regular NODEs:\n";
}
foreach ($this -> nodes as $node) {
if (!preg_match("/^::\s/", $node -> name)) {
if ($node -> defined_in == $this -> configfile) {
if ($which == "template" && $node -> x === NULL) {
wm_debug("TEMPLATE\n");
$output .= $node -> WriteConfig();
}
if ($which == "normal" && $node -> x !== NULL) {
$output .= $node -> WriteConfig();
}
}
}
}
if ($which == "template") {
$output .= "\n# TEMPLATE-only LINKs:\n";
}
if ($which == "normal") {
$output .= "\n# regular LINKs:\n";
}
foreach ($this -> links as $link) {
if (!preg_match('/^::\s/', $link -> name)) {
if ($link -> defined_in == $this -> configfile) {
if ($which == "template" && $link -> a === NULL) {
$output .= $link -> WriteConfig();
}
if ($which == "normal" && $link -> a !== NULL) {
$output .= $link -> WriteConfig();
}
}
}
}
}
//} else {
// wm_warn("Couldn't open config file $filename for writing");
// return (false);
//}
dbUpdate(['wmap_conf' => $output], 'weathermaps', "`wmap_name` = ?", [$filename]);
return (TRUE);
}
// pre-allocate colour slots for the colours used by the arrows
// this way, it's the pretty icons that suffer if there aren't enough colours, and
// not the actual useful data
// we skip any gradient scales
function AllocateScaleColours($im, $refname = 'gdref1')
{
# $colours=$this->colours['DEFAULT'];
foreach ($this -> colours as $scalename => $colours) {
foreach ($colours as $key => $colour) {
if ((!isset($this -> colours[$scalename][$key]['red2'])) && (!isset($this -> colours[$scalename][$key][$refname]))) {
$r = $colour['red1'];
$g = $colour['green1'];
$b = $colour['blue1'];
wm_debug("AllocateScaleColours: $scalename/$refname $key ($r,$g,$b)\n");
$this -> colours[$scalename][$key][$refname] = myimagecolorallocate($im, $r, $g, $b);
}
}
}
}
function DrawMap($filename = '', $thumbnailfile = '', $thumbnailmax = 250, $withnodes = TRUE, $use_via_overlay = FALSE, $use_rel_overlay = FALSE)
{
global $vars;
wm_debug("Trace: DrawMap()\n");
metadump("# start", TRUE);
$bgimage = NULL;
if ($this -> configfile != "") {
$this -> cachefile_version = crc32(file_get_contents($this -> configfile));
} else {
$this -> cachefile_version = crc32("........");
}
wm_debug("Running Post-Processing Plugins...\n");
foreach ($this -> postprocessclasses as $post_class) {
wm_debug("Running $post_class" . "->run()\n");
//call_user_func_array(array($post_class, 'run'), array(&$this));
$this -> plugins['post'][$post_class] -> run($this);
}
wm_debug("Finished Post-Processing Plugins...\n");
wm_debug("=====================================\n");
wm_debug("Start of Map Drawing\n");
// if we're running tests, we force the time to a particular value,
// so the output can be compared to a reference image more easily
$testmode = intval($this -> get_hint("testmode"));
if ($testmode == 1) {
$maptime = 1270813792;
date_default_timezone_set('UTC');
} else {
$maptime = time();
}
$this -> datestamp = strftime($this -> stamptext, $maptime);
// do the basic prep work
if ($this -> background != '') {
if (is_readable($this -> background)) {
$bgimage = imagecreatefromfile($this -> background);
if (!$bgimage) {
wm_warn("Failed to open background image. One possible reason: Is your BACKGROUND really a PNG?\n");
} else {
$this -> width = imagesx($bgimage);
$this -> height = imagesy($bgimage);
}
} else {
wm_warn("Your background image file could not be read. Check the filename, and permissions, for " . $this -> background . "\n");
}
}
$image = wimagecreatetruecolor($this -> width, $this -> height);
imageantialias($image, TRUE);
# $image = imagecreate($this->width, $this->height);
if (!$image) {
wm_warn("Couldn't create output image in memory (" . $this -> width . "x" . $this -> height . ").");
} else {
ImageAlphaBlending($image, TRUE);
if ($this -> get_hint("antialias") == 1) {
// Turn on anti-aliasing if it exists and it was requested
if (function_exists("imageantialias")) {
imageantialias($image, TRUE);
}
}
// by here, we should have a valid image handle
// save this away, now
$this -> image = $image;
$this -> white = myimagecolorallocate($image, 255, 255, 255);
$this -> black = myimagecolorallocate($image, 0, 0, 0);
$this -> grey = myimagecolorallocate($image, 192, 192, 192);
$this -> selected = myimagecolorallocate($image, 255, 0, 0); // for selections in the editor
$this -> AllocateScaleColours($image);
// fill with background colour anyway, in case the background image failed to load
wimagefilledrectangle($image, 0, 0, $this -> width, $this -> height, $this -> colours['DEFAULT']['BG']['gdref1']);
if ($bgimage) {
imagecopy($image, $bgimage, 0, 0, 0, 0, $this -> width, $this -> height);
imagedestroy($bgimage);
}
// Now it's time to draw a map
// do the node rendering stuff first, regardless of where they are actually drawn.
// this is so we can get the size of the nodes, which links will need if they use offsets
foreach ($this -> nodes as $node) {
// don't try and draw template nodes
wm_debug("Pre-rendering " . $node -> name . " to get bounding boxes.\n");
if (!is_null($node -> x)) {
$this -> nodes[$node -> name] -> pre_render($image, $this);
}
}
$all_layers = array_keys($this -> seen_zlayers);
sort($all_layers);
foreach ($all_layers as $z) {
$z_items = $this -> seen_zlayers[$z];
wm_debug("Drawing layer $z\n");
// all the map 'furniture' is fixed at z=1000
if ($z == 1000) {
foreach ($this -> colours as $scalename => $colours) {
wm_debug("Drawing KEY for $scalename if necessary.\n");
if ((isset($this -> numscales[$scalename])) && (isset($this -> keyx[$scalename])) && ($this -> keyx[$scalename] >= 0) && ($this -> keyy[$scalename] >= 0)) {
if ($this -> keystyle[$scalename] == 'classic') {
$this -> DrawLegend_Classic($image, $scalename, FALSE);
}
if ($this -> keystyle[$scalename] == 'horizontal') {
$this -> DrawLegend_Horizontal($image, $scalename, $this -> keysize[$scalename]);
}
if ($this -> keystyle[$scalename] == 'vertical') {
$this -> DrawLegend_Vertical($image, $scalename, $this -> keysize[$scalename]);
}
if ($this -> keystyle[$scalename] == 'inverted') {
$this -> DrawLegend_Vertical($image, $scalename, $this -> keysize[$scalename], TRUE);
}
if ($this -> keystyle[$scalename] == 'tags') {
$this -> DrawLegend_Classic($image, $scalename, TRUE);
}
}
}
$this -> DrawTimestamp($image, $this -> timefont, $this -> colours['DEFAULT']['TIME']['gdref1']);
if (!is_null($this -> min_data_time)) {
$this -> DrawTimestamp($image, $this -> timefont, $this -> colours['DEFAULT']['TIME']['gdref1'], "MIN");
$this -> DrawTimestamp($image, $this -> timefont, $this -> colours['DEFAULT']['TIME']['gdref1'], "MAX");
}
$this -> DrawTitle($image, $this -> titlefont, $this -> colours['DEFAULT']['TITLE']['gdref1']);
}
if (is_array($z_items)) {
foreach ($z_items as $it) {
if (strtolower(get_class($it)) == 'weathermaplink') {
// only draw LINKs if they have NODES defined (not templates)
// (also, check if the link still exists - if this is in the editor, it may have been deleted by now)
if (isset($this -> links[$it -> name]) && isset($it -> a) && isset($it -> b)) {
wm_debug("Drawing LINK " . $it -> name . "\n");
$this -> links[$it -> name] -> Draw($image, $this);
}
}
if (strtolower(get_class($it)) == 'weathermapnode') {
// if(!is_null($it->x)) $it->pre_render($image, $this);
if ($withnodes) {
// don't try and draw template nodes
if (isset($this -> nodes[$it -> name]) && !is_null($it -> x)) {
# print "::".get_class($it)."\n";
wm_debug("Drawing NODE " . $it -> name . "\n");
$this -> nodes[$it -> name] -> NewDraw($image, $this);
$ii = 0;
foreach ($this -> nodes[$it -> name] -> boundingboxes as $bbox) {
# $areaname = "NODE:" . $it->name . ':'.$ii;
$areaname = "NODE:N" . $it -> id . ":" . $ii;
$this -> imap -> addArea("Rectangle", $areaname, '', $bbox);
wm_debug("Adding imagemap area");
$ii++;
}
wm_debug("Added $ii bounding boxes too\n");
}
}
}
}
}
}
$overlay = myimagecolorallocate($image, 200, 0, 0);
// for the editor, we can optionally overlay some other stuff
if ($this -> context == 'editor') {
if ($use_rel_overlay) {
# $overlay = myimagecolorallocate($image, 200, 0, 0);
// first, we can show relatively positioned NODEs
foreach ($this -> nodes as $node) {
if ($node -> relative_to != '') {
$rel_x = $this -> nodes[$node -> relative_to] -> x;
$rel_y = $this -> nodes[$node -> relative_to] -> y;
imagearc($image, $node -> x, $node -> y, 15, 15, 0, 360, $overlay);
imagearc($image, $node -> x, $node -> y, 16, 16, 0, 360, $overlay);
imageline($image, $node -> x, $node -> y, $rel_x, $rel_y, $overlay);
}
}
}
if ($use_via_overlay) {
// then overlay VIAs, so they can be seen
foreach ($this -> links as $link) {
foreach ($link -> vialist as $via) {
if (isset($via[2])) {
$x = $this -> nodes[$via[2]] -> x + $via[0];
$y = $this -> nodes[$via[2]] -> y + $via[1];
} else {
$x = $via[0];
$y = $via[1];
}
imagearc($image, $x, $y, 10, 10, 0, 360, $overlay);
imagearc($image, $x, $y, 12, 12, 0, 360, $overlay);
}
}
}
}
#$this->myimagestring($image, 3, 200, 100, "Test 1\nLine 2", $overlay,0);
# $this->myimagestring($image, 30, 100, 100, "Test 1\nLine 2", $overlay,0);
#$this->myimagestring($image, 30, 200, 200, "Test 1\nLine 2", $overlay,45);
// Ready to output the results...
if ($filename == 'null') {
// do nothing at all - we just wanted the HTML AREAs for the editor or HTML output
} else {
if ($filename == '') {
if (isset($vars['width']) || isset($vars['height'])) {
resizeImage($image, $targetImage = NULL, $vars['width'], $vars['height']);
} else {
imagepng($image);
}
} elseif ($filename == 'return') {
} else {
$result = FALSE;
$functions = TRUE;
if (function_exists('imagejpeg') && preg_match("/\.jpg/i", $filename)) {
wm_debug("Writing JPEG file to $filename\n");
$result = imagejpeg($image, $filename);
} elseif (function_exists('imagegif') && preg_match("/\.gif/i", $filename)) {
wm_debug("Writing GIF file to $filename\n");
$result = imagegif($image, $filename);
} elseif (function_exists('imagepng') && preg_match("/\.png/i", $filename)) {
wm_debug("Writing PNG file to $filename\n");
$result = imagepng($image, $filename);
} else {
wm_warn("Failed to write map image. No function existed for the image format you requested. [WMWARN12]\n");
$functions = FALSE;
}
if (($result == FALSE) && ($functions == TRUE)) {
if (file_exists($filename)) {
wm_warn("Failed to overwrite existing image file $filename - permissions of existing file are wrong? [WMWARN13]");
} else {
wm_warn("Failed to create image file $filename - permissions of output directory are wrong? [WMWARN14]");
}
}
}
}
if ($this -> context == 'editor2') {
$cachefile = $this -> cachefolder . DIRECTORY_SEPARATOR . dechex(crc32($this -> configfile)) . "_bg." . $this -> cachefile_version . ".png";
imagepng($image, $cachefile);
$cacheuri = $this -> cachefolder . '/' . dechex(crc32($this -> configfile)) . "_bg." . $this -> cachefile_version . ".png";
$this -> mapcache = $cacheuri;
}
if (function_exists('imagecopyresampled')) {
// if one is specified, and we can, write a thumbnail too
if ($thumbnailfile != '') {
$result = FALSE;
if ($this -> width > $this -> height) {
$factor = ($thumbnailmax / $this -> width);
} else {
$factor = ($thumbnailmax / $this -> height);
}
$this -> thumb_width = $this -> width * $factor;
$this -> thumb_height = $this -> height * $factor;
$imagethumb = imagecreatetruecolor($this -> thumb_width, $this -> thumb_height);
imagecopyresampled($imagethumb, $image, 0, 0, 0, 0, $this -> thumb_width, $this -> thumb_height, $this -> width, $this -> height);
$result = imagepng($imagethumb, $thumbnailfile);
imagedestroy($imagethumb);
if (($result == FALSE)) {
if (file_exists($filename)) {
wm_warn("Failed to overwrite existing image file $filename - permissions of existing file are wrong? [WMWARN15]");
} else {
wm_warn("Failed to create image file $filename - permissions of output directory are wrong? [WMWARN16]");
}
}
}
} else {
wm_warn("Skipping thumbnail creation, since we don't have the necessary function. [WMWARN17]");
}
imagedestroy($image);
}
}
function CleanUp()
{
$all_layers = array_keys($this -> seen_zlayers);
foreach ($all_layers as $z) {
$this -> seen_zlayers[$z] = NULL;
}
foreach ($this -> links as $link) {
$link -> owner = NULL;
$link -> a = NULL;
$link -> b = NULL;
unset($link);
}
foreach ($this -> nodes as $node) {
// destroy all the images we created, to prevent memory leaks
if (isset($node -> image)) {
imagedestroy($node -> image);
}
$node -> owner = NULL;
unset($node);
}
// Clear up the other random hashes of information
$this -> dsinfocache = NULL;
$this -> colourtable = NULL;
$this -> usage_stats = NULL;
$this -> scales = NULL;
}
function PreloadMapHTML()
{
wm_debug("Trace: PreloadMapHTML()\n");
// onmouseover="return overlib('',DELAY,250,CAPTION,'$caption');" onmouseout="return nd();"
// find the middle of the map
$center_x = $this -> width / 2;
$center_y = $this -> height / 2;
// loop through everything. Figure out along the way if it's a node or a link
$allitems = $this -> buildAllItemsList();
foreach ($allitems as $myobj) {
$type = $myobj -> my_type();
$prefix = substr($type, 0, 1);
$dirs = [];
//print "\n\nConsidering a $type - ".$myobj->name.".\n";
if ($type == 'LINK') {
$dirs = [IN => [0, 2], OUT => [1, 3]];
}
if ($type == 'NODE') {
$dirs = [IN => [0, 1, 2, 3]];
}
// check to see if any of the relevant things have a value
$change = "";
foreach ($dirs as $d => $parts) {
//print "$d - ".join(" ",$parts)."\n";
$change .= join('', $myobj -> overliburl[$d]);
$change .= $myobj -> notestext[$d];
}
if ($this -> htmlstyle == "overlib") {
//print "CHANGE: $change\n";
// skip all this if it's a template node
if ($type == 'LINK' && !isset($myobj -> a -> name)) {
$change = '';
}
if ($type == 'NODE' && !isset($myobj -> x)) {
$change = '';
}
if ($change != '') {
//print "Something to be done.\n";
if ($type == 'NODE') {
$mid_x = $myobj -> x;
$mid_y = $myobj -> y;
}
if ($type == 'LINK') {
$a_x = $this -> nodes[$myobj -> a -> name] -> x;
$a_y = $this -> nodes[$myobj -> a -> name] -> y;
$b_x = $this -> nodes[$myobj -> b -> name] -> x;
$b_y = $this -> nodes[$myobj -> b -> name] -> y;
$mid_x = ($a_x + $b_x) / 2;
$mid_y = ($a_y + $b_y) / 2;
}
$left = "";
$above = "";
$img_extra = "";
if ($myobj -> overlibwidth != 0) {
$left = "WIDTH," . $myobj -> overlibwidth . ",";
$img_extra .= " WIDTH=$myobj->overlibwidth";
if ($mid_x > $center_x) {
$left .= "LEFT,";
}
}
if ($myobj -> overlibheight != 0) {
$above = "HEIGHT," . $myobj -> overlibheight . ",";
$img_extra .= " HEIGHT=$myobj->overlibheight";
if ($mid_y > $center_y) {
$above .= "ABOVE,";
}
}
foreach ($dirs as $dir => $parts) {
$caption = ($myobj -> overlibcaption[$dir] != '' ? $myobj -> overlibcaption[$dir] : $myobj -> name);
$caption = $this -> ProcessString($caption, $myobj);
$overlibhtml = "onmouseover=\"return overlib('";
$n = 0;
if (sizeof($myobj -> overliburl[$dir]) > 0) {
// print "ARRAY:".is_array($link->overliburl[$dir])."\n";
foreach ($myobj -> overliburl[$dir] as $url) {
if ($n > 0) {
$overlibhtml .= '<br />';
}
$overlibhtml .= "<img $img_extra src=" . $this -> ProcessString($url, $myobj) . ">";
$n++;
}
}
# print "Added $n for $dir\n";
if (trim($myobj -> notestext[$dir]) != '') {
# put in a linebreak if there was an image AND notes
if ($n > 0) {
$overlibhtml .= '<br />';
}
$note = $this -> ProcessString($myobj -> notestext[$dir], $myobj);
$note = htmlspecialchars($note, ENT_NOQUOTES);
$note = str_replace("'", "\\'", $note);
$note = str_replace('"', """, $note);
$overlibhtml .= $note;
}
$overlibhtml .= "',DELAY,250,{$left}{$above}CAPTION,'" . $caption . "');\" onmouseout=\"return nd();\"";
//r($myobj);
foreach ($myobj->targets as $target) {
if(strstr($target[4],"obs_port")) {
if ($port_id = extract_port_id($target[4])) {
$extra_html = 'class="entity-popup" data-etype="port" data-eid="'.$port_id.'"';
}
}
}
foreach ($parts as $part) {
$areaname = $type . ":" . $prefix . $myobj -> id . ":" . $part;
//print "INFOURL for $areaname - ";
//$this->imap->setProp("extrahtml", $overlibhtml, $areaname);
if(isset($extra_html)) {
$this -> imap -> setProp("extrahtml", $extra_html, $areaname);
}
}
unset($extra_html);
}
} // if change
} // overlib?
if ($this -> htmlstyle == "qtip") {
// class="entity-popup " data-eid="520" data-etype="device" data-hasqtip="1" aria-describedby="qtip-1"
}
// now look at inforurls
foreach ($dirs as $dir => $parts) {
foreach ($parts as $part) {
# $areaname = $type.":" . $myobj->name . ":" . $part;
$areaname = $type . ":" . $prefix . $myobj -> id . ":" . $part;
//print "INFOURL for $areaname - ";
if (($this -> htmlstyle != 'editor') && ($myobj -> infourl[$dir] != '')) {
$this -> imap -> setProp("href", $this -> ProcessString($myobj -> infourl[$dir], $myobj), $areaname);
//print "Setting.\n";
} else {
//print "NOT Setting.\n";
}
}
}
}
// }
}
function asJS()
{
$js = '';
$js .= "var Links = new Array();\n";
$js .= "var LinkIDs = new Array();\n";
# $js.=$this->defaultlink->asJS();
foreach ($this -> links as $link) {
$js .= $link -> asJS();
}
$js .= "var Nodes = new Array();\n";
$js .= "var NodeIDs = new Array();\n";
# $js.=$this->defaultnode->asJS();
foreach ($this -> nodes as $node) {
$js .= $node -> asJS();
}
return $js;
}
function asJSON()
{
$json = '';
$json .= "{ \n";
$json .= "\"map\": { \n";
foreach (array_keys($this -> inherit_fieldlist) as $fld) {
$json .= js_escape($fld) . ": ";
$json .= js_escape($this -> $fld);
$json .= ",\n";
}
$json = rtrim($json, ", \n");
$json .= "\n},\n";
$json .= "\"nodes\": {\n";
$json .= $this -> defaultnode -> asJSON();
foreach ($this -> nodes as $node) {
$json .= $node -> asJSON();
}
$json = rtrim($json, ", \n");
$json .= "\n},\n";
$json .= "\"links\": {\n";
$json .= $this -> defaultlink -> asJSON();
foreach ($this -> links as $link) {
$json .= $link -> asJSON();
}
$json = rtrim($json, ", \n");
$json .= "\n},\n";
$json .= "'imap': [\n";
$json .= $this -> imap -> subJSON("NODE:");
// should check if there WERE nodes...
$json .= ",\n";
$json .= $this -> imap -> subJSON("LINK:");
$json .= "\n]\n";
$json .= "\n";
$json .= ", 'valid': 1}\n";
return ($json);
}
// This method MUST run *after* DrawMap. It relies on DrawMap to call the map-drawing bits
// which will populate the ImageMap with regions.
//
// imagemapname is a parameter, so we can stack up several maps in the Cacti plugin with their own imagemaps
function MakeHTML($imagemapname = "weathermap_imap")
{
wm_debug("Trace: MakeHTML()\n");
// PreloadMapHTML fills in the ImageMap info, ready for the HTML to be created.
$this -> PreloadMapHTML();
$html = '';
$html .= '