From fa0bb3d55ae031ebfec89b05c3cc6c71baeaed5a Mon Sep 17 00:00:00 2001 From: pwlin <160443+pwlin@users.noreply.github.com> Date: Thu, 8 Aug 2024 20:14:47 +0100 Subject: [PATCH] fixed all the bugs when running the backend on wasmer --- .github/workflows/ci.yml | 42 +++++++++++++++++++++++++++ .gitignore | 1 + client/config.example.json | 2 +- client/index.html | 2 +- client/js/config.js | 2 +- client/js/model.js | 7 ++--- package.json | 2 +- server/index.php | 5 ++++ server/salad-api.php | 59 +++++++++++++++++++------------------- wasmer.toml | 13 +++++++++ 10 files changed, 97 insertions(+), 38 deletions(-) create mode 100644 .github/workflows/ci.yml create mode 100644 wasmer.toml diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 0000000..5597213 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,42 @@ +name: Continuous Integration + +on: + pull_request: + push: + branches: + - main + +env: + GITHUB_TOKEN: ${{ github.token }} + +concurrency: + group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} + cancel-in-progress: true + +jobs: + build_and_deploy: + name: Build and Deploy PHP site to Wasmer Edge + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v2 + - name: Install Wasmer + uses: wasmerio/setup-wasmer@v2 + - name: Install PHP + uses: shivammathur/setup-php@v2 + with: + php-version: '8.3' + # tools: composer + # - name: Get composer cache directory + # id: composer-cache + # run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT + # - name: Cache dependencies + # uses: actions/cache@v4 + # with: + # path: ${{ steps.composer-cache.outputs.dir }} + # key: ${{ runner.os }}-composer-${{ hashFiles('**/composer.lock') }} + # restore-keys: ${{ runner.os }}-composer- + # - name: Install dependencies + # run: composer install --prefer-dist + - name: Deploy app to Wasmer Edge + run: wasmer deploy --token=${{ secrets.WASMER_CIUSER_PROD_TOKEN }} --non-interactive --no-wait --no-persist-id diff --git a/.gitignore b/.gitignore index 3878834..891158f 100644 --- a/.gitignore +++ b/.gitignore @@ -8,3 +8,4 @@ client/feeds.json client/Neo.toml package-lock.json server/cache +app.yaml diff --git a/client/config.example.json b/client/config.example.json index 97209ca..9ded174 100644 --- a/client/config.example.json +++ b/client/config.example.json @@ -3,7 +3,7 @@ "main_color": "red", "second_color": "black", "api_key": "dev-api-key", - "api_url": "http://127.0.0.1:8080/index.php", + "api_url": "http://127.0.0.1:8081/index.php", "favicons_enabled": false, "redirlinks_enabled": false } \ No newline at end of file diff --git a/client/index.html b/client/index.html index d2046c7..833d1a2 100644 --- a/client/index.html +++ b/client/index.html @@ -38,7 +38,7 @@ diff --git a/client/js/config.js b/client/js/config.js index b147f00..2fa9f48 100644 --- a/client/js/config.js +++ b/client/js/config.js @@ -32,7 +32,7 @@ class Config { constructor() { this.config = null; this.defaultConfig = { - 'xml2json_endpoint': '/xml2json?url=__URL__&_t=__TS__', + 'xml2json_endpoint': '/xml2json?url=__URL__&_t=__TS__&api_key=__API_KEY__', 'favicon_endpoint': '/favicon?url=__URL__&api_key=__API_KEY__', 'revredir_endpoint': '/revredir?url=__URL__&api_key=__API_KEY__' }; diff --git a/client/js/model.js b/client/js/model.js index f93fce5..fe086db 100644 --- a/client/js/model.js +++ b/client/js/model.js @@ -38,13 +38,12 @@ class Model { * @returns {Promise} A promise that resolves to the normalized feed data. */ async fetchFeed(feed) { - const apiKey = config.get('api_key'); - const serviceUrl = `${config.get('api_url')}${config.get('xml2json_endpoint').replace('__URL__', encodeURIComponent(feed.feedUrl)).replace('__TS__', utils.ts())}`; - const response = await fetch(serviceUrl, { + const serviceUrl = `${config.get('api_url')}${config.get('xml2json_endpoint').replace('__URL__', encodeURIComponent(feed.feedUrl)).replace('__TS__', utils.ts()).replace('__API_KEY__', config.get('api_key'))}`; + const response = await fetch(serviceUrl/*, { headers: { 'Authorization': `Bearer ${apiKey}` } - }); + }*/); const content = await response.json(); // Normalize the general structure of the feed entries diff --git a/package.json b/package.json index 6c45399..3879617 100644 --- a/package.json +++ b/package.json @@ -5,7 +5,7 @@ "dev-client": "node node_modules/vite/bin/vite.js --clearScreen false", "build-client": "node node_modules/vite/bin/vite.js build", "preview-client": "node node_modules/vite/bin/vite.js preview", - "dev-server": "php -S localhost:8080 -t ./server" + "dev-server": "php -S localhost:8081 -t ./server" }, "devDependencies": { "vite": "^5.3.4", diff --git a/server/index.php b/server/index.php index 693b9a1..4c0f5a8 100644 --- a/server/index.php +++ b/server/index.php @@ -1,4 +1,9 @@ salad=$salad;}private function loginWithAPIKey($selectedApiKey,$accessLevelNeeded){$configApiKeys=$this->salad->config->get('api_keys');if(!empty($configApiKeys)){$apiKeys=['result'=>$configApiKeys];}else{$apiKeys=$this->salad->db->select('SELECT * FROM '.$this->salad->config->get('db_api_keys_table_name'));}$ret=false;foreach($apiKeys['result'] as $v){if( $v['key1']===$selectedApiKey && intval($v['access_level']) >= $accessLevelNeeded && in_array($v['enabled'],['yes','true',true]) ){$ret=true;if(!empty($v['tags_whitelist'])){$this->salad->request->setParam('tags_whitelist',$v['tags_whitelist']);}break;}} return $ret;}private function loginWithHttpDigest($accessLevelNeeded){$realm='Salad Restricted Area';if(empty($_SERVER['PHP_AUTH_DIGEST'])){$this->showLoginDialog($realm);return false;}else{$data=$this->httpDigestParse($_SERVER['PHP_AUTH_DIGEST']);$digestUsers=$this->salad->config->get('auth_digest_users',[]);$selectedUser=[];foreach($digestUsers as $u){if( $u['username']===@$data['username'] && $u['access_level'] >= $accessLevelNeeded && in_array($u['enabled'],['yes','true',true]) ){$selectedUser=$u;break;}} if(empty($selectedUser)){return false;}$A1=md5($data['username'].':'.$realm.':'.$selectedUser['password']);$A2=md5($_SERVER['REQUEST_METHOD'].':'.$data['uri']);$validResponse=md5($A1.':'.$data['nonce'].':'.$data['nc'].':'.$data['cnonce'].':'.$data['qop'].':'.$A2);if($data['response']!==$validResponse){return false;}else{return true;}}}public function login($args=[]){$args['access_level_needed']=intval($args['access_level_needed']);if($args['access_level_needed']!==0 && !isset($args['access_level_needed'])){$args['access_level_needed']=9;}if( $this->salad->request->getParam('is_cli')===true || $args['access_level_needed']===0 || @$args['skip_auth']===true ){return true;}$selectedApiKey=$this->apiKeyFromHeader();if(empty($selectedApiKey)){$selectedApiKey=@$args['api_key'];}if(!empty($selectedApiKey) && $this->salad->config->get('auth_api_keys_enabled')===true){return $this->loginWithAPIKey($selectedApiKey,$args['access_level_needed']);}elseif($this->salad->config->get('auth_digest_enabled')===true){return $this->loginWithHttpDigest($args['access_level_needed']);}else{return false;}} private function apiKeyFromHeader(){$apiKey='';$headers=getallheaders();if(isset($headers['Authorization'])){$authHeader=$headers['Authorization'];if(preg_match('/Bearer\s(\S+)/',$authHeader,$matches)){$apiKey=$matches[1];}} elseif(isset($headers['X-Api-Key'])){$apiKey=$headers['X-Api-Key'];}else if(isset($headers['X-Auth-Token'])){$apiKey=$headers['X-Auth-Token'];}return $apiKey;}private function showLoginDialog($realm){header('HTTP/1.1 401 Unauthorized');header('WWW-Authenticate:Digest realm="'.$realm.'",qop="auth",nonce="'.uniqid().'",opaque="'.md5($realm).'"');}private function httpDigestParse($txt){$needed_parts=array('nonce'=>1,'nc'=>1,'cnonce'=>1,'qop'=>1,'username'=>1,'uri'=>1,'response'=>1);$data=array();$keys=implode('|',array_keys($needed_parts));preg_match_all('@('.$keys.')=(?:([\'"])([^\2]+?)\2|([^\s,]+))@',$txt,$matches,PREG_SET_ORDER);foreach($matches as $m){$data[$m[1]]=$m[3]?$m[3]:$m[4];unset($needed_parts[$m[1]]);}return $needed_parts?false:$data;}} -class Cache{private int $ttl=86400;private $salad;public function __construct($salad){$this->salad=$salad;}public function makeCacheKey($key=[],$prefix=''){return $prefix.md5($prefix.serialize($key));}public function get($key,$default=false){if($this->salad->config->get('cache_type')==='none'){return false;}$cacheFile=$this->salad->config->get('cache_path').'/'.$key;if(is_file($cacheFile)){$data=json_decode(file_get_contents($cacheFile),true);if($data['expires']>0 && $data['expires']salad->config->get('cache_type')==='none'){return;}if($ttl<0){$ttl=0;}else{if($ttl===false){$ttl=intval($this->salad->config->get('cache_ttl',1440)) * 60;}$ttl=time() + $ttl;}$data=array( 'expires'=>$ttl,'value'=>$value );$path=$this->salad->config->get('cache_path').'/';if(!is_dir($path.dirname($key))){$this->mkdirRecursive($path.dirname($key));}@file_put_contents($path.$key,json_encode($data));@chmod($path.$key,0666);}private function mkdirRecursive($pathname,$mode=0777){if(!is_dir(dirname($pathname))){return $this->mkdirRecursive(dirname($pathname),$mode);}if(!is_dir($pathname)){mkdir($pathname);@chmod($pathname,$mode);}return is_dir($pathname);}public function cleanFolder($folderName){if($this->salad->config->get('cache_type')==='none'){return;}foreach(glob($this->salad->config->get('cache_path').'/'.$folderName.'/'.'*') as $filename){@unlink($filename);}@rmdir($this->salad->config->get('cache_path').'/'.$folderName);}} +class Cache{private $salad;public function __construct($salad){$this->salad=$salad;}public function makeCacheKey($key=[],$prefix=''){return $prefix.md5($prefix.serialize($key));}public function get($key,$default=false){if($this->salad->config->get('cache_type')==='none'){return false;}$cacheFile=$this->salad->config->get('cache_path').'/'.$key;if(is_file($cacheFile)){$data=json_decode(file_get_contents($cacheFile),true);if($data['expires']>0 && $data['expires']salad->config->get('cache_type')==='none'){return;}if($ttl<0){$ttl=0;}else{if($ttl===false){$ttl=intval($this->salad->config->get('cache_ttl',1440)) * 60;}$ttl=time() + $ttl;}$data=array( 'expires'=>$ttl,'value'=>$value );$path=$this->salad->config->get('cache_path').'/';if(!is_dir($path.dirname($key))){$this->mkdirRecursive($path.dirname($key));}@file_put_contents($path.$key,json_encode($data));@chmod($path.$key,0666);}private function mkdirRecursive($pathname,$mode=0777){if(!is_dir(dirname($pathname))){return $this->mkdirRecursive(dirname($pathname),$mode);}if(!is_dir($pathname)){mkdir($pathname);@chmod($pathname,$mode);}return is_dir($pathname);}public function cleanFolder($folderName){if($this->salad->config->get('cache_type')==='none'){return;}foreach(glob($this->salad->config->get('cache_path').'/'.$folderName.'/'.'*') as $filename){@unlink($filename);}@rmdir($this->salad->config->get('cache_path').'/'.$folderName);}} class Config{private array $params=[];private function setDefaultParams(){$this->params=['cache_type'=>'none','cache_path'=>'../cache','cache_ttl'=>1440,'db_bookmarks_table_name'=>'bookmarks','db_api_keys_table_name'=>'api_keys','db_type'=>'mysql','db_file'=>'../db/salad.sqlite','db_host'=>'127.0.0.1','db_port'=>'3306','db_name'=>'salad','db_username'=>'root','db_password'=>'','enable_cors_headers'=>false,'enable_web_api'=>false,'fetch_favicons'=>false,'fetch_popular_tags'=>false,'minify_html_output'=>false,'insert_example_bookmarks'=>true,'auth_api_keys_enabled'=>false,'api_keys'=>[['key1'=>'dev-api-key','access_level'=>1,'enabled'=>false]],'auth_digest_enabled'=>false,'auth_digest_users'=>[['username'=>'admin','password'=>'admin','access_level'=>9,'enabled'=>false],],'route_not_found'=>['func'=>'\Salad\API\Controllers\NotFound::index','format'=>!empty($_GET['format'])?$_GET['format']:'html','args'=>['access_level_needed'=>0]],'routes'=>[['uri'=>'/','func'=>'\Salad\API\Controllers\Home::index','format'=>!empty($_GET['format'])?$_GET['format']:'html','args'=>['access_level_needed'=>0]],['uri'=>'/tags','func'=>'\Salad\API\Controllers\Home::index','format'=>!empty($_GET['format'])?$_GET['format']:'html','args'=>['access_level_needed'=>9]],['uri'=>'/posts','func'=>'\Salad\API\Controllers\Home::index','format'=>!empty($_GET['format'])?$_GET['format']:'html','args'=>['access_level_needed'=>9]],['uri'=>'/tags/get','func'=>'\Salad\API\Controllers\Tags\Get::index','format'=>@$_GET['format'],'args'=>['tag'=>@$_GET['tag'],'search'=>@$_GET['search'],'sort'=>@$_GET['sort'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/tags/recent','func'=>'\Salad\API\Controllers\Tags\Recent::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/tags/rename','func'=>'\Salad\API\Controllers\Tags\Rename::index','format'=>@$_GET['format'],'args'=>['old'=>@$_GET['old'],'new'=>@$_GET['new'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/tags/delete','func'=>'\Salad\API\Controllers\Tags\Delete::index','format'=>@$_GET['format'],'args'=>['tag'=>@$_GET['tag'],'also_delete_bookmarks'=>@$_GET['also_delete_bookmarks'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/update','func'=>'\Salad\API\Controllers\Posts\Update::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/add','func'=>'\Salad\API\Controllers\Posts\Add::index','format'=>@$_GET['format'],'args'=>['url'=>@$_GET['url'],'description'=>@$_GET['description'],'extended'=>@$_GET['extended'],'tags'=>@$_GET['tags'],'dt'=>@$_GET['dt'],'replace'=>@$_GET['replace'],'shared'=>@$_GET['shared'],'old_hash'=>@$_GET['old_hash'],'shared'=>@$_GET['shared'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/delete','func'=>'\Salad\API\Controllers\Posts\Delete::index','format'=>@$_GET['format'],'args'=>['url'=>@$_GET['url'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/get','func'=>'\Salad\API\Controllers\Posts\Get::index','format'=>@$_GET['format'],'args'=>['tag'=>@$_GET['tag'],'search'=>@$_GET['search'],'dt'=>@$_GET['dt'],'url'=>@$_GET['url'],'hashes'=>@$_GET['hashes'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/recent','func'=>'\Salad\API\Controllers\Posts\Recent::index','format'=>@$_GET['format'],'args'=>['tag'=>@$_GET['tag'],'count'=>@$_GET['count'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/dates','func'=>'\Salad\API\Controllers\Posts\Dates::index','format'=>@$_GET['format'],'args'=>['tag'=>@$_GET['tag'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/all?hashes','func'=>'\Salad\API\Controllers\Posts\AllHashes::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/all','func'=>'\Salad\API\Controllers\Posts\All::index','format'=>@$_GET['format'],'args'=>['tag'=>@$_GET['tag'],'search'=>@$_GET['search'],'start'=>@$_GET['start'],'results'=>@$_GET['results'],'count'=>@$_GET['count'],'count_only'=>@$_GET['count_only'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/suggest','func'=>'\Salad\API\Controllers\Posts\Suggest::index','format'=>@$_GET['format'],'args'=>['url'=>@$_GET['url'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/posts/favicon','func'=>'\Salad\API\Controllers\Posts\FavIcon::index','format'=>'png','args'=>['hash'=>@$_GET['hash'],'url'=>@$_GET['url'],'api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/keys/gen','func'=>'\Salad\API\Controllers\Keys\Gen::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/keys/all','func'=>'\Salad\API\Controllers\Keys\All::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'access_level_needed'=>9]],['uri'=>'/keys/add','func'=>'\Salad\API\Controllers\Keys\Add::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'key1'=>@$_GET['key1'],'description'=>@$_GET['description'],'access_level'=>@$_GET['access_level'],'enabled'=>@$_GET['enabled'],'tags_whitelist'=>@$_GET['tags_whitelist'],'access_level_needed'=>9]],['uri'=>'/keys/update','func'=>'\Salad\API\Controllers\Keys\Update::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'id'=>@$_GET['id'],'key1'=>@$_GET['key1'],'description'=>@$_GET['description'],'access_level'=>@$_GET['access_level'],'enabled'=>@$_GET['enabled'],'tags_whitelist'=>@$_GET['tags_whitelist'],'access_level_needed'=>9]],['uri'=>'/keys/delete','func'=>'\Salad\API\Controllers\Keys\Delete::index','format'=>@$_GET['format'],'args'=>['api_key'=>@$_GET['api_key'],'id'=>@$_GET['id'],'access_level_needed'=>9]]]];}public function __construct($salad){$this->setDefaultParams();if(is_file('./config.php')){$config=include('./config.php');if(is_array($config)){$this->params=array_merge($this->params,$config);}} if(is_file('./config.user.php')){$userConfig=include('./config.user.php');if(is_array($userConfig)){$this->params=array_merge($this->params,$userConfig);}}}public function get($key,$default=false){return isset($this->params[$key])?$this->params[$key]:$default;}public function getAll(){return $this->params;}public function set($key,$val=''){$configItems=[];if(!is_array($key)){$configItems[$key]=$val;}else{$configItems=$key;}foreach($configItems as $k=>$v){$this->params[$k]=$v;}} } -class DB{private $dbh=false;private $salad;public function __construct($salad){$this->salad=$salad;}public function query($sql){return $this->dbh->query($sql);}public function init(){try{switch($this->salad->config->get('db_type')){case 'sqlite';default:$this->dbh=new \PDO('sqlite:'.$this->salad->config->get('db_file'));$this->exec('PRAGMA encoding="UTF-8"');break;case 'mysql':$this->dbh=new \PDO('mysql:host='.$this->salad->config->get('db_host').';port='.$this->salad->config->get('db_port').';charset=utf8mb4;dbname='.$this->salad->config->get('db_name'),$this->salad->config->get('db_username'),$this->salad->config->get('db_password'),[]);break;}$this->dbh->setAttribute(\PDO::ATTR_ERRMODE,\PDO::ERRMODE_EXCEPTION);if(!$this->tableExists($this->salad->config->get('db_bookmarks_table_name'))){$this->createInitialDB();if($this->salad->config->get('insert_example_bookmarks')===true){$this->insertSaladBookmarks();$this->insertExampleAPIKeys();}}}catch (\PDOException $e){exit('DB Error:'.$e->getMessage());}} public function exec($query,$params=[]){if($this->dbh===false){$this->init();}$queries=[];if(is_array($query)){$queries=$query;}else{$queries[]=['sql'=>$query,'params'=>$params];}$this->dbh->beginTransaction();try{foreach($queries as $q){$qParams=[];if(is_array($q)){$qParams=!empty($q['params'])?$q['params']:[];$qSQL=$q['sql'];}else{$qSQL=$q;}$stmt=$this->dbh->prepare($qSQL);$stmt->execute($qParams);}$result=$this->dbh->commit();return ['msg'=>'ok','result'=>$result];}catch (\PDOException $e){$this->dbh->rollBack();return ['msg'=>$e->getMessage(),'result'=>0,'sql'=>$query,'params'=>$params];}} public function select($query,$params=[]){if($this->dbh===false){$this->init();}try{$sth=$this->dbh->prepare($query);$sth->setFetchMode(\PDO::FETCH_ASSOC);$sth->execute($params);$result=$sth->fetchAll();return ['msg'=>'ok','result'=>$result];}catch (\PDOException $e){return ['msg'=>$e->getMessage(),'result'=>[],'sql'=>$query,'params'=>$params];}} private function createInitialDB(){switch($this->salad->config->get('db_type')){case 'mysql':$schema=explode(';','CREATE TABLE `bookmarks` ( `id` INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, `url` TEXT NOT NULL, `description` TEXT NOT NULL, `extended` TEXT DEFAULT "", `tags` TEXT DEFAULT "", `created` VARCHAR(15) NOT NULL, `modified` VARCHAR(15) NOT NULL, `hash` VARCHAR(32) NOT NULL, `shared` VARCHAR(3) NOT NULL DEFAULT "yes", INDEX (`created`), INDEX (`modified`), INDEX (`shared`), UNIQUE (`hash`)) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;CREATE TABLE `api_keys` ( `id` INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, `key1` VARCHAR(255) NOT NULL, `description` VARCHAR(255) DEFAULT "", `access_level` VARCHAR(2) NOT NULL DEFAULT "1", `enabled` VARCHAR(3) NOT NULL DEFAULT "no", `tags_whitelist` TEXT DEFAULT "", INDEX (`enabled`), UNIQUE (`key1`)) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;');break;case 'sqlite':default:$schema=explode(';','CREATE TABLE "bookmarks" ("id" INTEGER PRIMARY KEY AUTOINCREMENT,"url" TEXT NOT NULL,"description" TEXT NOT NULL,"extended" TEXT DEFAULT "","tags" TEXT DEFAULT "","created" TEXT NOT NULL,"modified" TEXT NOT NULL,"hash" TEXT NOT NULL,"shared" TEXT NOT NULL DEFAULT "yes");CREATE INDEX "created" ON "bookmarks" ("created");CREATE INDEX "modified" ON "bookmarks" ("modified");CREATE INDEX "shared" ON "bookmarks" ("shared");CREATE UNIQUE INDEX "hash" ON "bookmarks" ("hash");CREATE TABLE "api_keys" ("id" INTEGER PRIMARY KEY AUTOINCREMENT,"key1" TEXT NOT NULL,"description" TEXT DEFAULT "","access_level" TEXT NOT NULL DEFAULT "1","enabled" TEXT NOT NULL DEFAULT "no","tags_whitelist" TEXT DEFAULT "");CREATE INDEX "enabled" ON "api_keys" ("enabled");CREATE UNIQUE INDEX "key1" ON "api_keys" ("key1");');break;}$schema=array_filter($schema,'trim');$schema=array_filter($schema,'strlen');$schema=array_map(function ($value){return str_replace( ['CREATE TABLE `bookmarks`','CREATE TABLE "bookmarks"','CREATE TABLE `api_keys`','CREATE TABLE "api_keys"'],['CREATE TABLE `'.$this->salad->config->get('db_bookmarks_table_name').'`','CREATE TABLE "'.$this->salad->config->get('db_bookmarks_table_name').'"','CREATE TABLE `'.$this->salad->config->get('db_api_keys_table_name').'`','CREATE TABLE "'.$this->salad->config->get('db_api_keys_table_name').'"'],$value );},$schema);foreach($schema as $query){$this->dbh->exec($query);}} private function tableExists($table){try{$this->dbh->query('SELECT 1 FROM '.$table);}catch (\PDOException $e){return false;}return true;}private function insertExampleBookmarks(){$bookmarks=json_decode('[ { "url": "https://github.com/notwaldorf/dear-sir-or-madam", "description": "notwaldorf/dear-sir-or-madam: 💌 Bookmarklet that ransomifies your internets", "tags": "javascript bookmarklet github color" }, { "url": "http://www.srehttp.org/apps/gif_text/mkgiftxt.htm", "description": "GIF_TEXT: A Graphics Text Generator", "tags": "image generator gif" }, { "url": "https://browsers.evolt.org/", "description": "evolt.org - Browser Archive", "tags": "browsers archive javascript" }, { "url": "https://www.mercurytheatre.info/", "description": "The Mercury Theatre on the Air", "tags": "audio media radio" }, { "url": "color:FFFFFF,81C757,333333,D3D2D2,0000FF,DDDDDD,FF0000", "description": "Salad color scheme", "tags": "color swatches" }, { "url": "https://web.archive.org/web/20050207095344/http://home.earthlink.net/~sarasohn/aboutgs.html", "description": "Father Guido Sarducci", "tags": "media comedy tv" }]',true);$queries=[];$url='';$tags='';foreach($bookmarks as $link){if(!empty($link['post'])){$link=$link['post'];}if(!empty($link['url'])){$url=$link['url'];}elseif(!empty($link['href'])){$url=$link['href'];}if(!empty($link['tag'])){$tags=$link['tag'];}elseif(!empty($link['tags'])){$tags=$link['tags'];}if(!empty($link['time'])){$dt=strtotime($link['time']);}else{$dt=strtotime('now');}$url2=$url;$queries[]=['sql'=>'INSERT INTO '.$this->salad->config->get('db_bookmarks_table_name').' (url,description,extended,tags,created,modified,hash,shared) VALUES (:url,:description,:extended,:tags,:created,:modified,:hash,:shared)','params'=>['url'=>$url2,'description'=>@$link['description'],'extended'=>@$link['extended'],'tags'=>' '.$tags.' ','created'=>$dt,'modified'=>$dt,'hash'=>md5($url),'shared'=>'yes']];}$this->exec($queries);}private function insertExampleAPIKeys(){$queries=[];$queries[]=['sql'=>'INSERT INTO '.$this->salad->config->get('db_api_keys_table_name').' (key1,description,access_level,enabled) VALUES (:key1,:description,:access_level,:enabled)','params'=>['key1'=>'1234','description'=>'key1','access_level'=>'1','enabled'=>'no']];$this->exec($queries);}private function insertSaladBookmarks(){$saladBookmarksFile=realpath(__DIR__.'/../Utils/Schema/salad.bookmarks.json');if(!is_file($saladBookmarksFile)){$this->insertExampleBookmarks();return;}$bookmarks=json_decode(file_get_contents(realpath(__DIR__.'/../Utils/Schema/salad.bookmarks.json')),true);$queries=[];$bookmarks=$bookmarks[2]['data'];foreach($bookmarks as $link){$link['url']=str_replace('+',' ',$link['url']);$queries[]=['sql'=>'INSERT INTO '.$this->salad->config->get('db_bookmarks_table_name').' (url,description,extended,tags,created,modified,hash,shared) VALUES (:url,:description,:extended,:tags,:created,:modified,:hash,:shared)','params'=>['url'=>$link['url'],'description'=>@$link['description'],'extended'=>@$link['extended'],'tags'=>' '.trim($link['tags']).' ','created'=>$link['created'],'modified'=>$link['modified'],'hash'=>$link['hash'],'shared'=>'yes']];}$exec=$this->exec($queries);}} +class DB{private $dbh;private $salad;public function __construct($salad){$this->salad=$salad;}public function query($sql){return $this->dbh->query($sql);}public function init(){try{switch($this->salad->config->get('db_type')){case 'sqlite';default:$this->dbh=new \PDO('sqlite:'.$this->salad->config->get('db_file'));$this->exec('PRAGMA encoding="UTF-8"');break;case 'mysql':$this->dbh=new \PDO('mysql:host='.$this->salad->config->get('db_host').';port='.$this->salad->config->get('db_port').';charset=utf8mb4;dbname='.$this->salad->config->get('db_name'),$this->salad->config->get('db_username'),$this->salad->config->get('db_password'),[]);break;}$this->dbh->setAttribute(\PDO::ATTR_ERRMODE,\PDO::ERRMODE_EXCEPTION);if(!$this->tableExists($this->salad->config->get('db_bookmarks_table_name'))){$this->createInitialDB();if($this->salad->config->get('insert_example_bookmarks')===true){$this->insertExampleBookmarks();$this->insertExampleAPIKeys();}}}catch (\PDOException $e){exit('DB Error:'.$e->getMessage());}} public function exec($query,$params=[]){if($this->dbh===null){$this->init();}$queries=[];if(is_array($query)){$queries=$query;}else{$queries[]=['sql'=>$query,'params'=>$params];}$this->dbh->beginTransaction();try{foreach($queries as $q){$qParams=[];if(is_array($q)){$qParams=!empty($q['params'])?$q['params']:[];$qSQL=$q['sql'];}else{$qSQL=$q;}$stmt=$this->dbh->prepare($qSQL);$stmt->execute($qParams);}$result=$this->dbh->commit();return ['msg'=>'ok','result'=>$result];}catch (\PDOException $e){$this->dbh->rollBack();return ['msg'=>$e->getMessage(),'result'=>0,'sql'=>$query,'params'=>$params];}} public function select($query,$params=[]){if($this->dbh===null){$this->init();}try{$sth=$this->dbh->prepare($query);$sth->setFetchMode(\PDO::FETCH_ASSOC);$sth->execute($params);$result=$sth->fetchAll();return ['msg'=>'ok','result'=>$result];}catch (\PDOException $e){return ['msg'=>$e->getMessage(),'result'=>[],'sql'=>$query,'params'=>$params];}} private function createInitialDB(){switch($this->salad->config->get('db_type')){case 'mysql':$schema=explode(';','CREATE TABLE `bookmarks` ( `id` INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, `url` TEXT NOT NULL, `description` TEXT NOT NULL, `extended` TEXT DEFAULT "", `tags` TEXT DEFAULT "", `created` VARCHAR(15) NOT NULL, `modified` VARCHAR(15) NOT NULL, `hash` VARCHAR(32) NOT NULL, `shared` VARCHAR(3) NOT NULL DEFAULT "yes", INDEX (`created`), INDEX (`modified`), INDEX (`shared`), UNIQUE (`hash`)) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;CREATE TABLE `api_keys` ( `id` INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY, `key1` VARCHAR(255) NOT NULL, `description` VARCHAR(255) DEFAULT "", `access_level` VARCHAR(2) NOT NULL DEFAULT "1", `enabled` VARCHAR(3) NOT NULL DEFAULT "no", `tags_whitelist` TEXT DEFAULT "", INDEX (`enabled`), UNIQUE (`key1`)) ENGINE=InnoDB DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;');break;case 'sqlite':default:$schema=explode(';','CREATE TABLE "bookmarks" ("id" INTEGER PRIMARY KEY AUTOINCREMENT,"url" TEXT NOT NULL,"description" TEXT NOT NULL,"extended" TEXT DEFAULT "","tags" TEXT DEFAULT "","created" TEXT NOT NULL,"modified" TEXT NOT NULL,"hash" TEXT NOT NULL,"shared" TEXT NOT NULL DEFAULT "yes");CREATE INDEX "created" ON "bookmarks" ("created");CREATE INDEX "modified" ON "bookmarks" ("modified");CREATE INDEX "shared" ON "bookmarks" ("shared");CREATE UNIQUE INDEX "hash" ON "bookmarks" ("hash");CREATE TABLE "api_keys" ("id" INTEGER PRIMARY KEY AUTOINCREMENT,"key1" TEXT NOT NULL,"description" TEXT DEFAULT "","access_level" TEXT NOT NULL DEFAULT "1","enabled" TEXT NOT NULL DEFAULT "no","tags_whitelist" TEXT DEFAULT "");CREATE INDEX "enabled" ON "api_keys" ("enabled");CREATE UNIQUE INDEX "key1" ON "api_keys" ("key1");');break;}$schema=array_filter($schema,'trim');$schema=array_filter($schema,'strlen');$schema=array_map(function ($value){return str_replace( ['CREATE TABLE `bookmarks`','CREATE TABLE "bookmarks"','CREATE TABLE `api_keys`','CREATE TABLE "api_keys"'],['CREATE TABLE `'.$this->salad->config->get('db_bookmarks_table_name').'`','CREATE TABLE "'.$this->salad->config->get('db_bookmarks_table_name').'"','CREATE TABLE `'.$this->salad->config->get('db_api_keys_table_name').'`','CREATE TABLE "'.$this->salad->config->get('db_api_keys_table_name').'"'],$value );},$schema);foreach($schema as $query){$this->dbh->exec($query);}} private function tableExists($table){try{$this->dbh->query('SELECT 1 FROM '.$table);}catch (\PDOException $e){return false;}return true;}private function insertExampleBookmarks(){$bookmarks=json_decode('[ { "url": "https://github.com/notwaldorf/dear-sir-or-madam", "description": "notwaldorf/dear-sir-or-madam: 💌 Bookmarklet that ransomifies your internets", "tags": "javascript bookmarklet github color" }, { "url": "http://www.srehttp.org/apps/gif_text/mkgiftxt.htm", "description": "GIF_TEXT: A Graphics Text Generator", "tags": "image generator gif" }, { "url": "https://browsers.evolt.org/", "description": "evolt.org - Browser Archive", "tags": "browsers archive javascript" }, { "url": "https://www.mercurytheatre.info/", "description": "The Mercury Theatre on the Air", "tags": "audio media radio" }, { "url": "color:FFFFFF,81C757,333333,D3D2D2,0000FF,DDDDDD,FF0000", "description": "Salad color scheme", "tags": "color swatches" }, { "url": "https://web.archive.org/web/20050207095344/http://home.earthlink.net/~sarasohn/aboutgs.html", "description": "Father Guido Sarducci", "tags": "media comedy tv" }]',true);$queries=[];$url='';$tags='';foreach($bookmarks as $link){if(!empty($link['post'])){$link=$link['post'];}if(!empty($link['url'])){$url=$link['url'];}elseif(!empty($link['href'])){$url=$link['href'];}if(!empty($link['tag'])){$tags=$link['tag'];}elseif(!empty($link['tags'])){$tags=$link['tags'];}if(!empty($link['time'])){$dt=strtotime($link['time']);}else{$dt=strtotime('now');}$url2=$url;$queries[]=['sql'=>'INSERT INTO '.$this->salad->config->get('db_bookmarks_table_name').' (url,description,extended,tags,created,modified,hash,shared) VALUES (:url,:description,:extended,:tags,:created,:modified,:hash,:shared)','params'=>['url'=>$url2,'description'=>@$link['description'],'extended'=>@$link['extended'],'tags'=>' '.$tags.' ','created'=>$dt,'modified'=>$dt,'hash'=>md5($url),'shared'=>'yes']];}$this->exec($queries);}private function insertExampleAPIKeys(){$queries=[];$queries[]=['sql'=>'INSERT INTO '.$this->salad->config->get('db_api_keys_table_name').' (key1,description,access_level,enabled) VALUES (:key1,:description,:access_level,:enabled)','params'=>['key1'=>'1234','description'=>'key1','access_level'=>'1','enabled'=>'no']];$this->exec($queries);}} class Main{public $config;public $auth;public $db;public $cache;public $request;public $response;public $utils;public $sqlUtils;public $view;public $models;public $ext;public function __construct($api_file_included=false){$this->utils=new \Salad\Utils($this);$this->config=new \Salad\Config($this);$this->auth=new \Salad\Auth($this);$this->db=new \Salad\DB($this);$this->cache=new \Salad\Cache($this);$this->request=new \Salad\Request($this);$this->response=new \Salad\Response($this);$this->sqlUtils=new \Salad\SQLUtils($this);$this->view=new \Salad\View($this);if($api_file_included===true){$this->request->setParam('file_api_included',true);}} public function registerMyObjects($objects=[]){if(!empty($objects['models'])){$this->models=new \stdClass();foreach($objects['models']['classes'] as $modelClass){$class=$objects['models']['namespace'].'\\'.ucfirst($modelClass);$this->models->$modelClass=new $class($this);}} if(!empty($objects['ext'])){$this->ext=new \stdClass();foreach($objects['ext']['classes'] as $extClass){$class=$objects['ext']['namespace'].'\\'.ucfirst($extClass);$this->ext->$extClass=new $class($this);}}}} class Request{protected $salad;public function __construct($salad){$this->salad=$salad;$this->setParams();}private array $params=['uri'=>'/','base_uri'=>'/','format'=>'json','is_cli'=>false,'file_api_included'=>false,'callback'=>'','context'=>''];protected function setParams(){switch(PHP_SAPI){case 'cli':$this->setParam('is_cli',true);$this->setCliParams();break;default:self::setWebParams();break;}$this->setParam('callback',$this->salad->utils->safeForJSONP(''.@$_GET['callback']));$this->setParam('context',$this->salad->utils->safeForJSONP(''.@$_GET['context']));}private function setCliParams(){global $argv;if(count($argv)<2){die("Usage:php [key=value]...\n");}$url=$argv[1];$this->setParam('uri',$url);for ($i=2;$isetParam('base_uri','/');}private function setWebParams(){$uri=rtrim(filter_var(@$_SERVER['REQUEST_URI'],FILTER_SANITIZE_URL),'/').'/';$pattern='/^\/[^\/]*\.php/';$uri=preg_replace($pattern,'',$uri);if(strpos($uri,'all?hashes')===false){$uri=preg_replace('/\?(.*)$/','',$uri);}$uri=rtrim($uri,'/');if(empty($uri)){$uri='/';}$this->setUri($uri,preg_replace("/\/[^\/]+$/",'/',@$_SERVER['SCRIPT_NAME']));}private function setFileIncludedParams($options){preg_match('/(.*)\?(.*)/',$options['uri'],$urlMatches);$this->setUri($urlMatches[1],@$options['base_uri']);parse_str($urlMatches[2],$urlMatches);$_GET=array_merge($_GET,$urlMatches);if(!empty($_GET['format'])){$this->setParam('format',$_GET['format']);}} public function setParam($key,$val){$this->params[$key]=$val;}public function getParam($key,$default=''){return !empty($this->params[$key])?$this->params[$key]:$default;}public function setUri($uri='',$baseUri=''){if(!empty($uri)){$this->setParam('uri',$uri);if(!empty($baseUri)){$this->setParam('base_uri',$baseUri);}}}public function handle($options=[]){if(!empty($options['uri']) && $this->salad->request->getParam('file_api_included')===true){$this->setFileIncludedParams($options);}$router=new \Salad\Router($this->salad);$selectedRoute=$router->findRoute();$this->salad->response->setOutputData(call_user_func($selectedRoute[0],$selectedRoute[1],$selectedRoute[2]));}private string $fullBaseUri='';public function getFullBaseUri(){if(!empty($this->fullBaseUri)){return $this->fullBaseUri;}$fullBaseUri='';$port='';if(!empty($_SERVER['HTTP_HOST'])){switch($_SERVER['SERVER_PORT']){case '80':case '443':break;default:$port=':'.$_SERVER['SERVER_PORT'];break;}$fullBaseUri .= $_SERVER['REQUEST_SCHEME'].'://'.$_SERVER['HTTP_HOST'].$port;}$fullBaseUri .= $this->getParam('base_uri');$this->fullBaseUri=$fullBaseUri;return $this->fullBaseUri;}} -class Response{private array $outputData=[];private $salad;public function __construct($salad){$this->salad=$salad;}public function setOutputData($outputData){$outputData=$outputData===null?[]:$outputData;$this->outputData=$outputData;}public function getOutput($setHeaders=false){if($setHeaders===true){$this->setHeaders($this->salad->request->getParam('format'));}return $this->generateOutput($this->salad->request->getParam('format'));}public function setHeaders($format){if($this->salad->request->getParam('is_cli')===true || $this->salad->request->getParam('api_self_included')===true){return;}$this->setCorsHeaders();$this->setExtraHeaders();$this->setContentTypeHeader($format);}private function setContentTypeHeader($format){$contentType='';switch($format){case 'xml':$contentType='text/xml;charset=utf-8';break;case 'json':case 'js':$contentType='text/javascript;charset=utf-8';break;case 'css':$contentType='text/css;charset=utf-8';break;case 'webmanifest':$contentType='application/manifest+json;charset=utf-8';break;case 'html':case 'html-partial':$contentType='text/html;charset=utf-8';break;case 'png':$contentType='image/png';break;case 'svg':$contentType='image/svg+xml';break;case 'text':case 'txt':default:$contentType='text/plain;charset=utf-8';break;}header('Content-type:'.$contentType);}private function setCorsHeaders(){if($this->salad->config->get('enable_cors_headers')===true){header('Access-Control-Allow-Origin:*',true);header('Access-Control-Allow-Methods:GET,PUT,POST,DELETE,OPTIONS,post,get',true);header('Access-Control-Allow-Headers:Origin,Content-Type,X-Auth-Token,Authorization,X-Api-Key',true);header("Access-Control-Allow-Credentials",true);}} private function setExtraHeaders(){header('Referrer-Policy:no-referrer',true);header('X-Content-Type-Options:nosniff',true);}private function generateOutput($format){$data='';switch($format){case 'xml':default:$data=$this->generateXML();break;case 'json':$data=$this->generateJSON();$cb=$this->salad->request->getParam('callback');$cnx=$this->salad->request->getParam('context');if(!empty($cb)){$data=$cb.'('.$data;if(!empty($cnx)){$data .= ',"'.$cnx.'"';}$data .= ');';}break;case 'html':$data=$this->generateHTML(true);break;case 'html-partial':$data=$this->generateHTML(false);break;case 'css':case 'js':case 'webmanifest':case 'png':case 'svg':case 'text':case 'txt':$data=$this->generateText();break;}return $data;}private function generateXML(){if(empty($this->outputData['xml'])){$xml='';}else{$xml=$this->outputData['xml'];}$xml=new \SimpleXMLElement($xml);return $xml->asXML();}private function generateHTML($fullHTML=false){if($fullHTML===false){$partialHTML=@$this->outputData['html']['body'];if($this->salad->config->get('minify_html_output')===true){$partialHTML=$this->salad->utils->minifyOutput(['body'=>$partialHTML])['body'];}return $partialHTML;}else{$htmlSkeleton=''.trim(''.@$this->outputData['html']['title']).'';if($this->salad->config->get('minify_html_output')===true){$htmlSkeleton=$this->salad->utils->minifyOutput(['body'=>$htmlSkeleton])['body'];}if(!empty($this->outputData['html']['head'])){if($this->salad->config->get('minify_html_output')===true){$this->outputData['html']['head']=$this->salad->utils->minifyOutput(['head'=>$this->outputData['html']['head']])['head'];}$htmlSkeleton .= $this->outputData['html']['head'];}if(!empty($this->outputData['html']['css'])){$htmlSkeleton .= '';}$htmlSkeleton .= '';if(!empty($this->outputData['html']['body'])){if($this->salad->config->get('minify_html_output')===true){$this->outputData['html']['body']=$this->salad->utils->minifyOutput(['body'=>$this->outputData['html']['body']])['body'];}$htmlSkeleton .= $this->outputData['html']['body'];}if(!empty($this->outputData['html']['js'])){$htmlSkeleton .= '';}$htmlSkeleton .= '';return $htmlSkeleton;}} private function generateText(){if(!empty($this->outputData['bin-data'])){return $this->outputData['bin-data'];}$text=$this->outputData['text'];if($this->salad->config->get('minify_html_output')===true){$text=$this->salad->utils->minifyOutput(['body'=>$text])['body'];}return $text;}private function generateJSON(){return isset($this->outputData['array'])?json_encode($this->outputData['array']):$this->xmlToJson(@$this->outputData['xml']);}private string $nonce='';public function nonce(){if(!empty($this->nonce)){return $this->nonce;}$this->nonce=bin2hex(random_bytes(10));return $this->nonce;}public function debug($obj,$type="pr"){switch($type){case 'pr':default:header('content-type:text/plain;');print_r($obj);break;case 'vd':header('content-type:text/html;');var_dump($obj);break;}die();}public function debugF($context,$data){if(!is_array($data)){$data=[$data];}$data=json_encode($data,JSON_PRETTY_PRINT);$txt="\n".date('g:i:s A',strtotime('now')).":".$data."\n";file_put_contents($this->salad->config->get('cache_path').'/'.$context.'.txt',$txt,FILE_APPEND | LOCK_EX);}public function generateErrorMessage($type,$message=''){$message=!empty($message)?$message:'something went wrong';return $this->generateGenericMessage($type,$message);}public function generateDoneMessage($type,$message=''){$message=!empty($message)?$message:'done';return $this->generateGenericMessage($type,$message,true);}public function generateAuthFailedMessage(){return $this->generateErrorMessage('for:posts','Authentication Failed');}public function generateNotFoundMessage(){return $this->generateErrorMessage('for:posts','404 Not Found');}private function generateGenericMessage($type='for:tags',$message='',$isPositive=false){switch($type){case 'for:posts':$xml='';$array=['result'=>['code'=>$message]];break;case 'for:tags':default:$xml=''.$message.'';$array=['result'=>$message];break;}if($isPositive===true){$html=['title'=>$message,'body'=>$message,];}else{$sadKitty='';$html=['title'=>$message,'body'=>'
'.$message.'
'.$message.'
','css'=>'body {background-color:#000000;} figure {margin:0;padding:0;text-align:left;} figcaption{ text-align:center;width:191px;color:aqua;}'];}return ['xml'=>$xml,'array'=>$array,'html'=>$html];}private function xmlToJson($xmlContent){if(!$xmlContent){$xml=['result'=>['code'=>'no data for this format']];}else{$xml=new \SimpleXMLElement($xmlContent);}return json_encode($xml);}public function redirect($url){header('Location:'.$url);exit(0);}} +class Response{private array $outputData=[];private $salad;public function __construct($salad){$this->salad=$salad;}public function setOutputData($outputData){$outputData=$outputData===null?[]:$outputData;$this->outputData=$outputData;}public function getOutput($setHeaders=false){if($setHeaders===true){$this->setHeaders($this->salad->request->getParam('format'));}return $this->generateOutput($this->salad->request->getParam('format'));}public function setHeaders($format){if($this->salad->request->getParam('is_cli')===true || $this->salad->request->getParam('api_self_included')===true){return;}$this->setCorsHeaders();$this->setExtraHeaders();$this->setContentTypeHeader($format);}private function setContentTypeHeader($format){$contentType='';switch($format){case 'xml':$contentType='text/xml;charset=utf-8';break;case 'json':case 'js':$contentType='text/javascript;charset=utf-8';break;case 'css':$contentType='text/css;charset=utf-8';break;case 'webmanifest':$contentType='application/manifest+json;charset=utf-8';break;case 'html':case 'html-partial':$contentType='text/html;charset=utf-8';break;case 'png':$contentType='image/png';break;case 'svg':$contentType='image/svg+xml';break;case 'text':case 'txt':default:$contentType='text/plain;charset=utf-8';break;}header('Content-type:'.$contentType);}private function setCorsHeaders(){if($this->salad->config->get('enable_cors_headers')===true){header('Access-Control-Allow-Origin:*',true);header('Access-Control-Allow-Methods:GET,PUT,POST,DELETE,OPTIONS',true);header('Access-Control-Allow-Headers:Origin,Content-Type,X-Auth-Token,Authorization,X-Api-Key',true);}} private function setExtraHeaders(){header('Referrer-Policy:no-referrer',true);header('X-Content-Type-Options:nosniff',true);}private function generateOutput($format){$data='';switch($format){case 'xml':default:$data=$this->generateXML();break;case 'json':$data=$this->generateJSON();$cb=$this->salad->request->getParam('callback');$cnx=$this->salad->request->getParam('context');if(!empty($cb)){$data=$cb.'('.$data;if(!empty($cnx)){$data .= ',"'.$cnx.'"';}$data .= ');';}break;case 'html':$data=$this->generateHTML(true);break;case 'html-partial':$data=$this->generateHTML(false);break;case 'css':case 'js':case 'webmanifest':case 'png':case 'svg':case 'text':case 'txt':$data=$this->generateText();break;}return $data;}private function generateXML(){if(empty($this->outputData['xml'])){$xml='';}else{$xml=$this->outputData['xml'];}$xml=new \SimpleXMLElement($xml);return $xml->asXML();}private function generateHTML($fullHTML=false){if($fullHTML===false){$partialHTML=@$this->outputData['html']['body'];if($this->salad->config->get('minify_html_output')===true){$partialHTML=$this->salad->utils->minifyOutput(['body'=>$partialHTML])['body'];}return $partialHTML;}else{$htmlSkeleton=''.trim(''.@$this->outputData['html']['title']).'';if($this->salad->config->get('minify_html_output')===true){$htmlSkeleton=$this->salad->utils->minifyOutput(['body'=>$htmlSkeleton])['body'];}if(!empty($this->outputData['html']['head'])){if($this->salad->config->get('minify_html_output')===true){$this->outputData['html']['head']=$this->salad->utils->minifyOutput(['head'=>$this->outputData['html']['head']])['head'];}$htmlSkeleton .= $this->outputData['html']['head'];}if(!empty($this->outputData['html']['css'])){$htmlSkeleton .= '';}$htmlSkeleton .= '';if(!empty($this->outputData['html']['body'])){if($this->salad->config->get('minify_html_output')===true){$this->outputData['html']['body']=$this->salad->utils->minifyOutput(['body'=>$this->outputData['html']['body']])['body'];}$htmlSkeleton .= $this->outputData['html']['body'];}if(!empty($this->outputData['html']['js'])){$htmlSkeleton .= '';}$htmlSkeleton .= '';return $htmlSkeleton;}} private function generateText(){if(!empty($this->outputData['bin-data'])){return $this->outputData['bin-data'];}$text=$this->outputData['text'];if($this->salad->config->get('minify_html_output')===true){$text=$this->salad->utils->minifyOutput(['body'=>$text])['body'];}return $text;}private function generateJSON(){return isset($this->outputData['array'])?json_encode($this->outputData['array']):$this->xmlToJson(@$this->outputData['xml']);}private string $nonce='';public function nonce(){if(!empty($this->nonce)){return $this->nonce;}$this->nonce=bin2hex(random_bytes(10));return $this->nonce;}public function debug($obj,$type="pr"){switch($type){case 'pr':default:header('content-type:text/plain;');print_r($obj);break;case 'vd':header('content-type:text/html;');var_dump($obj);break;}die();}public function debugF($context,$data){if(!is_array($data)){$data=[$data];}$data=json_encode($data,JSON_PRETTY_PRINT);$txt="\n".date('g:i:s A',strtotime('now')).":".$data."\n";file_put_contents($this->salad->config->get('cache_path').'/'.$context.'.txt',$txt,FILE_APPEND | LOCK_EX);}public function generateErrorMessage($type,$message=''){$message=!empty($message)?$message:'something went wrong';return $this->generateGenericMessage($type,$message);}public function generateDoneMessage($type,$message=''){$message=!empty($message)?$message:'done';return $this->generateGenericMessage($type,$message,true);}public function generateAuthFailedMessage(){return $this->generateErrorMessage('for:posts','Authentication Failed');}public function generateNotFoundMessage(){return $this->generateErrorMessage('for:posts','404 Not Found');}private function generateGenericMessage($type='for:tags',$message='',$isPositive=false){switch($type){case 'for:posts':$xml='';$array=['result'=>['code'=>$message]];break;case 'for:tags':default:$xml=''.$message.'';$array=['result'=>$message];break;}if($isPositive===true){$html=['title'=>$message,'body'=>$message,];}else{$sadKitty='';$html=['title'=>$message,'body'=>'
'.$message.'
'.$message.'
','css'=>'body {background-color:#000000;} figure {margin:0;padding:0;text-align:left;} figcaption{ text-align:center;width:191px;color:aqua;}'];}return ['xml'=>$xml,'array'=>$array,'html'=>$html];}private function xmlToJson($xmlContent){if(!$xmlContent){$xml=['result'=>['code'=>'no data for this format']];}else{$xml=new \SimpleXMLElement($xmlContent);}return json_encode($xml);}public function redirect($url){header('Location:'.$url);exit(0);}} class Router{private $salad;public function __construct($salad){$this->salad=$salad;}private function setRouteAliases(){$routeAliases=[];foreach($this->salad->config->get('routes') as $cr){if(!empty($cr['aliases'])){$cr['aliases']=is_array($cr['aliases'])?$cr['aliases']:[$cr['aliases']];foreach($cr['aliases'] as $alias){$routeAliases[]=['uri'=>$alias,'func'=>$cr['func'],'format'=>@$cr['format'],'args'=>@$cr['args']];}}}$this->salad->config->set('routes',array_merge($routeAliases,$this->salad->config->get('routes')));}public function findRoute(){$requestedUri=$this->salad->request->getParam('uri');$baseUri=rtrim($this->salad->request->getParam('base_uri'),'/');if(empty($baseUri)){$baseUri='/';}if($this->salad->request->getParam('base_uri')!=='/'){$requestedUri=str_replace($baseUri,'',$requestedUri);}$requestedUri='/'.trim($requestedUri,'/');$this->setRouteAliases();$currentRoute=[];foreach($this->salad->config->get('routes') as $route){$pattern=$route['uri'];$args=[];if(preg_match("#^$pattern$#",$requestedUri,$matches)){array_shift($matches);$route['args']=!empty($route['args'])?$route['args']:[];$route['args']=array_filter($route['args'],fn ($value)=>$value!==null && $value!=='' && $value!=='undefined');foreach($route['args'] as $key=>$value){if(preg_match('/\$\d/',$value)){$index=intval(str_replace('$','',$value)) - 1;if(preg_match('/\$\d/',$key)){$k=intval(str_replace('$','',$key)) - 1;$args[$matches[$k]]=$this->salad->utils->urldecode($matches[$index]);}else{$args[$key]=$this->salad->utils->urldecode($matches[$index]);}} else{$args[$key]=$this->salad->utils->urldecode($value);}} foreach($_GET as $key=>$value){if(!isset($args[$key]) && !in_array($key,['access_level','access_level_needed','skip_auth','skip_url_check'])){$args[$key]=$this->salad->utils->urldecode($value);}} $currentRoute=[$route['func'],$this->salad,$args];$this->setRouteFormat(@$route['format']);break;}} if(empty($currentRoute)){$r=$this->salad->config->get('route_not_found');$this->setRouteFormat(@$r['format']);$currentRoute=[$r['func'],$this->salad,@$r['args']];}return $currentRoute;}private function setRouteFormat($format=''){if($this->salad->request->getParam('is_cli')===true){if(!empty($_GET['format'])){$this->salad->request->setParam('format',$_GET['format']);}} else{if(!empty($format)){$this->salad->request->setParam('format',$format);}}}} class SQLUtils{private $salad;public function __construct($salad){$this->salad=$salad;}public function buildWhereQueryForTags($tags=''){$where='';$data=[];$selected_tags=[];$tagsWhitelist=$this->salad->request->getParam('tags_whitelist');if(!empty($tags) || !empty($tagsWhitelist)){if(!empty($tagsWhitelist)){$tags .= ' '.$tagsWhitelist;}$tags=str_replace(',',' ',$tags);$tags=str_replace('+',' ',$tags);$tags_arr=array_unique(explode(' ',$tags));$i=0;foreach($tags_arr as $tag){$tag=trim($tag);if(!empty($tag)){$i++;$where .= ' tags LIKE :tag'.$i.' AND ';$data['tag'.$i]='% '.$tag.' %';$selected_tags[]=$tag;}} unset($i,$tag);}$where=rtrim($where,' AND ');return [$where,$data,$selected_tags];}public function buildWhereQueryForSearch($search=''){$where='';$data=[];if(!empty($search)){$search_arr=array_unique(explode(' ',$search));$where .= ' ( ';$search_columns=['url','description','extended'];$i=0;foreach($search_columns as $s_column){$where .= '( ';foreach($search_arr as $s){$i++;$where .= ' '.$s_column.' LIKE :'.$s_column.$i.' AND ';$data[$s_column.$i]='%'.$s.'%';}$where=rtrim($where,' AND ');$where .= ' ) OR ';}$where=rtrim($where,' OR ');$where .= ' ) ';}return [$where,$data];}public function buildWhereQueryForUrl($url){$where='';$data=[];if(!empty($url)){$where .= ' url=:url ';$data['url']=$url;}return [$where,$data];}public function buildWhereQueryForHashes($hashes=''){$where='';$data=[];if(!empty($hashes)){$hashes_arr=array_unique(explode(' ',$hashes));$i=0;foreach($hashes_arr as $hash){$hash=trim($hash);if(!empty($hash)){$i++;$where .= ' hash=:hashes'.$i.' OR ';$data['hashes'.$i]=$hash;}} unset($i,$hash);}$where=rtrim($where,' OR ');return [$where,$data];}public function buildWhereQueryForDate($dt=''){$where='';$data=[];if(!empty($dt)){$dtStart=strtotime(gmdate('d.m.Y 00:00:00',strtotime($dt)));$dtEnd=strtotime(gmdate('d.m.Y 23:59:59',strtotime($dt)));$where .= ' created BETWEEN "'.$dtStart.'" AND "'.$dtEnd.'" ';}return [$where,$data];}} -class Utils{private $salad;public function __construct($salad){$this->salad=$salad;}public function isValidUrl($url){if(!$url){return false;}$path=parse_url($url,PHP_URL_PATH);$encoded_path=array_map('urlencode',explode('/',$path));$url=str_replace($path,implode('/',$encoded_path),$url);return filter_var($url,FILTER_VALIDATE_URL)?true:false;}private function removeJsComments($jsString=''){$jsString=preg_replace('~(?%=&|!])\s*~','$1',$jsString);$jsString=preg_replace('~;+~',';',$jsString);$jsString=preg_replace('~\s+~',' ',$jsString);return $jsString;}private function removeCssComments($cssString=''){$cssString=preg_replace(['/\/\*[\s\S]*?\*\//','/\s*([\{\};:,\>])\s*/','/;}/','/\s+/'],['','$1','}',' '],$cssString);return trim($cssString);}private function removeHtmlComments($htmlString=''){$htmlString=preg_replace('//','',$htmlString);$htmlString=preg_replace('/\s+/',' ',$htmlString);$htmlString=preg_replace('/\s*([<>])\s*/','$1',$htmlString);$htmlString=str_replace(['" />'],['"/>'],$htmlString);return $htmlString;}public function minifyOutput($html){if(!empty($html['body'])){$html['body']=$this->removeHtmlComments($html['body']);}if(!empty($html['css'])){$html['css']=$this->removeCssComments($html['css']);}if(!empty($html['head'])){$html['head']=$this->removeHtmlComments($html['head']);}if(!empty($html['js'])){$html['js']=$this->removeJsComments($html['js']);}return $html;}private array $safeHTMLStrings=[];public function safeForHTML($text=''){if(!empty($this->safeHTMLStrings[$text])){return $this->safeHTMLStrings[$text];}$this->safeHTMLStrings[$text]=htmlspecialchars($text,ENT_QUOTES,'UTF-8');return $this->safeHTMLStrings[$text];}private array $safeJSONPStrings=[];public function safeForJSONP($text=''){if(!empty($this->safeJSONPStrings[$text])){return $this->safeJSONPStrings[$text];}$this->safeJSONPStrings[$text]=preg_replace('/[^a-zA-Z0-9_\.-]/','',$text);return $this->safeJSONPStrings[$text];}public function strEndsWith(string $haystack,string $needle){$needle_len=strlen($needle);return ($needle_len===0 || 0===substr_compare($haystack,$needle,-$needle_len));}public function strStartsWith(string $haystack,string $needle){return strncmp($haystack,$needle,strlen($needle))===0;}private $executionTimes=[];public function startExecTime($key){$this->executionTimes[$key]['start']=microtime(true);}public function endExecTime($key){$executionTime=microtime(true) - $this->executionTimes[$key]['start'];$executionTime=round($executionTime,2);unset($this->executionTimes[$key]);return $executionTime;}public function arrayDiff2D($arr1,$arr2){$arr1Assoc=[];foreach($arr1 as $item){if(isset($item['tag']) && isset($item['count'])){$arr1Assoc[$item['tag']]=$item['count'];}} $arr2Assoc=[];foreach($arr2 as $item){if(isset($item['tag']) && isset($item['count'])){$arr2Assoc[$item['tag']]=$item['count'];}} $diff=[];foreach($arr1Assoc as $tag=>$count){if(!isset($arr2Assoc[$tag])){$diff[]=['tag'=>$tag,'count'=>$count,'status'=>'removed'];}elseif($arr2Assoc[$tag] != $count){$diff[]=['tag'=>$tag,'count'=>$count,'status'=>'modified'];}} foreach($arr2Assoc as $tag=>$count){if(!isset($arr1Assoc[$tag])){$diff[]=['tag'=>$tag,'count'=>$count,'status'=>'added'];}} return $diff;}public function urlDecode($url){$placeholder='__PLUS__';$placeholder2='__ENCODED_AMPERSAND__';$urlWithPlaceholder=str_replace('+',$placeholder,$url);$urlWithPlaceholder=str_replace('%26',$placeholder2,$url);$decodedUrl=urldecode($urlWithPlaceholder);$finalUrl=str_replace($placeholder,'+',$decodedUrl);$finalUrl=str_replace($placeholder2,'%26',$decodedUrl);return $finalUrl;}public function setCacheHeaders($days=1){$daysToCache=($days * 86400);$ts=gmdate("D,d M Y H:i:s",time() + $daysToCache)." GMT";header("Expires:$ts",true);header("Pragma:cache",true);header("Cache-Control:Max-Age=$daysToCache",true);}public function fetch($url,$type){$options=array( 'http'=>array( 'user_agent'=>'Mozilla/5.0 (Windows NT 10.0;Win64;x64) AppleWebKit/537.36 (KHTML,like Gecko) Chrome/124.0.0.0 Safari/537.36' ),"ssl"=>array( "verify_peer"=>false,"verify_peer_name"=>false,),);$context=stream_context_create($options);$content=file_get_contents($url,false,$context);foreach($http_response_header as $c=>$h){if(stristr($h,'content-encoding') and stristr($h,'gzip')){$content=gzinflate(substr($content,10,-8));}} switch($type){case 'xml':$content=str_replace(["\0"," "],[""," "],$content);$simpleXml=simplexml_load_string($content,"SimpleXMLElement",LIBXML_NOCDATA);return $simpleXml;break;case 'text':default:return $content;break;}} } +class Utils{private $salad;public function __construct($salad){$this->salad=$salad;}public function isValidUrl($url){if(!$url){return false;}$path=parse_url($url,PHP_URL_PATH);$encoded_path=array_map('urlencode',explode('/',$path));$url=str_replace($path,implode('/',$encoded_path),$url);return filter_var($url,FILTER_VALIDATE_URL)?true:false;}private function removeJsComments($jsString=''){$jsString=preg_replace('~(?%=&|!])\s*~','$1',$jsString);$jsString=preg_replace('~;+~',';',$jsString);$jsString=preg_replace('~\s+~',' ',$jsString);return $jsString;}private function removeCssComments($cssString=''){$cssString=preg_replace(['/\/\*[\s\S]*?\*\//','/\s*([\{\};:,\>])\s*/','/;}/','/\s+/'],['','$1','}',' '],$cssString);return trim($cssString);}private function removeHtmlComments($htmlString=''){$htmlString=preg_replace('//','',$htmlString);$htmlString=preg_replace('/\s+/',' ',$htmlString);$htmlString=preg_replace('/\s*([<>])\s*/','$1',$htmlString);$htmlString=str_replace(['" />'],['"/>'],$htmlString);return $htmlString;}public function minifyOutput($html){if(!empty($html['body'])){$html['body']=$this->removeHtmlComments($html['body']);}if(!empty($html['css'])){$html['css']=$this->removeCssComments($html['css']);}if(!empty($html['head'])){$html['head']=$this->removeHtmlComments($html['head']);}if(!empty($html['js'])){$html['js']=$this->removeJsComments($html['js']);}return $html;}private array $safeHTMLStrings=[];public function safeForHTML($text=''){if(!empty($this->safeHTMLStrings[$text])){return $this->safeHTMLStrings[$text];}$this->safeHTMLStrings[$text]=htmlspecialchars($text,ENT_QUOTES,'UTF-8');return $this->safeHTMLStrings[$text];}private array $safeJSONPStrings=[];public function safeForJSONP($text=''){if(!empty($this->safeJSONPStrings[$text])){return $this->safeJSONPStrings[$text];}$this->safeJSONPStrings[$text]=preg_replace('/[^a-zA-Z0-9_\.-]/','',$text);return $this->safeJSONPStrings[$text];}public function strEndsWith(string $haystack,string $needle){$needle_len=strlen($needle);return ($needle_len===0 || 0===substr_compare($haystack,$needle,-$needle_len));}public function strStartsWith(string $haystack,string $needle){return strncmp($haystack,$needle,strlen($needle))===0;}private $executionTimes=[];public function startExecTime($key){$this->executionTimes[$key]['start']=microtime(true);}public function endExecTime($key){$executionTime=microtime(true) - $this->executionTimes[$key]['start'];$executionTime=round($executionTime,2);unset($this->executionTimes[$key]);return $executionTime;}public function arrayDiff2D($arr1,$arr2){$arr1Assoc=[];foreach($arr1 as $item){if(isset($item['tag']) && isset($item['count'])){$arr1Assoc[$item['tag']]=$item['count'];}} $arr2Assoc=[];foreach($arr2 as $item){if(isset($item['tag']) && isset($item['count'])){$arr2Assoc[$item['tag']]=$item['count'];}} $diff=[];foreach($arr1Assoc as $tag=>$count){if(!isset($arr2Assoc[$tag])){$diff[]=['tag'=>$tag,'count'=>$count,'status'=>'removed'];}elseif($arr2Assoc[$tag] != $count){$diff[]=['tag'=>$tag,'count'=>$count,'status'=>'modified'];}} foreach($arr2Assoc as $tag=>$count){if(!isset($arr1Assoc[$tag])){$diff[]=['tag'=>$tag,'count'=>$count,'status'=>'added'];}} return $diff;}public function urlDecode($url){$placeholder='__PLUS__';$placeholder2='__ENCODED_AMPERSAND__';$urlWithPlaceholder=str_replace('+',$placeholder,$url);$urlWithPlaceholder=str_replace('%26',$placeholder2,$url);$decodedUrl=urldecode($urlWithPlaceholder);$finalUrl=str_replace($placeholder,'+',$decodedUrl);$finalUrl=str_replace($placeholder2,'%26',$decodedUrl);return $finalUrl;}public function setCacheHeaders($days=1){$secondsToCache=$days * 24 * 60 * 60;header('Pragma:cache');header('Cache-Control:max-age='.$secondsToCache,true);}public function fetch($url,$type){$options=array( 'http'=>array( 'user_agent'=>'Mozilla/5.0 (Windows NT 10.0;Win64;x64) AppleWebKit/537.36 (KHTML,like Gecko) Chrome/124.0.0.0 Safari/537.36' ),"ssl"=>array( "verify_peer"=>false,"verify_peer_name"=>false,),);$context=stream_context_create($options);$content=file_get_contents($url,false,$context);foreach($http_response_header as $c=>$h){if(stristr($h,'content-encoding') and stristr($h,'gzip')){$content=gzinflate(substr($content,10,-8));}} switch($type){case 'xml':$content=str_replace(["\0"," "],[""," "],$content);$simpleXml=simplexml_load_string($content,"SimpleXMLElement",LIBXML_NOCDATA);return $simpleXml;break;case 'text':default:return $content;break;}} } class View{private $salad;public string $nonce='';public array $content=['html'=>['css'=>'','js'=>'','title'=>'','head'=>'','body'=>'']];public \stdClass $data;private string $tpl='';public function __construct($salad){$this->salad=$salad;$this->nonce=$this->salad->response->nonce();}public function render($tpl,$data=[]){if(!empty($data)){$this->data=new \stdClass();foreach($data as $dK=>$dV){$this->data->$dK=$dV;}} $this->tpl=$tpl;unset($data);ob_start();include('./themes/'.$this->salad->config->get('ui_theme','ui1').'/'.$this->tpl.'.php');$this->content['html']['body']=ob_get_clean();return $this->content;}public function addCss($file){if(is_file($file)){$file=file_get_contents($file);}$file=str_replace('$___FULL_BASE_URI___$',$this->salad->request->getFullBaseUri(),$file);$this->content['html']['css'] .= $file;}public function addJs($file){if(is_file($file)){$file=file_get_contents($file);}$file=str_replace('$___FULL_BASE_URI___$',$this->salad->request->getFullBaseUri(),$file);$this->content['html']['js'] .= $file;}public function addHtmlTitle($text){$this->content['html']['title']=$text;}public function addHtmlHead($head){if(is_file($head)){$head=file_get_contents($head);}$head=str_replace('$___FULL_BASE_URI___$',$this->salad->request->getFullBaseUri(),$head);$this->content['html']['head']=$head;}} } namespace Salad\API\Controllers{ diff --git a/wasmer.toml b/wasmer.toml new file mode 100644 index 0000000..ee1097e --- /dev/null +++ b/wasmer.toml @@ -0,0 +1,13 @@ +[dependencies] +"php/php" = "=8.3.4" + +[fs] +"/server" = "server" + +[[command]] +name = "run" +module = "php/php:php" +runner = "wasi" +[command.annotations.wasi] +main-args = ["-t", "/server", "-S", "localhost:8080"] +