Сделал первые функции под MRP
This commit is contained in:
@@ -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{})
|
||||
}
|
||||
|
||||
|
||||
@@ -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)
|
||||
}
|
||||
@@ -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
@@ -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")
|
||||
}
|
||||
|
||||
|
||||
@@ -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
@@ -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:Создан"`
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user