* GoogleSitemap::enable_google_notificaton(); * * * @see http://www.google.com/support/webmasters/bin/answer.py?hl=en&answer=34609 * * @package googlesitemaps */ class GoogleSitemap extends Controller { /** * @var boolean */ protected static $enabled = true; /** * @var DataObjectSet */ protected $Pages; /** * @var boolean */ protected static $google_notification_enabled = false; /** * @var boolean */ protected static $use_show_in_search = true; /** * List of DataObjects to show in sitemap.xml * * @var array */ public static $google_sitemap_dataobjects = array(); /** * List of DataObjects change frequency * * @var array */ public static $google_sitemap_dataobjects_changefreq = array(); /** * Decorates the given DataObject with {@link GoogleSitemapDecorator} * and pushes the class name to the registered DataObjects. * Note that all registered DataObjects need the method AbsoluteLink(). * * @param string $className name of DataObject to register * @param string $changeFreq * * @return void */ public static function register_dataobject($className, $changeFreq = 'monthly') { if (!self::is_registered($className)) { Object::add_extension($className, 'GoogleSitemapDecorator'); self::$google_sitemap_dataobjects[] = $className; self::$google_sitemap_dataobjects_changefreq[] = $changeFreq; } } /** * Checks whether the given class name is already registered or not. * * @param string $className Name of DataObject to check * * @return bool */ public static function is_registered($className) { return in_array($className, self::$google_sitemap_dataobjects); } /** * Adds DataObjects to the existing DataObjectSet with pages from the * site tree * * @param DataObjectSet $newPages * * @return DataObjectSet */ protected function addRegisteredDataObjects() { $output = new DataObjectSet(); foreach(self::$google_sitemap_dataobjects as $index => $className) { $dataObjectSet = DataObject::get($className); if($dataObjectSet) { foreach($dataObjectSet as $dataObject) { if($dataObject->canView() && (!isset($dataObject->Priority) || $dataObject->Priority > 0)) { $dataObject->ChangeFreq = self::$google_sitemap_dataobjects_changefreq[$index]; if(!isset($dataObject->Priority)) { $dataObject->Priority = 1.0; } $output->push($dataObject); } } } } return $output; } /** * Returns all the links to {@link SiteTree} pages and * {@link DataObject} urls on the page * * @return DataObjectSet */ public function Items() { $filter = ''; $bt = defined('DB::USE_ANSI_SQL') ? "\"" : "`"; if(self::$use_show_in_search) { $filter = "{$bt}ShowInSearch{$bt} = 1"; } $this->Pages = Versioned::get_by_stage('SiteTree', 'Live', $filter); $newPages = new DataObjectSet(); if($this->Pages) { foreach($this->Pages as $page) { // Only include pages from this host and pages which are not an // instance of ErrorPage. We prefix $_SERVER['HTTP_HOST'] with // 'http://' so that parse_url to help parse_url identify the // host name component; we could use another protocol (like ftp // as the prefix and the code would work the same. $pageHttp = parse_url($page->AbsoluteLink(), PHP_URL_HOST); $hostHttp = parse_url('http://' . $_SERVER['HTTP_HOST'], PHP_URL_HOST); if(($pageHttp == $hostHttp) && !($page instanceof ErrorPage)) { // If the page has been set to 0 priority, we set a flag so // it won't be included if($page->canView() && (!isset($page->Priority) || $page->Priority > 0)) { // The one field that isn't easy to deal with in the template is // Change frequency, so we set that here. $date = date('Y-m-d H:i:s'); $prop = $page->toMap(); $created = new SS_Datetime(); $created->value = (isset($prop['Created'])) ? $prop['Created'] : $date; $now = new SS_Datetime(); $now->value = $date; $versions = (isset($prop['Version'])) ? $prop['Version'] : 1; $timediff = $now->format('U') - $created->format('U'); // Check how many revisions have been made over the lifetime of the // Page for a rough estimate of it's changing frequency. $period = $timediff / ($versions + 1); if($period > 60*60*24*365) { // > 1 year $page->ChangeFreq = 'yearly'; } elseif($period > 60*60*24*30) { $page->ChangeFreq = 'monthly'; } elseif($period > 60*60*24*7) { // > 1 week $page->ChangeFreq = 'weekly'; } elseif($period > 60*60*24) { // > 1 day $page->ChangeFreq = 'daily'; } elseif($period > 60*60) { // > 1 hour $page->ChangeFreq = 'hourly'; } else { // < 1 hour $page->ChangeFreq = 'always'; } $newPages->push($page); } } } } $newPages->merge($this->addRegisteredDataObjects()); return $newPages; } /** * Notifies Google about changes to your sitemap. * Triggered automatically on every publish/unpublish of a page. * This behaviour is disabled by default, enable with: * GoogleSitemap::enable_google_notificaton(); * * If the site is in "dev-mode", no ping will be sent regardless wether * the Google notification is enabled. * * @return string Response text */ static function ping() { if(!self::$enabled) return false; //Don't ping if the site has disabled it, or if the site is in dev mode if(!GoogleSitemap::$google_notification_enabled || Director::isDev()) return; $location = urlencode(Controller::join_links( Director::absoluteBaseURL(), 'sitemap.xml' )); $response = HTTP::sendRequest( "www.google.com", "/webmasters/sitemaps/ping", sprintf("sitemap=%s", $location) ); return $response; } /** * Enable pings to google.com whenever sitemap changes. * * @return void */ public static function enable_google_notification() { self::$google_notification_enabled = true; } /** * Disables pings to google when the sitemap changes. * * @return void */ public static function disable_google_notification() { self::$google_notification_enabled = false; } /** * Default controller handler for the sitemap.xml file */ function index($url) { if(self::$enabled) { SSViewer::set_source_file_comments(false); $this->getResponse()->addHeader('Content-Type', 'application/xml; charset="utf-8"'); // But we want to still render. return array(); } else { return new SS_HTTPResponse('Not allowed', 405); } } /** * Enable the sitemap.xml file * * @return void */ public static function enable() { self::$enabled = true; } /** * Disable the sitemap.xml file * * @return void */ public static function disable() { self::$enabled = false; } }