Best Practices
- Use the Ambient logger with all it's different levels: error, warn, info, and debug. You shouldn't use
log
orfmt
package to output any messages because they are not standardized. - Plugins should not cause the application to halt or exit. They should also return an error where possible.
- If you run background jobs in your plugin, make sure you implement the
Enable()
function to start the background job and theDisable()
function to stop the background job. - When creating a FuncMap, you must prefix each one with your plugin name and then an underscore - this is to prevent collisions in the templates. A prefix will be added automatically if you don't which may cause inconsistencies.
- You must return every permission your plugin needs to use in the
GrantRequests()
function. Otherwise, the plugin will not be grant access to work properly. - Routes can be overwritten by other plugins so it's best to namespace with the plugin name since plugin names are unique.
- Use
Path()
in middleware and{{URLPrefix}}
in HTML templates so you always respect theAMB_URL_PREFIX
environment variable. It is already prefixed on routes and assets so you don't need to modify them. - to protect against cross-site request forgery, use
p.Site.SetCSRF(r)
andp.Site.CSRF(r, r.FormValue("token"))
to set and verify tokens on form submission.
What should determine the plugin boundary?
You should group required items together in a plugin. For instance, a session manager should implement both the SessionManager()
and Middleware()
functions - it wouldn't make sense to separate into different plugins since they require each other.
You could group by feature. A login plugin will have multiple routes (GET /login
, POST /login
, GET /logout
) so those should all be in the same plugin.
You could group by type as well. You may want to keep most of your middleware or code that modifies your header in the same plugin. For middleware, you may want to group these into a single plugin so you can use a single settings page to enable or disable each one: request logging, trail slash removal, and response gzip compression.
How should I approach wrapping a package as a plugin?
Regardless if you're using Ambient or not, Go packages should be designed so they can be reused by other people and projects without tightly coupling dependencies. If you are creating a new plugin, build the package so it follows Go best practices and then import it to the plugin package. That way you can use it with other apps as well.
Should I be worried about creating too many plugins?
If you feel like you have so many plugins that it's hard to find what you're looking for, it's probably too many plugins.
Does the app run slower with more plugins?
It will require a bit more startup time as you add plugins, but it shouldn't be noticeable.