from flask import render_template,redirect from flask_appbuilder.models.sqla.interface import SQLAInterface from flask import Blueprint, current_app, jsonify, make_response, request # 将model添加成视图,并控制在前端的显示 from myapp.models.model_serving import Service from myapp.models.model_train_model import Training_Model from myapp.models.model_serving import InferenceService from myapp.models.model_team import Project,Project_User from myapp.utils import core from flask_babel import gettext as __ from flask_babel import lazy_gettext as _ from flask_appbuilder.actions import action from myapp import app, appbuilder,db,event_logger import logging import re import uuid from myapp.views.view_team import Project_Filter,Project_Join_Filter,filter_join_org_project import requests from myapp.exceptions import MyappException from flask_appbuilder.security.decorators import has_access from myapp.models.model_job import Repository,Pipeline from myapp.project import push_message,push_admin from flask_wtf.file import FileAllowed, FileField, FileRequired from werkzeug.datastructures import FileStorage from wtforms.ext.sqlalchemy.fields import QuerySelectField from myapp import security_manager import os,sys from wtforms.validators import DataRequired, Length, NumberRange, Optional,Regexp from wtforms import BooleanField, IntegerField, SelectField, StringField,FloatField,DateField,DateTimeField,SelectMultipleField,FormField,FieldList from flask_appbuilder.fieldwidgets import BS3TextFieldWidget,BS3PasswordFieldWidget,DatePickerWidget,DateTimePickerWidget,Select2ManyWidget,Select2Widget from myapp.forms import MyBS3TextAreaFieldWidget,MySelect2Widget,MyCodeArea,MyLineSeparatedListField,MyJSONField,MyBS3TextFieldWidget,MySelectMultipleField from myapp.utils.py import py_k8s import os, zipfile import shutil from flask import ( current_app, abort, flash, g, Markup, make_response, redirect, render_template, request, send_from_directory, Response, url_for, ) from .base import ( DeleteMixin, api, BaseMyappView, check_ownership, data_payload_response, DeleteMixin, generate_download_headers, get_error_msg, get_user_roles, handle_api_exception, json_error_response, json_success, MyappFilter, MyappModelView, json_response ) from sqlalchemy import and_, or_, select from .baseApi import ( MyappModelRestApi ) from flask_appbuilder import CompactCRUDMixin, expose import pysnooper,datetime,time,json conf = app.config class Training_Model_Filter(MyappFilter): # @pysnooper.snoop() def apply(self, query, func): user_roles = [role.name.lower() for role in list(self.get_user_roles())] if "admin" in user_roles: return query return query.filter(self.model.created_by_fk == g.user.id) # 定义数据库视图 # class Training_Model_ModelView(JsonResModelView,DeleteMixin): class Training_Model_ModelView_Base(): datamodel = SQLAInterface(Training_Model) base_permissions = ['can_add', 'can_edit', 'can_delete', 'can_list', 'can_show'] # 默认为这些 base_order = ('changed_on', 'desc') order_columns = ['id'] list_columns = ['project_url','name','version','framework','api_type','pipeline_url','creator','modified','deploy'] add_columns = ['project','name','version','describe','path','framework','run_id','run_time','metrics','md5','api_type','pipeline_id'] edit_columns = add_columns add_form_query_rel_fields = { "project": [["name", Project_Join_Filter, 'org']] } edit_form_query_rel_fields = add_form_query_rel_fields cols_width={ "name":{"type": "ellip2", "width": 250}, "project_url": {"type": "ellip2", "width": 200}, "pipeline_url":{"type": "ellip2", "width": 300}, "version": {"type": "ellip2", "width": 200}, "modified": {"type": "ellip2", "width": 150}, "deploy": {"type": "ellip2", "width": 100}, } spec_label_columns = { "path": "模型文件", "framework":"算法框架", "api_type":"推理框架", "pipeline_id":"任务流id" } label_title = '模型' base_filters = [["id", Training_Model_Filter, lambda: []]] # 设置权限过滤器 path_describe= r''' tfserving:仅支持tf save_model方式的模型目录, /mnt/xx/../saved_model/
torch-server:torch-model-archiver编译后的mar模型文件地址, /mnt/xx/../xx.mar或torch script保存的模型
onnxruntime:onnx模型文件的地址, /mnt/xx/../xx.onnx
tensorrt:模型文件地址, /mnt/xx/../xx.plan
''' service_type_choices= [x.replace('_','-') for x in ['tfserving','torch-server','onnxruntime','triton-server']] add_form_extra_fields={ "path": StringField( _('模型文件地址'), default='/mnt/admin/xx/saved_model/', description=_(path_describe), validators=[DataRequired()] ), "describe": StringField( _(datamodel.obj.lab('describe')), description=_('模型描述'), validators=[DataRequired()] ), "pipeline_id": StringField( _(datamodel.obj.lab('pipeline_id')), description=_('任务流的id,0表示非任务流产生模型'), default='0' ), "version": StringField( _('版本'), widget=MyBS3TextFieldWidget(), description='模型版本', default=datetime.datetime.now().strftime('v%Y.%m.%d.1'), validators=[DataRequired()] ), "run_id":StringField( _(datamodel.obj.lab('run_id')), widget=MyBS3TextFieldWidget(), description='pipeline 训练的run id', default='random_run_id_'+uuid.uuid4().hex[:32] ), "run_time": StringField( _(datamodel.obj.lab('run_time')), widget=MyBS3TextFieldWidget(), description='pipeline 训练的 运行时间', default=datetime.datetime.now().strftime('%Y.%m.%d %H:%M:%S'), ), "name":StringField( _("模型名"), widget=MyBS3TextFieldWidget(), description='模型名(a-z0-9-字符组成,最长54个字符)', validators = [DataRequired(),Regexp("^[a-z0-9\-]*$"),Length(1,54)] ), "framework": SelectField( _('算法框架'), description="选项xgb、tf、pytorch、onnx、tensorrt等", widget=Select2Widget(), choices=[['xgb', 'xgb'],['tf', 'tf'], ['pytorch', 'pytorch'],['onnx','onnx'],['tensorrt','tensorrt']], validators=[DataRequired()] ), 'api_type': SelectField( _("部署类型"), description="推理框架类型", choices=[[x, x] for x in service_type_choices], validators=[DataRequired()] ) } edit_form_extra_fields=add_form_extra_fields # edit_form_extra_fields['path']=FileField( # _('模型压缩文件'), # description=_(path_describe), # validators=[ # FileAllowed(["zip",'tar.gz'],_("zip/tar.gz Files Only!")), # ] # ) # @pysnooper.snoop(watch_explode=('item')) def pre_add(self,item): if not item.run_id: item.run_id='random_run_id_'+uuid.uuid4().hex[:32] def pre_update(self,item): if not item.path: item.path=self.src_item_json['path'] self.pre_add(item) @expose("/deploy/", methods=["GET",'POST']) def deploy(self,model_id): train_model = db.session.query(Training_Model).filter_by(id=model_id).first() exist_inference = db.session.query(InferenceService).filter_by(model_name=train_model.name).filter_by(model_version=train_model.version).first() from myapp.views.view_inferenceserving import InferenceService_ModelView_base inference_class = InferenceService_ModelView_base() inference_class.src_item_json={} if not exist_inference: exist_inference = InferenceService() exist_inference.project_id=train_model.project_id exist_inference.project = train_model.project exist_inference.model_name=train_model.name exist_inference.label = train_model.describe exist_inference.model_version=train_model.version exist_inference.model_path=train_model.path exist_inference.service_type=train_model.api_type exist_inference.images='' exist_inference.name='%s-%s-%s'%(exist_inference.service_type,train_model.name,train_model.version.replace('v','').replace('.','')) inference_class.pre_add(exist_inference) db.session.add(exist_inference) db.session.commit() flash('新服务版本创建完成','success') else: flash('服务版本已存在', 'success') import urllib.parse url = conf.get('MODEL_URLS',{}).get('inferenceservice','')+'?filter='+urllib.parse.quote(json.dumps([{"key":"model_name","value":exist_inference.model_name}],ensure_ascii=False)) print(url) return redirect(url) class Training_Model_ModelView(Training_Model_ModelView_Base,MyappModelView,DeleteMixin): datamodel = SQLAInterface(Training_Model) # 添加视图和菜单 appbuilder.add_view(Training_Model_ModelView,"模型管理",icon = 'fa-hdd-o',category = '服务化',category_icon = 'fa-tasks') # 添加api class Training_Model_ModelView_Api(Training_Model_ModelView_Base,MyappModelRestApi): # noqa datamodel = SQLAInterface(Training_Model) # base_order = ('id', 'desc') route_base = '/training_model_modelview/api' appbuilder.add_api(Training_Model_ModelView_Api)