with one click
flow-convert-activities-to-steps
// Convert Flow SDK activities.ts to Output SDK steps.ts. Use when migrating activity functions to step definitions with typed parameters.
// Convert Flow SDK activities.ts to Output SDK steps.ts. Use when migrating activity functions to step definitions with typed parameters.
Validate and fix folder structure for Output SDK workflows. Use to ensure migrated workflows follow the correct structure conventions.
Convert inline prompts and prompt arrays to .prompt files with YAML frontmatter. Use when migrating prompts from Flow SDK format to Output SDK prompt files.
Convert Flow SDK workflow class to Output SDK workflow() function. Use when migrating workflow.ts files from class-based to functional definitions.
Create scenario files for testing migrated Output SDK workflows. Use to set up test inputs in the scenarios/ subfolder.
Fix ESLint issues in migrated Output SDK code. Use when seeing lint errors after migration, or when writing new Output SDK code that needs to follow project conventions.
Remove try-catch antipattern from step calls during Flow to Output SDK migration. Use when converting workflow code that wraps step calls in try-catch blocks.
| name | flow-convert-activities-to-steps |
| description | Convert Flow SDK activities.ts to Output SDK steps.ts. Use when migrating activity functions to step definitions with typed parameters. |
| allowed-tools | ["Bash","Read","Write","Grep","Edit"] |
This skill guides the conversion of Flow SDK activity functions (activities.ts) to Output SDK step definitions (steps.ts). This is one of the core migration tasks.
During Migration:
activities.ts file to steps.ts| Aspect | Flow SDK (activities.ts) | Output SDK (steps.ts) |
|---|---|---|
| Definition | Function with direct parameters | step() with inputSchema |
| Parameters | Individual function arguments | Single typed input object |
| Return Type | Direct Promise return | outputSchema validation |
| Imports | Various Flow SDK imports | @outputai/core |
| LLM Calls | Custom completion functions | generateText() with Output.object() |
// activities.ts
import { z } from 'zod';
import { completion } from '@flow/sdk';
export async function analyzeDocument(
documentText: string,
analysisType: string,
maxLength?: number
): Promise<AnalysisResult> {
const prompt = buildPrompt( documentText, analysisType );
const response = await completion( {
model: 'gpt-4',
messages: [ { role: 'user', content: prompt } ],
maxTokens: maxLength || 2000
} );
return parseAnalysisResult( response );
}
// steps.ts
import { z, step } from '@outputai/core';
import { generateText, Output } from '@outputai/llm';
import { AnalysisResultSchema, AnalysisResult } from './types.js';
const AnalyzeDocumentInputSchema = z.object( {
documentText: z.string(),
analysisType: z.string(),
maxLength: z.number().optional()
} );
export const analyzeDocument = step( {
name: 'analyzeDocument',
inputSchema: AnalyzeDocumentInputSchema,
outputSchema: AnalysisResultSchema,
fn: async ( input ) => {
const { documentText, analysisType, maxLength } = input;
const { output } = await generateText( {
prompt: 'analyzeDocument@v1',
variables: {
documentText,
analysisType
},
output: Output.object( {
schema: AnalysisResultSchema
} )
} );
return output;
}
} );
List all exported functions in activities.ts:
grep -E "^export (async )?function" src/workflows/my-workflow/activities.ts
For each function, create a Zod schema for its parameters:
// Original function signature
async function processUser( userId: string, options: ProcessOptions ): Promise<Result>
// Convert to input schema
const ProcessUserInputSchema = z.object( {
userId: z.string(),
options: ProcessOptionsSchema
} );
If the function returns structured data, create an output schema:
// types.ts
export const ResultSchema = z.object( {
success: z.boolean(),
data: z.any().optional(),
error: z.string().optional()
} );
export type Result = z.infer<typeof ResultSchema>;
Wrap the function body in a step() definition:
export const processUser = step( {
name: 'processUser',
inputSchema: ProcessUserInputSchema,
outputSchema: ResultSchema,
fn: async ( input ) => {
const { userId, options } = input;
// Original function body here
}
} );
Replace Flow SDK completion calls with Output SDK generators:
// Flow SDK
const response = await completion( { model: 'gpt-4', messages: [...] } );
// Output SDK
const { result } = await generateText( {
prompt: 'myPrompt@v1',
variables: { ... }
} );
import { z } from 'zod';
import { completion } from '@flow/sdk';
const UserSchema = z.object( {
id: z.string(),
name: z.string(),
email: z.string()
} );
type User = z.infer<typeof UserSchema>;
export async function fetchUser( userId: string ): Promise<User> {
const response = await fetch( `https://api.example.com/users/${userId}` );
return response.json();
}
export async function generateGreeting(
user: User,
style: 'formal' | 'casual'
): Promise<string> {
const prompt = style === 'formal'
? `Write a formal greeting for ${user.name}`
: `Write a casual greeting for ${user.name}`;
const response = await completion( {
model: 'gpt-4',
messages: [ { role: 'user', content: prompt } ]
} );
return response.content;
}
export async function sendEmail(
to: string,
subject: string,
body: string
): Promise<{ sent: boolean; messageId: string }> {
const result = await emailService.send( { to, subject, body } );
return { sent: true, messageId: result.id };
}
import { z, step } from '@outputai/core';
import { generateText } from '@outputai/llm';
import { UserSchema, User } from './types.js';
// Step 1: Fetch User
const FetchUserInputSchema = z.object( {
userId: z.string()
} );
export const fetchUser = step( {
name: 'fetchUser',
inputSchema: FetchUserInputSchema,
outputSchema: UserSchema,
fn: async ( input ) => {
const { userId } = input;
const response = await fetch( `https://api.example.com/users/${userId}` );
return response.json();
}
} );
// Step 2: Generate Greeting
const GenerateGreetingInputSchema = z.object( {
user: UserSchema,
style: z.enum( [ 'formal', 'casual' ] )
} );
export const generateGreeting = step( {
name: 'generateGreeting',
inputSchema: GenerateGreetingInputSchema,
outputSchema: z.string(),
fn: async ( input ) => {
const { user, style } = input;
const { result } = await generateText( {
prompt: 'generateGreeting@v1',
variables: {
userName: user.name,
style
}
} );
return result;
}
} );
// Step 3: Send Email
const SendEmailInputSchema = z.object( {
to: z.string(),
subject: z.string(),
body: z.string()
} );
const SendEmailOutputSchema = z.object( {
sent: z.boolean(),
messageId: z.string()
} );
export const sendEmail = step( {
name: 'sendEmail',
inputSchema: SendEmailInputSchema,
outputSchema: SendEmailOutputSchema,
fn: async ( input ) => {
const { to, subject, body } = input;
const result = await emailService.send( { to, subject, body } );
return { sent: true, messageId: result.id };
}
} );
import { z } from '@outputai/core';
export const UserSchema = z.object( {
id: z.string(),
name: z.string(),
email: z.string()
} );
export type User = z.infer<typeof UserSchema>;
Steps are called with a single input object:
// Flow SDK (direct parameters)
const user = await fetchUser( userId );
const greeting = await generateGreeting( user, 'formal' );
// Output SDK (object parameter)
const user = await fetchUser( { userId } );
const greeting = await generateGreeting( { user, style: 'formal' } );
// WRONG
fn: async ( userId, name ) => { ... }
// CORRECT
fn: async ( input ) => {
const { userId, name } = input;
...
}
// WRONG
import { UserSchema } from './types';
// CORRECT
import { UserSchema } from './types.js';
Keep schemas and types in types.ts for reuse across steps and workflows.
@outputai/core for zgenerateText() with Output.object() for structured output.js extensionflow-convert-workflow-definition - Workflow conversionflow-convert-prompts-to-files - Prompt file creationflow-error-zod-import - Zod import issuesflow-error-eslint-compliance - Code style compliance