2012-08-12 15:47:27 +02:00
< ? php
2014-04-14 12:29:03 +02:00
2012-08-12 15:47:27 +02:00
/*
* TileServer . php project
* ======================
* https :// github . com / klokantech / tileserver - php /
2016-02-17 21:29:53 +01:00
* Copyright ( C ) 2016 - Klokan Technologies GmbH
2012-08-12 15:47:27 +02:00
*/
2014-04-23 14:30:27 +02:00
global $config ;
2016-05-04 19:19:33 +02:00
$config [ 'serverTitle' ] = 'Maps hosted with TileServer-php v2.0' ;
2016-02-29 20:21:13 +01:00
$config [ 'availableFormats' ] = array ( 'png' , 'jpg' , 'jpeg' , 'gif' , 'webp' , 'pbf' , 'hybrid' );
2016-08-26 11:12:43 -04:00
$config [ 'mbtilesPrefix' ] = './' ;
2016-03-07 14:39:28 +01:00
//$config['template'] = 'template.php';
2016-02-01 20:51:03 +01:00
//$config['baseUrls'] = array('t0.server.com', 't1.server.com');
2014-04-23 14:30:27 +02:00
2014-04-11 10:51:19 +02:00
Router :: serve ( array (
2014-04-11 11:08:04 +02:00
'/' => 'Server:getHtml' ,
2016-03-04 12:07:37 +01:00
'/maps' => 'Server:getInfo' ,
2014-04-11 10:51:19 +02:00
'/html' => 'Server:getHtml' ,
2016-01-26 22:31:12 +01:00
'/:alpha/:number/:number/:number.grid.json' => 'Json:getUTFGrid' ,
2014-05-19 18:01:11 +02:00
'/:alpha.json' => 'Json:getJson' ,
'/:alpha.jsonp' => 'Json:getJsonp' ,
2014-04-24 14:56:01 +02:00
'/wmts' => 'Wmts:get' ,
2014-04-28 14:12:51 +02:00
'/wmts/1.0.0/WMTSCapabilities.xml' => 'Wmts:get' ,
2015-12-04 14:00:48 +01:00
'/wmts/:alpha/:number/:number/:alpha' => 'Wmts:getTile' ,
'/wmts/:alpha/:alpha/:number/:number/:alpha' => 'Wmts:getTile' ,
'/wmts/:alpha/:alpha/:alpha/:number/:number/:alpha' => 'Wmts:getTile' ,
'/:alpha/:number/:number/:alpha' => 'Wmts:getTile' ,
2014-04-11 10:51:19 +02:00
'/tms' => 'Tms:getCapabilities' ,
2014-05-19 18:01:11 +02:00
'/tms/:alpha' => 'Tms:getLayerCapabilities' ,
2014-04-28 14:12:51 +02:00
));
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* Server base
*/
class Server {
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
2014-04-28 14:12:51 +02:00
* Configuration of TileServer [ baseUrls , serverTitle ]
2016-01-26 22:55:34 +01:00
* @ var array
2014-04-11 10:51:19 +02:00
*/
public $config ;
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* Datasets stored in file structure
2016-01-26 22:55:34 +01:00
* @ var array
2014-04-11 10:51:19 +02:00
*/
public $fileLayer = array ();
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* Datasets stored in database
2016-01-26 22:55:34 +01:00
* @ var array
2014-04-11 10:51:19 +02:00
*/
public $dbLayer = array ();
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* PDO database connection
2016-01-26 22:55:34 +01:00
* @ var object
2014-04-11 10:51:19 +02:00
*/
2014-04-25 11:37:42 +02:00
public $db ;
2012-08-12 15:47:27 +02:00
2014-04-28 14:12:51 +02:00
/**
2014-04-11 10:51:19 +02:00
* Set config
*/
public function __construct () {
2016-03-01 15:05:58 +01:00
$this -> config = $GLOBALS [ 'config' ];
//Get config from enviroment
2016-03-01 19:17:46 +01:00
$envServerTitle = getenv ( 'serverTitle' );
if ( $envServerTitle !== FALSE ){
$this -> config [ 'serverTitle' ] = $envServerTitle ;
2016-03-01 15:05:58 +01:00
}
2016-03-01 19:17:46 +01:00
$envBaseUrls = getenv ( 'baseUrls' );
if ( $envBaseUrls !== FALSE ){
$this -> config [ 'baseUrls' ] = is_array ( $envBaseUrls ) ?
$envBaseUrls : explode ( ',' , $envBaseUrls );
2016-03-01 15:05:58 +01:00
}
2016-03-07 14:39:28 +01:00
$envTemplate = getenv ( 'template' );
if ( $envBaseUrls !== FALSE ){
$this -> config [ 'template' ] = $envTemplate ;
}
2014-04-11 10:51:19 +02:00
}
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* Looks for datasets
*/
public function setDatasets () {
$mjs = glob ( '*/metadata.json' );
2016-08-26 11:12:43 -04:00
$mbts = glob ( $this -> config [ 'mbtilesPrefix' ] . '*.mbtiles' );
2014-04-11 10:51:19 +02:00
if ( $mjs ) {
2014-06-20 12:24:09 +02:00
foreach ( array_filter ( $mjs , 'is_readable' ) as $mj ) {
2014-04-11 10:51:19 +02:00
$layer = $this -> metadataFromMetadataJson ( $mj );
array_push ( $this -> fileLayer , $layer );
}
2014-04-23 17:01:08 +02:00
}
if ( $mbts ) {
2014-06-20 12:24:09 +02:00
foreach ( array_filter ( $mbts , 'is_readable' ) as $mbt ) {
2014-04-11 10:51:19 +02:00
$this -> dbLayer [] = $this -> metadataFromMbtiles ( $mbt );
}
2014-04-23 17:01:08 +02:00
}
2014-04-11 10:51:19 +02:00
}
2014-04-28 14:12:51 +02:00
2014-04-11 10:51:19 +02:00
/**
2014-04-14 12:29:03 +02:00
* Processing params from router < server >/< layer >/< z >/< x >/< y >. ext
2014-04-11 10:51:19 +02:00
* @ param array $params
*/
public function setParams ( $params ) {
if ( isset ( $params [ 1 ])) {
$this -> layer = $params [ 1 ];
}
2014-04-28 14:12:51 +02:00
$params = array_reverse ( $params );
2015-12-04 14:00:48 +01:00
if ( isset ( $params [ 2 ])) {
$this -> z = $params [ 2 ];
$this -> x = $params [ 1 ];
$file = explode ( '.' , $params [ 0 ]);
$this -> y = $file [ 0 ];
$this -> ext = isset ( $file [ 1 ]) ? $file [ 1 ] : NULL ;
2014-04-11 10:51:19 +02:00
}
}
/**
* Get variable don ' t independent on sensitivity
* @ param string $key
2014-04-14 12:29:03 +02:00
* @ return boolean
2014-04-11 10:51:19 +02:00
*/
2014-04-24 14:56:01 +02:00
public function getGlobal ( $isKey ) {
$get = $_GET ;
foreach ( $get as $key => $value ) {
2014-04-28 14:12:51 +02:00
if ( strtolower ( $isKey ) == strtolower ( $key )) {
2014-04-24 14:56:01 +02:00
return $value ;
2014-04-11 10:51:19 +02:00
}
}
2014-04-14 12:29:03 +02:00
return FALSE ;
2014-04-11 10:51:19 +02:00
}
/**
* Testing if is a database layer
* @ param string $layer
* @ return boolean
*/
public function isDBLayer ( $layer ) {
2016-08-26 11:12:43 -04:00
if ( is_file ( $this -> config [ 'mbtilesPrefix' ] . $layer . '.mbtiles' )) {
2014-04-25 10:22:02 +02:00
return TRUE ;
2014-04-28 14:12:51 +02:00
} else {
2014-04-25 10:22:02 +02:00
return FALSE ;
2014-04-11 10:51:19 +02:00
}
}
/**
* Testing if is a file layer
* @ param string $layer
* @ return boolean
*/
2014-04-23 17:01:08 +02:00
public function isFileLayer ( $layer ) {
2014-04-28 14:12:51 +02:00
if ( is_dir ( $layer )) {
2014-04-25 10:22:02 +02:00
return TRUE ;
2014-04-28 14:12:51 +02:00
} else {
2014-04-25 10:22:02 +02:00
return FALSE ;
2014-04-11 10:51:19 +02:00
}
}
/**
2016-02-17 21:29:53 +01:00
* Get metadata from metadataJson
2014-04-11 10:51:19 +02:00
* @ param string $jsonFileName
* @ return array
*/
public function metadataFromMetadataJson ( $jsonFileName ) {
$metadata = json_decode ( file_get_contents ( $jsonFileName ), true );
$metadata [ 'basename' ] = str_replace ( '/metadata.json' , '' , $jsonFileName );
2016-02-29 12:16:24 +01:00
return $this -> metadataValidation ( $metadata );
2014-04-11 10:51:19 +02:00
}
/**
* Loads metadata from MBtiles
* @ param string $mbt
* @ return object
*/
public function metadataFromMbtiles ( $mbt ) {
$metadata = array ();
$this -> DBconnect ( $mbt );
$result = $this -> db -> query ( 'select * from metadata' );
$resultdata = $result -> fetchAll ();
foreach ( $resultdata as $r ) {
2016-02-17 21:29:53 +01:00
$value = preg_replace ( '/(\\n)+/' , '' , $r [ 'value' ]);
2014-06-05 21:46:00 +02:00
$metadata [ $r [ 'name' ]] = addslashes ( $value );
2014-04-11 10:51:19 +02:00
}
2014-06-20 23:15:01 +02:00
if ( ! array_key_exists ( 'minzoom' , $metadata )
|| ! array_key_exists ( 'maxzoom' , $metadata )
) {
// autodetect minzoom and maxzoom
$result = $this -> db -> query ( 'select min(zoom_level) as min, max(zoom_level) as max from tiles' );
$resultdata = $result -> fetchAll ();
2016-02-17 21:29:53 +01:00
if ( ! array_key_exists ( 'minzoom' , $metadata )){
2014-06-20 23:15:01 +02:00
$metadata [ 'minzoom' ] = $resultdata [ 0 ][ 'min' ];
2016-02-17 21:29:53 +01:00
}
if ( ! array_key_exists ( 'maxzoom' , $metadata )){
2014-06-20 23:15:01 +02:00
$metadata [ 'maxzoom' ] = $resultdata [ 0 ][ 'max' ];
2016-02-17 21:29:53 +01:00
}
2014-06-20 23:15:01 +02:00
}
// autodetect format using JPEG magic number FFD8
if ( ! array_key_exists ( 'format' , $metadata )) {
$result = $this -> db -> query ( 'select hex(substr(tile_data,1,2)) as magic from tiles limit 1' );
$resultdata = $result -> fetchAll ();
$metadata [ 'format' ] = ( $resultdata [ 0 ][ 'magic' ] == 'FFD8' )
? 'jpg'
: 'png' ;
}
2014-07-23 15:25:25 +02:00
// autodetect bounds
if ( ! array_key_exists ( 'bounds' , $metadata )) {
$result = $this -> db -> query ( 'select min(tile_column) as w, max(tile_column) as e, min(tile_row) as s, max(tile_row) as n from tiles where zoom_level=' . $metadata [ 'maxzoom' ]);
$resultdata = $result -> fetchAll ();
2016-02-17 21:29:53 +01:00
$w = - 180 + 360 * ( $resultdata [ 0 ][ 'w' ] / pow ( 2 , $metadata [ 'maxzoom' ]));
$e = - 180 + 360 * (( 1 + $resultdata [ 0 ][ 'e' ]) / pow ( 2 , $metadata [ 'maxzoom' ]));
2014-07-23 15:25:25 +02:00
$n = $this -> row2lat ( $resultdata [ 0 ][ 'n' ], $metadata [ 'maxzoom' ]);
2016-02-17 21:29:53 +01:00
$s = $this -> row2lat ( $resultdata [ 0 ][ 's' ] - 1 , $metadata [ 'maxzoom' ]);
2014-07-23 15:25:25 +02:00
$metadata [ 'bounds' ] = implode ( ',' , array ( $w , $s , $e , $n ));
}
2014-04-11 10:51:19 +02:00
$mbt = explode ( '.' , $mbt );
$metadata [ 'basename' ] = $mbt [ 0 ];
2016-02-29 12:16:24 +01:00
$metadata = $this -> metadataValidation ( $metadata );
2014-04-11 10:51:19 +02:00
return $metadata ;
}
2016-01-26 22:55:34 +01:00
2014-07-23 15:25:25 +02:00
/**
* Convert row number to latitude of the top of the row
* @ param integer $r
* @ param integer $zoom
* @ return integer
*/
public function row2lat ( $r , $zoom ) {
2016-02-17 21:29:53 +01:00
$y = $r / pow ( 2 , $zoom - 1 ) - 1 ;
return rad2deg ( 2.0 * atan ( exp ( 3.191459196 * $y )) - 1.57079632679489661922 );
2014-07-23 15:25:25 +02:00
}
2014-04-11 10:51:19 +02:00
/**
* Valids metaJSON
* @ param object $metadata
* @ return object
*/
public function metadataValidation ( $metadata ) {
2016-02-17 11:39:16 +01:00
if ( ! array_key_exists ( 'bounds' , $metadata )) {
$metadata [ 'bounds' ] = array ( - 180 , - 85.06 , 180 , 85.06 );
} elseif ( ! is_array ( $metadata [ 'bounds' ])) {
2014-04-11 10:51:19 +02:00
$metadata [ 'bounds' ] = array_map ( 'floatval' , explode ( ',' , $metadata [ 'bounds' ]));
}
if ( ! array_key_exists ( 'profile' , $metadata )) {
$metadata [ 'profile' ] = 'mercator' ;
}
2016-02-17 21:29:53 +01:00
if ( array_key_exists ( 'minzoom' , $metadata )){
2014-04-11 10:51:19 +02:00
$metadata [ 'minzoom' ] = intval ( $metadata [ 'minzoom' ]);
2016-02-17 21:29:53 +01:00
} else {
2014-04-11 10:51:19 +02:00
$metadata [ 'minzoom' ] = 0 ;
2016-02-17 21:29:53 +01:00
}
if ( array_key_exists ( 'maxzoom' , $metadata )){
2014-04-11 10:51:19 +02:00
$metadata [ 'maxzoom' ] = intval ( $metadata [ 'maxzoom' ]);
2016-02-17 21:29:53 +01:00
} else {
2014-04-11 10:51:19 +02:00
$metadata [ 'maxzoom' ] = 18 ;
2016-02-17 21:29:53 +01:00
}
2014-04-11 10:51:19 +02:00
if ( ! array_key_exists ( 'format' , $metadata )) {
2016-02-29 14:16:17 +01:00
if ( array_key_exists ( 'tiles' , $metadata )){
$pos = strrpos ( $metadata [ 'tiles' ][ 0 ], '.' );
$metadata [ 'format' ] = trim ( substr ( $metadata [ 'tiles' ][ 0 ], $pos + 1 ));
}
}
$formats = $this -> config [ 'availableFormats' ];
if ( ! in_array ( strtolower ( $metadata [ 'format' ]), $formats )){
$metadata [ 'format' ] = 'png' ;
2014-04-11 10:51:19 +02:00
}
2016-02-17 21:19:16 +01:00
if ( ! array_key_exists ( 'scale' , $metadata )) {
$metadata [ 'scale' ] = 1 ;
}
2016-02-29 12:16:24 +01:00
if ( ! array_key_exists ( 'tiles' , $metadata )){
$tiles = array ();
foreach ( $this -> config [ 'baseUrls' ] as $url ) {
$url = '' . $this -> config [ 'protocol' ] . '://' . $url . '/' .
$metadata [ 'basename' ] . '/{z}/{x}/{y}' ;
if ( strlen ( $metadata [ 'format' ]) <= 4 ){
$url .= '.' . $metadata [ 'format' ];
}
$tiles [] = $url ;
}
$metadata [ 'tiles' ] = $tiles ;
}
2014-04-11 10:51:19 +02:00
return $metadata ;
}
/**
* SQLite connection
* @ param string $tileset
*/
public function DBconnect ( $tileset ) {
try {
$this -> db = new PDO ( 'sqlite:' . $tileset , '' , '' , array ( PDO :: ATTR_PERSISTENT => true ));
} catch ( Exception $exc ) {
echo $exc -> getTraceAsString ();
die ;
}
if ( ! isset ( $this -> db )) {
header ( 'Content-type: text/plain' );
echo 'Incorrect tileset name: ' . $tileset ;
2014-04-28 14:12:51 +02:00
die ;
2014-04-11 10:51:19 +02:00
}
}
/**
* Check if file is modified and set Etag headers
* @ param string $filename
* @ return boolean
*/
public function isModified ( $filename ) {
2016-08-26 11:12:43 -04:00
$filename = $this -> config [ 'mbtilesPrefix' ] . $filename . '.mbtiles' ;
2014-04-11 10:51:19 +02:00
$lastModifiedTime = filemtime ( $filename );
2014-04-28 14:24:25 +02:00
$eTag = md5 ( $lastModifiedTime );
2016-02-17 21:29:53 +01:00
header ( 'Last-Modified: ' . gmdate ( 'D, d M Y H:i:s' , $lastModifiedTime ) . ' GMT' );
header ( 'Etag:' . $eTag );
2014-04-11 10:51:19 +02:00
if ( @ strtotime ( $_SERVER [ 'HTTP_IF_MODIFIED_SINCE' ]) == $lastModifiedTime ||
2014-05-19 19:47:11 +02:00
@ trim ( $_SERVER [ 'HTTP_IF_NONE_MATCH' ]) == $eTag ) {
2014-04-11 10:51:19 +02:00
return TRUE ;
} else {
return FALSE ;
}
}
/**
* Returns tile of dataset
* @ param string $tileset
* @ param integer $z
* @ param integer $y
* @ param integer $x
* @ param string $ext
*/
2014-06-05 21:54:40 +02:00
public function renderTile ( $tileset , $z , $y , $x , $ext ) {
2014-04-11 10:51:19 +02:00
if ( $this -> isDBLayer ( $tileset )) {
if ( $this -> isModified ( $tileset ) == TRUE ) {
2014-06-27 09:19:26 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2014-04-11 10:51:19 +02:00
header ( 'HTTP/1.1 304 Not Modified' );
2014-04-28 14:24:25 +02:00
die ;
2014-04-11 10:51:19 +02:00
}
2016-08-26 11:12:43 -04:00
$this -> DBconnect ( $this -> config [ 'mbtilesPrefix' ] . $tileset . '.mbtiles' );
2014-04-11 10:51:19 +02:00
$z = floatval ( $z );
$y = floatval ( $y );
$x = floatval ( $x );
2014-04-14 12:29:03 +02:00
$flip = true ;
if ( $flip ) {
$y = pow ( 2 , $z ) - 1 - $y ;
}
2014-04-11 10:51:19 +02:00
$result = $this -> db -> query ( 'select tile_data as t from tiles where zoom_level=' . $z . ' and tile_column=' . $x . ' and tile_row=' . $y );
$data = $result -> fetchColumn ();
if ( ! isset ( $data ) || $data === FALSE ) {
2015-12-15 16:27:28 +01:00
//if tile doesn't exist
//select scale of tile (for retina tiles)
2014-08-15 01:01:35 +02:00
$result = $this -> db -> query ( 'select value from metadata where name="scale"' );
$resultdata = $result -> fetchColumn ();
$scale = isset ( $resultdata ) && $resultdata !== FALSE ? $resultdata : 1 ;
2015-12-15 16:27:28 +01:00
$this -> getCleanTile ( $scale , $ext );
2014-04-11 10:51:19 +02:00
} else {
$result = $this -> db -> query ( 'select value from metadata where name="format"' );
$resultdata = $result -> fetchColumn ();
$format = isset ( $resultdata ) && $resultdata !== FALSE ? $resultdata : 'png' ;
if ( $format == 'jpg' ) {
$format = 'jpeg' ;
}
2015-09-05 04:44:45 +02:00
if ( $format == 'pbf' ) {
header ( 'Content-type: application/x-protobuf' );
header ( 'Content-Encoding:gzip' );
} else {
header ( 'Content-type: image/' . $format );
}
2014-06-27 09:19:26 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2014-04-11 10:51:19 +02:00
echo $data ;
}
2014-04-23 17:01:08 +02:00
} elseif ( $this -> isFileLayer ( $tileset )) {
2015-12-04 14:00:48 +01:00
$name = './' . $tileset . '/' . $z . '/' . $x . '/' . $y ;
$mime = 'image/' ;
if ( $ext != NULL ){
$name .= '.' . $ext ;
}
2014-04-11 10:51:19 +02:00
if ( $fp = @ fopen ( $name , 'rb' )) {
2015-12-04 14:00:48 +01:00
if ( $ext != NULL ){
$mime .= $ext ;
} else {
//detect image type from file
2016-02-01 20:51:03 +01:00
$mimetypes = array ( 'gif' , 'jpeg' , 'png' );
2015-12-04 14:00:48 +01:00
$mime .= $mimetypes [ exif_imagetype ( $name ) - 1 ];
}
2014-06-27 09:19:26 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2015-12-04 14:00:48 +01:00
header ( 'Content-Type: ' . $mime );
2014-04-11 10:51:19 +02:00
header ( 'Content-Length: ' . filesize ( $name ));
fpassthru ( $fp );
die ;
} else {
2014-08-15 01:01:35 +02:00
//scale of tile (for retina tiles)
2016-02-17 21:29:53 +01:00
$meta = json_decode ( file_get_contents ( $tileset . '/metadata.json' ));
2014-08-15 01:01:35 +02:00
if ( ! isset ( $meta -> scale )){
$meta -> scale = 1 ;
}
2014-04-11 10:51:19 +02:00
}
2015-12-15 16:27:28 +01:00
$this -> getCleanTile ( $meta -> scale , $ext );
2014-04-11 10:51:19 +02:00
} else {
2015-09-05 04:44:45 +02:00
header ( 'HTTP/1.1 404 Not Found' );
2016-02-17 21:29:53 +01:00
echo 'Server: Unknown or not specified dataset "' . $tileset . '"' ;
2014-04-11 10:51:19 +02:00
die ;
}
}
2014-04-11 11:08:04 +02:00
/**
* Returns clean tile
2014-08-15 01:01:35 +02:00
* @ param integer $scale Default 1
2014-04-11 11:08:04 +02:00
*/
2015-12-15 16:27:28 +01:00
public function getCleanTile ( $scale = 1 , $format = 'png' ) {
switch ( $format ) {
case 'pbf' :
2016-08-26 11:12:43 -04:00
header ( 'Access-Control-Allow-Origin: *' );
header ( 'HTTP/1.1 204 No Content' );
2015-12-15 16:27:28 +01:00
header ( 'Content-Type: application/json; charset=utf-8' );
echo '{"message":"Tile does not exist"}' ;
2016-02-29 20:21:13 +01:00
break ;
2016-05-03 23:21:40 +02:00
case 'webp' :
header ( 'Access-Control-Allow-Origin: *' );
header ( 'Content-type: image/webp' );
echo base64_decode ( 'UklGRhIAAABXRUJQVlA4TAYAAAAvQWxvAGs=' );
break ;
case 'jpg' :
header ( 'Access-Control-Allow-Origin: *' );
header ( 'Content-type: image/jpg' );
echo base64_decode ( '/9j/2wBDAAMCAgICAgMCAgIDAwMDBAYEBAQEBAgGBgUGCQgKCgkICQkKDA8MCgsOCwkJDRENDg8QEBEQCgwSExIQEw8QEBD/yQALCAABAAEBAREA/8wABgAQEAX/2gAIAQEAAD8A0s8g/9k=' );
break ;
2015-12-15 16:27:28 +01:00
case 'png' :
default :
header ( 'Access-Control-Allow-Origin: *' );
header ( 'Content-type: image/png' );
2016-04-18 21:25:38 +03:00
// 256x256 transparent optimised png tile
2016-05-03 22:48:23 +02:00
echo unpack ( 'H' , '89504e470d0a1a0a0000000d494844520000010000000100010300000066bc3a2500000003504c5445000000a77a3dda0000000174524e530040e6d8660000001f494441541819edc1010d000000c220fba77e0e37600000000000000000e70221000001f5a2bd040000000049454e44ae426082' );
2016-02-29 20:21:13 +01:00
break ;
2015-12-15 16:27:28 +01:00
}
2016-02-29 20:21:13 +01:00
die ;
2014-04-11 10:51:19 +02:00
}
/**
* Returns tile ' s UTFGrid
* @ param string $tileset
* @ param integer $z
* @ param integer $y
* @ param integer $x
*/
2014-06-05 21:54:40 +02:00
public function renderUTFGrid ( $tileset , $z , $y , $x , $flip = TRUE ) {
2014-04-11 10:51:19 +02:00
if ( $this -> isDBLayer ( $tileset )) {
if ( $this -> isModified ( $tileset ) == TRUE ) {
header ( 'HTTP/1.1 304 Not Modified' );
}
2014-04-28 14:12:51 +02:00
if ( $flip ) {
$y = pow ( 2 , $z ) - 1 - $y ;
}
2014-04-11 10:51:19 +02:00
try {
2016-08-26 11:12:43 -04:00
$this -> DBconnect ( $this -> config [ 'mbtilesPrefix' ] . $tileset . '.mbtiles' );
2014-04-28 14:12:51 +02:00
2016-01-26 22:55:34 +01:00
$query = 'SELECT grid FROM grids WHERE tile_column = ' . $x . ' AND '
. 'tile_row = ' . $y . ' AND zoom_level = ' . $z ;
$result = $this -> db -> query ( $query );
$data = $result -> fetch ( PDO :: FETCH_ASSOC );
if ( $data !== FALSE ) {
$grid = gzuncompress ( $data [ 'grid' ]);
2014-04-11 10:51:19 +02:00
$grid = substr ( trim ( $grid ), 0 , - 1 );
//adds legend (data) to output
$grid .= ',"data":{' ;
2016-01-26 22:55:34 +01:00
$kquery = 'SELECT key_name as key, key_json as json FROM grid_data '
. 'WHERE zoom_level=' . $z . ' and '
. 'tile_column=' . $x . ' and tile_row=' . $y ;
$result = $this -> db -> query ( $kquery );
2014-04-11 10:51:19 +02:00
while ( $r = $result -> fetch ( PDO :: FETCH_ASSOC )) {
$grid .= '"' . $r [ 'key' ] . '":' . $r [ 'json' ] . ',' ;
}
$grid = rtrim ( $grid , ',' ) . '}}' ;
header ( 'Access-Control-Allow-Origin: *' );
2014-05-19 19:47:11 +02:00
2014-05-19 18:58:54 +02:00
if ( isset ( $_GET [ 'callback' ]) && ! empty ( $_GET [ 'callback' ])) {
2016-01-26 22:55:34 +01:00
header ( 'Content-Type:text/javascript charset=utf-8' );
2014-04-11 10:51:19 +02:00
echo $_GET [ 'callback' ] . '(' . $grid . ');' ;
} else {
2016-01-26 22:55:34 +01:00
header ( 'Content-Type:text/javascript; charset=utf-8' );
2014-05-19 18:58:54 +02:00
echo $grid ;
2014-04-11 10:51:19 +02:00
}
2016-01-26 22:55:34 +01:00
} else {
header ( 'Access-Control-Allow-Origin: *' );
echo '{}' ;
die ;
}
} catch ( Exception $e ) {
2014-04-11 10:51:19 +02:00
header ( 'Content-type: text/plain' );
print 'Error querying the database: ' . $e -> getMessage ();
}
} else {
echo 'Server: no MBTiles tileset' ;
die ;
}
}
/**
* Returns server info
*/
public function getInfo () {
$this -> setDatasets ();
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
header ( 'Content-Type: text/html;charset=UTF-8' );
2016-03-04 12:07:37 +01:00
echo '<!DOCTYPE html><html><head><meta charset="UTF-8"><title>' . $this -> config [ 'serverTitle' ] . '</title></head><body>' .
'<h1>' . $this -> config [ 'serverTitle' ] . '</h1>' .
'TileJSON service: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/index.json">' . $this -> config [ 'baseUrls' ][ 0 ] . '/index.json</a><br>' .
'WMTS service: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/wmts">' . $this -> config [ 'baseUrls' ][ 0 ] . '/wmts</a><br>' .
'TMS service: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/tms">' . $this -> config [ 'baseUrls' ][ 0 ] . '/tms</a>' ;
2014-04-11 10:51:19 +02:00
foreach ( $maps as $map ) {
2016-03-04 12:07:37 +01:00
$extend = '[' . implode ( $map [ 'bounds' ], ', ' ) . ']' ;
echo '<p>Tileset: <b>' . $map [ 'basename' ] . '</b><br>' .
'Metadata: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/' . $map [ 'basename' ] . '.json">' .
$this -> config [ 'baseUrls' ][ 0 ] . '/' . $map [ 'basename' ] . '.json</a><br>' .
'Bounds: ' . $extend ;
if ( isset ( $map [ 'crs' ])){ echo '<br>CRS: ' . $map [ 'crs' ];}
echo '</p>' ;
2014-04-11 10:51:19 +02:00
}
2016-02-17 21:29:53 +01:00
echo '<p>Copyright (C) 2016 - Klokan Technologies GmbH</p>' ;
2014-04-11 10:51:19 +02:00
echo '</body></html>' ;
}
/**
* Returns html viewer
*/
public function getHtml () {
$this -> setDatasets ();
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
2016-03-07 14:39:28 +01:00
if ( isset ( $this -> config [ 'template' ]) && file_exists ( $this -> config [ 'template' ])) {
$baseUrls = $this -> config [ 'baseUrls' ];
$serverTitle = $this -> config [ 'serverTitle' ];
include_once $this -> config [ 'template' ];
2014-04-11 10:51:19 +02:00
} else {
2016-03-07 14:39:28 +01:00
header ( 'Content-Type: text/html;charset=UTF-8' );
echo '<!DOCTYPE html><html><head><meta charset="UTF-8"><title>' . $this -> config [ 'serverTitle' ] . '</title>' ;
echo ' < link rel = " stylesheet " type = " text/css " href = " //cdn.klokantech.com/tileviewer/v1/index.css " />
< script src = " //cdn.klokantech.com/tileviewer/v1/index.js " ></ script >< body >
< script > tileserver ({ index : " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/index.json " , tilejson : " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/%n.json " , tms : " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/tms " , wmts : " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/wmts " }); </ script >
< h1 > Welcome to ' . $this->config[' serverTitle '] . ' </ h1 >
< p > This server distributes maps to desktop , web , and mobile applications .</ p >
< p > The mapping data are available as OpenGIS Web Map Tiling Service ( OGC WMTS ), OSGEO Tile Map Service ( TMS ), and popular XYZ urls described with TileJSON metadata .</ p > ' ;
if ( ! isset ( $maps )) {
echo ' < h3 style = " color:darkred; " > No maps available yet </ h3 >
< p style = " color:darkred; font-style: italic; " >
Ready to go - just upload some maps into directory : ' . getcwd() . ' / on this server .</ p >
< p > Note : The maps can be a directory with tiles in XYZ format with metadata . json file .< br />
You can easily convert existing geodata ( GeoTIFF , ECW , MrSID , etc ) to this tile structure with < a href = " http://www.maptiler.com " > MapTiler Cluster </ a > or open - source projects such as < a href = " http://www.klokan.cz/projects/gdal2tiles/ " > GDAL2Tiles </ a > or < a href = " http://www.maptiler.org/ " > MapTiler </ a > or simply upload any maps in MBTiles format made by < a href = " http://www.tilemill.com/ " > TileMill </ a >. Helpful is also the < a href = " https://github.com/mapbox/mbutil " > mbutil </ a > tool . Serving directly from . mbtiles files is supported , but with decreased performance .</ p > ' ;
} else {
echo '<ul>' ;
foreach ( $maps as $map ) {
echo " <li> " . $map [ 'name' ] . '</li>' ;
}
echo '</ul>' ;
2014-04-11 10:51:19 +02:00
}
2016-03-07 14:39:28 +01:00
echo '</body></html>' ;
2014-04-11 10:51:19 +02:00
}
}
2012-08-12 15:47:27 +02:00
}
2014-04-11 10:51:19 +02:00
/**
* JSON service
*/
class Json extends Server {
/**
* Callback for JSONP default grid
2016-01-26 22:55:34 +01:00
* @ var string
2014-04-11 10:51:19 +02:00
*/
private $callback = 'grid' ;
/**
* @ param array $params
*/
public $layer = 'index' ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $z ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $y ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $x ;
/**
2016-01-26 22:55:34 +01:00
* @ var string
2014-04-11 10:51:19 +02:00
*/
public $ext ;
/**
2016-01-26 22:55:34 +01:00
*
2014-04-11 10:51:19 +02:00
* @ param array $params
*/
public function __construct ( $params ) {
parent :: __construct ();
parent :: setParams ( $params );
if ( isset ( $_GET [ 'callback' ]) && ! empty ( $_GET [ 'callback' ])) {
$this -> callback = $_GET [ 'callback' ];
}
}
/**
* Adds metadata about layer
* @ param array $metadata
* @ return array
*/
public function metadataTileJson ( $metadata ) {
$metadata [ 'tilejson' ] = '2.0.0' ;
$metadata [ 'scheme' ] = 'xyz' ;
2014-04-28 14:12:51 +02:00
if ( $this -> isDBLayer ( $metadata [ 'basename' ])) {
2016-08-26 11:12:43 -04:00
$this -> DBconnect ( $this -> config [ 'mbtilesPrefix' ] . $metadata [ 'basename' ] . '.mbtiles' );
2016-03-07 14:00:29 +01:00
$res = $this -> db -> query ( 'SELECT name FROM sqlite_master WHERE name="grids";' );
2014-04-28 14:12:51 +02:00
if ( $res ) {
2014-04-25 11:37:42 +02:00
foreach ( $this -> config [ 'baseUrls' ] as $url ) {
2014-08-04 17:47:47 +02:00
$grids [] = '' . $this -> config [ 'protocol' ] . '://' . $url . '/' . $metadata [ 'basename' ] . '/{z}/{x}/{y}.grid.json' ;
2014-04-25 11:37:42 +02:00
}
$metadata [ 'grids' ] = $grids ;
}
}
2015-09-05 04:44:45 +02:00
if ( array_key_exists ( 'json' , $metadata )) {
$mjson = json_decode ( stripslashes ( $metadata [ 'json' ]));
foreach ( $mjson as $key => $value ) {
$metadata [ $key ] = $value ;
}
unset ( $metadata [ 'json' ]);
}
2014-04-11 10:51:19 +02:00
return $metadata ;
}
/**
* Creates JSON from array
* @ param string $basename
* @ return string
*/
private function createJson ( $basename ) {
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
if ( $basename == 'index' ) {
$output = '[' ;
foreach ( $maps as $map ) {
2014-04-23 16:08:09 +02:00
$output = $output . json_encode ( $this -> metadataTileJson ( $map )) . ',' ;
2014-04-11 10:51:19 +02:00
}
2014-05-01 10:25:33 +02:00
if ( strlen ( $output ) > 1 ) {
2014-04-29 09:22:19 +02:00
$output = substr_replace ( $output , ']' , - 1 );
2014-05-01 10:25:33 +02:00
} else {
$output = $output . ']' ;
2014-04-29 09:22:19 +02:00
}
2014-04-11 10:51:19 +02:00
} else {
foreach ( $maps as $map ) {
if ( strpos ( $map [ 'basename' ], $basename ) !== false ) {
2014-04-23 16:08:09 +02:00
$output = json_encode ( $this -> metadataTileJson ( $map ));
2014-04-11 10:51:19 +02:00
break ;
}
}
2014-04-23 14:30:27 +02:00
}
if ( ! isset ( $output )) {
echo 'TileServer: unknown map ' . $basename ;
die ;
2014-04-11 10:51:19 +02:00
}
2014-06-04 09:55:25 +02:00
return stripslashes ( $output );
2014-04-11 10:51:19 +02:00
}
/**
* Returns JSON with callback
*/
public function getJson () {
2014-04-25 10:22:02 +02:00
parent :: setDatasets ();
2014-04-11 10:51:19 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2016-02-17 21:29:53 +01:00
header ( 'Content-Type: application/json; charset=utf-8' );
2014-04-23 17:01:08 +02:00
if ( $this -> callback !== 'grid' ) {
2014-05-19 19:47:11 +02:00
echo $this -> callback . '(' . $this -> createJson ( $this -> layer ) . ');' ; die ;
2014-04-23 17:01:08 +02:00
} else {
2014-05-19 19:47:11 +02:00
echo $this -> createJson ( $this -> layer ); die ;
2014-04-23 17:01:08 +02:00
}
2014-04-11 10:51:19 +02:00
}
/**
* Returns JSONP with callback
*/
public function getJsonp () {
2014-04-25 10:22:02 +02:00
parent :: setDatasets ();
2014-04-11 10:51:19 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2016-02-17 21:29:53 +01:00
header ( 'Content-Type: application/javascript; charset=utf-8' );
2014-04-11 10:51:19 +02:00
echo $this -> callback . '(' . $this -> createJson ( $this -> layer ) . ');' ;
}
/**
* Returns UTFGrid in JSON format
*/
public function getUTFGrid () {
2014-06-05 21:54:40 +02:00
parent :: renderUTFGrid ( $this -> layer , $this -> z , $this -> y , $this -> x );
2014-04-11 10:51:19 +02:00
}
2012-08-12 15:47:27 +02:00
}
2014-04-11 10:51:19 +02:00
/**
* Web map tile service
*/
class Wmts extends Server {
/**
* @ param array $params
*/
public $layer ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $z ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $y ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $x ;
/**
2016-01-26 22:55:34 +01:00
* @ var string
2014-04-11 10:51:19 +02:00
*/
public $ext ;
/**
2016-01-26 22:55:34 +01:00
*
2014-04-11 10:51:19 +02:00
* @ param array $params
*/
public function __construct ( $params ) {
parent :: __construct ();
2014-04-14 12:29:03 +02:00
if ( isset ( $params )) {
parent :: setParams ( $params );
}
2014-04-11 10:51:19 +02:00
}
2014-04-24 14:56:01 +02:00
/**
* Tests request from url and call method
*/
2014-04-28 14:12:51 +02:00
public function get () {
2014-04-24 14:56:01 +02:00
$request = $this -> getGlobal ( 'Request' );
2014-04-28 14:12:51 +02:00
if ( $request !== FALSE && $request == 'gettile' ) {
2014-04-24 14:56:01 +02:00
$this -> getTile ();
2014-04-28 14:12:51 +02:00
} else {
2014-04-25 10:22:02 +02:00
parent :: setDatasets ();
2014-04-24 14:56:01 +02:00
$this -> getCapabilities ();
}
}
2016-02-09 14:37:26 +01:00
/**
* Validates tilematrixset , calculates missing params
2016-02-29 20:21:13 +01:00
* @ param Object $tileMatrix
2016-02-09 14:37:26 +01:00
* @ return Object
*/
2016-02-17 21:19:16 +01:00
public function parseTileMatrix ( $layer , $tileMatrix ){
2016-02-26 08:51:47 +01:00
2016-02-25 23:21:40 +01:00
//process projection
if ( isset ( $layer [ 'proj4' ])){
preg_match_all ( " /([^+= ]+)=([^= ]+)/ " , $layer [ 'proj4' ], $res );
$proj4 = array_combine ( $res [ 1 ], $res [ 2 ]);
}
2016-02-26 08:51:47 +01:00
2016-02-18 08:43:49 +01:00
for ( $i = 0 ; $i < count ( $tileMatrix ); $i ++ ){
2016-02-18 16:49:48 +01:00
2016-02-17 21:19:16 +01:00
if ( ! isset ( $tileMatrix [ $i ][ 'id' ])){
$tileMatrix [ $i ][ 'id' ] = ( string ) $i ;
2016-02-09 14:37:26 +01:00
}
2016-02-17 21:19:16 +01:00
if ( ! isset ( $tileMatrix [ $i ][ 'extent' ]) && isset ( $layer [ 'extent' ])) {
$tileMatrix [ $i ][ 'extent' ] = $layer [ 'extent' ];
}
2016-02-09 14:37:26 +01:00
if ( ! isset ( $tileMatrix [ $i ][ 'matrix_size' ])) {
2016-02-18 16:49:48 +01:00
$tileExtent = $this -> tilesOfExtent (
$tileMatrix [ $i ][ 'extent' ],
$tileMatrix [ $i ][ 'origin' ],
$tileMatrix [ $i ][ 'pixel_size' ],
$tileMatrix [ $i ][ 'tile_size' ]
);
$tileMatrix [ $i ][ 'matrix_size' ] = array (
2016-02-25 19:23:48 +01:00
$tileExtent [ 2 ] + 1 ,
$tileExtent [ 1 ] + 1
2016-02-18 16:49:48 +01:00
);
2016-02-09 14:37:26 +01:00
}
if ( ! isset ( $tileMatrix [ $i ][ 'origin' ]) && isset ( $tileMatrix [ $i ][ 'extent' ])){
2016-02-18 08:43:49 +01:00
$tileMatrix [ $i ][ 'origin' ] = array (
2016-02-25 19:23:48 +01:00
$tileMatrix [ $i ][ 'extent' ][ 0 ], $tileMatrix [ $i ][ 'extent' ][ 3 ]
2016-02-18 08:43:49 +01:00
);
2016-02-09 14:37:26 +01:00
}
2016-02-26 08:51:47 +01:00
// Origins of geographic coordinate systems are setting in opposite order
2016-02-25 23:21:40 +01:00
if ( isset ( $proj4 ) && $proj4 [ 'proj' ] === 'longlat' ) {
2016-02-25 19:23:48 +01:00
$tileMatrix [ $i ][ 'origin' ] = array_reverse ( $tileMatrix [ $i ][ 'origin' ]);
}
2016-02-09 14:37:26 +01:00
if ( ! isset ( $tileMatrix [ $i ][ 'scale_denominator' ])){
2016-02-17 21:19:16 +01:00
$tileMatrix [ $i ][ 'scale_denominator' ] = count ( $tileMatrix ) - $i ;
}
if ( ! isset ( $tileMatrix [ $i ][ 'tile_size' ])){
$tileSize = 256 * ( int ) $layer [ 'scale' ];
$tileMatrix [ $i ][ 'tile_size' ] = array ( $tileSize , $tileSize );
2016-02-09 14:37:26 +01:00
}
}
2016-02-17 20:29:39 +01:00
2016-02-09 14:37:26 +01:00
return $tileMatrix ;
}
2016-02-17 20:29:39 +01:00
2016-02-09 14:37:26 +01:00
/**
* Calculates corners of tilematrix
* @ param array $extent
* @ param array $origin
* @ param array $pixel_size
* @ param array $tile_size
* @ return array
*/
public function tilesOfExtent ( $extent , $origin , $pixel_size , $tile_size ) {
2016-02-18 16:49:48 +01:00
$tiles = array (
$this -> minsample ( $extent [ 0 ] - $origin [ 0 ], $pixel_size [ 0 ] * $tile_size [ 0 ]),
$this -> minsample ( $extent [ 1 ] - $origin [ 1 ], $pixel_size [ 1 ] * $tile_size [ 1 ]),
$this -> maxsample ( $extent [ 2 ] - $origin [ 0 ], $pixel_size [ 0 ] * $tile_size [ 0 ]),
$this -> maxsample ( $extent [ 3 ] - $origin [ 1 ], $pixel_size [ 1 ] * $tile_size [ 1 ]),
);
return $tiles ;
}
2016-02-17 20:29:39 +01:00
2016-02-18 16:49:48 +01:00
private function minsample ( $x , $f ){
return $f > 0 ? floor ( $x / $f ) : ceil (( $x / $f ) - 1 );
}
2016-02-17 20:29:39 +01:00
2016-02-18 16:49:48 +01:00
private function maxsample ( $x , $f ){
return $f < 0 ? floor ( $x / $f ) : ceil (( $x / $f ) - 1 );
2016-02-09 14:37:26 +01:00
}
2016-02-17 20:29:39 +01:00
2016-02-03 15:13:56 +01:00
/**
* Default TileMetrixSet for Pseudo Mercator projection 3857
2016-03-21 10:59:29 +01:00
* @ param ? number $maxZoom
2016-02-03 15:13:56 +01:00
* @ return string TileMatrixSet xml
*/
2016-03-21 10:59:29 +01:00
public function getMercatorTileMatrixSet ( $maxZoom = 18 ){
2016-02-17 20:29:39 +01:00
$denominatorBase = 559082264.0287178 ;
2016-02-09 14:36:24 +01:00
$extent = array ( - 20037508.34 , - 20037508.34 , 20037508.34 , 20037508.34 );
2016-02-17 20:29:39 +01:00
$tileMatrixSet = array ();
2016-02-03 16:50:07 +01:00
2016-03-21 10:59:29 +01:00
for ( $i = 0 ; $i <= $maxZoom ; $i ++ ){
2016-02-17 20:29:39 +01:00
$matrixSize = pow ( 2 , $i );
2016-02-24 18:55:42 +01:00
$tileMatrixSet [] = array (
'extent' => $extent ,
'id' => ( string ) $i ,
'matrix_size' => array ( $matrixSize , $matrixSize ),
'origin' => array ( $extent [ 0 ], $extent [ 3 ]),
'scale_denominator' => $denominatorBase / pow ( 2 , $i ),
'tile_size' => array ( 256 , 256 )
);
2016-02-17 20:29:39 +01:00
}
return $this -> getTileMatrixSet ( 'GoogleMapsCompatible' , $tileMatrixSet , 'EPSG:3857' );
}
2016-02-17 21:19:16 +01:00
2016-02-17 20:29:39 +01:00
/**
* Default TileMetrixSet for WGS84 projection 4326
* @ return string Xml
*/
public function getWGS84TileMatrixSet (){
$extent = array ( - 180.000000 , - 90.000000 , 180.000000 , 90.000000 );
$scaleDenominators = array ( 279541132.01435887813568115234 , 139770566.00717943906784057617 ,
69885283.00358971953392028809 , 34942641.50179485976696014404 , 17471320.75089742988348007202 ,
8735660.37544871494174003601 , 4367830.18772435747087001801 , 2183915.09386217873543500900 ,
1091957.54693108936771750450 , 545978.77346554468385875225 , 272989.38673277234192937613 ,
136494.69336638617096468806 , 68247.34668319308548234403 , 34123.67334159654274117202 ,
17061.83667079825318069197 , 8530.91833539912659034599 , 4265.45916769956329517299 ,
2132.72958384978574031265 );
$tileMatrixSet = array ();
2016-03-21 13:40:33 +01:00
for ( $i = 0 ; $i <= 17 ; $i ++ ){
2016-02-17 20:29:39 +01:00
$matrixSize = pow ( 2 , $i );
2016-02-24 18:55:42 +01:00
$tileMatrixSet [] = array (
'extent' => $extent ,
'id' => ( string ) $i ,
'matrix_size' => array ( $matrixSize * 2 , $matrixSize ),
2016-02-26 08:51:47 +01:00
'origin' => array ( $extent [ 3 ], $extent [ 0 ]),
2016-02-24 18:55:42 +01:00
'scale_denominator' => $scaleDenominators [ $i ],
'tile_size' => array ( 256 , 256 )
);
2016-02-03 15:13:56 +01:00
}
2016-02-17 20:29:39 +01:00
return $this -> getTileMatrixSet ( 'WGS84' , $tileMatrixSet , 'EPSG:4326' );
2016-02-03 15:13:56 +01:00
}
2016-02-03 16:50:07 +01:00
2016-02-03 15:13:56 +01:00
/**
2016-02-17 20:29:39 +01:00
* Prints WMTS TileMatrixSet
2016-02-03 15:13:56 +01:00
* @ param string $name
2016-02-17 20:29:39 +01:00
* @ param array $tileMatrixSet Array of levels
2016-02-03 15:13:56 +01:00
* @ param string $crs Code of crs eg : EPSG : 3857
* @ return string TileMatrixSet xml
*/
2016-02-17 20:29:39 +01:00
public function getTileMatrixSet ( $name , $tileMatrixSet , $crs = 'EPSG:3857' ){
2016-02-03 15:13:56 +01:00
$srs = explode ( ':' , $crs );
$TileMatrixSet = ' < TileMatrixSet >
< ows : Title > ' . $name . ' </ ows : Title >
< ows : Abstract > ' . $name . ' '. $crs .' </ ows : Abstract >
< ows : Identifier > ' . $name . ' </ ows : Identifier >
< ows : SupportedCRS > urn : ogc : def : crs : '.$srs[0].' :: '.$srs[1].' </ ows : SupportedCRS > ' ;
2016-02-17 20:29:39 +01:00
// <WellKnownScaleSet>urn:ogc:def:wkss:OGC:1.0:GoogleMapsCompatible</WellKnownScaleSet>;
foreach ( $tileMatrixSet as $level ){
2016-02-03 15:13:56 +01:00
$TileMatrixSet .= '
< TileMatrix >
2016-02-17 20:29:39 +01:00
< ows : Identifier > ' . $level[' id '] . ' </ ows : Identifier >
< ScaleDenominator > ' . $level[' scale_denominator '] . ' </ ScaleDenominator >
< TopLeftCorner > '. $level[' origin '][0] . ' ' . $level[' origin '][1] .' </ TopLeftCorner >
< TileWidth > ' . $level[' tile_size '][0] . ' </ TileWidth >
< TileHeight > ' . $level[' tile_size '][1] . ' </ TileHeight >
< MatrixWidth > ' . $level[' matrix_size '][0] . ' </ MatrixWidth >
< MatrixHeight > ' . $level[' matrix_size '][1] . ' </ MatrixHeight >
2016-02-03 15:13:56 +01:00
</ TileMatrix > ' ;
}
$TileMatrixSet .= '</TileMatrixSet>' ;
2016-02-03 16:50:07 +01:00
2016-02-03 15:13:56 +01:00
return $TileMatrixSet ;
}
2014-04-24 14:56:01 +02:00
2014-04-11 10:51:19 +02:00
/**
2016-01-26 22:55:34 +01:00
* Returns tilesets getCapabilities
2014-04-11 10:51:19 +02:00
*/
public function getCapabilities () {
2016-02-17 21:19:16 +01:00
$layers = array_merge ( $this -> fileLayer , $this -> dbLayer );
2016-02-18 08:43:49 +01:00
//if TileMatrixSet is provided validate it
for ( $i = 0 ; $i < count ( $layers ); $i ++ ){
2016-02-18 16:49:48 +01:00
if ( $layers [ $i ][ 'profile' ] == 'custom' ){
2016-02-17 21:19:16 +01:00
$layers [ $i ][ 'tile_matrix' ] = $this -> parseTileMatrix (
2016-02-18 16:49:48 +01:00
$layers [ $i ],
$layers [ $i ][ 'tile_matrix' ]
);
2016-02-17 21:19:16 +01:00
}
}
2016-02-26 08:51:47 +01:00
2016-02-17 21:29:53 +01:00
header ( 'Content-type: application/xml' );
2014-04-28 14:12:51 +02:00
echo ' < ? xml version = " 1.0 " encoding = " UTF-8 " ?>
< Capabilities xmlns = " http://www.opengis.net/wmts/1.0 " xmlns : ows = " http://www.opengis.net/ows/1.1 " xmlns : xlink = " http://www.w3.org/1999/xlink " xmlns : xsi = " http://www.w3.org/2001/XMLSchema-instance " xmlns : gml = " http://www.opengis.net/gml " xsi : schemaLocation = " http://www.opengis.net/wmts/1.0 http://schemas.opengis.net/wmts/1.0/wmtsGetCapabilities_response.xsd " version = " 1.0.0 " >
2014-04-11 10:51:19 +02:00
<!-- Service Identification -->
< ows : ServiceIdentification >
2014-04-28 14:12:51 +02:00
< ows : Title > tileserverphp </ ows : Title >
2014-04-11 10:51:19 +02:00
< ows : ServiceType > OGC WMTS </ ows : ServiceType >
< ows : ServiceTypeVersion > 1.0 . 0 </ ows : ServiceTypeVersion >
</ ows : ServiceIdentification >
<!-- Operations Metadata -->
< ows : OperationsMetadata >
< ows : Operation name = " GetCapabilities " >
< ows : DCP >
< ows : HTTP >
2014-08-04 17:47:47 +02:00
< ows : Get xlink : href = " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/wmts/1.0.0/WMTSCapabilities.xml " >
2014-04-11 10:51:19 +02:00
< ows : Constraint name = " GetEncoding " >
< ows : AllowedValues >
< ows : Value > RESTful </ ows : Value >
</ ows : AllowedValues >
</ ows : Constraint >
</ ows : Get >
<!-- add KVP binding in 10.1 -->
2014-08-04 17:47:47 +02:00
< ows : Get xlink : href = " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/wmts? " >
2014-04-11 10:51:19 +02:00
< ows : Constraint name = " GetEncoding " >
< ows : AllowedValues >
< ows : Value > KVP </ ows : Value >
</ ows : AllowedValues >
</ ows : Constraint >
</ ows : Get >
</ ows : HTTP >
</ ows : DCP >
</ ows : Operation >
< ows : Operation name = " GetTile " >
< ows : DCP >
< ows : HTTP >
2014-08-04 17:47:47 +02:00
< ows : Get xlink : href = " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/wmts/ " >
2014-04-11 10:51:19 +02:00
< ows : Constraint name = " GetEncoding " >
< ows : AllowedValues >
< ows : Value > RESTful </ ows : Value >
</ ows : AllowedValues >
</ ows : Constraint >
</ ows : Get >
2014-08-04 17:47:47 +02:00
< ows : Get xlink : href = " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/wmts? " >
2014-04-11 10:51:19 +02:00
< ows : Constraint name = " GetEncoding " >
< ows : AllowedValues >
< ows : Value > KVP </ ows : Value >
</ ows : AllowedValues >
</ ows : Constraint >
</ ows : Get >
</ ows : HTTP >
</ ows : DCP >
</ ows : Operation >
</ ows : OperationsMetadata >
< Contents > ' ;
2016-02-17 20:29:39 +01:00
2016-02-09 14:36:24 +01:00
$customtileMatrixSets = '' ;
2016-03-21 13:40:33 +01:00
$maxMercatorZoom = 18 ;
2016-02-17 20:29:39 +01:00
2016-02-03 16:50:07 +01:00
//layers
2016-02-17 21:19:16 +01:00
foreach ( $layers as $m ) {
$basename = $m [ 'basename' ];
2014-04-11 10:51:19 +02:00
$title = ( array_key_exists ( 'name' , $m )) ? $m [ 'name' ] : $basename ;
$profile = $m [ 'profile' ];
$bounds = $m [ 'bounds' ];
2015-12-04 14:00:48 +01:00
$format = $m [ 'format' ] == 'hybrid' ? 'jpgpng' : $m [ 'format' ];
2015-12-01 23:03:54 +01:00
$mime = ( $format == 'jpg' ) ? 'image/jpeg' : 'image/' . $format ;
2016-02-17 20:29:39 +01:00
2014-04-11 10:51:19 +02:00
if ( $profile == 'geodetic' ) {
2016-02-17 21:29:53 +01:00
$tileMatrixSet = 'WGS84' ;
2016-02-03 16:50:07 +01:00
} elseif ( $m [ 'profile' ] == 'custom' ) {
$crs = explode ( ':' , $m [ 'crs' ]);
2016-02-18 16:49:48 +01:00
$tileMatrixSet = 'custom' . $crs [ 1 ] . $m [ 'basename' ];
2016-02-17 20:29:39 +01:00
$customtileMatrixSets .= $this -> getTileMatrixSet (
2016-02-17 21:19:16 +01:00
$tileMatrixSet ,
$m [ 'tile_matrix' ],
2016-02-17 20:29:39 +01:00
$m [ 'crs' ]
);
2014-04-11 10:51:19 +02:00
} else {
2016-02-17 21:29:53 +01:00
$tileMatrixSet = 'GoogleMapsCompatible' ;
2016-03-21 13:40:33 +01:00
$maxMercatorZoom = max ( $maxMercatorZoom , $m [ 'maxzoom' ]);
2014-04-11 10:51:19 +02:00
}
2016-03-01 11:57:18 +01:00
2016-02-29 12:16:24 +01:00
$wmtsHost = substr ( $m [ 'tiles' ][ 0 ], 0 , strpos ( $m [ 'tiles' ][ 0 ], $m [ 'basename' ]));
2016-03-07 14:57:29 +01:00
$resourceUrlTemplate = $wmtsHost . $basename
. '/{TileMatrix}/{TileCol}/{TileRow}' ;
2015-12-04 14:00:48 +01:00
if ( strlen ( $format ) <= 4 ){
$resourceUrlTemplate .= '.' . $format ;
}
2016-03-01 11:57:18 +01:00
2014-04-28 14:12:51 +02:00
echo '
< Layer >
2014-04-11 10:51:19 +02:00
< ows : Title > ' . $title . ' </ ows : Title >
2014-04-28 14:12:51 +02:00
< ows : Identifier > ' . $basename . ' </ ows : Identifier >
< ows : WGS84BoundingBox crs = " urn:ogc:def:crs:OGC:2:84 " >
2014-04-11 10:51:19 +02:00
< ows : LowerCorner > ' . $bounds[0] . ' ' . $bounds[1] . ' </ ows : LowerCorner >
< ows : UpperCorner > ' . $bounds[2] . ' ' . $bounds[3] . ' </ ows : UpperCorner >
</ ows : WGS84BoundingBox >
< Style isDefault = " true " >
< ows : Identifier > default </ ows : Identifier >
</ Style >
< Format > ' . $mime . ' </ Format >
< TileMatrixSetLink >
< TileMatrixSet > ' . $tileMatrixSet . ' </ TileMatrixSet >
</ TileMatrixSetLink >
2015-12-04 14:00:48 +01:00
< ResourceURL format = " ' . $mime . ' " resourceType = " tile " template = " ' . $resourceUrlTemplate . ' " />
2014-04-11 10:51:19 +02:00
</ Layer > ' ;
2012-08-12 15:47:27 +02:00
}
2016-02-26 08:51:47 +01:00
2016-02-24 18:55:42 +01:00
// Print custom TileMatrixSets
if ( strlen ( $customtileMatrixSets ) > 0 ) {
echo $customtileMatrixSets ;
}
2016-02-17 21:19:16 +01:00
2016-02-09 14:36:24 +01:00
// Print PseudoMercator TileMatrixSet
2016-03-21 10:59:29 +01:00
echo $this -> getMercatorTileMatrixSet ( $maxMercatorZoom );
2016-02-09 14:36:24 +01:00
2016-02-17 20:29:39 +01:00
// Print WGS84 TileMatrixSet
echo $this -> getWGS84TileMatrixSet ();
2016-02-03 15:13:56 +01:00
echo ' </ Contents >
2014-08-04 17:47:47 +02:00
< ServiceMetadataURL xlink : href = " ' . $this->config ['protocol'] . '://' . $this->config ['baseUrls'][0] . '/wmts/1.0.0/WMTSCapabilities.xml " />
2014-04-11 10:51:19 +02:00
</ Capabilities > ' ;
}
/**
* Returns tile via WMTS specification
*/
public function getTile () {
$request = $this -> getGlobal ( 'Request' );
2014-04-14 12:29:03 +02:00
if ( $request ) {
2014-04-11 10:51:19 +02:00
if ( strpos ( '/' , $_GET [ 'Format' ]) !== FALSE ) {
$format = explode ( '/' , $_GET [ 'Format' ]);
2014-04-24 14:56:01 +02:00
$format = $format [ 1 ];
2014-04-11 10:51:19 +02:00
} else {
$format = $this -> getGlobal ( 'Format' );
}
2015-12-04 14:00:48 +01:00
parent :: renderTile (
2016-01-26 22:55:34 +01:00
$this -> getGlobal ( 'Layer' ),
$this -> getGlobal ( 'TileMatrix' ),
$this -> getGlobal ( 'TileRow' ),
$this -> getGlobal ( 'TileCol' ),
2015-12-04 14:00:48 +01:00
$format
);
2014-04-11 10:51:19 +02:00
} else {
2014-06-05 21:54:40 +02:00
parent :: renderTile ( $this -> layer , $this -> z , $this -> y , $this -> x , $this -> ext );
2014-04-11 10:51:19 +02:00
}
}
2012-08-12 15:47:27 +02:00
}
2014-04-11 10:51:19 +02:00
/**
* Tile map service
*/
class Tms extends Server {
/**
* @ param array $params
*/
public $layer ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $z ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $y ;
/**
2016-01-26 22:55:34 +01:00
* @ var integer
2014-04-11 10:51:19 +02:00
*/
public $x ;
/**
2016-01-26 22:55:34 +01:00
* @ var string
2014-04-11 10:51:19 +02:00
*/
public $ext ;
/**
2016-01-26 22:55:34 +01:00
*
2014-04-11 10:51:19 +02:00
* @ param array $params
*/
public function __construct ( $params ) {
parent :: __construct ();
parent :: setParams ( $params );
}
/**
* Returns getCapabilities metadata request
*/
public function getCapabilities () {
2014-04-25 10:22:02 +02:00
parent :: setDatasets ();
2014-04-11 10:51:19 +02:00
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
2016-02-17 21:29:53 +01:00
header ( 'Content-type: application/xml' );
2014-04-11 10:51:19 +02:00
echo '<TileMapService version="1.0.0"><TileMaps>' ;
foreach ( $maps as $m ) {
$basename = $m [ 'basename' ];
$title = ( array_key_exists ( 'name' , $m ) ) ? $m [ 'name' ] : $basename ;
$profile = $m [ 'profile' ];
if ( $profile == 'geodetic' ) {
2016-02-17 21:29:53 +01:00
$srs = 'EPSG:4326' ;
2014-04-11 10:51:19 +02:00
} else {
2016-02-17 21:29:53 +01:00
$srs = 'EPSG:3857' ;
2014-04-11 10:51:19 +02:00
}
2016-08-26 11:12:43 -04:00
$url = $this -> config [ 'protocol' ] . '://' . $this -> config [ 'baseUrls' ][ 0 ]
2016-03-04 13:20:46 +01:00
. '/tms/' . $basename ;
echo '<TileMap title="' . $title . '" srs="' . $srs
. '" type="InvertedTMS" ' . 'profile="global-' . $profile
. '" href="' . $url . '" />' ;
2014-04-11 10:51:19 +02:00
}
echo '</TileMaps></TileMapService>' ;
}
/**
* Prints metadata about layer
*/
public function getLayerCapabilities () {
2014-04-25 10:22:02 +02:00
parent :: setDatasets ();
2014-04-11 10:51:19 +02:00
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
foreach ( $maps as $map ) {
if ( strpos ( $map [ 'basename' ], $this -> layer ) !== false ) {
$m = $map ;
break ;
}
}
$title = ( array_key_exists ( 'name' , $m )) ? $m [ 'name' ] : $m [ 'basename' ];
$description = ( array_key_exists ( 'description' , $m )) ? $m [ 'description' ] : " " ;
$bounds = $m [ 'bounds' ];
if ( $m [ 'profile' ] == 'geodetic' ) {
2016-02-17 21:29:53 +01:00
$srs = 'EPSG:4326' ;
2016-03-04 13:20:46 +01:00
$initRes = 0.703125 ;
} elseif ( $m [ 'profile' ] == 'custom' ) {
$srs = $m [ 'crs' ];
$bounds = $m [ 'extent' ];
if ( isset ( $m [ 'tile_matrix' ][ 0 ][ 'pixel_size' ][ 0 ])){
$initRes = $m [ 'tile_matrix' ][ 0 ][ 'pixel_size' ][ 0 ];
} else {
$initRes = 1 ;
}
2014-04-11 10:51:19 +02:00
} else {
2016-02-17 21:29:53 +01:00
$srs = 'EPSG:3857' ;
2016-03-04 13:20:46 +01:00
$bounds = array ( - 20037508.34 , - 20037508.34 , 20037508.34 , 20037508.34 );
$initRes = 156543.03392804062 ;
2014-04-11 10:51:19 +02:00
}
$mime = ( $m [ 'format' ] == 'jpg' ) ? 'image/jpeg' : 'image/png' ;
header ( " Content-type: application/xml " );
2016-03-04 13:20:46 +01:00
$serviceUrl = $this -> config [ 'protocol' ] . '://' . $this -> config [ 'baseUrls' ][ 0 ] . '/' . $m [ 'basename' ];
echo '<TileMap version="1.0.0" tilemapservice="' . $serviceUrl . ' " type= " InvertedTMS " >
2014-04-11 10:51:19 +02:00
< Title > ' . htmlspecialchars($title) . ' </ Title >
< Abstract > ' . htmlspecialchars($description) . ' </ Abstract >
< SRS > ' . $srs . ' </ SRS >
< BoundingBox minx = " ' . $bounds[0] . ' " miny = " ' . $bounds[1] . ' " maxx = " ' . $bounds[2] . ' " maxy = " ' . $bounds[3] . ' " />
2016-03-04 13:20:46 +01:00
< Origin x = " ' . $bounds[0] . ' " y = " ' . $bounds[1] . ' " />
2014-04-11 10:51:19 +02:00
< TileFormat width = " 256 " height = " 256 " mime - type = " ' . $mime . ' " extension = " ' . $m['format'] . ' " />
< TileSets profile = " global-' . $m['profile'] . ' " > ' ;
for ( $zoom = $m [ 'minzoom' ]; $zoom < $m [ 'maxzoom' ] + 1 ; $zoom ++ ) {
2016-03-04 13:20:46 +01:00
$res = $initRes / pow ( 2 , $zoom );
$url = $this -> config [ 'protocol' ] . '://' . $this -> config [ 'baseUrls' ][ 0 ] . '/' . $m [ 'basename' ] . '/' . $zoom ;
echo '<TileSet href="' . $url . '" units-per-pixel="' . $res . '" order="' . $zoom . '" />' ;
2014-04-11 10:51:19 +02:00
}
echo '</TileSets></TileMap>' ;
}
/**
* Process getTile request
*/
public function getTile () {
2014-06-05 21:54:40 +02:00
parent :: renderTile ( $this -> layer , $this -> z , $this -> y , $this -> x , $this -> ext );
2014-04-11 10:51:19 +02:00
}
2012-08-12 15:47:27 +02:00
}
2014-04-11 10:51:19 +02:00
/**
* Simple router
*/
class Router {
/**
* @ param array $routes
*/
2014-04-28 14:12:51 +02:00
public static function serve ( $routes ) {
2014-04-11 10:51:19 +02:00
$request_method = strtolower ( $_SERVER [ 'REQUEST_METHOD' ]);
$path_info = '/' ;
2014-08-04 17:47:47 +02:00
global $config ;
$config [ 'protocol' ] = ( isset ( $_SERVER [ " HTTPS " ]) or $_SERVER [ 'SERVER_PORT' ] == '443' ) ? " https " : " http " ;
2014-04-11 10:51:19 +02:00
if ( ! empty ( $_SERVER [ 'PATH_INFO' ])) {
$path_info = $_SERVER [ 'PATH_INFO' ];
2016-03-04 13:28:05 +01:00
} else if ( ! empty ( $_SERVER [ 'ORIG_PATH_INFO' ]) && strpos ( $_SERVER [ 'ORIG_PATH_INFO' ], 'tileserver.php' ) === false ) {
2014-04-11 10:51:19 +02:00
$path_info = $_SERVER [ 'ORIG_PATH_INFO' ];
2014-05-19 19:47:11 +02:00
} else if ( ! empty ( $_SERVER [ 'REQUEST_URI' ]) && strpos ( $_SERVER [ 'REQUEST_URI' ], '/tileserver.php' ) !== false ) {
2014-05-19 17:55:00 +02:00
$path_info = $_SERVER [ 'HTTP_HOST' ] . $_SERVER [ 'REQUEST_URI' ];
$config [ 'baseUrls' ][ 0 ] = $_SERVER [ 'HTTP_HOST' ] . $_SERVER [ 'REQUEST_URI' ] . '?' ;
2014-04-11 10:51:19 +02:00
} else {
if ( ! empty ( $_SERVER [ 'REQUEST_URI' ])) {
$path_info = ( strpos ( $_SERVER [ 'REQUEST_URI' ], '?' ) > 0 ) ? strstr ( $_SERVER [ 'REQUEST_URI' ], '?' , true ) : $_SERVER [ 'REQUEST_URI' ];
}
}
$discovered_handler = null ;
$regex_matches = array ();
2014-04-28 14:12:51 +02:00
2014-04-24 14:56:01 +02:00
if ( $routes ) {
2014-04-11 10:51:19 +02:00
$tokens = array (
':string' => '([a-zA-Z]+)' ,
':number' => '([0-9]+)' ,
2016-01-25 18:32:07 +01:00
':alpha' => '([a-zA-Z0-9-_@\.]+)'
2014-04-11 10:51:19 +02:00
);
2014-08-04 17:47:47 +02:00
//global $config;
2014-04-11 10:51:19 +02:00
foreach ( $routes as $pattern => $handler_name ) {
$pattern = strtr ( $pattern , $tokens );
2014-04-24 14:56:01 +02:00
if ( preg_match ( '#/?' . $pattern . '/?$#' , $path_info , $matches )) {
2014-04-28 14:12:51 +02:00
if ( ! isset ( $config [ 'baseUrls' ])) {
$config [ 'baseUrls' ][ 0 ] = $_SERVER [ 'HTTP_HOST' ] . preg_replace ( '#/?' . $pattern . '/?$#' , '' , $path_info );
2014-04-24 14:56:01 +02:00
}
2014-04-11 10:51:19 +02:00
$discovered_handler = $handler_name ;
$regex_matches = $matches ;
break ;
}
}
}
$handler_instance = null ;
if ( $discovered_handler ) {
if ( is_string ( $discovered_handler )) {
if ( strpos ( $discovered_handler , ':' ) !== false ) {
$discoverered_class = explode ( ':' , $discovered_handler );
$discoverered_method = explode ( ':' , $discovered_handler );
$handler_instance = new $discoverered_class [ 0 ]( $regex_matches );
call_user_func ( array ( $handler_instance , $discoverered_method [ 1 ]));
} else {
$handler_instance = new $discovered_handler ( $regex_matches );
}
} elseif ( is_callable ( $discovered_handler )) {
$handler_instance = $discovered_handler ();
}
} else {
2014-05-19 19:47:11 +02:00
if ( ! isset ( $config [ 'baseUrls' ][ 0 ])) {
2016-03-04 13:28:05 +01:00
$config [ 'baseUrls' ][ 0 ] = $_SERVER [ 'HTTP_HOST' ] . $_SERVER [ 'REQUEST_URI' ];
2014-05-19 19:47:11 +02:00
}
if ( strpos ( $_SERVER [ 'REQUEST_URI' ], '=' ) != FALSE ) {
$kvp = explode ( '=' , $_SERVER [ 'REQUEST_URI' ]);
$_GET [ 'callback' ] = $kvp [ 1 ];
$params [ 0 ] = 'index' ;
$handler_instance = new Json ( $params );
$handler_instance -> getJson ();
2014-05-01 10:25:33 +02:00
}
2014-05-19 17:55:00 +02:00
$handler_instance = new Server ;
$handler_instance -> getHtml ();
2014-04-11 10:51:19 +02:00
}
2014-05-19 17:55:00 +02:00
}
2012-08-12 15:47:27 +02:00
}