Skip to content

Commit

Permalink
add openAI ingester
Browse files Browse the repository at this point in the history
Signed-off-by: Xun Zhang <xunzh@amazon.com>
  • Loading branch information
Zhangxunmt committed Aug 28, 2024
1 parent a6b7b4a commit b3337d8
Show file tree
Hide file tree
Showing 2 changed files with 140 additions and 1 deletion.
Original file line number Diff line number Diff line change
Expand Up @@ -166,7 +166,7 @@ private void batchIngest(
private void processFieldMapping(Map<String, Object> jsonMap, Map<String, String> fieldMapping) {
List<List> smOutput = (List<List>) jsonMap.get("SageMakerOutput");
List<String> smInput = (List<String>) jsonMap.get("content");
if (smInput.size() == smInput.size() && smInput.size() == fieldMapping.size()) {
if (smInput.size() == smOutput.size() && smInput.size() == fieldMapping.size()) {
int index = 0;
for (Map.Entry<String, String> mapping : fieldMapping.entrySet()) {
// key is the field name for input String, value is the field name for embedded output
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,28 @@
package org.opensearch.ml.engine.ingest;

import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.net.HttpURLConnection;
import java.net.URL;
import java.security.AccessController;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.atomic.AtomicInteger;

import org.json.JSONArray;
import org.json.JSONObject;
import org.opensearch.OpenSearchStatusException;
import org.opensearch.action.bulk.BulkRequest;
import org.opensearch.action.bulk.BulkResponse;
import org.opensearch.action.index.IndexRequest;
import org.opensearch.client.Client;
import org.opensearch.core.action.ActionListener;
import org.opensearch.core.rest.RestStatus;
import org.opensearch.ml.common.transport.batch.MLBatchIngestionInput;
import org.opensearch.ml.engine.annotation.Ingester;

Expand All @@ -9,6 +31,9 @@
@Log4j2
@Ingester("openai")
public class openAIDataIngestion implements Ingestable {
private static final String API_KEY = "openAI_key";
private static final String API_URL = "https://api.openai.com/v1/files/";

public static final String SOURCE = "source";
private final Client client;

Expand All @@ -19,7 +44,121 @@ public openAIDataIngestion(Client client) {
@Override
public double ingest(MLBatchIngestionInput mlBatchIngestionInput) {
double successRate = 0;
try {
String apiKey = mlBatchIngestionInput.getCredential().get(API_KEY);
String fileId = mlBatchIngestionInput.getDataSources().get(SOURCE);
URL url = new URL(API_URL + fileId + "/content");

HttpURLConnection connection = (HttpURLConnection) url.openConnection();
connection.setRequestMethod("GET");
connection.setRequestProperty("Authorization", "Bearer " + apiKey);

InputStreamReader inputStreamReader = AccessController
.doPrivileged((PrivilegedExceptionAction<InputStreamReader>) () -> new InputStreamReader(connection.getInputStream()));
BufferedReader reader = new BufferedReader(inputStreamReader);

List<String> linesBuffer = new ArrayList<>();
String line;
int lineCount = 0;
// Atomic counters for tracking success and failure
AtomicInteger successfulBatches = new AtomicInteger(0);
AtomicInteger failedBatches = new AtomicInteger(0);
// List of CompletableFutures to track batch ingestion operations
List<CompletableFuture<Void>> futures = new ArrayList<>();

while ((line = reader.readLine()) != null) {
linesBuffer.add(line);
lineCount++;

// Process every 100 lines
if (lineCount == 100) {
// Create a CompletableFuture that will be completed by the bulkResponseListener
CompletableFuture<Void> future = new CompletableFuture<>();
batchIngest(linesBuffer, mlBatchIngestionInput, getBulkResponseListener(successfulBatches, failedBatches, future));

futures.add(future);
linesBuffer.clear();
lineCount = 0;
}
}
// Process any remaining lines in the buffer
if (!linesBuffer.isEmpty()) {
CompletableFuture<Void> future = new CompletableFuture<>();
batchIngest(linesBuffer, mlBatchIngestionInput, getBulkResponseListener(successfulBatches, failedBatches, future));
futures.add(future);
}

reader.close();
// Combine all futures and wait for completion
CompletableFuture<Void> allFutures = CompletableFuture.allOf(futures.toArray(new CompletableFuture[0]));
// Wait for all tasks to complete
allFutures.join();
int totalBatches = successfulBatches.get() + failedBatches.get();
successRate = (double) successfulBatches.get() / totalBatches * 100;
} catch (PrivilegedActionException e) {
throw new RuntimeException("Failed to read from OpenAI file API: ", e);
} catch (Exception e) {
log.error(e.getMessage());
throw new OpenSearchStatusException("Failed to batch ingest: " + e.getMessage(), RestStatus.INTERNAL_SERVER_ERROR);
}

return successRate;
}

private ActionListener<BulkResponse> getBulkResponseListener(
AtomicInteger successfulBatches,
AtomicInteger failedBatches,
CompletableFuture<Void> future
) {
return ActionListener.wrap(bulkResponse -> {
if (bulkResponse.hasFailures()) {
failedBatches.incrementAndGet();
future.completeExceptionally(new RuntimeException(bulkResponse.buildFailureMessage())); // Mark the future as completed
// with an exception
}
log.debug("Batch Ingestion successfully");
successfulBatches.incrementAndGet();
future.complete(null); // Mark the future as completed successfully
}, e -> {
log.error("Failed to bulk update model state", e);
failedBatches.incrementAndGet();
future.completeExceptionally(e); // Mark the future as completed with an exception
});
}

private void batchIngest(
List<String> sourceLines,
MLBatchIngestionInput mlBatchIngestionInput,
ActionListener<BulkResponse> bulkResponseListener
) {
BulkRequest bulkRequest = new BulkRequest();
sourceLines.stream().forEach(jsonStr -> {
JSONObject jsonObject = new JSONObject(jsonStr);
String customId = jsonObject.getString("custom_id");
JSONObject responseBody = jsonObject.getJSONObject("response").getJSONObject("body");
JSONArray dataArray = responseBody.getJSONArray("data");
Map<String, Object> jsonMap = processFieldMapping(customId, dataArray, mlBatchIngestionInput.getFieldMapping());
IndexRequest indexRequest = new IndexRequest(mlBatchIngestionInput.getIndexName()).source(jsonMap);

bulkRequest.add(indexRequest);
});
client.bulk(bulkRequest, bulkResponseListener);
}

private Map<String, Object> processFieldMapping(String customId, JSONArray dataArray, Map<String, String> fieldMapping) {
Map<String, Object> jsonMap = new HashMap<>();
if (dataArray.length() == fieldMapping.size()) {
int index = 0;
for (Map.Entry<String, String> mapping : fieldMapping.entrySet()) {
// key is the field name for input String, value is the field name for embedded output
JSONObject dataItem = dataArray.getJSONObject(index);
jsonMap.put(mapping.getValue(), dataItem.getJSONArray("embedding"));
index++;
}
jsonMap.put("id", customId);
} else {
throw new IllegalArgumentException("the fieldMapping and source data do not match");
}
return jsonMap;
}
}

0 comments on commit b3337d8

Please sign in to comment.