2
0
mirror of https://gitee.com/hotlcc/wechat4j.git synced 2025-06-29 13:42:48 +08:00

提交代码

This commit is contained in:
hotlcc 2018-07-24 20:33:53 +08:00
parent 2144e6c9f9
commit 7944be3b97
6 changed files with 251 additions and 32 deletions

View File

@ -3,17 +3,27 @@ package com.hotlcc.wechat4j;
import com.alibaba.fastjson.JSONArray; import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject; import com.alibaba.fastjson.JSONObject;
import com.hotlcc.wechat4j.api.WebWeixinApi; import com.hotlcc.wechat4j.api.WebWeixinApi;
import com.hotlcc.wechat4j.enums.ExitTypeEnum;
import com.hotlcc.wechat4j.enums.LoginTipEnum; import com.hotlcc.wechat4j.enums.LoginTipEnum;
import com.hotlcc.wechat4j.util.CommonUtil;
import com.hotlcc.wechat4j.util.PropertiesUtil; import com.hotlcc.wechat4j.util.PropertiesUtil;
import com.hotlcc.wechat4j.util.QRCodeUtil; import com.hotlcc.wechat4j.util.QRCodeUtil;
import com.hotlcc.wechat4j.util.StringUtil; import com.hotlcc.wechat4j.util.StringUtil;
import org.apache.http.HttpException;
import org.apache.http.HttpRequest;
import org.apache.http.HttpRequestInterceptor;
import org.apache.http.HttpResponse;
import org.apache.http.client.CookieStore; import org.apache.http.client.CookieStore;
import org.apache.http.client.HttpClient; import org.apache.http.client.HttpClient;
import org.apache.http.conn.ConnectionKeepAliveStrategy;
import org.apache.http.impl.client.BasicCookieStore; import org.apache.http.impl.client.BasicCookieStore;
import org.apache.http.impl.client.DefaultConnectionKeepAliveStrategy;
import org.apache.http.impl.client.HttpClients; import org.apache.http.impl.client.HttpClients;
import org.apache.http.protocol.HttpContext;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import java.io.IOException;
import java.io.OutputStream; import java.io.OutputStream;
import java.io.PrintStream; import java.io.PrintStream;
import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.Lock;
@ -33,8 +43,6 @@ public class Wechat {
private CookieStore cookieStore; private CookieStore cookieStore;
private HttpClient httpClient; private HttpClient httpClient;
//在线状态
private volatile boolean isOnline = false;
//认证码 //认证码
private volatile String wxsid; private volatile String wxsid;
private volatile String passTicket; private volatile String passTicket;
@ -47,21 +55,51 @@ public class Wechat {
private final Lock SyncKeyLock = new ReentrantLock(); private final Lock SyncKeyLock = new ReentrantLock();
private volatile JSONArray ContactList; private volatile JSONArray ContactList;
private final Lock ContactListLock = new ReentrantLock(); private final Lock ContactListLock = new ReentrantLock();
//在线状态
private volatile boolean isOnline = false;
private final Lock isOnlineLock = new ReentrantLock();
//同步监听器
private volatile SyncMonitor syncMonitor;
public Wechat(CookieStore cookieStore) { public Wechat(CookieStore cookieStore) {
this.cookieStore = cookieStore; this.cookieStore = cookieStore;
this.httpClient = HttpClients.custom().setDefaultCookieStore(cookieStore).build(); this.httpClient = buildHttpClient(cookieStore);
} }
public Wechat() { public Wechat() {
this.cookieStore = new BasicCookieStore(); this(new BasicCookieStore());
this.httpClient = HttpClients.custom().setDefaultCookieStore(cookieStore).build();
} }
public void setWebWeixinApi(WebWeixinApi webWeixinApi) { public void setWebWeixinApi(WebWeixinApi webWeixinApi) {
this.webWeixinApi = webWeixinApi; this.webWeixinApi = webWeixinApi;
} }
private HttpClient buildHttpClient(CookieStore cookieStore) {
ConnectionKeepAliveStrategy keepAliveStrategy = new DefaultConnectionKeepAliveStrategy() {
@Override
public long getKeepAliveDuration(HttpResponse response, HttpContext context) {
long keepAlive = super.getKeepAliveDuration(response, context);
if (keepAlive == -1) {
//如果服务器没有设置keep-alive这个参数我们就把它设置成1分钟
keepAlive = 5000;
}
return keepAlive;
}
};
HttpRequestInterceptor interceptor = new HttpRequestInterceptor() {
@Override
public void process(HttpRequest httpRequest, HttpContext httpContext) throws HttpException, IOException {
httpRequest.addHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
}
};
HttpClient httpClient = HttpClients.custom()
.setDefaultCookieStore(cookieStore)
// .setKeepAliveStrategy(keepAliveStrategy)
.addInterceptorFirst(interceptor)
.build();
return httpClient;
}
/** /**
* 获取uuid登录时 * 获取uuid登录时
* *
@ -137,6 +175,8 @@ public class Wechat {
ps.println(); ps.println();
ps.flush(); ps.flush();
getWxUuid(ps, 0); getWxUuid(ps, 0);
CommonUtil.threadSleep(2000);
continue; continue;
} }
@ -224,7 +264,6 @@ public class Wechat {
passTicket = result.getString("pass_ticket"); passTicket = result.getString("pass_ticket");
skey = result.getString("skey"); skey = result.getString("skey");
wxuin = result.getString("wxuin"); wxuin = result.getString("wxuin");
isOnline = true;
ps.println("\t成功"); ps.println("\t成功");
ps.flush(); ps.flush();
@ -302,7 +341,7 @@ public class Wechat {
* *
* @return * @return
*/ */
private boolean wxInit(PrintStream ps, int time) { private boolean wxInitWithRetry(PrintStream ps, int time) {
ps.print("正在初始化数据..."); ps.print("正在初始化数据...");
ps.flush(); ps.flush();
@ -319,6 +358,8 @@ public class Wechat {
} }
ps.println(); ps.println();
ps.flush(); ps.flush();
CommonUtil.threadSleep(2000);
continue; continue;
} }
@ -330,6 +371,35 @@ public class Wechat {
return false; return false;
} }
/**
* 开启状态通知
*
* @param time
* @return
*/
private boolean statusNotify(int time) {
for (int i = 0; i < time; i++) {
JSONObject result = webWeixinApi.statusNotify(httpClient, passTicket, wxsid, skey, wxuin, getLoginUserName(false));
if (result == null) {
continue;
}
JSONObject BaseResponse = result.getJSONObject("BaseResponse");
if (result == null) {
continue;
}
int Ret = BaseResponse.getIntValue("Ret");
if (Ret != 0) {
continue;
}
return true;
}
return false;
}
/** /**
* 自动登录 * 自动登录
*/ */
@ -386,14 +456,25 @@ public class Wechat {
} }
// 3初始化数据 // 3初始化数据
if (!wxInit(ps, time)) { if (!wxInitWithRetry(ps, time)) {
ps.println("初始化数据失败"); ps.println("初始化数据失败,请重新登录");
ps.flush(); ps.flush();
return false;
} }
ps.println("微信登录成功,欢迎你:" + getLoginUserNickName(false)); ps.println("微信登录成功,欢迎你:" + getLoginUserNickName(false));
ps.flush(); ps.flush();
try {
isOnlineLock.lock();
statusNotify(time);
isOnline = true;
syncMonitor = new SyncMonitor();
syncMonitor.start();
} finally {
isOnlineLock.unlock();
}
return true; return true;
} }
@ -410,8 +491,96 @@ public class Wechat {
* 退出登录 * 退出登录
*/ */
public void logout() { public void logout() {
webWeixinApi.logout(httpClient, wxsid, skey, wxuin); try {
isOnline = false; isOnlineLock.lock();
webWeixinApi.logout(httpClient, wxsid, skey, wxuin);
isOnline = false;
} finally {
isOnlineLock.unlock();
}
}
/**
* 微信同步监听器心跳
*/
private class SyncMonitor extends Thread {
@Override
public void run() {
int time = PropertiesUtil.getIntValue("wechat4j.syncCheck.retry.time", 5);
int i = 0;
while (isOnline) {
long start = System.currentTimeMillis();
try {
//API调用异常导致退出
JSONObject result = webWeixinApi.syncCheck(httpClient, wxsid, skey, wxuin, getSyncKeyList(false));
logger.debug("微信同步监听心跳返回数据:{}", result);
if (result == null) {
throw new RuntimeException("微信API调用异常");
} else {
i = 0;
}
//人为退出
int retcode = result.getIntValue("retcode");
if (retcode != 0) {
logger.info("微信退出或从其它设备登录");
logout();
processExitEvent(ExitTypeEnum.REMOTE_EXIT, null);
return;
}
int selector = result.getIntValue("selector");
processSelector(selector);
} catch (Exception e) {
logger.error("同步监听心跳异常", e);
if (i == 0) {
logger.info("同步监听请求失败,正在重试...");
} else if (i > 0) {
logger.info("第{}次重试失败" + i);
}
if (i >= time) {
logger.info("重复{}次仍然失败,退出微信", i);
logout();
processExitEvent(ExitTypeEnum.ERROR_EXIT, e);
return;
}
i++;
}
//如果时间太短则阻塞2秒
long end = System.currentTimeMillis();
if (end - start < 2000) {
CommonUtil.threadSleep(2000);
}
}
processExitEvent(ExitTypeEnum.LOCAL_EXIT, null);
}
/**
* 处理退出事件
*/
private void processExitEvent(ExitTypeEnum type, Throwable t) {
try {
} catch (Exception e) {
logger.error("Exit event process error.", e);
}
}
/**
* 处理selector值
*
* @param selector
*/
private void processSelector(int selector) {
System.out.println(selector);
}
} }
/** /**
@ -482,7 +651,7 @@ public class Wechat {
* @param update * @param update
* @return * @return
*/ */
public JSONObject getSyncKey(boolean update) { private JSONObject getSyncKey(boolean update) {
if (SyncKey == null || update) { if (SyncKey == null || update) {
JSONObject result = webWeixinApi.webWeixinInit(httpClient, passTicket, wxsid, skey, wxuin); JSONObject result = webWeixinApi.webWeixinInit(httpClient, passTicket, wxsid, skey, wxuin);
if (result == null) { if (result == null) {
@ -513,6 +682,20 @@ public class Wechat {
return SyncKey; return SyncKey;
} }
/**
* 获取SyncKey的List
*
* @param update
* @return
*/
private JSONArray getSyncKeyList(boolean update) {
JSONObject SyncKey = getSyncKey(update);
if (SyncKey == null) {
return null;
}
return SyncKey.getJSONArray("List");
}
/** /**
* 获取联系人列表 * 获取联系人列表
* *
@ -644,8 +827,6 @@ public class Wechat {
} }
public void test() { public void test() {
System.out.println(SyncKey);
JSONObject result = webWeixinApi.webWeixinInit(httpClient, passTicket, wxsid, skey, wxuin);
System.out.println(result);
} }
} }

View File

@ -52,7 +52,6 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build());
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
@ -108,7 +107,6 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build());
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
@ -148,7 +146,6 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build());
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
@ -204,7 +201,6 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build());
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
@ -245,7 +241,6 @@ public class WebWeixinApi {
.render(); .render();
HttpPost httpPost = new HttpPost(url); HttpPost httpPost = new HttpPost(url);
httpPost.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpPost.setHeader("Content-type", ContentType.APPLICATION_FORM_URLENCODED.toString()); httpPost.setHeader("Content-type", ContentType.APPLICATION_FORM_URLENCODED.toString());
HttpEntity paramEntity = new UrlEncodedFormEntity(pairList); HttpEntity paramEntity = new UrlEncodedFormEntity(pairList);
@ -270,7 +265,6 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build());
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
@ -304,7 +298,6 @@ public class WebWeixinApi {
.render(); .render();
HttpPost httpPost = new HttpPost(url); HttpPost httpPost = new HttpPost(url);
httpPost.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString()); httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString());
JSONObject paramJson = new JSONObject(); JSONObject paramJson = new JSONObject();
@ -345,7 +338,6 @@ public class WebWeixinApi {
.render(); .render();
HttpPost httpPost = new HttpPost(url); HttpPost httpPost = new HttpPost(url);
httpPost.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString()); httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString());
JSONObject paramJson = new JSONObject(); JSONObject paramJson = new JSONObject();
@ -394,8 +386,10 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent")); RequestConfig config = RequestConfig.custom().
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); setRedirectsEnabled(false)
.build();
httpGet.setConfig(config);
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
int statusCode = response.getStatusLine().getStatusCode(); int statusCode = response.getStatusLine().getStatusCode();
@ -437,7 +431,6 @@ public class WebWeixinApi {
.render(); .render();
HttpGet httpGet = new HttpGet(url); HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build()); httpGet.setConfig(RequestConfig.custom().setRedirectsEnabled(false).build());
HttpResponse response = httpClient.execute(httpGet); HttpResponse response = httpClient.execute(httpGet);
int statusCode = response.getStatusLine().getStatusCode(); int statusCode = response.getStatusLine().getStatusCode();
@ -471,7 +464,6 @@ public class WebWeixinApi {
.render(); .render();
HttpPost httpPost = new HttpPost(url); HttpPost httpPost = new HttpPost(url);
httpPost.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString()); httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString());
JSONObject paramJson = new JSONObject(); JSONObject paramJson = new JSONObject();
@ -514,7 +506,6 @@ public class WebWeixinApi {
.render(); .render();
HttpPost httpPost = new HttpPost(url); HttpPost httpPost = new HttpPost(url);
httpPost.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString()); httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString());
JSONObject paramJson = new JSONObject(); JSONObject paramJson = new JSONObject();
@ -557,7 +548,6 @@ public class WebWeixinApi {
.render(); .render();
HttpPost httpPost = new HttpPost(url); HttpPost httpPost = new HttpPost(url);
httpPost.setHeader("User-Agent", PropertiesUtil.getProperty("wechat4j.userAgent"));
httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString()); httpPost.setHeader("Content-type", ContentType.APPLICATION_JSON.toString());
JSONObject paramJson = new JSONObject(); JSONObject paramJson = new JSONObject();

View File

@ -0,0 +1,17 @@
package com.hotlcc.wechat4j.enums;
/**
* 微信退出类型
*/
public enum ExitTypeEnum {
ERROR_EXIT("错误导致退出"),
LOCAL_EXIT("本次手动退出"),
REMOTE_EXIT("远程操作退出");
private String desc;
ExitTypeEnum(String desc) {
this.desc = desc;
}
}

View File

@ -0,0 +1,27 @@
package com.hotlcc.wechat4j.util;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public final class CommonUtil {
private static Logger logger = LoggerFactory.getLogger(CommonUtil.class);
private CommonUtil() {
}
public static void threadSleep(long millis, int nanos) {
try {
Thread.sleep(millis, nanos);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
public static void threadSleep(long millis) {
try {
Thread.sleep(millis);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
}

View File

@ -5,4 +5,6 @@ wechat4j.userAgent=Mozilla/5.0 (Macintosh; Intel Mac OS X 10_11_6) AppleWebKit/5
# QRCode图片临时文件前缀 # QRCode图片临时文件前缀
wechat4j.qrcode.tmpfile.prefix=wechat4j_tmp_ wechat4j.qrcode.tmpfile.prefix=wechat4j_tmp_
# 全局重试次数 # 全局重试次数
wechat4j.retry.time=3 wechat4j.retry.time=3
# 同步监听请求重试次数
wechat4j.syncCheck.retry.time=5

View File

@ -1,13 +1,15 @@
import com.hotlcc.wechat4j.Wechat; import com.hotlcc.wechat4j.Wechat;
import com.hotlcc.wechat4j.api.WebWeixinApi; import com.hotlcc.wechat4j.api.WebWeixinApi;
import com.hotlcc.wechat4j.util.CommonUtil;
public class TestClass2 { public class TestClass2 {
public static void main(String[] args) { public static void main(String[] args) {
WebWeixinApi api = new WebWeixinApi(); WebWeixinApi api = new WebWeixinApi();
Wechat wechat = new Wechat(); Wechat wechat = new Wechat();
wechat.setWebWeixinApi(api); wechat.setWebWeixinApi(api);
System.out.println(wechat.autoLogin()); wechat.autoLogin();
wechat.test(); wechat.test();
CommonUtil.threadSleep(1000 * 60 * 10);
wechat.logout(); wechat.logout();
} }
} }