命令行在线词典|青训营笔记

190 阅读3分钟

命令行在线词典

这是我参与「第三届青训营 -后端场」笔记创作活动的第1篇笔记。在青训营第一天的学习中,我们跟随王克纯老师学习了Go语言的一些基础知识,以及两个小程序的编写。其中一个小程序就是这次要记录的主题:命令行在线词典。我们在使用这个词典时,只需要在运行命令后加入需要翻译的单词,就可以得到对应单词的中文翻译了。

使用效果

我们可以在学习前先运行一下程序,观察一下词典的使用效果:

image.png

可以发现我们输入home单词以后,程序会正确地根据彩云翻译引擎给出home的中文翻译,词性,甚至音标。那么这一切究竟是怎么做到的呢?

抓包

首先需要进入彩云翻译的官网:fanyi.caiyunapp.com/#/

按下F12键并进入开发者模式,在这个模式下,我们可以抓取并看到网页的一切Web请求。接下来我们在翻译框中输入我们需要翻译的单词,并单击翻译按钮,如下图所示:

image.png

可以看见我们已经抓取到了多条Web请求,我们找到名称为dict的POST请求。观察后发现正是这个请求返回了单词的翻译结果。所以我们接下来只要模拟了这个dict请求,即可完成程序的主体内容。

代码生成工具

首先我们抓取这个请求的cURL:

image.png

在复制好了cURL请求后,我们使用一个网页小工具,帮助我们把cURL请求转化为我们可以使用的Go代码,工具网址: curlconverter.com/#go ,使用效果图如下:

image.png

可以看到cURL被直接转化为了可以运行的Go代码,使用go run命令运行,可以得到如下图的返回值:

image.png

很显然,我们并不需要如此庞大的返回信息,我们只需要将最有用的翻译信息获取下来即可。但是为了接收这个庞大的json,我们首先需要创建一个与之对应的结构体。在这里,会用到另外一个网页工具,帮助我们根据json自动生成对应的Go语言结构体,工具网址:oktools.net/json2go 使用效果如下:

image.png

至此,我们已经完成了程序编写的必要准备工作,接下来就是代码展示环节

代码展示

//根据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)
}