Dodavanje modula dobavljača i nabavki
This commit is contained in:
@@ -0,0 +1,130 @@
|
||||
package sqlite
|
||||
|
||||
import (
|
||||
"context"
|
||||
"database/sql"
|
||||
"fmt"
|
||||
|
||||
"ntech/internal/model"
|
||||
)
|
||||
|
||||
// DobavljacRepo je SQLite implementacija DobavljacRepository interfejsa
|
||||
type DobavljacRepo struct {
|
||||
db *sql.DB
|
||||
}
|
||||
|
||||
// NoviDobavljacRepo kreira novi DobavljacRepo
|
||||
func NoviDobavljacRepo(db *sql.DB) *DobavljacRepo {
|
||||
return &DobavljacRepo{db: db}
|
||||
}
|
||||
|
||||
// Lista vraća listu dobavljača sa opcionom pretragom po nazivu
|
||||
func (r *DobavljacRepo) Lista(ctx context.Context, pretraga string) ([]model.Dobavljac, error) {
|
||||
upit := `
|
||||
SELECT id, naziv, kontakt_osoba, telefon, email, napomena, datum_unosa
|
||||
FROM dobavljaci
|
||||
WHERE 1=1`
|
||||
|
||||
args := []any{}
|
||||
|
||||
if pretraga != "" {
|
||||
upit += " AND naziv LIKE ?"
|
||||
args = append(args, "%"+pretraga+"%")
|
||||
}
|
||||
|
||||
upit += " ORDER BY naziv ASC"
|
||||
|
||||
redovi, err := r.db.QueryContext(ctx, upit, args...)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: DobavljacRepo.Lista: %w", err)
|
||||
}
|
||||
defer redovi.Close()
|
||||
|
||||
var rezultat []model.Dobavljac
|
||||
for redovi.Next() {
|
||||
var d model.Dobavljac
|
||||
var kontaktOsoba, telefon, email, napomena sql.NullString
|
||||
err := redovi.Scan(
|
||||
&d.ID, &d.Naziv, &kontaktOsoba, &telefon, &email, &napomena, &d.DatumUnosa,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: DobavljacRepo.Lista: scan: %w", err)
|
||||
}
|
||||
d.KontaktOsoba = kontaktOsoba.String
|
||||
d.Telefon = telefon.String
|
||||
d.Email = email.String
|
||||
d.Napomena = napomena.String
|
||||
rezultat = append(rezultat, d)
|
||||
}
|
||||
|
||||
return rezultat, nil
|
||||
}
|
||||
|
||||
// DohvatiID vraća jednog dobavljača po ID-u
|
||||
func (r *DobavljacRepo) DohvatiID(ctx context.Context, id int64) (*model.Dobavljac, error) {
|
||||
var d model.Dobavljac
|
||||
var kontaktOsoba, telefon, email, napomena sql.NullString
|
||||
|
||||
err := r.db.QueryRowContext(ctx, `
|
||||
SELECT id, naziv, kontakt_osoba, telefon, email, napomena, datum_unosa
|
||||
FROM dobavljaci WHERE id = ?`, id).Scan(
|
||||
&d.ID, &d.Naziv, &kontaktOsoba, &telefon, &email, &napomena, &d.DatumUnosa,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: DobavljacRepo.DohvatiID: %w", err)
|
||||
}
|
||||
|
||||
d.KontaktOsoba = kontaktOsoba.String
|
||||
d.Telefon = telefon.String
|
||||
d.Email = email.String
|
||||
d.Napomena = napomena.String
|
||||
|
||||
return &d, nil
|
||||
}
|
||||
|
||||
// Kreiraj dodaje novog dobavljača u bazu
|
||||
func (r *DobavljacRepo) Kreiraj(ctx context.Context, d *model.Dobavljac) (int64, error) {
|
||||
rezultat, err := r.db.ExecContext(ctx, `
|
||||
INSERT INTO dobavljaci (naziv, kontakt_osoba, telefon, email, napomena)
|
||||
VALUES (?, ?, ?, ?, ?)`,
|
||||
d.Naziv, nullString(d.KontaktOsoba), nullString(d.Telefon),
|
||||
nullString(d.Email), nullString(d.Napomena),
|
||||
)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: DobavljacRepo.Kreiraj: %w", err)
|
||||
}
|
||||
|
||||
id, err := rezultat.LastInsertId()
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: DobavljacRepo.Kreiraj: last insert id: %w", err)
|
||||
}
|
||||
|
||||
return id, nil
|
||||
}
|
||||
|
||||
// Izmeni ažurira postojećeg dobavljača
|
||||
func (r *DobavljacRepo) Izmeni(ctx context.Context, d *model.Dobavljac) error {
|
||||
_, err := r.db.ExecContext(ctx, `
|
||||
UPDATE dobavljaci SET
|
||||
naziv = ?, kontakt_osoba = ?, telefon = ?, email = ?, napomena = ?
|
||||
WHERE id = ?`,
|
||||
d.Naziv, nullString(d.KontaktOsoba), nullString(d.Telefon),
|
||||
nullString(d.Email), nullString(d.Napomena), d.ID,
|
||||
)
|
||||
if err != nil {
|
||||
return fmt.Errorf("ntech: DobavljacRepo.Izmeni: %w", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// Obrisi briše dobavljača po ID-u
|
||||
func (r *DobavljacRepo) Obrisi(ctx context.Context, id int64) error {
|
||||
_, err := r.db.ExecContext(ctx, "DELETE FROM dobavljaci WHERE id = ?", id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("ntech: DobavljacRepo.Obrisi: %w", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -0,0 +1,181 @@
|
||||
package sqlite
|
||||
|
||||
import (
|
||||
"context"
|
||||
"database/sql"
|
||||
"fmt"
|
||||
|
||||
"ntech/internal/model"
|
||||
)
|
||||
|
||||
// NabavkaRepo je SQLite implementacija NabavkaRepository interfejsa
|
||||
type NabavkaRepo struct {
|
||||
db *sql.DB
|
||||
}
|
||||
|
||||
// NoviNabavkaRepo kreira novi NabavkaRepo
|
||||
func NoviNabavkaRepo(db *sql.DB) *NabavkaRepo {
|
||||
return &NabavkaRepo{db: db}
|
||||
}
|
||||
|
||||
// Lista vraća sve nabavke sa nazivom dobavljača, sortirano od najnovije
|
||||
func (r *NabavkaRepo) Lista(ctx context.Context) ([]model.NabavkaSaDetaljem, error) {
|
||||
redovi, err := r.db.QueryContext(ctx, `
|
||||
SELECT
|
||||
n.id, n.dobavljac_id, n.napomena, n.ukupno, n.datum,
|
||||
COALESCE(d.naziv, '') AS dobavljac_naziv
|
||||
FROM nabavke n
|
||||
LEFT JOIN dobavljaci d ON n.dobavljac_id = d.id
|
||||
ORDER BY n.datum DESC`)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NabavkaRepo.Lista: %w", err)
|
||||
}
|
||||
defer redovi.Close()
|
||||
|
||||
var rezultat []model.NabavkaSaDetaljem
|
||||
for redovi.Next() {
|
||||
var n model.NabavkaSaDetaljem
|
||||
var dobavljacID sql.NullInt64
|
||||
var napomena sql.NullString
|
||||
|
||||
err := redovi.Scan(
|
||||
&n.ID, &dobavljacID, &napomena, &n.Ukupno, &n.Datum,
|
||||
&n.DobavljacNaziv,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NabavkaRepo.Lista: scan: %w", err)
|
||||
}
|
||||
|
||||
if dobavljacID.Valid {
|
||||
n.DobavljacID = &dobavljacID.Int64
|
||||
}
|
||||
n.Napomena = napomena.String
|
||||
|
||||
rezultat = append(rezultat, n)
|
||||
}
|
||||
|
||||
return rezultat, nil
|
||||
}
|
||||
|
||||
// DohvatiID vraća zaglavlje jedne nabavke po ID-u
|
||||
func (r *NabavkaRepo) DohvatiID(ctx context.Context, id int64) (*model.Nabavka, error) {
|
||||
var n model.Nabavka
|
||||
var dobavljacID sql.NullInt64
|
||||
var napomena sql.NullString
|
||||
|
||||
err := r.db.QueryRowContext(ctx, `
|
||||
SELECT id, dobavljac_id, napomena, ukupno, datum
|
||||
FROM nabavke WHERE id = ?`, id).Scan(
|
||||
&n.ID, &dobavljacID, &napomena, &n.Ukupno, &n.Datum,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NabavkaRepo.DohvatiID: %w", err)
|
||||
}
|
||||
|
||||
if dobavljacID.Valid {
|
||||
n.DobavljacID = &dobavljacID.Int64
|
||||
}
|
||||
n.Napomena = napomena.String
|
||||
|
||||
return &n, nil
|
||||
}
|
||||
|
||||
// DohvatiStavke vraća sve stavke jedne nabavke sa nazivima artikala
|
||||
func (r *NabavkaRepo) DohvatiStavke(ctx context.Context, nabavkaID int64) ([]model.StavkaSaArtiklom, error) {
|
||||
redovi, err := r.db.QueryContext(ctx, `
|
||||
SELECT
|
||||
s.id, s.nabavka_id, s.artikal_id, s.kolicina,
|
||||
s.cena_po_komadu, s.ukupno,
|
||||
a.naziv AS artikal_naziv
|
||||
FROM stavke_nabavke s
|
||||
JOIN artikli a ON s.artikal_id = a.id
|
||||
WHERE s.nabavka_id = ?
|
||||
ORDER BY s.id ASC`, nabavkaID)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NabavkaRepo.DohvatiStavke: %w", err)
|
||||
}
|
||||
defer redovi.Close()
|
||||
|
||||
var rezultat []model.StavkaSaArtiklom
|
||||
for redovi.Next() {
|
||||
var s model.StavkaSaArtiklom
|
||||
err := redovi.Scan(
|
||||
&s.ID, &s.NabavkaID, &s.ArtikalID, &s.Kolicina,
|
||||
&s.CenaPoKomadu, &s.Ukupno,
|
||||
&s.ArtikalNaziv,
|
||||
)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("ntech: NabavkaRepo.DohvatiStavke: scan: %w", err)
|
||||
}
|
||||
rezultat = append(rezultat, s)
|
||||
}
|
||||
|
||||
return rezultat, nil
|
||||
}
|
||||
|
||||
// Kreiraj upisuje novu nabavku sa svim stavkama u jednoj transakciji i ažurira stanje magacina
|
||||
func (r *NabavkaRepo) Kreiraj(ctx context.Context, n *model.Nabavka, stavke []model.StavkaNabavke) (int64, error) {
|
||||
tx, err := r.db.BeginTx(ctx, nil)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NabavkaRepo.Kreiraj: begin: %w", err)
|
||||
}
|
||||
defer tx.Rollback()
|
||||
|
||||
// računamo ukupan iznos nabavke kao zbir svih stavki
|
||||
var ukupno float64
|
||||
for i := range stavke {
|
||||
stavke[i].Ukupno = float64(stavke[i].Kolicina) * stavke[i].CenaPoKomadu
|
||||
ukupno += stavke[i].Ukupno
|
||||
}
|
||||
|
||||
// upisujemo zaglavlje nabavke
|
||||
rezultat, err := tx.ExecContext(ctx, `
|
||||
INSERT INTO nabavke (dobavljac_id, napomena, ukupno)
|
||||
VALUES (?, ?, ?)`,
|
||||
nullInt64(n.DobavljacID), nullString(n.Napomena), ukupno,
|
||||
)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NabavkaRepo.Kreiraj: insert nabavka: %w", err)
|
||||
}
|
||||
|
||||
nabavkaID, err := rezultat.LastInsertId()
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NabavkaRepo.Kreiraj: last insert id: %w", err)
|
||||
}
|
||||
|
||||
// upisujemo svaku stavku i ažuriramo stanje artikla u magacinu
|
||||
for _, s := range stavke {
|
||||
_, err := tx.ExecContext(ctx, `
|
||||
INSERT INTO stavke_nabavke (nabavka_id, artikal_id, kolicina, cena_po_komadu, ukupno)
|
||||
VALUES (?, ?, ?, ?, ?)`,
|
||||
nabavkaID, s.ArtikalID, s.Kolicina, s.CenaPoKomadu, s.Ukupno,
|
||||
)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NabavkaRepo.Kreiraj: insert stavka: %w", err)
|
||||
}
|
||||
|
||||
_, err = tx.ExecContext(ctx,
|
||||
"UPDATE artikli SET kolicina = kolicina + ? WHERE id = ?",
|
||||
s.Kolicina, s.ArtikalID,
|
||||
)
|
||||
if err != nil {
|
||||
return 0, fmt.Errorf("ntech: NabavkaRepo.Kreiraj: update kolicina: %w", err)
|
||||
}
|
||||
}
|
||||
|
||||
if err := tx.Commit(); err != nil {
|
||||
return 0, fmt.Errorf("ntech: NabavkaRepo.Kreiraj: commit: %w", err)
|
||||
}
|
||||
|
||||
return nabavkaID, nil
|
||||
}
|
||||
|
||||
// Obrisi briše nabavku po ID-u — stavke se brišu automatski (ON DELETE CASCADE)
|
||||
// Napomena: brisanje ne vraća količine artikala u magacin
|
||||
func (r *NabavkaRepo) Obrisi(ctx context.Context, id int64) error {
|
||||
_, err := r.db.ExecContext(ctx, "DELETE FROM nabavke WHERE id = ?", id)
|
||||
if err != nil {
|
||||
return fmt.Errorf("ntech: NabavkaRepo.Obrisi: %w", err)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
@@ -0,0 +1,18 @@
|
||||
package sqlite
|
||||
|
||||
import "database/sql"
|
||||
|
||||
// nullString pretvara prazan Go string u sql.NullString sa NULL vrednošću —
|
||||
// koristi se pri unosu i izmeni kada polje u bazi sme biti NULL
|
||||
func nullString(s string) sql.NullString {
|
||||
return sql.NullString{String: s, Valid: s != ""}
|
||||
}
|
||||
|
||||
// nullInt64 pretvara *int64 pokazivač u sql.NullInt64 —
|
||||
// koristi se za opciona FK polja koja smeju biti NULL u bazi
|
||||
func nullInt64(v *int64) sql.NullInt64 {
|
||||
if v == nil {
|
||||
return sql.NullInt64{}
|
||||
}
|
||||
return sql.NullInt64{Int64: *v, Valid: true}
|
||||
}
|
||||
Reference in New Issue
Block a user