login_ad.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. from selenium import webdriver
  2. from selenium.webdriver import ActionChains
  3. from selenium.webdriver.common.by import By
  4. from selenium.webdriver.support import expected_conditions as EC
  5. from selenium.webdriver.support.wait import WebDriverWait
  6. from wechat_action.create_ad_plan_idea import IdeaAction
  7. from wechat_action.human_ad import HumanAd
  8. from selenium.webdriver import ChromeOptions
  9. from selenium.webdriver.common.keys import Keys
  10. from sqlalchemy import Table
  11. from wechat_action import sql_tools
  12. import time
  13. import pickle
  14. from settings import using_config
  15. from wechat_action.sql_models import DB
  16. import requests
  17. import json
  18. import logging
  19. import re
  20. class LogIn:
  21. # TODO:整体运行使用逻辑,需要修改几次
  22. def __init__(self, user_id):
  23. # 获取到单独服务商下的独立公众号页面
  24. self.user_id = user_id
  25. self.driver = self.get_driver()
  26. self.db = DB(config=using_config)
  27. self.wechat_cookies_table = Table('wechat_cookies', self.db.metadata,
  28. autoload=True, autoload_with=self.db.engine)
  29. # self.log_in()
  30. # self.select_ad_master()
  31. def get_driver(self):
  32. options = ChromeOptions()
  33. # 防止selenium快速崩坏
  34. options.add_argument("--disable-dev-shm-usage")
  35. options.add_experimental_option('excludeSwitches', ['enable-automation'])
  36. prefs = {"profile.managed_default_content_settings.images": 2, 'permissions.default.stylesheet': 2}
  37. options.add_experimental_option("prefs", prefs)
  38. driver = webdriver.Remote(
  39. # command_executor='http://192.168.0.103/wd/hub',
  40. # command_executor='http://118.31.53.105:4555/wd/hub',
  41. command_executor='http://47.96.116.151/wd/hub',
  42. options=options)
  43. # driver = webdriver.Chrome(options=options)
  44. driver.maximize_window()
  45. return driver
  46. def log_in(self):
  47. # self.wechat_cookie_use()
  48. logging.info('开始登录')
  49. self.driver.get('https://a.weixin.qq.com/index.html')
  50. img_selector = 'body > div.old-template > div > div > div.waiting.panelContent > div.wrp_code > img'
  51. frame_login = self.driver.find_element_by_xpath('//*[@id="login_container"]/iframe')
  52. self.driver.switch_to.frame(frame_login)
  53. # WebDriverWait(driver, 3).until(EC.invisibility_of_element_located((By.CSS_SELECTOR, img_selector)))
  54. # time.sleep(3)
  55. img_url = self.driver.find_element_by_css_selector(img_selector)
  56. return img_url.get_attribute('src')
  57. def log_in_wait(self):
  58. # 默认等待6分钟
  59. WebDriverWait(self.driver, 6 * 60).until(lambda driver: self.driver.find_elements_by_link_text('广告投放'))
  60. logging.info('登录成功')
  61. @staticmethod
  62. def cookies_save(log_ad, sql_session):
  63. logging.info('update db cookie')
  64. # 切换窗口,点击创建广告,切到广告页面
  65. log_ad.driver.switch_to.window(log_ad.driver.window_handles[0])
  66. WebDriverWait(log_ad.driver, 100).until(lambda driver: driver.find_element_by_class_name(
  67. 'ui-mr-medium'))
  68. wechat_cookies, wechat_id_tmp = log_ad.wechat_cookie_pickle()
  69. wechat_id = sql_tools.get_wechat_id_from_cookies(log_ad.user_id, sql_session)
  70. if wechat_id:
  71. if wechat_id_tmp != wechat_id:
  72. raise ValueError("微信账号,非以前老账号,登录失败")
  73. # cookie 进行数据库保存
  74. update_res = log_ad.wechat_cookies_table.update() \
  75. .where(log_ad.wechat_cookies_table.c.user_id == log_ad.user_id) \
  76. .values(cookies=wechat_cookies,
  77. scan_action='done')
  78. sql_session.execute(update_res)
  79. sql_session.commit()
  80. else:
  81. wechat_cookies_info = {'user_id': log_ad.user_id, 'cookies': wechat_cookies, 'wechat_id': wechat_id_tmp,
  82. 'scan_action': 'done'}
  83. wechat_insert = sql_tools.save_wechat_cookies(wechat_cookies_info=wechat_cookies_info,
  84. table_wechat_cookies=log_ad.wechat_cookies_table)
  85. sql_session.execute(wechat_insert)
  86. sql_session.commit()
  87. log_ad.driver.switch_to.window(log_ad.driver.window_handles[-1])
  88. logging.info('update db cookies over')
  89. def select_ad_master(self, service_name, wechat_name, sql_session):
  90. logging.info('开始切换服务商')
  91. WebDriverWait(self.driver, 100).until(
  92. lambda driver: self.driver.find_element_by_xpath('//*[@id="root"]/div/header/div/div[3]/div/div[1]'))
  93. tmp_service_name = self.driver.find_element_by_xpath('//*[@id="root"]/div/header/div/div[3]/div/div[1]')
  94. if service_name != tmp_service_name.text:
  95. self.driver.execute_script('''
  96. window.scroll(0,1000000);
  97. ''')
  98. self.driver.find_element_by_css_selector(
  99. '#root > div > header > div > div.CoreLayout__account-2lIr0 > div').click()
  100. time.sleep(0.1)
  101. self.driver.find_element_by_css_selector(
  102. '#root > div > div.CoreLayout__headerDropdown-3xWkD > div > div:nth-child(1) > button').click()
  103. service_names = self.driver.find_elements_by_class_name('CoreLayout__headerDropdownItem-X4S98')
  104. choice_service = None
  105. for _ in service_names:
  106. if service_name in _.text:
  107. choice_service = _
  108. choice_service.click()
  109. # 挑选广告投放位置
  110. WebDriverWait(self.driver, 100).until(
  111. lambda driver: self.driver.find_element_by_xpath(
  112. '//*[@id="root"]/div/header/div/div[3]/div/div[1]').text == service_name)
  113. input_wechat_name = self.driver.find_element_by_class_name('TextInput_new__iconRight-pekjS')
  114. input_wechat_name.click()
  115. input_wechat_name.send_keys(wechat_name)
  116. input_wechat_name.send_keys(Keys.RETURN)
  117. self.driver.execute_script('''
  118. window.scroll(100000,1000000);
  119. var e_one=document.getElementsByClassName('Table_new__wrapper-1cpZN')[0];
  120. e_one.scroll(10000,100000);
  121. ''')
  122. for i in range(10):
  123. try:
  124. WebDriverWait(self.driver, 100).until(
  125. lambda driver: len([self.driver.find_elements_by_link_text('广告投放')]) == 1)
  126. WebDriverWait(self.driver, 100).until(
  127. lambda driver: len([_ for _ in self.driver.find_elements_by_link_text('广告投放') if
  128. _.is_enabled() and _.is_displayed()]) == 1)
  129. elements = self.driver.find_elements_by_link_text('广告投放')
  130. elements[0].click()
  131. WebDriverWait(self.driver, 100).until(lambda driver: len(self.driver.window_handles) > 1)
  132. logging.info('切换服务商成功')
  133. self.cookies_save(self, sql_session)
  134. break
  135. except Exception as e:
  136. pass
  137. if i == 10:
  138. raise ValueError('切换服务商出错')
  139. @staticmethod
  140. def get_cookie(driver, login_cookie=True):
  141. if login_cookie:
  142. WebDriverWait(driver, 100).until(
  143. lambda x: [True for _ in driver.get_cookies() if 'token_ticket' == _['name']]
  144. )
  145. cookies = driver.get_cookies()
  146. cookie_dict = {}
  147. for _ in cookies:
  148. cookie_dict[_['name']] = _['value']
  149. return cookie_dict
  150. def wechat_cookie_pickle(self):
  151. wechat_id = None
  152. self.driver.get('https://a.weixin.qq.com/client')
  153. WebDriverWait(self.driver, 100).until(
  154. lambda x: [True for _ in self.driver.get_cookies() if 'token_ticket' == _['name']]
  155. )
  156. WebDriverWait(self.driver, 100).until(
  157. lambda x: re.findall('g_tk=(\d+)&', self.driver.page_source)
  158. )
  159. cookies = self.driver.get_cookies()
  160. cookies_obj = pickle.dumps(cookies)
  161. g_tk = re.findall('g_tk=(\d+)&', self.driver.page_source)[0]
  162. wechat_cookies = self.get_cookie(self.driver)
  163. wechat_url = 'https://a.weixin.qq.com/cgi-bin/agency/check_login?g_tk={g_tk}&_={time_p}'.format(g_tk=g_tk,
  164. time_p=int(
  165. time.time() * 1000))
  166. rsp = requests.get(url=wechat_url, cookies=wechat_cookies)
  167. wechat_id = rsp.json()['data'][0]['wx_id']
  168. return cookies_obj, wechat_id
  169. def wechat_cookies_check_alive(self, driver_cookies):
  170. # wechat 检查cookies 是否可用
  171. # 可用返回ture
  172. self.driver.get('https://www.baidu.com')
  173. self.driver.get('https://a.weixin.qq.com/client')
  174. for _ in driver_cookies:
  175. self.driver.add_cookie(_)
  176. self.driver.get('https://a.weixin.qq.com/index.html')
  177. result = False
  178. try:
  179. WebDriverWait(self.driver, 5).until(
  180. lambda driver: self.driver.find_elements_by_xpath('//*[@class="headerInner"]'))
  181. except:
  182. result = True
  183. return result
  184. def upadte_user_info(self):
  185. # TODO: 更新 用户相关信息
  186. # 每次登录就更新一次相关数据,------公众号相关数据,人群报相关数据
  187. pass
  188. def get_driver_loged(self):
  189. return self.driver
  190. def refresh_driver(self):
  191. err_num = 0
  192. logging.info('开始刷新chrome')
  193. while True:
  194. try:
  195. if len(self.driver.window_handles) > 1:
  196. self.driver.switch_to.window(self.driver.window_handles[-1])
  197. self.driver.execute_script('window.close();')
  198. time.sleep(1)
  199. # 规避有弹窗的情况
  200. self.driver.switch_to.alert.accept()
  201. else:
  202. self.driver.switch_to.window(self.driver.window_handles[-1])
  203. self.driver.get('https://a.weixin.qq.com')
  204. break
  205. except Exception as e:
  206. logging.error(e)
  207. err_num = err_num + 1
  208. if err_num > 3:
  209. break
  210. logging.info('刷新chrome 结束')