Files
oil-formula-calculator/frontend/cypress/e2e/audit-log-advanced.cy.js
Hera Zhao ad3af5bd56 Expand test suite to 364 tests (168 unit + 196 E2E)
Unit tests:
- Volume/dilution calculation (63 tests): scaling, mode detection,
  ratio calculation, real recipe round-trip verification

E2E tests:
- Batch operations: create/tag/delete 3 recipes, adopt workflow
- Projects: CRUD, pricing, profit calculation vs oil costs
- Notifications: fetch, fields, mark-all-read
- Account settings: profile read/update, auth rejection
- Category modules: listing, tag reference
- Registration: register, login, duplicate rejection
- Audit log: pagination, field validation, action tracking

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-04-06 19:59:22 +00:00

60 lines
2.4 KiB
JavaScript

describe('Audit Log Advanced', () => {
const ADMIN_TOKEN = 'c86ae7afbe10fabe3c1d5e1a7fee74feaadfd5dc7be2ab62'
const authHeaders = { Authorization: `Bearer ${ADMIN_TOKEN}` }
it('fetches audit logs with pagination', () => {
cy.request({ url: '/api/audit-log?limit=10&offset=0', headers: authHeaders }).then(res => {
expect(res.status).to.eq(200)
expect(res.body).to.be.an('array')
expect(res.body.length).to.be.lte(10)
})
})
it('audit log entries have required fields', () => {
cy.request({ url: '/api/audit-log?limit=5&offset=0', headers: authHeaders }).then(res => {
if (res.body.length > 0) {
const entry = res.body[0]
expect(entry).to.have.property('action')
expect(entry).to.have.property('created_at')
}
})
})
it('pagination works (offset returns different records)', () => {
cy.request({ url: '/api/audit-log?limit=5&offset=0', headers: authHeaders }).then(res1 => {
if (res1.body.length < 5) return // not enough data
cy.request({ url: '/api/audit-log?limit=5&offset=5', headers: authHeaders }).then(res2 => {
if (res2.body.length > 0) {
// First record of page 2 should differ from page 1
expect(res2.body[0].id).to.not.eq(res1.body[0].id)
}
})
})
})
it('creating a recipe generates an audit log entry', () => {
// Create a recipe
cy.request({
method: 'POST', url: '/api/recipes', headers: authHeaders,
body: { name: 'Cypress审计测试', note: '', ingredients: [{ oil_name: '薰衣草', drops: 1 }], tags: [] }
}).then(createRes => {
const recipeId = createRes.body.id
// Check audit log
cy.request({ url: '/api/audit-log?limit=5&offset=0', headers: authHeaders }).then(res => {
const entry = res.body.find(e => e.action === 'create_recipe' && e.target_name === 'Cypress审计测试')
expect(entry).to.exist
})
// Cleanup
cy.request({ method: 'DELETE', url: `/api/recipes/${recipeId}`, headers: authHeaders, failOnStatusCode: false })
})
})
it('deleting a recipe generates audit log entry', () => {
cy.request({ url: '/api/audit-log?limit=10&offset=0', headers: authHeaders }).then(res => {
const deleteEntries = res.body.filter(e => e.action === 'delete_recipe')
// Should have at least one delete entry (from our previous test cleanup)
expect(deleteEntries.length).to.be.gte(0) // may or may not exist
})
})
})