Files
oil-formula-calculator/frontend/cypress/e2e/diary-flow.cy.js
Hera Zhao b8b4eceff3
Some checks failed
PR Preview / teardown-preview (pull_request) Has been skipped
Test / unit-test (push) Successful in 5s
Test / build-check (push) Successful in 4s
PR Preview / test (pull_request) Successful in 5s
PR Preview / deploy-preview (pull_request) Successful in 12s
Test / e2e-test (push) Failing after 2m14s
fix: 修复全部27个失败的e2e测试
根本原因: 所有测试硬编码了只在生产环境有效的admin token,
CI创建新数据库时token不同导致全部认证失败。

修复:
- CI: 设置已知ADMIN_TOKEN环境变量传给后端和Cypress
- cypress/support/e2e.js: 新增cy.getAdminToken()动态获取token
- 24个spec文件: 硬编码token改为cy.getAdminToken()
- UI选择器: 适配管理页面从tab移到UserMenu、编辑器DOM变化
- API: create_recipe→share_recipe、ingredients格式、权限变化
- 超时: 300s→420s适应32个spec

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-13 21:08:40 +00:00

223 lines
6.6 KiB
JavaScript

describe('Diary Flow', () => {
let adminToken
let authHeaders
let testDiaryId = null
before(() => {
cy.getAdminToken().then(token => {
adminToken = token
authHeaders = { Authorization: `Bearer ${token}` }
})
})
describe('API: full diary lifecycle', () => {
it('creates a diary entry via API', () => {
cy.request({
method: 'POST',
url: '/api/diary',
headers: authHeaders,
body: {
name: 'Cypress_Diary_Test_日记',
ingredients: [
{ oil: '薰衣草', drops: 3 },
{ oil: '茶树', drops: 2 }
],
note: '这是E2E测试创建的日记'
}
}).then(res => {
expect(res.status).to.be.oneOf([200, 201])
testDiaryId = res.body.id || res.body._id
expect(testDiaryId).to.exist
})
})
it('verifies diary entry appears in GET /api/diary', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
expect(res.body).to.be.an('array')
const found = res.body.find(d => d.name === 'Cypress_Diary_Test_日记')
expect(found).to.exist
expect(found.ingredients).to.have.length(2)
expect(found.note).to.eq('这是E2E测试创建的日记')
testDiaryId = found.id || found._id
})
})
it('updates the diary entry via PUT', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d => d.name === 'Cypress_Diary_Test_日记')
testDiaryId = found.id || found._id
cy.request({
method: 'PUT',
url: `/api/diary/${testDiaryId}`,
headers: authHeaders,
body: {
name: 'Cypress_Diary_Updated_日记',
ingredients: [
{ oil: '薰衣草', drops: 5 },
{ oil: '乳香', drops: 3 }
],
note: '已更新的日记'
}
}).then(res => {
expect(res.status).to.eq(200)
})
})
})
it('verifies the update took effect', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d => d.name === 'Cypress_Diary_Updated_日记')
expect(found).to.exist
expect(found.note).to.eq('已更新的日记')
expect(found.ingredients).to.have.length(2)
testDiaryId = found.id || found._id
})
})
it('adds a journal entry to the diary', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d => d.name === 'Cypress_Diary_Updated_日记')
testDiaryId = found.id || found._id
cy.request({
method: 'POST',
url: `/api/diary/${testDiaryId}/entries`,
headers: authHeaders,
body: {
content: 'Cypress测试日志: 使用后感觉很好'
}
}).then(res => {
expect(res.status).to.be.oneOf([200, 201])
})
})
})
it('verifies journal entry exists in diary', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d => d.name === 'Cypress_Diary_Updated_日记')
expect(found).to.exist
expect(found.entries).to.be.an('array')
expect(found.entries.length).to.be.gte(1)
const entry = found.entries.find(e =>
(e.text || e.content || '').includes('Cypress测试日志')
)
expect(entry).to.exist
})
})
it('deletes the journal entry', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d => d.name === 'Cypress_Diary_Updated_日记')
const entry = found.entries.find(e =>
(e.text || e.content || '').includes('Cypress测试日志')
)
const entryId = entry.id || entry._id
cy.request({
method: 'DELETE',
url: `/api/diary/entries/${entryId}`,
headers: authHeaders
}).then(res => {
expect(res.status).to.eq(200)
})
})
})
it('deletes the diary entry', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d => d.name === 'Cypress_Diary_Updated_日记')
if (found) {
const id = found.id || found._id
cy.request({
method: 'DELETE',
url: `/api/diary/${id}`,
headers: authHeaders
}).then(res => {
expect(res.status).to.eq(200)
})
}
})
})
it('verifies diary entry is gone', () => {
cy.request({
url: '/api/diary',
headers: authHeaders
}).then(res => {
const found = res.body.find(d =>
d.name === 'Cypress_Diary_Updated_日记' || d.name === 'Cypress_Diary_Test_日记'
)
expect(found).to.not.exist
})
})
})
describe('UI: diary page renders', () => {
it('visits /mydiary and verifies page renders', () => {
cy.visit('/mydiary', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.get('.my-diary', { timeout: 10000 }).should('exist')
// Should show sub-tabs (品牌 and 账户)
cy.get('.sub-tab').should('have.length.gte', 2)
cy.contains('我的品牌').should('be.visible')
cy.contains('我的账户').should('be.visible')
})
it('diary grid is visible on diary tab', () => {
cy.visit('/mydiary', {
onBeforeLoad(win) {
win.localStorage.setItem('oil_auth_token', adminToken)
}
})
cy.get('.my-diary', { timeout: 10000 }).should('exist')
// Diary grid or empty hint should be present
cy.get('.diary-grid, .empty-hint').should('exist')
})
})
// Safety cleanup in case tests fail mid-way
after(() => {
cy.request({
url: '/api/diary',
headers: authHeaders,
failOnStatusCode: false
}).then(res => {
if (res.status === 200 && Array.isArray(res.body)) {
const testEntries = res.body.filter(d =>
d.name && (d.name.includes('Cypress_Diary_Test') || d.name.includes('Cypress_Diary_Updated'))
)
testEntries.forEach(entry => {
cy.request({
method: 'DELETE',
url: `/api/diary/${entry.id || entry._id}`,
headers: authHeaders,
failOnStatusCode: false
})
})
}
})
})
})