develop #2
@@ -45,3 +45,32 @@ CREATE TABLE IF NOT EXISTS user_role (
|
|||||||
-- 创建索引
|
-- 创建索引
|
||||||
CREATE INDEX IF NOT EXISTS idx_user_role_user_id ON user_role(user_id);
|
CREATE INDEX IF NOT EXISTS idx_user_role_user_id ON user_role(user_id);
|
||||||
CREATE INDEX IF NOT EXISTS idx_user_role_role_id ON user_role(role_id);
|
CREATE INDEX IF NOT EXISTS idx_user_role_role_id ON user_role(role_id);
|
||||||
|
|
||||||
|
-- 创建审计日志表
|
||||||
|
CREATE TABLE IF NOT EXISTS audit_log (
|
||||||
|
id BIGINT AUTO_INCREMENT PRIMARY KEY,
|
||||||
|
entity_type VARCHAR(100) NOT NULL,
|
||||||
|
entity_id BIGINT,
|
||||||
|
operation_type VARCHAR(20) NOT NULL,
|
||||||
|
operator VARCHAR(100),
|
||||||
|
operation_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
before_data CLOB,
|
||||||
|
after_data CLOB,
|
||||||
|
changed_fields CLOB,
|
||||||
|
ip_address VARCHAR(50),
|
||||||
|
user_agent CLOB,
|
||||||
|
description CLOB,
|
||||||
|
create_by VARCHAR(50),
|
||||||
|
update_by VARCHAR(50),
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
deleted_at TIMESTAMP
|
||||||
|
);
|
||||||
|
|
||||||
|
-- 创建审计日志索引
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_audit_log_entity_type ON audit_log(entity_type);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_audit_log_entity_id ON audit_log(entity_id);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_audit_log_operation_type ON audit_log(operation_type);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_audit_log_operator ON audit_log(operator);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_audit_log_operation_time ON audit_log(operation_time);
|
||||||
|
CREATE INDEX IF NOT EXISTS idx_audit_log_entity ON audit_log(entity_type, entity_id);
|
||||||
|
|||||||
@@ -1,6 +1,10 @@
|
|||||||
package cn.novalon.manage.db.entity;
|
package cn.novalon.manage.db.entity;
|
||||||
|
|
||||||
|
import org.springframework.data.annotation.CreatedBy;
|
||||||
|
import org.springframework.data.annotation.CreatedDate;
|
||||||
import org.springframework.data.annotation.Id;
|
import org.springframework.data.annotation.Id;
|
||||||
|
import org.springframework.data.annotation.LastModifiedBy;
|
||||||
|
import org.springframework.data.annotation.LastModifiedDate;
|
||||||
import org.springframework.data.domain.Persistable;
|
import org.springframework.data.domain.Persistable;
|
||||||
import org.springframework.data.relational.core.mapping.Column;
|
import org.springframework.data.relational.core.mapping.Column;
|
||||||
|
|
||||||
@@ -17,15 +21,19 @@ public abstract class BaseEntity implements Persistable<Long> {
|
|||||||
@Id
|
@Id
|
||||||
private Long id;
|
private Long id;
|
||||||
|
|
||||||
|
@CreatedBy
|
||||||
@Column("create_by")
|
@Column("create_by")
|
||||||
private String createBy;
|
private String createBy;
|
||||||
|
|
||||||
|
@LastModifiedBy
|
||||||
@Column("update_by")
|
@Column("update_by")
|
||||||
private String updateBy;
|
private String updateBy;
|
||||||
|
|
||||||
|
@CreatedDate
|
||||||
@Column("created_at")
|
@Column("created_at")
|
||||||
private LocalDateTime createdAt;
|
private LocalDateTime createdAt;
|
||||||
|
|
||||||
|
@LastModifiedDate
|
||||||
@Column("updated_at")
|
@Column("updated_at")
|
||||||
private LocalDateTime updatedAt;
|
private LocalDateTime updatedAt;
|
||||||
|
|
||||||
|
|||||||
+4
@@ -5,6 +5,8 @@ import cn.novalon.manage.sys.audit.repository.IAuditLogRepository;
|
|||||||
import cn.novalon.manage.db.converter.AuditLogConverter;
|
import cn.novalon.manage.db.converter.AuditLogConverter;
|
||||||
import cn.novalon.manage.db.dao.AuditLogDao;
|
import cn.novalon.manage.db.dao.AuditLogDao;
|
||||||
import cn.novalon.manage.db.entity.AuditLogEntity;
|
import cn.novalon.manage.db.entity.AuditLogEntity;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.stereotype.Repository;
|
import org.springframework.stereotype.Repository;
|
||||||
import reactor.core.publisher.Flux;
|
import reactor.core.publisher.Flux;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
@@ -20,6 +22,8 @@ import java.time.LocalDateTime;
|
|||||||
@Repository
|
@Repository
|
||||||
public class AuditLogRepository implements IAuditLogRepository {
|
public class AuditLogRepository implements IAuditLogRepository {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(AuditLogRepository.class);
|
||||||
|
|
||||||
private final AuditLogDao auditLogDao;
|
private final AuditLogDao auditLogDao;
|
||||||
private final AuditLogConverter auditLogConverter;
|
private final AuditLogConverter auditLogConverter;
|
||||||
|
|
||||||
|
|||||||
+7
-8
@@ -1,7 +1,6 @@
|
|||||||
-- Novalon管理系统审计日志表
|
-- Novalon管理系统审计日志表
|
||||||
-- 版本: V7
|
-- 版本: V7
|
||||||
-- 描述: 创建审计日志表,记录数据变更前后的完整对比
|
-- 描述: 创建审计日志表,记录数据变更前后的完整对比
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS audit_log (
|
CREATE TABLE IF NOT EXISTS audit_log (
|
||||||
id BIGSERIAL PRIMARY KEY,
|
id BIGSERIAL PRIMARY KEY,
|
||||||
entity_type VARCHAR(100) NOT NULL,
|
entity_type VARCHAR(100) NOT NULL,
|
||||||
@@ -11,20 +10,22 @@ CREATE TABLE IF NOT EXISTS audit_log (
|
|||||||
operation_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
operation_time TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
before_data JSONB,
|
before_data JSONB,
|
||||||
after_data JSONB,
|
after_data JSONB,
|
||||||
changed_fields TEXT[],
|
changed_fields TEXT [],
|
||||||
ip_address VARCHAR(50),
|
ip_address VARCHAR(50),
|
||||||
user_agent TEXT,
|
user_agent TEXT,
|
||||||
description TEXT,
|
description TEXT,
|
||||||
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
|
create_by VARCHAR(50),
|
||||||
|
update_by VARCHAR(50),
|
||||||
|
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
deleted_at TIMESTAMP
|
||||||
);
|
);
|
||||||
|
|
||||||
CREATE INDEX idx_audit_log_entity_type ON audit_log(entity_type);
|
CREATE INDEX idx_audit_log_entity_type ON audit_log(entity_type);
|
||||||
CREATE INDEX idx_audit_log_entity_id ON audit_log(entity_id);
|
CREATE INDEX idx_audit_log_entity_id ON audit_log(entity_id);
|
||||||
CREATE INDEX idx_audit_log_operation_type ON audit_log(operation_type);
|
CREATE INDEX idx_audit_log_operation_type ON audit_log(operation_type);
|
||||||
CREATE INDEX idx_audit_log_operator ON audit_log(operator);
|
CREATE INDEX idx_audit_log_operator ON audit_log(operator);
|
||||||
CREATE INDEX idx_audit_log_operation_time ON audit_log(operation_time);
|
CREATE INDEX idx_audit_log_operation_time ON audit_log(operation_time);
|
||||||
CREATE INDEX idx_audit_log_entity ON audit_log(entity_type, entity_id);
|
CREATE INDEX idx_audit_log_entity ON audit_log(entity_type, entity_id);
|
||||||
|
|
||||||
COMMENT ON TABLE audit_log IS '审计日志表';
|
COMMENT ON TABLE audit_log IS '审计日志表';
|
||||||
COMMENT ON COLUMN audit_log.id IS '主键ID';
|
COMMENT ON COLUMN audit_log.id IS '主键ID';
|
||||||
COMMENT ON COLUMN audit_log.entity_type IS '实体类型(如User, Role等)';
|
COMMENT ON COLUMN audit_log.entity_type IS '实体类型(如User, Role等)';
|
||||||
@@ -35,7 +36,5 @@ COMMENT ON COLUMN audit_log.operation_time IS '操作时间';
|
|||||||
COMMENT ON COLUMN audit_log.before_data IS '变更前数据(JSON格式)';
|
COMMENT ON COLUMN audit_log.before_data IS '变更前数据(JSON格式)';
|
||||||
COMMENT ON COLUMN audit_log.after_data IS '变更后数据(JSON格式)';
|
COMMENT ON COLUMN audit_log.after_data IS '变更后数据(JSON格式)';
|
||||||
COMMENT ON COLUMN audit_log.changed_fields IS '变更字段列表';
|
COMMENT ON COLUMN audit_log.changed_fields IS '变更字段列表';
|
||||||
COMMENT ON COLUMN audit_log.ip_address IS 'IP地址';
|
COMMENT ON COLUMN audit_log.ip_address IS 'IP地址';COMMENT ON COLUMN audit_log.description IS '操作描述';
|
||||||
COMMENT ON COLUMN audit_log.user_agent IS '用户代理';
|
|
||||||
COMMENT ON COLUMN audit_log.description IS '操作描述';
|
|
||||||
COMMENT ON COLUMN audit_log.created_at IS '记录创建时间';
|
COMMENT ON COLUMN audit_log.created_at IS '记录创建时间';
|
||||||
|
|||||||
+15
-5
@@ -1,7 +1,7 @@
|
|||||||
package cn.novalon.manage.sys.audit;
|
package cn.novalon.manage.sys.audit;
|
||||||
|
|
||||||
import cn.novalon.manage.sys.audit.domain.AuditLog;
|
import cn.novalon.manage.sys.audit.domain.AuditLog;
|
||||||
import cn.novalon.manage.sys.audit.repository.IAuditLogRepository;
|
import cn.novalon.manage.sys.audit.service.IAuditLogService;
|
||||||
import com.fasterxml.jackson.databind.JsonNode;
|
import com.fasterxml.jackson.databind.JsonNode;
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper;
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import org.aspectj.lang.ProceedingJoinPoint;
|
import org.aspectj.lang.ProceedingJoinPoint;
|
||||||
@@ -34,11 +34,11 @@ public class AuditLogAspect {
|
|||||||
|
|
||||||
private static final Logger logger = LoggerFactory.getLogger(AuditLogAspect.class);
|
private static final Logger logger = LoggerFactory.getLogger(AuditLogAspect.class);
|
||||||
|
|
||||||
private final IAuditLogRepository auditLogRepository;
|
private final IAuditLogService auditLogService;
|
||||||
private final ObjectMapper objectMapper;
|
private final ObjectMapper objectMapper;
|
||||||
|
|
||||||
public AuditLogAspect(IAuditLogRepository auditLogRepository, ObjectMapper objectMapper) {
|
public AuditLogAspect(IAuditLogService auditLogService, ObjectMapper objectMapper) {
|
||||||
this.auditLogRepository = auditLogRepository;
|
this.auditLogService = auditLogService;
|
||||||
this.objectMapper = objectMapper;
|
this.objectMapper = objectMapper;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -99,6 +99,9 @@ public class AuditLogAspect {
|
|||||||
String finalOperationType = operationTypeHolder[0];
|
String finalOperationType = operationTypeHolder[0];
|
||||||
String finalBeforeData = beforeDataHolder[0];
|
String finalBeforeData = beforeDataHolder[0];
|
||||||
|
|
||||||
|
logger.debug("保存操作审计日志: entityType={}, entityIdHolder={}, extractedEntityId={}, finalEntityId={}",
|
||||||
|
entityType, entityIdHolder[0], extractEntityId(savedEntity), finalEntityId);
|
||||||
|
|
||||||
return createAndSaveAuditLog(
|
return createAndSaveAuditLog(
|
||||||
entityType, finalEntityId, finalOperationType,
|
entityType, finalEntityId, finalOperationType,
|
||||||
finalBeforeData, afterData, savedEntity
|
finalBeforeData, afterData, savedEntity
|
||||||
@@ -163,6 +166,7 @@ public class AuditLogAspect {
|
|||||||
private Mono<Void> createAndSaveAuditLog(String entityType, Long entityId,
|
private Mono<Void> createAndSaveAuditLog(String entityType, Long entityId,
|
||||||
String operationType, String beforeData,
|
String operationType, String beforeData,
|
||||||
String afterData, Object entity) {
|
String afterData, Object entity) {
|
||||||
|
logger.debug("创建审计日志: entityType={}, entityId={}, operationType={}", entityType, entityId, operationType);
|
||||||
return ReactiveSecurityContextHolder.getContext()
|
return ReactiveSecurityContextHolder.getContext()
|
||||||
.map(ctx -> ctx.getAuthentication().getPrincipal())
|
.map(ctx -> ctx.getAuthentication().getPrincipal())
|
||||||
.defaultIfEmpty("system")
|
.defaultIfEmpty("system")
|
||||||
@@ -175,6 +179,9 @@ public class AuditLogAspect {
|
|||||||
auditLog.setBeforeData(beforeData);
|
auditLog.setBeforeData(beforeData);
|
||||||
auditLog.setAfterData(afterData);
|
auditLog.setAfterData(afterData);
|
||||||
|
|
||||||
|
logger.debug("审计日志对象: entityId={}, entityType={}, operationType={}",
|
||||||
|
auditLog.getEntityId(), auditLog.getEntityType(), auditLog.getOperationType());
|
||||||
|
|
||||||
if (beforeData != null && afterData != null) {
|
if (beforeData != null && afterData != null) {
|
||||||
String[] changedFields = extractChangedFields(beforeData, afterData);
|
String[] changedFields = extractChangedFields(beforeData, afterData);
|
||||||
auditLog.setChangedFields(changedFields);
|
auditLog.setChangedFields(changedFields);
|
||||||
@@ -182,7 +189,7 @@ public class AuditLogAspect {
|
|||||||
|
|
||||||
auditLog.setDescription(generateDescription(entityType, operationType, entityId));
|
auditLog.setDescription(generateDescription(entityType, operationType, entityId));
|
||||||
|
|
||||||
return auditLogRepository.save(auditLog)
|
return auditLogService.save(auditLog)
|
||||||
.doOnSuccess(saved -> logger.debug("审计日志保存成功: {} - {}",
|
.doOnSuccess(saved -> logger.debug("审计日志保存成功: {} - {}",
|
||||||
entityType, operationType))
|
entityType, operationType))
|
||||||
.doOnError(error -> logger.error("审计日志保存失败: {}",
|
.doOnError(error -> logger.error("审计日志保存失败: {}",
|
||||||
@@ -231,11 +238,14 @@ public class AuditLogAspect {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private Long extractEntityId(Object entity) {
|
private Long extractEntityId(Object entity) {
|
||||||
|
logger.debug("提取实体ID: entity class={}", entity.getClass().getName());
|
||||||
if (entity instanceof Persistable) {
|
if (entity instanceof Persistable) {
|
||||||
Persistable<?> persistable = (Persistable<?>) entity;
|
Persistable<?> persistable = (Persistable<?>) entity;
|
||||||
Object id = persistable.getId();
|
Object id = persistable.getId();
|
||||||
|
logger.debug("Persistable实体ID: id={}, isNew={}", id, persistable.isNew());
|
||||||
return id != null ? ((Number) id).longValue() : null;
|
return id != null ? ((Number) id).longValue() : null;
|
||||||
}
|
}
|
||||||
|
logger.debug("实体不是Persistable类型");
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
-1
@@ -49,7 +49,6 @@ public class AuditLog extends BaseDomain {
|
|||||||
|
|
||||||
public AuditLog() {
|
public AuditLog() {
|
||||||
this.operationTime = LocalDateTime.now();
|
this.operationTime = LocalDateTime.now();
|
||||||
this.createdAt = LocalDateTime.now();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getEntityType() {
|
public String getEntityType() {
|
||||||
|
|||||||
+30
@@ -0,0 +1,30 @@
|
|||||||
|
package cn.novalon.manage.sys.audit.service;
|
||||||
|
|
||||||
|
import cn.novalon.manage.sys.audit.domain.AuditLog;
|
||||||
|
import reactor.core.publisher.Flux;
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 审计日志服务接口
|
||||||
|
*
|
||||||
|
* @author 张翔
|
||||||
|
* @date 2026-04-08
|
||||||
|
*/
|
||||||
|
public interface IAuditLogService {
|
||||||
|
|
||||||
|
Mono<AuditLog> save(AuditLog auditLog);
|
||||||
|
|
||||||
|
Mono<AuditLog> findById(Long id);
|
||||||
|
|
||||||
|
Flux<AuditLog> findAll();
|
||||||
|
|
||||||
|
Flux<AuditLog> findByEntityType(String entityType);
|
||||||
|
|
||||||
|
Flux<AuditLog> findByEntityId(Long entityId);
|
||||||
|
|
||||||
|
Flux<AuditLog> findByEntityTypeAndEntityId(String entityType, Long entityId);
|
||||||
|
|
||||||
|
Flux<AuditLog> findByOperator(String operator);
|
||||||
|
|
||||||
|
Flux<AuditLog> findByOperationType(String operationType);
|
||||||
|
}
|
||||||
+68
@@ -0,0 +1,68 @@
|
|||||||
|
package cn.novalon.manage.sys.audit.service.impl;
|
||||||
|
|
||||||
|
import cn.novalon.manage.sys.audit.domain.AuditLog;
|
||||||
|
import cn.novalon.manage.sys.audit.repository.IAuditLogRepository;
|
||||||
|
import cn.novalon.manage.sys.audit.service.IAuditLogService;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
import reactor.core.publisher.Flux;
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 审计日志服务实现类
|
||||||
|
*
|
||||||
|
* @author 张翔
|
||||||
|
* @date 2026-04-08
|
||||||
|
*/
|
||||||
|
@Service
|
||||||
|
public class AuditLogService implements IAuditLogService {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(AuditLogService.class);
|
||||||
|
|
||||||
|
private final IAuditLogRepository auditLogRepository;
|
||||||
|
|
||||||
|
public AuditLogService(IAuditLogRepository auditLogRepository) {
|
||||||
|
this.auditLogRepository = auditLogRepository;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Mono<AuditLog> save(AuditLog auditLog) {
|
||||||
|
return auditLogRepository.save(auditLog);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Mono<AuditLog> findById(Long id) {
|
||||||
|
return auditLogRepository.findById(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<AuditLog> findAll() {
|
||||||
|
return auditLogRepository.findAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<AuditLog> findByEntityType(String entityType) {
|
||||||
|
return auditLogRepository.findByEntityType(entityType);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<AuditLog> findByEntityId(Long entityId) {
|
||||||
|
return auditLogRepository.findByEntityId(entityId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<AuditLog> findByEntityTypeAndEntityId(String entityType, Long entityId) {
|
||||||
|
return auditLogRepository.findByEntityTypeAndEntityId(entityType, entityId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<AuditLog> findByOperator(String operator) {
|
||||||
|
return auditLogRepository.findByOperator(operator);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Flux<AuditLog> findByOperationType(String operationType) {
|
||||||
|
return auditLogRepository.findByOperationType(operationType);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,140 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { SystemConfigPage } from '../pages/SystemConfigPage';
|
||||||
|
|
||||||
|
test.describe('系统配置工作流', () => {
|
||||||
|
let configPage: SystemConfigPage;
|
||||||
|
const timestamp = Date.now();
|
||||||
|
const configKey = `test_config_${timestamp}`;
|
||||||
|
const configName = `测试配置_${timestamp}`;
|
||||||
|
const configValue = `测试值_${timestamp}`;
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
configPage = new SystemConfigPage(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('查看系统配置列表', async ({ page }) => {
|
||||||
|
await test.step('导航到系统配置页面', async () => {
|
||||||
|
await configPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证表格显示', async () => {
|
||||||
|
await expect(configPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证数据加载', async () => {
|
||||||
|
const rowCount = await configPage.getTableRowCount();
|
||||||
|
console.log(`系统配置列表包含 ${rowCount} 条记录`);
|
||||||
|
expect(rowCount).toBeGreaterThanOrEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('新增系统配置', async ({ page }) => {
|
||||||
|
await test.step('导航到系统配置页面', async () => {
|
||||||
|
await configPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击新增配置按钮', async () => {
|
||||||
|
await configPage.addButton.click();
|
||||||
|
await configPage.dialog.waitFor({ state: 'visible', timeout: 5000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('填写配置表单', async () => {
|
||||||
|
await configPage.configNameInput.fill(configName);
|
||||||
|
await configPage.configKeyInput.fill(configKey);
|
||||||
|
await configPage.configValueInput.fill(configValue);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('提交表单', async () => {
|
||||||
|
await configPage.saveButton.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证创建成功', async () => {
|
||||||
|
await expect(configPage.dialog).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`配置 ${configName} 创建完成`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('编辑系统配置', async ({ page }) => {
|
||||||
|
await test.step('导航到系统配置页面', async () => {
|
||||||
|
await configPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(configPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击编辑按钮', async () => {
|
||||||
|
const rows = await configPage.getTableRowCount();
|
||||||
|
if (rows > 0) {
|
||||||
|
const firstRow = configPage.table.locator('tr').first();
|
||||||
|
const editBtn = firstRow.getByRole('button', { name: '编辑' });
|
||||||
|
if (await editBtn.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await editBtn.click();
|
||||||
|
await configPage.dialog.waitFor({ state: 'visible', timeout: 5000 });
|
||||||
|
|
||||||
|
await test.step('修改配置值', async () => {
|
||||||
|
const newValue = `更新值_${timestamp}`;
|
||||||
|
await configPage.configValueInput.clear();
|
||||||
|
await configPage.configValueInput.fill(newValue);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('提交表单', async () => {
|
||||||
|
await configPage.saveButton.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证更新成功', async () => {
|
||||||
|
await expect(configPage.dialog).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`配置已更新`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('未找到编辑按钮,跳过编辑测试');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('当前没有配置记录,跳过编辑测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('删除系统配置', async ({ page }) => {
|
||||||
|
await test.step('导航到系统配置页面', async () => {
|
||||||
|
await configPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(configPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击删除按钮', async () => {
|
||||||
|
const rows = await configPage.getTableRowCount();
|
||||||
|
if (rows > 0) {
|
||||||
|
const firstRow = configPage.table.locator('tr').first();
|
||||||
|
const deleteBtn = firstRow.getByRole('button', { name: '删除' });
|
||||||
|
if (await deleteBtn.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await deleteBtn.click();
|
||||||
|
const confirmBtn = page.locator('.el-message-box');
|
||||||
|
await confirmBtn.waitFor({ state: 'visible', timeout: 3000 });
|
||||||
|
|
||||||
|
await test.step('确认删除', async () => {
|
||||||
|
const confirmBtn = page.locator('.el-message-box').getByRole('button', { name: '确定' });
|
||||||
|
if (await confirmBtn.isVisible({ timeout: 2000 }).catch(() => false)) {
|
||||||
|
await confirmBtn.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证删除成功', async () => {
|
||||||
|
const messageBox = page.locator('.el-message-box');
|
||||||
|
await expect(messageBox).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`配置已删除`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('未找到删除按钮,跳过删除测试');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('当前没有配置记录,跳过删除测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,138 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { DictionaryManagementPage } from '../pages/DictionaryManagementPage';
|
||||||
|
|
||||||
|
test.describe('字典管理工作流', () => {
|
||||||
|
let dictPage: DictionaryManagementPage;
|
||||||
|
const timestamp = Date.now();
|
||||||
|
const dictType = `test_dict_${timestamp}`;
|
||||||
|
const dictName = `测试字典_${timestamp}`;
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
dictPage = new DictionaryManagementPage(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('查看字典列表', async ({ page }) => {
|
||||||
|
await test.step('导航到字典管理页面', async () => {
|
||||||
|
await dictPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证表格显示', async () => {
|
||||||
|
await expect(dictPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证数据加载', async () => {
|
||||||
|
const rowCount = await dictPage.getDictCount();
|
||||||
|
console.log(`字典列表包含 ${rowCount} 条记录`);
|
||||||
|
expect(rowCount).toBeGreaterThanOrEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('新增字典', async ({ page }) => {
|
||||||
|
await test.step('导航到字典管理页面', async () => {
|
||||||
|
await dictPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击新增字典按钮', async () => {
|
||||||
|
await dictPage.createDictButton.click();
|
||||||
|
await dictPage.dialog.waitFor({ state: 'visible', timeout: 5000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('填写字典表单', async () => {
|
||||||
|
await dictPage.dictNameInput.fill(dictName);
|
||||||
|
await dictPage.dictTypeInput.fill(dictType);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('提交表单', async () => {
|
||||||
|
await dictPage.saveButton.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证创建成功', async () => {
|
||||||
|
await expect(dictPage.dialog).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`字典 ${dictName} 创建完成`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('编辑字典', async ({ page }) => {
|
||||||
|
await test.step('导航到字典管理页面', async () => {
|
||||||
|
await dictPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(dictPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击编辑按钮', async () => {
|
||||||
|
const rows = await dictPage.getDictCount();
|
||||||
|
if (rows > 0) {
|
||||||
|
const firstRow = dictPage.table.locator('tr').first();
|
||||||
|
const editBtn = firstRow.getByRole('button', { name: '编辑' });
|
||||||
|
if (await editBtn.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await editBtn.click();
|
||||||
|
await dictPage.dialog.waitFor({ state: 'visible', timeout: 5000 });
|
||||||
|
|
||||||
|
await test.step('修改字典名称', async () => {
|
||||||
|
const newName = `更新字典_${timestamp}`;
|
||||||
|
await dictPage.dictNameInput.clear();
|
||||||
|
await dictPage.dictNameInput.fill(newName);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('提交表单', async () => {
|
||||||
|
await dictPage.saveButton.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证更新成功', async () => {
|
||||||
|
await expect(dictPage.dialog).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`字典已更新`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('未找到编辑按钮,跳过编辑测试');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('当前没有字典记录,跳过编辑测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('删除字典', async ({ page }) => {
|
||||||
|
await test.step('导航到字典管理页面', async () => {
|
||||||
|
await dictPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(dictPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击删除按钮', async () => {
|
||||||
|
const rows = await dictPage.getDictCount();
|
||||||
|
if (rows > 0) {
|
||||||
|
const firstRow = dictPage.table.locator('tr').first();
|
||||||
|
const deleteBtn = firstRow.getByRole('button', { name: '删除' });
|
||||||
|
if (await deleteBtn.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await deleteBtn.click();
|
||||||
|
const confirmBtn = page.locator('.el-message-box');
|
||||||
|
await confirmBtn.waitFor({ state: 'visible', timeout: 3000 });
|
||||||
|
|
||||||
|
await test.step('确认删除', async () => {
|
||||||
|
const confirmBtn = page.locator('.el-message-box').getByRole('button', { name: '确定' });
|
||||||
|
if (await confirmBtn.isVisible({ timeout: 2000 }).catch(() => false)) {
|
||||||
|
await confirmBtn.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证删除成功', async () => {
|
||||||
|
const messageBox = page.locator('.el-message-box');
|
||||||
|
await expect(messageBox).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`字典已删除`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('未找到删除按钮,跳过删除测试');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('当前没有字典记录,跳过删除测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { ExceptionLogPage } from '../pages/ExceptionLogPage';
|
||||||
|
|
||||||
|
test.describe('异常日志工作流', () => {
|
||||||
|
let exceptionLogPage: ExceptionLogPage;
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
exceptionLogPage = new ExceptionLogPage(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('查看异常日志列表', async ({ page }) => {
|
||||||
|
await test.step('导航到异常日志页面', async () => {
|
||||||
|
await exceptionLogPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证表格显示', async () => {
|
||||||
|
await expect(exceptionLogPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证数据加载', async () => {
|
||||||
|
const rowCount = await exceptionLogPage.getLogCount();
|
||||||
|
console.log(`异常日志列表包含 ${rowCount} 条记录`);
|
||||||
|
expect(rowCount).toBeGreaterThanOrEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('搜索异常日志', async ({ page }) => {
|
||||||
|
await test.step('导航到异常日志页面', async () => {
|
||||||
|
await exceptionLogPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('输入搜索关键词', async () => {
|
||||||
|
const searchKeyword = 'NullPointerException';
|
||||||
|
await exceptionLogPage.search(searchKeyword);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证搜索结果', async () => {
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
const rowCount = await exceptionLogPage.getLogCount();
|
||||||
|
console.log(`搜索结果包含 ${rowCount} 条记录`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('查看异常日志详情', async ({ page }) => {
|
||||||
|
await test.step('导航到异常日志页面', async () => {
|
||||||
|
await exceptionLogPage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(exceptionLogPage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击查看详情按钮', async () => {
|
||||||
|
const detailButton = page.locator('button:has-text("详情")').or(page.locator('.detail-button')).first();
|
||||||
|
if (await detailButton.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await detailButton.click();
|
||||||
|
|
||||||
|
await test.step('验证详情对话框显示', async () => {
|
||||||
|
const dialog = page.locator('.el-dialog');
|
||||||
|
await expect(dialog).toBeVisible({ timeout: 5000 });
|
||||||
|
console.log('异常日志详情对话框已打开');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('关闭详情对话框', async () => {
|
||||||
|
await exceptionLogPage.closeDetailDialog();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('当前没有异常日志记录,跳过详情查看测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,138 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
import { NotificationPage } from '../pages/NotificationPage';
|
||||||
|
|
||||||
|
test.describe('通知管理工作流', () => {
|
||||||
|
let noticePage: NotificationPage;
|
||||||
|
const timestamp = Date.now();
|
||||||
|
const noticeTitle = `测试通知_${timestamp}`;
|
||||||
|
const noticeContent = `这是测试通知内容_${timestamp}`;
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
noticePage = new NotificationPage(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('查看通知列表', async ({ page }) => {
|
||||||
|
await test.step('导航到通知管理页面', async () => {
|
||||||
|
await noticePage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证表格显示', async () => {
|
||||||
|
await expect(noticePage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证数据加载', async () => {
|
||||||
|
const rowCount = await noticePage.getTableRowCount();
|
||||||
|
console.log(`通知列表包含 ${rowCount} 条记录`);
|
||||||
|
expect(rowCount).toBeGreaterThanOrEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('新增通知', async ({ page }) => {
|
||||||
|
await test.step('导航到通知管理页面', async () => {
|
||||||
|
await noticePage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击新增通知按钮', async () => {
|
||||||
|
await noticePage.addButton.click();
|
||||||
|
await noticePage.dialog.waitFor({ state: 'visible', timeout: 5000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('填写通知表单', async () => {
|
||||||
|
await noticePage.titleInput.fill(noticeTitle);
|
||||||
|
await noticePage.contentInput.fill(noticeContent);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('提交表单', async () => {
|
||||||
|
await noticePage.saveButton.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证创建成功', async () => {
|
||||||
|
await expect(noticePage.dialog).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`通知 ${noticeTitle} 创建完成`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('编辑通知', async ({ page }) => {
|
||||||
|
await test.step('导航到通知管理页面', async () => {
|
||||||
|
await noticePage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(noticePage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击编辑按钮', async () => {
|
||||||
|
const rows = await noticePage.getTableRowCount();
|
||||||
|
if (rows > 0) {
|
||||||
|
const firstRow = noticePage.table.locator('tr').first();
|
||||||
|
const editBtn = firstRow.getByRole('button', { name: '编辑' });
|
||||||
|
if (await editBtn.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await editBtn.click();
|
||||||
|
await noticePage.dialog.waitFor({ state: 'visible', timeout: 5000 });
|
||||||
|
|
||||||
|
await test.step('修改通知内容', async () => {
|
||||||
|
const newContent = `更新通知内容_${timestamp}`;
|
||||||
|
await noticePage.contentInput.clear();
|
||||||
|
await noticePage.contentInput.fill(newContent);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('提交表单', async () => {
|
||||||
|
await noticePage.saveButton.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证更新成功', async () => {
|
||||||
|
await expect(noticePage.dialog).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`通知已更新`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('未找到编辑按钮,跳过编辑测试');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('当前没有通知记录,跳过编辑测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('删除通知', async ({ page }) => {
|
||||||
|
await test.step('导航到通知管理页面', async () => {
|
||||||
|
await noticePage.goto();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('等待数据加载', async () => {
|
||||||
|
await expect(noticePage.table).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('点击删除按钮', async () => {
|
||||||
|
const rows = await noticePage.getTableRowCount();
|
||||||
|
if (rows > 0) {
|
||||||
|
const firstRow = noticePage.table.locator('tr').first();
|
||||||
|
const deleteBtn = firstRow.getByRole('button', { name: '删除' });
|
||||||
|
if (await deleteBtn.isVisible({ timeout: 3000 }).catch(() => false)) {
|
||||||
|
await deleteBtn.click();
|
||||||
|
const confirmBtn = page.locator('.el-message-box');
|
||||||
|
await confirmBtn.waitFor({ state: 'visible', timeout: 3000 });
|
||||||
|
|
||||||
|
await test.step('确认删除', async () => {
|
||||||
|
const confirmBtn = page.locator('.el-message-box').getByRole('button', { name: '确定' });
|
||||||
|
if (await confirmBtn.isVisible({ timeout: 2000 }).catch(() => false)) {
|
||||||
|
await confirmBtn.click();
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('验证删除成功', async () => {
|
||||||
|
const messageBox = page.locator('.el-message-box');
|
||||||
|
await expect(messageBox).not.toBeVisible({ timeout: 5000 });
|
||||||
|
console.log(`通知已删除`);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('未找到删除按钮,跳过删除测试');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('当前没有通知记录,跳过删除测试');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -3,24 +3,24 @@ import { Page, Locator, expect } from '@playwright/test';
|
|||||||
export class DictionaryManagementPage {
|
export class DictionaryManagementPage {
|
||||||
readonly page: Page;
|
readonly page: Page;
|
||||||
readonly table: Locator;
|
readonly table: Locator;
|
||||||
readonly createDictTypeButton: Locator;
|
readonly createDictButton: Locator;
|
||||||
readonly createDictDataButton: Locator;
|
readonly saveButton: Locator;
|
||||||
readonly searchInput: Locator;
|
readonly dialog: Locator;
|
||||||
readonly searchButton: Locator;
|
readonly dictNameInput: Locator;
|
||||||
readonly successMessage: Locator;
|
readonly dictTypeInput: Locator;
|
||||||
readonly dictTypeTable: Locator;
|
readonly statusSelect: Locator;
|
||||||
readonly dictDataTable: Locator;
|
readonly remarkInput: Locator;
|
||||||
|
|
||||||
constructor(page: Page) {
|
constructor(page: Page) {
|
||||||
this.page = page;
|
this.page = page;
|
||||||
this.table = page.locator('.el-table').or(page.locator('.dict-table'));
|
this.table = page.locator('.el-table');
|
||||||
this.createDictTypeButton = page.getByRole('button', { name: '新增字典类型' }).or(page.locator('button:has-text("新增字典类型")'));
|
this.createDictButton = page.getByRole('button', { name: '新增字典' });
|
||||||
this.createDictDataButton = page.getByRole('button', { name: '新增字典数据' }).or(page.locator('button:has-text("新增字典数据")'));
|
this.saveButton = page.getByRole('button', { name: '确定' });
|
||||||
this.searchInput = page.locator('input[placeholder*="搜索"]').or(page.locator('input[name*="keyword"]'));
|
this.dialog = page.locator('.el-dialog');
|
||||||
this.searchButton = page.getByRole('button', { name: '搜索' }).or(page.locator('button:has-text("搜索")'));
|
this.dictNameInput = page.locator('.el-dialog').getByRole('textbox', { name: '字典名称' });
|
||||||
this.successMessage = page.locator('.el-message--success').or(page.locator('.success-message'));
|
this.dictTypeInput = page.locator('.el-dialog').getByRole('textbox', { name: '字典类型' });
|
||||||
this.dictTypeTable = page.locator('.dict-type-table').or(page.locator('.el-table').first());
|
this.statusSelect = page.locator('.el-dialog').getByRole('combobox', { name: '状态' });
|
||||||
this.dictDataTable = page.locator('.dict-data-table').or(page.locator('.el-table').nth(1));
|
this.remarkInput = page.locator('.el-dialog').getByRole('textbox', { name: '备注' });
|
||||||
}
|
}
|
||||||
|
|
||||||
async goto() {
|
async goto() {
|
||||||
@@ -40,156 +40,57 @@ export class DictionaryManagementPage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async clickCreateDictType() {
|
async createDict(dictName: string, dictType: string, status: string = '0', remark?: string) {
|
||||||
await this.createDictTypeButton.click();
|
await this.createDictButton.click();
|
||||||
await this.page.waitForTimeout(500);
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
|
await this.dictNameInput.fill(dictName);
|
||||||
|
await this.dictTypeInput.fill(dictType);
|
||||||
|
|
||||||
|
if (status) {
|
||||||
|
await this.statusSelect.click();
|
||||||
|
await this.page.waitForTimeout(300);
|
||||||
|
await this.page.getByRole('option', { name: status === '0' ? '正常' : '停用' }).click();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (remark) {
|
||||||
|
await this.remarkInput.fill(remark);
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.saveButton.click();
|
||||||
|
await this.page.waitForLoadState('networkidle');
|
||||||
}
|
}
|
||||||
|
|
||||||
async clickCreateDictData() {
|
async editDict(dictName: string, newDictName: string) {
|
||||||
await this.createDictDataButton.click();
|
const row = this.table.locator('tr').filter({ hasText: dictName }).first();
|
||||||
|
const editBtn = row.getByRole('button', { name: '编辑' });
|
||||||
|
await editBtn.click();
|
||||||
await this.page.waitForTimeout(500);
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
|
await this.dictNameInput.clear();
|
||||||
|
await this.dictNameInput.fill(newDictName);
|
||||||
|
|
||||||
|
await this.saveButton.click();
|
||||||
|
await this.page.waitForLoadState('networkidle');
|
||||||
}
|
}
|
||||||
|
|
||||||
async fillDictTypeForm(dictTypeData: {
|
async deleteDict(dictName: string) {
|
||||||
dictName: string;
|
const row = this.table.locator('tr').filter({ hasText: dictName }).first();
|
||||||
dictType: string;
|
const deleteBtn = row.getByRole('button', { name: '删除' });
|
||||||
status?: string;
|
await deleteBtn.click();
|
||||||
remark?: string;
|
await this.page.waitForTimeout(500);
|
||||||
}) {
|
|
||||||
const dialog = this.page.locator('.el-dialog');
|
|
||||||
|
|
||||||
await dialog.locator('input').first().fill(dictTypeData.dictName);
|
const confirmBtn = this.page.locator('.el-message-box').getByRole('button', { name: '确定' });
|
||||||
await dialog.locator('input').nth(1).fill(dictTypeData.dictType);
|
await confirmBtn.click();
|
||||||
|
await this.page.waitForLoadState('networkidle');
|
||||||
if (dictTypeData.status) {
|
|
||||||
const statusRadio = dialog.locator(`input[value="${dictTypeData.status}"]`);
|
|
||||||
if (await statusRadio.count() > 0) {
|
|
||||||
await statusRadio.check();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dictTypeData.remark) {
|
|
||||||
const remarkInput = dialog.locator('textarea');
|
|
||||||
if (await remarkInput.count() > 0) {
|
|
||||||
await remarkInput.fill(dictTypeData.remark);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fillDictDataForm(dictData: {
|
async getDictCount() {
|
||||||
dictLabel: string;
|
const rows = await this.table.locator('.el-table__row').count();
|
||||||
dictValue: string;
|
return rows;
|
||||||
dictType?: string;
|
|
||||||
cssClass?: string;
|
|
||||||
listClass?: string;
|
|
||||||
isDefault?: string;
|
|
||||||
status?: string;
|
|
||||||
sort?: number;
|
|
||||||
}) {
|
|
||||||
const dialog = this.page.locator('.el-dialog');
|
|
||||||
|
|
||||||
await dialog.locator('input').first().fill(dictData.dictLabel);
|
|
||||||
await dialog.locator('input').nth(1).fill(dictData.dictValue);
|
|
||||||
|
|
||||||
if (dictData.dictType) {
|
|
||||||
const dictTypeSelect = dialog.locator('.el-select');
|
|
||||||
if (await dictTypeSelect.count() > 0) {
|
|
||||||
await dictTypeSelect.click();
|
|
||||||
await this.page.waitForTimeout(300);
|
|
||||||
await this.page.getByRole('option', { name: dictData.dictType }).click();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dictData.cssClass) {
|
|
||||||
const cssClassInput = dialog.locator('input[placeholder*="CSS"]');
|
|
||||||
if (await cssClassInput.count() > 0) {
|
|
||||||
await cssClassInput.fill(dictData.cssClass);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dictData.listClass) {
|
|
||||||
const listClassInput = dialog.locator('input[placeholder*="列表"]');
|
|
||||||
if (await listClassInput.count() > 0) {
|
|
||||||
await listClassInput.fill(dictData.listClass);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dictData.isDefault) {
|
|
||||||
const defaultRadio = dialog.locator(`input[value="${dictData.isDefault}"]`);
|
|
||||||
if (await defaultRadio.count() > 0) {
|
|
||||||
await defaultRadio.check();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dictData.status) {
|
|
||||||
const statusRadio = dialog.locator(`input[value="${dictData.status}"]`);
|
|
||||||
if (await statusRadio.count() > 0) {
|
|
||||||
await statusRadio.check();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (dictData.sort !== undefined) {
|
|
||||||
const sortInput = dialog.locator('input[type="number"]');
|
|
||||||
if (await sortInput.count() > 0) {
|
|
||||||
await sortInput.fill(String(dictData.sort));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async submitForm() {
|
|
||||||
await this.page.getByRole('button', { name: '确定' }).or(this.page.locator('button:has-text("确定")')).click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async editDictType(dictName: string) {
|
|
||||||
const dictTypeRow = this.dictTypeTable.locator('tbody tr').filter({ hasText: dictName });
|
|
||||||
await dictTypeRow.getByRole('button', { name: '编辑' }).or(this.page.locator('.edit-button')).click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async editDictData(dictLabel: string) {
|
|
||||||
const dictDataRow = this.dictDataTable.locator('tbody tr').filter({ hasText: dictLabel });
|
|
||||||
await dictDataRow.getByRole('button', { name: '编辑' }).or(this.page.locator('.edit-button')).click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async deleteDictType(dictName: string) {
|
|
||||||
const dictTypeRow = this.dictTypeTable.locator('tbody tr').filter({ hasText: dictName });
|
|
||||||
await dictTypeRow.getByRole('button', { name: '删除' }).or(this.page.locator('.delete-button')).click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async deleteDictData(dictLabel: string) {
|
|
||||||
const dictDataRow = this.dictDataTable.locator('tbody tr').filter({ hasText: dictLabel });
|
|
||||||
await dictDataRow.getByRole('button', { name: '删除' }).or(this.page.locator('.delete-button')).click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async confirmDelete() {
|
|
||||||
await this.page.getByRole('button', { name: '确定' }).or(this.page.locator('.confirm-dialog .confirm-button')).click();
|
|
||||||
}
|
|
||||||
|
|
||||||
async search(keyword: string) {
|
|
||||||
await this.searchInput.fill(keyword);
|
|
||||||
await this.searchButton.click();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async containsText(text: string): Promise<boolean> {
|
async containsText(text: string): Promise<boolean> {
|
||||||
return await this.table.getByText(text).count() > 0;
|
return await this.table.getByText(text).count() > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
async isSuccessMessageVisible(): Promise<boolean> {
|
|
||||||
try {
|
|
||||||
return await this.successMessage.isVisible({ timeout: 3000 });
|
|
||||||
} catch {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDictTypeCount(): Promise<number> {
|
|
||||||
return await this.dictTypeTable.locator('tbody tr').count();
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDictDataCount(): Promise<number> {
|
|
||||||
return await this.dictDataTable.locator('tbody tr').count();
|
|
||||||
}
|
|
||||||
|
|
||||||
async reload() {
|
|
||||||
await this.page.reload();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,40 +4,47 @@ export class NotificationPage {
|
|||||||
readonly page: Page;
|
readonly page: Page;
|
||||||
readonly table;
|
readonly table;
|
||||||
readonly addButton;
|
readonly addButton;
|
||||||
readonly editButton;
|
|
||||||
readonly deleteButton;
|
|
||||||
readonly saveButton;
|
readonly saveButton;
|
||||||
readonly cancelButton;
|
readonly cancelButton;
|
||||||
readonly searchInput;
|
readonly dialog;
|
||||||
readonly searchButton;
|
|
||||||
readonly titleInput;
|
readonly titleInput;
|
||||||
readonly contentInput;
|
readonly contentInput;
|
||||||
readonly typeSelect;
|
readonly noticeTypeSelect;
|
||||||
readonly statusSelect;
|
readonly statusSelect;
|
||||||
|
|
||||||
constructor(page: Page) {
|
constructor(page: Page) {
|
||||||
this.page = page;
|
this.page = page;
|
||||||
this.table = page.locator('.el-table');
|
this.table = page.locator('.el-table');
|
||||||
this.addButton = page.getByRole('button', { name: '新增' });
|
this.addButton = page.getByRole('button', { name: '新增公告' });
|
||||||
this.editButton = page.getByRole('button', { name: '修改' });
|
|
||||||
this.deleteButton = page.getByRole('button', { name: '删除' });
|
|
||||||
this.saveButton = page.getByRole('button', { name: '确定' });
|
this.saveButton = page.getByRole('button', { name: '确定' });
|
||||||
this.cancelButton = page.getByRole('button', { name: '取消' });
|
this.cancelButton = page.getByRole('button', { name: '取消' });
|
||||||
this.searchInput = page.getByPlaceholder('搜索通知标题');
|
this.dialog = page.locator('.el-dialog');
|
||||||
this.searchButton = page.getByRole('button', { name: '搜索' });
|
this.titleInput = page.locator('.el-dialog').getByRole('textbox', { name: '公告标题' });
|
||||||
this.titleInput = page.getByPlaceholder('请输入通知标题');
|
this.contentInput = page.locator('.el-dialog').getByRole('textbox', { name: '公告内容' });
|
||||||
this.contentInput = page.getByPlaceholder('请输入通知内容');
|
this.noticeTypeSelect = page.locator('.el-dialog').getByRole('combobox', { name: '公告类型' });
|
||||||
this.typeSelect = page.locator('.el-select');
|
this.statusSelect = page.locator('.el-dialog').getByRole('combobox', { name: '状态' });
|
||||||
this.statusSelect = page.locator('.el-select');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async goto() {
|
async goto() {
|
||||||
await this.page.goto('/system/notice');
|
try {
|
||||||
await this.page.waitForLoadState('networkidle');
|
console.log('导航到通知管理页面...');
|
||||||
|
await this.page.goto('/notice');
|
||||||
|
|
||||||
|
await this.page.waitForLoadState('networkidle');
|
||||||
|
await this.table.waitFor({ state: 'visible', timeout: 10000 });
|
||||||
|
await expect(this.page).toHaveURL(/.*notice/);
|
||||||
|
|
||||||
|
console.log('通知管理页面加载完成');
|
||||||
|
} catch (error) {
|
||||||
|
await this.page.screenshot({ path: `test-results/notification-error-${Date.now()}.png` });
|
||||||
|
console.error('导航到通知管理页面失败:', error);
|
||||||
|
throw new Error(`导航到通知管理页面失败: ${error instanceof Error ? error.message : String(error)}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async addNotification(title: string, content: string, type: string = '1', status: string = '0') {
|
async addNotification(title: string, content: string) {
|
||||||
await this.addButton.click();
|
await this.addButton.click();
|
||||||
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
await this.titleInput.fill(title);
|
await this.titleInput.fill(title);
|
||||||
await this.contentInput.fill(content);
|
await this.contentInput.fill(content);
|
||||||
@@ -48,7 +55,9 @@ export class NotificationPage {
|
|||||||
|
|
||||||
async editNotification(title: string, newContent: string) {
|
async editNotification(title: string, newContent: string) {
|
||||||
const row = this.table.locator('tr').filter({ hasText: title }).first();
|
const row = this.table.locator('tr').filter({ hasText: title }).first();
|
||||||
await row.locator('.el-button--primary').click();
|
const editBtn = row.getByRole('button', { name: '编辑' });
|
||||||
|
await editBtn.click();
|
||||||
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
await this.contentInput.clear();
|
await this.contentInput.clear();
|
||||||
await this.contentInput.fill(newContent);
|
await this.contentInput.fill(newContent);
|
||||||
@@ -59,34 +68,21 @@ export class NotificationPage {
|
|||||||
|
|
||||||
async deleteNotification(title: string) {
|
async deleteNotification(title: string) {
|
||||||
const row = this.table.locator('tr').filter({ hasText: title }).first();
|
const row = this.table.locator('tr').filter({ hasText: title }).first();
|
||||||
await row.locator('.el-button--danger').click();
|
const deleteBtn = row.getByRole('button', { name: '删除' });
|
||||||
|
await deleteBtn.click();
|
||||||
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
await this.saveButton.click();
|
const confirmBtn = this.page.locator('.el-message-box').getByRole('button', { name: '确定' });
|
||||||
|
await confirmBtn.click();
|
||||||
await this.page.waitForLoadState('networkidle');
|
await this.page.waitForLoadState('networkidle');
|
||||||
}
|
}
|
||||||
|
|
||||||
async searchNotification(keyword: string) {
|
|
||||||
await this.searchInput.fill(keyword);
|
|
||||||
await this.searchButton.click();
|
|
||||||
await this.page.waitForLoadState('networkidle');
|
|
||||||
}
|
|
||||||
|
|
||||||
async clearSearch() {
|
|
||||||
await this.searchInput.clear();
|
|
||||||
await this.searchButton.click();
|
|
||||||
await this.page.waitForLoadState('networkidle');
|
|
||||||
}
|
|
||||||
|
|
||||||
async verifyTableContains(text: string) {
|
|
||||||
await expect(this.table).toContainText(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
async verifyTableNotContains(text: string) {
|
|
||||||
await expect(this.table).not.toContainText(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getTableRowCount() {
|
async getTableRowCount() {
|
||||||
const rows = await this.table.locator('.el-table__row').count();
|
const rows = await this.table.locator('.el-table__row').count();
|
||||||
return rows;
|
return rows;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async verifyTableContains(text: string) {
|
||||||
|
await expect(this.table).toContainText(text);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -4,31 +4,23 @@ export class SystemConfigPage {
|
|||||||
readonly page: Page;
|
readonly page: Page;
|
||||||
readonly table;
|
readonly table;
|
||||||
readonly addButton;
|
readonly addButton;
|
||||||
readonly editButton;
|
|
||||||
readonly deleteButton;
|
|
||||||
readonly saveButton;
|
readonly saveButton;
|
||||||
readonly cancelButton;
|
readonly cancelButton;
|
||||||
readonly searchInput;
|
readonly dialog;
|
||||||
readonly searchButton;
|
|
||||||
readonly configNameInput;
|
readonly configNameInput;
|
||||||
readonly configKeyInput;
|
readonly configKeyInput;
|
||||||
readonly configValueInput;
|
readonly configValueInput;
|
||||||
readonly configTypeSelect;
|
|
||||||
|
|
||||||
constructor(page: Page) {
|
constructor(page: Page) {
|
||||||
this.page = page;
|
this.page = page;
|
||||||
this.table = page.locator('.el-table');
|
this.table = page.locator('.el-table');
|
||||||
this.addButton = page.getByRole('button', { name: '新增配置' });
|
this.addButton = page.getByRole('button', { name: '新增配置' });
|
||||||
this.editButton = page.getByRole('button', { name: '编辑' });
|
|
||||||
this.deleteButton = page.getByRole('button', { name: '删除' });
|
|
||||||
this.saveButton = page.getByRole('button', { name: '确定' });
|
this.saveButton = page.getByRole('button', { name: '确定' });
|
||||||
this.cancelButton = page.getByRole('button', { name: '取消' });
|
this.cancelButton = page.getByRole('button', { name: '取消' });
|
||||||
this.searchInput = page.getByPlaceholder('搜索配置名称');
|
this.dialog = page.locator('.el-dialog');
|
||||||
this.searchButton = page.getByRole('button', { name: '搜索' });
|
this.configNameInput = page.locator('.el-dialog').getByRole('textbox', { name: '参数名称' });
|
||||||
this.configNameInput = page.getByPlaceholder('请输入配置名称');
|
this.configKeyInput = page.locator('.el-dialog').getByRole('textbox', { name: '参数键名' });
|
||||||
this.configKeyInput = page.getByPlaceholder('请输入配置键名');
|
this.configValueInput = page.locator('.el-dialog').getByRole('textbox', { name: '参数值' });
|
||||||
this.configValueInput = page.getByPlaceholder('请输入配置键值');
|
|
||||||
this.configTypeSelect = page.locator('.el-select');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async goto() {
|
async goto() {
|
||||||
@@ -48,8 +40,9 @@ export class SystemConfigPage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async addConfig(configName: string, configKey: string, configValue: string, configType: string = 'Y') {
|
async addConfig(configName: string, configKey: string, configValue: string) {
|
||||||
await this.addButton.click();
|
await this.addButton.click();
|
||||||
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
await this.configNameInput.fill(configName);
|
await this.configNameInput.fill(configName);
|
||||||
await this.configKeyInput.fill(configKey);
|
await this.configKeyInput.fill(configKey);
|
||||||
@@ -61,7 +54,9 @@ export class SystemConfigPage {
|
|||||||
|
|
||||||
async editConfig(configKey: string, newValue: string) {
|
async editConfig(configKey: string, newValue: string) {
|
||||||
const row = this.table.locator('tr').filter({ hasText: configKey }).first();
|
const row = this.table.locator('tr').filter({ hasText: configKey }).first();
|
||||||
await row.locator('.el-button--primary').click();
|
const editBtn = row.getByRole('button', { name: '编辑' });
|
||||||
|
await editBtn.click();
|
||||||
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
await this.configValueInput.clear();
|
await this.configValueInput.clear();
|
||||||
await this.configValueInput.fill(newValue);
|
await this.configValueInput.fill(newValue);
|
||||||
@@ -72,34 +67,21 @@ export class SystemConfigPage {
|
|||||||
|
|
||||||
async deleteConfig(configKey: string) {
|
async deleteConfig(configKey: string) {
|
||||||
const row = this.table.locator('tr').filter({ hasText: configKey }).first();
|
const row = this.table.locator('tr').filter({ hasText: configKey }).first();
|
||||||
await row.locator('.el-button--danger').click();
|
const deleteBtn = row.getByRole('button', { name: '删除' });
|
||||||
|
await deleteBtn.click();
|
||||||
|
await this.page.waitForTimeout(500);
|
||||||
|
|
||||||
await this.saveButton.click();
|
const confirmBtn = this.page.locator('.el-message-box').getByRole('button', { name: '确定' });
|
||||||
|
await confirmBtn.click();
|
||||||
await this.page.waitForLoadState('networkidle');
|
await this.page.waitForLoadState('networkidle');
|
||||||
}
|
}
|
||||||
|
|
||||||
async searchConfig(keyword: string) {
|
|
||||||
await this.searchInput.fill(keyword);
|
|
||||||
await this.searchButton.click();
|
|
||||||
await this.page.waitForLoadState('networkidle');
|
|
||||||
}
|
|
||||||
|
|
||||||
async clearSearch() {
|
|
||||||
await this.searchInput.clear();
|
|
||||||
await this.searchButton.click();
|
|
||||||
await this.page.waitForLoadState('networkidle');
|
|
||||||
}
|
|
||||||
|
|
||||||
async verifyTableContains(text: string) {
|
|
||||||
await expect(this.table).toContainText(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
async verifyTableNotContains(text: string) {
|
|
||||||
await expect(this.table).not.toContainText(text);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getTableRowCount() {
|
async getTableRowCount() {
|
||||||
const rows = await this.table.locator('.el-table__row').count();
|
const rows = await this.table.locator('.el-table__row').count();
|
||||||
return rows;
|
return rows;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async verifyTableContains(text: string) {
|
||||||
|
await expect(this.table).toContainText(text);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -6,7 +6,7 @@
|
|||||||
"localStorage": [
|
"localStorage": [
|
||||||
{
|
{
|
||||||
"name": "token",
|
"name": "token",
|
||||||
"value": "eyJhbGciOiJIUzM4NCJ9.eyJyb2xlcyI6WyJhZG1pbiJdLCJ1c2VySWQiOjEsInVzZXJuYW1lIjoiYWRtaW4iLCJzdWIiOiJhZG1pbiIsImlhdCI6MTc3NTYzMzUxMiwiZXhwIjoxNzc1NzE5OTEyfQ.z8GEm_YKF0jGh_KWAk9NVW41TL9JDY5RcgQaLEpktrk2JRDWs9gfuLV0fQSs3EJ5"
|
"value": "eyJhbGciOiJIUzM4NCJ9.eyJyb2xlcyI6WyJhZG1pbiJdLCJ1c2VySWQiOjEsInVzZXJuYW1lIjoiYWRtaW4iLCJzdWIiOiJhZG1pbiIsImlhdCI6MTc3NTY0ODAzOCwiZXhwIjoxNzc1NzM0NDM4fQ.jCpkwk034HQKIYBWdZ5qjIe8rkxrar6fSLNauoJM0UgOFfVSBuoxaMpIzRHC7KDS"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "permission",
|
"name": "permission",
|
||||||
|
|||||||
Reference in New Issue
Block a user