Android模拟登录V2EX

最近在撸一个V2EX的客户端,官方API缺少一些功能如登录,发帖等,撸完官方API总觉得少了什么,本篇文章主要通过模拟登录实现一些官方没提供API的功能

观察登录传输的数据

在网页上登录帐号,通过chrome的调试模式可以看到 ,我们传了4个数据给服务器,分别是帐号,密码,once,和next,once是用来验证是否人为操作的标志,POST时必须带上这个字段,否则会认为是非人为操作而被禁止访问。

Paste_Image.png

知道了需要传什么参数后,退出登陆,回到登录之前的页面,https://www.v2ex.com/signin, 我们需要POST的数据都需要在这个页面里捉取下来

Paste_Image.png

代码实现

1.添加请求头
在chrome登陆成功页面查看POST的请求头,在OkHttp中相应进行配置。

        mClient = new OkHttpClient.Builder()
                .cookieJar(new CustomCookieJar(new CookieManager(mPersistentCookieStore, CookiePolicy.ACCEPT_ALL)))
                .readTimeout(12, TimeUnit.SECONDS)
                .addInterceptor(htmlInterceptor)
                .addInterceptor(loggingInterceptor)
                .build();

    private Interceptor htmlInterceptor = new Interceptor() {
        @Override
        public Response intercept(Chain chain) throws IOException {
            Request newRequest = chain.request().newBuilder()
                    .addHeader("Origin", "https://www.v2ex.com")
                    .addHeader("Content-Type", "application/x-www-form-urlencoded")
                    .build();
            return chain.proceed(newRequest);
        }
    };

2.定义登录接口api
分别定义get和Post 两种,get取得帐号,密码,once的值,用于POST提交

    @GET("signin")      //登录
    Observable<String> login();

    @FormUrlEncoded
    @Headers("Referer: https://www.v2ex.com/signin")
    @POST("signin")
    Observable<String> postLogin(@FieldMap HashMap<String, String> hashMap);

3获取数据,发送登陆请求

        Subscription subscription = LoginService.getInstance().auth().login()
                .map(new Func1<String, HashMap>() {
                    @Override
                    public HashMap call(String stringResponse) {
                        return JsoupUtil.parseUserNameAndPwd(stringResponse, username, password);
                    }
                }).flatMap(new Func1<HashMap, Observable<String>>() {
                    @Override
                    public Observable<String> call(HashMap requestMap) {
                        return LoginService.getInstance().auth().postLogin(requestMap);
                    }
                }).map(new Func1<String, LoginResult>() {
                    @Override
                    public LoginResult call(String response) {
                        String errorMsg = ApiErrorUtil.getErrorMsg(response);
                        if (errorMsg == null) {
                            return JsoupUtil.parseLoginResult(response);
                        } else {
                            LoginResult result = new LoginResult();
                            result.setMessage(errorMsg);
                            return result;
                        }
                    }
                })
                .subscribeOn(Schedulers.io())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(observer);

这里使用了Jsoup来解析html可以很便捷的获取需要的数据,没了解过的同学可以查看这里Jsoup开发指南

登录成功后返回的html里面就有用户的数据了

Paste_Image.png

Cookie

当你完成已上步骤还是无法获取个人信息,会自动跳转到登陆页面,因为http请求是无状态的,服务器并不知道你已经登录了,这里就需要配置Cookie,登录成功后,服务会返回一个Cookie(通行证),客户端需保存这个Cookie,再次发送请求时带上这个Cookie,服务器就能识别该用户,并且返回用户相关的信息了。
在OKHttp3中配置Cookie持久化只需一句话

.cookieJar(new CustomCookieJar(new CookieManager(mPersistentCookieStore, CookiePolicy.ACCEPT_ALL)))

实现Cookie后就可以获取所有数据了,如用户关注节点等。

对cookie不了解的可以看这里 Cookie/Session机制详解
在OkHttp中如何实现Cookie的持久化可以看这里OkHttp3之Cookies管理及持久化

以上完整的代码在这里 https://github.com/sooola/V2EX-Android ,RxJava &Retrofit 的V2EX第三方客户端,项目在龟速更新中,有任何建议欢迎issues,开源不易,路过求start鼓励下(⊙ᗜ⊙)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 174,167评论 25 709
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,199评论 19 139
  • 一说到REST,我想大家的第一反应就是“啊,就是那种前后台通信方式。”但是在要求详细讲述它所提出的各个约束,以及如...
    时待吾阅读 3,483评论 0 19
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,395评论 11 349
  • 大家好,欢迎您加入幸福心理学读书群! 我是名医务工作者,是国家二级心理咨询师,多年来一直坚持学习,并自我成长,学习...
    琥珀1阅读 496评论 0 0