文章结构: 按照逆向过程中的可疑类作为单元
VoiceIflytekCallBackBean
包路径: com.tanma.unirun.ui.activity.running.iflytek.VoiceIflytekCallBackBean
来自: SearchOneFeature类中的doSearchOneFeature()
方法中创建了该类的实例
作用: 声纹相关抓包中看, 这个类的数据域与body的message字段
SearchOneFeature
包路径: com.tanma.unirun.ui.activity.running.iflytek.SearchOneFeature
来自: VocalVerifyPresenterImpl.resolveStopRecord.4.callback.1.onSuccess.1
作用:
方法
doSearchOneFeature()
根据名字是发送一个请求报文, 来查询什么东西
public static VoiceIflytekCallBackBean doSearchOneFeature(String requestUrl, String APPID, String apiSecret, String apiKey, String AUDIO_PATH) {
int myCode;
SearchOneFeature searchOneFeature0 = new SearchOneFeature(requestUrl, APPID, apiSecret, apiKey, AUDIO_PATH);
VoiceIflytekCallBackBean myBean = new VoiceIflytekCallBackBean();
try {
String s5 = searchOneFeature0.doRequest();
System.out.println("resp=>" + s5);
JsonParse myJsonParse = (JsonParse)SearchOneFeature.json.fromJson(s5, JsonParse.class);
if(myJsonParse != null && myJsonParse.header != null) {
myCode = myJsonParse.header.code;
myBean.setSid(myJsonParse.header.sid);
myBean.setMessage(myJsonParse.header.message);
}
else {
myCode = 0xFFFFFCF7;
myBean.setSid("sid:myJsonParseOrheader is NULL");
myBean.setMessage("message : myJsonParseOrheader is NULL");
}
myBean.setCode(myCode);
if(myCode == 0xFFFFFCF7) {
myBean.setScore(0.0f);
return myBean;
}
if(myCode == 0) {
JSONObject jSONObject0 = JSON.parseObject(new String(Base64.getDecoder().decode(myJsonParse.payload.searchScoreFeaRes.text), "UTF-8"));
if(jSONObject0 != null) {
myBean.setScore(((VoiceIflytekInfoBean)JSON.parseObject(jSONObject0.toJSONString(), VoiceIflytekInfoBean.class)).getScore());
}
System.out.println("text字段Base64解码后=>" + jSONObject0);
return myBean;
}
myBean.setScore(0.0f);
}
catch(Exception e) {
e.printStackTrace();
myBean.setCode(0xFFFFFCF7);
myBean.setScore(0.0f);
myBean.setSid("sid:Exception");
myBean.setMessage("message : Exception=" + e.getMessage());
}
return myBean;
}
- 创建了一个本身类的对象
searchOneFeature0
, 创建了一个CallBackBean对象(其中的数据域与发送报文的字段一样) - 调用了
searchOneFeature0
的doRequest()
方法
doRequest()
使用了java的http连接
public String doRequest() throws Exception {
HttpURLConnection httpURLConnection = (HttpURLConnection)new URL(this.buildRequetUrl()).openConnection();
httpURLConnection.setDoInput(true);
httpURLConnection.setDoOutput(true);
httpURLConnection.setRequestMethod("POST");
httpURLConnection.setRequestProperty("Content-type", "application/json");
OutputStream outputStream0 = httpURLConnection.getOutputStream();
String s = this.buildParam();
System.out.println("params=>" + s);
outputStream0.write(s.getBytes());
outputStream0.flush();
try {
return this.readAllBytes(httpURLConnection.getInputStream());
}
catch(Exception e) {
return this.readAllBytes(httpURLConnection.getErrorStream());
}
}
buildRequetUrl()
加密更改url的参数
buildParam()
简单生成符合格式的字符串
VocalVerifyPresenterImpl
包路径: com.tanma.unirun.ui.activity.vocalverify.VocalVerifyPresenterImpl
来自: VocalVerifyPresenterImpl.resolveStopRecord.4.callback.1.onSuccess.1
就是该类的一个回调
作用: 用于管理声纹识别的UI
内部类
里面有一些字符串常量有用
public static final class Companion {
private Companion() {
}
public Companion(DefaultConstructorMarker $constructor_marker) {
}
public final String getAPPID() {
return "3607f345";
}
public final String getApiKey() {
return "68e1b1d4883b468e3e4a718e660bbc48";
}
public final String getApiSecret() {
return "ZTk3Mjg4MWM4ZTVhYzJjZWNhMmFiZTE1";
}
public final String getRequestUrl() {
return "https://api.xf-yun.com/v1/private/s782b4996";
}
}
VocalVerifyPresenterImpl.resolveStopRecord.4.callback.1.onSuccess.1
包路径: com.tanma.unirun.ui.activity.vocalverify.VocalVerifyPresenterImpl.resolveStopRecord.4.callback.1.onSuccess.1
来自: RunVocalStatusLogBean
的交叉引用
作用: VocalVerifyPresenterImpl类中的一个回调(根据名字可知), 同时是成功时的回调(根据名字可知)
方法
该类继承了Runnable, 最重要的就是run()方法
run() (关键函数)
具体分析见注释
public final void run() {
String s = this.$convertedFile.getPath();
VocalVerifyPresenterImpl.this.setFilePath(s);
VoiceIflytekCallBackBean voiceIflytekCallBackBean0 = SearchOneFeature.doSearchOneFeature("https://api.xf-yun.com/v1/private/s782b4996", "3607f345", "ZTk3Mjg4MWM4ZTVhYzJjZWNhMmFiZTE1", "68e1b1d4883b468e3e4a718e660bbc48", VocalVerifyPresenterImpl.this.getFilePath());
if(voiceIflytekCallBackBean0 != null) {
if(voiceIflytekCallBackBean0.getCode() == 0) {
float standard = 0.6f; // 应该是相似度标准
Float float0 = (Float)new PreUtil("sp_name_user").getValue("sp_user_recognittion_score", Reflection.getOrCreateKotlinClass(Float.TYPE), Float.valueOf(0.6f));
if(float0 != null) {
standard = (float)float0;
}
Log.i("VocalVerifyPresenterImp", "---> Score=" + standard + ",score=" + voiceIflytekCallBackBean0.getScore());
if(voiceIflytekCallBackBean0.getScore() >= standard) { // 如果声音识别大于standard
VocalVerifyPresenterImpl.this.getContext().runOnUiThread(((Runnable)new Runnable() { // 运行在主线程
@Override
public final void run() {
Toast toast0 = Toast.makeText(((Context)VocalVerifyPresenterImpl.this.getContext()), "验证通过", 0);
toast0.show(); // 显式通过的Toast
Intrinsics.checkExpressionValueIsNotNull(toast0, "Toast\n .makeText(… show()\n }");
Logger.e("验证通过", new Object[0]);
ProgressLoadingDialog progressLoadingDialog0 = VocalVerifyPresenterImpl.this.mWaitDialog;
if(progressLoadingDialog0 != null) {
progressLoadingDialog0.dissmissDialog();
}
new PreUtil("rundata").setValue("sp_run_vocalverify_status", "1"); // 设置了sp_run_vocalverify_status的值为1
View view0 = ((Activity)VocalVerifyPresenterImpl.this.getContext()).findViewById(0x1020002);
View view1 = null;
if(!(view0 instanceof ViewGroup)) {
view0 = null;
}
ViewGroup viewGroup0 = (ViewGroup)view0;
if(viewGroup0 != null) {
view1 = viewGroup0.getChildAt(0);
}
if(view1 != null) {
view1.postDelayed(((Runnable)new Runnable() {
@Override
public final void run() {
((ImageView)((FragmentActivity)VocalVerifyPresenterImpl.this.getContext()).findViewById(id.iv_hold)).setImageResource(0x7F080188);
VocalVerifyPresenterImpl.this.getContext().setResult(-1);
VocalVerifyPresenterImpl.this.getContext().finish();
}
}), 1000L);
return;
}
}
}));
}
else {
VocalVerifyPresenterImpl.this.getContext().runOnUiThread(((Runnable)new Runnable() {
@Override
public final void run() {
ProgressLoadingDialog progressLoadingDialog0 = VocalVerifyPresenterImpl.this.mWaitDialog;
if(progressLoadingDialog0 != null) {
progressLoadingDialog0.dissmissDialog();
}
new PreUtil("rundata").setValue("sp_run_vocalverify_status", "2");
Logger.e("验证失败,分数太低", new Object[0]);
Toast toast0 = Toast.makeText(((Context)VocalVerifyPresenterImpl.this.getContext()), "认证失败,请重新念出数字", 0);
toast0.show();
Intrinsics.checkExpressionValueIsNotNull(toast0, "Toast\n .makeText(… show()\n }");
}
}));
}
}
else {
VocalVerifyPresenterImpl.this.getContext().runOnUiThread(((Runnable)new Runnable() {
@Override
public final void run() {
ProgressLoadingDialog progressLoadingDialog0 = VocalVerifyPresenterImpl.this.mWaitDialog;
if(progressLoadingDialog0 != null) {
progressLoadingDialog0.dissmissDialog();
}
new PreUtil("rundata").setValue("sp_run_vocalverify_status", "2");
Logger.e("验证失败", new Object[0]);
Toast toast0 = Toast.makeText(((Context)VocalVerifyPresenterImpl.this.getContext()), "认证失败,请重新念出数字", 0);
toast0.show();
Intrinsics.checkExpressionValueIsNotNull(toast0, "Toast\n .makeText(… show()\n }");
}
}));
}
RunVocalStatusLogBean mlogBean = new RunVocalStatusLogBean(null, null, 3, null);
if(TextUtils.isEmpty(((CharSequence)voiceIflytekCallBackBean0.getSid()))) {
mlogBean.setCheckCode(Integer.valueOf(0xFFFFFCF7));
mlogBean.setMessge("defult message,isAndroid");
}
else {
mlogBean.setCheckCode(Integer.valueOf(voiceIflytekCallBackBean0.getCode()));
mlogBean.setMessge("Sid=" + voiceIflytekCallBackBean0.getSid() + ",Score=" + voiceIflytekCallBackBean0.getScore() + ",Message=" + voiceIflytekCallBackBean0.getMessage() + ",myGroupId=" + GroupIdUtil.myGroupId + ",isAndroid");
}
VocalVerifyPresenterImpl.this.getModelImpl().studentRunVocalStatusLog(mlogBean);
}
}
RunVocalStatusLogBean
包路径: com.tanma.unirun.entities.RunVocalStatusLogBean
来自: 抓了Unirun的包, 然后声纹识别的包的body里面有一个"checkCode"字段, 通过字符串查找找到了该类
作用: 声纹验证的body
RunVocalStatusLogBean
设置Activity
里面有getVocalReset()方法
public final void getVocalReset() {
User user0 = UnirunApplication.Companion.instance().getUser();
ApiClient.Companion.getInstance().getVocalVerifyApi().getVocalReset((user0 == null ? 0 : user0.getUserId())).compose(((ObservableTransformer)this.bindToLifecycle())).compose(ResponseTransformer.Companion.handleResult()).compose(SchedulerProvider.Companion.getInstance().applySchedulers()).subscribe(((Consumer)new SettingsActivity.getVocalReset.1(this)), ((Consumer)new SettingsActivity.getVocalReset.2(this)));
}
private final void getVocalStatus() {
User user0 = UnirunApplication.Companion.instance().getUser();
ApiClient.Companion.getInstance().getVocalVerifyApi().getVocalStatus((user0 == null ? ((int)0) : ((int)user0.getUserId()))).compose(((ObservableTransformer)this.bindToLifecycle())).compose(ResponseTransformer.Companion.handleResult()).compose(SchedulerProvider.Companion.getInstance().applySchedulers()).subscribe(((Consumer)new SettingsActivity.getVocalStatus.1(this)), ((Consumer)new SettingsActivity.getVocalStatus.2(this)));
}