skills/compute/text-to-image/SKILL.md
# Text-to-Image Generation ## Metadata - **Category**: compute - **SDK**: `@0glabs/0g-serving-broker` ^0.6.5, `ethers` ^6.13.0 - **Activation Triggers**: "generate image", "text-to-image", "Flux", "image generation", "create image" ## Purpose Generate images from text prompts using 0G Compute Network providers running Flux Turbo. Supports multiple resolutions and batch generation. ## Prerequisites - Node.js >= 22 - `@0glabs/0g-serving-broker` and `ethers` installed - Funded and acknowled
npx skillsauth add 0gfoundation/0g-agent-skills skills/compute/text-to-imageInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
4 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
@0glabs/0g-serving-broker ^0.6.5, ethers ^6.13.0Generate images from text prompts using 0G Compute Network providers running Flux Turbo. Supports multiple resolutions and batch generation.
@0glabs/0g-serving-broker and ethers installedtext-to-image service.env with PRIVATE_KEY, RPC_URL, PROVIDER_ADDRESSZG-Res-Key header ONLY (no body fallback)processResponse(providerAddress, chatID) — no usage data neededZG-Res-Key header only (no body fallback for images)processResponse() after every generationprocessResponse() param order: (providerAddress, chatID, usageData)getRequestHeaders() for image requestsimport { ethers } from 'ethers';
import { createZGComputeNetworkBroker } from '@0glabs/0g-serving-broker';
import * as fs from 'fs';
import 'dotenv/config';
async function generateImage(prompt: string, size = '1024x1024'): Promise<string> {
const provider = new ethers.JsonRpcProvider(process.env.RPC_URL);
const wallet = new ethers.Wallet(process.env.PRIVATE_KEY!, provider);
const broker = await createZGComputeNetworkBroker(wallet);
const providerAddress = process.env.PROVIDER_ADDRESS!;
const { endpoint, model } = await broker.inference.getServiceMetadata(providerAddress);
const requestBody = { model, prompt, n: 1, size };
// IMPORTANT: Include body for request signing
const headers = await broker.inference.getRequestHeaders(
providerAddress,
JSON.stringify(requestBody),
);
const response = await fetch(`${endpoint}/images/generations`, {
method: 'POST',
headers: { 'Content-Type': 'application/json', ...headers },
body: JSON.stringify(requestBody),
});
const data = await response.json();
const imageUrl = data.data[0].url;
// ChatID from header ONLY for images
const chatID = response.headers.get('ZG-Res-Key') || response.headers.get('zg-res-key');
if (chatID) {
await broker.inference.processResponse(providerAddress, chatID);
}
return imageUrl;
}
// Usage
const imageUrl = await generateImage('A futuristic city skyline at sunset');
console.log('Image URL:', imageUrl);
async function generateAndSave(
prompt: string,
outputPath: string,
size = '512x512',
): Promise<void> {
const provider = new ethers.JsonRpcProvider(process.env.RPC_URL);
const wallet = new ethers.Wallet(process.env.PRIVATE_KEY!, provider);
const broker = await createZGComputeNetworkBroker(wallet);
const providerAddress = process.env.PROVIDER_ADDRESS!;
const { endpoint, model } = await broker.inference.getServiceMetadata(providerAddress);
const requestBody = { model, prompt, n: 1, size };
const headers = await broker.inference.getRequestHeaders(
providerAddress,
JSON.stringify(requestBody),
);
const response = await fetch(`${endpoint}/images/generations`, {
method: 'POST',
headers: { 'Content-Type': 'application/json', ...headers },
body: JSON.stringify(requestBody),
});
const data = await response.json();
// Process response for fee settlement
const chatID = response.headers.get('ZG-Res-Key') || response.headers.get('zg-res-key');
if (chatID) {
await broker.inference.processResponse(providerAddress, chatID);
}
// Download and save image
const imageData = data.data[0];
if (imageData.b64_json) {
fs.writeFileSync(outputPath, Buffer.from(imageData.b64_json, 'base64'));
} else if (imageData.url) {
const imgResponse = await fetch(imageData.url);
const buffer = Buffer.from(await imgResponse.arrayBuffer());
fs.writeFileSync(outputPath, buffer);
}
console.log(`Image saved to ${outputPath}`);
}
async function batchGenerate(
prompts: string[],
outputDir: string,
size = '512x512',
): Promise<string[]> {
const provider = new ethers.JsonRpcProvider(process.env.RPC_URL);
const wallet = new ethers.Wallet(process.env.PRIVATE_KEY!, provider);
const broker = await createZGComputeNetworkBroker(wallet);
const providerAddress = process.env.PROVIDER_ADDRESS!;
const { endpoint, model } = await broker.inference.getServiceMetadata(providerAddress);
if (!fs.existsSync(outputDir)) fs.mkdirSync(outputDir, { recursive: true });
const results: string[] = [];
for (let i = 0; i < prompts.length; i++) {
console.log(`Generating ${i + 1}/${prompts.length}: "${prompts[i]}"`);
const requestBody = { model, prompt: prompts[i], n: 1, size };
const headers = await broker.inference.getRequestHeaders(
providerAddress,
JSON.stringify(requestBody),
);
const response = await fetch(`${endpoint}/images/generations`, {
method: 'POST',
headers: { 'Content-Type': 'application/json', ...headers },
body: JSON.stringify(requestBody),
});
const data = await response.json();
const chatID = response.headers.get('ZG-Res-Key') || response.headers.get('zg-res-key');
if (chatID) {
await broker.inference.processResponse(providerAddress, chatID);
}
const outputPath = `${outputDir}/image-${i + 1}.png`;
if (data.data[0].url) {
const imgResponse = await fetch(data.data[0].url);
fs.writeFileSync(outputPath, Buffer.from(await imgResponse.arrayBuffer()));
}
results.push(outputPath);
}
return results;
}
async function safeGenerateImage(prompt: string, size = '512x512'): Promise<string | null> {
const provider = new ethers.JsonRpcProvider(process.env.RPC_URL);
const wallet = new ethers.Wallet(process.env.PRIVATE_KEY!, provider);
const broker = await createZGComputeNetworkBroker(wallet);
const providerAddress = process.env.PROVIDER_ADDRESS!;
try {
const { endpoint, model } = await broker.inference.getServiceMetadata(providerAddress);
const requestBody = { model, prompt, n: 1, size };
const headers = await broker.inference.getRequestHeaders(
providerAddress,
JSON.stringify(requestBody),
);
const response = await fetch(`${endpoint}/images/generations`, {
method: 'POST',
headers: { 'Content-Type': 'application/json', ...headers },
body: JSON.stringify(requestBody),
});
if (!response.ok) {
throw new Error(`HTTP ${response.status}: ${await response.text()}`);
}
const data = await response.json();
const chatID = response.headers.get('ZG-Res-Key') || response.headers.get('zg-res-key');
if (chatID) {
await broker.inference.processResponse(providerAddress, chatID);
}
return data.data[0].url || null;
} catch (error) {
console.error('Image generation failed:', error);
return null;
}
}
| Size | Cost (approx.) | Best For |
| ----------- | -------------- | ------------------- |
| 256x256 | ~0.001 0G | Thumbnails, testing |
| 512x512 | ~0.002 0G | Standard images |
| 1024x1024 | ~0.003 0G | High quality |
// BAD: Missing body in getRequestHeaders for images
const headers = await broker.inference.getRequestHeaders(providerAddress);
// Must include: getRequestHeaders(providerAddress, JSON.stringify(requestBody))
// BAD: Trying to get chatID from response body
const chatID = data.id; // WRONG for images — use header only
// BAD: Large batch without balance check
await batchGenerate(hundredPrompts, './out'); // May run out of funds
// BAD: Hardcoding private keys
const wallet = new ethers.Wallet('0xabc123...', provider); // NEVER do this
// BAD: ethers v5 syntax
const provider = new ethers.providers.JsonRpcProvider(url); // v5!
| Error | Cause | Fix |
| ------------------------- | --------------------------------- | ------------------------------------- |
| Insufficient balance | Sub-account empty | Transfer more funds |
| Invalid request headers | Missing body in header generation | Include body in getRequestHeaders() |
| Fee verification failed | Missing chatID | Ensure ZG-Res-Key header exists |
| invalid size | Unsupported dimensions | Use 256x256, 512x512, or 1024x1024 |
development
# Upload File to 0G Storage ## Metadata - **Category**: storage - **SDK**: `@0glabs/0g-ts-sdk` ^0.3.3, `ethers` ^6.13.0 - **Activation Triggers**: "upload file", "store on 0G", "ZgFile", "save to storage" ## Purpose Upload files to 0G decentralized storage using the ZgFile API and Indexer. Files are split into chunks, organized as a Merkle tree, and distributed across storage nodes. Returns a root hash for later retrieval. ## Prerequisites - Node.js >= 18 - `@0glabs/0g-ts-sdk` and `ethers`
development
# Merkle Verification ## Metadata - **Category**: storage - **SDK**: `@0glabs/0g-ts-sdk` ^0.3.3 - **Activation Triggers**: "verify file", "merkle proof", "data integrity", "root hash", "check file" ## Purpose Compute root hashes and verify data integrity for files stored on 0G Storage. Uses Merkle tree proofs to cryptographically verify that downloaded data matches what was originally uploaded. ## Prerequisites - Node.js >= 18 - `@0glabs/0g-ts-sdk` installed ## Quick Workflow 1. Create
development
# Download File from 0G Storage ## Metadata - **Category**: storage - **SDK**: `@0glabs/0g-ts-sdk` ^0.3.3, `ethers` ^6.13.0 - **Activation Triggers**: "download file", "retrieve from 0G", "get file", "fetch from storage" ## Purpose Download and verify files from 0G decentralized storage using a root hash. Supports verified downloads with Merkle proof validation to ensure data integrity. ## Prerequisites - Node.js >= 18 - `@0glabs/0g-ts-sdk` installed - Root hash of the file to download - `
development
# Storage + Chain Integration ## Metadata - **Category**: cross-layer - **SDK**: `@0glabs/0g-ts-sdk` ^0.3.3, `ethers` ^6.13.0 - **Activation Triggers**: "on-chain reference", "NFT metadata on 0G", "store hash on-chain", "registry contract", "chain and storage" ## Purpose Combine 0G Storage with 0G Chain smart contracts to create on-chain references to off-chain data. Common patterns include NFT metadata storage, content registries, and verifiable document systems. ## Prerequisites - Node