resource: Implement Resource.Content
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sun, 15 Apr 2018 15:07:49 +0000 (17:07 +0200)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sun, 15 Apr 2018 16:08:06 +0000 (18:08 +0200)
Fixes #4622

resource/image.go
resource/image_test.go
resource/resource.go

index 84dc56fa7fbcd3293423a1e713782c07802238b7..19b68a2966ddb2a4890eb153c6596bad9abfe7ef 100644 (file)
@@ -568,6 +568,7 @@ func (i *Image) encodeToDestinations(img image.Image, conf imageConfig, resource
 
 func (i *Image) clone() *Image {
        g := *i.genericResource
+       g.resourceContent = &resourceContent{}
 
        return &Image{
                imaging:         i.imaging,
index 39e538d33b1a82925f0f4cb7e62ea2aeb8277ba5..11807d69500dee350ce558bf1406b4286a441da4 100644 (file)
@@ -322,6 +322,18 @@ func TestSVGImage(t *testing.T) {
        assert.NotNil(svg)
 }
 
+func TestSVGImageContent(t *testing.T) {
+       assert := require.New(t)
+       spec := newTestResourceSpec(assert)
+       svg := fetchResourceForSpec(spec, assert, "circle.svg")
+       assert.NotNil(svg)
+
+       content, err := svg.Content()
+       assert.NoError(err)
+       assert.IsType("", content)
+       assert.Contains(content.(string), `<svg height="100" width="100">`)
+}
+
 func BenchmarkResizeParallel(b *testing.B) {
        assert := require.New(b)
        img := fetchSunset(assert)
index 7fe3b4ff9f726bd52bd166bc736bacf924a6a83e..12e1160cfe71b662019f658f28a0073fcca68e77 100644 (file)
@@ -87,6 +87,14 @@ type Resource interface {
 
        // Params set in front matter for this resource.
        Params() map[string]interface{}
+
+       // Content returns this resource's content. It will be equivalent to reading the content
+       // that RelPermalink points to in the published folder.
+       // The return type will be contextual, and should be what you would expect:
+       // * Page: template.HTML
+       // * JSON: String
+       // * Etc.
+       Content() (interface{}, error)
 }
 
 // Resources represents a slice of resources, which can be a mix of different types.
@@ -360,6 +368,11 @@ func (d dirFile) path() string {
        return path.Join(d.dir, d.file)
 }
 
+type resourceContent struct {
+       content     string
+       contentInit sync.Once
+}
+
 // genericResource represents a generic linkable resource.
 type genericResource struct {
        // The relative path to this resource.
@@ -390,6 +403,26 @@ type genericResource struct {
        osFileInfo   os.FileInfo
 
        targetPathBuilder func(rel string) string
+
+       // We create copies of this struct, so this needs to be a pointer.
+       *resourceContent
+}
+
+func (l *genericResource) Content() (interface{}, error) {
+       var err error
+       l.contentInit.Do(func() {
+               var b []byte
+
+               b, err := afero.ReadFile(l.sourceFs(), l.AbsSourceFilename())
+               if err != nil {
+                       return
+               }
+
+               l.content = string(b)
+
+       })
+
+       return l.content, err
 }
 
 func (l *genericResource) sourceFs() afero.Fs {
@@ -444,6 +477,7 @@ func (l *genericResource) updateParams(params map[string]interface{}) {
 // Implement the Cloner interface.
 func (l genericResource) WithNewBase(base string) Resource {
        l.base = base
+       l.resourceContent = &resourceContent{}
        return &l
 }
 
@@ -611,5 +645,6 @@ func (r *Spec) newGenericResource(
                params:            make(map[string]interface{}),
                name:              baseFilename,
                title:             baseFilename,
+               resourceContent:   &resourceContent{},
        }
 }