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>
75 lines
2.6 KiB
JavaScript
75 lines
2.6 KiB
JavaScript
describe('Batch Operations', () => {
|
|
const ADMIN_TOKEN = 'c86ae7afbe10fabe3c1d5e1a7fee74feaadfd5dc7be2ab62'
|
|
const authHeaders = { Authorization: `Bearer ${ADMIN_TOKEN}` }
|
|
|
|
describe('Batch tag operations via API', () => {
|
|
let testRecipeIds = []
|
|
|
|
before(() => {
|
|
// Create 3 test recipes
|
|
const recipes = ['Cypress批量1', 'Cypress批量2', 'Cypress批量3']
|
|
recipes.forEach(name => {
|
|
cy.request({
|
|
method: 'POST', url: '/api/recipes', headers: authHeaders,
|
|
body: { name, note: 'batch test', ingredients: [{ oil_name: '薰衣草', drops: 5 }], tags: [] }
|
|
}).then(res => testRecipeIds.push(res.body.id))
|
|
})
|
|
})
|
|
|
|
it('created 3 test recipes', () => {
|
|
expect(testRecipeIds).to.have.length(3)
|
|
})
|
|
|
|
it('can update tags on each recipe', () => {
|
|
testRecipeIds.forEach(id => {
|
|
cy.request({
|
|
method: 'PUT', url: `/api/recipes/${id}`, headers: authHeaders,
|
|
body: { tags: ['cypress-batch-tag'] }
|
|
}).then(res => expect(res.status).to.eq(200))
|
|
})
|
|
})
|
|
|
|
it('verifies tags were applied', () => {
|
|
cy.request('/api/recipes').then(res => {
|
|
const tagged = res.body.filter(r => (r.tags || []).includes('cypress-batch-tag'))
|
|
expect(tagged.length).to.be.gte(3)
|
|
})
|
|
})
|
|
|
|
it('can delete all test recipes', () => {
|
|
testRecipeIds.forEach(id => {
|
|
cy.request({
|
|
method: 'DELETE', url: `/api/recipes/${id}`, headers: authHeaders
|
|
}).then(res => expect(res.status).to.eq(200))
|
|
})
|
|
})
|
|
|
|
it('verifies recipes are deleted', () => {
|
|
cy.request('/api/recipes').then(res => {
|
|
const found = res.body.filter(r => r.name && r.name.startsWith('Cypress批量'))
|
|
expect(found).to.have.length(0)
|
|
})
|
|
})
|
|
|
|
after(() => {
|
|
// Cleanup tag
|
|
cy.request({ method: 'DELETE', url: '/api/tags/cypress-batch-tag', headers: authHeaders, failOnStatusCode: false })
|
|
// Cleanup any remaining test recipes
|
|
cy.request('/api/recipes').then(res => {
|
|
res.body.filter(r => r.name && r.name.startsWith('Cypress批量')).forEach(r => {
|
|
cy.request({ method: 'DELETE', url: `/api/recipes/${r.id}`, headers: authHeaders, failOnStatusCode: false })
|
|
})
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('Recipe adopt workflow (admin)', () => {
|
|
// Test the adopt/review workflow that admin uses to approve user-submitted recipes
|
|
it('lists recipes and checks for owner_id field', () => {
|
|
cy.request('/api/recipes').then(res => {
|
|
expect(res.body[0]).to.have.property('owner_id')
|
|
})
|
|
})
|
|
})
|
|
})
|