golang 腾讯通用OCR 文字识别 Api 调用

一时的想法,想调着玩一玩,结果一下午没了(欸,菜)
首先,图片文字识别(我认为可行的)大概有两个途径,百度Api一天5w次通用调用机会(也有qps限制),腾讯每秒1或2次(个人认证完2次),这里觉得百度确实有点抠
腾讯的话应用广泛一些,qq里日常的语音识别,文字识别都大量用到,短期不会有大变化
但是腾讯目前不支持提交图片url,所以就要先下载下来,再提交

实现了三个功能,从本地上传识别,从url下载识别和base64编码的字符串识别
2020-6-1版本,结合了一些新想法,比以前好了点,放在一个文件里可能有些乱,见谅了
自定义 Param 结构体及方法,可以拿出来用在腾讯AI的其他 api 调用上,例:
中英互译

package TencentOCR

import (
	"bytes"
	"crypto/md5"
	"encoding/base64"
	"encoding/hex"
	"fmt"
	"golang.org/x/time/rate"
	"io"
	"io/ioutil"
	"log"
	"net/http"
	"net/url"
	"os"
	"sort"
	"strconv"
	"strings"
	"time"
)

/*
可以直接拿去使用
把自己开通好的 id 和 key 换了就可以,需要在应用中添加对应能力
返回的数据是原始字节数组(json格式),根据需要可以直接返回给前端解析
或者在服务端解析
 */

var (
	// 自己去官网开账号,开应用
	appId  = "*"
	appKey = "*"
	// Tencent's OCR QPS is 2 for me
	OcrLimiter = rate.NewLimiter(2, 2)
)

// 为了排序
type Param struct {
	key, value string
}

func SortParams(p []Param) {
	// 升序
	sort.Slice(p, func(i, j int) bool {
		if p[i].key < p[j].key {
			return true
		}
		return false
	})
}

func ParamsToString(p []Param) string {
	s := ""
	for _, v := range p {
		if v.value == "" {
			continue
		}
		// value 需要进行 url 编码
		s += v.key + "=" + url.QueryEscape(v.value) + "&"
	}
	return s[:len(s)-1]
}

func GetByteFromUrl(url string) (ret []byte, ok bool) {
	for {
		if ok := OcrLimiter.Allow(); ok {
			break
		}
		time.Sleep(time.Millisecond * 550)
	}
	fileName := "pic" + strconv.FormatInt(time.Now().UnixNano(), 10)
	if ok := downloadPic(url, fileName); !ok {
		return
	}
	defer delFile("./temp/" + fileName)
	return GetByteFromFile("./temp/" + fileName)
}

func GetByteFromFile(filePath string) (ret []byte, ok bool) {
	file, err := os.Open(filePath)
	if err != nil {
		log.Println(err)
		return
	}
	defer file.Close()

	pic, err := ioutil.ReadAll(file)
	if err != nil {
		log.Println(err)
		return
	}

	encodeToString := base64.StdEncoding.EncodeToString(pic)

	return GetByteFromBase64String(encodeToString)
}

func GetByteFromBase64String(encodeToString string) (ret []byte, ok bool) {
	client := http.Client{}
	params := make([]Param, 0, 10)
	params = append(params, Param{"app_id", appId})
	params = append(params, Param{"image", encodeToString})

	// 随便啥都行,非空且长度小于32
	params = append(params, Param{"nonce_str", "asbfiuasbhjbcuicg"})
	params = append(params, Param{"time_stamp", fmt.Sprintf("%d", time.Now().Unix())})

	// 取得动态加密标志,组装body
	params = getSign(params)
	form := ParamsToString(params)
	body := bytes.NewBufferString(form)

	request, err := http.NewRequest("POST", "https://api.ai.qq.com/fcgi-bin/ocr/ocr_generalocr", body)
	if err != nil {
		log.Println(err)
		return
	}

	// 重要的!!!
	request.Header.Set("Content-Type", "application/x-www-form-urlencoded")
	response, err := client.Do(request)
	if err != nil {
		log.Println(err)
		return
	}

	defer response.Body.Close()
	ret, err = ioutil.ReadAll(response.Body)
	if err != nil {
		log.Println(err)
		return
	}

	return ret, true
}

// 下载图片
func downloadPic(url, fileName string) (ok bool) {
	resp, err := http.Get(url)
	if err != nil {
		log.Println(err)
		return
	}
	defer resp.Body.Close()

	err = os.Mkdir("./temp", os.ModePerm)
	if !os.IsExist(err) {
		log.Println(err)
		return
	}

	// 后缀不重要,就省去了
	file, err := os.OpenFile("./temp/"+fileName, os.O_CREATE|os.O_WRONLY, os.ModePerm)
	if err != nil {
		log.Println(err)
		return
	}
	defer file.Close()

	_, err = io.Copy(file, resp.Body)
	if err != nil {
		log.Println(err)
		return
	}
	return true
}

func getSign(p []Param) []Param {
	s := strings.Builder{}

	SortParams(p)
	s.WriteString(ParamsToString(p))

	s.WriteString("&app_key=" + appKey)

	// MD5
	hash := md5.New()
	hash.Write([]byte(s.String()))
	encodeToString := strings.ToUpper(hex.EncodeToString(hash.Sum(nil)))
	p = append(p, Param{"sign", encodeToString})
	return p
}

func delFile(filePath string) {
	os.Remove(filePath)
}

End
代码对于各位来说可能不很优雅,但要使用的话基本框架应该是有的,
各位有好的意见的话欢迎指点,bye!
Golang调用 RabbitMQ 的节点(Node)API,你可以使用标准的 `net/http` 包来发送 HTTP 请求并处理响应。以下是一个简单的示例代码,用于获取 RabbitMQ 节点的信息: ```go package main import ( "encoding/json" "fmt" "io/ioutil" "net/http" ) type NodeInfo struct { Name string `json:"name"` Type string `json:"type"` Running bool `json:"running"` MemoryUsed int `json:"mem_used"` DiskFree int `json:"disk_free"` } func main() { // RabbitMQ Node API URL url := "http://localhost:15672/api/nodes" // 创建 HTTP 客户端 client := &http.Client{} // 创建 HTTP 请求 req, err := http.NewRequest("GET", url, nil) if err != nil { fmt.Println("创建请求失败:", err) return } // 设置 Basic Auth 认证(如果需要) req.SetBasicAuth("username", "password") // 发送请求并获取响应 resp, err := client.Do(req) if err != nil { fmt.Println("发送请求失败:", err) return } defer resp.Body.Close() // 读取响应内容 body, err := ioutil.ReadAll(resp.Body) if err != nil { fmt.Println("读取响应失败:", err) return } // 解析 JSON 响应 var nodes []NodeInfo err = json.Unmarshal(body, &nodes) if err != nil { fmt.Println("解析响应失败:", err) return } // 输出节点信息 for _, node := range nodes { fmt.Printf("节点名称: %s\n", node.Name) fmt.Printf("节点类型: %s\n", node.Type) fmt.Printf("运行状态: %v\n", node.Running) fmt.Printf("内存使用: %d\n", node.MemoryUsed) fmt.Printf("磁盘可用: %d\n", node.DiskFree) fmt.Println("----------------------") } } ``` 请注意,上述代码中的 `url` 变量需要根据你的 RabbitMQ 配置进行相应调整。如果 RabbitMQ 设置了用户名和密码的认证,你需要使用 `SetBasicAuth` 方法设置 Basic Auth 认证信息。 该示例代码发送一个 HTTP GET 请求到 RabbitMQ 的 `/api/nodes` 接口,获取所有节点的信息,并将其解析为 `NodeInfo` 结构体。然后,你可以根据需要处理和使用这些节点信息。 希望这个示例能帮助到你。如果有任何其他问题,请随时提问。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值