From d5b3a12e66e5dafb2058ca7a86410e36186d4223 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Fri, 16 Feb 2024 19:56:05 +0530 Subject: [PATCH 01/10] feat: use db --- cmd/server/main.go | 78 +++---- go.mod | 16 +- go.sum | 58 +++++- http-nostr_1.db | Bin 0 -> 16384 bytes internal/nostr/models.go | 34 +++- internal/nostr/nostr.go | 191 +++++++++++++----- internal/nostr/nostr_test.go | 6 +- migrations/202402161653_initial_migration.go | 22 ++ migrations/202402161653_initial_migration.sql | 1 + migrations/README.md | 25 +++ migrations/migrate.go | 15 ++ 11 files changed, 332 insertions(+), 114 deletions(-) create mode 100644 http-nostr_1.db create mode 100644 migrations/202402161653_initial_migration.go create mode 100644 migrations/202402161653_initial_migration.sql create mode 100644 migrations/README.md create mode 100644 migrations/migrate.go diff --git a/cmd/server/main.go b/cmd/server/main.go index c87d6f2..766882b 100644 --- a/cmd/server/main.go +++ b/cmd/server/main.go @@ -1,72 +1,54 @@ package main import ( + "context" "fmt" + "net/http" + "time" "http-nostr/internal/nostr" - "time" - "github.com/joho/godotenv" - "github.com/kelseyhightower/envconfig" + echologrus "github.com/davrux/echo-logrus/v4" "github.com/labstack/echo/v4" "github.com/sirupsen/logrus" ddEcho "gopkg.in/DataDog/dd-trace-go.v1/contrib/labstack/echo.v4" "gopkg.in/DataDog/dd-trace-go.v1/ddtrace/tracer" - - "github.com/getsentry/sentry-go" ) + func main() { - logrus.SetFormatter(&logrus.JSONFormatter{}) - // Load env file as env variables - err := godotenv.Load(".env") + ctx := context.Background() + svc, err := nostr.NewService(ctx) if err != nil { - logrus.Errorf("Error loading environment variables: %v", err) - } - //load global config - type config struct { - SentryDSN string `envconfig:"SENTRY_DSN"` - DatadogAgentUrl string `envconfig:"DATADOG_AGENT_URL"` - DefaultRelayURL string `envconfig:"DEFAULT_RELAY_URL"` - Port int `default:"8080"` - } - globalConf := &config{} - err = envconfig.Process("", globalConf) - if err != nil { - logrus.Fatal(err) - } - - // Setup exception tracking with Sentry if configured - if globalConf.SentryDSN != "" { - if err = sentry.Init(sentry.ClientOptions{ - Dsn: globalConf.SentryDSN, - IgnoreErrors: []string{"401"}, - }); err != nil { - logrus.Errorf("sentry init error: %v", err) - } - defer sentry.Flush(2 * time.Second) + logrus.Fatalf("Failed to initialize service: %v", err) } + echologrus.Logger = svc.Logger e := echo.New() - if globalConf.DatadogAgentUrl != "" { - tracer.Start(tracer.WithAgentAddr(globalConf.DatadogAgentUrl)) + if svc.Cfg.DatadogAgentUrl != "" { + tracer.Start(tracer.WithAgentAddr(svc.Cfg.DatadogAgentUrl)) defer tracer.Stop() e.Use(ddEcho.Middleware(ddEcho.WithServiceName("http-nostr"))) } - service, err := nostr.NewNostrService(&nostr.Config{ - DefaultRelayURL: globalConf.DefaultRelayURL, - }) - if err != nil { - logrus.Fatalf("Failed to initialize NostrService: %v", err) - } + e.GET("/info", svc.InfoHandler) + e.POST("/nip47", svc.NIP47Handler) + e.Use(echologrus.Middleware()) - e.GET("/info", service.InfoHandler) - e.POST("/nip47", service.NIP47Handler) - // r.Use(loggingMiddleware) - - logrus.Infof("Server starting on port %d", globalConf.Port) - if err := e.Start(fmt.Sprintf(":%d", globalConf.Port)); err != nil { - logrus.Fatalf("Server failed to start: %v", err) - } + //start Echo server + go func() { + if err := e.Start(fmt.Sprintf(":%v", svc.Cfg.Port)); err != nil && err != http.ErrServerClosed { + svc.Logger.Fatalf("Shutting down the server: %v", err) + } + }() + //handle graceful shutdown + <-svc.Ctx.Done() + svc.Logger.Infof("Shutting down echo server...") + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + e.Shutdown(ctx) + svc.Logger.Info("Echo server exited") + svc.Logger.Info("Waiting for service to exit...") + svc.Wg.Wait() + svc.Logger.Info("Service exited") } diff --git a/go.mod b/go.mod index e13c20f..1532606 100644 --- a/go.mod +++ b/go.mod @@ -3,7 +3,6 @@ module http-nostr go 1.21.3 require ( - github.com/getsentry/sentry-go v0.25.0 github.com/joho/godotenv v1.5.1 github.com/kelseyhightower/envconfig v1.4.0 github.com/labstack/echo/v4 v4.11.1 @@ -12,6 +11,17 @@ require ( gopkg.in/DataDog/dd-trace-go.v1 v1.58.0 ) +require ( + github.com/glebarez/go-sqlite v1.21.2 // indirect + github.com/jinzhu/inflection v1.0.0 // indirect + github.com/jinzhu/now v1.1.5 // indirect + github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect + modernc.org/libc v1.22.5 // indirect + modernc.org/mathutil v1.5.0 // indirect + modernc.org/memory v1.5.0 // indirect + modernc.org/sqlite v1.23.1 // indirect +) + require ( github.com/DataDog/appsec-internal-go v1.0.2 // indirect github.com/DataDog/datadog-agent/pkg/obfuscate v0.48.0 // indirect @@ -24,11 +34,14 @@ require ( github.com/btcsuite/btcd/btcec/v2 v2.3.2 // indirect github.com/btcsuite/btcd/chaincfg/chainhash v1.0.2 // indirect github.com/cespare/xxhash/v2 v2.2.0 // indirect + github.com/davrux/echo-logrus/v4 v4.0.3 github.com/decred/dcrd/crypto/blake256 v1.0.1 // indirect github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0 // indirect github.com/dgraph-io/ristretto v0.1.1 // indirect github.com/dustin/go-humanize v1.0.1 // indirect github.com/ebitengine/purego v0.5.0 // indirect + github.com/glebarez/sqlite v1.10.0 + github.com/go-gormigrate/gormigrate/v2 v2.1.1 github.com/gobwas/httphead v0.1.0 // indirect github.com/gobwas/pool v0.2.1 // indirect github.com/gobwas/ws v1.2.0 // indirect @@ -64,5 +77,6 @@ require ( golang.org/x/tools v0.12.1-0.20230815132531-74c255bcf846 // indirect golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect google.golang.org/protobuf v1.30.0 // indirect + gorm.io/gorm v1.25.7 inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a // indirect ) diff --git a/go.sum b/go.sum index a7f2f49..e7d11a8 100644 --- a/go.sum +++ b/go.sum @@ -28,12 +28,15 @@ github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSs github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davrux/echo-logrus/v4 v4.0.3 h1:V5bM43A+3PNdpiGC2TS8HKAeaUWQph/j8utG7/mwQ5w= +github.com/davrux/echo-logrus/v4 v4.0.3/go.mod h1:+1y03d0joOKfwnPN4GSFhh/ViG3newZtYZfAPB6yf+g= github.com/decred/dcrd/crypto/blake256 v1.0.1 h1:7PltbUIQB7u/FfZ39+DGa/ShuMyJ5ilcvdfma9wOH6Y= github.com/decred/dcrd/crypto/blake256 v1.0.1/go.mod h1:2OfgNZ5wDpcsFmHmCK5gZTPcCXqlm2ArzUIkw9czNJo= github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0 h1:8UrgZ3GkP4i/CLijOJx79Yu+etlyjdBU4sfcs2WYQMs= github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0/go.mod h1:v57UDF4pDQJcEfFUCRop3lJL149eHGSe9Jvczhzjo/0= github.com/dgraph-io/ristretto v0.1.1 h1:6CWw5tJNgpegArSHpNHJKldNeq03FQCwYvfMVWajOK8= github.com/dgraph-io/ristretto v0.1.1/go.mod h1:S1GPSBCYCIhmVNfcth17y2zZtQT6wzkzgwUve0VDWWA= +github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2 h1:tdlZCpZ/P9DhczCTSixgIKmwPv6+wP5DGjqLYw5SUiA= github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= @@ -42,10 +45,12 @@ github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+m github.com/dvyukov/go-fuzz v0.0.0-20210103155950-6a8e9d1f2415/go.mod h1:11Gm+ccJnvAhCNLlf5+cS9KjtbaD5I5zaZpFMsTHWTw= github.com/ebitengine/purego v0.5.0 h1:JrMGKfRIAM4/QVKaesIIT7m/UVjTj5GYhRSQYwfVdpo= github.com/ebitengine/purego v0.5.0/go.mod h1:ah1In8AOtksoNK6yk5z1HTJeUkC1Ez4Wk2idgGslMwQ= -github.com/getsentry/sentry-go v0.25.0 h1:q6Eo+hS+yoJlTO3uu/azhQadsD8V+jQn2D8VvX1eOyI= -github.com/getsentry/sentry-go v0.25.0/go.mod h1:lc76E2QywIyW8WuBnwl8Lc4bkmQH4+w1gwTf25trprY= -github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA= -github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og= +github.com/glebarez/go-sqlite v1.21.2 h1:3a6LFC4sKahUunAmynQKLZceZCOzUthkRkEAl9gAXWo= +github.com/glebarez/go-sqlite v1.21.2/go.mod h1:sfxdZyhQjTM2Wry3gVYWaW072Ri1WMdWJi0k6+3382k= +github.com/glebarez/sqlite v1.10.0 h1:u4gt8y7OND/cCei/NMHmfbLxF6xP2wgKcT/BJf2pYkc= +github.com/glebarez/sqlite v1.10.0/go.mod h1:IJ+lfSOmiekhQsFTJRx/lHtGYmCdtAiTaf5wI9u5uHA= +github.com/go-gormigrate/gormigrate/v2 v2.1.1 h1:eGS0WTFRV30r103lU8JNXY27KbviRnqqIDobW3EV3iY= +github.com/go-gormigrate/gormigrate/v2 v2.1.1/go.mod h1:L7nJ620PFDKei9QOhJzqA8kRCk+E3UbV2f5gv+1ndLc= github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU= github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM= github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og= @@ -69,25 +74,40 @@ github.com/google/pprof v0.0.0-20230817174616-7a8ec2ada47b h1:h9U78+dx9a4BKdQkBB github.com/google/pprof v0.0.0-20230817174616-7a8ec2ada47b/go.mod h1:czg5+yv1E0ZGTi6S6vVK1mke0fV+FaUhNGcd6VRS9Ik= github.com/google/uuid v1.3.1 h1:KjJaJ9iWZ3jOFZIf1Lqf4laDRCasjl0BCmnEGxkdLb4= github.com/google/uuid v1.3.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= +github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= +github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY= github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y= github.com/kelseyhightower/envconfig v1.4.0 h1:Im6hONhd3pLkfDFsbRgu68RDNkGF1r3dvMUtDTo2cv8= github.com/kelseyhightower/envconfig v1.4.0/go.mod h1:cccZRl6mQpaq41TPp5QxidR+Sa3axMbJDNb//FQX6Gg= +github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/pretty v0.2.1 h1:Fmg33tUaq4/8ym9TJN1x7sLJnHVwhP33CNkpYV/7rwI= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/labstack/echo v3.3.10+incompatible/go.mod h1:0INS7j/VjnFxD4E2wkz67b8cVwCLbBmJyDaka6Cmk1s= +github.com/labstack/echo/v4 v4.1.13/go.mod h1:3WZNypykZ3tnqpF2Qb4fPg27XDunFqgP3HGDmCMgv7U= github.com/labstack/echo/v4 v4.11.1 h1:dEpLU2FLg4UVmvCGPuk/APjlH6GDpbEPti61srUUUs4= github.com/labstack/echo/v4 v4.11.1/go.mod h1:YuYRTSM3CHs2ybfrL8Px48bO6BAnYIN4l8wSTMP6BDQ= +github.com/labstack/gommon v0.3.0/go.mod h1:MULnywXg0yavhxWKc+lOruYdAhDwPK9wf0OL7NoOu+k= github.com/labstack/gommon v0.4.0 h1:y7cvthEAEbU0yHOf4axH8ZG2NH8knB9iNSoTO8dyIk8= github.com/labstack/gommon v0.4.0/go.mod h1:uW6kP17uPlLJsD3ijUYn3/M5bAxtlZhMI6m3MFxTMTM= github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= +github.com/mattn/go-colorable v0.1.2/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= +github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= github.com/mattn/go-colorable v0.1.11/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= +github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= +github.com/mattn/go-isatty v0.0.9/go.mod h1:YNRxwqDuOph6SZLI9vUUz6OYw3QyUt7WiY2yME+cCiQ= +github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= +github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= @@ -100,8 +120,6 @@ github.com/outcaste-io/ristretto v0.2.3 h1:AK4zt/fJ76kjlYObOeNwh4T3asEuaCmp26pOv github.com/outcaste-io/ristretto v0.2.3/go.mod h1:W8HywhmtlopSB1jeMg3JtdIhf+DYkLAr0VN/s4+MHac= github.com/philhofer/fwd v1.1.2 h1:bnDivRJ1EWPjUIRXV5KfORO897HTbpFAQddBdE8t7Gw= github.com/philhofer/fwd v1.1.2/go.mod h1:qkPdfjR2SIEbspLqpe1tO4n5yICnr2DY7mqEx2tUTP0= -github.com/pingcap/errors v0.11.4 h1:lFuQV/oaUMGcD2tqt+01ROSmJs75VG1ToEOkZIZ4nE4= -github.com/pingcap/errors v0.11.4/go.mod h1:Oi8TUi2kEtXXLMJk9l1cGmz20kV3TaQ0usTwv5KuLY8= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= @@ -109,16 +127,21 @@ github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRI github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/puzpuzpuz/xsync/v2 v2.5.1 h1:mVGYAvzDSu52+zaGyNjC+24Xw2bQi3kTr4QJ6N9pIIU= github.com/puzpuzpuz/xsync/v2 v2.5.1/go.mod h1:gD2H2krq/w52MfPLE+Uy64TzJDVY7lP2znR9qmR35kU= +github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= +github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= +github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= github.com/richardartoul/molecule v1.0.1-0.20221107223329-32cfee06a052 h1:Qp27Idfgi6ACvFQat5+VJvlYToylpM/hcyLBI3WaKPA= github.com/richardartoul/molecule v1.0.1-0.20221107223329-32cfee06a052/go.mod h1:uvX/8buq8uVeiZiFht+0lqSLBHF+uGV8BrTv8W/SIwk= github.com/secure-systems-lab/go-securesystemslib v0.7.0 h1:OwvJ5jQf9LnIAS83waAjPbcMsODrTQUpJ02eNLUoxBg= github.com/secure-systems-lab/go-securesystemslib v0.7.0/go.mod h1:/2gYnlnHVQ6xeGtfIqFy7Do03K4cdCY0A/GlJLDKLHI= +github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ= github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= github.com/spaolacci/murmur3 v1.1.0 h1:7c1g84S4BPRrfL5Xrdp6fOJ206sU9y293DDHaoy0bLI= github.com/spaolacci/murmur3 v1.1.0/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.1.1/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/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/objx v0.5.1 h1:4VhoImhV/Bm0ToFkXFi8hXNXwpDRZ/ynw3amt82mzq0= @@ -144,6 +167,8 @@ github.com/tinylib/msgp v1.1.8 h1:FCXC1xanKO4I8plpHGH2P7koL/RzZs12l/+r7vakfm0= github.com/tinylib/msgp v1.1.8/go.mod h1:qkpG+2ldGg4xRFmx+jfTvZPxfGFhi64BcnL9vkCm/Tw= github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= +github.com/valyala/fasttemplate v1.0.1/go.mod h1:UQGH1tvbgY+Nz5t2n7tXsz52dQxojPUpymEIMZ47gx8= +github.com/valyala/fasttemplate v1.1.0/go.mod h1:UQGH1tvbgY+Nz5t2n7tXsz52dQxojPUpymEIMZ47gx8= github.com/valyala/fasttemplate v1.2.1/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= github.com/valyala/fasttemplate v1.2.2 h1:lxLXG0uE3Qnshl9QyaK6XJxMXlQZELvChBOCmQD0Loo= github.com/valyala/fasttemplate v1.2.2/go.mod h1:KHLXt3tVN2HBp8eijSv/kGJopbvo7S+qRAEEKiv+SiQ= @@ -161,6 +186,8 @@ go4.org/unsafe/assume-no-moving-gc v0.0.0-20230525183740-e7c30c78aeb2 h1:WJhcL4p go4.org/unsafe/assume-no-moving-gc v0.0.0-20230525183740-e7c30c78aeb2/go.mod h1:FftLjUGFEDu5k8lt0ddY+HcrH/qU/0qk+H8j9/nTl3E= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20191227163750-53104e6ec876/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20200214034016-1d94cc7ab1c6/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= @@ -175,6 +202,8 @@ golang.org/x/mod v0.12.0 h1:rmsUpXtvNzj340zd98LZ4KntptpfRHwpFOHG188oHXc= golang.org/x/mod v0.12.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= @@ -190,8 +219,14 @@ golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.3.0 h1:ftCYgMx6zT/asHUrPw8BLLscYtGznsLAnjq5RH9P66E= golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200217220822-9197077df867/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -216,6 +251,7 @@ golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9sn golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.3.0/go.mod h1:q750SLmJuPmVoN1blW3UFBPREJfb1KmY3vwxfr+nFDA= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.5.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= @@ -254,7 +290,17 @@ gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gorm.io/gorm v1.25.7 h1:VsD6acwRjz2zFxGO50gPO6AkNs7KKnvfzUjHQhZDz/A= +gorm.io/gorm v1.25.7/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= honnef.co/go/gotraceui v0.2.0 h1:dmNsfQ9Vl3GwbiVD7Z8d/osC6WtGGrasyrC2suc4ZIQ= honnef.co/go/gotraceui v0.2.0/go.mod h1:qHo4/W75cA3bX0QQoSvDjbJa4R8mAyyFjbWAj63XElc= inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a h1:1XCVEdxrvL6c0TGOhecLuB7U9zYNdxZEjvOqJreKZiM= inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a/go.mod h1:e83i32mAQOW1LAqEIweALsuK2Uw4mhQadA5r7b0Wobo= +modernc.org/libc v1.22.5 h1:91BNch/e5B0uPbJFgqbxXuOnxBQjlS//icfQEGmvyjE= +modernc.org/libc v1.22.5/go.mod h1:jj+Z7dTNX8fBScMVNRAYZ/jF91K8fdT2hYMThc3YjBY= +modernc.org/mathutil v1.5.0 h1:rV0Ko/6SfM+8G+yKiyI830l3Wuz1zRutdslNoQ0kfiQ= +modernc.org/mathutil v1.5.0/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E= +modernc.org/memory v1.5.0 h1:N+/8c5rE6EqugZwHii4IFsaJ7MUhoWX07J5tC/iI5Ds= +modernc.org/memory v1.5.0/go.mod h1:PkUhL0Mugw21sHPeskwZW4D6VscE/GQJOnIpCnW6pSU= +modernc.org/sqlite v1.23.1 h1:nrSBg4aRQQwq59JpvGEQ15tNxoO5pX/kUjcRNwSAGQM= +modernc.org/sqlite v1.23.1/go.mod h1:OrDj17Mggn6MhE+iPbBNf7RGKODDE9NFT0f3EwDzJqk= diff --git a/http-nostr_1.db b/http-nostr_1.db new file mode 100644 index 0000000000000000000000000000000000000000..2f5078daaabb4f8e432d3036f130266209d2d186 GIT binary patch literal 16384 zcmeI&+in|07y#g1JEYLcNiQnp3bDB=RTFE@JL4M&C_)j91fds{8+#7_wyQXHSg%Q1 ziHlUc1&_mf@BlmkSBSA4$4PL@LAhwqzuNWg%)bZ!-_DG@w!iuMNm)Zy467qsXFR!| zq-pYLmL*B@ATBF$nNN+lak-q%18-SBNZQ43t@YoN=GtzuUatQVS2&>n3ZMWApa2S> z01BW03ZMWA+*X0to6S2L4(V?SLJbC4hN$vg_X|_c6XlcW>0rMd$ODL zuEcuT$GtN2vT{(vbEvj@hvndM@x=D~P*ECN@WGd{Srdg*1asmo$K-lwtBvTxP85Ld_39X zTJ2W{d;2>F&$2IfpUtv*^yu4kbz^fg{c$RER6eillxMNkxN5Tq%weV(E;%!AtZsa< znI@47PDkJO01BW0 z3ZMWApa2S>z`4L?bMJ$b;B1KYTYbufRk31B8E_Jnab4Hbe`PFHJtiWt-(W73cBS>B^<#_uq-zehPjZ|4@CA z{IwfMJ1lQYr6fDQfy?5QO2YUX3}(kGePPhNX#)mA@z^W4B*8lgAiWV_y^^kAR_C4w zrCh;v> Date: Wed, 21 Feb 2024 18:14:15 +0530 Subject: [PATCH 02/10] chore: use postgres db --- go.mod | 11 +++---- go.sum | 66 +++++++++++++++++++++++++++++++---------- internal/nostr/nostr.go | 37 +++++++++++++++++------ 3 files changed, 82 insertions(+), 32 deletions(-) diff --git a/go.mod b/go.mod index 1532606..7b28dc7 100644 --- a/go.mod +++ b/go.mod @@ -3,23 +3,21 @@ module http-nostr go 1.21.3 require ( + github.com/jackc/pgx/v5 v5.3.1 github.com/joho/godotenv v1.5.1 github.com/kelseyhightower/envconfig v1.4.0 github.com/labstack/echo/v4 v4.11.1 github.com/nbd-wtf/go-nostr v0.25.7 github.com/sirupsen/logrus v1.9.3 gopkg.in/DataDog/dd-trace-go.v1 v1.58.0 + gorm.io/driver/postgres v1.4.6 ) require ( - github.com/glebarez/go-sqlite v1.21.2 // indirect + github.com/jackc/pgpassfile v1.0.0 // indirect + github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect github.com/jinzhu/inflection v1.0.0 // indirect github.com/jinzhu/now v1.1.5 // indirect - github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect - modernc.org/libc v1.22.5 // indirect - modernc.org/mathutil v1.5.0 // indirect - modernc.org/memory v1.5.0 // indirect - modernc.org/sqlite v1.23.1 // indirect ) require ( @@ -40,7 +38,6 @@ require ( github.com/dgraph-io/ristretto v0.1.1 // indirect github.com/dustin/go-humanize v1.0.1 // indirect github.com/ebitengine/purego v0.5.0 // indirect - github.com/glebarez/sqlite v1.10.0 github.com/go-gormigrate/gormigrate/v2 v2.1.1 github.com/gobwas/httphead v0.1.0 // indirect github.com/gobwas/pool v0.2.1 // indirect diff --git a/go.sum b/go.sum index e7d11a8..b2c7620 100644 --- a/go.sum +++ b/go.sum @@ -24,6 +24,7 @@ github.com/btcsuite/btcd/chaincfg/chainhash v1.0.2/go.mod h1:7SFka0XMvUgj3hfZtyd github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44= github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +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/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= @@ -34,6 +35,8 @@ github.com/decred/dcrd/crypto/blake256 v1.0.1 h1:7PltbUIQB7u/FfZ39+DGa/ShuMyJ5il github.com/decred/dcrd/crypto/blake256 v1.0.1/go.mod h1:2OfgNZ5wDpcsFmHmCK5gZTPcCXqlm2ArzUIkw9czNJo= github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0 h1:8UrgZ3GkP4i/CLijOJx79Yu+etlyjdBU4sfcs2WYQMs= github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0/go.mod h1:v57UDF4pDQJcEfFUCRop3lJL149eHGSe9Jvczhzjo/0= +github.com/denisenkom/go-mssqldb v0.11.0 h1:9rHa233rhdOyrz2GcP9NM+gi2psgJZ4GWDpL/7ND8HI= +github.com/denisenkom/go-mssqldb v0.11.0/go.mod h1:xbL0rPBG9cCiLr28tMa8zpbdarY27NDyej4t/EjAShU= github.com/dgraph-io/ristretto v0.1.1 h1:6CWw5tJNgpegArSHpNHJKldNeq03FQCwYvfMVWajOK8= github.com/dgraph-io/ristretto v0.1.1/go.mod h1:S1GPSBCYCIhmVNfcth17y2zZtQT6wzkzgwUve0VDWWA= github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= @@ -45,18 +48,20 @@ github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+m github.com/dvyukov/go-fuzz v0.0.0-20210103155950-6a8e9d1f2415/go.mod h1:11Gm+ccJnvAhCNLlf5+cS9KjtbaD5I5zaZpFMsTHWTw= github.com/ebitengine/purego v0.5.0 h1:JrMGKfRIAM4/QVKaesIIT7m/UVjTj5GYhRSQYwfVdpo= github.com/ebitengine/purego v0.5.0/go.mod h1:ah1In8AOtksoNK6yk5z1HTJeUkC1Ez4Wk2idgGslMwQ= -github.com/glebarez/go-sqlite v1.21.2 h1:3a6LFC4sKahUunAmynQKLZceZCOzUthkRkEAl9gAXWo= -github.com/glebarez/go-sqlite v1.21.2/go.mod h1:sfxdZyhQjTM2Wry3gVYWaW072Ri1WMdWJi0k6+3382k= -github.com/glebarez/sqlite v1.10.0 h1:u4gt8y7OND/cCei/NMHmfbLxF6xP2wgKcT/BJf2pYkc= -github.com/glebarez/sqlite v1.10.0/go.mod h1:IJ+lfSOmiekhQsFTJRx/lHtGYmCdtAiTaf5wI9u5uHA= github.com/go-gormigrate/gormigrate/v2 v2.1.1 h1:eGS0WTFRV30r103lU8JNXY27KbviRnqqIDobW3EV3iY= github.com/go-gormigrate/gormigrate/v2 v2.1.1/go.mod h1:L7nJ620PFDKei9QOhJzqA8kRCk+E3UbV2f5gv+1ndLc= +github.com/go-sql-driver/mysql v1.6.0 h1:BCTh4TKNUYmOmMUcQ3IipzF5prigylS7XXjEkfCHuOE= +github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg= github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU= github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM= github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og= github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= github.com/gobwas/ws v1.2.0 h1:u0p9s3xLYpZCA1z5JgCkMeB34CKCMMQbM+G8Ii7YD0I= github.com/gobwas/ws v1.2.0/go.mod h1:hRKAFb8wOxFROYNsT1bqfWnhX+b5MFeJM9r2ZSwg/KY= +github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9 h1:au07oEsX2xN0ktxqI+Sida1w446QrXBRJ0nee3SNZlA= +github.com/golang-sql/civil v0.0.0-20220223132316-b832511892a9/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0= +github.com/golang-sql/sqlexp v0.1.0 h1:ZCD6MBpcuOVfGVqsEmY5/4FtYiKz6tSyUv9LPEDei6A= +github.com/golang-sql/sqlexp v0.1.0/go.mod h1:J4ad9Vo8ZCWQ2GMrC4UCQy1JpCbwU9m3EOqtpKwwwHI= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= github.com/golang/glog v1.0.0 h1:nfP3RFugxnNRyKgeWd4oI1nYvXpxrx8ck8ZrcizshdQ= github.com/golang/glog v1.0.0/go.mod h1:EWib/APOK0SL3dFbYqvxE3UYd8E6s1ouQ7iEp/0LWV4= @@ -74,8 +79,18 @@ github.com/google/pprof v0.0.0-20230817174616-7a8ec2ada47b h1:h9U78+dx9a4BKdQkBB github.com/google/pprof v0.0.0-20230817174616-7a8ec2ada47b/go.mod h1:czg5+yv1E0ZGTi6S6vVK1mke0fV+FaUhNGcd6VRS9Ik= github.com/google/uuid v1.3.1 h1:KjJaJ9iWZ3jOFZIf1Lqf4laDRCasjl0BCmnEGxkdLb4= github.com/google/uuid v1.3.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM= +github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg= +github.com/jackc/pgservicefile v0.0.0-20200714003250-2b9c44734f2b/go.mod h1:vsD4gTJCa9TptPL8sPkXrLZ+hDuNrZCnj29CQpr4X1E= +github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk= +github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM= +github.com/jackc/pgx/v5 v5.2.0/go.mod h1:Ptn7zmohNsWEsdxRawMzk3gaKma2obW+NWTnKa0S4nk= +github.com/jackc/pgx/v5 v5.3.1 h1:Fcr8QJ1ZeLi5zsPZqQeUZhNhxfkkKBOgJuYkJHoBOtU= +github.com/jackc/pgx/v5 v5.3.1/go.mod h1:t3JDKnCBlYIc0ewLF0Q7B8MXmoIaBOZj/ic7iHozM/8= +github.com/jackc/puddle/v2 v2.1.2/go.mod h1:2lpufsF5mRHO6SuZkm0fNYxM6SWHfvyFj62KwNzgels= github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/jinzhu/now v1.1.4/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/jinzhu/now v1.1.5 h1:/o9tlHleP7gOFmsnYNz3RGnqzefHA47wQpKrrdTIwXQ= github.com/jinzhu/now v1.1.5/go.mod h1:d3SSVoowX0Lcu0IBviAWJpolVfI5UJVZZ7cO71lE/z8= github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= @@ -86,8 +101,12 @@ github.com/kelseyhightower/envconfig v1.4.0 h1:Im6hONhd3pLkfDFsbRgu68RDNkGF1r3dv github.com/kelseyhightower/envconfig v1.4.0/go.mod h1:cccZRl6mQpaq41TPp5QxidR+Sa3axMbJDNb//FQX6Gg= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= -github.com/kr/pretty v0.2.1 h1:Fmg33tUaq4/8ym9TJN1x7sLJnHVwhP33CNkpYV/7rwI= +github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= +github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= +github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= +github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= +github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/labstack/echo v3.3.10+incompatible/go.mod h1:0INS7j/VjnFxD4E2wkz67b8cVwCLbBmJyDaka6Cmk1s= @@ -97,6 +116,8 @@ github.com/labstack/echo/v4 v4.11.1/go.mod h1:YuYRTSM3CHs2ybfrL8Px48bO6BAnYIN4l8 github.com/labstack/gommon v0.3.0/go.mod h1:MULnywXg0yavhxWKc+lOruYdAhDwPK9wf0OL7NoOu+k= github.com/labstack/gommon v0.4.0 h1:y7cvthEAEbU0yHOf4axH8ZG2NH8knB9iNSoTO8dyIk8= github.com/labstack/gommon v0.4.0/go.mod h1:uW6kP17uPlLJsD3ijUYn3/M5bAxtlZhMI6m3MFxTMTM= +github.com/lib/pq v1.10.2 h1:AqzbZs4ZoCBp+GtejcpCpcxM3zlSMx29dXbUSeVtJb8= +github.com/lib/pq v1.10.2/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o= github.com/mailru/easyjson v0.7.7 h1:UGYAvKxe3sBsEDzO8ZeWOSlIQfWFlxbzLZe7hwFURr0= github.com/mailru/easyjson v0.7.7/go.mod h1:xzfreul335JAWq5oZzymOObrkdz5UnU4kGfJJLY9Nlc= github.com/mattn/go-colorable v0.1.2/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= @@ -112,6 +133,10 @@ github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27k github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/mattn/go-sqlite3 v1.14.16 h1:yOQRA0RpS5PFz/oikGwBEqvAWhWg5ufRz4ETLjwpU1Y= +github.com/mattn/go-sqlite3 v1.14.16/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= +github.com/microsoft/go-mssqldb v0.21.0 h1:p2rpHIL7TlSv1QrbXJUAcbyRKnIT0C9rRkH2E4OjLn8= +github.com/microsoft/go-mssqldb v0.21.0/go.mod h1:+4wZTUnz/SV6nffv+RRRB/ss8jPng5Sho2SmM1l2ts4= github.com/nbd-wtf/go-nostr v0.25.7 h1:DcGOSgKVr/L6w62tRtKeV2t46sRyFcq9pWcyIFkh0eM= github.com/nbd-wtf/go-nostr v0.25.7/go.mod h1:bkffJI+x914sPQWum9ZRUn66D7NpDnAoWo1yICvj3/0= github.com/opentracing/opentracing-go v1.2.0 h1:uEJPy/1a5RIPAJ0Ov+OIO8OxWu77jEv+1B0VhjKrZUs= @@ -127,11 +152,10 @@ github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRI github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/puzpuzpuz/xsync/v2 v2.5.1 h1:mVGYAvzDSu52+zaGyNjC+24Xw2bQi3kTr4QJ6N9pIIU= github.com/puzpuzpuz/xsync/v2 v2.5.1/go.mod h1:gD2H2krq/w52MfPLE+Uy64TzJDVY7lP2znR9qmR35kU= -github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= -github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= -github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= github.com/richardartoul/molecule v1.0.1-0.20221107223329-32cfee06a052 h1:Qp27Idfgi6ACvFQat5+VJvlYToylpM/hcyLBI3WaKPA= github.com/richardartoul/molecule v1.0.1-0.20221107223329-32cfee06a052/go.mod h1:uvX/8buq8uVeiZiFht+0lqSLBHF+uGV8BrTv8W/SIwk= +github.com/rogpeppe/go-internal v1.6.1 h1:/FiVV8dS/e+YqF2JvO3yXRFbBLTIuSDkuC7aBOAvL+k= +github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= github.com/secure-systems-lab/go-securesystemslib v0.7.0 h1:OwvJ5jQf9LnIAS83waAjPbcMsODrTQUpJ02eNLUoxBg= github.com/secure-systems-lab/go-securesystemslib v0.7.0/go.mod h1:/2gYnlnHVQ6xeGtfIqFy7Do03K4cdCY0A/GlJLDKLHI= github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= @@ -149,6 +173,7 @@ github.com/stretchr/objx v0.5.1/go.mod h1:/iHQpkQwBD6DLUmQ4pE+s1TXdob1mORJ4/UFdr 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= +github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= @@ -176,6 +201,7 @@ github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9dec 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= go.uber.org/atomic v1.9.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/atomic v1.10.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= go.uber.org/atomic v1.11.0 h1:ZvwS0R+56ePWxUNi+Atn9dWONBPp/AUETXlHW0DxSjE= go.uber.org/atomic v1.11.0/go.mod h1:LUxbIzbOniOlMKjJjyPfpl4v+PKK2cNJn91OQbhoJI0= go4.org/intern v0.0.0-20211027215823-ae77deb06f29/go.mod h1:cS2ma+47FKrLPdXFpr7CuxiTW3eyJbWew4qx0qtQWDA= @@ -190,6 +216,8 @@ golang.org/x/crypto v0.0.0-20191227163750-53104e6ec876/go.mod h1:LzIPMQfyMNhhGPh golang.org/x/crypto v0.0.0-20200214034016-1d94cc7ab1c6/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20220829220503-c86fa9a7ed90/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= +golang.org/x/crypto v0.4.0/go.mod h1:3quD/ATkf6oY+rnes5c3ExXTbLc8mueNue5/DoinL80= golang.org/x/crypto v0.14.0 h1:wBqGXzWJW6m1XrIKlAH0Hs1JJ7+9KBwnIO8v66Q9cHc= golang.org/x/crypto v0.14.0/go.mod h1:MVFd36DqK4CsrnJYDkBA3VC4m2GkXAM0PvzMCn4JQf4= golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53 h1:5llv2sWeaMSnA3w2kS57ouQQ4pudlXrR0dCgw51QK9o= @@ -207,6 +235,7 @@ golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLL golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +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.3.0/go.mod h1:MBQ8lrhLObU/6UmLb4fmbmk5OcyYmqtbGd/9yIeKjEE= golang.org/x/net v0.17.0 h1:pVaXccu2ozPjCXewfr1S7xza/zcXTity9cCdXQYSjIM= @@ -215,6 +244,7 @@ golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220923202941-7f9b1623fab7/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.3.0 h1:ftCYgMx6zT/asHUrPw8BLLscYtGznsLAnjq5RH9P66E= golang.org/x/sync v0.3.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y= @@ -232,6 +262,7 @@ golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -253,7 +284,9 @@ golang.org/x/term v0.3.0/go.mod h1:q750SLmJuPmVoN1blW3UFBPREJfb1KmY3vwxfr+nFDA= golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/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.5.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k= golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= @@ -283,24 +316,25 @@ google.golang.org/protobuf v1.30.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqw gopkg.in/DataDog/dd-trace-go.v1 v1.58.0 h1:ixIUarsu0RrOt7xfdrE5YSFvjgaWsP3cC3G342jTIuw= gopkg.in/DataDog/dd-trace-go.v1 v1.58.0/go.mod h1:SmnEjjV9ZQr4MWRSUYEpoPyNtmtRK5J6UuJdAma+Yxw= 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= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gorm.io/driver/mysql v1.0.1 h1:omJoilUzyrAp0xNoio88lGJCroGdIOen9hq2A/+3ifw= +gorm.io/driver/mysql v1.0.1/go.mod h1:KtqSthtg55lFp3S5kUXqlGaelnWpKitn4k1xZTnoiPw= +gorm.io/driver/postgres v1.4.6 h1:1FPESNXqIKG5JmraaH2bfCVlMQ7paLoCreFxDtqzwdc= +gorm.io/driver/postgres v1.4.6/go.mod h1:UJChCNLFKeBqQRE+HrkFUbKbq9idPXmTOk2u4Wok8S4= +gorm.io/driver/sqlserver v1.4.2 h1:nMtEeKqv2R/vv9FoHUFWfXfP6SskAgRar0TPlZV1stk= +gorm.io/driver/sqlserver v1.4.2/go.mod h1:XHwBuB4Tlh7DqO0x7Ema8dmyWsQW7wi38VQOAFkrbXY= +gorm.io/gorm v1.24.2/go.mod h1:DVrVomtaYTbqs7gB/x2uVvqnXzv0nqjB396B8cG4dBA= gorm.io/gorm v1.25.7 h1:VsD6acwRjz2zFxGO50gPO6AkNs7KKnvfzUjHQhZDz/A= gorm.io/gorm v1.25.7/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= honnef.co/go/gotraceui v0.2.0 h1:dmNsfQ9Vl3GwbiVD7Z8d/osC6WtGGrasyrC2suc4ZIQ= honnef.co/go/gotraceui v0.2.0/go.mod h1:qHo4/W75cA3bX0QQoSvDjbJa4R8mAyyFjbWAj63XElc= inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a h1:1XCVEdxrvL6c0TGOhecLuB7U9zYNdxZEjvOqJreKZiM= inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a/go.mod h1:e83i32mAQOW1LAqEIweALsuK2Uw4mhQadA5r7b0Wobo= -modernc.org/libc v1.22.5 h1:91BNch/e5B0uPbJFgqbxXuOnxBQjlS//icfQEGmvyjE= -modernc.org/libc v1.22.5/go.mod h1:jj+Z7dTNX8fBScMVNRAYZ/jF91K8fdT2hYMThc3YjBY= -modernc.org/mathutil v1.5.0 h1:rV0Ko/6SfM+8G+yKiyI830l3Wuz1zRutdslNoQ0kfiQ= -modernc.org/mathutil v1.5.0/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E= -modernc.org/memory v1.5.0 h1:N+/8c5rE6EqugZwHii4IFsaJ7MUhoWX07J5tC/iI5Ds= -modernc.org/memory v1.5.0/go.mod h1:PkUhL0Mugw21sHPeskwZW4D6VscE/GQJOnIpCnW6pSU= -modernc.org/sqlite v1.23.1 h1:nrSBg4aRQQwq59JpvGEQ15tNxoO5pX/kUjcRNwSAGQM= -modernc.org/sqlite v1.23.1/go.mod h1:OrDj17Mggn6MhE+iPbBNf7RGKODDE9NFT0f3EwDzJqk= diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index 6bbc1e9..24310ac 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -13,13 +13,17 @@ import ( "sync" "time" - "github.com/glebarez/sqlite" "github.com/joho/godotenv" "github.com/kelseyhightower/envconfig" "github.com/labstack/echo/v4" "github.com/nbd-wtf/go-nostr" "github.com/sirupsen/logrus" + "gorm.io/driver/postgres" "gorm.io/gorm" + + "github.com/jackc/pgx/v5/stdlib" + sqltrace "gopkg.in/DataDog/dd-trace-go.v1/contrib/database/sql" + gormtrace "gopkg.in/DataDog/dd-trace-go.v1/contrib/gorm.io/gorm.v1" ) type Config struct { @@ -59,15 +63,30 @@ func NewService(ctx context.Context) (*Service, error) { var db *gorm.DB var sqlDb *sql.DB - db, err = gorm.Open(sqlite.Open(cfg.DatabaseUri), &gorm.Config{}) - if err != nil { - return nil, err - } - // db.Exec("PRAGMA foreign_keys=ON;") // if we wish to enable foreign keys - sqlDb, err = db.DB() - if err != nil { - return nil, err + if os.Getenv("DATADOG_AGENT_URL") != "" { + sqltrace.Register("pgx", &stdlib.Driver{}, sqltrace.WithServiceName("nostr-wallet-connect")) + sqlDb, err = sqltrace.Open("pgx", cfg.DatabaseUri) + if err != nil { + logger.Fatalf("Failed to open DB %v", err) + return nil, err + } + db, err = gormtrace.Open(postgres.New(postgres.Config{Conn: sqlDb}), &gorm.Config{}) + if err != nil { + logger.Fatalf("Failed to open DB %v", err) + return nil, err + } + } else { + db, err = gorm.Open(postgres.Open(cfg.DatabaseUri), &gorm.Config{}) + if err != nil { + logger.Fatalf("Failed to open DB %v", err) + return nil, err + } + sqlDb, err = db.DB() + if err != nil { + logger.Fatalf("Failed to set DB config: %v", err) + return nil, err + } } sqlDb.SetMaxOpenConns(cfg.DatabaseMaxConns) From 20b4a5b033f63b07db3d653ce8aadf52f034a847 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Wed, 21 Feb 2024 18:15:46 +0530 Subject: [PATCH 03/10] chore: remove db and add gitignore --- .gitignore | 3 ++- http-nostr_1.db | Bin 16384 -> 0 bytes 2 files changed, 2 insertions(+), 1 deletion(-) delete mode 100644 http-nostr_1.db diff --git a/.gitignore b/.gitignore index 2eea525..cb7b2f0 100644 --- a/.gitignore +++ b/.gitignore @@ -1 +1,2 @@ -.env \ No newline at end of file +.env +*.db \ No newline at end of file diff --git a/http-nostr_1.db b/http-nostr_1.db deleted file mode 100644 index 2f5078daaabb4f8e432d3036f130266209d2d186..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 16384 zcmeI&+in|07y#g1JEYLcNiQnp3bDB=RTFE@JL4M&C_)j91fds{8+#7_wyQXHSg%Q1 ziHlUc1&_mf@BlmkSBSA4$4PL@LAhwqzuNWg%)bZ!-_DG@w!iuMNm)Zy467qsXFR!| zq-pYLmL*B@ATBF$nNN+lak-q%18-SBNZQ43t@YoN=GtzuUatQVS2&>n3ZMWApa2S> z01BW03ZMWA+*X0to6S2L4(V?SLJbC4hN$vg_X|_c6XlcW>0rMd$ODL zuEcuT$GtN2vT{(vbEvj@hvndM@x=D~P*ECN@WGd{Srdg*1asmo$K-lwtBvTxP85Ld_39X zTJ2W{d;2>F&$2IfpUtv*^yu4kbz^fg{c$RER6eillxMNkxN5Tq%weV(E;%!AtZsa< znI@47PDkJO01BW0 z3ZMWApa2S>z`4L?bMJ$b;B1KYTYbufRk31B8E_Jnab4Hbe`PFHJtiWt-(W73cBS>B^<#_uq-zehPjZ|4@CA z{IwfMJ1lQYr6fDQfy?5QO2YUX3}(kGePPhNX#)mA@z^W4B*8lgAiWV_y^^kAR_C4w zrCh;v> Date: Thu, 22 Feb 2024 12:38:35 +0530 Subject: [PATCH 04/10] feat: add subscription, request and response tables --- cmd/server/main.go | 2 + go.mod | 2 +- go.sum | 4 +- internal/nostr/models.go | 48 +++++-- internal/nostr/nostr.go | 120 ++++++++++++------ migrations/202402161653_initial_migration.go | 30 ++++- migrations/202402161653_initial_migration.sql | 1 - migrations/initial_migration_postgres.sql | 93 ++++++++++++++ migrations/initial_migration_sqlite.sql | 3 + 9 files changed, 241 insertions(+), 62 deletions(-) delete mode 100644 migrations/202402161653_initial_migration.sql create mode 100644 migrations/initial_migration_postgres.sql create mode 100644 migrations/initial_migration_sqlite.sql diff --git a/cmd/server/main.go b/cmd/server/main.go index 766882b..faabb81 100644 --- a/cmd/server/main.go +++ b/cmd/server/main.go @@ -48,6 +48,8 @@ func main() { defer cancel() e.Shutdown(ctx) svc.Logger.Info("Echo server exited") + svc.Relay.Close() + svc.Logger.Info("Relay connection closed") svc.Logger.Info("Waiting for service to exit...") svc.Wg.Wait() svc.Logger.Info("Service exited") diff --git a/go.mod b/go.mod index 7b28dc7..574afce 100644 --- a/go.mod +++ b/go.mod @@ -74,6 +74,6 @@ require ( golang.org/x/tools v0.12.1-0.20230815132531-74c255bcf846 // indirect golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect google.golang.org/protobuf v1.30.0 // indirect - gorm.io/gorm v1.25.7 + gorm.io/gorm v1.25.4 inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a // indirect ) diff --git a/go.sum b/go.sum index b2c7620..81c4006 100644 --- a/go.sum +++ b/go.sum @@ -332,8 +332,8 @@ gorm.io/driver/postgres v1.4.6/go.mod h1:UJChCNLFKeBqQRE+HrkFUbKbq9idPXmTOk2u4Wo gorm.io/driver/sqlserver v1.4.2 h1:nMtEeKqv2R/vv9FoHUFWfXfP6SskAgRar0TPlZV1stk= gorm.io/driver/sqlserver v1.4.2/go.mod h1:XHwBuB4Tlh7DqO0x7Ema8dmyWsQW7wi38VQOAFkrbXY= gorm.io/gorm v1.24.2/go.mod h1:DVrVomtaYTbqs7gB/x2uVvqnXzv0nqjB396B8cG4dBA= -gorm.io/gorm v1.25.7 h1:VsD6acwRjz2zFxGO50gPO6AkNs7KKnvfzUjHQhZDz/A= -gorm.io/gorm v1.25.7/go.mod h1:hbnx/Oo0ChWMn1BIhpy1oYozzpM15i4YPuHDmfYtwg8= +gorm.io/gorm v1.25.4 h1:iyNd8fNAe8W9dvtlgeRI5zSVZPsq3OpcTu37cYcpCmw= +gorm.io/gorm v1.25.4/go.mod h1:L4uxeKpfBml98NYqVqwAdmV1a2nBtAec/cf3fpucW/k= honnef.co/go/gotraceui v0.2.0 h1:dmNsfQ9Vl3GwbiVD7Z8d/osC6WtGGrasyrC2suc4ZIQ= honnef.co/go/gotraceui v0.2.0/go.mod h1:qHo4/W75cA3bX0QQoSvDjbJa4R8mAyyFjbWAj63XElc= inet.af/netaddr v0.0.0-20230525184311-b8eac61e914a h1:1XCVEdxrvL6c0TGOhecLuB7U9zYNdxZEjvOqJreKZiM= diff --git a/internal/nostr/models.go b/internal/nostr/models.go index 0fdeb32..4331687 100644 --- a/internal/nostr/models.go +++ b/internal/nostr/models.go @@ -21,17 +21,41 @@ const ( ) type Subscription struct { - ID uint - Kind int - WalletPubkey string `validate:"required"` - NostrId string `validate:"required"` - RelayUrl string - WebhookUrl string - State string - PublishState string - RepliedAt time.Time - CreatedAt time.Time - UpdatedAt time.Time + ID uint + RelayUrl string `validate:"required"` + WebhookUrl string + Open bool // "open" / "closed" + Ids *[]string `gorm:"type:jsonb"` + Kinds *[]int `gorm:"type:jsonb"` + Authors *[]string `gorm:"type:jsonb"` // WalletPubkey is included in this + Tags *nostr.TagMap `gorm:"type:jsonb"` // RequestEvent ID goes in the "e" tag + Since time.Time + Until time.Time + Limit int + Search string + CreatedAt time.Time + UpdatedAt time.Time +} + +type RequestEvent struct { + ID uint + SubscriptionId uint + NostrId string `validate:"required"` + Content string + State string + CreatedAt time.Time + UpdatedAt time.Time +} + +type ResponseEvent struct { + ID uint + SubscriptionId uint + RequestId uint `validate:"required"` + NostrId string `validate:"required"` + Content string + RepliedAt time.Time + CreatedAt time.Time + UpdatedAt time.Time } type ErrorResponse struct { @@ -46,6 +70,6 @@ type InfoRequest struct { type NIP47Request struct { RelayUrl string `json:"relayUrl"` WalletPubkey string `json:"walletPubkey"` - SignedEvent *nostr.Event `json:"event"` WebhookUrl string `json:"webhookURL"` + SignedEvent *nostr.Event `json:"event"` } diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index 24310ac..161231d 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -41,7 +41,7 @@ type Service struct { db *gorm.DB Ctx context.Context Wg *sync.WaitGroup - relay *nostr.Relay + Relay *nostr.Relay Cfg *Config Logger *logrus.Logger } @@ -116,7 +116,7 @@ func NewService(ctx context.Context) (*Service, error) { Ctx: ctx, Wg: &wg, Logger: logger, - relay: relay, + Relay: relay, } return svc, nil @@ -181,39 +181,80 @@ func (svc *Service) NIP47Handler(c echo.Context) error { }) } - + if (requestData.SignedEvent == nil) { + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: "signed event is empty", + }) + } + + // TODO: check if the RequestEvent already exists in our db + + requestEvent := RequestEvent{ + NostrId: requestData.SignedEvent.ID, + Content: requestData.SignedEvent.Content, + } + subscription := Subscription{ - WalletPubkey: requestData.WalletPubkey, - NostrId: requestData.SignedEvent.ID, - State: SUBSCRIPTION_STATE_RECEIVED, - RelayUrl: requestData.RelayUrl, - WebhookUrl: requestData.WebhookUrl, - Kind: NIP_47_RESPONSE_KIND, + RelayUrl: requestData.RelayUrl, + WebhookUrl: requestData.WebhookUrl, + Open: true, + Ids: &[]string{}, + Authors: &[]string{requestData.WalletPubkey}, + Kinds: &[]int{NIP_47_RESPONSE_KIND}, + Tags: &nostr.TagMap{"e": []string{requestEvent.NostrId}}, + Since: time.Now(), + Limit: 1, } - // mu.Lock() + svc.db.Create(&subscription) - // mu.Unlock() - if requestData.WebhookUrl != "" { + requestEvent.SubscriptionId = subscription.ID + svc.db.Create(&requestEvent) + + if subscription.WebhookUrl != "" { go func() { - event, _, err := svc.processRequest(context.Background(), &subscription, requestData.SignedEvent) + event, publishState, _, err := svc.processRequest(context.Background(), &subscription, &requestEvent, requestData.SignedEvent) + subscription.Open = false + requestEvent.State = publishState + svc.db.Save(&subscription) + svc.db.Save(&requestEvent) if err != nil { svc.Logger.WithError(err).Error("failed to process request for webhook") // what to pass to the webhook? return } + responseEvent := ResponseEvent{ + SubscriptionId: subscription.ID, + RequestId: requestEvent.ID, + NostrId: event.ID, + Content: event.Content, + RepliedAt: time.Now(), + } + svc.db.Save(&responseEvent) svc.postEventToWebhook(event, requestData.WebhookUrl) }() return c.JSON(http.StatusOK, "webhook received") } else { ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) defer cancel() - event, code, err := svc.processRequest(ctx, &subscription, requestData.SignedEvent) + event, publishState, code, err := svc.processRequest(ctx, &subscription, &requestEvent, requestData.SignedEvent) + subscription.Open = false + requestEvent.State = publishState + svc.db.Save(&subscription) + svc.db.Save(&requestEvent) if err != nil { return c.JSON(code, ErrorResponse{ Message: err.Error(), }) } + responseEvent := ResponseEvent{ + SubscriptionId: subscription.ID, + RequestId: requestEvent.ID, + NostrId: event.ID, + Content: event.Content, + RepliedAt: time.Now(), + } + svc.db.Save(&responseEvent) return c.JSON(http.StatusOK, event) } } @@ -227,8 +268,8 @@ func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL strin return relay, true, err // true means custom and the relay should be closed } // check if the default relay is active - if svc.relay.IsConnected() { - return svc.relay, false, nil + if svc.Relay.IsConnected() { + return svc.Relay, false, nil } else { svc.Logger.Info("lost connection to default relay, reconnecting...") relay, err := nostr.RelayConnect(context.Background(), svc.Cfg.DefaultRelayURL) @@ -236,34 +277,40 @@ func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL strin } } -func (svc *Service) processRequest(ctx context.Context, subscription *Subscription, requestEvent *nostr.Event) (*nostr.Event, int, error) { +func (svc *Service) processRequest(ctx context.Context, subscription *Subscription, requestEvent *RequestEvent, signedEvent *nostr.Event) (*nostr.Event, string, int, error) { + publishState := SUBSCRIPTION_STATE_PUBLISH_FAILED relay, isCustomRelay, err := svc.getRelayConnection(ctx, subscription.RelayUrl) if err != nil { - return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error connecting to relay: %w", err) + return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error connecting to relay: %w", err) } if isCustomRelay { defer relay.Close() } svc.Logger.WithFields(logrus.Fields{ - "e": requestEvent.ID, - "author": subscription.WalletPubkey, + "e": requestEvent.ID, + "authors": subscription.Authors, }).Info("subscribing to events for response...") + since := nostr.Timestamp(subscription.Since.Unix()) filter := nostr.Filter{ - Authors: []string{subscription.WalletPubkey}, - Kinds: []int{subscription.Kind}, - Tags: nostr.TagMap{"e": []string{requestEvent.ID}}, + // IDs: *subscription.Ids, + Kinds: *subscription.Kinds, + Authors: *subscription.Authors, + Tags: *subscription.Tags, + Since: &since, + Limit: subscription.Limit, } + fmt.Println(filter.Kinds[0], filter.Authors[0], filter.Tags["e"]) sub, err := relay.Subscribe(ctx, []nostr.Filter{filter}) if err != nil { - return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error subscribing to relay: %w", err) + return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error subscribing to relay: %w", err) } - status, err := relay.Publish(ctx, *requestEvent) + status, err := relay.Publish(ctx, *signedEvent) if err != nil { - return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error publishing request event: %w", err) + return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error publishing request event: %w", err) } if status == nostr.PublishStatusSucceeded { @@ -271,40 +318,31 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti "status": status, "eventId": requestEvent.ID, }).Info("published request") - subscription.PublishState = SUBSCRIPTION_STATE_PUBLISH_CONFIRMED + publishState = SUBSCRIPTION_STATE_PUBLISH_CONFIRMED } else if status == nostr.PublishStatusFailed { svc.Logger.WithFields(logrus.Fields{ "status": status, "eventId": requestEvent.ID, }).Info("failed to publish request") - subscription.PublishState = SUBSCRIPTION_STATE_PUBLISH_FAILED - subscription.State = SUBSCRIPTION_STATE_ERROR - svc.db.Save(subscription) - return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error publishing request event: %s", err.Error()) + return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error publishing request event: %s", err.Error()) } else { svc.Logger.WithFields(logrus.Fields{ "status": status, "eventId": requestEvent.ID, }).Info("request sent but no response from relay (timeout)") - subscription.PublishState = SUBSCRIPTION_STATE_PUBLISH_UNCONFIRMED + // If can somehow handle this, then publishState can be removed + publishState = SUBSCRIPTION_STATE_PUBLISH_UNCONFIRMED } - svc.db.Save(subscription) - select { case <-ctx.Done(): - subscription.State = SUBSCRIPTION_STATE_ERROR - svc.db.Save(subscription) - return &nostr.Event{}, http.StatusRequestTimeout, fmt.Errorf("request canceled or timed out") + return &nostr.Event{}, publishState, http.StatusRequestTimeout, fmt.Errorf("request canceled or timed out") case event := <-sub.Events: svc.Logger.WithFields(logrus.Fields{ "eventId": event.ID, "eventKind": event.Kind, }).Infof("successfully received event") - subscription.State = SUBSCRIPTION_STATE_EXECUTED - subscription.RepliedAt = time.Now() - svc.db.Save(subscription) - return event, http.StatusOK, nil + return event, publishState, http.StatusOK, nil } } diff --git a/migrations/202402161653_initial_migration.go b/migrations/202402161653_initial_migration.go index 2b74ec9..d98a239 100644 --- a/migrations/202402161653_initial_migration.go +++ b/migrations/202402161653_initial_migration.go @@ -2,21 +2,41 @@ package migrations import ( _ "embed" + "log" "github.com/go-gormigrate/gormigrate/v2" "gorm.io/gorm" ) -//go:embed 202402161653_initial_migration.sql -var initialMigration string +//go:embed initial_migration_postgres.sql +var initialMigrationPostgres string +//go:embed initial_migration_sqlite.sql +var initialMigrationSqlite string + +var initialMigrations = map[string]string { + "postgres": initialMigrationPostgres, + "sqlite": initialMigrationSqlite, +} // Initial migration -var _202402161653_initial_migration = &gormigrate.Migration{ +var _202402161653_initial_migration = &gormigrate.Migration { ID: "202402161653_initial_migration", Migrate: func(tx *gorm.DB) error { - return tx.Exec(initialMigration).Error + // only execute migration if subscriptions table doesn't exist + err := tx.Exec("SELECT * FROM subscriptions").Error; + if err != nil { + // find which initial migration should be executed + initialMigration := initialMigrations[tx.Dialector.Name()] + if initialMigration == "" { + log.Fatalf("unsupported database type: %s", tx.Dialector.Name()) + } + + return tx.Exec(initialMigration).Error + } + + return nil }, Rollback: func(tx *gorm.DB) error { - return nil + return nil; }, } diff --git a/migrations/202402161653_initial_migration.sql b/migrations/202402161653_initial_migration.sql deleted file mode 100644 index 20808ea..0000000 --- a/migrations/202402161653_initial_migration.sql +++ /dev/null @@ -1 +0,0 @@ -CREATE TABLE `subscriptions` (`id` integer,`kind` integer,`wallet_pubkey` text,`nostr_id` text,`relay_url` text,`webhook_url` text,`state` text,`publish_state` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); \ No newline at end of file diff --git a/migrations/initial_migration_postgres.sql b/migrations/initial_migration_postgres.sql new file mode 100644 index 0000000..2781c93 --- /dev/null +++ b/migrations/initial_migration_postgres.sql @@ -0,0 +1,93 @@ +CREATE TABLE subscriptions ( + id bigint NOT NULL, + relay_url text, + webhook_url text, + ids jsonb, + kinds jsonb, + authors jsonb, + tags jsonb, + since timestamp with time zone, + until timestamp with time zone, + "limit" integer, + search text, + "open" boolean, + created_at timestamp with time zone, + updated_at timestamp with time zone +); + +-- Create sequence for subscriptions +CREATE SEQUENCE subscriptions_id_seq + START WITH 1 + INCREMENT BY 1 + NO MINVALUE + NO MAXVALUE + CACHE 1; + +ALTER SEQUENCE subscriptions_id_seq OWNED BY subscriptions.id; + +ALTER TABLE ONLY subscriptions ALTER COLUMN id SET DEFAULT nextval('subscriptions_id_seq'::regclass); + +ALTER TABLE ONLY subscriptions + ADD CONSTRAINT subscriptions_pkey PRIMARY KEY (id); + +-- Create request_events table +CREATE TABLE request_events ( + id bigint NOT NULL, + subscription_id bigint, + nostr_id text UNIQUE, + content text, + "state" text, + created_at timestamp with time zone, + updated_at timestamp with time zone +); + +-- Create sequence for request_events +CREATE SEQUENCE request_events_id_seq + START WITH 1 + INCREMENT BY 1 + NO MINVALUE + NO MAXVALUE + CACHE 1; + +ALTER SEQUENCE request_events_id_seq OWNED BY request_events.id; + +ALTER TABLE ONLY request_events ALTER COLUMN id SET DEFAULT nextval('request_events_id_seq'::regclass); + +ALTER TABLE ONLY request_events + ADD CONSTRAINT request_events_pkey PRIMARY KEY (id); + +ALTER TABLE ONLY request_events + ADD CONSTRAINT fk_request_events_subscription FOREIGN KEY (subscription_id) REFERENCES subscriptions(id) ON DELETE CASCADE; + +-- Create response_events table +CREATE TABLE response_events ( + id bigint NOT NULL, + subscription_id bigint, + request_id bigint, + nostr_id text UNIQUE, + content text, + replied_at timestamp with time zone, + created_at timestamp with time zone, + updated_at timestamp with time zone +); + +-- Create sequence for response_events +CREATE SEQUENCE response_events_id_seq + START WITH 1 + INCREMENT BY 1 + NO MINVALUE + NO MAXVALUE + CACHE 1; + +ALTER SEQUENCE response_events_id_seq OWNED BY response_events.id; + +ALTER TABLE ONLY response_events ALTER COLUMN id SET DEFAULT nextval('response_events_id_seq'::regclass); + +ALTER TABLE ONLY response_events + ADD CONSTRAINT response_events_pkey PRIMARY KEY (id); + +ALTER TABLE ONLY response_events + ADD CONSTRAINT fk_response_events_subscription FOREIGN KEY (subscription_id) REFERENCES subscriptions(id) ON DELETE CASCADE; + +ALTER TABLE ONLY response_events + ADD CONSTRAINT fk_response_events_request_event FOREIGN KEY (request_id) REFERENCES request_events(id) ON DELETE CASCADE; \ No newline at end of file diff --git a/migrations/initial_migration_sqlite.sql b/migrations/initial_migration_sqlite.sql new file mode 100644 index 0000000..61ae286 --- /dev/null +++ b/migrations/initial_migration_sqlite.sql @@ -0,0 +1,3 @@ +CREATE TABLE "subscriptions" (`id` integer,`relay_url` text,`webhook_url` text,`ids` json,`kinds` json,`authors` json,`tags` json,`since` datetime,`until` datetime,`limit` integer,`search` text,`open` boolean,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); +CREATE TABLE "request_events" (`id` integer,`subscription_id` integer,`nostr_id` text UNIQUE,`content` text,`state` text,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_request_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE); +CREATE TABLE "response_events" (`id` integer,`subscription_id` integer,`request_id` integer,`nostr_id` text UNIQUE,`content` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_response_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_response_events_request_event` FOREIGN KEY (`request_id`) REFERENCES `request_events`(`id`) ON DELETE CASCADE); From 995924abf1f3d77b5d9247eca2c7161653426f35 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Thu, 22 Feb 2024 15:54:24 +0530 Subject: [PATCH 05/10] fix: db issues beforesave and afterfind --- internal/nostr/models.go | 109 +++++++++++++++++++--- internal/nostr/nostr.go | 59 ++++++------ migrations/initial_migration_postgres.sql | 8 +- migrations/initial_migration_sqlite.sql | 2 +- 4 files changed, 133 insertions(+), 45 deletions(-) diff --git a/internal/nostr/models.go b/internal/nostr/models.go index 4331687..c18e400 100644 --- a/internal/nostr/models.go +++ b/internal/nostr/models.go @@ -1,9 +1,11 @@ package nostr import ( + "encoding/json" "time" "github.com/nbd-wtf/go-nostr" + "gorm.io/gorm" ) const ( @@ -21,20 +23,99 @@ const ( ) type Subscription struct { - ID uint - RelayUrl string `validate:"required"` - WebhookUrl string - Open bool // "open" / "closed" - Ids *[]string `gorm:"type:jsonb"` - Kinds *[]int `gorm:"type:jsonb"` - Authors *[]string `gorm:"type:jsonb"` // WalletPubkey is included in this - Tags *nostr.TagMap `gorm:"type:jsonb"` // RequestEvent ID goes in the "e" tag - Since time.Time - Until time.Time - Limit int - Search string - CreatedAt time.Time - UpdatedAt time.Time + ID uint + RelayUrl string `validate:"required"` + WebhookUrl string + Open bool // "open" / "closed" + Ids *[]string `gorm:"-"` + Kinds *[]int `gorm:"-"` + Authors *[]string `gorm:"-"` // WalletPubkey is included in this + Tags *nostr.TagMap `gorm:"-"` // RequestEvent ID goes in the "e" tag + Since time.Time + Until time.Time + Limit int + Search string + CreatedAt time.Time + UpdatedAt time.Time + + IdsString string + KindsString string + AuthorsString string + TagsString string +} + +func (s *Subscription) BeforeSave(tx *gorm.DB) error { + var err error + if s.Ids != nil { + var idsJson []byte + idsJson, err = json.Marshal(s.Ids) + if err != nil { + return err + } + s.IdsString = string(idsJson) + } + + if s.Kinds != nil { + var kindsJson []byte + kindsJson, err = json.Marshal(s.Kinds) + if err != nil { + return err + } + s.KindsString = string(kindsJson) + } + + if s.Authors != nil { + var authorsJson []byte + authorsJson, err = json.Marshal(s.Authors) + if err != nil { + return err + } + s.AuthorsString = string(authorsJson) + } + + if s.Tags != nil { + var tagsJson []byte + tagsJson, err = json.Marshal(s.Tags) + if err != nil { + return err + } + s.TagsString = string(tagsJson) + } + + return nil +} + +func (s *Subscription) AfterFind(tx *gorm.DB) error { + var err error + if s.IdsString != "" { + err = json.Unmarshal([]byte(s.IdsString), &s.Ids) + if err != nil { + return err + } + } + + if s.KindsString != "" { + err = json.Unmarshal([]byte(s.KindsString), &s.Kinds) + if err != nil { + return err + } + } + + if s.AuthorsString != "" { + err = json.Unmarshal([]byte(s.AuthorsString), &s.Authors) + if err != nil { + return err + } + } + + if s.TagsString != "" { + err = json.Unmarshal([]byte(s.TagsString), &s.Tags) + if err != nil { + return err + } + } + + return nil } type RequestEvent struct { diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index 161231d..d6a75f1 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -98,7 +98,7 @@ func NewService(ctx context.Context) (*Service, error) { logger.Fatalf("Failed to migrate: %v", err) return nil, err } - logger.Println("Any pending migrations ran successfully") + logger.Info("Any pending migrations ran successfully") ctx, _ = signal.NotifyContext(ctx, os.Interrupt) @@ -187,33 +187,41 @@ func (svc *Service) NIP47Handler(c echo.Context) error { }) } - // TODO: check if the RequestEvent already exists in our db - - requestEvent := RequestEvent{ - NostrId: requestData.SignedEvent.ID, - Content: requestData.SignedEvent.Content, - } + subscription := Subscription{} + requestEvent := RequestEvent{} + findRequestResult := svc.db.Where("nostr_id = ?", requestData.SignedEvent.ID).Find(&requestEvent) + if findRequestResult.RowsAffected != 0 { + svc.Logger.Info("request event is already processed") + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: "request event is already processed", + }) + } else { + subscription = Subscription{ + RelayUrl: requestData.RelayUrl, + WebhookUrl: requestData.WebhookUrl, + Open: true, + Ids: &[]string{}, + Authors: &[]string{requestData.WalletPubkey}, + Kinds: &[]int{NIP_47_RESPONSE_KIND}, + Tags: &nostr.TagMap{"e": []string{requestEvent.NostrId}}, + Since: time.Now(), + Limit: 1, + } + svc.db.Create(&subscription) - subscription := Subscription{ - RelayUrl: requestData.RelayUrl, - WebhookUrl: requestData.WebhookUrl, - Open: true, - Ids: &[]string{}, - Authors: &[]string{requestData.WalletPubkey}, - Kinds: &[]int{NIP_47_RESPONSE_KIND}, - Tags: &nostr.TagMap{"e": []string{requestEvent.NostrId}}, - Since: time.Now(), - Limit: 1, + requestEvent = RequestEvent{ + NostrId: requestData.SignedEvent.ID, + Content: requestData.SignedEvent.Content, + SubscriptionId: subscription.ID, + } + svc.db.Create(&requestEvent) } - svc.db.Create(&subscription) - - requestEvent.SubscriptionId = subscription.ID - svc.db.Create(&requestEvent) - if subscription.WebhookUrl != "" { go func() { - event, publishState, _, err := svc.processRequest(context.Background(), &subscription, &requestEvent, requestData.SignedEvent) + ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) + defer cancel() + event, publishState, _, err := svc.processRequest(ctx, &subscription, &requestEvent, requestData.SignedEvent) subscription.Open = false requestEvent.State = publishState svc.db.Save(&subscription) @@ -228,7 +236,7 @@ func (svc *Service) NIP47Handler(c echo.Context) error { RequestId: requestEvent.ID, NostrId: event.ID, Content: event.Content, - RepliedAt: time.Now(), + RepliedAt: event.CreatedAt.Time(), } svc.db.Save(&responseEvent) svc.postEventToWebhook(event, requestData.WebhookUrl) @@ -252,7 +260,7 @@ func (svc *Service) NIP47Handler(c echo.Context) error { RequestId: requestEvent.ID, NostrId: event.ID, Content: event.Content, - RepliedAt: time.Now(), + RepliedAt: event.CreatedAt.Time(), } svc.db.Save(&responseEvent) return c.JSON(http.StatusOK, event) @@ -301,7 +309,6 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti Since: &since, Limit: subscription.Limit, } - fmt.Println(filter.Kinds[0], filter.Authors[0], filter.Tags["e"]) sub, err := relay.Subscribe(ctx, []nostr.Filter{filter}) if err != nil { diff --git a/migrations/initial_migration_postgres.sql b/migrations/initial_migration_postgres.sql index 2781c93..2150c96 100644 --- a/migrations/initial_migration_postgres.sql +++ b/migrations/initial_migration_postgres.sql @@ -2,10 +2,10 @@ CREATE TABLE subscriptions ( id bigint NOT NULL, relay_url text, webhook_url text, - ids jsonb, - kinds jsonb, - authors jsonb, - tags jsonb, + ids_string text, + kinds_string text, + authors_string text, + tags_string text, since timestamp with time zone, until timestamp with time zone, "limit" integer, diff --git a/migrations/initial_migration_sqlite.sql b/migrations/initial_migration_sqlite.sql index 61ae286..18dcaec 100644 --- a/migrations/initial_migration_sqlite.sql +++ b/migrations/initial_migration_sqlite.sql @@ -1,3 +1,3 @@ -CREATE TABLE "subscriptions" (`id` integer,`relay_url` text,`webhook_url` text,`ids` json,`kinds` json,`authors` json,`tags` json,`since` datetime,`until` datetime,`limit` integer,`search` text,`open` boolean,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); +CREATE TABLE "subscriptions" (`id` integer,`relay_url` text,`webhook_url` text,`ids_string` text,`kinds_string` text,`authors_string` text,`tags_string` text,`since` datetime,`until` datetime,`limit` integer,`search` text,`open` boolean,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); CREATE TABLE "request_events" (`id` integer,`subscription_id` integer,`nostr_id` text UNIQUE,`content` text,`state` text,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_request_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE); CREATE TABLE "response_events" (`id` integer,`subscription_id` integer,`request_id` integer,`nostr_id` text UNIQUE,`content` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_response_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_response_events_request_event` FOREIGN KEY (`request_id`) REFERENCES `request_events`(`id`) ON DELETE CASCADE); From 74412d9e2c018c5d28f1290c9ff7f8228061edf5 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Thu, 22 Feb 2024 15:57:51 +0530 Subject: [PATCH 06/10] chore: publish state for request event --- internal/nostr/models.go | 13 +++++-------- internal/nostr/nostr.go | 8 ++++---- 2 files changed, 9 insertions(+), 12 deletions(-) diff --git a/internal/nostr/models.go b/internal/nostr/models.go index c18e400..0dff3de 100644 --- a/internal/nostr/models.go +++ b/internal/nostr/models.go @@ -13,20 +13,17 @@ const ( NIP_47_REQUEST_KIND = 23194 NIP_47_RESPONSE_KIND = 23195 - SUBSCRIPTION_STATE_RECEIVED = "received" - SUBSCRIPTION_STATE_EXECUTED = "executed" - SUBSCRIPTION_STATE_ERROR = "error" - - SUBSCRIPTION_STATE_PUBLISH_CONFIRMED = "confirmed" - SUBSCRIPTION_STATE_PUBLISH_FAILED = "failed" - SUBSCRIPTION_STATE_PUBLISH_UNCONFIRMED = "unconfirmed" + // state of request event + REQUEST_EVENT_PUBLISH_CONFIRMED = "confirmed" + REQUEST_EVENT_PUBLISH_FAILED = "failed" + REQUEST_EVENT_PUBLISH_UNCONFIRMED = "unconfirmed" ) type Subscription struct { ID uint RelayUrl string `validate:"required"` WebhookUrl string - Open bool // "open" / "closed" + Open bool Ids *[]string `gorm:"-"` Kinds *[]int `gorm:"-"` Authors *[]string `gorm:"-"` // WalletPubkey is included in this diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index d6a75f1..998fc5c 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -286,7 +286,7 @@ func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL strin } func (svc *Service) processRequest(ctx context.Context, subscription *Subscription, requestEvent *RequestEvent, signedEvent *nostr.Event) (*nostr.Event, string, int, error) { - publishState := SUBSCRIPTION_STATE_PUBLISH_FAILED + publishState := REQUEST_EVENT_PUBLISH_FAILED relay, isCustomRelay, err := svc.getRelayConnection(ctx, subscription.RelayUrl) if err != nil { return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error connecting to relay: %w", err) @@ -325,7 +325,7 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti "status": status, "eventId": requestEvent.ID, }).Info("published request") - publishState = SUBSCRIPTION_STATE_PUBLISH_CONFIRMED + publishState = REQUEST_EVENT_PUBLISH_CONFIRMED } else if status == nostr.PublishStatusFailed { svc.Logger.WithFields(logrus.Fields{ "status": status, @@ -337,8 +337,8 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti "status": status, "eventId": requestEvent.ID, }).Info("request sent but no response from relay (timeout)") - // If can somehow handle this, then publishState can be removed - publishState = SUBSCRIPTION_STATE_PUBLISH_UNCONFIRMED + // If we can somehow handle this case, then publishState can be removed + publishState = REQUEST_EVENT_PUBLISH_UNCONFIRMED } select { From bf8d0dd7c50a9941752b9e4ab6673b1978843cf3 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Thu, 22 Feb 2024 17:56:48 +0530 Subject: [PATCH 07/10] chore: refactorings --- internal/nostr/models.go | 2 +- internal/nostr/nostr.go | 124 +++++++++++++++++++-------------------- 2 files changed, 61 insertions(+), 65 deletions(-) diff --git a/internal/nostr/models.go b/internal/nostr/models.go index 0dff3de..2d78f88 100644 --- a/internal/nostr/models.go +++ b/internal/nostr/models.go @@ -148,6 +148,6 @@ type InfoRequest struct { type NIP47Request struct { RelayUrl string `json:"relayUrl"` WalletPubkey string `json:"walletPubkey"` - WebhookUrl string `json:"webhookURL"` + WebhookUrl string `json:"webhookUrl"` SignedEvent *nostr.Event `json:"event"` } diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index 998fc5c..b927616 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -191,80 +191,54 @@ func (svc *Service) NIP47Handler(c echo.Context) error { requestEvent := RequestEvent{} findRequestResult := svc.db.Where("nostr_id = ?", requestData.SignedEvent.ID).Find(&requestEvent) if findRequestResult.RowsAffected != 0 { - svc.Logger.Info("request event is already processed") return c.JSON(http.StatusBadRequest, ErrorResponse{ Message: "request event is already processed", }) - } else { - subscription = Subscription{ - RelayUrl: requestData.RelayUrl, - WebhookUrl: requestData.WebhookUrl, - Open: true, - Ids: &[]string{}, - Authors: &[]string{requestData.WalletPubkey}, - Kinds: &[]int{NIP_47_RESPONSE_KIND}, - Tags: &nostr.TagMap{"e": []string{requestEvent.NostrId}}, - Since: time.Now(), - Limit: 1, - } - svc.db.Create(&subscription) + } - requestEvent = RequestEvent{ - NostrId: requestData.SignedEvent.ID, - Content: requestData.SignedEvent.Content, - SubscriptionId: subscription.ID, - } - svc.db.Create(&requestEvent) + subscription = Subscription{ + RelayUrl: requestData.RelayUrl, + WebhookUrl: requestData.WebhookUrl, + Open: true, + Authors: &[]string{requestData.WalletPubkey}, + Kinds: &[]int{NIP_47_RESPONSE_KIND}, + Tags: &nostr.TagMap{"e": []string{requestData.SignedEvent.ID}}, + Since: time.Now(), + Limit: 1, } + svc.db.Create(&subscription) + + requestEvent = RequestEvent{ + NostrId: requestData.SignedEvent.ID, + Content: requestData.SignedEvent.Content, + SubscriptionId: subscription.ID, + } + svc.db.Create(&requestEvent) if subscription.WebhookUrl != "" { go func() { ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) defer cancel() - event, publishState, _, err := svc.processRequest(ctx, &subscription, &requestEvent, requestData.SignedEvent) - subscription.Open = false - requestEvent.State = publishState - svc.db.Save(&subscription) - svc.db.Save(&requestEvent) + event, _, err := svc.processRequest(ctx, &subscription, &requestEvent, &requestData) if err != nil { svc.Logger.WithError(err).Error("failed to process request for webhook") // what to pass to the webhook? return } - responseEvent := ResponseEvent{ - SubscriptionId: subscription.ID, - RequestId: requestEvent.ID, - NostrId: event.ID, - Content: event.Content, - RepliedAt: event.CreatedAt.Time(), - } - svc.db.Save(&responseEvent) svc.postEventToWebhook(event, requestData.WebhookUrl) }() return c.JSON(http.StatusOK, "webhook received") - } else { - ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) - defer cancel() - event, publishState, code, err := svc.processRequest(ctx, &subscription, &requestEvent, requestData.SignedEvent) - subscription.Open = false - requestEvent.State = publishState - svc.db.Save(&subscription) - svc.db.Save(&requestEvent) - if err != nil { - return c.JSON(code, ErrorResponse{ - Message: err.Error(), - }) - } - responseEvent := ResponseEvent{ - SubscriptionId: subscription.ID, - RequestId: requestEvent.ID, - NostrId: event.ID, - Content: event.Content, - RepliedAt: event.CreatedAt.Time(), - } - svc.db.Save(&responseEvent) - return c.JSON(http.StatusOK, event) } + + ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) + defer cancel() + event, code, err := svc.processRequest(ctx, &subscription, &requestEvent, &requestData) + if err != nil { + return c.JSON(code, ErrorResponse{ + Message: err.Error(), + }) + } + return c.JSON(http.StatusOK, event) } func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL string) (*nostr.Relay, bool, error) { @@ -285,11 +259,17 @@ func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL strin } } -func (svc *Service) processRequest(ctx context.Context, subscription *Subscription, requestEvent *RequestEvent, signedEvent *nostr.Event) (*nostr.Event, string, int, error) { +func (svc *Service) processRequest(ctx context.Context, subscription *Subscription, requestEvent *RequestEvent, requestData *NIP47Request) (*nostr.Event, int, error) { publishState := REQUEST_EVENT_PUBLISH_FAILED + defer func() { + subscription.Open = false + requestEvent.State = publishState + svc.db.Save(&subscription) + svc.db.Save(&requestEvent) + }() relay, isCustomRelay, err := svc.getRelayConnection(ctx, subscription.RelayUrl) if err != nil { - return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error connecting to relay: %w", err) + return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error connecting to relay: %w", err) } if isCustomRelay { defer relay.Close() @@ -302,22 +282,30 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti since := nostr.Timestamp(subscription.Since.Unix()) filter := nostr.Filter{ - // IDs: *subscription.Ids, Kinds: *subscription.Kinds, Authors: *subscription.Authors, Tags: *subscription.Tags, Since: &since, Limit: subscription.Limit, + Search: subscription.Search, + } + + if subscription.Ids != nil { + filter.IDs = *subscription.Ids + } + if !subscription.Until.IsZero() { + until := nostr.Timestamp(subscription.Until.Unix()) + filter.Until = &until } sub, err := relay.Subscribe(ctx, []nostr.Filter{filter}) if err != nil { - return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error subscribing to relay: %w", err) + return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error subscribing to relay: %w", err) } - status, err := relay.Publish(ctx, *signedEvent) + status, err := relay.Publish(ctx, *requestData.SignedEvent) if err != nil { - return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error publishing request event: %w", err) + return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error publishing request event: %w", err) } if status == nostr.PublishStatusSucceeded { @@ -331,7 +319,7 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti "status": status, "eventId": requestEvent.ID, }).Info("failed to publish request") - return &nostr.Event{}, publishState, http.StatusBadRequest, fmt.Errorf("error publishing request event: %s", err.Error()) + return &nostr.Event{}, http.StatusBadRequest, fmt.Errorf("error publishing request event: %s", err.Error()) } else { svc.Logger.WithFields(logrus.Fields{ "status": status, @@ -343,13 +331,21 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti select { case <-ctx.Done(): - return &nostr.Event{}, publishState, http.StatusRequestTimeout, fmt.Errorf("request canceled or timed out") + return &nostr.Event{}, http.StatusRequestTimeout, fmt.Errorf("request canceled or timed out") case event := <-sub.Events: svc.Logger.WithFields(logrus.Fields{ "eventId": event.ID, "eventKind": event.Kind, }).Infof("successfully received event") - return event, publishState, http.StatusOK, nil + responseEvent := ResponseEvent{ + SubscriptionId: subscription.ID, + RequestId: requestEvent.ID, + NostrId: event.ID, + Content: event.Content, + RepliedAt: event.CreatedAt.Time(), + } + svc.db.Save(&responseEvent) + return event, http.StatusOK, nil } } From 937cc84765af5a10d5f1390fd7c248841ce722a0 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Thu, 22 Feb 2024 17:59:07 +0530 Subject: [PATCH 08/10] chore: fix migration README --- migrations/README.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/migrations/README.md b/migrations/README.md index f44f40e..1f7f95a 100644 --- a/migrations/README.md +++ b/migrations/README.md @@ -3,6 +3,9 @@ 1. Create a new file based on the current date and time (see existing migration format) 2. Copy the following code and update MY_ID_HERE and MY_COMMENT_HERE and DO_SOMETHING_HERE 3. Add the ID to the list of migrations in migrate.go +4. If possible, add a rollback function. + +*For Postgres/Sqlite specific migrations, see the [initial migration](202402161653.go)* ```go package migrations From e46ebbd2b9155631bc075f5cf36ccfc142adae70 Mon Sep 17 00:00:00 2001 From: im-adithya Date: Tue, 27 Feb 2024 15:27:21 +0530 Subject: [PATCH 09/10] feat: add subscriptions --- cmd/server/main.go | 2 + internal/nostr/models.go | 18 +- internal/nostr/nostr.go | 221 ++++++++++++++++++++-- migrations/initial_migration_postgres.sql | 2 +- migrations/initial_migration_sqlite.sql | 2 +- 5 files changed, 222 insertions(+), 23 deletions(-) diff --git a/cmd/server/main.go b/cmd/server/main.go index faabb81..197f2fa 100644 --- a/cmd/server/main.go +++ b/cmd/server/main.go @@ -33,6 +33,8 @@ func main() { e.GET("/info", svc.InfoHandler) e.POST("/nip47", svc.NIP47Handler) + e.POST("/subscribe", svc.SubscriptionHandler) + e.DELETE("/subscribe/:id", svc.StopSubscriptionHandler) e.Use(echologrus.Middleware()) //start Echo server diff --git a/internal/nostr/models.go b/internal/nostr/models.go index 2d78f88..c5fd1cc 100644 --- a/internal/nostr/models.go +++ b/internal/nostr/models.go @@ -35,6 +35,7 @@ type Subscription struct { CreatedAt time.Time UpdatedAt time.Time + // TODO: fix an elegant solution to store datatypes IdsString string KindsString string AuthorsString string @@ -117,7 +118,7 @@ func (s *Subscription) AfterFind(tx *gorm.DB) error { type RequestEvent struct { ID uint - SubscriptionId uint + SubscriptionId uint `validate:"required"` NostrId string `validate:"required"` Content string State string @@ -127,8 +128,8 @@ type RequestEvent struct { type ResponseEvent struct { ID uint - SubscriptionId uint - RequestId uint `validate:"required"` + RequestId *uint + SubscriptionId uint `validate:"required"` NostrId string `validate:"required"` Content string RepliedAt time.Time @@ -151,3 +152,14 @@ type NIP47Request struct { WebhookUrl string `json:"webhookUrl"` SignedEvent *nostr.Event `json:"event"` } + +type SubscriptionRequest struct { + RelayUrl string `json:"relayUrl"` + WebhookUrl string `json:"webhookUrl"` + Filter *nostr.Filter `json:"filter"` +} + +type SubscriptionResponse struct { + SubscriptionId uint `json:"subscription_id"` + WebhookUrl string `json:"webhookUrl"` +} \ No newline at end of file diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index b927616..c1d4f59 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -10,6 +10,7 @@ import ( "net/http" "os" "os/signal" + "strconv" "sync" "time" @@ -38,12 +39,14 @@ type Config struct { } type Service struct { - db *gorm.DB - Ctx context.Context - Wg *sync.WaitGroup - Relay *nostr.Relay - Cfg *Config - Logger *logrus.Logger + db *gorm.DB + Ctx context.Context + Wg *sync.WaitGroup + Relay *nostr.Relay + Cfg *Config + Logger *logrus.Logger + subscriptions map[uint]context.CancelFunc + mu sync.Mutex } func NewService(ctx context.Context) (*Service, error) { @@ -109,14 +112,43 @@ func NewService(ctx context.Context) (*Service, error) { return nil, err } + subscriptions := make(map[uint]context.CancelFunc) + var wg sync.WaitGroup svc := &Service{ - Cfg: cfg, - db: db, - Ctx: ctx, - Wg: &wg, - Logger: logger, - Relay: relay, + Cfg: cfg, + db: db, + Ctx: ctx, + Wg: &wg, + Logger: logger, + subscriptions: subscriptions, + Relay: relay, + } + + logger.Info("starting all open subscriptions...") + + var openSubscriptions []Subscription + if err := svc.db.Where("open = ?", true).Find(&openSubscriptions).Error; err != nil { + logger.Errorf("Failed to query open subscriptions: %v", err) + return nil, err + } + + for _, sub := range openSubscriptions { + go func(sub Subscription) { + ctx, cancel := context.WithCancel(ctx) + svc.mu.Lock() + svc.subscriptions[sub.ID] = cancel + svc.mu.Unlock() + errorChan := make(chan error) + svc.handleSubscription(ctx, &sub, errorChan) + + err := <-errorChan + if err != nil { + svc.stopSubscription(&sub) + svc.Logger.Errorf("error opening subscription %d: %v", sub.ID, err) + } + svc.Logger.Infof("opened subscription %d", sub.ID) + }(sub) } return svc, nil @@ -217,7 +249,7 @@ func (svc *Service) NIP47Handler(c echo.Context) error { if subscription.WebhookUrl != "" { go func() { - ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) + ctx, cancel := context.WithTimeout(svc.Ctx, 120*time.Second) defer cancel() event, _, err := svc.processRequest(ctx, &subscription, &requestEvent, &requestData) if err != nil { @@ -230,7 +262,7 @@ func (svc *Service) NIP47Handler(c echo.Context) error { return c.JSON(http.StatusOK, "webhook received") } - ctx, cancel := context.WithTimeout(context.Background(), 120*time.Second) + ctx, cancel := context.WithTimeout(svc.Ctx, 120*time.Second) defer cancel() event, code, err := svc.processRequest(ctx, &subscription, &requestEvent, &requestData) if err != nil { @@ -241,6 +273,158 @@ func (svc *Service) NIP47Handler(c echo.Context) error { return c.JSON(http.StatusOK, event) } +func (svc *Service) SubscriptionHandler(c echo.Context) error { + var requestData SubscriptionRequest + // send in a pubkey and authenticate by signing + if err := c.Bind(&requestData); err != nil { + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: fmt.Sprintf("error decoding subscription request: %s", err.Error()), + }) + } + + if (requestData.Filter == nil) { + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: "filters are empty", + }) + } + + if (requestData.WebhookUrl == "") { + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: "webhook url is empty", + }) + } + + subscription := Subscription{ + RelayUrl: requestData.RelayUrl, + WebhookUrl: requestData.WebhookUrl, + Open: true, + Ids: &requestData.Filter.IDs, + Authors: &requestData.Filter.Authors, + Kinds: &requestData.Filter.Kinds, + Tags: &requestData.Filter.Tags, + Limit: requestData.Filter.Limit, + Search: requestData.Filter.Search, + } + if requestData.Filter.Since != nil { + subscription.Since = requestData.Filter.Since.Time() + } + if requestData.Filter.Until != nil { + subscription.Until = requestData.Filter.Until.Time() + } + svc.db.Create(&subscription) + + errorChan := make(chan error, 1) + ctx, cancel := context.WithCancel(svc.Ctx) + svc.mu.Lock() + svc.subscriptions[subscription.ID] = cancel + svc.mu.Unlock() + go svc.handleSubscription(ctx, &subscription, errorChan) + + err := <-errorChan + if err != nil { + svc.stopSubscription(&subscription) + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: fmt.Sprintf("error setting up subscription %s",err.Error()), + }) + } + + return c.JSON(http.StatusOK, SubscriptionResponse{ + SubscriptionId: subscription.ID, + }) +} + +func (svc *Service) handleSubscription(ctx context.Context, subscription *Subscription, errorChan chan error) { + relay, isCustomRelay, err := svc.getRelayConnection(ctx, subscription.RelayUrl) + if err != nil { + errorChan <- err + return + } + if isCustomRelay { + defer relay.Close() + } + + filter := nostr.Filter{ + Limit: subscription.Limit, + Search: subscription.Search, + } + if subscription.Ids != nil { + filter.IDs = *subscription.Ids + } + if subscription.Kinds != nil { + filter.Kinds = *subscription.Kinds + } + if subscription.Authors != nil { + filter.Authors = *subscription.Authors + } + if subscription.Tags != nil { + filter.Tags = *subscription.Tags + } + if !subscription.Since.IsZero() { + since := nostr.Timestamp(subscription.Since.Unix()) + filter.Since = &since + } + if !subscription.Until.IsZero() { + until := nostr.Timestamp(subscription.Until.Unix()) + filter.Until = &until + } + + sub, err := relay.Subscribe(ctx, []nostr.Filter{filter}) + if err != nil { + errorChan <- err + return + } + errorChan <- nil + go func(){ + for event := range sub.Events { + svc.postEventToWebhook(event, subscription.WebhookUrl) + } + }() + svc.Logger.Infof("subscription %d started", subscription.ID) + <-ctx.Done() + svc.Logger.Infof("subscription %d closed", subscription.ID) + // delete svix app +} + +func (svc *Service) StopSubscriptionHandler(c echo.Context) error { + id := c.Param("id") + uint64Id, _ := strconv.ParseUint(id, 10, 64) + subId := uint(uint64Id) + + subscription := Subscription{} + findSubscriptionResult := svc.db.Where("id = ?", subId).Find(&subscription) + + if findSubscriptionResult.RowsAffected != 0 { + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: "subscription does not exist", + }) + } + + err := svc.stopSubscription(&subscription) + if err != nil { + return c.JSON(http.StatusNotFound, ErrorResponse{ + Message: err.Error(), + }) + } + + return c.JSON(http.StatusOK, fmt.Sprintf("subscription %d stopped", subId)) +} + +func (svc *Service) stopSubscription(sub *Subscription) error { + svc.mu.Lock() + cancel, exists := svc.subscriptions[sub.ID] + if exists { + cancel() + delete(svc.subscriptions, sub.ID) + sub.Open = false + svc.db.Save(sub) + } + svc.mu.Unlock() + if (!exists) { + return fmt.Errorf("cancel function of subscription doesn't exist") + } + return nil +} + func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL string) (*nostr.Relay, bool, error) { if customRelayURL != "" && customRelayURL != svc.Cfg.DefaultRelayURL { svc.Logger.WithFields(logrus.Fields{ @@ -254,7 +438,7 @@ func (svc *Service) getRelayConnection(ctx context.Context, customRelayURL strin return svc.Relay, false, nil } else { svc.Logger.Info("lost connection to default relay, reconnecting...") - relay, err := nostr.RelayConnect(context.Background(), svc.Cfg.DefaultRelayURL) + relay, err := nostr.RelayConnect(svc.Ctx, svc.Cfg.DefaultRelayURL) return relay, false, err } } @@ -264,8 +448,8 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti defer func() { subscription.Open = false requestEvent.State = publishState - svc.db.Save(&subscription) - svc.db.Save(&requestEvent) + svc.db.Save(subscription) + svc.db.Save(requestEvent) }() relay, isCustomRelay, err := svc.getRelayConnection(ctx, subscription.RelayUrl) if err != nil { @@ -339,7 +523,7 @@ func (svc *Service) processRequest(ctx context.Context, subscription *Subscripti }).Infof("successfully received event") responseEvent := ResponseEvent{ SubscriptionId: subscription.ID, - RequestId: requestEvent.ID, + RequestId: &requestEvent.ID, NostrId: event.ID, Content: event.Content, RepliedAt: event.CreatedAt.Time(), @@ -356,6 +540,7 @@ func (svc *Service) postEventToWebhook(event *nostr.Event, webhookURL string) { return } + // TODO: add svix functionality _, err = http.Post(webhookURL, "application/json", bytes.NewBuffer(eventData)) if err != nil { svc.Logger.WithError(err).Error("failed to post event to webhook") diff --git a/migrations/initial_migration_postgres.sql b/migrations/initial_migration_postgres.sql index 2150c96..1c6ecb3 100644 --- a/migrations/initial_migration_postgres.sql +++ b/migrations/initial_migration_postgres.sql @@ -63,7 +63,7 @@ ALTER TABLE ONLY request_events CREATE TABLE response_events ( id bigint NOT NULL, subscription_id bigint, - request_id bigint, + request_id bigint NULL, nostr_id text UNIQUE, content text, replied_at timestamp with time zone, diff --git a/migrations/initial_migration_sqlite.sql b/migrations/initial_migration_sqlite.sql index 18dcaec..5087d21 100644 --- a/migrations/initial_migration_sqlite.sql +++ b/migrations/initial_migration_sqlite.sql @@ -1,3 +1,3 @@ CREATE TABLE "subscriptions" (`id` integer,`relay_url` text,`webhook_url` text,`ids_string` text,`kinds_string` text,`authors_string` text,`tags_string` text,`since` datetime,`until` datetime,`limit` integer,`search` text,`open` boolean,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); CREATE TABLE "request_events" (`id` integer,`subscription_id` integer,`nostr_id` text UNIQUE,`content` text,`state` text,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_request_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE); -CREATE TABLE "response_events" (`id` integer,`subscription_id` integer,`request_id` integer,`nostr_id` text UNIQUE,`content` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_response_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_response_events_request_event` FOREIGN KEY (`request_id`) REFERENCES `request_events`(`id`) ON DELETE CASCADE); +CREATE TABLE "response_events" (`id` integer,`subscription_id` integer,`request_id` integer null,`nostr_id` text UNIQUE,`content` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_response_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_response_events_request_event` FOREIGN KEY (`request_id`) REFERENCES `request_events`(`id`) ON DELETE CASCADE); From ebfedfdbc4526a5cbbc5439ac38a0b24323fa31d Mon Sep 17 00:00:00 2001 From: im-adithya Date: Tue, 27 Feb 2024 16:00:15 +0530 Subject: [PATCH 10/10] chore: subscription fixes --- internal/nostr/nostr.go | 54 +++++++++++++++-------- migrations/initial_migration_postgres.sql | 2 +- migrations/initial_migration_sqlite.sql | 2 +- 3 files changed, 37 insertions(+), 21 deletions(-) diff --git a/internal/nostr/nostr.go b/internal/nostr/nostr.go index c1d4f59..c0f35b5 100644 --- a/internal/nostr/nostr.go +++ b/internal/nostr/nostr.go @@ -135,19 +135,19 @@ func NewService(ctx context.Context) (*Service, error) { for _, sub := range openSubscriptions { go func(sub Subscription) { - ctx, cancel := context.WithCancel(ctx) - svc.mu.Lock() - svc.subscriptions[sub.ID] = cancel - svc.mu.Unlock() - errorChan := make(chan error) - svc.handleSubscription(ctx, &sub, errorChan) - - err := <-errorChan - if err != nil { - svc.stopSubscription(&sub) - svc.Logger.Errorf("error opening subscription %d: %v", sub.ID, err) - } - svc.Logger.Infof("opened subscription %d", sub.ID) + ctx, cancel := context.WithCancel(svc.Ctx) + svc.mu.Lock() + svc.subscriptions[sub.ID] = cancel + svc.mu.Unlock() + errorChan := make(chan error) + go svc.handleSubscription(ctx, &sub, errorChan) + + err := <-errorChan + if err != nil { + svc.stopSubscription(&sub) + svc.Logger.Errorf("error opening subscription %d: %v", sub.ID, err) + } + svc.Logger.Infof("opened subscription %d", sub.ID) }(sub) } @@ -330,6 +330,7 @@ func (svc *Service) SubscriptionHandler(c echo.Context) error { return c.JSON(http.StatusOK, SubscriptionResponse{ SubscriptionId: subscription.ID, + WebhookUrl: requestData.WebhookUrl, }) } @@ -376,6 +377,17 @@ func (svc *Service) handleSubscription(ctx context.Context, subscription *Subscr errorChan <- nil go func(){ for event := range sub.Events { + svc.Logger.WithFields(logrus.Fields{ + "eventId": event.ID, + "eventKind": event.Kind, + }).Infof("received event on subscription %d", subscription.ID) + responseEvent := ResponseEvent{ + SubscriptionId: subscription.ID, + NostrId: event.ID, + Content: event.Content, + RepliedAt: event.CreatedAt.Time(), + } + svc.db.Save(&responseEvent) svc.postEventToWebhook(event, subscription.WebhookUrl) } }() @@ -391,12 +403,16 @@ func (svc *Service) StopSubscriptionHandler(c echo.Context) error { subId := uint(uint64Id) subscription := Subscription{} - findSubscriptionResult := svc.db.Where("id = ?", subId).Find(&subscription) - - if findSubscriptionResult.RowsAffected != 0 { - return c.JSON(http.StatusBadRequest, ErrorResponse{ - Message: "subscription does not exist", - }) + if err := svc.db.First(&subscription, subId).Error; err != nil { + if err == gorm.ErrRecordNotFound { + return c.JSON(http.StatusBadRequest, ErrorResponse{ + Message: "subscription does not exist", + }) + } else { + return c.JSON(http.StatusInternalServerError, ErrorResponse{ + Message: fmt.Sprintf("error occurred while fetching user: %s", err.Error()), + }) + } } err := svc.stopSubscription(&subscription) diff --git a/migrations/initial_migration_postgres.sql b/migrations/initial_migration_postgres.sql index 1c6ecb3..84fd1f1 100644 --- a/migrations/initial_migration_postgres.sql +++ b/migrations/initial_migration_postgres.sql @@ -64,7 +64,7 @@ CREATE TABLE response_events ( id bigint NOT NULL, subscription_id bigint, request_id bigint NULL, - nostr_id text UNIQUE, + nostr_id text, content text, replied_at timestamp with time zone, created_at timestamp with time zone, diff --git a/migrations/initial_migration_sqlite.sql b/migrations/initial_migration_sqlite.sql index 5087d21..2170ea7 100644 --- a/migrations/initial_migration_sqlite.sql +++ b/migrations/initial_migration_sqlite.sql @@ -1,3 +1,3 @@ CREATE TABLE "subscriptions" (`id` integer,`relay_url` text,`webhook_url` text,`ids_string` text,`kinds_string` text,`authors_string` text,`tags_string` text,`since` datetime,`until` datetime,`limit` integer,`search` text,`open` boolean,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`)); CREATE TABLE "request_events" (`id` integer,`subscription_id` integer,`nostr_id` text UNIQUE,`content` text,`state` text,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_request_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE); -CREATE TABLE "response_events" (`id` integer,`subscription_id` integer,`request_id` integer null,`nostr_id` text UNIQUE,`content` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_response_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_response_events_request_event` FOREIGN KEY (`request_id`) REFERENCES `request_events`(`id`) ON DELETE CASCADE); +CREATE TABLE "response_events" (`id` integer,`subscription_id` integer,`request_id` integer null,`nostr_id` text,`content` text,`replied_at` datetime,`created_at` datetime,`updated_at` datetime,PRIMARY KEY (`id`),CONSTRAINT `fk_response_events_subscription` FOREIGN KEY (`subscription_id`) REFERENCES `subscriptions`(`id`) ON DELETE CASCADE,CONSTRAINT `fk_response_events_request_event` FOREIGN KEY (`request_id`) REFERENCES `request_events`(`id`) ON DELETE CASCADE);