更改推理结构

This commit is contained in:
2023-06-28 14:29:44 +08:00
parent 9bcd9540ef
commit 723eca3353
7 changed files with 211 additions and 384 deletions

View File

@@ -37,6 +37,7 @@ type Image struct {
Progress int `json:"progress"` // 任务进度(0-100)
Public bool `json:"public"` // 是否公开
UserID int `json:"user_id"` // 用户ID
ModelID int `json:"model_id"` // 模型ID
CreatedAt time.Time `json:"created_at" gorm:"autoCreateTime"`
UpdatedAt time.Time `json:"updated_at" gorm:"autoUpdateTime"`
}

View File

@@ -11,6 +11,7 @@ import (
"net/url"
"os"
"path/filepath"
"strconv"
"time"
"encoding/base64"
@@ -43,184 +44,112 @@ type Model struct {
UpdatedAt time.Time `json:"updated_at" gorm:"autoUpdateTime"`
}
// 创建一个带缓冲的通道,缓冲区大小为 10
// var ch = make(chan int, 10)
func init() {
configs.ORMDB().AutoMigrate(&Model{})
// 检查 images 目录是否存在, 不存在则创建
if _, err := os.Stat("data/images"); err != nil {
if err := os.MkdirAll("data/images", 0777); err != nil {
log.Println(err)
}
}
// 处理推理任务
//go func() {
// for {
// // 从通道中取出一个数据
// model := <-ch
// // 模型状态变化时, 向监听此模型的所有连接发送消息
// }
//}()
}
// 从数据库加载
func (model *Model) Load() {
configs.ORMDB().First(&model)
}
// 推理模型
func (model *Model) Inference(image_list []Image, callback func(Image)) {
var server Server
// 模型未部署到推理機
if model.ServerID == "" {
//log.Println("模型未部署到推理機, 开始部署模型")
log.Println("模型已部署到推理機, 开始推理模型")
log.Println("模型未部署到推理機, 开始部署模型")
var server Server
server.IP = "106.15.192.42"
server.Port = 7860
//if err := configs.ORMDB().Where("models LIKE ?", "%"+model.Name+"%").Take(&server).Error; err != nil {
// log.Println(err)
// // 如果没有则寻找空闲服务器
// // 如果没有空闲则创建新服务器
// // 取一台空闲的推理机上传并切换到此模型
// // 新建一台推理机上传并切换到此模型
//}
// 执行生成任务
if model.Image == "" {
img := image_list[0]
// 发送的参数
var datx map[string]interface{} = make(map[string]interface{})
datx["prompt"] = img.Prompt // 提示词
datx["seed"] = img.Seed // 随机数种子
datx["n_iter"] = len(image_list) // 生成图像数量
datx["steps"] = img.Steps // 迭代步数
datx["cfg_scale"] = img.CfgScale // 提示词引导系数 (CFG Scale)
if img.SamplerName == "" {
datx["sampler_name"] = img.SamplerName // 采样器名称
// 寻找一台就绪的推理机, 且已部署模型目标模型
if err := configs.ORMDB().Where("type = ?", "推理").Where("status = ?", "就绪").Where("models LIKE ?", "%"+strconv.Itoa(model.ID)+"%").First(&server).Error; err != nil {
// 寻找一台就绪的推理机, 且模型位置仍有空余
if err := configs.ORMDB().Where("type = ?", "推理").Where("status = ?", "空闲").Where("length(models) < ?", 5).First(&server).Error; err != nil {
log.Println("创建一台新的推理机: 当前禁止创建新服务器")
return
}
fmt.Println("image_list:", datx)
// 上传目标模型到推理机
log.Println("上传模型到推理机: 当前禁止上传模型")
return
}
var data = struct {
//EnableHr bool `json:"enable_hr"`
//DenoisingStrength int `json:"denoising_strength"`
//FirstphaseWidth int `json:"firstphase_width"`
//FirstphaseHeight int `json:"firstphase_height"`
//HrScale int `json:"hr_scale"`
//HrUpscaler string `json:"hr_upscaler"`
//HrSecondPassSteps int `json:"hr_second_pass_steps"`
//HrResizeX int `json:"hr_resize_x"`
//HrResizeY int `json:"hr_resize_y"`
//HrSamplerName string `json:"hr_sampler_name"`
//HrPrompt string `json:"hr_prompt"`
//HrNegativePrompt string `json:"hr_negative_prompt"`
Prompt string `json:"prompt"`
//Styles []string `json:"styles"`
Seed int `json:"seed"`
//Subseed int `json:"subseed"`
//SubseedStrength int `json:"subseed_strength"`
//SeedResizeFromH int `json:"seed_resize_from_h"`
//SeedResizeFromW int `json:"seed_resize_from_w"`
SamplerName string `json:"sampler_name"`
//BatchSize int `json:"batch_size"`
NIter int `json:"n_iter"`
Steps int `json:"steps"`
CfgScale int `json:"cfg_scale"`
//Width int `json:"width"`
//Height int `json:"height"`
//RestoreFaces bool `json:"restore_faces"`
//Tiling bool `json:"tiling"`
//DoNotSaveSamples bool `json:"do_not_save_samples"`
//DoNotSaveGrid bool `json:"do_not_save_grid"`
//NegativePrompt string `json:"negative_prompt"`
//Eta int `json:"eta"`
//SMinUncond int `json:"s_min_uncond"`
//SChurn int `json:"s_churn"`
//STmax int `json:"s_tmax"`
//STmin int `json:"s_tmin"`
//SNoise int `json:"s_noise"`
//OverrideSettings map[string]string `json:"override_settings"`
//OverrideSettingsRestoreAfterwards bool `json:"override_settings_restore_afterwards"`
//ScriptArgs []interface{} `json:"script_args"`
//SamplerIndex string `json:"sampler_index"`
//ScriptName string `json:"script_name"`
//SendImages bool `json:"send_images"`
//SaveImages bool `json:"save_images"`
//AlwaysonScripts map[string]string `json:"alwayson_scripts"`
}{
//EnableHr: false,
//DenoisingStrength: 0,
//FirstphaseWidth: 0,
//FirstphaseHeight: 0,
//HrScale: 2,
//HrUpscaler: "nearest",
//HrSecondPassSteps: 0,
//HrResizeX: 0,
//HrResizeY: 0,
//HrSamplerName: "",
//HrPrompt: "",
//HrNegativePrompt: "",
Prompt: image_list[0].Prompt,
//Styles: []string{},
Seed: image_list[0].Seed,
//Subseed: -1,
//SubseedStrength: 0,
//SeedResizeFromH: -1,
//SeedResizeFromW: -1,
SamplerName: image_list[0].SamplerName, // 采样器名称
//BatchSize: 1,
NIter: len(image_list), // 1~100
Steps: 50, // 1~150
CfgScale: image_list[0].CfgScale,
//Width: 512,
//Height: 512,
//RestoreFaces: false,
//Tiling: false,
//DoNotSaveSamples: false,
//DoNotSaveGrid: false,
//NegativePrompt: "",
//Eta: 0,
//SMinUncond: 0,
//SChurn: 0,
//STmax: 0,
//STmin: 0,
//SNoise: 1,
//OverrideSettings: map[string]string{},
//OverrideSettingsRestoreAfterwards: false,
//ScriptArgs: []interface{}{},
//SamplerIndex: "Euler",
//ScriptName: "generate",
//SendImages: true,
//SaveImages: false,
//AlwaysonScripts: map[string]string{},
}
fmt.Println("data:", data)
// 接收到的图片列表
var rest = struct {
Images []string `json:"images"`
}{}
var url = fmt.Sprintf("http://%s:%d/sdapi/v1/txt2img", server.IP, server.Port)
if err := goreq.Post(url).SetJsonBody(datx).Do().BindJSON(&rest); err != nil {
log.Println("API 查询失败:", err)
}
for index, img := range rest.Images {
var filename = fmt.Sprintf("%x", md5.Sum([]byte(img)))
log.Println("保存图片:", filename)
if err := SaveBase64Image(img, "data/images/"+filename+".webp"); err != nil {
log.Println(err)
var form = struct {
Components []struct {
ID int `json:"id"`
Type string `json:"type"`
Props struct {
Value string `json:"value"`
}
image_list[index].Name = filename
image_list[index].Path = "data/images/" + filename + ".webp"
image_list[index].Hash = filename
image_list[index].Type = "image/webp"
image_list[index].Width = 512
image_list[index].Height = 512
image_list[index].Format = "webp"
image_list[index].Status = "success"
image_list[index].Progress = 100
callback(image_list[index])
} `json:"components"`
}{}
// 检查当前是否为目标模型, 不是则执行切换模型 http://106.15.192.42:7860/config
if err := goreq.Get(fmt.Sprintf("http://%s:%d/config", server.IP, server.Port)).Do().BindJSON(&form); err != nil {
log.Println("获取推理机配置失败:", err)
return
}
var isSet = false
for _, component := range form.Components {
if component.Type == "dropdown" && component.ID == 1514 && component.Props.Value == model.Name {
log.Println("当前推理机已经部署了目标模型")
isSet = true
break
}
}
return
if !isSet {
log.Println("当前推理机未部署目标模型, 开始部署目标模型")
// 没有切换模型接口
return
}
// 记录到模型
model.ServerID = server.ID
configs.ORMDB().Save(&model)
}
// 发送的参数
var img = image_list[0]
var datx map[string]interface{} = make(map[string]interface{})
datx["prompt"] = img.Prompt // 提示词
datx["seed"] = img.Seed // 随机数种子
datx["n_iter"] = len(image_list) // 生成图像数量
datx["steps"] = img.Steps // 迭代步数
datx["cfg_scale"] = img.CfgScale // 提示词引导系数 (CFG Scale)
if img.SamplerName == "" {
datx["sampler_name"] = img.SamplerName // 采样器名称
}
fmt.Println("image_list:", datx)
// 接收到的图片列表
var rest = struct {
Images []string `json:"images"`
}{}
var url = fmt.Sprintf("http://%s:%d/sdapi/v1/txt2img", server.IP, server.Port)
if err := goreq.Post(url).SetJsonBody(datx).Do().BindJSON(&rest); err != nil {
log.Println("API 查询失败:", err)
}
for index, img := range rest.Images {
var filename = fmt.Sprintf("%x", md5.Sum([]byte(img)))
log.Println("保存图片:", filename)
if err := SaveBase64Image(img, "data/images/"+filename+".webp"); err != nil {
log.Println(err)
}
image_list[index].Name = filename
image_list[index].Path = "data/images/" + filename + ".webp"
image_list[index].Hash = filename
image_list[index].Type = "image/webp"
image_list[index].Width = 512
image_list[index].Height = 512
image_list[index].Format = "webp"
image_list[index].Status = "success"
image_list[index].Progress = 100
callback(image_list[index])
}
log.Println("模型未部署到推理機, 取消推理模型")
}
@@ -254,6 +183,7 @@ func SaveBase64Image(base64Str string, filename string) error {
return nil
}
// 训练模型
func (model *Model) Train() (err error) {
// 獲取一臺空閒的訓練機
@@ -420,3 +350,101 @@ func (model *Model) Train() (err error) {
return nil
}
/**
var data = struct {
//EnableHr bool `json:"enable_hr"`
//DenoisingStrength int `json:"denoising_strength"`
//FirstphaseWidth int `json:"firstphase_width"`
//FirstphaseHeight int `json:"firstphase_height"`
//HrScale int `json:"hr_scale"`
//HrUpscaler string `json:"hr_upscaler"`
//HrSecondPassSteps int `json:"hr_second_pass_steps"`
//HrResizeX int `json:"hr_resize_x"`
//HrResizeY int `json:"hr_resize_y"`
//HrSamplerName string `json:"hr_sampler_name"`
//HrPrompt string `json:"hr_prompt"`
//HrNegativePrompt string `json:"hr_negative_prompt"`
Prompt string `json:"prompt"`
//Styles []string `json:"styles"`
Seed int `json:"seed"`
//Subseed int `json:"subseed"`
//SubseedStrength int `json:"subseed_strength"`
//SeedResizeFromH int `json:"seed_resize_from_h"`
//SeedResizeFromW int `json:"seed_resize_from_w"`
SamplerName string `json:"sampler_name"`
//BatchSize int `json:"batch_size"`
NIter int `json:"n_iter"`
Steps int `json:"steps"`
CfgScale int `json:"cfg_scale"`
//Width int `json:"width"`
//Height int `json:"height"`
//RestoreFaces bool `json:"restore_faces"`
//Tiling bool `json:"tiling"`
//DoNotSaveSamples bool `json:"do_not_save_samples"`
//DoNotSaveGrid bool `json:"do_not_save_grid"`
//NegativePrompt string `json:"negative_prompt"`
//Eta int `json:"eta"`
//SMinUncond int `json:"s_min_uncond"`
//SChurn int `json:"s_churn"`
//STmax int `json:"s_tmax"`
//STmin int `json:"s_tmin"`
//SNoise int `json:"s_noise"`
//OverrideSettings map[string]string `json:"override_settings"`
//OverrideSettingsRestoreAfterwards bool `json:"override_settings_restore_afterwards"`
//ScriptArgs []interface{} `json:"script_args"`
//SamplerIndex string `json:"sampler_index"`
//ScriptName string `json:"script_name"`
//SendImages bool `json:"send_images"`
//SaveImages bool `json:"save_images"`
//AlwaysonScripts map[string]string `json:"alwayson_scripts"`
}{
//EnableHr: false,
//DenoisingStrength: 0,
//FirstphaseWidth: 0,
//FirstphaseHeight: 0,
//HrScale: 2,
//HrUpscaler: "nearest",
//HrSecondPassSteps: 0,
//HrResizeX: 0,
//HrResizeY: 0,
//HrSamplerName: "",
//HrPrompt: "",
//HrNegativePrompt: "",
Prompt: image_list[0].Prompt,
//Styles: []string{},
Seed: image_list[0].Seed,
//Subseed: -1,
//SubseedStrength: 0,
//SeedResizeFromH: -1,
//SeedResizeFromW: -1,
SamplerName: image_list[0].SamplerName, // 采样器名称
//BatchSize: 1,
NIter: len(image_list), // 1~100
Steps: 50, // 1~150
CfgScale: image_list[0].CfgScale,
//Width: 512,
//Height: 512,
//RestoreFaces: false,
//Tiling: false,
//DoNotSaveSamples: false,
//DoNotSaveGrid: false,
//NegativePrompt: "",
//Eta: 0,
//SMinUncond: 0,
//SChurn: 0,
//STmax: 0,
//STmin: 0,
//SNoise: 1,
//OverrideSettings: map[string]string{},
//OverrideSettingsRestoreAfterwards: false,
//ScriptArgs: []interface{}{},
//SamplerIndex: "Euler",
//ScriptName: "generate",
//SendImages: true,
//SaveImages: false,
//AlwaysonScripts: map[string]string{},
}
fmt.Println("data:", data)
**/

View File

@@ -1,82 +0,0 @@
package models
import "time"
type Task struct {
ID int `json:"id" gorm:"primary_key"`
Name string `json:"name"`
Type string `json:"type"` // 任務類型(訓練|推理)
Status string `json:"status"` // (initial|ready|waiting|running|success|error)
Progress int `json:"progress"` // (0-100)
UserID int `json:"user_id"`
CreatedAt time.Time `json:"created_at" gorm:"autoCreateTime"`
UpdatedAt time.Time `json:"updated_at" gorm:"autoUpdateTime"`
}
// 异步任务调度管理
type AsynchronousTaskSchedulingManagement struct {
tasks map[int]Task // 任务队列
servers map[string]Server // 服务器队列
}
// 向任务队列添加任务
func (m *AsynchronousTaskSchedulingManagement) AddTask(task Task) {
// 1. 任务加入队列, 任务状态为 waiting, 每次从最后一个任务开始执行, 并向上全取同模型的任务, 任务状态更新为 waiting(排队), 并更新此模型的排队状态到所有关注此模型的用户(管理员每个连接都关注所有模型)
// 2. 任务从队列中取出, 任务队列长度超过12则增加机器, 模型类型持续占用机器则增加机器
// 加入任务队列
m.tasks[task.ID] = task
// 检查任务队列长度, 持续增长超过12则增加机器
if len(m.tasks) > 12 {
var server Server
m.servers[server.ID] = server
}
// 向目标机器发送模型
// 向目标机器切换模型
// 向目标机器发送任务
}
//// 推理任務
//func startInferenceTask(task *Task) {
//
// // 獲取一臺可用的 GPU 資源
// // ...
//
// // 執行推理任務
// // ...
//
// // 更新任務狀態
// task.Status = "running"
// task.Progress = 0
// task.Update()
//
// // 監聽任務狀態
// for {
// // 延遲 1 秒
// time.Sleep(1 * time.Second)
//
// // 查詢任務狀態
// resp, err := http.Get("http://localhost:5000/api/v1/tasks/" + strconv.Itoa(task.ID))
// if err != nil {
// log.Println(err)
// continue
// }
// defer resp.Body.Close()
//
// // 解析任務狀態
// // ...
//
// // 更新任務狀態
// task.Progress = 100
// task.Status = "success"
// task.Update()
//
// // 任務結束判定
// if task.Progress == 100 {
// break
// }
// }
//
//}

View File

@@ -51,6 +51,13 @@ var config = struct {
}{}
func init() {
configs.ORMDB().AutoMigrate(&Server{})
// 檢查所有服務器的狀態, 無效的服務器設置為異常
var servers []Server
configs.ORMDB().Find(&servers)
for _, server := range servers {
server.CheckStatus()
}
// 讀取配置文件
absPath, _ := filepath.Abs("./data/config.yaml")
configFile, err := ioutil.ReadFile(absPath)
@@ -60,10 +67,23 @@ func init() {
if err := yaml.Unmarshal(configFile, &config); err != nil {
panic(fmt.Errorf("格式化配置文件失敗: %v", err))
}
// 初始化检查默认服务器
if err := InitDefaultServer(); err != nil {
panic(fmt.Errorf("初始化默认服务器失败: %v", err))
}
}
// 检查默认服务器是否存在, 不存在则添加
func InitDefaultServer() (err error) {
if err = configs.ORMDB().Where("id = ?", "default").First(&Server{}).Error; err != nil {
server := Server{ID: "default", IP: "106.15.192.42", Port: 7860, Type: "推理", Status: "就绪"}
err = configs.ORMDB().Create(&server).Error
}
return
}
// 创建一台新服务器
func NewServer(server_type string) (server *Server, err error) {
func NewServer(server_type string) (server Server, err error) {
// 调用 API 创建一台新服务器(通過腾讯云API創建服務器)
client, err := cvm.NewClient(common.NewCredential(config.TencentCloud.SecretId, config.TencentCloud.SecretKey), config.TencentCloud.Region, profile.NewClientProfile())
if err != nil {
@@ -84,7 +104,7 @@ func NewServer(server_type string) (server *Server, err error) {
fmt.Println("創建服務器成功:", response.Response.InstanceIdSet[0])
// 获取服务器信息
var get_server_info = func(InstanceIdSet *string) (server *Server, err error) {
var get_server_info = func(InstanceIdSet *string) (server Server, err error) {
response2, err := client.DescribeInstances(cvm.NewDescribeInstancesRequest())
if err != nil {
return server, fmt.Errorf("獲取實例詳情失敗: %v", err)
@@ -169,13 +189,3 @@ func (server *Server) CheckStatus() error {
// 檢查服務器是否正常
return nil
}
func init() {
configs.ORMDB().AutoMigrate(&Server{})
// 檢查所有服務器的狀態, 無效的服務器設置為異常
var servers []Server
configs.ORMDB().Find(&servers)
for _, server := range servers {
server.CheckStatus()
}
}