gofiber/template

๐Ÿš€ [Feature]: Unified binary support out of the box

Opened this issue ยท 8 comments

Feature Description

I often create small Fiber apps, and I've been thinking about ways to make them easier to deploy and move around.

One of the main challenges I face is that certain files, like:

  • HTML templates,
  • CSS files,
  • JavaScript files
  • other static assets like favicons

are not included in the statically compiled binary. If all these files could be bundled within the binary, it would significantly simplify deployment. A few considerations arise once these assets are included in the binary:

Can CSS files still be served compressed after inclusion?
If so, it would be ideal to compress all static files at startup and store them in cache.
Iโ€™ve reviewed the following discussions:

From these, I believe the templ package could be a good solution for embedding HTML templates into the binary. I also read this comment #302 (comment) but thought it was a bad example. It only demonstrated rendering static text without passing variables, using templates/partials, or including other static files (CSS, JS, icons). It also didnโ€™t address whether those static assets can still be served compressed or cached.

While I'm not specifically set on templ, despite its increasing popularity, what I'd really like is a simple way to bundle HTML templates, CSS, JS, and other static files into the final statically compiled binary for Fiber apps.

I think this is something many developers would find useful. Ideally, it would be as simple as importing a Fiber sub-package or enabling the feature like this:

app := fiber.New(fiber.Config{
    IncludeAssets: true,
    IncludeTemplates: true,
})

This way, all assets and templates would be included in the binary. I understand implementing this isnโ€™t straightforward, but a possible approach could involve:

  • using the embed package, or
  • using spf13/afero.

However, I'm not familiar with those libraries.
From what I've gathered from other discussions, it seems that some of these features might already be achievable. However, they wouldnโ€™t work seamlessly with Fiber and would require a considerable amount of workaround to implement effectively.

I would love to have a discussion about and see if that is something the maintainers of fiber are considering beeing useful, or not.

This is not primarily about using templ. The main focus is on finding a easy straightforward way to include all assets - typically residing outside the statically compiled binary - into the binary itself.

Checklist:

  • I agree to follow Fiber's Code of Conduct.
  • I have checked for existing issues that describe my suggestion prior to opening this one.
  • I understand that improperly formatted feature requests may be closed without explanation.

I literally just finished playing with this while playing with combining Fiber and Hugo (and HTMX).

Got it working in a few minutes after reading: https://dev.to/aryaprakasa/serving-single-page-application-in-a-single-binary-file-with-go-12ij#serve-with-fiber-framework.

Note that it works as described for v2, but v3's change to remove filesystem and replace it with static had a side-effect that the static FS parameter won't accept an http.FS. I'm not a Go expert by any means... barely past beginner, so I just stayed with v2 for now.

Would love to see this documented formally though, it's actually pretty cool.

gaby commented

@idearat Can you open a ticket in https://github.com/gofiber/fiber, so we can look into that (http.FS) param.

Thanks

I actually also intended to create this issue at gofiber/fiber directly and not in gofiber/template, as I think the unification into one single static binary is not just about templates, but about all types of assets.

Anyway, I currently already switched to v3 (which is in beta) and would love to see an implementation which would let you include all types of assets into the binary.

gaby commented

@the-hotmann You can use embed.FS with the new static middleware.

//go:embed static.go config.go
var fsTestFilesystem embed.FS

app := fiber.New()

app.Get("/embed*", New("", static.Config{
    FS:     fsTestFilesystem,
	Browse: true,
}))

Ok it works. I have tried these two:

  • embed
  • os.DirFS

With os.DirFS I was able to server under this path /*
With embed I was not able to serve under this path /* - iit always defaulted to /static/*. The files have just been accessable through this prefix.

It works, but how can I serve a native favicon.ico under /favicon.ico if all static files are just served under /static/?
I understand that I can set the head-link like this:

<link rel="icon" href="favicon.png">

But if you open a picture in a new tab, your browser will not load the favicon, as it now does not know where it is. Just if html is served the head-link can be set.

Is there a workaround for this? Can files that are embedded also be served under the root-path?
Can I also embed template files like this?

I am talking about this:

var (
	engine      = html.New(template_path, ".html")
	fiberConfig = fiber.Config{
		Views:           engine,
	}
	app = fiber.New(fiberConfig)
)

Ok, got it to work on Fiber v3 aswell:

//go:embed static
embedDirStatic       embed.FS

func main() {

	var (
		newembedDirStatic fs.FS
		err               error
	)

	if newembedDirStatic, err = fs.Sub(embedDirStatic, "static"); err != nil {
		log.Panic(err)
	}

	app.Get("/*", static.New("/", static.Config{
		FS:        newembedDirStatic,
		Compress:  true,
		ByteRange: true,
	}))
	
	[...]
	
}

Hope this helps. As I mentioned above: everything is a little "hacky" and it would be cool if all this would be more tightly integrated into fiber :)

Now just the template embedding it missing

Now I have figured out how to also include the html/templates:

//go:embed templates
embedDirTemplates embed.FS
		
func main() {

	newembedDirTemplates, err := fs.Sub(embedDirTemplates, "templates")
	if err != nil {
		log.Panic(err)
	}

	var (
		engine      = html.NewFileSystem(http.FS(newembedDirTemplates), ".html")
		fiberConfig = fiber.Config{
			Views:           engine,
		}
		app               = fiber.New(fiberConfig)
	)
	
	[...]

}

I hope this helps someone.

gaby commented

Ok, got it to work on Fiber v3 aswell:

//go:embed static
embedDirStatic       embed.FS

func main() {

	var (
		newembedDirStatic fs.FS
		err               error
	)

	if newembedDirStatic, err = fs.Sub(embedDirStatic, "static"); err != nil {
		log.Panic(err)
	}

	app.Get("/*", static.New("/", static.Config{
		FS:        newembedDirStatic,
		Compress:  true,
		ByteRange: true,
	}))
	
	[...]
	
}

Hope this helps. As I mentioned above: everything is a little "hacky" and it would be cool if all this would be more tightly integrated into fiber :)

Now just the template embedding it missing

From my research golang doesnt support conditional embed. It's a compile time, not runtime feature. This limits how much we can do in Fiber. Will report back.