ckLogin.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. package login
  2. import (
  3. "context"
  4. "encoding/base64"
  5. "encoding/json"
  6. "fmt"
  7. "log"
  8. "strings"
  9. "time"
  10. "git.listensoft.net/tool/jspkit/common"
  11. "git.listensoft.net/tool/jspkit/common/models"
  12. "git.listensoft.net/tool/jspkit/logger"
  13. "git.listensoft.net/tool/jspkit/taxerr"
  14. "github.com/forgoer/openssl"
  15. "github.com/go-rod/rod"
  16. "github.com/go-rod/rod/lib/proto"
  17. "github.com/go-rod/rod/lib/utils"
  18. "github.com/tidwall/gjson"
  19. )
  20. func CheckSessionAndLogin(ctx context.Context, p *rod.Page, BaseUri string, info models.CompanyInfo) (err error) {
  21. //return taxerr.NewUser("税局代理机构登录存在风险,预计6号恢复,请等待税局修复后使用")//12.6 09:20左右 确认完了
  22. if info.Dlfs == "代理登录" && info.IdNo == "" {
  23. return taxerr.DlTaxNoError
  24. }
  25. return AutoCheckSessionAndLogin(ctx, p, BaseUri, info)
  26. }
  27. func LoadCookie(sessionInfos models.SsoSession, p *rod.Page, BaseUri string, info models.CompanyInfo) (name string, err error) {
  28. err = rod.Try(func() {
  29. defer AddErr()
  30. p.Navigate("")
  31. _ = p.Timeout(common.ClickTimeOut).MustNavigate(areaURLmap[info.Area+".pro"]).StopLoading() // 此时必须得去tpass域名 要不然写入cookie后进不去发票
  32. p.MustSetCookies(sessionInfos.Cookies...)
  33. {
  34. p.MustEval(`() => localStorage.clear()`)
  35. utils.Sleep(1)
  36. // 设置 localStorage 在第二个页面中,逐条设置
  37. logger.Info("Setting localStorage in second browser...")
  38. for key, value := range sessionInfos.LocalStorage {
  39. if key == "logConfig" {
  40. continue
  41. }
  42. script := fmt.Sprintf(`() => localStorage.setItem('%s', '%s')`, key, value)
  43. _, err := p.Eval(script)
  44. if err != nil {
  45. log.Printf("Error setting localStorage %s=%s: %v", key, value, err)
  46. }
  47. }
  48. }
  49. utils.Sleep(1)
  50. if info.Area == "shaanxi" {
  51. //陕西地区必须带后面/才能登录上去
  52. p.Timeout(common.ClickTimeOut).MustNavigate(BaseUri + "/loginb/")
  53. } else {
  54. p.Timeout(common.ClickTimeOut).MustNavigate(BaseUri + "/loginb")
  55. }
  56. utils.Sleep(4)
  57. _ = rod.Try(func() {
  58. p.Timeout(common.LoadTimeout * 2).MustWaitStable()
  59. p.Timeout(common.LoadTimeout * 2).MustWaitDOMStable()
  60. })
  61. //p.MustStopLoading()
  62. logger.Info("加载完成")
  63. if info.Area == "hebei" && !common.MustHasXV(p, `//div[@class='leftMain']//div[contains(@class,'title')]`) {
  64. logger.Info("河北重新访问loginb")
  65. p.Timeout(common.ClickTimeOut).MustNavigate(`https://tpass.hebei.chinatax.gov.cn:8443/#/login?redirect_uri=https%3A%2F%2Fetax.hebei.chinatax.gov.cn%3A8443%2Fmhzx%2Fapi%2Fmh%2Ftpass%2Fcode&client_id=ap5bfdavp66a49979apa5848fpaa5353&response_type=code&state=8f8f3db7ecb948f08b1da177cda40854&client_pid=ap5bfdavp66a49979apa5848fpaa5353`)
  66. utils.Sleep(4)
  67. p.MustWaitStable()
  68. p.MustWaitDOMStable()
  69. }
  70. if common.MustHasXV(p, "//div[text()=' 关于电子税务局“新办智能开业”的通知 ']") {
  71. p.Timeout(common.ClickTimeOut).MustSearch("//span[text()='我知道了']").MustClick()
  72. utils.Sleep(3)
  73. }
  74. if p.MustHasX(`//div[@class='leftMain']//div[contains(@class,'title')]`) {
  75. _ = p.Timeout(common.ClickTimeOut).MustElementX(`//div[@class='leftMain']//div[contains(@class,'title')]`).WaitStable(300 * time.Millisecond)
  76. if *p.Timeout(common.ClickTimeOut).MustElementX(`//div[@class='leftMain']//div[contains(@class,'title')]`).MustAttribute(`title`) == "" {
  77. logger.Info("公司名空刷新")
  78. p.Reload()
  79. utils.Sleep(4)
  80. p.MustWaitDOMStable()
  81. }
  82. }
  83. if common.MustHasXV(p, `//div[@class='leftMain']//div[contains(@class,'title')]`) {
  84. name = *common.WaitElementX(p, `//div[@class='leftMain']//div[contains(@class,'title')]`, 0.5).MustAttribute(`title`)
  85. if name == "" {
  86. err = taxerr.New("代理登录切换公司后异常公司名")
  87. return
  88. }
  89. return
  90. }
  91. DlSy := false //登录完是代理登录的首页
  92. if common.MustHasXV(p, `//a[@title="单户办理"]`) {
  93. common.MustElementX(p, `//a[@title="单户办理"]`).MustClick()
  94. DlSy = true
  95. utils.Sleep(3)
  96. }
  97. if !strings.Contains(p.MustInfo().URL, "tpass") {
  98. panic(taxerr.New("cookie失效"))
  99. }
  100. if !p.MustHasR(`[class="s_Breadcrumb"]`, "企业办税") {
  101. panic(taxerr.New("cookie失效"))
  102. }
  103. if !DlSy {
  104. r := p.MustWaitRequestIdle()
  105. p.Timeout(common.ClickTimeOut).MustSearch("返回").MustClick()
  106. r()
  107. } else {
  108. //这里需要手动切换到首页
  109. common.MustInputX(p, `//input[@placeholder="请输入统一社会信用代码"]`, info.TaxNo)
  110. common.MustElementX(p, `//span[text()="查询"]`).MustClick()
  111. utils.Sleep(2)
  112. tbody := common.MustElementsX(p, `//div[@class="el-table__body-wrapper is-scrolling-none"]//table//tr`)
  113. if len(tbody) == 0 {
  114. common.MustInputX(p, `//input[@placeholder="请输入统一社会信用代码"]`, "")
  115. p.Timeout(common.ClickTimeOut).MustElementX(`//input[@placeholder="请输入纳税人名称"]`).MustInput(info.ComName)
  116. common.MustElementX(p, `//span[text()="查询"]`).MustClick()
  117. utils.Sleep(2)
  118. tbody = common.MustElementsX(p, `//div[@class="el-table__body-wrapper is-scrolling-none"]//table//tr`)
  119. if len(tbody) == 0 {
  120. panic(taxerr.NewUserV3("未在当前账号下["+info.Zzridno+"]找到该企业", "请检查手机号与企业的关联关系"))
  121. }
  122. panic(taxerr.NewUserV3("该公司账号跟公司名称不匹配", "请检查后重新发起"))
  123. } else if len(tbody) == 1 {
  124. common.MustElementX(p, `//span[text()='进入']`).MustClick()
  125. if common.MustHasXV(p, `//span[text()="身份切换"]/../../..//span[contains(text(),"确定")]`) {
  126. common.MustElementX(p, `//span[text()="身份切换"]/../../..//span[contains(text(),"确定")]`).MustClick()
  127. time.Sleep(5 * time.Second)
  128. }
  129. } else {
  130. panic(taxerr.NewSystemV3("未在当前税号下["+info.TaxNo+"]找到多家企业", "请联系运维处理"))
  131. }
  132. }
  133. utils.Sleep(2)
  134. p.MustWaitDOMStable()
  135. name = *common.WaitElementX(p, `//div[@class='leftMain']//div[contains(@class,'title')]`, 0.5).MustAttribute(`title`)
  136. })
  137. // 这是panic的失效
  138. if err != nil {
  139. rod.Try(func() {
  140. patjh := common.GeneratePath("png")
  141. p.MustScreenshot(patjh)
  142. logger.Info("代理登录切换公司后异常公司名或判定为失效截图:" + patjh)
  143. })
  144. logger.Info(err.Error())
  145. return "", err
  146. }
  147. if (err != nil) && name == "" {
  148. logger.Info("cookie失效,当前url:" + p.MustInfo().URL)
  149. return "", taxerr.NewSystemV3("税局加载异常", "系统将于30分钟后重试")
  150. }
  151. return
  152. }
  153. func LoadCookieFast(sessionInfos models.SsoSession, p *rod.Page, BaseUri string, info models.CompanyInfo) (name string, err error) {
  154. err = rod.Try(func() {
  155. defer AddErr()
  156. p.Navigate("")
  157. p.Timeout(common.ClickTimeOut).MustNavigate(areaURLmap[info.Area+".pro"]).MustStopLoading() // 此时必须得去tpass域名 要不然写入cookie后进不去发票
  158. p.MustSetCookies(sessionInfos.Cookies...)
  159. {
  160. p.MustEval(`() => localStorage.clear()`)
  161. //utils.Sleep(1)
  162. // 设置 localStorage 在第二个页面中,逐条设置
  163. logger.Info("Setting localStorage in second browser...")
  164. for key, value := range sessionInfos.LocalStorage {
  165. if key == "logConfig" {
  166. continue
  167. }
  168. script := fmt.Sprintf(`() => localStorage.setItem('%s', '%s')`, key, value)
  169. _, err := p.Eval(script)
  170. if err != nil {
  171. log.Printf("Error setting localStorage %s=%s: %v", key, value, err)
  172. }
  173. }
  174. }
  175. utils.Sleep(0.5)
  176. if info.Area == "shaanxi" {
  177. //陕西地区必须带后面/才能登录上去
  178. p.Timeout(common.ClickTimeOut).MustNavigate(BaseUri + "/loginb/")
  179. } else {
  180. p.Timeout(common.ClickTimeOut).MustNavigate(BaseUri + "/loginb")
  181. }
  182. utils.Sleep(1)
  183. _ = rod.Try(func() {
  184. p.Timeout(2 * time.Second).MustWaitStable()
  185. p.Timeout(2 * time.Second).MustWaitDOMStable()
  186. })
  187. //p.MustStopLoading()
  188. logger.Info("加载完成")
  189. if info.Area == "hebei" && !common.MustHasXV(p, `//div[@class='leftMain']//div[contains(@class,'title')]`) {
  190. logger.Info("河北重新访问loginb")
  191. p.Timeout(common.ClickTimeOut).MustNavigate(`https://tpass.hebei.chinatax.gov.cn:8443/#/login?redirect_uri=https%3A%2F%2Fetax.hebei.chinatax.gov.cn%3A8443%2Fmhzx%2Fapi%2Fmh%2Ftpass%2Fcode&client_id=ap5bfdavp66a49979apa5848fpaa5353&response_type=code&state=8f8f3db7ecb948f08b1da177cda40854&client_pid=ap5bfdavp66a49979apa5848fpaa5353`)
  192. utils.Sleep(4)
  193. p.MustWaitStable()
  194. p.MustWaitDOMStable()
  195. }
  196. if common.MustHasXV(p, "//div[text()=' 关于电子税务局“新办智能开业”的通知 ']") {
  197. p.Timeout(common.ClickTimeOut).MustSearch("//span[text()='我知道了']").MustClick()
  198. utils.Sleep(3)
  199. }
  200. if p.MustHasX(`//div[@class='leftMain']//div[contains(@class,'title')]`) {
  201. _ = p.Timeout(common.ClickTimeOut).MustElementX(`//div[@class='leftMain']//div[contains(@class,'title')]`).WaitStable(300 * time.Millisecond)
  202. if *p.Timeout(common.ClickTimeOut).MustElementX(`//div[@class='leftMain']//div[contains(@class,'title')]`).MustAttribute(`title`) == "" {
  203. logger.Info("公司名空刷新")
  204. p.Reload()
  205. utils.Sleep(4)
  206. p.MustWaitDOMStable()
  207. }
  208. }
  209. if common.MustHasXV(p, `//div[@class='leftMain']//div[contains(@class,'title')]`) {
  210. name = *common.WaitElementX(p, `//div[@class='leftMain']//div[contains(@class,'title')]`, 0.5).MustAttribute(`title`)
  211. if name == "" {
  212. err = taxerr.New("代理登录切换公司后异常公司名")
  213. return
  214. }
  215. return
  216. }
  217. DlSy := false //登录完是代理登录的首页
  218. if common.MustHasXV(p, `//a[@title="单户办理"]`) {
  219. common.MustElementX(p, `//a[@title="单户办理"]`).MustClick()
  220. DlSy = true
  221. utils.Sleep(3)
  222. }
  223. if !strings.Contains(p.MustInfo().URL, "tpass") {
  224. panic(taxerr.New("cookie失效"))
  225. }
  226. if !p.MustHasR(`[class="s_Breadcrumb"]`, "企业办税") {
  227. panic(taxerr.New("cookie失效"))
  228. }
  229. if !DlSy {
  230. r := p.MustWaitRequestIdle()
  231. p.Timeout(common.ClickTimeOut).MustSearch("返回").MustClick()
  232. r()
  233. } else {
  234. //这里需要手动切换到首页
  235. common.MustInputX(p, `//input[@placeholder="请输入统一社会信用代码"]`, info.TaxNo)
  236. common.MustElementX(p, `//span[text()="查询"]`).MustClick()
  237. utils.Sleep(2)
  238. tbody := common.MustElementsX(p, `//div[@class="el-table__body-wrapper is-scrolling-none"]//table//tr`)
  239. if len(tbody) == 0 {
  240. common.MustInputX(p, `//input[@placeholder="请输入统一社会信用代码"]`, "")
  241. p.Timeout(common.ClickTimeOut).MustElementX(`//input[@placeholder="请输入纳税人名称"]`).MustInput(info.ComName)
  242. common.MustElementX(p, `//span[text()="查询"]`).MustClick()
  243. utils.Sleep(2)
  244. tbody = common.MustElementsX(p, `//div[@class="el-table__body-wrapper is-scrolling-none"]//table//tr`)
  245. if len(tbody) == 0 {
  246. panic(taxerr.NewUserV3("未在当前账号下["+info.Zzridno+"]找到该企业", "请检查手机号与企业的关联关系"))
  247. }
  248. panic(taxerr.NewUserV3("该公司账号跟公司名称不匹配", "请检查后重新发起"))
  249. } else if len(tbody) == 1 {
  250. common.MustElementX(p, `//span[text()='进入']`).MustClick()
  251. if common.MustHasXV(p, `//span[text()="身份切换"]/../../..//span[contains(text(),"确定")]`) {
  252. common.MustElementX(p, `//span[text()="身份切换"]/../../..//span[contains(text(),"确定")]`).MustClick()
  253. time.Sleep(5 * time.Second)
  254. }
  255. } else {
  256. panic(taxerr.NewSystemV3("未在当前税号下["+info.TaxNo+"]找到多家企业", "请联系运维处理"))
  257. }
  258. }
  259. utils.Sleep(0.5)
  260. p.MustWaitDOMStable()
  261. name = *common.WaitElementX(p, `//div[@class='leftMain']//div[contains(@class,'title')]`, 0.5).MustAttribute(`title`)
  262. logger.Info(name)
  263. })
  264. // 这是panic的失效
  265. if err != nil {
  266. rod.Try(func() {
  267. patjh := common.GeneratePath("png")
  268. p.MustScreenshot(patjh)
  269. logger.Info("代理登录切换公司后异常公司名或判定为失效截图:" + patjh)
  270. })
  271. logger.Info(err.Error())
  272. return "", err
  273. }
  274. if (err != nil) && name == "" {
  275. logger.Info("cookie失效,当前url:" + p.MustInfo().URL)
  276. return "", taxerr.NewSystemV3("税局加载异常", "系统将于30分钟后重试")
  277. }
  278. if name == "" {
  279. logger.Info("公司名加载异常,当前url:" + p.MustInfo().URL)
  280. path := common.GeneratePath("png")
  281. p.MustScreenshot(path)
  282. logger.Info(path)
  283. }
  284. return
  285. }
  286. func AddErr() {
  287. if err := recover(); err != nil {
  288. e := err.(error)
  289. if _, ok := err.(*taxerr.UserErr); ok {
  290. err = taxerr.NewUser(e.Error() + "(代理登录失败)")
  291. }
  292. if _, ok := err.(*taxerr.SystemErr); ok {
  293. err = taxerr.New(e.Error() + "(代理登录失败)")
  294. }
  295. panic(err)
  296. }
  297. }
  298. func AutoCheckSessionAndLogin(ctx context.Context, p *rod.Page, BaseUri string, info models.CompanyInfo) (err error) {
  299. if info.Area == "hubei" && info.Dlfs == "代理登录" && len(info.Tel) != 11 {
  300. return taxerr.NewUserV3("湖北代理登录需要接受验证码", "请在登录信息中修改为手机号")
  301. }
  302. if info.Zzrmm == "" {
  303. return taxerr.PasswdEmpty
  304. }
  305. var LoginResponse []byte
  306. var Session models.SsoSession
  307. // 直接登录获取cookie
  308. {
  309. logger.Info("开始代理登陆获取cookie")
  310. LoginData := MakeLoginData(ctx, info)
  311. a, _ := json.Marshal(LoginData)
  312. logger.Info("代理登录加密前参数" + string(a))
  313. R := AesECBEncrypt(LoginData)
  314. headder := map[string]string{
  315. "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/125.0.0.0 Safari/537.36",
  316. "content-type": "application/json",
  317. "x-token": XToken,
  318. }
  319. marshal, _ := json.Marshal(LoginData)
  320. logger.Info(fmt.Sprintf("PostLoin参数%s" + string(marshal)))
  321. LoginResponse, err = PostLoin(R, headder, info)
  322. logger.Info("代理登陆获取cookie成功")
  323. if err != nil {
  324. if info.Dlfs == "代理登录" {
  325. e := err.(error)
  326. if _, ok := err.(*taxerr.UserErr); ok {
  327. err = taxerr.NewUser(e.Error() + "(代理登录失败)")
  328. }
  329. if _, ok := err.(*taxerr.SystemErr); ok {
  330. err = taxerr.New(e.Error() + "(代理登录失败)")
  331. }
  332. }
  333. return err
  334. }
  335. }
  336. // 整理cookie
  337. {
  338. gdata := gjson.GetBytes(LoginResponse, "data")
  339. cookies := []*proto.NetworkCookieParam{}
  340. domain := ""
  341. gdata.Get("cookies").ForEach(func(key, value gjson.Result) bool {
  342. if key.String() == "lzkqow38189" {
  343. domain = "dppt." + info.Area + ".chinatax.gov.cn"
  344. } else if key.String() == "token" {
  345. domain = "tpass." + info.Area + ".chinatax.gov.cn"
  346. } else {
  347. domain = ".chinatax.gov.cn"
  348. }
  349. cookies = append(cookies, &proto.NetworkCookieParam{
  350. Name: key.String(),
  351. Value: value.String(),
  352. Domain: domain,
  353. Path: "/",
  354. })
  355. return true
  356. })
  357. localStorage := map[string]string{}
  358. gdata.Get("localStorage").ForEach(func(key, value gjson.Result) bool {
  359. localStorage[key.String()] = value.String()
  360. return true
  361. })
  362. Session.Cookies = cookies
  363. Session.LocalStorage = localStorage
  364. //dpptURL = gdata.Get("redirectUri").String()
  365. }
  366. //if ctxValue := ctx.Value("website"); ctxValue != nil { //只有发票任务会传website
  367. // // 写入浏览器
  368. // name, err := LoadCookieFast(Session, p, BaseUri, info)
  369. // if err != nil {
  370. // fmt.Println(err.Error())
  371. // return err
  372. // }
  373. // if name != info.ComName {
  374. // return taxerr.CompanyNameError(name)
  375. // }
  376. //} else {
  377. // 写入浏览器
  378. name, err := LoadCookie(Session, p, BaseUri, info)
  379. if err != nil {
  380. return err
  381. }
  382. if name != info.ComName {
  383. return taxerr.CompanyNameError(name)
  384. }
  385. //}
  386. return
  387. }
  388. func AesECBEncrypt(data map[string]any) []byte {
  389. jsonData, _ := json.Marshal(data)
  390. fmt.Println(string(jsonData))
  391. padded := make([]byte, 16)
  392. copy(padded, key)
  393. encryptData, err := openssl.AesECBEncrypt(jsonData, []byte(padded), openssl.PKCS5_PADDING)
  394. if err != nil {
  395. return []byte("")
  396. }
  397. jsonData, _ = json.Marshal(map[string]string{"data": base64.StdEncoding.EncodeToString(encryptData)})
  398. return jsonData
  399. }