object-remover.js
1 /** 2 * Yollomi object remover — POST /api/ai/object-remover 3 */ 4 import * as path from 'node:path'; 5 import { cli, Strategy } from '@jackwener/opencli/registry'; 6 import { CliError } from '@jackwener/opencli/errors'; 7 import { log } from '@jackwener/opencli/logger'; 8 import { YOLLOMI_DOMAIN, yollomiPost, downloadOutput, fmtBytes } from './utils.js'; 9 cli({ 10 site: 'yollomi', 11 name: 'object-remover', 12 description: 'Remove unwanted objects from images (3 credits)', 13 domain: YOLLOMI_DOMAIN, 14 strategy: Strategy.COOKIE, 15 args: [ 16 { name: 'image', positional: true, required: true, help: 'Image URL' }, 17 { name: 'mask', positional: true, required: true, help: 'Mask image URL (white = area to remove)' }, 18 { name: 'output', default: './yollomi-output', help: 'Output directory' }, 19 { name: 'no-download', type: 'boolean', default: false, help: 'Only show URL' }, 20 ], 21 columns: ['status', 'file', 'size', 'url'], 22 func: async (page, kwargs) => { 23 log.status('Removing object...'); 24 const data = await yollomiPost(page, '/api/ai/object-remover', { 25 image: kwargs.image, 26 mask: kwargs.mask, 27 }); 28 const url = data.image || (data.images?.[0]); 29 if (!url) 30 throw new CliError('EMPTY_RESPONSE', 'No result', 'Check image and mask'); 31 if (kwargs['no-download']) 32 return [{ status: 'removed', file: '-', size: '-', url }]; 33 try { 34 const filename = `yollomi_removed_${Date.now()}.png`; 35 const { path: fp, size } = await downloadOutput(url, kwargs.output, filename); 36 return [{ status: 'saved', file: path.relative('.', fp), size: fmtBytes(size), url }]; 37 } 38 catch { 39 return [{ status: 'download-failed', file: '-', size: '-', url }]; 40 } 41 }, 42 });