Typesetter-Original-gtbu/include/tool/Files.php

1058 lines
23 KiB
PHP
Raw Permalink Normal View History

2021-09-08 19:52:21 +02:00
<?php
namespace gp\tool{
defined('is_running') or die('Not an entry point...');
/**
* Contains functions for working with data files and directories
*
*/
class Files{
public static $last_modified; //the modified time of the last file retrieved with gp\tool\Files::Get();
public static $last_version; //the version of the last file retrieved with gp\tool\Files::Get();
public static $last_stats = array(); //the stats of the last file retrieved with gp\tool\Files::Get();
public static $last_meta = array(); //the meta data of the last file retrieved with gp\tool\Files::Get();
/**
* Make sure the $path is a subdirectory of $parent
*
* @param string The file path to check
* @param string The parent file path to check, null to check against $dataDir
* @return bool
*/
public static function CheckPath( $path, $parent = null){
global $dataDir;
if( is_null($parent) ){
$parent = $dataDir;
}
$path = self::Canonicalize($path);
if( strpos($path, $parent) === 0 ){
return true;
}
return false;
}
/**
* Return Canonicalized absolute pathname
* Similar to http://php.net/manual/en/function.realpath.php but does not check file existence
*
* @param string $path
* @return string
*/
public static function Canonicalize($path) {
$path = \gp\tool\Editing::Sanitize($path);
$path = str_replace( '\\', '/', $path);
$start_slash = $path[0] == '/' ? '/' : '';
$parts = explode('/', $path);
$parts = array_filter($parts);
$absolutes = array();
foreach( $parts as $part ){
if( '.' == $part ){
continue;
}
if( '..' == $part ){
array_pop($absolutes);
}else{
$absolutes[] = $part;
}
}
return $start_slash . implode('/', $absolutes);
}
/**
* Get array from data file
* Example:
* $config = gp\tool\Files::Get('_site/config','config'); or $config = gp\tool\Files::Get('_site/config');
* @since 4.4b1
*
*/
public static function Get( $file, $var_name=null ){
self::$last_modified = null;
self::$last_version = null;
self::$last_stats = array();
self::$last_meta = array();
$file_stats = array();
$fileModTime = time();
$fileVersion = gpversion;
$meta_data = array();
if( !$var_name ){
$var_name = basename($file);
}
$file = self::FilePath($file);
//json
if( gp_data_type === '.json' ){
return self::Get_Json($file,$var_name);
}
if( !file_exists($file) ){
return array();
}
include($file);
if( !isset(${$var_name}) || !is_array(${$var_name}) ){
return array();
}
// For data files older than 3.0
if( !isset($file_stats['modified']) ){
$file_stats['modified'] = $fileModTime;
}
if( !isset($file_stats['gpversion']) ){
$file_stats['gpversion'] = $fileVersion;
}
// File stats
self::$last_modified = $fileModTime;
self::$last_version = $fileVersion;
self::$last_stats = $file_stats;
if( isset($meta_data) ){
self::$last_meta = $meta_data;
}
return ${$var_name};
}
/**
* Experimental
*
*/
private static function Get_Json($file,$var_name){
if( !file_exists($file) ){
return array();
}
$contents = file_get_contents($file);
$data = json_decode($contents,true);
if( !isset($data[$var_name]) || !is_array($data[$var_name]) ){
return array();
}
// File stats
self::$last_modified = $data['file_stats']['modified'];
self::$last_version = $data['file_stats']['gpversion'];
self::$last_stats = $data['file_stats'];
self::$last_meta = $data['meta_data'];
return $data[$var_name];
}
/**
* Get the raw contents of a data file
*
*/
public static function GetRaw($file){
$file = self::FilePath($file);
return file_get_contents($file);
}
/**
* Return true if the data file exists
*
*/
public static function Exists($file){
$file = self::FilePath($file);
return file_exists($file);
}
/**
* Read directory and return an array with files corresponding to $filetype
*
* @param string $dir The path of the directory to be read
* @param mixed $filetype If false, all files in $dir will be included. false=all,1=directories,'php'='.php' files
* @return array() List of files in $dir
*/
public static function ReadDir($dir,$filetype='php'){
$files = array();
if( !file_exists($dir) ){
return $files;
}
$dh = @opendir($dir);
if( !$dh ){
return $files;
}
while( ($file = readdir($dh)) !== false){
if( $file == '.' || $file == '..' ){
continue;
}
//get all
if( $filetype === false ){
$files[$file] = $file;
continue;
}
//get directories
if( $filetype === 1 ){
$fullpath = $dir.'/'.$file;
if( is_dir($fullpath) ){
$files[$file] = $file;
}
continue;
}
$dot = strrpos($file, '.');
if( $dot === false ){
continue;
}
$type = substr($file, $dot + 1);
//if $filetype is an array
if( is_array($filetype) ){
if( in_array($type, $filetype) ){
$files[$file] = $file;
}
continue;
}
//if $filetype is a string
if( $type == $filetype ){
$file = substr($file, 0, $dot);
$files[$file] = $file;
}
}
closedir($dh);
return $files;
}
/**
* Read all of the folders and files within $dir and return them in an organized array
*
* @param string $dir The directory to be read
* @return array() The folders and files within $dir
*
*/
public static function ReadFolderAndFiles($dir){
$dh = @opendir($dir);
if( !$dh ){
return array();
}
$folders = array();
$files = array();
while( ($file = readdir($dh)) !== false){
if( strpos($file, '.') === 0){
continue;
}
$fullPath = $dir. '/'. $file;
if( is_dir($fullPath) ){
$folders[] = $file;
}else{
$files[] = $file;
}
}
natcasesort($folders);
natcasesort($files);
return array($folders, $files);
}
/**
* Get the Section Clipboard
* @since 5.1-b1
*
*/
public static function GetSectionClipboard(){
global $dataDir;
$clipboard_dir = $dataDir . '/data/_clipboard';
self::CheckDir($clipboard_dir);
$clipboard_data = self::Get($clipboard_dir . '/clipboard_data.php', 'clipboard_data');
return $clipboard_data;
}
/**
* Save the Section Clipboard
* @since 5.1-b1
*
*/
public static function SaveSectionClipboard($clipboard_data=array()){
global $dataDir;
$clipboard_dir = $dataDir . '/data/_clipboard';
self::CheckDir($clipboard_dir);
return self::SaveData($clipboard_dir . '/clipboard_data.php', 'clipboard_data', $clipboard_data);
}
/**
* Clean a string for use as a page label (displayed title)
* Similar to CleanTitle() but less restrictive
*
* @param string $title The title to be cleansed
* @return string The cleansed title
*/
public static function CleanLabel($title=''){
$title = str_replace(array('"'), array(''), $title);
$title = str_replace(array('<', '>'), array('_'), $title);
$title = trim($title);
// Remove control characters
return preg_replace('#[[:cntrl:]]#u', '', $title); // [\x00-\x1F\x7F]
}
/**
* Clean a string of html that may be used as file content
*
* @param string $text The string to be cleansed. Passed by reference
*/
public static function CleanText(&$text){
\gp\tool\Editing::tidyFix($text);
self::rmPHP($text);
self::FixTags($text);
$text = \gp\tool\Plugins::Filter('CleanText',array($text));
}
/**
* Use html parser to check the validity of $text
*
* @param string $text The html content to be checked. Passed by reference
*/
public static function FixTags(&$text){
$gp_html_output = new \gp\tool\Editing\HTML($text);
$text = $gp_html_output->result;
}
/**
* Remove php tags from $text
*
* @param string $text The html content to be checked. Passed by reference
*/
public static function rmPHP(&$text){
$search = array('<?', '<?php', '?>');
$replace = array('&lt;?', '&lt;?php', '?&gt;');
$text = str_replace($search, $replace, $text);
}
/**
* Removes any NULL characters in $string.
* @since 3.0.2
* @param string $string
* @return string
*/
public static function NoNull($string){
$string = preg_replace('/\0+/', '', $string);
return preg_replace('/(\\\\0)+/', '', $string);
}
/**
* Save the content for a new page in /data/_pages/<title>
* @since 1.8a1
*
*/
public static function NewTitle($title, $section_content=false, $type='text'){
// get the file for the title
if( empty($title) ){
return false;
}
$file = self::PageFile($title);
if( !$file ){
return false;
}
// organize section data
$file_sections = array();
if( is_array($section_content) && isset($section_content['type']) ){
$file_sections[0] = $section_content;
}elseif( is_array($section_content) ){
$file_sections = $section_content;
}else{
$file_sections[0] = array(
'type' => $type,
'content' => $section_content,
);
}
// add meta data
$meta_data = array(
'file_number' => self::NewFileNumber(),
'file_type' => $type,
);
return self::SaveData($file,'file_sections',$file_sections,$meta_data);
}
/**
* Return the data file location for a title
* Since v4.6, page files are within a subfolder
* As of v2.3.4, it defaults to an index based file name but falls back on title based file name for backwards compatibility
*
*
* @param string $title
* @return string The path of the data file
*/
public static function PageFile($title){
global $dataDir, $config, $gp_index;
$index_path = false;
// filename based on title index
if( gp_index_filenames && isset($gp_index[$title]) && isset($config['gpuniq']) ){
$index_path = $dataDir . '/data/_pages/' . substr($config['gpuniq'], 0, 7) . '_' . $gp_index[$title] . '/page.php';
}
// using file name instead of index
$normal_path = $dataDir . '/data/_pages/' . str_replace('/', '_', $title) . '/page.php';
if( !$index_path || self::Exists($normal_path) ){
return $normal_path;
}
return $index_path;
}
public static function NewFileNumber(){
global $config;
if( !isset($config['file_count']) ){
$config['file_count'] = 0;
}
$config['file_count']++;
\gp\admin\Tools::SaveConfig();
return $config['file_count'];
}
/**
* Get the meta data for the specified file
*
* @param string $file
* @return array
*/
public static function GetTitleMeta($file){
self::Get($file,'meta_data');
return self::$last_meta;
}
/**
* Return an array of info about the data file
*
*/
public static function GetFileStats($file){
$file_stats = self::Get($file, 'file_stats');
if( $file_stats ){
return $file_stats;
}
return array('created'=> time());
}
/**
* Save a file with content and data to the server
* This function will be deprecated in future releases. Using it is not recommended
*
* @param string $file The path of the file to be saved
* @param string $contents The contents of the file to be saved
* @param string $code The data to be saved
* @param string $time The unix timestamp to be used for the $fileVersion
* @return bool True on success
*/
public static function SaveFile($file, $contents, $code=false, $time=false){
$result = self::FileStart($file, $time);
if( $result !== false ){
$result .= "\n" . $code;
}
$result .= "\n\n?" . ">\n";
$result .= $contents;
return self::Save($file, $result);
}
/**
* Save raw content to a file to the server
*
* @param string $file The path of the file to be saved
* @param string $contents The contents of the file to be saved
* @return bool True on success
*/
public static function Save($file,$contents){
global $gp_not_writable;
$exists = self::Exists($file);
//make sure directory exists
if( !$exists ){
$dir = \gp\tool::DirName($file);
if( !file_exists($dir) ){
self::CheckDir($dir);
}
}
$fp = @fopen($file, 'wb');
if( $fp === false ){
$gp_not_writable[] = $file;
return false;
}
if( !flock($fp, LOCK_EX) ){
trigger_error('flock could not be obtained.');
return false;
}
if( !$exists ){
@chmod($file, gp_chmod_file);
}elseif( function_exists('opcache_invalidate') && substr($file, -4) === '.php' ){
opcache_invalidate($file);
}
$return = fwrite($fp, $contents);
flock($fp, LOCK_UN);
fclose($fp);
return ($return !== false);
}
/**
* Rename a file
* @since 4.6
*/
public static function Rename($from, $to){
global $gp_not_writable;
if( !self::WriteLock() ){
return false;
}
//make sure directory exists
$dir = \gp\tool::DirName($to);
if( !file_exists($dir) && !self::CheckDir($dir) ){
return false;
}
return rename($from, $to);
}
/**
* Replace $to with $from
*
*/
public static function Replace($from, $to){
$temp_dir = '';
// move the $to out of the way if it exists
if( file_exists($to) ){
$temp_dir = $to . '_' . time();
if( !self::rename($to, $temp_dir) ){
return false;
}
}
// rename $from -> $to
if( !self::rename($from, $to) ){
if( $temp_dir ){
self::rename($temp_dir, $to);
}
return false;
}
if( !empty($temp_dir) ){
self::RmAll($temp_dir);
}
return true;
}
/**
* Get a write lock to prevent simultaneous writing
* @since 3.5.3
*/
public static function WriteLock(){
if( defined('gp_has_lock') ){
return gp_has_lock;
}
$expires = gp_write_lock_time;
if( self::Lock('write', gp_random, $expires) ){
define('gp_has_lock', true);
return true;
}
trigger_error('CMS write lock could not be obtained.');
define('gp_has_lock', false);
return false;
}
/**
* Get a lock
* Loop and delay to wait for the removal of existing locks (maximum of about .2 of a second)
*
*/
public static function Lock($file, $value, &$expires){
global $dataDir;
$tries = 0;
$lock_file = $dataDir . '/data/_lock_' . sha1($file);
$file_time = 0;
$elapsed = 0;
while( $tries < 1000 ){
if( !file_exists($lock_file) ){
file_put_contents($lock_file, $value);
usleep(100);
}elseif( !$file_time ){
$file_time = filemtime($lock_file);
}
$contents = @file_get_contents($lock_file);
if( $value === $contents ){
@touch($lock_file);
return true;
}
if( $file_time ){
$elapsed = time() - $file_time;
if( $elapsed > $expires ){
@unlink($lock_file);
}
}
clearstatcache();
usleep(100);
$tries++;
}
if( $file_time ){
$expires -= $elapsed;
}
return false;
}
/**
* Remove a lock file if the value matches
*
*/
public static function Unlock($file, $value){
global $dataDir;
$lock_file = $dataDir . '/data/_lock_' . sha1($file);
if( !file_exists($lock_file) ){
return true;
}
$contents = @file_get_contents($lock_file);
if( $contents === false ){
return true;
}
if( $value === $contents ){
unlink($lock_file);
return true;
}
return false;
}
/**
* Save array(s) to a $file location
* Takes 2n+3 arguments
*
* @param string $file The location of the file to be saved
* @param string $varname The name of the variable being saved
* @param array $array The value of $varname to be saved
*
* @deprecated 4.3.5
*/
public static function SaveArray(){
if( gp_data_type === '.json' ){
throw new Exception('SaveArray() cannot be used for json data. Use SaveData() instead');
}
$args = func_get_args();
$count = count($args);
if( ($count %2 !== 1) || ($count < 3) ){
trigger_error('Wrong argument count ' . $count . ' for \gp\tool\Files::SaveArray() ');
return false;
}
$file = array_shift($args);
$file_stats = array();
$data = '';
while( count($args) ){
$varname = array_shift($args);
$array = array_shift($args);
if( $varname == 'file_stats' ){
$file_stats = $array;
}else{
$data .= self::ArrayToPHP($varname, $array);
$data .= "\n\n";
}
}
$data = self::FileStart($file, time(), $file_stats) . $data;
return self::Save($file, $data);
}
/**
* Save array to a $file location
*
* @param string $file The location of the file to be saved
* @param string $varname The name of the variable being saved
* @param array $array The value of $varname to be saved
* @param array $meta meta data to be saved along with $array
*
*/
public static function SaveData($file, $varname, $array, $meta=array()){
$file = self::FilePath($file);
if( gp_data_type === '.json' ){
$json = self::FileStart_Json($file);
$json[$varname] = $array;
$json['meta_data'] = $meta;
$content = json_encode($json);
}else{
$content = self::FileStart($file);
$content .= self::ArrayToPHP($varname, $array);
$content .= "\n\n";
$content .= self::ArrayToPHP('meta_data', $meta);
}
return self::Save($file, $content);
}
/**
* Experimental
*
*/
private static function FileStart_Json($file, $time=null ){
global $gpAdmin;
if( is_null($time) ){
$time = time();
}
//file stats
$file_stats = self::GetFileStats($file);
$file_stats['gpversion'] = gpversion;
$file_stats['modified'] = $time;
$file_stats['username'] = false;
if( \gp\tool::loggedIn() ){
$file_stats['username'] = $gpAdmin['username'];
}
$json = array();
$json['file_stats'] = $file_stats;
return $json;
}
/**
* Return the beginning content of a data file
*
*/
public static function FileStart($file, $time=null, $file_stats=array()){
global $gpAdmin;
if( is_null($time) ){
$time = time();
}
//file stats
$file_stats = (array)$file_stats + self::GetFileStats($file);
$file_stats['gpversion'] = gpversion;
$file_stats['modified'] = $time;
if( \gp\tool::loggedIn() ){
$file_stats['username'] = $gpAdmin['username'];
}else{
$file_stats['username'] = false;
}
return '<' . '?' . 'php'
. "\ndefined('is_running') or die('Not an entry point...');"
. "\n" . '$fileVersion = \'' . gpversion . '\';' // @deprecated 3.0
. "\n" . '$fileModTime = \'' . $time . '\';' // @deprecated 3.0
. "\n" . self::ArrayToPHP('file_stats', $file_stats)
. "\n\n";
}
public static function ArrayToPHP($varname, &$array){
return '$' . $varname . ' = ' . var_export($array, true) . ';';
}
/**
* Insert a key-value pair into an associative array
*
* @param mixed $search_key Value to search for in existing array to insert before
* @param mixed $new_key Key portion of key-value pair to insert
* @param mixed $new_value Value portion of key-value pair to insert
* @param array $array Array key-value pair will be added to
* @param int $offset Offset distance from where $search_key was found. A value of 1 would insert after $search_key, a value of 0 would insert before $search_key
* @param int $length If length is omitted, nothing is removed from $array. If positive, then that many elements will be removed starting with $search_key + $offset
* @return bool True on success
*/
public static function ArrayInsert($search_key, $new_key, $new_value, &$array, $offset=0, $length=0){
$array_keys = array_keys($array);
$array_values = array_values($array);
$insert_key = array_search($search_key,$array_keys);
if( ($insert_key === null) || ($insert_key === false) ){
return false;
}
array_splice($array_keys, $insert_key + $offset, $length, $new_key);
array_splice($array_values, $insert_key + $offset, $length, 'fill'); //use fill in case $new_value is an array
$array = array_combine($array_keys, $array_values);
$array[$new_key] = $new_value;
return true;
}
/**
* Replace a key-value pair in an associative array
* ArrayReplace() is a shortcut for using \gp\tool\Files::ArrayInsert() with $offset = 0 and $length = 1
*/
public static function ArrayReplace($search_key, $new_key, $new_value, &$array){
return self::ArrayInsert($search_key, $new_key, $new_value, $array, 0, 1);
}
/**
* Check recursively to see if a directory exists, if it doesn't attempt to create it
*
* @param string $dir The directory path
* @param bool $index Whether or not to add an index.hmtl file in the directory
* @return bool True on success
*/
public static function CheckDir($dir, $index=true){
global $config;
if( !file_exists($dir) ){
$parent = \gp\tool::DirName($dir);
self::CheckDir($parent, $index);
//ftp mkdir
if( !@mkdir($dir,gp_chmod_dir) ){
return false;
}
@chmod($dir, gp_chmod_dir); //some systems need more than just the 0755 in the mkdir() function
// make sure there's an index.html file
// only check if we just created the directory, we don't want to keep
// creating an index.html file if a user deletes it
if( $index && gp_dir_index ){
$indexFile = $dir . '/index.html';
if( !file_exists($indexFile) ){
//not using \gp\tool\Files::Save() so we can avoid infinite looping
// (it's safe since we already know the directory exists and we're not concerned about the content)
file_put_contents($indexFile, '<html></html>');
@chmod($indexFile, gp_chmod_file);
}
}
}
return true;
}
/**
* Remove a directory
* Will only work if directory is empty
*
*/
public static function RmDir($dir){
return @rmdir($dir);
}
/**
* Remove a file or directory and it's contents
*
*/
public static function RmAll($path){
if( empty($path) ){
return false;
}
if( is_link($path) ){
return @unlink($path);
}
if( !is_dir($path) ){
return @unlink($path);
}
$success = true;
$subDirs = array();
//$files = scandir($path);
$files = self::ReadDir($path, false);
foreach($files as $file){
$full_path = $path . '/' . $file;
if( !is_link($full_path) && is_dir($full_path) ){
$subDirs[] = $full_path;
continue;
}
if( !@unlink($full_path) ){
$success = false;
}
}
foreach($subDirs as $subDir){
if( !self::RmAll($subDir) ){
$success = false;
}
}
if( $success ){
return self::RmDir($path);
}
return false;
}
/**
* Get the correct path for the data file
* Two valid methods to get a data file path:
* Full path: /var/www/html/site/data/_site/config.php
* Relative: _site/config
*
*/
public static function FilePath($path){
global $dataDir;
$ext = pathinfo($path, PATHINFO_EXTENSION);
if( $ext === 'gpjson' ){
$path = substr($path,0,-7);
}elseif( $ext === 'php' ){
$path = substr($path,0,-4);
}else{
$path = $dataDir . '/data/' . ltrim($path, '/');
}
if( gp_data_type === '.json' ){
return $path . '.gpjson';
}
return $path . '.php';
}
/**
* @deprecated 3.0
* Use \gp\tool\Editing::CleanTitle() instead
* Used by Simple_Blog1
*/
public static function CleanTitle($title, $spaces='_'){
trigger_error('Deprecated Function');
return \gp\tool\Editing::CleanTitle($title, $spaces);
}
}
}
namespace{
class gpFiles extends gp\tool\Files{}
}