This commit is contained in:
2025-10-10 13:39:16 +08:00
parent a28d846638
commit d77ebab153
10 changed files with 420 additions and 8 deletions

70
.gitignore vendored
View File

@@ -4,6 +4,7 @@
*.temp *.temp
.DS_Store .DS_Store
Thumbs.db Thumbs.db
nul
# IDE 文件 # IDE 文件
## IntelliJ IDEA ## IntelliJ IDEA
@@ -14,6 +15,25 @@ Thumbs.db
*.ipr *.ipr
out/ out/
.idea_modules/ .idea_modules/
## VSCode
.vscode/
.history/
## Eclipse
.classpath
.project
.settings/
## AI 助手配置
.claude/
.cursor/
.starFactory/
.windsurf/
.aider/
.copilot/
# Maven 构建目录
/ruoyi-common/target/ /ruoyi-common/target/
/ruoyi-system/target/ /ruoyi-system/target/
/ruoyi-quartz/target/ /ruoyi-quartz/target/
@@ -22,3 +42,53 @@ out/
/ruoyi-admin/target/ /ruoyi-admin/target/
/erp_client_sb/target/ /erp_client_sb/target/
target/ target/
*.class
# Node.js 前端项目
node_modules/
npm-debug.log*
yarn-debug.log*
yarn-error.log*
pnpm-debug.log*
.npm
.yarn/
dist/
dist-ssr/
*.local
# Vite 构建缓存
.vite/
.vite-inspect/
# Electron 构建产物
build/
release/
out/
*.exe
*.dmg
*.AppImage
# Java 打包文件
*.jar
*.war
*.ear
*.zip
*.tar.gz
*.rar
# JRE 运行时环境
jre/
jdk/
# 数据库文件
*.db
*.db-shm
*.db-wal
*.sqlite
*.sqlite3
# 缓存和临时文件
.cache/
*.swp
*.swo
*~

View File

@@ -19,7 +19,7 @@
"@vitejs/plugin-vue": "^4.4.1", "@vitejs/plugin-vue": "^4.4.1",
"chalk": "^4.1.2", "chalk": "^4.1.2",
"chokidar": "^3.5.3", "chokidar": "^3.5.3",
"electron": "^28.2.0", "electron": "^28.3.3",
"electron-builder": "^25.1.6", "electron-builder": "^25.1.6",
"express": "^5.1.0", "express": "^5.1.0",
"fs-extra": "^11.3.2", "fs-extra": "^11.3.2",

View File

@@ -18,7 +18,6 @@ const PLATFORM_LABEL: Record<PlatformKey, string> = {
const accounts = ref<BanmaAccount[]>([]) const accounts = ref<BanmaAccount[]>([])
async function load() { async function load() {
// 目前后端只有斑马接口,其它平台先共用此接口占位
const res = await zebraApi.getAccounts() const res = await zebraApi.getAccounts()
const list = (res as any)?.data ?? res const list = (res as any)?.data ?? res
accounts.value = Array.isArray(list) ? list : [] accounts.value = Array.isArray(list) ? list : []

View File

@@ -0,0 +1,88 @@
@echo off
setlocal enabledelayedexpansion
set APP_ASAR=%~1
set UPDATE_FILE=%~2
set EXE_PATH=%~3
set LOG_FILE=%TEMP%\electron-update.log
echo ======================================== > "%LOG_FILE%"
echo Electron App Auto-Update Helper >> "%LOG_FILE%"
echo Started: %date% %time% >> "%LOG_FILE%"
echo ======================================== >> "%LOG_FILE%"
echo. >> "%LOG_FILE%"
if not exist "%UPDATE_FILE%" (
echo [ERROR] Update file not found: %UPDATE_FILE% >> "%LOG_FILE%"
goto :start_app
)
echo [INFO] Update file found: %UPDATE_FILE% >> "%LOG_FILE%"
echo [INFO] Target app.asar: %APP_ASAR% >> "%LOG_FILE%"
echo [INFO] Application exe: %EXE_PATH% >> "%LOG_FILE%"
echo [INFO] Waiting for application to close... >> "%LOG_FILE%"
REM 获取应用进程名
for /f "tokens=*" %%a in ("%EXE_PATH%") do set EXE_NAME=%%~nxa
echo [INFO] Waiting for %EXE_NAME% to close... >> "%LOG_FILE%"
REM 等待应用进程完全关闭最多等待10秒
set COUNT=0
:wait_loop
tasklist /FI "IMAGENAME eq %EXE_NAME%" 2>nul | find /I "%EXE_NAME%" >nul
if errorlevel 1 goto process_closed
set /a COUNT+=1
if %COUNT% GEQ 20 (
echo [WARN] Application still running after 10 seconds >> "%LOG_FILE%"
goto process_closed
)
timeout /t 1 /nobreak >nul
goto wait_loop
:process_closed
echo [INFO] Application process closed >> "%LOG_FILE%"
timeout /t 1 /nobreak >nul
echo [INFO] Backing up current app.asar... >> "%LOG_FILE%"
if exist "%APP_ASAR%.backup" (
del /f /q "%APP_ASAR%.backup" >nul 2>&1
)
if exist "%APP_ASAR%" (
move /y "%APP_ASAR%" "%APP_ASAR%.backup" >nul 2>&1
if errorlevel 1 (
echo [WARN] First move attempt failed, retrying... >> "%LOG_FILE%"
timeout /t 2 /nobreak >nul
move /y "%APP_ASAR%" "%APP_ASAR%.backup" >nul 2>&1
if errorlevel 1 (
echo [ERROR] Failed to backup app.asar, file is locked >> "%LOG_FILE%"
goto :start_app
)
)
echo [SUCCESS] Backup completed >> "%LOG_FILE%"
)
echo [INFO] Applying update... >> "%LOG_FILE%"
move /y "%UPDATE_FILE%" "%APP_ASAR%" >nul 2>&1
if errorlevel 1 (
echo [ERROR] Failed to apply update >> "%LOG_FILE%"
if exist "%APP_ASAR%.backup" (
echo [INFO] Restoring backup... >> "%LOG_FILE%"
move /y "%APP_ASAR%.backup" "%APP_ASAR%" >nul 2>&1
)
goto :start_app
)
echo [SUCCESS] Update applied successfully! >> "%LOG_FILE%"
if exist "%UPDATE_FILE%" (
del /f /q "%UPDATE_FILE%" >nul 2>&1
)
:start_app
echo [INFO] Restarting application... >> "%LOG_FILE%"
echo ======================================== >> "%LOG_FILE%"
start "" "%EXE_PATH%"
exit /b 0

View File

@@ -43,14 +43,11 @@ public class AuthController {
public JsonData saveAuth(@RequestBody Map<String, Object> data) { public JsonData saveAuth(@RequestBody Map<String, Object> data) {
String serviceName = (String) data.get("serviceName"); String serviceName = (String) data.get("serviceName");
String authKey = (String) data.get("authKey"); String authKey = (String) data.get("authKey");
if (serviceName == null || authKey == null) return JsonData.buildError("serviceName和authKey不能为空"); if (serviceName == null || authKey == null) return JsonData.buildError("serviceName和authKey不能为空");
AuthTokenEntity entity = authTokenRepository.findByServiceName(serviceName).orElse(new AuthTokenEntity()); AuthTokenEntity entity = authTokenRepository.findByServiceName(serviceName).orElse(new AuthTokenEntity());
entity.setServiceName(serviceName); entity.setServiceName(serviceName);
entity.setToken(authKey); entity.setToken(authKey);
authTokenRepository.save(entity); authTokenRepository.save(entity);
return JsonData.buildSuccess("认证信息保存成功"); return JsonData.buildSuccess("认证信息保存成功");
} }
@@ -104,7 +101,7 @@ public class AuthController {
} }
/** /**
* 删除缓存数据 - POST方式 * 删除缓存数据
*/ */
@PostMapping("/cache/delete") @PostMapping("/cache/delete")
public JsonData deleteCacheByPost(@RequestParam String key) { public JsonData deleteCacheByPost(@RequestParam String key) {
@@ -133,7 +130,7 @@ public class AuthController {
} }
/** /**
* 获取设备ID硬件UUID * 获取设备ID
*/ */
@GetMapping("/device-id") @GetMapping("/device-id")
public JsonData getDeviceId() { public JsonData getDeviceId() {

View File

@@ -0,0 +1,18 @@
package com.ruoyi.system.mapper;
import java.util.List;
import com.ruoyi.system.domain.BanmaAccount;
/**
* 斑马账号 Mapper
*/
public interface BanmaAccountMapper {
BanmaAccount selectById(Long id);
List<BanmaAccount> selectList(BanmaAccount query);
int insert(BanmaAccount entity);
int update(BanmaAccount entity);
int deleteById(Long id);
int clearDefault();
}

View File

@@ -0,0 +1,17 @@
package com.ruoyi.system.service;
import java.util.List;
import com.ruoyi.system.domain.BanmaAccount;
/**
* 斑马账号 Service 接口
*/
public interface IBanmaAccountService {
List<BanmaAccount> listSimple();
Long saveOrUpdate(BanmaAccount entity);
void remove(Long id);
boolean refreshToken(Long id);
void refreshAllTokens();
}

View File

@@ -0,0 +1,109 @@
package com.ruoyi.system.service.impl;
import java.util.Date;
import java.util.List;
import java.util.Objects;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.http.HttpEntity;
import org.springframework.http.HttpHeaders;
import org.springframework.http.MediaType;
import org.springframework.web.client.RestTemplate;
import com.ruoyi.system.domain.BanmaAccount;
import com.ruoyi.system.mapper.BanmaAccountMapper;
import com.ruoyi.system.service.IBanmaAccountService;
/**
* 斑马账号 Service 实现
*/
@Service
public class BanmaAccountServiceImpl implements IBanmaAccountService {
@Autowired
private BanmaAccountMapper mapper;
private final RestTemplate restTemplate = new RestTemplate();
private static final String LOGIN_URL = "https://banma365.cn/api/login";
@Override
public List<BanmaAccount> listSimple() {
List<BanmaAccount> list = mapper.selectList(new BanmaAccount());
// 隐藏密码
for (BanmaAccount a : list) { a.setPassword(null); }
return list;
}
@Override
public Long saveOrUpdate(BanmaAccount entity) {
if (entity.getId() == null) {
mapper.insert(entity);
} else {
mapper.update(entity);
}
if (Objects.equals(entity.getIsDefault(), 1)) {
mapper.clearDefault();
BanmaAccount only = new BanmaAccount();
only.setId(entity.getId());
only.setIsDefault(1);
mapper.update(only);
}
return entity.getId();
}
@Override
public void remove(Long id) { mapper.deleteById(id); }
@Override
public boolean refreshToken(Long id) {
BanmaAccount one = mapper.selectById(id);
if (one == null || one.getStatus() == null || one.getStatus() == 0) return false;
if (one.getUsername() == null || one.getPassword() == null) return false;
String token = validateAndGetToken(one.getUsername(), one.getPassword());
if (token != null) {
BanmaAccount upd = new BanmaAccount();
upd.setId(one.getId());
upd.setToken("Bearer " + token);
upd.setTokenExpireAt(new Date(System.currentTimeMillis() + 2L * 24 * 60 * 60 * 1000));
mapper.update(upd);
return true;
}
return false;
}
/**
* 验证账号密码并获取Token不保存到数据库
*/
public String validateAndGetToken(String username, String password) {
if (username == null || password == null) return null;
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.APPLICATION_JSON);
java.util.Map<String, String> body = new java.util.HashMap<>();
body.put("username", username);
body.put("password", password);
@SuppressWarnings("unchecked")
java.util.Map<String, Object> resp = restTemplate.postForObject(LOGIN_URL, new HttpEntity<>(body, headers), java.util.Map.class);
if (resp == null) return null;
Object code = resp.get("code");
if (code instanceof Number && ((Number) code).intValue() == 0) {
Object data = resp.get("data");
if (data instanceof java.util.Map) {
Object token = ((java.util.Map<?, ?>) data).get("token");
if (token instanceof String && !((String) token).isEmpty()) {
return (String) token;
}
}
}
return null;
}
@Override
public void refreshAllTokens() {
BanmaAccount q = new BanmaAccount();
q.setStatus(1);
List<BanmaAccount> list = mapper.selectList(q);
for (BanmaAccount a : list) {
try { refreshToken(a.getId()); } catch (Exception ignore) {}
}
}
}

View File

@@ -0,0 +1,94 @@
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ruoyi.system.mapper.BanmaAccountMapper">
<resultMap id="BanmaAccountResult" type="com.ruoyi.system.domain.BanmaAccount">
<result property="id" column="id"/>
<result property="name" column="name"/>
<result property="username" column="username"/>
<result property="password" column="password"/>
<result property="token" column="token"/>
<result property="tokenExpireAt" column="token_expire_at"/>
<result property="isDefault" column="is_default"/>
<result property="status" column="status"/>
<result property="remark" column="remark"/>
<result property="createBy" column="create_by"/>
<result property="createTime" column="create_time"/>
<result property="updateBy" column="update_by"/>
<result property="updateTime" column="update_time"/>
</resultMap>
<sql id="Base_Column_List">
id, name, username, password, token, token_expire_at, is_default, status, remark, create_by, create_time, update_by, update_time
</sql>
<select id="selectById" parameterType="long" resultMap="BanmaAccountResult">
select <include refid="Base_Column_List"/> from banma_account where id = #{id}
</select>
<select id="selectList" parameterType="com.ruoyi.system.domain.BanmaAccount" resultMap="BanmaAccountResult">
select <include refid="Base_Column_List"/> from banma_account
<where>
<if test="name != null and name != ''"> and name like concat('%', #{name}, '%')</if>
<if test="username != null and username != ''"> and username like concat('%', #{username}, '%')</if>
<if test="status != null"> and status = #{status}</if>
</where>
order by create_time desc
</select>
<insert id="insert" parameterType="com.ruoyi.system.domain.BanmaAccount" useGeneratedKeys="true" keyProperty="id">
insert into banma_account
<trim prefix="(" suffix=")" suffixOverrides=",">
<if test="name != null">name,</if>
<if test="username != null">username,</if>
<if test="password != null">password,</if>
<if test="token != null">token,</if>
<if test="tokenExpireAt != null">token_expire_at,</if>
<if test="isDefault != null">is_default,</if>
<if test="status != null">status,</if>
<if test="remark != null">remark,</if>
<if test="createBy != null">create_by,</if>
create_time
</trim>
<trim prefix="values (" suffix=")" suffixOverrides=",">
<if test="name != null">#{name},</if>
<if test="username != null">#{username},</if>
<if test="password != null">#{password},</if>
<if test="token != null">#{token},</if>
<if test="tokenExpireAt != null">#{tokenExpireAt},</if>
<if test="isDefault != null">#{isDefault},</if>
<if test="status != null">#{status},</if>
<if test="remark != null">#{remark},</if>
<if test="createBy != null">#{createBy},</if>
sysdate()
</trim>
</insert>
<update id="update" parameterType="com.ruoyi.system.domain.BanmaAccount">
update banma_account
<trim prefix="set" suffixOverrides=",">
<if test="name != null">name=#{name},</if>
<if test="username != null">username=#{username},</if>
<if test="password != null">password=#{password},</if>
<if test="token != null">token=#{token},</if>
<if test="tokenExpireAt != null">token_expire_at=#{tokenExpireAt},</if>
<if test="isDefault != null">is_default=#{isDefault},</if>
<if test="status != null">status=#{status},</if>
<if test="remark != null">remark=#{remark},</if>
<if test="updateBy != null">update_by=#{updateBy},</if>
update_time = sysdate()
</trim>
where id = #{id}
</update>
<update id="clearDefault">
update banma_account set is_default = 0 where is_default = 1
</update>
<delete id="deleteById" parameterType="long">
delete from banma_account where id = #{id}
</delete>
</mapper>

20
sql/banma_account.sql Normal file
View File

@@ -0,0 +1,20 @@
-- 斑马账号表(与 BanmaAccount 实体、BanmaAccountMapper.xml 一致)
DROP TABLE IF EXISTS `banma_account`;
CREATE TABLE `banma_account` (
`id` BIGINT NOT NULL AUTO_INCREMENT COMMENT '主键',
`name` VARCHAR(64) DEFAULT NULL COMMENT '显示名',
`username` VARCHAR(128) DEFAULT NULL COMMENT '登录用户名',
`token` VARCHAR(512) DEFAULT NULL COMMENT '访问Token客户端刷新后回写',
`token_expire_at` DATETIME DEFAULT NULL COMMENT 'Token过期时间',
`is_default` TINYINT(1) NOT NULL DEFAULT 0 COMMENT '是否默认 1是 0否',
`status` TINYINT(1) NOT NULL DEFAULT 1 COMMENT '状态 1启用 0停用',
`remark` VARCHAR(255) DEFAULT NULL COMMENT '备注',
`create_by` VARCHAR(64) DEFAULT NULL,
`create_time` DATETIME DEFAULT CURRENT_TIMESTAMP,
`update_by` VARCHAR(64) DEFAULT NULL,
`update_time` DATETIME DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
PRIMARY KEY (`id`),
KEY `idx_banma_is_default` (`is_default`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci COMMENT='斑马账号表';