mirror of
https://github.com/carlrobertoh/ProxyAI.git
synced 2026-05-12 14:10:29 +00:00
239 lines
8.1 KiB
Java
239 lines
8.1 KiB
Java
package ee.carlrobert.codegpt.completions;
|
|
|
|
import com.intellij.openapi.diagnostic.Logger;
|
|
import ee.carlrobert.codegpt.completions.you.YouUserManager;
|
|
import ee.carlrobert.codegpt.conversations.Conversation;
|
|
import ee.carlrobert.codegpt.conversations.message.Message;
|
|
import ee.carlrobert.codegpt.settings.state.AzureSettingsState;
|
|
import ee.carlrobert.codegpt.settings.state.OpenAISettingsState;
|
|
import ee.carlrobert.codegpt.settings.state.SettingsState;
|
|
import ee.carlrobert.codegpt.telemetry.TelemetryAction;
|
|
import ee.carlrobert.llm.client.openai.completion.ErrorDetails;
|
|
import ee.carlrobert.llm.client.you.completion.YouCompletionEventListener;
|
|
import ee.carlrobert.llm.client.you.completion.YouSerpResult;
|
|
import ee.carlrobert.llm.completion.CompletionEventListener;
|
|
import java.util.List;
|
|
import java.util.function.BiConsumer;
|
|
import java.util.function.Consumer;
|
|
import javax.swing.SwingWorker;
|
|
import okhttp3.sse.EventSource;
|
|
import org.jetbrains.annotations.NotNull;
|
|
import org.jetbrains.annotations.Nullable;
|
|
|
|
public class CompletionRequestHandler {
|
|
|
|
private static final Logger LOG = Logger.getInstance(CompletionRequestHandler.class);
|
|
|
|
private final StringBuilder messageBuilder = new StringBuilder();
|
|
private SwingWorker<Void, String> swingWorker;
|
|
private EventSource eventSource;
|
|
private @Nullable Consumer<String> messageListener;
|
|
private @Nullable BiConsumer<ErrorDetails, Throwable> errorListener;
|
|
private @Nullable Consumer<String> completedListener;
|
|
private @Nullable Consumer<List<YouSerpResult>> serpResultsListener;
|
|
private @Nullable Runnable tokensExceededListener;
|
|
private boolean useContextualSearch;
|
|
|
|
public CompletionRequestHandler withContextualSearch(boolean useContextualSearch) {
|
|
this.useContextualSearch = useContextualSearch;
|
|
return this;
|
|
}
|
|
|
|
public void addMessageListener(Consumer<String> messageListener) {
|
|
this.messageListener = messageListener;
|
|
}
|
|
|
|
public void addErrorListener(BiConsumer<ErrorDetails, Throwable> errorListener) {
|
|
this.errorListener = errorListener;
|
|
}
|
|
|
|
public void addRequestCompletedListener(Consumer<String> completedListener) {
|
|
this.completedListener = completedListener;
|
|
}
|
|
|
|
public void addTokensExceededListener(Runnable tokensExceededListener) {
|
|
this.tokensExceededListener = tokensExceededListener;
|
|
}
|
|
|
|
public void addSerpResultsListener(Consumer<List<YouSerpResult>> serpResultsListener) {
|
|
this.serpResultsListener = serpResultsListener;
|
|
}
|
|
|
|
public void call(Conversation conversation, Message message, boolean isRetry) {
|
|
swingWorker = new CompletionRequestWorker(conversation, message, isRetry);
|
|
swingWorker.execute();
|
|
}
|
|
|
|
public void cancel() {
|
|
if (eventSource != null) {
|
|
eventSource.cancel();
|
|
}
|
|
swingWorker.cancel(true);
|
|
}
|
|
|
|
private EventSource startCall(
|
|
@NotNull Conversation conversation,
|
|
@NotNull Message message,
|
|
boolean isRetry,
|
|
CompletionEventListener eventListener) {
|
|
var settings = SettingsState.getInstance();
|
|
var requestProvider = new CompletionRequestProvider(conversation);
|
|
|
|
try {
|
|
if (settings.isUseYouService()) {
|
|
var sessionId = "";
|
|
var accessToken = "";
|
|
var youUserManager = YouUserManager.getInstance();
|
|
if (youUserManager.isAuthenticated()) {
|
|
var authenticationResponse =
|
|
youUserManager.getAuthenticationResponse().getData();
|
|
sessionId = authenticationResponse.getSession().getSessionId();
|
|
accessToken = authenticationResponse.getSessionJwt();
|
|
}
|
|
var request = requestProvider.buildYouCompletionRequest(message);
|
|
LOG.info("Initiating completion request using model: " +
|
|
(request.isUseGPT4Model() ? "GPT-4" : "YouBot"));
|
|
|
|
return CompletionClientProvider.getYouClient(sessionId, accessToken)
|
|
.getChatCompletion(request, eventListener);
|
|
}
|
|
|
|
if (settings.isUseAzureService()) {
|
|
var azureSettings = AzureSettingsState.getInstance();
|
|
return CompletionClientProvider.getAzureClient().getChatCompletion(
|
|
requestProvider.buildOpenAIChatCompletionRequest(
|
|
azureSettings.getModel(),
|
|
message,
|
|
isRetry,
|
|
useContextualSearch,
|
|
azureSettings.isUsingCustomPath() ? azureSettings.getPath() : null),
|
|
eventListener);
|
|
}
|
|
|
|
var openAISettings = OpenAISettingsState.getInstance();
|
|
return CompletionClientProvider.getOpenAIClient().getChatCompletion(
|
|
requestProvider.buildOpenAIChatCompletionRequest(
|
|
openAISettings.getModel(),
|
|
message,
|
|
isRetry,
|
|
useContextualSearch,
|
|
openAISettings.isUsingCustomPath() ? openAISettings.getPath() : null),
|
|
eventListener);
|
|
} catch (Throwable t) {
|
|
if (errorListener != null) {
|
|
errorListener.accept(new ErrorDetails("Something went wrong"), t);
|
|
}
|
|
throw t;
|
|
}
|
|
}
|
|
|
|
private class CompletionRequestWorker extends SwingWorker<Void, String> {
|
|
|
|
private final Conversation conversation;
|
|
private final Message message;
|
|
private final boolean isRetry;
|
|
|
|
public CompletionRequestWorker(Conversation conversation, Message message, boolean isRetry) {
|
|
this.conversation = conversation;
|
|
this.message = message;
|
|
this.isRetry = isRetry;
|
|
}
|
|
|
|
protected Void doInBackground() {
|
|
var settings = SettingsState.getInstance();
|
|
try {
|
|
eventSource = startCall(
|
|
conversation,
|
|
message,
|
|
isRetry,
|
|
settings.isUseYouService() ?
|
|
new BaseCompletionEventListener() :
|
|
new YouRequestCompletionEventListener());
|
|
} catch (TotalUsageExceededException e) {
|
|
if (tokensExceededListener != null) {
|
|
tokensExceededListener.run();
|
|
}
|
|
} finally {
|
|
sendInfo(settings);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
protected void process(List<String> chunks) {
|
|
message.setResponse(messageBuilder.toString());
|
|
for (String text : chunks) {
|
|
messageBuilder.append(text);
|
|
if (messageListener != null) {
|
|
messageListener.accept(text);
|
|
}
|
|
}
|
|
}
|
|
|
|
class BaseCompletionEventListener implements CompletionEventListener {
|
|
|
|
@Override
|
|
public void onMessage(String message) {
|
|
publish(message);
|
|
}
|
|
|
|
@Override
|
|
public void onComplete(StringBuilder messageBuilder) {
|
|
if (completedListener != null) {
|
|
completedListener.accept(messageBuilder.toString());
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onError(ErrorDetails error, Throwable ex) {
|
|
try {
|
|
if (errorListener != null) {
|
|
errorListener.accept(error, ex);
|
|
}
|
|
} finally {
|
|
sendError(error, ex);
|
|
}
|
|
}
|
|
}
|
|
|
|
class YouRequestCompletionEventListener extends BaseCompletionEventListener
|
|
implements YouCompletionEventListener {
|
|
|
|
@Override
|
|
public void onSerpResults(List<YouSerpResult> results) {
|
|
if (serpResultsListener != null) {
|
|
serpResultsListener.accept(results);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void sendInfo(SettingsState settings) {
|
|
var service = "openai";
|
|
if (settings.isUseAzureService()) {
|
|
service = "azure";
|
|
}
|
|
if (settings.isUseYouService()) {
|
|
service = "you";
|
|
}
|
|
TelemetryAction.COMPLETION.createActionMessage()
|
|
.property("conversationId", conversation.getId().toString())
|
|
.property("model", conversation.getModel())
|
|
.property("service", service)
|
|
.send();
|
|
}
|
|
|
|
private void sendError(ErrorDetails error, Throwable ex) {
|
|
var telemetryMessage = TelemetryAction.COMPLETION_ERROR.createActionMessage();
|
|
if ("insufficient_quota".equals(error.getCode())) {
|
|
telemetryMessage
|
|
.property("type", "USER")
|
|
.property("code", "INSUFFICIENT_QUOTA");
|
|
} else {
|
|
telemetryMessage
|
|
.property("conversationId", conversation.getId().toString())
|
|
.property("model", conversation.getModel())
|
|
.error(new RuntimeException(error.toString(), ex));
|
|
}
|
|
telemetryMessage.send();
|
|
}
|
|
}
|
|
}
|