Spaces:
Running
on
Zero
Running
on
Zero
File size: 3,300 Bytes
c9cc441 e7d7d62 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 c08d8a8 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 ce12cd7 c9cc441 |
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 |
import argparse
import csv
import os
import json
from PIL import Image
import cv2
import numpy as np
from tensorflow.keras.layers import TFSMLayer
from huggingface_hub import hf_hub_download
from pathlib import Path
import spaces
# 画像サイズの設定
IMAGE_SIZE = 448
# デフォルトのタグ付けリポジトリとファイル構成
DEFAULT_REPO = "SmilingWolf/wd-v1-4-convnext-tagger-v2"
MODEL_FILES = ["keras_metadata.pb", "saved_model.pb", "selected_tags.csv"]
VAR_DIR = "variables"
VAR_FILES = ["variables.data-00000-of-00001", "variables.index"]
CSV_FILE = MODEL_FILES[-1]
def preprocess_image(image):
"""画像を前処理して正方形に変換"""
img = np.array(image)[:, :, ::-1] # RGB->BGR
size = max(img.shape[:2])
pad_x, pad_y = size - img.shape[1], size - img.shape[0]
img = np.pad(img, ((pad_y // 2, pad_y - pad_y // 2), (pad_x // 2, pad_x - pad_x // 2), (0, 0)), mode="constant", constant_values=255)
interp = cv2.INTER_AREA if size > IMAGE_SIZE else cv2.INTER_LANCZOS4
img = cv2.resize(img, (IMAGE_SIZE, IMAGE_SIZE), interpolation=interp)
return img.astype(np.float32)
def download_model_files(repo_id, model_dir, sub_dir, files, sub_files):
"""モデルファイルをHugging Face Hubからダウンロード"""
for file in files:
hf_hub_download(repo_id, file, cache_dir=model_dir, force_download=True, force_filename=file)
for file in sub_files:
hf_hub_download(repo_id, file, subfolder=sub_dir, cache_dir=os.path.join(model_dir, sub_dir), force_download=True, force_filename=file)
@spaces.GPU
def load_wd14_tagger_model():
"""WD14タグ付けモデルをロード"""
model_dir = "wd14_tagger_model"
if not os.path.exists(model_dir):
download_model_files(DEFAULT_REPO, model_dir, VAR_DIR, MODEL_FILES, VAR_FILES)
else:
print("Using existing model")
return TFSMLayer(model_dir, call_endpoint='serving_default')
def read_tags_from_csv(csv_path):
"""CSVファイルからタグを読み取る"""
with open(csv_path, "r", encoding="utf-8") as f:
reader = csv.reader(f)
tags = [row for row in reader]
header = tags[0]
rows = tags[1:]
assert header[:3] == ["tag_id", "name", "category"], f"Unexpected CSV format: {header}"
return rows
def generate_tags(images, model_dir, model):
"""画像にタグを生成"""
rows = read_tags_from_csv(os.path.join(model_dir, CSV_FILE))
general_tags = [row[1] for row in rows if row[2] == "0"]
character_tags = [row[1] for row in rows if row[2] == "4"]
tag_freq = {}
undesired_tags = {'one-piece_swimsuit', 'swimsuit', 'leotard', 'saitama_(one-punch_man)', '1boy'}
probs = model(images, training=False)['predictions_sigmoid'].numpy()
tag_text_list = []
for prob in probs:
tags_combined = []
for i, p in enumerate(prob[4:]):
tag_list = general_tags if i < len(general_tags) else character_tags
tag = tag_list[i - len(general_tags)] if i >= len(general_tags) else tag_list[i]
if p >= 0.35 and tag not in undesired_tags:
tag_freq[tag] = tag_freq.get(tag, 0) + 1
tags_combined.append(tag)
tag_text_list.append(", ".join(tags_combined))
return tag_text_list
|