blob: 01bf2ce50d61effbc9614c2ecf0100645db8568f [file] [log] [blame]
// Copyright 2024 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Package crawl implements a basic web crawler for crawling a portion of a web site.
// Construct a [Crawler], configure it, and then call its [Run] method.
// The crawler stores the crawled data in a [storage.DB], and then
// [Crawler.PageWatcher] can be used to watch for new pages.
package crawl
import (
"bytes"
"context"
"encoding/json"
"io"
"iter"
"log/slog"
"net/http"
"net/url"
"strings"
"time"
"golang.org/x/net/html"
"golang.org/x/oscar/internal/storage"
"golang.org/x/oscar/internal/storage/timed"
"rsc.io/ordered"
)
// This package stores timed entries in the database of the form:
//
// ["crawl.Page", URL] => [Raw(JSON(Page)), Raw(HTML)]
//
// The HTML is the raw HTML served at URL.
// Storing the raw HTML avoids having to re-download the site each time
// we change the way the HTML is processed.
// JSON and HTML are empty if the page has been found but not yet crawled.
const crawlKind = "crawl.Page"
const defaultRecrawl = 24 * time.Hour
// A Crawler is a basic web crawler.
//
// Note that this package does not load or process robots.txt.
// Instead the assumption is that the site owner is crawling a portion of their own site
// and will confiure the crawler appropriately.
// (In the case of Go's Oscar instance, we only crawl go.dev.)
type Crawler struct {
slog *slog.Logger
db storage.DB
http *http.Client
recrawl time.Duration
cleans []func(*url.URL) error
rules []rule
}
// A rule is a rule about which URLs can be crawled.
// See [Crawler.Allow] for more details.
type rule struct {
prefix string // URLs matching this prefix should be ...
allow bool // allowed or disallowed
}
// TODO(rsc): Store ETag and use to avoid redownloading?
// A Page records the result of crawling a single page.
type Page struct {
DBTime timed.DBTime
URL string // URL of page
From string // a page where we found the link to this one
LastCrawl time.Time // time of last crawl
Redirect string // HTTP redirect during fetch
HTML []byte // HTML content, if any
Error string // error fetching page, if any
}
// A crawlPage is the JSON form of Page.
// The fields and field order of crawlPage and Page must match exactly; only the struct tags differ.
// We omit the DBTime, URL, and HTML fields from JSON, because they are encoded separately.
// Using this separate copy of the struct avoids forcing the internal JSON needs of this package
// onto clients using Page.
type crawlPage struct {
DBTime timed.DBTime `json:"-"`
URL string `json:"-"`
From string
LastCrawl time.Time
Redirect string
HTML []byte `json:"-"`
Error string
}
// New returns a new [Crawler] that uses the given logger, database, and HTTP client.
// The caller should configure the Crawler further by calling [Crawler.Add],
// [Crawler.Allow], [Crawler.Deny], [Crawler.Clean], and [Crawler.SetRecrawl].
// Once configured, the crawler can be run by calling [Crawler.Run].
func New(lg *slog.Logger, db storage.DB, hc *http.Client) *Crawler {
if hc != nil {
// We want a client that does not follow redirects,
// but we cannot modify the caller's http.Client directly.
// Instead, make our own copy and override CheckRedirect.
hc1 := *hc
hc = &hc1
hc.CheckRedirect = func(*http.Request, []*http.Request) error { return http.ErrUseLastResponse }
}
c := &Crawler{
slog: lg,
db: db,
http: hc,
recrawl: defaultRecrawl,
}
return c
}
// Add adds the URL to the list of roots for the crawl.
// The added URL must not include a URL fragment (#name).
func (c *Crawler) Add(url string) {
if strings.Contains(url, "#") {
panic("crawl misuse: Add of URL with fragment")
}
if _, ok := c.Get(url); ok {
return
}
b := c.db.Batch()
c.set(b, &Page{URL: url})
b.Apply()
}
// SetRecrawl sets the time to wait before recrawling a page.
// The default is 24 hours.
func (c *Crawler) SetRecrawl(d time.Duration) {
c.recrawl = d
}
// decodePage decodes the timed.Entry into a Page.
func (c *Crawler) decodePage(e *timed.Entry) *Page {
var p Page
if err := ordered.Decode(e.Key, &p.URL); err != nil {
// unreachable unless database corruption
c.db.Panic("decode crawl.Page key", "key", storage.Fmt(e.Key), "err", err)
}
// The HTML is stored separately from the JSON describing the rest of the Page
// to avoid the bother and overhead of JSON-encoding the HTML.
var js, html ordered.Raw
if err := ordered.Decode(e.Val, &js, &html); err != nil {
// unreachable unless database corruption
c.db.Panic("decode crawl.Page val", "val", storage.Fmt(e.Val), "err", err)
}
if len(js) > 0 {
if err := json.Unmarshal(js, (*crawlPage)(&p)); err != nil {
// unreachable unless database corruption
c.db.Panic("decode crawl.Page js", "js", storage.Fmt(js), "err", err)
}
}
p.HTML = html
p.DBTime = e.ModTime
return &p
}
// Get returns the result of the most recent crawl for the given URL.
// If the page has been crawled, Get returns a non-nil *Page, true.
// If the page has not been crawled, Get returns nil, false.
func (c *Crawler) Get(url string) (*Page, bool) {
e, ok := timed.Get(c.db, crawlKind, ordered.Encode(url))
if !ok {
return nil, false
}
return c.decodePage(e), true
}
// Set adds p to the crawled page database.
// It is typically only used for setting up tests.
func (c *Crawler) Set(p *Page) {
b := c.db.Batch()
c.set(b, p)
b.Apply()
}
// set records p in the batch b.
func (c *Crawler) set(b storage.Batch, p *Page) {
if strings.Contains(p.URL, "#") {
// Unreachable without logic bug in this package.
panic("crawl misuse: Set of URL with fragment")
}
timed.Set(c.db, b, crawlKind,
ordered.Encode(p.URL),
ordered.Encode(
ordered.Raw(storage.JSON((*crawlPage)(p))),
ordered.Raw(p.HTML)))
}
// Run crawls all the pages it can, returning when the entire site has been
// crawled either during this run or within the crawl duration set by
// [Crawler.Recrawl].
func (c *Crawler) Run(ctx context.Context) error {
// Crawl every page in the database.
// The pages-by-time list serves as a work queue,
// but if there are link loops we may end up writing a Page
// we've already processed, making it appear again in our scan.
// We use the crawled map to make sure we only crawl each page at most once.
// We use the queued map to make sure we only queue each found link at most once.
crawled := make(map[string]bool)
queued := make(map[string]bool)
for e := range timed.ScanAfter(c.slog, c.db, crawlKind, 0, nil) {
p := c.decodePage(e)
if time.Since(p.LastCrawl) < c.recrawl || crawled[p.URL] {
continue
}
crawled[p.URL] = true
c.crawlPage(ctx, queued, p)
}
return nil
}
// crawlPage downloads the content for a page,
// saves it, and then queues all links it can find in that page's HTML.
func (c *Crawler) crawlPage(ctx context.Context, queued map[string]bool, p *Page) {
var slogBody []byte
slog := c.slog.With("page", p.URL, "lastcrawl", p.LastCrawl)
if strings.Contains(p.URL, "#") {
// Unreachable without logic bug in this package.
panic("crawl misuse: crawlPage of URL with fragment")
}
b := c.db.Batch()
defer func() {
if p.Error != "" {
if slogBody != nil {
slog = slog.With("body", string(slogBody[:min(len(slogBody), 1<<10)]))
}
slog.Warn("crawl error", "err", p.Error, "last", p.LastCrawl)
}
c.set(b, p)
b.Apply()
c.db.Flush()
}()
p.LastCrawl = time.Now()
p.Redirect = ""
p.Error = ""
p.HTML = nil
base, err := url.Parse(p.URL)
if err != nil {
// Unreachable unless Page was corrupted.
p.Error = err.Error()
return
}
u := base.String()
slog = slog.With("url", u)
req, err := http.NewRequestWithContext(ctx, "GET", u, nil)
if err != nil {
// Unreachable unless url.String doesn't round-trip back to url.Parse.
p.Error = err.Error()
return
}
resp, err := c.http.Do(req)
if err != nil {
p.Error = err.Error()
return
}
// TODO(rsc): Make max body length adjustable by policy.
// Also set HTML tokenizer max? For now the max body length
// takes care of it for us.
const maxBody = 4 << 20
body, err := io.ReadAll(io.LimitReader(resp.Body, maxBody+1))
resp.Body.Close()
slogBody = body
if err != nil {
p.Error = err.Error()
return
}
if len(body) > maxBody {
p.Error = "body too big"
return
}
slog = slog.With("status", resp.Status)
if resp.StatusCode/10 == 30 { // Redirect
loc := resp.Header.Get("Location")
if loc == "" {
p.Error = "redirect without location"
return
}
slog = slog.With("location", loc)
locURL, err := url.Parse(loc)
if err != nil {
// Unreachable: http.Client.Do processes the Location header
// to decide about following redirects (we disable that, but that
// check only happens after the Location header is processed),
// and will return an error if the Location has a bad URL.
p.Error = err.Error()
return
}
link := base.ResolveReference(locURL)
p.Redirect = link.String()
slog.Info("crawl redirect", "link", p.Redirect)
c.queue(queued, b, link, u)
return
}
if resp.StatusCode != 200 {
p.Error = "http status " + resp.Status
return
}
slogBody = nil
ctype := resp.Header.Get("Content-Type")
if ctype != "text/html" && !strings.HasPrefix(ctype, "text/html;") {
slog = slog.With("content-type", ctype)
p.Error = "Content-Type: " + ctype
return
}
p.HTML = body
slog = slog.With("htmlsize", len(body))
doc, err := html.Parse(bytes.NewReader(body))
if err != nil {
// Unreachable because it's either a read error
// (but bytes.NewReader has no read errors)
// or hitting the max HTML token limit (but we didn't set that limit).
p.Error = "html parse error: " + err.Error()
return
}
for link := range links(slog, base, doc) {
if queued[link.String()] {
// Quiet skip to avoid tons of repetitive logging about
// all the links in the page footers.
// (Calling c.queue will skip too but also log.)
continue
}
slog.Info("crawl html link", "link", link)
c.queue(queued, b, link, u)
}
slog.Info("crawl ok")
}
// queue queues the link for crawling, unless it has already been queued.
// It records that the link came from a page with URL fromURL.
func (c *Crawler) queue(queued map[string]bool, b storage.Batch, link *url.URL, fromURL string) {
old := link.String()
if queued[old] {
return
}
queued[old] = true
if err := c.clean(link); err != nil {
c.slog.Info("crawl queue clean error", "url", old, "from", fromURL, "err", err)
return
}
targ := link.String()
if targ != old && queued[targ] {
c.slog.Info("crawl queue seen", "url", targ, "old", old, "from", fromURL)
return
}
queued[targ] = true
if !c.allowed(targ) {
c.slog.Info("crawl queue disallow after clean", "url", targ, "old", old, "from", fromURL)
return
}
if strings.Contains(targ, "#") {
// Unreachable without logic bug in this package.
panic("crawl misuse: queue of URL with fragment")
}
p := &Page{
URL: targ,
From: fromURL,
}
if old, ok := c.Get(targ); ok {
if time.Since(old.LastCrawl) < c.recrawl {
c.slog.Debug("crawl queue already visited", "url", targ, "last", old.LastCrawl)
return
}
old.From = p.From
p = old
}
c.slog.Info("crawl queue", "url", p.URL, "old", old)
c.set(b, p)
}
// links returns an iterator over all HTML links in the doc,
// interpreted relative to base.
// It logs unexpected bad URLs to slog.
func links(slog *slog.Logger, base *url.URL, doc *html.Node) iter.Seq[*url.URL] {
return func(yield func(*url.URL) bool) {
// Walk HTML looking for <a href=...>.
var yieldLinks func(*html.Node) bool
yieldLinks = func(n *html.Node) bool {
for c := n.FirstChild; c != nil; c = c.NextSibling {
if !yieldLinks(c) {
return false
}
}
var targ string
if n.Type == html.ElementNode {
switch n.Data {
case "a":
targ = findAttr(n, "href")
}
}
// Ignore no target or #fragment.
if targ == "" || strings.HasPrefix(targ, "#") {
return true
}
// Parse target as URL.
u, err := url.Parse(targ)
if err != nil {
slog.Info("links bad url", "base", base.String(), "targ", targ, "err", err)
return true
}
return yield(base.ResolveReference(u))
}
yieldLinks(doc)
}
}
// findAttr returns the value for n's attribute with the given name.
func findAttr(n *html.Node, name string) string {
for _, a := range n.Attr {
if a.Key == name {
return a.Val
}
}
return ""
}
// Clean adds a cleaning function to the crawler's list of cleaners.
// Each time the crawler considers queuing a URL to be crawled,
// it calls the cleaning functions to canonicalize or otherwise clean the URL first.
// A cleaning function might remove unnecessary URL parameters or
// canonicalize host names or paths.
// The Crawler automatically removes any URL fragment before applying registered cleaners.
func (c *Crawler) Clean(clean func(*url.URL) error) {
c.cleans = append(c.cleans, clean)
}
// allowed reports whether c's configuration allows the target URL.
func (c *Crawler) allowed(targ string) bool {
allow := false
n := 0
for _, r := range c.rules {
if n <= len(r.prefix) && hasPrefix(targ, r.prefix) {
allow = r.allow
n = len(r.prefix)
}
}
return allow
}
// Allow records that the crawler is allowed to crawl URLs with the given list of prefixes.
// A URL is considered to match a prefix if one of the following is true:
//
// - The URL is exactly the prefix.
// - The URL begins with the prefix, and the prefix ends in /.
// - The URL begins with the prefix, and the next character in the URL is / or ?.
//
// The companion function [Crawler.Deny] records that the crawler is not allowed to
// crawl URLs with a list of prefixes. When deciding whether a URL can be crawled,
// longer prefixes take priority over shorter prefixes.
// If the same prefix is added to both [Crawler.Allow] and [Crawler.Deny],
// the last call wins. The default outcome is that a URL is not
// allowed to be crawled.
//
// For example, consider this call sequence:
//
// c.Allow("https://go.dev/a/")
// c.Allow("https://go.dev/a/b/c")
// c.Deny("https://go.dev/a/b")
//
// Given these rules, the crawler makes the following decisions about these URLs:
//
// - https://go.dev/a: not allowed
// - https://go.dev/a/: allowed
// - https://go.dev/a/?x=1: allowed
// - https://go.dev/a/x: allowed
// - https://go.dev/a/b: not allowed
// - https://go.dev/a/b/x: not allowed
// - https://go.dev/a/b/c: allowed
// - https://go.dev/a/b/c/x: allowed
// - https://go.dev/x: not allowed
func (c *Crawler) Allow(prefix ...string) {
for _, p := range prefix {
c.rules = append(c.rules, rule{p, true})
}
}
// Deny records that the crawler is allowed to crawl URLs with the given list of prefixes.
// See the [Crawler.Allow] documentation for details about prefixes and interactions with Allow.
func (c *Crawler) Deny(prefix ...string) {
for _, p := range prefix {
c.rules = append(c.rules, rule{p, false})
}
}
// hasPrefix reports whether targ is considered to have the given prefix,
// following the rules documented in [Crawler.Allow]'s doc comment.
func hasPrefix(targ, prefix string) bool {
if !strings.HasPrefix(targ, prefix) {
return false
}
if len(targ) == len(prefix) || prefix != "" && prefix[len(prefix)-1] == '/' {
return true
}
switch targ[len(prefix)] {
case '/', '?':
return true
}
return false
}
// clean removes the URL Fragment and then calls the registered cleaners on u.
// If any cleaner returns an error, clean returns that error and does not run any more cleaners.
func (c *Crawler) clean(u *url.URL) error {
u.Fragment = ""
for _, fn := range c.cleans {
if err := fn(u); err != nil {
return err
}
}
return nil
}
// PageWatcher returns a timed.Watcher over Pages that the Crawler
// has stored in its database.
func (c *Crawler) PageWatcher(name string) *timed.Watcher[*Page] {
return timed.NewWatcher(c.slog, c.db, name, crawlKind, c.decodePage)
}