From 23b629350257c242aededeb45d2acd99805721a4 Mon Sep 17 00:00:00 2001 From: Efril Date: Sat, 9 May 2026 12:24:44 +0700 Subject: [PATCH 1/3] fcm setup --- .gitignore | 3 + config/configs.go | 5 ++ config/fcm.go | 14 +++ go.mod | 71 ++++++++++++--- go.sum | 164 +++++++++++++++++++++++++++++----- infra/development.yaml | 6 +- internal/client/fcm_client.go | 142 +++++++++++++++++++++++++++++ 7 files changed, 372 insertions(+), 33 deletions(-) create mode 100644 config/fcm.go create mode 100644 internal/client/fcm_client.go diff --git a/.gitignore b/.gitignore index 497deaa..0e866c8 100644 --- a/.gitignore +++ b/.gitignore @@ -6,3 +6,6 @@ config/env/* !.env vendor + +# Firebase service account credentials +infra/firebase-service-account.json diff --git a/config/configs.go b/config/configs.go index 6a8dfe0..193a49a 100644 --- a/config/configs.go +++ b/config/configs.go @@ -30,6 +30,7 @@ type Config struct { Log Log `mapstructure:"log"` S3Config S3Config `mapstructure:"s3"` Fonnte Fonnte `mapstructure:"fonnte"` + FCM FCM `mapstructure:"fcm"` } var ( @@ -94,3 +95,7 @@ func (c *Config) LogFormat() string { func (c *Config) GetFonnte() *Fonnte { return &c.Fonnte } + +func (c *Config) GetFCM() *FCM { + return &c.FCM +} diff --git a/config/fcm.go b/config/fcm.go new file mode 100644 index 0000000..c8fe487 --- /dev/null +++ b/config/fcm.go @@ -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 +} diff --git a/go.mod b/go.mod index 3539fbb..add90a9 100644 --- a/go.mod +++ b/go.mod @@ -1,28 +1,54 @@ module apskel-pos-be -go 1.21 +go 1.23.0 require ( github.com/gin-gonic/gin v1.9.1 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/spf13/viper v1.16.0 gopkg.in/yaml.v3 v3.0.1 ) 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/cespare/xxhash/v2 v2.3.0 // indirect github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d // 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/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/gabriel-vasile/mimetype v1.4.3 // 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/universal-translator v0.18.1 // 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/jackc/pgpassfile v1.0.0 // indirect github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect @@ -39,34 +65,55 @@ require ( github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect github.com/modern-go/reflect2 v1.0.2 // 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/rogpeppe/go-internal v1.11.0 // indirect - github.com/spf13/afero v1.9.5 // indirect + github.com/spf13/afero v1.10.0 // indirect github.com/spf13/cast v1.5.1 // indirect github.com/spf13/jwalterweatherman v1.1.0 // 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/twitchyliquid64/golang-asm v0.15.1 // 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.10.0 // indirect go.uber.org/multierr v1.8.0 // indirect golang.org/x/arch v0.7.0 // indirect - golang.org/x/net v0.30.0 // indirect - golang.org/x/sys v0.26.0 // indirect - golang.org/x/text v0.20.0 // indirect - google.golang.org/protobuf v1.32.0 // indirect + golang.org/x/net v0.42.0 // indirect + golang.org/x/oauth2 v0.30.0 // indirect + golang.org/x/sync v0.16.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/yaml.v2 v2.4.0 // indirect ) require ( + firebase.google.com/go/v4 v4.19.0 github.com/aws/aws-sdk-go v1.55.7 github.com/golang-jwt/jwt/v5 v5.2.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 - 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/gorm v1.30.0 ) diff --git a/go.sum b/go.sum index 6cc295c..3d89507 100644 --- a/go.sum +++ b/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.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= 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.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.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.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.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.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.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.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= 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.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.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= +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/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/go.mod h1:eRwEWoyTWFMVYVQzKMNHWP5/RV4xIUGMQfXQHfHkpNU= github.com/benbjohnson/clock v1.1.0 h1:Q92kusRqC1XV2MjkWETPvjJVqKetz1OzxZB7mHJLju8= @@ -47,6 +83,8 @@ github.com/bytedance/sonic v1.10.0-rc/go.mod h1:ElCzW+ufi8qKqNW0FY314xriJhyJhuoJ github.com/bytedance/sonic v1.10.2 h1:GQebETVBxYB7JGWJtLBi07OVzWwt+8dWA00gEVW2ZFE= github.com/bytedance/sonic v1.10.2/go.mod h1:iZcSUejdk5aukTND/Eu/ivjQuEL0Cu9/rf50Hi0u/g4= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= +github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= github.com/chenzhuoyu/base64x v0.0.0-20230717121745-296ad89f973d h1:77cEq6EriyTZ0g/qfRdp61a3Uu/AWrgIq2s0ClJV1g0= @@ -61,6 +99,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-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/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/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= @@ -70,7 +110,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.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.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 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/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY= @@ -84,6 +134,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/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-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/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/oXslEjJA= @@ -94,6 +151,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/goccy/go-json v0.10.2 h1:CrxCmQqYDkv1z7lO7Wbh2HN93uovUHgrECaO5ZrCXAU= 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/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= @@ -121,6 +181,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.2/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 v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= @@ -132,12 +195,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.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.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= -github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +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/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/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.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-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= @@ -149,10 +216,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-20201218002935-b9804c9f04c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= 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.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.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/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= @@ -215,17 +289,19 @@ 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/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= 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/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= 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.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M= -github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA= +github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR38lUII= +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/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= -github.com/spf13/afero v1.9.5 h1:stMpOSZFs//0Lv29HduCmli3GUfpFoF3Y1Q/aXj/wVM= -github.com/spf13/afero v1.9.5/go.mod h1:UBogFpq8E9Hx+xc5CNTTEpTnuHVmXDwZcZcE1eb/UhQ= +github.com/spf13/afero v1.10.0 h1:EaGW2JJh15aKOejeuJ+wpFSHnbd7GE6Wvp3TsNhb6LY= +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/go.mod h1:b9PdjNptOpzXr7Rq1q9gJML/2cdGQAo69NKzQ10KN48= github.com/spf13/jwalterweatherman v1.1.0 h1:ue6voC5bR5F8YxI5S67j9i582FU4Qvo2bmqnqMYADFk= @@ -234,10 +310,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/viper v1.16.0 h1:rGGH0XDZhdUOryiDWjmIvUSWpbNqisK8Wk0Vyefw8hc= 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.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.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.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= @@ -247,8 +326,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.1/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.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/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0= github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= @@ -261,17 +341,40 @@ 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.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +github.com/zeebo/errs v1.4.0 h1:XNdoD/RRMKP7HD0UhJnIzUy74ISdGGxURlYG8HSWSfM= +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.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= 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.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= 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.10.0 h1:9qC72Qh0+3MqyJbAn8YU5xVq1frD8bn3JtD2oXtafVQ= go.uber.org/atomic v1.10.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= -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.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.8.0 h1:dg6GjLku4EH+249NNmoIciG9N/jURbDG+pFlTkhzIC8= go.uber.org/multierr v1.8.0/go.mod h1:7EAYxJLBy9rStEaz58O2t4Uvip6FSURkq8/ppBp95ak= @@ -289,8 +392,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-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= 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.28.0/go.mod h1:rmgy+3RHxRZMyY0jjAJShp2zgEdOqj2AO7U0pYmeQ7U= +golang.org/x/crypto v0.40.0 h1:r4x+VvoG5Fm+eJcxMaY8CQM7Lb0l1lsmjGBQ6s8BfKM= +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-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= @@ -361,8 +464,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-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.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4= -golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU= +golang.org/x/net v0.42.0 h1:jzkYrhi3YQWD6MLBJcsklgQsoAcw89EcZbJw8Z614hs= +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-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= @@ -372,6 +475,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-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.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-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -385,6 +490,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-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.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-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -428,8 +535,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.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo= -golang.org/x/sys v0.26.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/sys v0.34.0 h1:H5Y5sJ2L2JRdyv7ROF1he/lPdvFsd0mJHFw2ThKHxLA= +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-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= @@ -441,12 +548,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.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.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= 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.20.0/go.mod h1:D4IsuqiFMhST5bX19pQ9ikHC2GsaKyk/oF+pn3ducp4= +golang.org/x/text v0.27.0 h1:4fGWRpyh641NLlecmyl4LOe6yDdfaYNrGb2zdfo4JV4= +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-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.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-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= @@ -519,6 +629,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.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= 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.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= @@ -526,6 +638,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.6/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-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= @@ -562,6 +676,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-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-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.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= @@ -578,6 +698,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.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.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-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= @@ -588,8 +710,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.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= 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.32.0/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= +google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= +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 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= diff --git a/infra/development.yaml b/infra/development.yaml index 6880e35..4abcab3 100644 --- a/infra/development.yaml +++ b/infra/development.yaml @@ -42,4 +42,8 @@ log: fonnte: api_url: "https://api.fonnte.com/send" token: "bADQrf9NTXfLZQCK2wGg" - timeout: 30 \ No newline at end of file + timeout: 30 + +fcm: + credentials_file: "infra/firebase-service-account.json" + project_id: "your-firebase-project-id" \ No newline at end of file diff --git a/internal/client/fcm_client.go b/internal/client/fcm_client.go new file mode 100644 index 0000000..643f8d5 --- /dev/null +++ b/internal/client/fcm_client.go @@ -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 +} From bbd6666299b7dd0551a4135ba04cebf0cf51f458 Mon Sep 17 00:00:00 2001 From: Efril Date: Sun, 10 May 2026 10:42:09 +0700 Subject: [PATCH 2/3] user devices --- internal/app/app.go | 13 +- internal/constants/error.go | 25 +- internal/contract/user_contract.go | 21 +- internal/contract/user_device_contract.go | 59 +++++ internal/entities/entities.go | 1 + internal/entities/user_device.go | 50 ++++ internal/handler/user_device_handler.go | 215 ++++++++++++++++++ internal/mappers/user_device_mapper.go | 62 +++++ internal/models/user_device.go | 77 +++++++ internal/processor/user_device_processor.go | 165 ++++++++++++++ internal/repository/user_device_repository.go | 94 ++++++++ internal/router/router.go | 22 +- internal/service/auth_service.go | 45 +++- internal/service/user_device_service.go | 122 ++++++++++ .../transformer/user_device_transformer.go | 74 ++++++ internal/validator/user_device_validator.go | 126 ++++++++++ .../000063_create_user_devices_table.down.sql | 1 + .../000063_create_user_devices_table.up.sql | 22 ++ 18 files changed, 1162 insertions(+), 32 deletions(-) create mode 100644 internal/contract/user_device_contract.go create mode 100644 internal/entities/user_device.go create mode 100644 internal/handler/user_device_handler.go create mode 100644 internal/mappers/user_device_mapper.go create mode 100644 internal/models/user_device.go create mode 100644 internal/processor/user_device_processor.go create mode 100644 internal/repository/user_device_repository.go create mode 100644 internal/service/user_device_service.go create mode 100644 internal/transformer/user_device_transformer.go create mode 100644 internal/validator/user_device_validator.go create mode 100644 migrations/000063_create_user_devices_table.down.sql create mode 100644 migrations/000063_create_user_devices_table.up.sql diff --git a/internal/app/app.go b/internal/app/app.go index 0be0306..7e27f1e 100644 --- a/internal/app/app.go +++ b/internal/app/app.go @@ -105,6 +105,8 @@ func (a *App) Initialize(cfg *config.Config) error { services.customerPointsService, services.spinGameService, middleware.customerAuthMiddleware, + services.userDeviceService, + validators.userDeviceValidator, ) return nil @@ -192,6 +194,7 @@ type repositories struct { customerPointsRepo repository.CustomerPointsRepository otpRepo repository.OtpRepository txManager *repository.TxManager + userDeviceRepo *repository.UserDeviceRepositoryImpl } func (a *App) initRepositories() *repositories { @@ -238,6 +241,7 @@ func (a *App) initRepositories() *repositories { customerPointsRepo: repository.NewCustomerPointsRepository(a.db), otpRepo: repository.NewOtpRepository(a.db), txManager: repository.NewTxManager(a.db), + userDeviceRepo: repository.NewUserDeviceRepositoryImpl(a.db), } } @@ -280,6 +284,7 @@ type processors struct { otpProcessor processor.OtpProcessor fileClient processor.FileClient inventoryMovementService service.InventoryMovementService + userDeviceProcessor *processor.UserDeviceProcessorImpl } func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processors { @@ -327,6 +332,7 @@ func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processor otpProcessor: otpProcessor, fileClient: fileClient, inventoryMovementService: inventoryMovementService, + userDeviceProcessor: processor.NewUserDeviceProcessorImpl(repos.userDeviceRepo), } } @@ -363,11 +369,12 @@ type services struct { customerAuthService service.CustomerAuthService customerPointsService service.CustomerPointsService spinGameService service.SpinGameService + userDeviceService service.UserDeviceService } func (a *App) initServices(processors *processors, repos *repositories, cfg *config.Config) *services { authConfig := cfg.Auth() - authService := service.NewAuthService(processors.userProcessor, authConfig) + authService := service.NewAuthService(processors.userProcessor, processors.userDeviceProcessor, authConfig) organizationService := service.NewOrganizationService(processors.organizationProcessor) outletService := service.NewOutletService(processors.outletProcessor) outletSettingService := service.NewOutletSettingService(processors.outletSettingProcessor) @@ -398,6 +405,7 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con customerAuthService := service.NewCustomerAuthService(processors.customerAuthProcessor) customerPointsService := service.NewCustomerPointsService(processors.customerPointsProcessor) spinGameService := service.NewSpinGameService(processors.gamePlayProcessor, repos.txManager) + userDeviceService := service.NewUserDeviceService(processors.userDeviceProcessor) // Update order service with order ingredient transaction service orderService = service.NewOrderServiceImpl(processors.orderProcessor, repos.tableRepo, orderIngredientTransactionService, processors.orderIngredientTransactionProcessor, *repos.productRecipeRepo, repos.txManager) @@ -435,6 +443,7 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con customerAuthService: customerAuthService, customerPointsService: customerPointsService, spinGameService: spinGameService, + userDeviceService: userDeviceService, } } @@ -474,6 +483,7 @@ type validators struct { rewardValidator validator.RewardValidator campaignValidator validator.CampaignValidator customerAuthValidator validator.CustomerAuthValidator + userDeviceValidator *validator.UserDeviceValidatorImpl } func (a *App) initValidators() *validators { @@ -501,5 +511,6 @@ func (a *App) initValidators() *validators { rewardValidator: validator.NewRewardValidator(), campaignValidator: validator.NewCampaignValidator(), customerAuthValidator: validator.NewCustomerAuthValidator(), + userDeviceValidator: validator.NewUserDeviceValidator(), } } diff --git a/internal/constants/error.go b/internal/constants/error.go index 002ac43..200b303 100644 --- a/internal/constants/error.go +++ b/internal/constants/error.go @@ -44,18 +44,19 @@ const ( IngredientCompositionServiceEntity = "ingredient_composition_service" TableEntity = "table" // Gamification entities - CustomerPointsEntity = "customer_points" - CustomerTokensEntity = "customer_tokens" - TierEntity = "tier" - GameEntity = "game" - GamePrizeEntity = "game_prize" - GamePlayEntity = "game_play" - OmsetTrackerEntity = "omset_tracker" - RewardEntity = "reward" - CampaignEntity = "campaign" - CampaignRuleEntity = "campaign_rule" - CustomerEntity = "customer" - SpinGameHandlerEntity = "spin_game_handler" + CustomerPointsEntity = "customer_points" + CustomerTokensEntity = "customer_tokens" + TierEntity = "tier" + GameEntity = "game" + GamePrizeEntity = "game_prize" + GamePlayEntity = "game_play" + OmsetTrackerEntity = "omset_tracker" + RewardEntity = "reward" + CampaignEntity = "campaign" + CampaignRuleEntity = "campaign_rule" + CustomerEntity = "customer" + SpinGameHandlerEntity = "spin_game_handler" + UserDeviceServiceEntity = "user_device_service" ) var HttpErrorMap = map[string]int{ diff --git a/internal/contract/user_contract.go b/internal/contract/user_contract.go index 5607005..bbf0bb4 100644 --- a/internal/contract/user_contract.go +++ b/internal/contract/user_contract.go @@ -35,16 +35,23 @@ type UpdateUserOutletRequest struct { } type LoginRequest struct { - Email string `json:"email" validate:"required,email"` - Password string `json:"password" validate:"required"` + Email string `json:"email" validate:"required,email"` + 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 { - Token string `json:"token"` - RefreshToken string `json:"refresh_token"` - ExpiresAt time.Time `json:"expires_at"` - RefreshExpiresAt time.Time `json:"refresh_expires_at"` - User UserResponse `json:"user"` + Token string `json:"token"` + RefreshToken string `json:"refresh_token"` + ExpiresAt time.Time `json:"expires_at"` + RefreshExpiresAt time.Time `json:"refresh_expires_at"` + User UserResponse `json:"user"` } type UserResponse struct { diff --git a/internal/contract/user_device_contract.go b/internal/contract/user_device_contract.go new file mode 100644 index 0000000..1d0ee0b --- /dev/null +++ b/internal/contract/user_device_contract.go @@ -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"` +} diff --git a/internal/entities/entities.go b/internal/entities/entities.go index 1618997..e6eb8ee 100644 --- a/internal/entities/entities.go +++ b/internal/entities/entities.go @@ -36,6 +36,7 @@ func GetAllEntities() []interface{} { &CampaignRule{}, &OtpSession{}, // Analytics entities are not database tables, they are query results + &UserDevice{}, } } diff --git a/internal/entities/user_device.go b/internal/entities/user_device.go new file mode 100644 index 0000000..d07dcca --- /dev/null +++ b/internal/entities/user_device.go @@ -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" +} diff --git a/internal/handler/user_device_handler.go b/internal/handler/user_device_handler.go new file mode 100644 index 0000000..32d3086 --- /dev/null +++ b/internal/handler/user_device_handler.go @@ -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") +} diff --git a/internal/mappers/user_device_mapper.go b/internal/mappers/user_device_mapper.go new file mode 100644 index 0000000..190a944 --- /dev/null +++ b/internal/mappers/user_device_mapper.go @@ -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 +} diff --git a/internal/models/user_device.go b/internal/models/user_device.go new file mode 100644 index 0000000..bc9abde --- /dev/null +++ b/internal/models/user_device.go @@ -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"` +} diff --git a/internal/processor/user_device_processor.go b/internal/processor/user_device_processor.go new file mode 100644 index 0000000..f3bde33 --- /dev/null +++ b/internal/processor/user_device_processor.go @@ -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 +} diff --git a/internal/repository/user_device_repository.go b/internal/repository/user_device_repository.go new file mode 100644 index 0000000..6bab397 --- /dev/null +++ b/internal/repository/user_device_repository.go @@ -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 +} diff --git a/internal/router/router.go b/internal/router/router.go index 287ba3f..9c63318 100644 --- a/internal/router/router.go +++ b/internal/router/router.go @@ -46,11 +46,12 @@ type Router struct { customerAuthHandler *handler.CustomerAuthHandler customerPointsHandler *handler.CustomerPointsHandler spinGameHandler *handler.SpinGameHandler + userDeviceHandler *handler.UserDeviceHandler authMiddleware *middleware.AuthMiddleware 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) *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) *Router { return &Router{ config: cfg, @@ -89,6 +90,7 @@ func NewRouter(cfg *config.Config, healthHandler *handler.HealthHandler, authSer authMiddleware: authMiddleware, customerAuthMiddleware: customerAuthMiddleware, productVariantHandler: handler.NewProductVariantHandler(productVariantService, productVariantValidator), + userDeviceHandler: handler.NewUserDeviceHandler(userDeviceService, userDeviceValidator), } } @@ -559,6 +561,24 @@ func (r *Router) addAppRoutes(rg *gin.Engine) { // Reports 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) + } } } } diff --git a/internal/service/auth_service.go b/internal/service/auth_service.go index 547e072..1f3fc8e 100644 --- a/internal/service/auth_service.go +++ b/internal/service/auth_service.go @@ -8,7 +8,9 @@ import ( "apskel-pos-be/config" "apskel-pos-be/internal/contract" + "apskel-pos-be/internal/entities" "apskel-pos-be/internal/models" + "apskel-pos-be/internal/processor" "apskel-pos-be/internal/transformer" "github.com/golang-jwt/jwt/v5" @@ -24,11 +26,12 @@ type AuthService interface { } type AuthServiceImpl struct { - userProcessor UserProcessor - jwtSecret string - refreshSecret string - tokenTTL time.Duration - refreshTokenTTL time.Duration + userProcessor UserProcessor + userDeviceProcessor processor.UserDeviceProcessor + jwtSecret string + refreshSecret string + tokenTTL time.Duration + refreshTokenTTL time.Duration } type Claims struct { @@ -39,13 +42,14 @@ type Claims struct { jwt.RegisteredClaims } -func NewAuthService(userProcessor UserProcessor, authConfig *config.AuthConfig) AuthService { +func NewAuthService(userProcessor UserProcessor, userDeviceProcessor processor.UserDeviceProcessor, authConfig *config.AuthConfig) AuthService { return &AuthServiceImpl{ - userProcessor: userProcessor, - jwtSecret: authConfig.AccessTokenSecret(), - refreshSecret: authConfig.RefreshTokenSecret(), - tokenTTL: authConfig.AccessTokenTTL(), - refreshTokenTTL: authConfig.RefreshTokenTTL(), + userProcessor: userProcessor, + userDeviceProcessor: userDeviceProcessor, + jwtSecret: authConfig.AccessTokenSecret(), + refreshSecret: authConfig.RefreshTokenSecret(), + 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) } + // 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{ Token: token, RefreshToken: refreshToken, diff --git a/internal/service/user_device_service.go b/internal/service/user_device_service.go new file mode 100644 index 0000000..3f36c0b --- /dev/null +++ b/internal/service/user_device_service.go @@ -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) +} diff --git a/internal/transformer/user_device_transformer.go b/internal/transformer/user_device_transformer.go new file mode 100644 index 0000000..1814fb6 --- /dev/null +++ b/internal/transformer/user_device_transformer.go @@ -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 +} diff --git a/internal/validator/user_device_validator.go b/internal/validator/user_device_validator.go new file mode 100644 index 0000000..24c36b2 --- /dev/null +++ b/internal/validator/user_device_validator.go @@ -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, "" +} diff --git a/migrations/000063_create_user_devices_table.down.sql b/migrations/000063_create_user_devices_table.down.sql new file mode 100644 index 0000000..035f290 --- /dev/null +++ b/migrations/000063_create_user_devices_table.down.sql @@ -0,0 +1 @@ +DROP TABLE IF EXISTS user_devices; diff --git a/migrations/000063_create_user_devices_table.up.sql b/migrations/000063_create_user_devices_table.up.sql new file mode 100644 index 0000000..079c46c --- /dev/null +++ b/migrations/000063_create_user_devices_table.up.sql @@ -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); From 9d71b339b5621a4a41d97de5fa464574a8742f10 Mon Sep 17 00:00:00 2001 From: Efril Date: Sun, 10 May 2026 10:57:38 +0700 Subject: [PATCH 3/3] notification --- internal/app/app.go | 38 ++ internal/constants/error.go | 28 +- internal/contract/notification_contract.go | 92 +++++ internal/entities/notification.go | 150 ++++++++ internal/handler/notification_handler.go | 190 ++++++++++ internal/mappers/notification_mapper.go | 85 +++++ internal/models/notification.go | 118 ++++++ internal/processor/notification_processor.go | 338 ++++++++++++++++++ .../notification_delivery_repository.go | 59 +++ .../notification_receiver_repository.go | 108 ++++++ .../repository/notification_repository.go | 64 ++++ internal/router/router.go | 22 +- internal/service/notification_service.go | 154 ++++++++ .../transformer/notification_transformer.go | 63 ++++ internal/validator/notification_validator.go | 53 +++ 15 files changed, 1548 insertions(+), 14 deletions(-) create mode 100644 internal/contract/notification_contract.go create mode 100644 internal/entities/notification.go create mode 100644 internal/handler/notification_handler.go create mode 100644 internal/mappers/notification_mapper.go create mode 100644 internal/models/notification.go create mode 100644 internal/processor/notification_processor.go create mode 100644 internal/repository/notification_delivery_repository.go create mode 100644 internal/repository/notification_receiver_repository.go create mode 100644 internal/repository/notification_repository.go create mode 100644 internal/service/notification_service.go create mode 100644 internal/transformer/notification_transformer.go create mode 100644 internal/validator/notification_validator.go diff --git a/internal/app/app.go b/internal/app/app.go index 7e27f1e..e448b85 100644 --- a/internal/app/app.go +++ b/internal/app/app.go @@ -107,6 +107,8 @@ func (a *App) Initialize(cfg *config.Config) error { middleware.customerAuthMiddleware, services.userDeviceService, validators.userDeviceValidator, + services.notificationService, + validators.notificationValidator, ) return nil @@ -195,6 +197,9 @@ type repositories struct { otpRepo repository.OtpRepository txManager *repository.TxManager userDeviceRepo *repository.UserDeviceRepositoryImpl + notificationRepo *repository.NotificationRepositoryImpl + notificationReceiverRepo *repository.NotificationReceiverRepositoryImpl + notificationDeliveryRepo *repository.NotificationDeliveryRepositoryImpl } func (a *App) initRepositories() *repositories { @@ -242,6 +247,9 @@ func (a *App) initRepositories() *repositories { otpRepo: repository.NewOtpRepository(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), } } @@ -285,6 +293,7 @@ type processors struct { fileClient processor.FileClient inventoryMovementService service.InventoryMovementService userDeviceProcessor *processor.UserDeviceProcessorImpl + notificationProcessor *processor.NotificationProcessorImpl } func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processors { @@ -333,6 +342,7 @@ func (a *App) initProcessors(cfg *config.Config, repos *repositories) *processor fileClient: fileClient, inventoryMovementService: inventoryMovementService, userDeviceProcessor: processor.NewUserDeviceProcessorImpl(repos.userDeviceRepo), + notificationProcessor: buildNotificationProcessor(cfg, repos), } } @@ -370,6 +380,7 @@ type services struct { customerPointsService service.CustomerPointsService spinGameService service.SpinGameService userDeviceService service.UserDeviceService + notificationService service.NotificationService } func (a *App) initServices(processors *processors, repos *repositories, cfg *config.Config) *services { @@ -406,6 +417,7 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con customerPointsService := service.NewCustomerPointsService(processors.customerPointsProcessor) 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 orderService = service.NewOrderServiceImpl(processors.orderProcessor, repos.tableRepo, orderIngredientTransactionService, processors.orderIngredientTransactionProcessor, *repos.productRecipeRepo, repos.txManager) @@ -444,6 +456,7 @@ func (a *App) initServices(processors *processors, repos *repositories, cfg *con customerPointsService: customerPointsService, spinGameService: spinGameService, userDeviceService: userDeviceService, + notificationService: notificationService, } } @@ -484,6 +497,7 @@ type validators struct { campaignValidator validator.CampaignValidator customerAuthValidator validator.CustomerAuthValidator userDeviceValidator *validator.UserDeviceValidatorImpl + notificationValidator *validator.NotificationValidatorImpl } func (a *App) initValidators() *validators { @@ -512,5 +526,29 @@ func (a *App) initValidators() *validators { campaignValidator: validator.NewCampaignValidator(), customerAuthValidator: validator.NewCustomerAuthValidator(), userDeviceValidator: validator.NewUserDeviceValidator(), + notificationValidator: validator.NewNotificationValidator(), } } + +// 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, + ) +} diff --git a/internal/constants/error.go b/internal/constants/error.go index 200b303..a1e1ade 100644 --- a/internal/constants/error.go +++ b/internal/constants/error.go @@ -44,19 +44,21 @@ const ( IngredientCompositionServiceEntity = "ingredient_composition_service" TableEntity = "table" // Gamification entities - CustomerPointsEntity = "customer_points" - CustomerTokensEntity = "customer_tokens" - TierEntity = "tier" - GameEntity = "game" - GamePrizeEntity = "game_prize" - GamePlayEntity = "game_play" - OmsetTrackerEntity = "omset_tracker" - RewardEntity = "reward" - CampaignEntity = "campaign" - CampaignRuleEntity = "campaign_rule" - CustomerEntity = "customer" - SpinGameHandlerEntity = "spin_game_handler" - UserDeviceServiceEntity = "user_device_service" + CustomerPointsEntity = "customer_points" + CustomerTokensEntity = "customer_tokens" + TierEntity = "tier" + GameEntity = "game" + GamePrizeEntity = "game_prize" + GamePlayEntity = "game_play" + OmsetTrackerEntity = "omset_tracker" + RewardEntity = "reward" + CampaignEntity = "campaign" + CampaignRuleEntity = "campaign_rule" + CustomerEntity = "customer" + SpinGameHandlerEntity = "spin_game_handler" + UserDeviceServiceEntity = "user_device_service" + NotificationServiceEntity = "notification_service" + NotificationHandlerEntity = "notification_handler" ) var HttpErrorMap = map[string]int{ diff --git a/internal/contract/notification_contract.go b/internal/contract/notification_contract.go new file mode 100644 index 0000000..bab215a --- /dev/null +++ b/internal/contract/notification_contract.go @@ -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"` +} diff --git a/internal/entities/notification.go b/internal/entities/notification.go new file mode 100644 index 0000000..923184d --- /dev/null +++ b/internal/entities/notification.go @@ -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" +} diff --git a/internal/handler/notification_handler.go b/internal/handler/notification_handler.go new file mode 100644 index 0000000..abb4f79 --- /dev/null +++ b/internal/handler/notification_handler.go @@ -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") +} diff --git a/internal/mappers/notification_mapper.go b/internal/mappers/notification_mapper.go new file mode 100644 index 0000000..9ab06ff --- /dev/null +++ b/internal/mappers/notification_mapper.go @@ -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, + } +} diff --git a/internal/models/notification.go b/internal/models/notification.go new file mode 100644 index 0000000..d561dc9 --- /dev/null +++ b/internal/models/notification.go @@ -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"` +} diff --git a/internal/processor/notification_processor.go b/internal/processor/notification_processor.go new file mode 100644 index 0000000..0fbacbf --- /dev/null +++ b/internal/processor/notification_processor.go @@ -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) + } +} diff --git a/internal/repository/notification_delivery_repository.go b/internal/repository/notification_delivery_repository.go new file mode 100644 index 0000000..562db2c --- /dev/null +++ b/internal/repository/notification_delivery_repository.go @@ -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 +} diff --git a/internal/repository/notification_receiver_repository.go b/internal/repository/notification_receiver_repository.go new file mode 100644 index 0000000..69829d8 --- /dev/null +++ b/internal/repository/notification_receiver_repository.go @@ -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 +} diff --git a/internal/repository/notification_repository.go b/internal/repository/notification_repository.go new file mode 100644 index 0000000..72ecb7a --- /dev/null +++ b/internal/repository/notification_repository.go @@ -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 +} diff --git a/internal/router/router.go b/internal/router/router.go index 9c63318..13e2a36 100644 --- a/internal/router/router.go +++ b/internal/router/router.go @@ -47,11 +47,12 @@ type Router struct { customerPointsHandler *handler.CustomerPointsHandler spinGameHandler *handler.SpinGameHandler userDeviceHandler *handler.UserDeviceHandler + notificationHandler *handler.NotificationHandler authMiddleware *middleware.AuthMiddleware 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, userDeviceService service.UserDeviceService, userDeviceValidator validator.UserDeviceValidator) *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) *Router { return &Router{ config: cfg, @@ -91,6 +92,7 @@ func NewRouter(cfg *config.Config, healthHandler *handler.HealthHandler, authSer customerAuthMiddleware: customerAuthMiddleware, productVariantHandler: handler.NewProductVariantHandler(productVariantService, productVariantValidator), userDeviceHandler: handler.NewUserDeviceHandler(userDeviceService, userDeviceValidator), + notificationHandler: handler.NewNotificationHandler(notificationService, notificationValidator), } } @@ -579,6 +581,24 @@ func (r *Router) addAppRoutes(rg *gin.Engine) { 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) + } } } } diff --git a/internal/service/notification_service.go b/internal/service/notification_service.go new file mode 100644 index 0000000..140efe2 --- /dev/null +++ b/internal/service/notification_service.go @@ -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)) +} diff --git a/internal/transformer/notification_transformer.go b/internal/transformer/notification_transformer.go new file mode 100644 index 0000000..174885c --- /dev/null +++ b/internal/transformer/notification_transformer.go @@ -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 +} diff --git a/internal/validator/notification_validator.go b/internal/validator/notification_validator.go new file mode 100644 index 0000000..768252d --- /dev/null +++ b/internal/validator/notification_validator.go @@ -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, "" +}