312 lines
12 KiB
PHP
Executable File
312 lines
12 KiB
PHP
Executable File
<?php
|
|
/*
|
|
* 2007-2011 PrestaShop
|
|
*
|
|
* NOTICE OF LICENSE
|
|
*
|
|
* This source file is subject to the Academic Free License (AFL 3.0)
|
|
* that is bundled with this package in the file LICENSE.txt.
|
|
* It is also available through the world-wide-web at this URL:
|
|
* http://opensource.org/licenses/afl-3.0.php
|
|
* If you did not receive a copy of the license and are unable to
|
|
* obtain it through the world-wide-web, please send an email
|
|
* to license@prestashop.com so we can send you a copy immediately.
|
|
*
|
|
* DISCLAIMER
|
|
*
|
|
* Do not edit or add to this file if you wish to upgrade PrestaShop to newer
|
|
* versions in the future. If you wish to customize PrestaShop for your
|
|
* needs please refer to http://www.prestashop.com for more information.
|
|
*
|
|
* @author PrestaShop SA <contact@prestashop.com>
|
|
* @copyright 2007-2011 PrestaShop SA
|
|
* @version Release: $Revision: 8963 $
|
|
* @license http://opensource.org/licenses/afl-3.0.php Academic Free License (AFL 3.0)
|
|
* International Registered Trademark & Property of PrestaShop SA
|
|
*/
|
|
|
|
if (!defined('_PS_VERSION_'))
|
|
exit;
|
|
|
|
class Gsitemap extends Module
|
|
{
|
|
private $_html = '';
|
|
private $_postErrors = array();
|
|
|
|
public function __construct()
|
|
{
|
|
$this->name = 'gsitemap';
|
|
$this->tab = 'seo';
|
|
$this->version = '1.7';
|
|
$this->author = 'PrestaShop';
|
|
$this->need_instance = 0;
|
|
|
|
parent::__construct();
|
|
|
|
$this->displayName = $this->l('Google sitemap');
|
|
$this->description = $this->l('Generate your Google sitemap file');
|
|
|
|
if (!defined('GSITEMAP_FILE'))
|
|
define('GSITEMAP_FILE', dirname(__FILE__).'/../../sitemap.xml');
|
|
}
|
|
|
|
public function uninstall()
|
|
{
|
|
file_put_contents(GSITEMAP_FILE, '');
|
|
return parent::uninstall();
|
|
}
|
|
|
|
private function _postValidation()
|
|
{
|
|
file_put_contents(GSITEMAP_FILE, '');
|
|
if (!($fp = fopen(GSITEMAP_FILE, 'w')))
|
|
$this->_postErrors[] = $this->l('Cannot create').' '.realpath(dirname(__FILE__.'/../..')).'/'.$this->l('sitemap.xml file.');
|
|
else
|
|
fclose($fp);
|
|
}
|
|
|
|
private function getUrlWith($url, $key, $value)
|
|
{
|
|
if (empty($value))
|
|
return $url;
|
|
if (strpos($url, '?') !== false)
|
|
return $url.'&'.$key.'='.$value;
|
|
return $url.'?'.$key.'='.$value;
|
|
}
|
|
|
|
private function _postProcess()
|
|
{
|
|
Configuration::updateValue('GSITEMAP_ALL_CMS', (int)Tools::getValue('GSITEMAP_ALL_CMS'));
|
|
Configuration::updateValue('GSITEMAP_ALL_PRODUCTS', (int)Tools::getValue('GSITEMAP_ALL_PRODUCTS'));
|
|
$link = new Link();
|
|
$langs = Language::getLanguages();
|
|
|
|
$xmlString = <<<XML
|
|
<?xml version="1.0" encoding="UTF-8" ?>
|
|
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9"
|
|
xmlns:image="http://www.google.com/schemas/sitemap-image/1.1">
|
|
</urlset>
|
|
XML;
|
|
|
|
$xml = new SimpleXMLElement($xmlString);
|
|
|
|
if (Configuration::get('PS_REWRITING_SETTINGS') AND sizeof($langs) > 1)
|
|
foreach($langs as $lang)
|
|
$this->_addSitemapNode($xml, Tools::getShopDomain(true, true).__PS_BASE_URI__.$lang['iso_code'].'/', '1.00', 'daily', date('Y-m-d'));
|
|
else
|
|
$this->_addSitemapNode($xml, Tools::getShopDomain(true, true).__PS_BASE_URI__, '1.00', 'daily', date('Y-m-d'));
|
|
|
|
/* CMS Generator */
|
|
if (Configuration::get('GSITEMAP_ALL_CMS') OR !Module::isInstalled('blockcms'))
|
|
$sql_cms = '
|
|
SELECT DISTINCT '.(Configuration::get('PS_REWRITING_SETTINGS') ? 'cl.id_cms, cl.link_rewrite, cl.id_lang' : 'cl.id_cms').
|
|
' FROM '._DB_PREFIX_.'cms_lang cl
|
|
LEFT JOIN '._DB_PREFIX_.'lang l ON (cl.id_lang = l.id_lang)
|
|
WHERE l.`active` = 1
|
|
ORDER BY cl.id_cms, cl.id_lang ASC';
|
|
elseif (Module::isInstalled('blockcms'))
|
|
$sql_cms = '
|
|
SELECT DISTINCT '.(Configuration::get('PS_REWRITING_SETTINGS') ? 'cl.id_cms, cl.link_rewrite, cl.id_lang' : 'cl.id_cms').
|
|
' FROM '._DB_PREFIX_.'cms_block_page b
|
|
LEFT JOIN '._DB_PREFIX_.'cms_lang cl ON (b.id_cms = cl.id_cms)
|
|
LEFT JOIN '._DB_PREFIX_.'lang l ON (cl.id_lang = l.id_lang)
|
|
WHERE l.`active` = 1
|
|
ORDER BY cl.id_cms, cl.id_lang ASC';
|
|
|
|
$cmss = Db::getInstance(_PS_USE_SQL_SLAVE_)->ExecuteS($sql_cms);
|
|
foreach($cmss AS $cms)
|
|
{
|
|
$tmpLink = Configuration::get('PS_REWRITING_SETTINGS') ? $link->getCMSLink((int)$cms['id_cms'], $cms['link_rewrite'], false, (int)$cms['id_lang']) : $link->getCMSLink((int)$cms['id_cms']);
|
|
$this->_addSitemapNode($xml, $tmpLink, '0.8', 'daily');
|
|
}
|
|
|
|
/* Categories Generator */
|
|
if (Configuration::get('PS_REWRITING_SETTINGS'))
|
|
$categories = Db::getInstance()->ExecuteS('
|
|
SELECT c.id_category, c.level_depth, link_rewrite, DATE_FORMAT(IF(date_upd,date_upd,date_add), \'%Y-%m-%d\') AS date_upd, cl.id_lang
|
|
FROM '._DB_PREFIX_.'category c
|
|
LEFT JOIN '._DB_PREFIX_.'category_lang cl ON c.id_category = cl.id_category
|
|
LEFT JOIN '._DB_PREFIX_.'lang l ON cl.id_lang = l.id_lang
|
|
WHERE l.`active` = 1 AND c.`active` = 1 AND c.id_category != 1
|
|
ORDER BY cl.id_category, cl.id_lang ASC');
|
|
else
|
|
$categories = Db::getInstance()->ExecuteS(
|
|
'SELECT c.id_category, c.level_depth, DATE_FORMAT(IF(date_upd,date_upd,date_add), \'%Y-%m-%d\') AS date_upd
|
|
FROM '._DB_PREFIX_.'category c
|
|
ORDER BY c.id_category ASC');
|
|
|
|
|
|
foreach($categories as $category)
|
|
{
|
|
if (($priority = 0.9 - ($category['level_depth'] / 10)) < 0.1)
|
|
$priority = 0.1;
|
|
|
|
$tmpLink = Configuration::get('PS_REWRITING_SETTINGS') ? $link->getCategoryLink((int)$category['id_category'], $category['link_rewrite'], (int)$category['id_lang']) : $link->getCategoryLink((int)$category['id_category']);
|
|
$this->_addSitemapNode($xml, htmlspecialchars($tmpLink), $priority, 'weekly', substr($category['date_upd'], 0, 10));
|
|
}
|
|
|
|
$products = Db::getInstance(_PS_USE_SQL_SLAVE_)->ExecuteS('
|
|
SELECT p.id_product, pl.link_rewrite, DATE_FORMAT(IF(date_upd,date_upd,date_add), \'%Y-%m-%d\') date_upd, pl.id_lang, cl.`link_rewrite` category, ean13, i.id_image, il.legend legend_image, (
|
|
SELECT MIN(level_depth)
|
|
FROM '._DB_PREFIX_.'product p2
|
|
LEFT JOIN '._DB_PREFIX_.'category_product cp2 ON p2.id_product = cp2.id_product
|
|
LEFT JOIN '._DB_PREFIX_.'category c2 ON cp2.id_category = c2.id_category
|
|
WHERE p2.id_product = p.id_product AND p2.`active` = 1 AND c2.`active` = 1) AS level_depth
|
|
FROM '._DB_PREFIX_.'product p
|
|
LEFT JOIN '._DB_PREFIX_.'product_lang pl ON (p.id_product = pl.id_product)
|
|
LEFT JOIN `'._DB_PREFIX_.'category_lang` cl ON (p.`id_category_default` = cl.`id_category` AND pl.`id_lang` = cl.`id_lang`)
|
|
LEFT JOIN '._DB_PREFIX_.'image i ON p.id_product = i.id_product
|
|
LEFT JOIN `'._DB_PREFIX_.'image_lang` il ON (i.`id_image` = il.`id_image` AND pl.`id_lang` = il.`id_lang`)
|
|
LEFT JOIN '._DB_PREFIX_.'lang l ON (pl.id_lang = l.id_lang)
|
|
WHERE l.`active` = 1 AND p.`active` = 1
|
|
'.(Configuration::get('GSITEMAP_ALL_PRODUCTS') ? '' : 'HAVING level_depth IS NOT NULL').'
|
|
ORDER BY pl.id_product, pl.id_lang ASC');
|
|
|
|
$tmp = null;
|
|
$res = null;
|
|
foreach($products AS $product)
|
|
{
|
|
if ($tmp == $product['id_product'])
|
|
$res[$tmp]['images'] []= array('id_image' => $product['id_image'], 'legend_image' => $product['legend_image']);
|
|
else
|
|
{
|
|
$tmp = $product['id_product'];
|
|
$res[$tmp] = $product;
|
|
unset($res[$tmp]['id_image'], $res[$tmp]['legend_image']);
|
|
$res[$tmp]['images'] []= array('id_image' => $product['id_image'], 'legend_image' => $product['legend_image']);
|
|
}
|
|
}
|
|
|
|
foreach ($res as $product)
|
|
{
|
|
if (($priority = 0.7 - ($product['level_depth'] / 10)) < 0.1)
|
|
$priority = 0.1;
|
|
|
|
$tmpLink = $link->getProductLink((int)($product['id_product']), $product['link_rewrite'], $product['category'], $product['ean13'], (int)($product['id_lang']));
|
|
$sitemap = $this->_addSitemapNode($xml, htmlspecialchars($tmpLink), $priority, 'weekly', substr($product['date_upd'], 0, 10));
|
|
$sitemap = $this->_addSitemapNodeImage($sitemap, $product);
|
|
}
|
|
|
|
/* Add classic pages (contact, best sales, new products...) */
|
|
$pages = array(
|
|
'authentication' => true,
|
|
'best-sales' => false,
|
|
'contact-form' => true,
|
|
'discount' => false,
|
|
'index' => false,
|
|
'manufacturer' => false,
|
|
'new-products' => false,
|
|
'prices-drop' => false,
|
|
'supplier' => false,
|
|
'store' => false);
|
|
|
|
// Don't show suppliers and manufacturers if they are disallowed
|
|
if (!Module::getInstanceByName('blockmanufacturer')->id && !Configuration::get('PS_DISPLAY_SUPPLIERS'))
|
|
unset($pages['manufacturer']);
|
|
|
|
if (!Module::getInstanceByName('blocksupplier')->id && !Configuration::get('PS_DISPLAY_SUPPLIERS'))
|
|
unset($pages['supplier']);
|
|
|
|
// Generate nodes for pages
|
|
if (Configuration::get('PS_REWRITING_SETTINGS'))
|
|
foreach ($pages AS $page => $ssl)
|
|
foreach($langs as $lang)
|
|
$this->_addSitemapNode($xml, $link->getPageLink($page.'.php', $ssl, $lang['id_lang']), '0.5', 'monthly');
|
|
else
|
|
foreach($pages AS $page => $ssl)
|
|
$this->_addSitemapNode($xml, $link->getPageLink($page.'.php', $ssl), '0.5', 'monthly');
|
|
|
|
$xmlString = $xml->asXML();
|
|
|
|
$fp = fopen(GSITEMAP_FILE, 'w');
|
|
fwrite($fp, $xmlString);
|
|
fclose($fp);
|
|
|
|
$res = file_exists(GSITEMAP_FILE);
|
|
$this->_html .= '<h3 class="'. ($res ? 'conf confirm' : 'alert error') .'" style="margin-bottom: 20px">';
|
|
$this->_html .= $res ? $this->l('Sitemap file generated') : $this->l('Error while creating sitemap file');
|
|
$this->_html .= '</h3>';
|
|
}
|
|
|
|
private function _addSitemapNode($xml, $loc, $priority, $change_freq, $last_mod = NULL)
|
|
{
|
|
$sitemap = $xml->addChild('url');
|
|
$sitemap->addChild('loc', $loc);
|
|
$sitemap->addChild('priority', $priority);
|
|
if ($last_mod)
|
|
$sitemap->addChild('lastmod', $last_mod);
|
|
$sitemap->addChild('changefreq', $change_freq);
|
|
return $sitemap;
|
|
}
|
|
|
|
private function _addSitemapNodeImage($xml, $product)
|
|
{
|
|
foreach ($product['images'] as $img)
|
|
{
|
|
$link = new Link();
|
|
$image = $xml->addChild('image', null, 'http://www.google.com/schemas/sitemap-image/1.1');
|
|
$image->addChild('loc', $link->getImageLink($product['link_rewrite'], (int)$product['id_product'].'-'.(int)$img['id_image']), 'http://www.google.com/schemas/sitemap-image/1.1');
|
|
|
|
$legend_image = preg_replace('/(&+)/i', '&', $img['legend_image']);
|
|
$image->addChild('caption', $legend_image, 'http://www.google.com/schemas/sitemap-image/1.1');
|
|
$image->addChild('title', $legend_image, 'http://www.google.com/schemas/sitemap-image/1.1');
|
|
}
|
|
}
|
|
|
|
private function _displaySitemap()
|
|
{
|
|
if (file_exists(GSITEMAP_FILE) AND filesize(GSITEMAP_FILE))
|
|
{
|
|
$fp = fopen(GSITEMAP_FILE, 'r');
|
|
$fstat = fstat($fp);
|
|
fclose($fp);
|
|
$xml = simplexml_load_file(GSITEMAP_FILE);
|
|
|
|
$nbPages = sizeof($xml->url);
|
|
|
|
$this->_html .= '<p>'.$this->l('Your Google sitemap file is online at the following address:').'<br />
|
|
<a href="'.Tools::getShopDomain(true, true).__PS_BASE_URI__.'sitemap.xml" target="_blank"><b>'.Tools::getShopDomain(true, true).__PS_BASE_URI__.'sitemap.xml</b></a></p><br />';
|
|
|
|
$this->_html .= $this->l('Update:').' <b>'.utf8_encode(strftime('%A %d %B %Y %H:%M:%S',$fstat['mtime'])).'</b><br />';
|
|
$this->_html .= $this->l('Filesize:').' <b>'.number_format(($fstat['size']*.000001), 3).'MB</b><br />';
|
|
$this->_html .= $this->l('Indexed pages:').' <b>'.$nbPages.'</b><br /><br />';
|
|
}
|
|
}
|
|
|
|
private function _displayForm()
|
|
{
|
|
$this->_html .=
|
|
'<form action="'.Tools::htmlentitiesUTF8($_SERVER['REQUEST_URI']).'" method="post">
|
|
<div style="margin:0 0 20px 0;">
|
|
<input type="checkbox" name="GSITEMAP_ALL_PRODUCTS" id="GSITEMAP_ALL_PRODUCTS" style="vertical-align: middle;" value="1" '.(Configuration::get('GSITEMAP_ALL_PRODUCTS') ? 'checked="checked"' : '').' /> <label class="t" for="GSITEMAP_ALL_PRODUCTS">'.$this->l('Sitemap also includes products from inactive categories').'</label>
|
|
</div>
|
|
<div style="margin:0 0 20px 0;">
|
|
<input type="checkbox" name="GSITEMAP_ALL_CMS" id="GSITEMAP_ALL_CMS" style="vertical-align: middle;" value="1" '.(Configuration::get('GSITEMAP_ALL_CMS') ? 'checked="checked"' : '').' /> <label class="t" for="GSITEMAP_ALL_CMS">'.$this->l('Sitemap also includes CMS pages which are not in a CMS block').'</label>
|
|
</div>
|
|
<input name="btnSubmit" class="button" type="submit"
|
|
value="'.((!file_exists(GSITEMAP_FILE)) ? $this->l('Generate sitemap file') : $this->l('Update sitemap file')).'" />
|
|
</form>';
|
|
}
|
|
|
|
public function getContent()
|
|
{
|
|
$this->_html .= '<h2>'.$this->l('Search Engine Optimization').'</h2>
|
|
'.$this->l('See').' <a href="https://www.google.com/webmasters/tools/docs/en/about.html" style="font-weight:bold;text-decoration:underline;" target="_blank">
|
|
'.$this->l('this page').'</a> '.$this->l('for more information').'<br /><br />';
|
|
if (Tools::isSubmit('btnSubmit'))
|
|
{
|
|
$this->_postValidation();
|
|
if (!sizeof($this->_postErrors))
|
|
$this->_postProcess();
|
|
else
|
|
foreach ($this->_postErrors AS $err)
|
|
$this->_html .= '<div class="alert error">'.$err.'</div>';
|
|
}
|
|
|
|
$this->_displaySitemap();
|
|
$this->_displayForm();
|
|
|
|
return $this->_html;
|
|
}
|
|
}
|