Spaces:
Runtime error
Runtime error
File size: 10,347 Bytes
bd81242 1cd7e2c 37e5f7e bd81242 88c3112 1cd7e2c bd81242 61b9df0 1cd7e2c bd81242 0af4039 cfaeada bd81242 6c73e7a bd81242 74a4765 1cd7e2c dfa75b2 1cd7e2c dfa75b2 1cd7e2c 0af4039 1cd7e2c 9748f58 37e5f7e 5cd1557 3c4d0e1 cfaeada 9748f58 593deb8 bd81242 cfaeada 593deb8 cfaeada 593deb8 cfaeada 37e5f7e bd81242 37e5f7e 88c3112 bd81242 88c3112 37e5f7e bd81242 37e5f7e bd81242 37e5f7e bd81242 37e5f7e bd81242 37e5f7e 3b1dbeb 593deb8 7b994e6 dfa75b2 cfaeada 593deb8 7982fcd 593deb8 7b994e6 593deb8 cfaeada 593deb8 3b1dbeb 9748f58 5cd1557 9748f58 5cd1557 3b1dbeb 593deb8 3b1dbeb 9748f58 3b1dbeb cfaeada bae56f1 3b1dbeb cfaeada 9748f58 3b1dbeb 593deb8 bd81242 593deb8 bd81242 9b36efd 593deb8 37e5f7e bd81242 1cd7e2c 6c73e7a bae56f1 6c73e7a 5cd1557 bae56f1 6c73e7a bae56f1 5cd1557 6c73e7a bae56f1 9b36efd bae56f1 6c73e7a bae56f1 6c73e7a bae56f1 cfaeada 593deb8 6c73e7a 3c4d0e1 bae56f1 37e5f7e bd81242 |
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 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 |
# file stuff
import os
import sys
import zipfile
import requests
import tempfile
from io import BytesIO
import random
import string
#image generation stuff
from PIL import Image
# gradio / hf stuff
import gradio as gr
from openai import OpenAI
from dotenv import load_dotenv
# stats stuff
from pymongo.mongo_client import MongoClient
from pymongo.server_api import ServerApi
import time
# countdown stuff
from datetime import datetime, timedelta
load_dotenv()
openai_key = os.getenv("OPENAI_API_KEY")
pw_key = os.getenv("PW")
if openai_key == "<YOUR_OPENAI_API_KEY>":
openai_key = ""
if pw_key == "<YOUR_PW>":
pw_key = ""
if pw_key == "":
sys.exit("Please Provide A Password in the Environment Variables")
if openai_key == "":
sys.exit("Please Provide Your OpenAI API Key")
# Connect to MongoDB
uri = os.getenv("MONGO_URI")
mongo_client = MongoClient(uri, server_api=ServerApi('1'))
mongo_db = mongo_client.pdr
mongo_collection = mongo_db["images"]
image_labels_global = []
image_paths_global = []
#load challenges
challenges = []
with open('challenges.txt', 'r') as file:
for line in file:
challenges.append(line.strip())
# pick a random challenge
def get_challenge():
global challenge
challenge = random.choice(challenges)
return challenge
# set initial challenge
challenge = get_challenge()
def update_labels(show_labels):
updated_gallery = [(path, label if show_labels else "") for path, label in zip(image_paths_global, image_labels_global)]
return updated_gallery
def generate_images_wrapper(prompts, pw, model, show_labels):
global image_paths_global, image_labels_global
image_paths, image_labels = generate_images(prompts, pw, model)
image_paths_global = image_paths
# store this as a global so we can handle toggle state
image_labels_global = image_labels
image_data = [(path, label if show_labels else "") for path, label in zip(image_paths, image_labels)]
return image_data
def download_image(url):
response = requests.get(url)
if response.status_code == 200:
return response.content
else:
raise Exception(f"Failed to download image from URL: {url}")
def zip_images(image_paths_and_labels):
zip_file_path = tempfile.NamedTemporaryFile(delete=False, suffix='.zip').name
with zipfile.ZipFile(zip_file_path, 'w') as zipf:
for image_url, _ in image_paths_and_labels:
image_content = download_image(image_url)
random_filename = ''.join(random.choices(string.ascii_letters + string.digits, k=10)) + ".png"
# Write the image content to the zip file with the random filename
zipf.writestr(random_filename, image_content)
return zip_file_path
def download_all_images():
global image_paths_global, image_labels_global
if not image_paths_global:
raise gr.Error("No images to download.")
image_paths_and_labels = list(zip(image_paths_global, image_labels_global))
zip_path = zip_images(image_paths_and_labels)
image_paths_global = [] # Reset the global variable
image_labels_global = [] # Reset the global variable
return zip_path
def generate_images(prompts, pw, model):
# Check for a valid password
if pw != os.getenv("PW"):
raise gr.Error("Invalid password. Please try again.")
image_paths = [] # holds urls of images
image_labels = [] # shows the prompt in the gallery above the image
users = [] # adds the user to the label
# Split the prompts string into individual prompts based on semicolon separation
prompts_list = [prompt for prompt in prompts.split(';') if prompt]
for i, entry in enumerate(prompts_list):
entry_parts = entry.split('-', 1) # Split by the first dash found
if len(entry_parts) == 2:
#raise gr.Error("Invalid prompt format. Please ensure it is in 'initials-prompt' format.")
user_initials, text = entry_parts[0].strip(), entry_parts[1].strip() # Extract user initials and the prompt
else:
text = entry.strip() # If no initials are provided, use the entire prompt as the text
user_initials = ""
users.append(user_initials) # Append user initials to the list
try:
openai_client = OpenAI(api_key=openai_key)
start_time = time.time()
#make a prompt with the challenge and text
prompt_w_challenge = f"{challenge}: {text}"
response = openai_client.images.generate(
prompt=prompt_w_challenge,
model=model, # dall-e-2 or dall-e-3
quality="standard", # standard or hd
size="512x512" if model == "dall-e-2" else "1024x1024", # varies for dalle-2 and dalle-3
n=1, # Number of images to generate
)
end_time = time.time()
gen_time = end_time - start_time # total generation time
image_url = response.data[0].url
# conditionally render the user to the label with the prompt
image_label = f"{i}: {text}" if user_initials == "" else f"{i}: {user_initials}-{text}, "
try:
# Save the prompt, model, image URL, generation time and creation timestamp to the database
mongo_collection.insert_one({"user": user_initials, "text": text, "model": model, "image_url": image_url, "gen_time": gen_time, "timestamp": time.time()})
except Exception as e:
print(e)
raise gr.Error("An error occurred while saving the prompt to the database.")
# Append the image URL and label to their respective lists
image_paths.append(image_url)
image_labels.append(image_label)
except Exception as e:
print(e)
raise gr.Error(f"An error occurred while generating the image for: {entry}")
return image_paths, image_labels # Return both image paths and labels
#timer stuff
timer_cancelled_global = False # when true, timer does not tick down
def countdown(seconds):
"""
This function takes the number of seconds as input and returns a string displaying the remaining time.
"""
target_time = datetime.now() + timedelta(seconds=int(seconds))
while target_time > datetime.now():
remaining_time = target_time - datetime.now()
remaining_seconds = int(remaining_time.total_seconds())
yield f"{remaining_seconds:02d}"
# Check if the countdown was cancelled
if timer_cancelled_global:
break
def stop_countdown():
"""
This function stops the countdown.
"""
global timer_cancelled_global
timer_cancelled_global = True
def reset_countdown(slider):
"""
This function resets the countdown.
"""
global timer_cancelled_global
timer_cancelled_global = False
return 60
#custom css
css = """
#gallery-images .caption-label {
white-space: normal !important;
}
"""
with gr.Blocks(css=css) as demo:
gr.Markdown("# <center>Prompt de Resistance Image Generator</center>")
pw = gr.Textbox(label="Password", type="password", placeholder="Enter the password to unlock the service")
#instructions
with gr.Accordion("Instructions & Tips",label="instructions",open=False):
with gr.Row():
gr.Markdown("**Instructions**: To use this service, please enter the password. Then generate an image from the prompt field below in response to the challenge, then click the download arrow from the top right of the image to save it.")
gr.Markdown("**Tips**: Use adjectives (size,color,mood), specify the visual style (realistic,cartoon,8-bit), explain the point of view (from above,first person,wide angle) ")
#challenge
challenge_display = gr.Textbox(label="Challenge", value=get_challenge())
challenge_display.disabled = True
regenerate_btn = gr.Button("New Challenge")
#countdown
with gr.Accordion("Countdown",label="Countdown",open=False):
with gr.Row():
with gr.Column(scale=3):
slider = gr.Slider(minimum=1, maximum=120, value=60,label="Countdown",info="Select duration in seconds")
with gr.Column(scale=1):
countdown_button = gr.Button("Start")
stop_countdown_button = gr.Button("Stop")
reset_countdown_button = gr.Button("Reset")
#prompts
with gr.Accordion("Prompts",label="Prompts",open=True):
with gr.Row():
with gr.Column(scale=3):
text = gr.Textbox(label="What do you want to create?", placeholder="Enter your text and then click on the \"Image Generate\" button")
with gr.Column(scale=1):
model = gr.Dropdown(choices=["dall-e-2", "dall-e-3"], label="Model", value="dall-e-3")
with gr.Row():
btn = gr.Button("Generate Images")
#output
with gr.Accordion("Image Outputs",label="Image Outputs",open=True):
output_images = gr.Gallery(label="Image Outputs", elem_id="gallery-images", show_label=True, columns=[3], rows=[1], object_fit="contain", height="auto", allow_preview=False)
show_labels = gr.Checkbox(label="Show Labels", value=False)
with gr.Accordion("Downloads",label="download",open=False):
download_all_btn = gr.Button("Download All")
download_link = gr.File(label="Download Zip")
# generate new challenge
regenerate_btn.click(fn=get_challenge, inputs=[], outputs=[challenge_display])
#countdown
countdown_button.click(fn=countdown, inputs=[slider], outputs=[slider])
stop_countdown_button.click(fn=stop_countdown)
reset_countdown_button.click(fn=reset_countdown,inputs=[slider],outputs=[slider])
#submissions
#trigger generation either through hitting enter in the text field, or clicking the button.
btn.click(fn=generate_images_wrapper, inputs=[text, pw, model, show_labels], outputs=output_images, api_name=False)
text.submit(fn=generate_images_wrapper, inputs=[text, pw, model, show_labels], outputs=output_images, api_name="generate_image") # Generate an api endpoint in Gradio / HF
show_labels.change(fn=update_labels, inputs=[show_labels], outputs=[output_images])
#downloads
download_all_btn.click(fn=download_all_images, inputs=[], outputs=download_link)
demo.launch(share=False) |