@@ -218,22 +218,83 @@ func (c *Client) match(name, pattern string, matchFunc func(r resource.Resource)
218
218
})
219
219
}
220
220
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
226
268
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
+ }
227
276
return c .rs .NewResource (
228
277
resources.ResourceSourceDescriptor {
229
278
LazyPublish : true ,
230
279
GroupIdentity : identity .Anonymous , // All usage of this resource are tracked via its string content.
231
280
OpenReadSeekCloser : func () (hugio.ReadSeekCloser , error ) {
232
- return hugio . NewReadSeekerNoOpCloserFromString ( content ), nil
281
+ return newReadSeeker ()
233
282
},
234
- TargetPath : targetPath ,
283
+ TargetPath : opts . TargetPath ,
235
284
})
236
285
})
237
286
238
287
return r , err
239
288
}
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