Documentation
¶
Overview ¶
Package framework is the public surface of the GoFastr framework.
It contains the App spine — App, Plugin, Registry, lifecycle, typed hooks, the in-memory test harness — plus thin re-exports of every subpackage's public API so callers can keep writing framework.Entity, framework.NewCrudHandler, framework.AutoMigrate, etc.
The actual implementations live in subpackages:
- framework/entity Entity model, columns, relations, validators
- framework/crud HTTP CRUD handler, eager loading, includes, typed query, MCP tool generator
- framework/hook HookRegistry + lifecycle constants
- framework/event EventBus + Event types
- framework/migrate AutoMigrate + DiffSchema + Dialect detection
- framework/openapi EntityOpenAPI spec generator
- framework/dsl ?dsl= query parser
- framework/filter query-string filter & sort parsing
- framework/pagination cursor + offset paging
- framework/tenant multi-tenancy (TenantConfig, TenantMiddleware)
- framework/softdelete soft-delete helpers
- framework/access RBAC (Permission, Policy, RolePolicy)
- framework/file FileField upload helpers
- framework/cron in-process cron scheduler
- framework/slowquery SlowQueryLogger DBExecutor wrapper
- framework/db shared Executor + tx context primitives
Both surfaces are first-class: the facade re-exports give callers short, one-import access (`framework.Entity`, `framework.AutoMigrate`), while the narrow subpackages give plugin authors and codegen tools a precise dependency graph.
See framework/ARCHITECTURE.md for the layering rules, cycle-breaking interfaces, and the recipe for extracting a new subpackage.
Index ¶
- Constants
- Variables
- func DefaultMiddleware(a *App) []router.Middleware
- func EnsureAuditTable(db *sql.DB, table string) error
- func GetAs[T any](bm *BatteryManager, name string) (T, error)
- func NewTypedQuery[T any](h *crud.CrudHandler) *crud.TypedQuery[T]
- func OnAfterCreate[T any](app *App, name string, fn func(ctx context.Context, value *T) error)
- func OnAfterDelete(app *App, name string, fn func(ctx context.Context, id string) error)
- func OnAfterGet(app *App, name string, fn func(ctx context.Context, p *hook.GetPayload) error)
- func OnAfterList(app *App, name string, fn func(ctx context.Context, p *hook.ListPayload) error)
- func OnAfterUpdate[T any](app *App, name string, fn func(ctx context.Context, value *T) error)
- func OnBeforeCreate[T any](app *App, name string, fn func(ctx context.Context, value *T) error)
- func OnBeforeDelete(app *App, name string, fn func(ctx context.Context, id string) error)
- func OnBeforeGet(app *App, name string, fn func(ctx context.Context, p *hook.GetPayload) error)
- func OnBeforeList(app *App, name string, fn func(ctx context.Context, p *hook.ListPayload) error)
- func OnBeforeUpdate[T any](app *App, name string, fn func(ctx context.Context, value *T) error)
- func TxFromContext(ctx context.Context) (*sql.Tx, bool)
- type AccessControl
- type App
- func (a *App) AddCron(s *cron.Scheduler) *App
- func (a *App) AddQueue(q schedulerStartStop) *App
- func (a *App) CrudHandler(name string) (*crud.CrudHandler, error)
- func (a *App) EntitiesFromDir(dir string) error
- func (a *App) Entity(name string, config entity.EntityConfig) *App
- func (a *App) EntityFromFile(path string) (*entity.Entity, error)
- func (a *App) Events() *event.EventBus
- func (a *App) Flags() *featureflag.Evaluator
- func (a *App) Group(prefix string, opts ...routegroup.GroupOption) *routegroup.RouteGroup
- func (a *App) GroupEntitiesFromDir(g *routegroup.RouteGroup, dir string) error
- func (a *App) GroupEntity(g *routegroup.RouteGroup, name string, config entity.EntityConfig) *App
- func (a *App) HookRegistry(entityName string) *hook.HookRegistry
- func (a *App) InTx(ctx context.Context, fn func(ctx context.Context, tx *sql.Tx) error) error
- func (a *App) InitPlugins() error
- func (a *App) IsEnabled(ctx context.Context, key string) bool
- func (a *App) JSONCasing() crud.JSONCase
- func (a *App) Lifecycle() *lifecycle.Lifecycle
- func (a *App) Logger() *slog.Logger
- func (a *App) Mount(m Mountable) *App
- func (a *App) Mountables() []Mountable
- func (a *App) MustCrudHandler(name string) *crud.CrudHandler
- func (a *App) OnStart(fn func(ctx context.Context) error) *App
- func (a *App) OnStop(fn func() error) *App
- func (a *App) OnStopFirst(fn func() error) *App
- func (a *App) RegisterBattery(b Battery, deps ...string) *App
- func (a *App) RegisterEntities(entities map[string]entity.EntityConfig) *App
- func (a *App) RegisterPlugin(plugin Plugin) *App
- func (a *App) RegisterReadiness(name string, check func(ctx context.Context) error) *App
- func (a *App) Router() *router.Router
- func (a *App) Routine(r migrate.Routine) *App
- func (a *App) RunWithSignals(ctx context.Context) error
- func (a *App) SetFlagStore(s featureflag.Store) *App
- func (a *App) SetLogger(l *slog.Logger)
- func (a *App) Shutdown(ctx context.Context) error
- func (a *App) Start(addr string) error
- func (a *App) T(ctx context.Context, key string, params ...map[string]any) string
- func (a *App) Table(t migrate.Table) *App
- func (a *App) Translator() *i18n.Translator
- func (a *App) TryEntity(name string, config entity.EntityConfig) (err error)
- func (a *App) Use(mw ...router.Middleware) *App
- func (a *App) View(v migrate.View) *App
- func (a *App) WithAuditLog(cfg AuditConfig) *App
- type AppConfig
- type AppOption
- func WithAPIPrefix(prefix string) AppOption
- func WithConfig(config AppConfig) AppOption
- func WithDB(db *sql.DB) AppOption
- func WithFileStorage(s upload.Storage) AppOption
- func WithI18n(tr *i18n.Translator) AppOption
- func WithIdempotency(cfg middleware.IdempotencyConfig) AppOption
- func WithLogger(l *slog.Logger) AppOption
- func WithMCPIntrospection() AppOption
- func WithMCPServer(s *mcp.Server) AppOption
- func WithMetrics() AppOption
- func WithPublicOpenAPI() AppOption
- func WithReadinessTimeout(d time.Duration) AppOption
- func WithRouter(r *router.Router) AppOption
- func WithTracing() AppOption
- func WithVerboseReadiness() AppOption
- func WithoutDefaultMiddleware() AppOption
- type ApplyOptions
- type AuditConfig
- type Battery
- type BatteryLifecycle
- type BatteryManager
- func (bm *BatteryManager) All() []Battery
- func (bm *BatteryManager) Get(name string) (Battery, error)
- func (bm *BatteryManager) InitAll(app *App) error
- func (bm *BatteryManager) Names() []string
- func (bm *BatteryManager) Register(b Battery, deps ...string) error
- func (bm *BatteryManager) StartAll(ctx context.Context) error
- func (bm *BatteryManager) StopAll(ctx context.Context) error
- type BoolColumn
- type Column
- type Condition
- type CronJob
- type CrudHandler
- type DBExecutor
- type DestructiveChangeError
- type Dialect
- type Endpoint
- type Entity
- type EntityConfig
- type EntityDeclaration
- type Event
- type EventBus
- type EventHandler
- type FieldDeclaration
- type FloatColumn
- type ForeignKey
- type GroupOption
- type HookFunc
- type HookGetPayload
- type HookListPayload
- type HookRegistry
- type HookType
- type HookWhereClause
- type IncludeNode
- type Index
- type IntColumn
- type JSONCase
- type ListOptions
- type ListResponse
- type Metrics
- type MigrationPlan
- type Mountable
- type Order
- type Permission
- type Plugin
- type PluginManager
- type Policy
- type ReadinessCheck
- type ReadinessRegistrar
- type ReadinessResponse
- type ReadinessResult
- type Registry
- type Relation
- type RelationType
- type RolePolicy
- type RouteGroup
- type Routine
- type Scheduler
- type SchemaChange
- type SchemaSnapshot
- type StringColumn
- type Table
- type TenantConfig
- type TestApp
- func (ta *TestApp) Close()
- func (ta *TestApp) Delete(path string) *TestResponse
- func (ta *TestApp) Get(path string) *TestResponse
- func (ta *TestApp) Post(path string, body any) *TestResponse
- func (ta *TestApp) Put(path string, body any) *TestResponse
- func (ta *TestApp) Request(method, path string, body io.Reader) *TestRequest
- type TestRequest
- type TestResponse
- func (tr *TestResponse) AssertBodyContains(t testing.TB, substr string) *TestResponse
- func (tr *TestResponse) AssertHeader(t testing.TB, key, expected string) *TestResponse
- func (tr *TestResponse) AssertJSON(t testing.TB, expected any) *TestResponse
- func (tr *TestResponse) AssertStatus(t testing.TB, expected int) *TestResponse
- func (tr *TestResponse) Body() string
- func (tr *TestResponse) Close()
- func (tr *TestResponse) JSON(v any) error
- func (tr *TestResponse) Status() int
- type TimestampColumn
- type TypedQuery
- type UUIDColumn
- type ValidationRegistry
- type ValidatorFunc
- type View
Constants ¶
const ( CaseCamel = crud.CaseCamel CaseSnake = crud.CaseSnake MaxBatchSize = crud.MaxBatchSize MaxMultipartMemory = crud.MaxMultipartMemory )
const ( RelHasOne = entity.RelHasOne RelHasMany = entity.RelHasMany RelManyToOne = entity.RelManyToOne RelManyToMany = entity.RelManyToMany )
const ( EntityCreated = event.EntityCreated EntityUpdated = event.EntityUpdated EntityDeleted = event.EntityDeleted )
const ( BeforeCreate = hook.BeforeCreate AfterCreate = hook.AfterCreate BeforeUpdate = hook.BeforeUpdate AfterUpdate = hook.AfterUpdate BeforeDelete = hook.BeforeDelete AfterDelete = hook.AfterDelete BeforeList = hook.BeforeList AfterList = hook.AfterList BeforeGet = hook.BeforeGet AfterGet = hook.AfterGet )
const ( DialectPostgres = migrate.DialectPostgres DialectSQLite = migrate.DialectSQLite )
Variables ¶
var ( NewRolePolicy = access.NewRolePolicy RequirePermission = access.RequirePermission GetPermissions = access.GetPermissions WithPolicy = access.WithPolicy WithRoles = access.WithRoles // Can reports whether the request context carries a permission. Can = access.Can // AccessMiddleware installs the RBAC policy + roles into request context // so RequirePermission and EntityConfig.Access gates can resolve. AccessMiddleware = access.Middleware )
var ( NewCrudHandler = crud.NewCrudHandler RegisterCrudRoutes = crud.RegisterCrudRoutes RegisterCrudRoutesFunc = crud.RegisterCrudRoutesFunc MarshalEntity = crud.MarshalEntity UnmarshalEntity = crud.UnmarshalEntity IsNotFound = crud.IsNotFound EagerLoad = crud.EagerLoad RegisterEntityMCPTools = crud.RegisterEntityMCPTools )
var ( Define = entity.Define LoadEntityDeclaration = entity.LoadEntityDeclaration LoadEntityDeclarations = entity.LoadEntityDeclarations HasOne = entity.HasOne HasMany = entity.HasMany BelongsTo = entity.BelongsTo ManyToMany = entity.ManyToMany NewStringColumn = entity.NewStringColumn NewIntColumn = entity.NewIntColumn NewFloatColumn = entity.NewFloatColumn NewBoolColumn = entity.NewBoolColumn NewTimestampColumn = entity.NewTimestampColumn NewUUIDColumn = entity.NewUUIDColumn NewValidationRegistry = entity.NewValidationRegistry Required = entity.Required Unique = entity.Unique Custom = entity.Custom FormatValidationErrors = entity.FormatValidationErrors And = entity.And Or = entity.Or Not = entity.Not )
var ( AutoMigrate = migrate.AutoMigrate AutoMigrateContext = migrate.AutoMigrateContext AutoMigratePlanContext = migrate.AutoMigratePlanContext MigrateEntity = migrate.MigrateEntity MigrateEntityDialect = migrate.MigrateEntityDialect DiffSchema = migrate.DiffSchema ApplySchemaDiff = migrate.ApplySchemaDiff ApplySchemaDiffWithOptions = migrate.ApplySchemaDiffWithOptions DetectDialect = migrate.DetectDialect GenerateMigration = migrate.GenerateMigration GeneratePlan = migrate.GeneratePlan SnapshotFromRegistry = migrate.SnapshotFromRegistry SnapshotFromPlan = migrate.SnapshotFromPlan RenderMigrationFile = migrate.RenderMigrationFile LoadSnapshot = migrate.LoadSnapshot SaveSnapshot = migrate.SaveSnapshot )
var ( NewMetrics = middleware.NewMetrics MetricsMiddleware = middleware.MetricsMiddleware MetricsHandler = middleware.MetricsHandler Tracing = middleware.Tracing )
var ( DefaultTenantConfig = tenant.DefaultTenantConfig WithMultiTenant = tenant.WithMultiTenant ApplyTenantFilter = tenant.ApplyTenantFilter TenantMiddleware = tenant.TenantMiddleware SetTenantID = tenant.SetTenantID GetTenantID = tenant.GetTenantID InjectTenantID = tenant.InjectTenantID )
var EntityOpenAPI = openapi.EntityOpenAPI
var NewEventBus = event.NewEventBus
var NewHookRegistry = hook.NewHookRegistry
var NewScheduler = cron.NewScheduler
Functions ¶
func DefaultMiddleware ¶
func DefaultMiddleware(a *App) []router.Middleware
DefaultMiddleware is the framework's standard safety chain in canonical order:
recovery → request-id → [idempotency] → [i18n] → security headers → timeout
The optional entries are present when the App was configured with WithIdempotency / WithI18n; the timeout entry is omitted when AppConfig.DisableRequestTimeout is true.
Access logging is deliberately NOT in this list. battery/log owns structured access logging when registered, and ad-hoc apps that just want a basic line can add middleware.LoggingFn(app.Logger) themselves — having both fire produces duplicate entries with mismatched fields (`request` from the framework, `http.access` from the plugin).
Takes the App so the recovery middleware can route panics through app.Logger (late-binding) and the timeout reflects AppConfig.RequestTimeout. Pass nil only in tests; the recovery falls back to slog.Default and the timeout to 30s.
func EnsureAuditTable ¶
EnsureAuditTable creates the audit_log table if it does not exist. Idempotent. Dialect-aware via the existing migrate.DetectDialect helper.
func GetAs ¶
func GetAs[T any](bm *BatteryManager, name string) (T, error)
GetAs retrieves a battery by name and type-asserts it to T. Returns an error if the battery is not found or doesn't implement T.
func NewTypedQuery ¶
func NewTypedQuery[T any](h *crud.CrudHandler) *crud.TypedQuery[T]
func OnAfterCreate ¶
OnAfterCreate registers a typed AfterCreate hook. The callback receives the just-inserted row (already includes server-generated fields like id). Mutations are not reflected — Create has already committed the row's shape; modifying the struct is harmless but pointless.
func OnAfterDelete ¶
OnAfterDelete registers a typed AfterDelete hook. Same shape as OnBeforeDelete.
func OnAfterGet ¶
OnAfterGet registers a typed AfterGet hook. The callback receives *hook.GetPayload with Result populated — mutate the map in place to redact fields before the response is serialised.
func OnAfterList ¶
OnAfterList registers a typed AfterList hook. The callback receives the *hook.ListPayload with Results populated — mutate the slice in place to redact / drop rows.
func OnAfterUpdate ¶
OnAfterUpdate registers a typed AfterUpdate hook receiving the post-update row.
func OnBeforeCreate ¶
OnBeforeCreate registers a typed BeforeCreate hook on the entity named `name`. Mutations the callback makes to *T are reflected back into the pending body so the subsequent INSERT picks them up.
func OnBeforeDelete ¶
OnBeforeDelete registers a typed BeforeDelete hook. The payload is the record id; no generic parameter needed.
func OnBeforeGet ¶
OnBeforeGet registers a typed BeforeGet hook. The callback receives *hook.GetPayload; ID is the request's path-value and AddWhere lets the host scope the lookup (mismatch → 404).
func OnBeforeList ¶
OnBeforeList registers a typed BeforeList hook. The callback receives the framework's *hook.ListPayload directly so callers can append WHERE clauses (p.AddWhere) without type-asserting from any. Symmetric with OnBeforeCreate/OnBeforeUpdate.
func OnBeforeUpdate ¶
OnBeforeUpdate registers a typed BeforeUpdate hook. *T is sparse — it holds whatever the caller sent, not the full row.
Types ¶
type AccessControl ¶
type AccessControl = entity.AccessControl
type App ¶
type App struct {
Registry *Registry
MCP *mcp.Server
DB *sql.DB
Config AppConfig
Plugins *PluginManager
Storage upload.Storage // optional; enables multipart on Image/File fields
Batteries *BatteryManager
// contains filtered or unexported fields
}
App is the top-level application container. It wires together the entity registry, router, MCP server, and database.
func NewApp ¶
NewApp creates a new App with the given options. It initializes default Registry, Router, and MCP Server if not provided.
func NewUIHostApp ¶
NewUIHostApp builds an App and mounts the given host on it in one call — the near-universal shape for SSR/UIHost apps, which otherwise repeat
app := framework.NewApp(opts...) app.Mount(host)
host is any Mountable (typically a *uihost.Host). Returns the App for fluent chaining.
func (*App) AddCron ¶
AddCron registers a Scheduler with the app's lifecycle: it starts when Start runs and stops when Stop runs. Returns the App for chaining so users can wire several schedulers in one expression.
func (*App) AddQueue ¶
AddQueue registers any queue/worker that exposes Start(ctx) and Close(). The DBQueue from battery/queue satisfies this directly; in-memory and Redis variants can be wrapped.
func (*App) CrudHandler ¶
func (a *App) CrudHandler(name string) (*crud.CrudHandler, error)
CrudHandler returns a fully-wired in-process CRUD handler for a registered entity — the same handler shape the HTTP routes use (hooks, events, storage, JSON casing, registry). Use it to call CreateOne/UpdateOne/DeleteOne/ListAll directly, e.g. to compose several writes inside App.InTx (pass the InTx ctx so they join the same transaction). Returns an error if no entity is registered under name or the app has no DB.
func (*App) EntitiesFromDir ¶
EntitiesFromDir loads and registers every *.json declaration in dir.
func (*App) Entity ¶
func (a *App) Entity(name string, config entity.EntityConfig) *App
Entity registers an entity with the given name and configuration. Returns the App for fluent chaining. Panics on any misconfiguration — convenient for static, hand-written declarations where a bad config is a programming error you want to fail fast on. For generated or untrusted configs (e.g. an AI-authored field, a dynamic schema) where one bad entity should not crash the process, use TryEntity, which returns the error.
func (*App) EntityFromFile ¶
EntityFromFile loads and registers one JSON entity declaration.
func (*App) Flags ¶
func (a *App) Flags() *featureflag.Evaluator
Flags returns the app's feature-flag evaluator, creating one on first call. The default backing store is in-memory; for clustered deployments call SetFlagStore with a Redis- or DB-backed implementation BEFORE any caller invokes Flags — once the lazy default fires, subsequent SetFlagStore calls panic to avoid the silent race where some goroutines still hold a reference to the previous evaluator.
The evaluator is also installed as featureflag.Default() so package- level featureflag.Bool(ctx, "...") calls work from anywhere in the app.
func (*App) Group ¶
func (a *App) Group(prefix string, opts ...routegroup.GroupOption) *routegroup.RouteGroup
Group creates a route group with the given prefix and optional configuration. The group supports its own middleware stack, access policy, OpenAPI tags, and MCP namespacing. Nested groups compose prefixes and middleware.
api := app.Group("/api")
api.Use(authMiddleware)
api.Get("/health", healthHandler)
admin := app.Group("/admin", routegroup.WithAccess(access.RequirePermission("admin:access")))
admin.Entity("settings", settingsConfig)
func (*App) GroupEntitiesFromDir ¶
func (a *App) GroupEntitiesFromDir(g *routegroup.RouteGroup, dir string) error
GroupEntitiesFromDir loads every *.json declaration in dir and registers each one inside the given RouteGroup — the group-scoped equivalent of EntitiesFromDir. CRUD routes mount at <group-prefix>/<entity-table>, MCP tools are namespaced under the group's MCPNamespace.
Use this with a /api group when every JSON entity should live behind a single prefix:
api := app.Group("/api")
app.GroupEntitiesFromDir(api, "entities")
func (*App) GroupEntity ¶
func (a *App) GroupEntity(g *routegroup.RouteGroup, name string, config entity.EntityConfig) *App
GroupEntity registers an entity with the given configuration inside a RouteGroup. CRUD routes mount at <group-prefix>/<entity-table>, MCP tools are namespaced under the group's MCPNamespace, and the OpenAPI tag reflects the group's OpenAPITag if set.
This is the group-scoped equivalent of App.Entity.
func (*App) HookRegistry ¶
func (a *App) HookRegistry(entityName string) *hook.HookRegistry
HookRegistry returns (or creates) the hook registry for a named entity.
func (*App) InTx ¶
InTx runs fn inside a database transaction opened on the App's DB. The inner context carries the *sql.Tx so any code path that calls TxFromContext (typed hooks, the various do* helpers, generated repo methods invoked via WithTx) participates atomically.
Convenience wrapper for callers that aren't already inside a CRUD hook — e.g. seeders, batch jobs, multi-entity write paths that need an explicit boundary. If fn returns an error, the tx rolls back and that error is returned unchanged.
func (*App) InitPlugins ¶
InitPlugins initializes all registered plugins and batteries by calling their Init(app) method. Plugins go first (registration order), then batteries (dependency-resolved order). Each module does everything it needs from inside Init — register routes, add middleware, register MCP tools, attach hooks, swap the logger, etc.
Idempotent: the first successful call latches an internal flag so any later call returns nil without re-running plugin Inits. This lets tests call InitPlugins() manually pre-Start without colliding with the implicit call inside Start.
func (*App) IsEnabled ¶
IsEnabled is a convenience wrapper that calls Flags().Bool with the supplied context. Use it from handler code that doesn't otherwise need a direct reference to the evaluator.
The context is expected to already carry an EvalContext (via featureflag.WithContext) when user/tenant gating matters; without one, the call still works but only the kill-switch (Flag.Enabled) and uniform rollout are consulted.
func (*App) JSONCasing ¶
JSONCasing returns the configured JSON casing strategy. Defaults to CaseCamel if not explicitly set.
func (*App) Lifecycle ¶
Lifecycle returns the App's graceful-shutdown coordinator. Batteries and plugins use Lifecycle().RegisterDrainer / RegisterHealthChecker to participate in Shutdown beyond the simple OnStop hook.
func (*App) Logger ¶
Logger returns the App-local *slog.Logger. Middleware and plugins should call this — not slog.Default() — so that a logging plugin can replace the destination without rewiring globals.
Always non-nil. NewApp seeds the App with a JSON-to-stderr logger that is independent of slog.Default; an unrelated slog.SetDefault elsewhere in the process does not redirect this App's logs.
func (*App) Mount ¶
Mount attaches a Mountable and registers its routes on the app's router immediately. The default middleware chain is already in place (committed during NewApp), so any handler the Mountable registers is wrapped with it. Returns the app for fluent chaining.
Mountables typically register a NotFound catch-all (e.g. a UI host that renders pages for any unrouted path), so call Mount AFTER any explicit routes you want to take precedence (entity CRUD, custom endpoints).
IMPORTANT — ordering with plugins/batteries: plugin.Init runs at App.Start (or InitPlugins), AFTER Mount has already registered the Mountable's routes. If a plugin's Init registers a more-specific route that overlaps with a Mountable's NotFound catch-all, the plugin's route still wins (ServeMux dispatches by specificity, not by registration order). But if a plugin registers a Mountable-style catch-all itself, it shadows any user routes added after Mount but before InitPlugins. Mount last unless you know what you're doing.
func (*App) Mountables ¶
Mountables returns the Mountables registered via Mount, in registration order. Batteries use it to discover a mounted UI host (type-asserting to *uihost.Host) so they can register screens on the host's render pipeline rather than spinning up a second host. Returns a copy — callers must not mutate the App's internal slice.
func (*App) MustCrudHandler ¶
func (a *App) MustCrudHandler(name string) *crud.CrudHandler
MustCrudHandler is CrudHandler that panics on error — for app setup where a missing entity is a programming mistake.
func (*App) OnStart ¶
OnStart registers a function to run once during App.Start, before the HTTP server begins accepting connections. The context passed in is cancelled when Stop is called, so workers should respect it.
Hooks run in registration order; the first to return a non-nil error aborts Start.
func (*App) OnStop ¶
OnStop registers a function to run during App.Shutdown, after the HTTP server has shut down. Hooks run in reverse registration order — the last thing started is the first thing stopped. Internally the hook is wrapped as a lifecycle.Drainer so app-level cleanup and battery drains share one coordinator.
func (*App) OnStopFirst ¶
OnStopFirst registers an OnStop hook that runs LAST under the reverse-order Stop iteration. Useful for plugins (battery/log especially) that must outlive every other shutdown step: their close hook needs to fire AFTER every other OnStop has had a chance to emit log entries.
Without this, a user that registers app.OnStop BEFORE RegisterPlugin(log) gets the order inverted on reverse iteration — log's close runs first, the user's OnStop logs into closed sinks.
func (*App) RegisterBattery ¶
RegisterBattery registers a heavyweight, lifecycle-aware battery module (auth, search, cache, etc.) with the application. deps lists battery names that must be initialized before this one. Returns the App for chaining.
Batteries are initialized in dependency-resolved order during App.Start, before the HTTP server binds. Each battery's Init does whatever it needs by calling into the App (routes, middleware, hooks, MCP tools). Batteries that also implement BatteryLifecycle get their OnStart/OnStop fired by the App at the appropriate moment.
Example:
app.RegisterBattery(auth.New(auth.Config{...}), "search") // depends on search battery
func (*App) RegisterEntities ¶
func (a *App) RegisterEntities(entities map[string]entity.EntityConfig) *App
RegisterEntities registers each (name, config) pair via App.Entity in alphabetical-by-name order. Sorting matters: Entity has order-sensitive side effects — router registration, MCP tool list order, OpenAPI tag emission — and Go's map iteration is randomised, so unsorted iteration would mean non-deterministic /openapi.json bytes across restarts (breaking ETag caching) and non-deterministic MCP tools/list responses. FK relations stay safe because AutoMigrate also topologically sorts.
Returns the App for fluent chaining.
app.RegisterEntities(map[string]entity.EntityConfig{
"foods": foodsConfig,
"meals": mealsConfig,
"users": usersConfig,
})
func (*App) RegisterPlugin ¶
RegisterPlugin registers a plugin with the application's plugin manager. Returns the App for fluent chaining.
Panics if InitPlugins has already run — plugins must be registered before App.Start (or the explicit InitPlugins call) so their Init fires. The panic is a clear contract violation rather than a silent no-op that would have the new plugin's routes / middleware vanish.
func (*App) RegisterReadiness ¶
RegisterReadiness adds a readiness check. Names are not required to be unique — duplicates surface in the /readyz response as repeated rows, which is occasionally useful (the same backend probed at two layers).
Pass a timeout on the context if your check could hang; /readyz also applies an overall deadline, but per-check timeouts give better signal in the response.
func (*App) Router ¶
Router returns the App's *router.Router for advanced use (plugin authors, batteries that need to register routes with custom matching, sub-router construction). Application code should prefer the App-level helpers:
- App.Use(mw) — register middleware (instead of Router().Use)
- App.Get/Post/... — register routes (instead of Router().Handle)
- App.Group(prefix) — sub-routes (instead of Router().Group)
Both forms are functionally equivalent — App.Use forwards to Router().Use — but the App-level surface is the canonical one in docs and examples.
Exposed as a method (rather than a field) so plugins and batteries can swap or wrap the router during Init without callers depending on direct field assignment.
func (*App) Routine ¶
Routine registers a stored routine (function, procedure, trigger, or view) as a first-class migration object. Its Up runs on every boot (idempotent CREATE OR REPLACE) after tables are migrated, and `migrate generate` tracks it for reversible versioned migrations. Returns App for chaining.
func (*App) RunWithSignals ¶
RunWithSignals blocks until SIGINT or SIGTERM is received, then runs Shutdown. Returns Shutdown's error, or nil if ctx is cancelled before a signal arrives.
func (*App) SetFlagStore ¶
func (a *App) SetFlagStore(s featureflag.Store) *App
SetFlagStore swaps the underlying store. Must be called before any caller has triggered the lazy default; a second SetFlagStore call (or any call after Flags() / IsEnabled has already been used) panics to avoid the silent race where stale references to the previous evaluator persist in handler closures.
Useful for tests that want a preconfigured store, or for production wiring that uses a persistent backend — wire it during NewApp setup, before any request runs.
func (*App) SetLogger ¶
SetLogger replaces the App's logger. Atomic; safe to call concurrently with in-flight requests — atomic.Pointer.Store is race-free, and middleware reading via App.Logger() sees the new value on the next request.
Panics if l is nil — the App's logger is always non-nil; pass a discard logger (slog.New(slog.DiscardHandler)) to silence output.
func (*App) Shutdown ¶
Shutdown gracefully stops the HTTP server, stops every registered battery in reverse dependency order, then runs each OnStop hook in reverse registration order. Matches net/http.Server.Shutdown's signature (takes a deadline ctx) but does the FULL lifecycle teardown. Safe to call multiple times — subsequent calls are no-ops.
Call this from your signal handler.
func (*App) Start ¶
Start starts the HTTP server on the given address. Auto-migrates tables, registers OpenAPI/Swagger, debug stats, applies the default middleware chain (unless disabled), and calls Mount on every attached Mountable.
Sets the process title to the app name for visibility in ps/Activity Monitor.
func (*App) T ¶
T is a convenience wrapper around the app's Translator. Returns the bare key when no Translator has been wired via WithI18n.
The ctx is expected to already carry a Locale (the i18n middleware attaches one per request from Accept-Language); when it doesn't, the Translator's fallback locale is used.
func (*App) Table ¶
Table registers a raw, non-entity table for migration only — no CRUD, no HTTP routes, no validation, no auto-injected columns. The table participates in auto-migrate, diffing, and generation alongside entities (including foreign keys that cross between the two). For users who want migration coverage of a table without the entity machinery. Returns App for chaining.
func (*App) Translator ¶
func (a *App) Translator() *i18n.Translator
Translator returns the wired Translator, or nil if WithI18n was never called. Useful for handlers that need to drive locale-aware formatting beyond simple lookups.
func (*App) TryEntity ¶
func (a *App) TryEntity(name string, config entity.EntityConfig) (err error)
TryEntity is the error-returning variant of Entity: it registers an entity and returns an error on any misconfiguration instead of panicking. It also recovers panics from deeper validation (e.g. an invalid TenantField) and converts them to errors, so a single bad config can never take down the process — the property an agent-driven authoring loop needs.
func (*App) Use ¶
func (a *App) Use(mw ...router.Middleware) *App
Use appends middleware to the app's router chain. The default chain (installed by NewApp unless WithoutDefaultMiddleware is set) stays in place — Use adds to it, never silently replaces it. Plugins call Use from their Init to contribute middleware; router late-binding means these additions also wrap routes registered before the plugin loaded.
func (*App) View ¶
View registers a database view — a virtual table built from other entities. The view is created on boot after its source tables (and tracked reversibly by `migrate generate`), and, when it declares Columns, it is also exposed through the ORM as a READ-ONLY entity: List/Get and the query layer work, but no write routes are registered. Returns App for chaining.
func (*App) WithAuditLog ¶
func (a *App) WithAuditLog(cfg AuditConfig) *App
WithAuditLog enables audit logging on every entity registered on the app (or the subset named in cfg.Entities). Call AFTER Entity registrations.
Returns the app for fluent chaining. Panics if the audit table cannot be created — this is initialization-time work so loud failure is preferable to silent log loss.
type AppConfig ¶
type AppConfig struct {
Name string // application name
JSONCase crud.JSONCase // JSON key casing: "camelCase" (default) or "snake_case"
DebugEndpoints bool // opt-in for /.debug/* endpoints
NoLLMMD bool // disable auto-generated /llm.md entity docs
// PublicOpenAPI serves /openapi.json without the auth gate. By default
// the spec is auth-gated (it enumerates every route), so a minimal app
// returns 401 there — which surprised users following the quickstart
// curl. Set true (or use WithPublicOpenAPI) when the spec is meant to be
// public, e.g. a docs site or an internal API behind a network boundary.
// The Swagger UI at /api/docs/ is always reachable; this only governs
// the raw spec JSON.
PublicOpenAPI bool
// APIPrefix mounts every auto-CRUD entity route (list/get/create/update/
// delete + _batch + _events + per-entity llm.md) under this path — e.g.
// "/api" serves GET /api/posts instead of GET /posts. Empty (default)
// keeps the bare entity-name mounts, so this is not a breaking change.
// The generated OpenAPI spec expresses the prefix via its server URL.
// GroupEntity routes are unaffected — a group owns its own prefix. MCP
// tool names are unchanged.
APIPrefix string
// RequestTimeout caps the per-request wall-clock budget enforced by
// the default middleware chain. Zero (default) installs a 30s cap.
// Set a positive duration to override. To disable the timeout
// middleware entirely, set DisableRequestTimeout — overloading
// sign for "disable" is too easy to trip on (e.g. accidentally
// subtracting two timestamps).
RequestTimeout time.Duration
// DisableRequestTimeout removes the Timeout middleware from the
// default chain entirely. Useful for long-running uploads / SSE;
// pair with per-handler ctx deadlines if you still need bounded
// request lifetime.
DisableRequestTimeout bool
}
AppConfig holds application-level configuration.
type AppOption ¶
type AppOption func(*App)
AppOption is a functional option for configuring an App.
func WithAPIPrefix ¶
WithAPIPrefix mounts auto-CRUD entity routes under prefix (e.g. "/api"). See AppConfig.APIPrefix. A leading slash is added and a trailing slash trimmed, so "api", "/api", and "/api/" all behave identically.
func WithConfig ¶
WithConfig sets the application config.
func WithFileStorage ¶
WithFileStorage sets the default upload.Storage used by CRUD handlers to persist files for Image and File entity fields when a multipart request arrives. Without this option, multipart requests on those fields fail with a clear error.
func WithI18n ¶
func WithI18n(tr *i18n.Translator) AppOption
WithI18n installs a Translator and wires its locale-negotiation middleware into the default chain. Handlers downstream can call App.T(ctx, key, ...) for translated strings driven by the caller's Accept-Language. Also installed as i18n.Default() so the package- level i18n.T helper works from anywhere.
Panics when paired with WithoutDefaultMiddleware — register the middleware explicitly in your custom chain in that case.
func WithIdempotency ¶
func WithIdempotency(cfg middleware.IdempotencyConfig) AppOption
WithIdempotency adds an Idempotency-Key middleware to the default chain. Pass middleware.IdempotencyConfig{} to take all defaults; the option is otherwise idiomatic-Go composition over the existing middleware.Idempotency primitive.
Has no effect when WithoutDefaultMiddleware is also set — wire your own chain explicitly in that case.
func WithLogger ¶
WithLogger sets the App's *slog.Logger. Same effect as calling App.SetLogger after NewApp; available as an option for symmetry. Panics if l is nil — the App's logger is always non-nil; pass a discard logger (slog.New(slog.DiscardHandler)) if you want to silence output.
func WithMCPIntrospection ¶
func WithMCPIntrospection() AppOption
WithMCPIntrospection installs a set of MCP tools that expose the running App's structure to an agent: registered routes, plugins, batteries, app config, readiness status. Opt-in because:
- The tools are read-only but cumulatively reveal a lot about the app's shape (every mounted route, every loaded module, every config field). Production apps that expose MCP to untrusted callers should weigh the disclosure surface before enabling.
- The framework should stay zero-config-friendly: nothing should show up on the MCP server unless the operator asked for it.
Tools registered:
- app_routes: list every (method, pattern) registered on the router.
- app_plugins: list registered plugins (name only).
- app_batteries: list registered batteries with deps + lifecycle status.
- app_config: return the AppConfig snapshot (Name, JSONCase, timeouts…).
- app_readiness: run every registered readiness check and report results.
- framework_docs_list / framework_docs_get / framework_docs_search: expose the framework's markdown docs (embedded at build time, so they match the framework version this binary was built against — no GitHub fetch).
func WithMCPServer ¶
WithMCPServer sets a custom MCP server.
func WithMetrics ¶
func WithMetrics() AppOption
WithMetrics enables HTTP request metrics (per-route counts, status classes, latency histograms) in the default middleware chain and mounts a Prometheus-format /metrics endpoint. The endpoint is unauthenticated by design (scrape it from inside your network / behind your ingress). Panics when paired with WithoutDefaultMiddleware — mount middleware.MetricsMiddleware and middleware.MetricsHandler yourself in that case.
func WithPublicOpenAPI ¶
func WithPublicOpenAPI() AppOption
WithPublicOpenAPI serves /openapi.json without the auth gate. Equivalent to setting AppConfig.PublicOpenAPI. Use when the spec is meant to be public (docs sites, internal APIs behind a network boundary).
func WithReadinessTimeout ¶
WithReadinessTimeout overrides the per-request /readyz deadline. Default 5 seconds. Checks that don't return within the deadline are reported as errors with status "timeout".
func WithTracing ¶
func WithTracing() AppOption
WithTracing enables the OpenTelemetry tracing middleware in the default chain. Each request runs in a span with method/route/status attributes. Spans no-op until you install a TracerProvider via otel.SetTracerProvider (e.g. an OTLP exporter), so this is safe to leave on. Panics when paired with WithoutDefaultMiddleware.
func WithVerboseReadiness ¶
func WithVerboseReadiness() AppOption
WithVerboseReadiness opts in to including each check's error.Error() string in the /readyz JSON response. Default is to omit error text because /readyz is typically reachable without authentication and raw error strings frequently leak internal IPs, connection strings, or other infrastructure detail.
Turn this on only when the probes are scoped to a trusted listener or behind auth.
func WithoutDefaultMiddleware ¶
func WithoutDefaultMiddleware() AppOption
WithoutDefaultMiddleware disables the default middleware chain (recovery, request-id, logging, security headers, timeout). Use this when you want full control over middleware composition via Use().
type ApplyOptions ¶
type ApplyOptions = migrate.ApplyOptions
type AuditConfig ¶
type AuditConfig struct {
// Table is the destination table for audit rows. Defaults to "audit_log".
Table string
// Actor resolves the actor id (typically a user id) from the request
// context. Return "" when no actor is attached (e.g. system writes).
Actor func(context.Context) string
// Entities restricts auditing to the named entities. Empty means audit
// every registered entity.
Entities []string
// Redact, when non-nil, is called with the entity name and a
// defensive copy of the row about to be serialised into the
// `diff` column. Return either the modified input (safe to mutate
// — the framework already copied it) or a fresh map. Either works.
//
// When nil, the framework applies a default sensitive-field scrub
// (see defaultSensitiveSuffixes) — fields whose names look like
// passwords, tokens, secrets, or keys are dropped from the diff so
// a host that forgot to configure Redact doesn't accidentally
// stream credentials into the audit log.
//
// Redact is invoked for AfterCreate, AfterUpdate, AND AfterDelete.
// On delete the input is `map[string]any{"id": <record_id>}`; the
// returned map's "id" value (if any) becomes the audit row's
// record_id, letting hosts pseudonymise the natural key on delete
// too. If the callback returns a map with no `id` key the framework
// falls back to the original — silently erasing the record_id is
// an audit-forensics erasure primitive we don't want to expose.
//
// Returning nil is equivalent to returning an empty map. A panic
// inside Redact is recovered: the audit row is still written using
// the original pre-redact value so audit coverage isn't lost to a
// misbehaving callback.
Redact func(entity string, row map[string]any) map[string]any
}
AuditConfig configures the audit log helper.
Audit rows are written via lifecycle hooks: AfterCreate, AfterUpdate, and AfterDelete. The hook fires inside the same transaction as the operation it audits, so partial writes are impossible — a rollback drops the audit row along with the change.
type Battery ¶
type Battery interface {
// Name returns the unique battery identifier. Used for dependency
// resolution and logging.
Name() string
// Init wires the battery into the App. Called once during App.Start,
// before lifecycle hooks fire and after any batteries this one
// depends on have themselves initialized.
Init(app *App) error
}
Battery is the interface for heavyweight, lifecycle-aware modules that extend a GoFastr application (auth, search, cache, queue, etc.).
A Battery is a Plugin (Init(*App) error is the same shape) plus two things a plain Plugin doesn't have:
- Dependency declarations. Pass dependency names at RegisterBattery time and the framework topologically sorts the Init order so dependents see their dependencies wired first.
- Structured start/stop. Implement BatteryLifecycle to get per-battery OnStart(ctx) / OnStop(ctx) hooks, distinct from the App-wide app.OnStart / app.OnStop. The framework runs them in dependency order on Start and reverse-dependency order on Stop.
If neither of those applies, prefer Plugin — there is less ceremony and the Plugin/Battery distinction stays meaningful only for the modules that genuinely need it.
From Init the battery does everything by calling into the App — register routes via app.Router, middleware via app.Use, swap the logger via app.SetLogger, MCP tools via app.MCP, hooks via app.HookRegistry(name).RegisterHook(...), and so on. There are no optional interfaces for routes / middleware / hooks / tools — Init owns it all.
type BatteryLifecycle ¶
type BatteryLifecycle interface {
Battery
// OnStart is called after all batteries are initialized and before the
// HTTP server begins accepting connections. The context is cancelled
// when the app shuts down, so long-running workers should respect it.
// The first error aborts startup.
OnStart(ctx context.Context) error
// OnStop is called during graceful shutdown, after the HTTP server has
// stopped. Batteries stop in reverse dependency order (dependents
// first, then their dependencies).
OnStop(ctx context.Context) error
}
BatteryLifecycle is the optional interface for batteries that need to participate in the App's startup and shutdown sequence.
type BatteryManager ¶
type BatteryManager struct {
// contains filtered or unexported fields
}
BatteryManager manages registered batteries, resolves dependencies, and orchestrates initialization and lifecycle.
func NewBatteryManager ¶
func NewBatteryManager() *BatteryManager
NewBatteryManager creates a new BatteryManager.
func (*BatteryManager) All ¶
func (bm *BatteryManager) All() []Battery
All returns all registered batteries in dependency-resolved order.
func (*BatteryManager) Get ¶
func (bm *BatteryManager) Get(name string) (Battery, error)
Get retrieves a battery by name. Returns the Battery interface so callers can type-assert to the concrete type or any optional interface.
func (*BatteryManager) InitAll ¶
func (bm *BatteryManager) InitAll(app *App) error
InitAll initializes all batteries in dependency order. Called during App.Start before the HTTP server binds.
func (*BatteryManager) Names ¶
func (bm *BatteryManager) Names() []string
Names returns battery names in dependency-resolved order.
func (*BatteryManager) Register ¶
func (bm *BatteryManager) Register(b Battery, deps ...string) error
Register adds a battery with optional dependency declarations. Deps lists battery names that must be initialized before this one. Returns an error on duplicate name or unknown dependency.
type BoolColumn ¶
type BoolColumn = entity.BoolColumn
type CrudHandler ¶
type CrudHandler = crud.CrudHandler
type DBExecutor ¶
type DestructiveChangeError ¶
type DestructiveChangeError = migrate.DestructiveChangeError
type EntityConfig ¶
type EntityConfig = entity.EntityConfig
type EntityDeclaration ¶
type EntityDeclaration = entity.EntityDeclaration
type EventHandler ¶
type EventHandler = event.EventHandler
type FieldDeclaration ¶
type FieldDeclaration = entity.FieldDeclaration
type FloatColumn ¶
type FloatColumn = entity.FloatColumn
type ForeignKey ¶
type ForeignKey = migrate.ForeignKey
type GroupOption ¶
type GroupOption = routegroup.GroupOption
GroupOption is re-exported for routegroup configuration.
type HookGetPayload ¶
type HookGetPayload = hook.GetPayload
type HookListPayload ¶
type HookListPayload = hook.ListPayload
type HookRegistry ¶
type HookRegistry = hook.HookRegistry
type HookWhereClause ¶
type HookWhereClause = hook.WhereClause
type IncludeNode ¶
type IncludeNode = crud.IncludeNode
type ListOptions ¶
type ListOptions = crud.ListOptions
type ListResponse ¶
type ListResponse = crud.ListResponse
type Metrics ¶
type Metrics = middleware.Metrics
type MigrationPlan ¶
type Mountable ¶
Mountable is anything that can register routes on the framework's router. UI hosts, admin panels, websocket pubsub layers, etc. all satisfy this interface and are attached via App.Mount.
type Permission ¶
type Permission = access.Permission
type Plugin ¶
type Plugin interface {
// Name returns the unique plugin identifier.
Name() string
// Init wires the plugin into the App. Called once during App.Start.
Init(app *App) error
}
Plugin is the interface for lightweight GoFastr extensions — anything that needs to register routes, middleware, hooks, or MCP tools but has no dependency on other modules and no structured start/stop lifecycle of its own.
Plugins have a single integration point: Init(app). From there a plugin does everything it needs by calling into the App — register routes via app.Router, add middleware via app.Use, swap the logger via app.SetLogger, register MCP tools via app.MCP, attach hooks via app.HookRegistry(name).RegisterHook(...), and so on.
There are no optional interfaces. The router resolves middleware late-bound, so middleware added from Init wraps routes registered before the plugin loaded — there is no ordering footgun to dodge.
When to pick Plugin vs Battery
- Plugin: stateless or self-contained; no dependency on other modules; uses app.OnStart / app.OnStop if it needs lifecycle.
- Battery: depends on another module being initialised first (e.g. auth needs the user store wired before login), OR needs its own structured OnStart/OnStop distinct from the App-wide hooks (e.g. background queue workers with their own ctx).
Both share the Init(*App) shape; Battery just adds dependency declarations and a separate BatteryLifecycle. Most extensions should start as Plugin and graduate to Battery only when one of those two conditions appears.
type PluginManager ¶
type PluginManager struct {
// contains filtered or unexported fields
}
PluginManager manages registered plugins.
func NewPluginManager ¶
func NewPluginManager() *PluginManager
NewPluginManager creates a new plugin manager.
func (*PluginManager) All ¶
func (pm *PluginManager) All() []Plugin
All returns all registered plugins in order.
func (*PluginManager) Get ¶
func (pm *PluginManager) Get(name string) (Plugin, error)
Get retrieves a plugin by name.
func (*PluginManager) InitAll ¶
func (pm *PluginManager) InitAll(app *App) error
InitAll initializes all plugins in registration order.
Wraps each Init in a deferred recover so a panic — most commonly `http: multiple registrations for ...` from a duplicate route pattern — gets tagged with the offending plugin's name. Without this the panic surfaces deep in ServeMux with no context about which plugin registered the conflicting route.
Tracks per-plugin init state so that a retry after partial failure (App.InitPlugins rolls back its global latch on error) only re-runs plugins that haven't already applied side effects. Without this a retry would re-Register the routes the successful plugins already added and panic on the ServeMux duplicate-pattern check.
func (*PluginManager) Names ¶
func (pm *PluginManager) Names() []string
Names returns the names of all registered plugins in registration order.
func (*PluginManager) Register ¶
func (pm *PluginManager) Register(plugin Plugin) error
Register adds a plugin to the manager. Returns an error if a plugin with the same name is already registered, or if the name is invalid (empty, whitespace-only, contains a control character, or longer than maxModuleNameLen).
type ReadinessCheck ¶
ReadinessCheck is a named probe run by GET /readyz. Implementations must be fast (sub-second), idempotent, and safe to invoke concurrently. Return a non-nil error to mark the app "not ready" — clients (load balancers, k8s, Fly health checks) treat a 503 from /readyz as a signal to stop routing traffic.
type ReadinessRegistrar ¶
type ReadinessRegistrar interface {
RegisterReadinessChecks(app *App)
}
ReadinessRegistrar is the optional interface plugins and batteries implement to contribute checks. The framework's plugin/battery machinery probes for it during InitPlugins and calls RegisterReadinessChecks before health endpoints mount.
The method is deliberately named RegisterReadinessChecks (plural) to avoid colliding with App.RegisterReadiness(name, fn) — the two are different surfaces, and embedding *App into a battery would otherwise produce an ambiguous selector.
type ReadinessResponse ¶
type ReadinessResponse struct {
Status string `json:"status"`
Checks []ReadinessResult `json:"checks"`
}
ReadinessResponse is the JSON shape returned by /readyz.
type ReadinessResult ¶
type ReadinessResult struct {
Name string `json:"name"`
Status string `json:"status"` // "ok", "error", or "timeout"
Error string `json:"error,omitempty"`
DurMS int64 `json:"durationMs"`
}
ReadinessResult is one row in the /readyz response.
type Registry ¶
type Registry struct {
// contains filtered or unexported fields
}
Registry stores and retrieves Entity definitions by name. It is safe for concurrent use.
func (*Registry) All ¶
All returns a copy of the map of all registered entities. Map iteration order is randomised by Go; use AllSorted() for stable iteration in code paths that emit order-sensitive output.
func (*Registry) AllSorted ¶
AllSorted returns every registered entity in alphabetical order by name. Use this whenever the iteration order affects bytes-on-the-wire (OpenAPI tag emission, LLM-markdown, codegen output) so the same registry produces the same artefact across restarts.
func (*Registry) Get ¶
Get retrieves an Entity by name. Returns an error if no entity with that name is registered.
type RelationType ¶
type RelationType = entity.RelationType
type RolePolicy ¶
type RolePolicy = access.RolePolicy
type RouteGroup ¶
type RouteGroup = routegroup.RouteGroup
RouteGroup is the App-level route group abstraction. Created via App.Group().
type SchemaChange ¶
type SchemaChange = migrate.SchemaChange
type SchemaSnapshot ¶
type SchemaSnapshot = migrate.SchemaSnapshot
type StringColumn ¶
type StringColumn = entity.StringColumn
type TenantConfig ¶
type TenantConfig = tenant.TenantConfig
type TestApp ¶
type TestApp struct {
App *App
// contains filtered or unexported fields
}
TestApp wraps an App for in-memory testing (no real HTTP listener). Uses httptest.NewRequest + http.Handler.ServeHTTP for speed.
func TestHarness ¶
TestHarness creates an in-memory test harness around an App. No real HTTP server is started — requests go directly through the router.
Calls app.InitPlugins() internally so plugin / battery wiring is in place before the first request. Without this, RegisterPlugin'd behaviour silently does nothing under the harness (Init never fires). Idempotent guard inside InitPlugins makes this safe even if Start is called later by the same test.
func (*TestApp) Close ¶
func (ta *TestApp) Close()
Close is a no-op for in-memory testing (provided for API consistency).
func (*TestApp) Delete ¶
func (ta *TestApp) Delete(path string) *TestResponse
Delete performs a DELETE request.
func (*TestApp) Get ¶
func (ta *TestApp) Get(path string) *TestResponse
Get performs a GET request and returns a TestResponse for assertions.
func (*TestApp) Post ¶
func (ta *TestApp) Post(path string, body any) *TestResponse
Post performs a POST request with a JSON body.
type TestRequest ¶
type TestRequest struct {
// contains filtered or unexported fields
}
TestRequest wraps an http.Request with a fluent builder API.
func (*TestRequest) Execute ¶
func (tr *TestRequest) Execute() *TestResponse
Execute sends the request and returns a TestResponse.
func (*TestRequest) WithBody ¶
func (tr *TestRequest) WithBody(body any) *TestRequest
WithBody sets the request body. Non-reader values are marshalled as JSON.
func (*TestRequest) WithHeader ¶
func (tr *TestRequest) WithHeader(key, value string) *TestRequest
WithHeader adds a header to the request.
type TestResponse ¶
type TestResponse struct {
// contains filtered or unexported fields
}
TestResponse wraps the recorded response with assertion helpers.
func (*TestResponse) AssertBodyContains ¶
func (tr *TestResponse) AssertBodyContains(t testing.TB, substr string) *TestResponse
AssertBodyContains asserts the body contains the given substring.
func (*TestResponse) AssertHeader ¶
func (tr *TestResponse) AssertHeader(t testing.TB, key, expected string) *TestResponse
AssertHeader asserts a response header value. Chainable.
func (*TestResponse) AssertJSON ¶
func (tr *TestResponse) AssertJSON(t testing.TB, expected any) *TestResponse
AssertJSON asserts the response body equals expected after JSON normalisation. Compares decoded values (not raw strings) so key order and number types don't matter.
func (*TestResponse) AssertStatus ¶
func (tr *TestResponse) AssertStatus(t testing.TB, expected int) *TestResponse
AssertStatus asserts the HTTP status code. Chainable.
func (*TestResponse) Body ¶
func (tr *TestResponse) Body() string
Body returns the response body as a string.
func (*TestResponse) JSON ¶
func (tr *TestResponse) JSON(v any) error
JSON decodes the response body into v.
func (*TestResponse) Status ¶
func (tr *TestResponse) Status() int
Status returns the HTTP status code.
type TimestampColumn ¶
type TimestampColumn = entity.TimestampColumn
type TypedQuery ¶
type TypedQuery[T any] = crud.TypedQuery[T]
type UUIDColumn ¶
type UUIDColumn = entity.UUIDColumn
type ValidationRegistry ¶
type ValidationRegistry = entity.ValidationRegistry
type ValidatorFunc ¶
type ValidatorFunc = entity.ValidatorFunc
Source Files
¶
- agents.go
- app.go
- audit.go
- battery.go
- doc.go
- flags.go
- health.go
- i18n.go
- mcp_introspection.go
- plugin.go
- reexports_access.go
- reexports_cron.go
- reexports_crud.go
- reexports_entity.go
- reexports_event.go
- reexports_hook.go
- reexports_migrate.go
- reexports_observability.go
- reexports_openapi.go
- reexports_routegroup.go
- reexports_tenant.go
- registry.go
- testharness.go
- tx.go
- typed_hooks.go
Directories
¶
| Path | Synopsis |
|---|---|
|
Package agentsinv is a process-wide registry of agent-onboarding snippets contributed by batteries and the framework root.
|
Package agentsinv is a process-wide registry of agent-onboarding snippets contributed by batteries and the framework root. |
|
Package db holds shared low-level database abstractions used across the GoFastr framework subpackages.
|
Package db holds shared low-level database abstractions used across the GoFastr framework subpackages. |
|
Package dev provides dev-mode-only helpers (livereload, debug surfaces).
|
Package dev provides dev-mode-only helpers (livereload, debug surfaces). |
|
Package docs ships the framework's user-facing markdown docs as an embedded filesystem.
|
Package docs ships the framework's user-facing markdown docs as an embedded filesystem. |
|
experimental
|
|
|
apiversions
Package apiversions provides first-class API versioning built on top of route groups.
|
Package apiversions provides first-class API versioning built on top of route groups. |
|
Package factory provides Rails-style fixture / factory helpers for GoFastr tests and dev-time seeders.
|
Package factory provides Rails-style fixture / factory helpers for GoFastr tests and dev-time seeders. |
|
Package harness is part of the GoFastr harness.
|
Package harness is part of the GoFastr harness. |
|
client
Package client is part of the GoFastr harness.
|
Package client is part of the GoFastr harness. |
|
client/tui
Package tui is part of the GoFastr harness.
|
Package tui is part of the GoFastr harness. |
|
client/web
Package web is part of the GoFastr harness.
|
Package web is part of the GoFastr harness. |
|
context
Package context is part of the GoFastr harness.
|
Package context is part of the GoFastr harness. |
|
control
Package control is part of the GoFastr harness.
|
Package control is part of the GoFastr harness. |
|
control/auth
Package auth implements the capability-token model: claim set, internal JWT-like encoding (no third-party dep), revocation list, and the issuance flow with TTY/notification confirmation.
|
Package auth implements the capability-token model: claim set, internal JWT-like encoding (no third-party dep), revocation list, and the issuance flow with TTY/notification confirmation. |
|
control/conformance
Package conformance is the cross-transport parity test framework.
|
Package conformance is the cross-transport parity test framework. |
|
control/inproc
Package inproc is part of the GoFastr harness.
|
Package inproc is part of the GoFastr harness. |
|
control/mcpserver
Package mcpserver will expose the harness engine as an MCP server.
|
Package mcpserver will expose the harness engine as an MCP server. |
|
control/multiplex
Package multiplex is part of the GoFastr harness.
|
Package multiplex is part of the GoFastr harness. |
|
control/resources
Package resources is part of the GoFastr harness.
|
Package resources is part of the GoFastr harness. |
|
control/rest
Package rest is part of the GoFastr harness.
|
Package rest is part of the GoFastr harness. |
|
control/ws
Package ws will implement the WebSocket transport for the control plane.
|
Package ws will implement the WebSocket transport for the control plane. |
|
engine
Package engine is part of the GoFastr harness.
|
Package engine is part of the GoFastr harness. |
|
hook
Package hook is part of the GoFastr harness.
|
Package hook is part of the GoFastr harness. |
|
ids
Package ids is part of the GoFastr harness.
|
Package ids is part of the GoFastr harness. |
|
internal/clock
Package clock provides a swap-able clock for tests.
|
Package clock provides a swap-able clock for tests. |
|
internal/ulid
Package ulid is part of the GoFastr harness.
|
Package ulid is part of the GoFastr harness. |
|
logging
Package logging is part of the GoFastr harness.
|
Package logging is part of the GoFastr harness. |
|
mcpclient
Package mcpclient implements the MCP client (consumer side) the harness uses to talk to external MCP servers.
|
Package mcpclient implements the MCP client (consumer side) the harness uses to talk to external MCP servers. |
|
memory
Package memory is part of the GoFastr harness.
|
Package memory is part of the GoFastr harness. |
|
plugin
Package plugin is part of the GoFastr harness.
|
Package plugin is part of the GoFastr harness. |
|
profile
Package profile is part of the GoFastr harness.
|
Package profile is part of the GoFastr harness. |
|
provider
Package provider is part of the GoFastr harness.
|
Package provider is part of the GoFastr harness. |
|
provider/copilot
Package copilot implements the GitHub Copilot Provider.
|
Package copilot implements the GitHub Copilot Provider. |
|
provider/credstore
Package credstore implements credential storage.
|
Package credstore implements credential storage. |
|
provider/failover
Package failover composes a chain of Providers with a circuit-breaker per upstream.
|
Package failover composes a chain of Providers with a circuit-breaker per upstream. |
|
provider/helper
Package helper is part of the GoFastr harness.
|
Package helper is part of the GoFastr harness. |
|
provider/internal/openai
Package openai is an internal OpenAI-compatible adapter used by the OpenRouter and ZAI providers (both speak the same wire shape).
|
Package openai is an internal OpenAI-compatible adapter used by the OpenRouter and ZAI providers (both speak the same wire shape). |
|
provider/openrouter
Package openrouter is part of the GoFastr harness.
|
Package openrouter is part of the GoFastr harness. |
|
provider/routing
Package routing will implement RoutingProvider: a Provider that composes {router, executors[]} so a single turn can use a cheap model for routing and an expensive model for execution.
|
Package routing will implement RoutingProvider: a Provider that composes {router, executors[]} so a single turn can use a cheap model for routing and an expensive model for execution. |
|
provider/zai
Package zai is part of the GoFastr harness.
|
Package zai is part of the GoFastr harness. |
|
secrets
Package secrets locates and loads the repo-local .harness-secrets/env file.
|
Package secrets locates and loads the repo-local .harness-secrets/env file. |
|
session
Package session is part of the GoFastr harness.
|
Package session is part of the GoFastr harness. |
|
session/sqlite
Package sqlite is part of the GoFastr harness.
|
Package sqlite is part of the GoFastr harness. |
|
skill
Package skill is part of the GoFastr harness.
|
Package skill is part of the GoFastr harness. |
|
skill/skillmd
Package skillmd is part of the GoFastr harness.
|
Package skillmd is part of the GoFastr harness. |
|
slash
Package slash is part of the GoFastr harness.
|
Package slash is part of the GoFastr harness. |
|
tool
Package tool is part of the GoFastr harness.
|
Package tool is part of the GoFastr harness. |
|
tool/builtins
Package builtins is part of the GoFastr harness.
|
Package builtins is part of the GoFastr harness. |
|
tool/pack
Package pack is part of the GoFastr harness.
|
Package pack is part of the GoFastr harness. |
|
tool/permission
Package permission is part of the GoFastr harness.
|
Package permission is part of the GoFastr harness. |
|
tracing
Package tracing is part of the GoFastr harness.
|
Package tracing is part of the GoFastr harness. |
|
Package i18nui provides translated default strings for framework UI surfaces.
|
Package i18nui provides translated default strings for framework UI surfaces. |
|
Package image is a chainable image pipeline: decode → transform → encode, pure Go with only the standard library and golang.org/x/image as dependencies.
|
Package image is a chainable image pipeline: decode → transform → encode, pure Go with only the standard library and golang.org/x/image as dependencies. |
|
internal/vp8l
Package vp8l implements a pure-Go VP8L (WebP lossless) encoder.
|
Package vp8l implements a pure-Go VP8L (WebP lossless) encoder. |
|
internal
|
|
|
casing
Package casing holds snake_case <-> camelCase helpers used internally by the GoFastr framework.
|
Package casing holds snake_case <-> camelCase helpers used internally by the GoFastr framework. |
|
testdb
Package testdb provides shared per-test database helpers used by the framework's internal tests AND by framework_test (external) tests that can't access package-private helpers.
|
Package testdb provides shared per-test database helpers used by the framework's internal tests AND by framework_test (external) tests that can't access package-private helpers. |
|
Package isolation resolves worktree-specific local runtime resources.
|
Package isolation resolves worktree-specific local runtime resources. |
|
Package lifecycle provides a documented, cooperative graceful-shutdown contract for GoFastr applications.
|
Package lifecycle provides a documented, cooperative graceful-shutdown contract for GoFastr applications. |
|
Package owner provides a single seam for "who owns this row" lookups during CRUD operations.
|
Package owner provides a single seam for "who owns this row" lookups during CRUD operations. |
|
Package routegroup provides the App-level route group abstraction.
|
Package routegroup provides the App-level route group abstraction. |
|
Package static implements static-site generation for a framework.App with a UIHost mounted on it.
|
Package static implements static-site generation for a framework.App with a UIHost mounted on it. |
|
Package testkit provides PUBLIC test helpers for host apps that use the GoFastr framework.
|
Package testkit provides PUBLIC test helpers for host apps that use the GoFastr framework. |
|
Package ui is the framework's opinionated component layer on top of core-ui.
|
Package ui is the framework's opinionated component layer on top of core-ui. |
|
theme
Package theme is the canonical home for the framework's visual design system.
|
Package theme is the canonical home for the framework's visual design system. |
|
Package uihost wires a core-ui application onto a framework.App's router.
|
Package uihost wires a core-ui application onto a framework.App's router. |