专栏首页SeanCheney的专栏Python模拟登陆 —— 征服验证码 3 CSDN

Python模拟登陆 —— 征服验证码 3 CSDN

CSDN使用了spring的webflow

<input type="hidden" name="lt" value="LT-58187-6D0O7tpTNBVqzLTbyULoyeLt9fMZd1" />

这个隐藏参数可以理解成每个需要登录的用户都有一个流水号。只有有了webflow发放的有效的流水号,用户才可以说明是已经进入了webflow流程。否则,没有流水号的情况下,webflow会认为用户还没有进入webflow流程,从而会重新进入一次webflow流程,从而会重新出现登录界面。

import requests
from bs4 import BeautifulSoup
#######################################################
#  Replace special characters in string using the %xx escape. Letters, digits,
#          and the characters '_.-' are never quoted.
# Python 3 可以这么引入
from urllib.parse import quote


#### 控制台输入账号密码相关, 如有特殊密码保护,应使用getpass(需要注意的是Pycharm对此支持的不好)
import getpass


# Python 2 需要这么引入
# import urllib.quote

class CSDN(object):
    """
    CSDN模拟登陆并加上点赞,评论,私信等功能。
    """
    def __init__(self, headers):
        """
        :param session: 创建全局的session对象,保证会话的一致性,有效性。
        :param headers: 防止服务器端反爬虫,添加伪装头部信息
        """
        self.session = requests.Session()
        self.headers = headers

    def login(self, account, passwd):
        """
        模拟登陆,点赞, 发评论,发私信的前提都是已在登录状态下进行的,这是前提。
        :param account: 用户名
        :param passwd:  密码
        :return:
        """

        self.username = account
        self.password = passwd

        # 只有获取到webflow流水号,才会正式进入登陆通道,服务器端对此进行了限制。
        lt, execution = self.get_webflow()
        # 要提交的表单数据
        postdata = {
            'username': account,
            'password': passwd,
            "lt": lt,
            "execution": execution,
            "_eventId": "submit"
        }

        ## 开始登陆
        loginurl = 'https://passport.csdn.net/account/login'
        response = self.session.post(url=loginurl, headers=self.headers, data=postdata)
        if response.status_code == 200:
            print('恭喜您登陆成功!')
        else:
            print(response.text)
            print('登录失败,请重试!')


    def get_webflow(self):
        """
        流水号webflow获取。随便访问包含登陆页链接的CSDN网页就可以得到这串数据。应为是动态变化的
        所以,先获取下来,以备使用。
        :return:
        """
        url = 'https://passport.csdn.net/account/login?ref=toolbar'
        response = self.session.get(url=url, headers=self.headers)
        soup = BeautifulSoup(response.text, 'html.parser')
        lt = soup.find('input', {'name': 'lt'})['value']
        execution = soup.find('input', {'name': 'execution'})['value']
        # 释放不必要的对象
        soup.clear()
        return (lt, execution)

    def digg(self, articleurl, digg=True):
        """
        把给定的文章路径 http://blog.csdn.net/marksinoberg/article/details/69569353
        先转化一下为: http://blog.csdn.net/marksinoberg/article/digg?ArticleId=69569353

        :param articleurl 待操作的文章路径
        :param digg: 给文章点赞还是踩一下
        :return:
        """
        try:
            bloguser, blogid = articleurl.split('/')[3], articleurl.split('/')[-1]
            if digg==True:
                diggurl = 'http://blog.csdn.net/{}/article/digg?ArticleId={}'.format(bloguser, blogid)
            else:
                diggurl = 'http://blog.csdn.net/{}/article/bury?ArticleId={}'.format(bloguser, blogid)
        except:
            print('您输入的文章路径非法!')
        print("待操作文章的路径: ", diggurl)
        self.headers['X-Requested-With'] = 'XMLHttpRequest'
        self.headers['Host'] = 'blog.csdn.net'
        self.headers['Referer'] = articleurl

        response = self.session.get(url=diggurl, headers=self.headers)
        if response.status_code == 200:
            # print("Digg 操作成功", response.text)
            articlejson = response.json()
            digg, bury = articlejson['digg'], articlejson['bury']
            print('文章:{}\n:被点赞数:{}, 被踩数:{}'.format(articleurl, digg, bury))

        else:
            print('网络或服务器出现了问题,点赞操作出现了点故障!')
        self.headers['Referer'] = ''

    def comment(self, articleurl, content):
        """
        给定一个文章的路径http://blog.csdn.net/marksinoberg/article/details/69569353,
        需要转化为形如: http://blog.csdn.net/Marksinoberg/comment/submit?id=69569353
        :param articleurl:
        :param content:
        :return:
        """
        try:
            bloguser, blogid = articleurl.split('/')[3], articleurl.split('/')[-1]
            commenturl = 'http://blog.csdn.net/{}/comment/submit?id={}'.format(bloguser, blogid)
        except:
            print(commenturl, ' 不是一个合法的路径!')
        print(commenturl)
        postdata = {
            'commentid': self.username,
            'replyid': bloguser,
            'content': content
        }
        response = self.session.post(url=commenturl, headers=self.headers, data=postdata)
        if response.status_code == 200:
            print(response.json())
            if response.json()['result'] == 1:
                print('评论成功咯!')
            else:
                print('服务器访问成功,但评论操作失败了!')
        else:
            print('评论出现了点异常!')

    def letter(self, receiver, content):
        letterurl = 'http://msg.csdn.net/letters/send_message?receiver={0}&body={1}'.format(receiver, quote(content))
        response = self.session.get(url=letterurl, headers=self.headers)
        if response.status_code == 200:
            print("私信内容发送成功!")
            # print(response.text)
            ## 这里服务器返回的是一大串HTML代码。通过解析还可以得到本人和其他博友的私信记录。
        else:
            print('私信发送失败!请检查网络是否通畅。')


    def publish_article(self):
        """
        核心URL: http://write.blog.csdn.net/postedit?edit=1&isPub=1&joinblogcontest=undefined&r=0.14573376474383326
        有兴趣的可以尝试着做一下。

        需要提交的表单信息为:
                titl:1234567                           # 博客标题
                typ:1                                  # 原创1, 转载2, 翻译3
                cont:自动发现一篇好文章的前提是什么?    #  发表的文章内容
                desc:自动发现一篇好文章的前提是什么?    # 发表的摘要信息
                tags:                                  # 标签们
                flnm:69803183                          # 博客ID(如果是新文章默认没有的)
                chnl:0                                 # 文章类型: 编程语言啊, 系统架构啊什么的
                comm:2
                level:0
                tag2:
                artid:0
                checkcode:undefined
                userinfo1:713
                stat:publish
        :return:
        """

if __name__ == '__main__':
    headers = {
        "User-Agent": "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/49.0.2623.87 Safari/537.36",
    }
    csdn = CSDN(headers=headers)
    account = input('请输入您的账号:')
    password = getpass.getpass(prompt='请输入您的密码:')
    csdn.login(account=account, passwd=password)
    # 评论测试
    # csdn.comment('http://blog.csdn.net/marksinoberg/article/details/69569353', '哈哈,爬虫评论')
    # 点赞,踩测试。digg为True为顶, 为False即踩
    # csdn.digg('http://blog.csdn.net/marksinoberg/article/details/69569353', digg=True)
    # 私信测试
    # csdn.letter(receiver='marksinoberg', content='僚机呼叫主机,收到请回复,Over!')

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 使用Newspaper框架抓取新闻

    推荐安装Python3版本:pip3 install newspaper3k (pip install newspaper是Python2版本)

    SeanCheney
  • 《Scikit-Learn、Keras与TensorFlow机器学习实用指南(第二版)》第19章 规模化训练和部署TensorFlow模型

    有了能做出惊人预测的模型之后,要做什么呢?当然是部署生产了。这只要用模型运行一批数据就成,可能需要写一个脚本让模型每夜都跑着。但是,现实通常会更复杂。系统基础组...

    SeanCheney
  • 《Learning Scrapy》(中文版)0 序言

    Dimitris Kouzis – Loukas有超过15年的软件开发经历。同时他也参与到教学活动中,受众广泛。

    SeanCheney
  • 类方法,静态方法和实例方法

    2.类名的规范是: 1 数字字母下划线组成 2 不能以数字开头 3 首字母大写,驼峰命名 (例如:TestResult)

    清菡
  • [WordPress主题]2019大前端dux6.0最新无限制版

    DUX主题是基于WordPress程序的主题,由themebetter团队原创开发,是目前比较火的wordpress主题,和大前端主题一样比较优秀的主题还有be...

    爱游博客
  • Fetch API 教程

    fetch()是 XMLHttpRequest 的升级版,用于在 JavaScript 脚本里面发出 HTTP 请求。

    ruanyf
  • win2003 64位系统IIS配置方法

    64位系统IIS配置方法,主要是在 32与64位下运行iis的一些常见问题与方法,推荐服务器配置者参考。1、因用模版安装ASP.NET 2.0需先卸载64位的....

    习惯说一说
  • Flask(从wsgi开始 一)

    zx钟
  • python面向对象之反射

    类方法:有个默认参数cls,并且可以直接用类名去调用,可以与类属×××互(也就是可以使用类属性)

    py3study
  • PE解析器的编写(四)——数据目录表的解析

    在PE结构中最重要的就是区块表和数据目录表,上节已经说明了如何解析区块表,下面就是数据目录表,在数据目录表中一般只关心导入表,导出表和资源这几个部分,但是资源实...

    Masimaro

扫码关注云+社区

领取腾讯云代金券