Android 开发 有道翻译
爬取有道翻译api接口,定制专属于你的翻译官
1.首先是通过浏览器的开发者工具抓取有道翻译的数据包,解析request与response。
2.然后会发现这个是一个post请求,而且居然数据还加密,而这就需要通过js逆向来分析,这里面有MD5,时间戳,随机数,然后再糅合在一起。
3.通过正则解析响应回来的json数据
4.最后就是安卓的UI设计以及业务逻辑代码了。
抓取有道翻译数据
打开有道翻译网页版开始抓取数据包,可以发现salt,sign,lts,bv,这几个请求参数是加密了的,通过js逆向分析可以得出:
- i :待翻译的文本;
- lts:时间戳;
- salt:时间戳+10以内的随机数;
- bv:UA请求头通过MD5信息摘要法产生的32位16进制数;
- sign:固定字符串通过MD5产生的16进制数;
- 接口就抓取下来了,留着等会用 ;
Android 网络权限设置
问题:无法进行网络请求;
原因:Android9.0以后只支持HTTPS请求了,所有的http请求都认为是不安全的,所以不能访问。;
解决方法:
第一步:首先是最基本的网络权限 在AndroidManif.xml中添加
<!-- 获取网络权限 -->
<uses-permissionandroid:name="android.permission.INTERNET" />**;
第二步:在res文件夹中新建xml文件夹添加network_security_config.xml文件,文件内容如下:
<?xml version="1.0" encoding="utf-8"?> <network-security-config>
<base-config cleartextTrafficPermitted="true" />
</network-security-config>
第三步:在AndroidManif.xml文件中注册写命令文件:
android:networkSecurityConfig=
"@xml/network_security_config"
Android 逻辑代码
- 创建全局变量
- 在onCreat方法中找到组件
- 重写onResume方法
- 添加按钮监听事件
- 发送post请求
- 解析翻译结果
- setText的值
//debug调试
String TAG = "mytag";
//声明组件component
Button button0;
TextView text0,text1,text2;
//创建全局OkHttp对象
private OkHttpClient okHttpClient;
//全局翻译结果变量
String you_dao ="";
//更多翻译的词性
String entries = "";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//找到组件
button0 = findViewById(R.id.button0);
text0 = findViewById(R.id.text0);
text1 = findViewById(R.id.text1);
text2 = findViewById(R.id.text2);
//创建okhttp对象
okHttpClient = new OkHttpClient();
}
//交互
@Override
protected void onResume() {
super.onResume();
//翻译按钮监听事件
button0.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//获取文本框内容
String string = text0.getText().toString();
//请求有道翻译
//interpret(string);
youdao_interpret(string);
//线程延时
sleep(300);
//通过全局变量在results显示翻译结果
text1.setText(you_dao);
text2.setText(entries);
}
});
post请求
上面已经说完逻辑了,现在就要实现okhttp向有道翻译发送post请求,并返回翻译结果。
我们刚刚已经解析出加密的数据了,而我们现在要做的就是产生加密的数据并发送给
有道翻译服务器。也就是post请求添加请求头和表单请求体了。
(1)产生加密数据(补充:Android进行网络请求是必须新开线程)
public void youdao_interpret(String arg){
new Thread(){
final String string = arg;
@Override
public void run(){
//时间戳 r = ts
String ts = String.valueOf(System.currentTimeMillis());
//随机数 i = salt
Random random = new Random();
String salt = ts+String.valueOf(random.nextInt(10));
//请求头
String ua = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36";
// md5 t = bv
String bv = "";
try {
bv = MD5(ua);
} catch (Exception e) {
e.printStackTrace();
}
// x
String x="fanyideskweb" + this.string + salt + "Tbh5E8=q6U3EXe+&L[4c@";
// sign
String sign = "";
try {
sign=MD5(x);
} catch (Exception e) {
e.printStackTrace();
}
(2)封装MD5函数
//MD5函数
public static String MD5(String data) throws Exception {
java.security.MessageDigest md = MessageDigest.getInstance("MD5");
byte[] array = md.digest(data.getBytes("UTF-8"));
StringBuilder sb = new StringBuilder();
for (byte item : array) {
sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));
}
return sb.toString().toUpperCase();
}
(3)http请求添加请求体和请求头
//请求有道翻译的完整网址
String url="http://fanyi.youdao.com/translate_o?smartresult=dict&smartresult=rule";
//创建请求体表单 >>填入请求翻译的内容
FormBody formBody = new FormBody.Builder()
.add("i",this.string)
.add("smartresult","dict")
.add("salt",salt)
.add("sign",sign)
.add("lts",ts)
.add("bv",bv)
.add("client","fanyideskweb")
.add("doctype","json")
.add("version","2.1")
.add("keyfrom","fanyi.web")
.build();
formBody.contentType();
//创建请求对象 >>并填入form表单
Request request = new Request.Builder()
.url(url)
.addHeader("User-Agent",ua) //请求头
.addHeader("Cookie","OUTFOX_SEARCH_USER_ID=-1876541902@10.169.0.84; OUTFOX_SEARCH_USER_ID_NCOO=2111552283.5701284; JSESSIONID=aaaNARG2_9FBJ-Ce81qBx; ___rl__test__cookies=1609821642788") //请求头
.addHeader("Referer","http://fanyi.youdao.com/") //请求头
.post(formBody).build();
//准备请求对象 Client:客户
Call call = okHttpClient.newCall(request);
(5)解析返回的json数据,通过正则表达式提取翻译结果
try{
Response response = call.execute();
//读取响应
String string = response.body().string(); //获取响应的字符串 还有byteStream方法,获取二进制数据
Log.d(TAG, "post请求成功");
//显示返回的数据
Log.d(TAG, "返回的全部:"+string);
//提取数据
Pattern compile =Pattern.compile("\"tgt\":\".*?\"");
Pattern compile1 =Pattern.compile("entries\":\\[\"\",\".*");
//匹配
Matcher matcher = compile.matcher(string);
while(matcher.find()){
//输出每个匹配的字符
you_dao = matcher.group().substring(7,matcher.group().length()-1);
}
//更多翻译匹配
Matcher matcher1 = compile1.matcher(string);
while(matcher1.find()){
//输出每个匹配的字符
entries = matcher1.group().substring(14,matcher1.group().length()-17).replace("\\r\\n\",\""," ");
Log.d(TAG, "词根匹配的为"+entries);
}
}catch (IOException e ){
String string = e.getMessage();
Log.d(TAG, "post失败");
}
}
调试,打包,签名,加固,发布。
1.build 构建apk
2.360加固签名
完整代码
package com.example.translate;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
import java.io.IOException;
import java.security.MessageDigest;
import java.util.Date;
import java.util.Random;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import okhttp3.Call;
import okhttp3.FormBody;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.OkHttpClient;
public class MainActivity extends AppCompatActivity {
//debug调试
String TAG = "mytag";
//声明组件component
Button button0;
TextView text0,text1,text2;
//创建全局OkHttp对象
private OkHttpClient okHttpClient;
//全局翻译结果变量
String you_dao ="";
//更多翻译的词性
String entries = "";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//找到组件
button0 = findViewById(R.id.button0);
text0 = findViewById(R.id.text0);
text1 = findViewById(R.id.text1);
text2 = findViewById(R.id.text2);
//创建okhttp对象
okHttpClient = new OkHttpClient();
}
//交互
@Override
protected void onResume() {
super.onResume();
//翻译按钮监听事件
button0.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//获取文本框内容
String string = text0.getText().toString();
//请求有道翻译
//interpret(string);
youdao_interpret(string);
//线程延时
sleep(300);
//通过全局变量在results显示翻译结果
text1.setText(you_dao);
text2.setText(entries);
}
});
}
//有道翻译函数
public void interpret(String arg){
new Thread(){
final String string = arg;
@Override
public void run(){
//请求有道翻译的简化网址
String url="http://fanyi.youdao.com/translate?smartresult=dict&smartresult=rule";
//创建请求体表单 >>填入请求翻译的内容
FormBody formBody = new FormBody.Builder().add("i",this.string).add("doctype","json").build();
formBody.contentType();
//创建请求对象 >>并填入form表单
Request request = new Request.Builder()
.url(url)
.addHeader("User-Agent","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36") //请求头
.post(formBody).build();
//准备请求对象 Client:客户
Call call = okHttpClient.newCall(request);
//response 是响应对象
try{
Response response = call.execute();
//读取响应
String string = response.body().string(); //获取响应的字符串 还有byteStream方法,获取二进制数据
Log.d(TAG, "post请求");
//显示返回的数据
Log.d(TAG, string);
//提取数据
Pattern compile =Pattern.compile("\"tgt\":\".*?\"");
//匹配
Matcher matcher = compile.matcher(string);
while(matcher.find()){
//输出每个匹配的字符
you_dao = matcher.group().substring(7,matcher.group().length()-1);
}
}catch (IOException e ){
String string = e.getMessage();
Log.d(TAG, "post失败");
}
}
}.start();
}
//js逆向的完整请求函数
public void youdao_interpret(String arg){
new Thread(){
final String string = arg;
@Override
public void run(){
//时间戳 r = ts
String ts = String.valueOf(System.currentTimeMillis());
//随机数 i = salt
Random random = new Random();
String salt = ts+String.valueOf(random.nextInt(10));
//请求头
String ua = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36";
// md5 t = bv
String bv = "";
try {
bv = MD5(ua);
} catch (Exception e) {
e.printStackTrace();
}
// x
String x="fanyideskweb" + this.string + salt + "Tbh5E8=q6U3EXe+&L[4c@";
// sign
String sign = "";
try {
sign=MD5(x);
} catch (Exception e) {
e.printStackTrace();
}
//请求有道翻译的完整网址
String url="http://fanyi.youdao.com/translate_o?smartresult=dict&smartresult=rule";
//创建请求体表单 >>填入请求翻译的内容
FormBody formBody = new FormBody.Builder()
.add("i",this.string)
.add("smartresult","dict")
.add("salt",salt)
.add("sign",sign)
.add("lts",ts)
.add("bv",bv)
.add("client","fanyideskweb")
.add("doctype","json")
.add("version","2.1")
.add("keyfrom","fanyi.web")
.build();
formBody.contentType();
//创建请求对象 >>并填入form表单
Request request = new Request.Builder()
.url(url)
.addHeader("User-Agent",ua) //请求头
.addHeader("Cookie","OUTFOX_SEARCH_USER_ID=-1876541902@10.169.0.84; OUTFOX_SEARCH_USER_ID_NCOO=2111552283.5701284; JSESSIONID=aaaNARG2_9FBJ-Ce81qBx; ___rl__test__cookies=1609821642788") //请求头
.addHeader("Referer","http://fanyi.youdao.com/") //请求头
.post(formBody).build();
//准备请求对象 Client:客户
Call call = okHttpClient.newCall(request);
//response 是响应对象
try{
Response response = call.execute();
//读取响应
String string = response.body().string(); //获取响应的字符串 还有byteStream方法,获取二进制数据
Log.d(TAG, "post请求成功");
//显示返回的数据
Log.d(TAG, "返回的全部:"+string);
//提取数据
Pattern compile =Pattern.compile("\"tgt\":\".*?\"");
Pattern compile1 =Pattern.compile("entries\":\\[\"\",\".*");
//匹配
Matcher matcher = compile.matcher(string);
while(matcher.find()){
//输出每个匹配的字符
you_dao = matcher.group().substring(7,matcher.group().length()-1);
}
//更多翻译匹配
Matcher matcher1 = compile1.matcher(string);
while(matcher1.find()){
//输出每个匹配的字符
entries = matcher1.group().substring(14,matcher1.group().length()-17).replace("\\r\\n\",\""," ");
Log.d(TAG, "词根匹配的为"+entries);
}
}catch (IOException e ){
String string = e.getMessage();
Log.d(TAG, "post失败");
}
}
}.start();
}
//线程阻塞method
public void sleep(int ms){
try{
Thread.sleep(ms);
}catch(InterruptedException e){
e.getMessage();
}
}
//MD5加密函数
public static String MD5(String data) throws Exception {
java.security.MessageDigest md = MessageDigest.getInstance("MD5");
byte[] array = md.digest(data.getBytes("UTF-8"));
StringBuilder sb = new StringBuilder();
for (byte item : array) {
sb.append(Integer.toHexString((item & 0xFF) | 0x100).substring(1, 3));
}
return sb.toString().toUpperCase();
}
}