Also allow timeout to be set as a duration string, e.g. `30s`.
Fixes #6501
 // If not found a new file is created and passed to create, which should close
 // it when done.
 func (c *Cache) ReadOrCreate(id string,
-       read func(info ItemInfo, r io.Reader) error,
+       read func(info ItemInfo, r io.ReadSeeker) error,
        create func(info ItemInfo, w io.WriteCloser) error) (info ItemInfo, err error) {
        id = cleanID(id)
 
 
 
        var result string
 
-       rf := func(failLevel int) func(info ItemInfo, r io.Reader) error {
+       rf := func(failLevel int) func(info ItemInfo, r io.ReadSeeker) error {
 
-               return func(info ItemInfo, r io.Reader) error {
+               return func(info ItemInfo, r io.ReadSeeker) error {
                        if failLevel > 0 {
                                if failLevel > 1 {
                                        return ErrFatal
 
        v.SetDefault("disableAliases", false)
        v.SetDefault("debug", false)
        v.SetDefault("disableFastRender", false)
-       v.SetDefault("timeout", 30000) // 30 seconds
+       v.SetDefault("timeout", "30s")
        v.SetDefault("enableInlineShortcodes", false)
 
        return nil
 
        c.Assert(err, qt.IsNil)
        defer clean()
 
-       newBuilder := func() *sitesBuilder {
+       newBuilder := func(timeout string) *sitesBuilder {
 
                v := viper.New()
                v.Set("workingDir", workDir)
                v.Set("baseURL", "https://example.org")
+               v.Set("timeout", timeout)
 
                b := newTestSitesBuilder(t).WithWorkingDir(workDir)
                b.Fs = hugofs.NewDefault(v)
 title: "My bundle"
 ---
 
+{{< imgproc >}}
+
 `)
 
-               b.WithTemplatesAdded("index.html", `
+               b.WithTemplatesAdded(
+                       "shortcodes/imgproc.html", `
+{{ $img := resources.Get "images/sunset.jpg" }}
+{{ $r := $img.Resize "129x239" }}
+IMG SHORTCODE: {{ $r.RelPermalink }}/{{ $r.Width }}
+`,
+                       "index.html", `
 {{ $p := .Site.GetPage "mybundle" }}
 {{ $img1 := resources.Get "images/sunset.jpg" }}
 {{ $img2 := $p.Resources.GetMatch "sunset.jpg" }}
 {{ $blurryGrayscale4 := $r.Filter $filters }}
 BG4: {{ $blurryGrayscale4.RelPermalink }}/{{ $blurryGrayscale4.Width }}
 
-
+{{ $p.Content }}
 
 `)
 
        out.Close()
        src.Close()
 
-       b := newBuilder()
-       b.Build(BuildCfg{})
+       // First build it with a very short timeout to trigger errors.
+       b := newBuilder("10ns")
 
        imgExpect := `
 Resized1: images/sunset.jpg|123|234|image/jpg|/images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_123x234_resize_q75_box.jpg|
 BG2: /images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_2ae8bb993431ec1aec40fe59927b46b4.jpg/123
 BG3: /images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_ed7740a90b82802261c2fbdb98bc8082.jpg/123
 BG4: /images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_ed7740a90b82802261c2fbdb98bc8082.jpg/123
+IMG SHORTCODE: /images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_129x239_resize_q75_box.jpg/129
 `
 
-       b.AssertFileContent(filepath.Join(workDir, "public/index.html"), imgExpect)
-       b.AssertImage(350, 219, "public/images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_350x0_resize_q75_box.a86fe88d894e5db613f6aa8a80538fefc25b20fa24ba0d782c057adcef616f56.jpg")
+       assertImages := func() {
+               b.Helper()
+               b.AssertFileContent(filepath.Join(workDir, "public/index.html"), imgExpect)
+               b.AssertImage(350, 219, "public/images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_350x0_resize_q75_box.a86fe88d894e5db613f6aa8a80538fefc25b20fa24ba0d782c057adcef616f56.jpg")
+               b.AssertImage(129, 239, "public/images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_129x239_resize_q75_box.jpg")
+       }
+
+       err = b.BuildE(BuildCfg{})
+       c.Assert(err, qt.Not(qt.IsNil))
+
+       b = newBuilder("30s")
+       b.Build(BuildCfg{})
+
+       assertImages()
+
+       // Truncate one image.
+       imgInCache := filepath.Join(workDir, "resources/_gen/images/sunset_hu59e56ffff1bc1d8d122b1403d34e039f_90587_ed7740a90b82802261c2fbdb98bc8082.jpg")
+       f, err := os.Create(imgInCache)
+       c.Assert(err, qt.IsNil)
+       f.Close()
 
        // Build it again to make sure we read images from file cache.
-       b = newBuilder()
+       b = newBuilder("30s")
        b.Build(BuildCfg{})
 
-       b.AssertFileContent(filepath.Join(workDir, "public/index.html"), imgExpect)
+       assertImages()
 
 }
 
 
                needTimeout := !p.renderable || p.shortcodeState.hasShortcodes()
 
                if needTimeout {
-                       cp.initMain = parent.BranchdWithTimeout(p.s.siteCfg.timeout, func(ctx context.Context) (interface{}, error) {
+                       cp.initMain = parent.BranchWithTimeout(p.s.siteCfg.timeout, func(ctx context.Context) (interface{}, error) {
                                return nil, initContent()
                        })
                } else {
 }
 
 func (p *pageContentOutput) Content() (interface{}, error) {
-       p.p.s.initInit(p.initMain, p.p)
-       return p.content, nil
+       if p.p.s.initInit(p.initMain, p.p) {
+               return p.content, nil
+       }
+       return nil, nil
 }
 
 func (p *pageContentOutput) FuzzyWordCount() int {
 
        init.menus.Reset()
 }
 
-func (s *Site) initInit(init *lazy.Init, pctx pageContext) {
+func (s *Site) initInit(init *lazy.Init, pctx pageContext) bool {
        _, err := init.Do()
        if err != nil {
                s.h.FatalError(pctx.wrapError(err))
        }
+       return err == nil
 }
 
 func (s *Site) prepareInits() {
                return nil, err
        }
 
+       timeout := 30 * time.Second
+       if cfg.Language.IsSet("timeout") {
+               switch v := cfg.Language.Get("timeout").(type) {
+               case int:
+                       timeout = time.Duration(v) * time.Millisecond
+               case string:
+                       d, err := time.ParseDuration(v)
+                       if err == nil {
+                               timeout = d
+                       }
+               }
+       }
+
        siteConfig := siteConfigHolder{
                sitemap:          config.DecodeSitemap(config.Sitemap{Priority: -1, Filename: "sitemap.xml"}, cfg.Language.GetStringMap("sitemap")),
                taxonomiesConfig: taxonomies,
-               timeout:          time.Duration(cfg.Language.GetInt("timeout")) * time.Millisecond,
+               timeout:          timeout,
                hasCJKLanguage:   cfg.Language.GetBool("hasCJKLanguage"),
                enableEmoji:      cfg.Language.Cfg.GetBool("enableEmoji"),
        }
 
 }
 
 // BranchdWithTimeout is same as Branch, but with a timeout.
-func (ini *Init) BranchdWithTimeout(timeout time.Duration, f func(ctx context.Context) (interface{}, error)) *Init {
+func (ini *Init) BranchWithTimeout(timeout time.Duration, f func(ctx context.Context) (interface{}, error)) *Init {
        return ini.Branch(func() (interface{}, error) {
                return ini.withTimeout(timeout, f)
        })
 
                        name := fmt.Sprintf("goldmark_%s.txt", c.ctx.DocumentID)
                        filename := filepath.Join(dir, name)
                        afero.WriteFile(hugofs.Os, filename, ctx.Src, 07555)
-                       err = errors.Errorf("[BUG] goldmark: create an issue on GitHub attaching the file in: %s", filename)
+                       err = errors.Errorf("[BUG] goldmark: %s: create an issue on GitHub attaching the file in: %s", r, filename)
 
                }
        }()
 
 
                key := i.getImageMetaCacheTargetPath()
 
-               read := func(info filecache.ItemInfo, r io.Reader) error {
+               read := func(info filecache.ItemInfo, r io.ReadSeeker) error {
                        meta := &imageMeta{}
                        data, err := ioutil.ReadAll(r)
                        if err != nil {
 
        // These funcs are protected by a named lock.
        // read clones the parent to its new name and copies
        // the content to the destinations.
-       read := func(info filecache.ItemInfo, r io.Reader) error {
+       read := func(info filecache.ItemInfo, r io.ReadSeeker) error {
                img = parent.clone(nil)
                rp := img.getResourcePaths()
                rp.relTargetDirFile.file = relTarget.file
                img.setSourceFilename(info.Name)
 
+               if err := img.InitConfig(r); err != nil {
+                       return err
+               }
+
+               r.Seek(0, 0)
+
                w, err := img.openDestinationsForWriting()
                if err != nil {
                        return err
 
                defer w.Close()
                _, err = io.Copy(w, r)
+
                return err
        }
 
 
        return &i
 }
 
+// InitConfig reads the image config from the given reader.
+func (i *Image) InitConfig(r io.Reader) error {
+       var err error
+       i.configInit.Do(func() {
+               i.config, _, err = image.DecodeConfig(r)
+       })
+       return err
+}
+
 func (i *Image) initConfig() error {
        var err error
        i.configInit.Do(func() {
                        return
                }
 
-               var (
-                       f      hugio.ReadSeekCloser
-                       config image.Config
-               )
+               var f hugio.ReadSeekCloser
 
                f, err = i.Spec.ReadSeekCloser()
                if err != nil {
                }
                defer f.Close()
 
-               config, _, err = image.DecodeConfig(f)
-               if err != nil {
-                       return
-               }
-               i.config = config
+               i.config, _, err = image.DecodeConfig(f)
        })
 
        if err != nil {