168 lines
5.6 KiB
Python
168 lines
5.6 KiB
Python
#!/usr/bin/env python3
|
|
import tempfile
|
|
import os
|
|
from flask import request, jsonify, current_app, abort, Response
|
|
from flask_classful import FlaskView, route
|
|
from model import db, Default, AIModel, AIModelType, SVMDetails
|
|
from minio.error import NoSuchKey
|
|
from schemas import AIModelSchema, DefaultSchema, InfoSchema
|
|
from marshmallow.exceptions import ValidationError
|
|
from utils import json_required, storage, ensure_buckets
|
|
from pyAudioAnalysis.audioTrainTest import load_model, load_model_knn
|
|
|
|
|
|
class SVMView(FlaskView):
|
|
route_base = 'svm'
|
|
|
|
aimodel_schema = AIModelSchema(many=False)
|
|
aimodels_schema = AIModelSchema(many=True, exclude=['timestamp', 'details'])
|
|
default_schema = DefaultSchema(many=False)
|
|
info_schema = InfoSchema(many=False)
|
|
|
|
def index(self):
|
|
models = AIModel.query.filter_by(type=AIModelType.SVM).all()
|
|
return jsonify(self.aimodels_schema.dump(models)), 200
|
|
|
|
def post(self):
|
|
|
|
# get important data from the request
|
|
try:
|
|
info = self.info_schema.loads(request.form.get('info'))
|
|
except ValidationError as e:
|
|
abort(400, str(e))
|
|
|
|
# check for conflict
|
|
m = AIModel.query.filter_by(id=info['id']).first()
|
|
if m:
|
|
abort(409)
|
|
|
|
# get and validate file
|
|
model_file = request.files['modelFile']
|
|
|
|
if model_file.content_length <= 0:
|
|
abort(411, f"Content length for modelFile is not a positive integer or missing.")
|
|
|
|
means_file = request.files['meansFile']
|
|
|
|
if means_file.content_length <= 0:
|
|
abort(411, f"Content length for meansFile is not a positive integer or missing.")
|
|
|
|
# create bucket if necessary
|
|
ensure_buckets()
|
|
|
|
# Temporarily save the file, because pyAudioAnalysis can only read files
|
|
temp_model_handle, temp_model_filename = tempfile.mkstemp()
|
|
temp_means_filename = temp_model_filename + "MEANS"
|
|
|
|
os.close(temp_model_handle) # BRUUUUH
|
|
|
|
model_file.save(temp_model_filename)
|
|
means_file.save(temp_means_filename)
|
|
|
|
try:
|
|
|
|
_, _, _, _, mid_window, mid_step, short_window, short_step, compute_beat \
|
|
= load_model(temp_model_filename)
|
|
|
|
# Because of pyAudiomeme the files already saved, so we just use the file uploader functions
|
|
storage.connection.fput_object(
|
|
current_app.config['MINIO_SVM_BUCKET_NAME'],
|
|
"model/" + str(info['id']),
|
|
temp_model_filename
|
|
)
|
|
|
|
storage.connection.fput_object(
|
|
current_app.config['MINIO_SVM_BUCKET_NAME'],
|
|
"means/" + str(info['id']),
|
|
temp_means_filename
|
|
)
|
|
|
|
finally:
|
|
os.remove(temp_model_filename)
|
|
os.remove(temp_means_filename)
|
|
|
|
m = AIModel(id=info['id'], type=AIModelType.SVM, target_class_name=info['target_class_name'])
|
|
|
|
d = SVMDetails(
|
|
aimodel=m,
|
|
mid_window=mid_window,
|
|
mid_step=mid_step,
|
|
short_window=short_window,
|
|
short_step=short_step,
|
|
compute_beat=compute_beat
|
|
)
|
|
|
|
db.session.add(m)
|
|
db.session.add(d)
|
|
db.session.commit()
|
|
|
|
return jsonify(self.aimodel_schema.dump(m)), 200
|
|
|
|
def get(self, _id: str):
|
|
|
|
if _id == "$default":
|
|
# TODO: Kitalálni, hogy inkább a latestestest-el térjen-e vissza
|
|
default = Default.query.filter_by(type=AIModelType.SVM).first_or_404()
|
|
m = default.aimodel
|
|
else:
|
|
m = AIModel.query.filter_by(type=AIModelType.SVM, id=_id).first_or_404()
|
|
|
|
if "means" in request.args:
|
|
path = "means/" + str(m.id)
|
|
else:
|
|
path = "model/" + str(m.id)
|
|
|
|
try:
|
|
data = storage.connection.get_object(current_app.config['MINIO_SVM_BUCKET_NAME'], path)
|
|
except NoSuchKey:
|
|
abort(500, "The ID is stored in the database but not int the Object Store")
|
|
|
|
return Response(data.stream(), mimetype=data.headers['Content-type'])
|
|
|
|
@route('<_id>/details')
|
|
def get_details(self, _id: str):
|
|
|
|
if _id == "$default":
|
|
# TODO: Kitalálni, hogy inkább a latestestest-el térjen-e vissza
|
|
default = Default.query.filter_by(type=AIModelType.SVM).first_or_404()
|
|
m = default.aimodel
|
|
else:
|
|
m = AIModel.query.filter_by(type=AIModelType.SVM, id=_id).first_or_404()
|
|
|
|
return jsonify(self.aimodel_schema.dump(m))
|
|
|
|
def delete(self, _id: str):
|
|
|
|
if _id == "$default":
|
|
# TODO: Kitalálni, hogy inkább a latestestest-el térjen-e vissza
|
|
default = Default.query.filter_by(type=AIModelType.SVM).first_or_404()
|
|
m = default.aimodel
|
|
else:
|
|
m = AIModel.query.filter_by(type=AIModelType.SVM, id=_id).first_or_404()
|
|
|
|
storage.connection.remove_object(current_app.config['MINIO_SVM_BUCKET_NAME'], "means/" + str(m.id))
|
|
storage.connection.remove_object(current_app.config['MINIO_SVM_BUCKET_NAME'], "model/" + str(m.id))
|
|
|
|
db.session.delete(m)
|
|
db.session.commit()
|
|
|
|
return '', 204
|
|
|
|
@json_required
|
|
@route('$default', methods=['PUT'])
|
|
def put_default(self):
|
|
|
|
try:
|
|
req = self.default_schema.load(request.json)
|
|
except ValidationError as e:
|
|
abort(400, str(e))
|
|
|
|
m = AIModel.query.filter_by(type=AIModelType.SVM, id=req['id']).first_or_404()
|
|
|
|
Default.query.filter_by(type=AIModelType.SVM).delete()
|
|
new_default = Default(type=AIModelType.SVM, aimodel=m)
|
|
db.session.add(new_default)
|
|
db.session.commit()
|
|
|
|
return '', 204
|