This proposal is an early design sketch by the Chrome built-in AI team to describe the problem below and solicit feedback on the proposed solution. It has not been approved to ship in Chrome.
Browsers and operating systems are increasingly expected to gain access to a language model. (Example, example, example.) Language models are known for their versatility. With enough creative prompting, they can help accomplish tasks as diverse as:
Although the Chrome built-in AI team is exploring purpose-built APIs for some of these use cases (e.g. translation, and perhaps in the future summarization and compose), we are also exploring a general-purpose "prompt API" which allows web developers to prompt a language model directly. This gives web developers access to many more capabilities, at the cost of requiring them to do their own prompt engineering.
Currently, web developers wishing to use language models must either call out to cloud APIs, or bring their own and run them using technologies like WebAssembly and WebGPU. By providing access to the browser or operating system's existing language model, we can provide the following benefits compared to cloud APIs:
Similarly, compared to bring-your-own-AI approaches, using a built-in language model can save the user's bandwidth, likely benefit from more optimizations, and have a lower barrier to entry for web developers.
Even more so than many other behind-a-flag APIs, the prompt API is an experiment, designed to help us understand web developers' use cases to inform a roadmap of purpose-built APIs. However, we want to publish an explainer to provide documentation and a public discussion place for the experiment while it is ongoing.
Our goals are to:
The following are explicit non-goals:
The following are potential goals we are not yet certain of:
Both of these potential goals could pose challenges to interoperability, so we want to investigate more how important such functionality is to developers to find the right tradeoff.
In this example, a single string is used to prompt the API, which is assumed to come from the user. The returned response is from the language model.
const session = await ai.languageModel.create();
// Prompt the model and wait for the whole result to come back.
const result = await session.prompt("Write me a poem.");
console.log(result);
// Prompt the model and stream the result:
const stream = await session.promptStreaming("Write me an extra-long poem.");
for await (const chunk of stream) {
console.log(chunk);
}
The language model can be configured with a special "system prompt" which gives it the context for future interactions:
const session = await ai.languageModel.create({
systemPrompt: "Pretend to be an eloquent hamster."
});
console.log(await session.prompt("What is your favorite food?"));
The system prompt is special, in that the language model will not respond to it, and it will be preserved even if the context window otherwise overflows due to too many calls to prompt()
.
If the system prompt is too large (see below), then the promise will be rejected with a "QuotaExceededError"
DOMException
.
If developers want to provide examples of the user/assistant interaction, they can use the initialPrompts
array. This aligns with the common "chat completions API" format of { role, content }
pairs, including a "system"
role which can be used instead of the systemPrompt
option shown above.
const session = await ai.languageModel.create({
initialPrompts: [
{ role: "system", content: "Predict up to 5 emojis as a response to a comment. Output emojis, comma-separated." },
{ role: "user", content: "This is amazing!" },
{ role: "assistant", content: "❤️, ➕" },
{ role: "user", content: "LGTM" },
{ role: "assistant", content: "👍, 🚢" }
]
});
// Clone an existing session for efficiency, instead of recreating one each time.
async function predictEmoji(comment) {
const freshSession = await session.clone();
return await freshSession.prompt(comment);
}
const result1 = await predictEmoji("Back to the drawing board");
const result2 = await predictEmoji("This code is so good you should get promoted");
(Note that merely creating a session does not cause any new responses from the language model. We need to call prompt()
or promptStreaming()
to get a response.)
Some details on error cases:
systemPrompt
and a { role: "system" }
prompt in initialPrompts
, or using multiple { role: "system" }
prompts, or placing the { role: "system" }
prompt anywhere besides at the 0th position in initialPrompts
, will reject with a TypeError
.systemPrompt
, if provided) is too large, then the promise will be rejected with a "QuotaExceededError"
DOMException
.Our examples so far have provided prompt()
and promptStreaming()
with a single string. Such cases assume messages will come from the user role. These methods can also take in objects in the { role, content }
format, or arrays of such objects, in case you want to provide multiple user or assistant messages before getting another assistant message:
const multiUserSession = await ai.languageModel.create({
systemPrompt: "You are a mediator in a discussion between two departments."
});
const result = await multiUserSession.prompt([
{ role: "user", content: "Marketing: We need more budget for advertising campaigns." },
{ role: "user", content: "Finance: We need to cut costs and advertising is on the list." },
{ role: "assistant", content: "Let's explore a compromise that satisfies both departments." }
]);
// `result` will contain a compromise proposal from the assistant.
Because of their special behavior of being preserved on context window overflow, system prompts cannot be provided this way.
A special case of the above is using the assistant role to emulate tool use or function-calling, by marking a response as coming from the assistant side of the conversation:
const session = await ai.languageModel.create({
systemPrompt: `
You are a helpful assistant. You have access to the following tools:
- calculator: A calculator. To use it, write "CALCULATOR: <expression>" where <expression> is a valid mathematical expression.
`
});
async function promptWithCalculator(prompt) {
const result = await session.prompt(prompt);
// Check if the assistant wants to use the calculator tool.
const match = /^CALCULATOR: (.*)$/.exec(result);
if (match) {
const expression = match[1];
const mathResult = evaluateMathExpression(expression);
// Add the result to the session so it's in context going forward.
await session.prompt({ role: "assistant", content: mathResult });
// Return it as if that's what the assistant said to the user.
return mathResult;
}
// The assistant didn't want to use the calculator. Just return its response.
return result;
}
console.log(await promptWithCalculator("What is 2 + 2?"));
We'll likely explore more specific APIs for tool- and function-calling in the future; follow along in issue #7.
In addition to the systemPrompt
and initialPrompts
options shown above, the currently-configurable options are temperature and top-K. More information about the values for these parameters can be found using the capabilities()
API explained below.
const customSession = await ai.languageModel.create({
temperature: 0.8,
topK: 10
});
const capabilities = await ai.languageModel.capabilities();
const slightlyHighTemperatureSession = await ai.languageModel.create({
temperature: Math.max(
capabilities.defaultTemperature * 1.2,
capabilities.maxTemperature
),
topK: 10
});
// capabilities also contains defaultTopK and maxTopK.
Each language model session consists of a persistent series of interactions with the model:
const session = await ai.languageModel.create({
systemPrompt: "You are a friendly, helpful assistant specialized in clothing choices."
});
const result = await session.prompt(`
What should I wear today? It's sunny and I'm unsure between a t-shirt and a polo.
`);
console.log(result);
const result2 = await session.prompt(`
That sounds great, but oh no, it's actually going to rain! New advice??
`);
Multiple unrelated continuations of the same prompt can be set up by creating a session and then cloning it:
const session = await ai.languageModel.create({
systemPrompt: "You are a friendly, helpful assistant specialized in clothing choices."
});
const session2 = await session.clone();
The clone operation can be aborted using an AbortSignal
:
const controller = new AbortController();
const session2 = await session.clone({ signal: controller.signal });
A language model session can be destroyed, either by using an AbortSignal
passed to the create()
method call:
const controller = new AbortController();
stopButton.onclick = () => controller.abort();
const session = await ai.languageModel.create({ signal: controller.signal });
or by calling destroy()
on the session:
stopButton.onclick = () => session.destroy();
Destroying a session will have the following effects:
If done before the promise returned by create()
is settled:
Stop signaling any ongoing download progress for the language model. (The browser may also abort the download, or may continue it. Either way, no further downloadprogress
events will fire.)
Reject the create()
promise.
Otherwise:
Reject any ongoing calls to prompt()
.
Error any ReadableStream
s returned by promptStreaming()
.
Most importantly, destroying the session allows the user agent to unload the language model from memory, if no other APIs or sessions are using it.
In all cases the exception used for rejecting promises or erroring ReadableStream
s will be an "AbortError"
DOMException
, or the given abort reason.
The ability to manually destroy a session allows applications to free up memory without waiting for garbage collection, which can be useful since language models can be quite large.
Specific calls to prompt()
or promptStreaming()
can be aborted by passing an AbortSignal
to them:
const controller = new AbortController();
stopButton.onclick = () => controller.abort();
const result = await session.prompt("Write me a poem", { signal: controller.signal });
Note that because sessions are stateful, and prompts can be queued, aborting a specific prompt is slightly complicated:
A given language model session will have a maximum number of tokens it can process. Developers can check their current usage and progress toward that limit by using the following properties on the session object:
console.log(`${session.tokensSoFar}/${session.maxTokens} (${session.tokensLeft} left)`);
To know how many tokens a string will consume, without actually processing it, developers can use the countPromptTokens()
method:
const numTokens = await session.countPromptTokens(promptString);
Some notes on this API:
AbortSignal
, i.e. session.countPromptTokens(promptString, { signal })
.It's possible to send a prompt that causes the context window to overflow. That is, consider a case where session.countPromptTokens(promptString) > session.tokensLeft
before calling session.prompt(promptString)
, and then the web developer calls session.prompt(promptString)
anyway. In such cases, the initial portions of the conversation with the language model will be removed, one prompt/response pair at a time, until enough tokens are available to process the new prompt. The exception is the system prompt, which is never removed. If it's not possible to remove enough tokens from the conversation history to process the new prompt, then the prompt()
or promptStreaming()
call will fail with an "QuotaExceededError"
DOMException
and nothing will be removed.
Such overflows can be detected by listening for the "contextoverflow"
event on the session:
session.addEventListener("contextoverflow", () => {
console.log("Context overflow!");
});
In all our above examples, we call ai.languageModel.create()
and assume it will always succeed.
However, sometimes a language model needs to be downloaded before the API can be used. In such cases, immediately calling create()
will start the download, which might take a long time. The capabilities API gives you insight into the download status of the model:
const capabilities = await ai.languageModel.capabilities();
console.log(capabilities.available);
The capabilities.available
property is a string that can take one of three values:
"no"
, indicating the device or browser does not support prompting a language model at all."after-download"
, indicating the device or browser supports prompting a language model, but it needs to be downloaded before it can be used."readily"
, indicating the device or browser supports prompting a language model and it’s ready to be used without any downloading steps.In the "after-download"
case, developers might want to have users confirm before you call create()
to start the download, since doing so uses up significant bandwidth and users might not be willing to wait for a large download before using the site or feature.
Note that regardless of the return value of available
, create()
might also fail, if either the download fails or the session creation fails.
The capabilities API also contains other information about the model:
defaultTemperature
, maxTemperature
, defaultTopK
, and maxTopK
properties giving information about the model's sampling parameters.languageAvailable(languageTag)
, which returns "no"
, "after-download"
, or "readily"
to indicate whether the model supports conversing in a given human language.In cases where the model needs to be downloaded as part of creation, you can monitor the download progress (e.g. in order to show your users a progress bar) using code such as the following:
const session = await ai.languageModel.create({
monitor(m) {
m.addEventListener("downloadprogress", e => {
console.log(`Downloaded ${e.loaded} of ${e.total} bytes.`);
});
}
});
If the download fails, then downloadprogress
events will stop being emitted, and the promise returned by create()
will be rejected with a "NetworkError
" DOMException
.
// Shared self.ai APIs
partial interface WindowOrWorkerGlobalScope {
[Replaceable, SecureContext] readonly attribute AI ai;
};
[Exposed=(Window,Worker), SecureContext]
interface AI {
readonly attribute AILanguageModelFactory languageModel;
};
[Exposed=(Window,Worker), SecureContext]
interface AICreateMonitor : EventTarget {
attribute EventHandler ondownloadprogress;
// Might get more stuff in the future, e.g. for
// https://github.com/explainers-by-googlers/prompt-api/issues/4
};
callback AICreateMonitorCallback = undefined (AICreateMonitor monitor);
enum AICapabilityAvailability { "readily", "after-download", "no" };
// Language Model
[Exposed=(Window,Worker), SecureContext]
interface AILanguageModelFactory {
Promise<AILanguageModel> create(optional AILanguageModelCreateOptions options = {});
Promise<AILanguageModelCapabilities> capabilities();
};
[Exposed=(Window,Worker), SecureContext]
interface AILanguageModel : EventTarget {
Promise<DOMString> prompt(AILanguageModelPromptInput input, optional AILanguageModelPromptOptions options = {});
ReadableStream promptStreaming(AILanguageModelPromptInput input, optional AILanguageModelPromptOptions options = {});
Promise<unsigned long long> countPromptTokens(AILanguageModelPromptInput input, optional AILanguageModelPromptOptions options = {});
readonly attribute unsigned long long maxTokens;
readonly attribute unsigned long long tokensSoFar;
readonly attribute unsigned long long tokensLeft;
readonly attribute unsigned long topK;
readonly attribute float temperature;
attribute EventHandler oncontextoverflow;
Promise<AILanguageModel> clone(optional AILanguageModelCloneOptions options = {});
undefined destroy();
};
[Exposed=(Window,Worker), SecureContext]
interface AILanguageModelCapabilities {
readonly attribute AICapabilityAvailability available;
AICapabilityAvailability languageAvailable(DOMString languageTag);
// Always null if available === "no"
readonly attribute unsigned long? defaultTopK;
readonly attribute unsigned long? maxTopK;
readonly attribute float? defaultTemperature;
readonly attribute float? maxTemperature;
};
dictionary AILanguageModelCreateOptions {
AbortSignal signal;
AICreateMonitorCallback monitor;
DOMString systemPrompt;
sequence<AILanguageModelInitialPrompt> initialPrompts;
[EnforceRange] unsigned long topK;
float temperature;
};
dictionary AILanguageModelInitialPrompt {
required AILanguageModelInitialPromptRole role;
required DOMString content;
};
dictionary AILanguageModelPrompt {
required AILanguageModelPromptRole role;
required DOMString content;
};
dictionary AILanguageModelPromptOptions {
AbortSignal signal;
};
dictionary AILanguageModelCloneOptions {
AbortSignal signal;
};
typedef (DOMString or AILanguageModelPrompt or sequence<AILanguageModelPrompt>) AILanguageModelPromptInput;
enum AILanguageModelInitialPromptRole { "system", "user", "assistant" };
enum AILanguageModelPromptRole { "user", "assistant" };
We intend for this API to expose instruction-tuned models. Although we cannot mandate any particular level of quality or instruction-following capability, we think setting this base expectation can help ensure that what browsers ship is aligned with what web developers expect.
To illustrate the difference and how it impacts web developer expectations:
To ensure the API can be used by web developers across multiple implementations, all browsers should be sure their models behave like instruction-tuned models.
To actually get a response back from the model given a prompt, the following possible stages are involved:
We've chosen to manifest these 3-4 stages into the API as two methods, ai.languageModel.create()
and session.prompt()
/session.promptStreaming()
, with some additional facilities for dealing with the fact that ai.languageModel.create()
can include a download step. Some APIs simplify this into a single method, and some split it up into three (usually not four).
Our design here uses sessions. An alternate design, seen in some APIs, is to require the developer to feed in the entire conversation history to the model each time, keeping track of the results.
This can be slightly more flexible; for example, it allows manually correcting the model's responses before feeding them back into the context window.
However, our understanding is that the session-based model can be more efficiently implemented, at least for browsers with on-device models. (Implementing it for a cloud-based model would likely be more work.) And, developers can always achieve a stateless model by using a new session for each interaction.
If cloud-based language models are exposed through this API, then there are potential privacy issues with exposing user or website data to the relevant cloud and model providers. This is not a concern specific to this API, as websites can already choose to expose user or website data to other origins using APIs such as fetch()
. However, it's worth keeping in mind, and in particular as discussed in our Goals, perhaps we should make it easier for web developers to know whether a cloud-based model is in use, or which one.
If on-device language models are updated separately from browser and operating system versions, this API could enhance the web's fingerprinting service by providing extra identifying bits. Mandating that older browser versions not receive updates or be able to download models from too far into the future might be a possible remediation for this.
Finally, we intend to prohibit (in the specification) any use of user-specific information that is not directly supplied through the API. For example, it would not be permissible to fine-tune the language model based on information the user has entered into the browser in the past.