🆗 增加定时打卡前模拟重新登录

This commit is contained in:
jimlee
2023-04-26 13:09:45 +08:00
parent e26ef9fa3b
commit 24cf5c7035
4 changed files with 109 additions and 99 deletions

View File

@@ -3,10 +3,7 @@ package com.pomelotea.hoperun.sign.api
import com.alibaba.fastjson.JSON import com.alibaba.fastjson.JSON
import com.alibaba.fastjson.JSONObject import com.alibaba.fastjson.JSONObject
import com.alibaba.fastjson.TypeReference import com.alibaba.fastjson.TypeReference
import com.pomelotea.hoperun.sign.api.model.AttendancesDetail
import com.pomelotea.hoperun.sign.api.model.AttendancesDetailResponse
import com.pomelotea.hoperun.sign.common.* import com.pomelotea.hoperun.sign.common.*
import com.pomelotea.hoperun.sign.config.HoperunUserConfig
import com.pomelotea.hoperun.sign.config.HoperunUserConfig.deviceMap import com.pomelotea.hoperun.sign.config.HoperunUserConfig.deviceMap
import com.pomelotea.hoperun.sign.config.HoperunUserConfig.getUserConfig import com.pomelotea.hoperun.sign.config.HoperunUserConfig.getUserConfig
import com.pomelotea.hoperun.sign.config.HoperunUserConfig.userConfigMap import com.pomelotea.hoperun.sign.config.HoperunUserConfig.userConfigMap
@@ -17,7 +14,6 @@ import okhttp3.FormBody
import okhttp3.MediaType.Companion.toMediaTypeOrNull import okhttp3.MediaType.Companion.toMediaTypeOrNull
import okhttp3.Request import okhttp3.Request
import okhttp3.RequestBody.Companion.toRequestBody import okhttp3.RequestBody.Companion.toRequestBody
import org.jsoup.Jsoup
import org.springframework.web.bind.annotation.* import org.springframework.web.bind.annotation.*
import java.text.SimpleDateFormat import java.text.SimpleDateFormat
import java.util.* import java.util.*
@@ -35,7 +31,7 @@ import java.util.*
@RequestMapping("/api/daka") @RequestMapping("/api/daka")
class HoperunSignController { class HoperunSignController {
val LOGIN_URL = "http://pom.hoperun.com:8187/attm/login/login"
init { init {
AutoDakaScheduler() AutoDakaScheduler()
@@ -61,7 +57,7 @@ class HoperunSignController {
}*/ }*/
val userConfig = getUserConfig(employeeNo).let { val userConfig = getUserConfig(employeeNo).let {
oldLogin(employeeNo) defaultLogin(employeeNo)
getUserConfig(employeeNo) getUserConfig(employeeNo)
} }
userConfig ?: return WebResult.getFailed("登陆失败") userConfig ?: return WebResult.getFailed("登陆失败")
@@ -145,13 +141,6 @@ class HoperunSignController {
return WebResult.getSuccess(JSONObject.parseObject(result, DakaResponse::class.java)) return WebResult.getSuccess(JSONObject.parseObject(result, DakaResponse::class.java))
} }
private fun getJsessionIdAutoLogin(employeeNo: String, jsessionId: String): String? {
if (sessionMap.get(employeeNo) == null) {
login(employeeNo, jsessionId)
}
sessionMap[employeeNo] = jsessionId
return sessionMap.get(employeeNo)
}
/** /**
* 查询近两个月的 * 查询近两个月的
@@ -194,30 +183,11 @@ class HoperunSignController {
object : TypeReference<List<MonthAttLog>?>() {}) object : TypeReference<List<MonthAttLog>?>() {})
} }
private fun login(employeeNo: String, jsessionId: String) { private fun defaultLogin(employeeNo: String) {
// 读取员工姓名 resetJSessionId(employeeNo)
if (userConfigMap[employeeNo]?.username == null) { setUserConfig(employeeNo)
setUserConfig(employeeNo, jsessionId)
}
} }
/*
private fun oldLogin(employeeNo: String) {
val jsessionId: String?
val loginRequest = Request.Builder()
.url(LOGIN_URL)
.post(
FormBody.Builder()
.add("login_id", padEmployeeNumber(employeeNo))
.add("password", "123456")
.build()
)
.build()
val response = client.newCall(loginRequest).execute()
jsessionId = response.request.url.pathSegments[2].substring(19)
sessionMap.put(employeeNo, jsessionId)
setUserConfigOld(employeeNo)
}
private fun setUserConfig(employeeNo: String, jsessionId: String) { private fun setUserConfig(employeeNo: String, jsessionId: String) {
val attendancesDetailRequest = Request.Builder() val attendancesDetailRequest = Request.Builder()
.url("http://pom.hoperun.com:8187/attm/calendar/monthAtt?staff_code=${padEmployeeNumber(employeeNo)}&yearmonth=${getNowDateyyyy_MM()}-01") .url("http://pom.hoperun.com:8187/attm/calendar/monthAtt?staff_code=${padEmployeeNumber(employeeNo)}&yearmonth=${getNowDateyyyy_MM()}-01")
@@ -227,7 +197,7 @@ class HoperunSignController {
"User-Agent", "User-Agent",
"Qing/0.9.113;iOS 16.4.1;Apple;iPhone13,2;deviceId:a8baf66f-fdeb-4f4d-b1e5-9fafcd5045b6;deviceName:iOS;clientId:10200;os:iOS 16.3.1;brand:Apple;model:iPhone13,2;lang:zh-CN;fontNum:0;fontScale:1.0;ver:10.7.14;Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148" "Qing/0.9.113;iOS 16.4.1;Apple;iPhone13,2;deviceId:a8baf66f-fdeb-4f4d-b1e5-9fafcd5045b6;deviceName:iOS;clientId:10200;os:iOS 16.3.1;brand:Apple;model:iPhone13,2;lang:zh-CN;fontNum:0;fontScale:1.0;ver:10.7.14;Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148"
) )
.addHeader("accept", "*/*") .addHeader("accept", "*")
.addHeader("Origin", "http://pom.hoperun.com:8187") .addHeader("Origin", "http://pom.hoperun.com:8187")
.addHeader("Referer", "http://pom.hoperun.com:8187/attm/attence/getInfo") .addHeader("Referer", "http://pom.hoperun.com:8187/attm/attence/getInfo")
.addHeader("content-type", "application/x-www-form-urlencoded; charset=UTF-8") .addHeader("content-type", "application/x-www-form-urlencoded; charset=UTF-8")
@@ -263,7 +233,7 @@ class HoperunSignController {
"User-Agent", "User-Agent",
"Qing/0.9.113;iOS 16.3.1;Apple;iPhone13,2;deviceId:a8baf66f-fdeb-4f4d-b1e5-9fafcd5045b6;deviceName:iOS;clientId:10200;os:iOS 16.3.1;brand:Apple;model:iPhone13,2;lang:zh-CN;fontNum:0;fontScale:1.0;ver:10.7.14;Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148" "Qing/0.9.113;iOS 16.3.1;Apple;iPhone13,2;deviceId:a8baf66f-fdeb-4f4d-b1e5-9fafcd5045b6;deviceName:iOS;clientId:10200;os:iOS 16.3.1;brand:Apple;model:iPhone13,2;lang:zh-CN;fontNum:0;fontScale:1.0;ver:10.7.14;Mozilla/5.0 (iPhone; CPU iPhone OS 16_3_1 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Mobile/15E148"
) )
.addHeader("accept", "*/*") .addHeader("accept", "*")
.addHeader("Origin", "http://pom.hoperun.com:8187") .addHeader("Origin", "http://pom.hoperun.com:8187")
.addHeader("Referer", "http://pom.hoperun.com:8187/attm/attence/getInfo") .addHeader("Referer", "http://pom.hoperun.com:8187/attm/attence/getInfo")
.addHeader("content-type", "application/x-www-form-urlencoded; charset=UTF-8") .addHeader("content-type", "application/x-www-form-urlencoded; charset=UTF-8")
@@ -278,10 +248,10 @@ class HoperunSignController {
HoperunUserConfig.addUserConfig( HoperunUserConfig.addUserConfig(
employeeNo, userConfig employeeNo, userConfig
) )
} }*/
private fun setUserConfigOld(employeeNo: String) { private fun setUserConfig(employeeNo: String) {
// 获取deviceua // 获取deviceua
val loginRequest = Request.Builder() val loginRequest = Request.Builder()
.url("http://pom.hoperun.com:8187/attp/login/login.do") .url("http://pom.hoperun.com:8187/attp/login/login.do")

View File

@@ -9,6 +9,7 @@ import com.pomelotea.hoperun.sign.config.HoperunUserConfig.getUserConfig
import com.pomelotea.hoperun.sign.config.UserConfig import com.pomelotea.hoperun.sign.config.UserConfig
import com.pomelotea.hoperun.sign.holidays import com.pomelotea.hoperun.sign.holidays
import com.pomelotea.hoperun.sign.workdays import com.pomelotea.hoperun.sign.workdays
import okhttp3.FormBody
import okhttp3.MediaType.Companion.toMediaTypeOrNull import okhttp3.MediaType.Companion.toMediaTypeOrNull
import okhttp3.OkHttpClient import okhttp3.OkHttpClient
import okhttp3.Request import okhttp3.Request
@@ -160,6 +161,22 @@ fun endTimeHoperunDakaRequest(
return hoperunDakaRequest return hoperunDakaRequest
} }
const val LOGIN_URL = "http://pom.hoperun.com:8187/attm/login/login"
fun resetJSessionId(employeeNo: String) {
val jsessionId: String?
val loginRequest = Request.Builder()
.url(LOGIN_URL)
.post(
FormBody.Builder()
.add("login_id", padEmployeeNumber(employeeNo))
.add("password", "123456")
.build()
)
.build()
val response = client.newCall(loginRequest).execute()
jsessionId = response.request.url.pathSegments[2].substring(19)
sessionMap.put(employeeNo, jsessionId)
}
fun randowSecond(): String { fun randowSecond(): String {
val second = Random().nextInt(59) val second = Random().nextInt(59)

View File

@@ -4,12 +4,14 @@ import com.pomelotea.hoperun.sign.api.DakaResponse
import com.pomelotea.hoperun.sign.common.* import com.pomelotea.hoperun.sign.common.*
import com.pomelotea.hoperun.sign.config.HoperunUserConfig import com.pomelotea.hoperun.sign.config.HoperunUserConfig
import com.pomelotea.hoperun.sign.config.UserConfig import com.pomelotea.hoperun.sign.config.UserConfig
import com.pomelotea.hoperun.sign.logger import org.slf4j.Logger
import org.slf4j.LoggerFactory
import java.time.LocalDateTime import java.time.LocalDateTime
import java.time.format.DateTimeFormatter import java.time.format.DateTimeFormatter
import java.util.* import java.util.*
import java.util.concurrent.Executors import java.util.concurrent.Executors
import java.util.concurrent.ScheduledExecutorService import java.util.concurrent.ScheduledExecutorService
import java.util.concurrent.ScheduledThreadPoolExecutor
import java.util.concurrent.TimeUnit import java.util.concurrent.TimeUnit
/** /**
@@ -25,11 +27,12 @@ class AutoDakaScheduler {
val schedulerThreadPool: ScheduledExecutorService = Executors.newScheduledThreadPool(10) val schedulerThreadPool: ScheduledExecutorService = Executors.newScheduledThreadPool(10)
init { init {
timeThreadPool.scheduleAtFixedRate({addAutoDakaScheduled()}, 1, 60, TimeUnit.MINUTES) timeThreadPool.scheduleAtFixedRate({ addAutoDakaScheduled() }, 1, 60, TimeUnit.MINUTES)
} }
companion object { companion object {
val dakaQueue: MutableList<Daka> = LinkedList() val dakaQueue: MutableList<Daka> = LinkedList()
val logger: Logger = LoggerFactory.getLogger("DAKA-SHCEDULER")
} }
fun addAutoDakaScheduled(dakaDate: String = getNowDateyyyy_MM_dd()) { fun addAutoDakaScheduled(dakaDate: String = getNowDateyyyy_MM_dd()) {
@@ -48,39 +51,77 @@ class AutoDakaScheduler {
} }
} }
dakaQueue.filter { !it.added }.forEach { dakaQueue.forEach {
it.added = true if (!it.added) {
val beginDate = LocalDateTime.parse(it.dakaDate + "T" + it.beginTime + ":" + randowSecond(), DateTimeFormatter.ISO_LOCAL_DATE_TIME) it.added = true
val endDate = LocalDateTime.parse(it.dakaDate + "T" + it.endTime + ":" + randowSecond(), DateTimeFormatter.ISO_LOCAL_DATE_TIME) val beginDate = LocalDateTime.parse(
val beginSeconds = beginDate.toEpochSecond(DEFAULT_ZONE) - (System.currentTimeMillis() / 1000) it.dakaDate + "T" + it.beginTime + ":" + randowSecond(),
val endSeconds = endDate.toEpochSecond(DEFAULT_ZONE) - (System.currentTimeMillis() / 1000) DateTimeFormatter.ISO_LOCAL_DATE_TIME
if (beginSeconds > 0) { )
logger.info("[ADD-SCHEDULE]BEGIN:${it.employeeNo}:DATE:${it.dakaDate}TIME:${it.beginTime}") val endDate = LocalDateTime.parse(
schedulerThreadPool.schedule({beginDaka(it)}, beginSeconds, TimeUnit.SECONDS) it.dakaDate + "T" + it.endTime + ":" + randowSecond(),
} DateTimeFormatter.ISO_LOCAL_DATE_TIME
if (endSeconds > 0) { )
logger.info("[ADD-SCHEDULE]END:${it.employeeNo}:DATE:${it.dakaDate}TIME:${it.endTime}") val beginSeconds = beginDate.toEpochSecond(DEFAULT_ZONE) - (System.currentTimeMillis() / 1000)
schedulerThreadPool.schedule({endDaka(it)}, endSeconds, TimeUnit.SECONDS) val endSeconds = endDate.toEpochSecond(DEFAULT_ZONE) - (System.currentTimeMillis() / 1000)
if (beginSeconds > 0) {
logger.info("[ADD-SCHEDULE]BEGIN:${it.employeeNo}:DATE:${it.dakaDate}TIME:${it.beginTime}")
schedulerThreadPool.schedule({ beginDaka(it) }, beginSeconds, TimeUnit.SECONDS)
}
if (endSeconds > 0) {
logger.info("[ADD-SCHEDULE]END:${it.employeeNo}:DATE:${it.dakaDate}TIME:${it.endTime}")
schedulerThreadPool.schedule({ endDaka(it) }, endSeconds, TimeUnit.SECONDS)
}
} }
// schedulerThreadPool.schedule({beginDaka(it)}, 5, TimeUnit.SECONDS) // schedulerThreadPool.schedule({beginDaka(it)}, 5, TimeUnit.SECONDS)
// schedulerThreadPool.schedule({ endDaka(it) }, 5, TimeUnit.SECONDS) // schedulerThreadPool.schedule({ endDaka(it) }, 10, TimeUnit.SECONDS)
} }
dakaQueue.removeIf { it.dakaDate.compareTo(dakaDate) == -1 } dakaQueue.removeIf { it.dakaDate.compareTo(dakaDate) == -1 }
printScheduler()
dakaQueue.forEach {
logger.info("Task: ${it.toCsv()}")
}
} }
private fun beginDaka(daka: Daka) { private fun beginDaka(daka: Daka) {
// println("begin:${daka.toCsv()}") // println("begin:${daka.toCsv()}")
logger.info("[EXECUTE]BEGIN:${daka.employeeNo}:DATE:${daka.dakaDate}:TIME:${daka.beginTime}") try {
val resp: DakaResponse = beginTime(employeeNo = daka.employeeNo, date = daka.dakaDate, time = daka.beginTime) logger.info("[EXECUTE]BEGIN:${daka.employeeNo}:DATE:${daka.dakaDate}:TIME:${daka.beginTime}")
logger.info(resp.toString()) // 模拟重新登录打卡
logger.info("${daka.employeeNo}OLD-JSESSIONID: ${sessionMap[daka.employeeNo]}")
resetJSessionId(employeeNo = daka.employeeNo)
logger.info("${daka.employeeNo}NEW-JSESSIONID: ${sessionMap[daka.employeeNo]}")
val resp: DakaResponse =
beginTime(employeeNo = daka.employeeNo, date = daka.dakaDate, time = daka.beginTime)
if (resp.result != "success") {
logger.error("打上班卡失败")
} else {
logger.info("打上班卡成功")
}
} catch (e: Exception) {
logger.error(e.message, e)
}
} }
private fun endDaka(daka: Daka) { private fun endDaka(daka: Daka) {
// println("end:${daka.toCsv()}") // println("end:${daka.toCsv()}")
logger.info("[EXECUTE]END:${daka.employeeNo}:DATE:${daka.dakaDate}:TIME:${daka.endTime}") try {
val resp: DakaResponse = endTime(employeeNo = daka.employeeNo, date = daka.dakaDate, time = daka.endTime) logger.info("[EXECUTE]END:${daka.employeeNo}:DATE:${daka.dakaDate}:TIME:${daka.endTime}")
logger.info(resp.toString()) // 模拟重新登录打卡
logger.info("${daka.employeeNo}OLD-JSESSIONID: ${sessionMap[daka.employeeNo]}")
resetJSessionId(employeeNo = daka.employeeNo)
logger.info("${daka.employeeNo}NEW-JSESSIONID: ${sessionMap[daka.employeeNo]}")
val resp: DakaResponse = endTime(employeeNo = daka.employeeNo, date = daka.dakaDate, time = daka.endTime)
if (resp.result != "success") {
logger.error("打下班卡失败")
} else {
logger.info("打下班卡成功")
}
} catch (e: Exception) {
logger.error(e.message, e)
}
} }
private fun generateDakaInfo(userConfig: UserConfig, dakaDate: String): Daka { private fun generateDakaInfo(userConfig: UserConfig, dakaDate: String): Daka {
@@ -92,8 +133,14 @@ class AutoDakaScheduler {
) )
} }
fun printScheduler() {
schedulerThreadPool as ScheduledThreadPoolExecutor
val queue = schedulerThreadPool.queue
logger.info("任务队列数量: ${queue.size}")
}
private fun getRandomEndTime(): String { private fun getRandomEndTime(): String {
val hourArray = intArrayOf(18, 19, 20, 21, 22, 23, 20, 21, 22, 19, 18, 20) val hourArray = intArrayOf(18, 19, 20, 21, 22, 23, 19, 20, 20, 19, 18, 20)
val randomHour = hourArray[Math.round(Math.random() * 9).toInt()] val randomHour = hourArray[Math.round(Math.random() * 9).toInt()]
val randomMinute = if (randomHour == 18) { val randomMinute = if (randomHour == 18) {
30 + Random().nextInt(19) 30 + Random().nextInt(19)

View File

@@ -58,18 +58,19 @@
</encoder> </encoder>
</appender> </appender>
<!-- REST接口日志文件 -->
<appender name="REST-APPENDER" class="ch.qos.logback.core.rolling.RollingFileAppender"> <!-- 默认系统日志文件 -->
<appender name="SCHEDULER-APPENDER" class="ch.qos.logback.core.rolling.RollingFileAppender">
<append>true</append> <append>true</append>
<filter class="ch.qos.logback.classic.filter.LevelFilter"> <filter class="ch.qos.logback.classic.filter.LevelFilter">
<level>${logging.level}</level> <level>${logging.level}</level>
<onMatch>ACCEPT</onMatch> <onMatch>ACCEPT</onMatch>
<onMismatch>DENY</onMismatch> <onMismatch>DENY</onMismatch>
</filter> </filter>
<file>${logging.path}/rest.log</file> <file>${logging.path}/scheduler.log</file>
<rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy"> <rollingPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedRollingPolicy">
<maxFileSize>${logging.size}</maxFileSize> <maxFileSize>${logging.size}</maxFileSize>
<FileNamePattern>${logging.path}/rest.log.%d{yyyy-MM-dd}.%i</FileNamePattern> <FileNamePattern>${logging.path}/scheduler.log.%d{yyyy-MM-dd}.%i</FileNamePattern>
<MaxHistory>${logging.maxHistory}</MaxHistory> <MaxHistory>${logging.maxHistory}</MaxHistory>
</rollingPolicy> </rollingPolicy>
<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder"> <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
@@ -78,14 +79,6 @@
</encoder> </encoder>
</appender> </appender>
<!--==================async appender======================= -->
<appender name="ASYNC-REST-APPENDER" class="ch.qos.logback.classic.AsyncAppender">
<queueSize>256</queueSize>
<discardingThreshold>0</discardingThreshold>
<neverBlock>true</neverBlock>
<appender-ref ref="REST-APPENDER"/>
</appender>
<!--==================logger======================= --> <!--==================logger======================= -->
<!-- boot logger --> <!-- boot logger -->
<logger name="APPLICATION-STARTER" level="${logging.level}" additivity="false"> <logger name="APPLICATION-STARTER" level="${logging.level}" additivity="false">
@@ -93,37 +86,20 @@
<appender-ref ref="ROOT-APPENDER"/> <appender-ref ref="ROOT-APPENDER"/>
</logger> </logger>
<!-- scheduler logger -->
<logger name="DAKA-SHCEDULER" level="${logging.level}">
<appender-ref ref="SCHEDULER-APPENDER"/>
</logger>
<!-- application logger --> <!-- application logger -->
<logger name="com.pomelotea.hoperun.sign" level="${logging.level}" additivity="false"> <logger name="com.pomelotea.hoperun.sign" level="${logging.level}" additivity="false">
<appender-ref ref="ROOT-APPENDER"/> <appender-ref ref="ROOT-APPENDER"/>
<appender-ref ref="ERROR-APPENDER"/> <appender-ref ref="ERROR-APPENDER"/>
</logger> </logger>
<root level="${logging.level}"> <root level="${logging.level}">
<appender-ref ref="ROOT-APPENDER"/> <appender-ref ref="STDOUT" />
<appender-ref ref="ERROR-APPENDER"/>
</root> </root>
<!-- 本地调试日志输出到控制台 -->
<springProfile name="dev">
<!-- boot logger -->
<logger name="APPLICATION-STARTER" level="debug" additivity="false">
<appender-ref ref="STDOUT"/>
<appender-ref ref="ERROR-APPENDER"/>
<appender-ref ref="ROOT-APPENDER"/>
</logger>
<logger name="com.pomelotea.hoperun.sign" level="debug" additivity="false">
<appender-ref ref="STDOUT"/>
<appender-ref ref="ROOT-APPENDER"/>
<appender-ref ref="ERROR-APPENDER"/>
</logger>
<root level="debug">
<appender-ref ref="STDOUT"/>
<appender-ref ref="ROOT-APPENDER"/>
<appender-ref ref="ERROR-APPENDER"/>
</root>
</springProfile>
</configuration> </configuration>