Spaces:
Runtime error
Runtime error
jharrison27
commited on
Commit
·
67f0a55
1
Parent(s):
fcb5f3a
Upload 13 files
Browse files- .gitattributes +1 -0
- app.py +21 -0
- class_names/bccd_classes.txt +3 -0
- class_names/coco_classes.txt +80 -0
- coco_classes.txt +80 -0
- config.py +17 -0
- custom_callbacks.py +15 -0
- custom_layers.py +298 -0
- loss.py +212 -0
- models.py +530 -0
- requirements.txt +96 -0
- utils.py +475 -0
- xml_to_txt.py +42 -0
- yolov4.weights +3 -0
.gitattributes
CHANGED
@@ -32,3 +32,4 @@ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
|
|
32 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
33 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
34 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
|
|
|
32 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
33 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
34 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
35 |
+
yolov4.weights filter=lfs diff=lfs merge=lfs -text
|
app.py
ADDED
@@ -0,0 +1,21 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import tensorflow as tf
|
2 |
+
import cv2
|
3 |
+
import numpy as np
|
4 |
+
from glob import glob
|
5 |
+
from models import Yolov4
|
6 |
+
import gradio as gr
|
7 |
+
model = Yolov4(weight_path="yolov4.weights", class_name_path='coco_classes.txt')
|
8 |
+
def gradio_wrapper(img):
|
9 |
+
global model
|
10 |
+
#print(np.shape(img))
|
11 |
+
results = model.predict(img)
|
12 |
+
return results[0]
|
13 |
+
demo = gr.Interface(
|
14 |
+
gradio_wrapper,
|
15 |
+
#gr.Image(source="webcam", streaming=True, flip=True),
|
16 |
+
gr.Image(source="webcam", streaming=True),
|
17 |
+
"image",
|
18 |
+
live=True
|
19 |
+
)
|
20 |
+
|
21 |
+
demo.launch()
|
class_names/bccd_classes.txt
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
WBC
|
2 |
+
Platelets
|
3 |
+
RBC
|
class_names/coco_classes.txt
ADDED
@@ -0,0 +1,80 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
person
|
2 |
+
bicycle
|
3 |
+
car
|
4 |
+
motorbike
|
5 |
+
aeroplane
|
6 |
+
bus
|
7 |
+
train
|
8 |
+
truck
|
9 |
+
boat
|
10 |
+
traffic light
|
11 |
+
fire hydrant
|
12 |
+
stop sign
|
13 |
+
parking meter
|
14 |
+
bench
|
15 |
+
bird
|
16 |
+
cat
|
17 |
+
dog
|
18 |
+
horse
|
19 |
+
sheep
|
20 |
+
cow
|
21 |
+
elephant
|
22 |
+
bear
|
23 |
+
zebra
|
24 |
+
giraffe
|
25 |
+
backpack
|
26 |
+
umbrella
|
27 |
+
handbag
|
28 |
+
tie
|
29 |
+
suitcase
|
30 |
+
frisbee
|
31 |
+
skis
|
32 |
+
snowboard
|
33 |
+
sports ball
|
34 |
+
kite
|
35 |
+
baseball bat
|
36 |
+
baseball glove
|
37 |
+
skateboard
|
38 |
+
surfboard
|
39 |
+
tennis racket
|
40 |
+
bottle
|
41 |
+
wine glass
|
42 |
+
cup
|
43 |
+
fork
|
44 |
+
knife
|
45 |
+
spoon
|
46 |
+
bowl
|
47 |
+
banana
|
48 |
+
apple
|
49 |
+
sandwich
|
50 |
+
orange
|
51 |
+
broccoli
|
52 |
+
carrot
|
53 |
+
hot dog
|
54 |
+
pizza
|
55 |
+
donut
|
56 |
+
cake
|
57 |
+
chair
|
58 |
+
sofa
|
59 |
+
pottedplant
|
60 |
+
bed
|
61 |
+
diningtable
|
62 |
+
toilet
|
63 |
+
tvmonitor
|
64 |
+
laptop
|
65 |
+
mouse
|
66 |
+
remote
|
67 |
+
keyboard
|
68 |
+
cell phone
|
69 |
+
microwave
|
70 |
+
oven
|
71 |
+
toaster
|
72 |
+
sink
|
73 |
+
refrigerator
|
74 |
+
book
|
75 |
+
clock
|
76 |
+
vase
|
77 |
+
scissors
|
78 |
+
teddy bear
|
79 |
+
hair drier
|
80 |
+
toothbrush
|
coco_classes.txt
ADDED
@@ -0,0 +1,80 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
person
|
2 |
+
bicycle
|
3 |
+
car
|
4 |
+
motorbike
|
5 |
+
aeroplane
|
6 |
+
bus
|
7 |
+
train
|
8 |
+
truck
|
9 |
+
boat
|
10 |
+
traffic light
|
11 |
+
fire hydrant
|
12 |
+
stop sign
|
13 |
+
parking meter
|
14 |
+
bench
|
15 |
+
bird
|
16 |
+
cat
|
17 |
+
dog
|
18 |
+
horse
|
19 |
+
sheep
|
20 |
+
cow
|
21 |
+
elephant
|
22 |
+
bear
|
23 |
+
zebra
|
24 |
+
giraffe
|
25 |
+
backpack
|
26 |
+
umbrella
|
27 |
+
handbag
|
28 |
+
tie
|
29 |
+
suitcase
|
30 |
+
frisbee
|
31 |
+
skis
|
32 |
+
snowboard
|
33 |
+
sports ball
|
34 |
+
kite
|
35 |
+
baseball bat
|
36 |
+
baseball glove
|
37 |
+
skateboard
|
38 |
+
surfboard
|
39 |
+
tennis racket
|
40 |
+
bottle
|
41 |
+
wine glass
|
42 |
+
cup
|
43 |
+
fork
|
44 |
+
knife
|
45 |
+
spoon
|
46 |
+
bowl
|
47 |
+
banana
|
48 |
+
apple
|
49 |
+
sandwich
|
50 |
+
orange
|
51 |
+
broccoli
|
52 |
+
carrot
|
53 |
+
hot dog
|
54 |
+
pizza
|
55 |
+
donut
|
56 |
+
cake
|
57 |
+
chair
|
58 |
+
sofa
|
59 |
+
pottedplant
|
60 |
+
bed
|
61 |
+
diningtable
|
62 |
+
toilet
|
63 |
+
tvmonitor
|
64 |
+
laptop
|
65 |
+
mouse
|
66 |
+
remote
|
67 |
+
keyboard
|
68 |
+
cell phone
|
69 |
+
microwave
|
70 |
+
oven
|
71 |
+
toaster
|
72 |
+
sink
|
73 |
+
refrigerator
|
74 |
+
book
|
75 |
+
clock
|
76 |
+
vase
|
77 |
+
scissors
|
78 |
+
teddy bear
|
79 |
+
hair drier
|
80 |
+
toothbrush
|
config.py
ADDED
@@ -0,0 +1,17 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
yolo_config = {
|
2 |
+
# Basic
|
3 |
+
'img_size': (416, 416, 3),
|
4 |
+
'anchors': [12, 16, 19, 36, 40, 28, 36, 75, 76, 55, 72, 146, 142, 110, 192, 243, 459, 401],
|
5 |
+
'strides': [8, 16, 32],
|
6 |
+
'xyscale': [1.2, 1.1, 1.05],
|
7 |
+
|
8 |
+
# Training
|
9 |
+
'iou_loss_thresh': 0.5,
|
10 |
+
'batch_size': 8,
|
11 |
+
'num_gpu': 1, # 2,
|
12 |
+
|
13 |
+
# Inference
|
14 |
+
'max_boxes': 100,
|
15 |
+
'iou_threshold': 0.413,
|
16 |
+
'score_threshold': 0.3,
|
17 |
+
}
|
custom_callbacks.py
ADDED
@@ -0,0 +1,15 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from tensorflow.keras import callbacks
|
2 |
+
import math
|
3 |
+
|
4 |
+
|
5 |
+
class CosineAnnealingScheduler(callbacks.LearningRateScheduler):
|
6 |
+
def __init__(self, epochs_per_cycle, lr_min, lr_max, verbose=0):
|
7 |
+
super(callbacks.LearningRateScheduler, self).__init__()
|
8 |
+
self.verbose = verbose
|
9 |
+
self.lr_min = lr_min
|
10 |
+
self.lr_max = lr_max
|
11 |
+
self.epochs_per_cycle = epochs_per_cycle
|
12 |
+
|
13 |
+
def schedule(self, epoch, lr):
|
14 |
+
return self.lr_min + (self.lr_max - self.lr_min) *\
|
15 |
+
(1 + math.cos(math.pi * (epoch % self.epochs_per_cycle) / self.epochs_per_cycle)) / 2
|
custom_layers.py
ADDED
@@ -0,0 +1,298 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import tensorflow as tf
|
2 |
+
from tensorflow.keras import layers, initializers, models
|
3 |
+
|
4 |
+
|
5 |
+
def conv(x, filters, kernel_size, downsampling=False, activation='leaky', batch_norm=True):
|
6 |
+
def mish(x):
|
7 |
+
return x * tf.math.tanh(tf.math.softplus(x))
|
8 |
+
|
9 |
+
if downsampling:
|
10 |
+
x = layers.ZeroPadding2D(padding=((1, 0), (1, 0)))(x) # top & left padding
|
11 |
+
padding = 'valid'
|
12 |
+
strides = 2
|
13 |
+
else:
|
14 |
+
padding = 'same'
|
15 |
+
strides = 1
|
16 |
+
x = layers.Conv2D(filters,
|
17 |
+
kernel_size,
|
18 |
+
strides=strides,
|
19 |
+
padding=padding,
|
20 |
+
use_bias=not batch_norm,
|
21 |
+
# kernel_regularizer=regularizers.l2(0.0005),
|
22 |
+
kernel_initializer=initializers.RandomNormal(mean=0.0, stddev=0.01),
|
23 |
+
# bias_initializer=initializers.Zeros()
|
24 |
+
)(x)
|
25 |
+
if batch_norm:
|
26 |
+
x = layers.BatchNormalization()(x)
|
27 |
+
if activation == 'mish':
|
28 |
+
x = mish(x)
|
29 |
+
elif activation == 'leaky':
|
30 |
+
x = layers.LeakyReLU(alpha=0.1)(x)
|
31 |
+
return x
|
32 |
+
|
33 |
+
|
34 |
+
def residual_block(x, filters1, filters2, activation='leaky'):
|
35 |
+
"""
|
36 |
+
:param x: input tensor
|
37 |
+
:param filters1: num of filter for 1x1 conv
|
38 |
+
:param filters2: num of filter for 3x3 conv
|
39 |
+
:param activation: default activation function: leaky relu
|
40 |
+
:return:
|
41 |
+
"""
|
42 |
+
y = conv(x, filters1, kernel_size=1, activation=activation)
|
43 |
+
y = conv(y, filters2, kernel_size=3, activation=activation)
|
44 |
+
return layers.Add()([x, y])
|
45 |
+
|
46 |
+
|
47 |
+
def csp_block(x, residual_out, repeat, residual_bottleneck=False):
|
48 |
+
"""
|
49 |
+
Cross Stage Partial Network (CSPNet)
|
50 |
+
transition_bottleneck_dims: 1x1 bottleneck
|
51 |
+
output_dims: 3x3
|
52 |
+
:param x:
|
53 |
+
:param residual_out:
|
54 |
+
:param repeat:
|
55 |
+
:param residual_bottleneck:
|
56 |
+
:return:
|
57 |
+
"""
|
58 |
+
route = x
|
59 |
+
route = conv(route, residual_out, 1, activation="mish")
|
60 |
+
x = conv(x, residual_out, 1, activation="mish")
|
61 |
+
for i in range(repeat):
|
62 |
+
x = residual_block(x,
|
63 |
+
residual_out // 2 if residual_bottleneck else residual_out,
|
64 |
+
residual_out,
|
65 |
+
activation="mish")
|
66 |
+
x = conv(x, residual_out, 1, activation="mish")
|
67 |
+
|
68 |
+
x = layers.Concatenate()([x, route])
|
69 |
+
return x
|
70 |
+
|
71 |
+
|
72 |
+
def darknet53(x):
|
73 |
+
x = conv(x, 32, 3)
|
74 |
+
x = conv(x, 64, 3, downsampling=True)
|
75 |
+
|
76 |
+
for i in range(1):
|
77 |
+
x = residual_block(x, 32, 64)
|
78 |
+
x = conv(x, 128, 3, downsampling=True)
|
79 |
+
|
80 |
+
for i in range(2):
|
81 |
+
x = residual_block(x, 64, 128)
|
82 |
+
x = conv(x, 256, 3, downsampling=True)
|
83 |
+
|
84 |
+
for i in range(8):
|
85 |
+
x = residual_block(x, 128, 256)
|
86 |
+
route_1 = x
|
87 |
+
x = conv(x, 512, 3, downsampling=True)
|
88 |
+
|
89 |
+
for i in range(8):
|
90 |
+
x = residual_block(x, 256, 512)
|
91 |
+
route_2 = x
|
92 |
+
x = conv(x, 1024, 3, downsampling=True)
|
93 |
+
|
94 |
+
for i in range(4):
|
95 |
+
x = residual_block(x, 512, 1024)
|
96 |
+
|
97 |
+
return route_1, route_2, x
|
98 |
+
|
99 |
+
|
100 |
+
def cspdarknet53(input):
|
101 |
+
x = conv(input, 32, 3)
|
102 |
+
x = conv(x, 64, 3, downsampling=True)
|
103 |
+
|
104 |
+
x = csp_block(x, residual_out=64, repeat=1, residual_bottleneck=True)
|
105 |
+
x = conv(x, 64, 1, activation='mish')
|
106 |
+
x = conv(x, 128, 3, activation='mish', downsampling=True)
|
107 |
+
|
108 |
+
x = csp_block(x, residual_out=64, repeat=2)
|
109 |
+
x = conv(x, 128, 1, activation='mish')
|
110 |
+
x = conv(x, 256, 3, activation='mish', downsampling=True)
|
111 |
+
|
112 |
+
x = csp_block(x, residual_out=128, repeat=8)
|
113 |
+
x = conv(x, 256, 1, activation='mish')
|
114 |
+
route0 = x
|
115 |
+
x = conv(x, 512, 3, activation='mish', downsampling=True)
|
116 |
+
|
117 |
+
x = csp_block(x, residual_out=256, repeat=8)
|
118 |
+
x = conv(x, 512, 1, activation='mish')
|
119 |
+
route1 = x
|
120 |
+
x = conv(x, 1024, 3, activation='mish', downsampling=True)
|
121 |
+
|
122 |
+
x = csp_block(x, residual_out=512, repeat=4)
|
123 |
+
|
124 |
+
x = conv(x, 1024, 1, activation="mish")
|
125 |
+
|
126 |
+
x = conv(x, 512, 1)
|
127 |
+
x = conv(x, 1024, 3)
|
128 |
+
x = conv(x, 512, 1)
|
129 |
+
|
130 |
+
x = layers.Concatenate()([layers.MaxPooling2D(pool_size=13, strides=1, padding='same')(x),
|
131 |
+
layers.MaxPooling2D(pool_size=9, strides=1, padding='same')(x),
|
132 |
+
layers.MaxPooling2D(pool_size=5, strides=1, padding='same')(x),
|
133 |
+
x
|
134 |
+
])
|
135 |
+
x = conv(x, 512, 1)
|
136 |
+
x = conv(x, 1024, 3)
|
137 |
+
route2 = conv(x, 512, 1)
|
138 |
+
return models.Model(input, [route0, route1, route2])
|
139 |
+
|
140 |
+
|
141 |
+
def yolov4_neck(x, num_classes):
|
142 |
+
backbone_model = cspdarknet53(x)
|
143 |
+
route0, route1, route2 = backbone_model.output
|
144 |
+
|
145 |
+
route_input = route2
|
146 |
+
x = conv(route2, 256, 1)
|
147 |
+
x = layers.UpSampling2D()(x)
|
148 |
+
route1 = conv(route1, 256, 1)
|
149 |
+
x = layers.Concatenate()([route1, x])
|
150 |
+
|
151 |
+
x = conv(x, 256, 1)
|
152 |
+
x = conv(x, 512, 3)
|
153 |
+
x = conv(x, 256, 1)
|
154 |
+
x = conv(x, 512, 3)
|
155 |
+
x = conv(x, 256, 1)
|
156 |
+
|
157 |
+
route1 = x
|
158 |
+
x = conv(x, 128, 1)
|
159 |
+
x = layers.UpSampling2D()(x)
|
160 |
+
route0 = conv(route0, 128, 1)
|
161 |
+
x = layers.Concatenate()([route0, x])
|
162 |
+
|
163 |
+
x = conv(x, 128, 1)
|
164 |
+
x = conv(x, 256, 3)
|
165 |
+
x = conv(x, 128, 1)
|
166 |
+
x = conv(x, 256, 3)
|
167 |
+
x = conv(x, 128, 1)
|
168 |
+
|
169 |
+
route0 = x
|
170 |
+
x = conv(x, 256, 3)
|
171 |
+
conv_sbbox = conv(x, 3 * (num_classes + 5), 1, activation=None, batch_norm=False)
|
172 |
+
|
173 |
+
x = conv(route0, 256, 3, downsampling=True)
|
174 |
+
x = layers.Concatenate()([x, route1])
|
175 |
+
|
176 |
+
x = conv(x, 256, 1)
|
177 |
+
x = conv(x, 512, 3)
|
178 |
+
x = conv(x, 256, 1)
|
179 |
+
x = conv(x, 512, 3)
|
180 |
+
x = conv(x, 256, 1)
|
181 |
+
|
182 |
+
route1 = x
|
183 |
+
x = conv(x, 512, 3)
|
184 |
+
conv_mbbox = conv(x, 3 * (num_classes + 5), 1, activation=None, batch_norm=False)
|
185 |
+
|
186 |
+
x = conv(route1, 512, 3, downsampling=True)
|
187 |
+
x = layers.Concatenate()([x, route_input])
|
188 |
+
|
189 |
+
x = conv(x, 512, 1)
|
190 |
+
x = conv(x, 1024, 3)
|
191 |
+
x = conv(x, 512, 1)
|
192 |
+
x = conv(x, 1024, 3)
|
193 |
+
x = conv(x, 512, 1)
|
194 |
+
|
195 |
+
x = conv(x, 1024, 3)
|
196 |
+
conv_lbbox = conv(x, 3 * (num_classes + 5), 1, activation=None, batch_norm=False)
|
197 |
+
|
198 |
+
return [conv_sbbox, conv_mbbox, conv_lbbox]
|
199 |
+
|
200 |
+
|
201 |
+
def yolov4_head(yolo_neck_outputs, classes, anchors, xyscale):
|
202 |
+
bbox0, object_probability0, class_probabilities0, pred_box0 = get_boxes(yolo_neck_outputs[0],
|
203 |
+
anchors=anchors[0, :, :], classes=classes,
|
204 |
+
grid_size=52, strides=8,
|
205 |
+
xyscale=xyscale[0])
|
206 |
+
bbox1, object_probability1, class_probabilities1, pred_box1 = get_boxes(yolo_neck_outputs[1],
|
207 |
+
anchors=anchors[1, :, :], classes=classes,
|
208 |
+
grid_size=26, strides=16,
|
209 |
+
xyscale=xyscale[1])
|
210 |
+
bbox2, object_probability2, class_probabilities2, pred_box2 = get_boxes(yolo_neck_outputs[2],
|
211 |
+
anchors=anchors[2, :, :], classes=classes,
|
212 |
+
grid_size=13, strides=32,
|
213 |
+
xyscale=xyscale[2])
|
214 |
+
x = [bbox0, object_probability0, class_probabilities0, pred_box0,
|
215 |
+
bbox1, object_probability1, class_probabilities1, pred_box1,
|
216 |
+
bbox2, object_probability2, class_probabilities2, pred_box2]
|
217 |
+
|
218 |
+
return x
|
219 |
+
|
220 |
+
|
221 |
+
def get_boxes(pred, anchors, classes, grid_size, strides, xyscale):
|
222 |
+
"""
|
223 |
+
|
224 |
+
:param pred:
|
225 |
+
:param anchors:
|
226 |
+
:param classes:
|
227 |
+
:param grid_size:
|
228 |
+
:param strides:
|
229 |
+
:param xyscale:
|
230 |
+
:return:
|
231 |
+
"""
|
232 |
+
pred = tf.reshape(pred,
|
233 |
+
(tf.shape(pred)[0],
|
234 |
+
grid_size,
|
235 |
+
grid_size,
|
236 |
+
3,
|
237 |
+
5 + classes)) # (batch_size, grid_size, grid_size, 3, 5+classes)
|
238 |
+
box_xy, box_wh, obj_prob, class_prob = tf.split(
|
239 |
+
pred, (2, 2, 1, classes), axis=-1
|
240 |
+
) # (?, 52, 52, 3, 2) (?, 52, 52, 3, 2) (?, 52, 52, 3, 1) (?, 52, 52, 3, 80)
|
241 |
+
|
242 |
+
box_xy = tf.sigmoid(box_xy) # (?, 52, 52, 3, 2)
|
243 |
+
obj_prob = tf.sigmoid(obj_prob) # (?, 52, 52, 3, 1)
|
244 |
+
class_prob = tf.sigmoid(class_prob) # (?, 52, 52, 3, 80)
|
245 |
+
pred_box_xywh = tf.concat((box_xy, box_wh), axis=-1) # (?, 52, 52, 3, 4)
|
246 |
+
|
247 |
+
grid = tf.meshgrid(tf.range(grid_size), tf.range(grid_size)) # (52, 52) (52, 52)
|
248 |
+
grid = tf.expand_dims(tf.stack(grid, axis=-1), axis=2) # (52, 52, 1, 2)
|
249 |
+
grid = tf.cast(grid, dtype=tf.float32)
|
250 |
+
|
251 |
+
box_xy = ((box_xy * xyscale) - 0.5 * (xyscale - 1) + grid) * strides # (?, 52, 52, 1, 4)
|
252 |
+
|
253 |
+
box_wh = tf.exp(box_wh) * anchors # (?, 52, 52, 3, 2)
|
254 |
+
box_x1y1 = box_xy - box_wh / 2 # (?, 52, 52, 3, 2)
|
255 |
+
box_x2y2 = box_xy + box_wh / 2 # (?, 52, 52, 3, 2)
|
256 |
+
pred_box_x1y1x2y2 = tf.concat([box_x1y1, box_x2y2], axis=-1) # (?, 52, 52, 3, 4)
|
257 |
+
return pred_box_x1y1x2y2, obj_prob, class_prob, pred_box_xywh
|
258 |
+
# pred_box_x1y1x2y2: absolute xy value
|
259 |
+
|
260 |
+
|
261 |
+
def nms(model_ouputs, input_shape, num_class, iou_threshold=0.413, score_threshold=0.3):
|
262 |
+
"""
|
263 |
+
Apply Non-Maximum suppression
|
264 |
+
ref: https://www.tensorflow.org/api_docs/python/tf/image/combined_non_max_suppression
|
265 |
+
:param model_ouputs: yolo model model_ouputs
|
266 |
+
:param input_shape: size of input image
|
267 |
+
:return: nmsed_boxes, nmsed_scores, nmsed_classes, valid_detections
|
268 |
+
"""
|
269 |
+
bs = tf.shape(model_ouputs[0])[0]
|
270 |
+
boxes = tf.zeros((bs, 0, 4))
|
271 |
+
confidence = tf.zeros((bs, 0, 1))
|
272 |
+
class_probabilities = tf.zeros((bs, 0, num_class))
|
273 |
+
|
274 |
+
for output_idx in range(0, len(model_ouputs), 4):
|
275 |
+
output_xy = model_ouputs[output_idx]
|
276 |
+
output_conf = model_ouputs[output_idx + 1]
|
277 |
+
output_classes = model_ouputs[output_idx + 2]
|
278 |
+
boxes = tf.concat([boxes, tf.reshape(output_xy, (bs, -1, 4))], axis=1)
|
279 |
+
confidence = tf.concat([confidence, tf.reshape(output_conf, (bs, -1, 1))], axis=1)
|
280 |
+
class_probabilities = tf.concat([class_probabilities, tf.reshape(output_classes, (bs, -1, num_class))], axis=1)
|
281 |
+
|
282 |
+
scores = confidence * class_probabilities
|
283 |
+
boxes = tf.expand_dims(boxes, axis=-2)
|
284 |
+
boxes = boxes / input_shape[0] # box normalization: relative img size
|
285 |
+
print(f'nms iou: {iou_threshold} score: {score_threshold}')
|
286 |
+
(nmsed_boxes, # [bs, max_detections, 4]
|
287 |
+
nmsed_scores, # [bs, max_detections]
|
288 |
+
nmsed_classes, # [bs, max_detections]
|
289 |
+
valid_detections # [batch_size]
|
290 |
+
) = tf.image.combined_non_max_suppression(
|
291 |
+
boxes=boxes, # y1x1, y2x2 [0~1]
|
292 |
+
scores=scores,
|
293 |
+
max_output_size_per_class=100,
|
294 |
+
max_total_size=100, # max_boxes: Maximum nmsed_boxes in a single img.
|
295 |
+
iou_threshold=iou_threshold, # iou_threshold: Minimum overlap that counts as a valid detection.
|
296 |
+
score_threshold=score_threshold, # # Minimum confidence that counts as a valid detection.
|
297 |
+
)
|
298 |
+
return nmsed_boxes, nmsed_scores, nmsed_classes, valid_detections
|
loss.py
ADDED
@@ -0,0 +1,212 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
#!/usr/bin/env python
|
2 |
+
# coding: utf-8
|
3 |
+
|
4 |
+
import numpy as np
|
5 |
+
import math
|
6 |
+
import tensorflow.keras.backend as K
|
7 |
+
import tensorflow as tf
|
8 |
+
|
9 |
+
|
10 |
+
def xywh_to_x1y1x2y2(boxes):
|
11 |
+
return tf.concat([boxes[..., :2] - boxes[..., 2:] * 0.5, boxes[..., :2] + boxes[..., 2:] * 0.5], axis=-1)
|
12 |
+
|
13 |
+
|
14 |
+
# x,y,w,h
|
15 |
+
def bbox_iou(boxes1, boxes2):
|
16 |
+
boxes1_area = boxes1[..., 2] * boxes1[..., 3] # w * h
|
17 |
+
boxes2_area = boxes2[..., 2] * boxes2[..., 3]
|
18 |
+
|
19 |
+
# (x, y, w, h) -> (x0, y0, x1, y1)
|
20 |
+
boxes1 = xywh_to_x1y1x2y2(boxes1)
|
21 |
+
boxes2 = xywh_to_x1y1x2y2(boxes2)
|
22 |
+
|
23 |
+
# coordinates of intersection
|
24 |
+
top_left = tf.maximum(boxes1[..., :2], boxes2[..., :2])
|
25 |
+
bottom_right = tf.minimum(boxes1[..., 2:], boxes2[..., 2:])
|
26 |
+
intersection_xy = tf.maximum(bottom_right - top_left, 0.0)
|
27 |
+
|
28 |
+
intersection_area = intersection_xy[..., 0] * intersection_xy[..., 1]
|
29 |
+
union_area = boxes1_area + boxes2_area - intersection_area
|
30 |
+
|
31 |
+
return 1.0 * intersection_area / (union_area + tf.keras.backend.epsilon())
|
32 |
+
|
33 |
+
|
34 |
+
def bbox_giou(boxes1, boxes2):
|
35 |
+
boxes1_area = boxes1[..., 2] * boxes1[..., 3] # w*h
|
36 |
+
boxes2_area = boxes2[..., 2] * boxes2[..., 3]
|
37 |
+
|
38 |
+
# (x, y, w, h) -> (x0, y0, x1, y1)
|
39 |
+
boxes1 = xywh_to_x1y1x2y2(boxes1)
|
40 |
+
boxes2 = xywh_to_x1y1x2y2(boxes2)
|
41 |
+
|
42 |
+
top_left = tf.maximum(boxes1[..., :2], boxes2[..., :2])
|
43 |
+
bottom_right = tf.minimum(boxes1[..., 2:], boxes2[..., 2:])
|
44 |
+
|
45 |
+
intersection_xy = tf.maximum(bottom_right - top_left, 0.0)
|
46 |
+
intersection_area = intersection_xy[..., 0] * intersection_xy[..., 1]
|
47 |
+
|
48 |
+
union_area = boxes1_area + boxes2_area - intersection_area
|
49 |
+
|
50 |
+
iou = 1.0 * intersection_area / (union_area + tf.keras.backend.epsilon())
|
51 |
+
|
52 |
+
enclose_top_left = tf.minimum(boxes1[..., :2], boxes2[..., :2])
|
53 |
+
enclose_bottom_right = tf.maximum(boxes1[..., 2:], boxes2[..., 2:])
|
54 |
+
|
55 |
+
enclose_xy = enclose_bottom_right - enclose_top_left
|
56 |
+
enclose_area = enclose_xy[..., 0] * enclose_xy[..., 1]
|
57 |
+
|
58 |
+
giou = iou - tf.math.divide_no_nan(enclose_area - union_area, enclose_area)
|
59 |
+
|
60 |
+
return giou
|
61 |
+
|
62 |
+
|
63 |
+
def bbox_ciou(boxes1, boxes2):
|
64 |
+
'''
|
65 |
+
ciou = iou - p2/c2 - av
|
66 |
+
:param boxes1: (8, 13, 13, 3, 4) pred_xywh
|
67 |
+
:param boxes2: (8, 13, 13, 3, 4) label_xywh
|
68 |
+
:return:
|
69 |
+
'''
|
70 |
+
boxes1_x0y0x1y1 = tf.concat([boxes1[..., :2] - boxes1[..., 2:] * 0.5,
|
71 |
+
boxes1[..., :2] + boxes1[..., 2:] * 0.5], axis=-1)
|
72 |
+
boxes2_x0y0x1y1 = tf.concat([boxes2[..., :2] - boxes2[..., 2:] * 0.5,
|
73 |
+
boxes2[..., :2] + boxes2[..., 2:] * 0.5], axis=-1)
|
74 |
+
boxes1_x0y0x1y1 = tf.concat([tf.minimum(boxes1_x0y0x1y1[..., :2], boxes1_x0y0x1y1[..., 2:]),
|
75 |
+
tf.maximum(boxes1_x0y0x1y1[..., :2], boxes1_x0y0x1y1[..., 2:])], axis=-1)
|
76 |
+
boxes2_x0y0x1y1 = tf.concat([tf.minimum(boxes2_x0y0x1y1[..., :2], boxes2_x0y0x1y1[..., 2:]),
|
77 |
+
tf.maximum(boxes2_x0y0x1y1[..., :2], boxes2_x0y0x1y1[..., 2:])], axis=-1)
|
78 |
+
|
79 |
+
# area
|
80 |
+
boxes1_area = (boxes1_x0y0x1y1[..., 2] - boxes1_x0y0x1y1[..., 0]) * (
|
81 |
+
boxes1_x0y0x1y1[..., 3] - boxes1_x0y0x1y1[..., 1])
|
82 |
+
boxes2_area = (boxes2_x0y0x1y1[..., 2] - boxes2_x0y0x1y1[..., 0]) * (
|
83 |
+
boxes2_x0y0x1y1[..., 3] - boxes2_x0y0x1y1[..., 1])
|
84 |
+
|
85 |
+
# top-left and bottom-right coord, shape: (8, 13, 13, 3, 2)
|
86 |
+
left_up = tf.maximum(boxes1_x0y0x1y1[..., :2], boxes2_x0y0x1y1[..., :2])
|
87 |
+
right_down = tf.minimum(boxes1_x0y0x1y1[..., 2:], boxes2_x0y0x1y1[..., 2:])
|
88 |
+
|
89 |
+
# intersection area and iou
|
90 |
+
inter_section = tf.maximum(right_down - left_up, 0.0)
|
91 |
+
inter_area = inter_section[..., 0] * inter_section[..., 1]
|
92 |
+
union_area = boxes1_area + boxes2_area - inter_area
|
93 |
+
iou = inter_area / (union_area + 1e-9)
|
94 |
+
|
95 |
+
# top-left and bottom-right coord of the enclosing rectangle, shape: (8, 13, 13, 3, 2)
|
96 |
+
enclose_left_up = tf.minimum(boxes1_x0y0x1y1[..., :2], boxes2_x0y0x1y1[..., :2])
|
97 |
+
enclose_right_down = tf.maximum(boxes1_x0y0x1y1[..., 2:], boxes2_x0y0x1y1[..., 2:])
|
98 |
+
|
99 |
+
# diagnal ** 2
|
100 |
+
enclose_wh = enclose_right_down - enclose_left_up
|
101 |
+
enclose_c2 = K.pow(enclose_wh[..., 0], 2) + K.pow(enclose_wh[..., 1], 2)
|
102 |
+
|
103 |
+
# center distances between two rectangles
|
104 |
+
p2 = K.pow(boxes1[..., 0] - boxes2[..., 0], 2) + K.pow(boxes1[..., 1] - boxes2[..., 1], 2)
|
105 |
+
|
106 |
+
# add av
|
107 |
+
atan1 = tf.atan(boxes1[..., 2] / (boxes1[..., 3] + 1e-9))
|
108 |
+
atan2 = tf.atan(boxes2[..., 2] / (boxes2[..., 3] + 1e-9))
|
109 |
+
v = 4.0 * K.pow(atan1 - atan2, 2) / (math.pi ** 2)
|
110 |
+
a = v / (1 - iou + v)
|
111 |
+
|
112 |
+
ciou = iou - 1.0 * p2 / enclose_c2 - 1.0 * a * v
|
113 |
+
return ciou
|
114 |
+
|
115 |
+
|
116 |
+
def yolo_loss(args, num_classes, iou_loss_thresh, anchors):
|
117 |
+
conv_lbbox = args[2] # (?, ?, ?, 3*(num_classes+5))
|
118 |
+
conv_mbbox = args[1] # (?, ?, ?, 3*(num_classes+5))
|
119 |
+
conv_sbbox = args[0] # (?, ?, ?, 3*(num_classes+5))
|
120 |
+
label_sbbox = args[3] # (?, ?, ?, 3, num_classes+5)
|
121 |
+
label_mbbox = args[4] # (?, ?, ?, 3, num_classes+5)
|
122 |
+
label_lbbox = args[5] # (?, ?, ?, 3, num_classes+5)
|
123 |
+
true_bboxes = args[6] # (?, 50, 4)
|
124 |
+
pred_sbbox = decode(conv_sbbox, anchors[0], 8, num_classes)
|
125 |
+
pred_mbbox = decode(conv_mbbox, anchors[1], 16, num_classes)
|
126 |
+
pred_lbbox = decode(conv_lbbox, anchors[2], 32, num_classes)
|
127 |
+
sbbox_ciou_loss, sbbox_conf_loss, sbbox_prob_loss = loss_layer(conv_sbbox, pred_sbbox, label_sbbox, true_bboxes, 8, num_classes, iou_loss_thresh)
|
128 |
+
mbbox_ciou_loss, mbbox_conf_loss, mbbox_prob_loss = loss_layer(conv_mbbox, pred_mbbox, label_mbbox, true_bboxes, 16, num_classes, iou_loss_thresh)
|
129 |
+
lbbox_ciou_loss, lbbox_conf_loss, lbbox_prob_loss = loss_layer(conv_lbbox, pred_lbbox, label_lbbox, true_bboxes, 32, num_classes, iou_loss_thresh)
|
130 |
+
|
131 |
+
ciou_loss = (lbbox_ciou_loss + sbbox_ciou_loss + mbbox_ciou_loss) * 3.54
|
132 |
+
conf_loss = (lbbox_conf_loss + sbbox_conf_loss + mbbox_conf_loss) * 64.3
|
133 |
+
prob_loss = (lbbox_prob_loss + sbbox_prob_loss + mbbox_prob_loss) * 1
|
134 |
+
|
135 |
+
return ciou_loss+conf_loss+prob_loss
|
136 |
+
|
137 |
+
|
138 |
+
def loss_layer(conv, pred, label, bboxes, stride, num_class, iou_loss_thresh):
|
139 |
+
conv_shape = tf.shape(conv)
|
140 |
+
batch_size = conv_shape[0]
|
141 |
+
output_size = conv_shape[1]
|
142 |
+
input_size = stride * output_size
|
143 |
+
conv = tf.reshape(conv, (batch_size, output_size, output_size,
|
144 |
+
3, 5 + num_class))
|
145 |
+
conv_raw_prob = conv[:, :, :, :, 5:]
|
146 |
+
conv_raw_conf = conv[:, :, :, :, 4:5]
|
147 |
+
|
148 |
+
pred_xywh = pred[:, :, :, :, 0:4]
|
149 |
+
pred_conf = pred[:, :, :, :, 4:5]
|
150 |
+
|
151 |
+
label_xywh = label[:, :, :, :, 0:4]
|
152 |
+
respond_bbox = label[:, :, :, :, 4:5]
|
153 |
+
label_prob = label[:, :, :, :, 5:]
|
154 |
+
|
155 |
+
# Coordinate loss
|
156 |
+
ciou = tf.expand_dims(bbox_giou(pred_xywh, label_xywh), axis=-1) # (8, 13, 13, 3, 1)
|
157 |
+
# ciou = tf.expand_dims(bbox_ciou(pred_xywh, label_xywh), axis=-1) # (8, 13, 13, 3, 1)
|
158 |
+
input_size = tf.cast(input_size, tf.float32)
|
159 |
+
|
160 |
+
# loss weight of the gt bbox: 2-(gt area/img area)
|
161 |
+
bbox_loss_scale = 2.0 - 1.0 * label_xywh[:, :, :, :, 2:3] * label_xywh[:, :, :, :, 3:4] / (input_size ** 2)
|
162 |
+
ciou_loss = respond_bbox * bbox_loss_scale * (1 - ciou) # iou loss for respond bbox
|
163 |
+
|
164 |
+
# Classification loss for respond bbox
|
165 |
+
prob_loss = respond_bbox * tf.nn.sigmoid_cross_entropy_with_logits(labels=label_prob, logits=conv_raw_prob)
|
166 |
+
|
167 |
+
expand_pred_xywh = pred_xywh[:, :, :, :, np.newaxis, :] # (?, grid_h, grid_w, 3, 1, 4)
|
168 |
+
expand_bboxes = bboxes[:, np.newaxis, np.newaxis, np.newaxis, :, :] # (?, 1, 1, 1, 70, 4)
|
169 |
+
iou = bbox_iou(expand_pred_xywh, expand_bboxes) # IoU between all pred bbox and all gt (?, grid_h, grid_w, 3, 70)
|
170 |
+
max_iou = tf.expand_dims(tf.reduce_max(iou, axis=-1), axis=-1) # max iou: (?, grid_h, grid_w, 3, 1)
|
171 |
+
|
172 |
+
# ignore the bbox which is not respond bbox and max iou < threshold
|
173 |
+
respond_bgd = (1.0 - respond_bbox) * tf.cast(max_iou < iou_loss_thresh, tf.float32)
|
174 |
+
|
175 |
+
# Confidence loss
|
176 |
+
conf_focal = tf.pow(respond_bbox - pred_conf, 2)
|
177 |
+
|
178 |
+
conf_loss = conf_focal * (
|
179 |
+
respond_bbox * tf.nn.sigmoid_cross_entropy_with_logits(labels=respond_bbox, logits=conv_raw_conf)
|
180 |
+
+
|
181 |
+
respond_bgd * tf.nn.sigmoid_cross_entropy_with_logits(labels=respond_bbox, logits=conv_raw_conf)
|
182 |
+
)
|
183 |
+
|
184 |
+
ciou_loss = tf.reduce_mean(tf.reduce_sum(ciou_loss, axis=[1, 2, 3, 4]))
|
185 |
+
conf_loss = tf.reduce_mean(tf.reduce_sum(conf_loss, axis=[1, 2, 3, 4]))
|
186 |
+
prob_loss = tf.reduce_mean(tf.reduce_sum(prob_loss, axis=[1, 2, 3, 4]))
|
187 |
+
|
188 |
+
return ciou_loss, conf_loss, prob_loss
|
189 |
+
|
190 |
+
|
191 |
+
def decode(conv_output, anchors, stride, num_class):
|
192 |
+
conv_shape = tf.shape(conv_output)
|
193 |
+
batch_size = conv_shape[0]
|
194 |
+
output_size = conv_shape[1]
|
195 |
+
anchor_per_scale = len(anchors)
|
196 |
+
conv_output = tf.reshape(conv_output, (batch_size, output_size, output_size, anchor_per_scale, 5 + num_class))
|
197 |
+
conv_raw_dxdy = conv_output[:, :, :, :, 0:2]
|
198 |
+
conv_raw_dwdh = conv_output[:, :, :, :, 2:4]
|
199 |
+
conv_raw_conf = conv_output[:, :, :, :, 4:5]
|
200 |
+
conv_raw_prob = conv_output[:, :, :, :, 5:]
|
201 |
+
y = tf.tile(tf.range(output_size, dtype=tf.int32)[:, tf.newaxis], [1, output_size])
|
202 |
+
x = tf.tile(tf.range(output_size, dtype=tf.int32)[tf.newaxis, :], [output_size, 1])
|
203 |
+
xy_grid = tf.concat([x[:, :, tf.newaxis], y[:, :, tf.newaxis]], axis=-1)
|
204 |
+
xy_grid = tf.tile(xy_grid[tf.newaxis, :, :, tf.newaxis, :], [batch_size, 1, 1, anchor_per_scale, 1])
|
205 |
+
xy_grid = tf.cast(xy_grid, tf.float32)
|
206 |
+
pred_xy = (tf.sigmoid(conv_raw_dxdy) + xy_grid) * stride
|
207 |
+
pred_wh = (tf.exp(conv_raw_dwdh) * anchors)
|
208 |
+
pred_xywh = tf.concat([pred_xy, pred_wh], axis=-1)
|
209 |
+
pred_conf = tf.sigmoid(conv_raw_conf)
|
210 |
+
pred_prob = tf.sigmoid(conv_raw_prob)
|
211 |
+
return tf.concat([pred_xywh, pred_conf, pred_prob], axis=-1)
|
212 |
+
|
models.py
ADDED
@@ -0,0 +1,530 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import numpy as np
|
2 |
+
import cv2
|
3 |
+
import os
|
4 |
+
import json
|
5 |
+
from tqdm import tqdm
|
6 |
+
from glob import glob
|
7 |
+
import matplotlib.pyplot as plt
|
8 |
+
import tensorflow as tf
|
9 |
+
from tensorflow.keras import layers, models, optimizers
|
10 |
+
|
11 |
+
from custom_layers import yolov4_neck, yolov4_head, nms
|
12 |
+
from utils import load_weights, get_detection_data, draw_bbox, voc_ap, draw_plot_func, read_txt_to_list
|
13 |
+
from config import yolo_config
|
14 |
+
from loss import yolo_loss
|
15 |
+
|
16 |
+
|
17 |
+
class Yolov4(object):
|
18 |
+
def __init__(self,
|
19 |
+
weight_path=None,
|
20 |
+
class_name_path='coco_classes.txt',
|
21 |
+
config=yolo_config,
|
22 |
+
):
|
23 |
+
assert config['img_size'][0] == config['img_size'][1], 'not support yet'
|
24 |
+
assert config['img_size'][0] % config['strides'][-1] == 0, 'must be a multiple of last stride'
|
25 |
+
self.class_names = [line.strip() for line in open(class_name_path).readlines()]
|
26 |
+
self.img_size = yolo_config['img_size']
|
27 |
+
self.num_classes = len(self.class_names)
|
28 |
+
self.weight_path = weight_path
|
29 |
+
self.anchors = np.array(yolo_config['anchors']).reshape((3, 3, 2))
|
30 |
+
self.xyscale = yolo_config['xyscale']
|
31 |
+
self.strides = yolo_config['strides']
|
32 |
+
self.output_sizes = [self.img_size[0] // s for s in self.strides]
|
33 |
+
self.class_color = {name: list(np.random.random(size=3)*255) for name in self.class_names}
|
34 |
+
# Training
|
35 |
+
self.max_boxes = yolo_config['max_boxes']
|
36 |
+
self.iou_loss_thresh = yolo_config['iou_loss_thresh']
|
37 |
+
self.config = yolo_config
|
38 |
+
assert self.num_classes > 0, 'no classes detected!'
|
39 |
+
|
40 |
+
tf.keras.backend.clear_session()
|
41 |
+
if yolo_config['num_gpu'] > 1:
|
42 |
+
mirrored_strategy = tf.distribute.MirroredStrategy()
|
43 |
+
with mirrored_strategy.scope():
|
44 |
+
self.build_model(load_pretrained=True if self.weight_path else False)
|
45 |
+
else:
|
46 |
+
self.build_model(load_pretrained=True if self.weight_path else False)
|
47 |
+
|
48 |
+
def build_model(self, load_pretrained=True):
|
49 |
+
# core yolo model
|
50 |
+
input_layer = layers.Input(self.img_size)
|
51 |
+
yolov4_output = yolov4_neck(input_layer, self.num_classes)
|
52 |
+
self.yolo_model = models.Model(input_layer, yolov4_output)
|
53 |
+
|
54 |
+
# Build training model
|
55 |
+
y_true = [
|
56 |
+
layers.Input(name='input_2', shape=(52, 52, 3, (self.num_classes + 5))), # label small boxes
|
57 |
+
layers.Input(name='input_3', shape=(26, 26, 3, (self.num_classes + 5))), # label medium boxes
|
58 |
+
layers.Input(name='input_4', shape=(13, 13, 3, (self.num_classes + 5))), # label large boxes
|
59 |
+
layers.Input(name='input_5', shape=(self.max_boxes, 4)), # true bboxes
|
60 |
+
]
|
61 |
+
loss_list = tf.keras.layers.Lambda(yolo_loss, name='yolo_loss',
|
62 |
+
arguments={'num_classes': self.num_classes,
|
63 |
+
'iou_loss_thresh': self.iou_loss_thresh,
|
64 |
+
'anchors': self.anchors})([*self.yolo_model.output, *y_true])
|
65 |
+
self.training_model = models.Model([self.yolo_model.input, *y_true], loss_list)
|
66 |
+
|
67 |
+
# Build inference model
|
68 |
+
yolov4_output = yolov4_head(yolov4_output, self.num_classes, self.anchors, self.xyscale)
|
69 |
+
# output: [boxes, scores, classes, valid_detections]
|
70 |
+
self.inference_model = models.Model(input_layer,
|
71 |
+
nms(yolov4_output, self.img_size, self.num_classes,
|
72 |
+
iou_threshold=self.config['iou_threshold'],
|
73 |
+
score_threshold=self.config['score_threshold']))
|
74 |
+
|
75 |
+
if load_pretrained and self.weight_path and self.weight_path.endswith('.weights'):
|
76 |
+
if self.weight_path.endswith('.weights'):
|
77 |
+
load_weights(self.yolo_model, self.weight_path)
|
78 |
+
print(f'load from {self.weight_path}')
|
79 |
+
elif self.weight_path.endswith('.h5'):
|
80 |
+
self.training_model.load_weights(self.weight_path)
|
81 |
+
print(f'load from {self.weight_path}')
|
82 |
+
|
83 |
+
self.training_model.compile(optimizer=optimizers.Adam(lr=1e-3),
|
84 |
+
loss={'yolo_loss': lambda y_true, y_pred: y_pred})
|
85 |
+
|
86 |
+
def load_model(self, path):
|
87 |
+
self.yolo_model = models.load_model(path, compile=False)
|
88 |
+
yolov4_output = yolov4_head(self.yolo_model.output, self.num_classes, self.anchors, self.xyscale)
|
89 |
+
self.inference_model = models.Model(self.yolo_model.input,
|
90 |
+
nms(yolov4_output, self.img_size, self.num_classes)) # [boxes, scores, classes, valid_detections]
|
91 |
+
|
92 |
+
def save_model(self, path):
|
93 |
+
self.yolo_model.save(path)
|
94 |
+
|
95 |
+
def preprocess_img(self, img):
|
96 |
+
img = cv2.resize(img, self.img_size[:2])
|
97 |
+
img = img / 255.
|
98 |
+
return img
|
99 |
+
|
100 |
+
def fit(self, train_data_gen, epochs, val_data_gen=None, initial_epoch=0, callbacks=None):
|
101 |
+
self.training_model.fit(train_data_gen,
|
102 |
+
steps_per_epoch=len(train_data_gen),
|
103 |
+
validation_data=val_data_gen,
|
104 |
+
validation_steps=len(val_data_gen),
|
105 |
+
epochs=epochs,
|
106 |
+
callbacks=callbacks,
|
107 |
+
initial_epoch=initial_epoch)
|
108 |
+
# raw_img: RGB
|
109 |
+
def predict_img(self, raw_img, random_color=True, plot_img=True, figsize=(10, 10), show_text=True, return_output=True):
|
110 |
+
print('img shape: ', raw_img.shape)
|
111 |
+
img = self.preprocess_img(raw_img)
|
112 |
+
imgs = np.expand_dims(img, axis=0)
|
113 |
+
pred_output = self.inference_model.predict(imgs)
|
114 |
+
detections = get_detection_data(img=raw_img,
|
115 |
+
model_outputs=pred_output,
|
116 |
+
class_names=self.class_names)
|
117 |
+
|
118 |
+
output_img = draw_bbox(raw_img, detections, cmap=self.class_color, random_color=random_color, figsize=figsize,
|
119 |
+
show_text=show_text, show_img=False)
|
120 |
+
if return_output:
|
121 |
+
return output_img, detections
|
122 |
+
else:
|
123 |
+
return detections
|
124 |
+
|
125 |
+
def predict(self, img_path, random_color=True, plot_img=True, figsize=(10, 10), show_text=True):
|
126 |
+
raw_img = img_path
|
127 |
+
return self.predict_img(raw_img, random_color, plot_img, figsize, show_text)
|
128 |
+
|
129 |
+
def export_gt(self, annotation_path, gt_folder_path):
|
130 |
+
with open(annotation_path) as file:
|
131 |
+
for line in file:
|
132 |
+
line = line.split(' ')
|
133 |
+
filename = line[0].split(os.sep)[-1].split('.')[0]
|
134 |
+
objs = line[1:]
|
135 |
+
# export txt file
|
136 |
+
with open(os.path.join(gt_folder_path, filename + '.txt'), 'w') as output_file:
|
137 |
+
for obj in objs:
|
138 |
+
x_min, y_min, x_max, y_max, class_id = [float(o) for o in obj.strip().split(',')]
|
139 |
+
output_file.write(f'{self.class_names[int(class_id)]} {x_min} {y_min} {x_max} {y_max}\n')
|
140 |
+
|
141 |
+
def export_prediction(self, annotation_path, pred_folder_path, img_folder_path, bs=2):
|
142 |
+
with open(annotation_path) as file:
|
143 |
+
img_paths = [os.path.join(img_folder_path, line.split(' ')[0].split(os.sep)[-1]) for line in file]
|
144 |
+
# print(img_paths[:20])
|
145 |
+
for batch_idx in tqdm(range(0, len(img_paths), bs)):
|
146 |
+
# print(len(img_paths), batch_idx, batch_idx*bs, (batch_idx+1)*bs)
|
147 |
+
paths = img_paths[batch_idx:batch_idx+bs]
|
148 |
+
# print(paths)
|
149 |
+
# read and process img
|
150 |
+
imgs = np.zeros((len(paths), *self.img_size))
|
151 |
+
raw_img_shapes = []
|
152 |
+
for j, path in enumerate(paths):
|
153 |
+
img = cv2.imread(path)
|
154 |
+
raw_img_shapes.append(img.shape)
|
155 |
+
img = self.preprocess_img(img)
|
156 |
+
imgs[j] = img
|
157 |
+
|
158 |
+
# process batch output
|
159 |
+
b_boxes, b_scores, b_classes, b_valid_detections = self.inference_model.predict(imgs)
|
160 |
+
for k in range(len(paths)):
|
161 |
+
num_boxes = b_valid_detections[k]
|
162 |
+
raw_img_shape = raw_img_shapes[k]
|
163 |
+
boxes = b_boxes[k, :num_boxes]
|
164 |
+
classes = b_classes[k, :num_boxes]
|
165 |
+
scores = b_scores[k, :num_boxes]
|
166 |
+
# print(raw_img_shape)
|
167 |
+
boxes[:, [0, 2]] = (boxes[:, [0, 2]] * raw_img_shape[1]) # w
|
168 |
+
boxes[:, [1, 3]] = (boxes[:, [1, 3]] * raw_img_shape[0]) # h
|
169 |
+
cls_names = [self.class_names[int(c)] for c in classes]
|
170 |
+
# print(raw_img_shape, boxes.astype(int), cls_names, scores)
|
171 |
+
|
172 |
+
img_path = paths[k]
|
173 |
+
filename = img_path.split(os.sep)[-1].split('.')[0]
|
174 |
+
# print(filename)
|
175 |
+
output_path = os.path.join(pred_folder_path, filename+'.txt')
|
176 |
+
with open(output_path, 'w') as pred_file:
|
177 |
+
for box_idx in range(num_boxes):
|
178 |
+
b = boxes[box_idx]
|
179 |
+
pred_file.write(f'{cls_names[box_idx]} {scores[box_idx]} {b[0]} {b[1]} {b[2]} {b[3]}\n')
|
180 |
+
|
181 |
+
|
182 |
+
def eval_map(self, gt_folder_path, pred_folder_path, temp_json_folder_path, output_files_path):
|
183 |
+
"""Process Gt"""
|
184 |
+
ground_truth_files_list = glob(gt_folder_path + '/*.txt')
|
185 |
+
assert len(ground_truth_files_list) > 0, 'no ground truth file'
|
186 |
+
ground_truth_files_list.sort()
|
187 |
+
# dictionary with counter per class
|
188 |
+
gt_counter_per_class = {}
|
189 |
+
counter_images_per_class = {}
|
190 |
+
|
191 |
+
gt_files = []
|
192 |
+
for txt_file in ground_truth_files_list:
|
193 |
+
file_id = txt_file.split(".txt", 1)[0]
|
194 |
+
file_id = os.path.basename(os.path.normpath(file_id))
|
195 |
+
# check if there is a correspondent detection-results file
|
196 |
+
temp_path = os.path.join(pred_folder_path, (file_id + ".txt"))
|
197 |
+
assert os.path.exists(temp_path), "Error. File not found: {}\n".format(temp_path)
|
198 |
+
lines_list = read_txt_to_list(txt_file)
|
199 |
+
# create ground-truth dictionary
|
200 |
+
bounding_boxes = []
|
201 |
+
is_difficult = False
|
202 |
+
already_seen_classes = []
|
203 |
+
for line in lines_list:
|
204 |
+
class_name, left, top, right, bottom = line.split()
|
205 |
+
# check if class is in the ignore list, if yes skip
|
206 |
+
bbox = left + " " + top + " " + right + " " + bottom
|
207 |
+
bounding_boxes.append({"class_name": class_name, "bbox": bbox, "used": False})
|
208 |
+
# count that object
|
209 |
+
if class_name in gt_counter_per_class:
|
210 |
+
gt_counter_per_class[class_name] += 1
|
211 |
+
else:
|
212 |
+
# if class didn't exist yet
|
213 |
+
gt_counter_per_class[class_name] = 1
|
214 |
+
|
215 |
+
if class_name not in already_seen_classes:
|
216 |
+
if class_name in counter_images_per_class:
|
217 |
+
counter_images_per_class[class_name] += 1
|
218 |
+
else:
|
219 |
+
# if class didn't exist yet
|
220 |
+
counter_images_per_class[class_name] = 1
|
221 |
+
already_seen_classes.append(class_name)
|
222 |
+
|
223 |
+
# dump bounding_boxes into a ".json" file
|
224 |
+
new_temp_file = os.path.join(temp_json_folder_path, file_id+"_ground_truth.json") #TEMP_FILES_PATH + "/" + file_id + "_ground_truth.json"
|
225 |
+
gt_files.append(new_temp_file)
|
226 |
+
with open(new_temp_file, 'w') as outfile:
|
227 |
+
json.dump(bounding_boxes, outfile)
|
228 |
+
|
229 |
+
gt_classes = list(gt_counter_per_class.keys())
|
230 |
+
# let's sort the classes alphabetically
|
231 |
+
gt_classes = sorted(gt_classes)
|
232 |
+
n_classes = len(gt_classes)
|
233 |
+
print(gt_classes, gt_counter_per_class)
|
234 |
+
|
235 |
+
"""Process prediction"""
|
236 |
+
|
237 |
+
dr_files_list = sorted(glob(os.path.join(pred_folder_path, '*.txt')))
|
238 |
+
|
239 |
+
for class_index, class_name in enumerate(gt_classes):
|
240 |
+
bounding_boxes = []
|
241 |
+
for txt_file in dr_files_list:
|
242 |
+
# the first time it checks if all the corresponding ground-truth files exist
|
243 |
+
file_id = txt_file.split(".txt", 1)[0]
|
244 |
+
file_id = os.path.basename(os.path.normpath(file_id))
|
245 |
+
temp_path = os.path.join(gt_folder_path, (file_id + ".txt"))
|
246 |
+
if class_index == 0:
|
247 |
+
if not os.path.exists(temp_path):
|
248 |
+
error_msg = f"Error. File not found: {temp_path}\n"
|
249 |
+
print(error_msg)
|
250 |
+
lines = read_txt_to_list(txt_file)
|
251 |
+
for line in lines:
|
252 |
+
try:
|
253 |
+
tmp_class_name, confidence, left, top, right, bottom = line.split()
|
254 |
+
except ValueError:
|
255 |
+
error_msg = f"""Error: File {txt_file} in the wrong format.\n
|
256 |
+
Expected: <class_name> <confidence> <left> <top> <right> <bottom>\n
|
257 |
+
Received: {line} \n"""
|
258 |
+
print(error_msg)
|
259 |
+
if tmp_class_name == class_name:
|
260 |
+
# print("match")
|
261 |
+
bbox = left + " " + top + " " + right + " " + bottom
|
262 |
+
bounding_boxes.append({"confidence": confidence, "file_id": file_id, "bbox": bbox})
|
263 |
+
# sort detection-results by decreasing confidence
|
264 |
+
bounding_boxes.sort(key=lambda x: float(x['confidence']), reverse=True)
|
265 |
+
with open(temp_json_folder_path + "/" + class_name + "_dr.json", 'w') as outfile:
|
266 |
+
json.dump(bounding_boxes, outfile)
|
267 |
+
|
268 |
+
"""
|
269 |
+
Calculate the AP for each class
|
270 |
+
"""
|
271 |
+
sum_AP = 0.0
|
272 |
+
ap_dictionary = {}
|
273 |
+
# open file to store the output
|
274 |
+
with open(output_files_path + "/output.txt", 'w') as output_file:
|
275 |
+
output_file.write("# AP and precision/recall per class\n")
|
276 |
+
count_true_positives = {}
|
277 |
+
for class_index, class_name in enumerate(gt_classes):
|
278 |
+
count_true_positives[class_name] = 0
|
279 |
+
"""
|
280 |
+
Load detection-results of that class
|
281 |
+
"""
|
282 |
+
dr_file = temp_json_folder_path + "/" + class_name + "_dr.json"
|
283 |
+
dr_data = json.load(open(dr_file))
|
284 |
+
|
285 |
+
"""
|
286 |
+
Assign detection-results to ground-truth objects
|
287 |
+
"""
|
288 |
+
nd = len(dr_data)
|
289 |
+
tp = [0] * nd # creates an array of zeros of size nd
|
290 |
+
fp = [0] * nd
|
291 |
+
for idx, detection in enumerate(dr_data):
|
292 |
+
file_id = detection["file_id"]
|
293 |
+
gt_file = temp_json_folder_path + "/" + file_id + "_ground_truth.json"
|
294 |
+
ground_truth_data = json.load(open(gt_file))
|
295 |
+
ovmax = -1
|
296 |
+
gt_match = -1
|
297 |
+
# load detected object bounding-box
|
298 |
+
bb = [float(x) for x in detection["bbox"].split()]
|
299 |
+
for obj in ground_truth_data:
|
300 |
+
# look for a class_name match
|
301 |
+
if obj["class_name"] == class_name:
|
302 |
+
bbgt = [float(x) for x in obj["bbox"].split()]
|
303 |
+
bi = [max(bb[0], bbgt[0]), max(bb[1], bbgt[1]), min(bb[2], bbgt[2]), min(bb[3], bbgt[3])]
|
304 |
+
iw = bi[2] - bi[0] + 1
|
305 |
+
ih = bi[3] - bi[1] + 1
|
306 |
+
if iw > 0 and ih > 0:
|
307 |
+
# compute overlap (IoU) = area of intersection / area of union
|
308 |
+
ua = (bb[2] - bb[0] + 1) * (bb[3] - bb[1] + 1) + \
|
309 |
+
(bbgt[2] - bbgt[0]+ 1) * (bbgt[3] - bbgt[1] + 1) - iw * ih
|
310 |
+
ov = iw * ih / ua
|
311 |
+
if ov > ovmax:
|
312 |
+
ovmax = ov
|
313 |
+
gt_match = obj
|
314 |
+
|
315 |
+
min_overlap = 0.5
|
316 |
+
if ovmax >= min_overlap:
|
317 |
+
# if "difficult" not in gt_match:
|
318 |
+
if not bool(gt_match["used"]):
|
319 |
+
# true positive
|
320 |
+
tp[idx] = 1
|
321 |
+
gt_match["used"] = True
|
322 |
+
count_true_positives[class_name] += 1
|
323 |
+
# update the ".json" file
|
324 |
+
with open(gt_file, 'w') as f:
|
325 |
+
f.write(json.dumps(ground_truth_data))
|
326 |
+
else:
|
327 |
+
# false positive (multiple detection)
|
328 |
+
fp[idx] = 1
|
329 |
+
else:
|
330 |
+
fp[idx] = 1
|
331 |
+
|
332 |
+
|
333 |
+
# compute precision/recall
|
334 |
+
cumsum = 0
|
335 |
+
for idx, val in enumerate(fp):
|
336 |
+
fp[idx] += cumsum
|
337 |
+
cumsum += val
|
338 |
+
print('fp ', cumsum)
|
339 |
+
cumsum = 0
|
340 |
+
for idx, val in enumerate(tp):
|
341 |
+
tp[idx] += cumsum
|
342 |
+
cumsum += val
|
343 |
+
print('tp ', cumsum)
|
344 |
+
rec = tp[:]
|
345 |
+
for idx, val in enumerate(tp):
|
346 |
+
rec[idx] = float(tp[idx]) / gt_counter_per_class[class_name]
|
347 |
+
print('recall ', cumsum)
|
348 |
+
prec = tp[:]
|
349 |
+
for idx, val in enumerate(tp):
|
350 |
+
prec[idx] = float(tp[idx]) / (fp[idx] + tp[idx])
|
351 |
+
print('prec ', cumsum)
|
352 |
+
|
353 |
+
ap, mrec, mprec = voc_ap(rec[:], prec[:])
|
354 |
+
sum_AP += ap
|
355 |
+
text = "{0:.2f}%".format(
|
356 |
+
ap * 100) + " = " + class_name + " AP " # class_name + " AP = {0:.2f}%".format(ap*100)
|
357 |
+
|
358 |
+
print(text)
|
359 |
+
ap_dictionary[class_name] = ap
|
360 |
+
|
361 |
+
n_images = counter_images_per_class[class_name]
|
362 |
+
# lamr, mr, fppi = log_average_miss_rate(np.array(prec), np.array(rec), n_images)
|
363 |
+
# lamr_dictionary[class_name] = lamr
|
364 |
+
|
365 |
+
"""
|
366 |
+
Draw plot
|
367 |
+
"""
|
368 |
+
if True:
|
369 |
+
plt.plot(rec, prec, '-o')
|
370 |
+
# add a new penultimate point to the list (mrec[-2], 0.0)
|
371 |
+
# since the last line segment (and respective area) do not affect the AP value
|
372 |
+
area_under_curve_x = mrec[:-1] + [mrec[-2]] + [mrec[-1]]
|
373 |
+
area_under_curve_y = mprec[:-1] + [0.0] + [mprec[-1]]
|
374 |
+
plt.fill_between(area_under_curve_x, 0, area_under_curve_y, alpha=0.2, edgecolor='r')
|
375 |
+
# set window title
|
376 |
+
fig = plt.gcf() # gcf - get current figure
|
377 |
+
fig.canvas.set_window_title('AP ' + class_name)
|
378 |
+
# set plot title
|
379 |
+
plt.title('class: ' + text)
|
380 |
+
# plt.suptitle('This is a somewhat long figure title', fontsize=16)
|
381 |
+
# set axis titles
|
382 |
+
plt.xlabel('Recall')
|
383 |
+
plt.ylabel('Precision')
|
384 |
+
# optional - set axes
|
385 |
+
axes = plt.gca() # gca - get current axes
|
386 |
+
axes.set_xlim([0.0, 1.0])
|
387 |
+
axes.set_ylim([0.0, 1.05]) # .05 to give some extra space
|
388 |
+
# Alternative option -> wait for button to be pressed
|
389 |
+
# while not plt.waitforbuttonpress(): pass # wait for key display
|
390 |
+
# Alternative option -> normal display
|
391 |
+
plt.show()
|
392 |
+
# save the plot
|
393 |
+
# fig.savefig(output_files_path + "/classes/" + class_name + ".png")
|
394 |
+
# plt.cla() # clear axes for next plot
|
395 |
+
|
396 |
+
# if show_animation:
|
397 |
+
# cv2.destroyAllWindows()
|
398 |
+
|
399 |
+
output_file.write("\n# mAP of all classes\n")
|
400 |
+
mAP = sum_AP / n_classes
|
401 |
+
text = "mAP = {0:.2f}%".format(mAP * 100)
|
402 |
+
output_file.write(text + "\n")
|
403 |
+
print(text)
|
404 |
+
|
405 |
+
"""
|
406 |
+
Count total of detection-results
|
407 |
+
"""
|
408 |
+
# iterate through all the files
|
409 |
+
det_counter_per_class = {}
|
410 |
+
for txt_file in dr_files_list:
|
411 |
+
# get lines to list
|
412 |
+
lines_list = read_txt_to_list(txt_file)
|
413 |
+
for line in lines_list:
|
414 |
+
class_name = line.split()[0]
|
415 |
+
# check if class is in the ignore list, if yes skip
|
416 |
+
# if class_name in args.ignore:
|
417 |
+
# continue
|
418 |
+
# count that object
|
419 |
+
if class_name in det_counter_per_class:
|
420 |
+
det_counter_per_class[class_name] += 1
|
421 |
+
else:
|
422 |
+
# if class didn't exist yet
|
423 |
+
det_counter_per_class[class_name] = 1
|
424 |
+
# print(det_counter_per_class)
|
425 |
+
dr_classes = list(det_counter_per_class.keys())
|
426 |
+
|
427 |
+
"""
|
428 |
+
Plot the total number of occurences of each class in the ground-truth
|
429 |
+
"""
|
430 |
+
if True:
|
431 |
+
window_title = "ground-truth-info"
|
432 |
+
plot_title = "ground-truth\n"
|
433 |
+
plot_title += "(" + str(len(ground_truth_files_list)) + " files and " + str(n_classes) + " classes)"
|
434 |
+
x_label = "Number of objects per class"
|
435 |
+
output_path = output_files_path + "/ground-truth-info.png"
|
436 |
+
to_show = False
|
437 |
+
plot_color = 'forestgreen'
|
438 |
+
draw_plot_func(
|
439 |
+
gt_counter_per_class,
|
440 |
+
n_classes,
|
441 |
+
window_title,
|
442 |
+
plot_title,
|
443 |
+
x_label,
|
444 |
+
output_path,
|
445 |
+
to_show,
|
446 |
+
plot_color,
|
447 |
+
'',
|
448 |
+
)
|
449 |
+
|
450 |
+
"""
|
451 |
+
Finish counting true positives
|
452 |
+
"""
|
453 |
+
for class_name in dr_classes:
|
454 |
+
# if class exists in detection-result but not in ground-truth then there are no true positives in that class
|
455 |
+
if class_name not in gt_classes:
|
456 |
+
count_true_positives[class_name] = 0
|
457 |
+
# print(count_true_positives)
|
458 |
+
|
459 |
+
"""
|
460 |
+
Plot the total number of occurences of each class in the "detection-results" folder
|
461 |
+
"""
|
462 |
+
if True:
|
463 |
+
window_title = "detection-results-info"
|
464 |
+
# Plot title
|
465 |
+
plot_title = "detection-results\n"
|
466 |
+
plot_title += "(" + str(len(dr_files_list)) + " files and "
|
467 |
+
count_non_zero_values_in_dictionary = sum(int(x) > 0 for x in list(det_counter_per_class.values()))
|
468 |
+
plot_title += str(count_non_zero_values_in_dictionary) + " detected classes)"
|
469 |
+
# end Plot title
|
470 |
+
x_label = "Number of objects per class"
|
471 |
+
output_path = output_files_path + "/detection-results-info.png"
|
472 |
+
to_show = False
|
473 |
+
plot_color = 'forestgreen'
|
474 |
+
true_p_bar = count_true_positives
|
475 |
+
draw_plot_func(
|
476 |
+
det_counter_per_class,
|
477 |
+
len(det_counter_per_class),
|
478 |
+
window_title,
|
479 |
+
plot_title,
|
480 |
+
x_label,
|
481 |
+
output_path,
|
482 |
+
to_show,
|
483 |
+
plot_color,
|
484 |
+
true_p_bar
|
485 |
+
)
|
486 |
+
|
487 |
+
"""
|
488 |
+
Draw mAP plot (Show AP's of all classes in decreasing order)
|
489 |
+
"""
|
490 |
+
if True:
|
491 |
+
window_title = "mAP"
|
492 |
+
plot_title = "mAP = {0:.2f}%".format(mAP * 100)
|
493 |
+
x_label = "Average Precision"
|
494 |
+
output_path = output_files_path + "/mAP.png"
|
495 |
+
to_show = True
|
496 |
+
plot_color = 'royalblue'
|
497 |
+
draw_plot_func(
|
498 |
+
ap_dictionary,
|
499 |
+
n_classes,
|
500 |
+
window_title,
|
501 |
+
plot_title,
|
502 |
+
x_label,
|
503 |
+
output_path,
|
504 |
+
to_show,
|
505 |
+
plot_color,
|
506 |
+
""
|
507 |
+
)
|
508 |
+
|
509 |
+
def predict_raw(self, img_path):
|
510 |
+
raw_img = cv2.imread(img_path)
|
511 |
+
print('img shape: ', raw_img.shape)
|
512 |
+
img = self.preprocess_img(raw_img)
|
513 |
+
imgs = np.expand_dims(img, axis=0)
|
514 |
+
return self.yolo_model.predict(imgs)
|
515 |
+
|
516 |
+
def predict_nonms(self, img_path, iou_threshold=0.413, score_threshold=0.1):
|
517 |
+
raw_img = cv2.imread(img_path)
|
518 |
+
print('img shape: ', raw_img.shape)
|
519 |
+
img = self.preprocess_img(raw_img)
|
520 |
+
imgs = np.expand_dims(img, axis=0)
|
521 |
+
yolov4_output = self.yolo_model.predict(imgs)
|
522 |
+
output = yolov4_head(yolov4_output, self.num_classes, self.anchors, self.xyscale)
|
523 |
+
pred_output = nms(output, self.img_size, self.num_classes, iou_threshold, score_threshold)
|
524 |
+
pred_output = [p.numpy() for p in pred_output]
|
525 |
+
detections = get_detection_data(img=raw_img,
|
526 |
+
model_outputs=pred_output,
|
527 |
+
class_names=self.class_names)
|
528 |
+
draw_bbox(raw_img, detections, cmap=self.class_color, random_color=True)
|
529 |
+
return detections
|
530 |
+
|
requirements.txt
ADDED
@@ -0,0 +1,96 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
absl-py==1.3.0
|
2 |
+
aiohttp==3.8.3
|
3 |
+
aiosignal==1.3.1
|
4 |
+
anyio==3.6.2
|
5 |
+
astunparse==1.6.3
|
6 |
+
async-timeout==4.0.2
|
7 |
+
attrs==22.1.0
|
8 |
+
bcrypt==4.0.1
|
9 |
+
cachetools==5.2.0
|
10 |
+
certifi==2022.9.24
|
11 |
+
cffi==1.15.1
|
12 |
+
charset-normalizer==2.1.1
|
13 |
+
click==8.1.3
|
14 |
+
colorama==0.4.6
|
15 |
+
contourpy==1.0.6
|
16 |
+
cryptography==38.0.3
|
17 |
+
cycler==0.11.0
|
18 |
+
fastapi==0.87.0
|
19 |
+
ffmpy==0.3.0
|
20 |
+
flatbuffers==22.10.26
|
21 |
+
fonttools==4.38.0
|
22 |
+
frozenlist==1.3.3
|
23 |
+
fsspec==2022.11.0
|
24 |
+
gast==0.4.0
|
25 |
+
google-auth==2.14.1
|
26 |
+
google-auth-oauthlib==0.4.6
|
27 |
+
google-pasta==0.2.0
|
28 |
+
gradio==3.10.0
|
29 |
+
grpcio==1.50.0
|
30 |
+
h11==0.12.0
|
31 |
+
h5py==3.7.0
|
32 |
+
httpcore==0.15.0
|
33 |
+
httpx==0.23.1
|
34 |
+
idna==3.4
|
35 |
+
importlib-metadata==5.0.0
|
36 |
+
Jinja2==3.1.2
|
37 |
+
joblib==1.2.0
|
38 |
+
keras==2.11.0
|
39 |
+
kiwisolver==1.4.4
|
40 |
+
libclang==14.0.6
|
41 |
+
linkify-it-py==1.0.3
|
42 |
+
Markdown==3.4.1
|
43 |
+
markdown-it-py==2.1.0
|
44 |
+
MarkupSafe==2.1.1
|
45 |
+
matplotlib==3.6.2
|
46 |
+
mdit-py-plugins==0.3.1
|
47 |
+
mdurl==0.1.2
|
48 |
+
multidict==6.0.2
|
49 |
+
numpy==1.23.4
|
50 |
+
oauthlib==3.2.2
|
51 |
+
opencv-python==4.6.0.66
|
52 |
+
opt-einsum==3.3.0
|
53 |
+
orjson==3.8.1
|
54 |
+
packaging==21.3
|
55 |
+
pandas==1.5.1
|
56 |
+
paramiko==2.12.0
|
57 |
+
Pillow==9.3.0
|
58 |
+
protobuf==3.19.6
|
59 |
+
pyasn1==0.4.8
|
60 |
+
pyasn1-modules==0.2.8
|
61 |
+
pycparser==2.21
|
62 |
+
pycryptodome==3.15.0
|
63 |
+
pydantic==1.10.2
|
64 |
+
pydub==0.25.1
|
65 |
+
PyNaCl==1.5.0
|
66 |
+
pyparsing==3.0.9
|
67 |
+
python-dateutil==2.8.2
|
68 |
+
python-multipart==0.0.5
|
69 |
+
pytz==2022.6
|
70 |
+
PyYAML==6.0
|
71 |
+
requests==2.28.1
|
72 |
+
requests-oauthlib==1.3.1
|
73 |
+
rfc3986==1.5.0
|
74 |
+
rsa==4.9
|
75 |
+
scikit-learn==1.1.3
|
76 |
+
scipy==1.9.3
|
77 |
+
six==1.16.0
|
78 |
+
sniffio==1.3.0
|
79 |
+
starlette==0.21.0
|
80 |
+
tensorboard==2.11.0
|
81 |
+
tensorboard-data-server==0.6.1
|
82 |
+
tensorboard-plugin-wit==1.8.1
|
83 |
+
tensorflow==2.11.0
|
84 |
+
tensorflow-estimator==2.11.0
|
85 |
+
termcolor==2.1.0
|
86 |
+
threadpoolctl==3.1.0
|
87 |
+
tqdm==4.64.1
|
88 |
+
typing_extensions==4.4.0
|
89 |
+
uc-micro-py==1.0.1
|
90 |
+
urllib3==1.26.12
|
91 |
+
uvicorn==0.19.0
|
92 |
+
websockets==10.4
|
93 |
+
Werkzeug==2.2.2
|
94 |
+
wrapt==1.14.1
|
95 |
+
yarl==1.8.1
|
96 |
+
zipp==3.10.0
|
utils.py
ADDED
@@ -0,0 +1,475 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import numpy as np
|
2 |
+
import cv2
|
3 |
+
import pandas as pd
|
4 |
+
import operator
|
5 |
+
import matplotlib.pyplot as plt
|
6 |
+
import os
|
7 |
+
from sklearn.model_selection import train_test_split
|
8 |
+
from tensorflow.keras.utils import Sequence
|
9 |
+
from config import yolo_config
|
10 |
+
|
11 |
+
|
12 |
+
def load_weights(model, weights_file_path):
|
13 |
+
conv_layer_size = 110
|
14 |
+
conv_output_idxs = [93, 101, 109]
|
15 |
+
with open(weights_file_path, 'rb') as file:
|
16 |
+
major, minor, revision, seen, _ = np.fromfile(file, dtype=np.int32, count=5)
|
17 |
+
|
18 |
+
bn_idx = 0
|
19 |
+
for conv_idx in range(conv_layer_size):
|
20 |
+
conv_layer_name = f'conv2d_{conv_idx}' if conv_idx > 0 else 'conv2d'
|
21 |
+
bn_layer_name = f'batch_normalization_{bn_idx}' if bn_idx > 0 else 'batch_normalization'
|
22 |
+
|
23 |
+
conv_layer = model.get_layer(conv_layer_name)
|
24 |
+
filters = conv_layer.filters
|
25 |
+
kernel_size = conv_layer.kernel_size[0]
|
26 |
+
input_dims = conv_layer.input_shape[-1]
|
27 |
+
|
28 |
+
if conv_idx not in conv_output_idxs:
|
29 |
+
# darknet bn layer weights: [beta, gamma, mean, variance]
|
30 |
+
bn_weights = np.fromfile(file, dtype=np.float32, count=4 * filters)
|
31 |
+
# tf bn layer weights: [gamma, beta, mean, variance]
|
32 |
+
bn_weights = bn_weights.reshape((4, filters))[[1, 0, 2, 3]]
|
33 |
+
bn_layer = model.get_layer(bn_layer_name)
|
34 |
+
bn_idx += 1
|
35 |
+
else:
|
36 |
+
conv_bias = np.fromfile(file, dtype=np.float32, count=filters)
|
37 |
+
|
38 |
+
# darknet shape: (out_dim, input_dims, height, width)
|
39 |
+
# tf shape: (height, width, input_dims, out_dim)
|
40 |
+
conv_shape = (filters, input_dims, kernel_size, kernel_size)
|
41 |
+
conv_weights = np.fromfile(file, dtype=np.float32, count=np.product(conv_shape))
|
42 |
+
conv_weights = conv_weights.reshape(conv_shape).transpose([2, 3, 1, 0])
|
43 |
+
|
44 |
+
if conv_idx not in conv_output_idxs:
|
45 |
+
conv_layer.set_weights([conv_weights])
|
46 |
+
bn_layer.set_weights(bn_weights)
|
47 |
+
else:
|
48 |
+
conv_layer.set_weights([conv_weights, conv_bias])
|
49 |
+
|
50 |
+
if len(file.read()) == 0:
|
51 |
+
print('all weights read')
|
52 |
+
else:
|
53 |
+
print(f'failed to read all weights, # of unread weights: {len(file.read())}')
|
54 |
+
|
55 |
+
|
56 |
+
def get_detection_data(img, model_outputs, class_names):
|
57 |
+
"""
|
58 |
+
|
59 |
+
:param img: target raw image
|
60 |
+
:param model_outputs: outputs from inference_model
|
61 |
+
:param class_names: list of object class names
|
62 |
+
:return:
|
63 |
+
"""
|
64 |
+
|
65 |
+
num_bboxes = model_outputs[-1][0]
|
66 |
+
boxes, scores, classes = [output[0][:num_bboxes] for output in model_outputs[:-1]]
|
67 |
+
|
68 |
+
h, w = img.shape[:2]
|
69 |
+
df = pd.DataFrame(boxes, columns=['x1', 'y1', 'x2', 'y2'])
|
70 |
+
df[['x1', 'x2']] = (df[['x1', 'x2']] * w).astype('int64')
|
71 |
+
df[['y1', 'y2']] = (df[['y1', 'y2']] * h).astype('int64')
|
72 |
+
df['class_name'] = np.array(class_names)[classes.astype('int64')]
|
73 |
+
df['score'] = scores
|
74 |
+
df['w'] = df['x2'] - df['x1']
|
75 |
+
df['h'] = df['y2'] - df['y1']
|
76 |
+
|
77 |
+
print(f'# of bboxes: {num_bboxes}')
|
78 |
+
return df
|
79 |
+
|
80 |
+
def read_annotation_lines(annotation_path, test_size=None, random_seed=5566):
|
81 |
+
with open(annotation_path) as f:
|
82 |
+
lines = f.readlines()
|
83 |
+
if test_size:
|
84 |
+
return train_test_split(lines, test_size=test_size, random_state=random_seed)
|
85 |
+
else:
|
86 |
+
return lines
|
87 |
+
|
88 |
+
def draw_bbox(img, detections, cmap, random_color=True, figsize=(10, 10), show_img=True, show_text=True):
|
89 |
+
"""
|
90 |
+
Draw bounding boxes on the img.
|
91 |
+
:param img: BGR img.
|
92 |
+
:param detections: pandas DataFrame containing detections
|
93 |
+
:param random_color: assign random color for each objects
|
94 |
+
:param cmap: object colormap
|
95 |
+
:param plot_img: if plot img with bboxes
|
96 |
+
:return: None
|
97 |
+
"""
|
98 |
+
img = np.array(img)
|
99 |
+
scale = max(img.shape[0:2]) / 416
|
100 |
+
line_width = int(2 * scale)
|
101 |
+
|
102 |
+
for _, row in detections.iterrows():
|
103 |
+
x1, y1, x2, y2, cls, score, w, h = row.values
|
104 |
+
color = list(np.random.random(size=3) * 255) if random_color else cmap[cls]
|
105 |
+
cv2.rectangle(img, (x1, y1), (x2, y2), color, line_width)
|
106 |
+
if show_text:
|
107 |
+
text = f'{cls} {score:.2f}'
|
108 |
+
font = cv2.FONT_HERSHEY_DUPLEX
|
109 |
+
font_scale = max(0.3 * scale, 0.3)
|
110 |
+
thickness = max(int(1 * scale), 1)
|
111 |
+
(text_width, text_height) = cv2.getTextSize(text, font, fontScale=font_scale, thickness=thickness)[0]
|
112 |
+
cv2.rectangle(img, (x1 - line_width//2, y1 - text_height), (x1 + text_width, y1), color, cv2.FILLED)
|
113 |
+
cv2.putText(img, text, (x1, y1), font, font_scale, (255, 255, 255), thickness, cv2.LINE_AA)
|
114 |
+
if show_img:
|
115 |
+
plt.figure(figsize=figsize)
|
116 |
+
plt.imshow(img)
|
117 |
+
plt.show()
|
118 |
+
return img
|
119 |
+
|
120 |
+
|
121 |
+
class DataGenerator(Sequence):
|
122 |
+
"""
|
123 |
+
Generates data for Keras
|
124 |
+
ref: https://stanford.edu/~shervine/blog/keras-how-to-generate-data-on-the-fly
|
125 |
+
"""
|
126 |
+
def __init__(self,
|
127 |
+
annotation_lines,
|
128 |
+
class_name_path,
|
129 |
+
folder_path,
|
130 |
+
max_boxes=100,
|
131 |
+
shuffle=True):
|
132 |
+
self.annotation_lines = annotation_lines
|
133 |
+
self.class_name_path = class_name_path
|
134 |
+
self.num_classes = len([line.strip() for line in open(class_name_path).readlines()])
|
135 |
+
self.num_gpu = yolo_config['num_gpu']
|
136 |
+
self.batch_size = yolo_config['batch_size'] * self.num_gpu
|
137 |
+
self.target_img_size = yolo_config['img_size']
|
138 |
+
self.anchors = np.array(yolo_config['anchors']).reshape((9, 2))
|
139 |
+
self.shuffle = shuffle
|
140 |
+
self.indexes = np.arange(len(self.annotation_lines))
|
141 |
+
self.folder_path = folder_path
|
142 |
+
self.max_boxes = max_boxes
|
143 |
+
self.on_epoch_end()
|
144 |
+
|
145 |
+
def __len__(self):
|
146 |
+
'number of batches per epoch'
|
147 |
+
return int(np.ceil(len(self.annotation_lines) / self.batch_size))
|
148 |
+
|
149 |
+
def __getitem__(self, index):
|
150 |
+
'Generate one batch of data'
|
151 |
+
|
152 |
+
# Generate indexes of the batch
|
153 |
+
idxs = self.indexes[index * self.batch_size:(index + 1) * self.batch_size]
|
154 |
+
|
155 |
+
# Find list of IDs
|
156 |
+
lines = [self.annotation_lines[i] for i in idxs]
|
157 |
+
|
158 |
+
# Generate data
|
159 |
+
X, y_tensor, y_bbox = self.__data_generation(lines)
|
160 |
+
|
161 |
+
return [X, *y_tensor, y_bbox], np.zeros(len(lines))
|
162 |
+
|
163 |
+
def on_epoch_end(self):
|
164 |
+
'Updates indexes after each epoch'
|
165 |
+
if self.shuffle:
|
166 |
+
np.random.shuffle(self.indexes)
|
167 |
+
|
168 |
+
def __data_generation(self, annotation_lines):
|
169 |
+
"""
|
170 |
+
Generates data containing batch_size samples
|
171 |
+
:param annotation_lines:
|
172 |
+
:return:
|
173 |
+
"""
|
174 |
+
|
175 |
+
X = np.empty((len(annotation_lines), *self.target_img_size), dtype=np.float32)
|
176 |
+
y_bbox = np.empty((len(annotation_lines), self.max_boxes, 5), dtype=np.float32) # x1y1x2y2
|
177 |
+
|
178 |
+
for i, line in enumerate(annotation_lines):
|
179 |
+
img_data, box_data = self.get_data(line)
|
180 |
+
X[i] = img_data
|
181 |
+
y_bbox[i] = box_data
|
182 |
+
|
183 |
+
y_tensor, y_true_boxes_xywh = preprocess_true_boxes(y_bbox, self.target_img_size[:2], self.anchors, self.num_classes)
|
184 |
+
|
185 |
+
return X, y_tensor, y_true_boxes_xywh
|
186 |
+
|
187 |
+
def get_data(self, annotation_line):
|
188 |
+
line = annotation_line.split()
|
189 |
+
img_path = line[0]
|
190 |
+
img = cv2.imread(os.path.join(self.folder_path, img_path))[:, :, ::-1]
|
191 |
+
ih, iw = img.shape[:2]
|
192 |
+
h, w, c = self.target_img_size
|
193 |
+
boxes = np.array([np.array(list(map(float, box.split(',')))) for box in line[1:]], dtype=np.float32) # x1y1x2y2
|
194 |
+
scale_w, scale_h = w / iw, h / ih
|
195 |
+
img = cv2.resize(img, (w, h))
|
196 |
+
image_data = np.array(img) / 255.
|
197 |
+
|
198 |
+
# correct boxes coordinates
|
199 |
+
box_data = np.zeros((self.max_boxes, 5))
|
200 |
+
if len(boxes) > 0:
|
201 |
+
np.random.shuffle(boxes)
|
202 |
+
boxes = boxes[:self.max_boxes]
|
203 |
+
boxes[:, [0, 2]] = boxes[:, [0, 2]] * scale_w # + dx
|
204 |
+
boxes[:, [1, 3]] = boxes[:, [1, 3]] * scale_h # + dy
|
205 |
+
box_data[:len(boxes)] = boxes
|
206 |
+
|
207 |
+
return image_data, box_data
|
208 |
+
|
209 |
+
|
210 |
+
def preprocess_true_boxes(true_boxes, input_shape, anchors, num_classes):
|
211 |
+
'''Preprocess true boxes to training input format
|
212 |
+
|
213 |
+
Parameters
|
214 |
+
----------
|
215 |
+
true_boxes: array, shape=(bs, max boxes per img, 5)
|
216 |
+
Absolute x_min, y_min, x_max, y_max, class_id relative to input_shape.
|
217 |
+
input_shape: array-like, hw, multiples of 32
|
218 |
+
anchors: array, shape=(N, 2), (9, wh)
|
219 |
+
num_classes: int
|
220 |
+
|
221 |
+
Returns
|
222 |
+
-------
|
223 |
+
y_true: list of array, shape like yolo_outputs, xywh are reletive value
|
224 |
+
|
225 |
+
'''
|
226 |
+
|
227 |
+
num_stages = 3 # default setting for yolo, tiny yolo will be 2
|
228 |
+
anchor_mask = [[0, 1, 2], [3, 4, 5], [6, 7, 8]]
|
229 |
+
bbox_per_grid = 3
|
230 |
+
true_boxes = np.array(true_boxes, dtype='float32')
|
231 |
+
true_boxes_abs = np.array(true_boxes, dtype='float32')
|
232 |
+
input_shape = np.array(input_shape, dtype='int32')
|
233 |
+
true_boxes_xy = (true_boxes_abs[..., 0:2] + true_boxes_abs[..., 2:4]) // 2 # (100, 2)
|
234 |
+
true_boxes_wh = true_boxes_abs[..., 2:4] - true_boxes_abs[..., 0:2] # (100, 2)
|
235 |
+
|
236 |
+
# Normalize x,y,w, h, relative to img size -> (0~1)
|
237 |
+
true_boxes[..., 0:2] = true_boxes_xy/input_shape[::-1] # xy
|
238 |
+
true_boxes[..., 2:4] = true_boxes_wh/input_shape[::-1] # wh
|
239 |
+
|
240 |
+
bs = true_boxes.shape[0]
|
241 |
+
grid_sizes = [input_shape//{0:8, 1:16, 2:32}[stage] for stage in range(num_stages)]
|
242 |
+
y_true = [np.zeros((bs,
|
243 |
+
grid_sizes[s][0],
|
244 |
+
grid_sizes[s][1],
|
245 |
+
bbox_per_grid,
|
246 |
+
5+num_classes), dtype='float32')
|
247 |
+
for s in range(num_stages)]
|
248 |
+
# [(?, 52, 52, 3, 5+num_classes) (?, 26, 26, 3, 5+num_classes) (?, 13, 13, 3, 5+num_classes) ]
|
249 |
+
y_true_boxes_xywh = np.concatenate((true_boxes_xy, true_boxes_wh), axis=-1)
|
250 |
+
# Expand dim to apply broadcasting.
|
251 |
+
anchors = np.expand_dims(anchors, 0) # (1, 9 , 2)
|
252 |
+
anchor_maxes = anchors / 2. # (1, 9 , 2)
|
253 |
+
anchor_mins = -anchor_maxes # (1, 9 , 2)
|
254 |
+
valid_mask = true_boxes_wh[..., 0] > 0 # (1, 100)
|
255 |
+
|
256 |
+
for batch_idx in range(bs):
|
257 |
+
# Discard zero rows.
|
258 |
+
wh = true_boxes_wh[batch_idx, valid_mask[batch_idx]] # (# of bbox, 2)
|
259 |
+
num_boxes = len(wh)
|
260 |
+
if num_boxes == 0: continue
|
261 |
+
wh = np.expand_dims(wh, -2) # (# of bbox, 1, 2)
|
262 |
+
box_maxes = wh / 2. # (# of bbox, 1, 2)
|
263 |
+
box_mins = -box_maxes # (# of bbox, 1, 2)
|
264 |
+
|
265 |
+
# Compute IoU between each anchors and true boxes for responsibility assignment
|
266 |
+
intersect_mins = np.maximum(box_mins, anchor_mins) # (# of bbox, 9, 2)
|
267 |
+
intersect_maxes = np.minimum(box_maxes, anchor_maxes)
|
268 |
+
intersect_wh = np.maximum(intersect_maxes - intersect_mins, 0.)
|
269 |
+
intersect_area = np.prod(intersect_wh, axis=-1) # (9,)
|
270 |
+
box_area = wh[..., 0] * wh[..., 1] # (# of bbox, 1)
|
271 |
+
anchor_area = anchors[..., 0] * anchors[..., 1] # (1, 9)
|
272 |
+
iou = intersect_area / (box_area + anchor_area - intersect_area) # (# of bbox, 9)
|
273 |
+
|
274 |
+
# Find best anchor for each true box
|
275 |
+
best_anchors = np.argmax(iou, axis=-1) # (# of bbox,)
|
276 |
+
for box_idx in range(num_boxes):
|
277 |
+
best_anchor = best_anchors[box_idx]
|
278 |
+
for stage in range(num_stages):
|
279 |
+
if best_anchor in anchor_mask[stage]:
|
280 |
+
x_offset = true_boxes[batch_idx, box_idx, 0]*grid_sizes[stage][1]
|
281 |
+
y_offset = true_boxes[batch_idx, box_idx, 1]*grid_sizes[stage][0]
|
282 |
+
# Grid Index
|
283 |
+
grid_col = np.floor(x_offset).astype('int32')
|
284 |
+
grid_row = np.floor(y_offset).astype('int32')
|
285 |
+
anchor_idx = anchor_mask[stage].index(best_anchor)
|
286 |
+
class_idx = true_boxes[batch_idx, box_idx, 4].astype('int32')
|
287 |
+
# y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, 0] = x_offset - grid_col # x
|
288 |
+
# y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, 1] = y_offset - grid_row # y
|
289 |
+
# y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, :4] = true_boxes_abs[batch_idx, box_idx, :4] # abs xywh
|
290 |
+
y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, :2] = true_boxes_xy[batch_idx, box_idx, :] # abs xy
|
291 |
+
y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, 2:4] = true_boxes_wh[batch_idx, box_idx, :] # abs wh
|
292 |
+
y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, 4] = 1 # confidence
|
293 |
+
|
294 |
+
y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, 5+class_idx] = 1 # one-hot encoding
|
295 |
+
# smooth
|
296 |
+
# onehot = np.zeros(num_classes, dtype=np.float)
|
297 |
+
# onehot[class_idx] = 1.0
|
298 |
+
# uniform_distribution = np.full(num_classes, 1.0 / num_classes)
|
299 |
+
# delta = 0.01
|
300 |
+
# smooth_onehot = onehot * (1 - delta) + delta * uniform_distribution
|
301 |
+
# y_true[stage][batch_idx, grid_row, grid_col, anchor_idx, 5:] = smooth_onehot
|
302 |
+
|
303 |
+
return y_true, y_true_boxes_xywh
|
304 |
+
|
305 |
+
"""
|
306 |
+
Calculate the AP given the recall and precision array
|
307 |
+
1st) We compute a version of the measured precision/recall curve with
|
308 |
+
precision monotonically decreasing
|
309 |
+
2nd) We compute the AP as the area under this curve by numerical integration.
|
310 |
+
"""
|
311 |
+
def voc_ap(rec, prec):
|
312 |
+
"""
|
313 |
+
--- Official matlab code VOC2012---
|
314 |
+
mrec=[0 ; rec ; 1];
|
315 |
+
mpre=[0 ; prec ; 0];
|
316 |
+
for i=numel(mpre)-1:-1:1
|
317 |
+
mpre(i)=max(mpre(i),mpre(i+1));
|
318 |
+
end
|
319 |
+
i=find(mrec(2:end)~=mrec(1:end-1))+1;
|
320 |
+
ap=sum((mrec(i)-mrec(i-1)).*mpre(i));
|
321 |
+
"""
|
322 |
+
rec.insert(0, 0.0) # insert 0.0 at begining of list
|
323 |
+
rec.append(1.0) # insert 1.0 at end of list
|
324 |
+
mrec = rec[:]
|
325 |
+
prec.insert(0, 0.0) # insert 0.0 at begining of list
|
326 |
+
prec.append(0.0) # insert 0.0 at end of list
|
327 |
+
mpre = prec[:]
|
328 |
+
"""
|
329 |
+
This part makes the precision monotonically decreasing
|
330 |
+
(goes from the end to the beginning)
|
331 |
+
matlab: for i=numel(mpre)-1:-1:1
|
332 |
+
mpre(i)=max(mpre(i),mpre(i+1));
|
333 |
+
"""
|
334 |
+
# matlab indexes start in 1 but python in 0, so I have to do:
|
335 |
+
# range(start=(len(mpre) - 2), end=0, step=-1)
|
336 |
+
# also the python function range excludes the end, resulting in:
|
337 |
+
# range(start=(len(mpre) - 2), end=-1, step=-1)
|
338 |
+
for i in range(len(mpre)-2, -1, -1):
|
339 |
+
mpre[i] = max(mpre[i], mpre[i+1])
|
340 |
+
"""
|
341 |
+
This part creates a list of indexes where the recall changes
|
342 |
+
matlab: i=find(mrec(2:end)~=mrec(1:end-1))+1;
|
343 |
+
"""
|
344 |
+
i_list = []
|
345 |
+
for i in range(1, len(mrec)):
|
346 |
+
if mrec[i] != mrec[i-1]:
|
347 |
+
i_list.append(i) # if it was matlab would be i + 1
|
348 |
+
"""
|
349 |
+
The Average Precision (AP) is the area under the curve
|
350 |
+
(numerical integration)
|
351 |
+
matlab: ap=sum((mrec(i)-mrec(i-1)).*mpre(i));
|
352 |
+
"""
|
353 |
+
ap = 0.0
|
354 |
+
for i in i_list:
|
355 |
+
ap += ((mrec[i]-mrec[i-1])*mpre[i])
|
356 |
+
return ap, mrec, mpre
|
357 |
+
|
358 |
+
"""
|
359 |
+
Draw plot using Matplotlib
|
360 |
+
"""
|
361 |
+
def draw_plot_func(dictionary, n_classes, window_title, plot_title, x_label, output_path, to_show, plot_color, true_p_bar):
|
362 |
+
# sort the dictionary by decreasing value, into a list of tuples
|
363 |
+
sorted_dic_by_value = sorted(dictionary.items(), key=operator.itemgetter(1))
|
364 |
+
print(sorted_dic_by_value)
|
365 |
+
# unpacking the list of tuples into two lists
|
366 |
+
sorted_keys, sorted_values = zip(*sorted_dic_by_value)
|
367 |
+
#
|
368 |
+
if true_p_bar != "":
|
369 |
+
"""
|
370 |
+
Special case to draw in:
|
371 |
+
- green -> TP: True Positives (object detected and matches ground-truth)
|
372 |
+
- red -> FP: False Positives (object detected but does not match ground-truth)
|
373 |
+
- pink -> FN: False Negatives (object not detected but present in the ground-truth)
|
374 |
+
"""
|
375 |
+
fp_sorted = []
|
376 |
+
tp_sorted = []
|
377 |
+
for key in sorted_keys:
|
378 |
+
fp_sorted.append(dictionary[key] - true_p_bar[key])
|
379 |
+
tp_sorted.append(true_p_bar[key])
|
380 |
+
plt.barh(range(n_classes), fp_sorted, align='center', color='crimson', label='False Positive')
|
381 |
+
plt.barh(range(n_classes), tp_sorted, align='center', color='forestgreen', label='True Positive', left=fp_sorted)
|
382 |
+
# add legend
|
383 |
+
plt.legend(loc='lower right')
|
384 |
+
"""
|
385 |
+
Write number on side of bar
|
386 |
+
"""
|
387 |
+
fig = plt.gcf() # gcf - get current figure
|
388 |
+
axes = plt.gca()
|
389 |
+
r = fig.canvas.get_renderer()
|
390 |
+
for i, val in enumerate(sorted_values):
|
391 |
+
fp_val = fp_sorted[i]
|
392 |
+
tp_val = tp_sorted[i]
|
393 |
+
fp_str_val = " " + str(fp_val)
|
394 |
+
tp_str_val = fp_str_val + " " + str(tp_val)
|
395 |
+
# trick to paint multicolor with offset:
|
396 |
+
# first paint everything and then repaint the first number
|
397 |
+
t = plt.text(val, i, tp_str_val, color='forestgreen', va='center', fontweight='bold')
|
398 |
+
plt.text(val, i, fp_str_val, color='crimson', va='center', fontweight='bold')
|
399 |
+
if i == (len(sorted_values)-1): # largest bar
|
400 |
+
adjust_axes(r, t, fig, axes)
|
401 |
+
else:
|
402 |
+
plt.barh(range(n_classes), sorted_values, color=plot_color)
|
403 |
+
"""
|
404 |
+
Write number on side of bar
|
405 |
+
"""
|
406 |
+
fig = plt.gcf() # gcf - get current figure
|
407 |
+
axes = plt.gca()
|
408 |
+
r = fig.canvas.get_renderer()
|
409 |
+
for i, val in enumerate(sorted_values):
|
410 |
+
str_val = " " + str(val) # add a space before
|
411 |
+
if val < 1.0:
|
412 |
+
str_val = " {0:.2f}".format(val)
|
413 |
+
t = plt.text(val, i, str_val, color=plot_color, va='center', fontweight='bold')
|
414 |
+
# re-set axes to show number inside the figure
|
415 |
+
if i == (len(sorted_values)-1): # largest bar
|
416 |
+
adjust_axes(r, t, fig, axes)
|
417 |
+
# set window title
|
418 |
+
fig.canvas.set_window_title(window_title)
|
419 |
+
# write classes in y axis
|
420 |
+
tick_font_size = 12
|
421 |
+
plt.yticks(range(n_classes), sorted_keys, fontsize=tick_font_size)
|
422 |
+
"""
|
423 |
+
Re-scale height accordingly
|
424 |
+
"""
|
425 |
+
init_height = fig.get_figheight()
|
426 |
+
# comput the matrix height in points and inches
|
427 |
+
dpi = fig.dpi
|
428 |
+
height_pt = n_classes * (tick_font_size * 1.4) # 1.4 (some spacing)
|
429 |
+
height_in = height_pt / dpi
|
430 |
+
# compute the required figure height
|
431 |
+
top_margin = 0.15 # in percentage of the figure height
|
432 |
+
bottom_margin = 0.05 # in percentage of the figure height
|
433 |
+
figure_height = height_in / (1 - top_margin - bottom_margin)
|
434 |
+
# set new height
|
435 |
+
if figure_height > init_height:
|
436 |
+
fig.set_figheight(figure_height)
|
437 |
+
|
438 |
+
# set plot title
|
439 |
+
plt.title(plot_title, fontsize=14)
|
440 |
+
# set axis titles
|
441 |
+
# plt.xlabel('classes')
|
442 |
+
plt.xlabel(x_label, fontsize='large')
|
443 |
+
# adjust size of window
|
444 |
+
fig.tight_layout()
|
445 |
+
# save the plot
|
446 |
+
fig.savefig(output_path)
|
447 |
+
# show image
|
448 |
+
# if to_show:
|
449 |
+
plt.show()
|
450 |
+
# close the plot
|
451 |
+
# plt.close()
|
452 |
+
|
453 |
+
"""
|
454 |
+
Plot - adjust axes
|
455 |
+
"""
|
456 |
+
def adjust_axes(r, t, fig, axes):
|
457 |
+
# get text width for re-scaling
|
458 |
+
bb = t.get_window_extent(renderer=r)
|
459 |
+
text_width_inches = bb.width / fig.dpi
|
460 |
+
# get axis width in inches
|
461 |
+
current_fig_width = fig.get_figwidth()
|
462 |
+
new_fig_width = current_fig_width + text_width_inches
|
463 |
+
propotion = new_fig_width / current_fig_width
|
464 |
+
# get axis limit
|
465 |
+
x_lim = axes.get_xlim()
|
466 |
+
axes.set_xlim([x_lim[0], x_lim[1]*propotion])
|
467 |
+
|
468 |
+
|
469 |
+
def read_txt_to_list(path):
|
470 |
+
# open txt file lines to a list
|
471 |
+
with open(path) as f:
|
472 |
+
content = f.readlines()
|
473 |
+
# remove whitespace characters like `\n` at the end of each line
|
474 |
+
content = [x.strip() for x in content]
|
475 |
+
return content
|
xml_to_txt.py
ADDED
@@ -0,0 +1,42 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import xml.etree.ElementTree as ET
|
2 |
+
import os
|
3 |
+
from glob import glob
|
4 |
+
|
5 |
+
XML_PATH = './dataset/xml'
|
6 |
+
CLASSES_PATH = './class_names/classes.txt'
|
7 |
+
TXT_PATH = './dataset/txt/anno.txt'
|
8 |
+
|
9 |
+
|
10 |
+
'''loads the classes'''
|
11 |
+
def get_classes(classes_path):
|
12 |
+
with open(classes_path) as f:
|
13 |
+
class_names = f.readlines()
|
14 |
+
class_names = [c.strip() for c in class_names]
|
15 |
+
return class_names
|
16 |
+
|
17 |
+
|
18 |
+
classes = get_classes(CLASSES_PATH)
|
19 |
+
assert len(classes) > 0, 'no class names detected!'
|
20 |
+
print(f'num classes: {len(classes)}')
|
21 |
+
|
22 |
+
# output file
|
23 |
+
list_file = open(TXT_PATH, 'w')
|
24 |
+
|
25 |
+
for path in glob(os.path.join(XML_PATH, '*.xml')):
|
26 |
+
in_file = open(path)
|
27 |
+
|
28 |
+
# Parse .xml file
|
29 |
+
tree = ET.parse(in_file)
|
30 |
+
root = tree.getroot()
|
31 |
+
# Write object information to .txt file
|
32 |
+
file_name = root.find('filename').text
|
33 |
+
print(file_name)
|
34 |
+
list_file.write(file_name)
|
35 |
+
for obj in root.iter('object'):
|
36 |
+
cls = obj.find('name').text
|
37 |
+
cls_id = classes.index(cls)
|
38 |
+
xmlbox = obj.find('bndbox')
|
39 |
+
b = (int(xmlbox.find('xmin').text), int(xmlbox.find('ymin').text), int(xmlbox.find('xmax').text), int(xmlbox.find('ymax').text))
|
40 |
+
list_file.write(" " + ",".join([str(a) for a in b]) + ',' + str(cls_id))
|
41 |
+
list_file.write('\n')
|
42 |
+
list_file.close()
|
yolov4.weights
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:e8a4f6c62188738d86dc6898d82724ec0964d0eb9d2ae0f0a9d53d65d108d562
|
3 |
+
size 257717640
|