fotor-skills
Fotor AI image generator and AI video generator for photo editing, background remover, background replacement, product photos, ad creatives, social media graphics, poster and banner design, image upscaling, photo restoration, portrait enhancement, text-to-video, and image-to-video. Built for e-commerce, marketing, branding, and content creation.
git clone https://github.com/fotor-ai/fotor-skills
T=$(mktemp -d) && git clone --depth=1 https://github.com/fotor-ai/fotor-skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/fotor-skills" ~/.claude/skills/fotor-ai-fotor-skills-fotor-skills && rm -rf "$T"
skills/fotor-skills/SKILL.mdfotor-skills
Fotor OpenAPI skill for AI image generation, AI photo editing, AI video generation, product photos, ad creatives, social media graphics, background removal, photo restoration, and image upscaling.
This skill should match user requests expressed in outcome language first, not SDK language. Keep technical details behind the scenes unless they are needed to unblock execution.
When This Skill Matches
Use this skill when the user asks for outcomes such as:
- Generate AI images from a text prompt
- Edit or restyle an existing photo
- Turn a product shot into an e-commerce or ad-ready asset
- Create posters, banners, covers, thumbnails, or social media graphics
- Remove or replace an image background
- Restore, enhance, or upscale a blurry or old photo
- Generate AI videos from text, one image, multiple images, or start/end frames
- Batch-produce visual assets for branding, content, or marketing campaigns
Search Intent Coverage
Common search phrases this skill should be able to match include:
- AI photo editor
- AI image generator
- AI video generator
- Image to image
- Text to image
- Text to video
- Product photo generator
- Ad creative generator
- Marketing visual generator
- Poster maker
- Banner maker
- Social media post generator
- Cover and thumbnail generator
- Background remover
- Background replacement
- Photo restoration
- Image upscaler
- E-commerce image generation
- Brand asset generation
For API key application and product details, see
https://developers.fotor.com/fotor-skills/.
Use
uv as the bootstrap layer. Prefer a skill-local Python 3.12 environment and run bundled scripts from that local environment instead of the system Python.
Runtime Setup
Keep setup lightweight and local to the skill directory.
Install
uv first if it is missing:
# macOS / Linux curl -LsSf https://astral.sh/uv/install.sh | sh # Windows (PowerShell) powershell -ExecutionPolicy ByPass -c "irm https://astral.sh/uv/install.ps1 | iex"
Typical first-run setup:
uv python install 3.12 uv venv --python 3.12 .venv ./.venv/bin/python scripts/ensure_sdk.py
Setup rules:
- Prefer a local Python 3.12 environment in the skill directory.
- Use
to prepare Python 3.12 and createuv
when the local environment is missing..venv - Run bundled scripts from the local skill environment, not the system Python.
- Ensure
is set. If the user asks where to get a key, wants the officialFOTOR_OPENAPI_KEY
homepage during setup, or needs a key + homepage walkthrough, readfotor-skills
first.references/get_api_key.md
Current default interpreter paths:
- POSIX:
./.venv/bin/python - Windows:
.venv\\Scripts\\python.exe
Interaction Rules
- Speak in user-task language first. Do not lead with SDK, scripts, JSON, model IDs, or parameter tables unless they are needed to unblock the task or the user explicitly asks.
- Ask for only one missing blocker at a time.
- Once the minimum required information is present, execute immediately. Do not send vague transition messages like "I’m starting now" unless execution has actually started and a result or clear in-progress status will follow.
- If execution will take noticeable time, say that the task is running and give a short expectation such as "usually takes a few seconds to a few dozen seconds; I’ll send the result when it’s ready."
- If credentials are missing, resolve that blocker quickly and then return to the original task instead of turning the conversation into a long setup lesson.
- When the local skill environment is missing, prepare it with
before installing dependencies or executing the task. Avoid installing into the system Python unless the user explicitly asks.uv - Choose the model and default parameters internally unless the user explicitly requests a specific model or technical control.
- Return the result as soon as it is ready. Do not make the user ask follow-up questions like "where is the image?"
- If the user asks how to recharge, buy credits, top up, or purchase tokens, use
and follow its recharge guidance flow.references/credits-and-recharge.md - If a task fails because credits are insufficient, do not stop at the raw error. Use
to explain the failure and provide recharge guidance.references/credits-and-recharge.md - If an update reminder is available, keep it to one short non-blocking sentence and continue the current task.
Scripts
scripts/ensure_sdk.py
scripts/ensure_sdk.pyCross-platform (Windows / macOS / Linux) script to install or upgrade
fotor-sdk to the latest PyPI release with uv pip install --python <interpreter>. Run before every task.
- No args — install or upgrade to the latest PyPI release
— same behavior, kept as an explicit alias--upgrade
scripts/run_task.py
scripts/run_task.pyExecute one or more Fotor tasks from JSON. Handles client init, polling, and progress.
Single task:
echo '{"task_type":"text2image","params":{"prompt":"A cat","model_id":"seedream-4-5-251128"}}' \ | ./.venv/bin/python scripts/run_task.py
Batch (array):
echo '[ {"task_type":"text2image","params":{"prompt":"A cat","model_id":"seedream-4-5-251128"},"tag":"cat"}, {"task_type":"text2video","params":{"prompt":"Sunset","model_id":"kling-v3","duration":5},"tag":"sunset"} ]' | ./.venv/bin/python scripts/run_task.py --concurrency 5
Options:
--input FILE, --concurrency N (default 5), --poll-interval S (default 2.0), --timeout S (default 1200).
Output: JSON with
task_id, status, success, result_url, error, elapsed_seconds, creditsIncrement, tag.
Automatic fallback:
- If a task fails on its primary model and the current
matches a built-in fallback mapping,task_type + model_id
automatically retries once with the fallback model.run_task.py - If the failure is insufficient credits (
/code=510
),No enough credits
returns the failure immediately and does not retry on a fallback model.run_task.py - The output includes
,fallback_used
, andoriginal_model_id
.fallback_model_id
scripts/upload_image.py
scripts/upload_image.pyUpload a local image file through Fotor's signed upload flow and return a reusable image URL.
./.venv/bin/python scripts/upload_image.py ./input.jpg --task-type image2image
The script:
- Calls
with the mapped upload/v1/upload/sign
andtypesuffix - Uploads the local file to the signed target
- Prints JSON containing
andfile_urlupload_url
Use
file_url as the image_url, start_image_url, end_image_url, or an item inside image_urls for image-based tasks.
Supported task-to-upload mapping:
->image2imageimg2img
->image_upscaleimg_upscale
->background_removebg_remove
->single_image2videoimg2video
->start_end_frame2videoimg2video
->multiple_image2videoimg2video
scripts/check_skill_update.py
scripts/check_skill_update.pyCheck whether the installed skill has a newer version available for the current install source.
./.venv/bin/python scripts/check_skill_update.py --mark-notified --check-interval-hours 24
For development/testing when install-source metadata is unavailable:
./.venv/bin/python scripts/check_skill_update.py --install-source skills-github --slug fotor-skills --current-version 1.0.0 --github-source fotor-ai/fotor-skills --mark-notified --check-interval-hours 24
The script:
- Detects the install source first:
orclawhubskills-github - For
, reads installedclawhub
and fetches the latest version via_meta.jsonclawhub inspect <slug> --json - For
, reads localskills-github
frontmatter top-levelSKILL.md
field, falls back to legacyversion
, finds the GitHub source, and fetches the remotemetadata.version
version plusSKILL.md
highlights when availableCHANGELOG.md - Prints JSON with
,install_source
,current_version
,latest_version
, andupdate_availableshould_notify - Stores the last-notified version in a local state file when
is used--mark-notified - Caches the last successful version check and supports a minimum recheck interval via
(default 24)--check-interval-hours - Includes
so the reminder can mention the main highlights without dumping the full changelogchangelog_preview - Supports development/testing overrides such as
,--install-source
,--slug
, and--current-version--github-source
Reference Files
Only read the reference files that match the current need. Do not load all of them by default.
Task Execution References
Read these when choosing a model, validating parameters, or mapping an ambiguous user request to a recommended workflow:
-- image model IDs, T2I/I2I capabilities, per-model parameter constraints (resolution, ratios, input limits, max refs)references/image_models.md
-- video model IDs, T2V/I2V/SE/MI capabilities, per-model parameter constraints (duration, resolution, ratios, input limits, audio)references/video_models.md
-- full function signatures and parameter tables for all 8 task typesreferences/parameter_reference.md
-- scenario-to-model mapping for image generation (T2I, I2I, utilities); read when user intent is ambiguousreferences/image_scenarios.md
-- scenario-to-model mapping for video generation (T2V, I2V, SE, MI); read when user intent is ambiguousreferences/video_scenarios.md
Operational References
Read these only when the user asks about installation, upgrade, workspace layout, API key acquisition, or the official
fotor-skills homepage. Do not read them for normal image/video task execution.
-- credit lookup, payment links, recharge guidance, and insufficient-credit handlingreferences/credits-and-recharge.md
-- guide for giving the officialreferences/get_api_key.md
homepage, sending users to the API-key dashboard entry, and configuringfotor-skillsFOTOR_OPENAPI_KEY
-- how to install or upgradereferences/install-or-upgrade.md
for both ClawHub andfotor-skills
, and how to phrase a concise non-pushy upgrade remindernpx skills
Workflow
- If
is missing, install it first using the platform-appropriate installer shown in Setup.uv - If Python 3.12 is not already available to
, runuv
.uv python install 3.12 - If
is missing, create it with.venv
.uv venv --python 3.12 .venv - Run
before every task to install or upgrade the latest./.venv/bin/python scripts/ensure_sdk.py
withfotor-sdk
.uv - After
is ready and.venv
completes, silently runensure_sdk.py
. Do not inspect the state file manually. Only if it reports./.venv/bin/python scripts/check_skill_update.py --mark-notified --check-interval-hours 24
, readshould_notify: true
and send one short non-blocking update reminder. If the check fails or times out, ignore it and continue without mentioning the failure.references/install-or-upgrade.md - If the user asks how to get an API key or wants the official
homepage during setup, readfotor-skills
and include both links before expanding into lower-level setup details.references/get_api_key.md - Verify
is set.FOTOR_OPENAPI_KEY - If the user asks about account credits, recharge, buying credits, or payment links, read
, then use the appropriate balance or payment-links flow from that reference.references/credits-and-recharge.md - For image-based tasks that start from a local file, first run
and keep the returned./.venv/bin/python scripts/upload_image.py <local-file> --task-type <task-type>
.file_url - Read the appropriate model reference to choose
. Each model's per-model spec section lists supported resolutions, aspect ratios, duration, input constraints, and max reference images.model_id - If user intent is ambiguous (no specific model requested), consult the scenario files (
/image_scenarios.md
) for recommended model + params.video_scenarios.md - Validate parameters against the chosen model's spec before calling -- check resolution, aspect ratio, duration, and multi-image limits.
- Quick path -- pipe JSON into
(works for both single and batch)../.venv/bin/python scripts/run_task.py - Custom path -- write inline Python using the SDK directly (see examples below), still preferring the local
interpreter..venv - Check
in output. Chainresult_url
if higher resolution needed.image_upscale
If the user asks to check account credits or remaining credits, read
references/credits-and-recharge.md and use the SDK client flow described there instead of run_task.py.
Built-in automatic fallback mappings:
:text2image
->gemini-3.1-flash-image-previewseedream-5-0-260128
:image2image
->gemini-3.1-flash-image-previewseedream-5-0-260128
:text2video
->seedance-1-5-pro-251215kling-v3
:single_image2video
->seedance-1-5-pro-251215kling-v3
:start_end_frame2video
->kling-video-o1viduq2-turbo
:multiple_image2video
->kling-v3-omnikling-video-o1
Available Task Types
| task_type | Function | Required Params |
|---|---|---|
| | , |
| | , , |
| | |
| | |
| | , |
| | , , |
| | , , , |
| | , , (≥2) |
For full parameter details (defaults,
on_poll, **extra), read references/parameter_reference.md.
Credits and Recharge
For any balance lookup, recharge guidance, or insufficient-credit case, read
references/credits-and-recharge.md.
Keep
SKILL.md focused on routing:
- Use the credits reference when the user asks about remaining balance, total credits, recharge, top-up, or payment links.
- Use the same reference when a task fails with
orcode=510
.No enough credits - Keep detailed API examples, field meanings, and user-facing recharge wording in the reference instead of expanding this main skill file.
Inline Python Examples
When
scripts/run_task.py is insufficient (custom logic, chaining, progress callbacks):
Client Init
import os from fotor_sdk import FotorClient client = FotorClient(api_key=os.environ["FOTOR_OPENAPI_KEY"])
Single Task
from fotor_sdk import text2image result = await text2image(client, prompt="A diamond kitten", model_id="seedream-4-5-251128") print(result.result_url)
Batch with TaskRunner
from fotor_sdk import TaskRunner, TaskSpec runner = TaskRunner(client, max_concurrent=5) specs = [ TaskSpec("text2image", {"prompt": "A cat", "model_id": "seedream-4-5-251128"}, tag="cat"), TaskSpec("text2video", {"prompt": "Sunset", "model_id": "kling-v3", "duration": 5}, tag="sunset"), ] results = await runner.run(specs)
Video with Audio
from fotor_sdk import text2video result = await text2video(client, prompt="Jazz band", model_id="kling-v3", audio_enable=True, audio_prompt="Smooth jazz")
TaskResult
result.success # bool: True when COMPLETED with result_url result.result_url # str | None result.status # TaskStatus: COMPLETED / FAILED / TIMEOUT / IN_PROGRESS / CANCELLED result.error # str | None (e.g. "NSFW_CONTENT") result.elapsed_seconds # float result.creditsIncrement # int | float: credits consumed by this task result.metadata # dict (includes "tag" from TaskRunner)
Error Handling
- Single task: catch
(hasFotorAPIError
attribute)..code - Batch: check
per item; runner never raises on individual failures.result.success - NSFW: appears as
in TaskResult.error="NSFW_CONTENT" - Insufficient credits: if
, exception text, or a combined fallback error containsresult.error
orcode=510
, treat it as a recharge case. Tell the user credits are insufficient, then fetch and present payment links.No enough credits
For troubleshooting, enable SDK debug logging:
logging.getLogger("fotor_sdk").setLevel(logging.DEBUG).