plugins/cmd/ndpgen/README.md
Navidrome Plugin Development Kit (PDK) code generator. It reads Go interface definitions with special annotations and generates client wrappers for WASM plugins.
This tool is the unified code generator that handle both host function wrappers and capability wrappers.
ndpgen -input <dir> -output <dir> [-package <name>] [-v] [-dry-run] [-host-only] [-go] [-python] [-rust]
| Flag | Description | Default |
|---|---|---|
-input | Directory containing Go source files with annotated interfaces | Required |
-output | Directory where generated files will be written | Same as input |
-package | Package name for generated files | Inferred from output |
-v | Verbose output | false |
-dry-run | Parse and validate without writing files | false |
-host-only | Generate only host function wrappers (capability support TBD) | true |
-go | Generate Go client wrappers | true* |
-python | Generate Python client wrappers | false |
-rust | Generate Rust client wrappers | false |
* -go is enabled by default when neither -python nor -rust is specified. Use combinations like -go -python -rust to generate multiple languages.
go run ./plugins/cmd/ndpgen \
-input ./plugins/host \
-output ./plugins/pdk
//nd:hostserviceMarks an interface as a host service that will have wrappers generated.
//nd:hostservice name=<ServiceName> permission=<permission>
type MyService interface { ... }
| Parameter | Description | Required |
|---|---|---|
name | Service name used in generated type names and function prefixes | Yes |
permission | Permission required by plugins to use this service | Yes |
//nd:hostfuncMarks a method within a host service interface for export to plugins.
//nd:hostfunc [name=<export_name>]
MethodName(ctx context.Context, ...) (result Type, err error)
| Parameter | Description | Required |
|---|---|---|
name | Custom export name (default: <servicename>_<methodname> in lowercase) | No |
Host service interfaces must follow these conventions:
context.Context - Required for all methodserror - For proper error handlingpackage host
import "context"
// SubsonicAPIService provides access to Navidrome's Subsonic API.
// This documentation becomes part of the generated code.
//nd:hostservice name=SubsonicAPI permission=subsonicapi
type SubsonicAPIService interface {
// Call executes a Subsonic API request and returns the response.
//nd:hostfunc
Call(ctx context.Context, uri string) (response string, err error)
}
Generated files are named nd_host_<servicename>.go (lowercase) and placed in $output/go/host/. The $output/go/ directory becomes a complete Go module (github.com/navidrome/navidrome/plugins/pdk/go) with package name host, intended for import by Navidrome plugins built with TinyGo.
The generator creates:
nd_host_<servicename>.go - Client wrapper code (WASM build)nd_host_<servicename>_stub.go - Mock implementations for non-WASM platforms (testing)doc.go - Package documentation listing all available servicesgo.mod - Go module file with required dependenciesEach service file includes:
// Code generated by ndpgen. DO NOT EDIT. headerencoding/json, errors, github.com/extism/go-pdk)//go:wasmimport declarations for each host functionThe stub files (*_stub.go) contain testify/mock implementations that allow plugin authors to unit test their code on non-WASM platforms.
Each host service has:
mock.Mockhost.CacheMock, host.ArtworkMock)Example: Testing a plugin that uses the Cache service
package myplugin
import (
"testing"
"github.com/navidrome/navidrome/plugins/pdk/go/host"
)
func TestMyPluginFunction(t *testing.T) {
// Set expectations on the mock
host.CacheMock.On("GetString", "my-key").Return("cached-value", true, nil)
host.CacheMock.On("SetString", "new-key", "new-value", int64(3600)).Return(nil)
// Call your plugin code that uses host.CacheGetString and host.CacheSetString
result := myPluginFunction()
// Assert the result
if result != "expected" {
t.Errorf("unexpected result: %s", result)
}
// Verify all expected calls were made
host.CacheMock.AssertExpectations(t)
}
Resetting mocks between tests:
If you need to reset mock state between tests, testify's mock doesn't have a built-in reset. Either use separate test functions (testify automatically resets between test runs), or create a helper to set up fresh expectations.
When using -python, Python client files are generated in a python/ subdirectory.
When using -rust, Rust client files are generated in a rust/ subdirectory.
ndpgen supports these Go types in method signatures:
| Type | JSON Representation |
|---|---|
string, int, bool, etc. | Native JSON types |
[]T (slices) | JSON arrays |
map[K]V (maps) | JSON objects |
*T (pointers) | Nullable fields |
interface{} / any | Converts to any |
| Custom structs | JSON objects (must be JSON-serializable) |
Methods can return multiple values (plus error):
//nd:hostfunc
Search(ctx context.Context, query string) (results []string, total int, hasMore bool, err error)
Generates:
type ServiceSearchResponse struct {
Results []string `json:"results,omitempty"`
Total int `json:"total,omitempty"`
HasMore bool `json:"hasMore,omitempty"`
Error string `json:"error,omitempty"`
}
go test ./plugins/cmd/ndpgen/...