iOS: нет результата от предсказания CoreML, используя модель, конвертированную из Keras
Проблема, с которой я сталкиваюсь, заключается в Xcode 9 (бета-версия 5), прогнозирующий вызов моей модели не дает никакого результата, просто зависает там, без ошибок.
Требуется распознать одно конкретное движение телефона пользователя и исключить другие подобные движения. Таким образом, мои входные данные - это данные от акселерометра для направлений x, y, z в отдельности, а выходные данные - [positiveProbability, негативные возможности].
Первоначально моя модель была написана с использованием TensorFlow, затем я преобразовал ее в модель Keras и преобразовал в модель CoreML. (TensorFlow-1.1.0 Keras-2.0.6 Protobuf-3.4.0)
Вот код Python моей модели:
import numpy as np
np.random.seed(123) # for reproducibility
from keras.models import Sequential
from keras.layers import Dense, Dropout, Flatten
from keras.layers import Convolution2D, MaxPooling2D
from keras.utils import np_utils
from keras.datasets import mnist
from keras import backend as K
from keras.optimizers import sgd
import json
from data_fetcher import pre_process_data
import coremltools
print K.image_data_format()
def process_data(datafile):
with open(datafile) as data_file:
data = json.load(data_file)
if K.image_data_format() == 'channels_last':
# NHWC: tensorflow backend, always channel last
_x = np.zeros(shape=(len(data), 200, 3, 1))
_y = np.zeros(shape=(len(data), 2))
for i in range(len(data)):
output = pre_process_data(data, i)
output.data_process()
output.fft()
#output.plot_curve('Walk')
temp = [output.acceHFX, output.acceHFY, output.acceHFZ]
temp = np.transpose(temp)
temp = np.reshape(temp, (200, 3, 1))
label = output.string
if label == 'Pattern1' or label == 'Pattern2':
l = np.array([[1., 0.]])
else:
l = np.array([[0., 1.]])
_x[i] = temp
_y[i] = l
# shuffle the data set (_x,_y shuffle according to the same order)
c = list(zip(_x,_y))
np.random.shuffle(c)
_x1,_y1 = zip(*c)
Data = (np.asarray(_x1), np.asarray(_y1))
return Data
def deepnn():
# 7. Define model architecture
model = Sequential()
# below is the model given by tensorflow example:
model.add(Convolution2D(10, (5, 1), activation='relu', input_shape=(200, 3, 1))) # a stack of 10 H*W =200*3 images
model.add(MaxPooling2D(pool_size=(5, 1))) # reduce size to (10,40,3)
model.add(Convolution2D(20, (5, 1), activation='relu')) # a stack of 20 40*3 images
model.add(MaxPooling2D(pool_size=(5, 1))) # reduce size to (20,8,3)
model.add(Dropout(0.25))
model.add(Flatten()) # flatten to be 20*8*3
model.add(Dense(50, activation='relu')) # fully connected layer shrink down to 50 features
model.add(Dropout(0.5))
model.add(Dense(2, activation='softmax')) # map 50 features to 2 classes
# 8. Compile model
SGD = sgd(lr=0.05, decay=0.0, momentum=0.0, nesterov=False)
model.compile(loss='categorical_crossentropy',
optimizer='SGD',
metrics=['accuracy'])
return model
# 9. Fit model on training data
def main():
model = deepnn()
filename = 'datacollector-07182017Combined.json'
xData = process_data(filename)
testing_size = 0.25 # reserve 20 percent for testing
num_inputs = len(xData[0])
cutoff = int((1.0-testing_size)*num_inputs)
x_train = xData[0][:cutoff]
y_train = xData[1][:cutoff]
x_test = xData[0][cutoff:]
y_test = xData[1][cutoff:]
model.fit(x_train, y_train,batch_size=5, epochs=8, verbose=1)
# 10. Evaluate model on test data
score = model.evaluate(x_test, y_test, verbose=0)
print score
coreml_model = coremltools.converters.keras.convert(model)
coreml_model.save('model.mlmodel')
if __name__ == '__main__':
main()
Это последовательная модель с формой 200x3x1. После импорта в Xcode это выглядит так:
Вот код в моем проекте iOS для создания ввода:
private typealias Accelerate = (x: Double, y: Double, z: Double)
private func transformToModelInput(data: [Accelerate]) -> MLMultiArray {
guard let input = try? MLMultiArray(shape:[1,200,3], dataType:MLMultiArrayDataType.double) else {
fatalError("Unexpected runtime error. MLMultiArray")
}
var dataArray = data.map { $0.x }
dataArray += data.map { $0.y }
dataArray += data.map { $0.z }
for (index, value) in dataArray.enumerated() {
input[index] = NSNumber(floatLiteral: value)
}
return input
}
Вот мои коды предсказания:
let input = transformToModelInput(data: sampledData)
do {
let output = try model.prediction(input1: input)
let multiArray = MultiArray<Double>.init(output.output1)
if multiArray[0] > multiArray[1] {
print("Prediction: POSITIVE \(multiArray)")
} else {
print("Prediction: NEGATIVE \(multiArray)")
}
} catch let error {
print("CoreML Prediction Failed: \(error)")
}
Однако вызов model.prediction никогда не дает никаких результатов, проект просто висит там навсегда. Мне действительно нужна помощь в выяснении того, что не так, или, по крайней мере, как отладить проблему.