<?php /** * COPYRIGHT © 2022 JOHN PEARCEY * All rights reserved */ const BASE64_DIGITS = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz$_'; if (version_compare( PHP_VERSION, '8', '<') ){ include_once __DIR__ . "/common-pre-v8.php"; } function getStackTrace(){ try{ throw new Exception; }catch( Exception $e ){ return $e->getTraceAsString(); } } /** * e.g. split * ../db/stanhopetest/img/img_52548 * into * [ ../db/stanhopetest/img/ , img_52548 ] * */ function split_file_path( $ffn ){ $fn = basename( $ffn ); // img_52548 //$fp = substr( $ffn, 0, strlen($ffn) - strlen($fn) ); $fp = dirname( $ffn ) . '/'; return [ $fp, $fn ]; } /** * Returns true if it is possible to create this directory on the file system. * Includes the possibility of multiple directories. */ function is_creatable_dir( $dirs ){ $dir = $dirs; while(!file_exists($dir)){ $dir = dirname( $dir ); } return is_writable($dir); } /** * Function to make directories and actually SET the permissions * https://www.php.net/manual/en/function.chmod.php * * $perms is an octal number * e.g. mkdirs( $dir, 02770 ); //770 and g+s */ function mkdirs( $dirs, $perms ){ $dir = $dirs; while(!file_exists($dir)){ $dir = dirname( $dir ); } $rootDir = $dir; if(!file_exists($dirs)) mkdir( $dirs, 0750, true ); //perms not always set $dir = $dirs; while( $dir!=$rootDir) { chmod( $dir, $perms ); $dir = dirname( $dir ); } } /** * Return the contents of a url as a string using the curl functions. */ function getContentsFromUrl( $url ){ $ch = curl_init( $url ); //$fp = fopen("example_homepage.txt", "w"); $fp = tmpfile(); curl_setopt($ch, CURLOPT_FILE, $fp); curl_setopt($ch, CURLOPT_HEADER, 0); curl_exec($ch); $c_err = curl_error($ch); curl_close($ch); if($c_err) { error_log( $c_err ); fclose($fp); return; } $f_path = stream_get_meta_data($fp)['uri']; $rtn = file_get_contents( $f_path ); fclose($fp); return $rtn; } /** * Set the given key-value pair. * * Returns the record ID. */ function params_set( $key, $val ) { $rec = R::findOne( 'params', 'pkey=?', [$key] ); if($rec) { $rec->pval = $val; R::store( $rec ); return $rec->id; } $rec = R::dispense( 'params' ); $rec->pkey = $key; $rec->pval = $val; return R::store( $rec ); } function params_get_date( $key ){ return strtotime( params_get($key) ); } /** * Returns the value for the given key. Returns null if no such key exists. */ function params_get( $key ){ $rec = R::findOne( 'params', 'pkey=?', [$key] ); if(!$rec) return null; return $rec->pval; } /** * A novel way of creating a unique ID each time a call is made. The new ID is * returned. * * e.g. * This can be used to give a client which might want to create temporary unique IDs. It is * called once per client refresh and used to ensure uniqueness of certain CSS data points. */ function params_unique( ) { $rec = R::dispense( 'params' ); $rec->pkey = 'unique'; $id = R::store( $rec ); R::exec( "delete from params where id<$id and pkey='unique'" ); return $id; } /** * Return a string representation for the given dom node suitable for debugging. */ function getNodeString( $rmElem ){ if($rmElem==null) return "Element is null"; $str=""; if( $rmElem->nodeType==XML_ELEMENT_NODE){ $str .= $rmElem->nodeName; $a = $rmElem->getAttribute("shb_compid"); if($a){ $str .= " shb_compid=$a"; } }else{ $str .= 'type = '.$rmElem->nodeType; } return $str; } /** * Simple logger to write to a file. Using php error_log distorts all crlf pairs. This * is a pain when you're trying to preserve them in html. */ function mylog( $msg ){ global $logging_on; if(!$logging_on) return; $p = __DIR__ . "/../../shb.log"; $h = fopen( $p, "a"); if($h===false) return; fwrite( $h, "$msg\n"); fclose($h); } function generateNonce( $digitCount = 12 ){ // e.g. 422466a05f24b09c978fa1f6 return bin2hex( random_bytes($digitCount) ); } /** * Generate a randon username starting with a letter and containing only letters and numbers. */ function generateRndUserName( $digitCount = 10 ){ $l1 = generateRndStrUsing( 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz', 1 ); return $l1 . generateRndStrUsing( BASE64_DIGITS, $digitCount-1 ); } /** * Generate a randon password with letters and numbers and some funny stuff. */ function generateRndPwd( $digitCount = 10 ){ return generateRndStrUsing( '0123456789_!"£$%^&*()_-|\?/<>#~ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz', $digitCount ); } /** * Generate a randon string of asc char. Default consisting of only [0-9][a-z][A-Z] */ function generateRndStrUsing( $using=BASE64_DIGITS, $digitCount = 8 ){ $str = ''; $maxlen = strlen($using); for($i=0; $i<$digitCount; $i++){ $iv = random_int(0, $maxlen-1); $str .= mb_substr( $using, $iv, 1); //use mb_substr because £ is not and asc char! } return $str; } /** * Remove all html from the given string returning just the text content. */ function removeHtml( $str ){ //txt = txt.replace(/<div>/gi, '\n'); return preg_replace( '/<\/?[^>]+>/i', '', $str ); } function getHtmlOkAsString( $data ){ $data = json_encode( $data ); return "[\"OK\", $data ]"; } function sendHtmlOk( $data="" ){ echo getHtmlOkAsString( $data ); } /** * An effort to rectify my previously ill-thought out return encoding. You do not * need to encode the data before calling this method. Any arrays will be json encoded. */ function sendHtmlOk_WithData( $data ){ $data = json_encode( $data ); echo "[\"OK\", $data ]"; } function getHtmlErrorAsString( $strErrDesc = "--blank--" ){ $data = json_encode( $strErrDesc ); return "[\"Error\", $data]"; } function sendHtmlError( $strErrDesc = "--blank--" ){ $rtn = getHtmlErrorAsString( $strErrDesc ); mylog( $rtn ); echo $rtn; } /** * */ function getAttForPath( $name, $prefPath, $path){ return "$name = \"$prefPath/$path\""; } global $g_pub_path; /** * If the browser=true, the file returned will be correct for adding to an href or src attribute so that the * browser will find the link. */ function getDirForType( $filetype='html', $browser=false, $userid=-1 ){ global $g_pub_path; if($g_pub_path==null){ $g_pub_path = params_get('pub_path'); if($g_pub_path==null){ throw new Exception("Publishing now requires a valid key('pub_path') in the params table for the publishing output path"); } if(str_ends_with( $g_pub_path, '/') ) $g_pub_path = substr($g_pub_path, 0, strlen($g_pub_path)-1 ); //ensure all the directories exist createUserDir( 'img', $userid ); createUserDir( 'css', $userid ); createUserDir( 'html', $userid ); createUserDir( 'js', $userid ); } return getDirForType_( $filetype, $browser, $userid ); } function createUserDir( $dName, $userid ){ if(!file_exists(getDirForType_( $dName, false, $userid ))) { $dir = getDirForType_( $dName, false, $userid ); error_log("common.php: createUserDir: $dir"); mkdir( $dir, 0770, true); } } /** * Part of the new path handling system is to sepatate out path names from real files */ function mkdirIfNotExists( $d ){ if(!file_exists( $d ) ) { error_log("common.php: createUserDir: $d"); mkdir( $d, 0770, true ); } } /** * For the moment, call getDirForType and not this function. In time, the setup routine will make the directories * and eliminate the need for the check during a user session. * * If the browser flag is set, we assuming a document root sub-dir. But in time, this will need to also cater for * a site serving the g_pub_path as root (wrt the browser). * */ function getDirForType_( $filetype, $browser, $userid ){ global $g_pub_path; $strPath = $g_pub_path; if($userid>0){ $strPath .= "_dbg_$userid"; } switch($filetype){ case 'img': if($browser) return "/$strPath/img/"; return fxdPth( "$strPath/img/" ); case 'js': if($browser) return "/$strPath/js/"; return fxdPth( "$strPath/js/" ); case 'css': if($browser) return "/$strPath/css/"; return fxdPth( "$strPath/css/" ); case 'html': default: if($browser) return "/$strPath/"; return fxdPth( "$strPath/" ); } } //NO white-space