|
@@ -4,8 +4,10 @@ import (
|
|
|
"encoding/json"
|
|
|
"eta/eta_api/models"
|
|
|
"eta/eta_api/models/ai_summary"
|
|
|
+ "eta/eta_api/models/aimod"
|
|
|
"eta/eta_api/models/sandbox"
|
|
|
aiSummaryService "eta/eta_api/services/ai_summary"
|
|
|
+ "eta/eta_api/services/aiser"
|
|
|
"eta/eta_api/utils"
|
|
|
"fmt"
|
|
|
"github.com/rdlucklib/rdluck_tools/paging"
|
|
@@ -38,7 +40,6 @@ func (this *AiController) AiSummaryClassifyItems() {
|
|
|
return
|
|
|
}
|
|
|
|
|
|
-
|
|
|
br.Ret = 200
|
|
|
br.Success = true
|
|
|
br.Msg = "获取成功"
|
|
@@ -86,7 +87,7 @@ func (this *AiController) AiSummaryClassifyItems() {
|
|
|
|
|
|
|
|
|
|
|
|
-func (this *AiController) AddSandboxClassify() {
|
|
|
+func (this *AiController) AddAiSummaryClassify() {
|
|
|
br := new(models.BaseResponse).Init()
|
|
|
defer func() {
|
|
|
this.Data["json"] = br
|
|
@@ -183,7 +184,6 @@ func (this *AiController) EditAiSummaryClassify() {
|
|
|
return
|
|
|
}
|
|
|
|
|
|
-
|
|
|
br.Ret = 200
|
|
|
br.Msg = "保存成功"
|
|
|
br.Success = true
|
|
@@ -315,7 +315,7 @@ func (this *AiController) DeleteAiSummaryClassify() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-
|
|
|
+
|
|
|
if req.AiSummaryId > 0 {
|
|
|
summaryInfo, err := ai_summary.GetAiSummaryById(req.AiSummaryId)
|
|
|
if err != nil {
|
|
@@ -333,6 +333,10 @@ func (this *AiController) DeleteAiSummaryClassify() {
|
|
|
br.Msg = "纪要已删除,请刷新页面"
|
|
|
return
|
|
|
}
|
|
|
+ if summaryInfo.SysUserId != sysUser.AdminId {
|
|
|
+ br.Msg = "仅纪要创建人可以删除"
|
|
|
+ return
|
|
|
+ }
|
|
|
err = ai_summary.DelAiSummaryById(req.AiSummaryId)
|
|
|
if err != nil {
|
|
|
br.Msg = err.Error()
|
|
@@ -898,3 +902,578 @@ func (this *AiController) AiSummaryList() {
|
|
|
br.Msg = "获取成功"
|
|
|
br.Data = resp
|
|
|
}
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+func (this *AiController) AiSummaryDetail() {
|
|
|
+ br := new(models.BaseResponse).Init()
|
|
|
+ defer func() {
|
|
|
+ this.Data["json"] = br
|
|
|
+ this.ServeJSON()
|
|
|
+ }()
|
|
|
+ sysUser := this.SysUser
|
|
|
+ if sysUser == nil {
|
|
|
+ br.Msg = "请登录"
|
|
|
+ br.ErrMsg = "请登录,SysUser Is Empty"
|
|
|
+ br.Ret = 408
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ aiSummaryId, _ := this.GetInt("AiSummaryId")
|
|
|
+
|
|
|
+ detail, err := ai_summary.GetAiSummaryById(aiSummaryId)
|
|
|
+ if err != nil && err.Error() != utils.ErrNoRow() {
|
|
|
+ br.Success = true
|
|
|
+ br.Msg = "获取沙盘信息失败"
|
|
|
+ br.ErrMsg = "获取沙盘信息失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ br.Ret = 200
|
|
|
+ br.Success = true
|
|
|
+ br.Msg = "获取成功"
|
|
|
+ br.Data = detail
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+func (this *AiController) AddAiSummary() {
|
|
|
+ br := new(models.BaseResponse).Init()
|
|
|
+ defer func() {
|
|
|
+ this.Data["json"] = br
|
|
|
+ this.ServeJSON()
|
|
|
+ }()
|
|
|
+ var req ai_summary.AddAiSummaryReq
|
|
|
+ err := json.Unmarshal(this.Ctx.Input.RequestBody, &req)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "参数解析异常!"
|
|
|
+ br.ErrMsg = "参数解析失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if req.Title == "" {
|
|
|
+ br.Msg = "请输入名称"
|
|
|
+ br.IsSendEmail = false
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if req.SummaryContent == "" {
|
|
|
+ br.Msg = "纪要内容为空"
|
|
|
+ br.IsSendEmail = false
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if req.ClassifyId < 0 {
|
|
|
+ br.Msg = "分类id错误"
|
|
|
+ br.IsSendEmail = false
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ maxSort, err := ai_summary.GetAiSummaryMaxSort(req.ClassifyId)
|
|
|
+
|
|
|
+ summary := &ai_summary.AiSummary{
|
|
|
+ SaDocId: req.SaDocId,
|
|
|
+ OriginContent: req.OriginContent,
|
|
|
+ ClassifyId: req.ClassifyId,
|
|
|
+ SysUserId: this.SysUser.AdminId,
|
|
|
+ SysUserRealName: this.SysUser.RealName,
|
|
|
+ Title: req.Title,
|
|
|
+ CreateTime: time.Now(),
|
|
|
+ ModifyTime: time.Now(),
|
|
|
+ OpenaiFileName: req.OpenaiFileName,
|
|
|
+ OpenaiFilePath: req.OpenaiFilePath,
|
|
|
+ Sort: maxSort + 1,
|
|
|
+ }
|
|
|
+
|
|
|
+ _, err = ai_summary.AddAiSummary(summary)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "保存分类失败"
|
|
|
+ br.ErrMsg = "保存分类失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ br.Ret = 200
|
|
|
+ br.Msg = "保存成功"
|
|
|
+ br.Success = true
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+func (this *AiController) GenerateAiSummary() {
|
|
|
+ br := new(models.BaseResponse).Init()
|
|
|
+ defer func() {
|
|
|
+ this.Data["json"] = br
|
|
|
+ this.ServeJSON()
|
|
|
+ }()
|
|
|
+ var req ai_summary.GenerateAiSummaryReq
|
|
|
+ err := json.Unmarshal(this.Ctx.Input.RequestBody, &req)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "参数解析异常!"
|
|
|
+ br.ErrMsg = "参数解析失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if req.Ask == "" {
|
|
|
+ br.Msg = "请输入问题"
|
|
|
+ br.IsSendEmail = false
|
|
|
+ return
|
|
|
+ }
|
|
|
+ if req.OriginContent == "" && len(req.OpenaiFileId) == 0 {
|
|
|
+ br.Msg = "参数错误"
|
|
|
+ br.IsSendEmail = false
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ if utils.Re == nil {
|
|
|
+ key := "CACHE_CHAT_" + strconv.Itoa(this.SysUser.AdminId)
|
|
|
+ cacheVal, err := utils.Rc.RedisInt(key)
|
|
|
+ fmt.Println("RedisString:", cacheVal, "err:", err)
|
|
|
+ if err != nil && (err.Error() != "redigo: nil returned" && err.Error() != "redis: nil") {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "获取数据失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ putVal := 0
|
|
|
+ if cacheVal <= 0 {
|
|
|
+ putVal = utils.AiChatLimit
|
|
|
+ } else {
|
|
|
+ putVal = cacheVal - 1
|
|
|
+ }
|
|
|
+
|
|
|
+ if putVal <= 0 {
|
|
|
+ br.Msg = "您今日" + strconv.Itoa(utils.AiChatLimit) + "次问答已达上限,请明天再来!"
|
|
|
+ br.ErrMsg = "您今日" + strconv.Itoa(utils.AiChatLimit) + "次问答已达上限,请明天再来!"
|
|
|
+ return
|
|
|
+ }
|
|
|
+ lastSecond := utils.GetTodayLastSecond()
|
|
|
+ utils.Rc.Put(key, putVal, lastSecond)
|
|
|
+ }
|
|
|
+
|
|
|
+ resp := new(aimod.ChatResp)
|
|
|
+
|
|
|
+ if req.OriginContent != "" {
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ req.Ask = req.Prompt + req.OriginContent
|
|
|
+
|
|
|
+ var answer string
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ historyList, err := aimod.GetAiChatList(req.AiChatTopicId)
|
|
|
+ if err != nil && err.Error() != utils.ErrNoRow() {
|
|
|
+ br.Msg = "获取主题历史数据失败!"
|
|
|
+ br.ErrMsg = "获取主题历史数据失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ historyChatList := make([]aimod.HistoryChat, 0)
|
|
|
+ for _, v := range historyList {
|
|
|
+ historyChat := new(aimod.HistoryChat)
|
|
|
+ historyChat.Ask = v.Ask
|
|
|
+ historyChat.Answer = v.Answer
|
|
|
+ historyChatList = append(historyChatList, *historyChat)
|
|
|
+ }
|
|
|
+
|
|
|
+ answer, err = aiser.ChatAutoMsg(req.Ask, historyChatList, req.Model)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "获取数据失败,ChatAutoMsg,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ resp.Ask = req.Ask
|
|
|
+ resp.Answer = answer
|
|
|
+ resp.Model = req.Model
|
|
|
+
|
|
|
+ topic := new(aimod.AiChatTopic)
|
|
|
+ topic.TopicName = req.Ask
|
|
|
+ topic.SysUserId = this.SysUser.AdminId
|
|
|
+ topic.SysUserRealName = this.SysUser.RealName
|
|
|
+ topic.CreateTime = time.Now()
|
|
|
+ topic.ModifyTime = time.Now()
|
|
|
+ topicId, err := aimod.AddAiChatTopic(topic)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "生成话题失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ resp.AiChatTopicId = int(topicId)
|
|
|
+ chatItem := new(aimod.AiChat)
|
|
|
+ chatItem.AiChatTopicId = resp.AiChatTopicId
|
|
|
+ chatItem.Ask = req.Ask
|
|
|
+ chatItem.AskUuid = utils.MD5(req.Ask)
|
|
|
+ chatItem.Answer = answer
|
|
|
+ chatItem.Model = EnabledModelsForMap[req.Model]
|
|
|
+ chatItem.SysUserId = this.SysUser.AdminId
|
|
|
+ chatItem.SysUserRealName = this.SysUser.RealName
|
|
|
+ chatItem.CreateTime = time.Now()
|
|
|
+ chatItem.ModifyTime = time.Now()
|
|
|
+ _, err = aimod.AddAiChat(chatItem)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "生成话题记录失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ } else if len(req.OpenaiFileId) > 0 {
|
|
|
+
|
|
|
+
|
|
|
+ askUuid := utils.MD5(req.Ask)
|
|
|
+
|
|
|
+ var assistantId, threadId string
|
|
|
+ if req.AiChatTopicId > 0 {
|
|
|
+ aiChatTopicObj := new(aimod.AiChatTopic)
|
|
|
+ aiChatTopicObj.AiChatTopicId = req.AiChatTopicId
|
|
|
+ topic, err := aiChatTopicObj.GetAiChatTopicById()
|
|
|
+ if err != nil {
|
|
|
+ if err.Error() == utils.ErrNoRow() {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "获取数据失败,主题不存在,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "获取数据失败,GetAiChatTopicById,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ assistantId = topic.AssistantId
|
|
|
+ threadId = topic.ThreadId
|
|
|
+ }
|
|
|
+
|
|
|
+ var answer string
|
|
|
+
|
|
|
+
|
|
|
+ historyList, err := aimod.GetAiChatList(req.AiChatTopicId)
|
|
|
+ if err != nil && err.Error() != utils.ErrNoRow() {
|
|
|
+ br.Msg = "获取主题历史数据失败!"
|
|
|
+ br.ErrMsg = "获取主题历史数据失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ frList := make([]aimod.HistoryChat, 0)
|
|
|
+ tmpFileIdList := make([]string, 0)
|
|
|
+
|
|
|
+
|
|
|
+ for _, v := range historyList {
|
|
|
+ if v.OpenaiFileId != "" {
|
|
|
+ tmpFileIdList = append(tmpFileIdList, v.OpenaiFileId)
|
|
|
+ } else {
|
|
|
+ historyFr := new(aimod.HistoryChat)
|
|
|
+ historyFr.Ask = v.Ask
|
|
|
+ historyFr.Answer = v.Answer
|
|
|
+ historyFr.OpenaiFileId = tmpFileIdList
|
|
|
+ frList = append(frList, *historyFr)
|
|
|
+ tmpFileIdList = []string{}
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ {
|
|
|
+ frItem := new(aimod.HistoryChat)
|
|
|
+ frItem.Ask = req.Ask
|
|
|
+ frItem.Answer = ""
|
|
|
+ frItem.OpenaiFileId = tmpFileIdList
|
|
|
+ frList = append(frList, *frItem)
|
|
|
+ }
|
|
|
+
|
|
|
+ fileRetrieveResp, err := aiser.FileRetrieve(assistantId, threadId, req.Model, frList, req.OpenaiFileId)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "获取数据失败,FileRetrieve,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ if fileRetrieveResp != nil {
|
|
|
+ if fileRetrieveResp.Ret == 200 {
|
|
|
+ assistantId = fileRetrieveResp.Data.AssistantId
|
|
|
+ threadId = fileRetrieveResp.Data.ThreadId
|
|
|
+ answer = fileRetrieveResp.Data.Answer
|
|
|
+ } else {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = fileRetrieveResp.Msg
|
|
|
+ return
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if req.AiChatTopicId <= 0 {
|
|
|
+ topic := new(aimod.AiChatTopic)
|
|
|
+ topic.TopicName = req.Ask
|
|
|
+ topic.SysUserId = this.SysUser.AdminId
|
|
|
+ topic.SysUserRealName = this.SysUser.RealName
|
|
|
+ topic.CreateTime = time.Now()
|
|
|
+ topic.ModifyTime = time.Now()
|
|
|
+ topic.AssistantId = assistantId
|
|
|
+ topic.ThreadId = threadId
|
|
|
+ topicId, err := aimod.AddAiChatTopic(topic)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "生成话题失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ chatItem := new(aimod.AiChat)
|
|
|
+ chatItem.AiChatTopicId = int(topicId)
|
|
|
+ chatItem.Ask = req.Ask
|
|
|
+ chatItem.AskUuid = utils.MD5(req.Ask)
|
|
|
+ chatItem.Answer = answer
|
|
|
+ chatItem.Model = EnabledModelsForMap[req.Model]
|
|
|
+ chatItem.SysUserId = this.SysUser.AdminId
|
|
|
+ chatItem.SysUserRealName = this.SysUser.RealName
|
|
|
+ chatItem.CreateTime = time.Now()
|
|
|
+ chatItem.ModifyTime = time.Now()
|
|
|
+ _, err = aimod.AddAiChat(chatItem)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "生成话题记录失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ req.AiChatTopicId = int(topicId)
|
|
|
+ } else {
|
|
|
+ chatItem := new(aimod.AiChat)
|
|
|
+ chatItem.AiChatTopicId = req.AiChatTopicId
|
|
|
+ chatItem.Ask = req.Ask
|
|
|
+ chatItem.AskUuid = askUuid
|
|
|
+ chatItem.Answer = answer
|
|
|
+ chatItem.Model = EnabledModelsForMap[req.Model]
|
|
|
+ chatItem.SysUserId = this.SysUser.AdminId
|
|
|
+ chatItem.SysUserRealName = this.SysUser.RealName
|
|
|
+ chatItem.CreateTime = time.Now()
|
|
|
+ chatItem.ModifyTime = time.Now()
|
|
|
+ _, err = aimod.AddAiChat(chatItem)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取数据失败!"
|
|
|
+ br.ErrMsg = "生成话题记录失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+
|
|
|
+ aiChatTopicObj := new(aimod.AiChatTopic)
|
|
|
+
|
|
|
+ updateParams := make(map[string]interface{})
|
|
|
+ updateParams["assistant_id"] = assistantId
|
|
|
+ updateParams["thread_id"] = threadId
|
|
|
+ updateParams["modify_time"] = time.Now()
|
|
|
+
|
|
|
+ whereParam := make(map[string]interface{})
|
|
|
+ whereParam["ai_chat_topic_id"] = req.AiChatTopicId
|
|
|
+
|
|
|
+ err = aiChatTopicObj.Update(updateParams, whereParam)
|
|
|
+ if err != nil {
|
|
|
+ br.Msg = "获取失败!"
|
|
|
+ br.ErrMsg = "修改助手标识失败,Err:" + err.Error()
|
|
|
+ return
|
|
|
+ }
|
|
|
+ }
|
|
|
+ resp.Model = aimod.ModelViewMap[req.Model]
|
|
|
+ resp.AiChatTopicId = req.AiChatTopicId
|
|
|
+ resp.Ask = req.Ask
|
|
|
+ resp.Answer = answer
|
|
|
+ }
|
|
|
+
|
|
|
+ br.Ret = 200
|
|
|
+ br.Success = true
|
|
|
+ br.Msg = "获取成功"
|
|
|
+ br.Data = resp
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+
|