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.) 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:
"no"
from canCreateTextSession()
, or to implement this API entirely by using cloud services instead of on-device models.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 assistant.
const session = await ai.createTextSession();
// 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 model can be configured with a special "system prompt" which gives it the context for future assistant/user interactions:
const session = await ai.createTextSession({
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 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 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.createTextSession({
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 session = await session.clone();
return await session.prompt(comment);
}
const result1 = await predictEmoji("Back to the drawing board");
const result2 = await predictEmoji("This code is so good you should get promoted");
(Using both 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
.)
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 by calling textModelInfo()
.
const customSession = await ai.createTextSession({
temperature: 0.8,
topK: 10
});
const modelInfo = await ai.textModelInfo();
const slightlyHighTemperatureSession = await ai.createTextSession({
temperature: Math.max(modelInfo.defaultTemperature * 1.2, 1.0),
});
// modelInfo also contains defaultTopK and maxTopK.
Each session consists of a persistent series of interactions with the model:
const session = await ai.createTextSession({
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.createTextSession({
systemPrompt: "You are a friendly, helpful assistant specialized in clothing choices."
});
const session2 = await session.clone();
A session can be destroyed to free some resources when it’s not needed anymore. After destroying it, it can’t be prompted anymore, and any ongoing prompt calls will be rejected with an "InvalidStateError"
DOMException
.
session.destroy();
// The promise will be rejected with an error explaining that the session is destroyed.
await session.prompt(`
What should I wear today? It's sunny and I'm unsure between a t-shirt and a polo.
`);
In all our above examples, we call ai.createTextSession()
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 createTextSession()
will start the download, which might take a long time. You can give your users a better experience by detecting this condition using canCreateTextSession()
, which returns 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 createTextSession()
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 canCreateTextSession()
, createTextSession()
might also fail, if either the download fails or the session creation fails.
When a download has been started by createTextSession()
, the ai
object will emit textmodeldownloadprogress
events, which are of type ProgressEvent
. They can be used to display a progress bar or similar.
See Downloading and session creation flow for the full details on the internal state machine.
const canCreate = await ai.canCreateTextSession();
switch (canCreate) {
case "no": {
console.log("This browser/device cannot provide a language model.");
break;
}
case "after-download": {
console.log("Going to download a language model; sit tight!");
ai.addEventListener("textmodeldownloadprogress", e => {
console.log(`Download progress: ${e.loaded} / ${e.total} bytes.`);
});
break;
}
case "readily": {
console.log("The language model is already downloaded; creating it will be quick.");
}
}
// * If canCreate was "no", this will throw a "NotSupportedError" DOMException.
//
// * If canCreate was "after-download", this will kick off the download and only fulfill
// after it completes, or reject if it fails or the session creation fails.
//
// * If canCreate was "readily", this will fulfill or reject relatively quickly, based
// on whether session creation succeeds or fails.
let session;
try {
session = await ai.createTextSession();
console.log("Creating the session succeeded!");
} catch (e) {
console.error("Creating the session failed, either in the downloading or session " +
"creation steps. More details: ", e);
}
// Now prompt it as usual...
partial interface WindowOrWorkerGlobalScope {
[Replaceable] readonly attribute AI ai;
}
[Exposed=(Window,Worker)]
interface AI {
Promise<AIModelAvailability> canCreateTextSession();
Promise<AITextSession> createTextSession(optional AITextSessionOptions options = {});
attribute EventHandler ontextmodeldownloadprogress;
Promise<AITextModelInfo> textModelInfo();
};
[Exposed=(Window,Worker)]
interface AITextSession {
Promise<DOMString> prompt(DOMString input);
ReadableStream promptStreaming(DOMString input);
readonly attribute unsigned long topK;
readonly attribute float temperature;
Promise<AITextSession> clone();
undefined destroy();
};
dictionary AITextSessionOptions {
[EnforceRange] unsigned long topK;
float temperature;
DOMString systemPrompt;
sequence<AIPrompt> initialPrompts;
};
dictionary AIPrompt {
AIPromptRole role;
DOMString content;
};
dictionary AITextModelInfo {
unsigned long defaultTopK;
unsigned long maxTopK;
float defaultTemperature;
};
enum AIModelAvailability { "readily", "after-download", "no" };
enum AIPromptRole { "system", "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.
The state machine for the model downloading and session creation is as follows.
There is a per-global session creation requested: a boolean, initially false.
There is a per-user agent model availability state. It moves through the following states:
canCreateTextSession()
returns "no"
createTextSession()
rejects with a "NotSupportedError"
DOMException
canCreateTextSession()
returns "after-download"
createTextSession()
:canCreateTextSession()
returns "after-download"
createTextSession()
:AI
object fires textmodeldownloadprogress
events on download progress updates.canCreateTextSession()
returns "readily"
createTextSession()
:Create a session operation: takes as input a download status
"NetworkError"
DOMException
We don't love the current naming of the API. Especially, the names using "text" would become confusing if the same API eventually gains multi-modal capabilities.
We've started a discussion on the issue tracker to explore alternatives: see issue #1.
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, createTextSession()
and prompt()
/promptStreaming()
, with some additional facilities for dealing with the fact that createTextSession()
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.