|
create or replace package body "UTL_OPENAI_BATCH_API" as |
|
|
|
/** |
|
* バッチを作成する。 |
|
*/ |
|
function create_batch( |
|
p_endpoint in varchar2 |
|
,p_completion_window in varchar2 |
|
) |
|
return number |
|
as |
|
l_submission_id openai_batch_submissions.submission_id%type; |
|
begin |
|
insert into openai_batch_submissions(endpoint, completion_window) values(p_endpoint, p_completion_window) |
|
returning submission_id into l_submission_id; |
|
return l_submission_id; |
|
end create_batch; |
|
|
|
/** |
|
* 個別のリクエストをバッチに追加する。戻り値はcustom_id。 |
|
* |
|
* methodはPOST、p_urlはバッチのendpointを引き継ぐので、指定は不要。 |
|
*/ |
|
function append_request( |
|
p_submission_id in number |
|
,p_method in varchar2 default 'POST' |
|
,p_url in varchar2 default null |
|
,p_body in clob -- JSON |
|
) |
|
return number |
|
as |
|
l_custom_id openai_batch_requests.custom_id%type; |
|
begin |
|
insert into openai_batch_requests(submission_id, method, url, body) values(p_submission_id, p_method, p_url, p_body) |
|
returning custom_id into l_custom_id; |
|
return l_custom_id; |
|
end append_request; |
|
|
|
/** |
|
* バッチにChat requestを追加する。 |
|
*/ |
|
function append_chat( |
|
p_submission_id in number |
|
,p_model in varchar2 |
|
,p_system_message in clob |
|
,p_user_message in clob |
|
,p_json_schema_name in varchar2 |
|
,p_json_schema_strict in boolean |
|
,p_json_schema in clob -- for Structured Output, response_format type is set to json_schema if supplied. |
|
,p_max_tokens in number |
|
) |
|
return number |
|
as |
|
l_request json_object_t; |
|
l_messages json_array_t; |
|
l_message json_object_t; |
|
l_response_format json_object_t; |
|
l_request_clob clob; |
|
l_json_schema json_object_t; |
|
begin |
|
/* |
|
* リクエストに含まれるメッセージとしてsystemとuserを、それぞれ1つに限定。 |
|
* 対話するわけでは無いので、それで十分なはず。 |
|
*/ |
|
l_messages := json_array_t(); |
|
/* systemメッセージの追加 */ |
|
l_message := json_object_t(); |
|
l_message.put('role', 'system'); |
|
l_message.put('content', p_system_message); |
|
l_messages.append(l_message); |
|
/* userメッセージの追加 */ |
|
l_message := json_object_t(); |
|
l_message.put('role', 'user'); |
|
l_message.put('content', p_user_message); |
|
l_messages.append(l_message); |
|
/* リクエストの作成 */ |
|
l_request := json_object_t(); |
|
l_request.put('model', p_model); |
|
l_request.put('messages', l_messages); |
|
/* |
|
* JSON Schema nameの指定があれば、Structured Outputを要求する |
|
*/ |
|
if p_json_schema_name is not null then |
|
l_response_format := json_object_t(); |
|
l_response_format.put('type', 'json_schema'); |
|
/* |
|
* schemaに与えているオブジェクトが本来のJSON Schemaだが、 |
|
* nameとstrictを含めたオブジェクトをjson_schemaとして与える。 |
|
*/ |
|
l_json_schema := json_object_t(); |
|
l_json_schema.put('name', p_json_schema_name); |
|
l_json_schema.put('schema', json_object_t(p_json_schema)); |
|
l_json_schema.put('strict', p_json_schema_strict); |
|
l_response_format.put('json_schema', l_json_schema); |
|
l_request.put('response_format', l_response_format); |
|
end if; |
|
/* |
|
* max_tokensの指定があれば、リクエストに含める。 |
|
*/ |
|
if p_max_tokens is not null then |
|
l_request.put('max_tokens', p_max_tokens); |
|
end if; |
|
l_request_clob := l_request.to_clob(); |
|
return append_request( |
|
p_submission_id => p_submission_id |
|
,p_body => l_request_clob |
|
); |
|
end append_chat; |
|
|
|
/** |
|
* バッチへEmbedding requestを追加する。 |
|
* Batchでリクエストするので、配列の対応は除く。 |
|
*/ |
|
function append_embedding( |
|
p_submission_id in number |
|
,p_model in varchar2 |
|
,p_input in clob |
|
,p_encoding_format in varchar2 |
|
,p_dimensions in number |
|
) |
|
return number |
|
as |
|
l_request json_object_t; |
|
l_request_clob clob; |
|
begin |
|
l_request := json_object_t(); |
|
l_request.put('input', p_input); |
|
l_request.put('model', p_model); |
|
if p_encoding_format is not null then |
|
l_request.put('encoding_format', p_encoding_format); |
|
end if; |
|
if p_dimensions is not null then |
|
l_request.put('dimensions', p_dimensions); |
|
end if; |
|
l_request_clob := l_request.to_clob(); |
|
return append_request( |
|
p_submission_id => p_submission_id |
|
,p_body => l_request_clob |
|
); |
|
end append_embedding; |
|
|
|
/** |
|
* OpenAIにバッチ・リクエストを送信する。 |
|
*/ |
|
function submit_batch( |
|
p_submission_id in number |
|
,p_credential_static_id in varchar2 |
|
) |
|
return varchar2 |
|
as |
|
l_request json_object_t; |
|
l_file_content_clob clob; |
|
l_file_content blob; |
|
l_filename varchar2(80); |
|
l_batch_request json_object_t; |
|
l_batch_request_clob clob; |
|
l_file clob; |
|
l_batch_object clob; |
|
l_batch_object_json json_object_t; |
|
e_api_call_failed exception; |
|
l_input_file_id openai_batch_submissions.input_file_id%type; |
|
l_endpoint openai_batch_submissions.endpoint%type; |
|
l_completion_window openai_batch_submissions.completion_window%type; |
|
l_batch_id openai_batch_submissions.batch_id%type; |
|
l_output_file_id openai_batch_submissions.output_file_id%type; |
|
l_status openai_batch_submissions.status%type; |
|
begin |
|
/* |
|
* endpointとcompletion_windowの指定を取り出す。 |
|
*/ |
|
select endpoint, completion_window into l_endpoint, l_completion_window |
|
from openai_batch_submissions |
|
where submission_id = p_submission_id; |
|
/* |
|
* リクエストをJSONLのフォーマットにまとめる。 |
|
*/ |
|
l_file_content_clob := ''; |
|
for r in ( |
|
select * from openai_batch_requests |
|
where submission_id = p_submission_id order by custom_id asc |
|
) |
|
loop |
|
l_request := json_object_t(); |
|
l_request.put('custom_id', to_char(r.custom_id)); |
|
l_request.put('method', coalesce(r.method, 'POST')); |
|
l_request.put('url', coalesce(r.url, l_endpoint)); |
|
l_request.put('body', json_object_t(r.body)); |
|
l_file_content_clob := l_file_content_clob || l_request.to_clob() || apex_application.LF; |
|
end loop; |
|
l_file_content := apex_util.clob_to_blob(l_file_content_clob); |
|
/* update batch request, debug purpose */ |
|
update openai_batch_submissions set request_file = l_file_content |
|
where submission_id = p_submission_id; |
|
l_filename := apex_string.format('batchinput-%s.jsonl', p_submission_id); |
|
/* upload file */ |
|
l_input_file_id := utl_openai_files_api.upload_file( |
|
p_filename => l_filename |
|
,p_content_type => 'application/jsonlines' |
|
,p_file_content => l_file_content |
|
,p_purpose => 'batch' |
|
,p_credential_static_id => p_credential_static_id |
|
,p_file => l_file |
|
); |
|
update openai_batch_submissions set input_file_id = l_input_file_id |
|
where submission_id = p_submission_id; |
|
/* バッチ要求の発行 */ |
|
l_batch_request := json_object_t(); |
|
l_batch_request.put('input_file_id', l_input_file_id); |
|
l_batch_request.put('endpoint', l_endpoint); |
|
l_batch_request.put('completion_window', l_completion_window); |
|
l_batch_request_clob := l_batch_request.to_clob(); |
|
apex_web_service.clear_request_headers(); |
|
apex_web_service.set_request_headers('Content-Type', 'application/json'); |
|
l_batch_object := apex_web_service.make_rest_request( |
|
p_url => 'https://api.openai.com/v1/batches' |
|
,p_http_method => 'POST' |
|
,p_body => l_batch_request_clob |
|
,p_credential_static_id => p_credential_static_id |
|
); |
|
if apex_web_service.g_status_code <> 200 then |
|
raise e_api_call_failed; |
|
end if; |
|
l_batch_object_json := json_object_t(l_batch_object); |
|
l_batch_id := l_batch_object_json.get_string('id'); |
|
l_output_file_id := l_batch_object_json.get_string('output_file_id'); -- サブミット時点ではNULLなはず。 |
|
l_status := l_batch_object_json.get_string('status'); |
|
update openai_batch_submissions set batch_id = l_batch_id, output_file_id = l_output_file_id |
|
,batch_object = l_batch_object, status = l_status |
|
where submission_id = p_submission_id; |
|
return l_batch_id; |
|
end submit_batch; |
|
|
|
/** |
|
* バッチのステータスを更新する。 |
|
*/ |
|
procedure update_batch_status( |
|
p_submission_id in number |
|
,p_credential_static_id in varchar2 |
|
) |
|
as |
|
l_url varchar2(400); |
|
l_batch_object clob; |
|
l_batch_object_json json_object_t; |
|
e_api_call_failed exception; |
|
l_batch_id openai_batch_submissions.batch_id%type; |
|
l_output_file_id openai_batch_submissions.output_file_id%type; |
|
l_status openai_batch_submissions.status%type; |
|
l_error_file_id openai_batch_submissions.error_file_id%type; |
|
begin |
|
select batch_id into l_batch_id from openai_batch_submissions |
|
where submission_id = p_submission_id; |
|
l_url := apex_string.format('https://api.openai.com/v1/batches/%s', l_batch_id); |
|
apex_web_service.clear_request_headers(); |
|
apex_web_service.set_request_headers('Content-Type', 'application/json'); |
|
l_batch_object := apex_web_service.make_rest_request( |
|
p_url => l_url |
|
,p_http_method => 'GET' |
|
,p_credential_static_id => p_credential_static_id |
|
); |
|
if apex_web_service.g_status_code <> 200 then |
|
raise e_api_call_failed; |
|
end if; |
|
l_batch_object_json := json_object_t(l_batch_object); |
|
l_batch_id := l_batch_object_json.get_string('id'); |
|
l_output_file_id := l_batch_object_json.get_string('output_file_id'); |
|
l_status := l_batch_object_json.get_string('status'); |
|
l_error_file_id := l_batch_object_json.get_string('error_file_id'); |
|
update openai_batch_submissions set batch_id = l_batch_id, output_file_id = l_output_file_id |
|
,batch_object = l_batch_object, status = l_status, error_file_id = l_error_file_id |
|
where submission_id = p_submission_id; |
|
end update_batch_status; |
|
|
|
/** |
|
* 結果のファイルをダウンロードする。 |
|
*/ |
|
procedure download_batch_response( |
|
p_submission_id in number |
|
,p_credential_static_id in varchar2 |
|
) |
|
as |
|
l_file clob; |
|
l_input_file_id openai_batch_submissions.input_file_id%type; |
|
l_file_id openai_batch_submissions.output_file_id%type; |
|
l_output_file_id openai_batch_submissions.output_file_id%type; |
|
l_status openai_batch_submissions.status%type; |
|
l_response_file openai_batch_submissions.response_file%type; |
|
l_error_file_id openai_batch_submissions.error_file_id%type; |
|
l_error_file openai_batch_submissions.error_file%type; |
|
begin |
|
select status, input_file_id, output_file_id, response_file, error_file_id, error_file |
|
into l_status, l_input_file_id, l_output_file_id, l_response_file, l_error_file_id, l_error_file |
|
from openai_batch_submissions |
|
where submission_id = p_submission_id; |
|
/* |
|
* 処理が完了していて、結果のファイルが未ダウンロードであれば、ダウンロードする。 |
|
*/ |
|
if l_status = 'completed' and l_output_file_id is not null and l_response_file is null then |
|
l_file_id := utl_openai_files_api.retrieve_file( |
|
p_file_id => l_output_file_id |
|
,p_credential_static_id => p_credential_static_id |
|
,p_file => l_file |
|
,p_file_content => l_response_file |
|
); |
|
if l_response_file is not null then |
|
update openai_batch_submissions set response_file = l_response_file where submission_id = p_submission_id; |
|
end if; |
|
/* |
|
* OpenAIにアップロードされているファイルを削除する。 |
|
* |
|
*/ |
|
begin |
|
l_file_id := utl_openai_files_api.delete_file( |
|
p_file_id => l_input_file_id |
|
,p_credential_static_id => p_credential_static_id |
|
,p_file => l_file |
|
); |
|
exception |
|
when others then |
|
apex_debug.info('failed to delete file: %s', l_input_file_id); |
|
end; |
|
begin |
|
l_file_id := utl_openai_files_api.delete_file( |
|
p_file_id => l_output_file_id |
|
,p_credential_static_id => p_credential_static_id |
|
,p_file => l_file |
|
); |
|
exception |
|
when others then |
|
apex_debug.info('failed to delete file: %s', l_output_file_id); |
|
end; |
|
end if; |
|
/* |
|
* エラー・ファイルがあればダウンロードする。 |
|
*/ |
|
if l_status = 'completed' and l_error_file_id is not null and l_error_file is null then |
|
l_file_id := utl_openai_files_api.retrieve_file( |
|
p_file_id => l_error_file_id |
|
,p_credential_static_id => p_credential_static_id |
|
,p_file => l_file |
|
,p_file_content => l_error_file |
|
); |
|
if l_error_file is not null then |
|
update openai_batch_submissions set error_file = l_error_file where submission_id = p_submission_id; |
|
end if; |
|
/* |
|
* OpenAIにアップロードされているファイルを削除する。 |
|
* |
|
*/ |
|
begin |
|
l_file_id := utl_openai_files_api.delete_file( |
|
p_file_id => l_error_file_id |
|
,p_credential_static_id => p_credential_static_id |
|
,p_file => l_file |
|
); |
|
exception |
|
when others then |
|
apex_debug.info('failed to delete file: %s', l_error_file_id); |
|
end; |
|
end if; |
|
end download_batch_response; |
|
|
|
/** |
|
* ダウンロードしたresponse_fileの内容で、openai_batch_responsesを更新する。 |
|
*/ |
|
procedure parse_batch_response( |
|
p_submission_id in number |
|
) |
|
as |
|
l_response_file blob; |
|
l_responses apex_t_varchar2; |
|
l_response_json json_object_t; |
|
l_line varchar2(32767); |
|
l_id openai_batch_responses.id%type; |
|
l_custom_id openai_batch_responses.custom_id%type; |
|
l_response json_object_t; |
|
l_response_clob clob; |
|
l_status_code openai_batch_responses.status_code%type; |
|
l_request_id openai_batch_responses.request_id%type; |
|
l_body json_object_t; |
|
l_body_clob clob; |
|
/* chat completionsの応答より、最初のメッセージを取り出す。 */ |
|
l_choices json_array_t; |
|
l_choice json_object_t; |
|
l_message json_object_t; |
|
l_content clob; |
|
begin |
|
select response_file into l_response_file from openai_batch_submissions where submission_id = p_submission_id; |
|
if l_response_file is null then |
|
return; -- ファイルの内容が無ければ、何もしない。 |
|
end if; |
|
/* |
|
* CLOBの内容をLFで分割する。結果はVARCHAR2の配列なので、それぞれのレスポンスのサイズが |
|
* 32kを超える場合の結果は不明。 |
|
*/ |
|
l_responses := apex_string.split(apex_util.blob_to_clob(l_response_file)); |
|
for i in 1..l_responses.count |
|
loop |
|
l_line := trim(l_responses(i)); |
|
if l_line is null then |
|
continue; |
|
end if; |
|
l_response_json := json_object_t(l_line); |
|
l_id := l_response_json.get_string('id'); |
|
l_custom_id := to_number(l_response_json.get_string('custom_id')); |
|
l_response := l_response_json.get_object('response'); |
|
l_response_clob := l_response.to_clob(); |
|
l_status_code := l_response.get_number('status_code'); |
|
l_request_id := l_response.get_string('request_id'); |
|
l_body := l_response.get_object('body'); |
|
l_body_clob := l_body.to_clob(); |
|
/* |
|
* bodyに含まれる最初のメッセージを取り出す。 |
|
*/ |
|
l_choices := l_body.get_array('choices'); |
|
if l_choices is not null then |
|
l_choice := treat(l_choices.get(0) as json_object_t); |
|
if l_choice is not null then |
|
l_message := l_choice.get_object('message'); |
|
if l_message is not null then |
|
l_content := l_message.get_clob('content'); |
|
-- 最初の文字が { であれば、pretty printを行う。 |
|
if l_content is not null and substr(l_content,1,1) = '{' then |
|
select json_serialize(l_content returning clob pretty) into l_content from dual; |
|
end if; |
|
end if; |
|
end if; |
|
end if; |
|
begin |
|
insert into openai_batch_responses(id, custom_id, response, status_code, request_id, body, first_message) |
|
values(l_id, l_custom_id, l_response_clob, l_status_code, l_request_id, l_body_clob, l_content); |
|
exception |
|
when dup_val_on_index then |
|
null; |
|
end; |
|
end loop; |
|
end parse_batch_response; |
|
|
|
/** |
|
* 継続中のバッチをキャンセルする。 |
|
*/ |
|
procedure cancel_batch( |
|
p_batch_id in varchar2 default null |
|
,p_submission_id in number default null |
|
,p_credential_static_id in varchar2 |
|
) |
|
as |
|
l_url varchar2(400); |
|
l_batch_object clob; |
|
l_batch_object_json json_object_t; |
|
e_api_call_failed exception; |
|
l_input_file_id openai_batch_submissions.input_file_id%type; |
|
l_endpoint openai_batch_submissions.endpoint%type; |
|
l_completion_window openai_batch_submissions.completion_window%type; |
|
l_batch_id openai_batch_submissions.batch_id%type; |
|
l_output_file_id openai_batch_submissions.output_file_id%type; |
|
l_status openai_batch_submissions.status%type; |
|
e_no_required_argument exception; |
|
begin |
|
if p_batch_id is null then |
|
if p_submission_id is null then |
|
raise e_no_required_argument; |
|
end if; |
|
select batch_id into l_batch_id from openai_batch_submissions |
|
where submission_id = p_submission_id; |
|
else |
|
l_batch_id := p_batch_id; |
|
end if; |
|
l_url := apex_string.format('https://api.openai.com/v1/batches/%s/cancel', l_batch_id); |
|
apex_web_service.clear_request_headers(); |
|
apex_web_service.set_request_headers('Content-Type', 'application/json'); |
|
l_batch_object := apex_web_service.make_rest_request( |
|
p_url => l_url |
|
,p_http_method => 'POST' |
|
,p_credential_static_id => p_credential_static_id |
|
); |
|
if apex_web_service.g_status_code <> 200 then |
|
raise e_api_call_failed; |
|
end if; |
|
/* |
|
* submission_idの指定があるときは、OPENAI_BATCH_SUBMISSIONを更新する。 |
|
*/ |
|
if p_submission_id is not null then |
|
l_batch_object_json := json_object_t(l_batch_object); |
|
l_status := l_batch_object_json.get_string('status'); |
|
update openai_batch_submissions set batch_object = l_batch_object, status = l_status |
|
where submission_id = p_submission_id; |
|
end if; |
|
end cancel_batch; |
|
|
|
end "UTL_OPENAI_BATCH_API"; |
|
/ |