276 lines
14 KiB
Go
276 lines
14 KiB
Go
package db
|
||
|
||
import (
|
||
"context"
|
||
"errors"
|
||
"time"
|
||
|
||
"ntech/internal/model"
|
||
)
|
||
|
||
// ErrArtikalUUpotrebi se vraća kad se artikal ne može obrisati jer postoji u prometu
|
||
// (prodaja, nabavka, magacinske promene ili servisni nalozi). Tada se artikal arhivira.
|
||
var ErrArtikalUUpotrebi = errors.New("ntech: artikal je u upotrebi")
|
||
|
||
// ArtikalRepository definiše operacije nad artiklima
|
||
type ArtikalRepository interface {
|
||
Lista(ctx context.Context, filter ArtikalFilter) ([]model.ArtikalSaKategorijom, error)
|
||
PrebrojiPoFilteru(ctx context.Context, filter ArtikalFilter) (int, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.Artikal, error)
|
||
Kreiraj(ctx context.Context, a *model.Artikal) (int64, error)
|
||
Izmeni(ctx context.Context, a *model.Artikal) error
|
||
// AzurirajCene menja samo nabavnu i prodajnu cenu (kalkulacija pri nabavci)
|
||
AzurirajCene(ctx context.Context, id int64, nabavna, prodajna float64) error
|
||
PremestiKategoriju(ctx context.Context, id int64, kategorijaID *int64) error
|
||
Obrisi(ctx context.Context, id int64) error
|
||
// Arhiviraj označava artikal kao arhiviran (skriva ga iz aktivne liste, čuva istoriju)
|
||
Arhiviraj(ctx context.Context, id int64) error
|
||
// Vrati poništava arhiviranje i vraća artikal u aktivnu listu
|
||
Vrati(ctx context.Context, id int64) error
|
||
// SledecaSifra vraća predlog sledeće auto-šifre (npr. KOMP-0042 ili ART-0042)
|
||
SledecaSifra(ctx context.Context, kategorijaID *int64) (string, error)
|
||
// KorigujKolicinu postavlja novu količinu artikla i upisuje korekciju u magacinske_promene
|
||
KorigujKolicinu(ctx context.Context, artikalID int64, novaKolicina int, korisnikID *int64, napomena string) error
|
||
// DobavljaciArtikla vraća ID-jeve dobavljača vezanih za artikal
|
||
DobavljaciArtikla(ctx context.Context, artikalID int64) ([]int64, error)
|
||
// PostaviDobavljaceArtikla zamenjuje skup dobavljača artikla datim ID-jevima
|
||
PostaviDobavljaceArtikla(ctx context.Context, artikalID int64, dobavljaciID []int64) error
|
||
// PoveziDobavljaca dodaje vezu artikal–dobavljač ako ne postoji (auto pri nabavci)
|
||
PoveziDobavljaca(ctx context.Context, artikalID, dobavljacID int64) error
|
||
// SveDobavljaceArtikala vraća mapu artikal_id → lista dobavljac_id (za filter u nabavci)
|
||
SveDobavljaceArtikala(ctx context.Context) (map[int64][]int64, error)
|
||
}
|
||
|
||
// KategorijaRepository definiše operacije nad kategorijama
|
||
type KategorijaRepository interface {
|
||
Lista(ctx context.Context) ([]model.Kategorija, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.Kategorija, error)
|
||
Kreiraj(ctx context.Context, k *model.Kategorija) (int64, error)
|
||
Izmeni(ctx context.Context, k *model.Kategorija) error
|
||
}
|
||
|
||
// PdvStopaRepository definiše operacije nad šifarnikom PDV stopa
|
||
type PdvStopaRepository interface {
|
||
Lista(ctx context.Context, samoAktivne bool) ([]model.PdvStopa, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.PdvStopa, error)
|
||
Kreiraj(ctx context.Context, s *model.PdvStopa) (int64, error)
|
||
Izmeni(ctx context.Context, s *model.PdvStopa) error
|
||
PostaviAktivnu(ctx context.Context, id int64, aktivna bool) error
|
||
}
|
||
|
||
// PdvKirRepository definiše operacije nad knjigom izdatih računa (KIR)
|
||
type PdvKirRepository interface {
|
||
Lista(ctx context.Context, od, do time.Time) ([]model.PdvKir, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.PdvKir, error)
|
||
Kreiraj(ctx context.Context, k *model.PdvKir) (int64, error)
|
||
Obrisi(ctx context.Context, id int64) error
|
||
// ObrisiPoIzvoru briše zapise vezane za dati izvor (npr. pri stornu prodaje)
|
||
ObrisiPoIzvoru(ctx context.Context, izvor string, izvorID int64) error
|
||
}
|
||
|
||
// PdvKprRepository definiše operacije nad knjigom primljenih računa (KPR)
|
||
type PdvKprRepository interface {
|
||
Lista(ctx context.Context, od, do time.Time) ([]model.PdvKpr, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.PdvKpr, error)
|
||
Kreiraj(ctx context.Context, k *model.PdvKpr) (int64, error)
|
||
Obrisi(ctx context.Context, id int64) error
|
||
// ObrisiPoIzvoru briše zapise vezane za dati izvor (npr. pri brisanju nabavke)
|
||
ObrisiPoIzvoru(ctx context.Context, izvor string, izvorID int64) error
|
||
}
|
||
|
||
// NivelacijaRepository definiše operacije nad evidencijom promene prodajnih cena
|
||
type NivelacijaRepository interface {
|
||
// PromeniCenu transakciono menja prodajnu cenu artikla i upisuje nivelacioni zapis;
|
||
// vraća kreirani zapis (sa starom i novom cenom). Izvor je "rucno".
|
||
PromeniCenu(ctx context.Context, artikalID int64, novaCena float64, razlog string, korisnikID *int64) (*model.Nivelacija, error)
|
||
// Kreiraj upisuje gotov nivelacioni zapis (npr. auto-trag pri izmeni artikla)
|
||
Kreiraj(ctx context.Context, n *model.Nivelacija) (int64, error)
|
||
// Lista vraća nivelacije u periodu (po datumu); nulti datum znači bez granice
|
||
Lista(ctx context.Context, od, do time.Time) ([]model.Nivelacija, error)
|
||
// ListaZaArtikal vraća sve nivelacije jednog artikla (najnovije prvo)
|
||
ListaZaArtikal(ctx context.Context, artikalID int64) ([]model.Nivelacija, error)
|
||
}
|
||
|
||
// ArtikalFilter definiše parametre za filtriranje liste artikala
|
||
type ArtikalFilter struct {
|
||
Pretraga string
|
||
KategorijaID *int64
|
||
SamoKriticni bool
|
||
Arhivirani bool // true → vrati samo arhivirane; false (podrazumevano) → samo aktivne
|
||
Limit int
|
||
Offset int
|
||
}
|
||
|
||
// NabavkaRepository definiše operacije nad nabavkama
|
||
type NabavkaRepository interface {
|
||
Lista(ctx context.Context) ([]model.NabavkaSaDetaljem, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.Nabavka, error)
|
||
DohvatiStavke(ctx context.Context, nabavkaID int64) ([]model.StavkaSaArtiklom, error)
|
||
DohvatiTroskove(ctx context.Context, nabavkaID int64) ([]model.NabavkaTrosak, error)
|
||
Kreiraj(ctx context.Context, n *model.Nabavka, stavke []model.StavkaNabavke, troskovi []model.NabavkaTrosak) (int64, error)
|
||
Obrisi(ctx context.Context, id int64) error
|
||
}
|
||
|
||
// DobavljacRepository definiše operacije nad dobavljačima
|
||
type DobavljacRepository interface {
|
||
Lista(ctx context.Context, pretraga string) ([]model.Dobavljac, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.Dobavljac, error)
|
||
Kreiraj(ctx context.Context, d *model.Dobavljac) (int64, error)
|
||
Izmeni(ctx context.Context, d *model.Dobavljac) error
|
||
Obrisi(ctx context.Context, id int64) error
|
||
}
|
||
|
||
// KlijentFilter definiše parametre za filtriranje liste klijenata
|
||
type KlijentFilter struct {
|
||
Pretraga string
|
||
Tip string // "fizicko", "pravno" ili "" za sve
|
||
Limit int
|
||
Offset int
|
||
}
|
||
|
||
// KlijentRepository definiše operacije nad klijentima
|
||
type KlijentRepository interface {
|
||
Lista(ctx context.Context, pretraga string) ([]model.Klijent, error)
|
||
ListaFilter(ctx context.Context, filter KlijentFilter) ([]model.Klijent, error)
|
||
PrebrojiPoFilteru(ctx context.Context, filter KlijentFilter) (int, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.Klijent, error)
|
||
Kreiraj(ctx context.Context, k *model.Klijent) (int64, error)
|
||
Izmeni(ctx context.Context, k *model.Klijent) error
|
||
Obrisi(ctx context.Context, id int64) error
|
||
}
|
||
|
||
// ServisRepository definiše operacije nad servisnim nalozima
|
||
type ServisRepository interface {
|
||
Lista(ctx context.Context, pretraga, status string) ([]model.ServisniNalogSaKlijentom, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.ServisniNalog, error)
|
||
DohvatiJavniToken(ctx context.Context, token string) (*model.ServisniNalog, error)
|
||
Kreiraj(ctx context.Context, n *model.ServisniNalog) (int64, error)
|
||
Izmeni(ctx context.Context, n *model.ServisniNalog) error
|
||
AzurirajStatus(ctx context.Context, id int64, status string) error
|
||
Obrisi(ctx context.Context, id int64) error
|
||
SledeciBroj(ctx context.Context) (string, error)
|
||
}
|
||
|
||
// ProdajaRepository definiše operacije nad prodajnim nalozima
|
||
type ProdajaRepository interface {
|
||
Lista(ctx context.Context, pretraga string) ([]model.ProdajniNalogSaDetaljem, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.ProdajniNalog, error)
|
||
DohvatiStavke(ctx context.Context, nalogID int64) ([]model.StavkaProdajeSaArtiklom, error)
|
||
Kreiraj(ctx context.Context, n *model.ProdajniNalog, stavke []model.StavkaProdaje, korisnikID *int64) (int64, error)
|
||
Storno(ctx context.Context, id int64, razlog string, korisnikID *int64) error
|
||
Obrisi(ctx context.Context, id int64, korisnikID *int64) error
|
||
SledeciBroj(ctx context.Context) (string, error)
|
||
}
|
||
|
||
// ServisniDeloviRepository definiše operacije nad ugrađenim delovima u servisu
|
||
type ServisniDeloviRepository interface {
|
||
DohvatiZaNalog(ctx context.Context, nalogID int64) ([]model.ServisniDeoSaArtiklom, error)
|
||
Dodaj(ctx context.Context, nalogID, artikalID int64, kolicina int, cenaKomada float64, korisnikID *int64) (int64, error)
|
||
Obrisi(ctx context.Context, id int64, korisnikID *int64) error
|
||
}
|
||
|
||
// MagacinskePromeneRepository definiše operacije nad revizijskim tragom magacina
|
||
type MagacinskePromeneRepository interface {
|
||
Lista(ctx context.Context, artikalID *int64, limit int) ([]model.MagacinskaPromenaSaDetaljem, error)
|
||
}
|
||
|
||
// KorisniciRepository definiše operacije nad korisnicima
|
||
type KorisniciRepository interface {
|
||
Kreiraj(ctx context.Context, korisnickoIme, lozinkaHash, uloga string) (*model.Korisnik, error)
|
||
DohvatiPoImenu(ctx context.Context, korisnickoIme string) (*model.Korisnik, error)
|
||
DohvatiPoID(ctx context.Context, id int64) (*model.Korisnik, error)
|
||
Lista(ctx context.Context) ([]model.Korisnik, error)
|
||
AzurirajUlogu(ctx context.Context, id int64, uloga string) error
|
||
AzurirajAktivan(ctx context.Context, id int64, aktivan bool) error
|
||
PromeniLozinku(ctx context.Context, id int64, hash string) error
|
||
SacuvajTotpTajnu(ctx context.Context, id int64, tajna string) error
|
||
SacuvajLokalnuTemu(ctx context.Context, id int64, lokalnaTema string, koristi bool) error
|
||
SacuvajLokalnuPozadinu(ctx context.Context, id int64, pozadina, opacity, blur, blurPozadine, glassOpacity string) error
|
||
SacuvajLokalnuAnimaciju(ctx context.Context, id int64, animacija string) error
|
||
SacuvajLokalniHover(ctx context.Context, id int64, hover string) error
|
||
SacuvajLokalnuBrzinuAnimacije(ctx context.Context, id int64, brzina string) error
|
||
SacuvajAvatar(ctx context.Context, id int64, putanja string) error
|
||
PostojiIjedan(ctx context.Context) (bool, error)
|
||
Obrisi(ctx context.Context, id int64) error
|
||
}
|
||
|
||
// SesijeRepository definiše operacije nad sesijama
|
||
type SesijeRepository interface {
|
||
Kreiraj(ctx context.Context, korisnikID int64, token string, istice time.Time, totpPotvrdjeno bool) error
|
||
DohvatiPoTokenu(ctx context.Context, token string) (*model.Sesija, error)
|
||
PotvrdiTotp(ctx context.Context, token string, novoIstice time.Time) error
|
||
Obrisi(ctx context.Context, token string) error
|
||
ObrisiIstekle(ctx context.Context) error
|
||
}
|
||
|
||
// PodsetnikFilter definiše parametre za filtriranje liste podsetnika
|
||
type PodsetnikFilter struct {
|
||
SamoAktivni bool // true = samo nezavršeni; false = svi
|
||
KorisnikID *int64 // ako nije nil — samo podsetnici tog korisnika
|
||
}
|
||
|
||
// PokusajiPrijaveRepository definiše operacije nad evidencijom pokušaja prijave
|
||
type PokusajiPrijaveRepository interface {
|
||
Zabeleži(ctx context.Context, ip, korisnickoIme string, uspeh bool) error
|
||
BrojNeuspeha(ctx context.Context, ip string, od time.Time) (int, error)
|
||
VremePoslednjeg(ctx context.Context, ip string, od time.Time) (time.Time, bool, error)
|
||
ObrisiStare(ctx context.Context, pre time.Time) error
|
||
}
|
||
|
||
// LoginIstorijsaRepository definiše operacije nad evidencijom prijava
|
||
type LoginIstorijsaRepository interface {
|
||
Zabeleži(ctx context.Context, korisnikID *int64, ip, userAgent, razlog string, uspeh bool) error
|
||
ListaZaKorisnika(ctx context.Context, korisnikID int64, limit int) ([]*model.LoginPokusaj, error)
|
||
}
|
||
|
||
// DozvoleRepository definiše operacije nad dozvolama po ulogama
|
||
type DozvoleRepository interface {
|
||
ImaDozvolu(ctx context.Context, uloga, akcija string) bool
|
||
SveDozvole(ctx context.Context, uloga string) map[string]bool
|
||
Sacuvaj(ctx context.Context, uloga, akcija string, dozvoljeno bool) error
|
||
Reset(ctx context.Context) error
|
||
}
|
||
|
||
// RezervniKodoviRepository definiše operacije nad rezervnim (jednokratnim) 2FA kodovima.
|
||
// Kodovi se čuvaju kao bcrypt heš; Iskoristi prima čist kod i poredi ga sa hešovima.
|
||
type RezervniKodoviRepository interface {
|
||
Zameni(ctx context.Context, korisnikID int64, hashevi []string) error
|
||
Iskoristi(ctx context.Context, korisnikID int64, kod string) (bool, error)
|
||
BrojPreostalih(ctx context.Context, korisnikID int64) (int, error)
|
||
Obrisi(ctx context.Context, korisnikID int64) error
|
||
}
|
||
|
||
// IzvestajRepository definiše read-only upite za dashboard i stranicu izveštaja.
|
||
// Vraća sirove podatke; prezentaciju (datumi, boje, rang) radi handler.
|
||
type IzvestajRepository interface {
|
||
// dashboard — brojači
|
||
BrojArtikala(ctx context.Context) (int, error)
|
||
BrojAktivnihServisa(ctx context.Context) (int, error)
|
||
PrihodTekuciMesec(ctx context.Context) (float64, error)
|
||
BrojKriticnihZaliha(ctx context.Context) (int, error)
|
||
// dashboard — liste
|
||
PoslednjiServisi(ctx context.Context, limit int) ([]model.ServisRedDashboard, error)
|
||
KriticneZalihe(ctx context.Context, limit int) ([]model.ZalihaRed, error)
|
||
PoslednjeProdaje(ctx context.Context, limit int) ([]model.ProdajaRedDashboard, error)
|
||
// izveštaji
|
||
MesecniPrihodProdaja(ctx context.Context) ([]model.MesecniIznos, error)
|
||
MesecniPrihodServis(ctx context.Context) ([]model.MesecniIznos, error)
|
||
StariOtvoreniNalozi(ctx context.Context) ([]model.StariNalogRed, error)
|
||
TopArtikli(ctx context.Context, limit int) ([]model.TopArtikalRed, error)
|
||
TopKlijenti(ctx context.Context, limit int) ([]model.TopKlijentRed, error)
|
||
// magacinski izveštaji
|
||
PrometniList(ctx context.Context, od, do time.Time) ([]model.PrometniRed, error)
|
||
StanjeZaliha(ctx context.Context) ([]model.StanjeZalihaRed, error)
|
||
}
|
||
|
||
// PodsetnikRepository definiše operacije nad podsetnicima
|
||
type PodsetnikRepository interface {
|
||
Lista(ctx context.Context, filter PodsetnikFilter) ([]model.Podsetnik, error)
|
||
DohvatiID(ctx context.Context, id int64) (*model.Podsetnik, error)
|
||
Kreiraj(ctx context.Context, p *model.Podsetnik) (int64, error)
|
||
Izmeni(ctx context.Context, p *model.Podsetnik) error
|
||
OznaciZavrsenim(ctx context.Context, id int64, zavrseno bool) error
|
||
Obrisi(ctx context.Context, id int64) error
|
||
BrojAktivnih(ctx context.Context, filter PodsetnikFilter) (int, error)
|
||
}
|