@@ -30,7 +30,6 @@ import (
30
30
31
31
"github.com/gohugoio/hugo/cache/filecache"
32
32
"github.com/gohugoio/hugo/common/hashing"
33
- "github.com/gohugoio/hugo/common/hstrings"
34
33
"github.com/gohugoio/hugo/common/paths"
35
34
36
35
"github.com/disintegration/gift"
@@ -205,15 +204,12 @@ func (i *imageResource) cloneWithUpdates(u *transformationUpdate) (baseResource,
205
204
}, nil
206
205
}
207
206
208
- var imageActions = []string {images .ActionResize , images .ActionCrop , images .ActionFit , images .ActionFill }
209
-
210
207
// Process processes the image with the given spec.
211
208
// The spec can contain an optional action, one of "resize", "crop", "fit" or "fill".
212
209
// This makes this method a more flexible version that covers all of Resize, Crop, Fit and Fill,
213
210
// but it also supports e.g. format conversions without any resize action.
214
211
func (i * imageResource ) Process (spec string ) (images.ImageResource , error ) {
215
- action , options := i .resolveActionOptions (spec )
216
- return i .processActionOptions (action , options )
212
+ return i .processActionSpec ("" , spec )
217
213
}
218
214
219
215
// Resize resizes the image to the specified width and height using the specified resampling
@@ -243,55 +239,38 @@ func (i *imageResource) Fill(spec string) (images.ImageResource, error) {
243
239
}
244
240
245
241
func (i * imageResource ) Filter (filters ... any ) (images.ImageResource , error ) {
246
- var conf images.ImageConfig
242
+ var confMain images.ImageConfig
247
243
248
244
var gfilters []gift.Filter
249
245
250
246
for _ , f := range filters {
251
247
gfilters = append (gfilters , images .ToFilters (f )... )
252
248
}
253
249
254
- var (
255
- targetFormat images.Format
256
- configSet bool
257
- )
250
+ var options []string
251
+
258
252
for _ , f := range gfilters {
259
253
f = images .UnwrapFilter (f )
260
254
if specProvider , ok := f .(images.ImageProcessSpecProvider ); ok {
261
- action , options := i .resolveActionOptions (specProvider .ImageProcessSpec ())
262
- var err error
263
- conf , err = images .DecodeImageConfig (action , options , i .Proc .Cfg , i .Format )
264
- if err != nil {
265
- return nil , err
266
- }
267
- configSet = true
268
- if conf .TargetFormat != 0 {
269
- targetFormat = conf .TargetFormat
270
- // We only support one target format, but prefer the last one,
271
- // so we keep going.
272
- }
255
+ options = append (options , strings .Fields (specProvider .ImageProcessSpec ())... )
273
256
}
274
257
}
275
258
276
- if ! configSet {
277
- conf = images .GetDefaultImageConfig ("filter" , i .Proc .Cfg )
259
+ confMain , err := images .DecodeImageConfig (options , i .Proc .Cfg , i .Format )
260
+ if err != nil {
261
+ return nil , err
278
262
}
279
263
280
- conf .Action = "filter"
281
- conf .Key = hashing .HashString (gfilters )
282
- conf .TargetFormat = targetFormat
283
- if conf .TargetFormat == 0 {
284
- conf .TargetFormat = i .Format
285
- }
264
+ confMain .Action = "filter"
265
+ confMain .Key = hashing .HashString (gfilters )
286
266
287
- return i .doWithImageConfig (conf , func (src image.Image ) (image.Image , error ) {
267
+ return i .doWithImageConfig (confMain , func (src image.Image ) (image.Image , error ) {
288
268
var filters []gift.Filter
289
269
for _ , f := range gfilters {
290
270
f = images .UnwrapFilter (f )
291
271
if specProvider , ok := f .(images.ImageProcessSpecProvider ); ok {
292
- processSpec := specProvider .ImageProcessSpec ()
293
- action , options := i .resolveActionOptions (processSpec )
294
- conf , err := images .DecodeImageConfig (action , options , i .Proc .Cfg , i .Format )
272
+ options := strings .Fields (specProvider .ImageProcessSpec ())
273
+ conf , err := images .DecodeImageConfig (options , i .Proc .Cfg , i .Format )
295
274
if err != nil {
296
275
return nil , err
297
276
}
@@ -313,25 +292,13 @@ func (i *imageResource) Filter(filters ...any) (images.ImageResource, error) {
313
292
})
314
293
}
315
294
316
- func (i * imageResource ) resolveActionOptions (spec string ) (string , []string ) {
317
- var action string
318
- options := strings .Fields (spec )
319
- for i , p := range options {
320
- if hstrings .InSlicEqualFold (imageActions , p ) {
321
- action = p
322
- options = append (options [:i ], options [i + 1 :]... )
323
- break
324
- }
325
- }
326
- return action , options
327
- }
328
-
329
295
func (i * imageResource ) processActionSpec (action , spec string ) (images.ImageResource , error ) {
330
- return i .processActionOptions (action , strings .Fields (spec ))
296
+ options := append ([]string {action }, strings .Fields (strings .ToLower (spec ))... )
297
+ return i .processOptions (options )
331
298
}
332
299
333
- func (i * imageResource ) processActionOptions ( action string , options []string ) (images.ImageResource , error ) {
334
- conf , err := images .DecodeImageConfig (action , options , i .Proc .Cfg , i .Format )
300
+ func (i * imageResource ) processOptions ( options []string ) (images.ImageResource , error ) {
301
+ conf , err := images .DecodeImageConfig (options , i .Proc .Cfg , i .Format )
335
302
if err != nil {
336
303
return nil , err
337
304
}
@@ -343,13 +310,12 @@ func (i *imageResource) processActionOptions(action string, options []string) (i
343
310
return nil , err
344
311
}
345
312
346
- if action == images .ActionFill {
347
- if conf .Anchor == 0 && img .Width () == 0 || img .Height () == 0 {
313
+ if conf . Action == images .ActionFill {
314
+ if conf .Anchor == images . SmartCropAnchor && img .Width () == 0 || img .Height () == 0 {
348
315
// See https://github.com/gohugoio/hugo/issues/7955
349
316
// Smartcrop fails silently in some rare cases.
350
317
// Fall back to a center fill.
351
- conf .Anchor = gift .CenterAnchor
352
- conf .AnchorStr = "center"
318
+ conf = conf .Reanchor (gift .CenterAnchor )
353
319
return i .doWithImageConfig (conf , func (src image.Image ) (image.Image , error ) {
354
320
return i .Proc .ApplyFiltersFromConfig (src , conf )
355
321
})
@@ -417,7 +383,7 @@ func (i *imageResource) doWithImageConfig(conf images.ImageConfig, f func(src im
417
383
}
418
384
419
385
ci := i .clone (converted )
420
- targetPath := i .relTargetPathFromConfig (conf )
386
+ targetPath := i .relTargetPathFromConfig (conf , i . getSpec (). imaging . Cfg . SourceHash )
421
387
ci .setTargetPath (targetPath )
422
388
ci .Format = conf .TargetFormat
423
389
ci .setMediaType (conf .TargetFormat .MediaType ())
@@ -485,26 +451,30 @@ func (i *imageResource) getImageMetaCacheTargetPath() string {
485
451
df := i .getResourcePaths ()
486
452
p1 , _ := paths .FileAndExt (df .File )
487
453
h := i .hash ()
488
- idStr := hashing .HashString (h , i .size (), imageMetaVersionNumber , cfgHash )
454
+ idStr := hashing .HashStringHex (h , i .size (), imageMetaVersionNumber , cfgHash )
489
455
df .File = fmt .Sprintf ("%s_%s.json" , p1 , idStr )
490
456
return df .TargetPath ()
491
457
}
492
458
493
- func (i * imageResource ) relTargetPathFromConfig (conf images.ImageConfig ) internal.ResourcePaths {
459
+ func (i * imageResource ) relTargetPathFromConfig (conf images.ImageConfig , imagingConfigSourceHash string ) internal.ResourcePaths {
494
460
p1 , p2 := paths .FileAndExt (i .getResourcePaths ().File )
495
461
if conf .TargetFormat != i .Format {
496
462
p2 = conf .TargetFormat .DefaultExtension ()
497
463
}
498
- const prefix = "_hu"
499
- huIdx := strings .LastIndex (p1 , prefix )
500
- incomingID := "i"
464
+
465
+ // Do not change.
466
+ const imageHashPrefix = "_hu_"
467
+
468
+ huIdx := strings .LastIndex (p1 , imageHashPrefix )
469
+ incomingID := ""
501
470
if huIdx > - 1 {
502
- incomingID = p1 [huIdx + len (prefix ):]
471
+ incomingID = p1 [huIdx + len (imageHashPrefix ):]
503
472
p1 = p1 [:huIdx ]
504
473
}
505
- hash := hashing .HashUint64 (incomingID , i .hash (), conf .GetKey (i .Format ))
474
+
475
+ hash := hashing .HashStringHex (incomingID , i .hash (), conf .Key , imagingConfigSourceHash )
506
476
rp := i .getResourcePaths ()
507
- rp .File = fmt .Sprintf ("%s%s%d %s" , p1 , prefix , hash , p2 )
477
+ rp .File = fmt .Sprintf ("%s%s%s %s" , p1 , imageHashPrefix , hash , p2 )
508
478
509
479
return rp
510
480
}
0 commit comments