ckLogin.go 15 KB

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