Hugo でサーチエンジン用に sitemap.xml や robots.txt ファイルを配置する

サーチエンジンに対して、インデックス登録させたいページを伝えるには、Web サイトに sitemap.xmlrobots.txt というファイルを配置します。 Google の検索結果に影響するため、正しく理解して配置しましょう。

sitemap.xml と robots.txt の概要

Web サイトのルートに sitemap.xmlrobots.txt を置くと、下記のようなヒント情報を Google などの検索エンジンに伝えることができます。

  • sitemap.xml … サイト内のページ一覧情報を記述します。検索エンジンは、サイト全体をクロールしなくても、この情報でページのインデックス情報を作成できるようになります。ページごとに更新日時 (YYYY-MM-DD) や優先度 (0.0〜1.0) を指定できます。
  • robots.txt … クロールしなくてもよいディレクトリやファイルを指定します。上記の sitemap.xml の URL を指定することもできます。

もちろん、これらのファイルを用意したところで、どこまで反映されるかは検索エンジン側の実装によりますが、SEO 対策として少しでもヒントとなる情報を与えておくのが望ましいでしょう。

sitemap.xml を作成する

sitemap.xml ファイルには、Google などの検索エンジンにインデックス登録してもらいたいページの一覧を記述します。

sitemap.xml のデフォルト出力

Hugo はデフォルトで Web サイトのルートに、Sitemap protocol に基づいた sitemap.xml ファイルを出力するようになっています。 下記はデフォルト設定で出力される sitemap.xml の例です。

sitemap.xml
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9" xmlns:xhtml="http://www.w3.org/1999/xhtml">
  <url>
    <loc>https://example.com/page1/</loc>
    <lastmod>2020-04-26T00:00:00+00:00</lastmod>
  </url>
  <url>
    <loc>https://example.com/page2/</loc>
    <lastmod>2020-04-25T00:00:00+00:00</lastmod>
  </url>
</urlset>

この出力は、Hugo にデフォルトで組み込まれている sitemap.xml テンプレートによって生成されています。 デフォルトのテンプレート定義は下記を見ると分かります。

このままで問題なければ、特に何も設定する必要はありませんが、出力内容をカスタマイズしたいときは、次のように sitemap.xml 用のテンプレートファイルを生成する必要があります。 Hugo はデフォルトですべてのページを sitemap.xml にリスト化しようとするので、Google などの検索結果に載せたくないページがある場合は、sitemap.xml のカスタマイズが必要です。

sitemap.xml のカスタマイズ

下記のいずれかのテンプレートファイルを作成することで、Web サイトのルートディレクトリに出力される sitemap.xml の内容をカスタマイズできます。

  • layouts/sitemap.xml
  • layouts/_default/sitemap.xml

ここでは、下記のようにカスタマイズしてみます。

  • タクソノミーリスト(あるタグのページ一覧)のページを出力しない(/tags/*/categories/* などをリストに載せない)
  • 特定の URL 以下のページを出力しない (URL が /search/ で始まるページをリストに載せない)
  • Markdown ファイルのフロントマターで noindex: true と記述されていたら出力しない
  • セクションページの優先度を上げる(デフォルトで priority=0.5 のところを priority=1.0 とする。ただしフロントマターでの指定があればそちらを優先)
layouts/sitemap.xml
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9" xmlns:xhtml="http://www.w3.org/1999/xhtml">
  {{- range .Data.Pages }}
  {{- if ne .Kind "term" }}
  {{- if not (hasPrefix .RelPermalink "/search/") }}
  {{- if ne .Params.noindex true }}
  <url>
    <loc>{{ .Permalink }}</loc>
    {{- if not .Lastmod.IsZero }}
    <lastmod>{{ .Lastmod.Format "2006-01-02" }}</lastmod>
    {{- end }}
    {{- with .Sitemap.ChangeFreq }}
    <changefreq>{{ . }}</changefreq>
    {{- end }}
    {{- if ge .Sitemap.Priority 0.0 }}
    <priority>{{ .Sitemap.Priority }}</priority>
    {{- else if eq .Kind "section" }}
    <priority>1.0</priority>
    {{- end }}
  </url>
  {{- end }}
  {{- end }}
  {{- end }}
  {{- end }}
</urlset>

ここでは、ページ変数の .Kind の値を見ることで、セクションやタクソノミーリスト (/tags/*) のページであるかどうかを判断しています。 このあたりのページ変数がどのような値を取るかに関しては、下記の記事を参考にしてください。

各ページのフロントマターでサイトマップ出力を制御する

ページごとにサイトマップ出力を制御したい場合は、フロントマターのプロパティを設定します。

サイトマップに列挙されないようにする

サイトマップには列挙したくない特殊なページがあるかもしれません。 簡単な方法として、Hugo 組み込みの _build プロパティで、list: never を指定してしまう方法があります(参考: Hugo でページリスト(記事一覧)に列挙されないページを作る)。 このフラグを設定すると、Hugo の .Pages.RegularPages などのコレクションに含まれなくなるので、結果的に sitemap.xml にも列挙されなくなります(サイトマップテンプレート内で .Data.Pages をループ処理していたことに着目してください)。

content/page.md(Hugo の _build プロパティで出力抑制)
---
title: "ページタイトル"
date: "2023-05-31"
_build: { list: never }
---

別の方法として、独自の noindex プロパティを定義してしまう方法もあります。 .Pages コレクションなどには含めたいけれど、サイトマップには列挙したくないといった事情があれば、このような独自プロパティを使って制御します。 前述のカスタマイズされた sitemap.xml 用テンプレートでは、この noindex プロパティがセットされていたら、そのページの URL を列挙しないようにしています。

content/page.md(独自の noindex プロパティで出力抑制)
---
title: "ページタイトル"
date: "2023-05-31"
noindex: true
---

ちなみに、独自の noindex プロパティを導入するのであれば、それと連動させて HTML の meta 要素を出力するようにしておくのがよいでしょう(noindex 用の meta 要素については後述)。

layouts/_default/baseof.html(ベーステンプレートの抜粋)
<head>
  <meta charset="UTF-8">
  {{- if .Params.noindex }}
  <meta name="robots" content="noindex" />
  {{- end }}

サイトマップに出力される情報を調整する (changefreq, priority)

Hugo 組み込みの sitemap プロパティを設定することで、sitemap.xml に出力する <changefreq><priority> 要素の値を制御することができます。 これらの値は、前述のテンプレート内で .Sitemap.ChangeFreq.Sitemap.Priority といった変数で参照しています。

content/page.md(ページのプライオリティなどを調整)
---
title: "ページタイトル"
date: "2020-04-26"
sitemap:
  changefreq: monthly
  priority: 0.8
---

sitemap.xml の出力先パスを変更する

sitemap.xml はデフォルトで Web サイトのルートに出力されますが、Hugo の設定ファイルで出力パスを変更することができます。

hugo.toml
[sitemap]
  filename = "assets/my-sitemap.xml"

robots.txt によるクロールの抑制

Web サイトのルートに robots.txt を配置しておくと、検索エンジンに対して、そのサイト内でクロールされたくないディレクトリやファイルを知らせることができます(参考: robotstxt.org)。 また、上記で作成した sitemap.xml の URL を知らせることもできます。

robots.txt を出力するための設定

静的なファイルとして robots.txt を作成するのであれば、単純に Hugo プロジェクト内に /static/robots.txt というファイルを配置しておけば OK です(robots.txt が出力ディレクトリのルートにコピーされます)。 テンプレート機能を使って、動的に robots.txt を生成したいのであれば、設定ファイルで下記のように指定しておく必要があります。

hugo.toml
enableRobotsTXT = true

この設定により、下記のテンプレートファイルを使って robots.txt を生成してくれるようになります。

  • /layouts/robots.txt
  • /themes/<THEME>/layouts/robots.txt

下記のようなテンプレートを作成すると、すべての種類のクローラーに対して sitemap.xml の URL を知らせるための robots.txt を生成できます。

/layouts/robots.txt
User-agent: *
Sitemap: {{ .Site.BaseURL }}sitemap.xml

sitemap.xml のパスは、FQDN(完全修飾 URL)で指定しなければいけないので、上記のようにテンプレート内で .Site.BaseURL 変数を使って完全な URL を構築しています。

robots.txt の Disallow ディレクティブはあまり使わない

よくある robots.txt の例として、Disallow ディレクティブを使ったクロールの防止方法が示されていたりしますが、実際に Disallow ディレクティブが必要になるケースはそれほど多くありません(それよりは後述の meta 要素を使う方が目的と合っていることが多いです)。 特に、JavaScript や CSS、画像ファイルなどは、検索エンジンがページ内容を正しく処理するために必要なので、Disallow 指定してクロール対象から外してはいけません。

Google の検索結果から除外する(meta 要素の noindex 設定)

robots.txt の Disallow ではなく meta 要素の noindex を使う

特定のページを Google の検索結果に表示されないようにするには、HTML の head 要素内に下記のような meta 要素を記述するのが正解みたいです。

<meta name="robots" content="noindex" />

robots.txt に下記のように指定すれば、/search/ 以下をクロールしないように知らせることにはなるのですが、すでにインデックスされてしまったページは検索結果に表示され続けることになってしまいます。

robots.txt
User-agent: *
Disallow: /search/

また、クロールを防いだとしても、他のページからリンクされている場合はインデックスされてしまう可能性があります。

よって、検索エンジンに特定のページがインデックスされることを防ぐには、

  1. meta 要素で noindex を指定する
  2. robots.txtDisallow は指定しない

とするのがよいようです。

  • 参考: noindex を使用して検索インデックス登録をブロックする - Search Console ヘルプ

    重要: noindex ディレクティブが有効に機能するようにするために、robots.txt ファイルでページをブロックしないでください。ページが robots.txt ファイルでブロックされると、クローラは noindex ディレクティブを認識しません。そのため、たとえば他のページからリンクされていると、ページは検索結果に引き続き表示される可能性があります。

(コラム)Google Search Console の sitemap.xml に関するエラー

sitemap.xml にリストアップされているにも関わらず、HTML ファイルの <meta name="robots" content="noindex"> でインデックス登録を抑制しているようなページがあると、Google Search Console にエラーが表示されることがあります。

送信された URL に noindex タグが追加されています
図: 送信された URL に noindex タグが追加されています

これは、「sitemap.xml でインデックス登録しろと指定されているけど、実際にページの内容を見ると meta 要素で noindex 指定されてるのでインデックス登録しないよ。それでいいの?」という意味の警告です。 インデックス登録されたくないページは、sitemap.xml にもリストアップされないように整合性を取りましょう。