* Copyright (C) 2024 Frédéric France * Copyright (C) 2024 MDW * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ /** * \file htdocs/core/lib/website.lib.php * \ingroup website * \brief Library for website module */ /** * Remove PHP code part from a string. * * @param string $str String to clean * @param string $replacewith String to use as replacement * @return string Result string without php code * @see dolKeepOnlyPhpCode() */ function dolStripPhpCode($str, $replacewith = '') { $str = str_replace('', $part); if (!empty($partlings)) { //$phppart = $partlings[0]; //remove content before closing tag if (count($partlings) > 1) { $partlings[0] = ''; // Todo why a count > 1 and not >= 1 ? } //append to out string //$newstr .= ''.$replacewith.''.implode('', $partlings); //$newstr .= ''.$replacewith.''.implode('', $partlings); $newstr .= ''.$replacewith.''.implode('', $partlings); //$newstr .= $replacewith.implode('', $partlings); } } } return $newstr; } /** * Keep only PHP code part from a HTML string page. * * @param string $str String to clean * @return string Result string with php code only * @see dolStripPhpCode(), checkPHPCode() */ function dolKeepOnlyPhpCode($str) { $str = str_replace('', $part, 2); if (!empty($partlings)) { $newstr .= $partlings[0].'?>'; } else { $newstr .= $part.'?>'; } } } return $newstr; } /** * Convert a page content to have correct links (based on DOL_URL_ROOT) into an html content. It replaces also dynamic content with '...php...' * Used to output the page on the Preview from backoffice. * * @param Website $website Web site object * @param string $content Content to replace * @param int $removephppart 0=Replace PHP sections with a PHP badge. 1=Remove completely PHP sections. * @param string $contenttype Content type * @param int $containerid Contenair id * @return string html content * @see dolWebsiteOutput() for function used to replace content in a web server context */ function dolWebsiteReplacementOfLinks($website, $content, $removephppart = 0, $contenttype = 'html', $containerid = 0) { $nbrep = 0; dol_syslog('dolWebsiteReplacementOfLinks start (contenttype='.$contenttype." containerid=".$containerid." USEDOLIBARREDITOR=".(defined('USEDOLIBARREDITOR') ? '1' : '')." USEDOLIBARRSERVER=".(defined('USEDOLIBARRSERVER') ? '1' : '').')', LOG_DEBUG); //if ($contenttype == 'html') { print $content;exit; } // Replace php code. Note $content may come from database and does not contain body tags. $replacewith = '...php...'; if ($removephppart) { $replacewith = ''; } $content = preg_replace('/value="<\?php((?!\?>).)*\?>\n*/ims', 'value="'.$replacewith.'"', $content); $replacewith = '"callto=#'; if ($removephppart) { $replacewith = ''; } $content = preg_replace('/"callto:<\?php((?!\?>).)*\?>\n*/ims', $replacewith, $content); $replacewith = '"mailto=#'; if ($removephppart) { $replacewith = ''; } $content = preg_replace('/"mailto:<\?php((?!\?>).)*\?>\n*/ims', $replacewith, $content); $replacewith = 'src="php'; if ($removephppart) { $replacewith = ''; } $content = preg_replace('/src="<\?php((?!\?>).)*\?>\n*/ims', $replacewith, $content); $replacewith = 'href="php'; if ($removephppart) { $replacewith = ''; } $content = preg_replace('/href="<\?php((?!\?>).)*\?>\n*/ims', $replacewith, $content); //$replacewith='...php...'; $replacewith = '...php...'; if ($removephppart) { $replacewith = ''; } //$content = preg_replace('/<\?php((?!\?toremove>).)*\?toremove>\n*/ims', $replacewith, $content); /*if ($content === null) { if (preg_last_error() == PREG_JIT_STACKLIMIT_ERROR) $content = 'preg_replace error (when removing php tags) PREG_JIT_STACKLIMIT_ERROR'; }*/ $content = dolStripPhpCode($content, $replacewith); // Protect the link styles.css.php to any replacement that we make after. $content = str_replace('href="styles.css.php', 'href="!~!~!~styles.css.php', $content); $content = str_replace('src="javascript.js.php', 'src="!~!~!~javascript.js.php', $content); $content = str_replace('href="http', 'href="!~!~!~http', $content); $content = str_replace('xlink:href="', 'xlink:href="!~!~!~', $content); $content = str_replace('href="//', 'href="!~!~!~//', $content); $content = str_replace('src="//', 'src="!~!~!~//', $content); $content = str_replace('src="viewimage.php', 'src="!~!~!~/viewimage.php', $content); $content = str_replace('src="/viewimage.php', 'src="!~!~!~/viewimage.php', $content); $content = str_replace('src="'.DOL_URL_ROOT.'/viewimage.php', 'src="!~!~!~'.DOL_URL_ROOT.'/viewimage.php', $content); $content = str_replace('href="document.php', 'href="!~!~!~/document.php', $content); $content = str_replace('href="/document.php', 'href="!~!~!~/document.php', $content); $content = str_replace('href="'.DOL_URL_ROOT.'/document.php', 'href="!~!~!~'.DOL_URL_ROOT.'/document.php', $content); // Replace relative link '/' with dolibarr URL $content = preg_replace('/(href=")\/(#[^\"<>]*)?\"/', '\1!~!~!~'.DOL_URL_ROOT.'/website/index.php?website='.$website->ref.'&pageid='.$website->fk_default_home.'\2"', $content, -1, $nbrep); // Replace relative link /xxx.php#aaa or /xxx.php with dolibarr URL (we discard param ?...) $content = preg_replace('/(href=")\/?([^:\"\!]*)\.php(#[^\"<>]*)?\"/', '\1!~!~!~'.DOL_URL_ROOT.'/website/index.php?website='.$website->ref.'&pageref=\2\3"', $content, -1, $nbrep); // Replace relative link /xxx.php?a=b&c=d#aaa or /xxx.php?a=b&c=d with dolibarr URL $content = preg_replace('/(href=")\/?([^:\"\!]*)\.php\?([^#\"<>]*)(#[^\"<>]*)?\"/', '\1!~!~!~'.DOL_URL_ROOT.'/website/index.php?website='.$website->ref.'&pageref=\2&\3\4"', $content, -1, $nbrep); // Fix relative link into medias with correct URL after the DOL_URL_ROOT: ../url("medias/ $content = preg_replace('/url\((["\']?)\/?medias\//', 'url(\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=', $content, -1, $nbrep); $content = preg_replace('/data-slide-bg=(["\']?)\/?medias\//', 'data-slide-bg=\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=', $content, -1, $nbrep); // ]*src=")\/?medias\//', '\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=', $content, -1, $nbrep); // ]*src=")\/?([^:\"\!]+)\"/', '\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=\2"', $content, -1, $nbrep); // ]*src=")(\/?viewimage\.php)/', '\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php', $content, -1, $nbrep); // action="newpage.php" => action="dolibarr/website/index.php?website=...&pageref=newpage $content = preg_replace('/(action=")\/?([^:\"]*)(\.php\")/', '\1!~!~!~'.DOL_URL_ROOT.'/website/index.php?website='.$website->ref.'&pageref=\2"', $content, -1, $nbrep); // Fix relative link /document.php with correct URL after the DOL_URL_ROOT: ...href="/document.php?modulepart=" $content = preg_replace('/(href=")(\/?document\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); $content = preg_replace('/(src=")(\/?document\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); // Fix relative link /viewimage.php with correct URL after the DOL_URL_ROOT: ...href="/viewimage.php?modulepart=" $content = preg_replace('/(url\(")(\/?viewimage\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); // Fix relative URL $content = str_replace('src="!~!~!~/viewimage.php', 'src="!~!~!~'.DOL_URL_ROOT.'/viewimage.php', $content); $content = str_replace('href="!~!~!~/document.php', 'href="!~!~!~'.DOL_URL_ROOT.'/document.php', $content); // Remove the protection tag !~!~!~ $content = str_replace('!~!~!~', '', $content); dol_syslog('dolWebsiteReplacementOfLinks end', LOG_DEBUG); //if ($contenttype == 'html') { print $content;exit; } return $content; } /** * Converts smiley string into the utf8 sequence. * @param string $content Content to replace * @return string Replacement of all smiley strings with their utf8 code * @see dolWebsiteOutput() */ function dolReplaceSmileyCodeWithUTF8($content) { $map = array( ":face_with_tears_of_joy:" => "\xF0\x9F\x98\x82", ":grinning_face_with_smiling_eyes:" => "\xF0\x9F\x98\x81", ":smiling_face_with_open_mouth:" => "\xF0\x9F\x98\x83", ":smiling_face_with_open_mouth_and_cold_sweat:" => "\xF0\x9F\x98\x85", ":smiling_face_with_open_mouth_and_tightly_closed_eyes:" => "\xF0\x9F\x98\x86", ":winking_face:" => "\xF0\x9F\x98\x89", ":smiling_face_with_smiling_eyes:" => "\xF0\x9F\x98\x8A", ":face_savouring_delicious_food:" => "\xF0\x9F\x98\x8B", ":relieved_face:" => "\xF0\x9F\x98\x8C", ":smiling_face_with_heart_shaped_eyes:" => "\xF0\x9F\x98\x8D", ":smiling_face_with_sunglasses:" => "\xF0\x9F\x98\x8E", ":smirking_face:" => "\xF0\x9F\x98\x8F", ":neutral_face:" => "\xF0\x9F\x98\x90", ":expressionless_face:" => "\xF0\x9F\x98\x91", ":unamused_face:" => "\xF0\x9F\x98\x92", ":face_with_cold_sweat:" => "\xF0\x9F\x98\x93", ":pensive_face:" => "\xF0\x9F\x98\x94", ":confused_face:" => "\xF0\x9F\x98\x95", ":confounded_face:" => "\xF0\x9F\x98\x96", ":kissing_face:" => "\xF0\x9F\x98\x97", ":face_throwing_a_kiss:" => "\xF0\x9F\x98\x98", ":kissing_face_with_smiling_eyes:" => "\xF0\x9F\x98\x99", ":kissing_face_with_closed_eyes:" => "\xF0\x9F\x98\x9A", ":face_with_stuck_out_tongue:" => "\xF0\x9F\x98\x9B", ":face_with_stuck_out_tongue_and_winking_eye:" => "\xF0\x9F\x98\x9C", ":face_with_stuck_out_tongue_and_tightly_closed_eyes:" => "\xF0\x9F\x98\x9D", ":disappointed_face:" => "\xF0\x9F\x98\x9E", ":worried_face:" => "\xF0\x9F\x98\x9F", ":angry_face:" => "\xF0\x9F\x98\xA0", ":face_with_symbols_on_mouth:" => "\xF0\x9F\x98\xA1", ); foreach ($map as $key => $value) { $content = str_replace($key, $value, $content); } return $content; } /** * Render a string of an HTML content and output it. * Used to output the page when viewed from a server (Dolibarr or Apache). * * @param string $content Content string * @param string $contenttype Content type * @param int $containerid Contenair id * @return void * @see dolWebsiteReplacementOfLinks() for function used to replace content in the backoffice context. */ function dolWebsiteOutput($content, $contenttype = 'html', $containerid = 0) { global $db, $langs, $conf, $user; global $dolibarr_main_url_root, $dolibarr_main_data_root; global $website; global $includehtmlcontentopened; $nbrep = 0; dol_syslog("dolWebsiteOutput start - contenttype=".$contenttype." containerid=".$containerid." USEDOLIBARREDITOR=".(defined('USEDOLIBARREDITOR') ? '1' : '')." USEDOLIBARRSERVER=".(defined('USEDOLIBARRSERVER') ? '1' : '').' includehtmlcontentopened='.$includehtmlcontentopened); //print $containerid.' '.$content; // Define $urlwithroot $urlwithouturlroot = preg_replace('/'.preg_quote(DOL_URL_ROOT, '/').'$/i', '', trim($dolibarr_main_url_root)); $urlwithroot = $urlwithouturlroot.DOL_URL_ROOT; // This is to use external domain name found into config file //$urlwithroot=DOL_MAIN_URL_ROOT; // This is to use same domain name than current if (defined('USEDOLIBARREDITOR')) { // REPLACEMENT OF LINKS When page called from Dolibarr editor // We remove the part of content if ($contenttype == 'html') { $content = preg_replace('/.*<\/head>/ims', '', $content); $content = preg_replace('/^.*]*)*>/ims', '', $content); $content = preg_replace('/<\/body(\s[^>]*)*>.*$/ims', '', $content); } } elseif (defined('USEDOLIBARRSERVER')) { // REPLACEMENT OF LINKS When page called from Dolibarr server $content = str_replace(']*)?\"/', '\1!~!~!~'.DOL_URL_ROOT.'/public/website/index.php?website='.$website->ref.'&pageref=\2\3"', $content, -1, $nbrep); // Replace relative link /xxx.php?a=b&c=d#aaa or /xxx.php?a=b&c=d with dolibarr URL // Warning: we may replace twice if href="..." was inside an include (dolWebsiteOutput called by include and the by final page), that's why // at end we replace the '!~!~!~' only if we are in final parent page. $content = preg_replace('/(href=")\/?([^:\"\!]*)\.php\?([^#\"<>]*)(#[^\"<>]*)?\"/', '\1!~!~!~'.DOL_URL_ROOT.'/public/website/index.php?website='.$website->ref.'&pageref=\2&\3\4"', $content, -1, $nbrep); // Replace occurrence like _service_XXX.php with dolibarr URL $content = preg_replace('/([\'"])_service_([^\'"]+)\.php\1/', '\1!~!~!~' . DOL_URL_ROOT . '/public/website/index.php?website=' . $website->ref . '&pageref=_service_\2\1', $content, -1, $nbrep); // Replace occurrence like _library_XXX.php with dolibarr URL $content = preg_replace('/([\'"])_library_([^\'"]+)\.php\1/', '\1!~!~!~' . DOL_URL_ROOT . '/public/website/index.php?website=' . $website->ref . '&pageref=_library_\2\1', $content, -1, $nbrep); // Replace relative link without .php like /xxx#aaa or /xxx with dolibarr URL: ...href="....php" $content = preg_replace('/(href=")\/?([a-zA-Z0-9\-_#]+)(\"|\?)/', '\1!~!~!~'.DOL_URL_ROOT.'/public/website/index.php?website='.$website->ref.'&pageref=\2\3', $content, -1, $nbrep); // Fix relative link /document.php with correct URL after the DOL_URL_ROOT: href="/document.php?modulepart=" => href="/dolibarr/document.php?modulepart=" $content = preg_replace('/(href=")(\/?document\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); $content = preg_replace('/(src=")(\/?document\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); // Fix relative link /viewimage.php with correct URL after the DOL_URL_ROOT: href="/viewimage.php?modulepart=" => href="/dolibarr/viewimage.php?modulepart=" $content = preg_replace('/(href=")(\/?viewimage\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); $content = preg_replace('/(src=")(\/?viewimage\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); $content = preg_replace('/(url\(")(\/?viewimage\.php\?[^\"]*modulepart=[^\"]*)(\")/', '\1!~!~!~'.DOL_URL_ROOT.'\2\3', $content, -1, $nbrep); // Fix relative link into medias with correct URL after the DOL_URL_ROOT: ../url("medias/ $content = preg_replace('/url\((["\']?)\/?medias\//', 'url(\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=', $content, -1, $nbrep); $content = preg_replace('/data-slide-bg=(["\']?)\/?medias\//', 'data-slide-bg=\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=', $content, -1, $nbrep); // ]*src=")\/?medias\//', '\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=', $content, -1, $nbrep); // ]*src=")\/?([^:\"\!]+)\"/', '\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file=\2"', $content, -1, $nbrep); // ]*src=")(\/?viewimage\.php)/', '\1!~!~!~'.DOL_URL_ROOT.'/viewimage.php', $content, -1, $nbrep); // action="newpage.php" => action="dolibarr/website/index.php?website=...&pageref=newpage $content = preg_replace('/(action=")\/?([^:\"]*)(\.php\")/', '\1!~!~!~'.DOL_URL_ROOT.'/public/website/index.php?website='.$website->ref.'&pageref=\2"', $content, -1, $nbrep); // Fix relative URL $content = str_replace('src="!~!~!~/viewimage.php', 'src="!~!~!~'.DOL_URL_ROOT.'/viewimage.php', $content); $content = str_replace('href="!~!~!~/document.php', 'href="!~!~!~'.DOL_URL_ROOT.'/document.php', $content); // Remove the protection tag !~!~!~, but only if this is the parent page and not an include if (empty($includehtmlcontentopened)) { $content = str_replace('!~!~!~', '', $content); } } else { // REPLACEMENT OF LINKS When page called from virtual host web server $symlinktomediaexists = 1; if ($website->virtualhost) { $content = preg_replace('/^(]*rel="canonical" href=")\//m', '\1'.$website->virtualhost.'/', $content, -1, $nbrep); } //print 'rrrrrrrrr'.$website->virtualhost.$content; // Make a change into HTML code to allow to include images from medias directory correct with direct link for virtual server // // become // if (!$symlinktomediaexists) { // ]*src=")\/?image\//', '\1/wrapper.php?modulepart=medias&file=medias/image/', $content, -1, $nbrep); $content = preg_replace('/(url\(["\']?)\/?image\//', '\1/wrapper.php?modulepart=medias&file=medias/image/', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*document\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2modulepart=medias\3file=\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*href=")[^\"]*document\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2modulepart=medias\3file=\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*href=")[^\"]*viewimage\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2modulepart=medias\3file=\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*viewimage\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2modulepart=medias\3file=\4\5', $content, -1, $nbrep); $content = preg_replace('/(url\(["\']?)[^\)]*viewimage\.php([^\)]*)modulepart=medias([^\)]*)file=([^\)]*)(["\']?\))/', '\1/wrapper.php\2modulepart=medias\3file=\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*href=")[^\"]*viewimage\.php([^\"]*)hashp=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2hashp=\3\4', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*viewimage\.php([^\"]*)hashp=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2hashp=\3\4', $content, -1, $nbrep); $content = preg_replace('/(url\(["\']?)[^\)]*viewimage\.php([^\)]*)hashp=([^\)]*)(["\']?\))/', '\1/wrapper.php\2hashp\3\4', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*viewimage\.php([^\"]*)modulepart=mycompany([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2modulepart=mycompany\3file=\4\5', $content, -1, $nbrep); // If some links to documents or viewimage remains, we replace with wrapper $content = preg_replace('/(]*src=")\/?viewimage\.php/', '\1/wrapper.php', $content, -1, $nbrep); $content = preg_replace('/(]*href=")\/?documents\.php/', '\1/wrapper.php', $content, -1, $nbrep); } else { // ]*src=")\/?image\//', '\1/medias/image/', $content, -1, $nbrep); $content = preg_replace('/(url\(["\']?)\/?image\//', '\1/medias/image/', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*document\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/medias/\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*href=")[^\"]*document\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/medias/\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*href=")[^\"]*viewimage\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/medias/\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*viewimage\.php([^\"]*)modulepart=medias([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/medias/\4\5', $content, -1, $nbrep); $content = preg_replace('/(url\(["\']?)[^\)]*viewimage\.php([^\)]*)modulepart=medias([^\)]*)file=([^\)]*)(["\']?\))/', '\1/medias/\4\5', $content, -1, $nbrep); $content = preg_replace('/(]*href=")[^\"]*viewimage\.php([^\"]*)hashp=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2hashp=\3\4', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*viewimage\.php([^\"]*)hashp=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2hashp=\3\4', $content, -1, $nbrep); $content = preg_replace('/(url\(["\']?)[^\)]*viewimage\.php([^\)]*)hashp=([^\)]*)(["\']?\))/', '\1/wrapper.php\2hashp=\3\4', $content, -1, $nbrep); $content = preg_replace('/(]*src=")[^\"]*viewimage\.php([^\"]*)modulepart=mycompany([^\"]*)file=([^\"]*)("[^>]*>)/', '\1/wrapper.php\2modulepart=mycompany\3file=\4\5', $content, -1, $nbrep); // If some links to documents or viewimage remains, we replace with wrapper $content = preg_replace('/(]*src=")\/?viewimage\.php/', '\1/wrapper.php', $content, -1, $nbrep); $content = preg_replace('/(]*href=")\/?document\.php/', '\1/wrapper.php', $content, -1, $nbrep); } } if (!defined('USEDOLIBARREDITOR')) { $content = str_replace(' contenteditable="true"', ' contenteditable="false"', $content); } if (getDolGlobalString('WEBSITE_ADD_CSS_TO_BODY')) { $content = str_replace('0 if OK */ function dolWebsiteIncrementCounter($websiteid, $websitepagetype, $websitepageid) { if (!getDolGlobalInt('WEBSITE_PERF_DISABLE_COUNTERS')) { //dol_syslog("dolWebsiteIncrementCounter websiteid=".$websiteid." websitepagetype=".$websitepagetype." websitepageid=".$websitepageid); if (in_array($websitepagetype, array('blogpost', 'page'))) { global $db; $tmpnow = dol_getdate(dol_now('gmt'), true, 'gmt'); $sql = "UPDATE ".$db->prefix()."website SET "; $sql .= " pageviews_total = pageviews_total + 1,"; $sql .= " pageviews_month = pageviews_month + 1,"; // if last access was done during previous month, we save pageview_month into pageviews_previous_month $sql .= " pageviews_previous_month = ".$db->ifsql("lastaccess < '".$db->idate(dol_mktime(0, 0, 0, $tmpnow['mon'], 1, $tmpnow['year'], 'gmt', 0), 'gmt')."'", 'pageviews_month', 'pageviews_previous_month').","; $sql .= " lastaccess = '".$db->idate(dol_now('gmt'), 'gmt')."',"; $sql .= " lastpageid = ".((int) $websitepageid); $sql .= " WHERE rowid = ".((int) $websiteid); $resql = $db->query($sql); if (! $resql) { return -1; } } } return 1; } /** * Format img tags to introduce viewimage on img src. * * @param string $content Content string * @return void * @see dolWebsiteOutput() */ /* function dolWebsiteSaveContent($content) { global $db, $langs, $conf, $user; global $dolibarr_main_url_root, $dolibarr_main_data_root; //dol_syslog("dolWebsiteSaveContent start (mode=".(defined('USEDOLIBARRSERVER')?'USEDOLIBARRSERVER':'').')'); // Define $urlwithroot $urlwithouturlroot=preg_replace('/'.preg_quote(DOL_URL_ROOT,'/').'$/i','',trim($dolibarr_main_url_root)); $urlwithroot=$urlwithouturlroot.DOL_URL_ROOT; // This is to use external domain name found into config file //$urlwithroot=DOL_MAIN_URL_ROOT; // This is to use same domain name than current //$content = preg_replace('/(fetch(0, $website->id, '', $containeraliasalt); if ($result > 0) { $containerref = $tmpwebsitepage->pageurl; } else { print "Error, page contains a redirect to the alternative alias '".$containeraliasalt."' that does not exists in web site (".$website->id." / ".$website->ref.")"; exit; } } if (defined('USEDOLIBARREDITOR')) { /*print '
'; print "This page contains dynamic code that make a redirect to '".$containerref."' in your current context. Redirect has been canceled as it is not supported in edition mode."; print '
';*/ $text = "This page contains dynamic code that make a redirect to '".$containerref."' in your current context. Redirect has been canceled as it is not supported in edition mode."; setEventMessages($text, null, 'warnings', 'WEBSITEREDIRECTDISABLED'.$containerref); return; } if (defined('USEDOLIBARRSERVER')) { // When page called from Dolibarr server // Check new container exists if (!$containeraliasalt) { // If containeraliasalt set, we already did the test include_once DOL_DOCUMENT_ROOT.'/website/class/websitepage.class.php'; $tmpwebsitepage = new WebsitePage($db); // @phan-suppress-next-line PhanPluginSuspiciousParamPosition $result = $tmpwebsitepage->fetch(0, $website->id, $containerref); unset($tmpwebsitepage); } if ($result > 0) { $currenturi = $_SERVER["REQUEST_URI"]; // Example: /public/website/index.php?website=mywebsite.com&pageref=mywebsite-home&nocache=1708177483 $regtmp = array(); if (preg_match('/&pageref=([^&]+)/', $currenturi, $regtmp)) { if ($regtmp[0] == $containerref) { print "Error, page with uri '.$currenturi.' try a redirect to the same alias page '".$containerref."' in web site '".$website->ref."'"; exit; } else { $newurl = preg_replace('/&pageref=([^&]+)/', '&pageref='.$containerref, $currenturi); } } else { $newurl = $currenturi.'&pageref='.urlencode($containerref); } } } else { // When page called from virtual host server $newurl = '/'.$containerref.'.php'; $newurl .= (empty($_SERVER["QUERY_STRING"]) ? '' : '?'.$_SERVER["QUERY_STRING"]); } if ($newurl) { if (!empty($parameters)) { $separator = (parse_url($newurl, PHP_URL_QUERY) == null) ? '?' : '&'; $newurl = $newurl . $separator . http_build_query($parameters); } if ($permanent) { header("Status: 301 Moved Permanently", false, 301); } header("Location: ".$newurl); exit; } else { print "Error, page contains a redirect to the alias page '".$containerref."' that does not exists in web site (".$website->id." / ".$website->ref.")"; exit; } } /** * Clean an HTML page to report only content, so we can include it into another page. * It outputs content of file sanitized from html and body part. * * @param string $containerref Path to file to include (must be a page from website root. Example: 'mypage.php' means 'mywebsite/mypage.php') * @return void */ function includeContainer($containerref) { global $conf, $db, $hookmanager, $langs, $mysoc, $user, $website, $websitepage, $weblangs; // Very important. Required to have var available when running included containers. global $includehtmlcontentopened; global $websitekey, $websitepagefile; $MAXLEVEL = 20; if (!preg_match('/\.php$/i', $containerref)) { $containerref .= '.php'; } $fullpathfile = DOL_DATA_ROOT.($conf->entity > 1 ? '/'.$conf->entity : '').'/website/'.$websitekey.'/'.$containerref; if (empty($includehtmlcontentopened)) { $includehtmlcontentopened = 0; } $includehtmlcontentopened++; if ($includehtmlcontentopened > $MAXLEVEL) { print 'ERROR: RECURSIVE CONTENT LEVEL. Depth of recursive call is more than the limit of '.((int) $MAXLEVEL).".\n"; return; } //dol_syslog("Include container ".$containerref.' includehtmlcontentopened='.$includehtmlcontentopened); // file_get_contents is not possible. We must execute code with include //$content = file_get_contents($fullpathfile); //print preg_replace(array('/^.*]*>/ims','/<\/body>.*$/ims'), array('', ''), $content);*/ ob_start(); $res = @include $fullpathfile; // Include because we want to execute code content $tmpoutput = ob_get_contents(); ob_end_clean(); // We don't print info messages for pages of type library or service if (!empty($websitepage->type_container) && !in_array($websitepage->type_container, array('library', 'service'))) { print "\n".''."\n"; } print preg_replace(array('/^.*]*>/ims', '/<\/body>.*$/ims'), array('', ''), $tmpoutput); if (!$res) { print 'ERROR: FAILED TO INCLUDE PAGE '.$containerref.".\n"; } $includehtmlcontentopened--; } /** * Return HTML content to add structured data for an article, news or Blog Post. Use the json-ld format. * Example: * * 'Name', 'os'=>'Windows', 'price'=>10)); ?> * * @param string $type 'blogpost', 'product', 'software', 'organization', 'qa', ... * @param array $data Array of data parameters for structured data * @return string HTML content */ function getStructuredData($type, $data = array()) { global $conf, $db, $hookmanager, $langs, $mysoc, $user, $website, $websitepage, $weblangs, $pagelangs; // Very important. Required to have var available when running included containers. $type = strtolower($type); if ($type == 'software') { $ret = ''."\n"; $ret .= ''."\n"; } elseif ($type == 'organization') { $companyname = $mysoc->name; $url = $mysoc->url; $ret = ''."\n"; $ret .= ''."\n"; } elseif ($type == 'blogpost') { if (!empty($websitepage->author_alias)) { //include_once DOL_DOCUMENT_ROOT.'/user/class/user.class.php'; //$tmpuser = new User($db); //$restmpuser = $tmpuser->fetch($websitepage->fk_user_creat); $pageurl = $websitepage->pageurl; $title = $websitepage->title; $image = $websitepage->image; $companyname = $mysoc->name; $description = $websitepage->description; $pageurl = str_replace('__WEBSITE_KEY__', $website->ref, $pageurl); $title = str_replace('__WEBSITE_KEY__', $website->ref, $title); $image = '/medias'.(preg_match('/^\//', $image) ? '' : '/').str_replace('__WEBSITE_KEY__', $website->ref, $image); $companyname = str_replace('__WEBSITE_KEY__', $website->ref, $companyname); $description = str_replace('__WEBSITE_KEY__', $website->ref, $description); $ret = ''."\n"; $ret .= ''."\n"; } else { $ret = ''."\n"; } } elseif ($type == 'product') { $ret = ''."\n"; $ret .= ''."\n"; } elseif ($type == 'qa') { $ret = ''."\n"; $ret .= ''."\n"; } else { $ret = ''; } return $ret; } /** * Return HTML content to add as header card for an article, news or Blog Post or home page. * * @param array $params Array of parameters * @return string HTML content */ function getSocialNetworkHeaderCards($params = null) { global $conf, $db, $hookmanager, $langs, $mysoc, $user, $website, $websitepage, $weblangs; // Very important. Required to have var available when running included containers. $out = ''; if ($website->virtualhost) { $pageurl = $websitepage->pageurl; $title = $websitepage->title; $image = $websitepage->image; $companyname = $mysoc->name; $description = $websitepage->description; $pageurl = str_replace('__WEBSITE_KEY__', $website->ref, $pageurl); $title = str_replace('__WEBSITE_KEY__', $website->ref, $title); $image = '/medias'.(preg_match('/^\//', $image) ? '' : '/').str_replace('__WEBSITE_KEY__', $website->ref, $image); $companyname = str_replace('__WEBSITE_KEY__', $website->ref, $companyname); $description = str_replace('__WEBSITE_KEY__', $website->ref, $description); $shortlangcode = ''; if ($websitepage->lang) { $shortlangcode = substr($websitepage->lang, 0, 2); // en_US or en-US -> en } if (empty($shortlangcode)) { $shortlangcode = substr($website->lang, 0, 2); // en_US or en-US -> en } $fullurl = $website->virtualhost.'/'.$websitepage->pageurl.'.php'; $canonicalurl = $website->virtualhost.(($websitepage->id == $website->fk_default_home) ? '/' : (($shortlangcode != substr($website->lang, 0, 2) ? '/'.$shortlangcode : '').'/'.$websitepage->pageurl.'.php')); $hashtags = trim(implode(' #', array_map('trim', explode(',', $websitepage->keywords)))); // Open Graph $out .= ''."\n"; // TODO If blogpost, use type article $out .= ''."\n"; if ($websitepage->image) { $out .= ''."\n"; } $out .= ''."\n"; // Twitter $out .= ''."\n"; if (!empty($params) && !empty($params['twitter_account'])) { $out .= ''."\n"; $out .= ''."\n"; } $out .= ''."\n"; if ($websitepage->description) { $out .= ''."\n"; } if ($websitepage->image) { $out .= ''."\n"; } //$out .= ''; /* $out .= ''; $out .= ''; $out .= ''; $out .= ''; $out .= ''; $out .= ''; $out .= ''; $out .= ''; $out .= ''; */ } return $out; } /** * Return HTML content to add structured data for an article, news or Blog Post. * * @param string $socialnetworks '' or list of social networks * @return string HTML content */ function getSocialNetworkSharingLinks($socialnetworks = '') { global $website, $websitepage; // Very important. Required to have var available when running included containers. $out = ''."\n"; if ($website->virtualhost) { $fullurl = $website->virtualhost.'/'.$websitepage->pageurl.'.php'; $hashtags = trim(implode(' #', array_map('trim', explode(',', $websitepage->keywords)))); $out .= '\n"; } else { $out .= ''."\n"; } $out .= ''."\n"; return $out; } /** * Return HTML content to add structured data for an article, news or Blog Post. * * @param Object $object Object * @return int HTML img content or '' if no image found * @see getImagePublicURLOfObject() */ function getNbOfImagePublicURLOfObject($object) { global $db; $nb = 0; include_once DOL_DOCUMENT_ROOT.'/core/lib/images.lib.php'; $regexforimg = getListOfPossibleImageExt(0); $regexforimg = '('.$regexforimg.')$'; $sql = "SELECT COUNT(rowid) as nb"; $sql .= " FROM ".MAIN_DB_PREFIX."ecm_files"; $sql .= " WHERE entity IN (".getEntity($object->element).")"; $sql .= " AND src_object_type = '".$db->escape($object->element)."' AND src_object_id = ".((int) $object->id); // Filter on object $sql .= " AND ".$db->regexpsql('filename', $regexforimg, 1); $sql .= " AND share IS NOT NULL"; // Only image that are public $resql = $db->query($sql); if ($resql) { $obj = $db->fetch_object($resql); if ($obj) { $nb = $obj->nb; } } return $nb; } /** * Return HTML content to add structured data for an article, news or Blog Post. * * @param Object $object Object * @param int $no Numero of image (if there is several images. 1st one by default) * @param string $extName Extension to differentiate thumb file name ('', '_small', '_mini') * @return string HTML img content or '' if no image found * @see getNbOfImagePublicURLOfObject() */ function getImagePublicURLOfObject($object, $no = 1, $extName = '') { global $db; $image_path = ''; include_once DOL_DOCUMENT_ROOT.'/core/lib/images.lib.php'; $regexforimg = getListOfPossibleImageExt(0); $regexforimg = '('.$regexforimg.')$'; $sql = "SELECT rowid, ref, share, filename, cover, position"; $sql .= " FROM ".MAIN_DB_PREFIX."ecm_files"; $sql .= " WHERE entity IN (".getEntity($object->element).")"; $sql .= " AND src_object_type = '".$db->escape($object->element)."' AND src_object_id = ".((int) $object->id); // Filter on object $sql .= " AND ".$db->regexpsql('filename', $regexforimg, 1); $sql .= $db->order("cover,position,rowid", "ASC,ASC,ASC"); $resql = $db->query($sql); if ($resql) { $num = $db->num_rows($resql); $i = 0; $found = 0; $foundnotshared = 0; while ($i < $num) { $obj = $db->fetch_object($resql); if ($obj) { if (empty($obj->share)) { $foundnotshared++; } else { $found++; if (defined('USEDOLIBARRSERVER') || defined('USEDOLIBARREDITOR')) { $image_path = DOL_URL_ROOT.'/viewimage.php?hashp='.urlencode($obj->share); } else { $image_path = '/wrapper.php?hashp='.urlencode($obj->share); } if ($extName) { $image_path .= '&extname='.urlencode($extName); } } } if ($found >= $no) { break; } $i++; } if (!$found && $foundnotshared) { if (defined('USEDOLIBARRSERVER') || defined('USEDOLIBARREDITOR')) { $image_path = DOL_URL_ROOT.'/viewimage.php?modulepart=common&file=nophotopublic.png'; } else { $image_path = '/wrapper.php?modulepart=common&file=nophotopublic.png'; } } } if (empty($image_path)) { if (defined('USEDOLIBARRSERVER') || defined('USEDOLIBARREDITOR')) { $image_path = DOL_URL_ROOT.'/viewimage.php?modulepart=common&file=nophoto.png'; } else { $image_path = '/wrapper.php?modulepart=common&file=nophoto.png'; } } return $image_path; } /** * Return list of public files of a given object. * * @param Object $object Object * @return array List of public files of object */ function getPublicFilesOfObject($object) { global $db; $files = array(); include_once DOL_DOCUMENT_ROOT.'/core/lib/images.lib.php'; $regexforimg = getListOfPossibleImageExt(0); $regexforimg = '/('.$regexforimg.')$/i'; $sql = "SELECT rowid, ref, share, filename, cover, position"; $sql .= " FROM ".MAIN_DB_PREFIX."ecm_files"; $sql .= " WHERE entity IN (".getEntity($object->element).")"; $sql .= " AND src_object_type = '".$db->escape($object->element)."' AND src_object_id = ".((int) $object->id); $sql .= $db->order("cover,position,rowid", "ASC,ASC,ASC"); $resql = $db->query($sql); if ($resql) { $num = $db->num_rows($resql); $i = 0; while ($i < $num) { $obj = $db->fetch_object($resql); if ($obj) { if (!empty($obj->share)) { $files[$obj->rowid]['filename'] = $obj->filename; $files[$obj->rowid]['position'] = $obj->position; if (defined('USEDOLIBARRSERVER') || defined('USEDOLIBARREDITOR')) { if (preg_match($regexforimg, $obj->filename)) { $files[$obj->rowid]['url'] = DOL_URL_ROOT.'/viewimage.php?hashp='.urlencode($obj->share); } else { $files[$obj->rowid]['url'] = DOL_URL_ROOT.'/document.php?hashp='.urlencode($obj->share); } } else { $files[$obj->rowid]['url'] = '/wrapper.php?hashp='.urlencode($obj->share); } } } $i++; } } return $files; } /** * Return list of containers object that match a criteria. * WARNING: This function can be used by websites. * * @param string $type Type of container to search into (Example: '', 'page', 'blogpost', 'page,blogpost', ...) * @param string $algo Algorithm used for search (Example: 'meta' is searching into meta information like title and description, 'content', 'sitefiles', or any combination 'meta,content,sitefiles') * @param string $searchstring Search string * @param int $max Max number of answers * @param string $sortfield Sort Fields * @param string $sortorder Sort order ('DESC' or 'ASC') * @param string $langcode Language code ('' or 'en', 'fr', 'es', ...) * @param array $otherfilters Other filters * @param int $status 0 or 1, or -1 for both * @return array Array with results of search */ function getPagesFromSearchCriterias($type, $algo, $searchstring, $max = 25, $sortfield = 'date_creation', $sortorder = 'DESC', $langcode = '', $otherfilters = [], $status = 1) { global $conf, $db, $hookmanager, $langs, $mysoc, $user, $website, $websitepage, $weblangs; // Very important. Required to have var available when running included containers. $error = 0; $arrayresult = array('code' => '', 'list' => array()); // Clean parameters if (!is_object($weblangs)) { $weblangs = $langs; } if (empty($algo)) { $algo = 'content'; } /* if (empty($searchstring) && empty($type) && empty($langcode) && empty($otherfilters)) { $error++; $arrayresult['code'] = 'KO'; $arrayresult['message'] = $weblangs->trans("EmptySearchString"); } elseif ($searchstring && dol_strlen($searchstring) < 2) { $weblangs->load("errors"); $error++; $arrayresult['code'] = 'KO'; $arrayresult['message'] = $weblangs->trans("ErrorSearchCriteriaTooSmall"); } else { */ $tmparrayoftype = explode(',', $type); /*foreach ($tmparrayoftype as $tmptype) { if (!in_array($tmptype, array('', 'page', 'blogpost'))) { $error++; $arrayresult['code'] = 'KO'; $arrayresult['message'] = 'Bad value for parameter type'; break; } }*/ //} $searchdone = 0; $found = 0; if (!$error && (empty($max) || ($found < $max)) && (preg_match('/meta/', $algo) || preg_match('/content/', $algo))) { include_once DOL_DOCUMENT_ROOT.'/website/class/websitepage.class.php'; $sql = 'SELECT wp.rowid FROM '.MAIN_DB_PREFIX.'website_page as wp'; if (is_array($otherfilters) && !empty($otherfilters['category'])) { $sql .= ', '.MAIN_DB_PREFIX.'categorie_website_page as cwp'; } $sql .= " WHERE wp.fk_website = ".((int) $website->id); if ($status >= 0) { $sql .= " AND wp.status = ".((int) $status); } if ($langcode) { $sql .= " AND wp.lang = '".$db->escape($langcode)."'"; } if ($type) { $tmparrayoftype = explode(',', $type); $typestring = ''; foreach ($tmparrayoftype as $tmptype) { $typestring .= ($typestring ? ", " : "")."'".$db->escape(trim($tmptype))."'"; } $sql .= " AND wp.type_container IN (".$db->sanitize($typestring, 1).")"; } $sql .= " AND ("; $searchalgo = ''; if (preg_match('/meta/', $algo)) { $searchalgo .= ($searchalgo ? ' OR ' : '')."wp.title LIKE '%".$db->escape($db->escapeforlike($searchstring))."%' OR wp.description LIKE '%".$db->escape($db->escapeforlike($searchstring))."%'"; $searchalgo .= ($searchalgo ? ' OR ' : '')."wp.keywords LIKE '".$db->escape($db->escapeforlike($searchstring)).",%' OR wp.keywords LIKE '% ".$db->escape($db->escapeforlike($searchstring))."%'"; // TODO Use a better way to scan keywords } if (preg_match('/content/', $algo)) { $searchalgo .= ($searchalgo ? ' OR ' : '')."wp.content LIKE '%".$db->escape($db->escapeforlike($searchstring))."%'"; } $sql .= $searchalgo; if (is_array($otherfilters) && !empty($otherfilters['category'])) { $sql .= ' AND cwp.fk_website_page = wp.rowid AND cwp.fk_categorie = '.((int) $otherfilters['category']); } $sql .= ")"; $sql .= $db->order($sortfield, $sortorder); $sql .= $db->plimit($max); //print $sql; $resql = $db->query($sql); if ($resql) { $i = 0; while (($obj = $db->fetch_object($resql)) && ($i < $max || $max == 0)) { if ($obj->rowid > 0) { $tmpwebsitepage = new WebsitePage($db); $tmpwebsitepage->fetch($obj->rowid); if ($tmpwebsitepage->id > 0) { $arrayresult['list'][$obj->rowid] = $tmpwebsitepage; } $found++; } $i++; } } else { $error++; $arrayresult['code'] = $db->lasterrno(); $arrayresult['message'] = $db->lasterror(); } $searchdone = 1; } if (!$error && (empty($max) || ($found < $max)) && (preg_match('/sitefiles/', $algo))) { global $dolibarr_main_data_root; $pathofwebsite = $dolibarr_main_data_root.($conf->entity > 1 ? '/'.$conf->entity : '').'/website/'.$website->ref; $filehtmlheader = $pathofwebsite.'/htmlheader.html'; $filecss = $pathofwebsite.'/styles.css.php'; $filejs = $pathofwebsite.'/javascript.js.php'; $filerobot = $pathofwebsite.'/robots.txt'; $filehtaccess = $pathofwebsite.'/.htaccess'; $filemanifestjson = $pathofwebsite.'/manifest.json.php'; $filereadme = $pathofwebsite.'/README.md'; $filecontent = file_get_contents($filehtmlheader); if ((empty($max) || ($found < $max)) && preg_match('/'.preg_quote($searchstring, '/').'/', $filecontent)) { $arrayresult['list'][] = array('type' => 'website_htmlheadercontent'); } $filecontent = file_get_contents($filecss); if ((empty($max) || ($found < $max)) && preg_match('/'.preg_quote($searchstring, '/').'/', $filecontent)) { $arrayresult['list'][] = array('type' => 'website_csscontent'); } $filecontent = file_get_contents($filejs); if ((empty($max) || ($found < $max)) && preg_match('/'.preg_quote($searchstring, '/').'/', $filecontent)) { $arrayresult['list'][] = array('type' => 'website_jscontent'); } $filerobot = file_get_contents($filerobot); if ((empty($max) || ($found < $max)) && preg_match('/'.preg_quote($searchstring, '/').'/', $filecontent)) { $arrayresult['list'][] = array('type' => 'website_robotcontent'); } $searchdone = 1; } if (!$error) { if ($searchdone) { $arrayresult['code'] = 'OK'; if (empty($arrayresult['list'])) { $arrayresult['code'] = 'KO'; $arrayresult['message'] = $weblangs->trans("NoRecordFound"); } } else { $error++; $arrayresult['code'] = 'KO'; $arrayresult['message'] = 'No supported algorithm found'; } } return $arrayresult; } /** * Download all images found into page content $tmp. * If $modifylinks is set, links to images will be replace with a link to viewimage wrapper. * * @param Website $object Object website * @param WebsitePage $objectpage Object website page * @param string $urltograb URL to grab (example: http://www.nltechno.com/ or http://www.nltechno.com/dir1/ or http://www.nltechno.com/dir1/mapage1) * @param string $tmp Content to parse * @param string $action Var $action * @param int $modifylinks 0=Do not modify content, 1=Replace links with a link to viewimage * @param int $grabimages 0=Do not grab images, 1=Grab images * @param string $grabimagesinto 'root' or 'subpage' * @return void */ function getAllImages($object, $objectpage, $urltograb, &$tmp, &$action, $modifylinks = 0, $grabimages = 1, $grabimagesinto = 'subpage') { global $conf; $error = 0; dol_syslog("Call getAllImages with grabimagesinto=".$grabimagesinto); $alreadygrabbed = array(); if (preg_match('/\/$/', $urltograb)) { $urltograb .= '.'; } $urltograb = dirname($urltograb); // So urltograb is now http://www.nltechno.com or http://www.nltechno.com/dir1 // Search X in "img...src=X" $regs = array(); preg_match_all('/]*)>/i', $tmp, $regs); foreach ($regs[0] as $key => $val) { if (preg_match('/^data:image/i', $regs[2][$key])) { continue; // We do nothing for such images } if (preg_match('/^\//', $regs[2][$key])) { $urltograbdirrootwithoutslash = getRootURLFromURL($urltograb); $urltograbbis = $urltograbdirrootwithoutslash.$regs[2][$key]; // We use dirroot } else { $urltograbbis = $urltograb.'/'.$regs[2][$key]; // We use dir of grabbed file } $linkwithoutdomain = $regs[2][$key]; $dirforimages = '/'.$objectpage->pageurl; if ($grabimagesinto == 'root') { $dirforimages = ''; } // Define $filetosave and $filename $filetosave = $conf->medias->multidir_output[$conf->entity].'/image/'.$object->ref.$dirforimages.(preg_match('/^\//', $regs[2][$key]) ? '' : '/').$regs[2][$key]; if (preg_match('/^http/', $regs[2][$key])) { $urltograbbis = $regs[2][$key]; $linkwithoutdomain = preg_replace('/^https?:\/\/[^\/]+\//i', '', $regs[2][$key]); $filetosave = $conf->medias->multidir_output[$conf->entity].'/image/'.$object->ref.$dirforimages.(preg_match('/^\//', $linkwithoutdomain) ? '' : '/').$linkwithoutdomain; } $filename = 'image/'.$object->ref.$dirforimages.(preg_match('/^\//', $linkwithoutdomain) ? '' : '/').$linkwithoutdomain; // Clean the aa/bb/../cc into aa/cc $filetosave = preg_replace('/\/[^\/]+\/\.\./', '', $filetosave); $filename = preg_replace('/\/[^\/]+\/\.\./', '', $filename); if (empty($alreadygrabbed[$urltograbbis])) { if ($grabimages) { $tmpgeturl = getURLContent($urltograbbis, 'GET', '', 1, array(), array('http', 'https'), 0); if ($tmpgeturl['curl_error_no']) { $error++; setEventMessages('Error getting '.$urltograbbis.': '.$tmpgeturl['curl_error_msg'], null, 'errors'); $action = 'create'; } elseif ($tmpgeturl['http_code'] != '200') { $error++; setEventMessages('Error getting '.$urltograbbis.': '.$tmpgeturl['http_code'], null, 'errors'); $action = 'create'; } else { $alreadygrabbed[$urltograbbis] = 1; // Track that file was already grabbed. dol_mkdir(dirname($filetosave)); $fp = fopen($filetosave, "w"); fwrite($fp, $tmpgeturl['content']); fclose($fp); dolChmod($filetosave); } } } if ($modifylinks) { $tmp = preg_replace('/'.preg_quote($regs[0][$key], '/').'/i', '', $tmp); } } // Search X in "background...url(X)" preg_match_all('/background([^\.\/\(;]+)url\([\"\']?([^\)\"\']*)[\"\']?\)/i', $tmp, $regs); foreach ($regs[0] as $key => $val) { if (preg_match('/^data:image/i', $regs[2][$key])) { continue; // We do nothing for such images } if (preg_match('/^\//', $regs[2][$key])) { $urltograbdirrootwithoutslash = getRootURLFromURL($urltograb); $urltograbbis = $urltograbdirrootwithoutslash.$regs[2][$key]; // We use dirroot } else { $urltograbbis = $urltograb.'/'.$regs[2][$key]; // We use dir of grabbed file } $linkwithoutdomain = $regs[2][$key]; $dirforimages = '/'.$objectpage->pageurl; if ($grabimagesinto == 'root') { $dirforimages = ''; } $filetosave = $conf->medias->multidir_output[$conf->entity].'/image/'.$object->ref.$dirforimages.(preg_match('/^\//', $regs[2][$key]) ? '' : '/').$regs[2][$key]; if (preg_match('/^http/', $regs[2][$key])) { $urltograbbis = $regs[2][$key]; $linkwithoutdomain = preg_replace('/^https?:\/\/[^\/]+\//i', '', $regs[2][$key]); $filetosave = $conf->medias->multidir_output[$conf->entity].'/image/'.$object->ref.$dirforimages.(preg_match('/^\//', $linkwithoutdomain) ? '' : '/').$linkwithoutdomain; } $filename = 'image/'.$object->ref.$dirforimages.(preg_match('/^\//', $linkwithoutdomain) ? '' : '/').$linkwithoutdomain; // Clean the aa/bb/../cc into aa/cc $filetosave = preg_replace('/\/[^\/]+\/\.\./', '', $filetosave); $filename = preg_replace('/\/[^\/]+\/\.\./', '', $filename); if (empty($alreadygrabbed[$urltograbbis])) { if ($grabimages) { $tmpgeturl = getURLContent($urltograbbis, 'GET', '', 1, array(), array('http', 'https'), 0); if ($tmpgeturl['curl_error_no']) { $error++; setEventMessages('Error getting '.$urltograbbis.': '.$tmpgeturl['curl_error_msg'], null, 'errors'); $action = 'create'; } elseif ($tmpgeturl['http_code'] != '200') { $error++; setEventMessages('Error getting '.$urltograbbis.': '.$tmpgeturl['http_code'], null, 'errors'); $action = 'create'; } else { $alreadygrabbed[$urltograbbis] = 1; // Track that file was already grabbed. dol_mkdir(dirname($filetosave)); $fp = fopen($filetosave, "w"); fwrite($fp, $tmpgeturl['content']); fclose($fp); dolChmod($filetosave); } } } if ($modifylinks) { $tmp = preg_replace('/'.preg_quote($regs[0][$key], '/').'/i', 'background'.$regs[1][$key].'url("'.DOL_URL_ROOT.'/viewimage.php?modulepart=medias&file='.$filename.'")', $tmp); } } }