Compare commits
34 Commits
enaklo-sel
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 6d735c20cb | |||
|
|
cb8a830345 | ||
| 9c143a43aa | |||
|
|
222cadd8df | ||
| cad4e6c816 | |||
|
|
50d633ee3a | ||
|
|
21fa21d089 | ||
| 5f379faf17 | |||
| 3b62504798 | |||
| 4130cb66df | |||
| 30dff17272 | |||
|
|
fa037b4d2a | ||
| d38a770ec5 | |||
| 015292e830 | |||
|
|
f8c732f0ff | ||
| e92c487815 | |||
| c573b23d76 | |||
|
|
f73a5d533c | ||
|
|
4ea8e32a8e | ||
|
|
06d79046d0 | ||
| 8eb19c57ba | |||
|
|
f123de7233 | ||
|
|
7ba776555e | ||
|
|
bccf02b5f7 | ||
|
|
c24a8a8c13 | ||
| 6064ef8fde | |||
|
|
1834dd0b19 | ||
|
|
9f653eef37 | ||
|
|
ddaf6df436 | ||
| 0708ce816e | |||
| 2c34578a98 | |||
|
|
9d71b339b5 | ||
|
|
bbd6666299 | ||
|
|
23b6293502 |
3
.gitignore
vendored
3
.gitignore
vendored
@ -6,3 +6,6 @@ config/env/*
|
|||||||
!.env
|
!.env
|
||||||
|
|
||||||
vendor
|
vendor
|
||||||
|
|
||||||
|
# Firebase service account credentials
|
||||||
|
infra/firebase-service-account.json
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
# 1) Build stage
|
# 1) Build stage
|
||||||
FROM golang:1.21-alpine AS build
|
FROM golang:1.24-alpine AS build
|
||||||
RUN apk --no-cache add ca-certificates tzdata git curl
|
RUN apk --no-cache add ca-certificates tzdata git curl
|
||||||
WORKDIR /src
|
WORKDIR /src
|
||||||
COPY go.mod go.sum ./
|
COPY go.mod go.sum ./
|
||||||
|
|||||||
@ -31,6 +31,7 @@ type Config struct {
|
|||||||
Log Log `mapstructure:"log"`
|
Log Log `mapstructure:"log"`
|
||||||
S3Config S3Config `mapstructure:"s3"`
|
S3Config S3Config `mapstructure:"s3"`
|
||||||
Fonnte Fonnte `mapstructure:"fonnte"`
|
Fonnte Fonnte `mapstructure:"fonnte"`
|
||||||
|
FCM FCM `mapstructure:"fcm"`
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@ -95,3 +96,7 @@ func (c *Config) LogFormat() string {
|
|||||||
func (c *Config) GetFonnte() *Fonnte {
|
func (c *Config) GetFonnte() *Fonnte {
|
||||||
return &c.Fonnte
|
return &c.Fonnte
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Config) GetFCM() *FCM {
|
||||||
|
return &c.FCM
|
||||||
|
}
|
||||||
|
|||||||
14
config/fcm.go
Normal file
14
config/fcm.go
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
package config
|
||||||
|
|
||||||
|
type FCM struct {
|
||||||
|
CredentialsFile string `mapstructure:"credentials_file"`
|
||||||
|
ProjectID string `mapstructure:"project_id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *FCM) GetCredentialsFile() string {
|
||||||
|
return f.CredentialsFile
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *FCM) GetProjectID() string {
|
||||||
|
return f.ProjectID
|
||||||
|
}
|
||||||
@ -1,7 +1,8 @@
|
|||||||
package config
|
package config
|
||||||
|
|
||||||
type Server struct {
|
type Server struct {
|
||||||
Port string `mapstructure:"port"`
|
Port string `mapstructure:"port"`
|
||||||
BaseUrl string `mapstructure:"common-url"`
|
BaseUrl string `mapstructure:"common-url"`
|
||||||
LocalUrl string `mapstructure:"local-url"`
|
LocalUrl string `mapstructure:"local-url"`
|
||||||
|
SelfOrderUrl string `mapstructure:"self-order-url"`
|
||||||
}
|
}
|
||||||
|
|||||||
68
go.mod
68
go.mod
@ -5,25 +5,50 @@ go 1.24
|
|||||||
require (
|
require (
|
||||||
github.com/gin-gonic/gin v1.9.1
|
github.com/gin-gonic/gin v1.9.1
|
||||||
github.com/go-playground/validator/v10 v10.17.0
|
github.com/go-playground/validator/v10 v10.17.0
|
||||||
github.com/google/uuid v1.1.2
|
github.com/google/uuid v1.6.0
|
||||||
github.com/lib/pq v1.2.0
|
github.com/lib/pq v1.2.0
|
||||||
github.com/spf13/viper v1.16.0
|
github.com/spf13/viper v1.16.0
|
||||||
gopkg.in/yaml.v3 v3.0.1
|
gopkg.in/yaml.v3 v3.0.1
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
cel.dev/expr v0.23.1 // indirect
|
||||||
|
cloud.google.com/go v0.121.0 // indirect
|
||||||
|
cloud.google.com/go/auth v0.16.1 // indirect
|
||||||
|
cloud.google.com/go/auth/oauth2adapt v0.2.8 // indirect
|
||||||
|
cloud.google.com/go/compute/metadata v0.6.0 // indirect
|
||||||
|
cloud.google.com/go/firestore v1.18.0 // indirect
|
||||||
|
cloud.google.com/go/iam v1.5.2 // indirect
|
||||||
|
cloud.google.com/go/longrunning v0.6.7 // indirect
|
||||||
|
cloud.google.com/go/monitoring v1.24.2 // indirect
|
||||||
|
cloud.google.com/go/storage v1.53.0 // indirect
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/detectors/gcp v1.27.0 // indirect
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/exporter/metric v0.51.0 // indirect
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/resourcemapping v0.51.0 // indirect
|
||||||
|
github.com/MicahParks/keyfunc v1.9.0 // indirect
|
||||||
github.com/bytedance/sonic v1.10.2 // indirect
|
github.com/bytedance/sonic v1.10.2 // indirect
|
||||||
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
github.com/cespare/xxhash/v2 v2.3.0 // indirect
|
||||||
github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect
|
github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // indirect
|
||||||
github.com/chenzhuoyu/iasm v0.9.1 // indirect
|
github.com/chenzhuoyu/iasm v0.9.1 // indirect
|
||||||
|
github.com/cncf/xds/go v0.0.0-20250501225837-2ac532fd4443 // indirect
|
||||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
|
github.com/envoyproxy/go-control-plane/envoy v1.32.4 // indirect
|
||||||
|
github.com/envoyproxy/protoc-gen-validate v1.2.1 // indirect
|
||||||
|
github.com/felixge/httpsnoop v1.0.4 // indirect
|
||||||
github.com/fsnotify/fsnotify v1.6.0 // indirect
|
github.com/fsnotify/fsnotify v1.6.0 // indirect
|
||||||
github.com/gabriel-vasile/mimetype v1.4.3 // indirect
|
github.com/gabriel-vasile/mimetype v1.4.3 // indirect
|
||||||
github.com/gin-contrib/sse v0.1.0 // indirect
|
github.com/gin-contrib/sse v0.1.0 // indirect
|
||||||
|
github.com/go-jose/go-jose/v4 v4.0.5 // indirect
|
||||||
|
github.com/go-logr/logr v1.4.2 // indirect
|
||||||
|
github.com/go-logr/stdr v1.2.2 // indirect
|
||||||
github.com/go-playground/locales v0.14.1 // indirect
|
github.com/go-playground/locales v0.14.1 // indirect
|
||||||
github.com/go-playground/universal-translator v0.18.1 // indirect
|
github.com/go-playground/universal-translator v0.18.1 // indirect
|
||||||
github.com/goccy/go-json v0.10.2 // indirect
|
github.com/goccy/go-json v0.10.2 // indirect
|
||||||
github.com/google/go-cmp v0.6.0 // indirect
|
github.com/golang-jwt/jwt/v4 v4.5.2 // indirect
|
||||||
|
github.com/golang/protobuf v1.5.4 // indirect
|
||||||
|
github.com/google/s2a-go v0.1.9 // indirect
|
||||||
|
github.com/googleapis/enterprise-certificate-proxy v0.3.6 // indirect
|
||||||
|
github.com/googleapis/gax-go/v2 v2.14.1 // indirect
|
||||||
github.com/hashicorp/hcl v1.0.0 // indirect
|
github.com/hashicorp/hcl v1.0.0 // indirect
|
||||||
github.com/jackc/pgpassfile v1.0.0 // indirect
|
github.com/jackc/pgpassfile v1.0.0 // indirect
|
||||||
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
|
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
|
||||||
@ -40,36 +65,57 @@ require (
|
|||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
github.com/pelletier/go-toml/v2 v2.1.1 // indirect
|
github.com/pelletier/go-toml/v2 v2.1.1 // indirect
|
||||||
|
github.com/planetscale/vtprotobuf v0.6.1-0.20240319094008-0393e58bdf10 // indirect
|
||||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
github.com/rogpeppe/go-internal v1.11.0 // indirect
|
github.com/spf13/afero v1.10.0 // indirect
|
||||||
github.com/spf13/afero v1.9.5 // indirect
|
|
||||||
github.com/spf13/cast v1.5.1 // indirect
|
github.com/spf13/cast v1.5.1 // indirect
|
||||||
github.com/spf13/jwalterweatherman v1.1.0 // indirect
|
github.com/spf13/jwalterweatherman v1.1.0 // indirect
|
||||||
github.com/spf13/pflag v1.0.5 // indirect
|
github.com/spf13/pflag v1.0.5 // indirect
|
||||||
github.com/stretchr/objx v0.5.0 // indirect
|
github.com/spiffe/go-spiffe/v2 v2.5.0 // indirect
|
||||||
|
github.com/stretchr/objx v0.5.2 // indirect
|
||||||
github.com/subosito/gotenv v1.4.2 // indirect
|
github.com/subosito/gotenv v1.4.2 // indirect
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
github.com/ugorji/go/codec v1.2.12 // indirect
|
github.com/ugorji/go/codec v1.2.12 // indirect
|
||||||
|
github.com/zeebo/errs v1.4.0 // indirect
|
||||||
|
go.opentelemetry.io/auto/sdk v1.1.0 // indirect
|
||||||
|
go.opentelemetry.io/contrib/detectors/gcp v1.35.0 // indirect
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.60.0 // indirect
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.60.0 // indirect
|
||||||
|
go.opentelemetry.io/otel v1.35.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/metric v1.35.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/sdk v1.35.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/sdk/metric v1.35.0 // indirect
|
||||||
|
go.opentelemetry.io/otel/trace v1.35.0 // indirect
|
||||||
go.uber.org/atomic v1.11.0 // indirect
|
go.uber.org/atomic v1.11.0 // indirect
|
||||||
go.uber.org/multierr v1.8.0 // indirect
|
go.uber.org/multierr v1.8.0 // indirect
|
||||||
golang.org/x/arch v0.7.0 // indirect
|
golang.org/x/arch v0.7.0 // indirect
|
||||||
golang.org/x/net v0.30.0 // indirect
|
golang.org/x/net v0.42.0 // indirect
|
||||||
golang.org/x/sys v0.30.0 // indirect
|
golang.org/x/oauth2 v0.30.0 // indirect
|
||||||
golang.org/x/text v0.20.0 // indirect
|
golang.org/x/sync v0.16.0 // indirect
|
||||||
google.golang.org/protobuf v1.32.0 // indirect
|
golang.org/x/sys v0.34.0 // indirect
|
||||||
|
golang.org/x/text v0.27.0 // indirect
|
||||||
|
golang.org/x/time v0.11.0 // indirect
|
||||||
|
google.golang.org/appengine/v2 v2.0.6 // indirect
|
||||||
|
google.golang.org/genproto v0.0.0-20250505200425-f936aa4a68b2 // indirect
|
||||||
|
google.golang.org/genproto/googleapis/api v0.0.0-20250505200425-f936aa4a68b2 // indirect
|
||||||
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250505200425-f936aa4a68b2 // indirect
|
||||||
|
google.golang.org/grpc v1.72.0 // indirect
|
||||||
|
google.golang.org/protobuf v1.36.6 // indirect
|
||||||
gopkg.in/ini.v1 v1.67.0 // indirect
|
gopkg.in/ini.v1 v1.67.0 // indirect
|
||||||
gopkg.in/yaml.v2 v2.4.0 // indirect
|
gopkg.in/yaml.v2 v2.4.0 // indirect
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
firebase.google.com/go/v4 v4.19.0
|
||||||
github.com/aws/aws-sdk-go v1.55.7
|
github.com/aws/aws-sdk-go v1.55.7
|
||||||
github.com/boombuler/barcode v1.1.0
|
github.com/boombuler/barcode v1.1.0
|
||||||
github.com/golang-jwt/jwt/v5 v5.2.3
|
github.com/golang-jwt/jwt/v5 v5.2.3
|
||||||
github.com/redis/go-redis/v9 v9.19.0
|
github.com/redis/go-redis/v9 v9.19.0
|
||||||
github.com/sirupsen/logrus v1.9.3
|
github.com/sirupsen/logrus v1.9.3
|
||||||
github.com/stretchr/testify v1.8.4
|
github.com/stretchr/testify v1.10.0
|
||||||
go.uber.org/zap v1.21.0
|
go.uber.org/zap v1.21.0
|
||||||
golang.org/x/crypto v0.28.0
|
golang.org/x/crypto v0.40.0
|
||||||
|
google.golang.org/api v0.231.0
|
||||||
gorm.io/driver/postgres v1.5.0
|
gorm.io/driver/postgres v1.5.0
|
||||||
gorm.io/gorm v1.30.0
|
gorm.io/gorm v1.30.0
|
||||||
)
|
)
|
||||||
|
|||||||
163
go.sum
163
go.sum
@ -1,3 +1,5 @@
|
|||||||
|
cel.dev/expr v0.23.1 h1:K4KOtPCJQjVggkARsjG9RWXP6O4R73aHeJMa/dmCQQg=
|
||||||
|
cel.dev/expr v0.23.1/go.mod h1:hLPLo1W4QUmuYdA72RBX06QTs6MXw941piREPl3Yfiw=
|
||||||
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||||
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||||
cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
|
cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
|
||||||
@ -17,14 +19,32 @@ cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHOb
|
|||||||
cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI=
|
cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI=
|
||||||
cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk=
|
cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk=
|
||||||
cloud.google.com/go v0.75.0/go.mod h1:VGuuCn7PG0dwsd5XPVm2Mm3wlh3EL55/79EKB6hlPTY=
|
cloud.google.com/go v0.75.0/go.mod h1:VGuuCn7PG0dwsd5XPVm2Mm3wlh3EL55/79EKB6hlPTY=
|
||||||
|
cloud.google.com/go v0.121.0 h1:pgfwva8nGw7vivjZiRfrmglGWiCJBP+0OmDpenG/Fwg=
|
||||||
|
cloud.google.com/go v0.121.0/go.mod h1:rS7Kytwheu/y9buoDmu5EIpMMCI4Mb8ND4aeN4Vwj7Q=
|
||||||
|
cloud.google.com/go/auth v0.16.1 h1:XrXauHMd30LhQYVRHLGvJiYeczweKQXZxsTbV9TiguU=
|
||||||
|
cloud.google.com/go/auth v0.16.1/go.mod h1:1howDHJ5IETh/LwYs3ZxvlkXF48aSqqJUM+5o02dNOI=
|
||||||
|
cloud.google.com/go/auth/oauth2adapt v0.2.8 h1:keo8NaayQZ6wimpNSmW5OPc283g65QNIiLpZnkHRbnc=
|
||||||
|
cloud.google.com/go/auth/oauth2adapt v0.2.8/go.mod h1:XQ9y31RkqZCcwJWNSx2Xvric3RrU88hAYYbjDWYDL+c=
|
||||||
cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o=
|
cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o=
|
||||||
cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE=
|
cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE=
|
||||||
cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc=
|
cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc=
|
||||||
cloud.google.com/go/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg=
|
cloud.google.com/go/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg=
|
||||||
cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc=
|
cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc=
|
||||||
cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ=
|
cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ=
|
||||||
|
cloud.google.com/go/compute/metadata v0.6.0 h1:A6hENjEsCDtC1k8byVsgwvVcioamEHvZ4j01OwKxG9I=
|
||||||
|
cloud.google.com/go/compute/metadata v0.6.0/go.mod h1:FjyFAW1MW0C203CEOMDTu3Dk1FlqW3Rga40jzHL4hfg=
|
||||||
cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE=
|
cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE=
|
||||||
cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk=
|
cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk=
|
||||||
|
cloud.google.com/go/firestore v1.18.0 h1:cuydCaLS7Vl2SatAeivXyhbhDEIR8BDmtn4egDhIn2s=
|
||||||
|
cloud.google.com/go/firestore v1.18.0/go.mod h1:5ye0v48PhseZBdcl0qbl3uttu7FIEwEYVaWm0UIEOEU=
|
||||||
|
cloud.google.com/go/iam v1.5.2 h1:qgFRAGEmd8z6dJ/qyEchAuL9jpswyODjA2lS+w234g8=
|
||||||
|
cloud.google.com/go/iam v1.5.2/go.mod h1:SE1vg0N81zQqLzQEwxL2WI6yhetBdbNQuTvIKCSkUHE=
|
||||||
|
cloud.google.com/go/logging v1.13.0 h1:7j0HgAp0B94o1YRDqiqm26w4q1rDMH7XNRU34lJXHYc=
|
||||||
|
cloud.google.com/go/logging v1.13.0/go.mod h1:36CoKh6KA/M0PbhPKMq6/qety2DCAErbhXT62TuXALA=
|
||||||
|
cloud.google.com/go/longrunning v0.6.7 h1:IGtfDWHhQCgCjwQjV9iiLnUta9LBCo8R9QmAFsS/PrE=
|
||||||
|
cloud.google.com/go/longrunning v0.6.7/go.mod h1:EAFV3IZAKmM56TyiE6VAP3VoTzhZzySwI/YI1s/nRsY=
|
||||||
|
cloud.google.com/go/monitoring v1.24.2 h1:5OTsoJ1dXYIiMiuL+sYscLc9BumrL3CarVLL7dd7lHM=
|
||||||
|
cloud.google.com/go/monitoring v1.24.2/go.mod h1:x7yzPWcgDRnPEv3sI+jJGBkwl5qINf+6qY4eq0I9B4U=
|
||||||
cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I=
|
cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I=
|
||||||
cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw=
|
cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw=
|
||||||
cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA=
|
cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA=
|
||||||
@ -35,9 +55,25 @@ cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohl
|
|||||||
cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs=
|
cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs=
|
||||||
cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0=
|
cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0=
|
||||||
cloud.google.com/go/storage v1.14.0/go.mod h1:GrKmX003DSIwi9o29oFT7YDnHYwZoctc3fOKtUw0Xmo=
|
cloud.google.com/go/storage v1.14.0/go.mod h1:GrKmX003DSIwi9o29oFT7YDnHYwZoctc3fOKtUw0Xmo=
|
||||||
|
cloud.google.com/go/storage v1.53.0 h1:gg0ERZwL17pJ+Cz3cD2qS60w1WMDnwcm5YPAIQBHUAw=
|
||||||
|
cloud.google.com/go/storage v1.53.0/go.mod h1:7/eO2a/srr9ImZW9k5uufcNahT2+fPb8w5it1i5boaA=
|
||||||
|
cloud.google.com/go/trace v1.11.6 h1:2O2zjPzqPYAHrn3OKl029qlqG6W8ZdYaOWRyr8NgMT4=
|
||||||
|
cloud.google.com/go/trace v1.11.6/go.mod h1:GA855OeDEBiBMzcckLPE2kDunIpC72N+Pq8WFieFjnI=
|
||||||
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
|
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
|
||||||
|
firebase.google.com/go/v4 v4.19.0 h1:f5NMlC2YHFsncz00c2+ecBr+ZYlRMhKIhj1z8Iz0lD8=
|
||||||
|
firebase.google.com/go/v4 v4.19.0/go.mod h1:P7UfBpzc8+Z3MckX79+zsWzKVfpGryr6HLbAe7gCWfs=
|
||||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||||
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
|
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/detectors/gcp v1.27.0 h1:ErKg/3iS1AKcTkf3yixlZ54f9U1rljCkQyEXWUnIUxc=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/detectors/gcp v1.27.0/go.mod h1:yAZHSGnqScoU556rBOVkwLze6WP5N+U11RHuWaGVxwY=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/exporter/metric v0.51.0 h1:fYE9p3esPxA/C0rQ0AHhP0drtPXDRhaWiwg1DPqO7IU=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/exporter/metric v0.51.0/go.mod h1:BnBReJLvVYx2CS/UHOgVz2BXKXD9wsQPxZug20nZhd0=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/cloudmock v0.51.0 h1:OqVGm6Ei3x5+yZmSJG1Mh2NwHvpVmZ08CB5qJhT9Nuk=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/cloudmock v0.51.0/go.mod h1:SZiPHWGOOk3bl8tkevxkoiwPgsIl6CwrWcbwjfHZpdM=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/resourcemapping v0.51.0 h1:6/0iUd0xrnX7qt+mLNRwg5c0PGv8wpE8K90ryANQwMI=
|
||||||
|
github.com/GoogleCloudPlatform/opentelemetry-operations-go/internal/resourcemapping v0.51.0/go.mod h1:otE2jQekW/PqXk1Awf5lmfokJx4uwuqcj1ab5SpGeW0=
|
||||||
|
github.com/MicahParks/keyfunc v1.9.0 h1:lhKd5xrFHLNOWrDc4Tyb/Q1AJ4LCzQ48GVJyVIID3+o=
|
||||||
|
github.com/MicahParks/keyfunc v1.9.0/go.mod h1:IdnCilugA0O/99dW+/MkvlyrsX8+L8+x95xuVNtM5jw=
|
||||||
github.com/aws/aws-sdk-go v1.55.7 h1:UJrkFq7es5CShfBwlWAC8DA077vp8PyVbQd3lqLiztE=
|
github.com/aws/aws-sdk-go v1.55.7 h1:UJrkFq7es5CShfBwlWAC8DA077vp8PyVbQd3lqLiztE=
|
||||||
github.com/aws/aws-sdk-go v1.55.7/go.mod h1:eRwEWoyTWFMVYVQzKMNHWP5/RV4xIUGMQfXQHfHkpNU=
|
github.com/aws/aws-sdk-go v1.55.7/go.mod h1:eRwEWoyTWFMVYVQzKMNHWP5/RV4xIUGMQfXQHfHkpNU=
|
||||||
github.com/benbjohnson/clock v1.1.0 h1:Q92kusRqC1XV2MjkWETPvjJVqKetz1OzxZB7mHJLju8=
|
github.com/benbjohnson/clock v1.1.0 h1:Q92kusRqC1XV2MjkWETPvjJVqKetz1OzxZB7mHJLju8=
|
||||||
@ -69,6 +105,8 @@ github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDk
|
|||||||
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
|
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
|
||||||
github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
|
github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
|
||||||
github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
|
github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
|
||||||
|
github.com/cncf/xds/go v0.0.0-20250501225837-2ac532fd4443 h1:aQ3y1lwWyqYPiWZThqv1aFbZMiM9vblcSArJRf2Irls=
|
||||||
|
github.com/cncf/xds/go v0.0.0-20250501225837-2ac532fd4443/go.mod h1:W+zGtBO5Y1IgJhy4+A9GOqVhqLpfZi+vwmdNXUehLA8=
|
||||||
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
|
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
@ -78,7 +116,17 @@ github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.m
|
|||||||
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
|
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
|
||||||
github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po=
|
github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po=
|
||||||
github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk=
|
github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk=
|
||||||
|
github.com/envoyproxy/go-control-plane v0.13.4 h1:zEqyPVyku6IvWCFwux4x9RxkLOMUL+1vC9xUFv5l2/M=
|
||||||
|
github.com/envoyproxy/go-control-plane v0.13.4/go.mod h1:kDfuBlDVsSj2MjrLEtRWtHlsWIFcGyB2RMO44Dc5GZA=
|
||||||
|
github.com/envoyproxy/go-control-plane/envoy v1.32.4 h1:jb83lalDRZSpPWW2Z7Mck/8kXZ5CQAFYVjQcdVIr83A=
|
||||||
|
github.com/envoyproxy/go-control-plane/envoy v1.32.4/go.mod h1:Gzjc5k8JcJswLjAx1Zm+wSYE20UrLtt7JZMWiWQXQEw=
|
||||||
|
github.com/envoyproxy/go-control-plane/ratelimit v0.1.0 h1:/G9QYbddjL25KvtKTv3an9lx6VBE2cnb8wp1vEGNYGI=
|
||||||
|
github.com/envoyproxy/go-control-plane/ratelimit v0.1.0/go.mod h1:Wk+tMFAFbCXaJPzVVHnPgRKdUdwW/KdbRt94AzgRee4=
|
||||||
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
|
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
|
||||||
|
github.com/envoyproxy/protoc-gen-validate v1.2.1 h1:DEo3O99U8j4hBFwbJfrz9VtgcDfUKS7KJ7spH3d86P8=
|
||||||
|
github.com/envoyproxy/protoc-gen-validate v1.2.1/go.mod h1:d/C80l/jxXLdfEIhX1W2TmLfsJ31lvEjwamM4DxlWXU=
|
||||||
|
github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg=
|
||||||
|
github.com/felixge/httpsnoop v1.0.4/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
|
||||||
github.com/frankban/quicktest v1.14.4 h1:g2rn0vABPOOXmZUj+vbmUp0lPoXEMuhTpIluN0XL9UY=
|
github.com/frankban/quicktest v1.14.4 h1:g2rn0vABPOOXmZUj+vbmUp0lPoXEMuhTpIluN0XL9UY=
|
||||||
github.com/frankban/quicktest v1.14.4/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
|
github.com/frankban/quicktest v1.14.4/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
|
||||||
github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY=
|
github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY=
|
||||||
@ -92,6 +140,13 @@ github.com/gin-gonic/gin v1.9.1/go.mod h1:hPrL7YrpYKXt5YId3A/Tnip5kqbEAP+KLuI3SU
|
|||||||
github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU=
|
github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU=
|
||||||
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
||||||
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
||||||
|
github.com/go-jose/go-jose/v4 v4.0.5 h1:M6T8+mKZl/+fNNuFHvGIzDz7BTLQPIounk/b9dw3AaE=
|
||||||
|
github.com/go-jose/go-jose/v4 v4.0.5/go.mod h1:s3P1lRrkT8igV8D9OjyL4WRyHvjB6a4JSllnOrmmBOA=
|
||||||
|
github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
|
||||||
|
github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY=
|
||||||
|
github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
|
||||||
|
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
|
||||||
|
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
|
||||||
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
github.com/go-playground/assert/v2 v2.2.0 h1:JvknZsQTYeFEAhQwI4qEt9cyV5ONwRHC+lYKSsYSR8s=
|
||||||
github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
github.com/go-playground/assert/v2 v2.2.0/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4=
|
||||||
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA=
|
||||||
@ -102,6 +157,9 @@ github.com/go-playground/validator/v10 v10.17.0 h1:SmVVlfAOtlZncTxRuinDPomC2DkXJ
|
|||||||
github.com/go-playground/validator/v10 v10.17.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
github.com/go-playground/validator/v10 v10.17.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU=
|
||||||
github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU=
|
github.com/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU=
|
||||||
github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I=
|
github.com/goccy/go-json v0.10.2/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I=
|
||||||
|
github.com/golang-jwt/jwt/v4 v4.4.2/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0=
|
||||||
|
github.com/golang-jwt/jwt/v4 v4.5.2 h1:YtQM7lnr8iZ+j5q71MGKkNw9Mn7AjHM68uc9g5fXeUI=
|
||||||
|
github.com/golang-jwt/jwt/v4 v4.5.2/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0=
|
||||||
github.com/golang-jwt/jwt/v5 v5.2.3 h1:kkGXqQOBSDDWRhWNXTFpqGSCMyh/PLnqUvMGJPDJDs0=
|
github.com/golang-jwt/jwt/v5 v5.2.3 h1:kkGXqQOBSDDWRhWNXTFpqGSCMyh/PLnqUvMGJPDJDs0=
|
||||||
github.com/golang-jwt/jwt/v5 v5.2.3/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
github.com/golang-jwt/jwt/v5 v5.2.3/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk=
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
||||||
@ -129,6 +187,9 @@ github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvq
|
|||||||
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
|
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
|
||||||
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
||||||
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
||||||
|
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
|
||||||
|
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
|
||||||
|
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
|
||||||
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
||||||
github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
||||||
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
||||||
@ -140,12 +201,16 @@ github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/
|
|||||||
github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||||
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
|
||||||
|
github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
|
||||||
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
||||||
|
github.com/google/martian v2.1.0+incompatible h1:/CP5g8u/VJHijgedC/Legn3BAbAaWPgecwXBIDzw5no=
|
||||||
github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
|
github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
|
||||||
github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0=
|
github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0=
|
||||||
github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0=
|
github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0=
|
||||||
|
github.com/google/martian/v3 v3.3.3 h1:DIhPTQrbPkgs2yJYdXU/eNACCG5DVQjySNRNlflZ9Fc=
|
||||||
|
github.com/google/martian/v3 v3.3.3/go.mod h1:iEPrYcgCF7jA9OtScMFQyAlZZ4YXTKEtJ1E6RWzmBA0=
|
||||||
github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
||||||
github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
||||||
github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
||||||
@ -157,10 +222,17 @@ github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLe
|
|||||||
github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
||||||
github.com/google/pprof v0.0.0-20201218002935-b9804c9f04c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
github.com/google/pprof v0.0.0-20201218002935-b9804c9f04c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
||||||
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
|
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
|
||||||
github.com/google/uuid v1.1.2 h1:EVhdT+1Kseyi1/pUmXKaFxYsDNy9RQYkMWRH68J/W7Y=
|
github.com/google/s2a-go v0.1.9 h1:LGD7gtMgezd8a/Xak7mEWL0PjoTQFvpRudN895yqKW0=
|
||||||
|
github.com/google/s2a-go v0.1.9/go.mod h1:YA0Ei2ZQL3acow2O62kdp9UlnvMmU7kA6Eutn0dXayM=
|
||||||
github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
|
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||||
|
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||||
|
github.com/googleapis/enterprise-certificate-proxy v0.3.6 h1:GW/XbdyBFQ8Qe+YAmFU9uHLo7OnF5tL52HFAgMmyrf4=
|
||||||
|
github.com/googleapis/enterprise-certificate-proxy v0.3.6/go.mod h1:MkHOF77EYAE7qfSuSS9PU6g4Nt4e11cnsDUowfwewLA=
|
||||||
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
||||||
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
|
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
|
||||||
|
github.com/googleapis/gax-go/v2 v2.14.1 h1:hb0FFeiPaQskmvakKu5EbCbpntQn48jyHuvrkurSS/Q=
|
||||||
|
github.com/googleapis/gax-go/v2 v2.14.1/go.mod h1:Hb/NubMaVM88SrNkvl8X/o8XWwDJEPqouaLeN2IUxoA=
|
||||||
github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g=
|
github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g=
|
||||||
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
@ -223,6 +295,8 @@ github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINE
|
|||||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||||
github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg=
|
github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg=
|
||||||
|
github.com/planetscale/vtprotobuf v0.6.1-0.20240319094008-0393e58bdf10 h1:GFCKgmp0tecUJ0sJuv4pzYCqS9+RGSn52M3FUwPs+uo=
|
||||||
|
github.com/planetscale/vtprotobuf v0.6.1-0.20240319094008-0393e58bdf10/go.mod h1:t/avpk3KcrXxUnYOhZhMXJlSEyie6gQbtLq5NM3loB8=
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||||
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
||||||
@ -230,12 +304,12 @@ github.com/redis/go-redis/v9 v9.19.0 h1:XPVaaPSnG6RhYf7p+rmSa9zZfeVAnWsH5h3lxthO
|
|||||||
github.com/redis/go-redis/v9 v9.19.0/go.mod h1:v/M13XI1PVCDcm01VtPFOADfZtHf8YW3baQf57KlIkA=
|
github.com/redis/go-redis/v9 v9.19.0/go.mod h1:v/M13XI1PVCDcm01VtPFOADfZtHf8YW3baQf57KlIkA=
|
||||||
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
|
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
|
||||||
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
||||||
github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M=
|
github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR38lUII=
|
||||||
github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA=
|
github.com/rogpeppe/go-internal v1.13.1/go.mod h1:uMEvuHeurkdAXX61udpOXGD/AzZDWNMNyH2VO9fmH0o=
|
||||||
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
|
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
|
||||||
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
|
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
|
||||||
github.com/spf13/afero v1.9.5 h1:stMpOSZFs//0Lv29HduCmli3GUfpFoF3Y1Q/aXj/wVM=
|
github.com/spf13/afero v1.10.0 h1:EaGW2JJh15aKOejeuJ+wpFSHnbd7GE6Wvp3TsNhb6LY=
|
||||||
github.com/spf13/afero v1.9.5/go.mod h1:UBogFpq8E9Hx+xc5CNTTEpTnuHVmXDwZcZcE1eb/UhQ=
|
github.com/spf13/afero v1.10.0/go.mod h1:UBogFpq8E9Hx+xc5CNTTEpTnuHVmXDwZcZcE1eb/UhQ=
|
||||||
github.com/spf13/cast v1.5.1 h1:R+kOtfhWQE6TVQzY+4D7wJLBgkdVasCEFxSUBYBYIlA=
|
github.com/spf13/cast v1.5.1 h1:R+kOtfhWQE6TVQzY+4D7wJLBgkdVasCEFxSUBYBYIlA=
|
||||||
github.com/spf13/cast v1.5.1/go.mod h1:b9PdjNptOpzXr7Rq1q9gJML/2cdGQAo69NKzQ10KN48=
|
github.com/spf13/cast v1.5.1/go.mod h1:b9PdjNptOpzXr7Rq1q9gJML/2cdGQAo69NKzQ10KN48=
|
||||||
github.com/spf13/jwalterweatherman v1.1.0 h1:ue6voC5bR5F8YxI5S67j9i582FU4Qvo2bmqnqMYADFk=
|
github.com/spf13/jwalterweatherman v1.1.0 h1:ue6voC5bR5F8YxI5S67j9i582FU4Qvo2bmqnqMYADFk=
|
||||||
@ -244,10 +318,13 @@ github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
|
|||||||
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||||
github.com/spf13/viper v1.16.0 h1:rGGH0XDZhdUOryiDWjmIvUSWpbNqisK8Wk0Vyefw8hc=
|
github.com/spf13/viper v1.16.0 h1:rGGH0XDZhdUOryiDWjmIvUSWpbNqisK8Wk0Vyefw8hc=
|
||||||
github.com/spf13/viper v1.16.0/go.mod h1:yg78JgCJcbrQOvV9YLXgkLaZqUidkY9K+Dd1FofRzQg=
|
github.com/spf13/viper v1.16.0/go.mod h1:yg78JgCJcbrQOvV9YLXgkLaZqUidkY9K+Dd1FofRzQg=
|
||||||
|
github.com/spiffe/go-spiffe/v2 v2.5.0 h1:N2I01KCUkv1FAjZXJMwh95KK1ZIQLYbPfhaxw8WS0hE=
|
||||||
|
github.com/spiffe/go-spiffe/v2 v2.5.0/go.mod h1:P+NxobPc6wXhVtINNtFjNWGBTreew1GBUCwT2wPmb7g=
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
||||||
github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c=
|
|
||||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
|
||||||
|
github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY=
|
||||||
|
github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA=
|
||||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
||||||
@ -257,8 +334,9 @@ github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/
|
|||||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||||
github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk=
|
|
||||||
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
|
||||||
|
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
|
||||||
|
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||||
github.com/subosito/gotenv v1.4.2 h1:X1TuBLAMDFbaTAChgCBLu3DU3UPyELpnF2jjJ2cz/S8=
|
github.com/subosito/gotenv v1.4.2 h1:X1TuBLAMDFbaTAChgCBLu3DU3UPyELpnF2jjJ2cz/S8=
|
||||||
github.com/subosito/gotenv v1.4.2/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0=
|
github.com/subosito/gotenv v1.4.2/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0=
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI=
|
||||||
@ -271,19 +349,41 @@ github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9de
|
|||||||
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||||
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k=
|
||||||
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
||||||
github.com/zeebo/xxh3 v1.1.0 h1:s7DLGDK45Dyfg7++yxI0khrfwq9661w9EN78eP/UZVs=
|
github.com/zeebo/errs v1.4.0 h1:XNdoD/RRMKP7HD0UhJnIzUy74ISdGGxURlYG8HSWSfM=
|
||||||
github.com/zeebo/xxh3 v1.1.0/go.mod h1:IisAie1LELR4xhVinxWS5+zf1lA4p0MW4T+w+W07F5s=
|
github.com/zeebo/errs v1.4.0/go.mod h1:sgbWHsvVuTPHcqJJGQ1WhI5KbWlHYz+2+2C/LSEtCw4=
|
||||||
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
|
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
|
||||||
go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
|
go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
|
||||||
go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
||||||
go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
||||||
go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
||||||
go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk=
|
go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk=
|
||||||
|
go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA=
|
||||||
|
go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A=
|
||||||
|
go.opentelemetry.io/contrib/detectors/gcp v1.35.0 h1:bGvFt68+KTiAKFlacHW6AhA56GF2rS0bdD3aJYEnmzA=
|
||||||
|
go.opentelemetry.io/contrib/detectors/gcp v1.35.0/go.mod h1:qGWP8/+ILwMRIUf9uIVLloR1uo5ZYAslM4O6OqUi1DA=
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.60.0 h1:x7wzEgXfnzJcHDwStJT+mxOz4etr2EcexjqhBvmoakw=
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.60.0/go.mod h1:rg+RlpR5dKwaS95IyyZqj5Wd4E13lk/msnTS0Xl9lJM=
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.60.0 h1:sbiXRNDSWJOTobXh5HyQKjq6wUC5tNybqjIqDpAY4CU=
|
||||||
|
go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.60.0/go.mod h1:69uWxva0WgAA/4bu2Yy70SLDBwZXuQ6PbBpbsa5iZrQ=
|
||||||
|
go.opentelemetry.io/otel v1.35.0 h1:xKWKPxrxB6OtMCbmMY021CqC45J+3Onta9MqjhnusiQ=
|
||||||
|
go.opentelemetry.io/otel v1.35.0/go.mod h1:UEqy8Zp11hpkUrL73gSlELM0DupHoiq72dR+Zqel/+Y=
|
||||||
|
go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.35.0 h1:PB3Zrjs1sG1GBX51SXyTSoOTqcDglmsk7nT6tkKPb/k=
|
||||||
|
go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.35.0/go.mod h1:U2R3XyVPzn0WX7wOIypPuptulsMcPDPs/oiSVOMVnHY=
|
||||||
|
go.opentelemetry.io/otel/metric v1.35.0 h1:0znxYu2SNyuMSQT4Y9WDWej0VpcsxkuklLa4/siN90M=
|
||||||
|
go.opentelemetry.io/otel/metric v1.35.0/go.mod h1:nKVFgxBZ2fReX6IlyW28MgZojkoAkJGaE8CpgeAU3oE=
|
||||||
|
go.opentelemetry.io/otel/sdk v1.35.0 h1:iPctf8iprVySXSKJffSS79eOjl9pvxV9ZqOWT0QejKY=
|
||||||
|
go.opentelemetry.io/otel/sdk v1.35.0/go.mod h1:+ga1bZliga3DxJ3CQGg3updiaAJoNECOgJREo9KHGQg=
|
||||||
|
go.opentelemetry.io/otel/sdk/metric v1.35.0 h1:1RriWBmCKgkeHEhM7a2uMjMUfP7MsOF5JpUCaEqEI9o=
|
||||||
|
go.opentelemetry.io/otel/sdk/metric v1.35.0/go.mod h1:is6XYCUMpcKi+ZsOvfluY5YstFnhW0BidkR+gL+qN+w=
|
||||||
|
go.opentelemetry.io/otel/trace v1.35.0 h1:dPpEfJu1sDIqruz7BHFG3c7528f6ddfSWfFDVt/xgMs=
|
||||||
|
go.opentelemetry.io/otel/trace v1.35.0/go.mod h1:WUk7DtFp1Aw2MkvqGdwiXYDZZNvA/1J8o6xRXLrIkyc=
|
||||||
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||||
go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
|
go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE=
|
||||||
go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
|
go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0=
|
||||||
go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI=
|
go.uber.org/goleak v1.1.11 h1:wy28qYRKZgnJTxGxvye5/wgWr1EKjmUDGYox5mGlRlI=
|
||||||
go.uber.org/goleak v1.1.11/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ=
|
go.uber.org/goleak v1.1.11/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ=
|
||||||
|
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
|
||||||
|
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
|
||||||
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
|
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
|
||||||
go.uber.org/multierr v1.8.0 h1:dg6GjLku4EH+249NNmoIciG9N/jURbDG+pFlTkhzIC8=
|
go.uber.org/multierr v1.8.0 h1:dg6GjLku4EH+249NNmoIciG9N/jURbDG+pFlTkhzIC8=
|
||||||
go.uber.org/multierr v1.8.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak=
|
go.uber.org/multierr v1.8.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak=
|
||||||
@ -301,8 +401,8 @@ golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm
|
|||||||
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
||||||
golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||||
golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58=
|
golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58=
|
||||||
golang.org/x/crypto v0.28.0 h1:GBDwsMXVQi34v5CCYUm2jkJvu4cbtru2U4TN2PSyQnw=
|
golang.org/x/crypto v0.40.0 h1:r4x+VvoG5Fm+eJcxMaY8CQM7Lb0l1lsmjGBQ6s8BfKM=
|
||||||
golang.org/x/crypto v0.28.0/go.mod h1:rmgy+3RHxRZMyY0jjAJShp2zgEdOqj2AO7U0pYmeQ7U=
|
golang.org/x/crypto v0.40.0/go.mod h1:Qr1vMER5WyS2dfPHAlsOj01wgLbsyWtFn/aY+5+ZdxY=
|
||||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||||
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||||
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
|
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
|
||||||
@ -373,8 +473,8 @@ golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96b
|
|||||||
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
|
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
|
||||||
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
|
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
|
||||||
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
|
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
|
||||||
golang.org/x/net v0.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4=
|
golang.org/x/net v0.42.0 h1:jzkYrhi3YQWD6MLBJcsklgQsoAcw89EcZbJw8Z614hs=
|
||||||
golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU=
|
golang.org/x/net v0.42.0/go.mod h1:FF1RA5d3u7nAYA4z2TkclSCKh68eSXtiFwcWQpPXdt8=
|
||||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||||
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||||
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||||
@ -384,6 +484,8 @@ golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ
|
|||||||
golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||||
golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||||
golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||||
|
golang.org/x/oauth2 v0.30.0 h1:dnDm7JmhM45NNpd8FDDeLhK6FwqbOf4MLCM9zb1BOHI=
|
||||||
|
golang.org/x/oauth2 v0.30.0/go.mod h1:B++QgG3ZKulg6sRPGD/mqlHQs5rB3Ml9erfeDY7xKlU=
|
||||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
@ -397,6 +499,8 @@ golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJ
|
|||||||
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||||
|
golang.org/x/sync v0.16.0 h1:ycBJEhp9p4vXvUZNszeOq0kGTPghopOL8q0fq3vstxw=
|
||||||
|
golang.org/x/sync v0.16.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA=
|
||||||
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||||
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||||
@ -440,8 +544,8 @@ golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBc
|
|||||||
golang.org/x/sys v0.0.0-20220908164124-27713097b956/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.0.0-20220908164124-27713097b956/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.30.0 h1:QjkSwP/36a20jFYWkSue1YwXzLmsV5Gfq7Eiy72C1uc=
|
golang.org/x/sys v0.34.0 h1:H5Y5sJ2L2JRdyv7ROF1he/lPdvFsd0mJHFw2ThKHxLA=
|
||||||
golang.org/x/sys v0.30.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
|
golang.org/x/sys v0.34.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
||||||
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
|
||||||
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
||||||
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
||||||
@ -453,12 +557,15 @@ golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
|||||||
golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||||
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
|
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
|
||||||
|
golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ=
|
||||||
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
||||||
golang.org/x/text v0.20.0 h1:gK/Kv2otX8gz+wn7Rmb3vT96ZwuoxnQlY+HlJVj7Qug=
|
golang.org/x/text v0.27.0 h1:4fGWRpyh641NLlecmyl4LOe6yDdfaYNrGb2zdfo4JV4=
|
||||||
golang.org/x/text v0.20.0/go.mod h1:D4IsuqiFMhST5bX19pQ9ikHC2GsaKyk/oF+pn3ducp4=
|
golang.org/x/text v0.27.0/go.mod h1:1D28KMCvyooCX9hBiosv5Tz/+YLxj0j7XhWjpSUF7CU=
|
||||||
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
|
golang.org/x/time v0.11.0 h1:/bpjEDfN9tkoN/ryeYHnv5hcMlc8ncjMcM4XBk5NWV0=
|
||||||
|
golang.org/x/time v0.11.0/go.mod h1:CDIdPxbZBQxdj6cxyCIdrNogrJKMJ7pr37NYpMcMDSg=
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
||||||
@ -531,6 +638,8 @@ google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz513
|
|||||||
google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg=
|
google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg=
|
||||||
google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE=
|
google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE=
|
||||||
google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8=
|
google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8=
|
||||||
|
google.golang.org/api v0.231.0 h1:LbUD5FUl0C4qwia2bjXhCMH65yz1MLPzA/0OYEsYY7Q=
|
||||||
|
google.golang.org/api v0.231.0/go.mod h1:H52180fPI/QQlUc0F4xWfGZILdv09GCWKt2bcsn164A=
|
||||||
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
||||||
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
||||||
google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
||||||
@ -538,6 +647,8 @@ google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww
|
|||||||
google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
||||||
google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
||||||
google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
||||||
|
google.golang.org/appengine/v2 v2.0.6 h1:LvPZLGuchSBslPBp+LAhihBeGSiRh1myRoYK4NtuBIw=
|
||||||
|
google.golang.org/appengine/v2 v2.0.6/go.mod h1:WoEXGoXNfa0mLvaH5sV3ZSGXwVmy8yf7Z1JKf3J3wLI=
|
||||||
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
|
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
|
||||||
google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
||||||
google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
||||||
@ -574,6 +685,12 @@ google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6D
|
|||||||
google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||||
google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||||
google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||||
|
google.golang.org/genproto v0.0.0-20250505200425-f936aa4a68b2 h1:1tXaIXCracvtsRxSBsYDiSBN0cuJvM7QYW+MrpIRY78=
|
||||||
|
google.golang.org/genproto v0.0.0-20250505200425-f936aa4a68b2/go.mod h1:49MsLSx0oWMOZqcpB3uL8ZOkAh1+TndpJ8ONoCBWiZk=
|
||||||
|
google.golang.org/genproto/googleapis/api v0.0.0-20250505200425-f936aa4a68b2 h1:vPV0tzlsK6EzEDHNNH5sa7Hs9bd7iXR7B1tSiPepkV0=
|
||||||
|
google.golang.org/genproto/googleapis/api v0.0.0-20250505200425-f936aa4a68b2/go.mod h1:pKLAc5OolXC3ViWGI62vvC0n10CpwAtRcTNCFwTKBEw=
|
||||||
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250505200425-f936aa4a68b2 h1:IqsN8hx+lWLqlN+Sc3DoMy/watjofWiU8sRFgQ8fhKM=
|
||||||
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20250505200425-f936aa4a68b2/go.mod h1:qQ0YXyHHx3XkvlzUtpXDkS29lDSafHMZBAZDc03LQ3A=
|
||||||
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
|
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
|
||||||
google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
|
google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
|
||||||
google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
|
google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
|
||||||
@ -590,6 +707,8 @@ google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM
|
|||||||
google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc=
|
google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc=
|
||||||
google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8=
|
google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8=
|
||||||
google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU=
|
google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU=
|
||||||
|
google.golang.org/grpc v1.72.0 h1:S7UkcVa60b5AAQTaO6ZKamFp1zMZSU0fGDK2WZLbBnM=
|
||||||
|
google.golang.org/grpc v1.72.0/go.mod h1:wH5Aktxcg25y1I3w7H69nHfXdOG3UiadoBtjh3izSDM=
|
||||||
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
|
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
|
||||||
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
|
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
|
||||||
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
|
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
|
||||||
@ -600,8 +719,10 @@ google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2
|
|||||||
google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
||||||
google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4=
|
google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4=
|
||||||
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
|
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
|
||||||
google.golang.org/protobuf v1.32.0 h1:pPC6BG5ex8PDFnkbrGU3EixyhKcQ2aDuBS36lqK/C7I=
|
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
||||||
google.golang.org/protobuf v1.32.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos=
|
google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I=
|
||||||
|
google.golang.org/protobuf v1.36.6 h1:z1NpPI8ku2WgiWnf+t9wTPsn6eP1L7ksHUlkfLvd9xY=
|
||||||
|
google.golang.org/protobuf v1.36.6/go.mod h1:jduwjTPXsFjZGTmRluh+L6NjiWu7pchiJ2/5YcXBHnY=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
||||||
|
|||||||
@ -1,6 +1,7 @@
|
|||||||
server:
|
server:
|
||||||
base-url:
|
base-url:
|
||||||
local-url:
|
local-url:
|
||||||
|
self-order-url: http://localhost:5173
|
||||||
port: 4000
|
port: 4000
|
||||||
|
|
||||||
jwt:
|
jwt:
|
||||||
@ -28,7 +29,7 @@ postgresql:
|
|||||||
debug: false
|
debug: false
|
||||||
|
|
||||||
redis:
|
redis:
|
||||||
host: 127.0.0.1
|
host: 194.233.78.1
|
||||||
port: 6379
|
port: 6379
|
||||||
password: "CmICdmnX1EZPhVBYzQPEGw==U"
|
password: "CmICdmnX1EZPhVBYzQPEGw==U"
|
||||||
db: 0
|
db: 0
|
||||||
@ -53,4 +54,8 @@ log:
|
|||||||
fonnte:
|
fonnte:
|
||||||
api_url: "https://api.fonnte.com/send"
|
api_url: "https://api.fonnte.com/send"
|
||||||
token: "bADQrf9NTXfLZQCK2wGg"
|
token: "bADQrf9NTXfLZQCK2wGg"
|
||||||
timeout: 30
|
timeout: 30
|
||||||
|
|
||||||
|
fcm:
|
||||||
|
credentials_file: "infra/firebase-service-account.json"
|
||||||
|
project_id: "apskel-pos-v2"
|
||||||
@ -25,11 +25,12 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type App struct {
|
type App struct {
|
||||||
server *http.Server
|
server *http.Server
|
||||||
db *gorm.DB
|
db *gorm.DB
|
||||||
redisClient *redis.Client
|
redisClient *redis.Client
|
||||||
router *router.Router
|
router *router.Router
|
||||||
shutdown chan os.Signal
|
shutdown chan os.Signal
|
||||||
|
omsetScheduler *service.OmsetMilestoneScheduler
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewApp(db *gorm.DB, redisClient *redis.Client) *App {
|
func NewApp(db *gorm.DB, redisClient *redis.Client) *App {
|
||||||
@ -43,6 +44,14 @@ func NewApp(db *gorm.DB, redisClient *redis.Client) *App {
|
|||||||
func (a *App) Initialize(cfg *config.Config) error {
|
func (a *App) Initialize(cfg *config.Config) error {
|
||||||
repos := a.initRepositories()
|
repos := a.initRepositories()
|
||||||
processors := a.initProcessors(cfg, repos)
|
processors := a.initProcessors(cfg, repos)
|
||||||
|
|
||||||
|
// Initialize omset milestone scheduler
|
||||||
|
a.omsetScheduler = service.NewOmsetMilestoneScheduler(
|
||||||
|
repos.organizationRepo,
|
||||||
|
repos.userRepo,
|
||||||
|
processors.notificationProcessor,
|
||||||
|
)
|
||||||
|
|
||||||
services := a.initServices(processors, repos, cfg)
|
services := a.initServices(processors, repos, cfg)
|
||||||
validators := a.initValidators()
|
validators := a.initValidators()
|
||||||
middleware := a.initMiddleware(services, cfg)
|
middleware := a.initMiddleware(services, cfg)
|
||||||
@ -56,6 +65,7 @@ func (a *App) Initialize(cfg *config.Config) error {
|
|||||||
repos.userRepo,
|
repos.userRepo,
|
||||||
repos.sessionRepo,
|
repos.sessionRepo,
|
||||||
repos.orderRepo,
|
repos.orderRepo,
|
||||||
|
services.productOutletPriceService,
|
||||||
)
|
)
|
||||||
|
|
||||||
a.router = router.NewRouter(
|
a.router = router.NewRouter(
|
||||||
@ -118,6 +128,12 @@ func (a *App) Initialize(cfg *config.Config) error {
|
|||||||
services.customerPointsService,
|
services.customerPointsService,
|
||||||
services.spinGameService,
|
services.spinGameService,
|
||||||
middleware.customerAuthMiddleware,
|
middleware.customerAuthMiddleware,
|
||||||
|
services.userDeviceService,
|
||||||
|
validators.userDeviceValidator,
|
||||||
|
services.notificationService,
|
||||||
|
validators.notificationValidator,
|
||||||
|
services.productOutletPriceService,
|
||||||
|
validators.productOutletPriceValidator,
|
||||||
selfOrderHandler,
|
selfOrderHandler,
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -125,6 +141,11 @@ func (a *App) Initialize(cfg *config.Config) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (a *App) Start(port string) error {
|
func (a *App) Start(port string) error {
|
||||||
|
// Start the omset milestone scheduler (checks every hour)
|
||||||
|
if a.omsetScheduler != nil {
|
||||||
|
a.omsetScheduler.Start(1 * time.Hour)
|
||||||
|
}
|
||||||
|
|
||||||
engine := a.router.Init()
|
engine := a.router.Init()
|
||||||
|
|
||||||
a.server = &http.Server{
|
a.server = &http.Server{
|
||||||
@ -160,6 +181,9 @@ func (a *App) Start(port string) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (a *App) Shutdown() {
|
func (a *App) Shutdown() {
|
||||||
|
if a.omsetScheduler != nil {
|
||||||
|
a.omsetScheduler.Stop()
|
||||||
|
}
|
||||||
close(a.shutdown)
|
close(a.shutdown)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -207,6 +231,11 @@ type repositories struct {
|
|||||||
otpRepo repository.OtpRepository
|
otpRepo repository.OtpRepository
|
||||||
sessionRepo repository.SessionRepository
|
sessionRepo repository.SessionRepository
|
||||||
txManager *repository.TxManager
|
txManager *repository.TxManager
|
||||||
|
userDeviceRepo *repository.UserDeviceRepositoryImpl
|
||||||
|
notificationRepo *repository.NotificationRepositoryImpl
|
||||||
|
notificationReceiverRepo *repository.NotificationReceiverRepositoryImpl
|
||||||
|
notificationDeliveryRepo *repository.NotificationDeliveryRepositoryImpl
|
||||||
|
productOutletPriceRepo *repository.ProductOutletPriceRepositoryImpl
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *App) initRepositories() *repositories {
|
func (a *App) initRepositories() *repositories {
|
||||||
@ -254,6 +283,11 @@ func (a *App) initRepositories() *repositories {
|
|||||||
otpRepo: repository.NewOtpRepository(a.db),
|
otpRepo: repository.NewOtpRepository(a.db),
|
||||||
sessionRepo: repository.NewSessionRepository(a.redisClient),
|
sessionRepo: repository.NewSessionRepository(a.redisClient),
|
||||||
txManager: repository.NewTxManager(a.db),
|
txManager: repository.NewTxManager(a.db),
|
||||||
|
userDeviceRepo: repository.NewUserDeviceRepositoryImpl(a.db),
|
||||||
|
notificationRepo: repository.NewNotificationRepository(a.db),
|
||||||
|
notificationReceiverRepo: repository.NewNotificationReceiverRepository(a.db),
|
||||||
|
notificationDeliveryRepo: repository.NewNotificationDeliveryRepository(a.db),
|
||||||
|
productOutletPriceRepo: repository.NewProductOutletPriceRepositoryImpl(a.db),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -296,6 +330,9 @@ type processors struct {
|
|||||||
otpProcessor processor.OtpProcessor
|
otpProcessor processor.OtpProcessor
|
||||||
fileClient processor.FileClient
|
fileClient processor.FileClient
|
||||||
inventoryMovementService service.InventoryMovementService
|
inventoryMovementService service.InventoryMovementService
|
||||||
|
userDeviceProcessor *processor.UserDeviceProcessorImpl
|
||||||
|
notificationProcessor *processor.NotificationProcessorImpl
|
||||||
|
productOutletPriceProcessor processor.ProductOutletPriceProcessor
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processors {
|
func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processors {
|
||||||
@ -310,10 +347,10 @@ func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processor
|
|||||||
outletProcessor: processor.NewOutletProcessorImpl(repos.outletRepo),
|
outletProcessor: processor.NewOutletProcessorImpl(repos.outletRepo),
|
||||||
outletSettingProcessor: processor.NewOutletSettingProcessorImpl(repos.outletSettingRepo, repos.outletRepo),
|
outletSettingProcessor: processor.NewOutletSettingProcessorImpl(repos.outletSettingRepo, repos.outletRepo),
|
||||||
categoryProcessor: processor.NewCategoryProcessorImpl(repos.categoryRepo),
|
categoryProcessor: processor.NewCategoryProcessorImpl(repos.categoryRepo),
|
||||||
productProcessor: processor.NewProductProcessorImpl(repos.productRepo, repos.categoryRepo, repos.productVariantRepo, repos.inventoryRepo, repos.outletRepo),
|
productProcessor: processor.NewProductProcessorImpl(repos.productRepo, repos.categoryRepo, repos.productVariantRepo, repos.inventoryRepo, repos.outletRepo, repos.productOutletPriceRepo),
|
||||||
productVariantProcessor: processor.NewProductVariantProcessorImpl(repos.productVariantRepo, repos.productRepo),
|
productVariantProcessor: processor.NewProductVariantProcessorImpl(repos.productVariantRepo, repos.productRepo),
|
||||||
inventoryProcessor: processor.NewInventoryProcessorImpl(repos.inventoryRepo, repos.productRepo, repos.outletRepo, repos.ingredientRepo, repos.inventoryMovementRepo),
|
inventoryProcessor: processor.NewInventoryProcessorImpl(repos.inventoryRepo, repos.productRepo, repos.outletRepo, repos.ingredientRepo, repos.inventoryMovementRepo),
|
||||||
orderProcessor: processor.NewOrderProcessorImpl(repos.orderRepo, repos.orderItemRepo, repos.paymentRepo, repos.paymentOrderItemRepo, repos.productRepo, repos.paymentMethodRepo, repos.inventoryRepo, repos.inventoryMovementRepo, repos.productVariantRepo, repos.outletRepo, repos.customerRepo, repos.txManager, repos.productRecipeRepo, repos.ingredientRepo, inventoryMovementService),
|
orderProcessor: processor.NewOrderProcessorImpl(repos.orderRepo, repos.orderItemRepo, repos.paymentRepo, repos.paymentOrderItemRepo, repos.productRepo, repos.paymentMethodRepo, repos.inventoryRepo, repos.inventoryMovementRepo, repos.productVariantRepo, repos.outletRepo, repos.customerRepo, repos.txManager, repos.productRecipeRepo, repos.ingredientRepo, inventoryMovementService, repos.productOutletPriceRepo),
|
||||||
paymentMethodProcessor: processor.NewPaymentMethodProcessorImpl(repos.paymentMethodRepo),
|
paymentMethodProcessor: processor.NewPaymentMethodProcessorImpl(repos.paymentMethodRepo),
|
||||||
fileProcessor: processor.NewFileProcessorImpl(repos.fileRepo, fileClient),
|
fileProcessor: processor.NewFileProcessorImpl(repos.fileRepo, fileClient),
|
||||||
customerProcessor: processor.NewCustomerProcessor(repos.customerRepo),
|
customerProcessor: processor.NewCustomerProcessor(repos.customerRepo),
|
||||||
@ -343,6 +380,9 @@ func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processor
|
|||||||
otpProcessor: otpProcessor,
|
otpProcessor: otpProcessor,
|
||||||
fileClient: fileClient,
|
fileClient: fileClient,
|
||||||
inventoryMovementService: inventoryMovementService,
|
inventoryMovementService: inventoryMovementService,
|
||||||
|
userDeviceProcessor: processor.NewUserDeviceProcessorImpl(repos.userDeviceRepo),
|
||||||
|
notificationProcessor: buildNotificationProcessor(cfg, repos),
|
||||||
|
productOutletPriceProcessor: processor.NewProductOutletPriceProcessorImpl(repos.productOutletPriceRepo, repos.productRepo, repos.outletRepo),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -379,11 +419,14 @@ type services struct {
|
|||||||
customerAuthService service.CustomerAuthService
|
customerAuthService service.CustomerAuthService
|
||||||
customerPointsService service.CustomerPointsService
|
customerPointsService service.CustomerPointsService
|
||||||
spinGameService service.SpinGameService
|
spinGameService service.SpinGameService
|
||||||
|
userDeviceService service.UserDeviceService
|
||||||
|
notificationService service.NotificationService
|
||||||
|
productOutletPriceService service.ProductOutletPriceService
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *App) initServices(processors *processors, repos *repositories, cfg *config.Config) *services {
|
func (a *App) initServices(processors *processors, repos *repositories, cfg *config.Config) *services {
|
||||||
authConfig := cfg.Auth()
|
authConfig := cfg.Auth()
|
||||||
authService := service.NewAuthService(processors.userProcessor, authConfig)
|
authService := service.NewAuthService(processors.userProcessor, processors.userDeviceProcessor, authConfig)
|
||||||
organizationService := service.NewOrganizationService(processors.organizationProcessor)
|
organizationService := service.NewOrganizationService(processors.organizationProcessor)
|
||||||
outletService := service.NewOutletService(processors.outletProcessor)
|
outletService := service.NewOutletService(processors.outletProcessor)
|
||||||
outletSettingService := service.NewOutletSettingService(processors.outletSettingProcessor)
|
outletSettingService := service.NewOutletSettingService(processors.outletSettingProcessor)
|
||||||
@ -391,7 +434,7 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con
|
|||||||
productService := service.NewProductService(processors.productProcessor)
|
productService := service.NewProductService(processors.productProcessor)
|
||||||
productVariantService := service.NewProductVariantService(processors.productVariantProcessor)
|
productVariantService := service.NewProductVariantService(processors.productVariantProcessor)
|
||||||
inventoryService := service.NewInventoryService(processors.inventoryProcessor)
|
inventoryService := service.NewInventoryService(processors.inventoryProcessor)
|
||||||
orderService := service.NewOrderServiceImpl(processors.orderProcessor, repos.tableRepo, nil, processors.orderIngredientTransactionProcessor, *repos.productRecipeRepo, repos.txManager, repos.sessionRepo) // Will be updated after orderIngredientTransactionService is created
|
orderService := service.NewOrderServiceImpl(processors.orderProcessor, repos.tableRepo, nil, processors.orderIngredientTransactionProcessor, *repos.productRecipeRepo, repos.txManager, repos.sessionRepo, processors.notificationProcessor, repos.userRepo) // Will be updated after orderIngredientTransactionService is created
|
||||||
paymentMethodService := service.NewPaymentMethodService(processors.paymentMethodProcessor)
|
paymentMethodService := service.NewPaymentMethodService(processors.paymentMethodProcessor)
|
||||||
fileService := service.NewFileServiceImpl(processors.fileProcessor)
|
fileService := service.NewFileServiceImpl(processors.fileProcessor)
|
||||||
var customerService service.CustomerService = service.NewCustomerService(processors.customerProcessor)
|
var customerService service.CustomerService = service.NewCustomerService(processors.customerProcessor)
|
||||||
@ -414,9 +457,11 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con
|
|||||||
customerAuthService := service.NewCustomerAuthService(processors.customerAuthProcessor)
|
customerAuthService := service.NewCustomerAuthService(processors.customerAuthProcessor)
|
||||||
customerPointsService := service.NewCustomerPointsService(processors.customerPointsProcessor)
|
customerPointsService := service.NewCustomerPointsService(processors.customerPointsProcessor)
|
||||||
spinGameService := service.NewSpinGameService(processors.gamePlayProcessor, repos.txManager)
|
spinGameService := service.NewSpinGameService(processors.gamePlayProcessor, repos.txManager)
|
||||||
|
userDeviceService := service.NewUserDeviceService(processors.userDeviceProcessor)
|
||||||
|
notificationService := service.NewNotificationService(processors.notificationProcessor)
|
||||||
|
|
||||||
// Update order service with order ingredient transaction service
|
// Update order service with order ingredient transaction service
|
||||||
orderService = service.NewOrderServiceImpl(processors.orderProcessor, repos.tableRepo, orderIngredientTransactionService, processors.orderIngredientTransactionProcessor, *repos.productRecipeRepo, repos.txManager, repos.sessionRepo)
|
orderService = service.NewOrderServiceImpl(processors.orderProcessor, repos.tableRepo, orderIngredientTransactionService, processors.orderIngredientTransactionProcessor, *repos.productRecipeRepo, repos.txManager, repos.sessionRepo, processors.notificationProcessor, repos.userRepo)
|
||||||
|
|
||||||
return &services{
|
return &services{
|
||||||
userService: service.NewUserService(processors.userProcessor),
|
userService: service.NewUserService(processors.userProcessor),
|
||||||
@ -451,6 +496,9 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con
|
|||||||
customerAuthService: customerAuthService,
|
customerAuthService: customerAuthService,
|
||||||
customerPointsService: customerPointsService,
|
customerPointsService: customerPointsService,
|
||||||
spinGameService: spinGameService,
|
spinGameService: spinGameService,
|
||||||
|
userDeviceService: userDeviceService,
|
||||||
|
notificationService: notificationService,
|
||||||
|
productOutletPriceService: service.NewProductOutletPriceService(processors.productOutletPriceProcessor),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -490,6 +538,9 @@ type validators struct {
|
|||||||
rewardValidator validator.RewardValidator
|
rewardValidator validator.RewardValidator
|
||||||
campaignValidator validator.CampaignValidator
|
campaignValidator validator.CampaignValidator
|
||||||
customerAuthValidator validator.CustomerAuthValidator
|
customerAuthValidator validator.CustomerAuthValidator
|
||||||
|
userDeviceValidator *validator.UserDeviceValidatorImpl
|
||||||
|
notificationValidator *validator.NotificationValidatorImpl
|
||||||
|
productOutletPriceValidator *validator.ProductOutletPriceValidatorImpl
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *App) initValidators() *validators {
|
func (a *App) initValidators() *validators {
|
||||||
@ -517,5 +568,31 @@ func (a *App) initValidators() *validators {
|
|||||||
rewardValidator: validator.NewRewardValidator(),
|
rewardValidator: validator.NewRewardValidator(),
|
||||||
campaignValidator: validator.NewCampaignValidator(),
|
campaignValidator: validator.NewCampaignValidator(),
|
||||||
customerAuthValidator: validator.NewCustomerAuthValidator(),
|
customerAuthValidator: validator.NewCustomerAuthValidator(),
|
||||||
|
userDeviceValidator: validator.NewUserDeviceValidator(),
|
||||||
|
notificationValidator: validator.NewNotificationValidator(),
|
||||||
|
productOutletPriceValidator: validator.NewProductOutletPriceValidator(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// buildNotificationProcessor creates the notification processor with FCM integration.
|
||||||
|
// If FCM is not configured, it returns a processor with a nil FCM client (FCM dispatch will be skipped).
|
||||||
|
func buildNotificationProcessor(cfg *config.Config, repos *repositories) *processor.NotificationProcessorImpl {
|
||||||
|
var fcmClient client.FCMClient
|
||||||
|
if cfg.FCM.CredentialsFile != "" {
|
||||||
|
var err error
|
||||||
|
fcmClient, err = client.NewFCMClient(&cfg.FCM)
|
||||||
|
if err != nil {
|
||||||
|
// FCM init failure is non-fatal; notifications will still be persisted.
|
||||||
|
fcmClient = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return processor.NewNotificationProcessor(
|
||||||
|
repos.notificationRepo,
|
||||||
|
repos.notificationReceiverRepo,
|
||||||
|
repos.notificationDeliveryRepo,
|
||||||
|
repos.userDeviceRepo,
|
||||||
|
repos.userRepo,
|
||||||
|
fcmClient,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|||||||
142
internal/client/fcm_client.go
Normal file
142
internal/client/fcm_client.go
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
package client
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
firebase "firebase.google.com/go/v4"
|
||||||
|
"firebase.google.com/go/v4/messaging"
|
||||||
|
"google.golang.org/api/option"
|
||||||
|
)
|
||||||
|
|
||||||
|
type FCMConfig interface {
|
||||||
|
GetCredentialsFile() string
|
||||||
|
GetProjectID() string
|
||||||
|
}
|
||||||
|
|
||||||
|
type FCMClient interface {
|
||||||
|
SendNotification(ctx context.Context, token string, title string, body string, data map[string]string) error
|
||||||
|
SendMulticastNotification(ctx context.Context, tokens []string, title string, body string, data map[string]string) error
|
||||||
|
SendToTopic(ctx context.Context, topic string, title string, body string, data map[string]string) error
|
||||||
|
}
|
||||||
|
|
||||||
|
type fcmClient struct {
|
||||||
|
messaging *messaging.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewFCMClient(cfg FCMConfig) (FCMClient, error) {
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
opt := option.WithCredentialsFile(cfg.GetCredentialsFile())
|
||||||
|
|
||||||
|
app, err := firebase.NewApp(ctx, &firebase.Config{
|
||||||
|
ProjectID: cfg.GetProjectID(),
|
||||||
|
}, opt)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to initialize firebase app: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
msgClient, err := app.Messaging(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to initialize firebase messaging client: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &fcmClient{
|
||||||
|
messaging: msgClient,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendNotification sends a push notification to a single device token.
|
||||||
|
func (f *fcmClient) SendNotification(ctx context.Context, token string, title string, body string, data map[string]string) error {
|
||||||
|
message := &messaging.Message{
|
||||||
|
Token: token,
|
||||||
|
Notification: &messaging.Notification{
|
||||||
|
Title: title,
|
||||||
|
Body: body,
|
||||||
|
},
|
||||||
|
Data: data,
|
||||||
|
Android: &messaging.AndroidConfig{
|
||||||
|
Priority: "high",
|
||||||
|
},
|
||||||
|
APNS: &messaging.APNSConfig{
|
||||||
|
Payload: &messaging.APNSPayload{
|
||||||
|
Aps: &messaging.Aps{
|
||||||
|
Sound: "default",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := f.messaging.Send(ctx, message)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to send FCM notification: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendMulticastNotification sends a push notification to multiple device tokens.
|
||||||
|
func (f *fcmClient) SendMulticastNotification(ctx context.Context, tokens []string, title string, body string, data map[string]string) error {
|
||||||
|
if len(tokens) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
message := &messaging.MulticastMessage{
|
||||||
|
Tokens: tokens,
|
||||||
|
Notification: &messaging.Notification{
|
||||||
|
Title: title,
|
||||||
|
Body: body,
|
||||||
|
},
|
||||||
|
Data: data,
|
||||||
|
Android: &messaging.AndroidConfig{
|
||||||
|
Priority: "high",
|
||||||
|
},
|
||||||
|
APNS: &messaging.APNSConfig{
|
||||||
|
Payload: &messaging.APNSPayload{
|
||||||
|
Aps: &messaging.Aps{
|
||||||
|
Sound: "default",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
batchResp, err := f.messaging.SendEachForMulticast(ctx, message)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to send FCM multicast notification: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if batchResp.FailureCount > 0 {
|
||||||
|
return fmt.Errorf("FCM multicast: %d/%d messages failed to send", batchResp.FailureCount, len(tokens))
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendToTopic sends a push notification to all devices subscribed to a topic.
|
||||||
|
func (f *fcmClient) SendToTopic(ctx context.Context, topic string, title string, body string, data map[string]string) error {
|
||||||
|
message := &messaging.Message{
|
||||||
|
Topic: topic,
|
||||||
|
Notification: &messaging.Notification{
|
||||||
|
Title: title,
|
||||||
|
Body: body,
|
||||||
|
},
|
||||||
|
Data: data,
|
||||||
|
Android: &messaging.AndroidConfig{
|
||||||
|
Priority: "high",
|
||||||
|
},
|
||||||
|
APNS: &messaging.APNSConfig{
|
||||||
|
Payload: &messaging.APNSPayload{
|
||||||
|
Aps: &messaging.Aps{
|
||||||
|
Sound: "default",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := f.messaging.Send(ctx, message)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to send FCM topic notification: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@ -44,18 +44,22 @@ const (
|
|||||||
IngredientCompositionServiceEntity = "ingredient_composition_service"
|
IngredientCompositionServiceEntity = "ingredient_composition_service"
|
||||||
TableEntity = "table"
|
TableEntity = "table"
|
||||||
// Gamification entities
|
// Gamification entities
|
||||||
CustomerPointsEntity = "customer_points"
|
CustomerPointsEntity = "customer_points"
|
||||||
CustomerTokensEntity = "customer_tokens"
|
CustomerTokensEntity = "customer_tokens"
|
||||||
TierEntity = "tier"
|
TierEntity = "tier"
|
||||||
GameEntity = "game"
|
GameEntity = "game"
|
||||||
GamePrizeEntity = "game_prize"
|
GamePrizeEntity = "game_prize"
|
||||||
GamePlayEntity = "game_play"
|
GamePlayEntity = "game_play"
|
||||||
OmsetTrackerEntity = "omset_tracker"
|
OmsetTrackerEntity = "omset_tracker"
|
||||||
RewardEntity = "reward"
|
RewardEntity = "reward"
|
||||||
CampaignEntity = "campaign"
|
CampaignEntity = "campaign"
|
||||||
CampaignRuleEntity = "campaign_rule"
|
CampaignRuleEntity = "campaign_rule"
|
||||||
CustomerEntity = "customer"
|
CustomerEntity = "customer"
|
||||||
SpinGameHandlerEntity = "spin_game_handler"
|
SpinGameHandlerEntity = "spin_game_handler"
|
||||||
|
UserDeviceServiceEntity = "user_device_service"
|
||||||
|
NotificationServiceEntity = "notification_service"
|
||||||
|
NotificationHandlerEntity = "notification_handler"
|
||||||
|
ProductOutletPriceServiceEntity = "product_outlet_price_service"
|
||||||
)
|
)
|
||||||
|
|
||||||
var HttpErrorMap = map[string]int{
|
var HttpErrorMap = map[string]int{
|
||||||
|
|||||||
@ -7,6 +7,7 @@ const (
|
|||||||
RoleManager UserRole = "manager"
|
RoleManager UserRole = "manager"
|
||||||
RoleCashier UserRole = "cashier"
|
RoleCashier UserRole = "cashier"
|
||||||
RoleWaiter UserRole = "waiter"
|
RoleWaiter UserRole = "waiter"
|
||||||
|
RoleOwner UserRole = "owner"
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetAllUserRoles() []UserRole {
|
func GetAllUserRoles() []UserRole {
|
||||||
@ -15,6 +16,7 @@ func GetAllUserRoles() []UserRole {
|
|||||||
RoleManager,
|
RoleManager,
|
||||||
RoleCashier,
|
RoleCashier,
|
||||||
RoleWaiter,
|
RoleWaiter,
|
||||||
|
RoleOwner,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -7,11 +7,11 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type PaymentMethodAnalyticsRequest struct {
|
type PaymentMethodAnalyticsRequest struct {
|
||||||
OrganizationID uuid.UUID `form:"organization_id"`
|
OrganizationID uuid.UUID `form:"organization_id"`
|
||||||
OutletID *uuid.UUID `form:"outlet_id,omitempty"`
|
OutletID *string `form:"outlet_id,omitempty"`
|
||||||
DateFrom string `form:"date_from" validate:"required"`
|
DateFrom string `form:"date_from" validate:"required"`
|
||||||
DateTo string `form:"date_to" validate:"required"`
|
DateTo string `form:"date_to" validate:"required"`
|
||||||
GroupBy string `form:"group_by,default=day" validate:"omitempty,oneof=day hour week month"`
|
GroupBy string `form:"group_by,default=day" validate:"omitempty,oneof=day hour week month"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// PaymentMethodAnalyticsResponse represents the response for payment method analytics
|
// PaymentMethodAnalyticsResponse represents the response for payment method analytics
|
||||||
@ -45,10 +45,10 @@ type PaymentMethodAnalyticsData struct {
|
|||||||
|
|
||||||
type SalesAnalyticsRequest struct {
|
type SalesAnalyticsRequest struct {
|
||||||
OrganizationID uuid.UUID
|
OrganizationID uuid.UUID
|
||||||
OutletID *uuid.UUID `form:"outlet_id,omitempty"`
|
OutletID *string `form:"outlet_id,omitempty"`
|
||||||
DateFrom string `form:"date_from" validate:"required"`
|
DateFrom string `form:"date_from" validate:"required"`
|
||||||
DateTo string `form:"date_to" validate:"required"`
|
DateTo string `form:"date_to" validate:"required"`
|
||||||
GroupBy string `form:"group_by,default=day" validate:"omitempty,oneof=day hour week month"`
|
GroupBy string `form:"group_by,default=day" validate:"omitempty,oneof=day hour week month"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type SalesAnalyticsResponse struct {
|
type SalesAnalyticsResponse struct {
|
||||||
@ -86,10 +86,10 @@ type SalesAnalyticsData struct {
|
|||||||
// ProductAnalyticsRequest represents the request for product analytics
|
// ProductAnalyticsRequest represents the request for product analytics
|
||||||
type ProductAnalyticsRequest struct {
|
type ProductAnalyticsRequest struct {
|
||||||
OrganizationID uuid.UUID
|
OrganizationID uuid.UUID
|
||||||
OutletID *uuid.UUID `form:"outlet_id,omitempty"`
|
OutletID *string `form:"outlet_id,omitempty"`
|
||||||
DateFrom string `form:"date_from" validate:"required"`
|
DateFrom string `form:"date_from" validate:"required"`
|
||||||
DateTo string `form:"date_to" validate:"required"`
|
DateTo string `form:"date_to" validate:"required"`
|
||||||
Limit int `form:"limit,default=1000" validate:"min=1,max=1000"`
|
Limit int `form:"limit,default=1000" validate:"min=1,max=1000"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProductAnalyticsResponse represents the response for product analytics
|
// ProductAnalyticsResponse represents the response for product analytics
|
||||||
@ -123,9 +123,9 @@ type ProductAnalyticsData struct {
|
|||||||
// ProductAnalyticsPerCategoryRequest represents the request for product analytics per category
|
// ProductAnalyticsPerCategoryRequest represents the request for product analytics per category
|
||||||
type ProductAnalyticsPerCategoryRequest struct {
|
type ProductAnalyticsPerCategoryRequest struct {
|
||||||
OrganizationID uuid.UUID
|
OrganizationID uuid.UUID
|
||||||
OutletID *uuid.UUID `form:"outlet_id,omitempty"`
|
OutletID *string `form:"outlet_id,omitempty"`
|
||||||
DateFrom string `form:"date_from" validate:"required"`
|
DateFrom string `form:"date_from" validate:"required"`
|
||||||
DateTo string `form:"date_to" validate:"required"`
|
DateTo string `form:"date_to" validate:"required"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProductAnalyticsPerCategoryResponse represents the response for product analytics per category
|
// ProductAnalyticsPerCategoryResponse represents the response for product analytics per category
|
||||||
@ -152,9 +152,9 @@ type ProductAnalyticsPerCategoryData struct {
|
|||||||
// DashboardAnalyticsRequest represents the request for dashboard analytics
|
// DashboardAnalyticsRequest represents the request for dashboard analytics
|
||||||
type DashboardAnalyticsRequest struct {
|
type DashboardAnalyticsRequest struct {
|
||||||
OrganizationID uuid.UUID
|
OrganizationID uuid.UUID
|
||||||
OutletID *uuid.UUID `form:"outlet_id,omitempty"`
|
OutletID *string `form:"outlet_id,omitempty"`
|
||||||
DateFrom string `form:"date_from" validate:"required"`
|
DateFrom string `form:"date_from" validate:"required"`
|
||||||
DateTo string `form:"date_to" validate:"required"`
|
DateTo string `form:"date_to" validate:"required"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// DashboardAnalyticsResponse represents the response for dashboard analytics
|
// DashboardAnalyticsResponse represents the response for dashboard analytics
|
||||||
@ -182,10 +182,10 @@ type DashboardOverview struct {
|
|||||||
// ProfitLossAnalyticsRequest represents the request for profit and loss analytics
|
// ProfitLossAnalyticsRequest represents the request for profit and loss analytics
|
||||||
type ProfitLossAnalyticsRequest struct {
|
type ProfitLossAnalyticsRequest struct {
|
||||||
OrganizationID uuid.UUID
|
OrganizationID uuid.UUID
|
||||||
OutletID *uuid.UUID `form:"outlet_id,omitempty"`
|
OutletID *string `form:"outlet_id,omitempty"`
|
||||||
DateFrom string `form:"date_from" validate:"required"`
|
DateFrom string `form:"date_from" validate:"required"`
|
||||||
DateTo string `form:"date_to" validate:"required"`
|
DateTo string `form:"date_to" validate:"required"`
|
||||||
GroupBy string `form:"group_by,default=day" validate:"omitempty,oneof=day hour week month"`
|
GroupBy string `form:"group_by,default=day" validate:"omitempty,oneof=day hour week month"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProfitLossAnalyticsResponse represents the response for profit and loss analytics
|
// ProfitLossAnalyticsResponse represents the response for profit and loss analytics
|
||||||
|
|||||||
92
internal/contract/notification_contract.go
Normal file
92
internal/contract/notification_contract.go
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
package contract
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ---- Request contracts ----
|
||||||
|
|
||||||
|
type SendNotificationRequest struct {
|
||||||
|
Title string `json:"title" validate:"required,min=1,max=255"`
|
||||||
|
Body string `json:"body" validate:"required"`
|
||||||
|
Type string `json:"type,omitempty" validate:"omitempty,max=100"`
|
||||||
|
Category string `json:"category,omitempty" validate:"omitempty,max=100"`
|
||||||
|
Priority entities.NotificationPriority `json:"priority,omitempty" validate:"omitempty,oneof=low normal high"`
|
||||||
|
ImageURL string `json:"image_url,omitempty" validate:"omitempty,max=512"`
|
||||||
|
ActionURL string `json:"action_url,omitempty" validate:"omitempty,max=512"`
|
||||||
|
NotifiableType string `json:"notifiable_type,omitempty" validate:"omitempty,max=100"`
|
||||||
|
NotifiableID *uuid.UUID `json:"notifiable_id,omitempty"`
|
||||||
|
Data map[string]interface{} `json:"data,omitempty"`
|
||||||
|
ReceiverIDs []uuid.UUID `json:"receiver_ids" validate:"required,min=1"`
|
||||||
|
ScheduledAt *time.Time `json:"scheduled_at,omitempty"`
|
||||||
|
ExpiredAt *time.Time `json:"expired_at,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type BroadcastNotificationRequest struct {
|
||||||
|
Title string `json:"title" validate:"required,min=1,max=255"`
|
||||||
|
Body string `json:"body" validate:"required"`
|
||||||
|
Type string `json:"type,omitempty" validate:"omitempty,max=100"`
|
||||||
|
Category string `json:"category,omitempty" validate:"omitempty,max=100"`
|
||||||
|
Priority entities.NotificationPriority `json:"priority,omitempty" validate:"omitempty,oneof=low normal high"`
|
||||||
|
ImageURL string `json:"image_url,omitempty" validate:"omitempty,max=512"`
|
||||||
|
ActionURL string `json:"action_url,omitempty" validate:"omitempty,max=512"`
|
||||||
|
NotifiableType string `json:"notifiable_type,omitempty" validate:"omitempty,max=100"`
|
||||||
|
NotifiableID *uuid.UUID `json:"notifiable_id,omitempty"`
|
||||||
|
Data map[string]interface{} `json:"data,omitempty"`
|
||||||
|
ScheduledAt *time.Time `json:"scheduled_at,omitempty"`
|
||||||
|
ExpiredAt *time.Time `json:"expired_at,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListNotificationsRequest struct {
|
||||||
|
Page int `form:"page" validate:"min=1"`
|
||||||
|
Limit int `form:"limit" validate:"min=1,max=100"`
|
||||||
|
IsRead *bool `form:"is_read"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---- Response contracts ----
|
||||||
|
|
||||||
|
type NotificationResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Body string `json:"body"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
Category string `json:"category"`
|
||||||
|
Priority entities.NotificationPriority `json:"priority"`
|
||||||
|
ImageURL string `json:"image_url"`
|
||||||
|
ActionURL string `json:"action_url"`
|
||||||
|
NotifiableType string `json:"notifiable_type"`
|
||||||
|
NotifiableID *uuid.UUID `json:"notifiable_id"`
|
||||||
|
Data map[string]interface{} `json:"data"`
|
||||||
|
ScheduledAt *time.Time `json:"scheduled_at"`
|
||||||
|
SentAt *time.Time `json:"sent_at"`
|
||||||
|
ExpiredAt *time.Time `json:"expired_at"`
|
||||||
|
CreatedBy *uuid.UUID `json:"created_by"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationReceiverResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
NotificationID uuid.UUID `json:"notification_id"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
IsRead bool `json:"is_read"`
|
||||||
|
ReadAt *time.Time `json:"read_at"`
|
||||||
|
IsDeleted bool `json:"is_deleted"`
|
||||||
|
DeletedAt *time.Time `json:"deleted_at"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
Notification *NotificationResponse `json:"notification,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListNotificationsResponse struct {
|
||||||
|
Notifications []*NotificationReceiverResponse `json:"notifications"`
|
||||||
|
TotalCount int64 `json:"total_count"`
|
||||||
|
UnreadCount int64 `json:"unread_count"`
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
TotalPages int `json:"total_pages"`
|
||||||
|
}
|
||||||
@ -56,24 +56,26 @@ type UpdateProductVariantRequest struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type ProductResponse struct {
|
type ProductResponse struct {
|
||||||
ID uuid.UUID `json:"id"`
|
ID uuid.UUID `json:"id"`
|
||||||
OrganizationID uuid.UUID `json:"organization_id"`
|
OrganizationID uuid.UUID `json:"organization_id"`
|
||||||
CategoryID uuid.UUID `json:"category_id"`
|
CategoryID uuid.UUID `json:"category_id"`
|
||||||
CategoryName string `json:"category_name"`
|
CategoryName string `json:"category_name"`
|
||||||
SKU *string `json:"sku"`
|
SKU *string `json:"sku"`
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Description *string `json:"description"`
|
Description *string `json:"description"`
|
||||||
Price float64 `json:"price"`
|
Price float64 `json:"price"`
|
||||||
Cost float64 `json:"cost"`
|
OutletPrice *float64 `json:"outlet_price,omitempty"`
|
||||||
BusinessType string `json:"business_type"`
|
OutletPrices []ProductOutletPriceResponse `json:"outlet_prices,omitempty"`
|
||||||
ImageURL *string `json:"image_url"`
|
Cost float64 `json:"cost"`
|
||||||
PrinterType string `json:"printer_type"`
|
BusinessType string `json:"business_type"`
|
||||||
Metadata map[string]interface{} `json:"metadata"`
|
ImageURL *string `json:"image_url"`
|
||||||
IsActive bool `json:"is_active"`
|
PrinterType string `json:"printer_type"`
|
||||||
CreatedAt time.Time `json:"created_at"`
|
Metadata map[string]interface{} `json:"metadata"`
|
||||||
UpdatedAt time.Time `json:"updated_at"`
|
IsActive bool `json:"is_active"`
|
||||||
Category *CategoryResponse `json:"category,omitempty"`
|
CreatedAt time.Time `json:"created_at"`
|
||||||
Variants []ProductVariantResponse `json:"variants,omitempty"`
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
Category *CategoryResponse `json:"category,omitempty"`
|
||||||
|
Variants []ProductVariantResponse `json:"variants,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ProductVariantResponse struct {
|
type ProductVariantResponse struct {
|
||||||
@ -89,6 +91,7 @@ type ProductVariantResponse struct {
|
|||||||
|
|
||||||
type ListProductsRequest struct {
|
type ListProductsRequest struct {
|
||||||
OrganizationID *uuid.UUID `json:"organization_id,omitempty"`
|
OrganizationID *uuid.UUID `json:"organization_id,omitempty"`
|
||||||
|
OutletID *uuid.UUID `json:"outlet_id,omitempty"`
|
||||||
CategoryID *uuid.UUID `json:"category_id,omitempty"`
|
CategoryID *uuid.UUID `json:"category_id,omitempty"`
|
||||||
BusinessType string `json:"business_type,omitempty"`
|
BusinessType string `json:"business_type,omitempty"`
|
||||||
IsActive *bool `json:"is_active,omitempty"`
|
IsActive *bool `json:"is_active,omitempty"`
|
||||||
|
|||||||
42
internal/contract/product_outlet_price_contract.go
Normal file
42
internal/contract/product_outlet_price_contract.go
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
package contract
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CreateProductOutletPriceRequest struct {
|
||||||
|
ProductID uuid.UUID `json:"product_id" validate:"required"`
|
||||||
|
OutletID uuid.UUID `json:"outlet_id" validate:"required"`
|
||||||
|
Price float64 `json:"price" validate:"required,min=0"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UpdateProductOutletPriceRequest struct {
|
||||||
|
Price float64 `json:"price" validate:"required,min=0"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProductOutletPriceResponse struct {
|
||||||
|
ID uuid.UUID `json:"id,omitempty"`
|
||||||
|
ProductID uuid.UUID `json:"product_id,omitempty"`
|
||||||
|
OutletID uuid.UUID `json:"outlet_id"`
|
||||||
|
OutletName string `json:"outlet_name,omitempty"`
|
||||||
|
Price float64 `json:"price"`
|
||||||
|
CreatedAt time.Time `json:"created_at,omitempty"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListProductOutletPricesResponse struct {
|
||||||
|
Prices []ProductOutletPriceResponse `json:"prices"`
|
||||||
|
TotalCount int `json:"total_count"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type BulkCreateProductOutletPriceRequest struct {
|
||||||
|
ProductID uuid.UUID `json:"product_id" validate:"required"`
|
||||||
|
Prices []CreateProductOutletPricePerOutletRequest `json:"prices" validate:"required,dive"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateProductOutletPricePerOutletRequest struct {
|
||||||
|
OutletID uuid.UUID `json:"outlet_id" validate:"required"`
|
||||||
|
Price float64 `json:"price" validate:"required,min=0"`
|
||||||
|
}
|
||||||
@ -1,8 +1,6 @@
|
|||||||
package contract
|
package contract
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -50,8 +48,10 @@ type SelfOrderMenuVariant struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type SelfOrderCreateOrderRequest struct {
|
type SelfOrderCreateOrderRequest struct {
|
||||||
SessionID string `json:"session_id" validate:"required"`
|
SessionID string `json:"session_id" validate:"required"`
|
||||||
OrderItems []SelfOrderCreateOrderItem `json:"order_items" validate:"required,min=1,dive"`
|
CustomerName string `json:"customer_name" validate:"required"`
|
||||||
|
OrderType string `json:"order_type" validate:"required,oneof=dine_in takeaway delivery"`
|
||||||
|
OrderItems []SelfOrderCreateOrderItem `json:"order_items" validate:"required,min=1,dive"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type SelfOrderCreateOrderItem struct {
|
type SelfOrderCreateOrderItem struct {
|
||||||
@ -62,7 +62,7 @@ type SelfOrderCreateOrderItem struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type SelfOrderListCategoriesRequest struct {
|
type SelfOrderListCategoriesRequest struct {
|
||||||
OrganizationID string `form:"organisasi_id" validate:"required"`
|
OrganizationID string `form:"organization_id" validate:"required"`
|
||||||
OutletID string `form:"outlet_id" validate:"required"`
|
OutletID string `form:"outlet_id" validate:"required"`
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -78,35 +78,5 @@ type SelfOrderListCategoriesResponse struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type SelfOrderListOrdersResponse struct {
|
type SelfOrderListOrdersResponse struct {
|
||||||
Orders []SelfOrderOrderItem `json:"orders"`
|
Orders []OrderResponse `json:"orders"`
|
||||||
}
|
|
||||||
|
|
||||||
type SelfOrderOrderItem struct {
|
|
||||||
ID uuid.UUID `json:"id"`
|
|
||||||
OrderNumber string `json:"order_number"`
|
|
||||||
TableNumber *string `json:"table_number,omitempty"`
|
|
||||||
OrderType string `json:"order_type"`
|
|
||||||
Status string `json:"status"`
|
|
||||||
Subtotal float64 `json:"subtotal"`
|
|
||||||
TaxAmount float64 `json:"tax_amount"`
|
|
||||||
DiscountAmount float64 `json:"discount_amount"`
|
|
||||||
TotalAmount float64 `json:"total_amount"`
|
|
||||||
RemainingAmount float64 `json:"remaining_amount"`
|
|
||||||
PaymentStatus string `json:"payment_status"`
|
|
||||||
IsVoid bool `json:"is_void"`
|
|
||||||
IsRefund bool `json:"is_refund"`
|
|
||||||
Items []SelfOrderOrderLineItem `json:"items,omitempty"`
|
|
||||||
CreatedAt time.Time `json:"created_at"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type SelfOrderOrderLineItem struct {
|
|
||||||
ProductID uuid.UUID `json:"product_id"`
|
|
||||||
ProductName string `json:"product_name"`
|
|
||||||
ProductVariantID *uuid.UUID `json:"product_variant_id,omitempty"`
|
|
||||||
ProductVariantNam *string `json:"product_variant_name,omitempty"`
|
|
||||||
Quantity int `json:"quantity"`
|
|
||||||
UnitPrice float64 `json:"unit_price"`
|
|
||||||
TotalPrice float64 `json:"total_price"`
|
|
||||||
Notes *string `json:"notes,omitempty"`
|
|
||||||
Status string `json:"status"`
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -35,16 +35,23 @@ type UpdateUserOutletRequest struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type LoginRequest struct {
|
type LoginRequest struct {
|
||||||
Email string `json:"email" validate:"required,email"`
|
Email string `json:"email" validate:"required,email"`
|
||||||
Password string `json:"password" validate:"required"`
|
Password string `json:"password" validate:"required"`
|
||||||
|
DeviceID string `json:"device_id,omitempty"`
|
||||||
|
DeviceName string `json:"device_name,omitempty"`
|
||||||
|
DeviceType string `json:"device_type,omitempty"`
|
||||||
|
Platform string `json:"platform,omitempty"`
|
||||||
|
FCMToken string `json:"fcm_token,omitempty"`
|
||||||
|
AppVersion string `json:"app_version,omitempty"`
|
||||||
|
OsVersion string `json:"os_version,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type LoginResponse struct {
|
type LoginResponse struct {
|
||||||
Token string `json:"token"`
|
Token string `json:"token"`
|
||||||
RefreshToken string `json:"refresh_token"`
|
RefreshToken string `json:"refresh_token"`
|
||||||
ExpiresAt time.Time `json:"expires_at"`
|
ExpiresAt time.Time `json:"expires_at"`
|
||||||
RefreshExpiresAt time.Time `json:"refresh_expires_at"`
|
RefreshExpiresAt time.Time `json:"refresh_expires_at"`
|
||||||
User UserResponse `json:"user"`
|
User UserResponse `json:"user"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type UserResponse struct {
|
type UserResponse struct {
|
||||||
|
|||||||
59
internal/contract/user_device_contract.go
Normal file
59
internal/contract/user_device_contract.go
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
package contract
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type RegisterUserDeviceRequest struct {
|
||||||
|
DeviceID string `json:"device_id" validate:"required,min=1,max=255"`
|
||||||
|
DeviceName string `json:"device_name,omitempty" validate:"omitempty,max=255"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type,omitempty" validate:"omitempty,oneof=mobile tablet desktop"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform,omitempty" validate:"omitempty,oneof=android ios web"`
|
||||||
|
FCMToken string `json:"fcm_token,omitempty" validate:"omitempty,max=512"`
|
||||||
|
AppVersion string `json:"app_version,omitempty" validate:"omitempty,max=50"`
|
||||||
|
OsVersion string `json:"os_version,omitempty" validate:"omitempty,max=50"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UpdateUserDeviceRequest struct {
|
||||||
|
DeviceName string `json:"device_name,omitempty" validate:"omitempty,max=255"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type,omitempty" validate:"omitempty,oneof=mobile tablet desktop"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform,omitempty" validate:"omitempty,oneof=android ios web"`
|
||||||
|
FCMToken string `json:"fcm_token,omitempty" validate:"omitempty,max=512"`
|
||||||
|
AppVersion string `json:"app_version,omitempty" validate:"omitempty,max=50"`
|
||||||
|
OsVersion string `json:"os_version,omitempty" validate:"omitempty,max=50"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserDeviceResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
DeviceID string `json:"device_id"`
|
||||||
|
DeviceName string `json:"device_name"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform"`
|
||||||
|
FCMToken string `json:"fcm_token"`
|
||||||
|
AppVersion string `json:"app_version"`
|
||||||
|
OsVersion string `json:"os_version"`
|
||||||
|
IPAddress string `json:"ip_address"`
|
||||||
|
LastActiveAt *time.Time `json:"last_active_at"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListUserDevicesRequest struct {
|
||||||
|
Page int `json:"page" validate:"min=1"`
|
||||||
|
Limit int `json:"limit" validate:"min=1,max=100"`
|
||||||
|
UserID string `json:"user_id,omitempty"`
|
||||||
|
Platform string `json:"platform,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListUserDevicesResponse struct {
|
||||||
|
Devices []UserDeviceResponse `json:"devices"`
|
||||||
|
TotalCount int `json:"total_count"`
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
TotalPages int `json:"total_pages"`
|
||||||
|
}
|
||||||
@ -36,6 +36,12 @@ func GetAllEntities() []interface{} {
|
|||||||
&CampaignRule{},
|
&CampaignRule{},
|
||||||
&OtpSession{},
|
&OtpSession{},
|
||||||
// Analytics entities are not database tables, they are query results
|
// Analytics entities are not database tables, they are query results
|
||||||
|
&UserDevice{},
|
||||||
|
// Notification entities
|
||||||
|
&Notification{},
|
||||||
|
&NotificationReceiver{},
|
||||||
|
&NotificationDelivery{},
|
||||||
|
&ProductOutletPrice{},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
150
internal/entities/notification.go
Normal file
150
internal/entities/notification.go
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
package entities
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql/driver"
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationPriority string
|
||||||
|
type NotificationDeliveryStatus string
|
||||||
|
type NotificationChannel string
|
||||||
|
type NotificationProvider string
|
||||||
|
|
||||||
|
const (
|
||||||
|
NotificationPriorityLow NotificationPriority = "low"
|
||||||
|
NotificationPriorityNormal NotificationPriority = "normal"
|
||||||
|
NotificationPriorityHigh NotificationPriority = "high"
|
||||||
|
|
||||||
|
NotificationDeliveryStatusPending NotificationDeliveryStatus = "pending"
|
||||||
|
NotificationDeliveryStatusSent NotificationDeliveryStatus = "sent"
|
||||||
|
NotificationDeliveryStatusDelivered NotificationDeliveryStatus = "delivered"
|
||||||
|
NotificationDeliveryStatusFailed NotificationDeliveryStatus = "failed"
|
||||||
|
|
||||||
|
NotificationChannelPush NotificationChannel = "push"
|
||||||
|
NotificationChannelWebsocket NotificationChannel = "websocket"
|
||||||
|
NotificationChannelEmail NotificationChannel = "email"
|
||||||
|
|
||||||
|
NotificationProviderFirebase NotificationProvider = "firebase"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NotificationData is a JSON-serializable map for extra notification payload.
|
||||||
|
type NotificationData map[string]interface{}
|
||||||
|
|
||||||
|
func (d NotificationData) Value() (driver.Value, error) {
|
||||||
|
if d == nil {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return json.Marshal(d)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *NotificationData) Scan(value interface{}) error {
|
||||||
|
if value == nil {
|
||||||
|
*d = nil
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
bytes, ok := value.([]byte)
|
||||||
|
if !ok {
|
||||||
|
return errors.New("type assertion to []byte failed")
|
||||||
|
}
|
||||||
|
return json.Unmarshal(bytes, d)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notification is the master notification record.
|
||||||
|
type Notification struct {
|
||||||
|
ID uuid.UUID `gorm:"type:uuid;primary_key;default:gen_random_uuid()" json:"id"`
|
||||||
|
Title string `gorm:"not null;size:255" json:"title"`
|
||||||
|
Body string `gorm:"type:text" json:"body"`
|
||||||
|
Type string `gorm:"size:100" json:"type"`
|
||||||
|
Category string `gorm:"size:100" json:"category"`
|
||||||
|
Priority NotificationPriority `gorm:"size:50;default:'normal'" json:"priority"`
|
||||||
|
ImageURL string `gorm:"size:512" json:"image_url"`
|
||||||
|
ActionURL string `gorm:"size:512" json:"action_url"`
|
||||||
|
NotifiableType string `gorm:"size:100" json:"notifiable_type"`
|
||||||
|
NotifiableID *uuid.UUID `gorm:"type:uuid" json:"notifiable_id"`
|
||||||
|
Data NotificationData `gorm:"type:jsonb" json:"data"`
|
||||||
|
ScheduledAt *time.Time `gorm:"type:timestamptz" json:"scheduled_at"`
|
||||||
|
SentAt *time.Time `gorm:"type:timestamptz" json:"sent_at"`
|
||||||
|
ExpiredAt *time.Time `gorm:"type:timestamptz" json:"expired_at"`
|
||||||
|
CreatedBy *uuid.UUID `gorm:"type:uuid" json:"created_by"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
||||||
|
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
||||||
|
|
||||||
|
Creator *User `gorm:"foreignKey:CreatedBy" json:"creator,omitempty"`
|
||||||
|
Receivers []*NotificationReceiver `gorm:"foreignKey:NotificationID" json:"receivers,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *Notification) BeforeCreate(tx *gorm.DB) error {
|
||||||
|
if n.ID == uuid.Nil {
|
||||||
|
n.ID = uuid.New()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (Notification) TableName() string {
|
||||||
|
return "notifications"
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationReceiver links a notification to a specific user.
|
||||||
|
type NotificationReceiver struct {
|
||||||
|
ID uuid.UUID `gorm:"type:uuid;primary_key;default:gen_random_uuid()" json:"id"`
|
||||||
|
NotificationID uuid.UUID `gorm:"type:uuid;not null;index" json:"notification_id"`
|
||||||
|
UserID uuid.UUID `gorm:"type:uuid;not null;index" json:"user_id"`
|
||||||
|
IsRead bool `gorm:"default:false" json:"is_read"`
|
||||||
|
ReadAt *time.Time `gorm:"type:timestamptz" json:"read_at"`
|
||||||
|
IsDeleted bool `gorm:"default:false" json:"is_deleted"`
|
||||||
|
DeletedAt *time.Time `gorm:"type:timestamptz" json:"deleted_at"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
||||||
|
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
||||||
|
|
||||||
|
Notification *Notification `gorm:"foreignKey:NotificationID" json:"notification,omitempty"`
|
||||||
|
User *User `gorm:"foreignKey:UserID" json:"user,omitempty"`
|
||||||
|
Deliveries []*NotificationDelivery `gorm:"foreignKey:NotificationReceiverID" json:"deliveries,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *NotificationReceiver) BeforeCreate(tx *gorm.DB) error {
|
||||||
|
if n.ID == uuid.Nil {
|
||||||
|
n.ID = uuid.New()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (NotificationReceiver) TableName() string {
|
||||||
|
return "notification_receivers"
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationDelivery tracks per-device delivery attempts.
|
||||||
|
type NotificationDelivery struct {
|
||||||
|
ID uuid.UUID `gorm:"type:uuid;primary_key;default:gen_random_uuid()" json:"id"`
|
||||||
|
NotificationReceiverID uuid.UUID `gorm:"type:uuid;not null;index" json:"notification_receiver_id"`
|
||||||
|
UserDeviceID uuid.UUID `gorm:"type:uuid;not null;index" json:"user_device_id"`
|
||||||
|
Channel NotificationChannel `gorm:"size:50;default:'push'" json:"channel"`
|
||||||
|
DeliveryStatus NotificationDeliveryStatus `gorm:"size:50;default:'pending'" json:"delivery_status"`
|
||||||
|
Provider NotificationProvider `gorm:"size:50" json:"provider"`
|
||||||
|
ProviderMessageID string `gorm:"size:255" json:"provider_message_id"`
|
||||||
|
SentAt *time.Time `gorm:"type:timestamptz" json:"sent_at"`
|
||||||
|
DeliveredAt *time.Time `gorm:"type:timestamptz" json:"delivered_at"`
|
||||||
|
FailedAt *time.Time `gorm:"type:timestamptz" json:"failed_at"`
|
||||||
|
FailureReason string `gorm:"type:text" json:"failure_reason"`
|
||||||
|
RetryCount int `gorm:"default:0" json:"retry_count"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
||||||
|
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
||||||
|
|
||||||
|
NotificationReceiver *NotificationReceiver `gorm:"foreignKey:NotificationReceiverID" json:"notification_receiver,omitempty"`
|
||||||
|
UserDevice *UserDevice `gorm:"foreignKey:UserDeviceID" json:"user_device,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (n *NotificationDelivery) BeforeCreate(tx *gorm.DB) error {
|
||||||
|
if n.ID == uuid.Nil {
|
||||||
|
n.ID = uuid.New()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (NotificationDelivery) TableName() string {
|
||||||
|
return "notification_deliveries"
|
||||||
|
}
|
||||||
@ -26,13 +26,13 @@ type Product struct {
|
|||||||
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
||||||
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
||||||
|
|
||||||
Organization Organization `gorm:"foreignKey:OrganizationID" json:"organization,omitempty"`
|
Organization Organization `gorm:"foreignKey:OrganizationID" json:"organization,omitempty"`
|
||||||
Category Category `gorm:"foreignKey:CategoryID" json:"category,omitempty"`
|
Category Category `gorm:"foreignKey:CategoryID" json:"category,omitempty"`
|
||||||
Unit *Unit `gorm:"foreignKey:UnitID" json:"unit,omitempty"`
|
Unit *Unit `gorm:"foreignKey:UnitID" json:"unit,omitempty"`
|
||||||
ProductVariants []ProductVariant `gorm:"foreignKey:ProductID" json:"variants,omitempty"`
|
ProductVariants []ProductVariant `gorm:"foreignKey:ProductID" json:"variants,omitempty"`
|
||||||
ProductRecipes []ProductRecipe `gorm:"foreignKey:ProductID" json:"product_recipes,omitempty"`
|
ProductRecipes []ProductRecipe `gorm:"foreignKey:ProductID" json:"product_recipes,omitempty"`
|
||||||
Inventory []Inventory `gorm:"foreignKey:ProductID" json:"inventory,omitempty"`
|
Inventory []Inventory `gorm:"foreignKey:ProductID" json:"inventory,omitempty"`
|
||||||
OrderItems []OrderItem `gorm:"foreignKey:ProductID" json:"order_items,omitempty"`
|
OrderItems []OrderItem `gorm:"foreignKey:ProductID" json:"order_items,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *Product) BeforeCreate(tx *gorm.DB) error {
|
func (p *Product) BeforeCreate(tx *gorm.DB) error {
|
||||||
|
|||||||
31
internal/entities/product_outlet_price.go
Normal file
31
internal/entities/product_outlet_price.go
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
package entities
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPrice struct {
|
||||||
|
ID uuid.UUID `gorm:"type:uuid;primary_key;default:gen_random_uuid()" json:"id"`
|
||||||
|
ProductID uuid.UUID `gorm:"type:uuid;not null;index" json:"product_id"`
|
||||||
|
OutletID uuid.UUID `gorm:"type:uuid;not null;index" json:"outlet_id"`
|
||||||
|
Price float64 `gorm:"type:decimal(10,2);not null" json:"price"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
||||||
|
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
||||||
|
|
||||||
|
Product Product `gorm:"foreignKey:ProductID" json:"product,omitempty"`
|
||||||
|
Outlet Outlet `gorm:"foreignKey:OutletID" json:"outlet,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPrice) BeforeCreate(tx *gorm.DB) error {
|
||||||
|
if p.ID == uuid.Nil {
|
||||||
|
p.ID = uuid.New()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (ProductOutletPrice) TableName() string {
|
||||||
|
return "product_outlet_prices"
|
||||||
|
}
|
||||||
50
internal/entities/user_device.go
Normal file
50
internal/entities/user_device.go
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
package entities
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type DeviceType string
|
||||||
|
type DevicePlatform string
|
||||||
|
|
||||||
|
const (
|
||||||
|
DeviceTypeMobile DeviceType = "mobile"
|
||||||
|
DeviceTypeTablet DeviceType = "tablet"
|
||||||
|
DeviceTypeDesktop DeviceType = "desktop"
|
||||||
|
|
||||||
|
DevicePlatformAndroid DevicePlatform = "android"
|
||||||
|
DevicePlatformIOS DevicePlatform = "ios"
|
||||||
|
DevicePlatformWeb DevicePlatform = "web"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDevice struct {
|
||||||
|
ID uuid.UUID `gorm:"type:uuid;primary_key;default:gen_random_uuid()" json:"id"`
|
||||||
|
UserID uuid.UUID `gorm:"type:uuid;not null;index" json:"user_id"`
|
||||||
|
DeviceID string `gorm:"not null;size:255;index" json:"device_id"`
|
||||||
|
DeviceName string `gorm:"size:255" json:"device_name"`
|
||||||
|
DeviceType DeviceType `gorm:"size:50" json:"device_type"`
|
||||||
|
Platform DevicePlatform `gorm:"size:50" json:"platform"`
|
||||||
|
FCMToken string `gorm:"size:512" json:"fcm_token"`
|
||||||
|
AppVersion string `gorm:"size:50" json:"app_version"`
|
||||||
|
OsVersion string `gorm:"size:50" json:"os_version"`
|
||||||
|
IPAddress string `gorm:"size:45" json:"ip_address"`
|
||||||
|
LastActiveAt *time.Time `gorm:"type:timestamptz" json:"last_active_at"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime" json:"created_at"`
|
||||||
|
UpdatedAt time.Time `gorm:"autoUpdateTime" json:"updated_at"`
|
||||||
|
|
||||||
|
User User `gorm:"foreignKey:UserID" json:"user,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (u *UserDevice) BeforeCreate(tx *gorm.DB) error {
|
||||||
|
if u.ID == uuid.Nil {
|
||||||
|
u.ID = uuid.New()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (UserDevice) TableName() string {
|
||||||
|
return "user_devices"
|
||||||
|
}
|
||||||
@ -8,6 +8,7 @@ import (
|
|||||||
"apskel-pos-be/internal/util"
|
"apskel-pos-be/internal/util"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type AnalyticsHandler struct {
|
type AnalyticsHandler struct {
|
||||||
@ -25,6 +26,17 @@ func NewAnalyticsHandler(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *AnalyticsHandler) resolveOutletID(c *gin.Context, contextOutletID uuid.UUID) *string {
|
||||||
|
if outletIDStr := c.Query("outlet_id"); outletIDStr != "" {
|
||||||
|
return &outletIDStr
|
||||||
|
}
|
||||||
|
if contextOutletID != uuid.Nil {
|
||||||
|
s := contextOutletID.String()
|
||||||
|
return &s
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (h *AnalyticsHandler) GetPaymentMethodAnalytics(c *gin.Context) {
|
func (h *AnalyticsHandler) GetPaymentMethodAnalytics(c *gin.Context) {
|
||||||
ctx := c.Request.Context()
|
ctx := c.Request.Context()
|
||||||
contextInfo := appcontext.FromGinContext(ctx)
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
@ -36,7 +48,7 @@ func (h *AnalyticsHandler) GetPaymentMethodAnalytics(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
req.OrganizationID = contextInfo.OrganizationID
|
req.OrganizationID = contextInfo.OrganizationID
|
||||||
req.OutletID = &contextInfo.OutletID
|
req.OutletID = h.resolveOutletID(c, contextInfo.OutletID)
|
||||||
modelReq := transformer.PaymentMethodAnalyticsContractToModel(&req)
|
modelReq := transformer.PaymentMethodAnalyticsContractToModel(&req)
|
||||||
|
|
||||||
response, err := h.analyticsService.GetPaymentMethodAnalytics(ctx, modelReq)
|
response, err := h.analyticsService.GetPaymentMethodAnalytics(ctx, modelReq)
|
||||||
@ -60,7 +72,7 @@ func (h *AnalyticsHandler) GetSalesAnalytics(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
req.OrganizationID = contextInfo.OrganizationID
|
req.OrganizationID = contextInfo.OrganizationID
|
||||||
req.OutletID = &contextInfo.OutletID
|
req.OutletID = h.resolveOutletID(c, contextInfo.OutletID)
|
||||||
modelReq := transformer.SalesAnalyticsContractToModel(&req)
|
modelReq := transformer.SalesAnalyticsContractToModel(&req)
|
||||||
|
|
||||||
response, err := h.analyticsService.GetSalesAnalytics(ctx, modelReq)
|
response, err := h.analyticsService.GetSalesAnalytics(ctx, modelReq)
|
||||||
@ -84,7 +96,7 @@ func (h *AnalyticsHandler) GetProductAnalytics(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
req.OrganizationID = contextInfo.OrganizationID
|
req.OrganizationID = contextInfo.OrganizationID
|
||||||
req.OutletID = &contextInfo.OutletID
|
req.OutletID = h.resolveOutletID(c, contextInfo.OutletID)
|
||||||
modelReq := transformer.ProductAnalyticsContractToModel(&req)
|
modelReq := transformer.ProductAnalyticsContractToModel(&req)
|
||||||
|
|
||||||
response, err := h.analyticsService.GetProductAnalytics(ctx, modelReq)
|
response, err := h.analyticsService.GetProductAnalytics(ctx, modelReq)
|
||||||
@ -108,7 +120,7 @@ func (h *AnalyticsHandler) GetProductAnalyticsPerCategory(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
req.OrganizationID = contextInfo.OrganizationID
|
req.OrganizationID = contextInfo.OrganizationID
|
||||||
req.OutletID = &contextInfo.OutletID
|
req.OutletID = h.resolveOutletID(c, contextInfo.OutletID)
|
||||||
modelReq := transformer.ProductAnalyticsPerCategoryContractToModel(&req)
|
modelReq := transformer.ProductAnalyticsPerCategoryContractToModel(&req)
|
||||||
|
|
||||||
response, err := h.analyticsService.GetProductAnalyticsPerCategory(ctx, modelReq)
|
response, err := h.analyticsService.GetProductAnalyticsPerCategory(ctx, modelReq)
|
||||||
@ -132,7 +144,7 @@ func (h *AnalyticsHandler) GetDashboardAnalytics(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
req.OrganizationID = contextInfo.OrganizationID
|
req.OrganizationID = contextInfo.OrganizationID
|
||||||
req.OutletID = &contextInfo.OutletID
|
req.OutletID = h.resolveOutletID(c, contextInfo.OutletID)
|
||||||
modelReq := transformer.DashboardAnalyticsContractToModel(&req)
|
modelReq := transformer.DashboardAnalyticsContractToModel(&req)
|
||||||
|
|
||||||
response, err := h.analyticsService.GetDashboardAnalytics(ctx, modelReq)
|
response, err := h.analyticsService.GetDashboardAnalytics(ctx, modelReq)
|
||||||
@ -156,6 +168,7 @@ func (h *AnalyticsHandler) GetProfitLossAnalytics(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
req.OrganizationID = contextInfo.OrganizationID
|
req.OrganizationID = contextInfo.OrganizationID
|
||||||
|
req.OutletID = h.resolveOutletID(c, contextInfo.OutletID)
|
||||||
modelReq, err := transformer.ProfitLossAnalyticsContractToModel(&req)
|
modelReq, err := transformer.ProfitLossAnalyticsContractToModel(&req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{contract.NewResponseError("invalid_request", "AnalyticsHandler::GetProfitLossAnalytics", err.Error())}), "AnalyticsHandler::GetProfitLossAnalytics")
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{contract.NewResponseError("invalid_request", "AnalyticsHandler::GetProfitLossAnalytics", err.Error())}), "AnalyticsHandler::GetProfitLossAnalytics")
|
||||||
|
|||||||
190
internal/handler/notification_handler.go
Normal file
190
internal/handler/notification_handler.go
Normal file
@ -0,0 +1,190 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/appcontext"
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/logger"
|
||||||
|
"apskel-pos-be/internal/service"
|
||||||
|
"apskel-pos-be/internal/util"
|
||||||
|
"apskel-pos-be/internal/validator"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationHandler struct {
|
||||||
|
notificationService service.NotificationService
|
||||||
|
notificationValidator validator.NotificationValidator
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNotificationHandler(
|
||||||
|
notificationService service.NotificationService,
|
||||||
|
notificationValidator validator.NotificationValidator,
|
||||||
|
) *NotificationHandler {
|
||||||
|
return &NotificationHandler{
|
||||||
|
notificationService: notificationService,
|
||||||
|
notificationValidator: notificationValidator,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send godoc
|
||||||
|
// POST /api/v1/notifications/send
|
||||||
|
// Sends a notification to specific users.
|
||||||
|
func (h *NotificationHandler) Send(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
var req contract.SendNotificationRequest
|
||||||
|
if err := c.ShouldBindJSON(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("NotificationHandler::Send -> request binding failed")
|
||||||
|
validationErr := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationErr}), "NotificationHandler::Send")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if validationErr, errCode := h.notificationValidator.ValidateSendRequest(&req); validationErr != nil {
|
||||||
|
respErr := contract.NewResponseError(errCode, constants.RequestEntity, validationErr.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{respErr}), "NotificationHandler::Send")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.notificationService.Send(ctx, &req, contextInfo.UserID)
|
||||||
|
if resp.HasErrors() {
|
||||||
|
logger.FromContext(ctx).WithError(resp.GetErrors()[0]).Error("NotificationHandler::Send -> service error")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::Send")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Broadcast godoc
|
||||||
|
// POST /api/v1/notifications/broadcast
|
||||||
|
// Sends a notification to all active users in the caller's organization.
|
||||||
|
func (h *NotificationHandler) Broadcast(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
var req contract.BroadcastNotificationRequest
|
||||||
|
if err := c.ShouldBindJSON(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("NotificationHandler::Broadcast -> request binding failed")
|
||||||
|
validationErr := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationErr}), "NotificationHandler::Broadcast")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if validationErr, errCode := h.notificationValidator.ValidateBroadcastRequest(&req); validationErr != nil {
|
||||||
|
respErr := contract.NewResponseError(errCode, constants.RequestEntity, validationErr.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{respErr}), "NotificationHandler::Broadcast")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.notificationService.Broadcast(ctx, &req, contextInfo.OrganizationID, contextInfo.UserID)
|
||||||
|
if resp.HasErrors() {
|
||||||
|
logger.FromContext(ctx).WithError(resp.GetErrors()[0]).Error("NotificationHandler::Broadcast -> service error")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::Broadcast")
|
||||||
|
}
|
||||||
|
|
||||||
|
// List godoc
|
||||||
|
// GET /api/v1/notifications
|
||||||
|
// Returns paginated notifications for the authenticated user.
|
||||||
|
func (h *NotificationHandler) List(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
req := contract.ListNotificationsRequest{
|
||||||
|
Page: 1,
|
||||||
|
Limit: 20,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := c.ShouldBindQuery(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("NotificationHandler::List -> query binding failed")
|
||||||
|
validationErr := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationErr}), "NotificationHandler::List")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if validationErr, errCode := h.notificationValidator.ValidateListRequest(&req); validationErr != nil {
|
||||||
|
respErr := contract.NewResponseError(errCode, constants.RequestEntity, validationErr.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{respErr}), "NotificationHandler::List")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.notificationService.ListForUser(ctx, &req, contextInfo.UserID)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::List")
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetByID godoc
|
||||||
|
// GET /api/v1/notifications/:id
|
||||||
|
func (h *NotificationHandler) GetByID(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
idStr := c.Param("id")
|
||||||
|
id, err := uuid.Parse(idStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("NotificationHandler::GetByID -> invalid notification ID")
|
||||||
|
validationErr := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid notification ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationErr}), "NotificationHandler::GetByID")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.notificationService.GetByID(ctx, id)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::GetByID")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MarkAsRead godoc
|
||||||
|
// PUT /api/v1/notifications/:id/read
|
||||||
|
func (h *NotificationHandler) MarkAsRead(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
idStr := c.Param("id")
|
||||||
|
receiverID, err := uuid.Parse(idStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("NotificationHandler::MarkAsRead -> invalid receiver ID")
|
||||||
|
validationErr := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid notification receiver ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationErr}), "NotificationHandler::MarkAsRead")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.notificationService.MarkAsRead(ctx, receiverID, contextInfo.UserID)
|
||||||
|
if resp.HasErrors() {
|
||||||
|
logger.FromContext(ctx).WithError(resp.GetErrors()[0]).Error("NotificationHandler::MarkAsRead -> service error")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::MarkAsRead")
|
||||||
|
}
|
||||||
|
|
||||||
|
// MarkAllAsRead godoc
|
||||||
|
// PUT /api/v1/notifications/read-all
|
||||||
|
func (h *NotificationHandler) MarkAllAsRead(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
resp := h.notificationService.MarkAllAsRead(ctx, contextInfo.UserID)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::MarkAllAsRead")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete godoc
|
||||||
|
// DELETE /api/v1/notifications/:id
|
||||||
|
func (h *NotificationHandler) Delete(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
idStr := c.Param("id")
|
||||||
|
receiverID, err := uuid.Parse(idStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("NotificationHandler::Delete -> invalid receiver ID")
|
||||||
|
validationErr := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid notification receiver ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationErr}), "NotificationHandler::Delete")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.notificationService.DeleteForUser(ctx, receiverID, contextInfo.UserID)
|
||||||
|
if resp.HasErrors() {
|
||||||
|
logger.FromContext(ctx).WithError(resp.GetErrors()[0]).Error("NotificationHandler::Delete -> service error")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "NotificationHandler::Delete")
|
||||||
|
}
|
||||||
@ -137,6 +137,9 @@ func (h *OrderHandler) ListOrders(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
modelReq.OrganizationID = &contextInfo.OrganizationID
|
modelReq.OrganizationID = &contextInfo.OrganizationID
|
||||||
|
if modelReq.OutletID == nil && contextInfo.OutletID != uuid.Nil {
|
||||||
|
modelReq.OutletID = &contextInfo.OutletID
|
||||||
|
}
|
||||||
response, err := h.orderService.ListOrders(c.Request.Context(), modelReq)
|
response, err := h.orderService.ListOrders(c.Request.Context(), modelReq)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{contract.NewResponseError("internal_error", "OrderHandler::ListOrders", err.Error())}), "OrderHandler::ListOrders")
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{contract.NewResponseError("internal_error", "OrderHandler::ListOrders", err.Error())}), "OrderHandler::ListOrders")
|
||||||
|
|||||||
@ -117,6 +117,7 @@ func (h *ProductHandler) DeleteProduct(c *gin.Context) {
|
|||||||
|
|
||||||
func (h *ProductHandler) GetProduct(c *gin.Context) {
|
func (h *ProductHandler) GetProduct(c *gin.Context) {
|
||||||
ctx := c.Request.Context()
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
productIDStr := c.Param("id")
|
productIDStr := c.Param("id")
|
||||||
productID, err := uuid.Parse(productIDStr)
|
productID, err := uuid.Parse(productIDStr)
|
||||||
@ -127,7 +128,7 @@ func (h *ProductHandler) GetProduct(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
productResponse := h.productService.GetProductByID(ctx, productID)
|
productResponse := h.productService.GetProductByID(ctx, productID, contextInfo.OutletID)
|
||||||
if productResponse.HasErrors() {
|
if productResponse.HasErrors() {
|
||||||
errorResp := productResponse.GetErrors()[0]
|
errorResp := productResponse.GetErrors()[0]
|
||||||
logger.FromContext(ctx).WithError(errorResp).Error("ProductHandler::GetProduct -> Failed to get product from service")
|
logger.FromContext(ctx).WithError(errorResp).Error("ProductHandler::GetProduct -> Failed to get product from service")
|
||||||
@ -184,6 +185,97 @@ func (h *ProductHandler) ListProducts(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if outletIDStr := c.Query("outlet_id"); outletIDStr != "" {
|
||||||
|
if outletID, err := uuid.Parse(outletIDStr); err == nil {
|
||||||
|
req.OutletID = &outletID
|
||||||
|
}
|
||||||
|
} else if contextInfo.OutletID != uuid.Nil {
|
||||||
|
req.OutletID = &contextInfo.OutletID
|
||||||
|
}
|
||||||
|
|
||||||
|
if minPriceStr := c.Query("min_price"); minPriceStr != "" {
|
||||||
|
if minPrice, err := strconv.ParseFloat(minPriceStr, 64); err == nil {
|
||||||
|
req.MinPrice = &minPrice
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if maxPriceStr := c.Query("max_price"); maxPriceStr != "" {
|
||||||
|
if maxPrice, err := strconv.ParseFloat(maxPriceStr, 64); err == nil {
|
||||||
|
req.MaxPrice = &maxPrice
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
validationError, validationErrorCode := h.productValidator.ValidateListProductsRequest(req)
|
||||||
|
if validationError != nil {
|
||||||
|
logger.FromContext(ctx).WithError(validationError).Error("ProductHandler::ListProducts -> request validation failed")
|
||||||
|
validationResponseError := contract.NewResponseError(validationErrorCode, constants.RequestEntity, validationError.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductHandler::ListProducts")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
productsResponse := h.productService.ListProducts(ctx, req)
|
||||||
|
if productsResponse.HasErrors() {
|
||||||
|
errorResp := productsResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("ProductHandler::ListProducts -> Failed to list products from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, productsResponse, "ProductHandler::ListProducts")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductHandler) ListProductAll(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
req := &contract.ListProductsRequest{
|
||||||
|
Page: 1,
|
||||||
|
Limit: 10,
|
||||||
|
OrganizationID: &contextInfo.OrganizationID,
|
||||||
|
}
|
||||||
|
|
||||||
|
if pageStr := c.Query("page"); pageStr != "" {
|
||||||
|
if page, err := strconv.Atoi(pageStr); err == nil {
|
||||||
|
req.Page = page
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if limitStr := c.Query("limit"); limitStr != "" {
|
||||||
|
if limit, err := strconv.Atoi(limitStr); err == nil {
|
||||||
|
req.Limit = limit
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if search := c.Query("search"); search != "" {
|
||||||
|
req.Search = search
|
||||||
|
}
|
||||||
|
|
||||||
|
if businessType := c.Query("business_type"); businessType != "" {
|
||||||
|
req.BusinessType = businessType
|
||||||
|
}
|
||||||
|
|
||||||
|
if organizationIDStr := c.Query("organization_id"); organizationIDStr != "" {
|
||||||
|
if organizationID, err := uuid.Parse(organizationIDStr); err == nil {
|
||||||
|
req.OrganizationID = &organizationID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if categoryIDStr := c.Query("category_id"); categoryIDStr != "" {
|
||||||
|
if categoryID, err := uuid.Parse(categoryIDStr); err == nil {
|
||||||
|
req.CategoryID = &categoryID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if isActiveStr := c.Query("is_active"); isActiveStr != "" {
|
||||||
|
if isActive, err := strconv.ParseBool(isActiveStr); err == nil {
|
||||||
|
req.IsActive = &isActive
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if outletIDStr := c.Query("outlet_id"); outletIDStr != "" {
|
||||||
|
if outletID, err := uuid.Parse(outletIDStr); err == nil {
|
||||||
|
req.OutletID = &outletID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if minPriceStr := c.Query("min_price"); minPriceStr != "" {
|
if minPriceStr := c.Query("min_price"); minPriceStr != "" {
|
||||||
if minPrice, err := strconv.ParseFloat(minPriceStr, 64); err == nil {
|
if minPrice, err := strconv.ParseFloat(minPriceStr, 64); err == nil {
|
||||||
req.MinPrice = &minPrice
|
req.MinPrice = &minPrice
|
||||||
|
|||||||
135
internal/handler/product_outlet_price_handler.go
Normal file
135
internal/handler/product_outlet_price_handler.go
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/logger"
|
||||||
|
"apskel-pos-be/internal/service"
|
||||||
|
"apskel-pos-be/internal/util"
|
||||||
|
"apskel-pos-be/internal/validator"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPriceHandler struct {
|
||||||
|
service service.ProductOutletPriceService
|
||||||
|
validator validator.ProductOutletPriceValidator
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewProductOutletPriceHandler(svc service.ProductOutletPriceService, v validator.ProductOutletPriceValidator) *ProductOutletPriceHandler {
|
||||||
|
return &ProductOutletPriceHandler{
|
||||||
|
service: svc,
|
||||||
|
validator: v,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductOutletPriceHandler) Upsert(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
var req contract.CreateProductOutletPriceRequest
|
||||||
|
if err := c.ShouldBindJSON(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("ProductOutletPriceHandler::Upsert -> request binding failed")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::Upsert")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if validationErr, code := h.validator.ValidateCreateRequest(&req); validationErr != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(code, constants.RequestEntity, validationErr.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::Upsert")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.service.Upsert(ctx, &req)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "ProductOutletPriceHandler::Upsert")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductOutletPriceHandler) GetByProductAndOutlet(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
productIDStr := c.Param("product_id")
|
||||||
|
productID, err := uuid.Parse(productIDStr)
|
||||||
|
if err != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid product ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::GetByProductAndOutlet")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
outletIDStr := c.Param("outlet_id")
|
||||||
|
outletID, err := uuid.Parse(outletIDStr)
|
||||||
|
if err != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid outlet ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::GetByProductAndOutlet")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.service.GetByProductAndOutlet(ctx, productID, outletID)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "ProductOutletPriceHandler::GetByProductAndOutlet")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductOutletPriceHandler) GetByProduct(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
productIDStr := c.Param("product_id")
|
||||||
|
productID, err := uuid.Parse(productIDStr)
|
||||||
|
if err != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid product ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::GetByProduct")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.service.GetByProduct(ctx, productID)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "ProductOutletPriceHandler::GetByProduct")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductOutletPriceHandler) GetByOutlet(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
outletIDStr := c.Param("outlet_id")
|
||||||
|
outletID, err := uuid.Parse(outletIDStr)
|
||||||
|
if err != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid outlet ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::GetByOutlet")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.service.GetByOutlet(ctx, outletID)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "ProductOutletPriceHandler::GetByOutlet")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductOutletPriceHandler) Delete(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
idStr := c.Param("id")
|
||||||
|
id, err := uuid.Parse(idStr)
|
||||||
|
if err != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::Delete")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.service.Delete(ctx, id)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "ProductOutletPriceHandler::Delete")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *ProductOutletPriceHandler) BulkUpsert(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
var req contract.BulkCreateProductOutletPriceRequest
|
||||||
|
if err := c.ShouldBindJSON(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("ProductOutletPriceHandler::BulkUpsert -> request binding failed")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::BulkUpsert")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if validationErr, code := h.validator.ValidateBulkCreateRequest(&req); validationErr != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(code, constants.RequestEntity, validationErr.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "ProductOutletPriceHandler::BulkUpsert")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
resp := h.service.BulkUpsert(ctx, &req)
|
||||||
|
util.HandleResponse(c.Writer, c.Request, resp, "ProductOutletPriceHandler::BulkUpsert")
|
||||||
|
}
|
||||||
@ -8,6 +8,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ReportHandler struct {
|
type ReportHandler struct {
|
||||||
@ -19,11 +20,26 @@ func NewReportHandler(reportService service.ReportService, userService UserServi
|
|||||||
return &ReportHandler{reportService: reportService, userService: userService}
|
return &ReportHandler{reportService: reportService, userService: userService}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *ReportHandler) resolveOutletID(c *gin.Context, contextOutletID uuid.UUID) string {
|
||||||
|
if outletIDStr := c.Query("outlet_id"); outletIDStr != "" {
|
||||||
|
if _, err := uuid.Parse(outletIDStr); err == nil {
|
||||||
|
return outletIDStr
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if pathOutletID := c.Param("outlet_id"); pathOutletID != "" {
|
||||||
|
return pathOutletID
|
||||||
|
}
|
||||||
|
if contextOutletID != uuid.Nil {
|
||||||
|
return contextOutletID.String()
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
func (h *ReportHandler) GetDailyTransactionReportPDF(c *gin.Context) {
|
func (h *ReportHandler) GetDailyTransactionReportPDF(c *gin.Context) {
|
||||||
ctx := c.Request.Context()
|
ctx := c.Request.Context()
|
||||||
ci := appcontext.FromGinContext(ctx)
|
ci := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
outletID := c.Param("outlet_id")
|
outletID := h.resolveOutletID(c, ci.OutletID)
|
||||||
var dayPtr *time.Time
|
var dayPtr *time.Time
|
||||||
if d := c.Query("date"); d != "" {
|
if d := c.Query("date"); d != "" {
|
||||||
if t, err := time.Parse("2006-01-02", d); err == nil {
|
if t, err := time.Parse("2006-01-02", d); err == nil {
|
||||||
|
|||||||
@ -5,6 +5,7 @@ import (
|
|||||||
"apskel-pos-be/internal/contract"
|
"apskel-pos-be/internal/contract"
|
||||||
"apskel-pos-be/internal/entities"
|
"apskel-pos-be/internal/entities"
|
||||||
"apskel-pos-be/internal/logger"
|
"apskel-pos-be/internal/logger"
|
||||||
|
"apskel-pos-be/internal/mappers"
|
||||||
"apskel-pos-be/internal/models"
|
"apskel-pos-be/internal/models"
|
||||||
"apskel-pos-be/internal/pkg/tabletoken"
|
"apskel-pos-be/internal/pkg/tabletoken"
|
||||||
"apskel-pos-be/internal/processor"
|
"apskel-pos-be/internal/processor"
|
||||||
@ -20,14 +21,15 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type SelfOrderHandler struct {
|
type SelfOrderHandler struct {
|
||||||
orderService service.OrderService
|
orderService service.OrderService
|
||||||
categoryService service.CategoryService
|
categoryService service.CategoryService
|
||||||
productService service.ProductService
|
productService service.ProductService
|
||||||
tableRepo repository.TableRepositoryInterface
|
tableRepo repository.TableRepositoryInterface
|
||||||
outletRepo processor.OutletRepository
|
outletRepo processor.OutletRepository
|
||||||
userRepo processor.UserRepository
|
userRepo processor.UserRepository
|
||||||
sessionRepo repository.SessionRepository
|
sessionRepo repository.SessionRepository
|
||||||
orderRepo repository.OrderRepository
|
orderRepo repository.OrderRepository
|
||||||
|
productOutletPriceService service.ProductOutletPriceService
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSelfOrderHandler(
|
func NewSelfOrderHandler(
|
||||||
@ -39,16 +41,18 @@ func NewSelfOrderHandler(
|
|||||||
userRepo processor.UserRepository,
|
userRepo processor.UserRepository,
|
||||||
sessionRepo repository.SessionRepository,
|
sessionRepo repository.SessionRepository,
|
||||||
orderRepo repository.OrderRepository,
|
orderRepo repository.OrderRepository,
|
||||||
|
productOutletPriceService service.ProductOutletPriceService,
|
||||||
) *SelfOrderHandler {
|
) *SelfOrderHandler {
|
||||||
return &SelfOrderHandler{
|
return &SelfOrderHandler{
|
||||||
orderService: orderService,
|
orderService: orderService,
|
||||||
categoryService: categoryService,
|
categoryService: categoryService,
|
||||||
productService: productService,
|
productService: productService,
|
||||||
tableRepo: tableRepo,
|
tableRepo: tableRepo,
|
||||||
outletRepo: outletRepo,
|
outletRepo: outletRepo,
|
||||||
userRepo: userRepo,
|
userRepo: userRepo,
|
||||||
sessionRepo: sessionRepo,
|
sessionRepo: sessionRepo,
|
||||||
orderRepo: orderRepo,
|
orderRepo: orderRepo,
|
||||||
|
productOutletPriceService: productOutletPriceService,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -215,16 +219,29 @@ func (h *SelfOrderHandler) GetMenu(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
menu := h.buildMenuResponse(outlet, table, catList.Categories, prodList.Products)
|
menu := h.buildMenuResponse(ctx, outlet, table, catList.Categories, prodList.Products)
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(menu), "SelfOrderHandler::GetMenu")
|
util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(menu), "SelfOrderHandler::GetMenu")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *SelfOrderHandler) buildMenuResponse(
|
func (h *SelfOrderHandler) buildMenuResponse(
|
||||||
|
ctx context.Context,
|
||||||
outlet *entities.Outlet,
|
outlet *entities.Outlet,
|
||||||
table *entities.Table,
|
table *entities.Table,
|
||||||
categories []contract.CategoryResponse,
|
categories []contract.CategoryResponse,
|
||||||
products []contract.ProductResponse,
|
products []contract.ProductResponse,
|
||||||
) *contract.SelfOrderMenuResponse {
|
) *contract.SelfOrderMenuResponse {
|
||||||
|
outletPriceMap := make(map[uuid.UUID]float64)
|
||||||
|
if h.productOutletPriceService != nil {
|
||||||
|
priceResp := h.productOutletPriceService.GetByOutlet(ctx, outlet.ID)
|
||||||
|
if priceResp != nil && !priceResp.HasErrors() {
|
||||||
|
if priceList, ok := priceResp.Data.(*contract.ListProductOutletPricesResponse); ok {
|
||||||
|
for _, p := range priceList.Prices {
|
||||||
|
outletPriceMap[p.ProductID] = p.Price
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
productMap := make(map[uuid.UUID][]contract.ProductResponse)
|
productMap := make(map[uuid.UUID][]contract.ProductResponse)
|
||||||
for _, p := range products {
|
for _, p := range products {
|
||||||
productMap[p.CategoryID] = append(productMap[p.CategoryID], p)
|
productMap[p.CategoryID] = append(productMap[p.CategoryID], p)
|
||||||
@ -235,11 +252,15 @@ func (h *SelfOrderHandler) buildMenuResponse(
|
|||||||
menuItems := make([]contract.SelfOrderMenuItem, 0)
|
menuItems := make([]contract.SelfOrderMenuItem, 0)
|
||||||
if prods, ok := productMap[cat.ID]; ok {
|
if prods, ok := productMap[cat.ID]; ok {
|
||||||
for _, p := range prods {
|
for _, p := range prods {
|
||||||
|
price := p.Price
|
||||||
|
if outletPrice, exists := outletPriceMap[p.ID]; exists {
|
||||||
|
price = outletPrice
|
||||||
|
}
|
||||||
item := contract.SelfOrderMenuItem{
|
item := contract.SelfOrderMenuItem{
|
||||||
ID: p.ID,
|
ID: p.ID,
|
||||||
Name: p.Name,
|
Name: p.Name,
|
||||||
Description: p.Description,
|
Description: p.Description,
|
||||||
Price: p.Price,
|
Price: price,
|
||||||
ImageURL: p.ImageURL,
|
ImageURL: p.ImageURL,
|
||||||
}
|
}
|
||||||
for _, v := range p.Variants {
|
for _, v := range p.Variants {
|
||||||
@ -330,6 +351,7 @@ func (h *SelfOrderHandler) CreateOrder(c *gin.Context) {
|
|||||||
metadata := make(map[string]interface{})
|
metadata := make(map[string]interface{})
|
||||||
metadata["self_order"] = true
|
metadata["self_order"] = true
|
||||||
metadata["session_id"] = session.ID
|
metadata["session_id"] = session.ID
|
||||||
|
metadata["customer_name"] = req.CustomerName
|
||||||
|
|
||||||
tableID := table.ID
|
tableID := table.ID
|
||||||
modelReq := &models.CreateOrderRequest{
|
modelReq := &models.CreateOrderRequest{
|
||||||
@ -337,7 +359,7 @@ func (h *SelfOrderHandler) CreateOrder(c *gin.Context) {
|
|||||||
UserID: userID,
|
UserID: userID,
|
||||||
TableID: &tableID,
|
TableID: &tableID,
|
||||||
TableNumber: &table.TableName,
|
TableNumber: &table.TableName,
|
||||||
OrderType: constants.OrderTypeDineIn,
|
OrderType: constants.OrderType(req.OrderType),
|
||||||
OrderItems: orderItems,
|
OrderItems: orderItems,
|
||||||
Metadata: metadata,
|
Metadata: metadata,
|
||||||
}
|
}
|
||||||
@ -357,7 +379,7 @@ func (h *SelfOrderHandler) CreateOrder(c *gin.Context) {
|
|||||||
|
|
||||||
func (h *SelfOrderHandler) GetOrdersBySession(c *gin.Context) {
|
func (h *SelfOrderHandler) GetOrdersBySession(c *gin.Context) {
|
||||||
ctx := c.Request.Context()
|
ctx := c.Request.Context()
|
||||||
sessionID := c.Param("sessionId")
|
sessionID := c.Param("session_id")
|
||||||
|
|
||||||
if sessionID == "" {
|
if sessionID == "" {
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{
|
||||||
@ -390,47 +412,15 @@ func (h *SelfOrderHandler) GetOrdersBySession(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
resp := &contract.SelfOrderListOrdersResponse{
|
modelOrders := mappers.OrderEntitiesToResponses(orders)
|
||||||
Orders: make([]contract.SelfOrderOrderItem, 0, len(orders)),
|
contractOrders := make([]contract.OrderResponse, len(modelOrders))
|
||||||
}
|
for i := range modelOrders {
|
||||||
for _, o := range orders {
|
contractOrders[i] = *transformer.OrderModelToContract(&modelOrders[i])
|
||||||
item := contract.SelfOrderOrderItem{
|
|
||||||
ID: o.ID,
|
|
||||||
OrderNumber: o.OrderNumber,
|
|
||||||
TableNumber: o.TableNumber,
|
|
||||||
OrderType: string(o.OrderType),
|
|
||||||
Status: string(o.Status),
|
|
||||||
Subtotal: o.Subtotal,
|
|
||||||
TaxAmount: o.TaxAmount,
|
|
||||||
DiscountAmount: o.DiscountAmount,
|
|
||||||
TotalAmount: o.TotalAmount,
|
|
||||||
RemainingAmount: o.RemainingAmount,
|
|
||||||
PaymentStatus: string(o.PaymentStatus),
|
|
||||||
IsVoid: o.IsVoid,
|
|
||||||
IsRefund: o.IsRefund,
|
|
||||||
CreatedAt: o.CreatedAt,
|
|
||||||
}
|
|
||||||
for _, oi := range o.OrderItems {
|
|
||||||
lineItem := contract.SelfOrderOrderLineItem{
|
|
||||||
ProductID: oi.ProductID,
|
|
||||||
Quantity: oi.Quantity,
|
|
||||||
UnitPrice: oi.UnitPrice,
|
|
||||||
TotalPrice: oi.TotalPrice,
|
|
||||||
Notes: oi.Notes,
|
|
||||||
Status: string(oi.Status),
|
|
||||||
ProductVariantID: oi.ProductVariantID,
|
|
||||||
}
|
|
||||||
if oi.Product.ID != uuid.Nil {
|
|
||||||
lineItem.ProductName = oi.Product.Name
|
|
||||||
}
|
|
||||||
if oi.ProductVariant != nil {
|
|
||||||
lineItem.ProductVariantNam = &oi.ProductVariant.Name
|
|
||||||
}
|
|
||||||
item.Items = append(item.Items, lineItem)
|
|
||||||
}
|
|
||||||
resp.Orders = append(resp.Orders, item)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
resp := &contract.SelfOrderListOrdersResponse{
|
||||||
|
Orders: contractOrders,
|
||||||
|
}
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(resp), "SelfOrderHandler::GetOrdersBySession")
|
util.HandleResponse(c.Writer, c.Request, contract.BuildSuccessResponse(resp), "SelfOrderHandler::GetOrdersBySession")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -439,6 +429,7 @@ func (h *SelfOrderHandler) validateCreateOrderRequest(req *contract.SelfOrderCre
|
|||||||
return fmt.Errorf("session_id is required")
|
return fmt.Errorf("session_id is required")
|
||||||
}
|
}
|
||||||
if len(req.OrderItems) == 0 {
|
if len(req.OrderItems) == 0 {
|
||||||
|
|
||||||
return fmt.Errorf("at least one order item is required")
|
return fmt.Errorf("at least one order item is required")
|
||||||
}
|
}
|
||||||
for i, item := range req.OrderItems {
|
for i, item := range req.OrderItems {
|
||||||
@ -466,7 +457,7 @@ func (h *SelfOrderHandler) ListCategories(c *gin.Context) {
|
|||||||
|
|
||||||
if req.OrganizationID == "" {
|
if req.OrganizationID == "" {
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{
|
||||||
contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, "organisasi_id is required"),
|
contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, "organization_id is required"),
|
||||||
}), "SelfOrderHandler::ListCategories")
|
}), "SelfOrderHandler::ListCategories")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -481,7 +472,7 @@ func (h *SelfOrderHandler) ListCategories(c *gin.Context) {
|
|||||||
orgID, err := uuid.Parse(req.OrganizationID)
|
orgID, err := uuid.Parse(req.OrganizationID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{
|
||||||
contract.NewResponseError(constants.ValidationErrorCode, constants.RequestEntity, "invalid organisasi_id format"),
|
contract.NewResponseError(constants.ValidationErrorCode, constants.RequestEntity, "invalid organization_id format"),
|
||||||
}), "SelfOrderHandler::ListCategories")
|
}), "SelfOrderHandler::ListCategories")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|||||||
@ -19,14 +19,14 @@ import (
|
|||||||
type TableHandler struct {
|
type TableHandler struct {
|
||||||
tableService TableService
|
tableService TableService
|
||||||
tableValidator *validator.TableValidator
|
tableValidator *validator.TableValidator
|
||||||
baseURL string
|
selfOrderURL string
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewTableHandler(tableService TableService, tableValidator *validator.TableValidator, baseURL string) *TableHandler {
|
func NewTableHandler(tableService TableService, tableValidator *validator.TableValidator, selfOrderURL string) *TableHandler {
|
||||||
return &TableHandler{
|
return &TableHandler{
|
||||||
tableService: tableService,
|
tableService: tableService,
|
||||||
tableValidator: tableValidator,
|
tableValidator: tableValidator,
|
||||||
baseURL: baseURL,
|
selfOrderURL: selfOrderURL,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -150,6 +150,11 @@ func (h *TableHandler) List(c *gin.Context) {
|
|||||||
Limit: 100,
|
Limit: 100,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Fallback to context outlet ID if not provided in query
|
||||||
|
if query.OutletID == "" && contextInfo.OutletID != uuid.Nil {
|
||||||
|
query.OutletID = contextInfo.OutletID.String()
|
||||||
|
}
|
||||||
|
|
||||||
if pageStr := c.Query("page"); pageStr != "" {
|
if pageStr := c.Query("page"); pageStr != "" {
|
||||||
if page, err := strconv.Atoi(pageStr); err == nil && page > 0 {
|
if page, err := strconv.Atoi(pageStr); err == nil && page > 0 {
|
||||||
query.Page = page
|
query.Page = page
|
||||||
@ -312,7 +317,7 @@ func (h *TableHandler) GenerateQRCode(c *gin.Context) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
selfOrderURL := fmt.Sprintf("%s/api/v1/self-order/table/%s", h.baseURL, token)
|
selfOrderURLResult := fmt.Sprintf("%s/menu?token=%s", h.selfOrderURL, token)
|
||||||
|
|
||||||
size := 256
|
size := 256
|
||||||
if sizeStr := c.Query("size"); sizeStr != "" {
|
if sizeStr := c.Query("size"); sizeStr != "" {
|
||||||
@ -321,7 +326,7 @@ func (h *TableHandler) GenerateQRCode(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pngBytes, err := qrcode.GeneratePNG(selfOrderURL, size)
|
pngBytes, err := qrcode.GeneratePNG(selfOrderURLResult, size)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.FromContext(ctx).WithError(err).Error("TableHandler::GenerateQRCode -> QR generation failed")
|
logger.FromContext(ctx).WithError(err).Error("TableHandler::GenerateQRCode -> QR generation failed")
|
||||||
validationResponseError := contract.NewResponseError(constants.InternalServerErrorCode, constants.TableEntity, "Failed to generate QR code")
|
validationResponseError := contract.NewResponseError(constants.InternalServerErrorCode, constants.TableEntity, "Failed to generate QR code")
|
||||||
|
|||||||
215
internal/handler/user_device_handler.go
Normal file
215
internal/handler/user_device_handler.go
Normal file
@ -0,0 +1,215 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/appcontext"
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/logger"
|
||||||
|
"apskel-pos-be/internal/service"
|
||||||
|
"apskel-pos-be/internal/util"
|
||||||
|
"apskel-pos-be/internal/validator"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDeviceHandler struct {
|
||||||
|
userDeviceService service.UserDeviceService
|
||||||
|
userDeviceValidator validator.UserDeviceValidator
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUserDeviceHandler(
|
||||||
|
userDeviceService service.UserDeviceService,
|
||||||
|
userDeviceValidator validator.UserDeviceValidator,
|
||||||
|
) *UserDeviceHandler {
|
||||||
|
return &UserDeviceHandler{
|
||||||
|
userDeviceService: userDeviceService,
|
||||||
|
userDeviceValidator: userDeviceValidator,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) RegisterDevice(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
var req contract.RegisterUserDeviceRequest
|
||||||
|
if err := c.ShouldBindJSON(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("UserDeviceHandler::RegisterDevice -> request binding failed")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, err.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::RegisterDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
validationError, validationErrorCode := h.userDeviceValidator.ValidateRegisterDeviceRequest(&req)
|
||||||
|
if validationError != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(validationErrorCode, constants.RequestEntity, validationError.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::RegisterDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.RegisterDevice(ctx, contextInfo.UserID, &req)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::RegisterDevice -> Failed to register device from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::RegisterDevice")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) UpdateDevice(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
deviceIDStr := c.Param("id")
|
||||||
|
deviceID, err := uuid.Parse(deviceIDStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("UserDeviceHandler::UpdateDevice -> Invalid device ID")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid device ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::UpdateDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
var req contract.UpdateUserDeviceRequest
|
||||||
|
if err := c.ShouldBindJSON(&req); err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("UserDeviceHandler::UpdateDevice -> request binding failed")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MissingFieldErrorCode, constants.RequestEntity, "Invalid request body")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::UpdateDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
validationError, validationErrorCode := h.userDeviceValidator.ValidateUpdateDeviceRequest(&req)
|
||||||
|
if validationError != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(validationErrorCode, constants.RequestEntity, validationError.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::UpdateDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.UpdateDevice(ctx, deviceID, &req)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::UpdateDevice -> Failed to update device from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::UpdateDevice")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) DeleteDevice(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
deviceIDStr := c.Param("id")
|
||||||
|
deviceID, err := uuid.Parse(deviceIDStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("UserDeviceHandler::DeleteDevice -> Invalid device ID")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid device ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::DeleteDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.DeleteDevice(ctx, deviceID)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::DeleteDevice -> Failed to delete device from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::DeleteDevice")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) GetDevice(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
deviceIDStr := c.Param("id")
|
||||||
|
deviceID, err := uuid.Parse(deviceIDStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("UserDeviceHandler::GetDevice -> Invalid device ID")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid device ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::GetDevice")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.GetDeviceByID(ctx, deviceID)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::GetDevice -> Failed to get device from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::GetDevice")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) GetMyDevices(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
contextInfo := appcontext.FromGinContext(ctx)
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.GetDevicesByUserID(ctx, contextInfo.UserID)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::GetMyDevices -> Failed to get devices from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::GetMyDevices")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) GetDevicesByUser(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
userIDStr := c.Param("user_id")
|
||||||
|
userID, err := uuid.Parse(userIDStr)
|
||||||
|
if err != nil {
|
||||||
|
logger.FromContext(ctx).WithError(err).Error("UserDeviceHandler::GetDevicesByUser -> Invalid user ID")
|
||||||
|
validationResponseError := contract.NewResponseError(constants.MalformedFieldErrorCode, constants.RequestEntity, "Invalid user ID")
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::GetDevicesByUser")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.GetDevicesByUserID(ctx, userID)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::GetDevicesByUser -> Failed to get devices from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::GetDevicesByUser")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *UserDeviceHandler) ListDevices(c *gin.Context) {
|
||||||
|
ctx := c.Request.Context()
|
||||||
|
|
||||||
|
req := &contract.ListUserDevicesRequest{
|
||||||
|
Page: 1,
|
||||||
|
Limit: 10,
|
||||||
|
}
|
||||||
|
|
||||||
|
if pageStr := c.Query("page"); pageStr != "" {
|
||||||
|
if page, err := strconv.Atoi(pageStr); err == nil {
|
||||||
|
req.Page = page
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if limitStr := c.Query("limit"); limitStr != "" {
|
||||||
|
if limit, err := strconv.Atoi(limitStr); err == nil {
|
||||||
|
req.Limit = limit
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if userID := c.Query("user_id"); userID != "" {
|
||||||
|
req.UserID = userID
|
||||||
|
}
|
||||||
|
|
||||||
|
if platform := c.Query("platform"); platform != "" {
|
||||||
|
req.Platform = platform
|
||||||
|
}
|
||||||
|
|
||||||
|
validationError, validationErrorCode := h.userDeviceValidator.ValidateListDevicesRequest(req)
|
||||||
|
if validationError != nil {
|
||||||
|
validationResponseError := contract.NewResponseError(validationErrorCode, constants.RequestEntity, validationError.Error())
|
||||||
|
util.HandleResponse(c.Writer, c.Request, contract.BuildErrorResponse([]*contract.ResponseError{validationResponseError}), "UserDeviceHandler::ListDevices")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponse := h.userDeviceService.ListDevices(ctx, req)
|
||||||
|
if deviceResponse.HasErrors() {
|
||||||
|
errorResp := deviceResponse.GetErrors()[0]
|
||||||
|
logger.FromContext(ctx).WithError(errorResp).Error("UserDeviceHandler::ListDevices -> Failed to list devices from service")
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HandleResponse(c.Writer, c.Request, deviceResponse, "UserDeviceHandler::ListDevices")
|
||||||
|
}
|
||||||
85
internal/mappers/notification_mapper.go
Normal file
85
internal/mappers/notification_mapper.go
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
package mappers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func NotificationEntityToResponse(e *entities.Notification) *models.NotificationResponse {
|
||||||
|
if e == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &models.NotificationResponse{
|
||||||
|
ID: e.ID,
|
||||||
|
Title: e.Title,
|
||||||
|
Body: e.Body,
|
||||||
|
Type: e.Type,
|
||||||
|
Category: e.Category,
|
||||||
|
Priority: e.Priority,
|
||||||
|
ImageURL: e.ImageURL,
|
||||||
|
ActionURL: e.ActionURL,
|
||||||
|
NotifiableType: e.NotifiableType,
|
||||||
|
NotifiableID: e.NotifiableID,
|
||||||
|
Data: e.Data,
|
||||||
|
ScheduledAt: e.ScheduledAt,
|
||||||
|
SentAt: e.SentAt,
|
||||||
|
ExpiredAt: e.ExpiredAt,
|
||||||
|
CreatedBy: e.CreatedBy,
|
||||||
|
CreatedAt: e.CreatedAt,
|
||||||
|
UpdatedAt: e.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NotificationReceiverEntityToResponse(e *entities.NotificationReceiver) *models.NotificationReceiverResponse {
|
||||||
|
if e == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
resp := &models.NotificationReceiverResponse{
|
||||||
|
ID: e.ID,
|
||||||
|
NotificationID: e.NotificationID,
|
||||||
|
UserID: e.UserID,
|
||||||
|
IsRead: e.IsRead,
|
||||||
|
ReadAt: e.ReadAt,
|
||||||
|
IsDeleted: e.IsDeleted,
|
||||||
|
DeletedAt: e.DeletedAt,
|
||||||
|
CreatedAt: e.CreatedAt,
|
||||||
|
UpdatedAt: e.UpdatedAt,
|
||||||
|
}
|
||||||
|
if e.Notification != nil {
|
||||||
|
resp.Notification = NotificationEntityToResponse(e.Notification)
|
||||||
|
}
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
|
func NotificationReceiverEntitiesToResponses(entities []*entities.NotificationReceiver) []*models.NotificationReceiverResponse {
|
||||||
|
if entities == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
responses := make([]*models.NotificationReceiverResponse, len(entities))
|
||||||
|
for i, e := range entities {
|
||||||
|
responses[i] = NotificationReceiverEntityToResponse(e)
|
||||||
|
}
|
||||||
|
return responses
|
||||||
|
}
|
||||||
|
|
||||||
|
func NotificationDeliveryEntityToResponse(e *entities.NotificationDelivery) *models.NotificationDeliveryResponse {
|
||||||
|
if e == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &models.NotificationDeliveryResponse{
|
||||||
|
ID: e.ID,
|
||||||
|
NotificationReceiverID: e.NotificationReceiverID,
|
||||||
|
UserDeviceID: e.UserDeviceID,
|
||||||
|
Channel: e.Channel,
|
||||||
|
DeliveryStatus: e.DeliveryStatus,
|
||||||
|
Provider: e.Provider,
|
||||||
|
ProviderMessageID: e.ProviderMessageID,
|
||||||
|
SentAt: e.SentAt,
|
||||||
|
DeliveredAt: e.DeliveredAt,
|
||||||
|
FailedAt: e.FailedAt,
|
||||||
|
FailureReason: e.FailureReason,
|
||||||
|
RetryCount: e.RetryCount,
|
||||||
|
CreatedAt: e.CreatedAt,
|
||||||
|
UpdatedAt: e.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -135,6 +135,7 @@ func ProductEntityToResponse(entity *entities.Product) *models.ProductResponse {
|
|||||||
Name: entity.Name,
|
Name: entity.Name,
|
||||||
Description: entity.Description,
|
Description: entity.Description,
|
||||||
Price: entity.Price,
|
Price: entity.Price,
|
||||||
|
OutletPrice: nil, // populated by processor when outletID is available
|
||||||
Cost: entity.Cost,
|
Cost: entity.Cost,
|
||||||
BusinessType: constants.BusinessType(entity.BusinessType),
|
BusinessType: constants.BusinessType(entity.BusinessType),
|
||||||
ImageURL: entity.ImageURL,
|
ImageURL: entity.ImageURL,
|
||||||
|
|||||||
48
internal/mappers/product_outlet_price_mapper.go
Normal file
48
internal/mappers/product_outlet_price_mapper.go
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
package mappers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func ProductOutletPriceEntityToModel(entity *entities.ProductOutletPrice) *models.ProductOutletPrice {
|
||||||
|
if entity == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &models.ProductOutletPrice{
|
||||||
|
ID: entity.ID,
|
||||||
|
ProductID: entity.ProductID,
|
||||||
|
OutletID: entity.OutletID,
|
||||||
|
Price: entity.Price,
|
||||||
|
CreatedAt: entity.CreatedAt,
|
||||||
|
UpdatedAt: entity.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ProductOutletPriceModelToEntity(model *models.ProductOutletPrice) *entities.ProductOutletPrice {
|
||||||
|
if model == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &entities.ProductOutletPrice{
|
||||||
|
ID: model.ID,
|
||||||
|
ProductID: model.ProductID,
|
||||||
|
OutletID: model.OutletID,
|
||||||
|
Price: model.Price,
|
||||||
|
CreatedAt: model.CreatedAt,
|
||||||
|
UpdatedAt: model.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ProductOutletPriceEntitiesToModels(entities []*entities.ProductOutletPrice) []*models.ProductOutletPrice {
|
||||||
|
if entities == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
models := make([]*models.ProductOutletPrice, len(entities))
|
||||||
|
for i, entity := range entities {
|
||||||
|
models[i] = ProductOutletPriceEntityToModel(entity)
|
||||||
|
}
|
||||||
|
return models
|
||||||
|
}
|
||||||
62
internal/mappers/user_device_mapper.go
Normal file
62
internal/mappers/user_device_mapper.go
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
package mappers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func UserDeviceEntityToModel(entity *entities.UserDevice) *models.UserDevice {
|
||||||
|
if entity == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &models.UserDevice{
|
||||||
|
ID: entity.ID,
|
||||||
|
UserID: entity.UserID,
|
||||||
|
DeviceID: entity.DeviceID,
|
||||||
|
DeviceName: entity.DeviceName,
|
||||||
|
DeviceType: entity.DeviceType,
|
||||||
|
Platform: entity.Platform,
|
||||||
|
FCMToken: entity.FCMToken,
|
||||||
|
AppVersion: entity.AppVersion,
|
||||||
|
OsVersion: entity.OsVersion,
|
||||||
|
IPAddress: entity.IPAddress,
|
||||||
|
LastActiveAt: entity.LastActiveAt,
|
||||||
|
CreatedAt: entity.CreatedAt,
|
||||||
|
UpdatedAt: entity.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UserDeviceEntityToResponse(entity *entities.UserDevice) *models.UserDeviceResponse {
|
||||||
|
if entity == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &models.UserDeviceResponse{
|
||||||
|
ID: entity.ID,
|
||||||
|
UserID: entity.UserID,
|
||||||
|
DeviceID: entity.DeviceID,
|
||||||
|
DeviceName: entity.DeviceName,
|
||||||
|
DeviceType: entity.DeviceType,
|
||||||
|
Platform: entity.Platform,
|
||||||
|
FCMToken: entity.FCMToken,
|
||||||
|
AppVersion: entity.AppVersion,
|
||||||
|
OsVersion: entity.OsVersion,
|
||||||
|
IPAddress: entity.IPAddress,
|
||||||
|
LastActiveAt: entity.LastActiveAt,
|
||||||
|
CreatedAt: entity.CreatedAt,
|
||||||
|
UpdatedAt: entity.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UserDeviceEntitiesToResponses(entities []*entities.UserDevice) []*models.UserDeviceResponse {
|
||||||
|
if entities == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
responses := make([]*models.UserDeviceResponse, len(entities))
|
||||||
|
for i, entity := range entities {
|
||||||
|
responses[i] = UserDeviceEntityToResponse(entity)
|
||||||
|
}
|
||||||
|
return responses
|
||||||
|
}
|
||||||
@ -11,6 +11,7 @@ import (
|
|||||||
"apskel-pos-be/internal/service"
|
"apskel-pos-be/internal/service"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type AuthMiddleware struct {
|
type AuthMiddleware struct {
|
||||||
@ -45,9 +46,13 @@ func (m *AuthMiddleware) RequireAuth() gin.HandlerFunc {
|
|||||||
setKeyInContext(c, appcontext.OrganizationIDKey, userResponse.OrganizationID.String())
|
setKeyInContext(c, appcontext.OrganizationIDKey, userResponse.OrganizationID.String())
|
||||||
setKeyInContext(c, appcontext.UserIDKey, userResponse.ID.String())
|
setKeyInContext(c, appcontext.UserIDKey, userResponse.ID.String())
|
||||||
|
|
||||||
if userResponse.Role != "superadmin" {
|
// Always override OutletID from token to prevent header injection.
|
||||||
setKeyInContext(c, appcontext.OutletIDKey, userResponse.OutletID.String())
|
// Set empty string if user has no outlet, so PopulateContext header value is ignored.
|
||||||
|
outletIDStr := ""
|
||||||
|
if userResponse.OutletID != nil && *userResponse.OutletID != uuid.Nil {
|
||||||
|
outletIDStr = userResponse.OutletID.String()
|
||||||
}
|
}
|
||||||
|
setKeyInContext(c, appcontext.OutletIDKey, outletIDStr)
|
||||||
|
|
||||||
logger.FromContext(c.Request.Context()).Infof("AuthMiddleware::RequireAuth -> User authenticated: %s", userResponse.Email)
|
logger.FromContext(c.Request.Context()).Infof("AuthMiddleware::RequireAuth -> User authenticated: %s", userResponse.Email)
|
||||||
c.Next()
|
c.Next()
|
||||||
|
|||||||
118
internal/models/notification.go
Normal file
118
internal/models/notification.go
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ---- Request models ----
|
||||||
|
|
||||||
|
type SendNotificationRequest struct {
|
||||||
|
Title string `json:"title"`
|
||||||
|
Body string `json:"body"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
Category string `json:"category"`
|
||||||
|
Priority entities.NotificationPriority `json:"priority"`
|
||||||
|
ImageURL string `json:"image_url"`
|
||||||
|
ActionURL string `json:"action_url"`
|
||||||
|
NotifiableType string `json:"notifiable_type"`
|
||||||
|
NotifiableID *uuid.UUID `json:"notifiable_id"`
|
||||||
|
Data map[string]interface{} `json:"data"`
|
||||||
|
ReceiverIDs []uuid.UUID `json:"receiver_ids"`
|
||||||
|
ScheduledAt *time.Time `json:"scheduled_at"`
|
||||||
|
ExpiredAt *time.Time `json:"expired_at"`
|
||||||
|
CreatedBy *uuid.UUID `json:"created_by"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type BroadcastNotificationRequest struct {
|
||||||
|
Title string `json:"title"`
|
||||||
|
Body string `json:"body"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
Category string `json:"category"`
|
||||||
|
Priority entities.NotificationPriority `json:"priority"`
|
||||||
|
ImageURL string `json:"image_url"`
|
||||||
|
ActionURL string `json:"action_url"`
|
||||||
|
NotifiableType string `json:"notifiable_type"`
|
||||||
|
NotifiableID *uuid.UUID `json:"notifiable_id"`
|
||||||
|
Data map[string]interface{} `json:"data"`
|
||||||
|
OrganizationID uuid.UUID `json:"organization_id"`
|
||||||
|
ScheduledAt *time.Time `json:"scheduled_at"`
|
||||||
|
ExpiredAt *time.Time `json:"expired_at"`
|
||||||
|
CreatedBy *uuid.UUID `json:"created_by"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type MarkNotificationReadRequest struct {
|
||||||
|
NotificationReceiverID uuid.UUID `json:"notification_receiver_id"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListNotificationsRequest struct {
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
IsRead *bool `json:"is_read"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// ---- Response models ----
|
||||||
|
|
||||||
|
type NotificationResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Body string `json:"body"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
Category string `json:"category"`
|
||||||
|
Priority entities.NotificationPriority `json:"priority"`
|
||||||
|
ImageURL string `json:"image_url"`
|
||||||
|
ActionURL string `json:"action_url"`
|
||||||
|
NotifiableType string `json:"notifiable_type"`
|
||||||
|
NotifiableID *uuid.UUID `json:"notifiable_id"`
|
||||||
|
Data map[string]interface{} `json:"data"`
|
||||||
|
ScheduledAt *time.Time `json:"scheduled_at"`
|
||||||
|
SentAt *time.Time `json:"sent_at"`
|
||||||
|
ExpiredAt *time.Time `json:"expired_at"`
|
||||||
|
CreatedBy *uuid.UUID `json:"created_by"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationReceiverResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
NotificationID uuid.UUID `json:"notification_id"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
IsRead bool `json:"is_read"`
|
||||||
|
ReadAt *time.Time `json:"read_at"`
|
||||||
|
IsDeleted bool `json:"is_deleted"`
|
||||||
|
DeletedAt *time.Time `json:"deleted_at"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
Notification *NotificationResponse `json:"notification,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationDeliveryResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
NotificationReceiverID uuid.UUID `json:"notification_receiver_id"`
|
||||||
|
UserDeviceID uuid.UUID `json:"user_device_id"`
|
||||||
|
Channel entities.NotificationChannel `json:"channel"`
|
||||||
|
DeliveryStatus entities.NotificationDeliveryStatus `json:"delivery_status"`
|
||||||
|
Provider entities.NotificationProvider `json:"provider"`
|
||||||
|
ProviderMessageID string `json:"provider_message_id"`
|
||||||
|
SentAt *time.Time `json:"sent_at"`
|
||||||
|
DeliveredAt *time.Time `json:"delivered_at"`
|
||||||
|
FailedAt *time.Time `json:"failed_at"`
|
||||||
|
FailureReason string `json:"failure_reason"`
|
||||||
|
RetryCount int `json:"retry_count"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListNotificationsResponse struct {
|
||||||
|
Notifications []*NotificationReceiverResponse `json:"notifications"`
|
||||||
|
TotalCount int `json:"total_count"`
|
||||||
|
UnreadCount int `json:"unread_count"`
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
TotalPages int `json:"total_pages"`
|
||||||
|
}
|
||||||
@ -100,6 +100,8 @@ type ProductResponse struct {
|
|||||||
Name string
|
Name string
|
||||||
Description *string
|
Description *string
|
||||||
Price float64
|
Price float64
|
||||||
|
OutletPrice *float64 // outlet-specific price, nil if not set
|
||||||
|
OutletPrices []OutletPrice // all outlet prices, populated when no outletID in context
|
||||||
Cost float64
|
Cost float64
|
||||||
BusinessType constants.BusinessType
|
BusinessType constants.BusinessType
|
||||||
ImageURL *string
|
ImageURL *string
|
||||||
@ -113,6 +115,12 @@ type ProductResponse struct {
|
|||||||
Variants []ProductVariantResponse
|
Variants []ProductVariantResponse
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type OutletPrice struct {
|
||||||
|
OutletID uuid.UUID
|
||||||
|
OutletName string
|
||||||
|
Price float64
|
||||||
|
}
|
||||||
|
|
||||||
type ProductVariantResponse struct {
|
type ProductVariantResponse struct {
|
||||||
ID uuid.UUID
|
ID uuid.UUID
|
||||||
ProductID uuid.UUID
|
ProductID uuid.UUID
|
||||||
|
|||||||
35
internal/models/product_outlet_price.go
Normal file
35
internal/models/product_outlet_price.go
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPrice struct {
|
||||||
|
ID uuid.UUID
|
||||||
|
ProductID uuid.UUID
|
||||||
|
OutletID uuid.UUID
|
||||||
|
Price float64
|
||||||
|
CreatedAt time.Time
|
||||||
|
UpdatedAt time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
type CreateProductOutletPriceRequest struct {
|
||||||
|
ProductID uuid.UUID `validate:"required"`
|
||||||
|
OutletID uuid.UUID `validate:"required"`
|
||||||
|
Price float64 `validate:"required,min=0"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UpdateProductOutletPriceRequest struct {
|
||||||
|
Price *float64 `validate:"required,min=0"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProductOutletPriceResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
ProductID uuid.UUID `json:"product_id"`
|
||||||
|
OutletID uuid.UUID `json:"outlet_id"`
|
||||||
|
Price float64 `json:"price"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
77
internal/models/user_device.go
Normal file
77
internal/models/user_device.go
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDevice struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
DeviceID string `json:"device_id"`
|
||||||
|
DeviceName string `json:"device_name"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform"`
|
||||||
|
FCMToken string `json:"fcm_token"`
|
||||||
|
AppVersion string `json:"app_version"`
|
||||||
|
OsVersion string `json:"os_version"`
|
||||||
|
IPAddress string `json:"ip_address"`
|
||||||
|
LastActiveAt *time.Time `json:"last_active_at"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserDeviceResponse struct {
|
||||||
|
ID uuid.UUID `json:"id"`
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
DeviceID string `json:"device_id"`
|
||||||
|
DeviceName string `json:"device_name"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform"`
|
||||||
|
FCMToken string `json:"fcm_token"`
|
||||||
|
AppVersion string `json:"app_version"`
|
||||||
|
OsVersion string `json:"os_version"`
|
||||||
|
IPAddress string `json:"ip_address"`
|
||||||
|
LastActiveAt *time.Time `json:"last_active_at"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type RegisterUserDeviceRequest struct {
|
||||||
|
UserID uuid.UUID `json:"user_id"`
|
||||||
|
DeviceID string `json:"device_id"`
|
||||||
|
DeviceName string `json:"device_name"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform"`
|
||||||
|
FCMToken string `json:"fcm_token"`
|
||||||
|
AppVersion string `json:"app_version"`
|
||||||
|
OsVersion string `json:"os_version"`
|
||||||
|
IPAddress string `json:"ip_address"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UpdateUserDeviceRequest struct {
|
||||||
|
DeviceName string `json:"device_name"`
|
||||||
|
DeviceType entities.DeviceType `json:"device_type"`
|
||||||
|
Platform entities.DevicePlatform `json:"platform"`
|
||||||
|
FCMToken string `json:"fcm_token"`
|
||||||
|
AppVersion string `json:"app_version"`
|
||||||
|
OsVersion string `json:"os_version"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListUserDevicesRequest struct {
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
UserID string `json:"user_id,omitempty"`
|
||||||
|
Platform string `json:"platform,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ListUserDevicesResponse struct {
|
||||||
|
Devices []*UserDeviceResponse `json:"devices"`
|
||||||
|
TotalCount int `json:"total_count"`
|
||||||
|
Page int `json:"page"`
|
||||||
|
Limit int `json:"limit"`
|
||||||
|
TotalPages int `json:"total_pages"`
|
||||||
|
}
|
||||||
338
internal/processor/notification_processor.go
Normal file
338
internal/processor/notification_processor.go
Normal file
@ -0,0 +1,338 @@
|
|||||||
|
package processor
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/client"
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/mappers"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NotificationRepository is the interface the processor depends on.
|
||||||
|
type NotificationRepository interface {
|
||||||
|
Create(ctx context.Context, notification *entities.Notification) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.Notification, error)
|
||||||
|
Update(ctx context.Context, notification *entities.Notification) error
|
||||||
|
Delete(ctx context.Context, id uuid.UUID) error
|
||||||
|
List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.Notification, int64, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationReceiverRepository is the interface the processor depends on.
|
||||||
|
type NotificationReceiverRepository interface {
|
||||||
|
Create(ctx context.Context, receiver *entities.NotificationReceiver) error
|
||||||
|
BulkCreate(ctx context.Context, receivers []*entities.NotificationReceiver) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.NotificationReceiver, error)
|
||||||
|
GetByNotificationAndUser(ctx context.Context, notificationID, userID uuid.UUID) (*entities.NotificationReceiver, error)
|
||||||
|
Update(ctx context.Context, receiver *entities.NotificationReceiver) error
|
||||||
|
ListByUserID(ctx context.Context, userID uuid.UUID, isRead *bool, limit, offset int) ([]*entities.NotificationReceiver, int64, error)
|
||||||
|
CountUnreadByUserID(ctx context.Context, userID uuid.UUID) (int64, error)
|
||||||
|
SoftDeleteByID(ctx context.Context, id uuid.UUID) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationDeliveryRepository is the interface the processor depends on.
|
||||||
|
type NotificationDeliveryRepository interface {
|
||||||
|
Create(ctx context.Context, delivery *entities.NotificationDelivery) error
|
||||||
|
BulkCreate(ctx context.Context, deliveries []*entities.NotificationDelivery) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.NotificationDelivery, error)
|
||||||
|
Update(ctx context.Context, delivery *entities.NotificationDelivery) error
|
||||||
|
ListByReceiverID(ctx context.Context, receiverID uuid.UUID) ([]*entities.NotificationDelivery, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationUserRepository is a minimal interface to fetch user devices.
|
||||||
|
type NotificationUserDeviceRepository interface {
|
||||||
|
GetByUserID(ctx context.Context, userID uuid.UUID) ([]*entities.UserDevice, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationUserRepository is a minimal interface to fetch users by org.
|
||||||
|
type NotificationUserRepository interface {
|
||||||
|
GetActiveUsers(ctx context.Context, organizationID uuid.UUID) ([]*entities.User, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NotificationProcessor defines the business logic interface.
|
||||||
|
type NotificationProcessor interface {
|
||||||
|
Send(ctx context.Context, req *models.SendNotificationRequest) (*models.NotificationResponse, error)
|
||||||
|
Broadcast(ctx context.Context, req *models.BroadcastNotificationRequest) (*models.NotificationResponse, error)
|
||||||
|
MarkAsRead(ctx context.Context, receiverID, userID uuid.UUID) (*models.NotificationReceiverResponse, error)
|
||||||
|
MarkAllAsRead(ctx context.Context, userID uuid.UUID) error
|
||||||
|
DeleteForUser(ctx context.Context, receiverID, userID uuid.UUID) error
|
||||||
|
ListForUser(ctx context.Context, req *models.ListNotificationsRequest) ([]*models.NotificationReceiverResponse, int64, int64, error)
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*models.NotificationResponse, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationProcessorImpl struct {
|
||||||
|
notificationRepo NotificationRepository
|
||||||
|
receiverRepo NotificationReceiverRepository
|
||||||
|
deliveryRepo NotificationDeliveryRepository
|
||||||
|
userDeviceRepo NotificationUserDeviceRepository
|
||||||
|
userRepo NotificationUserRepository
|
||||||
|
fcmClient client.FCMClient
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNotificationProcessor(
|
||||||
|
notificationRepo NotificationRepository,
|
||||||
|
receiverRepo NotificationReceiverRepository,
|
||||||
|
deliveryRepo NotificationDeliveryRepository,
|
||||||
|
userDeviceRepo NotificationUserDeviceRepository,
|
||||||
|
userRepo NotificationUserRepository,
|
||||||
|
fcmClient client.FCMClient,
|
||||||
|
) *NotificationProcessorImpl {
|
||||||
|
return &NotificationProcessorImpl{
|
||||||
|
notificationRepo: notificationRepo,
|
||||||
|
receiverRepo: receiverRepo,
|
||||||
|
deliveryRepo: deliveryRepo,
|
||||||
|
userDeviceRepo: userDeviceRepo,
|
||||||
|
userRepo: userRepo,
|
||||||
|
fcmClient: fcmClient,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send creates a notification and dispatches it to the given receiver user IDs via FCM.
|
||||||
|
func (p *NotificationProcessorImpl) Send(ctx context.Context, req *models.SendNotificationRequest) (*models.NotificationResponse, error) {
|
||||||
|
if len(req.ReceiverIDs) == 0 {
|
||||||
|
return nil, fmt.Errorf("at least one receiver_id is required")
|
||||||
|
}
|
||||||
|
|
||||||
|
notification := &entities.Notification{
|
||||||
|
Title: req.Title,
|
||||||
|
Body: req.Body,
|
||||||
|
Type: req.Type,
|
||||||
|
Category: req.Category,
|
||||||
|
Priority: req.Priority,
|
||||||
|
ImageURL: req.ImageURL,
|
||||||
|
ActionURL: req.ActionURL,
|
||||||
|
NotifiableType: req.NotifiableType,
|
||||||
|
NotifiableID: req.NotifiableID,
|
||||||
|
Data: req.Data,
|
||||||
|
ScheduledAt: req.ScheduledAt,
|
||||||
|
ExpiredAt: req.ExpiredAt,
|
||||||
|
CreatedBy: req.CreatedBy,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := p.notificationRepo.Create(ctx, notification); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create notification: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create receiver records and dispatch FCM per user.
|
||||||
|
for _, userID := range req.ReceiverIDs {
|
||||||
|
receiver := &entities.NotificationReceiver{
|
||||||
|
NotificationID: notification.ID,
|
||||||
|
UserID: userID,
|
||||||
|
}
|
||||||
|
if err := p.receiverRepo.Create(ctx, receiver); err != nil {
|
||||||
|
// Log but continue for other receivers.
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
p.dispatchFCMToUser(ctx, receiver, notification)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mark notification as sent.
|
||||||
|
now := time.Now()
|
||||||
|
notification.SentAt = &now
|
||||||
|
_ = p.notificationRepo.Update(ctx, notification)
|
||||||
|
|
||||||
|
return mappers.NotificationEntityToResponse(notification), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Broadcast sends a notification to all active users in an organization.
|
||||||
|
func (p *NotificationProcessorImpl) Broadcast(ctx context.Context, req *models.BroadcastNotificationRequest) (*models.NotificationResponse, error) {
|
||||||
|
users, err := p.userRepo.GetActiveUsers(ctx, req.OrganizationID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to fetch organization users: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
notification := &entities.Notification{
|
||||||
|
Title: req.Title,
|
||||||
|
Body: req.Body,
|
||||||
|
Type: req.Type,
|
||||||
|
Category: req.Category,
|
||||||
|
Priority: req.Priority,
|
||||||
|
ImageURL: req.ImageURL,
|
||||||
|
ActionURL: req.ActionURL,
|
||||||
|
NotifiableType: req.NotifiableType,
|
||||||
|
NotifiableID: req.NotifiableID,
|
||||||
|
Data: req.Data,
|
||||||
|
ScheduledAt: req.ScheduledAt,
|
||||||
|
ExpiredAt: req.ExpiredAt,
|
||||||
|
CreatedBy: req.CreatedBy,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := p.notificationRepo.Create(ctx, notification); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create notification: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build receiver records in bulk.
|
||||||
|
receivers := make([]*entities.NotificationReceiver, 0, len(users))
|
||||||
|
for _, u := range users {
|
||||||
|
receivers = append(receivers, &entities.NotificationReceiver{
|
||||||
|
NotificationID: notification.ID,
|
||||||
|
UserID: u.ID,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := p.receiverRepo.BulkCreate(ctx, receivers); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create notification receivers: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Dispatch FCM for each receiver.
|
||||||
|
for _, receiver := range receivers {
|
||||||
|
p.dispatchFCMToUser(ctx, receiver, notification)
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
notification.SentAt = &now
|
||||||
|
_ = p.notificationRepo.Update(ctx, notification)
|
||||||
|
|
||||||
|
return mappers.NotificationEntityToResponse(notification), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// MarkAsRead marks a single notification receiver record as read.
|
||||||
|
func (p *NotificationProcessorImpl) MarkAsRead(ctx context.Context, receiverID, userID uuid.UUID) (*models.NotificationReceiverResponse, error) {
|
||||||
|
receiver, err := p.receiverRepo.GetByID(ctx, receiverID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("notification not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if receiver.UserID != userID {
|
||||||
|
return nil, fmt.Errorf("unauthorized: notification does not belong to user")
|
||||||
|
}
|
||||||
|
|
||||||
|
if !receiver.IsRead {
|
||||||
|
now := time.Now()
|
||||||
|
receiver.IsRead = true
|
||||||
|
receiver.ReadAt = &now
|
||||||
|
if err := p.receiverRepo.Update(ctx, receiver); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to mark notification as read: %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.NotificationReceiverEntityToResponse(receiver), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// MarkAllAsRead marks all unread notifications for a user as read.
|
||||||
|
func (p *NotificationProcessorImpl) MarkAllAsRead(ctx context.Context, userID uuid.UUID) error {
|
||||||
|
isRead := false
|
||||||
|
receivers, _, err := p.receiverRepo.ListByUserID(ctx, userID, &isRead, 1000, 0)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to fetch unread notifications: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
for _, r := range receivers {
|
||||||
|
r.IsRead = true
|
||||||
|
r.ReadAt = &now
|
||||||
|
_ = p.receiverRepo.Update(ctx, r)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// DeleteForUser soft-deletes a notification receiver record for a user.
|
||||||
|
func (p *NotificationProcessorImpl) DeleteForUser(ctx context.Context, receiverID, userID uuid.UUID) error {
|
||||||
|
receiver, err := p.receiverRepo.GetByID(ctx, receiverID)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("notification not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if receiver.UserID != userID {
|
||||||
|
return fmt.Errorf("unauthorized: notification does not belong to user")
|
||||||
|
}
|
||||||
|
|
||||||
|
return p.receiverRepo.SoftDeleteByID(ctx, receiverID)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListForUser returns paginated notifications for a user.
|
||||||
|
// Returns: receivers, total, unreadCount, error
|
||||||
|
func (p *NotificationProcessorImpl) ListForUser(ctx context.Context, req *models.ListNotificationsRequest) ([]*models.NotificationReceiverResponse, int64, int64, error) {
|
||||||
|
offset := (req.Page - 1) * req.Limit
|
||||||
|
|
||||||
|
receivers, total, err := p.receiverRepo.ListByUserID(ctx, req.UserID, req.IsRead, req.Limit, offset)
|
||||||
|
if err != nil {
|
||||||
|
return nil, 0, 0, fmt.Errorf("failed to list notifications: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
unreadCount, err := p.receiverRepo.CountUnreadByUserID(ctx, req.UserID)
|
||||||
|
if err != nil {
|
||||||
|
unreadCount = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
responses := mappers.NotificationReceiverEntitiesToResponses(receivers)
|
||||||
|
return responses, total, unreadCount, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetByID returns a single notification by its ID.
|
||||||
|
func (p *NotificationProcessorImpl) GetByID(ctx context.Context, id uuid.UUID) (*models.NotificationResponse, error) {
|
||||||
|
notification, err := p.notificationRepo.GetByID(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("notification not found: %w", err)
|
||||||
|
}
|
||||||
|
return mappers.NotificationEntityToResponse(notification), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// dispatchFCMToUser fetches all FCM tokens for a user and sends the push notification.
|
||||||
|
func (p *NotificationProcessorImpl) dispatchFCMToUser(ctx context.Context, receiver *entities.NotificationReceiver, notification *entities.Notification) {
|
||||||
|
if p.fcmClient == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
devices, err := p.userDeviceRepo.GetByUserID(ctx, receiver.UserID)
|
||||||
|
if err != nil || len(devices) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Build FCM data payload.
|
||||||
|
data := map[string]string{
|
||||||
|
"notification_id": notification.ID.String(),
|
||||||
|
"notification_receiver_id": receiver.ID.String(),
|
||||||
|
"type": notification.Type,
|
||||||
|
"category": notification.Category,
|
||||||
|
"action_url": notification.ActionURL,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect valid FCM tokens and create delivery records.
|
||||||
|
tokens := make([]string, 0, len(devices))
|
||||||
|
deliveries := make([]*entities.NotificationDelivery, 0, len(devices))
|
||||||
|
|
||||||
|
for _, device := range devices {
|
||||||
|
if device.FCMToken == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
tokens = append(tokens, device.FCMToken)
|
||||||
|
deliveries = append(deliveries, &entities.NotificationDelivery{
|
||||||
|
NotificationReceiverID: receiver.ID,
|
||||||
|
UserDeviceID: device.ID,
|
||||||
|
Channel: entities.NotificationChannelPush,
|
||||||
|
DeliveryStatus: entities.NotificationDeliveryStatusPending,
|
||||||
|
Provider: entities.NotificationProviderFirebase,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(tokens) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Persist delivery records before sending.
|
||||||
|
_ = p.deliveryRepo.BulkCreate(ctx, deliveries)
|
||||||
|
|
||||||
|
// Send via FCM multicast.
|
||||||
|
now := time.Now()
|
||||||
|
sendErr := p.fcmClient.SendMulticastNotification(ctx, tokens, notification.Title, notification.Body, data)
|
||||||
|
|
||||||
|
// Update delivery status.
|
||||||
|
for _, delivery := range deliveries {
|
||||||
|
if sendErr != nil {
|
||||||
|
delivery.DeliveryStatus = entities.NotificationDeliveryStatusFailed
|
||||||
|
delivery.FailedAt = &now
|
||||||
|
delivery.FailureReason = sendErr.Error()
|
||||||
|
} else {
|
||||||
|
delivery.DeliveryStatus = entities.NotificationDeliveryStatusSent
|
||||||
|
delivery.SentAt = &now
|
||||||
|
}
|
||||||
|
_ = p.deliveryRepo.Update(ctx, delivery)
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -108,6 +108,7 @@ type OrderProcessorImpl struct {
|
|||||||
productRecipeRepo *repository.ProductRecipeRepository
|
productRecipeRepo *repository.ProductRecipeRepository
|
||||||
ingredientRepo IngredientRepository
|
ingredientRepo IngredientRepository
|
||||||
inventoryMovementService InventoryMovementService
|
inventoryMovementService InventoryMovementService
|
||||||
|
productOutletPriceRepo repository.ProductOutletPriceRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewOrderProcessorImpl(
|
func NewOrderProcessorImpl(
|
||||||
@ -126,6 +127,7 @@ func NewOrderProcessorImpl(
|
|||||||
productRecipeRepo *repository.ProductRecipeRepository,
|
productRecipeRepo *repository.ProductRecipeRepository,
|
||||||
ingredientRepo IngredientRepository,
|
ingredientRepo IngredientRepository,
|
||||||
inventoryMovementService InventoryMovementService,
|
inventoryMovementService InventoryMovementService,
|
||||||
|
productOutletPriceRepo repository.ProductOutletPriceRepository,
|
||||||
) *OrderProcessorImpl {
|
) *OrderProcessorImpl {
|
||||||
return &OrderProcessorImpl{
|
return &OrderProcessorImpl{
|
||||||
orderRepo: orderRepo,
|
orderRepo: orderRepo,
|
||||||
@ -144,6 +146,7 @@ func NewOrderProcessorImpl(
|
|||||||
productRecipeRepo: productRecipeRepo,
|
productRecipeRepo: productRecipeRepo,
|
||||||
ingredientRepo: ingredientRepo,
|
ingredientRepo: ingredientRepo,
|
||||||
inventoryMovementService: inventoryMovementService,
|
inventoryMovementService: inventoryMovementService,
|
||||||
|
productOutletPriceRepo: productOutletPriceRepo,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -170,6 +173,12 @@ func (p *OrderProcessorImpl) CreateOrder(ctx context.Context, req *models.Create
|
|||||||
unitPrice := product.Price
|
unitPrice := product.Price
|
||||||
unitCost := product.Cost
|
unitCost := product.Cost
|
||||||
|
|
||||||
|
if p.productOutletPriceRepo != nil {
|
||||||
|
if outletPrice, err := p.productOutletPriceRepo.GetByProductAndOutlet(ctx, itemReq.ProductID, req.OutletID); err == nil {
|
||||||
|
unitPrice = outletPrice.Price
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if itemReq.ProductVariantID != nil {
|
if itemReq.ProductVariantID != nil {
|
||||||
variant, err := p.productVariantRepo.GetByID(ctx, *itemReq.ProductVariantID)
|
variant, err := p.productVariantRepo.GetByID(ctx, *itemReq.ProductVariantID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -293,6 +302,12 @@ func (p *OrderProcessorImpl) AddToOrder(ctx context.Context, orderID uuid.UUID,
|
|||||||
unitPrice := product.Price
|
unitPrice := product.Price
|
||||||
unitCost := product.Cost
|
unitCost := product.Cost
|
||||||
|
|
||||||
|
if p.productOutletPriceRepo != nil {
|
||||||
|
if outletPrice, err := p.productOutletPriceRepo.GetByProductAndOutlet(ctx, itemReq.ProductID, order.OutletID); err == nil {
|
||||||
|
unitPrice = outletPrice.Price
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Handle product variant if specified
|
// Handle product variant if specified
|
||||||
if itemReq.ProductVariantID != nil {
|
if itemReq.ProductVariantID != nil {
|
||||||
variant, err := p.productVariantRepo.GetByID(ctx, *itemReq.ProductVariantID)
|
variant, err := p.productVariantRepo.GetByID(ctx, *itemReq.ProductVariantID)
|
||||||
|
|||||||
121
internal/processor/product_outlet_price_processor.go
Normal file
121
internal/processor/product_outlet_price_processor.go
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
package processor
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/mappers"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
"apskel-pos-be/internal/repository"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPriceProcessor interface {
|
||||||
|
Upsert(ctx context.Context, req *models.CreateProductOutletPriceRequest) (*models.ProductOutletPrice, error)
|
||||||
|
GetByProductAndOutlet(ctx context.Context, productID, outletID uuid.UUID) (*models.ProductOutletPrice, error)
|
||||||
|
GetByProduct(ctx context.Context, productID uuid.UUID) ([]*models.ProductOutletPrice, error)
|
||||||
|
GetByOutlet(ctx context.Context, outletID uuid.UUID) ([]*models.ProductOutletPrice, error)
|
||||||
|
Delete(ctx context.Context, id uuid.UUID) error
|
||||||
|
ResolvePrice(ctx context.Context, productID, outletID uuid.UUID, fallbackPrice float64) float64
|
||||||
|
BulkUpsert(ctx context.Context, productID uuid.UUID, prices []models.CreateProductOutletPriceRequest) ([]*models.ProductOutletPrice, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProductOutletPriceProcessorImpl struct {
|
||||||
|
repo repository.ProductOutletPriceRepository
|
||||||
|
productRepo ProductRepository
|
||||||
|
outletRepo OutletRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewProductOutletPriceProcessorImpl(repo repository.ProductOutletPriceRepository, productRepo ProductRepository, outletRepo OutletRepository) *ProductOutletPriceProcessorImpl {
|
||||||
|
return &ProductOutletPriceProcessorImpl{
|
||||||
|
repo: repo,
|
||||||
|
productRepo: productRepo,
|
||||||
|
outletRepo: outletRepo,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) Upsert(ctx context.Context, req *models.CreateProductOutletPriceRequest) (*models.ProductOutletPrice, error) {
|
||||||
|
if _, err := p.productRepo.GetByID(ctx, req.ProductID); err != nil {
|
||||||
|
return nil, fmt.Errorf("product not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := p.outletRepo.GetByID(ctx, req.OutletID); err != nil {
|
||||||
|
return nil, fmt.Errorf("outlet not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
entity := &entities.ProductOutletPrice{
|
||||||
|
ProductID: req.ProductID,
|
||||||
|
OutletID: req.OutletID,
|
||||||
|
Price: req.Price,
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := p.repo.Upsert(ctx, entity); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to upsert product outlet price: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
actual, err := p.repo.GetByProductAndOutlet(ctx, req.ProductID, req.OutletID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to retrieve upserted product outlet price: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.ProductOutletPriceEntityToModel(actual), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) GetByProductAndOutlet(ctx context.Context, productID, outletID uuid.UUID) (*models.ProductOutletPrice, error) {
|
||||||
|
entity, err := p.repo.GetByProductAndOutlet(ctx, productID, outletID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("product outlet price not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.ProductOutletPriceEntityToModel(entity), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) GetByProduct(ctx context.Context, productID uuid.UUID) ([]*models.ProductOutletPrice, error) {
|
||||||
|
entities, err := p.repo.GetByProduct(ctx, productID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to get product outlet prices: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.ProductOutletPriceEntitiesToModels(entities), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) GetByOutlet(ctx context.Context, outletID uuid.UUID) ([]*models.ProductOutletPrice, error) {
|
||||||
|
entities, err := p.repo.GetByOutlet(ctx, outletID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to get outlet prices: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.ProductOutletPriceEntitiesToModels(entities), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) Delete(ctx context.Context, id uuid.UUID) error {
|
||||||
|
if err := p.repo.Delete(ctx, id); err != nil {
|
||||||
|
return fmt.Errorf("failed to delete product outlet price: %w", err)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) ResolvePrice(ctx context.Context, productID, outletID uuid.UUID, fallbackPrice float64) float64 {
|
||||||
|
outletPrice, err := p.repo.GetByProductAndOutlet(ctx, productID, outletID)
|
||||||
|
if err != nil {
|
||||||
|
return fallbackPrice
|
||||||
|
}
|
||||||
|
return outletPrice.Price
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductOutletPriceProcessorImpl) BulkUpsert(ctx context.Context, productID uuid.UUID, prices []models.CreateProductOutletPriceRequest) ([]*models.ProductOutletPrice, error) {
|
||||||
|
var results []*models.ProductOutletPrice
|
||||||
|
|
||||||
|
for _, req := range prices {
|
||||||
|
req.ProductID = productID
|
||||||
|
result, err := p.Upsert(ctx, &req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to upsert price for outlet %s: %w", req.OutletID, err)
|
||||||
|
}
|
||||||
|
results = append(results, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
return results, nil
|
||||||
|
}
|
||||||
@ -16,8 +16,9 @@ type ProductProcessor interface {
|
|||||||
CreateProduct(ctx context.Context, req *models.CreateProductRequest) (*models.ProductResponse, error)
|
CreateProduct(ctx context.Context, req *models.CreateProductRequest) (*models.ProductResponse, error)
|
||||||
UpdateProduct(ctx context.Context, id uuid.UUID, req *models.UpdateProductRequest) (*models.ProductResponse, error)
|
UpdateProduct(ctx context.Context, id uuid.UUID, req *models.UpdateProductRequest) (*models.ProductResponse, error)
|
||||||
DeleteProduct(ctx context.Context, id uuid.UUID) error
|
DeleteProduct(ctx context.Context, id uuid.UUID) error
|
||||||
GetProductByID(ctx context.Context, id uuid.UUID) (*models.ProductResponse, error)
|
GetProductByID(ctx context.Context, id uuid.UUID, outletID uuid.UUID) (*models.ProductResponse, error)
|
||||||
ListProducts(ctx context.Context, filters map[string]interface{}, page, limit int) ([]models.ProductResponse, int, error)
|
ListProducts(ctx context.Context, filters map[string]interface{}, page, limit int) ([]models.ProductResponse, int, error)
|
||||||
|
ListProductsAll(ctx context.Context, filters map[string]interface{}, page, limit int) ([]models.ProductResponse, int, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type ProductRepository interface {
|
type ProductRepository interface {
|
||||||
@ -32,6 +33,7 @@ type ProductRepository interface {
|
|||||||
Update(ctx context.Context, product *entities.Product) error
|
Update(ctx context.Context, product *entities.Product) error
|
||||||
Delete(ctx context.Context, id uuid.UUID) error
|
Delete(ctx context.Context, id uuid.UUID) error
|
||||||
List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.Product, int64, error)
|
List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.Product, int64, error)
|
||||||
|
ListWithOutletPrice(ctx context.Context, filters map[string]interface{}, outletID uuid.UUID, limit, offset int) ([]*entities.Product, int64, error)
|
||||||
Count(ctx context.Context, filters map[string]interface{}) (int64, error)
|
Count(ctx context.Context, filters map[string]interface{}) (int64, error)
|
||||||
GetBySKU(ctx context.Context, organizationID uuid.UUID, sku string) (*entities.Product, error)
|
GetBySKU(ctx context.Context, organizationID uuid.UUID, sku string) (*entities.Product, error)
|
||||||
ExistsBySKU(ctx context.Context, organizationID uuid.UUID, sku string, excludeID *uuid.UUID) (bool, error)
|
ExistsBySKU(ctx context.Context, organizationID uuid.UUID, sku string, excludeID *uuid.UUID) (bool, error)
|
||||||
@ -47,15 +49,17 @@ type ProductProcessorImpl struct {
|
|||||||
productVariantRepo repository.ProductVariantRepository
|
productVariantRepo repository.ProductVariantRepository
|
||||||
inventoryRepo repository.InventoryRepository
|
inventoryRepo repository.InventoryRepository
|
||||||
outletRepo OutletRepository
|
outletRepo OutletRepository
|
||||||
|
outletPriceRepo repository.ProductOutletPriceRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewProductProcessorImpl(productRepo ProductRepository, categoryRepo CategoryRepository, productVariantRepo repository.ProductVariantRepository, inventoryRepo repository.InventoryRepository, outletRepo OutletRepository) *ProductProcessorImpl {
|
func NewProductProcessorImpl(productRepo ProductRepository, categoryRepo CategoryRepository, productVariantRepo repository.ProductVariantRepository, inventoryRepo repository.InventoryRepository, outletRepo OutletRepository, outletPriceRepo repository.ProductOutletPriceRepository) *ProductProcessorImpl {
|
||||||
return &ProductProcessorImpl{
|
return &ProductProcessorImpl{
|
||||||
productRepo: productRepo,
|
productRepo: productRepo,
|
||||||
categoryRepo: categoryRepo,
|
categoryRepo: categoryRepo,
|
||||||
productVariantRepo: productVariantRepo,
|
productVariantRepo: productVariantRepo,
|
||||||
inventoryRepo: inventoryRepo,
|
inventoryRepo: inventoryRepo,
|
||||||
outletRepo: outletRepo,
|
outletRepo: outletRepo,
|
||||||
|
outletPriceRepo: outletPriceRepo,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -214,19 +218,79 @@ func (p *ProductProcessorImpl) DeleteProduct(ctx context.Context, id uuid.UUID)
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ProductProcessorImpl) GetProductByID(ctx context.Context, id uuid.UUID) (*models.ProductResponse, error) {
|
func (p *ProductProcessorImpl) GetProductByID(ctx context.Context, id uuid.UUID, outletID uuid.UUID) (*models.ProductResponse, error) {
|
||||||
productEntity, err := p.productRepo.GetWithCategory(ctx, id)
|
productEntity, err := p.productRepo.GetWithCategory(ctx, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("product not found: %w", err)
|
return nil, fmt.Errorf("product not found: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
response := mappers.ProductEntityToResponse(productEntity)
|
response := mappers.ProductEntityToResponse(productEntity)
|
||||||
|
|
||||||
|
if outletID != uuid.Nil {
|
||||||
|
// Attach outlet-specific price
|
||||||
|
outletPrice, err := p.outletPriceRepo.GetByProductAndOutlet(ctx, id, outletID)
|
||||||
|
if err == nil {
|
||||||
|
response.OutletPrice = &outletPrice.Price
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// No outlet context — return all outlet prices for this product
|
||||||
|
outletPrices, err := p.outletPriceRepo.GetByProductWithOutlet(ctx, id)
|
||||||
|
if err == nil && len(outletPrices) > 0 {
|
||||||
|
prices := make([]models.OutletPrice, len(outletPrices))
|
||||||
|
for i, op := range outletPrices {
|
||||||
|
prices[i] = models.OutletPrice{
|
||||||
|
OutletID: op.OutletID,
|
||||||
|
OutletName: op.Outlet.Name,
|
||||||
|
Price: op.Price,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
response.OutletPrices = prices
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return response, nil
|
return response, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *ProductProcessorImpl) ListProducts(ctx context.Context, filters map[string]interface{}, page, limit int) ([]models.ProductResponse, int, error) {
|
func (p *ProductProcessorImpl) ListProducts(ctx context.Context, filters map[string]interface{}, page, limit int) ([]models.ProductResponse, int, error) {
|
||||||
offset := (page - 1) * limit
|
offset := (page - 1) * limit
|
||||||
|
|
||||||
|
// Extract outletID from filters — it's not a products column so remove it before querying
|
||||||
|
var outletID uuid.UUID
|
||||||
|
if oid, ok := filters["outlet_id"]; ok {
|
||||||
|
outletID = oid.(uuid.UUID)
|
||||||
|
delete(filters, "outlet_id")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use the JOIN-based query when an outlet is specified so we get outlet-specific
|
||||||
|
// prices in a single round-trip; fall back to the plain List otherwise.
|
||||||
|
var (
|
||||||
|
productEntities []*entities.Product
|
||||||
|
total int64
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
if outletID != uuid.Nil {
|
||||||
|
productEntities, total, err = p.productRepo.ListWithOutletPrice(ctx, filters, outletID, limit, offset)
|
||||||
|
} else {
|
||||||
|
productEntities, total, err = p.productRepo.List(ctx, filters, limit, offset)
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return nil, 0, fmt.Errorf("failed to list products: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
responses := make([]models.ProductResponse, len(productEntities))
|
||||||
|
for i, entity := range productEntities {
|
||||||
|
response := mappers.ProductEntityToResponse(entity)
|
||||||
|
if response != nil {
|
||||||
|
responses[i] = *response
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return responses, int(total), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *ProductProcessorImpl) ListProductsAll(ctx context.Context, filters map[string]interface{}, page, limit int) ([]models.ProductResponse, int, error) {
|
||||||
|
offset := (page - 1) * limit
|
||||||
|
|
||||||
productEntities, total, err := p.productRepo.List(ctx, filters, limit, offset)
|
productEntities, total, err := p.productRepo.List(ctx, filters, limit, offset)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, 0, fmt.Errorf("failed to list products: %w", err)
|
return nil, 0, fmt.Errorf("failed to list products: %w", err)
|
||||||
|
|||||||
@ -4,6 +4,7 @@ import (
|
|||||||
"apskel-pos-be/internal/constants"
|
"apskel-pos-be/internal/constants"
|
||||||
"apskel-pos-be/internal/entities"
|
"apskel-pos-be/internal/entities"
|
||||||
"apskel-pos-be/internal/models"
|
"apskel-pos-be/internal/models"
|
||||||
|
"apskel-pos-be/internal/pkg/tabletoken"
|
||||||
"apskel-pos-be/internal/repository"
|
"apskel-pos-be/internal/repository"
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
@ -212,6 +213,15 @@ func (p *TableProcessor) GetTokenByID(ctx context.Context, id uuid.UUID) (string
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if _, _, _, err := tabletoken.Decode(table.Token); err != nil {
|
||||||
|
newToken := tabletoken.Encode(table.ID, table.OrganizationID, table.OutletID)
|
||||||
|
if updateErr := p.tableRepo.UpdateToken(ctx, table.ID, newToken); updateErr != nil {
|
||||||
|
return "", updateErr
|
||||||
|
}
|
||||||
|
return newToken, nil
|
||||||
|
}
|
||||||
|
|
||||||
return table.Token, nil
|
return table.Token, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
165
internal/processor/user_device_processor.go
Normal file
165
internal/processor/user_device_processor.go
Normal file
@ -0,0 +1,165 @@
|
|||||||
|
package processor
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/mappers"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDeviceRepository interface {
|
||||||
|
Create(ctx context.Context, device *entities.UserDevice) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.UserDevice, error)
|
||||||
|
GetByDeviceID(ctx context.Context, deviceID string, userID uuid.UUID) (*entities.UserDevice, error)
|
||||||
|
GetByUserID(ctx context.Context, userID uuid.UUID) ([]*entities.UserDevice, error)
|
||||||
|
Update(ctx context.Context, device *entities.UserDevice) error
|
||||||
|
Delete(ctx context.Context, id uuid.UUID) error
|
||||||
|
DeleteByUserID(ctx context.Context, userID uuid.UUID) error
|
||||||
|
List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.UserDevice, int64, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserDeviceProcessor interface {
|
||||||
|
RegisterDevice(ctx context.Context, req *models.RegisterUserDeviceRequest) (*models.UserDeviceResponse, error)
|
||||||
|
UpdateDevice(ctx context.Context, id uuid.UUID, req *models.UpdateUserDeviceRequest) (*models.UserDeviceResponse, error)
|
||||||
|
DeleteDevice(ctx context.Context, id uuid.UUID) error
|
||||||
|
GetDeviceByID(ctx context.Context, id uuid.UUID) (*models.UserDeviceResponse, error)
|
||||||
|
GetDevicesByUserID(ctx context.Context, userID uuid.UUID) ([]*models.UserDeviceResponse, error)
|
||||||
|
ListDevices(ctx context.Context, filters map[string]interface{}, page, limit int) ([]*models.UserDeviceResponse, int, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserDeviceProcessorImpl struct {
|
||||||
|
userDeviceRepo UserDeviceRepository
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUserDeviceProcessorImpl(userDeviceRepo UserDeviceRepository) *UserDeviceProcessorImpl {
|
||||||
|
return &UserDeviceProcessorImpl{
|
||||||
|
userDeviceRepo: userDeviceRepo,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *UserDeviceProcessorImpl) RegisterDevice(ctx context.Context, req *models.RegisterUserDeviceRequest) (*models.UserDeviceResponse, error) {
|
||||||
|
// Upsert: if device already registered for this user, update it
|
||||||
|
existing, err := p.userDeviceRepo.GetByDeviceID(ctx, req.DeviceID, req.UserID)
|
||||||
|
if err == nil && existing != nil {
|
||||||
|
existing.DeviceName = req.DeviceName
|
||||||
|
existing.DeviceType = req.DeviceType
|
||||||
|
existing.Platform = req.Platform
|
||||||
|
existing.FCMToken = req.FCMToken
|
||||||
|
existing.AppVersion = req.AppVersion
|
||||||
|
existing.OsVersion = req.OsVersion
|
||||||
|
existing.IPAddress = req.IPAddress
|
||||||
|
now := time.Now()
|
||||||
|
existing.LastActiveAt = &now
|
||||||
|
|
||||||
|
if err := p.userDeviceRepo.Update(ctx, existing); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to update device: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.UserDeviceEntityToResponse(existing), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceEntity := &entities.UserDevice{
|
||||||
|
UserID: req.UserID,
|
||||||
|
DeviceID: req.DeviceID,
|
||||||
|
DeviceName: req.DeviceName,
|
||||||
|
DeviceType: req.DeviceType,
|
||||||
|
Platform: req.Platform,
|
||||||
|
FCMToken: req.FCMToken,
|
||||||
|
AppVersion: req.AppVersion,
|
||||||
|
OsVersion: req.OsVersion,
|
||||||
|
IPAddress: req.IPAddress,
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
deviceEntity.LastActiveAt = &now
|
||||||
|
|
||||||
|
if err := p.userDeviceRepo.Create(ctx, deviceEntity); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to register device: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.UserDeviceEntityToResponse(deviceEntity), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *UserDeviceProcessorImpl) UpdateDevice(ctx context.Context, id uuid.UUID, req *models.UpdateUserDeviceRequest) (*models.UserDeviceResponse, error) {
|
||||||
|
deviceEntity, err := p.userDeviceRepo.GetByID(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("device not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.DeviceName != "" {
|
||||||
|
deviceEntity.DeviceName = req.DeviceName
|
||||||
|
}
|
||||||
|
if req.DeviceType != "" {
|
||||||
|
deviceEntity.DeviceType = req.DeviceType
|
||||||
|
}
|
||||||
|
if req.Platform != "" {
|
||||||
|
deviceEntity.Platform = req.Platform
|
||||||
|
}
|
||||||
|
if req.FCMToken != "" {
|
||||||
|
deviceEntity.FCMToken = req.FCMToken
|
||||||
|
}
|
||||||
|
if req.AppVersion != "" {
|
||||||
|
deviceEntity.AppVersion = req.AppVersion
|
||||||
|
}
|
||||||
|
if req.OsVersion != "" {
|
||||||
|
deviceEntity.OsVersion = req.OsVersion
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now()
|
||||||
|
deviceEntity.LastActiveAt = &now
|
||||||
|
|
||||||
|
if err := p.userDeviceRepo.Update(ctx, deviceEntity); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to update device: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.UserDeviceEntityToResponse(deviceEntity), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *UserDeviceProcessorImpl) DeleteDevice(ctx context.Context, id uuid.UUID) error {
|
||||||
|
_, err := p.userDeviceRepo.GetByID(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("device not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := p.userDeviceRepo.Delete(ctx, id); err != nil {
|
||||||
|
return fmt.Errorf("failed to delete device: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *UserDeviceProcessorImpl) GetDeviceByID(ctx context.Context, id uuid.UUID) (*models.UserDeviceResponse, error) {
|
||||||
|
deviceEntity, err := p.userDeviceRepo.GetByID(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("device not found: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.UserDeviceEntityToResponse(deviceEntity), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *UserDeviceProcessorImpl) GetDevicesByUserID(ctx context.Context, userID uuid.UUID) ([]*models.UserDeviceResponse, error) {
|
||||||
|
deviceEntities, err := p.userDeviceRepo.GetByUserID(ctx, userID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to get devices: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappers.UserDeviceEntitiesToResponses(deviceEntities), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *UserDeviceProcessorImpl) ListDevices(ctx context.Context, filters map[string]interface{}, page, limit int) ([]*models.UserDeviceResponse, int, error) {
|
||||||
|
offset := (page - 1) * limit
|
||||||
|
deviceEntities, total, err := p.userDeviceRepo.List(ctx, filters, limit, offset)
|
||||||
|
if err != nil {
|
||||||
|
return nil, 0, fmt.Errorf("failed to list devices: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
deviceResponses := mappers.UserDeviceEntitiesToResponses(deviceEntities)
|
||||||
|
totalPages := int((total + int64(limit) - 1) / int64(limit))
|
||||||
|
|
||||||
|
return deviceResponses, totalPages, nil
|
||||||
|
}
|
||||||
@ -29,6 +29,13 @@ func NewAnalyticsRepositoryImpl(db *gorm.DB) *AnalyticsRepositoryImpl {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *AnalyticsRepositoryImpl) resolveOutletID(query *gorm.DB, outletID *uuid.UUID, column string) *gorm.DB {
|
||||||
|
if outletID != nil {
|
||||||
|
return query.Where(column+" = ?", *outletID)
|
||||||
|
}
|
||||||
|
return query
|
||||||
|
}
|
||||||
|
|
||||||
func (r *AnalyticsRepositoryImpl) GetPaymentMethodAnalytics(ctx context.Context, organizationID uuid.UUID, outletID *uuid.UUID, dateFrom, dateTo time.Time) ([]*entities.PaymentMethodAnalytics, error) {
|
func (r *AnalyticsRepositoryImpl) GetPaymentMethodAnalytics(ctx context.Context, organizationID uuid.UUID, outletID *uuid.UUID, dateFrom, dateTo time.Time) ([]*entities.PaymentMethodAnalytics, error) {
|
||||||
var results []*entities.PaymentMethodAnalytics
|
var results []*entities.PaymentMethodAnalytics
|
||||||
|
|
||||||
@ -50,9 +57,7 @@ func (r *AnalyticsRepositoryImpl) GetPaymentMethodAnalytics(ctx context.Context,
|
|||||||
Where("p.status = ?", entities.PaymentTransactionStatusCompleted).
|
Where("p.status = ?", entities.PaymentTransactionStatusCompleted).
|
||||||
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
||||||
|
|
||||||
if outletID != nil {
|
query = r.resolveOutletID(query, outletID, "o.outlet_id")
|
||||||
query = query.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err := query.
|
err := query.
|
||||||
Group("pm.id, pm.name, pm.type").
|
Group("pm.id, pm.name, pm.type").
|
||||||
@ -180,9 +185,7 @@ func (r *AnalyticsRepositoryImpl) GetProductAnalytics(ctx context.Context, organ
|
|||||||
Where("oi.status != ?", entities.OrderItemStatusCancelled).
|
Where("oi.status != ?", entities.OrderItemStatusCancelled).
|
||||||
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
||||||
|
|
||||||
if outletID != nil {
|
query = r.resolveOutletID(query, outletID, "o.outlet_id")
|
||||||
query = query.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err := query.
|
err := query.
|
||||||
Group("p.id, p.name, p.cost, c.id, c.name, c.order, mahpp.hpp_per_unit").
|
Group("p.id, p.name, p.cost, c.id, c.name, c.order, mahpp.hpp_per_unit").
|
||||||
@ -235,9 +238,7 @@ func (r *AnalyticsRepositoryImpl) GetProductAnalyticsPerCategory(ctx context.Con
|
|||||||
Where("oi.status != ?", entities.OrderItemStatusCancelled).
|
Where("oi.status != ?", entities.OrderItemStatusCancelled).
|
||||||
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
||||||
|
|
||||||
if outletID != nil {
|
query = r.resolveOutletID(query, outletID, "o.outlet_id")
|
||||||
query = query.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err := query.
|
err := query.
|
||||||
Group("c.id, c.name").
|
Group("c.id, c.name").
|
||||||
@ -267,9 +268,7 @@ func (r *AnalyticsRepositoryImpl) GetDashboardOverview(ctx context.Context, orga
|
|||||||
Where("o.organization_id = ?", organizationID).
|
Where("o.organization_id = ?", organizationID).
|
||||||
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
||||||
|
|
||||||
if outletID != nil {
|
query = r.resolveOutletID(query, outletID, "o.outlet_id")
|
||||||
query = query.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err := query.Scan(&result).Error
|
err := query.Scan(&result).Error
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -320,9 +319,7 @@ func (r *AnalyticsRepositoryImpl) GetProfitLossAnalytics(ctx context.Context, or
|
|||||||
Where("o.is_void = false AND o.is_refund = false").
|
Where("o.is_void = false AND o.is_refund = false").
|
||||||
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
Where("o.created_at >= ? AND o.created_at <= ?", dateFrom, dateTo)
|
||||||
|
|
||||||
if outletID != nil {
|
summaryQuery = r.resolveOutletID(summaryQuery, outletID, "o.outlet_id")
|
||||||
summaryQuery = summaryQuery.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err := summaryQuery.Scan(&summary).Error
|
err := summaryQuery.Scan(&summary).Error
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -374,9 +371,7 @@ func (r *AnalyticsRepositoryImpl) GetProfitLossAnalytics(ctx context.Context, or
|
|||||||
Group(timeFormat).
|
Group(timeFormat).
|
||||||
Order(timeFormat)
|
Order(timeFormat)
|
||||||
|
|
||||||
if outletID != nil {
|
dataQuery = r.resolveOutletID(dataQuery, outletID, "o.outlet_id")
|
||||||
dataQuery = dataQuery.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err = dataQuery.Scan(&data).Error
|
err = dataQuery.Scan(&data).Error
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -419,9 +414,7 @@ func (r *AnalyticsRepositoryImpl) GetProfitLossAnalytics(ctx context.Context, or
|
|||||||
Order("p.name ASC").
|
Order("p.name ASC").
|
||||||
Limit(1000)
|
Limit(1000)
|
||||||
|
|
||||||
if outletID != nil {
|
productQuery = r.resolveOutletID(productQuery, outletID, "o.outlet_id")
|
||||||
productQuery = productQuery.Where("o.outlet_id = ?", *outletID)
|
|
||||||
}
|
|
||||||
|
|
||||||
err = productQuery.Scan(&productData).Error
|
err = productQuery.Scan(&productData).Error
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@ -11,6 +11,7 @@ import (
|
|||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
|
"gorm.io/gorm/clause"
|
||||||
)
|
)
|
||||||
|
|
||||||
type InventoryRepository interface {
|
type InventoryRepository interface {
|
||||||
@ -278,7 +279,12 @@ func (r *InventoryRepositoryImpl) UpdateReorderLevel(ctx context.Context, id uui
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (r *InventoryRepositoryImpl) BulkCreate(ctx context.Context, inventoryItems []*entities.Inventory) error {
|
func (r *InventoryRepositoryImpl) BulkCreate(ctx context.Context, inventoryItems []*entities.Inventory) error {
|
||||||
return r.db.WithContext(ctx).CreateInBatches(inventoryItems, 100).Error
|
return r.db.WithContext(ctx).
|
||||||
|
Clauses(clause.OnConflict{
|
||||||
|
Columns: []clause.Column{{Name: "outlet_id"}, {Name: "product_id"}},
|
||||||
|
DoNothing: true,
|
||||||
|
}).
|
||||||
|
CreateInBatches(inventoryItems, 100).Error
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *InventoryRepositoryImpl) BulkUpdate(ctx context.Context, inventoryItems []*entities.Inventory) error {
|
func (r *InventoryRepositoryImpl) BulkUpdate(ctx context.Context, inventoryItems []*entities.Inventory) error {
|
||||||
@ -301,21 +307,25 @@ func (r *InventoryRepositoryImpl) BulkAdjustQuantity(ctx context.Context, adjust
|
|||||||
return r.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
|
return r.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
|
||||||
for productID, delta := range adjustments {
|
for productID, delta := range adjustments {
|
||||||
var inventory entities.Inventory
|
var inventory entities.Inventory
|
||||||
if err := tx.Where("product_id = ? AND outlet_id = ?", productID, outletID).First(&inventory).Error; err != nil {
|
err := tx.Set("gorm:query_option", "FOR UPDATE").
|
||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
Where("product_id = ? AND outlet_id = ?", productID, outletID).
|
||||||
// Inventory doesn't exist, create it with initial quantity
|
First(&inventory).Error
|
||||||
inventory = entities.Inventory{
|
if err != nil {
|
||||||
ProductID: productID,
|
if !errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
OutletID: outletID,
|
|
||||||
Quantity: 0,
|
|
||||||
ReorderLevel: 0,
|
|
||||||
}
|
|
||||||
if err := tx.Create(&inventory).Error; err != nil {
|
|
||||||
return fmt.Errorf("failed to create inventory record for product %s: %w", productID, err)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
// Use FirstOrCreate to handle race conditions — avoids duplicate key
|
||||||
|
// if another transaction already inserted this row concurrently.
|
||||||
|
inventory = entities.Inventory{
|
||||||
|
ProductID: productID,
|
||||||
|
OutletID: outletID,
|
||||||
|
Quantity: 0,
|
||||||
|
ReorderLevel: 0,
|
||||||
|
}
|
||||||
|
if err := tx.Where(entities.Inventory{ProductID: productID, OutletID: outletID}).
|
||||||
|
FirstOrCreate(&inventory).Error; err != nil {
|
||||||
|
return fmt.Errorf("failed to create inventory record for product %s: %w", productID, err)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inventory.UpdateQuantity(delta)
|
inventory.UpdateQuantity(delta)
|
||||||
|
|||||||
59
internal/repository/notification_delivery_repository.go
Normal file
59
internal/repository/notification_delivery_repository.go
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationDeliveryRepository interface {
|
||||||
|
Create(ctx context.Context, delivery *entities.NotificationDelivery) error
|
||||||
|
BulkCreate(ctx context.Context, deliveries []*entities.NotificationDelivery) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.NotificationDelivery, error)
|
||||||
|
Update(ctx context.Context, delivery *entities.NotificationDelivery) error
|
||||||
|
ListByReceiverID(ctx context.Context, receiverID uuid.UUID) ([]*entities.NotificationDelivery, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationDeliveryRepositoryImpl struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNotificationDeliveryRepository(db *gorm.DB) *NotificationDeliveryRepositoryImpl {
|
||||||
|
return &NotificationDeliveryRepositoryImpl{db: db}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationDeliveryRepositoryImpl) Create(ctx context.Context, delivery *entities.NotificationDelivery) error {
|
||||||
|
return r.db.WithContext(ctx).Create(delivery).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationDeliveryRepositoryImpl) BulkCreate(ctx context.Context, deliveries []*entities.NotificationDelivery) error {
|
||||||
|
if len(deliveries) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return r.db.WithContext(ctx).Create(&deliveries).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationDeliveryRepositoryImpl) GetByID(ctx context.Context, id uuid.UUID) (*entities.NotificationDelivery, error) {
|
||||||
|
var delivery entities.NotificationDelivery
|
||||||
|
err := r.db.WithContext(ctx).First(&delivery, "id = ?", id).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &delivery, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationDeliveryRepositoryImpl) Update(ctx context.Context, delivery *entities.NotificationDelivery) error {
|
||||||
|
return r.db.WithContext(ctx).Save(delivery).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationDeliveryRepositoryImpl) ListByReceiverID(ctx context.Context, receiverID uuid.UUID) ([]*entities.NotificationDelivery, error) {
|
||||||
|
var deliveries []*entities.NotificationDelivery
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Where("notification_receiver_id = ?", receiverID).
|
||||||
|
Order("created_at DESC").
|
||||||
|
Find(&deliveries).Error
|
||||||
|
return deliveries, err
|
||||||
|
}
|
||||||
108
internal/repository/notification_receiver_repository.go
Normal file
108
internal/repository/notification_receiver_repository.go
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationReceiverRepository interface {
|
||||||
|
Create(ctx context.Context, receiver *entities.NotificationReceiver) error
|
||||||
|
BulkCreate(ctx context.Context, receivers []*entities.NotificationReceiver) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.NotificationReceiver, error)
|
||||||
|
GetByNotificationAndUser(ctx context.Context, notificationID, userID uuid.UUID) (*entities.NotificationReceiver, error)
|
||||||
|
Update(ctx context.Context, receiver *entities.NotificationReceiver) error
|
||||||
|
ListByUserID(ctx context.Context, userID uuid.UUID, isRead *bool, limit, offset int) ([]*entities.NotificationReceiver, int64, error)
|
||||||
|
CountUnreadByUserID(ctx context.Context, userID uuid.UUID) (int64, error)
|
||||||
|
SoftDeleteByID(ctx context.Context, id uuid.UUID) error
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationReceiverRepositoryImpl struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNotificationReceiverRepository(db *gorm.DB) *NotificationReceiverRepositoryImpl {
|
||||||
|
return &NotificationReceiverRepositoryImpl{db: db}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) Create(ctx context.Context, receiver *entities.NotificationReceiver) error {
|
||||||
|
return r.db.WithContext(ctx).Create(receiver).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) BulkCreate(ctx context.Context, receivers []*entities.NotificationReceiver) error {
|
||||||
|
if len(receivers) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return r.db.WithContext(ctx).Create(&receivers).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) GetByID(ctx context.Context, id uuid.UUID) (*entities.NotificationReceiver, error) {
|
||||||
|
var receiver entities.NotificationReceiver
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Preload("Notification").
|
||||||
|
First(&receiver, "id = ? AND is_deleted = false", id).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &receiver, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) GetByNotificationAndUser(ctx context.Context, notificationID, userID uuid.UUID) (*entities.NotificationReceiver, error) {
|
||||||
|
var receiver entities.NotificationReceiver
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Where("notification_id = ? AND user_id = ? AND is_deleted = false", notificationID, userID).
|
||||||
|
First(&receiver).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &receiver, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) Update(ctx context.Context, receiver *entities.NotificationReceiver) error {
|
||||||
|
return r.db.WithContext(ctx).Save(receiver).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) ListByUserID(ctx context.Context, userID uuid.UUID, isRead *bool, limit, offset int) ([]*entities.NotificationReceiver, int64, error) {
|
||||||
|
var receivers []*entities.NotificationReceiver
|
||||||
|
var total int64
|
||||||
|
|
||||||
|
query := r.db.WithContext(ctx).
|
||||||
|
Model(&entities.NotificationReceiver{}).
|
||||||
|
Where("user_id = ? AND is_deleted = false", userID)
|
||||||
|
|
||||||
|
if isRead != nil {
|
||||||
|
query = query.Where("is_read = ?", *isRead)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := query.Count(&total).Error; err != nil {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err := query.
|
||||||
|
Preload("Notification").
|
||||||
|
Order("created_at DESC").
|
||||||
|
Limit(limit).
|
||||||
|
Offset(offset).
|
||||||
|
Find(&receivers).Error
|
||||||
|
|
||||||
|
return receivers, total, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) CountUnreadByUserID(ctx context.Context, userID uuid.UUID) (int64, error) {
|
||||||
|
var count int64
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Model(&entities.NotificationReceiver{}).
|
||||||
|
Where("user_id = ? AND is_read = false AND is_deleted = false", userID).
|
||||||
|
Count(&count).Error
|
||||||
|
return count, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationReceiverRepositoryImpl) SoftDeleteByID(ctx context.Context, id uuid.UUID) error {
|
||||||
|
return r.db.WithContext(ctx).
|
||||||
|
Model(&entities.NotificationReceiver{}).
|
||||||
|
Where("id = ?", id).
|
||||||
|
Updates(map[string]interface{}{"is_deleted": true}).Error
|
||||||
|
}
|
||||||
64
internal/repository/notification_repository.go
Normal file
64
internal/repository/notification_repository.go
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationRepository interface {
|
||||||
|
Create(ctx context.Context, notification *entities.Notification) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.Notification, error)
|
||||||
|
Update(ctx context.Context, notification *entities.Notification) error
|
||||||
|
Delete(ctx context.Context, id uuid.UUID) error
|
||||||
|
List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.Notification, int64, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationRepositoryImpl struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNotificationRepository(db *gorm.DB) *NotificationRepositoryImpl {
|
||||||
|
return &NotificationRepositoryImpl{db: db}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationRepositoryImpl) Create(ctx context.Context, notification *entities.Notification) error {
|
||||||
|
return r.db.WithContext(ctx).Create(notification).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationRepositoryImpl) GetByID(ctx context.Context, id uuid.UUID) (*entities.Notification, error) {
|
||||||
|
var notification entities.Notification
|
||||||
|
err := r.db.WithContext(ctx).First(¬ification, "id = ?", id).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return ¬ification, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationRepositoryImpl) Update(ctx context.Context, notification *entities.Notification) error {
|
||||||
|
return r.db.WithContext(ctx).Save(notification).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationRepositoryImpl) Delete(ctx context.Context, id uuid.UUID) error {
|
||||||
|
return r.db.WithContext(ctx).Delete(&entities.Notification{}, "id = ?", id).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *NotificationRepositoryImpl) List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.Notification, int64, error) {
|
||||||
|
var notifications []*entities.Notification
|
||||||
|
var total int64
|
||||||
|
|
||||||
|
query := r.db.WithContext(ctx).Model(&entities.Notification{})
|
||||||
|
for key, value := range filters {
|
||||||
|
query = query.Where(key+" = ?", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := query.Count(&total).Error; err != nil {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err := query.Order("created_at DESC").Limit(limit).Offset(offset).Find(¬ifications).Error
|
||||||
|
return notifications, total, err
|
||||||
|
}
|
||||||
@ -99,3 +99,14 @@ func (r *OrganizationRepositoryImpl) GetByEmail(ctx context.Context, email strin
|
|||||||
}
|
}
|
||||||
return &org, nil
|
return &org, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetTotalOmset returns the total revenue from completed orders for an organization.
|
||||||
|
func (r *OrganizationRepositoryImpl) GetTotalOmset(ctx context.Context, organizationID uuid.UUID) (float64, error) {
|
||||||
|
var total float64
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Table("orders").
|
||||||
|
Where("organization_id = ? AND payment_status = ?", organizationID, "completed").
|
||||||
|
Select("COALESCE(SUM(total_amount), 0)").
|
||||||
|
Scan(&total).Error
|
||||||
|
return total, err
|
||||||
|
}
|
||||||
|
|||||||
85
internal/repository/product_outlet_price_repository.go
Normal file
85
internal/repository/product_outlet_price_repository.go
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
"gorm.io/gorm/clause"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPriceRepository interface {
|
||||||
|
GetByProductAndOutlet(ctx context.Context, productID, outletID uuid.UUID) (*entities.ProductOutletPrice, error)
|
||||||
|
GetByProduct(ctx context.Context, productID uuid.UUID) ([]*entities.ProductOutletPrice, error)
|
||||||
|
GetByProductWithOutlet(ctx context.Context, productID uuid.UUID) ([]*entities.ProductOutletPrice, error)
|
||||||
|
GetByOutlet(ctx context.Context, outletID uuid.UUID) ([]*entities.ProductOutletPrice, error)
|
||||||
|
GetByProductsAndOutlet(ctx context.Context, productIDs []uuid.UUID, outletID uuid.UUID) ([]*entities.ProductOutletPrice, error)
|
||||||
|
Upsert(ctx context.Context, price *entities.ProductOutletPrice) error
|
||||||
|
Delete(ctx context.Context, id uuid.UUID) error
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) (*entities.ProductOutletPrice, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProductOutletPriceRepositoryImpl struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewProductOutletPriceRepositoryImpl(db *gorm.DB) *ProductOutletPriceRepositoryImpl {
|
||||||
|
return &ProductOutletPriceRepositoryImpl{
|
||||||
|
db: db,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) GetByProductAndOutlet(ctx context.Context, productID, outletID uuid.UUID) (*entities.ProductOutletPrice, error) {
|
||||||
|
var price entities.ProductOutletPrice
|
||||||
|
err := r.db.WithContext(ctx).Where("product_id = ? AND outlet_id = ?", productID, outletID).First(&price).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &price, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) GetByProduct(ctx context.Context, productID uuid.UUID) ([]*entities.ProductOutletPrice, error) {
|
||||||
|
var prices []*entities.ProductOutletPrice
|
||||||
|
err := r.db.WithContext(ctx).Where("product_id = ?", productID).Find(&prices).Error
|
||||||
|
return prices, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) GetByOutlet(ctx context.Context, outletID uuid.UUID) ([]*entities.ProductOutletPrice, error) {
|
||||||
|
var prices []*entities.ProductOutletPrice
|
||||||
|
err := r.db.WithContext(ctx).Where("outlet_id = ?", outletID).Find(&prices).Error
|
||||||
|
return prices, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) Upsert(ctx context.Context, price *entities.ProductOutletPrice) error {
|
||||||
|
return r.db.WithContext(ctx).Clauses(clause.OnConflict{
|
||||||
|
Columns: []clause.Column{{Name: "product_id"}, {Name: "outlet_id"}},
|
||||||
|
DoUpdates: clause.AssignmentColumns([]string{"price", "updated_at"}),
|
||||||
|
}).Create(price).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) Delete(ctx context.Context, id uuid.UUID) error {
|
||||||
|
return r.db.WithContext(ctx).Delete(&entities.ProductOutletPrice{}, "id = ?", id).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) GetByID(ctx context.Context, id uuid.UUID) (*entities.ProductOutletPrice, error) {
|
||||||
|
var price entities.ProductOutletPrice
|
||||||
|
err := r.db.WithContext(ctx).First(&price, "id = ?", id).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &price, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) GetByProductsAndOutlet(ctx context.Context, productIDs []uuid.UUID, outletID uuid.UUID) ([]*entities.ProductOutletPrice, error) {
|
||||||
|
var prices []*entities.ProductOutletPrice
|
||||||
|
err := r.db.WithContext(ctx).Where("product_id IN ? AND outlet_id = ?", productIDs, outletID).Find(&prices).Error
|
||||||
|
return prices, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ProductOutletPriceRepositoryImpl) GetByProductWithOutlet(ctx context.Context, productID uuid.UUID) ([]*entities.ProductOutletPrice, error) {
|
||||||
|
var prices []*entities.ProductOutletPrice
|
||||||
|
err := r.db.WithContext(ctx).Preload("Outlet").Where("product_id = ?", productID).Find(&prices).Error
|
||||||
|
return prices, err
|
||||||
|
}
|
||||||
@ -189,3 +189,47 @@ func (r *ProductRepositoryImpl) GetLowCostProducts(ctx context.Context, organiza
|
|||||||
err := r.db.WithContext(ctx).Where("organization_id = ? AND cost <= ? AND is_active = ?", organizationID, maxCost, true).Find(&products).Error
|
err := r.db.WithContext(ctx).Where("organization_id = ? AND cost <= ? AND is_active = ?", organizationID, maxCost, true).Find(&products).Error
|
||||||
return products, err
|
return products, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ListWithOutletPrice fetches products with the same filters as List, but overrides
|
||||||
|
// each product's Price with the outlet-specific price from product_outlet_prices when
|
||||||
|
// outletID is provided. A single LEFT JOIN is used so no second round-trip is needed.
|
||||||
|
func (r *ProductRepositoryImpl) ListWithOutletPrice(ctx context.Context, filters map[string]interface{}, outletID uuid.UUID, limit, offset int) ([]*entities.Product, int64, error) {
|
||||||
|
var products []*entities.Product
|
||||||
|
var total int64
|
||||||
|
|
||||||
|
// Base query with category and variant preloads
|
||||||
|
query := r.db.WithContext(ctx).Model(&entities.Product{}).
|
||||||
|
Preload("Category").
|
||||||
|
Preload("ProductVariants")
|
||||||
|
|
||||||
|
// Apply filters
|
||||||
|
for key, value := range filters {
|
||||||
|
switch key {
|
||||||
|
case "search":
|
||||||
|
searchValue := "%" + value.(string) + "%"
|
||||||
|
query = query.Where("products.name ILIKE ? OR products.description ILIKE ? OR products.sku ILIKE ?", searchValue, searchValue, searchValue)
|
||||||
|
case "price_min":
|
||||||
|
query = query.Where("products.price >= ?", value)
|
||||||
|
case "price_max":
|
||||||
|
query = query.Where("products.price <= ?", value)
|
||||||
|
default:
|
||||||
|
query = query.Where("products."+key+" = ?", value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When outletID is provided, INNER JOIN product_outlet_prices so only products
|
||||||
|
// that have been explicitly assigned to this outlet are returned, with their
|
||||||
|
// outlet-specific price.
|
||||||
|
if outletID != uuid.Nil {
|
||||||
|
query = query.
|
||||||
|
Joins("INNER JOIN product_outlet_prices pop ON pop.product_id = products.id AND pop.outlet_id = ?", outletID).
|
||||||
|
Select("products.*, pop.price AS price")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := query.Count(&total).Error; err != nil {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err := query.Limit(limit).Offset(offset).Find(&products).Error
|
||||||
|
return products, total, err
|
||||||
|
}
|
||||||
|
|||||||
@ -171,6 +171,13 @@ func (r *TableRepository) ReleaseTable(ctx context.Context, tableID uuid.UUID, p
|
|||||||
}).Error
|
}).Error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *TableRepository) UpdateToken(ctx context.Context, tableID uuid.UUID, token string) error {
|
||||||
|
return r.db.WithContext(ctx).
|
||||||
|
Model(&entities.Table{}).
|
||||||
|
Where("id = ?", tableID).
|
||||||
|
Update("token", token).Error
|
||||||
|
}
|
||||||
|
|
||||||
func (r *TableRepository) GetByOrderID(ctx context.Context, orderID uuid.UUID) (*entities.Table, error) {
|
func (r *TableRepository) GetByOrderID(ctx context.Context, orderID uuid.UUID) (*entities.Table, error) {
|
||||||
var table entities.Table
|
var table entities.Table
|
||||||
err := r.db.WithContext(ctx).
|
err := r.db.WithContext(ctx).
|
||||||
|
|||||||
@ -24,4 +24,5 @@ type TableRepositoryInterface interface {
|
|||||||
OccupyTable(ctx context.Context, tableID, orderID uuid.UUID, startTime *time.Time) error
|
OccupyTable(ctx context.Context, tableID, orderID uuid.UUID, startTime *time.Time) error
|
||||||
ReleaseTable(ctx context.Context, tableID uuid.UUID, paymentAmount float64) error
|
ReleaseTable(ctx context.Context, tableID uuid.UUID, paymentAmount float64) error
|
||||||
GetByOrderID(ctx context.Context, orderID uuid.UUID) (*entities.Table, error)
|
GetByOrderID(ctx context.Context, orderID uuid.UUID) (*entities.Table, error)
|
||||||
|
UpdateToken(ctx context.Context, tableID uuid.UUID, token string) error
|
||||||
}
|
}
|
||||||
|
|||||||
94
internal/repository/user_device_repository.go
Normal file
94
internal/repository/user_device_repository.go
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDeviceRepositoryImpl struct {
|
||||||
|
db *gorm.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUserDeviceRepositoryImpl(db *gorm.DB) *UserDeviceRepositoryImpl {
|
||||||
|
return &UserDeviceRepositoryImpl{
|
||||||
|
db: db,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) Create(ctx context.Context, device *entities.UserDevice) error {
|
||||||
|
return r.db.WithContext(ctx).Create(device).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) GetByID(ctx context.Context, id uuid.UUID) (*entities.UserDevice, error) {
|
||||||
|
var device entities.UserDevice
|
||||||
|
err := r.db.WithContext(ctx).First(&device, "id = ?", id).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &device, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) GetByDeviceID(ctx context.Context, deviceID string, userID uuid.UUID) (*entities.UserDevice, error) {
|
||||||
|
var device entities.UserDevice
|
||||||
|
err := r.db.WithContext(ctx).Where("device_id = ? AND user_id = ?", deviceID, userID).First(&device).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &device, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) GetByUserID(ctx context.Context, userID uuid.UUID) ([]*entities.UserDevice, error) {
|
||||||
|
var devices []*entities.UserDevice
|
||||||
|
err := r.db.WithContext(ctx).Where("user_id = ?", userID).Order("created_at DESC").Find(&devices).Error
|
||||||
|
return devices, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) Update(ctx context.Context, device *entities.UserDevice) error {
|
||||||
|
return r.db.WithContext(ctx).Save(device).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) Delete(ctx context.Context, id uuid.UUID) error {
|
||||||
|
return r.db.WithContext(ctx).Delete(&entities.UserDevice{}, "id = ?", id).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) DeleteByUserID(ctx context.Context, userID uuid.UUID) error {
|
||||||
|
return r.db.WithContext(ctx).Delete(&entities.UserDevice{}, "user_id = ?", userID).Error
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *UserDeviceRepositoryImpl) List(ctx context.Context, filters map[string]interface{}, limit, offset int) ([]*entities.UserDevice, int64, error) {
|
||||||
|
var devices []*entities.UserDevice
|
||||||
|
var total int64
|
||||||
|
|
||||||
|
query := r.db.WithContext(ctx).Model(&entities.UserDevice{})
|
||||||
|
|
||||||
|
for key, value := range filters {
|
||||||
|
switch key {
|
||||||
|
case "user_id":
|
||||||
|
query = query.Where("user_id = ?", value)
|
||||||
|
case "platform":
|
||||||
|
if platform, ok := value.(string); ok && platform != "" {
|
||||||
|
query = query.Where("platform = ?", platform)
|
||||||
|
}
|
||||||
|
case "search":
|
||||||
|
if searchStr, ok := value.(string); ok && searchStr != "" {
|
||||||
|
searchPattern := "%" + strings.ToLower(searchStr) + "%"
|
||||||
|
query = query.Where("LOWER(device_name) LIKE ? OR LOWER(device_id) LIKE ?",
|
||||||
|
searchPattern, searchPattern)
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
query = query.Where(key+" = ?", value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := query.Count(&total).Error; err != nil {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err := query.Order("created_at DESC").Limit(limit).Offset(offset).Find(&devices).Error
|
||||||
|
return devices, total, err
|
||||||
|
}
|
||||||
@ -61,6 +61,17 @@ func (r *UserRepositoryImpl) GetActiveUsers(ctx context.Context, organizationID
|
|||||||
return users, err
|
return users, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *UserRepositoryImpl) GetActiveByOutletID(ctx context.Context, organizationID, outletID uuid.UUID) ([]*entities.User, error) {
|
||||||
|
var users []*entities.User
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Where(
|
||||||
|
"organization_id = ? AND is_active = ? AND (outlet_id = ? OR role IN ?)",
|
||||||
|
organizationID, true, outletID, []string{"admin", "manager"},
|
||||||
|
).
|
||||||
|
Find(&users).Error
|
||||||
|
return users, err
|
||||||
|
}
|
||||||
|
|
||||||
func (r *UserRepositoryImpl) Update(ctx context.Context, user *entities.User) error {
|
func (r *UserRepositoryImpl) Update(ctx context.Context, user *entities.User) error {
|
||||||
return r.db.WithContext(ctx).Save(user).Error
|
return r.db.WithContext(ctx).Save(user).Error
|
||||||
}
|
}
|
||||||
|
|||||||
@ -46,12 +46,15 @@ type Router struct {
|
|||||||
customerAuthHandler *handler.CustomerAuthHandler
|
customerAuthHandler *handler.CustomerAuthHandler
|
||||||
customerPointsHandler *handler.CustomerPointsHandler
|
customerPointsHandler *handler.CustomerPointsHandler
|
||||||
spinGameHandler *handler.SpinGameHandler
|
spinGameHandler *handler.SpinGameHandler
|
||||||
|
userDeviceHandler *handler.UserDeviceHandler
|
||||||
|
notificationHandler *handler.NotificationHandler
|
||||||
selfOrderHandler *handler.SelfOrderHandler
|
selfOrderHandler *handler.SelfOrderHandler
|
||||||
|
productOutletPriceHandler *handler.ProductOutletPriceHandler
|
||||||
authMiddleware *middleware.AuthMiddleware
|
authMiddleware *middleware.AuthMiddleware
|
||||||
customerAuthMiddleware *middleware.CustomerAuthMiddleware
|
customerAuthMiddleware *middleware.CustomerAuthMiddleware
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewRouter(cfg *config.Config, healthHandler *handler.HealthHandler, authService service.AuthService, authMiddleware *middleware.AuthMiddleware, userService *service.UserServiceImpl, userValidator *validator.UserValidatorImpl, organizationService service.OrganizationService, organizationValidator validator.OrganizationValidator, outletService service.OutletService, outletValidator validator.OutletValidator, outletSettingService service.OutletSettingService, categoryService service.CategoryService, categoryValidator validator.CategoryValidator, productService service.ProductService, productValidator validator.ProductValidator, productVariantService service.ProductVariantService, productVariantValidator validator.ProductVariantValidator, inventoryService service.InventoryService, inventoryValidator validator.InventoryValidator, orderService service.OrderService, orderValidator validator.OrderValidator, fileService service.FileService, fileValidator validator.FileValidator, customerService service.CustomerService, customerValidator validator.CustomerValidator, paymentMethodService service.PaymentMethodService, paymentMethodValidator validator.PaymentMethodValidator, analyticsService *service.AnalyticsServiceImpl, reportService service.ReportService, tableService *service.TableServiceImpl, tableValidator *validator.TableValidator, unitService handler.UnitService, ingredientService handler.IngredientService, productRecipeService service.ProductRecipeService, vendorService service.VendorService, vendorValidator validator.VendorValidator, purchaseOrderService service.PurchaseOrderService, purchaseOrderValidator validator.PurchaseOrderValidator, unitConverterService service.IngredientUnitConverterService, unitConverterValidator validator.IngredientUnitConverterValidator, chartOfAccountTypeService service.ChartOfAccountTypeService, chartOfAccountTypeValidator validator.ChartOfAccountTypeValidator, chartOfAccountService service.ChartOfAccountService, chartOfAccountValidator validator.ChartOfAccountValidator, accountService service.AccountService, accountValidator validator.AccountValidator, orderIngredientTransactionService service.OrderIngredientTransactionService, orderIngredientTransactionValidator validator.OrderIngredientTransactionValidator, gamificationService service.GamificationService, gamificationValidator validator.GamificationValidator, rewardService service.RewardService, rewardValidator validator.RewardValidator, campaignService service.CampaignService, campaignValidator validator.CampaignValidator, customerAuthService service.CustomerAuthService, customerAuthValidator validator.CustomerAuthValidator, customerPointsService service.CustomerPointsService, spinGameService service.SpinGameService, customerAuthMiddleware *middleware.CustomerAuthMiddleware, selfOrderHandler *handler.SelfOrderHandler) *Router {
|
func NewRouter(cfg *config.Config, healthHandler *handler.HealthHandler, authService service.AuthService, authMiddleware *middleware.AuthMiddleware, userService *service.UserServiceImpl, userValidator *validator.UserValidatorImpl, organizationService service.OrganizationService, organizationValidator validator.OrganizationValidator, outletService service.OutletService, outletValidator validator.OutletValidator, outletSettingService service.OutletSettingService, categoryService service.CategoryService, categoryValidator validator.CategoryValidator, productService service.ProductService, productValidator validator.ProductValidator, productVariantService service.ProductVariantService, productVariantValidator validator.ProductVariantValidator, inventoryService service.InventoryService, inventoryValidator validator.InventoryValidator, orderService service.OrderService, orderValidator validator.OrderValidator, fileService service.FileService, fileValidator validator.FileValidator, customerService service.CustomerService, customerValidator validator.CustomerValidator, paymentMethodService service.PaymentMethodService, paymentMethodValidator validator.PaymentMethodValidator, analyticsService *service.AnalyticsServiceImpl, reportService service.ReportService, tableService *service.TableServiceImpl, tableValidator *validator.TableValidator, unitService handler.UnitService, ingredientService handler.IngredientService, productRecipeService service.ProductRecipeService, vendorService service.VendorService, vendorValidator validator.VendorValidator, purchaseOrderService service.PurchaseOrderService, purchaseOrderValidator validator.PurchaseOrderValidator, unitConverterService service.IngredientUnitConverterService, unitConverterValidator validator.IngredientUnitConverterValidator, chartOfAccountTypeService service.ChartOfAccountTypeService, chartOfAccountTypeValidator validator.ChartOfAccountTypeValidator, chartOfAccountService service.ChartOfAccountService, chartOfAccountValidator validator.ChartOfAccountValidator, accountService service.AccountService, accountValidator validator.AccountValidator, orderIngredientTransactionService service.OrderIngredientTransactionService, orderIngredientTransactionValidator validator.OrderIngredientTransactionValidator, gamificationService service.GamificationService, gamificationValidator validator.GamificationValidator, rewardService service.RewardService, rewardValidator validator.RewardValidator, campaignService service.CampaignService, campaignValidator validator.CampaignValidator, customerAuthService service.CustomerAuthService, customerAuthValidator validator.CustomerAuthValidator, customerPointsService service.CustomerPointsService, spinGameService service.SpinGameService, customerAuthMiddleware *middleware.CustomerAuthMiddleware, userDeviceService service.UserDeviceService, userDeviceValidator validator.UserDeviceValidator, notificationService service.NotificationService, notificationValidator validator.NotificationValidator, productOutletPriceService service.ProductOutletPriceService, productOutletPriceValidator validator.ProductOutletPriceValidator, selfOrderHandler *handler.SelfOrderHandler) *Router {
|
||||||
|
|
||||||
return &Router{
|
return &Router{
|
||||||
config: cfg,
|
config: cfg,
|
||||||
@ -70,7 +73,7 @@ func NewRouter(cfg *config.Config, healthHandler *handler.HealthHandler, authSer
|
|||||||
paymentMethodHandler: handler.NewPaymentMethodHandler(paymentMethodService, paymentMethodValidator),
|
paymentMethodHandler: handler.NewPaymentMethodHandler(paymentMethodService, paymentMethodValidator),
|
||||||
analyticsHandler: handler.NewAnalyticsHandler(analyticsService, transformer.NewTransformer()),
|
analyticsHandler: handler.NewAnalyticsHandler(analyticsService, transformer.NewTransformer()),
|
||||||
reportHandler: handler.NewReportHandler(reportService, userService),
|
reportHandler: handler.NewReportHandler(reportService, userService),
|
||||||
tableHandler: handler.NewTableHandler(tableService, tableValidator, cfg.Server.BaseUrl),
|
tableHandler: handler.NewTableHandler(tableService, tableValidator, cfg.Server.SelfOrderUrl),
|
||||||
unitHandler: handler.NewUnitHandler(unitService),
|
unitHandler: handler.NewUnitHandler(unitService),
|
||||||
ingredientHandler: handler.NewIngredientHandler(ingredientService),
|
ingredientHandler: handler.NewIngredientHandler(ingredientService),
|
||||||
productRecipeHandler: handler.NewProductRecipeHandler(productRecipeService),
|
productRecipeHandler: handler.NewProductRecipeHandler(productRecipeService),
|
||||||
@ -90,7 +93,10 @@ func NewRouter(cfg *config.Config, healthHandler *handler.HealthHandler, authSer
|
|||||||
authMiddleware: authMiddleware,
|
authMiddleware: authMiddleware,
|
||||||
customerAuthMiddleware: customerAuthMiddleware,
|
customerAuthMiddleware: customerAuthMiddleware,
|
||||||
productVariantHandler: handler.NewProductVariantHandler(productVariantService, productVariantValidator),
|
productVariantHandler: handler.NewProductVariantHandler(productVariantService, productVariantValidator),
|
||||||
|
userDeviceHandler: handler.NewUserDeviceHandler(userDeviceService, userDeviceValidator),
|
||||||
|
notificationHandler: handler.NewNotificationHandler(notificationService, notificationValidator),
|
||||||
selfOrderHandler: selfOrderHandler,
|
selfOrderHandler: selfOrderHandler,
|
||||||
|
productOutletPriceHandler: handler.NewProductOutletPriceHandler(productOutletPriceService, productOutletPriceValidator),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -153,7 +159,7 @@ func (r *Router) addAppRoutes(rg *gin.Engine) {
|
|||||||
selfOrder.GET("/categories", r.selfOrderHandler.ListCategories)
|
selfOrder.GET("/categories", r.selfOrderHandler.ListCategories)
|
||||||
selfOrder.GET("/menu", r.selfOrderHandler.GetMenu)
|
selfOrder.GET("/menu", r.selfOrderHandler.GetMenu)
|
||||||
selfOrder.POST("/orders", r.selfOrderHandler.CreateOrder)
|
selfOrder.POST("/orders", r.selfOrderHandler.CreateOrder)
|
||||||
selfOrder.GET("/orders/:sessionId", r.selfOrderHandler.GetOrdersBySession)
|
selfOrder.GET("/orders/:session_id", r.selfOrderHandler.GetOrdersBySession)
|
||||||
}
|
}
|
||||||
|
|
||||||
organizations := v1.Group("/organizations")
|
organizations := v1.Group("/organizations")
|
||||||
@ -219,11 +225,23 @@ func (r *Router) addAppRoutes(rg *gin.Engine) {
|
|||||||
{
|
{
|
||||||
products.POST("", r.productHandler.CreateProduct)
|
products.POST("", r.productHandler.CreateProduct)
|
||||||
products.GET("", r.productHandler.ListProducts)
|
products.GET("", r.productHandler.ListProducts)
|
||||||
|
products.GET("/all", r.productHandler.ListProductAll)
|
||||||
products.GET("/:id", r.productHandler.GetProduct)
|
products.GET("/:id", r.productHandler.GetProduct)
|
||||||
products.PUT("/:id", r.productHandler.UpdateProduct)
|
products.PUT("/:id", r.productHandler.UpdateProduct)
|
||||||
products.DELETE("/:id", r.productHandler.DeleteProduct)
|
products.DELETE("/:id", r.productHandler.DeleteProduct)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
productOutletPrices := protected.Group("/product-outlet-prices")
|
||||||
|
productOutletPrices.Use(r.authMiddleware.RequireAdminOrManager())
|
||||||
|
{
|
||||||
|
productOutletPrices.POST("", r.productOutletPriceHandler.Upsert)
|
||||||
|
productOutletPrices.POST("/bulk", r.productOutletPriceHandler.BulkUpsert)
|
||||||
|
productOutletPrices.GET("/product/:product_id", r.productOutletPriceHandler.GetByProduct)
|
||||||
|
productOutletPrices.GET("/outlet/:outlet_id", r.productOutletPriceHandler.GetByOutlet)
|
||||||
|
productOutletPrices.GET("/product/:product_id/outlet/:outlet_id", r.productOutletPriceHandler.GetByProductAndOutlet)
|
||||||
|
productOutletPrices.DELETE("/:id", r.productOutletPriceHandler.Delete)
|
||||||
|
}
|
||||||
|
|
||||||
productVariants := protected.Group("/product-variants")
|
productVariants := protected.Group("/product-variants")
|
||||||
{
|
{
|
||||||
productVariants.POST("", r.productVariantHandler.CreateProductVariant)
|
productVariants.POST("", r.productVariantHandler.CreateProductVariant)
|
||||||
@ -571,6 +589,42 @@ func (r *Router) addAppRoutes(rg *gin.Engine) {
|
|||||||
// Reports
|
// Reports
|
||||||
outlets.GET("/:outlet_id/reports/daily-transaction.pdf", r.reportHandler.GetDailyTransactionReportPDF)
|
outlets.GET("/:outlet_id/reports/daily-transaction.pdf", r.reportHandler.GetDailyTransactionReportPDF)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// User device routes - accessible by authenticated users for their own devices
|
||||||
|
userDevices := protected.Group("/user-devices")
|
||||||
|
{
|
||||||
|
userDevices.POST("/register", r.userDeviceHandler.RegisterDevice)
|
||||||
|
userDevices.GET("/me", r.userDeviceHandler.GetMyDevices)
|
||||||
|
userDevices.GET("/:id", r.userDeviceHandler.GetDevice)
|
||||||
|
userDevices.PUT("/:id", r.userDeviceHandler.UpdateDevice)
|
||||||
|
userDevices.DELETE("/:id", r.userDeviceHandler.DeleteDevice)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Admin-only user device routes
|
||||||
|
adminUserDevices := protected.Group("/user-devices")
|
||||||
|
adminUserDevices.Use(r.authMiddleware.RequireAdminOrManager())
|
||||||
|
{
|
||||||
|
adminUserDevices.GET("", r.userDeviceHandler.ListDevices)
|
||||||
|
adminUserDevices.GET("/user/:user_id", r.userDeviceHandler.GetDevicesByUser)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notification routes - authenticated users manage their own notifications
|
||||||
|
notifications := protected.Group("/notifications")
|
||||||
|
{
|
||||||
|
notifications.GET("", r.notificationHandler.List)
|
||||||
|
notifications.GET("/:id", r.notificationHandler.GetByID)
|
||||||
|
notifications.PUT("/:id/read", r.notificationHandler.MarkAsRead)
|
||||||
|
notifications.PUT("/read-all", r.notificationHandler.MarkAllAsRead)
|
||||||
|
notifications.DELETE("/:id", r.notificationHandler.Delete)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Admin notification routes - send and broadcast
|
||||||
|
adminNotifications := protected.Group("/notifications")
|
||||||
|
adminNotifications.Use(r.authMiddleware.RequireAdminOrManager())
|
||||||
|
{
|
||||||
|
adminNotifications.POST("/send", r.notificationHandler.Send)
|
||||||
|
adminNotifications.POST("/broadcast", r.notificationHandler.Broadcast)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -8,7 +8,9 @@ import (
|
|||||||
|
|
||||||
"apskel-pos-be/config"
|
"apskel-pos-be/config"
|
||||||
"apskel-pos-be/internal/contract"
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
"apskel-pos-be/internal/models"
|
"apskel-pos-be/internal/models"
|
||||||
|
"apskel-pos-be/internal/processor"
|
||||||
"apskel-pos-be/internal/transformer"
|
"apskel-pos-be/internal/transformer"
|
||||||
|
|
||||||
"github.com/golang-jwt/jwt/v5"
|
"github.com/golang-jwt/jwt/v5"
|
||||||
@ -24,11 +26,12 @@ type AuthService interface {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type AuthServiceImpl struct {
|
type AuthServiceImpl struct {
|
||||||
userProcessor UserProcessor
|
userProcessor UserProcessor
|
||||||
jwtSecret string
|
userDeviceProcessor processor.UserDeviceProcessor
|
||||||
refreshSecret string
|
jwtSecret string
|
||||||
tokenTTL time.Duration
|
refreshSecret string
|
||||||
refreshTokenTTL time.Duration
|
tokenTTL time.Duration
|
||||||
|
refreshTokenTTL time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
type Claims struct {
|
type Claims struct {
|
||||||
@ -39,13 +42,14 @@ type Claims struct {
|
|||||||
jwt.RegisteredClaims
|
jwt.RegisteredClaims
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewAuthService(userProcessor UserProcessor, authConfig *config.AuthConfig) AuthService {
|
func NewAuthService(userProcessor UserProcessor, userDeviceProcessor processor.UserDeviceProcessor, authConfig *config.AuthConfig) AuthService {
|
||||||
return &AuthServiceImpl{
|
return &AuthServiceImpl{
|
||||||
userProcessor: userProcessor,
|
userProcessor: userProcessor,
|
||||||
jwtSecret: authConfig.AccessTokenSecret(),
|
userDeviceProcessor: userDeviceProcessor,
|
||||||
refreshSecret: authConfig.RefreshTokenSecret(),
|
jwtSecret: authConfig.AccessTokenSecret(),
|
||||||
tokenTTL: authConfig.AccessTokenTTL(),
|
refreshSecret: authConfig.RefreshTokenSecret(),
|
||||||
refreshTokenTTL: authConfig.RefreshTokenTTL(),
|
tokenTTL: authConfig.AccessTokenTTL(),
|
||||||
|
refreshTokenTTL: authConfig.RefreshTokenTTL(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -81,6 +85,25 @@ func (s *AuthServiceImpl) Login(ctx context.Context, req *contract.LoginRequest)
|
|||||||
return nil, fmt.Errorf("failed to generate refresh token: %w", err)
|
return nil, fmt.Errorf("failed to generate refresh token: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Register or update device info if provided
|
||||||
|
if req.DeviceID != "" && s.userDeviceProcessor != nil {
|
||||||
|
deviceReq := &models.RegisterUserDeviceRequest{
|
||||||
|
UserID: userResponse.ID,
|
||||||
|
DeviceID: req.DeviceID,
|
||||||
|
DeviceName: req.DeviceName,
|
||||||
|
DeviceType: entities.DeviceType(req.DeviceType),
|
||||||
|
Platform: entities.DevicePlatform(req.Platform),
|
||||||
|
FCMToken: req.FCMToken,
|
||||||
|
AppVersion: req.AppVersion,
|
||||||
|
OsVersion: req.OsVersion,
|
||||||
|
}
|
||||||
|
// Non-blocking: log error but don't fail login
|
||||||
|
if _, err := s.userDeviceProcessor.RegisterDevice(ctx, deviceReq); err != nil {
|
||||||
|
// Log but don't fail the login
|
||||||
|
_ = err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return &contract.LoginResponse{
|
return &contract.LoginResponse{
|
||||||
Token: token,
|
Token: token,
|
||||||
RefreshToken: refreshToken,
|
RefreshToken: refreshToken,
|
||||||
|
|||||||
154
internal/service/notification_service.go
Normal file
154
internal/service/notification_service.go
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"math"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
"apskel-pos-be/internal/processor"
|
||||||
|
"apskel-pos-be/internal/transformer"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationService interface {
|
||||||
|
Send(ctx context.Context, req *contract.SendNotificationRequest, createdBy uuid.UUID) *contract.Response
|
||||||
|
Broadcast(ctx context.Context, req *contract.BroadcastNotificationRequest, organizationID, createdBy uuid.UUID) *contract.Response
|
||||||
|
MarkAsRead(ctx context.Context, receiverID, userID uuid.UUID) *contract.Response
|
||||||
|
MarkAllAsRead(ctx context.Context, userID uuid.UUID) *contract.Response
|
||||||
|
DeleteForUser(ctx context.Context, receiverID, userID uuid.UUID) *contract.Response
|
||||||
|
ListForUser(ctx context.Context, req *contract.ListNotificationsRequest, userID uuid.UUID) *contract.Response
|
||||||
|
GetByID(ctx context.Context, id uuid.UUID) *contract.Response
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationServiceImpl struct {
|
||||||
|
notificationProcessor processor.NotificationProcessor
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewNotificationService(notificationProcessor processor.NotificationProcessor) *NotificationServiceImpl {
|
||||||
|
return &NotificationServiceImpl{
|
||||||
|
notificationProcessor: notificationProcessor,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) Send(ctx context.Context, req *contract.SendNotificationRequest, createdBy uuid.UUID) *contract.Response {
|
||||||
|
modelReq := &models.SendNotificationRequest{
|
||||||
|
Title: req.Title,
|
||||||
|
Body: req.Body,
|
||||||
|
Type: req.Type,
|
||||||
|
Category: req.Category,
|
||||||
|
Priority: req.Priority,
|
||||||
|
ImageURL: req.ImageURL,
|
||||||
|
ActionURL: req.ActionURL,
|
||||||
|
NotifiableType: req.NotifiableType,
|
||||||
|
NotifiableID: req.NotifiableID,
|
||||||
|
Data: req.Data,
|
||||||
|
ReceiverIDs: req.ReceiverIDs,
|
||||||
|
ScheduledAt: req.ScheduledAt,
|
||||||
|
ExpiredAt: req.ExpiredAt,
|
||||||
|
CreatedBy: &createdBy,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := s.notificationProcessor.Send(ctx, modelReq)
|
||||||
|
if err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(transformer.NotificationModelResponseToContract(resp))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) Broadcast(ctx context.Context, req *contract.BroadcastNotificationRequest, organizationID, createdBy uuid.UUID) *contract.Response {
|
||||||
|
modelReq := &models.BroadcastNotificationRequest{
|
||||||
|
Title: req.Title,
|
||||||
|
Body: req.Body,
|
||||||
|
Type: req.Type,
|
||||||
|
Category: req.Category,
|
||||||
|
Priority: req.Priority,
|
||||||
|
ImageURL: req.ImageURL,
|
||||||
|
ActionURL: req.ActionURL,
|
||||||
|
NotifiableType: req.NotifiableType,
|
||||||
|
NotifiableID: req.NotifiableID,
|
||||||
|
Data: req.Data,
|
||||||
|
OrganizationID: organizationID,
|
||||||
|
ScheduledAt: req.ScheduledAt,
|
||||||
|
ExpiredAt: req.ExpiredAt,
|
||||||
|
CreatedBy: &createdBy,
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := s.notificationProcessor.Broadcast(ctx, modelReq)
|
||||||
|
if err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(transformer.NotificationModelResponseToContract(resp))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) MarkAsRead(ctx context.Context, receiverID, userID uuid.UUID) *contract.Response {
|
||||||
|
resp, err := s.notificationProcessor.MarkAsRead(ctx, receiverID, userID)
|
||||||
|
if err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(transformer.NotificationReceiverModelResponseToContract(resp))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) MarkAllAsRead(ctx context.Context, userID uuid.UUID) *contract.Response {
|
||||||
|
if err := s.notificationProcessor.MarkAllAsRead(ctx, userID); err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(map[string]interface{}{"message": "All notifications marked as read"})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) DeleteForUser(ctx context.Context, receiverID, userID uuid.UUID) *contract.Response {
|
||||||
|
if err := s.notificationProcessor.DeleteForUser(ctx, receiverID, userID); err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(map[string]interface{}{"message": "Notification deleted"})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) ListForUser(ctx context.Context, req *contract.ListNotificationsRequest, userID uuid.UUID) *contract.Response {
|
||||||
|
modelReq := &models.ListNotificationsRequest{
|
||||||
|
Page: req.Page,
|
||||||
|
Limit: req.Limit,
|
||||||
|
UserID: userID,
|
||||||
|
IsRead: req.IsRead,
|
||||||
|
}
|
||||||
|
|
||||||
|
receivers, total, unreadCount, err := s.notificationProcessor.ListForUser(ctx, modelReq)
|
||||||
|
if err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
totalPages := int(math.Ceil(float64(total) / float64(req.Limit)))
|
||||||
|
|
||||||
|
response := contract.ListNotificationsResponse{
|
||||||
|
Notifications: transformer.NotificationReceiverModelResponsesToContracts(receivers),
|
||||||
|
TotalCount: total,
|
||||||
|
UnreadCount: unreadCount,
|
||||||
|
Page: req.Page,
|
||||||
|
Limit: req.Limit,
|
||||||
|
TotalPages: totalPages,
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(response)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *NotificationServiceImpl) GetByID(ctx context.Context, id uuid.UUID) *contract.Response {
|
||||||
|
resp, err := s.notificationProcessor.GetByID(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
errResp := contract.NewResponseError(constants.NotFoundErrorCode, constants.NotificationServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(transformer.NotificationModelResponseToContract(resp))
|
||||||
|
}
|
||||||
171
internal/service/omset_milestone_scheduler.go
Normal file
171
internal/service/omset_milestone_scheduler.go
Normal file
@ -0,0 +1,171 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
"apskel-pos-be/internal/processor"
|
||||||
|
"apskel-pos-be/internal/repository"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
defaultCheckInterval = 1 * time.Hour
|
||||||
|
OmsetMillionRupiah = 1_000_000.0
|
||||||
|
)
|
||||||
|
|
||||||
|
// OmsetMilestoneScheduler periodically checks each organization's total omset
|
||||||
|
// and sends a notification to owner/admin users when a milestone is reached.
|
||||||
|
//
|
||||||
|
// NOTE: Milestone tracking is in-memory; notifications may re-trigger after a restart.
|
||||||
|
// For persistent tracking, persist the notified state in the database.
|
||||||
|
type OmsetMilestoneScheduler struct {
|
||||||
|
orgRepo *repository.OrganizationRepositoryImpl
|
||||||
|
userRepo *repository.UserRepositoryImpl
|
||||||
|
notificationProc processor.NotificationProcessor
|
||||||
|
|
||||||
|
mu sync.Mutex
|
||||||
|
notified map[string]bool // "orgID:milestone" -> already notified
|
||||||
|
stopCh chan struct{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewOmsetMilestoneScheduler(
|
||||||
|
orgRepo *repository.OrganizationRepositoryImpl,
|
||||||
|
userRepo *repository.UserRepositoryImpl,
|
||||||
|
notificationProc processor.NotificationProcessor,
|
||||||
|
) *OmsetMilestoneScheduler {
|
||||||
|
return &OmsetMilestoneScheduler{
|
||||||
|
orgRepo: orgRepo,
|
||||||
|
userRepo: userRepo,
|
||||||
|
notificationProc: notificationProc,
|
||||||
|
notified: make(map[string]bool),
|
||||||
|
stopCh: make(chan struct{}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start begins the periodic milestone check in a background goroutine.
|
||||||
|
func (s *OmsetMilestoneScheduler) Start(interval time.Duration) {
|
||||||
|
if interval <= 0 {
|
||||||
|
interval = defaultCheckInterval
|
||||||
|
}
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
// Perform an initial check immediately.
|
||||||
|
s.checkAllOrganizations()
|
||||||
|
|
||||||
|
ticker := time.NewTicker(interval)
|
||||||
|
defer ticker.Stop()
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-ticker.C:
|
||||||
|
s.checkAllOrganizations()
|
||||||
|
case <-s.stopCh:
|
||||||
|
log.Println("Omset milestone scheduler stopped")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
log.Println("Omset milestone scheduler started")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stop signals the scheduler to stop.
|
||||||
|
func (s *OmsetMilestoneScheduler) Stop() {
|
||||||
|
close(s.stopCh)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *OmsetMilestoneScheduler) checkAllOrganizations() {
|
||||||
|
ctx := context.Background()
|
||||||
|
|
||||||
|
orgs, _, err := s.orgRepo.List(ctx, nil, 1000, 0)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("OmsetMilestoneScheduler: failed to list organizations: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, org := range orgs {
|
||||||
|
s.checkOrganization(ctx, org)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *OmsetMilestoneScheduler) checkOrganization(ctx context.Context, org *entities.Organization) {
|
||||||
|
totalOmset, err := s.orgRepo.GetTotalOmset(ctx, org.ID)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("OmsetMilestoneScheduler: failed to get total omset for org %s: %v", org.ID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
milestones := []float64{OmsetMillionRupiah}
|
||||||
|
|
||||||
|
for _, milestone := range milestones {
|
||||||
|
if totalOmset < milestone {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
key := fmt.Sprintf("%s:%.0f", org.ID.String(), milestone)
|
||||||
|
|
||||||
|
s.mu.Lock()
|
||||||
|
if s.notified[key] {
|
||||||
|
s.mu.Unlock()
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
s.notified[key] = true
|
||||||
|
s.mu.Unlock()
|
||||||
|
|
||||||
|
s.sendMilestoneNotification(ctx, org, totalOmset, milestone)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *OmsetMilestoneScheduler) sendMilestoneNotification(ctx context.Context, org *entities.Organization, totalOmset float64, milestone float64) {
|
||||||
|
users, err := s.userRepo.GetByOrganizationID(ctx, org.ID)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("OmsetMilestoneScheduler: failed to get users for org %s: %v", org.ID, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Notify owner and admin users.
|
||||||
|
var receiverIDs []uuid.UUID
|
||||||
|
for _, user := range users {
|
||||||
|
roleStr := string(user.Role)
|
||||||
|
if roleStr == string(constants.RoleOwner) || roleStr == string(constants.RoleAdmin) {
|
||||||
|
receiverIDs = append(receiverIDs, user.ID)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(receiverIDs) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
orgID := org.ID
|
||||||
|
title := "🎉 Selamat! Omset Telah Mencapai 1 Juta Rupiah"
|
||||||
|
body := fmt.Sprintf("Organisasi %s telah mencapai omset Rp %.0f. Terus tingkatkan prestasinya!", org.Name, totalOmset)
|
||||||
|
|
||||||
|
notifReq := &models.SendNotificationRequest{
|
||||||
|
Title: title,
|
||||||
|
Body: body,
|
||||||
|
Type: "milestone",
|
||||||
|
Category: "omset_milestone",
|
||||||
|
NotifiableType: "organization",
|
||||||
|
NotifiableID: &orgID,
|
||||||
|
ReceiverIDs: receiverIDs,
|
||||||
|
Data: map[string]interface{}{
|
||||||
|
"organization_id": org.ID.String(),
|
||||||
|
"total_omset": totalOmset,
|
||||||
|
"milestone": milestone,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := s.notificationProc.Send(ctx, notifReq); err != nil {
|
||||||
|
log.Printf("OmsetMilestoneScheduler: failed to send notification for org %s: %v", org.ID, err)
|
||||||
|
} else {
|
||||||
|
log.Printf("OmsetMilestoneScheduler: sent milestone notification to org %s (omset: %.0f)", org.ID, totalOmset)
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -16,6 +16,11 @@ import (
|
|||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// orderUserRepository is a minimal interface to fetch users by organization for notification purposes.
|
||||||
|
type orderUserRepository interface {
|
||||||
|
GetActiveByOutletID(ctx context.Context, organizationID, outletID uuid.UUID) ([]*entities.User, error)
|
||||||
|
}
|
||||||
|
|
||||||
type OrderService interface {
|
type OrderService interface {
|
||||||
CreateOrder(ctx context.Context, req *models.CreateOrderRequest, organizationID uuid.UUID) (*models.OrderResponse, error)
|
CreateOrder(ctx context.Context, req *models.CreateOrderRequest, organizationID uuid.UUID) (*models.OrderResponse, error)
|
||||||
AddToOrder(ctx context.Context, orderID uuid.UUID, req *models.AddToOrderRequest) (*models.AddToOrderResponse, error)
|
AddToOrder(ctx context.Context, orderID uuid.UUID, req *models.AddToOrderRequest) (*models.AddToOrderResponse, error)
|
||||||
@ -38,9 +43,11 @@ type OrderServiceImpl struct {
|
|||||||
productRecipeRepo repository.ProductRecipeRepository
|
productRecipeRepo repository.ProductRecipeRepository
|
||||||
txManager *repository.TxManager
|
txManager *repository.TxManager
|
||||||
sessionRepo repository.SessionRepository
|
sessionRepo repository.SessionRepository
|
||||||
|
notificationProcessor processor.NotificationProcessor
|
||||||
|
userRepo orderUserRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewOrderServiceImpl(orderProcessor processor.OrderProcessor, tableRepo repository.TableRepositoryInterface, orderIngredientTransactionService *OrderIngredientTransactionService, orderIngredientTransactionProcessor processor.OrderIngredientTransactionProcessor, productRecipeRepo repository.ProductRecipeRepository, txManager *repository.TxManager, sessionRepo repository.SessionRepository) *OrderServiceImpl {
|
func NewOrderServiceImpl(orderProcessor processor.OrderProcessor, tableRepo repository.TableRepositoryInterface, orderIngredientTransactionService *OrderIngredientTransactionService, orderIngredientTransactionProcessor processor.OrderIngredientTransactionProcessor, productRecipeRepo repository.ProductRecipeRepository, txManager *repository.TxManager, sessionRepo repository.SessionRepository, notificationProcessor processor.NotificationProcessor, userRepo orderUserRepository) *OrderServiceImpl {
|
||||||
return &OrderServiceImpl{
|
return &OrderServiceImpl{
|
||||||
orderProcessor: orderProcessor,
|
orderProcessor: orderProcessor,
|
||||||
tableRepo: tableRepo,
|
tableRepo: tableRepo,
|
||||||
@ -49,6 +56,8 @@ func NewOrderServiceImpl(orderProcessor processor.OrderProcessor, tableRepo repo
|
|||||||
productRecipeRepo: productRecipeRepo,
|
productRecipeRepo: productRecipeRepo,
|
||||||
txManager: txManager,
|
txManager: txManager,
|
||||||
sessionRepo: sessionRepo,
|
sessionRepo: sessionRepo,
|
||||||
|
notificationProcessor: notificationProcessor,
|
||||||
|
userRepo: userRepo,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -104,10 +113,73 @@ func (s *OrderServiceImpl) CreateOrder(ctx context.Context, req *models.CreateOr
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Send notification to all org users if this is a self-order
|
||||||
|
if isSelfOrder(req.Metadata) {
|
||||||
|
go s.sendSelfOrderNotification(context.Background(), response, organizationID)
|
||||||
|
}
|
||||||
|
|
||||||
return response, nil
|
return response, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// createIngredientTransactions creates ingredient transactions for order items efficiently
|
// isSelfOrder checks if the order metadata indicates a self-order.
|
||||||
|
func isSelfOrder(metadata map[string]interface{}) bool {
|
||||||
|
if metadata == nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
v, ok := metadata["self_order"]
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
b, ok := v.(bool)
|
||||||
|
return ok && b
|
||||||
|
}
|
||||||
|
|
||||||
|
// sendSelfOrderNotification sends a new-order notification to all active users
|
||||||
|
// that can access the outlet where the self-order was placed.
|
||||||
|
func (s *OrderServiceImpl) sendSelfOrderNotification(ctx context.Context, order *models.OrderResponse, organizationID uuid.UUID) {
|
||||||
|
if s.notificationProcessor == nil || s.userRepo == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
users, err := s.userRepo.GetActiveByOutletID(ctx, organizationID, order.OutletID)
|
||||||
|
if err != nil || len(users) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
receiverIDs := make([]uuid.UUID, 0, len(users))
|
||||||
|
for _, u := range users {
|
||||||
|
receiverIDs = append(receiverIDs, u.ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
tableName := ""
|
||||||
|
if order.TableNumber != nil {
|
||||||
|
tableName = *order.TableNumber
|
||||||
|
}
|
||||||
|
|
||||||
|
title := "Pesanan Baru Masuk"
|
||||||
|
body := fmt.Sprintf("Ada pesanan baru dari meja %s", tableName)
|
||||||
|
if tableName == "" {
|
||||||
|
body = "Ada pesanan baru masuk"
|
||||||
|
}
|
||||||
|
|
||||||
|
orderID := order.ID
|
||||||
|
notifReq := &models.SendNotificationRequest{
|
||||||
|
Title: title,
|
||||||
|
Body: body,
|
||||||
|
Type: "order",
|
||||||
|
Category: "self_order",
|
||||||
|
NotifiableType: "order",
|
||||||
|
NotifiableID: &orderID,
|
||||||
|
ReceiverIDs: receiverIDs,
|
||||||
|
Data: map[string]interface{}{
|
||||||
|
"order_id": order.ID.String(),
|
||||||
|
"order_number": order.OrderNumber,
|
||||||
|
"table_name": tableName,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, _ = s.notificationProcessor.Send(ctx, notifReq)
|
||||||
|
}
|
||||||
func (s *OrderServiceImpl) createIngredientTransactions(ctx context.Context, orderID uuid.UUID, orderItems []models.OrderItemResponse) ([]*contract.CreateOrderIngredientTransactionRequest, error) {
|
func (s *OrderServiceImpl) createIngredientTransactions(ctx context.Context, orderID uuid.UUID, orderItems []models.OrderItemResponse) ([]*contract.CreateOrderIngredientTransactionRequest, error) {
|
||||||
appCtx := appcontext.FromGinContext(ctx)
|
appCtx := appcontext.FromGinContext(ctx)
|
||||||
organizationID := appCtx.OrganizationID
|
organizationID := appCtx.OrganizationID
|
||||||
|
|||||||
125
internal/service/product_outlet_price_service.go
Normal file
125
internal/service/product_outlet_price_service.go
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
"apskel-pos-be/internal/processor"
|
||||||
|
"apskel-pos-be/internal/transformer"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPriceService interface {
|
||||||
|
Upsert(ctx context.Context, req *contract.CreateProductOutletPriceRequest) *contract.Response
|
||||||
|
GetByProductAndOutlet(ctx context.Context, productID, outletID uuid.UUID) *contract.Response
|
||||||
|
GetByProduct(ctx context.Context, productID uuid.UUID) *contract.Response
|
||||||
|
GetByOutlet(ctx context.Context, outletID uuid.UUID) *contract.Response
|
||||||
|
Delete(ctx context.Context, id uuid.UUID) *contract.Response
|
||||||
|
BulkUpsert(ctx context.Context, req *contract.BulkCreateProductOutletPriceRequest) *contract.Response
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProductOutletPriceServiceImpl struct {
|
||||||
|
processor processor.ProductOutletPriceProcessor
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewProductOutletPriceService(proc processor.ProductOutletPriceProcessor) *ProductOutletPriceServiceImpl {
|
||||||
|
return &ProductOutletPriceServiceImpl{
|
||||||
|
processor: proc,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductOutletPriceServiceImpl) Upsert(ctx context.Context, req *contract.CreateProductOutletPriceRequest) *contract.Response {
|
||||||
|
modelReq := transformer.CreateProductOutletPriceRequestToModel(req)
|
||||||
|
|
||||||
|
result, err := s.processor.Upsert(ctx, modelReq)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductOutletPriceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResp := transformer.ProductOutletPriceModelToResponse(result)
|
||||||
|
return contract.BuildSuccessResponse(contractResp)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductOutletPriceServiceImpl) GetByProductAndOutlet(ctx context.Context, productID, outletID uuid.UUID) *contract.Response {
|
||||||
|
result, err := s.processor.GetByProductAndOutlet(ctx, productID, outletID)
|
||||||
|
if err != nil {
|
||||||
|
code := constants.InternalServerErrorCode
|
||||||
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
|
code = constants.NotFoundErrorCode
|
||||||
|
}
|
||||||
|
errorResp := contract.NewResponseError(code, constants.ProductOutletPriceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResp := transformer.ProductOutletPriceModelToResponse(result)
|
||||||
|
return contract.BuildSuccessResponse(contractResp)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductOutletPriceServiceImpl) GetByProduct(ctx context.Context, productID uuid.UUID) *contract.Response {
|
||||||
|
results, err := s.processor.GetByProduct(ctx, productID)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductOutletPriceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResps := transformer.ProductOutletPriceModelsToResponses(results)
|
||||||
|
return contract.BuildSuccessResponse(&contract.ListProductOutletPricesResponse{
|
||||||
|
Prices: contractResps,
|
||||||
|
TotalCount: len(contractResps),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductOutletPriceServiceImpl) GetByOutlet(ctx context.Context, outletID uuid.UUID) *contract.Response {
|
||||||
|
results, err := s.processor.GetByOutlet(ctx, outletID)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductOutletPriceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResps := transformer.ProductOutletPriceModelsToResponses(results)
|
||||||
|
return contract.BuildSuccessResponse(&contract.ListProductOutletPricesResponse{
|
||||||
|
Prices: contractResps,
|
||||||
|
TotalCount: len(contractResps),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductOutletPriceServiceImpl) Delete(ctx context.Context, id uuid.UUID) *contract.Response {
|
||||||
|
err := s.processor.Delete(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductOutletPriceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(map[string]interface{}{
|
||||||
|
"message": "Product outlet price deleted successfully",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductOutletPriceServiceImpl) BulkUpsert(ctx context.Context, req *contract.BulkCreateProductOutletPriceRequest) *contract.Response {
|
||||||
|
prices := make([]models.CreateProductOutletPriceRequest, len(req.Prices))
|
||||||
|
for i, p := range req.Prices {
|
||||||
|
prices[i] = models.CreateProductOutletPriceRequest{
|
||||||
|
ProductID: req.ProductID,
|
||||||
|
OutletID: p.OutletID,
|
||||||
|
Price: p.Price,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
results, err := s.processor.BulkUpsert(ctx, req.ProductID, prices)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductOutletPriceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResps := transformer.ProductOutletPriceModelsToResponses(results)
|
||||||
|
return contract.BuildSuccessResponse(&contract.ListProductOutletPricesResponse{
|
||||||
|
Prices: contractResps,
|
||||||
|
TotalCount: len(contractResps),
|
||||||
|
})
|
||||||
|
}
|
||||||
@ -16,8 +16,9 @@ type ProductService interface {
|
|||||||
CreateProduct(ctx context.Context, apctx *appcontext.ContextInfo, req *contract.CreateProductRequest) *contract.Response
|
CreateProduct(ctx context.Context, apctx *appcontext.ContextInfo, req *contract.CreateProductRequest) *contract.Response
|
||||||
UpdateProduct(ctx context.Context, id uuid.UUID, req *contract.UpdateProductRequest) *contract.Response
|
UpdateProduct(ctx context.Context, id uuid.UUID, req *contract.UpdateProductRequest) *contract.Response
|
||||||
DeleteProduct(ctx context.Context, id uuid.UUID) *contract.Response
|
DeleteProduct(ctx context.Context, id uuid.UUID) *contract.Response
|
||||||
GetProductByID(ctx context.Context, id uuid.UUID) *contract.Response
|
GetProductByID(ctx context.Context, id uuid.UUID, outletID uuid.UUID) *contract.Response
|
||||||
ListProducts(ctx context.Context, req *contract.ListProductsRequest) *contract.Response
|
ListProducts(ctx context.Context, req *contract.ListProductsRequest) *contract.Response
|
||||||
|
ListProductsAll(ctx context.Context, req *contract.ListProductsRequest) *contract.Response
|
||||||
}
|
}
|
||||||
|
|
||||||
type ProductServiceImpl struct {
|
type ProductServiceImpl struct {
|
||||||
@ -68,8 +69,8 @@ func (s *ProductServiceImpl) DeleteProduct(ctx context.Context, id uuid.UUID) *c
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *ProductServiceImpl) GetProductByID(ctx context.Context, id uuid.UUID) *contract.Response {
|
func (s *ProductServiceImpl) GetProductByID(ctx context.Context, id uuid.UUID, outletID uuid.UUID) *contract.Response {
|
||||||
productResponse, err := s.productProcessor.GetProductByID(ctx, id)
|
productResponse, err := s.productProcessor.GetProductByID(ctx, id, outletID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductServiceEntity, err.Error())
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductServiceEntity, err.Error())
|
||||||
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
@ -85,6 +86,63 @@ func (s *ProductServiceImpl) ListProducts(ctx context.Context, req *contract.Lis
|
|||||||
if req.OrganizationID != nil {
|
if req.OrganizationID != nil {
|
||||||
filters["organization_id"] = *req.OrganizationID
|
filters["organization_id"] = *req.OrganizationID
|
||||||
}
|
}
|
||||||
|
if req.OutletID != nil {
|
||||||
|
filters["outlet_id"] = *req.OutletID
|
||||||
|
}
|
||||||
|
if req.CategoryID != nil {
|
||||||
|
filters["category_id"] = *req.CategoryID
|
||||||
|
}
|
||||||
|
if req.BusinessType != "" {
|
||||||
|
filters["business_type"] = req.BusinessType
|
||||||
|
}
|
||||||
|
if req.IsActive != nil {
|
||||||
|
filters["is_active"] = *req.IsActive
|
||||||
|
}
|
||||||
|
if req.Search != "" {
|
||||||
|
filters["search"] = req.Search
|
||||||
|
}
|
||||||
|
if req.MinPrice != nil {
|
||||||
|
filters["price_min"] = *req.MinPrice
|
||||||
|
}
|
||||||
|
if req.MaxPrice != nil {
|
||||||
|
filters["price_max"] = *req.MaxPrice
|
||||||
|
}
|
||||||
|
|
||||||
|
products, totalCount, err := s.productProcessor.ListProducts(ctx, filters, req.Page, req.Limit)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.ProductServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Convert to contract responses
|
||||||
|
contractResponses := transformer.ProductsToResponses(products)
|
||||||
|
|
||||||
|
// Calculate total pages
|
||||||
|
totalPages := totalCount / req.Limit
|
||||||
|
if totalCount%req.Limit > 0 {
|
||||||
|
totalPages++
|
||||||
|
}
|
||||||
|
|
||||||
|
listResponse := &contract.ListProductsResponse{
|
||||||
|
Products: contractResponses,
|
||||||
|
TotalCount: totalCount,
|
||||||
|
Page: req.Page,
|
||||||
|
Limit: req.Limit,
|
||||||
|
TotalPages: totalPages,
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(listResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *ProductServiceImpl) ListProductsAll(ctx context.Context, req *contract.ListProductsRequest) *contract.Response {
|
||||||
|
// Build filters
|
||||||
|
filters := make(map[string]interface{})
|
||||||
|
if req.OrganizationID != nil {
|
||||||
|
filters["organization_id"] = *req.OrganizationID
|
||||||
|
}
|
||||||
|
if req.OutletID != nil {
|
||||||
|
filters["outlet_id"] = *req.OutletID
|
||||||
|
}
|
||||||
if req.CategoryID != nil {
|
if req.CategoryID != nil {
|
||||||
filters["category_id"] = *req.CategoryID
|
filters["category_id"] = *req.CategoryID
|
||||||
}
|
}
|
||||||
|
|||||||
122
internal/service/user_device_service.go
Normal file
122
internal/service/user_device_service.go
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
package service
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/processor"
|
||||||
|
"apskel-pos-be/internal/transformer"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDeviceService interface {
|
||||||
|
RegisterDevice(ctx context.Context, userID uuid.UUID, req *contract.RegisterUserDeviceRequest) *contract.Response
|
||||||
|
UpdateDevice(ctx context.Context, id uuid.UUID, req *contract.UpdateUserDeviceRequest) *contract.Response
|
||||||
|
DeleteDevice(ctx context.Context, id uuid.UUID) *contract.Response
|
||||||
|
GetDeviceByID(ctx context.Context, id uuid.UUID) *contract.Response
|
||||||
|
GetDevicesByUserID(ctx context.Context, userID uuid.UUID) *contract.Response
|
||||||
|
ListDevices(ctx context.Context, req *contract.ListUserDevicesRequest) *contract.Response
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserDeviceServiceImpl struct {
|
||||||
|
userDeviceProcessor processor.UserDeviceProcessor
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewUserDeviceService(userDeviceProcessor processor.UserDeviceProcessor) *UserDeviceServiceImpl {
|
||||||
|
return &UserDeviceServiceImpl{
|
||||||
|
userDeviceProcessor: userDeviceProcessor,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserDeviceServiceImpl) RegisterDevice(ctx context.Context, userID uuid.UUID, req *contract.RegisterUserDeviceRequest) *contract.Response {
|
||||||
|
modelReq := transformer.RegisterUserDeviceRequestToModel(req)
|
||||||
|
modelReq.UserID = userID
|
||||||
|
|
||||||
|
deviceResponse, err := s.userDeviceProcessor.RegisterDevice(ctx, modelReq)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.UserDeviceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResponse := transformer.UserDeviceModelResponseToResponse(deviceResponse)
|
||||||
|
return contract.BuildSuccessResponse(contractResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserDeviceServiceImpl) UpdateDevice(ctx context.Context, id uuid.UUID, req *contract.UpdateUserDeviceRequest) *contract.Response {
|
||||||
|
modelReq := transformer.UpdateUserDeviceRequestToModel(req)
|
||||||
|
|
||||||
|
deviceResponse, err := s.userDeviceProcessor.UpdateDevice(ctx, id, modelReq)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.UserDeviceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResponse := transformer.UserDeviceModelResponseToResponse(deviceResponse)
|
||||||
|
return contract.BuildSuccessResponse(contractResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserDeviceServiceImpl) DeleteDevice(ctx context.Context, id uuid.UUID) *contract.Response {
|
||||||
|
err := s.userDeviceProcessor.DeleteDevice(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.UserDeviceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(map[string]interface{}{
|
||||||
|
"message": "Device deleted successfully",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserDeviceServiceImpl) GetDeviceByID(ctx context.Context, id uuid.UUID) *contract.Response {
|
||||||
|
deviceResponse, err := s.userDeviceProcessor.GetDeviceByID(ctx, id)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.NotFoundErrorCode, constants.UserDeviceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResponse := transformer.UserDeviceModelResponseToResponse(deviceResponse)
|
||||||
|
return contract.BuildSuccessResponse(contractResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserDeviceServiceImpl) GetDevicesByUserID(ctx context.Context, userID uuid.UUID) *contract.Response {
|
||||||
|
deviceResponses, err := s.userDeviceProcessor.GetDevicesByUserID(ctx, userID)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.UserDeviceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResponses := transformer.UserDeviceModelResponsesToResponses(deviceResponses)
|
||||||
|
return contract.BuildSuccessResponse(contractResponses)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *UserDeviceServiceImpl) ListDevices(ctx context.Context, req *contract.ListUserDevicesRequest) *contract.Response {
|
||||||
|
modelReq := transformer.ListUserDevicesRequestToModel(req)
|
||||||
|
|
||||||
|
filters := make(map[string]interface{})
|
||||||
|
if modelReq.UserID != "" {
|
||||||
|
filters["user_id"] = modelReq.UserID
|
||||||
|
}
|
||||||
|
if modelReq.Platform != "" {
|
||||||
|
filters["platform"] = modelReq.Platform
|
||||||
|
}
|
||||||
|
|
||||||
|
devices, totalPages, err := s.userDeviceProcessor.ListDevices(ctx, filters, modelReq.Page, modelReq.Limit)
|
||||||
|
if err != nil {
|
||||||
|
errorResp := contract.NewResponseError(constants.InternalServerErrorCode, constants.UserDeviceServiceEntity, err.Error())
|
||||||
|
return contract.BuildErrorResponse([]*contract.ResponseError{errorResp})
|
||||||
|
}
|
||||||
|
|
||||||
|
contractResponses := transformer.UserDeviceModelResponsesToResponses(devices)
|
||||||
|
|
||||||
|
response := contract.ListUserDevicesResponse{
|
||||||
|
Devices: contractResponses,
|
||||||
|
TotalCount: len(contractResponses),
|
||||||
|
Page: modelReq.Page,
|
||||||
|
Limit: modelReq.Limit,
|
||||||
|
TotalPages: totalPages,
|
||||||
|
}
|
||||||
|
|
||||||
|
return contract.BuildSuccessResponse(response)
|
||||||
|
}
|
||||||
@ -6,8 +6,22 @@ import (
|
|||||||
"apskel-pos-be/internal/util"
|
"apskel-pos-be/internal/util"
|
||||||
"fmt"
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// parseOutletID converts a *string outlet ID to *uuid.UUID, returning nil for invalid/empty values.
|
||||||
|
func parseOutletID(s *string) *uuid.UUID {
|
||||||
|
if s == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
id, err := uuid.Parse(*s)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &id
|
||||||
|
}
|
||||||
|
|
||||||
// PaymentMethodAnalyticsContractToModel converts contract request to model
|
// PaymentMethodAnalyticsContractToModel converts contract request to model
|
||||||
func PaymentMethodAnalyticsContractToModel(req *contract.PaymentMethodAnalyticsRequest) *models.PaymentMethodAnalyticsRequest {
|
func PaymentMethodAnalyticsContractToModel(req *contract.PaymentMethodAnalyticsRequest) *models.PaymentMethodAnalyticsRequest {
|
||||||
var dateFrom, dateTo time.Time
|
var dateFrom, dateTo time.Time
|
||||||
@ -23,7 +37,7 @@ func PaymentMethodAnalyticsContractToModel(req *contract.PaymentMethodAnalyticsR
|
|||||||
|
|
||||||
return &models.PaymentMethodAnalyticsRequest{
|
return &models.PaymentMethodAnalyticsRequest{
|
||||||
OrganizationID: req.OrganizationID,
|
OrganizationID: req.OrganizationID,
|
||||||
OutletID: req.OutletID,
|
OutletID: parseOutletID(req.OutletID),
|
||||||
DateFrom: dateFrom,
|
DateFrom: dateFrom,
|
||||||
DateTo: dateTo,
|
DateTo: dateTo,
|
||||||
GroupBy: req.GroupBy,
|
GroupBy: req.GroupBy,
|
||||||
@ -79,7 +93,7 @@ func SalesAnalyticsContractToModel(req *contract.SalesAnalyticsRequest) *models.
|
|||||||
|
|
||||||
return &models.SalesAnalyticsRequest{
|
return &models.SalesAnalyticsRequest{
|
||||||
OrganizationID: req.OrganizationID,
|
OrganizationID: req.OrganizationID,
|
||||||
OutletID: req.OutletID,
|
OutletID: parseOutletID(req.OutletID),
|
||||||
DateFrom: dateFrom,
|
DateFrom: dateFrom,
|
||||||
DateTo: dateTo,
|
DateTo: dateTo,
|
||||||
GroupBy: req.GroupBy,
|
GroupBy: req.GroupBy,
|
||||||
@ -139,7 +153,7 @@ func ProductAnalyticsContractToModel(req *contract.ProductAnalyticsRequest) *mod
|
|||||||
|
|
||||||
return &models.ProductAnalyticsRequest{
|
return &models.ProductAnalyticsRequest{
|
||||||
OrganizationID: req.OrganizationID,
|
OrganizationID: req.OrganizationID,
|
||||||
OutletID: req.OutletID,
|
OutletID: parseOutletID(req.OutletID),
|
||||||
DateFrom: dateFrom,
|
DateFrom: dateFrom,
|
||||||
DateTo: dateTo,
|
DateTo: dateTo,
|
||||||
Limit: req.Limit,
|
Limit: req.Limit,
|
||||||
@ -199,7 +213,7 @@ func ProductAnalyticsPerCategoryContractToModel(req *contract.ProductAnalyticsPe
|
|||||||
|
|
||||||
return &models.ProductAnalyticsPerCategoryRequest{
|
return &models.ProductAnalyticsPerCategoryRequest{
|
||||||
OrganizationID: req.OrganizationID,
|
OrganizationID: req.OrganizationID,
|
||||||
OutletID: req.OutletID,
|
OutletID: parseOutletID(req.OutletID),
|
||||||
DateFrom: dateFrom,
|
DateFrom: dateFrom,
|
||||||
DateTo: dateTo,
|
DateTo: dateTo,
|
||||||
}
|
}
|
||||||
@ -251,7 +265,7 @@ func DashboardAnalyticsContractToModel(req *contract.DashboardAnalyticsRequest)
|
|||||||
|
|
||||||
return &models.DashboardAnalyticsRequest{
|
return &models.DashboardAnalyticsRequest{
|
||||||
OrganizationID: req.OrganizationID,
|
OrganizationID: req.OrganizationID,
|
||||||
OutletID: req.OutletID,
|
OutletID: parseOutletID(req.OutletID),
|
||||||
DateFrom: dateFrom,
|
DateFrom: dateFrom,
|
||||||
DateTo: dateTo,
|
DateTo: dateTo,
|
||||||
}
|
}
|
||||||
@ -346,7 +360,7 @@ func ProfitLossAnalyticsContractToModel(req *contract.ProfitLossAnalyticsRequest
|
|||||||
|
|
||||||
return &models.ProfitLossAnalyticsRequest{
|
return &models.ProfitLossAnalyticsRequest{
|
||||||
OrganizationID: req.OrganizationID,
|
OrganizationID: req.OrganizationID,
|
||||||
OutletID: req.OutletID,
|
OutletID: parseOutletID(req.OutletID),
|
||||||
DateFrom: *dateFrom,
|
DateFrom: *dateFrom,
|
||||||
DateTo: *dateTo,
|
DateTo: *dateTo,
|
||||||
GroupBy: req.GroupBy,
|
GroupBy: req.GroupBy,
|
||||||
|
|||||||
63
internal/transformer/notification_transformer.go
Normal file
63
internal/transformer/notification_transformer.go
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
package transformer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func NotificationModelResponseToContract(m *models.NotificationResponse) *contract.NotificationResponse {
|
||||||
|
if m == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &contract.NotificationResponse{
|
||||||
|
ID: m.ID,
|
||||||
|
Title: m.Title,
|
||||||
|
Body: m.Body,
|
||||||
|
Type: m.Type,
|
||||||
|
Category: m.Category,
|
||||||
|
Priority: m.Priority,
|
||||||
|
ImageURL: m.ImageURL,
|
||||||
|
ActionURL: m.ActionURL,
|
||||||
|
NotifiableType: m.NotifiableType,
|
||||||
|
NotifiableID: m.NotifiableID,
|
||||||
|
Data: m.Data,
|
||||||
|
ScheduledAt: m.ScheduledAt,
|
||||||
|
SentAt: m.SentAt,
|
||||||
|
ExpiredAt: m.ExpiredAt,
|
||||||
|
CreatedBy: m.CreatedBy,
|
||||||
|
CreatedAt: m.CreatedAt,
|
||||||
|
UpdatedAt: m.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NotificationReceiverModelResponseToContract(m *models.NotificationReceiverResponse) *contract.NotificationReceiverResponse {
|
||||||
|
if m == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
resp := &contract.NotificationReceiverResponse{
|
||||||
|
ID: m.ID,
|
||||||
|
NotificationID: m.NotificationID,
|
||||||
|
UserID: m.UserID,
|
||||||
|
IsRead: m.IsRead,
|
||||||
|
ReadAt: m.ReadAt,
|
||||||
|
IsDeleted: m.IsDeleted,
|
||||||
|
DeletedAt: m.DeletedAt,
|
||||||
|
CreatedAt: m.CreatedAt,
|
||||||
|
UpdatedAt: m.UpdatedAt,
|
||||||
|
}
|
||||||
|
if m.Notification != nil {
|
||||||
|
resp.Notification = NotificationModelResponseToContract(m.Notification)
|
||||||
|
}
|
||||||
|
return resp
|
||||||
|
}
|
||||||
|
|
||||||
|
func NotificationReceiverModelResponsesToContracts(ms []*models.NotificationReceiverResponse) []*contract.NotificationReceiverResponse {
|
||||||
|
if ms == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
result := make([]*contract.NotificationReceiverResponse, len(ms))
|
||||||
|
for i, m := range ms {
|
||||||
|
result[i] = NotificationReceiverModelResponseToContract(m)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
55
internal/transformer/product_outlet_price_transformer.go
Normal file
55
internal/transformer/product_outlet_price_transformer.go
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
package transformer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func CreateProductOutletPriceRequestToModel(req *contract.CreateProductOutletPriceRequest) *models.CreateProductOutletPriceRequest {
|
||||||
|
if req == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &models.CreateProductOutletPriceRequest{
|
||||||
|
ProductID: req.ProductID,
|
||||||
|
OutletID: req.OutletID,
|
||||||
|
Price: req.Price,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UpdateProductOutletPriceRequestToModel(req *contract.UpdateProductOutletPriceRequest) *models.UpdateProductOutletPriceRequest {
|
||||||
|
if req == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &models.UpdateProductOutletPriceRequest{
|
||||||
|
Price: &req.Price,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ProductOutletPriceModelToResponse(m *models.ProductOutletPrice) *contract.ProductOutletPriceResponse {
|
||||||
|
if m == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return &contract.ProductOutletPriceResponse{
|
||||||
|
ID: m.ID,
|
||||||
|
ProductID: m.ProductID,
|
||||||
|
OutletID: m.OutletID,
|
||||||
|
Price: m.Price,
|
||||||
|
CreatedAt: m.CreatedAt,
|
||||||
|
UpdatedAt: m.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ProductOutletPriceModelsToResponses(ms []*models.ProductOutletPrice) []contract.ProductOutletPriceResponse {
|
||||||
|
if ms == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
responses := make([]contract.ProductOutletPriceResponse, len(ms))
|
||||||
|
for i, m := range ms {
|
||||||
|
responses[i] = *ProductOutletPriceModelToResponse(m)
|
||||||
|
}
|
||||||
|
return responses
|
||||||
|
}
|
||||||
@ -97,6 +97,19 @@ func ProductModelResponseToResponse(prod *models.ProductResponse) *contract.Prod
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Convert outlet prices
|
||||||
|
var outletPriceResponses []contract.ProductOutletPriceResponse
|
||||||
|
if len(prod.OutletPrices) > 0 {
|
||||||
|
outletPriceResponses = make([]contract.ProductOutletPriceResponse, len(prod.OutletPrices))
|
||||||
|
for i, op := range prod.OutletPrices {
|
||||||
|
outletPriceResponses[i] = contract.ProductOutletPriceResponse{
|
||||||
|
OutletID: op.OutletID,
|
||||||
|
OutletName: op.OutletName,
|
||||||
|
Price: op.Price,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return &contract.ProductResponse{
|
return &contract.ProductResponse{
|
||||||
ID: prod.ID,
|
ID: prod.ID,
|
||||||
OrganizationID: prod.OrganizationID,
|
OrganizationID: prod.OrganizationID,
|
||||||
@ -106,6 +119,8 @@ func ProductModelResponseToResponse(prod *models.ProductResponse) *contract.Prod
|
|||||||
Name: prod.Name,
|
Name: prod.Name,
|
||||||
Description: prod.Description,
|
Description: prod.Description,
|
||||||
Price: prod.Price,
|
Price: prod.Price,
|
||||||
|
OutletPrice: prod.OutletPrice,
|
||||||
|
OutletPrices: outletPriceResponses,
|
||||||
Cost: prod.Cost,
|
Cost: prod.Cost,
|
||||||
BusinessType: string(prod.BusinessType),
|
BusinessType: string(prod.BusinessType),
|
||||||
ImageURL: prod.ImageURL,
|
ImageURL: prod.ImageURL,
|
||||||
|
|||||||
74
internal/transformer/user_device_transformer.go
Normal file
74
internal/transformer/user_device_transformer.go
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
package transformer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func RegisterUserDeviceRequestToModel(req *contract.RegisterUserDeviceRequest) *models.RegisterUserDeviceRequest {
|
||||||
|
return &models.RegisterUserDeviceRequest{
|
||||||
|
DeviceID: req.DeviceID,
|
||||||
|
DeviceName: req.DeviceName,
|
||||||
|
DeviceType: req.DeviceType,
|
||||||
|
Platform: req.Platform,
|
||||||
|
FCMToken: req.FCMToken,
|
||||||
|
AppVersion: req.AppVersion,
|
||||||
|
OsVersion: req.OsVersion,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UpdateUserDeviceRequestToModel(req *contract.UpdateUserDeviceRequest) *models.UpdateUserDeviceRequest {
|
||||||
|
return &models.UpdateUserDeviceRequest{
|
||||||
|
DeviceName: req.DeviceName,
|
||||||
|
DeviceType: req.DeviceType,
|
||||||
|
Platform: req.Platform,
|
||||||
|
FCMToken: req.FCMToken,
|
||||||
|
AppVersion: req.AppVersion,
|
||||||
|
OsVersion: req.OsVersion,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ListUserDevicesRequestToModel(req *contract.ListUserDevicesRequest) *models.ListUserDevicesRequest {
|
||||||
|
return &models.ListUserDevicesRequest{
|
||||||
|
Page: req.Page,
|
||||||
|
Limit: req.Limit,
|
||||||
|
UserID: req.UserID,
|
||||||
|
Platform: req.Platform,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UserDeviceModelResponseToResponse(device *models.UserDeviceResponse) *contract.UserDeviceResponse {
|
||||||
|
if device == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return &contract.UserDeviceResponse{
|
||||||
|
ID: device.ID,
|
||||||
|
UserID: device.UserID,
|
||||||
|
DeviceID: device.DeviceID,
|
||||||
|
DeviceName: device.DeviceName,
|
||||||
|
DeviceType: device.DeviceType,
|
||||||
|
Platform: device.Platform,
|
||||||
|
FCMToken: device.FCMToken,
|
||||||
|
AppVersion: device.AppVersion,
|
||||||
|
OsVersion: device.OsVersion,
|
||||||
|
IPAddress: device.IPAddress,
|
||||||
|
LastActiveAt: device.LastActiveAt,
|
||||||
|
CreatedAt: device.CreatedAt,
|
||||||
|
UpdatedAt: device.UpdatedAt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func UserDeviceModelResponsesToResponses(devices []*models.UserDeviceResponse) []contract.UserDeviceResponse {
|
||||||
|
if devices == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
responses := make([]contract.UserDeviceResponse, len(devices))
|
||||||
|
for i, device := range devices {
|
||||||
|
response := UserDeviceModelResponseToResponse(device)
|
||||||
|
if response != nil {
|
||||||
|
responses[i] = *response
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return responses
|
||||||
|
}
|
||||||
53
internal/validator/notification_validator.go
Normal file
53
internal/validator/notification_validator.go
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
package validator
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
)
|
||||||
|
|
||||||
|
type NotificationValidator interface {
|
||||||
|
ValidateSendRequest(req *contract.SendNotificationRequest) (error, string)
|
||||||
|
ValidateBroadcastRequest(req *contract.BroadcastNotificationRequest) (error, string)
|
||||||
|
ValidateListRequest(req *contract.ListNotificationsRequest) (error, string)
|
||||||
|
}
|
||||||
|
|
||||||
|
type NotificationValidatorImpl struct{}
|
||||||
|
|
||||||
|
func NewNotificationValidator() *NotificationValidatorImpl {
|
||||||
|
return &NotificationValidatorImpl{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *NotificationValidatorImpl) ValidateSendRequest(req *contract.SendNotificationRequest) (error, string) {
|
||||||
|
if req.Title == "" {
|
||||||
|
return fmt.Errorf("title is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
if req.Body == "" {
|
||||||
|
return fmt.Errorf("body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
if len(req.ReceiverIDs) == 0 {
|
||||||
|
return fmt.Errorf("at least one receiver_id is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *NotificationValidatorImpl) ValidateBroadcastRequest(req *contract.BroadcastNotificationRequest) (error, string) {
|
||||||
|
if req.Title == "" {
|
||||||
|
return fmt.Errorf("title is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
if req.Body == "" {
|
||||||
|
return fmt.Errorf("body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *NotificationValidatorImpl) ValidateListRequest(req *contract.ListNotificationsRequest) (error, string) {
|
||||||
|
if req.Page < 1 {
|
||||||
|
return fmt.Errorf("page must be greater than 0"), constants.ValidationErrorCode
|
||||||
|
}
|
||||||
|
if req.Limit < 1 || req.Limit > 100 {
|
||||||
|
return fmt.Errorf("limit must be between 1 and 100"), constants.ValidationErrorCode
|
||||||
|
}
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
80
internal/validator/product_outlet_price_validator.go
Normal file
80
internal/validator/product_outlet_price_validator.go
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
package validator
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
|
||||||
|
"github.com/google/uuid"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProductOutletPriceValidator interface {
|
||||||
|
ValidateCreateRequest(req *contract.CreateProductOutletPriceRequest) (error, string)
|
||||||
|
ValidateUpdateRequest(req *contract.UpdateProductOutletPriceRequest) (error, string)
|
||||||
|
ValidateBulkCreateRequest(req *contract.BulkCreateProductOutletPriceRequest) (error, string)
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProductOutletPriceValidatorImpl struct{}
|
||||||
|
|
||||||
|
func NewProductOutletPriceValidator() *ProductOutletPriceValidatorImpl {
|
||||||
|
return &ProductOutletPriceValidatorImpl{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *ProductOutletPriceValidatorImpl) ValidateCreateRequest(req *contract.CreateProductOutletPriceRequest) (error, string) {
|
||||||
|
if req == nil {
|
||||||
|
return errors.New("request body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.ProductID == uuid.Nil {
|
||||||
|
return errors.New("product_id is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.OutletID == uuid.Nil {
|
||||||
|
return errors.New("outlet_id is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Price < 0 {
|
||||||
|
return errors.New("price must be non-negative"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *ProductOutletPriceValidatorImpl) ValidateUpdateRequest(req *contract.UpdateProductOutletPriceRequest) (error, string) {
|
||||||
|
if req == nil {
|
||||||
|
return errors.New("request body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Price < 0 {
|
||||||
|
return errors.New("price must be non-negative"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *ProductOutletPriceValidatorImpl) ValidateBulkCreateRequest(req *contract.BulkCreateProductOutletPriceRequest) (error, string) {
|
||||||
|
if req == nil {
|
||||||
|
return errors.New("request body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.ProductID == uuid.Nil {
|
||||||
|
return errors.New("product_id is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(req.Prices) == 0 {
|
||||||
|
return errors.New("at least one price entry is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, p := range req.Prices {
|
||||||
|
if p.OutletID == uuid.Nil {
|
||||||
|
return errors.New("outlet_id is required for each price entry"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
if p.Price < 0 {
|
||||||
|
return fmt.Errorf("price at index %d must be non-negative", i), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
126
internal/validator/user_device_validator.go
Normal file
126
internal/validator/user_device_validator.go
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
package validator
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"apskel-pos-be/internal/constants"
|
||||||
|
"apskel-pos-be/internal/contract"
|
||||||
|
"apskel-pos-be/internal/entities"
|
||||||
|
)
|
||||||
|
|
||||||
|
type UserDeviceValidator interface {
|
||||||
|
ValidateRegisterDeviceRequest(req *contract.RegisterUserDeviceRequest) (error, string)
|
||||||
|
ValidateUpdateDeviceRequest(req *contract.UpdateUserDeviceRequest) (error, string)
|
||||||
|
ValidateListDevicesRequest(req *contract.ListUserDevicesRequest) (error, string)
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserDeviceValidatorImpl struct{}
|
||||||
|
|
||||||
|
func NewUserDeviceValidator() *UserDeviceValidatorImpl {
|
||||||
|
return &UserDeviceValidatorImpl{}
|
||||||
|
}
|
||||||
|
|
||||||
|
var validDeviceTypes = map[entities.DeviceType]bool{
|
||||||
|
entities.DeviceTypeMobile: true,
|
||||||
|
entities.DeviceTypeTablet: true,
|
||||||
|
entities.DeviceTypeDesktop: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
var validPlatforms = map[entities.DevicePlatform]bool{
|
||||||
|
entities.DevicePlatformAndroid: true,
|
||||||
|
entities.DevicePlatformIOS: true,
|
||||||
|
entities.DevicePlatformWeb: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *UserDeviceValidatorImpl) ValidateRegisterDeviceRequest(req *contract.RegisterUserDeviceRequest) (error, string) {
|
||||||
|
if req == nil {
|
||||||
|
return errors.New("request body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.TrimSpace(req.DeviceID) == "" {
|
||||||
|
return errors.New("device_id is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(req.DeviceID) > 255 {
|
||||||
|
return errors.New("device_id must be at most 255 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.DeviceName != "" && len(req.DeviceName) > 255 {
|
||||||
|
return errors.New("device_name must be at most 255 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.DeviceType != "" && !validDeviceTypes[req.DeviceType] {
|
||||||
|
return errors.New("device_type must be one of: mobile, tablet, desktop"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Platform != "" && !validPlatforms[req.Platform] {
|
||||||
|
return errors.New("platform must be one of: android, ios, web"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.FCMToken != "" && len(req.FCMToken) > 512 {
|
||||||
|
return errors.New("fcm_token must be at most 512 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.AppVersion != "" && len(req.AppVersion) > 50 {
|
||||||
|
return errors.New("app_version must be at most 50 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.OsVersion != "" && len(req.OsVersion) > 50 {
|
||||||
|
return errors.New("os_version must be at most 50 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *UserDeviceValidatorImpl) ValidateUpdateDeviceRequest(req *contract.UpdateUserDeviceRequest) (error, string) {
|
||||||
|
if req == nil {
|
||||||
|
return errors.New("request body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.DeviceName != "" && len(req.DeviceName) > 255 {
|
||||||
|
return errors.New("device_name must be at most 255 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.DeviceType != "" && !validDeviceTypes[req.DeviceType] {
|
||||||
|
return errors.New("device_type must be one of: mobile, tablet, desktop"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Platform != "" && !validPlatforms[req.Platform] {
|
||||||
|
return errors.New("platform must be one of: android, ios, web"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.FCMToken != "" && len(req.FCMToken) > 512 {
|
||||||
|
return errors.New("fcm_token must be at most 512 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.AppVersion != "" && len(req.AppVersion) > 50 {
|
||||||
|
return errors.New("app_version must be at most 50 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.OsVersion != "" && len(req.OsVersion) > 50 {
|
||||||
|
return errors.New("os_version must be at most 50 characters"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v *UserDeviceValidatorImpl) ValidateListDevicesRequest(req *contract.ListUserDevicesRequest) (error, string) {
|
||||||
|
if req == nil {
|
||||||
|
return errors.New("request body is required"), constants.MissingFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Page < 1 {
|
||||||
|
return errors.New("page must be at least 1"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Limit < 1 || req.Limit > 100 {
|
||||||
|
return errors.New("limit must be between 1 and 100"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
if req.Platform != "" && !validPlatforms[entities.DevicePlatform(req.Platform)] {
|
||||||
|
return errors.New("platform must be one of: android, ios, web"), constants.MalformedFieldErrorCode
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, ""
|
||||||
|
}
|
||||||
1
migrations/000064_create_user_devices_table.down.sql
Normal file
1
migrations/000064_create_user_devices_table.down.sql
Normal file
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS user_devices;
|
||||||
22
migrations/000064_create_user_devices_table.up.sql
Normal file
22
migrations/000064_create_user_devices_table.up.sql
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
-- User devices table
|
||||||
|
CREATE TABLE user_devices (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||||
|
device_id VARCHAR(255) NOT NULL,
|
||||||
|
device_name VARCHAR(255),
|
||||||
|
device_type VARCHAR(50) CHECK (device_type IN ('mobile', 'tablet', 'desktop')),
|
||||||
|
platform VARCHAR(50) CHECK (platform IN ('android', 'ios', 'web')),
|
||||||
|
fcm_token VARCHAR(512),
|
||||||
|
app_version VARCHAR(50),
|
||||||
|
os_version VARCHAR(50),
|
||||||
|
ip_address VARCHAR(45),
|
||||||
|
last_active_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Indexes
|
||||||
|
CREATE INDEX idx_user_devices_user_id ON user_devices(user_id);
|
||||||
|
CREATE INDEX idx_user_devices_device_id ON user_devices(device_id);
|
||||||
|
CREATE INDEX idx_user_devices_fcm_token ON user_devices(fcm_token);
|
||||||
|
CREATE UNIQUE INDEX idx_user_devices_user_device ON user_devices(user_id, device_id);
|
||||||
1
migrations/000065_create_notifications_table.down.sql
Normal file
1
migrations/000065_create_notifications_table.down.sql
Normal file
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS notifications;
|
||||||
28
migrations/000065_create_notifications_table.up.sql
Normal file
28
migrations/000065_create_notifications_table.up.sql
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
-- Notifications table (master notification record)
|
||||||
|
CREATE TABLE notifications (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
title VARCHAR(255) NOT NULL,
|
||||||
|
body TEXT,
|
||||||
|
type VARCHAR(100),
|
||||||
|
category VARCHAR(100),
|
||||||
|
priority VARCHAR(50) NOT NULL DEFAULT 'normal' CHECK (priority IN ('low', 'normal', 'high')),
|
||||||
|
image_url VARCHAR(512),
|
||||||
|
action_url VARCHAR(512),
|
||||||
|
notifiable_type VARCHAR(100),
|
||||||
|
notifiable_id UUID,
|
||||||
|
data JSONB,
|
||||||
|
scheduled_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
sent_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
expired_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
created_by UUID REFERENCES users(id) ON DELETE SET NULL,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Indexes
|
||||||
|
CREATE INDEX idx_notifications_created_by ON notifications(created_by);
|
||||||
|
CREATE INDEX idx_notifications_type ON notifications(type);
|
||||||
|
CREATE INDEX idx_notifications_category ON notifications(category);
|
||||||
|
CREATE INDEX idx_notifications_notifiable ON notifications(notifiable_type, notifiable_id);
|
||||||
|
CREATE INDEX idx_notifications_scheduled_at ON notifications(scheduled_at);
|
||||||
|
CREATE INDEX idx_notifications_sent_at ON notifications(sent_at);
|
||||||
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS notification_receivers;
|
||||||
18
migrations/000066_create_notification_receivers_table.up.sql
Normal file
18
migrations/000066_create_notification_receivers_table.up.sql
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
-- Notification receivers table (links a notification to a specific user)
|
||||||
|
CREATE TABLE notification_receivers (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
notification_id UUID NOT NULL REFERENCES notifications(id) ON DELETE CASCADE,
|
||||||
|
user_id UUID NOT NULL REFERENCES users(id) ON DELETE CASCADE,
|
||||||
|
is_read BOOLEAN NOT NULL DEFAULT FALSE,
|
||||||
|
read_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
is_deleted BOOLEAN NOT NULL DEFAULT FALSE,
|
||||||
|
deleted_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Indexes
|
||||||
|
CREATE INDEX idx_notification_receivers_notification_id ON notification_receivers(notification_id);
|
||||||
|
CREATE INDEX idx_notification_receivers_user_id ON notification_receivers(user_id);
|
||||||
|
CREATE INDEX idx_notification_receivers_user_unread ON notification_receivers(user_id, is_read) WHERE is_deleted = FALSE;
|
||||||
|
CREATE UNIQUE INDEX idx_notification_receivers_unique ON notification_receivers(notification_id, user_id);
|
||||||
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS notification_deliveries;
|
||||||
@ -0,0 +1,23 @@
|
|||||||
|
-- Notification deliveries table (tracks per-device delivery attempts)
|
||||||
|
CREATE TABLE notification_deliveries (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
notification_receiver_id UUID NOT NULL REFERENCES notification_receivers(id) ON DELETE CASCADE,
|
||||||
|
user_device_id UUID NOT NULL REFERENCES user_devices(id) ON DELETE CASCADE,
|
||||||
|
channel VARCHAR(50) NOT NULL DEFAULT 'push' CHECK (channel IN ('push', 'websocket', 'email')),
|
||||||
|
delivery_status VARCHAR(50) NOT NULL DEFAULT 'pending' CHECK (delivery_status IN ('pending', 'sent', 'delivered', 'failed')),
|
||||||
|
provider VARCHAR(50) CHECK (provider IN ('firebase', 'onesignal')),
|
||||||
|
provider_message_id VARCHAR(255),
|
||||||
|
sent_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
delivered_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
failed_at TIMESTAMP WITH TIME ZONE,
|
||||||
|
failure_reason TEXT,
|
||||||
|
retry_count INT NOT NULL DEFAULT 0,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||||
|
);
|
||||||
|
|
||||||
|
-- Indexes
|
||||||
|
CREATE INDEX idx_notification_deliveries_receiver_id ON notification_deliveries(notification_receiver_id);
|
||||||
|
CREATE INDEX idx_notification_deliveries_device_id ON notification_deliveries(user_device_id);
|
||||||
|
CREATE INDEX idx_notification_deliveries_status ON notification_deliveries(delivery_status);
|
||||||
|
CREATE INDEX idx_notification_deliveries_provider ON notification_deliveries(provider);
|
||||||
@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS product_outlet_prices;
|
||||||
12
migrations/000068_create_product_outlet_prices_table.up.sql
Normal file
12
migrations/000068_create_product_outlet_prices_table.up.sql
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
CREATE TABLE product_outlet_prices (
|
||||||
|
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
||||||
|
product_id UUID NOT NULL REFERENCES products(id) ON DELETE CASCADE,
|
||||||
|
outlet_id UUID NOT NULL REFERENCES outlets(id) ON DELETE CASCADE,
|
||||||
|
price DECIMAL(10,2) NOT NULL,
|
||||||
|
created_at TIMESTAMP WITH TIME ZONE DEFAULT NOW(),
|
||||||
|
updated_at TIMESTAMP WITH TIME ZONE DEFAULT NOW()
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE UNIQUE INDEX idx_product_outlet_prices_product_outlet ON product_outlet_prices(product_id, outlet_id);
|
||||||
|
CREATE INDEX idx_product_outlet_prices_product_id ON product_outlet_prices(product_id);
|
||||||
|
CREATE INDEX idx_product_outlet_prices_outlet_id ON product_outlet_prices(outlet_id);
|
||||||
Loading…
x
Reference in New Issue
Block a user