feat(kalkulacija): Faza C — marža po kategoriji/artiklu + zavisni troškovi (backend)

Celina 1 (kompletna) — marža po kategoriji/artiklu:
- migracija 046: nullable marza na artikli i kategorije
- model Marza *float64 (Artikal, Kategorija) + KategorijaMarza u ArtikalSaKategorijom
- repo: čitanje/pisanje marže; nove DohvatiID/Izmeni za kategoriju
- dozvola kategorija.izmeni; handler IzmeniKategoriju + ruta
- UI: polje marže u formi artikla i kategorije; modal izmene kategorije
- nabavka: fallback predlog marže artikal → kategorija → globalna (izaberiArtikal)

Celina 2 (backend) — zavisni troškovi nabavke:
- migracija 047: tabela nabavka_troskovi + kolona metod_raspodele na nabavke
- model NabavkaTrosak, MetodRaspodele; čista funkcija RasporediTroskove + test
- repo: Kreiraj upisuje troškove i metod; DohvatiTroskove
- handler: parsiranje troškova/metoda; kalkulativna nabavna cena na serveru

UI forme troškova i prikaz u detaljima nabavke slede.
This commit is contained in:
2026-06-14 16:12:03 +02:00
parent cb1a3b21c3
commit 30db396ee6
18 changed files with 448 additions and 51 deletions
+74
View File
@@ -0,0 +1,74 @@
package model
import "testing"
// proverava raspodelu zavisnih troškova na stavke nabavke po oba metoda
func TestRasporediTroskove(t *testing.T) {
// dve stavke: A = 10×100 (vrednost 1000), B = 10×200 (vrednost 2000)
stavke := []StavkaNabavke{
{Kolicina: 10, CenaPoKomadu: 100},
{Kolicina: 10, CenaPoKomadu: 200},
}
slucajevi := []struct {
naziv string
stavke []StavkaNabavke
trosak float64
metod string
ocekuje []float64
}{
{
naziv: "bez troška — cena nepromenjena",
stavke: stavke,
trosak: 0,
metod: "vrednost",
ocekuje: []float64{100, 200},
},
{
// trošak 300 po vrednosti: A nosi 1/3 (100 → 10/kom), B nosi 2/3 (200 → 20/kom)
naziv: "po vrednosti",
stavke: stavke,
trosak: 300,
metod: "vrednost",
ocekuje: []float64{110, 220},
},
{
// trošak 300 po količini: svaka stavka 1/2 (150 → 15/kom)
naziv: "po količini",
stavke: stavke,
trosak: 300,
metod: "kolicina",
ocekuje: []float64{115, 215},
},
{
// osnovica po vrednosti je 0 (sve cene 0) — nema raspodele
naziv: "nulta osnovica vrednosti",
stavke: []StavkaNabavke{{Kolicina: 5, CenaPoKomadu: 0}},
trosak: 100,
metod: "vrednost",
ocekuje: []float64{0},
},
{
// stavka bez količine se preskače (bez deljenja nulom)
naziv: "količina nula",
stavke: []StavkaNabavke{{Kolicina: 0, CenaPoKomadu: 50}},
trosak: 100,
metod: "kolicina",
ocekuje: []float64{50},
},
}
for _, s := range slucajevi {
t.Run(s.naziv, func(t *testing.T) {
dobijeno := RasporediTroskove(s.stavke, s.trosak, s.metod)
if len(dobijeno) != len(s.ocekuje) {
t.Fatalf("dužina = %d, očekivano %d", len(dobijeno), len(s.ocekuje))
}
for i := range s.ocekuje {
if dobijeno[i] != s.ocekuje[i] {
t.Errorf("stavka %d: dobijeno %.2f, očekivano %.2f", i, dobijeno[i], s.ocekuje[i])
}
}
})
}
}