Skip to content

Commit 38629a1

Browse files
committed
Save the downstream service state after every interaction
This provides a basic implementation for #159
1 parent 4a41c84 commit 38629a1

File tree

22 files changed

+524
-45
lines changed

22 files changed

+524
-45
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
package ai.wanaku.api.types.management;
2+
3+
public record State(String service, boolean healthy, String message) {
4+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
package ai.wanaku.cli.main.commands.targets;
2+
3+
import java.net.URI;
4+
5+
import ai.wanaku.cli.main.commands.BaseCommand;
6+
import ai.wanaku.cli.main.services.LinkService;
7+
import io.quarkus.rest.client.reactive.QuarkusRestClientBuilder;
8+
import picocli.CommandLine;
9+
10+
@CommandLine.Command(name = "state",
11+
description = "Get the state of the targeted services")
12+
public abstract class AbstractTargetState extends BaseCommand {
13+
@CommandLine.Option(names = {"--host"}, description = "The API host", defaultValue = "http://localhost:8080",
14+
arity = "0..1")
15+
protected String host;
16+
17+
protected LinkService linkService;
18+
19+
protected void initService() {
20+
linkService = QuarkusRestClientBuilder.newBuilder()
21+
.baseUri(URI.create(host))
22+
.build(LinkService.class);
23+
}
24+
25+
}

cli/src/main/java/ai/wanaku/cli/main/commands/targets/resources/Resources.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
import picocli.CommandLine;
55

66
@CommandLine.Command(name = "resources",
7-
description = "Manage targets", subcommands = { ResourcesLinkedList.class, ResourcesConfigure.class })
7+
description = "Manage targets", subcommands = { ResourcesLinkedList.class, ResourcesConfigure.class, ResourcesState.class })
88
public class Resources extends BaseCommand {
99
@Override
1010
public void run() {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
package ai.wanaku.cli.main.commands.targets.resources;
2+
3+
import java.util.List;
4+
import java.util.Map;
5+
6+
import ai.wanaku.api.types.WanakuResponse;
7+
import ai.wanaku.api.types.management.State;
8+
import ai.wanaku.cli.main.commands.targets.AbstractTargetsList;
9+
import ai.wanaku.cli.main.support.PrettyPrinter;
10+
import picocli.CommandLine;
11+
12+
@CommandLine.Command(name = "state",
13+
description = "List service states")
14+
public class ResourcesState extends AbstractTargetsList {
15+
@Override
16+
public void run() {
17+
initService();
18+
19+
WanakuResponse<Map<String, List<State>>> list = linkService.resourcesState();
20+
PrettyPrinter.printStates(list.data());
21+
}
22+
}

cli/src/main/java/ai/wanaku/cli/main/commands/targets/tools/Tools.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
import picocli.CommandLine;
55

66
@CommandLine.Command(name = "tools",
7-
description = "Manage targets", subcommands = { ToolsLinkedList.class, ToolsConfigure.class })
7+
description = "Manage targets", subcommands = { ToolsLinkedList.class, ToolsConfigure.class, ToolsState.class })
88
public class Tools extends BaseCommand {
99
@Override
1010
public void run() {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
package ai.wanaku.cli.main.commands.targets.tools;
2+
3+
import java.util.List;
4+
import java.util.Map;
5+
6+
import ai.wanaku.api.types.management.State;
7+
import ai.wanaku.cli.main.commands.targets.AbstractTargetsList;
8+
import ai.wanaku.cli.main.support.PrettyPrinter;
9+
import picocli.CommandLine;
10+
11+
@CommandLine.Command(name = "state",
12+
description = "List services states")
13+
public class ToolsState extends AbstractTargetsList {
14+
@Override
15+
public void run() {
16+
initService();
17+
18+
Map<String, List<State>> states = linkService.toolsState().data();
19+
PrettyPrinter.printStates(states);
20+
}
21+
}

cli/src/main/java/ai/wanaku/cli/main/services/LinkService.java

+12
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
11
package ai.wanaku.cli.main.services;
22

3+
import java.util.List;
34
import java.util.Map;
45

56
import ai.wanaku.api.types.WanakuResponse;
7+
import ai.wanaku.api.types.management.State;
68
import jakarta.ws.rs.Consumes;
79
import jakarta.ws.rs.GET;
810
import jakarta.ws.rs.PUT;
@@ -22,6 +24,11 @@ public interface LinkService {
2224
@Consumes(MediaType.TEXT_PLAIN)
2325
WanakuResponse<Map<String, Service>> toolsList();
2426

27+
@Path("/tools/state")
28+
@GET
29+
@Consumes(MediaType.TEXT_PLAIN)
30+
WanakuResponse<Map<String, List<State>>> toolsState();
31+
2532
@Path("/tools/configure/{service}")
2633
@PUT
2734
@Consumes(MediaType.TEXT_PLAIN)
@@ -36,4 +43,9 @@ public interface LinkService {
3643
@PUT
3744
@Consumes(MediaType.TEXT_PLAIN)
3845
Response resourcesConfigure(@RestPath("service") String service, @QueryParam("option") String option, @QueryParam("value") String value);
46+
47+
@Path("/resources/state")
48+
@GET
49+
@Consumes(MediaType.TEXT_PLAIN)
50+
WanakuResponse<Map<String, List<State>>> resourcesState();
3951
}

cli/src/main/java/ai/wanaku/cli/main/support/PrettyPrinter.java

+23
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
import ai.wanaku.api.types.ToolReference;
99
import ai.wanaku.api.types.management.Configuration;
1010
import ai.wanaku.api.types.management.Service;
11+
import ai.wanaku.api.types.management.State;
1112

1213
public class PrettyPrinter {
1314

@@ -68,4 +69,26 @@ public static void printTargets(final Map<String, Service> map) {
6869
}
6970
}
7071

72+
73+
/**
74+
* Prints a map of entries
75+
* @param states the map of states
76+
*/
77+
public static void printStates(final Map<String, List<State>> states) {
78+
System.out.printf("%-20s %-10s %-60s%n",
79+
"Service", "Healthy", "Message");
80+
81+
for (var entry : states.entrySet()) {
82+
for (var state : entry.getValue()) {
83+
printParseableState(entry.getKey(), state.healthy(), state.message());
84+
}
85+
}
86+
}
87+
88+
private static void printParseableState(String service, boolean healthy, String message) {
89+
System.out.printf("%-15s => %-15s => %-30s %n",
90+
service, Boolean.valueOf(healthy), message);
91+
}
92+
93+
7194
}

core/core-mcp/src/main/java/ai/wanaku/core/mcp/providers/ServiceRegistry.java

+20-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
package ai.wanaku.core.mcp.providers;
22

33
import ai.wanaku.api.types.management.Service;
4+
import ai.wanaku.api.types.management.State;
5+
6+
import java.util.List;
47
import java.util.Map;
58

69
/**
@@ -19,7 +22,7 @@ public interface ServiceRegistry {
1922
* De-register a service from the registry
2023
* @param service the service name
2124
*/
22-
void deregister(String service);
25+
void deregister(String service, ServiceType serviceType);
2326

2427
/**
2528
* Gets a registered service by name
@@ -28,6 +31,22 @@ public interface ServiceRegistry {
2831
*/
2932
Service getService(String service);
3033

34+
/**
35+
* Saves the current state of the service
36+
* @param service the service to save the state
37+
* @param healthy whether it is healthy (true for healthy, false otherwise)
38+
* @param message Optional state message (ignored if healthy)
39+
*/
40+
void saveState(String service, boolean healthy, String message);
41+
42+
/**
43+
* Gets the state of the given service
44+
* @param service the service name
45+
* @param count the number of states to get
46+
* @return the last count states of the given service
47+
*/
48+
List<State> getState(String service, int count);
49+
3150
/**
3251
* Get a map of all registered services and their configurations
3352
* @param serviceType the type of service to get

core/core-service-discovery/src/main/java/ai/wanaku/core/service/discovery/ReservedKeys.java

+21
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22

33
import java.util.Set;
44

5+
import ai.wanaku.core.mcp.providers.ServiceType;
6+
57
/**
68
* Reserved keys
79
*/
@@ -16,6 +18,25 @@ class ReservedKeys {
1618
*/
1719
public static final String WANAKU_TARGET_TYPE = "wanaku-target-type";
1820

21+
/**
22+
* This key stores the set of Wanaku resource services
23+
*/
24+
public static final String WANAKU_SERVICES_RESOURCES = "wanaku-services-resources";
25+
26+
/**
27+
* This key stores the set of Wanaku tools services
28+
*/
29+
public static final String WANAKU_SERVICES_TOOLS = "wanaku-services-tools";
30+
1931

2032
public static final Set<String> ALL_KEYS = Set.of(WANAKU_TARGET_ADDRESS, WANAKU_TARGET_TYPE);
33+
34+
public static String getServiceKey(ServiceType serviceType) {
35+
switch (serviceType) {
36+
case TOOL_INVOKER: return ReservedKeys.WANAKU_TARGET_ADDRESS;
37+
case RESOURCE_PROVIDER: return ReservedKeys.WANAKU_SERVICES_RESOURCES;
38+
}
39+
40+
throw new IllegalArgumentException("Unknown service type: " + serviceType);
41+
}
2142
}

core/core-service-discovery/src/main/java/ai/wanaku/core/service/discovery/ValkeyRegistry.java

+74-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,9 @@
11
package ai.wanaku.core.service.discovery;
22

3+
import ai.wanaku.api.types.management.State;
4+
import io.valkey.StreamEntryID;
5+
import io.valkey.params.XAddParams;
6+
import io.valkey.resps.StreamEntry;
37
import jakarta.enterprise.context.ApplicationScoped;
48
import jakarta.enterprise.event.Observes;
59
import jakarta.inject.Inject;
@@ -13,7 +17,11 @@
1317
import io.quarkus.runtime.ShutdownEvent;
1418
import io.valkey.Jedis;
1519
import io.valkey.JedisPool;
20+
21+
import java.time.Instant;
22+
import java.util.ArrayList;
1623
import java.util.HashMap;
24+
import java.util.List;
1725
import java.util.Map;
1826
import java.util.Set;
1927
import org.jboss.logging.Logger;
@@ -46,6 +54,10 @@ public class ValkeyRegistry implements ServiceRegistry {
4654
@Override
4755
public void register(ServiceTarget serviceTarget, Map<String, String> configurations) {
4856
try (io.valkey.Jedis jedis = jedisPool.getResource()) {
57+
// Register the service on the specific set
58+
String serviceKey = ReservedKeys.getServiceKey(serviceTarget.getServiceType());
59+
jedis.sadd(serviceKey, serviceTarget.getService());
60+
4961
jedis.hset(serviceTarget.getService(), ReservedKeys.WANAKU_TARGET_ADDRESS, serviceTarget.toAddress());
5062
jedis.hset(serviceTarget.getService(), ReservedKeys.WANAKU_TARGET_TYPE, serviceTarget.getServiceType().asValue());
5163

@@ -64,17 +76,70 @@ public void register(ServiceTarget serviceTarget, Map<String, String> configurat
6476
* Deregisters a service with the given name.
6577
*
6678
* @param service The name of the service to deregister.
79+
* @param serviceType the type of service to deregister
6780
*/
6881
@Override
69-
public void deregister(String service) {
82+
public void deregister(String service, ServiceType serviceType) {
7083
try (io.valkey.Jedis jedis = jedisPool.getResource()) {
71-
jedis.del(service);
84+
String serviceKey = ReservedKeys.getServiceKey(serviceType);
85+
jedis.srem(serviceKey, service);
86+
7287
LOG.infof("Service %s registered", service);
7388
} catch (Exception e) {
7489
LOG.errorf(e, "Failed to register service %s: %s", service, e.getMessage());
7590
}
7691
}
7792

93+
94+
@Override
95+
public void saveState(String service, boolean healthy, String message) {
96+
try (io.valkey.Jedis jedis = jedisPool.getResource()) {
97+
Map<String, String> state = Map.of("service", service, "healthy",
98+
Boolean.toString(healthy), "message", (healthy ? "healthy" : message));
99+
100+
jedis.xadd(stateKey(service), state, XAddParams.xAddParams());
101+
} catch (Exception e) {
102+
LOG.errorf(e, "Failed to save state for %s: %s", service, e.getMessage());
103+
}
104+
}
105+
106+
@Override
107+
public List<State> getState(String service, int count) {
108+
try (io.valkey.Jedis jedis = jedisPool.getResource()) {
109+
110+
String stateKey = stateKey(service);
111+
Instant now = Instant.now();
112+
long endEpoch = now.toEpochMilli();
113+
long startEpoch = now.minusSeconds(60).toEpochMilli();
114+
115+
List<StreamEntry> streamEntries = jedis.xrange(stateKey, new StreamEntryID(startEpoch), new StreamEntryID(endEpoch));
116+
117+
List<State> states = new ArrayList<>(streamEntries.size());
118+
119+
for (StreamEntry streamEntry : streamEntries) {
120+
LOG.debugf("Entry %s", streamEntry);
121+
122+
Map<String, String> fields = streamEntry.getFields();
123+
String serviceName = fields.get("service");
124+
String message = fields.get("message");
125+
String healthy = fields.get("healthy");
126+
127+
State state = new State(serviceName, Boolean.parseBoolean(healthy), message);
128+
states.add(state);
129+
}
130+
131+
return states;
132+
} catch (Exception e) {
133+
LOG.errorf(e, "Failed to get state for %s: %s", service, e.getMessage());
134+
}
135+
136+
return List.of();
137+
}
138+
139+
private static String stateKey(String service) {
140+
return "state:" + service;
141+
}
142+
78143
/**
79144
* Retrieves a service with the given name.
80145
*
@@ -98,14 +163,13 @@ public Service getService(String service) {
98163
public Map<String, Service> getEntries(ServiceType serviceType) {
99164
Map<String, Service> entries = new HashMap<>();
100165
try (io.valkey.Jedis jedis = jedisPool.getResource()) {
101-
Set<String> keys = jedis.keys("*");
102-
for (String key : keys) {
103-
String sType = jedis.hget(key, ReservedKeys.WANAKU_TARGET_TYPE);
104-
if (serviceType.asValue().equals(sType)) {
105-
Service service = newService(jedis, key);
106-
107-
entries.put(key, service);
108-
}
166+
String serviceKey = ReservedKeys.getServiceKey(serviceType);
167+
Set<String> services = jedis.smembers(serviceKey);
168+
169+
for (String key : services) {
170+
Service service = newService(jedis, key);
171+
172+
entries.put(key, service);
109173
}
110174

111175
} catch (Exception e) {

0 commit comments

Comments
 (0)