mirror of
https://github.com/navidrome/navidrome.git
synced 2026-02-09 14:31:11 -05:00
* feat: implement raw binary framing for host function responses Signed-off-by: Deluan <deluan@navidrome.org> * feat: add CallRaw method for Subsonic API to handle binary responses Signed-off-by: Deluan <deluan@navidrome.org> * test: add tests for raw=true methods and binary framing generation Signed-off-by: Deluan <deluan@navidrome.org> * fix: improve error message for malformed raw responses to indicate incomplete header Signed-off-by: Deluan <deluan@navidrome.org> * fix: add wasm_import_module attribute for raw methods and improve content-type handling Signed-off-by: Deluan <deluan@navidrome.org> --------- Signed-off-by: Deluan <deluan@navidrome.org>
151 lines
4.3 KiB
Go
151 lines
4.3 KiB
Go
// Code generated by ndpgen. DO NOT EDIT.
|
|
|
|
package host
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"encoding/json"
|
|
|
|
extism "github.com/extism/go-sdk"
|
|
)
|
|
|
|
// SubsonicAPICallRequest is the request type for SubsonicAPI.Call.
|
|
type SubsonicAPICallRequest struct {
|
|
Uri string `json:"uri"`
|
|
}
|
|
|
|
// SubsonicAPICallResponse is the response type for SubsonicAPI.Call.
|
|
type SubsonicAPICallResponse struct {
|
|
ResponseJSON string `json:"responseJson,omitempty"`
|
|
Error string `json:"error,omitempty"`
|
|
}
|
|
|
|
// SubsonicAPICallRawRequest is the request type for SubsonicAPI.CallRaw.
|
|
type SubsonicAPICallRawRequest struct {
|
|
Uri string `json:"uri"`
|
|
}
|
|
|
|
// RegisterSubsonicAPIHostFunctions registers SubsonicAPI service host functions.
|
|
// The returned host functions should be added to the plugin's configuration.
|
|
func RegisterSubsonicAPIHostFunctions(service SubsonicAPIService) []extism.HostFunction {
|
|
return []extism.HostFunction{
|
|
newSubsonicAPICallHostFunction(service),
|
|
newSubsonicAPICallRawHostFunction(service),
|
|
}
|
|
}
|
|
|
|
func newSubsonicAPICallHostFunction(service SubsonicAPIService) extism.HostFunction {
|
|
return extism.NewHostFunctionWithStack(
|
|
"subsonicapi_call",
|
|
func(ctx context.Context, p *extism.CurrentPlugin, stack []uint64) {
|
|
// Read JSON request from plugin memory
|
|
reqBytes, err := p.ReadBytes(stack[0])
|
|
if err != nil {
|
|
subsonicapiWriteError(p, stack, err)
|
|
return
|
|
}
|
|
var req SubsonicAPICallRequest
|
|
if err := json.Unmarshal(reqBytes, &req); err != nil {
|
|
subsonicapiWriteError(p, stack, err)
|
|
return
|
|
}
|
|
|
|
// Call the service method
|
|
responsejson, svcErr := service.Call(ctx, req.Uri)
|
|
if svcErr != nil {
|
|
subsonicapiWriteError(p, stack, svcErr)
|
|
return
|
|
}
|
|
|
|
// Write JSON response to plugin memory
|
|
resp := SubsonicAPICallResponse{
|
|
ResponseJSON: responsejson,
|
|
}
|
|
subsonicapiWriteResponse(p, stack, resp)
|
|
},
|
|
[]extism.ValueType{extism.ValueTypePTR},
|
|
[]extism.ValueType{extism.ValueTypePTR},
|
|
)
|
|
}
|
|
|
|
func newSubsonicAPICallRawHostFunction(service SubsonicAPIService) extism.HostFunction {
|
|
return extism.NewHostFunctionWithStack(
|
|
"subsonicapi_callraw",
|
|
func(ctx context.Context, p *extism.CurrentPlugin, stack []uint64) {
|
|
// Read JSON request from plugin memory
|
|
reqBytes, err := p.ReadBytes(stack[0])
|
|
if err != nil {
|
|
subsonicapiWriteRawError(p, stack, err)
|
|
return
|
|
}
|
|
var req SubsonicAPICallRawRequest
|
|
if err := json.Unmarshal(reqBytes, &req); err != nil {
|
|
subsonicapiWriteRawError(p, stack, err)
|
|
return
|
|
}
|
|
|
|
// Call the service method
|
|
contenttype, data, svcErr := service.CallRaw(ctx, req.Uri)
|
|
if svcErr != nil {
|
|
subsonicapiWriteRawError(p, stack, svcErr)
|
|
return
|
|
}
|
|
|
|
// Write binary-framed response to plugin memory:
|
|
// [0x00][4-byte content-type length (big-endian)][content-type string][raw data]
|
|
ctBytes := []byte(contenttype)
|
|
frame := make([]byte, 1+4+len(ctBytes)+len(data))
|
|
frame[0] = 0x00 // success
|
|
binary.BigEndian.PutUint32(frame[1:5], uint32(len(ctBytes)))
|
|
copy(frame[5:5+len(ctBytes)], ctBytes)
|
|
copy(frame[5+len(ctBytes):], data)
|
|
|
|
respPtr, err := p.WriteBytes(frame)
|
|
if err != nil {
|
|
stack[0] = 0
|
|
return
|
|
}
|
|
stack[0] = respPtr
|
|
},
|
|
[]extism.ValueType{extism.ValueTypePTR},
|
|
[]extism.ValueType{extism.ValueTypePTR},
|
|
)
|
|
}
|
|
|
|
// subsonicapiWriteResponse writes a JSON response to plugin memory.
|
|
func subsonicapiWriteResponse(p *extism.CurrentPlugin, stack []uint64, resp any) {
|
|
respBytes, err := json.Marshal(resp)
|
|
if err != nil {
|
|
subsonicapiWriteError(p, stack, err)
|
|
return
|
|
}
|
|
respPtr, err := p.WriteBytes(respBytes)
|
|
if err != nil {
|
|
stack[0] = 0
|
|
return
|
|
}
|
|
stack[0] = respPtr
|
|
}
|
|
|
|
// subsonicapiWriteError writes an error response to plugin memory.
|
|
func subsonicapiWriteError(p *extism.CurrentPlugin, stack []uint64, err error) {
|
|
errResp := struct {
|
|
Error string `json:"error"`
|
|
}{Error: err.Error()}
|
|
respBytes, _ := json.Marshal(errResp)
|
|
respPtr, _ := p.WriteBytes(respBytes)
|
|
stack[0] = respPtr
|
|
}
|
|
|
|
// subsonicapiWriteRawError writes a binary-framed error response to plugin memory.
|
|
// Format: [0x01][UTF-8 error message]
|
|
func subsonicapiWriteRawError(p *extism.CurrentPlugin, stack []uint64, err error) {
|
|
errMsg := []byte(err.Error())
|
|
frame := make([]byte, 1+len(errMsg))
|
|
frame[0] = 0x01 // error
|
|
copy(frame[1:], errMsg)
|
|
respPtr, _ := p.WriteBytes(frame)
|
|
stack[0] = respPtr
|
|
}
|