much better errors
This commit is contained in:
parent
4e72beb433
commit
1913a0429f
@ -27,14 +27,14 @@ func (api *Api) Authenticate(next http.Handler) http.Handler {
|
|||||||
|
|
||||||
headerParts := strings.Split(authorizationHeader, " ")
|
headerParts := strings.Split(authorizationHeader, " ")
|
||||||
if len(headerParts) != 2 || headerParts[0] != "Bearer" {
|
if len(headerParts) != 2 || headerParts[0] != "Bearer" {
|
||||||
api.InvalidAuthenticationTokenResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidAuthToken)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
token := headerParts[1]
|
token := headerParts[1]
|
||||||
v := validator.New()
|
v := validator.New()
|
||||||
if data.ValidateTokenPlaintext(v, token); !v.Valid() {
|
if data.ValidateTokenPlaintext(v, token); !v.Valid() {
|
||||||
api.InvalidAuthenticationTokenResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidAuthToken)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -42,7 +42,7 @@ func (api *Api) Authenticate(next http.Handler) http.Handler {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
switch {
|
switch {
|
||||||
case errors.Is(err, data.ErrRecordNotFound):
|
case errors.Is(err, data.ErrRecordNotFound):
|
||||||
api.InvalidAuthenticationTokenResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidAuthToken)
|
||||||
default:
|
default:
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -53,7 +53,7 @@ func (api *Api) Authenticate(next http.Handler) http.Handler {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
switch {
|
switch {
|
||||||
case errors.Is(err, data.ErrRecordNotFound):
|
case errors.Is(err, data.ErrRecordNotFound):
|
||||||
api.InvalidCredentialsResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidCredentials)
|
||||||
default:
|
default:
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,11 +1,11 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"party.at/party/cmd/party/common"
|
"party.at/party/cmd/party/common"
|
||||||
|
"party.at/party/internal/data"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (api *Api) RegisterDeviceToken(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) RegisterDeviceToken(w http.ResponseWriter, r *http.Request) {
|
||||||
@ -23,14 +23,14 @@ func (api *Api) RegisterDeviceToken(w http.ResponseWriter, r *http.Request) {
|
|||||||
"user_id": fmt.Sprint(user.ID),
|
"user_id": fmt.Sprint(user.ID),
|
||||||
"error": err.Error(),
|
"error": err.Error(),
|
||||||
})
|
})
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if input.Token == "" {
|
if input.Token == "" {
|
||||||
api.App.Logger.PrintInfo("register device token: empty token", map[string]string{
|
api.App.Logger.PrintInfo("register device token: empty token", map[string]string{
|
||||||
"user_id": fmt.Sprint(user.ID),
|
"user_id": fmt.Sprint(user.ID),
|
||||||
})
|
})
|
||||||
api.BadRequestResponse(w, r, errors.New("token must be provided"))
|
api.errorResponse(w, r, data.ErrNoToken)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -55,11 +55,11 @@ func (api *Api) DeleteDeviceToken(w http.ResponseWriter, r *http.Request) {
|
|||||||
Token string `json:"token"`
|
Token string `json:"token"`
|
||||||
}
|
}
|
||||||
if err := common.ReadJSON(w, r, &input); err != nil {
|
if err := common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if input.Token == "" {
|
if input.Token == "" {
|
||||||
api.BadRequestResponse(w, r, errors.New("token must be provided"))
|
api.errorResponse(w, r, data.ErrNoToken)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -76,7 +76,7 @@ func (api *Api) DeleteDeviceToken(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !owns {
|
if !owns {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -1,20 +1,13 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import(
|
import(
|
||||||
"fmt"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"errors"
|
||||||
|
|
||||||
"party.at/party/cmd/party/common"
|
"party.at/party/cmd/party/common"
|
||||||
|
"party.at/party/internal/data"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ── Error responses ──────────────────────────────────────────────────────────
|
|
||||||
|
|
||||||
type apiError struct {
|
|
||||||
Code common.ErrorCode `json:"code,omitempty"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
Details map[string]string `json:"details,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) LogError(r *http.Request, err error) {
|
func (api *Api) LogError(r *http.Request, err error) {
|
||||||
api.App.Logger.PrintError(err, map[string]string{
|
api.App.Logger.PrintError(err, map[string]string{
|
||||||
"request_method": r.Method,
|
"request_method": r.Method,
|
||||||
@ -22,124 +15,26 @@ func (api *Api) LogError(r *http.Request, err error) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (api *Api) errorResponse(w http.ResponseWriter, r *http.Request, status int, ae apiError) {
|
func (api *Api) errorResponse(w http.ResponseWriter, r *http.Request, err error) {
|
||||||
if err := common.WriteJSON(w, status, common.Envelope{"error": ae}, nil); err != nil {
|
apiErr := &data.Error{
|
||||||
|
HttpCode: http.StatusInternalServerError,
|
||||||
|
Message: "the server encountered a problem",
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try to "unbox" the error to see if it's our rich *Error type
|
||||||
|
var customErr *data.Error
|
||||||
|
if errors.As(err, &customErr) {
|
||||||
|
apiErr = customErr
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := common.WriteJSON(w, apiErr.HttpCode, common.Envelope{"error": apiErr}, nil); err != nil {
|
||||||
api.App.LogError(r, err)
|
api.App.LogError(r, err)
|
||||||
w.WriteHeader(500)
|
w.WriteHeader(500)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (api *Api) ServerErrorResponse(w http.ResponseWriter, r *http.Request, err error) {
|
func (api *Api) ServerErrorResponse(w http.ResponseWriter, r *http.Request, err error) {
|
||||||
api.App.LogError(r, err)
|
api.App.LogError(r, err)
|
||||||
api.errorResponse(w, r, http.StatusInternalServerError, apiError{
|
|
||||||
Message: "the server encountered a problem and could not process your request",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) NotFoundResponse(w http.ResponseWriter, r *http.Request) {
|
api.errorResponse(w, r, err)
|
||||||
api.errorResponse(w, r, http.StatusNotFound, apiError{
|
|
||||||
Message: "the requested resource could not be found",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) MethodNotAllowedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusMethodNotAllowed, apiError{
|
|
||||||
Message: fmt.Sprintf("the %s method is not supported for this resource", r.Method),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) BadRequestResponse(w http.ResponseWriter, r *http.Request, err error) {
|
|
||||||
api.errorResponse(w, r, http.StatusBadRequest, apiError{Message: err.Error()})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) FailedValidationResponse(w http.ResponseWriter, r *http.Request, errors map[string]string) {
|
|
||||||
api.errorResponse(w, r, http.StatusUnprocessableEntity, apiError{
|
|
||||||
Code: common.ErrCodeValidationFailed,
|
|
||||||
Message: "validation failed",
|
|
||||||
Details: errors,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) EditConflictResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusConflict, apiError{
|
|
||||||
Code: common.ErrCodeEditConflict,
|
|
||||||
Message: "unable to update the record due to an edit conflict, please try again",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) InvalidCredentialsResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusUnauthorized, apiError{
|
|
||||||
Code: common.ErrCodeInvalidCredentials,
|
|
||||||
Message: "invalid authentication credentials",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) InvalidAuthenticationTokenResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
w.Header().Set("WWW-Authenticate", "Bearer")
|
|
||||||
api.errorResponse(w, r, http.StatusUnauthorized, apiError{
|
|
||||||
Code: common.ErrCodeInvalidAuthToken,
|
|
||||||
Message: "invalid or missing authentication token",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) RateLimitExceededResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusTooManyRequests, apiError{
|
|
||||||
Message: "rate limit exceeded",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) AuthenticationRequiredResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusUnauthorized, apiError{
|
|
||||||
Code: common.ErrCodeAuthRequired,
|
|
||||||
Message: "you must be authenticated to access this resource",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) InactiveAccountResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusForbidden, apiError{
|
|
||||||
Code: common.ErrCodeInactiveAccount,
|
|
||||||
Message: "your user account must be activated to access this resource",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) NotPermittedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusForbidden, apiError{
|
|
||||||
Code: common.ErrCodeNotPermitted,
|
|
||||||
Message: "your user account doesn't have the necessary permissions to access this resource",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) AlreadyVotedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusConflict, apiError{
|
|
||||||
Code: common.ErrCodeAlreadyVoted,
|
|
||||||
Message: "your user account already voted for this issue",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) AlreadyBlindSignedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusConflict, apiError{
|
|
||||||
Code: common.ErrCodeAlreadyBlindSigned,
|
|
||||||
Message: "already requested a blind signature for this issue",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) BlindedVoteOutOfRangeResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusUnprocessableEntity, apiError{
|
|
||||||
Code: common.ErrCodeBlindedVoteRange,
|
|
||||||
Message: "blinded_vote is out of valid range",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) InvalidSignatureResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusUnprocessableEntity, apiError{
|
|
||||||
Code: common.ErrCodeInvalidSignature,
|
|
||||||
Message: "invalid signature",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (api *Api) VoteAlreadyCastResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
api.errorResponse(w, r, http.StatusConflict, apiError{
|
|
||||||
Code: common.ErrCodeVoteAlreadyCast,
|
|
||||||
Message: "this vote has already been cast",
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -27,7 +27,7 @@ func (api *Api) ListIssues(w http.ResponseWriter, r *http.Request) {
|
|||||||
input.Filters.SortSafelist = []string{"id", "-id", "title", "-title", "description", "-description"}
|
input.Filters.SortSafelist = []string{"id", "-id", "title", "-title", "description", "-description"}
|
||||||
|
|
||||||
if data.ValidateFilters(v, input.Filters); !v.Valid() {
|
if data.ValidateFilters(v, input.Filters); !v.Valid() {
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
api.errorResponse(w, r, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,18 +52,13 @@ func (api *Api) CreateIssue(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := common.ReadJSON(w, r, &input); err != nil {
|
if err := common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
issue, options, err := api.App.CreateIssue(input.Title, input.Description, input.StartTime, input.EndTime, input.Options)
|
issue, options, err := api.App.CreateIssue(input.Title, input.Description, input.StartTime, input.EndTime, input.Options)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
var ve *common.ValidationError
|
api.errorResponse(w, r, err)
|
||||||
if errors.As(err, &ve) {
|
|
||||||
api.FailedValidationResponse(w, r, ve.Errors)
|
|
||||||
} else {
|
|
||||||
api.ServerErrorResponse(w, r, err)
|
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -77,14 +72,14 @@ func (api *Api) CreateIssue(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (api *Api) ReadIssue(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) ReadIssue(w http.ResponseWriter, r *http.Request) {
|
||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
result, err := api.App.GetIssue(id, common.GetUser(r))
|
result, err := api.App.GetIssue(id, common.GetUser(r))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.Is(err, data.ErrRecordNotFound) {
|
if errors.Is(err, data.ErrRecordNotFound) {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
} else {
|
} else {
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -99,7 +94,7 @@ func (api *Api) ReadIssue(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (api *Api) UpdateIssue(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) UpdateIssue(w http.ResponseWriter, r *http.Request) {
|
||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,23 +106,13 @@ func (api *Api) UpdateIssue(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err = common.ReadJSON(w, r, &input); err != nil {
|
if err = common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
issue, err := api.App.UpdateIssue(id, input.Title, input.Description, input.StartTime, input.EndTime)
|
issue, err := api.App.UpdateIssue(id, input.Title, input.Description, input.StartTime, input.EndTime)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
var ve *common.ValidationError
|
api.errorResponse(w, r, err)
|
||||||
switch {
|
|
||||||
case errors.As(err, &ve):
|
|
||||||
api.FailedValidationResponse(w, r, ve.Errors)
|
|
||||||
case errors.Is(err, data.ErrRecordNotFound):
|
|
||||||
api.NotFoundResponse(w, r)
|
|
||||||
case errors.Is(err, data.ErrEditConflict):
|
|
||||||
api.EditConflictResponse(w, r)
|
|
||||||
default:
|
|
||||||
api.ServerErrorResponse(w, r, err)
|
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -139,13 +124,13 @@ func (api *Api) UpdateIssue(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (api *Api) DeleteIssue(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) DeleteIssue(w http.ResponseWriter, r *http.Request) {
|
||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = api.App.DeleteIssue(id); err != nil {
|
if err = api.App.DeleteIssue(id); err != nil {
|
||||||
if errors.Is(err, data.ErrRecordNotFound) {
|
if errors.Is(err, data.ErrRecordNotFound) {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
} else {
|
} else {
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -160,14 +145,14 @@ func (api *Api) DeleteIssue(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (api *Api) ReadIssuePubKey(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) ReadIssuePubKey(w http.ResponseWriter, r *http.Request) {
|
||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
pubKey, err := api.App.GetIssuePublicKey(id)
|
pubKey, err := api.App.GetIssuePublicKey(id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.Is(err, data.ErrRecordNotFound) {
|
if errors.Is(err, data.ErrRecordNotFound) {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
} else {
|
} else {
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -182,7 +167,7 @@ func (api *Api) ReadIssuePubKey(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (api *Api) BlindSignIssueVote(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) BlindSignIssueVote(w http.ResponseWriter, r *http.Request) {
|
||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -190,22 +175,13 @@ func (api *Api) BlindSignIssueVote(w http.ResponseWriter, r *http.Request) {
|
|||||||
BlindedVote []byte `json:"blinded_vote"`
|
BlindedVote []byte `json:"blinded_vote"`
|
||||||
}
|
}
|
||||||
if err = common.ReadJSON(w, r, &input); err != nil {
|
if err = common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
signed, err := api.App.BlindSign(id, input.BlindedVote, common.GetUser(r))
|
signed, err := api.App.BlindSign(id, input.BlindedVote, common.GetUser(r))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
switch {
|
api.errorResponse(w, r, err)
|
||||||
case errors.Is(err, data.ErrRecordNotFound):
|
|
||||||
api.NotFoundResponse(w, r)
|
|
||||||
case errors.Is(err, data.ErrDuplicateBlindSign):
|
|
||||||
api.AlreadyBlindSignedResponse(w, r)
|
|
||||||
case errors.Is(err, data.ErrInvalidBlindedVote):
|
|
||||||
api.BlindedVoteOutOfRangeResponse(w, r)
|
|
||||||
default:
|
|
||||||
api.ServerErrorResponse(w, r, err)
|
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -1,13 +1,15 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"party.at/party/cmd/party/common"
|
|
||||||
"time"
|
|
||||||
"sync"
|
|
||||||
"golang.org/x/time/rate"
|
|
||||||
"net"
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/time/rate"
|
||||||
|
"party.at/party/cmd/party/common"
|
||||||
|
"party.at/party/internal/data"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (api *Api) RecoverPanic(next http.Handler) http.Handler {
|
func (api *Api) RecoverPanic(next http.Handler) http.Handler {
|
||||||
@ -25,7 +27,7 @@ func (api *Api) RecoverPanic(next http.Handler) http.Handler {
|
|||||||
func (api *Api) RequireAuthenticatedUser(next http.HandlerFunc) http.HandlerFunc {
|
func (api *Api) RequireAuthenticatedUser(next http.HandlerFunc) http.HandlerFunc {
|
||||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
if common.GetUser(r).IsAnonymous() {
|
if common.GetUser(r).IsAnonymous() {
|
||||||
api.AuthenticationRequiredResponse(w, r)
|
api.errorResponse(w, r, data.ErrAuthRequired)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
next.ServeHTTP(w, r)
|
next.ServeHTTP(w, r)
|
||||||
@ -35,7 +37,7 @@ func (api *Api) RequireAuthenticatedUser(next http.HandlerFunc) http.HandlerFunc
|
|||||||
func (api *Api) RequireActivatedUser(next http.HandlerFunc) http.HandlerFunc {
|
func (api *Api) RequireActivatedUser(next http.HandlerFunc) http.HandlerFunc {
|
||||||
fn := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
fn := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
if !common.GetUser(r).Activated {
|
if !common.GetUser(r).Activated {
|
||||||
api.InactiveAccountResponse(w, r)
|
api.errorResponse(w, r, data.ErrInactiveAccount)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
next.ServeHTTP(w, r)
|
next.ServeHTTP(w, r)
|
||||||
@ -51,7 +53,7 @@ func (api *Api) RequirePermission(code string, next http.HandlerFunc) http.Handl
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
if !permissions.Include(code) {
|
if !permissions.Include(code) {
|
||||||
api.NotPermittedResponse(w, r)
|
api.errorResponse(w, r, data.ErrNotPermitted)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
next.ServeHTTP(w, r)
|
next.ServeHTTP(w, r)
|
||||||
@ -96,7 +98,7 @@ func (api *Api) RateLimit(next http.Handler) http.Handler {
|
|||||||
clients[ip].lastSeen = time.Now()
|
clients[ip].lastSeen = time.Now()
|
||||||
if !clients[ip].limiter.Allow() {
|
if !clients[ip].limiter.Allow() {
|
||||||
mu.Unlock()
|
mu.Unlock()
|
||||||
api.RateLimitExceededResponse(w, r)
|
api.errorResponse(w, r, data.ErrRateLimitExceeded)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
mu.Unlock()
|
mu.Unlock()
|
||||||
|
|||||||
@ -1,18 +1,19 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"party.at/party/cmd/party/common"
|
"party.at/party/cmd/party/common"
|
||||||
|
"party.at/party/internal/data"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (api *Api) GetParlVoteDetail(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) GetParlVoteDetail(w http.ResponseWriter, r *http.Request) {
|
||||||
routePath := r.PathValue("path")
|
routePath := r.PathValue("path")
|
||||||
routePath = strings.TrimPrefix(routePath, "/")
|
routePath = strings.TrimPrefix(routePath, "/")
|
||||||
if routePath == "" {
|
if routePath == "" {
|
||||||
api.BadRequestResponse(w, r, errors.New("path is required"))
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -24,7 +25,7 @@ func (api *Api) GetParlVoteDetail(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
if detail == nil {
|
if detail == nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -17,7 +17,7 @@ func (api *Api) CreateAuthenticationToken(w http.ResponseWriter, r *http.Request
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := common.ReadJSON(w, r, &input); err != nil {
|
if err := common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -25,7 +25,7 @@ func (api *Api) CreateAuthenticationToken(w http.ResponseWriter, r *http.Request
|
|||||||
v.Check(input.Email != "", "email", "must be provided")
|
v.Check(input.Email != "", "email", "must be provided")
|
||||||
v.Check(input.Password != "", "password", "must be provided")
|
v.Check(input.Password != "", "password", "must be provided")
|
||||||
if !v.Valid() {
|
if !v.Valid() {
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
api.errorResponse(w, r, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -33,7 +33,7 @@ func (api *Api) CreateAuthenticationToken(w http.ResponseWriter, r *http.Request
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
switch {
|
switch {
|
||||||
case errors.Is(err, data.ErrInvalidCredentials):
|
case errors.Is(err, data.ErrInvalidCredentials):
|
||||||
api.InvalidCredentialsResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidCredentials)
|
||||||
default:
|
default:
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -49,14 +49,14 @@ func (api *Api) DeleteAuthenticationToken(w http.ResponseWriter, r *http.Request
|
|||||||
authHeader := r.Header.Get("Authorization")
|
authHeader := r.Header.Get("Authorization")
|
||||||
parts := strings.Split(authHeader, " ")
|
parts := strings.Split(authHeader, " ")
|
||||||
if len(parts) != 2 || parts[0] != "Bearer" {
|
if len(parts) != 2 || parts[0] != "Bearer" {
|
||||||
api.InvalidAuthenticationTokenResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidAuthToken)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
token := parts[1]
|
token := parts[1]
|
||||||
v := validator.New()
|
v := validator.New()
|
||||||
if data.ValidateTokenPlaintext(v, token); !v.Valid() {
|
if data.ValidateTokenPlaintext(v, token); !v.Valid() {
|
||||||
api.InvalidAuthenticationTokenResponse(w, r)
|
api.errorResponse(w, r, data.ErrInvalidAuthToken)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -25,7 +25,7 @@ func (api *Api) ListUsers(w http.ResponseWriter, r *http.Request) {
|
|||||||
input.Filters.SortSafelist = []string{"id", "-id", "name", "-name", "email", "-email", "created", "-created"}
|
input.Filters.SortSafelist = []string{"id", "-id", "name", "-name", "email", "-email", "created", "-created"}
|
||||||
|
|
||||||
if data.ValidateFilters(v, input.Filters); !v.Valid() {
|
if data.ValidateFilters(v, input.Filters); !v.Valid() {
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
api.errorResponse(w, r, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -55,10 +55,12 @@ func (api *Api) CreateUser(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := common.ReadJSON(w, r, &input); err != nil {
|
if err := common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
user, authToken, err := api.App.RegisterUser(common.RegisterUserInput{
|
user, authToken, err := api.App.RegisterUser(common.RegisterUserInput{
|
||||||
ProviderID: input.ProviderId,
|
ProviderID: input.ProviderId,
|
||||||
Username: input.Username,
|
Username: input.Username,
|
||||||
@ -72,21 +74,7 @@ func (api *Api) CreateUser(w http.ResponseWriter, r *http.Request) {
|
|||||||
Address: input.Address,
|
Address: input.Address,
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
var ve *common.ValidationError
|
api.errorResponse(w, r, err)
|
||||||
switch {
|
|
||||||
case errors.As(err, &ve):
|
|
||||||
api.FailedValidationResponse(w, r, ve.Errors)
|
|
||||||
case errors.Is(err, data.ErrDuplicateEmail):
|
|
||||||
v := validator.New()
|
|
||||||
v.AddError("email", "a user with this email address already exists")
|
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
|
||||||
case errors.Is(err, data.ErrDuplicateUser):
|
|
||||||
v := validator.New()
|
|
||||||
v.AddError("username", "a user with this username already exists")
|
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
|
||||||
default:
|
|
||||||
api.ServerErrorResponse(w, r, err)
|
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -105,23 +93,20 @@ func (api *Api) ReadUser(w http.ResponseWriter, r *http.Request) {
|
|||||||
var err error
|
var err error
|
||||||
id, err = strconv.ParseInt(param, 10, 64)
|
id, err = strconv.ParseInt(param, 10, 64)
|
||||||
if err != nil || id < 1 {
|
if err != nil || id < 1 {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if common.GetUser(r).ID != id {
|
if common.GetUser(r).ID != id {
|
||||||
api.errorResponse(w, r, http.StatusForbidden, apiError{
|
api.errorResponse(w, r, data.ErrNotPermitted)
|
||||||
Code: common.ErrCodeNotPermitted,
|
|
||||||
Message: "your user account doesn't have the necessary permissions to access this resource",
|
|
||||||
})
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
user, err := api.App.GetUser(id)
|
user, err := api.App.GetUser(id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.Is(err, data.ErrRecordNotFound) {
|
if errors.Is(err, data.ErrRecordNotFound) {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
} else {
|
} else {
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -137,7 +122,7 @@ func (api *Api) ReadUser(w http.ResponseWriter, r *http.Request) {
|
|||||||
// user := common.GetUser(r).ID
|
// user := common.GetUser(r).ID
|
||||||
// // if err != nil {
|
// // if err != nil {
|
||||||
// // if errors.Is(err, data.ErrRecordNotFound) {
|
// // if errors.Is(err, data.ErrRecordNotFound) {
|
||||||
// // api.NotFoundResponse(w, r)
|
// // api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
// // } else {
|
// // } else {
|
||||||
// // api.ServerErrorResponse(w, r, err)
|
// // api.ServerErrorResponse(w, r, err)
|
||||||
// // }
|
// // }
|
||||||
@ -152,13 +137,13 @@ func (api *Api) ReadUser(w http.ResponseWriter, r *http.Request) {
|
|||||||
func (api *Api) DeleteUser(w http.ResponseWriter, r *http.Request) {
|
func (api *Api) DeleteUser(w http.ResponseWriter, r *http.Request) {
|
||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = api.App.DeleteUser(id); err != nil {
|
if err = api.App.DeleteUser(id); err != nil {
|
||||||
if errors.Is(err, data.ErrRecordNotFound) {
|
if errors.Is(err, data.ErrRecordNotFound) {
|
||||||
api.NotFoundResponse(w, r)
|
api.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
} else {
|
} else {
|
||||||
api.ServerErrorResponse(w, r, err)
|
api.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
@ -176,27 +161,19 @@ func (api *Api) ActivateUser(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := common.ReadJSON(w, r, &input); err != nil {
|
if err := common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
v := validator.New()
|
v := validator.New()
|
||||||
if data.ValidateTokenPlaintext(v, input.TokenPlaintext); !v.Valid() {
|
if data.ValidateTokenPlaintext(v, input.TokenPlaintext); !v.Valid() {
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
api.errorResponse(w, r, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors))
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
user, err := api.App.ActivateUser(input.TokenPlaintext)
|
user, err := api.App.ActivateUser(input.TokenPlaintext)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
switch {
|
api.errorResponse(w, r, err)
|
||||||
case errors.Is(err, data.ErrRecordNotFound):
|
|
||||||
v.AddError("token", "invalid or expired activation token")
|
|
||||||
api.FailedValidationResponse(w, r, v.Errors)
|
|
||||||
case errors.Is(err, data.ErrEditConflict):
|
|
||||||
api.EditConflictResponse(w, r)
|
|
||||||
default:
|
|
||||||
api.ServerErrorResponse(w, r, err)
|
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
package api
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"party.at/party/internal/data"
|
"party.at/party/internal/data"
|
||||||
@ -17,21 +16,13 @@ func (api *Api) Vote(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := common.ReadJSON(w, r, &input); err != nil {
|
if err := common.ReadJSON(w, r, &input); err != nil {
|
||||||
api.BadRequestResponse(w, r, err)
|
api.errorResponse(w, r, data.ErrBadRequest)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := api.App.CastVote(input.IssueID, input.OptionID, input.Nonce, input.Signature); err != nil {
|
if err := api.App.CastVote(input.IssueID, input.OptionID, input.Nonce, input.Signature); err != nil {
|
||||||
switch {
|
api.ServerErrorResponse(w, r, err)
|
||||||
case errors.Is(err, data.ErrRecordNotFound):
|
|
||||||
api.NotFoundResponse(w, r)
|
|
||||||
case errors.Is(err, data.ErrInvalidSignature):
|
|
||||||
api.InvalidSignatureResponse(w, r)
|
|
||||||
case errors.Is(err, data.ErrDuplicateVote):
|
|
||||||
api.AlreadyVotedResponse(w, r)
|
|
||||||
default:
|
|
||||||
api.ServerErrorResponse(w, r, err)
|
|
||||||
}
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -64,14 +64,6 @@ type Application struct {
|
|||||||
CORSTrustedOrigins []string
|
CORSTrustedOrigins []string
|
||||||
}
|
}
|
||||||
|
|
||||||
type ValidationError struct {
|
|
||||||
Errors map[string]string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (e *ValidationError) Error() string {
|
|
||||||
return "validation failed"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *Application) background(fn func()) {
|
func (app *Application) background(fn func()) {
|
||||||
go func() {
|
go func() {
|
||||||
defer func() {
|
defer func() {
|
||||||
|
|||||||
@ -1,25 +0,0 @@
|
|||||||
package common
|
|
||||||
|
|
||||||
type ErrorCode int
|
|
||||||
|
|
||||||
const (
|
|
||||||
// 401 variants
|
|
||||||
ErrCodeInvalidCredentials ErrorCode = 4011
|
|
||||||
ErrCodeInvalidAuthToken ErrorCode = 4012
|
|
||||||
ErrCodeAuthRequired ErrorCode = 4013
|
|
||||||
|
|
||||||
// 403 variants
|
|
||||||
ErrCodeInactiveAccount ErrorCode = 4031
|
|
||||||
ErrCodeNotPermitted ErrorCode = 4032
|
|
||||||
|
|
||||||
// 409 variants
|
|
||||||
ErrCodeEditConflict ErrorCode = 4091
|
|
||||||
ErrCodeAlreadyVoted ErrorCode = 4092
|
|
||||||
ErrCodeAlreadyBlindSigned ErrorCode = 4093
|
|
||||||
ErrCodeVoteAlreadyCast ErrorCode = 4094
|
|
||||||
|
|
||||||
// 422 variants
|
|
||||||
ErrCodeValidationFailed ErrorCode = 4221
|
|
||||||
ErrCodeBlindedVoteRange ErrorCode = 4222
|
|
||||||
ErrCodeInvalidSignature ErrorCode = 4223
|
|
||||||
)
|
|
||||||
@ -15,6 +15,7 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"party.at/party/internal/validator"
|
"party.at/party/internal/validator"
|
||||||
|
"party.at/party/internal/data"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Envelope map[string]interface{}
|
type Envelope map[string]interface{}
|
||||||
@ -49,14 +50,14 @@ func ReadJSON(w http.ResponseWriter, r *http.Request, dst interface{}) error {
|
|||||||
case errors.As(err, &syntaxError):
|
case errors.As(err, &syntaxError):
|
||||||
return fmt.Errorf("body contains badly-formed JSON (at character %d)", syntaxError.Offset)
|
return fmt.Errorf("body contains badly-formed JSON (at character %d)", syntaxError.Offset)
|
||||||
case errors.Is(err, io.ErrUnexpectedEOF):
|
case errors.Is(err, io.ErrUnexpectedEOF):
|
||||||
return errors.New("body contains badly-formed JSON")
|
return data.ErrBadlyFormedJSON
|
||||||
case errors.As(err, &unmarshalTypeError):
|
case errors.As(err, &unmarshalTypeError):
|
||||||
if unmarshalTypeError.Field != "" {
|
if unmarshalTypeError.Field != "" {
|
||||||
return fmt.Errorf("body contains incorrect JSON type for field %q", unmarshalTypeError.Field)
|
return fmt.Errorf("body contains incorrect JSON type for field %q", unmarshalTypeError.Field)
|
||||||
}
|
}
|
||||||
return fmt.Errorf("body contains incorrect JSON type (at character %d)", unmarshalTypeError.Offset)
|
return fmt.Errorf("body contains incorrect JSON type (at character %d)", unmarshalTypeError.Offset)
|
||||||
case errors.Is(err, io.EOF):
|
case errors.Is(err, io.EOF):
|
||||||
return errors.New("body must not be empty")
|
return data.ErrBodyEmpty
|
||||||
case strings.HasPrefix(err.Error(), "json: unknown field "):
|
case strings.HasPrefix(err.Error(), "json: unknown field "):
|
||||||
fieldName := strings.TrimPrefix(err.Error(), "json: unknown field ")
|
fieldName := strings.TrimPrefix(err.Error(), "json: unknown field ")
|
||||||
return fmt.Errorf("body contains unknown key %s", fieldName)
|
return fmt.Errorf("body contains unknown key %s", fieldName)
|
||||||
@ -70,7 +71,7 @@ func ReadJSON(w http.ResponseWriter, r *http.Request, dst interface{}) error {
|
|||||||
}
|
}
|
||||||
err = dec.Decode(&struct{}{})
|
err = dec.Decode(&struct{}{})
|
||||||
if err != io.EOF {
|
if err != io.EOF {
|
||||||
return errors.New("body must only contain a single JSON value")
|
return data.ErrSingleValue
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -91,7 +92,7 @@ func GenerateIssueKey() ([]byte, int, []byte, error) {
|
|||||||
func ReadIDParam(r *http.Request) (int64, error) {
|
func ReadIDParam(r *http.Request) (int64, error) {
|
||||||
id, err := strconv.ParseInt(r.PathValue("id"), 10, 64)
|
id, err := strconv.ParseInt(r.PathValue("id"), 10, 64)
|
||||||
if err != nil || id < 1 {
|
if err != nil || id < 1 {
|
||||||
return 0, errors.New("invalid id parameter")
|
return 0, data.ErrInvalidID
|
||||||
}
|
}
|
||||||
return id, nil
|
return id, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@ -77,7 +77,7 @@ func (app *Application) CreateIssue(title, description string, startTime, endTim
|
|||||||
data.ValidateOption(v, option)
|
data.ValidateOption(v, option)
|
||||||
}
|
}
|
||||||
if data.ValidateIssue(v, issue); !v.Valid() {
|
if data.ValidateIssue(v, issue); !v.Valid() {
|
||||||
return nil, nil, &ValidationError{Errors: v.Errors}
|
return nil, nil, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors)
|
||||||
}
|
}
|
||||||
|
|
||||||
n, e, privatePEM, err := generateIssueKey()
|
n, e, privatePEM, err := generateIssueKey()
|
||||||
@ -145,7 +145,7 @@ func (app *Application) UpdateIssue(id int64, title, description *string, startT
|
|||||||
|
|
||||||
v := validator.New()
|
v := validator.New()
|
||||||
if data.ValidateIssue(v, issue); !v.Valid() {
|
if data.ValidateIssue(v, issue); !v.Valid() {
|
||||||
return nil, &ValidationError{Errors: v.Errors}
|
return nil, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = app.Models.Issues.Update(issue); err != nil {
|
if err = app.Models.Issues.Update(issue); err != nil {
|
||||||
@ -167,12 +167,16 @@ func (app *Application) GetIssuePublicKey(id int64) (*PublicKey, error) {
|
|||||||
return &PublicKey{N: hex.EncodeToString(issue.N), E: issue.E}, nil
|
return &PublicKey{N: hex.EncodeToString(issue.N), E: issue.E}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (app *Application) BlindSign(issueID int64, blindedVote []byte, user *data.User) ([]byte, error) {
|
func (app *Application) BlindSign(issueID int64, blindedVote []byte, user *data.User) ([]byte, error) {
|
||||||
issue, err := app.Models.Issues.Get(issueID)
|
issue, err := app.Models.Issues.Get(issueID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if issue.StartTime.After(time.Now()) {
|
||||||
|
return nil, data.ErrHasNotStarted
|
||||||
|
}
|
||||||
|
|
||||||
blindSign := &data.BlindSign{UserID: user.ID, IssueID: issue.ID}
|
blindSign := &data.BlindSign{UserID: user.ID, IssueID: issue.ID}
|
||||||
if err = app.Models.BlindSigns.Insert(blindSign); err != nil {
|
if err = app.Models.BlindSigns.Insert(blindSign); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
|||||||
@ -45,7 +45,7 @@ func (app *Application) RegisterUser(input RegisterUserInput) (*data.User, *data
|
|||||||
data.ValidateUser(v, user)
|
data.ValidateUser(v, user)
|
||||||
data.ValidateUserIdentity(v, userIdentity)
|
data.ValidateUserIdentity(v, userIdentity)
|
||||||
if !v.Valid() {
|
if !v.Valid() {
|
||||||
return nil, nil, &ValidationError{Errors: v.Errors}
|
return nil, nil, data.ErrValidationFailed.(*data.Error).WithDetails(v.Errors)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := app.Models.Users.ExecuteRegistrationTx(user, userIdentity); err != nil {
|
if err := app.Models.Users.ExecuteRegistrationTx(user, userIdentity); err != nil {
|
||||||
@ -54,7 +54,7 @@ func (app *Application) RegisterUser(input RegisterUserInput) (*data.User, *data
|
|||||||
|
|
||||||
role := "viewer"
|
role := "viewer"
|
||||||
if app.Config.Env == "development" {
|
if app.Config.Env == "development" {
|
||||||
role = ""
|
role = "admin"
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := app.Models.Roles.AssignToUser(user.ID, role); err != nil {
|
if err := app.Models.Roles.AssignToUser(user.ID, role); err != nil {
|
||||||
|
|||||||
@ -1,8 +1,8 @@
|
|||||||
package web
|
package web
|
||||||
|
|
||||||
import(
|
import(
|
||||||
"fmt"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"errors"
|
||||||
|
|
||||||
"party.at/party/internal/data"
|
"party.at/party/internal/data"
|
||||||
"party.at/party/cmd/party/common"
|
"party.at/party/cmd/party/common"
|
||||||
@ -10,12 +10,6 @@ import(
|
|||||||
|
|
||||||
// ── Error responses ──────────────────────────────────────────────────────────
|
// ── Error responses ──────────────────────────────────────────────────────────
|
||||||
|
|
||||||
type webError struct {
|
|
||||||
Code common.ErrorCode `json:"code,omitempty"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
Details map[string]string `json:"details,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) LogError(r *http.Request, err error) {
|
func (web *Web) LogError(r *http.Request, err error) {
|
||||||
web.App.Logger.PrintError(err, map[string]string{
|
web.App.Logger.PrintError(err, map[string]string{
|
||||||
"request_method": r.Method,
|
"request_method": r.Method,
|
||||||
@ -23,131 +17,32 @@ func (web *Web) LogError(r *http.Request, err error) {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (web *Web) errorResponse(w http.ResponseWriter, r *http.Request, status int, we webError) {
|
func (web *Web) errorResponse(w http.ResponseWriter, r *http.Request, err error) {
|
||||||
// if err := common.WriteJSON(w, status, common.Envelope{"error": we}, nil); err != nil {
|
displayError := data.Error{
|
||||||
// web.LogError(r, err)
|
HttpCode: http.StatusInternalServerError,
|
||||||
// w.WriteHeader(500)
|
Message: "the server encountered a problem and could not process your request",
|
||||||
// }
|
}
|
||||||
|
|
||||||
|
var customErr *data.Error
|
||||||
|
if errors.As(err, &customErr) {
|
||||||
|
displayError = *customErr
|
||||||
|
}
|
||||||
|
|
||||||
user := common.GetUser(r)
|
user := common.GetUser(r)
|
||||||
|
|
||||||
web.render(w, r, http.StatusOK, "error", struct {
|
web.render(w, r, displayError.HttpCode, "error", struct {
|
||||||
AuthenticatedUser *data.User
|
AuthenticatedUser *data.User
|
||||||
FormErrors []string
|
|
||||||
IsDevelopment bool
|
IsDevelopment bool
|
||||||
Status int
|
Error data.Error
|
||||||
We webError
|
|
||||||
}{
|
}{
|
||||||
AuthenticatedUser: user,
|
AuthenticatedUser: user,
|
||||||
IsDevelopment: web.App.Config.Env == "development",
|
IsDevelopment: web.App.Config.Env == "development",
|
||||||
Status: status,
|
Error: displayError,
|
||||||
We: we,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (web *Web) ServerErrorResponse(w http.ResponseWriter, r *http.Request, err error) {
|
func (web *Web) ServerErrorResponse(w http.ResponseWriter, r *http.Request, err error) {
|
||||||
web.LogError(r, err)
|
web.LogError(r, err)
|
||||||
web.errorResponse(w, r, http.StatusInternalServerError, webError{
|
|
||||||
Message: "the server encountered a problem and could not process your request",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) NotFoundResponse(w http.ResponseWriter, r *http.Request) {
|
web.errorResponse(w, r, err)
|
||||||
web.errorResponse(w, r, http.StatusNotFound, webError{
|
|
||||||
Message: "the requested resource could not be found",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) MethodNotAllowedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusMethodNotAllowed, webError{
|
|
||||||
Message: fmt.Sprintf("the %s method is not supported for this resource", r.Method),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) BadRequestResponse(w http.ResponseWriter, r *http.Request, err error) {
|
|
||||||
web.errorResponse(w, r, http.StatusBadRequest, webError{Message: err.Error()})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) FailedValidationResponse(w http.ResponseWriter, r *http.Request, errors map[string]string) {
|
|
||||||
web.errorResponse(w, r, http.StatusUnprocessableEntity, webError{
|
|
||||||
Code: common.ErrCodeValidationFailed,
|
|
||||||
Message: "validation failed",
|
|
||||||
Details: errors,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) EditConflictResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusConflict, webError{
|
|
||||||
Code: common.ErrCodeEditConflict,
|
|
||||||
Message: "unable to update the record due to an edit conflict, please try again",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) InvalidCredentialsResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusUnauthorized, webError{
|
|
||||||
Code: common.ErrCodeInvalidCredentials,
|
|
||||||
Message: "invalid authentication credentials",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) RateLimitExceededResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusTooManyRequests, webError{
|
|
||||||
Message: "rate limit exceeded",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) AuthenticationRequiredResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusUnauthorized, webError{
|
|
||||||
Code: common.ErrCodeAuthRequired,
|
|
||||||
Message: "you must be authenticated to access this resource",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) InactiveAccountResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusForbidden, webError{
|
|
||||||
Code: common.ErrCodeInactiveAccount,
|
|
||||||
Message: "your user account must be activated to access this resource",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) NotPermittedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusForbidden, webError{
|
|
||||||
Code: common.ErrCodeNotPermitted,
|
|
||||||
Message: "your user account doesn't have the necessary permissions to access this resource",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) AlreadyVotedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusConflict, webError{
|
|
||||||
Code: common.ErrCodeAlreadyVoted,
|
|
||||||
Message: "your user account already voted for this issue",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) AlreadyBlindSignedResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusConflict, webError{
|
|
||||||
Code: common.ErrCodeAlreadyBlindSigned,
|
|
||||||
Message: "already requested a blind signature for this issue",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) BlindedVoteOutOfRangeResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusUnprocessableEntity, webError{
|
|
||||||
Code: common.ErrCodeBlindedVoteRange,
|
|
||||||
Message: "blinded_vote is out of valid range",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) InvalidSignatureResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusUnprocessableEntity, webError{
|
|
||||||
Code: common.ErrCodeInvalidSignature,
|
|
||||||
Message: "invalid signature",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (web *Web) VoteAlreadyCastResponse(w http.ResponseWriter, r *http.Request) {
|
|
||||||
web.errorResponse(w, r, http.StatusConflict, webError{
|
|
||||||
Code: common.ErrCodeVoteAlreadyCast,
|
|
||||||
Message: "this vote has already been cast",
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -93,6 +93,7 @@ func (web *Web) IssuePage(w http.ResponseWriter, r *http.Request) {
|
|||||||
id, err := common.ReadIDParam(r)
|
id, err := common.ReadIDParam(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.NotFound(w, r)
|
http.NotFound(w, r)
|
||||||
|
web.errorResponse(w, r, data.ErrRecordNotFound)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -239,9 +240,11 @@ func (web *Web) BlindSignIssueVote(w http.ResponseWriter, r *http.Request) {
|
|||||||
case errors.Is(err, data.ErrRecordNotFound):
|
case errors.Is(err, data.ErrRecordNotFound):
|
||||||
http.NotFound(w, r)
|
http.NotFound(w, r)
|
||||||
case errors.Is(err, data.ErrDuplicateBlindSign):
|
case errors.Is(err, data.ErrDuplicateBlindSign):
|
||||||
common.WriteJSON(w, http.StatusConflict, common.Envelope{"error": map[string]string{"message": "bereits eine Blindsignatur für diese Abstimmung angefordert"}}, nil)
|
common.WriteJSON(w, http.StatusConflict, common.Envelope{"error": map[string]string{"message": err.Error()}}, nil)
|
||||||
case errors.Is(err, data.ErrInvalidBlindedVote):
|
case errors.Is(err, data.ErrInvalidBlindedVote):
|
||||||
common.WriteJSON(w, http.StatusUnprocessableEntity, common.Envelope{"error": map[string]string{"message": err.Error()}}, nil)
|
common.WriteJSON(w, http.StatusUnprocessableEntity, common.Envelope{"error": map[string]string{"message": err.Error()}}, nil)
|
||||||
|
case errors.Is(err, data.ErrHasNotStarted):
|
||||||
|
common.WriteJSON(w, http.StatusUnprocessableEntity, common.Envelope{"error": map[string]string{"message": err.Error()}}, nil)
|
||||||
default:
|
default:
|
||||||
web.App.LogError(r, err)
|
web.App.LogError(r, err)
|
||||||
common.WriteJSON(w, http.StatusInternalServerError, common.Envelope{"error": map[string]string{"message": "internal server error"}}, nil)
|
common.WriteJSON(w, http.StatusInternalServerError, common.Envelope{"error": map[string]string{"message": "internal server error"}}, nil)
|
||||||
@ -250,6 +253,6 @@ func (web *Web) BlindSignIssueVote(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err = common.WriteJSON(w, http.StatusOK, common.Envelope{"signed": signed}, nil); err != nil {
|
if err = common.WriteJSON(w, http.StatusOK, common.Envelope{"signed": signed}, nil); err != nil {
|
||||||
web.App.LogError(r, err)
|
web.ServerErrorResponse(w, r, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -91,7 +91,7 @@ func (web *Web) RequireActivatedUser(next http.HandlerFunc) http.HandlerFunc {
|
|||||||
func (web *Web) RequirePermission(code string, next http.HandlerFunc) http.HandlerFunc {
|
func (web *Web) RequirePermission(code string, next http.HandlerFunc) http.HandlerFunc {
|
||||||
return web.RequireActivatedUser(func(w http.ResponseWriter, r *http.Request) {
|
return web.RequireActivatedUser(func(w http.ResponseWriter, r *http.Request) {
|
||||||
if !common.GetPermissions(r).Include(code) {
|
if !common.GetPermissions(r).Include(code) {
|
||||||
web.NotPermittedResponse(w, r)
|
web.errorResponse(w, r, data.ErrNotPermitted)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
next.ServeHTTP(w, r)
|
next.ServeHTTP(w, r)
|
||||||
|
|||||||
@ -48,20 +48,12 @@ func (web *Web) RegisterUserPage(w http.ResponseWriter, r *http.Request) {
|
|||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
var formErrors []string
|
var formErrors []string
|
||||||
var ve *common.ValidationError
|
var customErr *data.Error
|
||||||
switch {
|
|
||||||
case errors.As(err, &ve):
|
if errors.As(err, &customErr) && len(customErr.Details) > 0 {
|
||||||
for _, msg := range ve.Errors {
|
formErrors = customErr.DetailMessages()
|
||||||
formErrors = append(formErrors, msg)
|
|
||||||
}
|
|
||||||
case errors.Is(err, data.ErrDuplicateEmail):
|
|
||||||
formErrors = []string{"Diese E-Mail-Adresse wird bereits verwendet."}
|
|
||||||
case errors.Is(err, data.ErrDuplicateUser):
|
|
||||||
formErrors = []string{"Dieser Benutzername wird bereits verwendet."}
|
|
||||||
default:
|
|
||||||
web.App.LogError(r, err)
|
|
||||||
formErrors = []string{"Ein Fehler ist aufgetreten. Bitte versuchen Sie es erneut."}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
web.render(w, r, http.StatusUnprocessableEntity, "register", struct {
|
web.render(w, r, http.StatusUnprocessableEntity, "register", struct {
|
||||||
AuthenticatedUser *data.User
|
AuthenticatedUser *data.User
|
||||||
FormErrors []string
|
FormErrors []string
|
||||||
|
|||||||
100
internal/data/errors.go
Normal file
100
internal/data/errors.go
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
package data
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ErrorCode int
|
||||||
|
|
||||||
|
const (
|
||||||
|
// 401 variants
|
||||||
|
ErrCodeInvalidCredentials ErrorCode = 4011
|
||||||
|
ErrCodeInvalidAuthToken ErrorCode = 4012
|
||||||
|
ErrCodeAuthRequired ErrorCode = 4013
|
||||||
|
|
||||||
|
// 403 variants
|
||||||
|
ErrCodeInactiveAccount ErrorCode = 4031
|
||||||
|
ErrCodeNotPermitted ErrorCode = 4032
|
||||||
|
|
||||||
|
// 409 variants
|
||||||
|
ErrCodeEditConflict ErrorCode = 4091
|
||||||
|
ErrCodeAlreadyVoted ErrorCode = 4092
|
||||||
|
ErrCodeAlreadyBlindSigned ErrorCode = 4093
|
||||||
|
ErrCodeVoteAlreadyCast ErrorCode = 4094
|
||||||
|
|
||||||
|
// 422 variants
|
||||||
|
ErrCodeValidationFailed ErrorCode = 4221
|
||||||
|
ErrCodeBlindedVoteRange ErrorCode = 4222
|
||||||
|
ErrCodeInvalidSignature ErrorCode = 4223
|
||||||
|
ErrCodeHasNotStarted ErrorCode = 4224
|
||||||
|
)
|
||||||
|
|
||||||
|
type Error struct {
|
||||||
|
HttpCode int `json:"http_code,omitempty"`
|
||||||
|
Code ErrorCode `json:"code,omitempty"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Details map[string]string `json:"details,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *Error) Error() string {
|
||||||
|
return e.Message
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *Error) WithDetails(details map[string]string) error {
|
||||||
|
e.Details = details
|
||||||
|
return e
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
func (e *Error) DetailMessages() []string {
|
||||||
|
msgs := make([]string, 0, len(e.Details))
|
||||||
|
for key, msg := range e.Details {
|
||||||
|
msgs = append(msgs, fmt.Sprintf("%s: %s", key, msg))
|
||||||
|
}
|
||||||
|
return msgs
|
||||||
|
}
|
||||||
|
|
||||||
|
func New(httpCode int, code ErrorCode, text string) error {
|
||||||
|
return &Error{HttpCode: httpCode, Code: code, Message: text}
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
// 400 Bad Request
|
||||||
|
ErrFailedPEM = New(400, 0, "failed to decode PEM block")
|
||||||
|
ErrBadlyFormedJSON = New(400, 0, "body contains badly-formed JSON")
|
||||||
|
ErrBodyEmpty = New(400, 0, "body must not be empty")
|
||||||
|
ErrSingleValue = New(400, 0, "body must only contain a single JSON value")
|
||||||
|
ErrInvalidID = New(400, 0, "invalid id parameter")
|
||||||
|
ErrBadRequest = New(400, 0, "the server cannot process the request due to a client error")
|
||||||
|
|
||||||
|
// 401 Unauthorized
|
||||||
|
ErrInvalidCredentials = New(401, 4011, "invalid credentials")
|
||||||
|
ErrInvalidAuthToken = New(401, 4012, "invalid or missing authentication token")
|
||||||
|
ErrNoToken = New(401, 4012, "token must be provided")
|
||||||
|
ErrAuthRequired = New(401, 4013, "you must be authenticated to access this resource")
|
||||||
|
ErrHasNotStarted = New(401, 4224, "the vote has not yet started")
|
||||||
|
|
||||||
|
// 403 Forbidden
|
||||||
|
ErrInactiveAccount = New(403, 4031, "your user account must be activated to access this resource")
|
||||||
|
ErrNotPermitted = New(403, 4032, "your user account doesn't have the necessary permissions to access this resource")
|
||||||
|
|
||||||
|
// 404 Not Found
|
||||||
|
ErrRecordNotFound = New(404, 0, "record not found")
|
||||||
|
ErrNoPath = New(404, 0, "path is required")
|
||||||
|
|
||||||
|
// 409 Conflict
|
||||||
|
ErrEditConflict = New(409, 4091, "edit conflict")
|
||||||
|
ErrDuplicateVote = New(409, 4092, "this signature has already been used to cast a vote")
|
||||||
|
ErrDuplicateBlindSign = New(409, 4093, "user has already requested a blind signature for this issue")
|
||||||
|
ErrDuplicateSignature = New(409, 4094, "this signature has already been used to cast a vote")
|
||||||
|
ErrDuplicateEmail = New(409, 0, "duplicate email")
|
||||||
|
ErrDuplicateUser = New(409, 0, "duplicate username")
|
||||||
|
|
||||||
|
// 422 Unprocessable Entity
|
||||||
|
ErrValidationFailed = New(422, 4221, "validation failed")
|
||||||
|
ErrInvalidBlindedVote = New(422, 4222, "blinded_vote is out of valid range [1, n-1]")
|
||||||
|
ErrInvalidSignature = New(422, 4223, "signature verification failed")
|
||||||
|
|
||||||
|
// 429 Too Many Requests
|
||||||
|
ErrRateLimitExceeded = New(429, 0, "rate limit exceeded")
|
||||||
|
)
|
||||||
@ -3,14 +3,13 @@ package data
|
|||||||
import(
|
import(
|
||||||
"crypto/rsa"
|
"crypto/rsa"
|
||||||
"encoding/pem"
|
"encoding/pem"
|
||||||
"errors"
|
|
||||||
"crypto/x509"
|
"crypto/x509"
|
||||||
)
|
)
|
||||||
|
|
||||||
func parsePrivateKey(pemBytes []byte) (*rsa.PrivateKey, error) {
|
func parsePrivateKey(pemBytes []byte) (*rsa.PrivateKey, error) {
|
||||||
block, _ := pem.Decode(pemBytes)
|
block, _ := pem.Decode(pemBytes)
|
||||||
if block == nil {
|
if block == nil {
|
||||||
return nil, errors.New("failed to decode PEM block")
|
return nil, ErrFailedPEM
|
||||||
}
|
}
|
||||||
return x509.ParsePKCS1PrivateKey(block.Bytes)
|
return x509.ParsePKCS1PrivateKey(block.Bytes)
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,12 +2,6 @@ package data
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"errors"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrRecordNotFound = errors.New("record not found")
|
|
||||||
ErrEditConflict = errors.New("edit conflict")
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type Models struct {
|
type Models struct {
|
||||||
|
|||||||
@ -12,12 +12,6 @@ import (
|
|||||||
"party.at/party/internal/validator"
|
"party.at/party/internal/validator"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
|
||||||
ErrDuplicateEmail = errors.New("duplicate email")
|
|
||||||
ErrDuplicateUser = errors.New("duplicate username")
|
|
||||||
ErrInvalidCredentials = errors.New("invalid credentials")
|
|
||||||
)
|
|
||||||
|
|
||||||
var AnonymousUser = &User{}
|
var AnonymousUser = &User{}
|
||||||
|
|
||||||
type User struct {
|
type User struct {
|
||||||
|
|||||||
@ -1,7 +1,6 @@
|
|||||||
package data
|
package data
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"time"
|
"time"
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"math/big"
|
"math/big"
|
||||||
@ -12,14 +11,6 @@ import (
|
|||||||
"github.com/lib/pq"
|
"github.com/lib/pq"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
|
||||||
ErrDuplicateBlindSign = errors.New("user has already requested a blind signature for this issue")
|
|
||||||
ErrInvalidBlindedVote = errors.New("blinded_vote is out of valid range [1, n-1]")
|
|
||||||
ErrInvalidSignature = errors.New("signature verification failed")
|
|
||||||
ErrDuplicateVote = errors.New("this signature has already been used to cast a vote")
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
type Vote struct {
|
type Vote struct {
|
||||||
ID int64 `json:"id"`
|
ID int64 `json:"id"`
|
||||||
OptionID int64 `json:"option_id"`
|
OptionID int64 `json:"option_id"`
|
||||||
|
|||||||
@ -27,10 +27,6 @@
|
|||||||
</nav>
|
</nav>
|
||||||
{{end}}
|
{{end}}
|
||||||
|
|
||||||
<div style="text-align:center; padding: 40px 0 24px;">
|
|
||||||
<img src="/static/logo.svg" alt="DPÖ Logo" style="height: 600px; width: auto;">
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<main class="site-main">
|
<main class="site-main">
|
||||||
<div class="container">
|
<div class="container">
|
||||||
{{template "body" .}}
|
{{template "body" .}}
|
||||||
|
|||||||
@ -3,6 +3,10 @@
|
|||||||
{{define "title"}}Übersicht{{end}}
|
{{define "title"}}Übersicht{{end}}
|
||||||
|
|
||||||
{{define "body"}}
|
{{define "body"}}
|
||||||
|
<div style="text-align:center; padding: 40px 0 24px;">
|
||||||
|
<img src="/static/logo.svg" alt="DPÖ Logo" style="height: 600px; width: auto;">
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="page-header">
|
<div class="page-header">
|
||||||
<h1 class="page-title">Willkommen, {{.Name}}</h1>
|
<h1 class="page-title">Willkommen, {{.Name}}</h1>
|
||||||
<p class="page-subtitle">Ihre demokratische Plattform für Österreich</p>
|
<p class="page-subtitle">Ihre demokratische Plattform für Österreich</p>
|
||||||
|
|||||||
@ -3,6 +3,10 @@
|
|||||||
{{define "title"}}Willkommen{{end}}
|
{{define "title"}}Willkommen{{end}}
|
||||||
|
|
||||||
{{define "body"}}
|
{{define "body"}}
|
||||||
|
<div style="text-align:center; padding: 40px 0 24px;">
|
||||||
|
<img src="/static/logo.svg" alt="DPÖ Logo" style="height: 600px; width: auto;">
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="hero">
|
<div class="hero">
|
||||||
<div class="hero-content">
|
<div class="hero-content">
|
||||||
<div class="hero-label">Digitale Demokratie für Österreich</div>
|
<div class="hero-label">Digitale Demokratie für Österreich</div>
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user