This commit is contained in:
Oscar
2026-06-03 20:21:43 +03:00
commit a38e905b29
25 changed files with 19011 additions and 0 deletions

20
backend/src/index.ts Normal file
View File

@@ -0,0 +1,20 @@
import cors from 'cors'
import express from 'express'
import { itemsRouter } from './routes/items.js'
import { setupSwagger } from './swagger.js'
const app = express()
app.use(cors({ origin: 'http://localhost:3000' }))
app.use(express.json())
setupSwagger(app)
app.use('/api/items', itemsRouter)
const PORT = 1337
app.listen(PORT, () => {
// eslint-disable-next-line no-console
console.log(`Server running on http://localhost:${PORT}`)
// eslint-disable-next-line no-console
console.log(`Swagger UI: http://localhost:${PORT}/docs`)
})

View File

@@ -0,0 +1,5 @@
import type { NextFunction, Request, Response } from 'express'
export function batcherMiddleware(_req: Request, _res: Response, next: NextFunction): void {
next()
}

259
backend/src/routes/items.ts Normal file
View File

@@ -0,0 +1,259 @@
import { Router } from 'express'
import * as store from '../services/itemsStore.js'
import { queue } from '../services/queue.js'
export const itemsRouter = Router()
/**
* @openapi
* components:
* schemas:
* Item:
* type: object
* properties:
* id:
* type: integer
* PaginatedItems:
* type: object
* properties:
* data:
* type: array
* items:
* $ref: '#/components/schemas/Item'
* total:
* type: integer
* page:
* type: integer
* limit:
* type: integer
* hasMore:
* type: boolean
*/
/**
* @openapi
* /api/items:
* get:
* summary: Get unselected items
* tags: [Items]
* parameters:
* - in: query
* name: page
* schema:
* type: integer
* default: 1
* - in: query
* name: limit
* schema:
* type: integer
* default: 20
* maximum: 20
* - in: query
* name: search
* schema:
* type: string
* responses:
* 200:
* description: Paginated list of unselected items
* content:
* application/json:
* schema:
* $ref: '#/components/schemas/PaginatedItems'
*/
itemsRouter.get('/', (req, res) => {
const page = Math.max(1, Number(req.query.page) || 1)
const limit = Math.min(20, Math.max(1, Number(req.query.limit) || 20))
const search = req.query.search ? String(req.query.search) : undefined
res.json(store.getItems(page, limit, search))
})
/**
* @openapi
* /api/items/selected:
* get:
* summary: Get selected items in order
* tags: [Items]
* parameters:
* - in: query
* name: page
* schema:
* type: integer
* default: 1
* - in: query
* name: limit
* schema:
* type: integer
* default: 20
* maximum: 20
* - in: query
* name: search
* schema:
* type: string
* responses:
* 200:
* description: Paginated list of selected items
* content:
* application/json:
* schema:
* $ref: '#/components/schemas/PaginatedItems'
*/
itemsRouter.get('/selected', (req, res) => {
const page = Math.max(1, Number(req.query.page) || 1)
const limit = Math.min(20, Math.max(1, Number(req.query.limit) || 20))
const search = req.query.search ? String(req.query.search) : undefined
res.json(store.getSelectedItems(page, limit, search))
})
/**
* @openapi
* /api/items/add:
* post:
* summary: Queue an item for addition
* tags: [Items]
* requestBody:
* required: true
* content:
* application/json:
* schema:
* type: object
* required: [id]
* properties:
* id:
* type: integer
* responses:
* 200:
* description: Item queued
* content:
* application/json:
* schema:
* type: object
* properties:
* queued:
* type: boolean
* id:
* type: integer
* 400:
* description: Invalid id
*/
itemsRouter.post('/add', (req, res) => {
const id = req.body?.id
if (id === undefined || id === null || typeof id !== 'number' || !Number.isInteger(id)) {
res.status(400).json({ error: 'id must be an integer' })
return
}
queue.enqueue({ type: 'add', id })
res.json({ queued: true, id })
})
/**
* @openapi
* /api/items/select:
* post:
* summary: Queue an item for selection (move to right panel)
* tags: [Items]
* requestBody:
* required: true
* content:
* application/json:
* schema:
* type: object
* required: [id]
* properties:
* id:
* type: integer
* responses:
* 200:
* description: Item queued
* content:
* application/json:
* schema:
* type: object
* properties:
* queued:
* type: boolean
*/
itemsRouter.post('/select', (req, res) => {
const id = req.body?.id
if (id === undefined || id === null || typeof id !== 'number' || !Number.isInteger(id)) {
res.status(400).json({ error: 'id must be an integer' })
return
}
queue.enqueue({ type: 'select', id })
res.json({ queued: true })
})
/**
* @openapi
* /api/items/deselect:
* post:
* summary: Queue an item for deselection (move back to left panel)
* tags: [Items]
* requestBody:
* required: true
* content:
* application/json:
* schema:
* type: object
* required: [id]
* properties:
* id:
* type: integer
* responses:
* 200:
* description: Item queued
* content:
* application/json:
* schema:
* type: object
* properties:
* queued:
* type: boolean
*/
itemsRouter.post('/deselect', (req, res) => {
const id = req.body?.id
if (id === undefined || id === null || typeof id !== 'number' || !Number.isInteger(id)) {
res.status(400).json({ error: 'id must be an integer' })
return
}
queue.enqueue({ type: 'deselect', id })
res.json({ queued: true })
})
/**
* @openapi
* /api/items/reorder:
* put:
* summary: Queue a reorder of selected items
* tags: [Items]
* requestBody:
* required: true
* content:
* application/json:
* schema:
* type: object
* required: [ids]
* properties:
* ids:
* type: array
* items:
* type: integer
* responses:
* 200:
* description: Reorder queued
* content:
* application/json:
* schema:
* type: object
* properties:
* queued:
* type: boolean
*/
itemsRouter.put('/reorder', (req, res) => {
const ids = req.body?.ids
if (!Array.isArray(ids)) {
res.status(400).json({ error: 'ids must be an array' })
return
}
queue.enqueue({ type: 'reorder', ids })
res.json({ queued: true })
})

View File

@@ -0,0 +1,79 @@
export interface PaginatedResult {
data: { id: number }[]
total: number
page: number
limit: number
hasMore: boolean
}
const allItems: number[] = Array.from({ length: 1_000_000 }, (_, i) => i + 1)
const selectedIds = new Set<number>()
const orderedSelected: number[] = []
export function getItems(page: number, limit: number, search?: string): PaginatedResult {
let filtered: number[]
if (search) {
filtered = allItems.filter(id => !selectedIds.has(id) && String(id).includes(search))
}
else {
filtered = allItems.filter(id => !selectedIds.has(id))
}
const total = filtered.length
const start = (page - 1) * limit
const slice = filtered.slice(start, start + limit)
return {
data: slice.map(id => ({ id })),
total,
page,
limit,
hasMore: start + limit < total,
}
}
export function getSelectedItems(page: number, limit: number, search?: string): PaginatedResult {
let filtered: number[]
if (search) {
filtered = orderedSelected.filter(id => String(id).includes(search))
}
else {
filtered = [...orderedSelected]
}
const total = filtered.length
const start = (page - 1) * limit
const slice = filtered.slice(start, start + limit)
return {
data: slice.map(id => ({ id })),
total,
page,
limit,
hasMore: start + limit < total,
}
}
export function addItem(id: number): boolean {
if (allItems.includes(id) || selectedIds.has(id)) return false
allItems.push(id)
return true
}
export function selectItem(id: number): boolean {
if (selectedIds.has(id)) return false
selectedIds.add(id)
orderedSelected.push(id)
return true
}
export function deselectItem(id: number): boolean {
if (!selectedIds.has(id)) return false
selectedIds.delete(id)
const idx = orderedSelected.indexOf(id)
if (idx !== -1) orderedSelected.splice(idx, 1)
return true
}
export function reorderSelected(ids: number[]): void {
orderedSelected.length = 0
for (const id of ids) {
if (selectedIds.has(id)) orderedSelected.push(id)
}
}

View File

@@ -0,0 +1,65 @@
import * as store from './itemsStore.js'
interface AddTask { type: 'add', id: number }
interface SelectTask { type: 'select', id: number }
interface DeselectTask { type: 'deselect', id: number }
interface ReorderTask { type: 'reorder', ids: number[] }
type Task = AddTask | SelectTask | DeselectTask | ReorderTask
class RequestQueue {
private addQueue: AddTask[] = []
private actionQueue: ReorderTask[] = []
private pendingKeys = new Set<string>()
private pendingReorder: ReorderTask | null = null
constructor() {
setInterval(() => this.flushAdd(), 10_000)
setInterval(() => this.flushActions(), 1_000)
}
enqueue(task: Task): boolean {
if (task.type === 'reorder') {
this.pendingReorder = task
this.actionQueue = this.actionQueue.filter(t => t.type !== 'reorder')
this.actionQueue.push(task)
return true
}
const key = `${task.type}:${(task as AddTask | SelectTask | DeselectTask).id}`
if (this.pendingKeys.has(key)) return false
this.pendingKeys.add(key)
if (task.type === 'add') {
this.addQueue.push(task)
if (this.addQueue.length >= 100) this.flushAdd()
}
else {
// select/deselect применяются немедленно, чтобы GET после POST видел актуальный state
if (task.type === 'select') store.selectItem(task.id)
else if (task.type === 'deselect') store.deselectItem(task.id)
this.pendingKeys.delete(key)
}
return true
}
private flushAdd(): void {
const batch = this.addQueue.splice(0)
for (const task of batch) {
store.addItem(task.id)
this.pendingKeys.delete(`add:${task.id}`)
}
}
private flushActions(): void {
const batch = this.actionQueue.splice(0)
this.pendingReorder = null
for (const task of batch) {
if (task.type === 'reorder') {
store.reorderSelected(task.ids)
}
}
}
}
export const queue = new RequestQueue()

23
backend/src/swagger.ts Normal file
View File

@@ -0,0 +1,23 @@
import type { Express } from 'express'
import swaggerJsdoc from 'swagger-jsdoc'
import swaggerUi from 'swagger-ui-express'
const options: swaggerJsdoc.Options = {
definition: {
openapi: '3.0.0',
info: {
title: 'TMC Items API',
version: '1.0.0',
description: 'API for managing items with selection and ordering',
},
servers: [{ url: 'http://localhost:1337' }],
},
apis: ['./src/routes/*.ts'],
}
const spec = swaggerJsdoc(options)
export function setupSwagger(app: Express): void {
app.get('/docs/json', (_req, res) => res.json(spec))
app.use('/docs', swaggerUi.serve, swaggerUi.setup(spec))
}