aigis-lczs / backend.py
xinwei89
fixed gradio logic
79a62ec
"""
aerial-segmentation
Proof of concept showing effectiveness of a fine tuned instance segmentation model for detecting trees.
"""
import os
import gradio as gr
import cv2
os.system("pip install 'git+https://github.com/facebookresearch/detectron2.git'")
from transformers import DetrFeatureExtractor, DetrForSegmentation
from PIL import Image
import gradio as gr
import numpy as np
import torch
import torchvision
import detectron2
import json
# import some common detectron2 utilities
import itertools
import seaborn as sns
from detectron2 import model_zoo
from detectron2.engine import DefaultPredictor
from detectron2.config import get_cfg
from detectron2.utils.visualizer import Visualizer
from detectron2.utils.visualizer import ColorMode
from detectron2.data import MetadataCatalog, DatasetCatalog
from detectron2.checkpoint import DetectionCheckpointer
from detectron2.utils.visualizer import ColorMode
from detectron2.structures import Instances
def list_cfg_file_versions(directory):
files = os.listdir(directory)
# return files that contains substring version and end with .yml
cfg_files = [f.split("_")[0] for f in files if (f.endswith(".yml") or f.endswith(".yaml")) and f.startswith(f"{directory.split('_')[0]}v")]
return cfg_files
def list_pth_files_in_directory(directory, version="v1"):
files = os.listdir(directory)
version = version.split("v")[1]
# return files that contains substring version and end with .pth
pth_files = [f for f in files if version in f and f.endswith(".pth")]
return pth_files
def get_version_cfg_yml(path):
directory = path.split("/")[0]
version = path.split("/")[1]
files = os.listdir(directory)
cfg_file = [f for f in files if (f.endswith(".yml") or f.endswith(".yaml")) and version in f]
return directory + "/" + cfg_file[0]
def update_row_visibility(mode):
visibility = {
"tree": mode in ["Trees", "Trees & Buildings"],
"building": mode in ["Buildings", "Trees & Buildings"],
"lcz": mode in ["LCZ"]
}
tree_row, building_row, lcz_row = gr.Row(visible=visibility["tree"]), gr.Row(visible=visibility["building"]), gr.Row(visible=visibility["lcz"])
print(visibility)
return tree_row, building_row, lcz_row
def update_path_options(version):
if "tree" in version:
directory = "tree_model_weights"
elif "building" in version:
directory = "building_model_weights"
elif "lcz" in version:
directory = "lcz_model_weights"
return gr.Dropdown(choices=list_pth_files_in_directory(directory, version), label=f"Select a {version.split('v')[0]} model file", visible=True, interactive=True)
# Model for trees
def tree_model(tree_version_dropdown, tree_pth_dropdown, tree_threshold, device="cpu"):
tree_cfg = get_cfg()
tree_cfg.merge_from_file(get_version_cfg_yml(f"tree_model_weights/{tree_version_dropdown}"))
tree_cfg.MODEL.DEVICE=device
tree_cfg.MODEL.WEIGHTS = f"tree_model_weights/{tree_pth_dropdown}"
tree_cfg.MODEL.ROI_HEADS.NUM_CLASSES = 2 # TODO change this
tree_cfg.MODEL.ROI_HEADS.SCORE_THRESH_TEST = tree_threshold
tree_predictor = DefaultPredictor(tree_cfg)
return tree_predictor
# Model for buildings
def building_model(building_version_dropdown, building_pth_dropdown, building_threshold, device="cpu"):
building_cfg = get_cfg()
building_cfg.merge_from_file(get_version_cfg_yml(f"building_model_weights/{building_version_dropdown}"))
building_cfg.MODEL.DEVICE=device
building_cfg.MODEL.WEIGHTS = f"building_model_weights/{building_pth_dropdown}"
building_cfg.MODEL.ROI_HEADS.NUM_CLASSES = 8 # TODO change this
building_cfg.MODEL.ROI_HEADS.SCORE_THRESH_TEST = building_threshold
building_predictor = DefaultPredictor(building_cfg)
return building_predictor
# Model for LCZs
def lcz_model(lcz_version_dropdown, lcz_pth_dropdown, lcz_threshold, device="cpu"):
lcz_cfg = get_cfg()
lcz_cfg.merge_from_file(get_version_cfg_yml("lcz_model_weights/lczs_cfg.yaml"))
lcz_cfg.MODEL.DEVICE=device
lcz_cfg.MODEL.WEIGHTS = f"tree_model_weights/{lcz_pth_dropdown}"
lcz_cfg.MODEL.ROI_HEADS.NUM_CLASSES = 14 # TODO change this
lcz_cfg.MODEL.ROI_HEADS.SCORE_THRESH_TEST = lcz_threshold
lcz_predictor = DefaultPredictor(lcz_cfg)
return lcz_predictor
# A function that runs the buildings model on an given image and confidence threshold
def segment_building(im, building_predictor):
outputs = building_predictor(im)
building_instances = outputs["instances"].to("cpu")
return building_instances
# A function that runs the trees model on an given image and confidence threshold
def segment_tree(im, tree_predictor):
outputs = tree_predictor(im)
tree_instances = outputs["instances"].to("cpu")
return tree_instances
# A function that runs the trees model on an given image and confidence threshold
def segment_lcz(im, lcz_predictor):
outputs = lcz_predictor(im)
lcz_instances = outputs["instances"].to("cpu")
return lcz_instances
# Function to map strings to color mode
def map_color_mode(color_mode):
if color_mode == "Black/white":
return ColorMode.IMAGE_BW
elif color_mode == "Random":
return ColorMode.IMAGE
elif color_mode == "Segmentation" or color_mode == None:
return ColorMode.SEGMENTATION
def load_predictor(model, version, pth, threshold):
return model(version, pth, threshold)
def load_instances(image, predictor, segment_function):
return segment_function(image, predictor)
def combine_instances(tree_instances, building_instances):
return Instances.cat([tree_instances, building_instances])
def get_metadata(dataset_name, coco_file):
metadata = MetadataCatalog.get(dataset_name)
with open(coco_file, "r") as f:
coco = json.load(f)
categories = coco["categories"]
metadata.thing_classes = [c["name"] for c in categories]
return metadata
def visualize_image(im, mode, tree_threshold, building_threshold, color_mode, tree_version, tree_pth, building_version, building_pth, lcz_version, lcz_pth):
im = np.array(im)
color_mode = map_color_mode(color_mode)
instances = None
if mode in {"Trees", "Both"}:
tree_predictor = load_predictor(tree_model, tree_version, tree_pth, tree_threshold)
tree_instances = load_instances(im, tree_predictor, segment_tree)
instances = tree_instances
if mode in {"Buildings", "Both"}:
building_predictor = load_predictor(building_model, building_version, building_pth, building_threshold)
building_instances = load_instances(im, building_predictor, segment_building)
instances = building_instances if mode == "Buildings" else combine_instances(instances, building_instances)
if mode in {"LCZ", "Both"}:
lcz_predictor = load_predictor(lcz_model, lcz_version, lcz_pth, lcz_threshold)
lcz_instances = load_instances(im, lcz_predictor, segment_lcz)
instances = lcz_instances if mode == "LCZ" else combine_instances(instances, LCZ_instances)
# Assuming 'urban-small_train' is intended for both Trees and Buildings
metadata = get_metadata("urban-small_train", "building_model_weights/_annotations.coco.json")
visualizer = Visualizer(im[:, :, ::-1], metadata=metadata, scale=0.5, instance_mode=color_mode)
output_image = visualizer.draw_instance_predictions(instances)
return Image.fromarray(output_image.get_image()[:, :, ::-1])