42c74a725a
- migracija 048: kolona uvoz na pdv_kpr (0=domaća nabavka, 1=uvoz) - model PdvKpr.Uvoz; MapirajPPPDV(kir, kprDomace, kprUvoz) rutira uvoz u 006/106, domaće u 008/108; test ažuriran + uvozni scenario - repo: KPR Lista/DohvatiID/Kreiraj čitaju i pišu uvoz - obračun: KPR se razdvaja na domaće/uvozne; obaveza ostaje na ukupnom KPR-u - KPR forma: kvačica „Uvoz (JCI)"; lista: oznaka UVOZ uz broj dokumenta
162 lines
6.4 KiB
Go
162 lines
6.4 KiB
Go
package model
|
||
|
||
import (
|
||
"math"
|
||
"testing"
|
||
"time"
|
||
)
|
||
|
||
func blizu(a, b float64) bool { return math.Abs(a-b) < 0.01 }
|
||
|
||
func TestKirIzProdaje(t *testing.T) {
|
||
nalog := ProdajniNalog{ID: 5, BrojNaloga: "P-1", Datum: time.Date(2026, 6, 1, 0, 0, 0, 0, time.UTC)}
|
||
stavke := []StavkaProdaje{
|
||
// 20%: 2 × 120 = 240 (osnovica 200, PDV 40)
|
||
{Kolicina: 2, CenaPoKomadu: 120, PdvStopa: 20},
|
||
// 10%: 1 × 110 = 110 (osnovica 100, PDV 10)
|
||
{Kolicina: 1, CenaPoKomadu: 110, PdvStopa: 10},
|
||
// 0%: 1 × 50 = 50 (oslobođeno, bez PDV)
|
||
{Kolicina: 1, CenaPoKomadu: 50, PdvStopa: 0},
|
||
}
|
||
|
||
k := KirIzProdaje(nalog, stavke, "Kupac doo", "123456789", "Niš")
|
||
|
||
if k.Izvor != "prodaja" || k.IzvorID == nil || *k.IzvorID != 5 {
|
||
t.Errorf("izvor=%q izvor_id=%v, očekivano prodaja/5", k.Izvor, k.IzvorID)
|
||
}
|
||
if k.BrojDokumenta != "P-1" || k.KupacNaziv != "Kupac doo" || k.KupacPib != "123456789" {
|
||
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.OslobodenSaPravom, 50) {
|
||
t.Errorf("oslobođeno=%v, očekivano 50", k.OslobodenSaPravom)
|
||
}
|
||
if !blizu(k.Ukupno, 400) {
|
||
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)
|
||
}
|
||
}
|
||
|
||
func TestObracunajPdv(t *testing.T) {
|
||
// izlazni: 200 (opšta) + 50 (posebna) = 250; odbitni: 80 + 20 = 100; obaveza = 150
|
||
kir := PdvKirSume{PdvOpsta: 200, PdvPosebna: 50}
|
||
kpr := PdvKprSume{PdvOpsta: 80, PdvPosebna: 20, PdvBezOdbitka: 30}
|
||
|
||
o := ObracunajPdv(kir, kpr)
|
||
|
||
if !blizu(o.IzlazniPdvUkupno, 250) {
|
||
t.Errorf("izlazni=%v, očekivano 250", o.IzlazniPdvUkupno)
|
||
}
|
||
// PdvBezOdbitka (30) ne sme da uđe u odbitni PDV
|
||
if !blizu(o.OdbitniPdvUkupno, 100) {
|
||
t.Errorf("odbitni=%v, očekivano 100 (bez PdvBezOdbitka)", o.OdbitniPdvUkupno)
|
||
}
|
||
if !blizu(o.Obaveza, 150) || !o.ZaUplatu() {
|
||
t.Errorf("obaveza=%v ZaUplatu=%v, očekivano 150/true", o.Obaveza, o.ZaUplatu())
|
||
}
|
||
|
||
// kada je odbitni veći od izlaznog → negativna obaveza (povraćaj/prenos), ZaUplatu=false
|
||
o2 := ObracunajPdv(PdvKirSume{PdvOpsta: 40}, PdvKprSume{PdvOpsta: 100})
|
||
if !blizu(o2.Obaveza, -60) || o2.ZaUplatu() {
|
||
t.Errorf("obaveza=%v ZaUplatu=%v, očekivano -60/false", o2.Obaveza, o2.ZaUplatu())
|
||
}
|
||
}
|
||
|
||
func TestMapirajPPPDV(t *testing.T) {
|
||
kir := PdvKirSume{
|
||
OsnovicaOpsta: 1000.4, PdvOpsta: 200.08, // 003=1000, 103=200
|
||
OsnovicaPosebna: 500.5, PdvPosebna: 50.05, // 004=501 (round), 104=50
|
||
OslobodenSaPravom: 300.2, OslobodenBezPrava: 100.9, // 001=300, 002=101
|
||
}
|
||
kpr := PdvKprSume{
|
||
OsnovicaOpsta: 400, PdvOpsta: 80,
|
||
OsnovicaPosebna: 100, PdvPosebna: 10,
|
||
PdvBezOdbitka: 25, // ne sme da uđe nigde u PPPDV
|
||
}
|
||
|
||
p := MapirajPPPDV(kir, kpr, PdvKprSume{})
|
||
|
||
if p.Polje001 != 300 || p.Polje002 != 101 || p.Polje003 != 1000 || p.Polje103 != 200 {
|
||
t.Errorf("I deo (001/002/003/103) = %d/%d/%d/%d", p.Polje001, p.Polje002, p.Polje003, p.Polje103)
|
||
}
|
||
if p.Polje004 != 501 || p.Polje104 != 50 {
|
||
t.Errorf("posebna 004/104 = %d/%d, očekivano 501/50", p.Polje004, p.Polje104)
|
||
}
|
||
// 005 = 300+101+1000+501 = 1902; 105 = 200+50 = 250
|
||
if p.Polje005 != 1902 || p.Polje105 != 250 {
|
||
t.Errorf("zbir 005/105 = %d/%d, očekivano 1902/250", p.Polje005, p.Polje105)
|
||
}
|
||
// u ovom scenariju nema uvoza (prazan kprUvoz) ni poljoprivrednika → 006/106/007/107 = 0
|
||
if p.Polje006 != 0 || p.Polje106 != 0 || p.Polje007 != 0 || p.Polje107 != 0 {
|
||
t.Errorf("006/106/007/107 moraju biti 0")
|
||
}
|
||
// 008 = 400+100 = 500; 108 = 80+10 = 90 (PdvBezOdbitka 25 izostavljen)
|
||
if p.Polje008 != 500 || p.Polje108 != 90 {
|
||
t.Errorf("ostali prethodni 008/108 = %d/%d, očekivano 500/90", p.Polje008, p.Polje108)
|
||
}
|
||
if p.Polje009 != 500 || p.Polje109 != 90 {
|
||
t.Errorf("zbir prethodnog 009/109 = %d/%d, očekivano 500/90", p.Polje009, p.Polje109)
|
||
}
|
||
// 110 = 250 − 90 = 160 (za uplatu)
|
||
if p.Polje110 != 160 || p.Povracaj {
|
||
t.Errorf("110=%d Povracaj=%v, očekivano 160/false", p.Polje110, p.Povracaj)
|
||
}
|
||
|
||
// scenario povraćaja: izlazni < odbitni → 110 negativan, Povracaj=true
|
||
p2 := MapirajPPPDV(PdvKirSume{PdvOpsta: 30}, PdvKprSume{PdvOpsta: 90}, PdvKprSume{})
|
||
if p2.Polje110 != -60 || !p2.Povracaj || p2.Polje110Apsolutno() != 60 {
|
||
t.Errorf("povraćaj: 110=%d Povracaj=%v abs=%d, očekivano -60/true/60", p2.Polje110, p2.Povracaj, p2.Polje110Apsolutno())
|
||
}
|
||
|
||
// scenario uvoza: uvozni KPR ide u 006/106, domaći u 008/108
|
||
p3 := MapirajPPPDV(
|
||
PdvKirSume{},
|
||
PdvKprSume{OsnovicaOpsta: 1000, PdvOpsta: 200}, // domaće → 008/108
|
||
PdvKprSume{OsnovicaOpsta: 500, PdvOpsta: 100, OsnovicaPosebna: 50, PdvPosebna: 5}, // uvoz → 006/106
|
||
)
|
||
if p3.Polje006 != 550 || p3.Polje106 != 105 {
|
||
t.Errorf("uvoz 006/106 = %d/%d, očekivano 550/105", p3.Polje006, p3.Polje106)
|
||
}
|
||
if p3.Polje008 != 1000 || p3.Polje108 != 200 {
|
||
t.Errorf("domaće 008/108 = %d/%d, očekivano 1000/200", p3.Polje008, p3.Polje108)
|
||
}
|
||
// 009 = 006+007+008 = 550+0+1000 = 1550; 109 = 106+107+108 = 105+0+200 = 305
|
||
if p3.Polje009 != 1550 || p3.Polje109 != 305 {
|
||
t.Errorf("zbir prethodnog 009/109 = %d/%d, očekivano 1550/305", p3.Polje009, p3.Polje109)
|
||
}
|
||
}
|