Сделал первые функции под MRP

This commit is contained in:
2026-05-06 23:15:50 +03:00
parent 5417092796
commit 3b7a740081
27 changed files with 2996 additions and 767 deletions
+4 -3
View File
@@ -3,9 +3,11 @@ package database
import (
"log"
"os"
"viplight-mrp/models" // Замени viplight-mrp на имя своего модуля из go.mod
"gorm.io/driver/postgres"
"gorm.io/gorm"
"viplight-mrp/models"
)
var DB *gorm.DB
@@ -18,6 +20,5 @@ func InitDB() {
log.Fatal("Failed to connect to database:", err)
}
DB.AutoMigrate(&models.Part{})
DB.AutoMigrate(&models.Order{}, &models.Part{})
}
+21
View File
@@ -0,0 +1,21 @@
package handlers
import (
"net/http"
"github.com/gin-gonic/gin"
"viplight-mrp/database"
"viplight-mrp/models"
)
func GetsOrders(c *gin.Context) {
var orders []models.Order
if err := database.DB.Preload("Parts").Find(&orders).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Ошибка при получении заказов"})
return
}
c.JSON(http.StatusOK, orders)
}
+74 -42
View File
@@ -2,9 +2,11 @@ package handlers
import (
"net/http"
"github.com/gin-gonic/gin"
"viplight-mrp/database"
"viplight-mrp/models"
"github.com/gin-gonic/gin"
)
func GetPart(c *gin.Context) {
@@ -29,64 +31,94 @@ func CreatePart(c *gin.Context) {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
database.DB.Create(&newPart)
if newPart.OrderNo != "" {
var order models.Order
database.DB.Where(models.Order{OrderNo: newPart.OrderNo}).FirstOrCreate(&order)
}
if err := database.DB.Create(&newPart).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, newPart)
}
func UpdateStatus(c *gin.Context) {
id := c.Param("id")
var input struct { Status string `json:"status" binding:"required"` }
var input struct {
Status string `json:"status" binding:"required"`
}
if err := c.ShouldBindJSON(&input); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "Status is required"})
}
if err := c.ShouldBindJSON(&input); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": "Неверные данные"})
}
result := database.DB.Model(&models.Part{}).Where("id = ?", id).Update("status", input.Status)
if result.RowsAffected == 0{
c.JSON(http.StatusNotFound, gin.H{"error" : "Деталь не найдена"})
}
if err := database.DB.Model(&models.Part{}).Where("id = ?", id).Update("status", input.Status).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Не удалось обновить статус"})
return
}
c.JSON(http.StatusOK, gin.H{"status": "updated", "new_status": input.Status})
}
func DeletePart(c *gin.Context){
id := c.Param("id")
if err := database.DB.Delete(&models.Part{}, "id = ?", id).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Не удалось удалить элемент"})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Удалено"})
func DeletePart(c *gin.Context) {
id := c.Param("id")
if err := database.DB.Delete(&models.Part{}, "id = ?", id).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Удалено"})
}
func CreatePartsBulk(c *gin.Context){
var parts []models.Part
if err := c.ShouldBindJSON(&parts); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
func CreatePartsBulk(c *gin.Context) {
var parts []models.Part
if err := c.ShouldBindJSON(&parts); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if err := database.DB.Create(&parts).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error" : "Не могу создать деталь"})
return
}
ordersMap := make(map[string]bool)
for _, p := range parts {
if p.OrderNo != "" {
ordersMap[p.OrderNo] = true
}
}
c.JSON(http.StatusCreated, parts)
for orderNo := range ordersMap {
var order models.Order
database.DB.Where(models.Order{OrderNo: orderNo}).FirstOrCreate(&order)
}
if err := database.DB.Create(&parts).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
return
}
c.JSON(http.StatusCreated, parts)
}
func ImportParts(c *gin.Context) {
var parts []models.Part
if err := c.ShouldBindJSON(&parts); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
var parts []models.Part
// GORM сделает один быстрый INSERT для всех деталей
if err := database.DB.Create(&parts).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Ошибка импорта"})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Импортировано", "count": len(parts)})
if err := c.ShouldBindJSON(&parts); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}
if len(parts) > 0 {
orderNo := parts[0].OrderNo
var order models.Order
if err := database.DB.Where(models.Order{OrderNo: orderNo}).FirstOrCreate(&order).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Ошибка создания заголовка заказа"})
return
}
}
if err := database.DB.Create(&parts).Error; err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": "Ошибка при сохранении заказа"})
return
}
c.JSON(http.StatusOK, gin.H{"message": "Данные успешно импортированы", "count": len(parts), "order_no": parts[0].OrderNo})
}
+3 -3
View File
@@ -1,9 +1,10 @@
package main
import (
"github.com/gin-gonic/gin"
"viplight-mrp/database"
"viplight-mrp/handlers"
"github.com/gin-gonic/gin"
)
func main() {
@@ -26,12 +27,11 @@ func main() {
// Роуты теперь вызывают функции из handlers
r.GET("/api/parts/:id", handlers.GetPart)
r.GET("/api/parts", handlers.GetAllParts)
r.GET("/api/orders", handlers.GetsOrders)
r.POST("/api/parts", handlers.CreatePart)
r.POST("/api/parts/bulk", handlers.ImportParts)
r.PATCH("/api/parts/:id/status", handlers.UpdateStatus)
r.DELETE("/api/parts/:id", handlers.DeletePart)
r.Run(":8090")
}
+13
View File
@@ -0,0 +1,13 @@
package models
import "gorm.io/gorm"
type Order struct {
gorm.Model
OrderNo string `gorm:"uniqueIndex;not null" json:"order_no"`
ClientName string `json:"client_name"`
Deadline string `json:"deadline"`
Status string `gorm:"default:'Новый'" json:"status"`
Parts []Part `gorm:"foreignKey:OrderNo;references:OrderNo" json:"parts"`
}
+56 -58
View File
@@ -1,68 +1,66 @@
package models
const (
StatusImport = "Запланировано"
StatusWarehouse = "Склад"
StatusSawing = "Пила"
StatusEdging = "Кромка"
StatusEdgingHand = "Кромка ручная"
StatusCNCDelta = "ЧПУ Дельта"
StatusCNCTrepan = "ЧПУ Трепан"
StatusHandDrilling = "Присадка на ручном станке"
StatusSawHand = "Циркулярная пила"
StatusAssembly = "Сборка"
StatusAbrasive = "Шлифовка"
StatusDrying = "Сушка после малярки"
StatusPainting = "Малярная камера"
StatusGlass = "Стекольный цех"
StatusMetalSelf = "Сварочный цех"
StatusPaintingOutsource = "Порошковая покраска"
StatusAdv = "Рекламный участок"
StatusLight = "Светодиодный участок"
StatusSemiFinished = "Упаковка полуфабриката"
StatusFinished = "Упаковка изделия"
StatusShipment = "Отгружено на монтаж"
StatusInstallation = "Монтаж оборудования"
StatusRejected = "Отбраковано/сломано"
StatusImport = "Запланировано"
StatusWarehouse = "Склад"
StatusSawing = "Пила"
StatusEdging = "Кромка"
StatusEdgingHand = "Кромка ручная"
StatusCNCDelta = "ЧПУ Дельта"
StatusCNCTrepan = "ЧПУ Трепан"
StatusHandDrilling = "Присадка на ручном станке"
StatusSawHand = "Циркулярная пила"
StatusAssembly = "Сборка"
StatusAbrasive = "Шлифовка"
StatusDrying = "Сушка после малярки"
StatusPainting = "Малярная камера"
StatusGlass = "Стекольный цех"
StatusMetalSelf = "Сварочный цех"
StatusPaintingOutsource = "Порошковая покраска"
StatusAdv = "Рекламный участок"
StatusLight = "Светодиодный участок"
StatusSemiFinished = "Упаковка полуфабриката"
StatusFinished = "Упаковка изделия"
StatusShipment = "Отгружено на монтаж"
StatusInstallation = "Монтаж оборудования"
StatusRejected = "Отбраковано/сломано"
)
func IsValidStatus(s string) bool {
switch s {
case StatusWarehouse, StatusSawing, StatusEdging, StatusEdgingHand,
StatusCNCDelta, StatusCNCTrepan, StatusHandDrilling, StatusSawHand,
StatusAssembly, StatusAbrasive, StatusDrying, StatusPainting,
StatusGlass, StatusMetalSelf, StatusPaintingOutsource, StatusAdv,
StatusLight, StatusSemiFinished, StatusFinished, StatusShipment,
StatusInstallation, StatusRejected:
return true
}
return false
switch s {
case StatusWarehouse, StatusSawing, StatusEdging, StatusEdgingHand,
StatusCNCDelta, StatusCNCTrepan, StatusHandDrilling, StatusSawHand,
StatusAssembly, StatusAbrasive, StatusDrying, StatusPainting,
StatusGlass, StatusMetalSelf, StatusPaintingOutsource, StatusAdv,
StatusLight, StatusSemiFinished, StatusFinished, StatusShipment,
StatusInstallation, StatusRejected:
return true
}
return false
}
type Part struct {
ID string `gorm:"primaryKey" json:"id" binding:"required"` // № или Обозначение
Destignation string `json:"destignation" binding:"required"` // № или Обозначение
OrderNo string `json:"order_no" binding:"required"` // Заказ изделия
Name string `json:"name" binding:"required"` // Наименование детали
Material string `json:"material" binding:"required"` // Наименование материала
Thickness float64 `json:"thickness" binding:"required"` // Толщина
Quantity int `json:"quantity" binding:"required"` // Количество
// Размеры (используем Готовую деталь как основной стандарт)
Length float64 `json:"length" binding:"required"` // Готовая деталь [L]
Width float64 `json:"width" binding:"required"` // Готовая деталь [W]
// Кромка
EdgeL1 string `json:"edge_l1"` // Обозначение кромки [L1]
EdgeL2 string `json:"edge_l2"` // Обозначение кромки [L2]
EdgeW1 string `json:"edge_w1"` // Обозначение кромки [W1]
EdgeW2 string `json:"edge_w2"` // Обозначение кромки [W2]
// Доп. инфо
Groove string `json:"groove"` // Паз
Note string `json:"note"` // Примечание
ProductName string `json:"product_name" binding:"required"` // Наимен. изделия
Status string `json:"status" gorm:"default:Создан"`
}
ID uint `gorm:"primaryKey;autoIncrement" json:"id"`
Designation string `json:"designation" binding:"required"`
OrderNo string `json:"order_no" binding:"required"`
Name string `json:"name" binding:"required"`
Material string `json:"material" binding:"required"`
Thickness float64 `json:"thickness" binding:"required"`
Quantity int `json:"quantity" binding:"required"`
Length float64 `json:"length" binding:"required"`
Width float64 `json:"width" binding:"required"`
LengthFirst float64 `json:"length_first" binding:"required"`
WidthFirst float64 `json:"width_first" binding:"required"`
EdgeL1 string `json:"edge_l1"`
EdgeL2 string `json:"edge_l2"`
EdgeW1 string `json:"edge_w1"`
EdgeW2 string `json:"edge_w2"`
Groove string `json:"groove"`
Note string `json:"note"`
ProductName string `json:"product_name" binding:"required"`
Status string `json:"status" gorm:"default:Создан"`
}