Configuration
Copy
Ask AI
npm install @mastra/core @mendable/firecrawl-js zod
.env :
Copy
Ask AI
FIRECRAWL_API_KEY=votre_clé_firecrawl
OPENAI_API_KEY=votre_clé_openai
Remarque : Si vous utilisez Node < 20, installezdotenvet ajoutezimport 'dotenv/config'à votre code.
Flux de travail en plusieurs étapes
Copy
Ask AI
import { createWorkflow, createStep } from "@mastra/core/workflows";
import { z } from "zod";
import Firecrawl from "@mendable/firecrawl-js";
import { Agent } from "@mastra/core/agent";
const firecrawl = new Firecrawl({
apiKey: process.env.FIRECRAWL_API_KEY || "fc-YOUR_API_KEY"
});
const agent = new Agent({
name: "summarizer",
instructions: "Vous êtes un assistant qui crée des résumés concis de documentation.",
model: "openai/gpt-5-nano",
});
// Étape 1 : Recherche avec le SDK Firecrawl
const searchStep = createStep({
id: "search",
inputSchema: z.object({
query: z.string(),
}),
outputSchema: z.object({
url: z.string(),
title: z.string(),
}),
execute: async ({ inputData }: { inputData: { query: string } }) => {
console.log(`Recherche : ${inputData.query}`);
const searchResults = await firecrawl.search(inputData.query, { limit: 1 });
const webResults = (searchResults as any)?.web;
if (!webResults || !Array.isArray(webResults) || webResults.length === 0) {
throw new Error("Aucun résultat de recherche trouvé");
}
const firstResult = webResults[0];
console.log(`Trouvé : ${firstResult.title}`);
return {
url: firstResult.url,
title: firstResult.title,
};
},
});
// Étape 2 : Scraper l'URL avec le SDK Firecrawl
const scrapeStep = createStep({
id: "scrape",
inputSchema: z.object({
url: z.string(),
title: z.string(),
}),
outputSchema: z.object({
markdown: z.string(),
title: z.string(),
}),
execute: async ({ inputData }: { inputData: { url: string; title: string } }) => {
console.log(`Scraping : ${inputData.url}`);
const scrapeResult = await firecrawl.scrape(inputData.url, {
formats: ["markdown"],
});
console.log(`Scrapé : ${scrapeResult.markdown?.length || 0} caractères`);
return {
markdown: scrapeResult.markdown || "",
title: inputData.title,
};
},
});
// Étape 3 : Résumer avec Claude
const summarizeStep = createStep({
id: "summarize",
inputSchema: z.object({
markdown: z.string(),
title: z.string(),
}),
outputSchema: z.object({
summary: z.string(),
}),
execute: async ({ inputData }: { inputData: { markdown: string; title: string } }) => {
console.log(`Résumé en cours : ${inputData.title}`);
const prompt = `Résumez la documentation suivante en 2-3 phrases :\n\nTitre : ${inputData.title}\n\n${inputData.markdown}`;
const result = await agent.generate(prompt);
console.log(`Résumé généré`);
return { summary: result.text };
},
});
// Créer le workflow
export const workflow = createWorkflow({
id: "firecrawl-workflow",
inputSchema: z.object({
query: z.string(),
}),
outputSchema: z.object({
summary: z.string(),
}),
steps: [searchStep, scrapeStep, summarizeStep],
})
.then(searchStep)
.then(scrapeStep)
.then(summarizeStep)
.commit();
async function testWorkflow() {
const run = await workflow.createRunAsync();
const result = await run.start({
inputData: { query: "Documentation Firecrawl" }
});
if (result.status === "success") {
const { summarize } = result.steps;
if (summarize.status === "success") {
console.log(`\n${summarize.output.summary}`);
}
} else {
console.error("Échec du workflow :", result.status);
}
}
testWorkflow().catch(console.error);

