diff --git a/R/provider-openai.R b/R/provider-openai.R index d6e9003c..cadb6953 100644 --- a/R/provider-openai.R +++ b/R/provider-openai.R @@ -49,28 +49,17 @@ chat_openai <- function( api_key = openai_key(), model = NULL, params = NULL, - seed = lifecycle::deprecated(), api_args = list(), echo = c("none", "output", "all") ) { model <- set_default(model, "gpt-4.1") echo <- check_echo(echo) - params <- params %||% params() - if (lifecycle::is_present(seed) && !is.null(seed)) { - lifecycle::deprecate_warn( - when = "0.2.0", - what = "chat_openai(seed)", - with = "chat_openai(params)" - ) - params$seed <- seed - } - provider <- ProviderOpenAI( name = "OpenAI", base_url = base_url, model = model, - params = params, + params = params %||% params(), extra_args = api_args, api_key = api_key ) @@ -84,7 +73,6 @@ chat_openai_test <- function( echo = "none" ) { params <- params %||% params() - params$seed <- params$seed %||% 1014 params$temperature <- params$temperature %||% 0 chat_openai( @@ -145,10 +133,10 @@ method(base_request_error, ProviderOpenAI) <- function(provider, req) { # Chat endpoint ---------------------------------------------------------------- method(chat_path, ProviderOpenAI) <- function(provider) { - "/chat/completions" + "/responses" } -# https://platform.openai.com/docs/api-reference/chat/create +# https://platform.openai.com/docs/api-reference/responses method(chat_body, ProviderOpenAI) <- function( provider, stream = TRUE, @@ -156,33 +144,41 @@ method(chat_body, ProviderOpenAI) <- function( tools = list(), type = NULL ) { - messages <- compact(unlist(as_json(provider, turns), recursive = FALSE)) + input <- compact(unlist(as_json(provider, turns), recursive = FALSE)) tools <- as_json(provider, unname(tools)) if (!is.null(type)) { - response_format <- list( - type = "json_schema", - json_schema = list( + # https://platform.openai.com/docs/api-reference/responses/create#responses-create-text + text <- list( + format = list( + type = "json_schema", name = "structured_data", schema = as_json(provider, type), strict = TRUE ) ) } else { - response_format <- NULL + text <- NULL } + # https://platform.openai.com/docs/api-reference/responses/create#responses-create-include params <- chat_params(provider, provider@params) - params$seed <- params$seed %||% provider@seed + if (isTRUE(params$log_probs)) { + include <- list("message.output_text.logprobs") + } else { + include <- NULL + } + params$log_probs <- NULL compact(list2( - messages = messages, + input = input, + include = include, model = provider@model, !!!params, stream = stream, - stream_options = if (stream) list(include_usage = TRUE), tools = tools, - response_format = response_format + text = text, + store = FALSE )) } @@ -194,11 +190,8 @@ method(chat_params, ProviderOpenAI) <- function(provider, params) { temperature = "temperature", top_p = "top_p", frequency_penalty = "frequency_penalty", - presence_penalty = "presence_penalty", - seed = "seed", - max_tokens = "max_completion_tokens", - logprobs = "log_probs", - stop = "stop_sequences" + max_tokens = "max_output_tokens", + log_probs = "log_probs" ) ) } @@ -213,10 +206,9 @@ method(stream_parse, ProviderOpenAI) <- function(provider, event) { jsonlite::parse_json(event$data) } method(stream_text, ProviderOpenAI) <- function(provider, event) { - if (length(event$choices) == 0) { - NULL - } else { - event$choices[[1]]$delta[["content"]] + # https://platform.openai.com/docs/api-reference/responses-streaming/response/output_text/delta + if (event$type == "response.output_text.delta") { + event$delta } } method(stream_merge_chunks, ProviderOpenAI) <- function( @@ -224,56 +216,43 @@ method(stream_merge_chunks, ProviderOpenAI) <- function( result, chunk ) { - if (is.null(result)) { - chunk - } else { - merge_dicts(result, chunk) + # https://platform.openai.com/docs/api-reference/responses-streaming/response/completed + if (chunk$type == "response.completed") { + chunk$response } } + method(value_turn, ProviderOpenAI) <- function( provider, result, has_type = FALSE ) { - if (has_name(result$choices[[1]], "delta")) { - # streaming - message <- result$choices[[1]]$delta - } else { - message <- result$choices[[1]]$message - } - - if (has_type) { - if (is_string(message$content)) { - json <- jsonlite::parse_json(message$content[[1]]) + contents <- lapply(result$output, function(output) { + if (output$type == "message") { + if (has_type) { + ContentJson(jsonlite::parse_json(output$content[[1]]$text)) + } else { + ContentText(output$content[[1]]$text) + } + } else if (output$type == "function_call") { + arguments <- jsonlite::parse_json(output$arguments) + ContentToolRequest(output$id, output$name, arguments) } else { - json <- message$content - } - content <- list(ContentJson(json)) - } else { - content <- lapply(message$content, as_content) - } - if (has_name(message, "tool_calls")) { - calls <- lapply(message$tool_calls, function(call) { - name <- call$`function`$name - # TODO: record parsing error - args <- tryCatch( - jsonlite::parse_json(call$`function`$arguments), - error = function(cnd) list() + browser() + cli::cli_abort( + "Unknown content type {.str {content$type}}.", + .internal = TRUE ) - ContentToolRequest(name = name, arguments = args, id = call$id) - }) - content <- c(content, calls) - } + } + }) + + # cached_tokens <- result$usage$input_token_details$cached_tokens tokens <- tokens_log( provider, - input = result$usage$prompt_tokens, - output = result$usage$completion_tokens - ) - assistant_turn( - content, - json = result, - tokens = tokens + input = result$usage$input_tokens, + output = result$usage$output_tokens ) + assistant_turn(contents = contents, json = result, tokens = tokens) } # ellmer -> OpenAI -------------------------------------------------------------- @@ -284,51 +263,42 @@ method(as_json, list(ProviderOpenAI, Turn)) <- function(provider, x) { list(role = "system", content = x@contents[[1]]@text) ) } else if (x@role == "user") { - # Each tool result needs to go in its own message with role "tool" - is_tool <- map_lgl(x@contents, S7_inherits, ContentToolResult) - content <- as_json(provider, x@contents[!is_tool]) - if (length(content) > 0) { - user <- list(list(role = "user", content = content)) - } else { - user <- list() - } - - tools <- lapply(x@contents[is_tool], function(tool) { - list( - role = "tool", - content = tool_string(tool), - tool_call_id = tool@request@id - ) + lapply(x@contents, function(x) { + if (S7_inherits(x, ContentText)) { + list(role = "user", content = x@text) + } else { + as_json(provider, x) + } }) - - c(user, tools) } else if (x@role == "assistant") { - # Tool requests come out of content and go into own argument - is_tool <- map_lgl(x@contents, is_tool_request) - content <- as_json(provider, x@contents[!is_tool]) - tool_calls <- as_json(provider, x@contents[is_tool]) - - list( - compact(list( - role = "assistant", - content = content, - tool_calls = tool_calls - )) - ) + as_json(provider, x@contents) } else { cli::cli_abort("Unknown role {x@role}", .internal = TRUE) } } method(as_json, list(ProviderOpenAI, ContentText)) <- function(provider, x) { - list(type = "text", text = x@text) + # OpenAI uses a different format dependening on whether the text is provided + # by the user or generated by the assistant. Since ellmer content types don't + # distinguish, this method generates the assistant content and we special + # case the + list( + role = "assistant", + content = x@text + ) } method(as_json, list(ProviderOpenAI, ContentImageRemote)) <- function( provider, x ) { - list(type = "image_url", image_url = list(url = x@url)) + list( + type = "message", + role = "user", + content = list( + list(type = "input_image", image_url = x@url) + ) + ) } method(as_json, list(ProviderOpenAI, ContentImageInline)) <- function( @@ -336,9 +306,13 @@ method(as_json, list(ProviderOpenAI, ContentImageInline)) <- function( x ) { list( - type = "image_url", - image_url = list( - url = paste0("data:", x@type, ";base64,", x@data) + type = "message", + role = "user", + content = list( + list( + type = "input_image", + image_url = paste0("data:", x@type, ";base64,", x@data) + ) ) ) } @@ -347,23 +321,32 @@ method(as_json, list(ProviderOpenAI, ContentToolRequest)) <- function( provider, x ) { - json_args <- jsonlite::toJSON(x@arguments) list( - id = x@id, - `function` = list(name = x@name, arguments = json_args), - type = "function" + type = "function_call", + call_id = x@id, + name = x@name, + arguments = jsonlite::toJSON(x@arguments) + ) +} + +method(as_json, list(ProviderOpenAI, ContentToolResult)) <- function( + provider, + x +) { + list( + type = "function_call_output", + call_id = x@request@id, + output = tool_string(x) ) } method(as_json, list(ProviderOpenAI, ToolDef)) <- function(provider, x) { list( type = "function", - "function" = compact(list( - name = x@name, - description = x@description, - strict = TRUE, - parameters = as_json(provider, x@arguments) - )) + name = x@name, + description = x@description, + strict = TRUE, + parameters = as_json(provider, x@arguments) ) } @@ -422,7 +405,7 @@ method(batch_submit, ProviderOpenAI) <- function( list( custom_id = paste0("chat-", i), method = "POST", - url = "/v1/chat/completions", + url = "/v1/responses", body = body ) }) diff --git a/R/provider.R b/R/provider.R index f9b3a92f..d00f6cf9 100644 --- a/R/provider.R +++ b/R/provider.R @@ -137,6 +137,8 @@ stream_parse <- new_generic( S7_dispatch() } ) + +# Extract text that should be printed to the console stream_text <- new_generic( "stream_text", "provider", diff --git a/man/chat_openai.Rd b/man/chat_openai.Rd index 9c350270..84bfd64f 100644 --- a/man/chat_openai.Rd +++ b/man/chat_openai.Rd @@ -10,8 +10,7 @@ chat_openai( base_url = "https://api.openai.com/v1", api_key = openai_key(), model = NULL, - params = NULL, - seed = lifecycle::deprecated(), + params = params(), api_args = list(), echo = c("none", "output", "all") ) @@ -35,9 +34,6 @@ Use \code{models_openai()} to see all options.} \item{params}{Common model parameters, usually created by \code{\link[=params]{params()}}.} -\item{seed}{Optional integer seed that ChatGPT uses to try and make output -more reproducible.} - \item{api_args}{Named list of arbitrary extra arguments appended to the body of every chat API call. Combined with the body object generated by ellmer with \code{\link[=modifyList]{modifyList()}}.} @@ -51,6 +47,9 @@ when running at the console). } Note this only affects the \code{chat()} method.} + +\item{seed}{Optional integer seed that ChatGPT uses to try and make output +more reproducible.} } \value{ A \link{Chat} object. diff --git a/tests/testthat/_snaps/provider-openai.md b/tests/testthat/_snaps/provider-openai.md index 78b3be54..67b0cc15 100644 --- a/tests/testthat/_snaps/provider-openai.md +++ b/tests/testthat/_snaps/provider-openai.md @@ -13,12 +13,3 @@ Error in `method(as_json, list(ellmer::ProviderOpenAI, ellmer::TypeObject))`: ! `.additional_properties` not supported for OpenAI. -# seed is deprecated, but still honored - - Code - chat <- chat_openai_test(seed = 1) - Condition - Warning: - The `seed` argument of `chat_openai()` is deprecated as of ellmer 0.2.0. - i Please use the `params` argument instead. - diff --git a/tests/testthat/_vcr/openai-v2-image.yml b/tests/testthat/_vcr/openai-v2-image.yml new file mode 100644 index 00000000..4f4223c8 --- /dev/null +++ b/tests/testthat/_vcr/openai-v2-image.yml @@ -0,0 +1,192 @@ +http_interactions: +- request: + method: POST + uri: https://api.openai.com/v1/responses + body: + string: '{"input":[{"role":"system","content":"Be terse."},{"role":"user","content":"What''s + in this image? (Be sure to mention the outside shape)"},{"type":"message","role":"user","content":[{"type":"input_image","image_url":""}]}],"model":"gpt-4.1-mini","temperature":0,"stream":false}' + response: + status: 200 + headers: + alt-svc: h3=":443"; ma=86400 + cf-cache-status: DYNAMIC + cf-ray: 95ca7b5d9da3fbc2-IAH + content-encoding: gzip + content-type: application/json + date: Wed, 09 Jul 2025 20:13:27 GMT + openai-organization: user-wfrajlxs4zxa4ixjljhmt9vx + openai-processing-ms: '1609' + openai-version: '2020-10-01' + server: cloudflare + set-cookie: + - __cf_bm=zHPF_HVxmXn4KrxM3QnvQe_YZObUibuNDkpX6DeO6LU-1752092007-1.0.1.1-K68.JK3Gr4CHhNSiwypJBoh8SlXmN09NjgeG4Z9Eio090u9AfEOktfKIin4OC2T7poCYpcVW4baR7Sz1qQf4gBeZ8DT.eg.sVFLS1NDrDvU; + path=/; expires=Wed, 09-Jul-25 20:43:27 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=n2oPW9AhUbrVqcjpmLWZOw5pxQ0uW9ia7x5ADUNvlsI-1752092007742-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + strict-transport-security: max-age=31536000; includeSubDomains; preload + x-content-type-options: nosniff + x-ratelimit-limit-requests: '30000' + x-ratelimit-remaining-requests: '29999' + x-ratelimit-reset-requests: 2ms + x-request-id: req_154ff4d0d7a1e6226cb1600186802af1 + body: + string: |- + { + "id": "resp_686ecd6616d4819a998835d2109773a6098389edbd29dc68", + "object": "response", + "created_at": 1752092006, + "status": "completed", + "background": false, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-mini-2025-04-14", + "output": [ + { + "id": "msg_686ecd669164819abff00918809caf7e098389edbd29dc68", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The image is inside a hexagon shape. It shows a red silhouette of a baseball player swinging a bat at a ball with \"www\" on it. Above the player is the text \"htr2\" in white cursive. The background is dark blue." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "reasoning": { + "effort": null, + "summary": null + }, + "service_tier": "default", + "store": true, + "temperature": 0.0, + "text": { + "format": { + "type": "text" + } + }, + "tool_choice": "auto", + "tools": [], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 213, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 54, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 267 + }, + "user": null, + "metadata": {} + } + recorded_at: 2025-07-09 20:13:27 +- request: + method: POST + uri: https://api.openai.com/v1/responses + body: + string: '{"input":[{"role":"system","content":"Be terse."},{"role":"user","content":"What''s + in this image? (Be sure to mention the outside shape)"},{"type":"message","role":"user","content":[{"type":"input_image","image_url":"https://httr2.r-lib.org/logo.png"}]}],"model":"gpt-4.1-mini","temperature":0,"stream":false}' + response: + status: 200 + headers: + alt-svc: h3=":443"; ma=86400 + cf-cache-status: DYNAMIC + cf-ray: 95ca7b6979c4fbc2-IAH + content-encoding: gzip + content-type: application/json + date: Wed, 09 Jul 2025 20:13:29 GMT + openai-organization: user-wfrajlxs4zxa4ixjljhmt9vx + openai-processing-ms: '1466' + openai-version: '2020-10-01' + server: cloudflare + set-cookie: + - __cf_bm=N.yCHntMW_BbXjTGLKtVRYnmj532DjrqrbEt7OG8e1Y-1752092009-1.0.1.1-oPYdg.B80WyV41ZuttNvHo2kpf8Y6bFHFFEpH4HD_lhQvOQ2EN56mBaor3AMrWyzsSyrkazegngd2dwbwbNcjd6W16iBhjw8BRycpdJotlk; + path=/; expires=Wed, 09-Jul-25 20:43:29 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=C4HEucpbuQHVOb5UZJ9vhv9qJGIJVBfxcucsyyWT2BI-1752092009480-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + strict-transport-security: max-age=31536000; includeSubDomains; preload + x-content-type-options: nosniff + x-ratelimit-limit-requests: '30000' + x-ratelimit-remaining-requests: '29999' + x-ratelimit-reset-requests: 2ms + x-request-id: req_782cb6f9801f888542cccc6f4ed0eb9c + body: + string: |- + { + "id": "resp_686ecd67fbb08198aefcd3b0c40c452104413787094ed5c1", + "object": "response", + "created_at": 1752092008, + "status": "completed", + "background": false, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-mini-2025-04-14", + "output": [ + { + "id": "msg_686ecd68a2408198b209e5d3096b5eca04413787094ed5c1", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "The image is inside a hexagon shape. It shows a red silhouette of a baseball player swinging a bat, with the text \"httr2\" above and a small ball labeled \"www\" near the bat. The background is dark blue." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "reasoning": { + "effort": null, + "summary": null + }, + "service_tier": "default", + "store": true, + "temperature": 0.0, + "text": { + "format": { + "type": "text" + } + }, + "tool_choice": "auto", + "tools": [], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 151, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 50, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 201 + }, + "user": null, + "metadata": {} + } + recorded_at: 2025-07-09 20:13:29 +recorded_with: VCR-vcr/1.7.0.91, webmockr/2.1.0 diff --git a/tests/testthat/_vcr/openai-v2-tool.yml b/tests/testthat/_vcr/openai-v2-tool.yml new file mode 100644 index 00000000..dac97d64 --- /dev/null +++ b/tests/testthat/_vcr/openai-v2-tool.yml @@ -0,0 +1,478 @@ +http_interactions: +- request: + method: POST + uri: https://api.openai.com/v1/responses + body: + string: '{"input":[{"role":"system","content":"Always use a tool to answer. + Reply with ''It is ____.''."},{"role":"user","content":"What''s the current + date in Y-M-D format?"}],"model":"gpt-4.1-nano","temperature":0,"stream":false,"tools":[{"type":"function","name":"current_date","description":"Return + the current date","strict":true,"parameters":{"type":"object","description":"","properties":{},"required":[],"additionalProperties":false}}]}' + response: + status: 200 + headers: + alt-svc: h3=":443"; ma=86400 + cf-cache-status: DYNAMIC + cf-ray: 95ca7b355ffefbc2-IAH + content-encoding: gzip + content-type: application/json + date: Wed, 09 Jul 2025 20:13:20 GMT + openai-organization: user-wfrajlxs4zxa4ixjljhmt9vx + openai-processing-ms: '655' + openai-version: '2020-10-01' + server: cloudflare + set-cookie: + - __cf_bm=dEZLZ7JJO01ysLMlFZkiulBT_MURBo21xWXvlbfhGrc-1752092000-1.0.1.1-o_N3qQEumHJg_.RtBolD8mxG2t9wVOnEvn6bs2x8VNiAQnrwwYXFhguaQa4QjlZWIyWoNk5W3PP1Wj4ili_RenSPdGrmmGwzqrA6FP5wTxw; + path=/; expires=Wed, 09-Jul-25 20:43:20 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=0QZ8BzsIngM4fTc0IyAFkEAgdFPoDBvMHrKqew0xbeE-1752092000348-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + strict-transport-security: max-age=31536000; includeSubDomains; preload + x-content-type-options: nosniff + x-ratelimit-limit-requests: '30000' + x-ratelimit-limit-tokens: '150000000' + x-ratelimit-remaining-requests: '29999' + x-ratelimit-remaining-tokens: '149999724' + x-ratelimit-reset-requests: 2ms + x-ratelimit-reset-tokens: 0s + x-request-id: req_8f800070b1c98e15cc948fcdd03b3409 + body: + string: |- + { + "id": "resp_686ecd5fa7708198b54d73f42d3575e903f0dc7b529e5cc4", + "object": "response", + "created_at": 1752091999, + "status": "completed", + "background": false, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-nano-2025-04-14", + "output": [ + { + "id": "fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4", + "type": "function_call", + "status": "completed", + "arguments": "{}", + "call_id": "call_NVX5fmiDJcbNNksOv1ZLZpKd", + "name": "current_date" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "reasoning": { + "effort": null, + "summary": null + }, + "service_tier": "default", + "store": true, + "temperature": 0.0, + "text": { + "format": { + "type": "text" + } + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Return the current date", + "name": "current_date", + "parameters": { + "type": "object", + "description": "", + "properties": {}, + "required": [], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 59, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 11, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 70 + }, + "user": null, + "metadata": {} + } + recorded_at: 2025-07-09 20:13:20 +- request: + method: POST + uri: https://api.openai.com/v1/responses + body: + string: '{"input":[{"role":"system","content":"Always use a tool to answer. + Reply with ''It is ____.''."},{"role":"user","content":"What''s the current + date in Y-M-D format?"},{"type":"function_call","call_id":"fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4","name":"current_date","arguments":"{}"},{"type":"function_call_output","call_id":"fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4","output":"2024-01-01"}],"model":"gpt-4.1-nano","temperature":0,"stream":false,"tools":[{"type":"function","name":"current_date","description":"Return + the current date","strict":true,"parameters":{"type":"object","description":"","properties":{},"required":[],"additionalProperties":false}}]}' + response: + status: 200 + headers: + alt-svc: h3=":443"; ma=86400 + cf-cache-status: DYNAMIC + cf-ray: 95ca7b3bccccfbc2-IAH + content-encoding: gzip + content-type: application/json + date: Wed, 09 Jul 2025 20:13:21 GMT + openai-organization: user-wfrajlxs4zxa4ixjljhmt9vx + openai-processing-ms: '499' + openai-version: '2020-10-01' + server: cloudflare + set-cookie: + - __cf_bm=6HYLyA43jpx2_qbGyqK1bgEl_c5p3xPwcPrS9aaTSLI-1752092001-1.0.1.1-_zLVg1UtyhWABsZ8h442mXmh8kH8MgPcWbp6P2CTcgZ_qJqBRpvwWiKQpwcSZZbHZvLJi8I5yb73giA6gm0nk_AoCIii29UIyjazPj1l.Ss; + path=/; expires=Wed, 09-Jul-25 20:43:21 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=mDXOTZuG33jSJzFolMT_XRET5LIBzVgppvVW5tNNcyw-1752092001220-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + strict-transport-security: max-age=31536000; includeSubDomains; preload + x-content-type-options: nosniff + x-ratelimit-limit-requests: '30000' + x-ratelimit-limit-tokens: '150000000' + x-ratelimit-remaining-requests: '29999' + x-ratelimit-remaining-tokens: '149999702' + x-ratelimit-reset-requests: 2ms + x-ratelimit-reset-tokens: 0s + x-request-id: req_e4f2dae041ccd4d98ed439b899123393 + body: + string: |- + { + "id": "resp_686ecd60b1148199a1705bb6349b461908913f2ae1ee9fd1", + "object": "response", + "created_at": 1752092000, + "status": "completed", + "background": false, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-nano-2025-04-14", + "output": [ + { + "id": "msg_686ecd60f994819982c5e41636f7eb7f08913f2ae1ee9fd1", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "It is 2024-01-01." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "reasoning": { + "effort": null, + "summary": null + }, + "service_tier": "default", + "store": true, + "temperature": 0.0, + "text": { + "format": { + "type": "text" + } + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Return the current date", + "name": "current_date", + "parameters": { + "type": "object", + "description": "", + "properties": {}, + "required": [], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 81, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 12, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 93 + }, + "user": null, + "metadata": {} + } + recorded_at: 2025-07-09 20:13:21 +- request: + method: POST + uri: https://api.openai.com/v1/responses + body: + string: '{"input":[{"role":"system","content":"Always use a tool to answer. + Reply with ''It is ____.''."},{"role":"user","content":"What''s the current + date in Y-M-D format?"},{"type":"function_call","call_id":"fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4","name":"current_date","arguments":"{}"},{"type":"function_call_output","call_id":"fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4","output":"2024-01-01"},{"role":"assistant","content":"It + is 2024-01-01."},{"role":"user","content":"What month is it? Provide the full + name"}],"model":"gpt-4.1-nano","temperature":0,"stream":false,"tools":[{"type":"function","name":"current_date","description":"Return + the current date","strict":true,"parameters":{"type":"object","description":"","properties":{},"required":[],"additionalProperties":false}},{"type":"function","name":"current_month","description":"Return + the full name of the current month","strict":true,"parameters":{"type":"object","description":"","properties":{},"required":[],"additionalProperties":false}}]}' + response: + status: 200 + headers: + alt-svc: h3=":443"; ma=86400 + cf-cache-status: DYNAMIC + cf-ray: 95ca7b41d92bfbc2-IAH + content-encoding: gzip + content-type: application/json + date: Wed, 09 Jul 2025 20:13:22 GMT + openai-organization: user-wfrajlxs4zxa4ixjljhmt9vx + openai-processing-ms: '1080' + openai-version: '2020-10-01' + server: cloudflare + set-cookie: + - __cf_bm=Fr.ZCC_iuBwD.Gcqe7kNssqd1DDRhaM7WyufIV2P8VI-1752092002-1.0.1.1-o8lL0pufSaYNQtmJndgqoinSxAG8R_XO8XCan1FeJhFqasyhkBRx76uSBXbyt3nBiojRnwtKGGYe2e0jVLHL5NORSY2IaFDYmbTmL5A.APM; + path=/; expires=Wed, 09-Jul-25 20:43:22 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=o7BVmzjPXPVtjEgv2Hr9sW3W4x1lGlqcqdVLsw.C9TI-1752092002776-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + strict-transport-security: max-age=31536000; includeSubDomains; preload + x-content-type-options: nosniff + x-ratelimit-limit-requests: '30000' + x-ratelimit-limit-tokens: '150000000' + x-ratelimit-remaining-requests: '29999' + x-ratelimit-remaining-tokens: '149999658' + x-ratelimit-reset-requests: 2ms + x-ratelimit-reset-tokens: 0s + x-request-id: req_2ca939f5e035903f941c4e629da0554b + body: + string: |- + { + "id": "resp_686ecd61a7e4819ab1b4c1fe76efbf810d79be5ad2e4c9f9", + "object": "response", + "created_at": 1752092001, + "status": "completed", + "background": false, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-nano-2025-04-14", + "output": [ + { + "id": "fc_686ecd626ef4819aa92fa44a81f469780d79be5ad2e4c9f9", + "type": "function_call", + "status": "completed", + "arguments": "{}", + "call_id": "call_ywiVsy2WsoQDgndJCNJGlITT", + "name": "current_month" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "reasoning": { + "effort": null, + "summary": null + }, + "service_tier": "default", + "store": true, + "temperature": 0.0, + "text": { + "format": { + "type": "text" + } + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Return the current date", + "name": "current_date", + "parameters": { + "type": "object", + "description": "", + "properties": {}, + "required": [], + "additionalProperties": false + }, + "strict": true + }, + { + "type": "function", + "description": "Return the full name of the current month", + "name": "current_month", + "parameters": { + "type": "object", + "description": "", + "properties": {}, + "required": [], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 125, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 11, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 136 + }, + "user": null, + "metadata": {} + } + recorded_at: 2025-07-09 20:13:23 +- request: + method: POST + uri: https://api.openai.com/v1/responses + body: + string: '{"input":[{"role":"system","content":"Always use a tool to answer. + Reply with ''It is ____.''."},{"role":"user","content":"What''s the current + date in Y-M-D format?"},{"type":"function_call","call_id":"fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4","name":"current_date","arguments":"{}"},{"type":"function_call_output","call_id":"fc_686ecd602d648198912c8251e607a78803f0dc7b529e5cc4","output":"2024-01-01"},{"role":"assistant","content":"It + is 2024-01-01."},{"role":"user","content":"What month is it? Provide the full + name"},{"type":"function_call","call_id":"fc_686ecd626ef4819aa92fa44a81f469780d79be5ad2e4c9f9","name":"current_month","arguments":"{}"},{"type":"function_call_output","call_id":"fc_686ecd626ef4819aa92fa44a81f469780d79be5ad2e4c9f9","output":"February"}],"model":"gpt-4.1-nano","temperature":0,"stream":false,"tools":[{"type":"function","name":"current_date","description":"Return + the current date","strict":true,"parameters":{"type":"object","description":"","properties":{},"required":[],"additionalProperties":false}},{"type":"function","name":"current_month","description":"Return + the full name of the current month","strict":true,"parameters":{"type":"object","description":"","properties":{},"required":[],"additionalProperties":false}}]}' + response: + status: 200 + headers: + alt-svc: h3=":443"; ma=86400 + cf-cache-status: DYNAMIC + cf-ray: 95ca7b4c8a5bfbc2-IAH + content-encoding: gzip + content-type: application/json + date: Wed, 09 Jul 2025 20:13:23 GMT + openai-organization: user-wfrajlxs4zxa4ixjljhmt9vx + openai-processing-ms: '514' + openai-version: '2020-10-01' + server: cloudflare + set-cookie: + - __cf_bm=7C_SnpkDbGY5UE1B9R4Xl5aTYnR7ZfdafniZHk9eSxM-1752092003-1.0.1.1-gmDOz153YcR4aK_l3Xv6Ne45.0N4kTJMDhF4iB0TBC4iupgyyLC1kwGFYIYcBDVaKZ08K.2v.5Q9TSMDdJXDf9hKvop3JZfUFIUK_7UJlWg; + path=/; expires=Wed, 09-Jul-25 20:43:23 GMT; domain=.api.openai.com; HttpOnly; + Secure; SameSite=None + - _cfuvid=7D6A9t2qxmCAvbnPOkFV2dI1._VBT1OQuITvQvVvV3Y-1752092003908-0.0.1.1-604800000; + path=/; domain=.api.openai.com; HttpOnly; Secure; SameSite=None + strict-transport-security: max-age=31536000; includeSubDomains; preload + x-content-type-options: nosniff + x-ratelimit-limit-requests: '30000' + x-ratelimit-limit-tokens: '150000000' + x-ratelimit-remaining-requests: '29999' + x-ratelimit-remaining-tokens: '149999641' + x-ratelimit-reset-requests: 2ms + x-ratelimit-reset-tokens: 0s + x-request-id: req_10603c5fb3997c1f17abfc7ddcc8ceeb + body: + string: |- + { + "id": "resp_686ecd635f50819b953d823ae8db8f3e06801cb0f54bbb71", + "object": "response", + "created_at": 1752092003, + "status": "completed", + "background": false, + "error": null, + "incomplete_details": null, + "instructions": null, + "max_output_tokens": null, + "max_tool_calls": null, + "model": "gpt-4.1-nano-2025-04-14", + "output": [ + { + "id": "msg_686ecd63af7c819b89fdcf9ee80a23b406801cb0f54bbb71", + "type": "message", + "status": "completed", + "content": [ + { + "type": "output_text", + "annotations": [], + "logprobs": [], + "text": "It is February." + } + ], + "role": "assistant" + } + ], + "parallel_tool_calls": true, + "previous_response_id": null, + "reasoning": { + "effort": null, + "summary": null + }, + "service_tier": "default", + "store": true, + "temperature": 0.0, + "text": { + "format": { + "type": "text" + } + }, + "tool_choice": "auto", + "tools": [ + { + "type": "function", + "description": "Return the current date", + "name": "current_date", + "parameters": { + "type": "object", + "description": "", + "properties": {}, + "required": [], + "additionalProperties": false + }, + "strict": true + }, + { + "type": "function", + "description": "Return the full name of the current month", + "name": "current_month", + "parameters": { + "type": "object", + "description": "", + "properties": {}, + "required": [], + "additionalProperties": false + }, + "strict": true + } + ], + "top_logprobs": 0, + "top_p": 1.0, + "truncation": "disabled", + "usage": { + "input_tokens": 142, + "input_tokens_details": { + "cached_tokens": 0 + }, + "output_tokens": 6, + "output_tokens_details": { + "reasoning_tokens": 0 + }, + "total_tokens": 148 + }, + "user": null, + "metadata": {} + } + recorded_at: 2025-07-09 20:13:23 +recorded_with: VCR-vcr/1.7.0.91, webmockr/2.1.0 diff --git a/tests/testthat/helper-provider.R b/tests/testthat/helper-provider.R index 84cb9a43..b352de43 100644 --- a/tests/testthat/helper-provider.R +++ b/tests/testthat/helper-provider.R @@ -36,15 +36,15 @@ test_tools_simple <- function(chat_fun) { name = "current_date", description = "Return the current date" )) + result <- chat$chat("What's the current date in Y-M-D format?") + expect_match(result, "2024-01-01") + chat$register_tool(tool( function() "February", name = "current_month", description = "Return the full name of the current month" )) - result <- chat$chat("What's the current date in Y-M-D format?") - expect_match(result, "2024-01-01") - result <- chat$chat("What month is it? Provide the full name") expect_match(result, "February") } diff --git a/tests/testthat/test-provider-openai.R b/tests/testthat/test-provider-openai.R index 92fe5e91..d06d27db 100644 --- a/tests/testthat/test-provider-openai.R +++ b/tests/testthat/test-provider-openai.R @@ -5,6 +5,9 @@ test_that("can make simple request", { resp <- chat$chat("What is 1 + 1?", echo = FALSE) expect_match(resp, "2") expect_equal(chat$last_turn()@tokens > 0, c(TRUE, TRUE)) + + resp <- chat$chat("Double that", echo = FALSE) + expect_match(resp, "4") }) test_that("can make simple streaming request", { @@ -17,21 +20,21 @@ test_that("can list models", { test_models(models_openai) }) - # Common provider interface ----------------------------------------------- test_that("defaults are reported", { expect_snapshot(. <- chat_openai()) }) -test_that("supports standard parameters", { - chat_fun <- chat_openai_test +# No longer supports stop parameter +# test_that("supports standard parameters", { +# chat_fun <- chat_openai_test - test_params_stop(chat_fun) -}) +# test_params_stop(chat_fun) +# }) test_that("supports tool calling", { - vcr::local_cassette("openai-tool") + # vcr::local_cassette("openai-v2-tool") chat_fun <- chat_openai_test test_tools_simple(chat_fun) @@ -44,7 +47,7 @@ test_that("can extract data", { }) test_that("can use images", { - vcr::local_cassette("openai-image") + # vcr::local_cassette("openai-v2-image") # Needs mini to get shape correct chat_fun <- \(...) chat_openai_test(model = "gpt-4.1-mini", ...) @@ -56,13 +59,8 @@ test_that("can use images", { test_that("can retrieve log_probs (#115)", { chat <- chat_openai_test(params = params(log_probs = TRUE)) - pieces <- coro::collect(chat$stream("Hi")) - - logprobs <- chat$last_turn()@json$choices[[1]]$logprobs$content - expect_equal( - length(logprobs), - length(pieces) - 2 # leading "" + trailing \n - ) + chat$chat("Hi") + expect_length(chat$last_turn()@json$output[[1]]$content[[1]]$logprobs, 2) }) # Custom ----------------------------------------------------------------- @@ -87,11 +85,3 @@ test_that("as_json specialised for OpenAI", { ) ) }) - -test_that("seed is deprecated, but still honored", { - expect_snapshot(chat <- chat_openai_test(seed = 1)) - expect_equal(chat$get_provider()@params$seed, 1) - - # NULL is also ignored since that's what subclasses use - expect_no_warning(chat_openai_test(seed = NULL)) -})