nekoai-js
Version:
A lightweight JavaScript/TypeScript API for NovelAI image generation and director tools
600 lines (462 loc) • 18.9 kB
Markdown
# 🐾 NekoAI-JS
<div align="center">
<img src="https://raw.githubusercontent.com/Nya-Foundation/NekoAI-JS/main/assets/banner.png" alt="NekoAI-JS Banner" width="800" />
<h3>🎨 A lightweight JavaScript/TypeScript API for NovelAI image generation and director tools.</h3>
<div>
<a href="https://github.com/Nya-Foundation/NekoAI-JS/blob/main/LICENSE"><img src="https://img.shields.io/github/license/Nya-Foundation/nekoai-js.svg" alt="License"/></a>
<a href="https://github.com/Nya-Foundation/NekoAI-JS/actions/workflows/release.yml"><img src="https://github.com/Nya-Foundation/NekoAI-JS/actions/workflows/release.yml/badge.svg" alt="Builds & Release"/></a>
<a href="https://www.npmjs.com/package/nekoai-js"><img src="https://img.shields.io/npm/v/nekoai-js.svg" alt="npm version"/></a>
<a href="https://deepwiki.com/Nya-Foundation/NekoAI-JS"><img src="https://deepwiki.com/badge.svg" alt="Ask DeepWiki"/></a>
</div>
</div>
## 🌈 Introduction
> 🐾 **NekoAI-JS** is a **lightweight** and **easy-to-use** JavaScript/TypeScript wrapper for NovelAI's image generation capabilities. This package makes it simple to integrate NovelAI's powerful image generation and manipulation tools into your JavaScript applications with minimal code overhead.
>
> Built with modern JavaScript/TypeScript features for both browser and Node.js environments, it provides full access to NovelAI's latest models (V3, V4, V4.5) and Director tools while maintaining a clean interface. This project is based on the [NekoAI-API](https://github.com/Nya-Foundation/NekoAI-API) Python package.
### 📄 License Change Notice
> **Important**: This project has transitioned from MIT to **AGPL-3.0** license to ensure better compliance and alignment with our inspiration source. As this work builds significantly upon concepts and approaches from NekoAI-API, we've adopted a more appropriate license that better reflects the collaborative nature of open-source development and provides stronger copyleft protections for the community.
## 🌟 Core Capabilities
| Feature | Description |
| --------------------------- | ------------------------------------------------------------------------------------------------------ |
| 🚀 **Lightweight** | Focuses on image generation and Director tools, providing a simple and easy-to-use interface. |
| ⚙️ **Parameterized** | Provides strongly typed interfaces to easily set up generation parameters with validation. |
| 🔑 **Token Authentication** | Supports direct token authentication for API access. |
| 🎬 **Real-time Streaming** | Stream V4/V4.5 generation progress in real-time, watching each denoising step as it happens. |
| 🌐 **Cross-Platform** | Works in both browser and Node.js environments. |
| ✨ **Latest Models** | Full support for V3, V4, and V4.5 models including multi-character generation. |
| 🛠️ **Director Tools** | Complete support for all NovelAI Director tools like line art, background removal, and emotion change. |
| 🔄 **TypeScript Support** | Full TypeScript definitions for all API parameters and responses. |
| 🔁 **Automatic Retries** | Built-in retry mechanism for handling rate limits and temporary API failures. |
## 📦 Installation
```sh
# Using npm
npm install nekoai-js
# Using yarn
yarn add nekoai-js
# Using pnpm
pnpm add nekoai-js
```
For Node.js environments, you may need to install the optional canvas dependency for image processing:
```sh
# Using npm
npm install canvas
# Using yarn
yarn add canvas
# Using pnpm
pnpm add canvas
```
This is not required for browser environments, as they use the native Canvas API.
## 🚀 Usage
### 🔑 Initialization
Import the package and initialize a client with your NovelAI access token.
```javascript
// ESM
import { NovelAI } from "nekoai-js";
// CommonJS
const { NovelAI } = require("nekoai-js");
// Initialize with token
const client = new NovelAI({
token: "your_access_token",
});
```
### 🖼️ Image Generation
Generate images with the `generateImage` method. The method takes parameters directly or as a `Metadata` object.
```javascript
import { NovelAI, Model, Resolution, Sampler } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Generate using parameters directly
const images = await client.generateImage({
prompt: "1girl, cute, anime style, detailed",
model: Model.V4_5,
resPreset: Resolution.NORMAL_PORTRAIT,
n_samples: 1,
seed: 1234567890, // Fixed seed for reproducibility
});
// Save images (Node.js environment)
for (const image of images) {
await image.save("./output");
console.log(`Image saved: ${image.filename}`);
}
// Get image data URL (browser environment)
for (const image of images) {
const dataUrl = image.toDataURL();
console.log(`Image data URL: ${dataUrl.substring(0, 50)}...`);
}
```
### Streaming Generation (V4/V4.5 Models)
V4 and V4.5 models support real-time streaming, allowing you to watch the generation process as it happens. Enable streaming by passing `true` as the second parameter to `generateImage()`.
```javascript
import { NovelAI, Model, Resolution, EventType } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Generate with streaming enabled
const response = await client.generateImage(
{
prompt: "1girl, cute, anime style",
model: Model.V4_5,
resPreset: Resolution.NORMAL_PORTRAIT,
steps: 28,
seed: 3417044607,
},
true, // Enable streaming
);
// Handle streaming response
if (response && typeof response[Symbol.asyncIterator] === "function") {
console.log("Streaming generation steps...");
for await (const event of response) {
if (event.event_type === EventType.INTERMEDIATE) {
// Save intermediate steps
await event.image.save(`./output/step_${event.step_ix}.jpg`);
console.log(`Step ${event.step_ix} completed`);
} else if (event.event_type === EventType.FINAL) {
// Save final result
await event.image.save(`./output/final_result.png`);
console.log("Generation complete!");
}
}
}
```
### Multi-Character Generation (V4.5)
V4.5 models support generating multiple characters with character-specific prompts and positioning.
```javascript
import { NovelAI, Model, Resolution } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Create character prompts with positioning
const characterPrompts = [
{
prompt: "girl, red hair, red dress",
uc: "bad hands, bad anatomy",
center: { x: 0.3, y: 0.3 },
},
{
prompt: "boy, blue hair, blue uniform",
uc: "bad hands, bad anatomy",
center: { x: 0.7, y: 0.7 },
},
];
// Generate image with multiple characters
const images = await client.generateImage({
prompt: "two people standing together, park background",
model: Model.V4_5,
resPreset: Resolution.NORMAL_LANDSCAPE,
characterPrompts,
});
// Process the resulting images
for (const image of images) {
// Browser
const dataUrl = image.toDataURL();
// Node.js
await image.save("./output");
}
```
### Image to Image
To perform `img2img` action, set `action` parameter to `Action.IMG2IMG`, and provide a source image. Use the `parseImage` utility to handle multiple image formats seamlessly.
```javascript
import { NovelAI, Action, parseImage } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Parse image using the utility (supports multiple formats)
const sourceImage = await parseImage("./input/image.png");
const images = await client.generateImage({
prompt: "1girl, fantasy outfit",
action: Action.IMG2IMG,
width: sourceImage.width,
height: sourceImage.height,
image: sourceImage.base64,
strength: 0.5, // Lower = more similar to original
noise: 0.1,
});
for (const image of images) {
await image.save("./output");
}
```
### Inpainting (V4.5)
V4.5 supports advanced inpainting for selective image editing. Use `Model.V4_5_INP` for optimal inpainting results.
```javascript
import { NovelAI, Model, Action, parseImage } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Parse images using the utility function
const sourceImage = await parseImage("./input/source.png");
const maskImage = await parseImage("./input/mask.png");
const images = await client.generateImage({
prompt: "beautiful flower garden",
model: Model.V4_5_INP, // Use inpainting model
action: Action.INPAINT,
image: sourceImage.base64,
mask: maskImage.base64,
steps: 28,
seed: 3417044607,
});
for (const image of images) {
await image.save("./output");
}
```
### Streaming Inpainting
Combine streaming with inpainting to watch the inpainting process in real-time.
```javascript
import { NovelAI, Model, Action, EventType, parseImage } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Parse images
const sourceImage = await parseImage("./input/source.png");
const maskImage = await parseImage("./input/mask.png");
// Generate with streaming inpainting
const response = await client.generateImage(
{
prompt: "1girl, cute",
model: Model.V4_5_INP,
action: Action.INPAINT,
image: sourceImage.base64,
mask: maskImage.base64,
steps: 28,
},
true, // Enable streaming
);
// Process streaming results
for await (const event of response) {
if (event.event_type === EventType.INTERMEDIATE) {
await event.image.save(`./output/inpaint_step_${event.step_ix}.jpg`);
} else if (event.event_type === EventType.FINAL) {
await event.image.save(`./output/inpaint_final.png`);
}
}
```
### Director Tools
NovelAI offers several Director tools for image manipulation, all accessible through dedicated methods. These tools automatically handle various image input formats through the built-in `parseImage` utility.
```javascript
import { NovelAI } from "nekoai-js";
// Initialize client
const client = new NovelAI({
token: "your_access_token",
});
// Line Art - supports file paths, Blobs, Files, URLs, etc.
const lineArtResult = await client.lineArt("./input/image.png");
await lineArtResult.save("./output");
// Background Removal
const bgRemovalResult = await client.backgroundRemoval("./input/image.png");
await bgRemovalResult.save("./output");
// Change Emotion
const emotionResult = await client.changeEmotion(
"./input/image.png",
"happy", // Target emotion
"neutral", // Additional prompt
0, // Emotion level (0-5)
);
await emotionResult.save("./output");
// Other Director Tools
const declutterResult = await client.declutter("./input/image.png", "dreamy", 0);
const colorizeResult = await client.colorize("./input/image.png","dream, mirror",1,);
```
All Director Tool methods automatically handle ZIP-compressed responses from the API, extracting the image data for you. This works across both Node.js and browser environments.
### Image Input Support
NekoAI-JS provides comprehensive image input support through the `parseImage` utility function, which automatically handles format detection and conversion across different environments.
#### Supported Image Formats
The `parseImage` function supports multiple input types for maximum flexibility:
```javascript
import { NovelAI, parseImage } from "nekoai-js";
const client = new NovelAI({
token: "your_access_token",
});
// 1. File paths (Node.js only)
const image1 = await parseImage("./input/photo.png");
const image2 = await parseImage("./input/drawing.jpg");
// 2. Uint8Array / ArrayBuffer (both environments)
const imageData = new Uint8Array(/* image bytes */);
const image3 = await parseImage(imageData);
// 3. Base64 strings (both environments)
const base64String = "iVBORw0KGgoAAAANSUhEUgAA...";
const image4 = await parseImage(`data:image/png;base64,${base64String}`);
// 4. URLs (browser only)
const image5 = await parseImage("https://example.com/image.png");
const image6 = await parseImage("...");
// 5. File objects from input elements (browser only)
const fileInput = document.getElementById("imageInput");
const file = fileInput.files[0];
const image7 = await parseImage(file);
// 6. Blob objects (browser only)
const response = await fetch("https://example.com/image.png");
const blob = await response.blob();
const image8 = await parseImage(blob);
// 7. Canvas elements (browser only)
const canvas = document.getElementById("myCanvas");
const image9 = await parseImage(canvas);
// 8. Image elements (browser only)
const imgElement = document.getElementById("myImage");
const image10 = await parseImage(imgElement);
// The parseImage function returns { width, height, base64 }
console.log(`Image dimensions: ${image1.width}x${image1.height}`);
// Use with generation or director tools
const images = await client.generateImage({
prompt: "enhance this image",
action: Action.IMG2IMG,
image: image1.base64,
width: image1.width,
height: image1.height,
});
```
#### Cross-Platform Compatibility
- **Node.js**: Supports file paths, Uint8Array, ArrayBuffer, and URLs
- **Browser**: Supports File, Blob, Canvas, Image elements, Data URLs, and remote URLs
- **Both**: Supports Uint8Array, ArrayBuffer, and base64 data
#### Automatic Format Detection
The utility automatically:
- Detects image dimensions
- Converts to base64 format for API compatibility
- Handles JPEG, PNG, and other common formats
- Preserves image quality during conversion
- Works seamlessly across different environments
### Advanced Usage Examples
Here are practical examples showing how to use `parseImage` with various input formats and director tools:
#### Node.js Environment
```javascript
import { NovelAI, parseImage } from "nekoai-js";
const client = new NovelAI({
token: "your_access_token",
});
// Example 1: Processing multiple image formats
const imageFormats = [
"./input/photo.png",
"./input/sketch.jpg",
"./input/artwork.webp"
];
for (const imagePath of imageFormats) {
const parsedImage = await parseImage(imagePath);
console.log(`Processing ${imagePath}: ${parsedImage.width}x${parsedImage.height}`);
const result = await client.lineArt(imagePath);
await result.save(`./output/lineart_${Date.now()}.png`);
}
// Example 2: Using with binary data
const fs = require("fs");
const imageBuffer = fs.readFileSync("./input/photo.png");
const result = await client.backgroundRemoval(imageBuffer);
await result.save("./output");
```
#### Browser Environment
```javascript
import { NovelAI, parseImage } from "nekoai-js";
const client = new NovelAI({
token: "your_access_token",
});
// Example 1: File upload with preview
const fileInput = document.getElementById("fileInput");
fileInput.addEventListener("change", async (e) => {
const file = e.target.files[0];
if (!file) return;
// Parse the image to get dimensions
const parsedImage = await parseImage(file);
console.log(`Uploaded image: ${parsedImage.width}x${parsedImage.height}`);
// Process with director tool
const result = await client.lineArt(file);
// Display result
const img = document.createElement("img");
img.src = result.toDataURL();
img.style.maxWidth = "500px";
document.body.appendChild(img);
});
// Example 2: Canvas processing
const canvas = document.getElementById("drawingCanvas");
const processCanvas = async () => {
const result = await client.sketch(canvas);
// Create download link
const link = document.createElement("a");
link.download = "sketch_result.png";
link.href = result.toDataURL();
link.click();
};
// Example 3: URL processing with error handling
const processImageUrl = async (url) => {
try {
const result = await client.colorize(url);
return result.toDataURL();
} catch (error) {
console.error("Failed to process image:", error);
return null;
}
};
```
### Using Custom Hosts
NekoAI-JS supports using custom hosts for API requests. This is useful if you need to use a different endpoint or if you're using a proxy server.
```javascript
import { NovelAI, Model, Host } from "nekoai-js";
// Method 1: Use predefined hosts
const client1 = new NovelAI({
token: "your_access_token",
host: Host.API, // Use API host instead of default WEB host
});
const images1 = await client1.generateImage({
prompt: "1girl, cute, anime style",
model: Model.V3,
});
// Method 2: Use a custom host URL
const client2 = new NovelAI({
token: "your_access_token",
host: "https://your-custom-host.com", // Direct URL string
});
const images2 = await client2.generateImage({
prompt: "1girl, cute, anime style",
model: Model.V4,
});
// Custom hosts work with all other client methods
const lineArtResult = await client2.lineArt("./input/image.png");
```
You can use custom hosts for:
1. Connection to third-party API providers
2. Working with proxies
3. Connecting to local NovelAI servers
4. Load balancing between multiple endpoints
### Custom Retry Configuration
NekoAI-JS includes a built-in retry mechanism for handling rate limits and temporary API failures. By default, retries are enabled with reasonable defaults, but you can customize this behavior:
```javascript
import { NovelAI, Model } from "nekoai-js";
// Initialize client with custom retry settings
const client = new NovelAI({
token: "your_access_token",
retry: {
enabled: true, // Enable retries
maxRetries: 5, // Maximum 5 retry attempts
baseDelay: 2000, // Start with 2 second delay
maxDelay: 60000, // Maximum delay of 1 minute
retryStatusCodes: [429], // Only retry on rate limit errors
},
});
// Generate image with retry
try {
const images = await client.generateImage({
prompt: "1girl, cute, anime style",
model: Model.V4_5,
});
console.log("Success after potential retries!");
} catch (error) {
console.error("Failed even after retries:", error);
}
```
You can also disable retries completely if needed:
```javascript
const client = new NovelAI({
token: "your_access_token",
retry: {
enabled: false, // Disable retries
},
});
```
The retry mechanism uses exponential backoff with jitter to prevent overwhelming the API service when it's under stress.
## References
[NovelAI Documentation](https://docs.novelai.net/)
[NovelAI Backend API](https://api.novelai.net/docs)
[NovelAI Unofficial Knowledgebase](https://naidb.miraheze.org/wiki/Using_the_API)
[NekoAI-API Python Package](https://github.com/Nya-Foundation/NekoAI-API)