import gradio as gr from transformers import pipeline import re import torch import timm from PIL import Image import numpy as np import base64 from io import BytesIO try: eva02_model = timm.create_model('hf_hub:SmilingWolf/wd-eva02-large-tagger-v3', pretrained=True) eva02_model.eval() import requests tags_response = requests.get('https://huggingface.co/SmilingWolf/wd-eva02-large-tagger-v3/resolve/main/selected_tags.csv') tags_lines = tags_response.text.strip().split('\n') eva02_tags = [line.split(',')[1] for line in tags_lines[1:]] print(f"Modelo EVA02 carregado com {len(eva02_tags)} tags") except Exception as e: print(f"Erro ao carregar EVA02: {e}") eva02_model = None eva02_tags = [] content_model = pipeline("image-classification", model="facebook/convnext-base-224") nsfw_model = pipeline("image-classification", model="Falconsai/nsfw_image_detection") def decode_base64_image(base64_string): if ',' in base64_string: base64_string = base64_string.split(',')[1] try: image_data = base64.b64decode(base64_string) image = Image.open(BytesIO(image_data)) return image except Exception as e: print(f"Erro ao decodificar base64: {e}") return None def analyze_with_eva02(image): if eva02_model is None: return [], [] image_tensor = torch.from_numpy(np.array(image)).permute(2, 0, 1).float() / 255.0 image_tensor = torch.nn.functional.interpolate( image_tensor.unsqueeze(0), size=(448, 448), mode='bilinear', align_corners=False ) with torch.no_grad(): features = eva02_model(image_tensor) probs = torch.sigmoid(features[0]) detected_tags = [] tag_scores = [] for i, (tag, prob) in enumerate(zip(eva02_tags, probs)): if prob > 0.5: detected_tags.append(tag) tag_scores.append(float(prob)) return detected_tags, tag_scores def analyze_image(image_input, base64_input=None): image = None if base64_input and isinstance(base64_input, str) and base64_input.strip(): image = decode_base64_image(base64_input) if image is None: return { "content": "unknown", "adult_content": False, "violence": False, "sensitive_content": False, "content_description": "Erro ao processar imagem base64." } elif image_input is not None: image = image_input else: return { "content": "unknown", "adult_content": False, "violence": False, "sensitive_content": False, "content_description": "Nenhuma imagem fornecida." } if isinstance(image, np.ndarray): image = Image.fromarray(image) if eva02_model is not None: eva02_tags_detected, eva02_scores = analyze_with_eva02(image) combined_labels = " ".join(eva02_tags_detected).lower() print(f"EVA02 detectou: {eva02_tags_detected}") else: content_preds = content_model(image) top_content = max(content_preds, key=lambda x: x["score"]) nsfw_preds = nsfw_model(image) top_nsfw = max(nsfw_preds, key=lambda x: x["score"]) all_labels = [] for pred in content_preds: all_labels.append(pred["label"].lower()) for pred in nsfw_preds: all_labels.append(pred["label"].lower()) combined_labels = " ".join(all_labels) human_keywords = [ "1boy", "1girl", "2boys", "2girls", "multiple boys", "multiple girls", "human", "person", "people", "man", "woman", "child", "baby", "boy", "girl", "face", "portrait", "selfie", "crowd", "family", "couple", "teenager", "male", "female", "adult", "teen", "kid", "toddler", "infant" ] dog_keywords = [ "1dog", "2dogs", "multiple dogs", "dog", "puppy", "retriever", "labrador", "golden", "beagle", "bulldog", "poodle", "german shepherd", "chihuahua", "terrier", "hound", "mastiff", "canine", "pet", "animal", "malamute", "malemute", "alaskan", "doggy", "doggie", "pup", "husky", "border collie", "dachshund", "boxer", "rottweiler", "siberian husky", "australian shepherd" ] human_count = sum(1 for keyword in human_keywords if keyword in combined_labels) dog_count = sum(1 for keyword in dog_keywords if keyword in combined_labels) if dog_count > 0 and human_count > 0: if dog_count >= human_count: is_dog = True is_human = False else: is_dog = False is_human = True elif dog_count > 0: is_dog = True is_human = False elif human_count > 0: is_dog = False is_human = True else: is_dog = False is_human = False violence_keywords = [ "blood", "wound", "injury", "hurt", "pain", "fight", "violence", "weapon", "knife", "gun", "attack", "aggression", "conflict", "battle", "war", "bloody", "injured", "wounded", "bleeding", "scar", "bruise", "cut" ] suffering_keywords = [ "sad", "crying", "tears", "depressed", "miserable", "suffering", "pain", "distress", "anguish", "grief", "mourning", "funeral", "death", "dead", "dying", "illness", "sick", "injured", "abandoned", "neglected", "crying", "tears", "sad", "depressed", "miserable", "grief", "mourning" ] abuse_keywords = [ "abuse", "mistreatment", "cruelty", "torture", "beaten", "chained", "caged", "starving", "malnourished", "neglected", "abandoned", "chained", "caged", "starving", "malnourished", "abused", "mistreated" ] death_keywords = [ "death", "dead", "dying", "corpse", "carcass", "deceased", "lifeless", "motionless", "still", "rigid", "pale", "cold", "skull", "bones", "grave", "tombstone", "funeral", "coffin", "burial" ] suspicious_keywords = [ "unconscious", "motionless", "lifeless", "rigid", "cold", "pale", "injured", "wounded", "bleeding", "hurt", "pain", "distress", "abandoned", "neglected", "starving", "malnourished", "chained", "caged" ] normal_dog_behavior = [ "sleeping", "resting", "lying", "sitting", "playing", "running", "walking", "happy", "excited", "alert", "awake", "active", "energetic", "playful" ] if eva02_model is not None: adult_keywords = ["nsfw", "explicit", "nude", "naked", "sexual", "adult", "mature"] adult_content = any(keyword in combined_labels for keyword in adult_keywords) else: adult_content = top_nsfw["label"].lower() == "nsfw" violence = any(keyword in combined_labels for keyword in violence_keywords) suffering = any(keyword in combined_labels for keyword in suffering_keywords) abuse = any(keyword in combined_labels for keyword in abuse_keywords) death = any(keyword in combined_labels for keyword in death_keywords) has_suspicious_behavior = any(keyword in combined_labels for keyword in suspicious_keywords) has_normal_behavior = any(keyword in combined_labels for keyword in normal_dog_behavior) has_death_indicators = any(keyword in combined_labels for keyword in death_keywords) if is_dog: suspicious_animal = has_death_indicators or (has_suspicious_behavior and not has_normal_behavior) dead_dog = has_death_indicators else: suspicious_animal = has_suspicious_behavior dead_dog = False sensitive = adult_content or violence or suffering or abuse or death or suspicious_animal or dead_dog if is_human: content_type = "human" elif is_dog: content_type = "dog" else: content_type = "unknown" description_parts = [] if content_type == "human": description_parts.append("Imagem contendo pessoa(s)") elif content_type == "dog": description_parts.append("Imagem contendo cão/cachorro") else: description_parts.append(f"Imagem contendo {label}") if adult_content: description_parts.append("com conteúdo adulto") if violence: description_parts.append("com violência") if suffering: description_parts.append("mostrando sofrimento") if abuse: description_parts.append("com maus tratos") if death: description_parts.append("mostrando morte") if suspicious_animal: description_parts.append("com características suspeitas") if dead_dog: description_parts.append("morto") if sensitive: description_parts.append("- CONTEÚDO SENSÍVEL") else: description_parts.append("- conteúdo seguro") description = " ".join(description_parts) + "." detected_human_tags = [tag for tag in human_keywords if tag in combined_labels] detected_dog_tags = [tag for tag in dog_keywords if tag in combined_labels] detected_suspicious = [tag for tag in suspicious_keywords if tag in combined_labels] detected_normal = [tag for tag in normal_dog_behavior if tag in combined_labels] detected_death = [tag for tag in death_keywords if tag in combined_labels] debug_info = f" [Debug: Human({human_count}): {detected_human_tags}, Dog({dog_count}): {detected_dog_tags}, Suspicious: {detected_suspicious}, Normal: {detected_normal}, Death: {detected_death}]" return { "content": content_type, "adult_content": adult_content, "violence": violence, "sensitive_content": sensitive, "content_description": description + debug_info } demo = gr.Interface( fn=analyze_image, inputs=[ gr.Image(type="pil", label="Upload de Imagem ou Cole Base64"), gr.Textbox(label="Ou Cole String Base64 Aqui", lines=3, placeholder="...") ], outputs="json", title="Dog / Human Safety Detector", description="Envie uma imagem ou cole uma string base64 para análise de moderação" ) if __name__ == "__main__": demo.launch()