Function Options in Go
Explore how to use functional options to create flexible and maintainable APIs in Go
Functional options in Go provide an idiomatic way to manage the configuration of functions, constructors, and methods, allowing for greater flexibility and readability. This idiom is especially useful when you have a function or constructor with several optional parameters.
Basic Example of Functional Options
Here's an example that demonstrates how to use functional options to configure a struct:
package main
import (
"fmt"
)
// Configuration structure.
type Config struct {
Host string
Port int
Timeout int
}
// Functional option type.
type Option func(*Config)
// Host sets the Host field in Config.
func Host(host string) Option {
return func(c *Config) {
c.Host = host
}
}
// Port sets the Port field in Config.
func Port(port int) Option {
return func(c *Config) {
c.Port = port
}
}
// Timeout sets the Timeout field in Config.
func Timeout(timeout int) Option {
return func(c *Config) {
c.Timeout = timeout
}
}
// NewConfig initializes a Config with the given options.
func NewConfig(options ...Option) *Config {
// Default configuration.
config := &Config{
Host: "localhost",
Port: 8080,
Timeout: 30,
}
for _, option := range options {
option(config)
}
return config
}
func main() {
config := NewConfig(Host("example.com"), Port(9090))
fmt.Printf("Config: %+v\n", config)
}
Using Functional Options for APIs
Functional options can make API methods more flexible and maintainable:
package main
import (
"fmt"
)
type Request struct {
URL string
Method string
Headers map[string]string
}
type ReqOption func(*Request)
func WithMethod(method string) ReqOption {
return func(r *Request) {
r.Method = method
}
}
func WithHeader(key, value string) ReqOption {
return func(r *Request) {
if r.Headers == nil {
r.Headers = make(map[string]string)
}
r.Headers[key] = value
}
}
func NewRequest(url string, options ...ReqOption) *Request {
req := &Request{
URL: url,
Method: "GET", // default method
}
for _, opt := range options {
opt(req)
}
return req
}
func main() {
req := NewRequest("https://api.example.com/v1/resource", WithMethod("POST"), WithHeader("Authorization", "Bearer token"))
fmt.Printf("Request: %+v\n", req)
}
Best Practices
- Use functional options to provide clear and concise APIs with many optional parameters.
- Set sensible defaults in the constructor for parameters that are not configured via options.
- Document each option clearly to ensure they're easily understandable and maintainable.
Common Pitfalls
- Not setting defaults can lead to unexpected behavior if the user forgets to configure some fields.
- Overusing functional options might make the API complex, consider the number of options that are genuinely necessary.
- If options interact in complex ways, it could lead to hard-to-debug errors.
Performance Tips
- Minimize memory allocations within option functions for better performance.
- Avoid deep copying of structs unless necessary, as this impacts performance, especially within high-load scenarios.
- Profile your application when adding numerous functional options to ensure it does not introduce performance bottlenecks.