使用go语言封装调用第三方接口的示例与AI算法模型在集群部署思考

使用go语言封装调用第三方接口的示例

package service

import (
	"xxx/src/config"
	"xxx/src/models"
	"encoding/json"
	"fmt"
	"io/ioutil"
	"net/http"
	"net/url"
	"strconv"
	"strings"
)

//创建数字人API接口时候返回的响应体
type HuaSoulEchoRespones struct {
	Code   int                    `from:"code" json:"code"`
	Status map[string]interface{} `from:"status" json:"status"`
	ISLIU  string                 `from:"ISLIU" json:"ISLIU"`
}

func (client *Client) HuaSoulEchoAPI(msg *SendMsg) (int, map[string]interface{}, string, error) {

	/*
	   创建一个请求参数对象
	*/

	ChatReustParam := models.EchoChatParm{
		Auth:    "1d527cbf2cd50ad7e97f0aa23xxxxxxx",
		Type:    msg.Type,
		IsliuId: client.conversation.IsliuId,
		Content: msg.Content,
	}

	fmt.Println(ChatReustParam)
	request_url := config.Conf.SoulUrl + "/soul/echo"
	fmt.Println("Echo request_url", request_url)
	
	// 准备POST请求的body数据,这里使用bytes.Buffer来构建一个简单的JSON格式数据作为示例
	//requestBody := []byte(`{"key1": "value1", "key2": "value2"}`)
	/*
		requestBody, err := json.Marshal(ChatReustParam)
		if err != nil {
			fmt.Println("Failed to marshal request body:", err)
			return -1, "请求参数体,解析失败", "解析chatParam参数失败,无法获取意识流"
		}
	*/
	postData := url.Values{}
	postData.Add("auth", ChatReustParam.Auth)
	postData.Add("type", strconv.Itoa(ChatReustParam.Type))
	postData.Add("ISLIUid", ChatReustParam.IsliuId)
	postData.Add("content", ChatReustParam.Content)
	// 发送POST请求
	resp, err := http.Post(request_url, "application/x-www-form-urlencoded", strings.NewReader(postData.Encode()))
	if err != nil {
		fmt.Println("会话响应失败,请稍后重试:", err)
		return 0, nil, "", err
	}

	defer resp.Body.Close()

	fmt.Println("Response status:", resp.Status)

	// 读取响应的body数据
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		fmt.Println("读取会话响应数据失败,请稍后重试:", err)
		return 0, nil, "", err
	}
	// 解析JSON
	var response HuaSoulEchoRespones

	err = json.Unmarshal(body, &response)
	if err != nil {
		fmt.Println("数字人调用会话接口失败,无法获取意识流:", err)
		return 0, nil, "", err
	}
	// 使用解析后的参数
	fmt.Println("Code:", response.Code)
	fmt.Println("ISLIUid:", response.Status)
	fmt.Println("ISLIUid:", response.ISLIU)

	return response.Code, response.Status, response.ISLIU, nil

}

go语言如何实现调用第三方接口传输文件流

package service

import (
	"bytes"
	"encoding/json"
	"fmt"
	"io"
	"io/ioutil"
	"log"
	"mime/multipart"
	"net/http"
	"os"
	"strconv"
)

type SoulNeuralRequest struct {
	Auth    string `json:"auth"`
	Type    int    `json:"type"`
	ISLIUid string `json:"ISLIUid"`
	//CFN     string `json:"CFN,omitempty"` // omitempty 表示如果 CFN 为空,则在 JSON 中省略这个字段
	CFN string `json:"CFN"`
}

type SoulNeuralResponse struct {
	Code int `json:"code"` // 仅当网络状态码为201时有效
}

// SoulNeuralFileTransfer 神经网络记忆文件上载
func SoulNeuralFileUpload(Neuralurl string, request SoulNeuralRequest, fileName string) ([]byte, error) {
	fmt.Println("NeuralFileUpload request_url", Neuralurl)
	// 创建一个HTTP客户端
	client := &http.Client{}

	// 创建一个buffer,用于存储请求的multipart/form-data数据
	var bodyBuf bytes.Buffer
	writer := multipart.NewWriter(&bodyBuf)

	// 添加参数字段
	writer.WriteField("auth", request.Auth)

	writer.WriteField("type", strconv.Itoa(request.Type))
	writer.WriteField("ISLIUid", request.ISLIUid)

	// 如果是上载(还魂)操作,需要添加静态意识网络文件
	// 这里假设你已经有一个名为 "static_network.cfn" 的文件

	staticNetworkFile, err := os.Open(fileName)
	if err != nil {
		fmt.Println("Failed to open file:", err)
		return nil, err
	}
	defer staticNetworkFile.Close()

	fileWriter, err := writer.CreateFormFile("CFN", fileName)
	if err != nil {
		fmt.Println("Failed to create form file:", err)
		return nil, err
	}

	_, err = io.Copy(fileWriter, staticNetworkFile)
	if err != nil {
		fmt.Println("Failed to copy file data:", err)
		return nil, err
	}

	// 关闭multipart writer
	writer.Close()

	// 创建POST请求
	req, err := http.NewRequest("POST", Neuralurl, &bodyBuf)
	if err != nil {
		fmt.Println("Failed to create request:", err)
		return nil, err
	}

	// 设置请求头
	req.Header.Set("Content-Type", writer.FormDataContentType())

	// 发送请求
	resp, err := client.Do(req)
	if err != nil {
		fmt.Println("Failed to send request:", err)
		return nil, err
	}
	defer resp.Body.Close()

	// 读取响应体
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		return nil, err
	}
	// 解析JSON
	var response SoulNeuralResponse

	// 根据不同的响应状态码处理数据
	if resp.StatusCode == 201 {
		err = json.Unmarshal(body, &response)
		log.Println("response.Code: ", response.Code)
		if err != nil {
			return nil, err
		}
		return nil, fmt.Errorf("error from server: %v", response.Code)
	} else if resp.StatusCode == 200 {
		// 正常响应,返回body
		fmt.Printf("响应状态码%d, 神经网络记忆文件上传【 %d字节 】 \n", resp.StatusCode, len(body))

		//log.Println("意识文件上传成功!", resp.StatusCode)
		return body, nil
	}

	return nil, fmt.Errorf("unexpected server response: %v", resp.Status)
}

// SoulNeuralFileUpload 神经网络记忆文件下载
func SoulNeuralFileDownload(Neuralurl string, request SoulNeuralRequest, fileName string) error {

	fmt.Println("NeuralFileDownload request_url", Neuralurl)
	// 创建一个HTTP客户端
	client := &http.Client{}

	// 创建一个buffer,用于存储请求的multipart/form-data数据
	bodyBuf := &bytes.Buffer{}
	writer := multipart.NewWriter(bodyBuf)

	// 添加参数字段
	writer.WriteField("auth", request.Auth)
	writer.WriteField("type", fmt.Sprintf("%d", request.Type))
	writer.WriteField("ISLIUid", request.ISLIUid)

	// 关闭multipart writer
	err := writer.Close()
	if err != nil {
		fmt.Println("Failed to close writer:", err)
		return err
	}

	// 创建POST请求
	req, err := http.NewRequest("POST", Neuralurl, bodyBuf)
	if err != nil {
		fmt.Println("Failed to create request:", err)
		return err
	}

	// 设置请求头
	req.Header.Set("Content-Type", writer.FormDataContentType())

	// 发送请求
	resp, err := client.Do(req)
	if err != nil {
		fmt.Println("Failed to send request:", err)
		return err
	}
	defer resp.Body.Close()

	// 读取响应体
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		return err
	}
	// 解析JSON
	var response SoulNeuralResponse

	// 检查网络状态码
	if resp.StatusCode == 200 {
		// 创建文件
		file, err := os.Create(fileName)
		if err != nil {
			fmt.Println("Failed to create file:", err)
			return err
		}
		defer file.Close()

		// 将响应体中的数据写入文件
		_, err = file.Write(body)
		if err != nil {
			fmt.Println("Failed to write file:", err)
			return err
		}
		fmt.Println("文件下载完成")
		fmt.Printf("响应状态码%d, 神经网络记忆文件下载【%d字节】\n", resp.StatusCode, len(body))

	} else {
		err = json.Unmarshal(body, &response)
		fmt.Println("请求失败,状态码:", resp.StatusCode, " response.Code: ", response.Code)

		//log.Println("response.Code: ", response.Code)

		return fmt.Errorf("error from server: %v", response.Code)
	}

	return nil
}


集群部署思考

  • 假设场景
    目前有20台服务器,每台服务器上需要部署8个算法模型,如何实现自动负载均衡?当请求量大的时候,服务数增多,请求量少的时候减少一些服务,如何实现这样的集群部署?

思路:
要实现模型服务的自动负载均衡,可以考虑以下步骤来进行集群部署:

  1. 选择合适的负载均衡器:选择一种合适的负载均衡器,例如Nginx、HAProxy等。负载均衡器可以接收客户端请求,并将请求转发到多台服务器上的模型服务。

  2. 设置服务器集群:将这20台服务器组成一个服务器集群。可以使用云服务提供商的负载均衡服务,或者自己设置主机名、IP地址等网络配置。

  3. 部署模型服务:在每台服务器上部署8个算法模型的服务。确保每个模型服务都能独立运行,并且可以接收和处理请求。

  4. 配置负载均衡器:在负载均衡器上配置后端服务器。将每台服务器及其对应的8个模型服务都添加到负载均衡器的配置中。

  5. 设置负载均衡策略:根据请求量的大小,调整负载均衡器的策略。可以根据服务器的负载情况、CPU利用率、请求响应时间等因素,自动调整请求的转发方式。例如,可以使用轮询、权重、最少连接数等策略来平衡负载。

  6. 监控和自动扩展:使用监控工具来监测服务器集群的负载情况,当请求量大于一定阈值时,自动创建新的服务器实例并部署模型服务。当请求量减少时,自动缩减服务器实例。这可以使用云服务提供商的自动扩展功能,或者使用监控脚本和自动化部署工具来实现。

通过以上步骤,你可以实现模型服务的自动负载均衡,根据请求量的变化自动增加或减少服务器实例,以确保高可用性和良好的性能。

请注意,在实际部署过程中,还需要考虑网络安全、数据同步、故障恢复等方面的问题。建议在实施前进行充分的规划和测试,确保部署方案的有效性和可靠性。


要实现模型服务的自动负载均衡,可以使用以下技术和工具:

  1. 负载均衡器:常见的负载均衡器有Nginx、HAProxy、F5等。它们能够接收客户端请求并将请求转发到后端服务器上。

  2. 容器化技术:使用容器化技术如Docker或Kubernetes可以更方便地管理和部署模型服务。通过将每个算法模型封装为一个独立的容器,可以实现快速的部署和水平扩展。

  3. 云服务提供商:大多数云服务提供商(如AWS、Azure、阿里云等)提供负载均衡服务和自动扩展功能。你可以使用这些平台来轻松地设置负载均衡器和自动扩展规则。

  4. 监控和自动化工具:使用监控工具如Prometheus、Grafana等可以实时监测服务器集群的负载情况,进行性能分析和故障排查。结合自动化工具如Ansible、Terraform等,可以实现自动化部署和伸缩。

  5. 负载均衡策略:负载均衡器通常支持不同的负载均衡策略,如轮询、加权轮询、最少连接数等。根据实际需求,选择合适的负载均衡策略来平衡服务器负载。

综上所述,上述技术和工具是实现模型服务自动负载均衡的常见选择。具体使用哪些技术,取决于你的要求、预算和技术栈。可以根据自身情况进行选择和定制。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值