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 /
2014-04-11 10:51:19 +02:00
* Copyright ( C ) 2014 - Klokan Technologies GmbH
2012-08-12 15:47:27 +02:00
*/
2014-04-23 14:30:27 +02:00
global $config ;
2014-04-28 14:12:51 +02:00
$config [ 'serverTitle' ] = 'TileServer-php v1' ;
2014-04-23 14:30:27 +02:00
//$config['baseUrls'] = ['t0.server.com', 't1.server.com'];
2014-04-11 10:51:19 +02:00
Router :: serve ( array (
2014-04-11 11:08:04 +02:00
'/' => 'Server:getHtml' ,
2014-04-11 10:51:19 +02:00
'/test' => 'Server:getInfo' ,
'/html' => 'Server:getHtml' ,
2014-05-19 18:01:11 +02:00
'/:alpha.json' => 'Json:getJson' ,
'/:alpha.jsonp' => 'Json:getJsonp' ,
'/:alpha/:number/:number/:number.:alpha.json' => 'Json:getUTFGrid' ,
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' ,
2014-05-19 18:01:11 +02:00
'/wmts/:alpha/:number/:number/:number.:alpha' => 'Wmts:getTile' ,
'/wmts/:alpha/:alpha/:number/:number/:number.:alpha' => 'Wmts:getTile' ,
'/wmts/:alpha/:alpha/:alpha/:number/:number/:number.:alpha' => 'Wmts:getTile' ,
'/:alpha/:number/: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 ]
2014-04-11 10:51:19 +02:00
* @ var array
*/
public $config ;
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* Datasets stored in file structure
* @ var array
*/
public $fileLayer = array ();
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* Datasets stored in database
* @ var array
*/
public $dbLayer = array ();
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
/**
* PDO database connection
* @ var object
*/
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 () {
2014-04-23 14:30:27 +02:00
$this -> config = $GLOBALS [ 'config' ];
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' );
$mbts = glob ( '*.mbtiles' );
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 );
if ( isset ( $params [ 3 ])) {
$this -> z = $params [ 3 ];
$this -> x = $params [ 2 ];
$this -> y = $params [ 1 ];
2014-04-11 10:51:19 +02:00
}
2014-04-28 14:12:51 +02:00
if ( isset ( $params [ 0 ])) {
$this -> ext = $params [ 0 ];
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 ) {
2014-04-28 14:12:51 +02:00
if ( is_file ( $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
}
}
/**
*
* @ param string $jsonFileName
* @ return array
*/
public function metadataFromMetadataJson ( $jsonFileName ) {
$metadata = json_decode ( file_get_contents ( $jsonFileName ), true );
$metadata = $this -> metadataValidation ( $metadata );
$metadata [ 'basename' ] = str_replace ( '/metadata.json' , '' , $jsonFileName );
return $metadata ;
}
/**
* 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 ) {
2014-06-05 21:46:00 +02:00
$value = preg_replace ( '/(\\n)+/' , '' , $r [ 'value' ]);
$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 ();
if ( ! array_key_exists ( 'minzoom' , $metadata ))
$metadata [ 'minzoom' ] = $resultdata [ 0 ][ 'min' ];
if ( ! array_key_exists ( 'maxzoom' , $metadata ))
$metadata [ 'maxzoom' ] = $resultdata [ 0 ][ 'max' ];
}
// 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 ();
$w = - 180 + 360 * ( $resultdata [ 0 ][ 'w' ] / pow ( 2 , $metadata [ 'maxzoom' ]));
$e = - 180 + 360 * (( 1 + $resultdata [ 0 ][ 'e' ]) / pow ( 2 , $metadata [ 'maxzoom' ]));
$n = $this -> row2lat ( $resultdata [ 0 ][ 'n' ], $metadata [ 'maxzoom' ]);
$s = $this -> row2lat ( $resultdata [ 0 ][ 's' ] - 1 , $metadata [ 'maxzoom' ]);
$metadata [ 'bounds' ] = implode ( ',' , array ( $w , $s , $e , $n ));
}
2014-04-11 10:51:19 +02:00
$metadata = $this -> metadataValidation ( $metadata );
$mbt = explode ( '.' , $mbt );
$metadata [ 'basename' ] = $mbt [ 0 ];
return $metadata ;
}
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 ) {
$y = $r / pow ( 2 , $zoom - 1 ) - 1 ;
return rad2deg ( 2.0 * atan ( exp ( 3.191459196 * $y )) - 1.57079632679489661922 );
}
2014-04-11 10:51:19 +02:00
/**
* Valids metaJSON
* @ param object $metadata
* @ return object
*/
public function metadataValidation ( $metadata ) {
if ( array_key_exists ( 'bounds' , $metadata )) {
$metadata [ 'bounds' ] = array_map ( 'floatval' , explode ( ',' , $metadata [ 'bounds' ]));
2014-04-23 14:30:27 +02:00
} else {
$metadata [ 'bounds' ] = array ( - 180 , - 85.051128779807 , 180 , 85.051128779807 );
2014-04-11 10:51:19 +02:00
}
if ( ! array_key_exists ( 'profile' , $metadata )) {
$metadata [ 'profile' ] = 'mercator' ;
}
2014-06-20 23:15:01 +02:00
// TODO: detect thumb / SQL for mbtiles
2014-04-11 10:51:19 +02:00
if ( array_key_exists ( 'minzoom' , $metadata ))
$metadata [ 'minzoom' ] = intval ( $metadata [ 'minzoom' ]);
else
$metadata [ 'minzoom' ] = 0 ;
if ( array_key_exists ( 'maxzoom' , $metadata ))
$metadata [ 'maxzoom' ] = intval ( $metadata [ 'maxzoom' ]);
else
$metadata [ 'maxzoom' ] = 18 ;
if ( ! array_key_exists ( 'format' , $metadata )) {
$metadata [ 'format' ] = 'png' ;
}
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 ) {
$filename = $filename . '.mbtiles' ;
$lastModifiedTime = filemtime ( $filename );
2014-04-28 14:24:25 +02:00
$eTag = md5 ( $lastModifiedTime );
2014-04-11 10:51:19 +02:00
header ( " Last-Modified: " . gmdate ( " D, d M Y H:i:s " , $lastModifiedTime ) . " GMT " );
header ( " Etag: " . $eTag );
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
}
$this -> DBconnect ( $tileset . '.mbtiles' );
$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 ) {
$this -> getCleanTile ();
} 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' ;
}
2014-06-27 09:19:26 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2014-04-11 10:51:19 +02:00
header ( 'Content-type: image/' . $format );
echo $data ;
}
2014-04-23 17:01:08 +02:00
} elseif ( $this -> isFileLayer ( $tileset )) {
2014-04-28 14:12:51 +02:00
$name = './' . $tileset . '/' . $z . '/' . $x . '/' . $y . '.' . $ext ;
2014-04-11 10:51:19 +02:00
if ( $fp = @ fopen ( $name , 'rb' )) {
2014-06-27 09:19:26 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2014-04-11 10:51:19 +02:00
header ( 'Content-Type: image/' . $ext );
header ( 'Content-Length: ' . filesize ( $name ));
fpassthru ( $fp );
die ;
} else {
$this -> getCleanTile ();
}
} else {
2014-04-14 12:29:03 +02:00
echo 'Server: Unknown or not specified dataset' ;
2014-04-11 10:51:19 +02:00
die ;
}
}
2014-04-11 11:08:04 +02:00
/**
* Returns clean tile
*/
2014-04-11 10:51:19 +02:00
public function getCleanTile () {
$png = imagecreatetruecolor ( 256 , 256 );
imagesavealpha ( $png , true );
$trans_colour = imagecolorallocatealpha ( $png , 0 , 0 , 0 , 127 );
imagefill ( $png , 0 , 0 , $trans_colour );
2014-06-27 09:19:26 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2014-04-11 10:51:19 +02:00
header ( 'Content-type: image/png' );
imagepng ( $png );
die ;
}
/**
* 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 {
2014-04-28 14:12:51 +02:00
$this -> DBconnect ( $tileset . '.mbtiles' );
2014-04-11 10:51:19 +02:00
$result = $this -> db -> query ( 'SELECT grid FROM grids WHERE tile_column = ' . $x . ' AND tile_row = ' . $y . ' AND zoom_level = ' . $z );
2014-04-28 14:12:51 +02:00
if ( ! isset ( $result ) || $result === FALSE ) {
2014-04-11 10:51:19 +02:00
header ( 'Access-Control-Allow-Origin: *' );
2014-05-19 18:58:54 +02:00
echo '{}' ;
2014-04-11 10:51:19 +02:00
die ;
} else {
2014-04-28 14:12:51 +02:00
$data = $result -> fetchColumn ();
2014-04-11 10:51:19 +02:00
$grid = gzuncompress ( $data );
$grid = substr ( trim ( $grid ), 0 , - 1 );
//adds legend (data) to output
$grid .= ',"data":{' ;
$result = $this -> db -> query ( 'SELECT key_name as key, key_json as json FROM grid_data WHERE zoom_level=' . $z . ' and tile_column=' . $x . ' and tile_row=' . $y );
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' ])) {
2014-04-11 10:51:19 +02:00
header ( " Content-Type:text/javascript charset=utf-8 " );
echo $_GET [ 'callback' ] . '(' . $grid . ');' ;
} else {
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
}
}
} catch ( PDOException $e ) {
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 () {
2014-06-05 22:04:18 +02:00
// echo $this->config['baseUrls'][0];die;
2014-04-11 10:51:19 +02:00
$this -> setDatasets ();
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
header ( 'Content-Type: text/html;charset=UTF-8' );
echo '<!DOCTYPE html><html><head><meta charset="UTF-8"><title>' . $this -> config [ 'serverTitle' ] . '</title></head><body>' ;
2014-04-28 14:12:51 +02:00
echo '<h1>' . $this -> config [ 'serverTitle' ] . '</h1>' ;
2014-06-05 22:04:18 +02:00
echo 'TileJSON service: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/index.json">' . $this -> config [ 'baseUrls' ][ 0 ] . '/index.json</a><br>' ;
echo 'WMTS service: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/wmts">' . $this -> config [ 'baseUrls' ][ 0 ] . '/wmts</a><br>' ;
echo '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 ) {
$extend = '[' ;
foreach ( $map [ 'bounds' ] as $ext ) {
$extend = $extend . ' ' . $ext ;
}
$extend = $extend . ' ]' ;
if ( strpos ( $map [ 'basename' ], 'mbtiles' ) !== false ) {
echo '<p>Available MBtiles tileset: ' . $map [ 'basename' ] . '<br>' ;
} else {
echo '<p>Available file tileset: ' . $map [ 'basename' ] . '<br>' ;
}
2014-06-05 22:04:18 +02:00
echo 'Metadata: <a href="//' . $this -> config [ 'baseUrls' ][ 0 ] . '/' . $map [ 'basename' ] . '.json">'
2014-04-28 14:12:51 +02:00
. $this -> config [ 'baseUrls' ][ 0 ] . '/' . $map [ 'basename' ] . '.json</a><br>' ;
2014-04-11 10:51:19 +02:00
echo 'Bounds: ' . $extend . '</p>' ;
}
2014-04-28 14:12:51 +02:00
echo '<p>Copyright (C) 2014 - 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 );
header ( 'Content-Type: text/html;charset=UTF-8' );
echo '<!DOCTYPE html><html><head><meta charset="UTF-8"><title>' . $this -> config [ 'serverTitle' ] . '</title>' ;
2014-04-24 14:56:01 +02:00
echo ' < link rel = " stylesheet " type = " text/css " href = " //tileserver.com/v1/index.css " />
< script src = " //tileserver.com/v1/index.js " ></ script >< body >
2014-05-01 10:25:33 +02:00
< script > tileserver ({ index : " http://' . $this->config ['baseUrls'][0] . '/index.json " , tilejson : " http://' . $this->config ['baseUrls'][0] . '/%n.json " , tms : " http://' . $this->config ['baseUrls'][0] . '/tms " , wmts : " http://' . $this->config ['baseUrls'][0] . '/wmts " }); </ script >
2014-04-11 10:51:19 +02:00
< 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>' ;
}
echo '</body></html>' ;
}
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
* @ var string
*/
private $callback = 'grid' ;
/**
* @ param array $params
*/
public $layer = 'index' ;
/**
* @ var integer
*/
public $z ;
/**
* @ var integer
*/
public $y ;
/**
* @ var integer
*/
public $x ;
/**
* @ var string
*/
public $ext ;
/**
*
* @ 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' ;
$tiles = array ();
2014-04-23 14:30:27 +02:00
foreach ( $this -> config [ 'baseUrls' ] as $url ) {
$tiles [] = 'http://' . $url . '/' . $metadata [ 'basename' ] . '/{z}/{x}/{y}.' . $metadata [ 'format' ];
}
2014-04-11 10:51:19 +02:00
$metadata [ 'tiles' ] = $tiles ;
2014-04-28 14:12:51 +02:00
if ( $this -> isDBLayer ( $metadata [ 'basename' ])) {
$this -> DBconnect ( $metadata [ 'basename' ] . '.mbtiles' );
2014-04-25 11:37:42 +02:00
$res = $this -> db -> query ( 'SELECT grid FROM grids LIMIT 1' );
2014-04-28 14:12:51 +02:00
if ( $res ) {
2014-04-25 11:37:42 +02:00
foreach ( $this -> config [ 'baseUrls' ] as $url ) {
$grids [] = 'http://' . $url . '/' . $metadata [ 'basename' ] . '/{z}/{x}/{y}.grid.json' ;
}
$metadata [ 'grids' ] = $grids ;
}
}
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: *' );
header ( " Content-Type:application/javascript 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: *' );
header ( " Content-Type:text/javascript charset=utf-8 " );
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 ;
/**
* @ var integer
*/
public $z ;
/**
* @ var integer
*/
public $y ;
/**
* @ var integer
*/
public $x ;
/**
* @ var string
*/
public $ext ;
/**
*
* @ 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 ();
}
}
2014-04-11 10:51:19 +02:00
/**
* Returns tilesets getCapabilities
*/
public function getCapabilities () {
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-04-23 14:30:27 +02:00
< ows : Get xlink : href = " http://' . $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-04-23 14:30:27 +02:00
< ows : Get xlink : href = " http://' . $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-04-28 14:12:51 +02:00
< ows : Get xlink : href = " http://' . $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-04-23 14:30:27 +02:00
< ows : Get xlink : href = " http://' . $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 > ' ;
$maps = array_merge ( $this -> fileLayer , $this -> dbLayer );
$mercator = new GlobalMercator ();
foreach ( $maps as $m ) {
if ( strpos ( $m [ 'basename' ], '.' ) !== false ) {
$basename = explode ( '.' , $m [ 'basename' ]);
} else {
$basename = $m [ 'basename' ];
}
$title = ( array_key_exists ( 'name' , $m )) ? $m [ 'name' ] : $basename ;
$profile = $m [ 'profile' ];
$bounds = $m [ 'bounds' ];
$format = $m [ 'format' ];
$mime = ( $format == 'jpg' ) ? 'image/jpeg' : 'image/png' ;
if ( $profile == 'geodetic' ) {
$tileMatrixSet = " WGS84 " ;
} else {
$tileMatrixSet = " GoogleMapsCompatible " ;
list ( $minx , $miny ) = $mercator -> LatLonToMeters ( $bounds [ 1 ], $bounds [ 0 ]);
list ( $maxx , $maxy ) = $mercator -> LatLonToMeters ( $bounds [ 3 ], $bounds [ 2 ]);
$bounds3857 = array ( $minx , $miny , $maxx , $maxy );
}
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 >
2014-04-23 14:30:27 +02:00
< ResourceURL format = " ' . $mime . ' " resourceType = " tile " template = " http://'
2014-04-28 14:12:51 +02:00
. $this -> config [ 'baseUrls' ][ 0 ] . '/wmts/' . $basename . '/{TileMatrixSet}/{TileMatrix}/{TileCol}/{TileRow}.' . $format . ' " />
2014-04-11 10:51:19 +02:00
</ Layer > ' ;
2012-08-12 15:47:27 +02:00
}
2014-04-28 14:12:51 +02:00
echo '
< TileMatrixSet >
2014-04-11 10:51:19 +02:00
< ows : Title > GoogleMapsCompatible </ ows : Title >
2014-04-28 14:12:51 +02:00
< ows : Abstract > the wellknown \ ' GoogleMapsCompatible\ ' tile matrix set defined by OGC WMTS specification </ ows : Abstract >
2014-04-11 10:51:19 +02:00
< ows : Identifier > GoogleMapsCompatible </ ows : Identifier >
< ows : SupportedCRS > urn : ogc : def : crs : EPSG : 6.18 : 3 : 3857 </ ows : SupportedCRS >
< WellKnownScaleSet > urn : ogc : def : wkss : OGC : 1.0 : GoogleMapsCompatible </ WellKnownScaleSet >
< TileMatrix >
< ows : Identifier > 0 </ ows : Identifier >
< ScaleDenominator > 559082264.0287178 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 1 </ MatrixWidth >
< MatrixHeight > 1 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 1 </ ows : Identifier >
< ScaleDenominator > 279541132.0143589 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 2 </ MatrixWidth >
< MatrixHeight > 2 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 2 </ ows : Identifier >
< ScaleDenominator > 139770566.0071794 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 4 </ MatrixWidth >
< MatrixHeight > 4 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 3 </ ows : Identifier >
< ScaleDenominator > 69885283.00358972 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 8 </ MatrixWidth >
< MatrixHeight > 8 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 4 </ ows : Identifier >
< ScaleDenominator > 34942641.50179486 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 16 </ MatrixWidth >
< MatrixHeight > 16 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 5 </ ows : Identifier >
< ScaleDenominator > 17471320.75089743 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 32 </ MatrixWidth >
< MatrixHeight > 32 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 6 </ ows : Identifier >
< ScaleDenominator > 8735660.375448715 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 64 </ MatrixWidth >
< MatrixHeight > 64 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 7 </ ows : Identifier >
< ScaleDenominator > 4367830.187724357 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 128 </ MatrixWidth >
< MatrixHeight > 128 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 8 </ ows : Identifier >
< ScaleDenominator > 2183915.093862179 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 256 </ MatrixWidth >
< MatrixHeight > 256 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 9 </ ows : Identifier >
< ScaleDenominator > 1091957.546931089 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 512 </ MatrixWidth >
< MatrixHeight > 512 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 10 </ ows : Identifier >
< ScaleDenominator > 545978.7734655447 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 1024 </ MatrixWidth >
< MatrixHeight > 1024 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 11 </ ows : Identifier >
< ScaleDenominator > 272989.3867327723 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 2048 </ MatrixWidth >
< MatrixHeight > 2048 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 12 </ ows : Identifier >
< ScaleDenominator > 136494.6933663862 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 4096 </ MatrixWidth >
< MatrixHeight > 4096 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 13 </ ows : Identifier >
< ScaleDenominator > 68247.34668319309 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 8192 </ MatrixWidth >
< MatrixHeight > 8192 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 14 </ ows : Identifier >
< ScaleDenominator > 34123.67334159654 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 16384 </ MatrixWidth >
< MatrixHeight > 16384 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 15 </ ows : Identifier >
< ScaleDenominator > 17061.83667079827 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 32768 </ MatrixWidth >
< MatrixHeight > 32768 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 16 </ ows : Identifier >
< ScaleDenominator > 8530.918335399136 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 65536 </ MatrixWidth >
< MatrixHeight > 65536 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 17 </ ows : Identifier >
< ScaleDenominator > 4265.459167699568 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 131072 </ MatrixWidth >
< MatrixHeight > 131072 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 18 </ ows : Identifier >
< ScaleDenominator > 2132.729583849784 </ ScaleDenominator >
< TopLeftCorner >- 20037508.34278925 20037508.34278925 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 262144 </ MatrixWidth >
< MatrixHeight > 262144 </ MatrixHeight >
</ TileMatrix >
</ TileMatrixSet >
< TileMatrixSet >
< ows : Identifier > WGS84 </ ows : Identifier >
< ows : Title > GoogleCRS84Quad </ ows : Title >
< ows : SupportedCRS > urn : ogc : def : crs : EPSG : 6.3 : 4326 </ ows : SupportedCRS >
< ows : BoundingBox crs = " urn:ogc:def:crs:EPSG:6.3:4326 " >
< LowerCorner >- 180.000000 - 90.000000 </ LowerCorner >
< UpperCorner > 180.000000 90.000000 </ UpperCorner >
</ ows : BoundingBox >
< WellKnownScaleSet > urn : ogc : def : wkss : OGC : 1.0 : GoogleCRS84Quad </ WellKnownScaleSet >
< TileMatrix >
< ows : Identifier > 0 </ ows : Identifier >
< ScaleDenominator > 279541132.01435887813568115234 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 2 </ MatrixWidth >
< MatrixHeight > 1 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 1 </ ows : Identifier >
< ScaleDenominator > 139770566.00717943906784057617 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 4 </ MatrixWidth >
< MatrixHeight > 2 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 2 </ ows : Identifier >
< ScaleDenominator > 69885283.00358971953392028809 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 8 </ MatrixWidth >
< MatrixHeight > 4 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 3 </ ows : Identifier >
< ScaleDenominator > 34942641.50179485976696014404 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 16 </ MatrixWidth >
< MatrixHeight > 8 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 4 </ ows : Identifier >
< ScaleDenominator > 17471320.75089742988348007202 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 32 </ MatrixWidth >
< MatrixHeight > 16 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 5 </ ows : Identifier >
< ScaleDenominator > 8735660.37544871494174003601 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 64 </ MatrixWidth >
< MatrixHeight > 32 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 6 </ ows : Identifier >
< ScaleDenominator > 4367830.18772435747087001801 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 128 </ MatrixWidth >
< MatrixHeight > 64 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 7 </ ows : Identifier >
< ScaleDenominator > 2183915.09386217873543500900 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 256 </ MatrixWidth >
< MatrixHeight > 128 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 8 </ ows : Identifier >
< ScaleDenominator > 1091957.54693108936771750450 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 512 </ MatrixWidth >
< MatrixHeight > 256 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 9 </ ows : Identifier >
< ScaleDenominator > 545978.77346554468385875225 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 1024 </ MatrixWidth >
< MatrixHeight > 512 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 10 </ ows : Identifier >
< ScaleDenominator > 272989.38673277234192937613 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 2048 </ MatrixWidth >
< MatrixHeight > 1024 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 11 </ ows : Identifier >
< ScaleDenominator > 136494.69336638617096468806 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 4096 </ MatrixWidth >
< MatrixHeight > 2048 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 12 </ ows : Identifier >
< ScaleDenominator > 68247.34668319308548234403 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 8192 </ MatrixWidth >
< MatrixHeight > 4096 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 13 </ ows : Identifier >
< ScaleDenominator > 34123.67334159654274117202 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 16384 </ MatrixWidth >
< MatrixHeight > 8192 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 14 </ ows : Identifier >
< ScaleDenominator > 17061.83667079825318069197 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 32768 </ MatrixWidth >
< MatrixHeight > 16384 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 15 </ ows : Identifier >
< ScaleDenominator > 8530.91833539912659034599 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 65536 </ MatrixWidth >
< MatrixHeight > 32768 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 16 </ ows : Identifier >
< ScaleDenominator > 4265.45916769956329517299 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 131072 </ MatrixWidth >
< MatrixHeight > 65536 </ MatrixHeight >
</ TileMatrix >
< TileMatrix >
< ows : Identifier > 17 </ ows : Identifier >
< ScaleDenominator > 2132.72958384978574031265 </ ScaleDenominator >
< TopLeftCorner > 90.000000 - 180.000000 </ TopLeftCorner >
< TileWidth > 256 </ TileWidth >
< TileHeight > 256 </ TileHeight >
< MatrixWidth > 262144 </ MatrixWidth >
< MatrixHeight > 131072 </ MatrixHeight >
</ TileMatrix >
</ TileMatrixSet >
</ Contents >
2014-04-23 14:30:27 +02:00
< ServiceMetadataURL xlink : href = " http://' . $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' );
}
2014-06-05 21:54:40 +02:00
parent :: renderTile ( $this -> getGlobal ( 'Layer' ), $this -> getGlobal ( 'TileMatrix' ), $this -> getGlobal ( 'TileRow' ), $this -> getGlobal ( 'TileCol' ), $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 ;
/**
* @ var integer
*/
public $z ;
/**
* @ var integer
*/
public $y ;
/**
* @ var integer
*/
public $x ;
/**
* @ var string
*/
public $ext ;
/**
*
* @ 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 );
header ( " Content-type: application/xml " );
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' ) {
$srs = " EPSG:4326 " ;
} else {
$srs = " EPSG:3857 " ;
echo '<TileMap title="' . $title . '" srs="' . $srs
. '" type="InvertedTMS" ' . 'profile="global-' . $profile
2014-04-24 14:56:01 +02:00
. '" href="http://' . $this -> config [ 'baseUrls' ][ 0 ] . '/tms/' . $basename . '" />' ;
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' ) {
$srs = " EPSG:4326 " ;
$originx = - 180.0 ;
$originy = - 90.0 ;
$initialResolution = 0.703125 ;
} else {
$srs = " EPSG:3857 " ;
$originx = - 20037508.342789 ;
$originy = - 20037508.342789 ;
$mercator = new GlobalMercator ();
list ( $minx , $miny ) = $mercator -> LatLonToMeters ( $bounds [ 1 ], $bounds [ 0 ]);
list ( $maxx , $maxy ) = $mercator -> LatLonToMeters ( $bounds [ 3 ], $bounds [ 2 ]);
$bounds = array ( $minx , $miny , $maxx , $maxy );
$initialResolution = 156543.03392804062 ;
}
$mime = ( $m [ 'format' ] == 'jpg' ) ? 'image/jpeg' : 'image/png' ;
header ( " Content-type: application/xml " );
2014-04-23 14:30:27 +02:00
echo '<TileMap version="1.0.0" tilemapservice="http://' . $this -> config [ 'baseUrls' ][ 0 ] . '/' . $m [ 'basename' ] . ' " 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] . ' " />
< Origin x = " ' . $originx . ' " y = " ' . $originy . ' " />
< 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 ++ ) {
2014-04-23 14:30:27 +02:00
echo '<TileSet href="http://' . $this -> config [ 'baseUrls' ] [ 0 ] . '/' . $m [ 'basename' ] . '/' . $zoom . '" units-per-pixel="' . $initialResolution / pow ( 2 , $zoom ) . '" 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
GlobalMapTiles - part of Aggregate Map Tools
Version 1.0
Copyright ( c ) 2009 The Bivings Group
All rights reserved .
Author : John Bafford
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
http :// www . bivings . com /
http :// bafford . com / softare / aggregate - map - tools /
2012-08-12 15:47:27 +02:00
2014-04-11 10:51:19 +02:00
Based on GDAL2Tiles / globalmaptiles . py
Original python version Copyright ( c ) 2008 Klokan Petr Pridal . All rights reserved .
http :// www . klokan . cz / projects / gdal2tiles /
Permission is hereby granted , free of charge , to any person obtaining a
copy of this software and associated documentation files ( the " Software " ),
to deal in the Software without restriction , including without limitation
the rights to use , copy , modify , merge , publish , distribute , sublic ense ,
and / or sell copies of the Software , and to permit persons to whom the
Software is furnished to do so , subject to the following conditions :
The abov
e copyright notice and this permission notice shall be included
in all copies or substantial portions of the Software .
THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS
OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING
FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE .
*/
class GlobalMercator {
var $tileSize ;
var $initialResolution ;
var $originShift ;
//Initialize the TMS Global Mercator pyramid
function __construct ( $tileSize = 256 ) {
$this -> tileSize = $tileSize ;
$this -> initialResolution = 2 * M_PI * 6378137 / $this -> tileSize ;
# 156543.03392804062 for tileSize 256 Pixels
$this -> originShift = 2 * M_PI * 6378137 / 2.0 ;
# 20037508.342789244
}
//Converts given lat/lon in WGS84 Datum to XY in Spherical Mercator EPSG:900913
function LatLonToMeters ( $lat , $lon ) {
$mx = $lon * $this -> originShift / 180.0 ;
$my = log ( tan (( 90 + $lat ) * M_PI / 360.0 )) / ( M_PI / 180.0 );
$my *= $this -> originShift / 180.0 ;
return array ( $mx , $my );
}
//Converts XY point from Spherical Mercator EPSG:900913 to lat/lon in WGS84 Datum
function MetersToLatLon ( $mx , $my ) {
$lon = ( $mx / $this -> originShift ) * 180.0 ;
$lat = ( $my / $this -> originShift ) * 180.0 ;
$lat = 180 / M_PI * ( 2 * atan ( exp ( $lat * M_PI / 180.0 )) - M_PI / 2.0 );
return array ( $lat , $lon );
}
//Converts pixel coordinates in given zoom level of pyramid to EPSG:900913
function PixelsToMeters ( $px , $py , $zoom ) {
$res = $this -> Resolution ( $zoom );
$mx = $px * $res - $this -> originShift ;
$my = $py * $res - $this -> originShift ;
return array ( $mx , $my );
}
//Converts EPSG:900913 to pyramid pixel coordinates in given zoom level
function MetersToPixels ( $mx , $my , $zoom ) {
$res = $this -> Resolution ( $zoom );
$px = ( $mx + $this -> originShift ) / $res ;
$py = ( $my + $this -> originShift ) / $res ;
return array ( $px , $py );
}
//Returns a tile covering region in given pixel coordinates
function PixelsToTile ( $px , $py ) {
$tx = ceil ( $px / $this -> tileSize ) - 1 ;
$ty = ceil ( $py / $this -> tileSize ) - 1 ;
return array ( $tx , $ty );
}
//Returns tile for given mercator coordinates
function MetersToTile ( $mx , $my , $zoom ) {
list ( $px , $py ) = $this -> MetersToPixels ( $mx , $my , $zoom );
return $this -> PixelsToTile ( $px , $py );
}
//Returns bounds of the given tile in EPSG:900913 coordinates
function TileBounds ( $tx , $ty , $zoom ) {
list ( $minx , $miny ) = $this -> PixelsToMeters ( $tx * $this -> tileSize , $ty * $this -> tileSize , $zoom );
list ( $maxx , $maxy ) = $this -> PixelsToMeters (( $tx + 1 ) * $this -> tileSize , ( $ty + 1 ) * $this -> tileSize , $zoom );
return array ( $minx , $miny , $maxx , $maxy );
}
//Returns bounds of the given tile in latutude/longitude using WGS84 datum
function TileLatLonBounds ( $tx , $ty , $zoom ) {
$bounds = $this -> TileBounds ( $tx , $ty , $zoom );
list ( $minLat , $minLon ) = $this -> MetersToLatLon ( $bounds [ 0 ], $bounds [ 1 ]);
list ( $maxLat , $maxLon ) = $this -> MetersToLatLon ( $bounds [ 2 ], $bounds [ 3 ]);
return array ( $minLat , $minLon , $maxLat , $maxLon );
}
//Resolution (meters/pixel) for given zoom level (measured at Equator)
function Resolution ( $zoom ) {
return $this -> initialResolution / ( 1 < $zoom );
}
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 = '/' ;
if ( ! empty ( $_SERVER [ 'PATH_INFO' ])) {
$path_info = $_SERVER [ 'PATH_INFO' ];
} else if ( ! empty ( $_SERVER [ 'ORIG_PATH_INFO' ]) && $_SERVER [ 'ORIG_PATH_INFO' ] !== '/tileserver.php' ) {
$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]+)' ,
2014-07-23 15:32:10 +02:00
':alpha' => '([a-zA-Z0-9-_@]+)'
2014-04-11 10:51:19 +02:00
);
2014-04-24 14:56:01 +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 ])) {
$config [ 'baseUrls' ][ 0 ] = $_SERVER [ 'HTTP_HOST' ] . $_SERVER [ 'REQUEST_URI' ] . '?' ;
}
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
}