Spaces:
Running
on
CPU Upgrade
Running
on
CPU Upgrade
File size: 4,442 Bytes
1373ff5 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 |
import { client } from "@gradio/client"
import { generateSeed } from "../../utils/misc/generateSeed.mts"
import { getValidNumber } from "../../utils/validators/getValidNumber.mts"
import { convertToWebp } from "../../utils/image/convertToWebp.mts"
import { addBase64HeaderToPng } from "../../utils/image/addBase64HeaderToPng.mts"
// TODO add a system to mark failed instances as "unavailable" for a couple of minutes
// console.log("process.env:", process.env)
// note: to reduce costs I use the small A10s (not the large)
// anyway, we will soon not need to use this cloud anymore
// since we will be able to leverage the Inference API
const gradioSpaceApiUrl = `https://jbilcke-hf-ai-tube-model-pulid.hf.space`
const gradioSpace = `jbilcke-hf/ai-tube-model-pulid`
const secretToken = `${process.env.VC_MICROSERVICE_SECRET_TOKEN || ""}`
// console.log("DEBUG:", JSON.stringify({ instances, secretToken }, null, 2))
export async function generateImagePulibAsBase64(options: {
positivePrompt: string;
negativePrompt?: string;
identityImage?: string;
seed?: number;
width?: number;
height?: number;
nbSteps?: number;
}): Promise<string> {
const positivePrompt = options?.positivePrompt || ""
if (!positivePrompt) {
throw new Error("missing prompt")
}
// the negative prompt CAN be missing, since we use a trick
// where we make the interface mandatory in the TS doc,
// but browsers might send something partial
const negativePrompt = options?.negativePrompt || ""
// we treat 0 as meaning "random seed"
const seed = (options?.seed ? options.seed : 0) || generateSeed()
const width = getValidNumber(options?.width, 256, 1024, 512)
const height = getValidNumber(options?.height, 256, 1024, 512)
const nbSteps = getValidNumber(options?.nbSteps, 1, 8, 4)
// console.log("SEED:", seed)
const identityImage = `${options.identityImage || ""}`
const positive = [
positivePrompt,
].filter(word => word)
.join(", ")
const negative = [
negativePrompt,
"watermark",
"copyright",
"blurry",
// "artificial",
// "cropped",
"low quality",
"ugly",
'flaws in the eyes',
'flaws in the face',
'flaws',
'lowres',
'non-HDRi',
'low quality',
'worst quality',
'artifacts noise',
'text',
'glitch',
'deformed',
'mutated',
'disfigured hands',
'low resolution',
'partially rendered objects',
'deformed or partially rendered eyes',
'ddeformed eyeballs',
'cross-eyed',
].filter(word => word)
.join(", ")
const api = await client(gradioSpaceApiUrl, {
hf_token: `${process.env.VC_HF_API_TOKEN}` as any
})
// we hardcode the number of steps to 4
const steps = 4
// console.log("querying " + gradioSpaceApiUrl + " with tons of params")
const rawResponse = (await api.predict("/run", [
secretToken, // # str in 'parameter_4' Textbox component
identityImage || "", // 'ID image (main)' Image component
"", // 'Additional ID image (auxiliary)' Image component
"", // 'Additional ID image (auxiliary)' Image component
"", // 'Additional ID image (auxiliary)' Image component
positive, // # str in 'Prompt' Textbox component
negative, // # str in 'Negative Prompt' Textbox component
1.2, // # int | float (numeric value between 1 and 1.5) in 'CFG, recommend value range [1, 1.5], 1 will be faster ' Slider component
generateSeed(), //, # int | float (numeric value between 0 and 4294967295) in 'Seed' Slider component
steps, // # int | float (numeric value between 1 and 100) in 'Steps' Slider component
height, // # int | float (numeric value between 512 and 1280) in 'Height' Slider component
width, // # int | float (numeric value between 512 and 1280) in 'Width' Slider component
0.8, // # int | float (numeric value between 0 and 5) in 'ID scale' Slider component
"fidelity", // # str (Option from: ['fidelity', 'extremely style']) in 'mode' Dropdown component
false, // 'ID Mix (if you want to mix two ID image, please turn this on, otherwise, turn this off)' Checkbox component
])) as any
const result = rawResponse?.data?.[0] as string
if (!result?.length) {
throw new Error(`the returned image was empty`)
}
try {
const finalImage = await convertToWebp(addBase64HeaderToPng(result))
return finalImage
} catch (err) {
// console.log("err:", err)
throw new Error(err)
}
} |