xmlsitemap.pages.inc 5.19 KB
Newer Older
1 2 3 4 5 6 7 8 9
<?php

/**
 * @file
 * Page callbacks for the xmlsitemap module.
 *
 * @ingroup xmlsitemap
 */

10 11 12
/**
 * Get the sitemap chunk/page of the current request.
 */
13
function xmlsitemap_get_current_chunk(stdClass $sitemap) {
14
  // Check if we should be displaing the index.
15
  if (!isset($_GET['page']) || !is_numeric($_GET['page'])) {
16
    if ($sitemap->chunks > 1) {
17
      return 'index';
18 19
    }
    else {
20
      return 1;
21 22 23
    }
  }
  else {
24
    return (int) $_GET['page'];
25
  }
26
}
27

28 29 30
/**
 * Output a sitemap page.
 *
31
 * @see xmlsitemap_sitemap_load_by_context()
32
 * @see xmlsitemap_get_current_chunk()
33
 * @see xmlsitemap_sitemap_get_file()
34 35 36
 * @see xmlsitemap_output_file()
 */
function xmlsitemap_output_chunk() {
37 38 39 40 41 42
  $sitemap = xmlsitemap_sitemap_load_by_context();
  if (!$sitemap) {
    return MENU_NOT_FOUND;
  }

  $chunk = xmlsitemap_get_current_chunk($sitemap);
43
  $file = xmlsitemap_sitemap_get_file($sitemap, $chunk);
44

45
  // Provide debugging information if enabled.
46
  if (variable_get('xmlsitemap_developer_mode', 0) && isset($_GET['debug'])) {
47
    $output = array();
48 49
    $context = xmlsitemap_get_current_context();
    $output[] = "Current context: " . print_r($context, TRUE);
50
    $output[] = "Sitemap: " . print_r($sitemap, TRUE);
51 52 53 54 55 56
    $output[] = "Chunk: $chunk";
    $output[] = "Cache file location: $file";
    $output[] = "Cache file exists: " . (file_exists($file) ? 'Yes' : 'No');
    return implode('<br />', $output);
  }

57
  return xmlsitemap_output_file($file);
58 59 60 61 62 63 64
}

/**
 * Output the contents of a file to the browser and check caching headers.
 */
function xmlsitemap_output_file($file, array $headers = array()) {
  if (!file_exists($file) || !is_readable($file)) {
65
    return MENU_NOT_FOUND;
66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
  }

  $mtime = filemtime($file);
  $last_modified = gmdate(DATE_RFC1123, $mtime);
  $etag = '"' . md5($last_modified) . '"';

  // See if the client has provided the required HTTP headers.
  $if_modified_since = isset($_SERVER['HTTP_IF_MODIFIED_SINCE']) ? stripslashes($_SERVER['HTTP_IF_MODIFIED_SINCE']) : FALSE;
  $if_none_match = isset($_SERVER['HTTP_IF_NONE_MATCH']) ? stripslashes($_SERVER['HTTP_IF_NONE_MATCH']) : FALSE;
  if ($if_modified_since && $if_none_match && $if_none_match == $etag && $if_modified_since == $last_modified) {
    header('HTTP/1.1 304 Not Modified');
    // All 304 responses must send an etag if the 200 response for the same object contained an etag
    header('Etag: ' . $etag);
    exit;
  }

  $headers += array(
83 84 85 86 87 88 89
    'Content-type' => 'text/xml; charset=utf-8',
    //'Content-length' => filesize($file),
    'Last-modified' => $last_modified,
    'Etag' => $etag,
    'Expires' => gmdate(DATE_RFC1123, $mtime + variable_get('xmlsitemap_minimum_lifetime', 0)),
    'Cache-Control' => 'must-revalidate',
    'X-Robots-Tag' => 'noindex, follow',
90 91 92 93 94 95 96 97 98 99 100
  );

  // Transfer the file as output.
  xmlsitemap_file_transfer($file, $headers);
}

/**
 * Modified version of file_transfer() that invokes hook_exit()s afterwards.
 *
 * @see file_transfer()
 */
101
function xmlsitemap_file_transfer($uri, $headers) {
102 103 104 105
  if (ob_get_level()) {
    ob_end_clean();
  }

106 107
  foreach ($headers as $name => $value) {
    drupal_add_http_header($name, $value);
108
  }
109
  drupal_send_headers();
110

111 112 113
  // Attempt to increase time to transfer file.
  drupal_set_time_limit(240);

114
  $scheme = variable_get('file_default_scheme', 'public');
115
  // Transfer file in 16 KB chunks to save memory usage.
116 117
  if ($scheme && file_stream_wrapper_valid_scheme($scheme) && $fd = fopen($uri, 'rb')) {
    while (!feof($fd)) {
118
      print fread($fd, 1024*16);
119
    }
120
    fclose($fd);
121 122 123 124
  }
  else {
    drupal_not_found();
  }
125
  drupal_exit();
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147
}

/**
 * Output an XML transformation file for the sitemap XML.
 */
function xmlsitemap_output_xsl() {
  // Read the XSL content from the file.
  $module_path = drupal_get_path('module', 'xmlsitemap');
  $xsl_content = file_get_contents($module_path . '/xsl/xmlsitemap.xsl');

  // Make sure the strings in the XSL content are translated properly.
  $replacements = array(
    'Sitemap file' => t('Sitemap file'),
    'Generated by the <a href="http://drupal.org/project/xmlsitemap">Drupal XML sitemap module</a>.' => t('Generated by the <a href="@link-xmlsitemap">Drupal XML sitemap module</a>.', array('@link-xmlsitemap' => 'http://drupal.org/project/xmlsitemap')),
    'Number of sitemaps in this index' => t('Number of sitemaps in this index'),
    'Click on the table headers to change sorting.' => t('Click on the table headers to change sorting.'),
    'Sitemap URL' => t('Sitemap URL'),
    'Last modification date' => t('Last modification date'),
    'Number of URLs in this sitemap' => t('Number of URLs in this sitemap'),
    'URL location' => t('URL location'),
    'Change frequency' => t('Change frequency'),
    'Priority' => t('Priority'),
148 149
    '[jquery]' => base_path() . 'misc/jquery.js',
    '[jquery-tablesort]' => base_path() . $module_path . '/xsl/jquery.tablesorter.min.js',
150
    '[xsl-js]' => base_path() . $module_path . '/xsl/xmlsitemap.xsl.js',
151
    '[xsl-css]' => base_path() . $module_path . '/xsl/xmlsitemap.xsl.css',
152 153 154 155
  );
  $xsl_content = strtr($xsl_content, $replacements);

  // Output the XSL content.
156 157
  drupal_add_http_header('Content-type', 'application/xml; charset=utf-8');
  drupal_add_http_header('X-Robots-Tag', 'noindex, follow');
158
  print $xsl_content;
159
}