diff --git a/agents-flex-core/src/main/java/com/agentsflex/core/llm/client/impl/SseClient.java b/agents-flex-core/src/main/java/com/agentsflex/core/llm/client/impl/SseClient.java
new file mode 100644
index 0000000..40594e3
--- /dev/null
+++ b/agents-flex-core/src/main/java/com/agentsflex/core/llm/client/impl/SseClient.java
@@ -0,0 +1,119 @@
+/*
+ * Copyright (c) 2023-2025, Agents-Flex (fuhai999@gmail.com).
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package com.agentsflex.core.llm.client.impl;
+
+import com.agentsflex.core.llm.LlmConfig;
+import com.agentsflex.core.llm.client.LlmClient;
+import com.agentsflex.core.llm.client.LlmClientListener;
+import com.agentsflex.core.llm.client.OkHttpClientUtil;
+import com.agentsflex.core.util.LogUtil;
+import okhttp3.*;
+import okhttp3.sse.EventSource;
+import okhttp3.sse.EventSourceListener;
+import okhttp3.sse.EventSources;
+import org.jetbrains.annotations.NotNull;
+import org.jetbrains.annotations.Nullable;
+
+import java.util.Map;
+
+public class SseClient extends EventSourceListener implements LlmClient {
+
+ private OkHttpClient okHttpClient;
+ private EventSource eventSource;
+ private LlmClientListener listener;
+ private LlmConfig config;
+ private boolean isStop = false;
+
+ @Override
+ public void start(String url, Map headers, String payload, LlmClientListener listener, LlmConfig config) {
+ this.listener = listener;
+ this.config = config;
+ this.isStop = false;
+
+ Request.Builder builder = new Request.Builder()
+ .url(url);
+
+ if (headers != null && !headers.isEmpty()) {
+ headers.forEach(builder::addHeader);
+ }
+
+ MediaType mediaType = MediaType.parse("application/json; charset=utf-8");
+ RequestBody body = RequestBody.create(payload, mediaType);
+ Request request = builder.post(body).build();
+
+
+ this.okHttpClient = OkHttpClientUtil.buildDefaultClient();
+
+ EventSource.Factory factory = EventSources.createFactory(this.okHttpClient);
+ this.eventSource = factory.newEventSource(request, this);
+
+ if (this.config.isDebug()) {
+ LogUtil.println(">>>>send payload:" + payload);
+ }
+
+ this.listener.onStart(this);
+ }
+
+ @Override
+ public void stop() {
+ tryToStop();
+ }
+
+
+ @Override
+ public void onClosed(@NotNull EventSource eventSource) {
+ tryToStop();
+ }
+
+ @Override
+ public void onEvent(@NotNull EventSource eventSource, @Nullable String id, @Nullable String type, @NotNull String data) {
+ if (this.config.isDebug()) {
+ LogUtil.println(">>>>receive payload:" + data);
+ }
+ this.listener.onMessage(this, data);
+ }
+
+ @Override
+ public void onFailure(@NotNull EventSource eventSource, @Nullable Throwable t, @Nullable Response response) {
+ try {
+ this.listener.onFailure(this, Util.getFailureThrowable(t, response));
+ } finally {
+ tryToStop();
+ }
+ }
+
+ @Override
+ public void onOpen(@NotNull EventSource eventSource, @NotNull Response response) {
+ //super.onOpen(eventSource, response);
+ }
+
+
+ private void tryToStop() {
+ if (!this.isStop) {
+ try {
+ this.isStop = true;
+ this.listener.onStop(this);
+ } finally {
+ if (eventSource != null) {
+ eventSource.cancel();
+ }
+ if (okHttpClient != null) {
+ okHttpClient.dispatcher().executorService().shutdown();
+ }
+ }
+ }
+ }
+}