foundryR and ellmer both make language-model work possible from R. They are not substitutes for every use case. ellmer is the better fit for provider-portable chat. foundryR is the better fit when the work is tied to Azure AI Foundry and the output needs to become data. The two are complementary: you can describe a structure once with ellmer’s type system and hand it to foundryR for strict, tibble-shaped extraction.
Comparison
| Question | Use foundryR | Use ellmer |
|---|---|---|
| Are you committed to Azure AI Foundry? | Yes | Sometimes |
| Do you need Azure AI Content Safety? | Yes | No |
| Do you need Azure’s Files and Batch APIs? | Yes | No |
| Do you need strict schema-constrained extraction as tibbles? | Yes | Sometimes |
| Do you need embeddings inside a dataframe workflow? | Yes | Sometimes |
Do you need step_foundry_embed() in a tidymodels
recipe? |
Yes | No |
| Do you need multi-provider chat? | No | Yes |
| Do you need interactive streaming chat? | No | Yes |
| Do you need a chat-first tool-calling agent interface? | Basic Responses API loop | Yes |
Where foundryR fits
Use foundryR when your work starts or ends in a dataframe. Common examples are:
- Coding open-ended survey responses with a strict JSON Schema.
- Running low-cost batch annotation jobs through Azure’s Batch API.
- Embedding documents for clustering, semantic search, or near-duplicate checks.
- Adding embeddings to tidymodels recipes.
- Moderating user text, detecting prompt injection, or checking groundedness.
- Keeping model metadata, citations, tool calls, and token use in tibble columns.
Azure’s own Batch API matters here. ellmer supports batch workflows for native OpenAI and Anthropic endpoints, but it does not target Azure’s Files and Batch API surface.
Where ellmer fits
Use ellmer when chat is the product interface. It gives you a provider-neutral chat abstraction, interactive streaming, and a mature chat-first tool-calling workflow. That is the right shape for assistants, notebooks where you want token streaming, or applications that may move between providers.
foundryR deliberately does not implement streaming. For streaming chat in R, use ellmer.
Interop: reuse an ellmer type as a foundryR schema
If you already describe your data with ellmer’s type system, you do
not have to rewrite it. as_foundry_schema() converts an
ellmer::type_object() into the strict JSON Schema that
foundry_extract() expects. This conversion is a local
operation – no network, no credentials – so its output is shown here
directly.
library(ellmer)
#>
#> Attaching package: 'ellmer'
#> The following objects are masked from 'package:foundryR':
#>
#> type_boolean, type_enum, type_number, type_string
# Describe the structure you want with ellmer's type system.
sentiment_spec <- type_object(
sentiment = type_enum(
c("positive", "negative", "neutral"),
description = "Overall sentiment of the response."
),
theme = type_string("A short theme label for the response.")
)
# Convert it into a foundryR schema for strict extraction.
sentiment_schema <- as_foundry_schema(sentiment_spec)
str(sentiment_schema)
#> List of 4
#> $ type : chr "object"
#> $ properties :List of 2
#> ..$ sentiment:List of 3
#> .. ..$ type : chr "string"
#> .. ..$ enum : 'AsIs' chr [1:3] "positive" "negative" "neutral"
#> .. ..$ description: chr "Overall sentiment of the response."
#> ..$ theme :List of 2
#> .. ..$ type : chr "string"
#> .. ..$ description: chr "A short theme label for the response."
#> $ required : 'AsIs' chr [1:2] "sentiment" "theme"
#> $ additionalProperties: logi FALSEExtraction with foundryR
Hand the converted schema to foundry_extract(). foundryR
sends each input through the Responses API with strict decoding and
returns one tidy row per input – the model’s output has already become
data.
foundry_extract(
c("The lesson was clear.", "I wanted more examples."),
schema = sentiment_schema
)
#> # A tibble: 2 × 10
#> .input_idx .input_text .response_id .status .output_text .error .error_msg
#> <int> <chr> <chr> <chr> <chr> <lgl> <chr>
#> 1 1 The lesson was… resp_06e39d… comple… "{\"sentime… FALSE NA
#> 2 2 I wanted more … resp_031136… comple… "{\"sentime… FALSE NA
#> # ℹ 3 more variables: raw_response <list>, sentiment <chr>, theme <chr>You can also build the same schema natively with foundryR’s
foundry_schema() and schema_*() constructors
when you would rather not depend on ellmer. See
vignette("responses-api") for the native path, and
vignette("content-safety") for using Azure Content Safety
as a pipeline gate.
The practical rule is simple: use ellmer when you need the best R
chat client, and use foundryR when you need Azure AI Foundry results as
data – and use as_foundry_schema() when you want both.