mirror of
https://github.com/gtbu/Typesetter-5.3-p8.git
synced 2025-04-03 13:33:15 +02:00
update Status.php
This commit is contained in:
parent
24ab68181c
commit
a886d2ab94
1 changed files with 240 additions and 191 deletions
|
@ -4,51 +4,51 @@ namespace gp\admin\Tools;
|
|||
|
||||
defined('is_running') or die('Not an entry point...');
|
||||
|
||||
class Status extends \gp\special\Base{
|
||||
class Status extends \gp\special\Base {
|
||||
|
||||
protected $check_dir_len = 0;
|
||||
protected $failed_count = 0;
|
||||
protected $failed = [];
|
||||
protected $passed_count = 0;
|
||||
protected $show_failed_max = 50;
|
||||
protected $deletable = [];
|
||||
protected $check_dir_len = 0;
|
||||
protected $failed_count = 0;
|
||||
protected $failed = [];
|
||||
protected $passed_count = 0;
|
||||
protected $show_failed_max = 50;
|
||||
protected $deletable = [];
|
||||
|
||||
protected $euid;
|
||||
|
||||
public function __construct(){
|
||||
public function __construct() {
|
||||
|
||||
}
|
||||
|
||||
public function RunScript(){
|
||||
public function RunScript() {
|
||||
global $langmessage;
|
||||
|
||||
echo '<h2>'.$langmessage['Site Status'].'</h2>';
|
||||
echo '<h2>' . $langmessage['Site Status'] . '</h2>';
|
||||
|
||||
$cmd = \gp\tool::GetCommand();
|
||||
switch($cmd){
|
||||
switch ($cmd) {
|
||||
case 'FixOwner':
|
||||
$this->FixOwner();
|
||||
break;
|
||||
break;
|
||||
}
|
||||
|
||||
$this->CheckDataDir();
|
||||
$this->DefaultDisplay();
|
||||
}
|
||||
|
||||
public function CheckDataDir(){
|
||||
public function CheckDataDir() {
|
||||
global $dataDir;
|
||||
|
||||
$this->check_dir_len = 0;
|
||||
$this->failed_count = 0;
|
||||
$this->passed_count = 0;
|
||||
$this->show_failed_max = 50;
|
||||
$this->check_dir_len = 0;
|
||||
$this->failed_count = 0;
|
||||
$this->passed_count = 0;
|
||||
$this->show_failed_max = 50;
|
||||
|
||||
|
||||
$check_dir = $dataDir.'/data';
|
||||
$this->check_dir_len = strlen($check_dir);
|
||||
$this->euid = '?';
|
||||
$check_dir = $dataDir . '/data';
|
||||
$this->check_dir_len = strlen($check_dir);
|
||||
$this->euid = '?';
|
||||
|
||||
if( function_exists('posix_geteuid') ){
|
||||
if (function_exists('posix_geteuid')) {
|
||||
$this->euid = posix_geteuid();
|
||||
}
|
||||
|
||||
|
@ -56,104 +56,107 @@ class Status extends \gp\special\Base{
|
|||
$this->CheckDir($check_dir);
|
||||
}
|
||||
|
||||
public function DefaultDisplay(){
|
||||
global $langmessage, $dataDir;
|
||||
public function DefaultDisplay() {
|
||||
global $langmessage, $dataDir;
|
||||
|
||||
$check_dir = $dataDir.'/data';
|
||||
$checked = $this->passed_count + $this->failed_count;
|
||||
$check_dir = $dataDir . '/data';
|
||||
$checked = $this->passed_count + $this->failed_count;
|
||||
|
||||
if( $this->failed_count === 0 ){
|
||||
echo '<p class="gp_passed">';
|
||||
echo sprintf($langmessage['data_check_passed'],$checked,$checked);
|
||||
echo '</p>';
|
||||
$this->ShowDeletable();
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<p class="gp_notice">';
|
||||
echo sprintf($langmessage['data_check_failed'],$this->failed_count,$checked);
|
||||
if ($this->failed_count === 0) {
|
||||
echo '<p class="gp_passed">';
|
||||
echo sprintf($langmessage['data_check_passed'], $checked, $checked);
|
||||
echo '</p>';
|
||||
|
||||
|
||||
// the /data directory isn't writable
|
||||
if( count($this->failed) == 1 && in_array($check_dir,$this->failed) ){
|
||||
echo '<p class="gp_notice">';
|
||||
echo '<b>WARNING:</b> Your data directory at is no longer writable: '.$check_dir;
|
||||
echo '</p>';
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if( $this->failed_count > $this->show_failed_max ){
|
||||
echo '<p class="gp_notice">';
|
||||
echo sprintf($langmessage['showing_max_failed'],$this->show_failed_max);
|
||||
echo '</p>';
|
||||
}
|
||||
|
||||
|
||||
echo '<table class="bordered">';
|
||||
echo '<tr><th>';
|
||||
echo $langmessage['file_name'];
|
||||
echo '</th><th>';
|
||||
echo $langmessage['File Owner'];
|
||||
echo '<br/>';
|
||||
echo $langmessage['Current_Value'];
|
||||
echo '</th><th>';
|
||||
echo '<br/>';
|
||||
echo $langmessage['Expected_Value'];
|
||||
echo '</th><th> ';
|
||||
echo '</th></tr>';
|
||||
|
||||
|
||||
// sort by strlen to get directories first
|
||||
usort($this->failed, function($a, $b) {
|
||||
return strlen($a) - strlen($b);
|
||||
});
|
||||
|
||||
foreach($this->failed as $i => $path){
|
||||
|
||||
if( $i > $this->show_failed_max ){
|
||||
break;
|
||||
}
|
||||
|
||||
$readable_path = substr($path,$this->check_dir_len);
|
||||
$euid = \gp\install\FilePermissions::file_uid($path);
|
||||
|
||||
echo '<tr><td>';
|
||||
echo $readable_path;
|
||||
echo '</td><td>';
|
||||
|
||||
echo $this->ShowUser($euid);
|
||||
echo '</td><td>';
|
||||
echo $this->ShowUser($this->euid);
|
||||
echo '</td><td>';
|
||||
echo \gp\tool::Link('Admin/Status','Fix','cmd=FixOwner&path='.rawurlencode($readable_path),'data-cmd="cnreq"');
|
||||
echo '</td></tr>';
|
||||
}
|
||||
|
||||
echo '</table>';
|
||||
|
||||
$this->CheckPageFiles();
|
||||
$this->ShowDeletable();
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<p class="gp_notice">';
|
||||
echo sprintf($langmessage['data_check_failed'], $this->failed_count, $checked);
|
||||
echo '</p>';
|
||||
|
||||
|
||||
// the /data directory isn't writable
|
||||
if (count($this->failed) == 1 && in_array($check_dir, $this->failed)) {
|
||||
echo '<p class="gp_notice">';
|
||||
echo '<b>WARNING:</b> Your data directory at is no longer writable: ' . htmlspecialchars($check_dir, ENT_QUOTES, 'UTF-8'); // escape output
|
||||
echo '</p>';
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if ($this->failed_count > $this->show_failed_max) {
|
||||
echo '<p class="gp_notice">';
|
||||
echo sprintf($langmessage['showing_max_failed'], $this->show_failed_max);
|
||||
echo '</p>';
|
||||
}
|
||||
|
||||
|
||||
echo '<table class="bordered">';
|
||||
echo '<tr><th>';
|
||||
echo $langmessage['file_name'];
|
||||
echo '</th><th>';
|
||||
echo $langmessage['File Owner'];
|
||||
echo '<br/>';
|
||||
echo $langmessage['Current_Value'];
|
||||
echo '</th><th>';
|
||||
echo '<br/>';
|
||||
echo $langmessage['Expected_Value'];
|
||||
echo '</th><th> ';
|
||||
echo '</th></tr>';
|
||||
|
||||
// sort by strlen to get directories first
|
||||
usort($this->failed, function ($a, $b) {
|
||||
return strlen($a) - strlen($b);
|
||||
});
|
||||
|
||||
foreach ($this->failed as $i => $path) {
|
||||
|
||||
if ($i > $this->show_failed_max) {
|
||||
break;
|
||||
}
|
||||
|
||||
$readable_path = substr($path, $this->check_dir_len);
|
||||
|
||||
// Sanitize the readable_path BEFORE passing it to rawurlencode and \gp\tool::Link
|
||||
$safe_readable_path = $this->sanitizePath($readable_path);
|
||||
|
||||
$euid = \gp\install\FilePermissions::file_uid($path);
|
||||
|
||||
echo '<tr><td>';
|
||||
echo htmlspecialchars($readable_path, ENT_QUOTES, 'UTF-8'); // escape output
|
||||
echo '</td><td>';
|
||||
|
||||
echo $this->ShowUser($euid);
|
||||
echo '</td><td>';
|
||||
echo $this->ShowUser($this->euid);
|
||||
echo '</td><td>';
|
||||
// Use the sanitized path
|
||||
echo \gp\tool::Link('Admin/Status', 'Fix', 'cmd=FixOwner&path=' . rawurlencode($safe_readable_path), 'data-cmd="cnreq"');
|
||||
echo '</td></tr>';
|
||||
}
|
||||
|
||||
echo '</table>';
|
||||
|
||||
$this->CheckPageFiles();
|
||||
$this->ShowDeletable();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Show Deletable Files
|
||||
*/
|
||||
protected function ShowDeletable(){
|
||||
if( empty($this->deletable) ){
|
||||
protected function ShowDeletable() {
|
||||
if (empty($this->deletable)) {
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<h3>Deletable Files</h3>';
|
||||
echo '<ol>';
|
||||
foreach($this->deletable as $file){
|
||||
echo '<li>'.htmlspecialchars($file).'</li>';
|
||||
foreach ($this->deletable as $file) {
|
||||
echo '<li>' . htmlspecialchars($file, ENT_QUOTES, 'UTF-8') . '</li>'; // escape output
|
||||
}
|
||||
echo '</ol>';
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
@ -162,32 +165,32 @@ class Status extends \gp\special\Base{
|
|||
* Check page files for orphaned data files
|
||||
*
|
||||
*/
|
||||
protected function CheckPageFiles(){
|
||||
global $dataDir,$gp_index;
|
||||
protected function CheckPageFiles() {
|
||||
global $dataDir, $gp_index;
|
||||
|
||||
$pages_dir = $dataDir.'/data/_pages';
|
||||
$all_files = \gp\tool\Files::ReadDir($pages_dir,'php');
|
||||
foreach($all_files as $key => $file){
|
||||
$all_files[$key] = $pages_dir.'/'.$file.'.php';
|
||||
$pages_dir = $dataDir . '/data/_pages';
|
||||
$all_files = \gp\tool\Files::ReadDir($pages_dir, 'php');
|
||||
foreach ($all_files as $key => $file) {
|
||||
$all_files[$key] = $pages_dir . '/' . $file . '.php';
|
||||
}
|
||||
|
||||
$page_files = array();
|
||||
foreach($gp_index as $slug => $index){
|
||||
foreach ($gp_index as $slug => $index) {
|
||||
$page_files[] = \gp\tool\Files::PageFile($slug);
|
||||
}
|
||||
|
||||
$diff = array_diff($all_files,$page_files);
|
||||
$diff = array_diff($all_files, $page_files);
|
||||
|
||||
if( !count($diff) ){
|
||||
if (!count($diff)) {
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<h2>Orphaned Data Files</h2>';
|
||||
echo '<p>The following data files appear to be orphaned and are most likely no longer needed. Before completely removing these files, we recommend backing them up first.</p>';
|
||||
echo '<table class="bordered"><tr><th>File</th></tr>';
|
||||
foreach($diff as $file){
|
||||
foreach ($diff as $file) {
|
||||
echo '<tr><td>'
|
||||
. $file
|
||||
. htmlspecialchars($file, ENT_QUOTES, 'UTF-8') // escape output
|
||||
. '</td></tr>';
|
||||
}
|
||||
echo '</table>';
|
||||
|
@ -198,59 +201,58 @@ class Status extends \gp\special\Base{
|
|||
* @param string $dir
|
||||
*
|
||||
*/
|
||||
protected function CheckDir($dir){
|
||||
protected function CheckDir($dir) {
|
||||
|
||||
if( !$this->CheckFile($dir) ){
|
||||
if (!$this->CheckFile($dir)) {
|
||||
return;
|
||||
}
|
||||
|
||||
$dh = @opendir($dir);
|
||||
if( $dh === false ){
|
||||
if ($dh === false) {
|
||||
$this->failed_count++;
|
||||
$this->failed[] = $dir;
|
||||
return;
|
||||
}
|
||||
|
||||
while( ($file = readdir($dh)) !== false){
|
||||
if( $file === '.' || $file === '..' ){
|
||||
while (($file = readdir($dh)) !== false) {
|
||||
if ($file === '.' || $file === '..') {
|
||||
continue;
|
||||
}
|
||||
|
||||
$full_path = $dir.'/'.$file;
|
||||
if( is_link($full_path) ){
|
||||
$full_path = $dir . '/' . $file;
|
||||
if (is_link($full_path)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if( preg_match('#x-deletable-[0-9]+#',$full_path) ){
|
||||
if (preg_match('#x-deletable-[0-9]+#', $full_path)) {
|
||||
$this->deletable[] = $full_path;
|
||||
continue;
|
||||
}
|
||||
|
||||
if( is_dir($full_path) ){
|
||||
if (is_dir($full_path)) {
|
||||
$this->CheckDir($full_path);
|
||||
}else{
|
||||
$this->CheckFile($full_path,'file');
|
||||
} else {
|
||||
$this->CheckFile($full_path, 'file');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
protected function CheckFile($path,$type='dir'){
|
||||
protected function CheckFile($path, $type = 'dir') {
|
||||
|
||||
if( \gp\install\FilePermissions::HasFunctions() ){
|
||||
$current = @substr(decoct( @fileperms($path)), -3);
|
||||
if (\gp\install\FilePermissions::HasFunctions()) {
|
||||
$current = @substr(decoct(@fileperms($path)), -3);
|
||||
|
||||
if( $type === 'file' ){
|
||||
if ($type === 'file') {
|
||||
$expected = \gp\install\FilePermissions::getExpectedPerms_file($path);
|
||||
}else{
|
||||
} else {
|
||||
$expected = \gp\install\FilePermissions::getExpectedPerms($path);
|
||||
}
|
||||
|
||||
if( \gp\install\FilePermissions::perm_compare($expected,$current) ){
|
||||
if (\gp\install\FilePermissions::perm_compare($expected, $current)) {
|
||||
$this->passed_count++;
|
||||
return true;
|
||||
}
|
||||
|
||||
}elseif( gp_is_writable($path) ){
|
||||
} elseif (gp_is_writable($path)) {
|
||||
$this->passed_count++;
|
||||
return true;
|
||||
}
|
||||
|
@ -266,10 +268,10 @@ class Status extends \gp\special\Base{
|
|||
* Display a user name and uid
|
||||
* @param int $uid
|
||||
*/
|
||||
protected function ShowUser($uid){
|
||||
protected function ShowUser($uid) {
|
||||
$user_info = posix_getpwuid($uid);
|
||||
if( $user_info ){
|
||||
return $user_info['name'].' ('.$uid.')';
|
||||
if ($user_info) {
|
||||
return htmlspecialchars($user_info['name'], ENT_QUOTES, 'UTF-8') . ' (' . $uid . ')'; // escape output
|
||||
}
|
||||
|
||||
return $uid;
|
||||
|
@ -284,65 +286,112 @@ class Status extends \gp\special\Base{
|
|||
* 4) attempt to delete temp folder
|
||||
*
|
||||
*/
|
||||
public function FixOwner(){
|
||||
global $dataDir, $langmessage;
|
||||
|
||||
|
||||
public function FixOwner() {
|
||||
global $dataDir, $langmessage;
|
||||
|
||||
// Sanitize the path from user input
|
||||
$unsafe_path = isset($_REQUEST['path']) ? $_REQUEST['path'] : '';
|
||||
$safe_path = $this->sanitizePath($unsafe_path);
|
||||
|
||||
$to_fix = '/data'.$_REQUEST['path'];
|
||||
$to_fix_full = $dataDir . $to_fix;
|
||||
$new_file = \gp\tool\FileSystem::TempFile($to_fix);
|
||||
$new_file_full = $dataDir . $new_file;
|
||||
$deletable = \gp\tool\FileSystem::TempFile(dirname($to_fix).'/x-deletable');
|
||||
$deletable_full = $dataDir . $deletable;
|
||||
|
||||
if( !\gp\tool\Files::CheckPath( $to_fix_full ) ){
|
||||
msg($langmessage['OOPS'].' Invalid Path');
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
echo '<ol>';
|
||||
echo '<li>Copy: '.$to_fix.' -> ' . $new_file . '</li>';
|
||||
|
||||
if( !\gp\admin\Tools\Port::CopyAll($to_fix_full,$new_file_full) ){
|
||||
echo '<li>Failed</li>';
|
||||
echo '</ol>';
|
||||
msg($langmessage['OOPS'].' Not Copied');
|
||||
\gp\tool\Files::RmAll($new_file_full);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
// move old to deletable
|
||||
echo '<li>Move: '.htmlspecialchars($to_fix, ENT_QUOTES, 'UTF-8').' -> ' . $deletable . '</li>';
|
||||
if( !rename($to_fix_full,$deletable_full) ){
|
||||
echo '<li>Failed</li>';
|
||||
echo '</ol>';
|
||||
msg($langmessage['OOPS'].' Rename to deletable failed');
|
||||
\gp\tool\Files::RmAll($new_file_full);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
// move
|
||||
echo '<li>Move: '.$new_file.' -> ' . $to_fix . '</li>';
|
||||
if( !rename($new_file_full, $to_fix_full) ){
|
||||
echo '<li>Failed</li>';
|
||||
echo '</ol>';
|
||||
msg($langmessage['OOPS'].' Rename to old failed');
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<li>Success</li>';
|
||||
|
||||
// attempt to remove deletable
|
||||
if( !\gp\tool\Files::RmAll($deletable_full) ){
|
||||
echo '<li>Note: '.$deletable.' was not deleted</li>';
|
||||
}
|
||||
|
||||
echo '</ol>';
|
||||
|
||||
// Validate path AFTER sanitization
|
||||
if (strpos($safe_path, 'javascript:') !== false) {
|
||||
msg($langmessage['OOPS'] . ' Invalid Path: javascript: URI is not allowed.');
|
||||
return;
|
||||
}
|
||||
if (strpos($safe_path, 'data:') !== false) {
|
||||
msg($langmessage['OOPS'] . ' Invalid Path: data: URI is not allowed.');
|
||||
return;
|
||||
}
|
||||
|
||||
$to_fix = '/data' . $safe_path;
|
||||
$to_fix_full = $dataDir . $to_fix;
|
||||
|
||||
$new_file = \gp\tool\FileSystem::TempFile($to_fix);
|
||||
$new_file_full = $dataDir . $new_file;
|
||||
|
||||
$deletable = \gp\tool\FileSystem::TempFile(dirname($to_fix) . '/x-deletable');
|
||||
$deletable_full = $dataDir . $deletable;
|
||||
|
||||
if (!\gp\tool\Files::CheckPath($to_fix_full)) {
|
||||
msg($langmessage['OOPS'] . ' Invalid Path');
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<ol>';
|
||||
echo '<li>Copy: ' . htmlspecialchars($to_fix, ENT_QUOTES, 'UTF-8') . ' -> ' . htmlspecialchars($new_file, ENT_QUOTES, 'UTF-8') . '</li>';
|
||||
|
||||
if (!\gp\admin\Tools\Port::CopyAll($to_fix_full, $new_file_full)) {
|
||||
echo '<li>Failed</li>';
|
||||
echo '</ol>';
|
||||
msg($langmessage['OOPS'] . ' Not Copied');
|
||||
\gp\tool\Files::RmAll($new_file_full);
|
||||
return;
|
||||
}
|
||||
|
||||
// move old to deletable
|
||||
echo '<li>Move: ' . htmlspecialchars($to_fix, ENT_QUOTES, 'UTF-8') . ' -> ' . htmlspecialchars($deletable, ENT_QUOTES, 'UTF-8') . '</li>';
|
||||
if (!rename($to_fix_full, $deletable_full)) {
|
||||
echo '<li>Failed</li>';
|
||||
echo '</ol>';
|
||||
msg($langmessage['OOPS'] . ' Rename to deletable failed');
|
||||
\gp\tool\Files::RmAll($new_file_full);
|
||||
return;
|
||||
}
|
||||
|
||||
// move
|
||||
echo '<li>Move: ' . htmlspecialchars($new_file, ENT_QUOTES, 'UTF-8') . ' -> ' . htmlspecialchars($to_fix, ENT_QUOTES, 'UTF-8') . '</li>';
|
||||
if (!rename($new_file_full, $to_fix_full)) {
|
||||
echo '<li>Failed</li>';
|
||||
echo '</ol>';
|
||||
msg($langmessage['OOPS'] . ' Rename to old failed');
|
||||
return;
|
||||
}
|
||||
|
||||
echo '<li>Success</li>';
|
||||
|
||||
// attempt to remove deletable
|
||||
if (!\gp\tool\Files::RmAll($deletable_full)) {
|
||||
echo '<li>Note: ' . htmlspecialchars($deletable, ENT_QUOTES, 'UTF-8') . ' was not deleted</li>';
|
||||
}
|
||||
|
||||
echo '</ol>';
|
||||
}
|
||||
/**
|
||||
* Sanitizes a path to prevent directory traversal and XSS.
|
||||
*
|
||||
* @param string $path The path to sanitize.
|
||||
* @return string The sanitized path.
|
||||
*/
|
||||
private function sanitizePath(string $path): string
|
||||
{
|
||||
// Block Javascript and data URI
|
||||
if (stripos($path, 'javascript:') !== false || stripos($path, 'data:') !== false) {
|
||||
return ''; // or throw an exception, or return a safe default path
|
||||
}
|
||||
|
||||
// Remove any characters that aren't alphanumeric, underscores, dashes, or slashes
|
||||
$sanitized = preg_replace('/[^a-zA-Z0-9_\-\/]/', '', $path);
|
||||
|
||||
// Resolve directory traversal attempts (e.g., ../../)
|
||||
$sanitized = str_replace('..', '', $sanitized);
|
||||
|
||||
// Remove potentially harmful URL components
|
||||
$sanitized = str_replace('href', '', strtolower($sanitized));
|
||||
|
||||
// Remove leading and trailing slashes
|
||||
$sanitized = trim($sanitized, '/');
|
||||
|
||||
// Ensure no double slashes
|
||||
$sanitized = preg_replace('/\/+/', '/', $sanitized);
|
||||
|
||||
// Add a leading slash
|
||||
if ($sanitized !== '') {
|
||||
$sanitized = '/' . $sanitized;
|
||||
}
|
||||
|
||||
return $sanitized;
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in a new issue