- Partial redesigned of resolver away from exceptions and returning ERROR Values instead
- Simplification of tftp_provisioner.php - Refactored the filename cache into standalone abstract class which can be implemented using different backends - Check cache for duplicate files when adding and report Signed-off-by: Diederik de Groot <ddegroot@talon.nl>
This commit is contained in:
81
lib/resolveCache.php
Normal file
81
lib/resolveCache.php
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
<?php
|
||||||
|
abstract class resolveCache {
|
||||||
|
abstract protected function addFile($filename, $realpath);
|
||||||
|
abstract protected function removeFile($filename);
|
||||||
|
abstract protected function check($filename);
|
||||||
|
abstract protected function getPath($filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
class fileCache extends resolveCache {
|
||||||
|
private $_isDirty = false;
|
||||||
|
private $_cache = array();
|
||||||
|
private $_cache_file = NULL;
|
||||||
|
|
||||||
|
function __construct($filename) {
|
||||||
|
$this->_cache_file = $filename;
|
||||||
|
if (file_exists($this->_cache_file)) {
|
||||||
|
$this->_cache = unserialize(file_get_contents($this->_cache_file));
|
||||||
|
$this->_isDirty = false;
|
||||||
|
} else {
|
||||||
|
$this->_isDirty = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function __destruct() {
|
||||||
|
if ($this->_isDirty) {
|
||||||
|
/*if (!is_writable($this->_cache_file)) {
|
||||||
|
log_error_and_throw("Could not write to file '".$this->_cache_file."' at Resolver::destruct");
|
||||||
|
}*/
|
||||||
|
if (!file_put_contents($this->_cache_file, serialize($this->_cache))) {
|
||||||
|
log_error_and_throw("Could not write to file '".$this->_cache_file."' at Resolver::destruct");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function isDirty() {
|
||||||
|
return $this->_isDirty;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function check($filename) {
|
||||||
|
return array_key_exists($filename, $this->_cache);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function addFile($filename, $realpath) {
|
||||||
|
if ($this->check($filename))
|
||||||
|
log_error("Duplicate file:$filename"); /* should we prevent this ? */
|
||||||
|
$this->_cache[$filename] = $realpath;
|
||||||
|
$this->_isDirty =true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function removeFile($filename) {
|
||||||
|
if ($this->check($filename)) {
|
||||||
|
unset($this->_cache[$filename]);
|
||||||
|
$this->_isDirty = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getPath($filename) {
|
||||||
|
if ($this->check($filename)) {
|
||||||
|
return $this->_cache[$filename];
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function printCache() {
|
||||||
|
print_r($this->_cache);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
class sqliteCache extends resolveCache {
|
||||||
|
function __construct() {
|
||||||
|
}
|
||||||
|
function __destruct() {
|
||||||
|
}
|
||||||
|
public function addFile($filename, $realpath);
|
||||||
|
public function removeFile($filename);
|
||||||
|
public function check($filename);
|
||||||
|
public function getPath($filename);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
?>
|
147
lib/resolver.php
147
lib/resolver.php
@@ -1,6 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
include_once("config.php");
|
include_once("config.php");
|
||||||
include_once("utils.php");
|
include_once("utils.php");
|
||||||
|
include_once("resolveCache.php");
|
||||||
|
|
||||||
/* Todo:
|
/* Todo:
|
||||||
✔️ setup logging
|
✔️ setup logging
|
||||||
@@ -15,52 +16,46 @@ include_once("utils.php");
|
|||||||
|
|
||||||
- Could use some more test-cases, especially error ones
|
- Could use some more test-cases, especially error ones
|
||||||
*/
|
*/
|
||||||
|
class ResolveResult {
|
||||||
|
const Ok = 0;
|
||||||
|
const EmptyRequest = 1;
|
||||||
|
const RequestNotAString = 2;
|
||||||
|
const RequestContainsInvalidChar = 3;
|
||||||
|
const RequestContainsPathWalk = 4;
|
||||||
|
const FileNotFound = 5;
|
||||||
|
const InvalidFilename = 6;
|
||||||
|
const InvalidPath = 7;
|
||||||
|
}
|
||||||
|
|
||||||
class Resolver {
|
class Resolver {
|
||||||
private $isDirty = FALSE;
|
private $isDirty = FALSE;
|
||||||
private $cache = array();
|
private $cache;
|
||||||
private $config;
|
private $config;
|
||||||
//private $logger;
|
|
||||||
function __construct($config) {
|
function __construct($config) {
|
||||||
//global $logger;
|
|
||||||
$this->config = $config;
|
$this->config = $config;
|
||||||
//$this->logger = $logger;
|
$this->cache = new fileCache($this->config['main']['cache_filename']);
|
||||||
if(file_exists($this->config['main']['cache_filename'])) {
|
if ($this->cache->isDirty()) {
|
||||||
$this->cache = unserialize(file_get_contents($config['main']['cache_filename']));
|
$this->rebuildCache();
|
||||||
} else {
|
|
||||||
$this->buildCleanCache();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
function __destruct() {
|
|
||||||
// $this->printCache()
|
public function searchForFile($filename) {
|
||||||
if ($this->isDirty) {
|
|
||||||
if (!file_put_contents($this->config['main']['cache_filename'], serialize($this->cache))) {
|
|
||||||
$this->log_error_and_throw("Could not write to file '".$this->config['cache_filename']."' at Resolver::destruct");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function log_error_and_throw($message) {
|
|
||||||
global $logger;
|
|
||||||
$logger->log('LOG_ERROR', $message);
|
|
||||||
throw new Exception($message);
|
|
||||||
}
|
|
||||||
function log_debug($message) {
|
|
||||||
global $logger;
|
|
||||||
$logger->log('LOG_DEBUG', $message);
|
|
||||||
}
|
|
||||||
function searchForFile($filename) {
|
|
||||||
foreach($this->config['subdirs'] as $key => $value) {
|
foreach($this->config['subdirs'] as $key => $value) {
|
||||||
if ($key === "firmware" || $key === "tftproot" ) {
|
if ($key === "firmware" || $key === "tftproot" ) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
$path = realpath($this->config['main']['base_path'] . "/" . $value['path'] . "/$filename");
|
$path = realpath($this->config['main']['base_path'] . "/" . $value['path'] . "/$filename");
|
||||||
if (file_exists($path)) {
|
if (file_exists($path)) {
|
||||||
$this-> addFile($filename, $path);
|
$this->cache->addFile($filename, $path);
|
||||||
return $path;
|
return $path;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$this->log_error_and_throw("File '$filename' does not exist");
|
log_error("File '$filename' does not exist");
|
||||||
|
return ResolveResult::FileNotFound;
|
||||||
}
|
}
|
||||||
function buildCleanCache() {
|
|
||||||
|
public function rebuildCache() {
|
||||||
|
log_debug("Rebuilding Cache, standby...");
|
||||||
foreach($this->config['subdirs'] as $key =>$value) {
|
foreach($this->config['subdirs'] as $key =>$value) {
|
||||||
if ($key === "tftproot") {
|
if ($key === "tftproot") {
|
||||||
continue;
|
continue;
|
||||||
@@ -71,60 +66,62 @@ class Resolver {
|
|||||||
foreach ($iterator as $file) {
|
foreach ($iterator as $file) {
|
||||||
if ($file->isFile()) {
|
if ($file->isFile()) {
|
||||||
if ($value['strip']) {
|
if ($value['strip']) {
|
||||||
$this->addFile($file->getFileName(), $file->getPathname());
|
$this->cache->addFile($file->getFileName(), $file->getPathname());
|
||||||
} else {
|
} else {
|
||||||
$subdir = basename(dirname($file->getPathname()));
|
$subdir = basename(dirname($file->getPathname()));
|
||||||
$this->addFile('$subpath/'.$file->getFileName(), $file->getPathname());
|
$this->cache->addFile('$subpath/'.$file->getFileName(), $file->getPathname());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$this->isDirty = TRUE;
|
$this->isDirty = TRUE;
|
||||||
}
|
}
|
||||||
function addFile($requestpath, $truepath) {
|
|
||||||
//$this->logger->log('LOG_DEBUG', "Adding $requestpath");
|
public function validateRequest($request) {
|
||||||
$this->log_debug("Adding $requestpath");
|
|
||||||
$this->cache[$requestpath] = $truepath;
|
|
||||||
$this->isDirty =TRUE;
|
|
||||||
}
|
|
||||||
function removeFile($requestpath) {
|
|
||||||
$this->log_debug("Removing $hash");
|
|
||||||
unset($this->cache[$requestpath]);
|
|
||||||
$this->isDirty = TRUE;
|
|
||||||
}
|
|
||||||
function validateRequest($request) {
|
|
||||||
/* make sure request does not startwith or contain: "/", "../" or "/./" */
|
/* make sure request does not startwith or contain: "/", "../" or "/./" */
|
||||||
/* make sure request only starts with filename or one of $config[$subdir]['locale'] or $config[$subdir]['wallpaper'] */
|
/* make sure request only starts with filename or one of $config[$subdir]['locale'] or $config[$subdir]['wallpaper'] */
|
||||||
/* check uri/url decode */
|
/* check uri/url decode */
|
||||||
if (!is_string($request)) {
|
if (!$request || empty($request)) {
|
||||||
$this->log_error_and_throw("Request is not a string");
|
log_error("Request is empty");
|
||||||
|
return ResolveResult::EmptyRequest;
|
||||||
}
|
}
|
||||||
$this->log_debug($request . ":" . escapeshellarg($request) . ":" . utf8_urldecode($request) . "\n");
|
if (!is_string($request)) {
|
||||||
|
log_error("Request is not a string");
|
||||||
|
return ResolveResult::RequestNotAString;
|
||||||
|
}
|
||||||
|
log_debug($request . ":" . escapeshellarg($request) . ":" . utf8_urldecode($request) . "\n");
|
||||||
$escaped_request = escapeshellarg(utf8_urldecode($request));
|
$escaped_request = escapeshellarg(utf8_urldecode($request));
|
||||||
if ($escaped_request !== "'" . $request . "'") {
|
if ($escaped_request !== "'" . $request . "'") {
|
||||||
$this->log_error_and_throw("Request '$request' contains invalid characters");
|
log_error("Request '$request' contains invalid characters");
|
||||||
|
return ResolveResult::RequestContainsInvalidChar;
|
||||||
}
|
}
|
||||||
if (strstr($escaped_request, "..")) {
|
if (strstr($escaped_request, "..")) {
|
||||||
$this->log_error_and_throw("Request '$request' containst '..'");
|
log_error("Request '$request' contains '..'");
|
||||||
|
return ResolveResult::RequestContainsPathWalk;
|
||||||
}
|
}
|
||||||
|
return ResolveResult::Ok;
|
||||||
}
|
}
|
||||||
function resolve($request) /* canthrow */ {
|
|
||||||
$this->validateRequest($request);
|
public function resolve($request) /* canthrow */ {
|
||||||
$path = '';
|
$path = '';
|
||||||
if (array_key_exists($request, $this->cache)) {
|
$result = $this->validateRequest($request);
|
||||||
if ($path = $this->cache[$request]) {
|
if ($result !== ResolveResult::Ok) {
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
if (($path = $this->cache->getPath($request))) {
|
||||||
if (!file_exists($path)) {
|
if (!file_exists($path)) {
|
||||||
$this->removeFile($request);
|
$this->cache->removeFile($request);
|
||||||
$this->log_error_and_throw("File '$request' does not exist on FS");
|
log_error("File '$request' does not exist on FS");
|
||||||
|
return ResolveResult::FileNotFound;
|
||||||
}
|
}
|
||||||
return $path;
|
return $path;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if ($this->searchForFile($request)) {
|
if ($this->searchForFile($request)) {
|
||||||
return $this->cache[$request];
|
$path = $this->cache->getPath($request);
|
||||||
}
|
}
|
||||||
return $path;
|
return $path;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* temporary */
|
/* temporary */
|
||||||
function printCache() {
|
function printCache() {
|
||||||
print_r($this->cache);
|
print_r($this->cache);
|
||||||
@@ -134,33 +131,37 @@ class Resolver {
|
|||||||
// Testing
|
// Testing
|
||||||
if(defined('STDIN') ) {
|
if(defined('STDIN') ) {
|
||||||
$resolver = new Resolver($config);
|
$resolver = new Resolver($config);
|
||||||
|
|
||||||
$test_cases = Array(
|
$test_cases = Array(
|
||||||
Array('request' => 'jar70sccp.9-4-2ES26.sbn', 'expected' => '/tftpboot/firmware/7970/jar70sccp.9-4-2ES26.sbn', 'throws' => FALSE),
|
Array('request' => 'jar70sccp.9-4-2ES26.sbn', 'expected' => '/tftpboot/firmware/7970/jar70sccp.9-4-2ES26.sbn'),
|
||||||
Array('request' => 'Russian_Russian_Federation/be-sccp.jar', 'expected' => '/tftpboot/locales/languages/Russian_Russian_Federation/be-sccp.jar', 'throws' => FALSE),
|
Array('request' => 'Russian_Russian_Federation/be-sccp.jar', 'expected' => '/tftpboot/locales/languages/Russian_Russian_Federation/be-sccp.jar'),
|
||||||
Array('request' => 'Spain/g3-tones.xml', 'expected' => '/tftpboot/locales/countries/Spain/g3-tones.xml', 'throws' => FALSE),
|
Array('request' => 'Spain/g3-tones.xml', 'expected' => '/tftpboot/locales/countries/Spain/g3-tones.xml'),
|
||||||
Array('request' => '320x196x4/Chan-SCCP-b.png', 'expected' => '/tftpboot/wallpapers/320x196x4/Chan-SCCP-b.png', 'throws' => FALSE),
|
Array('request' => '320x196x4/Chan-SCCP-b.png', 'expected' => '/tftpboot/wallpapers/320x196x4/Chan-SCCP-b.png'),
|
||||||
Array('request' => 'XMLDefault.cnf.xml', 'expected' => '/tftpboot/settings/bak/XMLDefault.cnf.xml', 'throws' => FALSE),
|
Array('request' => 'XMLDefault.cnf.xml', 'expected' => '/tftpboot/settings/bak/XMLDefault.cnf.xml'),
|
||||||
Array('request' => '../XMLDefault.cnf.xml', 'expected' => '', 'throws' => TRUE),
|
Array('request' => '../XMLDefault.cnf.xml', 'expected' => ResolveResult::RequestContainsPathWalk),
|
||||||
Array('request' => 'XMLDefault.cnf.xml/../../text.xml', 'expected' => '', 'throws' => TRUE),
|
Array('request' => 'XMLDefault.cnf.xml/../../text.xml', 'expected' => ResolveResult::RequestContainsPathWalk),
|
||||||
|
|
||||||
);
|
);
|
||||||
foreach($test_cases as $test) {
|
foreach($test_cases as $test) {
|
||||||
try {
|
try {
|
||||||
$result = $resolver->resolve($test['request']);
|
$result = $resolver->resolve($test['request']);
|
||||||
if ($result !== $base_path . $test['expected']) {
|
if (is_string($result)) {
|
||||||
print("Error: expected result does not match what we got\n");
|
if ($result === $base_path . $test['expected']) {
|
||||||
print("request:'".$test['request']."', result:'" . $base_path . $test['expected'] . "'\n");
|
|
||||||
} else {
|
|
||||||
print("'" . $test['request'] . "' => '" . $result . "'\n");
|
print("'" . $test['request'] . "' => '" . $result . "'\n");
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
} catch (Exception $e) {
|
|
||||||
if (!$test['throws']) {
|
|
||||||
print("Error: request was expected to throw: $e\n");
|
|
||||||
} else {
|
} else {
|
||||||
print("'" . $test['request'] . "' => throws error as expected\n");
|
if ($result === $test['expected']) {
|
||||||
|
print("'" . $test['request'] . "' => '" . $result . "'\n");
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
print("Error: expected result does not match what we got\n");
|
||||||
|
print("request:'".$test['request']."'\n");
|
||||||
|
print("expected:'" . $base_path . $test['expected'] . "'\n");
|
||||||
|
print("result:'" . $result . "'\n\n");
|
||||||
|
} catch (Exception $e) {
|
||||||
|
print("'" . $test['request'] . "' => throws error as expected\n");
|
||||||
|
print("Exception: " . $e->getMessage() . "\n");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
unset($resolver);
|
unset($resolver);
|
||||||
#unlink($CACHEFILE_NAME);
|
#unlink($CACHEFILE_NAME);
|
||||||
|
@@ -50,4 +50,19 @@ function parse_ini_file_multi($file, $process_sections = false, $scanner_mode =
|
|||||||
}
|
}
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function log_debug($message) {
|
||||||
|
global $logger;
|
||||||
|
$logger->log('LOG_DEBUG', $message);
|
||||||
|
}
|
||||||
|
|
||||||
|
function log_error($message) {
|
||||||
|
global $logger;
|
||||||
|
$logger->log('LOG_ERROR', $message);
|
||||||
|
}
|
||||||
|
|
||||||
|
function log_error_and_throw($message) {
|
||||||
|
log_error($message);
|
||||||
|
throw new Exception($message);
|
||||||
|
}
|
||||||
?>
|
?>
|
||||||
|
@@ -8,7 +8,7 @@ class TFTPProvisioner extends TFTPServer
|
|||||||
{
|
{
|
||||||
private $_debug;
|
private $_debug;
|
||||||
private $_resolver;
|
private $_resolver;
|
||||||
private $_filename;
|
private $_settings_path;
|
||||||
|
|
||||||
function __construct($server_url, $config, $logger = NULL, $debug = false)
|
function __construct($server_url, $config, $logger = NULL, $debug = false)
|
||||||
{
|
{
|
||||||
@@ -20,42 +20,32 @@ class TFTPProvisioner extends TFTPServer
|
|||||||
$this->_debug = $debug;
|
$this->_debug = $debug;
|
||||||
$this->max_put_size = 60000000;
|
$this->max_put_size = 60000000;
|
||||||
$this->_resolver = new Resolver($config);
|
$this->_resolver = new Resolver($config);
|
||||||
}
|
$this->_settings_path = $this->_config['main']['base_path'] . DIRECTORY_SEPARATOR
|
||||||
|
. $this->_config['subdirs']['settings']['path'] . DIRECTORY_SEPARATOR;
|
||||||
public function exists($peer, $req_filename)
|
|
||||||
{
|
|
||||||
if (($this->_filename = $this->_resolver->resolve($req_filename))) {
|
|
||||||
return file_exists($this->_filename);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function readable($peer, $req_filename)
|
|
||||||
{
|
|
||||||
return is_readable($this->_filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
public function get($peer, $req_filename, $mode)
|
|
||||||
{
|
|
||||||
return file_get_contents($this->_filename);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public function writable($peer, $req_filename)
|
public function writable($peer, $req_filename)
|
||||||
{
|
{
|
||||||
// check $req_filename starts with 'settings/' (SPA phones can write to tftpboot)
|
$filename = $this->_settings_path . basename($req_filename);
|
||||||
$settings_path = $this->_config['main']['base_path'] . DIRECTORY_SEPARATOR
|
if (file_exists($filename)) {
|
||||||
. $this->_config['subdirs']['settings']['path'] . DIRECTORY_SEPARATOR;
|
return is_writable($filename);
|
||||||
$filename = $settings_path . basename($req_filename);
|
|
||||||
if (is_writable($filename) || (!file_exists($filename) && is_writable($settings_path))) {
|
|
||||||
$this->_filename = $filename;
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
return is_writable($this->_settings_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function get($peer, $req_filename, $mode)
|
||||||
|
{
|
||||||
|
$filename = $this->_resolver->resolve($req_filename);
|
||||||
|
if (file_exists($filename) && is_readable($filename))
|
||||||
|
return file_get_contents($filename);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function put($peer, $filename, $mode, $content)
|
public function put($peer, $req_filename, $mode, $content)
|
||||||
{
|
{
|
||||||
return file_put_contents($this->_filename, $content);
|
// (SPA phones can write to tftpboot -> redirect PUT request to 'settings' folder)
|
||||||
|
$filename = $this->_settings_path . basename($req_filename);
|
||||||
|
return file_put_contents($filename, $content);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
Reference in New Issue
Block a user