From 4f37be2b26060e35d052f59784993fca798829bc Mon Sep 17 00:00:00 2001 From: Roland Bewick Date: Wed, 24 Jan 2024 17:40:07 +0700 Subject: [PATCH] chore: move api methods into separate object --- api.go | 82 ++++++++++++++++++++++++++--------------------- http_service.go | 16 +++++---- wails_app.go | 6 +++- wails_handlers.go | 30 ++++++++--------- 4 files changed, 74 insertions(+), 60 deletions(-) diff --git a/api.go b/api.go index aef8f085..25bae7db 100644 --- a/api.go +++ b/api.go @@ -8,14 +8,22 @@ import ( "strings" "time" - "github.com/getAlby/nostr-wallet-connect/models/api" + models "github.com/getAlby/nostr-wallet-connect/models/api" "github.com/nbd-wtf/go-nostr" "gorm.io/gorm" ) -// TODO: these methods should be moved to a separate object, not in Service +type API struct { + svc *Service +} + +func NewAPI(svc *Service) *API { + return &API{ + svc: svc, + } +} -func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.CreateAppResponse, error) { +func (api *API) CreateApp(createAppRequest *models.CreateAppRequest) (*models.CreateAppResponse, error) { name := createAppRequest.Name var pairingPublicKey string var pairingSecretKey string @@ -27,7 +35,7 @@ func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.Crea //validate public key decoded, err := hex.DecodeString(pairingPublicKey) if err != nil || len(decoded) != 32 { - svc.Logger.Errorf("Invalid public key format: %s", pairingPublicKey) + api.svc.Logger.Errorf("Invalid public key format: %s", pairingPublicKey) return nil, errors.New(fmt.Sprintf("Invalid public key format: %s", pairingPublicKey)) } @@ -42,7 +50,7 @@ func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.Crea var err error expiresAt, err = time.Parse(time.RFC3339, createAppRequest.ExpiresAt) if err != nil { - svc.Logger.Errorf("Invalid expiresAt: %s", pairingPublicKey) + api.svc.Logger.Errorf("Invalid expiresAt: %s", pairingPublicKey) return nil, errors.New(fmt.Sprintf("Invalid expiresAt: %v", err)) } } @@ -51,7 +59,7 @@ func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.Crea expiresAt = time.Date(expiresAt.Year(), expiresAt.Month(), expiresAt.Day(), 23, 59, 59, 0, expiresAt.Location()) } - err := svc.db.Transaction(func(tx *gorm.DB) error { + err := api.svc.db.Transaction(func(tx *gorm.DB) error { err := tx.Save(&app).Error if err != nil { return err @@ -89,9 +97,9 @@ func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.Crea return nil, err } - relayUrl, _ := svc.cfg.Get("Relay", "") + relayUrl, _ := api.svc.cfg.Get("Relay", "") - responseBody := &api.CreateAppResponse{} + responseBody := &models.CreateAppResponse{} responseBody.Name = name responseBody.Pubkey = pairingPublicKey responseBody.PairingSecret = pairingSecretKey @@ -101,7 +109,7 @@ func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.Crea if err == nil { query := returnToUrl.Query() query.Add("relay", relayUrl) - query.Add("pubkey", svc.cfg.NostrPublicKey) + query.Add("pubkey", api.svc.cfg.NostrPublicKey) // if user.LightningAddress != "" { // query.Add("lud16", user.LightningAddress) // } @@ -114,23 +122,23 @@ func (svc *Service) CreateApp(createAppRequest *api.CreateAppRequest) (*api.Crea // if user.LightningAddress != "" { // lud16 = fmt.Sprintf("&lud16=%s", user.LightningAddress) // } - responseBody.PairingUri = fmt.Sprintf("nostr+walletconnect://%s?relay=%s&secret=%s%s", svc.cfg.NostrPublicKey, relayUrl, pairingSecretKey, lud16) + responseBody.PairingUri = fmt.Sprintf("nostr+walletconnect://%s?relay=%s&secret=%s%s", api.svc.cfg.NostrPublicKey, relayUrl, pairingSecretKey, lud16) return responseBody, nil } -func (svc *Service) DeleteApp(userApp *App) error { - return svc.db.Delete(userApp).Error +func (api *API) DeleteApp(userApp *App) error { + return api.svc.db.Delete(userApp).Error } -func (svc *Service) GetApp(userApp *App) *api.App { +func (api *API) GetApp(userApp *App) *models.App { var lastEvent NostrEvent - lastEventResult := svc.db.Where("app_id = ?", userApp.ID).Order("id desc").Limit(1).Find(&lastEvent) + lastEventResult := api.svc.db.Where("app_id = ?", userApp.ID).Order("id desc").Limit(1).Find(&lastEvent) paySpecificPermission := AppPermission{} appPermissions := []AppPermission{} var expiresAt *time.Time - svc.db.Where("app_id = ?", userApp.ID).Find(&appPermissions) + api.svc.db.Where("app_id = ?", userApp.ID).Find(&appPermissions) requestMethods := []string{} for _, appPerm := range appPermissions { @@ -148,10 +156,10 @@ func (svc *Service) GetApp(userApp *App) *api.App { budgetUsage := int64(0) maxAmount := paySpecificPermission.MaxAmount if maxAmount > 0 { - budgetUsage = svc.GetBudgetUsage(&paySpecificPermission) + budgetUsage = api.svc.GetBudgetUsage(&paySpecificPermission) } - response := api.App{ + response := models.App{ Name: userApp.Name, Description: userApp.Description, CreatedAt: userApp.CreatedAt, @@ -172,13 +180,13 @@ func (svc *Service) GetApp(userApp *App) *api.App { } -func (svc *Service) ListApps() ([]api.App, error) { +func (api *API) ListApps() ([]models.App, error) { apps := []App{} - svc.db.Find(&apps) + api.svc.db.Find(&apps) - apiApps := []api.App{} + apiApps := []models.App{} for _, userApp := range apps { - apiApp := api.App{ + apiApp := models.App{ // ID: app.ID, Name: userApp.Name, Description: userApp.Description, @@ -188,9 +196,9 @@ func (svc *Service) ListApps() ([]api.App, error) { } var lastEvent NostrEvent - result := svc.db.Where("app_id = ?", userApp.ID).Order("id desc").Limit(1).Find(&lastEvent) + result := api.svc.db.Where("app_id = ?", userApp.ID).Order("id desc").Limit(1).Find(&lastEvent) if result.Error != nil { - svc.Logger.Errorf("Failed to fetch last event %v", result.Error) + api.svc.Logger.Errorf("Failed to fetch last event %v", result.Error) return nil, errors.New("Failed to fetch last event") } if result.RowsAffected > 0 { @@ -201,40 +209,40 @@ func (svc *Service) ListApps() ([]api.App, error) { return apiApps, nil } -func (svc *Service) GetInfo() *api.InfoResponse { - info := api.InfoResponse{} - backend, _ := svc.cfg.Get("LNBackendType", "") +func (api *API) GetInfo() *models.InfoResponse { + info := models.InfoResponse{} + backend, _ := api.svc.cfg.Get("LNBackendType", "") info.SetupCompleted = backend != "" - info.Running = svc.lnClient != nil + info.Running = api.svc.lnClient != nil return &info } -func (svc *Service) Start(startRequest *api.StartRequest) error { - return svc.StartApp(startRequest.UnlockPassword) +func (api *API) Start(startRequest *models.StartRequest) error { + return api.svc.StartApp(startRequest.UnlockPassword) } -func (svc *Service) Setup(setupRequest *api.SetupRequest) error { +func (api *API) Setup(setupRequest *models.SetupRequest) error { // only update non-empty values if setupRequest.LNBackendType != "" { - svc.cfg.SetUpdate("LNBackendType", setupRequest.LNBackendType, "") + api.svc.cfg.SetUpdate("LNBackendType", setupRequest.LNBackendType, "") } if setupRequest.BreezAPIKey != "" { - svc.cfg.SetUpdate("BreezAPIKey", setupRequest.BreezAPIKey, setupRequest.UnlockPassword) + api.svc.cfg.SetUpdate("BreezAPIKey", setupRequest.BreezAPIKey, setupRequest.UnlockPassword) } if setupRequest.BreezMnemonic != "" { - svc.cfg.SetUpdate("BreezMnemonic", setupRequest.BreezMnemonic, setupRequest.UnlockPassword) + api.svc.cfg.SetUpdate("BreezMnemonic", setupRequest.BreezMnemonic, setupRequest.UnlockPassword) } if setupRequest.GreenlightInviteCode != "" { - svc.cfg.SetUpdate("GreenlightInviteCode", setupRequest.GreenlightInviteCode, setupRequest.UnlockPassword) + api.svc.cfg.SetUpdate("GreenlightInviteCode", setupRequest.GreenlightInviteCode, setupRequest.UnlockPassword) } if setupRequest.LNDAddress != "" { - svc.cfg.SetUpdate("LNDAddress", setupRequest.LNDAddress, setupRequest.UnlockPassword) + api.svc.cfg.SetUpdate("LNDAddress", setupRequest.LNDAddress, setupRequest.UnlockPassword) } if setupRequest.LNDCertHex != "" { - svc.cfg.SetUpdate("LNDCertHex", setupRequest.LNDCertHex, setupRequest.UnlockPassword) + api.svc.cfg.SetUpdate("LNDCertHex", setupRequest.LNDCertHex, setupRequest.UnlockPassword) } if setupRequest.LNDMacaroonHex != "" { - svc.cfg.SetUpdate("LNDMacaroonHex", setupRequest.LNDMacaroonHex, setupRequest.UnlockPassword) + api.svc.cfg.SetUpdate("LNDMacaroonHex", setupRequest.LNDMacaroonHex, setupRequest.UnlockPassword) } return nil diff --git a/http_service.go b/http_service.go index ad19c79f..743e937b 100644 --- a/http_service.go +++ b/http_service.go @@ -17,11 +17,13 @@ import ( type HttpService struct { svc *Service + api *API } func NewHttpService(svc *Service) *HttpService { return &HttpService{ svc: svc, + api: NewAPI(svc), } } @@ -72,7 +74,7 @@ func (httpSvc *HttpService) csrfHandler(c echo.Context) error { } func (httpSvc *HttpService) infoHandler(c echo.Context) error { - responseBody := httpSvc.svc.GetInfo() + responseBody := httpSvc.api.GetInfo() return c.JSON(http.StatusOK, responseBody) } @@ -84,7 +86,7 @@ func (httpSvc *HttpService) startHandler(c echo.Context) error { }) } - err := httpSvc.svc.Start(&startRequest) + err := httpSvc.api.Start(&startRequest) if err != nil { return c.JSON(http.StatusInternalServerError, ErrorResponse{ Message: fmt.Sprintf("Failed to start node: %s", err.Error()), @@ -111,7 +113,7 @@ func (httpSvc *HttpService) logoutHandler(c echo.Context) error { func (httpSvc *HttpService) appsListHandler(c echo.Context) error { - apps, err := httpSvc.svc.ListApps() + apps, err := httpSvc.api.ListApps() if err != nil { return c.JSON(http.StatusInternalServerError, ErrorResponse{ @@ -132,7 +134,7 @@ func (httpSvc *HttpService) appsShowHandler(c echo.Context) error { }) } - response := httpSvc.svc.GetApp(&app) + response := httpSvc.api.GetApp(&app) return c.JSON(http.StatusOK, response) } @@ -157,7 +159,7 @@ func (httpSvc *HttpService) appsDeleteHandler(c echo.Context) error { }) } - if err := httpSvc.svc.DeleteApp(&app); err != nil { + if err := httpSvc.api.DeleteApp(&app); err != nil { return c.JSON(http.StatusInternalServerError, ErrorResponse{ Message: "Failed to delete app", }) @@ -173,7 +175,7 @@ func (httpSvc *HttpService) appsCreateHandler(c echo.Context) error { }) } - responseBody, err := httpSvc.svc.CreateApp(&requestData) + responseBody, err := httpSvc.api.CreateApp(&requestData) if err != nil { httpSvc.svc.Logger.Errorf("Failed to save app: %v", err) @@ -193,7 +195,7 @@ func (httpSvc *HttpService) setupHandler(c echo.Context) error { }) } - err := httpSvc.svc.Setup(&setupRequest) + err := httpSvc.api.Setup(&setupRequest) if err != nil { return c.JSON(http.StatusInternalServerError, ErrorResponse{ Message: fmt.Sprintf("Failed to setup node: %s", err.Error()), diff --git a/wails_app.go b/wails_app.go index 94e44406..15f4ebcc 100644 --- a/wails_app.go +++ b/wails_app.go @@ -17,10 +17,14 @@ var assets embed.FS type WailsApp struct { ctx context.Context svc *Service + api *API } func NewApp(svc *Service) *WailsApp { - return &WailsApp{svc: svc} + return &WailsApp{ + svc: svc, + api: NewAPI(svc), + } } // startup is called when the app starts. The context is saved diff --git a/wails_handlers.go b/wails_handlers.go index aa2ec291..b9690d64 100644 --- a/wails_handlers.go +++ b/wails_handlers.go @@ -15,7 +15,7 @@ type WailsRequestRouterResponse struct { } // TODO: make this match echo -func (a *WailsApp) WailsRequestRouter(route string, method string, body string) WailsRequestRouterResponse { +func (app *WailsApp) WailsRequestRouter(route string, method string, body string) WailsRequestRouterResponse { appRegex := regexp.MustCompile( `/api/apps/([0-9a-f]+)`, ) @@ -27,7 +27,7 @@ func (a *WailsApp) WailsRequestRouter(route string, method string, body string) pubkey := appMatch[1] userApp := App{} - findResult := a.svc.db.Where("nostr_pubkey = ?", pubkey).First(&userApp) + findResult := app.svc.db.Where("nostr_pubkey = ?", pubkey).First(&userApp) if findResult.RowsAffected == 0 { return WailsRequestRouterResponse{Body: nil, Error: "App does not exist"} @@ -35,10 +35,10 @@ func (a *WailsApp) WailsRequestRouter(route string, method string, body string) switch method { case "GET": - app := a.svc.GetApp(&userApp) + app := app.api.GetApp(&userApp) return WailsRequestRouterResponse{Body: app, Error: ""} case "DELETE": - err := a.svc.DeleteApp(&userApp) + err := app.api.DeleteApp(&userApp) if err != nil { return WailsRequestRouterResponse{Body: nil, Error: err.Error()} } @@ -50,7 +50,7 @@ func (a *WailsApp) WailsRequestRouter(route string, method string, body string) case "/api/apps": switch method { case "GET": - apps, err := a.svc.ListApps() + apps, err := app.api.ListApps() if err != nil { return WailsRequestRouterResponse{Body: nil, Error: err.Error()} } @@ -59,37 +59,37 @@ func (a *WailsApp) WailsRequestRouter(route string, method string, body string) createAppRequest := &api.CreateAppRequest{} err := json.Unmarshal([]byte(body), createAppRequest) if err != nil { - a.svc.Logger.WithFields(logrus.Fields{ + app.svc.Logger.WithFields(logrus.Fields{ "route": route, "method": method, "body": body, }).Errorf("Failed to decode request to wails router: %v", err) return WailsRequestRouterResponse{Body: nil, Error: err.Error()} } - createAppResponse, err := a.svc.CreateApp(createAppRequest) + createAppResponse, err := app.api.CreateApp(createAppRequest) if err != nil { return WailsRequestRouterResponse{Body: nil, Error: err.Error()} } return WailsRequestRouterResponse{Body: createAppResponse, Error: ""} } case "/api/info": - infoResponse := a.svc.GetInfo() + infoResponse := app.api.GetInfo() res := WailsRequestRouterResponse{Body: *infoResponse, Error: ""} return res case "/api/start": startRequest := &api.StartRequest{} err := json.Unmarshal([]byte(body), startRequest) if err != nil { - a.svc.Logger.WithFields(logrus.Fields{ + app.svc.Logger.WithFields(logrus.Fields{ "route": route, "method": method, "body": body, }).Errorf("Failed to decode request to wails router: %v", err) return WailsRequestRouterResponse{Body: nil, Error: err.Error()} } - err = a.svc.Start(startRequest) + err = app.api.Start(startRequest) if err != nil { - a.svc.Logger.WithFields(logrus.Fields{ + app.svc.Logger.WithFields(logrus.Fields{ "route": route, "method": method, "body": body, @@ -103,16 +103,16 @@ func (a *WailsApp) WailsRequestRouter(route string, method string, body string) setupRequest := &api.SetupRequest{} err := json.Unmarshal([]byte(body), setupRequest) if err != nil { - a.svc.Logger.WithFields(logrus.Fields{ + app.svc.Logger.WithFields(logrus.Fields{ "route": route, "method": method, "body": body, }).Errorf("Failed to decode request to wails router: %v", err) return WailsRequestRouterResponse{Body: nil, Error: err.Error()} } - err = a.svc.Setup(setupRequest) + err = app.api.Setup(setupRequest) if err != nil { - a.svc.Logger.WithFields(logrus.Fields{ + app.svc.Logger.WithFields(logrus.Fields{ "route": route, "method": method, "body": body, @@ -121,6 +121,6 @@ func (a *WailsApp) WailsRequestRouter(route string, method string, body string) } return WailsRequestRouterResponse{Body: nil, Error: ""} } - a.svc.Logger.Errorf("Unhandled route: %s", route) + app.svc.Logger.Errorf("Unhandled route: %s", route) return WailsRequestRouterResponse{Body: nil, Error: fmt.Sprintf("Unhandled route: %s %s", method, route)} }