feat(pdv): nabavka → KPR automatski (Faza 2b-2)

PDV se izvodi iz stope artikla po stavci (aproksimacija: nabavna cena
= osnovica bez PDV). Grupisanje po stopi (20→opšta, 10→posebna,
ostalo→oslobođena nabavka), broj dokumenta NAB-<id>, veza izvor/izvor_id.
Auto-zapisi se ne mogu ručno brisati u KPR; brisanje nabavke uklanja
vezani KPR zapis.
This commit is contained in:
2026-06-14 08:16:41 +02:00
parent 76b147074d
commit 7fc2e9bcc3
4 changed files with 110 additions and 2 deletions
+44 -1
View File
@@ -1,6 +1,9 @@
package model
import "time"
import (
"fmt"
"time"
)
// PdvKir je jedan zapis u knjizi izdatih računa (izlazni PDV).
// Iznosi se vode po vrsti stope (opšta/posebna) — vidi migraciju 041.
@@ -176,3 +179,43 @@ func SumirajKpr(zapisi []PdvKpr) PdvKprSume {
}
return s
}
// NabavkaStavkaPdv je jedna stavka nabavke sa pripadajućom PDV stopom (iz artikla).
type NabavkaStavkaPdv struct {
Osnovica float64 // nabavna vrednost stavke (cena × količina) — tretira se kao osnovica bez PDV
PdvStopa float64
}
// KprIzNabavke gradi KPR zapis iz nabavke. PDV se izvodi iz stope artikla po stavci
// (⚠ aproksimacija: nabavna cena = osnovica bez PDV; stvaran PDV sa računa dobavljača može
// se razlikovati). Grupiše po stopi (20→opšta, 10→posebna, ostalo→oslobođena nabavka).
// Broj dokumenta je sintetički ("NAB-<id>") jer nabavka ne čuva broj računa dobavljača.
func KprIzNabavke(nabavka Nabavka, dobavljacNaziv, dobavljacPib, dobavljacMesto string, stavke []NabavkaStavkaPdv) PdvKpr {
id := nabavka.ID
k := PdvKpr{
DatumPrometa: nabavka.Datum,
DatumKnjizenja: nabavka.Datum,
BrojDokumenta: fmt.Sprintf("NAB-%d", id),
DobavljacNaziv: dobavljacNaziv,
DobavljacPib: dobavljacPib,
DobavljacMesto: dobavljacMesto,
Napomena: nabavka.Napomena,
Izvor: "nabavka",
IzvorID: &id,
}
for _, s := range stavke {
pdv := s.Osnovica * s.PdvStopa / 100
switch s.PdvStopa {
case 20:
k.OsnovicaOpsta += s.Osnovica
k.PdvOpsta += pdv
case 10:
k.OsnovicaPosebna += s.Osnovica
k.PdvPosebna += pdv
default:
k.OslobodenNabavka += s.Osnovica
}
k.Ukupno += s.Osnovica + pdv
}
return k
}
+30
View File
@@ -40,3 +40,33 @@ func TestKirIzProdaje(t *testing.T) {
t.Errorf("ukupno=%v, očekivano 400 (240+110+50)", k.Ukupno)
}
}
func TestKprIzNabavke(t *testing.T) {
nabavka := Nabavka{ID: 3, Napomena: "test", Datum: time.Date(2026, 6, 2, 0, 0, 0, 0, time.UTC)}
stavke := []NabavkaStavkaPdv{
{Osnovica: 200, PdvStopa: 20}, // PDV 40
{Osnovica: 100, PdvStopa: 10}, // PDV 10
{Osnovica: 50, PdvStopa: 0}, // oslobođena nabavka
}
k := KprIzNabavke(nabavka, "Dobavljač doo", "987654321", "Beograd", stavke)
if k.Izvor != "nabavka" || k.IzvorID == nil || *k.IzvorID != 3 {
t.Errorf("izvor=%q izvor_id=%v, očekivano nabavka/3", k.Izvor, k.IzvorID)
}
if k.BrojDokumenta != "NAB-3" || k.DobavljacNaziv != "Dobavljač doo" || k.DobavljacPib != "987654321" {
t.Errorf("zaglavlje ne odgovara: %+v", k)
}
if !blizu(k.OsnovicaOpsta, 200) || !blizu(k.PdvOpsta, 40) {
t.Errorf("opšta: osnovica=%v pdv=%v, očekivano 200/40", k.OsnovicaOpsta, k.PdvOpsta)
}
if !blizu(k.OsnovicaPosebna, 100) || !blizu(k.PdvPosebna, 10) {
t.Errorf("posebna: osnovica=%v pdv=%v, očekivano 100/10", k.OsnovicaPosebna, k.PdvPosebna)
}
if !blizu(k.OslobodenNabavka, 50) {
t.Errorf("oslobođena nabavka=%v, očekivano 50", k.OslobodenNabavka)
}
if !blizu(k.Ukupno, 400) {
t.Errorf("ukupno=%v, očekivano 400 (240+110+50)", k.Ukupno)
}
}