This repository has been archived by the owner on May 28, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathpredict.py
118 lines (99 loc) · 3.63 KB
/
predict.py
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
import cv2
import torch
import argparse
import numpy as np
from utils import visualize_prediction, crop_inference_bbox
from model import get_torchvision_maskrcnn
from transform import get_transforms
def to_float_tensor(img: np.array) -> torch.tensor:
# Converts numpy images to pytorch format
return torch.from_numpy(img.transpose(2, 0, 1)).float()
def get_prediction(
image: np.array,
model,
category_mapping: dict = {},
threshold: float = 0.5,
verbose: int = 1,
) -> (list, list, list):
# normalize image
image = image / np.max(image)
# convert to tensor
image = to_float_tensor(image).unsqueeze(0)
# get prediction
model.eval()
pred = model(image)
# map prediction ids to labels if category_mapping is given as input
def map_category(ind):
if not (category_mapping == {}):
return category_mapping[str(ind)]
else:
return str(ind)
# get predictions with above threshold prediction scores
pred_score = list(pred[0]["scores"].detach().numpy())
num_predictions_above_threshold = sum([1 for x in pred_score if x > threshold])
pred_num = num_predictions_above_threshold
masks, pred_boxes, pred_class = [], [], []
# process predictions if there are any
if pred_num > 0:
masks = (pred[0]["masks"] > 0.5).squeeze().detach().cpu().numpy()
pred_class = [map_category(i) for i in list(pred[0]["labels"].numpy())]
pred_boxes = [
[(int(i[0]), int(i[1])), (int(i[2]), int(i[3]))]
for i in list(pred[0]["boxes"].detach().numpy())
]
if len(masks.shape) == 3:
masks = masks[:pred_num]
elif len(masks.shape) == 2:
masks = np.expand_dims(masks, 0)
pred_boxes = pred_boxes[:pred_num]
pred_class = pred_class[:pred_num]
# print the number of detections
if verbose == 1:
print("There are {} detected instances.".format(pred_num))
return masks, pred_boxes, pred_class
def instance_segmentation_api(image_path: str, weight_path: str):
# read image
image = cv2.imread(image_path)
image = cv2.cvtColor(image, cv2.COLOR_BGR2RGB)
# load model dict
DEVICE = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
model_dict = torch.load(weight_path, map_location=DEVICE)
# load config from model dict
config = model_dict["config"]
# load model
model = get_torchvision_maskrcnn(
num_classes=config["NUM_CLASSES"],
trainable_backbone_layers=config["TRAINABLE_BACKBONE_LAYERS"],
anchor_sizes=config["RPN_ANCHOR_SIZES"],
anchor_aspect_ratios=config["RPN_ANCHOR_ASPECT_RATIOS"],
pretrained=False,
)
# load weights
model.load_state_dict(model_dict["state_dict"])
# get prediction
masks, boxes, pred_cls = get_prediction(
image, model, category_mapping=config["CATEGORY_MAPPING"], threshold=0.75
)
# visualize result
visualize_prediction(
image, masks, boxes, pred_cls, rect_th=3, text_size=3, text_th=3
)
# crop detected region
crop_inference_bbox(image, boxes)
if __name__ == "__main__":
# construct the argument parser
ap = argparse.ArgumentParser()
# add the arguments to the parser
ap.add_argument(
"image_path",
default="test/test_files/CA/CA01_01.tif",
help="Path for input image.",
)
ap.add_argument(
"weight_path",
default="artifacts/maskrcnn-best.pt",
help="Path for trained MaskRCNN model.",
)
args = vars(ap.parse_args())
# perform instance segmentation
instance_segmentation_api(args["image_path"], args["weight_path"])