Skip to content

Commit 723e3f4

Browse files
committed
resources: Add FromOpts for more effective resource creation
E.g. when the targetPath already contains a hash or if the resource content is expensive to create.
1 parent d913f46 commit 723e3f4

File tree

2 files changed

+81
-7
lines changed

2 files changed

+81
-7
lines changed

‎common/hashing/hashing.go

+13
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,19 @@ func XXHashFromReader(r io.Reader) (uint64, int64, error) {
3838
return h.Sum64(), size, nil
3939
}
4040

41+
// XxHashFromReaderHexEncoded calculates the xxHash for the given reader
42+
// and returns the hash as a hex encoded string.
43+
func XxHashFromReaderHexEncoded(r io.Reader) (string, error) {
44+
h := getXxHashReadFrom()
45+
defer putXxHashReadFrom(h)
46+
_, err := io.Copy(h, r)
47+
if err != nil {
48+
return "", err
49+
}
50+
hash := h.Sum(nil)
51+
return hex.EncodeToString(hash), nil
52+
}
53+
4154
// XXHashFromString calculates the xxHash for the given string.
4255
func XXHashFromString(s string) (uint64, error) {
4356
h := xxhash.New()

‎resources/resource_factories/create/create.go

+68-7
Original file line numberDiff line numberDiff line change
@@ -218,22 +218,83 @@ func (c *Client) match(name, pattern string, matchFunc func(r resource.Resource)
218218
})
219219
}
220220

221-
// FromString creates a new Resource from a string with the given relative target path.
222-
// TODO(bep) see #10912; we currently emit a warning for this config scenario.
223-
func (c *Client) FromString(targetPath, content string) (resource.Resource, error) {
224-
targetPath = path.Clean(targetPath)
225-
key := dynacache.CleanKey(targetPath) + hashing.MD5FromStringHexEncoded(content)
221+
type Options struct {
222+
// The target path relative to the publish directory.
223+
// Unix style path, i.e. "images/logo.png".
224+
TargetPath string
225+
226+
// Whether the TargetPath has a hash in it which will change if the resource changes.
227+
// If not, we will calculate a hash from the content.
228+
TargetPathHasHash bool
229+
230+
// The content to create the Resource from.
231+
CreateContent func() (func() (hugio.ReadSeekCloser, error), error)
232+
}
233+
234+
// FromOpts creates a new Resource from the given Options.
235+
// Make sure to set optis.TargetPathHasHash if the TargetPath already contains a hash,
236+
// as this avoids the need to calculate it.
237+
// To create a new ReadSeekCloser from a string, use hugio.NewReadSeekerNoOpCloserFromString,
238+
// or hugio.NewReadSeekerNoOpCloserFromBytes for a byte slice.
239+
// See FromString.
240+
func (c *Client) FromOpts(opts Options) (resource.Resource, error) {
241+
opts.TargetPath = path.Clean(opts.TargetPath)
242+
var hash string
243+
var newReadSeeker func() (hugio.ReadSeekCloser, error) = nil
244+
if !opts.TargetPathHasHash {
245+
var err error
246+
newReadSeeker, err = opts.CreateContent()
247+
if err != nil {
248+
return nil, err
249+
}
250+
if err := func() error {
251+
r, err := newReadSeeker()
252+
if err != nil {
253+
return err
254+
}
255+
defer r.Close()
256+
257+
hash, err = hashing.XxHashFromReaderHexEncoded(r)
258+
if err != nil {
259+
return err
260+
}
261+
return nil
262+
}(); err != nil {
263+
return nil, err
264+
}
265+
}
266+
267+
key := dynacache.CleanKey(opts.TargetPath) + hash
226268
r, err := c.rs.ResourceCache.GetOrCreate(key, func() (resource.Resource, error) {
269+
if newReadSeeker == nil {
270+
var err error
271+
newReadSeeker, err = opts.CreateContent()
272+
if err != nil {
273+
return nil, err
274+
}
275+
}
227276
return c.rs.NewResource(
228277
resources.ResourceSourceDescriptor{
229278
LazyPublish: true,
230279
GroupIdentity: identity.Anonymous, // All usage of this resource are tracked via its string content.
231280
OpenReadSeekCloser: func() (hugio.ReadSeekCloser, error) {
232-
return hugio.NewReadSeekerNoOpCloserFromString(content), nil
281+
return newReadSeeker()
233282
},
234-
TargetPath: targetPath,
283+
TargetPath: opts.TargetPath,
235284
})
236285
})
237286

238287
return r, err
239288
}
289+
290+
// FromString creates a new Resource from a string with the given relative target path.
291+
func (c *Client) FromString(targetPath, content string) (resource.Resource, error) {
292+
return c.FromOpts(Options{
293+
TargetPath: targetPath,
294+
CreateContent: func() (func() (hugio.ReadSeekCloser, error), error) {
295+
return func() (hugio.ReadSeekCloser, error) {
296+
return hugio.NewReadSeekerNoOpCloserFromString(content), nil
297+
}, nil
298+
},
299+
})
300+
}

0 commit comments

Comments
 (0)