feat: add remote NodeService client backed by the HTTP API

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-04-01 13:18:31 +02:00
parent 3dfc46c3ff
commit 7292751ef7
4 changed files with 141 additions and 0 deletions

122
service/api_client.go Normal file
View File

@@ -0,0 +1,122 @@
package service
import (
"axolotl/models"
"bytes"
"encoding/json"
"fmt"
"net/http"
"net/url"
)
type apiClient struct {
base string
user string
http *http.Client
}
func (c *apiClient) User() string { return c.user }
func (c *apiClient) do(method, path string, body any) (*http.Response, error) {
var buf bytes.Buffer
if body != nil {
if err := json.NewEncoder(&buf).Encode(body); err != nil {
return nil, err
}
}
req, err := http.NewRequest(method, c.base+path, &buf)
if err != nil {
return nil, err
}
req.Header.Set("X-Ax-User", c.user)
if body != nil {
req.Header.Set("Content-Type", "application/json")
}
return c.http.Do(req)
}
func apiDecode[T any](resp *http.Response) (T, error) {
var v T
defer resp.Body.Close()
if resp.StatusCode >= 400 {
var e struct{ Error string }
json.NewDecoder(resp.Body).Decode(&e)
return v, fmt.Errorf("%s", e.Error)
}
return v, json.NewDecoder(resp.Body).Decode(&v)
}
func (c *apiClient) GetByID(id string) (*models.Node, error) {
resp, err := c.do("GET", "/nodes/"+id, nil)
if err != nil {
return nil, err
}
return apiDecode[*models.Node](resp)
}
func (c *apiClient) List(filter ListFilter) ([]*models.Node, error) {
q := url.Values{}
for _, r := range filter.Rels {
if r.Target == "" {
q.Add("rel", string(r.Type))
} else {
q.Add("rel", string(r.Type)+":"+r.Target)
}
}
path := "/nodes"
if len(q) > 0 {
path += "?" + q.Encode()
}
resp, err := c.do("GET", path, nil)
if err != nil {
return nil, err
}
return apiDecode[[]*models.Node](resp)
}
func (c *apiClient) Add(input AddInput) (*models.Node, error) {
resp, err := c.do("POST", "/nodes", input)
if err != nil {
return nil, err
}
return apiDecode[*models.Node](resp)
}
func (c *apiClient) Update(id string, input UpdateInput) (*models.Node, error) {
resp, err := c.do("PATCH", "/nodes/"+id, input)
if err != nil {
return nil, err
}
return apiDecode[*models.Node](resp)
}
func (c *apiClient) Delete(id string) error {
resp, err := c.do("DELETE", "/nodes/"+id, nil)
if err != nil {
return err
}
defer resp.Body.Close()
if resp.StatusCode >= 400 {
var e struct{ Error string }
json.NewDecoder(resp.Body).Decode(&e)
return fmt.Errorf("%s", e.Error)
}
return nil
}
func (c *apiClient) ListUsers() ([]*models.Node, error) {
resp, err := c.do("GET", "/users", nil)
if err != nil {
return nil, err
}
return apiDecode[[]*models.Node](resp)
}
func (c *apiClient) AddUser(name string) (*models.Node, error) {
resp, err := c.do("POST", "/users", map[string]string{"name": name})
if err != nil {
return nil, err
}
return apiDecode[*models.Node](resp)
}

View File

@@ -19,5 +19,7 @@ type Config interface {
DeleteAlias(name string) error
ListAliases() ([]*Alias, error)
GetServerConfig() ServerConfig
// GetRemoteConfig returns the remote server address and whether remote mode is enabled.
GetRemoteConfig() (ServerConfig, bool)
Save() error
}

View File

@@ -14,6 +14,7 @@ type fileConfig struct {
User string `json:"user"`
UserAliases []*Alias `json:"aliases"`
Serve ServerConfig `json:"serve"`
Remote ServerConfig `json:"remote"`
}
var defaultAliases = []*Alias{
@@ -141,6 +142,17 @@ func (c *fileConfig) ListAliases() ([]*Alias, error) {
return result, nil
}
func (c *fileConfig) GetRemoteConfig() (ServerConfig, bool) {
if c.Remote.Host == "" {
return ServerConfig{}, false
}
port := c.Remote.Port
if port == 0 {
port = 7000
}
return ServerConfig{Host: c.Remote.Host, Port: port}, true
}
func (c *fileConfig) GetServerConfig() ServerConfig {
host := c.Serve.Host
if host == "" {

View File

@@ -4,6 +4,7 @@ import (
"axolotl/models"
"axolotl/store"
"fmt"
"net/http"
)
// NodeService is the single entry point for all node operations.
@@ -76,6 +77,10 @@ func GetNodeService(cfg Config) (NodeService, error) {
if user == "" {
return nil, fmt.Errorf("no user configured: run 'ax user set <username>' first")
}
if rc, ok := cfg.GetRemoteConfig(); ok {
base := fmt.Sprintf("http://%s:%d", rc.Host, rc.Port)
return &apiClient{base: base, user: user, http: &http.Client{}}, nil
}
st, err := store.FindAndOpenSQLiteStore()
if err != nil {
return nil, err