LLMFlow is a powerful and flexible TypeScript-first library for working with Language Models (LLMs) in your Node.js applications. It provides an intuitive, type-safe interface for creating, managing, and executing prompts with various LLM providers.
npm i @mirror-ai/llmflow
LLMFlow uses API keys for OpenAI and Anthropic services. These keys should be set as environment variables:
- Set
OPENAI_API_KEY
for OpenAI services - Set
ANTHROPIC_API_KEY
for Anthropic services
Example:
OPENAI_API_KEY=your_openai_api_key ANTHROPIC_API_KEY=your_anthropic_api_key
The service automatically loads these keys from environment variables.
LLMFlow is built with TypeScript and provides first-class TypeScript support out of the box. This native TypeScript integration offers several key advantages that we'll explore throughout this guide.
Import the necessary functions:
import { createLLMFlow } from 'llm-flow';
Create an LLMFlow instance:
const flow = createLLMFlow<{name: string; age: number; balance: number;}>()(
"Hello {{name}}, you are {{age}} years old and your balance is {{balance}}",
{
model: "gpt-3.5-turbo-0125",
maxTokens: 50,
temperature: 0.5,
}
);
Use the flow:
// This will work fine:
flow.run({ name: "Alice", age: 30, balance: 1000 });
// This will cause a TypeScript error (Type 'string' is not assignable to type 'number'.):
flow.run({ name: "Charlie", age: "bob", balance: 1500 });
// This will cause a TypeScript error (extra property 'extra'):
flow.run({ name: "David", age: 35, balance: 2000, extra: true });
The prompt template is also parsed, so not providing a value for age will result in a TypeScript error, as the age property is expected but missing in the prompt string this example:
const flow = createLLMFlow<{name: string; age: number; balance: number;}>()(
"Hello {{name}}, you are age years old and your balance is {{balance}}")
Will create this error:
// TypeScript error (Property 'age' is missing):
Because LLM's usually return a string, this is the default return type and you do not need to explcitly type it.
The above example demonstrates several key advantages of LLMFlow's TypeScript integration:
-
Type Safety: The input type is explicitly defined as a generic type parameter in the createLLMFlow function.
-
Compile-Time Checking: TypeScript checks that the input object you provide matches the structure defined in the generic type parameter, not the template string.
-
Informative Error Messages: If you miss a required property or add an extra one, TypeScript will provide a clear error message indicating what's wrong.
-
Flexibility: The type of the input object is clearly defined, making it easy to understand what properties are required and is easily changed.
This approach provides strong type checking and a great developer experience, catching potential errors at compile-time and providing clear feedback about what's wrong when there's a mismatch between the template and the provided input.
LLMFlow can handle complex scenarios where structured output is required. This example demonstrates how to create a flow that assesses whether a user's prompt is possible given a set of available tools, returning a JSON object with a boolean success
flag and a feedback
string.
import { createLLMFlow } from 'llm-flow';
const assessmentFlow = createLLMFlow<{prompt: string; tools: string[]}>()(
`Assess whether the user prompt "{{prompt}}" is possible given the following tools: {{tools[]}}.
Your response should include a valid JSON object with two keys:
"success": boolean (true if the prompt is possible, false otherwise)
"feedback": string (explanation of your assessment)`,
{
model: 'gpt-4-2024-05-13',
maxTokens: 150,
temperature: 0.2
}
);
This creates a flow that:
- Takes a user prompt and a list of available tools as input
- Instructs the LLM to assess the feasibility of the prompt
- Returns a JSON object with
success
andfeedback
fields
TypeScript is enforcing type safety based on the explicit generic type provided ({prompt: string; tools: string[]})
const userPrompt = "Generate a 3D model of a house";
const availableTools = ["Text generation", "Image generation", "Code completion"];
assessmentFlow.run({ prompt: userPrompt, tools: availableTools })
.then((result) => {
// TypeScript infers that result is of type { success: boolean; feedback: string }
console.log(`Is the prompt possible? ${result.success}`);
console.log(`Feedback: ${result.feedback}`);
})
.catch((error) => {
console.error('Error in assessment:', error);
});
In this example, TypeScript ensures that the input to run()
matches the structure defined in the template string. It also infers the shape of the result
object based on the expected JSON structure described in the prompt.
-
Automatic Type Inference: LLMFlow automatically infers input types based on the prompt template. Output types are inferred based on the expected JSON structure described in the prompt.
-
Structured Output: The flow is designed to return a specific JSON structure, which is automatically parsed by the library.
-
Flexible Input: The flow accepts multiple input parameters (
prompt
andtools
) which are inserted into the prompt template. -
Error Handling: The example includes basic error handling to catch any issues during flow execution.
-
Compile-Time Checking: TypeScript checks that the input object provided to
run()
matches the structure defined by the template string.
- The library attempts to parse the LLM's response as JSON automatically. If parsing fails, it will return the raw string response.
- Adjust the
maxTokens
andtemperature
settings as needed for your specific use case. - The actual output depends on the LLM's interpretation of the prompt. You may need to iterate on the prompt template to get consistently well-structured responses.
By leveraging LLMFlow's capabilities, you can create complex, type-safe interactions with language models that produce structured data, making it easier to integrate LLM outputs into your applications.
JSON parsing is turned on by default in LLMFlow. However you easily turn it off if you want to parse the response in a custom way.
Simply pass the "dontParse" flag in the options. This tells TypeScript to treat the output as a string, which you can then parse manually if needed.
import { createLLMFlow } from '@mirror-ai/llmflow';
const flow = createLLMFlow<{ topic: string }>()(
"Write a short paragraph about {topic}",
{
// ... options ...
dontParse: true,
},
);
LLMFlow allows you to specify options directly when creating a flow. Here's how to use some of the key options inline:
import { createLLMFlow } from '@mirror-ai/llmflow';
const flow = createLLMFlow<{ topic: string }>()(
"Write a short paragraph about {topic}",
{
model: 'gpt-4-2024-05-13',
maxTokens: 100,
temperature: 0.7,
topP: 0.9,
frequencyPenalty: 0.5,
presencePenalty: 0.5,
stopSequences: ['\n', 'END'],
responseFormat: 'text',
stream: false
},
);
// Usage
const result = await flow.run({ topic: 'artificial intelligence' });
console.log(result);
model
: Specifies the LLM (e.g., 'gpt-4-2024-05-13', 'claude-3-opus-20240229')maxTokens
: Limits response lengthtemperature
: Controls randomness (0-1)topP
: Nucleus sampling (0-1)frequencyPenalty
andpresencePenalty
: Adjust repetition (-2.0 to 2.0)stopSequences
: Array of stop stringsresponseFormat
: 'text' or 'json_object'stream
: Enable streaming (boolean)
const jsonFlow = createLLMFlow<{ key1: string; key2: string }>()(
"Generate a JSON object with {key1} and {key2}",
{
model: 'gpt-4-2024-05-13',
maxTokens: 150,
temperature: 0.5,
responseFormat: 'json_object'
},
);
const toolFlow = createLLMFlow<{ task: string }>()(
"Use available tools to {task}",
{
model: 'gpt-4-2024-05-13',
maxTokens: 200,
tools: [{
name: "get_weather",
description: "Get current weather",
parameters: {
type: "object",
properties: {
location: { type: "string" }
},
required: ["location"]
}
}],
toolChoice: "auto"
},
);
const streamingFlow = createLLMFlow<{ topic: string }>()(
"Generate a story about {topic}",
{
model: 'gpt-4-2024-05-13',
maxTokens: 500,
stream: true
},
{}
);
// Usage
for await (const chunk of streamingFlow.run({ topic: "space exploration" })) {
console.log(chunk);
}
These examples demonstrate how to use LLMFlow's options inline when creating flow instances. Adjust the options as needed for your specific use cases.
Prompts in LLMFlow are treated as parameters of a machine learning model. The process of prompt engineering involves multiple iterations, similar to hyperparameter tuning in traditional machine learning. LLMFlow offers automatic versioning and serialization of prompts through static and dynamic analysis. Enable versioning to keep track of your prompt templates and LLM configurations:
const flowWithVersioning = createLLMFlow<{ topic: string; length: number }>()(
"Write a {length}-word paragraph about {topic}",
{ model: 'gpt-4-2024-05-13', maxTokens: 100 },
{ versioningEnabled: true, storePath: './prompt-checkpoints' }
);
In this example:
versioningEnabled: true
activates the automatic versioning system.
storePath: './prompt-checkpoints'
specifies the local directory where prompt versions will be stored, similar to saving model checkpoints.
LLMFlow's TypeScript integration provides intelligent autocompletion for versioning options, catching potential typos or type mismatches at compile-time. This ensures that your prompt versioning configuration is always valid, reducing errors in your ML pipeline. With this setup, each time you run your flow, LLMFlow will:
- Automatically version your prompt template
- Generate a commit message describing the changes
- Store the versioned prompt in the specified checkpoint directory
This allows you to track the evolution of your prompts over time, rollback to previous versions if needed, and maintain a clear history of your prompt engineering process - all integrated seamlessly into your development workflow.
LLMFlow supports various LLM providers. Simply specify the model in the options:
const openAIFlow = createLLMFlow(/* ... */, { model: 'gpt-4o-2024-05-13', /* ... */ });
const anthropicFlow = createLLMFlow(/* ... */, { model: 'claude-3-opus-20240229', /* ... */ });
TypeScript's union types ensure that only valid model names are accepted:
type Models = "gpt-4o-2024-05-13" | "claude-3-opus-20240229" | /* ... other valid models ... */;
By default, LLMFlow attempts to parse JSON responses. You can disable this:
const flow = createLLMFlow<InputType>()(
/* ... */,
{ dontParse: true, /* ... */ }
);
When dontParse
is true, TypeScript ensures the output type is string
.
LLMFlow leverages TypeScript's generic types to ensure type safety when defining prompt templates:
const flow = createLLMFlow<{ name: string; age: number }, UserProfile>()(
"Generate a user profile for {name}, age {age}",
{ model: 'gpt-4o-2024-05-13' }
);
TypeScript ensures that:
The input object must have name
(string) and age
(number) properties.
The output must conform to the UserProfile
interface.
IDE tools provide intelligent code completion for:
Input variables in your prompt templates
LLM options and their allowed values
Methods and properties of the LLMFlow
class
TypeScript catches potential errors at compile-time:
// TypeScript will catch these errors:
const flow = createLLMFlow<{ topic: number }, string>()( // Error: 'topic' should be string
"Write about {topic}",
{ model: 'invalid-model' } // Error: 'invalid-model' is not a valid model option
);
Static typing makes refactoring safer and improves code navigation, especially valuable in large projects using LLMFlow.
TypeScript interfaces serve as inline documentation:
interface VersioningOptions {
versioningEnabled: boolean;
storePath: string;
}
const flow = createLLMFlow<
Template extends string,
TInput extends Record<string, unknown> = Record<string, string>,
TOutput = string
>()(
template: Template,
options: LLMOptions,
input: ValidateTemplate<Template, TInput>,
versioningOptions?: Partial<VersioningOptions>
)
Leverage TypeScript's type system to define clear interfaces for your input and output types. Use TypeScript's strict mode to catch more potential issues. Take advantage of IDE features like autocompletion and quick documentation for LLMFlow APIs. Enable versioning in production to track changes in your prompts and configurations. Handle errors appropriately, as LLM calls can sometimes fail or produce unexpected results.
Explore the full API documentation for advanced usage and configuration options. Check out the examples directory for more complex use cases and patterns. Join our community forum to ask questions and share your experiences with LLMFlow.
Happy coding with LLMFlow!