命令行在线词典
这是我参与「第三届青训营 -后端场」笔记创作活动的第1篇笔记。在青训营第一天的学习中,我们跟随王克纯老师学习了Go语言的一些基础知识,以及两个小程序的编写。其中一个小程序就是这次要记录的主题:命令行在线词典。我们在使用这个词典时,只需要在运行命令后加入需要翻译的单词,就可以得到对应单词的中文翻译了。
使用效果
我们可以在学习前先运行一下程序,观察一下词典的使用效果:
可以发现我们输入home单词以后,程序会正确地根据彩云翻译引擎给出home的中文翻译,词性,甚至音标。那么这一切究竟是怎么做到的呢?
抓包
首先需要进入彩云翻译的官网:fanyi.caiyunapp.com/#/
按下F12键并进入开发者模式,在这个模式下,我们可以抓取并看到网页的一切Web请求。接下来我们在翻译框中输入我们需要翻译的单词,并单击翻译按钮,如下图所示:
可以看见我们已经抓取到了多条Web请求,我们找到名称为dict的POST请求。观察后发现正是这个请求返回了单词的翻译结果。所以我们接下来只要模拟了这个dict请求,即可完成程序的主体内容。
代码生成工具
首先我们抓取这个请求的cURL:
在复制好了cURL请求后,我们使用一个网页小工具,帮助我们把cURL请求转化为我们可以使用的Go代码,工具网址: curlconverter.com/#go ,使用效果图如下:
可以看到cURL被直接转化为了可以运行的Go代码,使用go run命令运行,可以得到如下图的返回值:
很显然,我们并不需要如此庞大的返回信息,我们只需要将最有用的翻译信息获取下来即可。但是为了接收这个庞大的json,我们首先需要创建一个与之对应的结构体。在这里,会用到另外一个网页工具,帮助我们根据json自动生成对应的Go语言结构体,工具网址:oktools.net/json2go 使用效果如下:
至此,我们已经完成了程序编写的必要准备工作,接下来就是代码展示环节
代码展示
//根据json返回值构建的Go语言结构体,用于json的反序列化
type CaiyunTrans struct {
Rc int `json:"rc"`
Wiki struct {
KnownInLaguages int `json:"known_in_laguages"`
Description struct {
Source string `json:"source"`
Target interface{} `json:"target"`
} `json:"description"`
ID string `json:"id"`
Item struct {
Source string `json:"source"`
Target string `json:"target"`
} `json:"item"`
ImageURL string `json:"image_url"`
IsSubject string `json:"is_subject"`
Sitelink string `json:"sitelink"`
} `json:"wiki"`
Dictionary struct {
Prons struct {
EnUs string `json:"en-us"`
En string `json:"en"`
} `json:"prons"`
Explanations []string `json:"explanations"`
Synonym []string `json:"synonym"`
Antonym []string `json:"antonym"`
WqxExample [][]string `json:"wqx_example"`
Entry string `json:"entry"`
Type string `json:"type"`
Related []interface{} `json:"related"`
Source string `json:"source"`
} `json:"dictionary"`
}
//模拟彩云翻译api的cURL
func caiyunTanslate(s string) {
client := &http.Client{}
var data = strings.NewReader(`{"trans_type":"en2zh","source":"` + s + `"}`)
req, err := http.NewRequest("POST", "https://api.interpreter.caiyunai.com/v1/dict", data)
if err != nil {
log.Fatal(err)
}
req.Header.Set("Accept", "application/json, text/plain, */*")
req.Header.Set("Accept-Language", "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6")
req.Header.Set("Connection", "keep-alive")
req.Header.Set("Content-Type", "application/json;charset=UTF-8")
req.Header.Set("Origin", "https://fanyi.caiyunapp.com")
req.Header.Set("Referer", "https://fanyi.caiyunapp.com/")
req.Header.Set("Sec-Fetch-Dest", "empty")
req.Header.Set("Sec-Fetch-Mode", "cors")
req.Header.Set("Sec-Fetch-Site", "cross-site")
req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/101.0.4951.41 Safari/537.36 Edg/101.0.1210.32")
req.Header.Set("X-Authorization", "token:qgemv4jr1y38jyq6vhvi")
req.Header.Set("app-name", "xy")
req.Header.Set("os-type", "web")
req.Header.Set("sec-ch-ua", `" Not A;Brand";v="99", "Chromium";v="101", "Microsoft Edge";v="101"`)
req.Header.Set("sec-ch-ua-mobile", "?0")
req.Header.Set("sec-ch-ua-platform", `"Windows"`)
resp, err := client.Do(req)
if err != nil {
log.Fatal(err)
}
defer resp.Body.Close()
bodyText, err := ioutil.ReadAll(resp.Body)
if resp.StatusCode != 200 { //防御性编码,方便排查错误
log.Fatal("2bad StatusCode:", resp.StatusCode, "body", string(bodyText))
}
if err != nil {
log.Fatal(err)
}
var dictR CaiyunTrans
json.Unmarshal(bodyText, &dictR)
lock.Lock()
fmt.Println("\n", "彩云翻译:")
fmt.Println(s, "UK:", dictR.Dictionary.Prons.En, "US:", dictR.Dictionary.Prons.EnUs)
for _, item := range dictR.Dictionary.Explanations {
fmt.Println(item)
}
}
//程序主入口
func main() {
if len(os.Args) != 2 {
fmt.Fprintf(os.Stderr, `usage: simpleDict WORD
example: simpleDict hello
`)
os.Exit(1)
}
word := os.Args[1]
caiyunTanslate(word)
}