mirror of
https://github.com/ente-io/ente.git
synced 2025-05-06 05:09:22 +00:00
91 lines
2.9 KiB
Go
91 lines
2.9 KiB
Go
package emergency
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/ente-io/museum/ente"
|
|
"github.com/ente-io/museum/pkg/controller/user"
|
|
"github.com/ente-io/museum/pkg/repo"
|
|
"github.com/ente-io/museum/pkg/repo/emergency"
|
|
"github.com/ente-io/stacktrace"
|
|
"github.com/gin-gonic/gin"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
type Controller struct {
|
|
Repo *emergency.Repository
|
|
UserRepo *repo.UserRepository
|
|
UserCtrl *user.UserController
|
|
}
|
|
|
|
func (c *Controller) UpdateContact(ctx *gin.Context,
|
|
userID int64,
|
|
req ente.UpdateContact) error {
|
|
if err := validateUpdateReq(userID, req); err != nil {
|
|
return stacktrace.Propagate(err, "")
|
|
}
|
|
if req.State == ente.ContactDenied || req.State == ente.ContactLeft || req.State == ente.UserRevokedContact {
|
|
activeSessions, sessionErr := c.Repo.GetActiveSessions(ctx, req.UserID, req.EmergencyContactID)
|
|
if sessionErr != nil {
|
|
return stacktrace.Propagate(sessionErr, "")
|
|
}
|
|
for _, session := range activeSessions {
|
|
if req.State == ente.UserRevokedContact {
|
|
rejErr := c.RejectRecovery(ctx, userID, ente.RecoveryIdentifier{
|
|
ID: session.ID,
|
|
UserID: session.UserID,
|
|
EmergencyContactID: session.EmergencyContactID,
|
|
})
|
|
if rejErr != nil {
|
|
return stacktrace.Propagate(rejErr, "failed to reject recovery")
|
|
}
|
|
} else {
|
|
stopErr := c.StopRecovery(ctx, userID, ente.RecoveryIdentifier{
|
|
ID: session.ID,
|
|
UserID: session.UserID,
|
|
EmergencyContactID: session.EmergencyContactID,
|
|
})
|
|
if stopErr != nil {
|
|
return stacktrace.Propagate(stopErr, "failed to stop recovery")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
hasUpdate, err := c.Repo.UpdateState(ctx, req.UserID, req.EmergencyContactID, req.State)
|
|
if !hasUpdate {
|
|
log.WithField("userID", userID).WithField("req", req).
|
|
Warn("No update applied for emergency contact")
|
|
} else {
|
|
go c.sendContactNotification(ctx, req.UserID, req.EmergencyContactID, req.State)
|
|
}
|
|
if err != nil {
|
|
return stacktrace.Propagate(err, "")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func validateUpdateReq(userID int64, req ente.UpdateContact) error {
|
|
if req.EmergencyContactID == req.UserID {
|
|
return stacktrace.Propagate(ente.NewBadRequestWithMessage("contact and user can not be same"), "")
|
|
}
|
|
if req.EmergencyContactID != userID && req.UserID != userID {
|
|
return stacktrace.Propagate(ente.ErrPermissionDenied, "user can only update his own state")
|
|
}
|
|
|
|
isActorContact := userID == req.EmergencyContactID
|
|
if isActorContact {
|
|
if req.State == ente.ContactAccepted ||
|
|
req.State == ente.ContactLeft ||
|
|
req.State == ente.ContactDenied {
|
|
return nil
|
|
}
|
|
return stacktrace.Propagate(ente.NewBadRequestWithMessage(fmt.Sprintf("Can not update state to %s", req.State)), "")
|
|
} else {
|
|
if req.State == ente.UserInvitedContact ||
|
|
req.State == ente.UserRevokedContact {
|
|
return nil
|
|
}
|
|
return stacktrace.Propagate(ente.NewBadRequestWithMessage(fmt.Sprintf("Can not update state to %s", req.State)), "")
|
|
}
|
|
}
|