Props¶
Overview¶
Props serves as the primary data structure that carries essential information about your tool and provides access to various services and configurations. It's designed to be passed to all major components and commands in your CLI application.
What's in a Name?
The name Props is not merely a shorthand for 'properties' (though we do shove plenty of those in there). It’s a direct reference to a prop, the heavy-duty timber or steel beam that prevents a structure from an embarrassing collapse. For the sports fans, it’s also a lovingly crafted nod to the rugby position: the broad-shouldered stalwarts who provide the primary structural support for the scrum. Much like its on-field namesake, our Props struct isn't here to score the flashy tries; it's here to do the unsung heavy lifting that keeps the entire framework from falling over.
Design Rationale¶
Props is intentionally designed as a concrete dependency injection container rather than using Go's context.Context for passing dependencies. This design choice provides several key benefits:
Type Safety and Compile-Time Checks¶
Unlike context.Context which stores values as interface{}, Props provides concrete types for all dependencies:
// Props approach - Type safe, IDE-friendly
func NewCommand(props *props.Props) *cobra.Command {
props.Logger.Info("Starting command") // ✅ Compile-time type checking
host := props.Config.GetString("db.host") // ✅ Known interface methods
return cmd
}
// Context approach - Runtime type assertions required
func NewCommand(ctx context.Context) *cobra.Command {
l := ctx.Value("logger").(logger.Logger) // ❌ Runtime panic risk
config := ctx.Value("config").(SomeInterface) // ❌ No compile-time guarantee
return cmd
}
Clear Dependency Declaration¶
Props makes dependencies explicit and discoverable:
- Discoverability: IDEs can provide accurate autocomplete and navigation
- Documentation: Each field is clearly documented with its purpose
- Refactoring: Changes to dependency interfaces are caught at compile time
- Testing: Easy to create test doubles with concrete interfaces
Performance Benefits¶
- No runtime type assertions: All types are known at compile time
- Reduced allocations: No boxing/unboxing of interface{} values
- Better inlining: Compiler can optimize concrete type access
Core Structure¶
[!NOTE] See pkg.go.dev/gitlab.com/phpboyscout/go-tool-base/pkg/props for the full API definition.
Collector is always non-nil
When telemetry is disabled, Collector is a noop implementation. Commands can safely call p.Collector.Track(...) without checking whether telemetry is enabled.
The root bootstrap upholds this invariant automatically: building the command tree (NewCmdRoot) defaults the field to props.NoopCollector{}, and the resolved *telemetry.Collector replaces it once config loads. A Props constructed directly as a struct literal — for example in tests that exercise a command without going through the bootstrap — should set Collector: props.NoopCollector{} itself, or run the command via root.Execute (which also defaults it).
ErrorHandler is an Interface
The ErrorHandler field is an interface type, not a pointer. This enables easy mocking and custom implementations for testing.
Constants and Types¶
Feature Commands¶
Feature commands are identifiers used to enable or disable built-in functionality:
type FeatureCmd string
const (
UpdateCmd = FeatureCmd("update") // Self-update functionality
InitCmd = FeatureCmd("init") // Configuration initialisation
McpCmd = FeatureCmd("mcp") // Model Context Protocol server
DocsCmd = FeatureCmd("docs") // Interactive documentation browser
AiCmd = FeatureCmd("ai") // AI-powered features (opt-in)
DoctorCmd = FeatureCmd("doctor") // Environment health checks
ConfigCmd = FeatureCmd("config") // Programmatic config access (opt-in)
TelemetryCmd = FeatureCmd("telemetry") // Anonymous usage telemetry (opt-in)
ChangelogCmd = FeatureCmd("changelog") // Embedded changelog display
)
Default Behavior¶
props.Tool automatically handles default feature states. IsEnabled prioritizes configured features but falls back to built-in defaults if no explicit configuration is found.
pkg/props defines a standard set of features enabled by default:
- update
- init
- mcp
- docs
- doctor
- changelog
The following features are opt-in (disabled by default):
- ai — AI provider configuration during init
- config — programmatic config access (config get/set/list/validate)
- telemetry — anonymous usage telemetry collection and CLI management commands
The SetFeatures Constructor¶
The preferred way to define a tool's feature set in code is using the props.SetFeatures constructor. It automatically applies all default features first, allowing you to only specify overrides:
// Returns defaults (Update, Init, Mcp, Docs, Doctor, Changelog enabled)
Features: props.SetFeatures(),
// Starts with defaults, but disables 'init' and enables 'ai'
Features: props.SetFeatures(
props.Disable(props.InitCmd),
props.Enable(props.AiCmd),
),
Enabling vs Disabling Features
To disable default features or enable optional features (like ai), use the SetFeatures helper in your tool configuration:
You can check feature status using the helper methods:
props.Tool.IsEnabled(props.AiCmd) or props.Tool.IsDisabled(props.InitCmd).
Narrow Interfaces¶
Props provides narrow role-based interfaces that *Props satisfies. When writing functions that only need a subset of Props, prefer these interfaces to declare minimal dependencies:
| Interface | Methods | Use When |
|---|---|---|
LoggerProvider |
GetLogger() |
You only need logging |
ConfigProvider |
GetConfig() |
You only need configuration |
FileSystemProvider |
GetFS() |
You only need filesystem access |
AssetProvider |
GetAssets() |
You only need embedded assets |
VersionProvider |
GetVersion() |
You only need version info |
ErrorHandlerProvider |
GetErrorHandler() |
You only need error handling |
ToolMetadataProvider |
GetTool() |
You only need tool metadata |
TelemetryProvider |
GetCollector() |
You only need the telemetry collector |
LoggingConfigProvider |
GetLogger(), GetConfig() |
You need logging + config |
CoreProvider |
GetLogger(), GetConfig(), GetFS() |
You need the three most common capabilities |
Example¶
// Before: opaque dependency on all of Props
func generateDocs(p *props.Props) error { ... }
// After: declares exactly what it needs
func generateDocs(p props.LoggingConfigProvider) error {
p.GetLogger().Info("generating docs")
dir := p.GetConfig().GetString("docs.output_dir")
...
}
Migration is optional and incremental — *Props continues to work everywhere.
Components¶
Tool Metadata¶
The Tool struct contains essential information about your CLI tool:
type Tool struct {
Name string `json:"name" yaml:"name"`
Summary string `json:"summary" yaml:"summary"`
Description string `json:"description" yaml:"description"`
Features []Feature `json:"features" yaml:"features"`
ReleaseSource ReleaseSource `json:"release_source" yaml:"release_source"`
Help errorhandling.HelpConfig `json:"-" yaml:"-"`
// InstallHint is shown when a feature needs a full release binary the
// running binary lacks (e.g. embedded docs after `go install`). Set it to
// your tool's recommended install command; empty falls back to a generic
// message referencing Name.
InstallHint string `json:"install_hint,omitempty" yaml:"install_hint,omitempty"`
}
// ReleaseSource identifies where the tool's releases are hosted.
type ReleaseSource struct {
Type string `json:"type" yaml:"type"` // "github" or "gitlab"
Host string `json:"host" yaml:"host"` // Custom host (e.g., self-hosted GitLab)
Owner string `json:"owner" yaml:"owner"` // Organisation or user
Repo string `json:"repo" yaml:"repo"` // Repository name
Private bool `json:"private" yaml:"private"` // Whether the repository is private
}
// Feature represents the configuration state of a feature (Enabled/Disabled).
type Feature struct {
Cmd FeatureCmd `json:"cmd" yaml:"cmd"`
Enabled bool `json:"enabled" yaml:"enabled"`
}
// FeatureState is a functional option used to mutate the feature list.
type FeatureState func([]Feature) []Feature
Help Configuration
Tool.Help accepts any value that implements the errorhandling.HelpConfig interface (SupportMessage() string). Use errorhandling.SlackHelp or errorhandling.TeamsHelp for built-in support channel messages, or pass nil for no help message. The field is set programmatically — it is not read from YAML/JSON config files.
Example:
p := &props.Props{
Tool: props.Tool{
Name: "awesome-cli",
Summary: "An awesome command-line tool",
Description: "A comprehensive CLI tool for managing awesome things",
ReleaseSource: props.ReleaseSource{
Type: "github",
Owner: "mycompany",
Repo: "awesome-cli",
},
Features: props.SetFeatures(
props.Enable(props.AiCmd),
),
},
// ... other fields
}
// Set the help channel after constructing Props
p.Tool.Help = errorhandling.SlackHelp{
Channel: "#support",
Team: "myteam",
}
Version Information¶
Version tracking for updates and display. The Version field on Props uses the version.Version interface from pkg/version:
[!NOTE] See pkg.go.dev/gitlab.com/phpboyscout/go-tool-base/pkg/props for the full API definition.
Example:
Logger Configuration¶
Structured logging with configurable output via the unified logger package:
l := logger.NewCharm(os.Stderr,
logger.WithCaller(),
logger.WithTimestamp(),
logger.WithLevel(logger.InfoLevel),
)
Log Levels:
logger.DebugLevel- Detailed debugging informationlogger.InfoLevel- General informationlogger.WarnLevel- Warning messageslogger.ErrorLevel- Error messageslogger.FatalLevel- Fatal messages
Filesystem Abstraction¶
The FS field uses the afero library for filesystem abstraction, enabling easy testing:
import "github.com/spf13/afero"
// Production: real filesystem
FS: afero.NewOsFs()
// Testing: in-memory filesystem
FS: afero.NewMemMapFs()
Embedded Assets¶
The Assets field holds a wrapper for embedded filesystems (configurations, templates, etc.):
//go:embed assets/*
var assets embed.FS
props := &props.Props{
Assets: props.NewAssets(props.AssetMap{"root": &assets}),
}
Subcommands can register their own assets:
Usage Patterns¶
Basic Initialization¶
func NewCmdRoot(v version.Info) (*cobra.Command, *props.Props) {
l := logger.NewCharm(os.Stderr,
logger.WithTimestamp(),
logger.WithLevel(logger.InfoLevel),
)
p := &props.Props{
Tool: props.Tool{
Name: "mytool",
Summary: "My CLI tool",
Description: "Does amazing things",
ReleaseSource: props.ReleaseSource{
Type: "github",
Owner: "myorg",
Repo: "mytool",
},
},
Logger: l,
Assets: props.NewAssets(props.AssetMap{"root": &assets}),
FS: afero.NewOsFs(),
Version: v,
}
p.ErrorHandler = errorhandling.New(l, p.Tool.Help)
rootCmd := root.NewCmdRoot(p)
return rootCmd, p
}
Passing to Custom Commands¶
func NewCustomCommand(props *props.Props) *cobra.Command {
cmd := &cobra.Command{
Use: "custom",
Short: "A custom command",
RunE: func(cmd *cobra.Command, args []string) error {
return runCustomCommand(cmd.Context(), props)
},
}
return cmd
}
func runCustomCommand(ctx context.Context, props *props.Props) error {
props.Logger.Info("Running custom command")
data, err := afero.ReadFile(props.FS, "data.txt")
if err != nil {
return errors.Wrap(err, "failed to read data file")
}
props.Logger.Info("Command completed successfully")
return nil
}
Configuration Integration¶
func runDatabaseCommand(ctx context.Context, props *props.Props) error {
dbHost := props.Config.GetString("database.host")
dbPort := props.Config.GetInt("database.port")
props.Logger.Info("Connecting to database", "host", dbHost, "port", dbPort)
return nil
}
func NewDatabaseCommand(props *props.Props) *cobra.Command {
return &cobra.Command{
Use: "database",
Short: "Database operations",
RunE: func(cmd *cobra.Command, args []string) error {
return runDatabaseCommand(cmd.Context(), props)
},
}
}
Advanced Configuration¶
Conditional Features¶
Tool: props.Tool{
Name: "enterprise-tool",
Features: props.SetFeatures(
props.Disable(props.UpdateCmd), // Disable auto-updates in enterprise
),
}
Copy-on-Write Filesystem¶
import "github.com/spf13/afero"
baseFs := afero.NewReadOnlyFs(afero.NewOsFs())
overlayFs := afero.NewMemMapFs()
cowFs := afero.NewCopyOnWriteFs(baseFs, overlayFs)
props.FS = cowFs
Testing with Props¶
propstest.New (recommended)¶
The pkg/props/propstest package — public, so tools built on GTB can use it too — distils the common "construct a fully-wired *props.Props" pattern into a single call. Every field gets a hermetic, safe default, so the documented invariants (notably non-nil Collector and a usable Config) hold without hand-assembly:
import "gitlab.com/phpboyscout/go-tool-base/pkg/props/propstest"
func TestMyCommand(t *testing.T) {
t.Parallel()
p := propstest.New() // all fields wired with safe defaults
// Override only what the test cares about:
p = propstest.New(
propstest.WithTool(props.Tool{Name: "mytool", EnvPrefix: "MYTOOL"}),
propstest.WithFS(afero.NewMemMapFs()),
)
// ... drive code that needs a *props.Props ...
}
Defaults applied by propstest.New:
| Field | Default |
|---|---|
Logger |
logger.NewNoop() |
FS |
afero.NewMemMapFs() (in-memory, isolated) |
Collector |
props.NoopCollector{} (upholds the non-nil invariant) |
ErrorHandler |
errorhandling.New(...) with an inert Exit and io.Discard writer — a Fatal under test never terminates the process |
Tool |
benign valid metadata (testtool, EnvPrefix: TESTTOOL, a GitHub ReleaseSource) |
Version |
deterministic version.NewInfo("v0.0.0-test", ...) |
Assets |
empty-but-valid props.NewAssets() |
Config |
empty-but-usable config.NewReaderContainer(fs) — Get* is always safe |
Each call returns a fresh, independent instance with no real filesystem, network, keychain or os.Exit side effects, so it is safe under t.Parallel(). Override options are: WithTool, WithLogger, WithFS, WithCollector, WithVersion, WithAssets, WithConfig, and WithErrorHandler.
Manual construction¶
For full control you can still assemble a Props literal directly. Remember to set Collector: props.NoopCollector{} so the non-nil invariant holds:
func createTestProps() *props.Props {
l := logger.NewNoop()
memFs := afero.NewMemMapFs()
return &props.Props{
Tool: props.Tool{
Name: "test-tool",
Summary: "Test tool",
},
Logger: l,
FS: memFs,
Version: version.NewInfo("0.0.0-test", "", ""),
Collector: props.NoopCollector{},
}
}
Best Practices¶
1. Use ReleaseSource for Repository Identity¶
ReleaseSource is the single source of truth for where the tool's releases are hosted. It supports both GitHub and GitLab:
// GitHub
ReleaseSource: props.ReleaseSource{
Type: "github",
Owner: "your-org",
Repo: "tool-name",
},
// GitLab (including self-hosted)
ReleaseSource: props.ReleaseSource{
Type: "gitlab",
Host: "gitlab.example.com", // Optional: defaults to gitlab.com
Owner: "your-group",
Repo: "tool-name",
Private: true, // Set to true for private repositories
},
2. Consistent Tool Metadata¶
Tool: props.Tool{
Name: "kebab-case-name",
Summary: "Brief description",
Description: "Longer description that explains the tool's purpose and capabilities",
ReleaseSource: props.ReleaseSource{
Type: "github",
Owner: "your-org",
Repo: "tool-name",
},
}
3. Set Help After Construction¶
Since Tool.Help is an interface (not serializable), assign it programmatically after building Props:
p := &props.Props{Tool: props.Tool{...}}
p.Tool.Help = errorhandling.SlackHelp{Team: "Platform", Channel: "#help"}
p.ErrorHandler = errorhandling.New(l, p.Tool.Help)
The Props component provides a robust foundation for building maintainable and testable CLI applications with GTB.