2019-06-22 17:48:44 -08:00
|
|
|
package user
|
|
|
|
|
|
|
|
import (
|
2019-06-25 22:31:54 -08:00
|
|
|
"context"
|
2019-06-22 17:48:44 -08:00
|
|
|
"database/sql"
|
2019-08-04 14:48:43 -08:00
|
|
|
"encoding/json"
|
2019-07-13 12:16:28 -08:00
|
|
|
"geeks-accelerator/oss/saas-starter-kit/internal/platform/web"
|
2019-06-22 17:48:44 -08:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/lib/pq"
|
|
|
|
)
|
|
|
|
|
|
|
|
// User represents someone with access to our system.
|
|
|
|
type User struct {
|
2019-06-25 22:31:54 -08:00
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
2019-07-31 13:47:30 -08:00
|
|
|
FirstName string `json:"first_name" validate:"required" example:"Gabi"`
|
|
|
|
LastName string `json:"last_name" validate:"required" example:"May"`
|
2019-06-25 22:31:54 -08:00
|
|
|
Email string `json:"email" validate:"required,email,unique" example:"gabi@geeksinthewoods.com"`
|
|
|
|
PasswordSalt string `json:"-" validate:"required"`
|
|
|
|
PasswordHash []byte `json:"-" validate:"required"`
|
2019-06-25 02:40:29 -08:00
|
|
|
PasswordReset *sql.NullString `json:"-"`
|
2019-06-25 22:31:54 -08:00
|
|
|
Timezone string `json:"timezone" validate:"omitempty" example:"America/Anchorage"`
|
|
|
|
CreatedAt time.Time `json:"created_at"`
|
|
|
|
UpdatedAt time.Time `json:"updated_at"`
|
|
|
|
ArchivedAt *pq.NullTime `json:"archived_at,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UserResponse represents someone with access to our system that is returned for display.
|
|
|
|
type UserResponse struct {
|
2019-08-04 14:48:43 -08:00
|
|
|
ID string `json:"id" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
Name string `json:"name" example:"Gabi"`
|
|
|
|
FirstName string `json:"first_name" example:"Gabi"`
|
|
|
|
LastName string `json:"last_name" example:"May"`
|
|
|
|
Email string `json:"email" example:"gabi@geeksinthewoods.com"`
|
|
|
|
Timezone string `json:"timezone" example:"America/Anchorage"`
|
|
|
|
CreatedAt web.TimeResponse `json:"created_at"` // CreatedAt contains multiple format options for display.
|
|
|
|
UpdatedAt web.TimeResponse `json:"updated_at"` // UpdatedAt contains multiple format options for display.
|
|
|
|
ArchivedAt *web.TimeResponse `json:"archived_at,omitempty"` // ArchivedAt contains multiple format options for display.
|
2019-08-03 16:35:57 -08:00
|
|
|
Gravatar web.GravatarResponse `json:"gravatar"`
|
2019-06-25 22:31:54 -08:00
|
|
|
}
|
2019-06-22 17:48:44 -08:00
|
|
|
|
2019-06-25 22:31:54 -08:00
|
|
|
// Response transforms User and UserResponse that is used for display.
|
|
|
|
// Additional filtering by context values or translations could be applied.
|
|
|
|
func (m *User) Response(ctx context.Context) *UserResponse {
|
|
|
|
if m == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2019-06-22 17:48:44 -08:00
|
|
|
|
2019-06-25 22:31:54 -08:00
|
|
|
r := &UserResponse{
|
|
|
|
ID: m.ID,
|
2019-08-04 14:48:43 -08:00
|
|
|
Name: m.FirstName + " " + m.LastName,
|
2019-07-31 13:47:30 -08:00
|
|
|
FirstName: m.FirstName,
|
|
|
|
LastName: m.LastName,
|
2019-06-25 22:31:54 -08:00
|
|
|
Email: m.Email,
|
|
|
|
Timezone: m.Timezone,
|
|
|
|
CreatedAt: web.NewTimeResponse(ctx, m.CreatedAt),
|
|
|
|
UpdatedAt: web.NewTimeResponse(ctx, m.UpdatedAt),
|
2019-08-04 14:48:43 -08:00
|
|
|
Gravatar: web.NewGravatarResponse(ctx, m.Email),
|
2019-06-25 22:31:54 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if m.ArchivedAt != nil && !m.ArchivedAt.Time.IsZero() {
|
|
|
|
at := web.NewTimeResponse(ctx, m.ArchivedAt.Time)
|
|
|
|
r.ArchivedAt = &at
|
|
|
|
}
|
|
|
|
|
|
|
|
return r
|
2019-06-22 17:48:44 -08:00
|
|
|
}
|
|
|
|
|
2019-08-04 14:48:43 -08:00
|
|
|
func (m *UserResponse) UnmarshalBinary(data []byte) error {
|
|
|
|
if data == nil || len(data) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// convert data to yours, let's assume its json data
|
|
|
|
return json.Unmarshal(data, m)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *UserResponse) MarshalBinary() ([]byte, error) {
|
|
|
|
return json.Marshal(m)
|
|
|
|
}
|
|
|
|
|
2019-06-24 17:36:42 -08:00
|
|
|
// UserCreateRequest contains information needed to create a new User.
|
|
|
|
type UserCreateRequest struct {
|
2019-07-31 13:47:30 -08:00
|
|
|
FirstName string `json:"first_name" validate:"required" example:"Gabi"`
|
|
|
|
LastName string `json:"last_name" validate:"required" example:"May"`
|
2019-06-25 22:31:54 -08:00
|
|
|
Email string `json:"email" validate:"required,email,unique" example:"gabi@geeksinthewoods.com"`
|
2019-06-25 02:40:29 -08:00
|
|
|
Password string `json:"password" validate:"required" example:"SecretString"`
|
2019-07-31 13:47:30 -08:00
|
|
|
PasswordConfirm string `json:"password_confirm" validate:"required,eqfield=Password" example:"SecretString"`
|
2019-06-25 02:40:29 -08:00
|
|
|
Timezone *string `json:"timezone,omitempty" validate:"omitempty" example:"America/Anchorage"`
|
2019-06-22 17:48:44 -08:00
|
|
|
}
|
|
|
|
|
2019-08-02 15:03:32 -08:00
|
|
|
// UserCreateInviteRequest contains information needed to create a new User.
|
|
|
|
type UserCreateInviteRequest struct {
|
|
|
|
Email string `json:"email" validate:"required,email,unique" example:"gabi@geeksinthewoods.com"`
|
|
|
|
}
|
|
|
|
|
2019-08-04 14:48:43 -08:00
|
|
|
// UserReadRequest defines the information needed to read an user.
|
|
|
|
type UserReadRequest struct {
|
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
IncludeArchived bool `json:"include-archived" example:"false"`
|
|
|
|
}
|
|
|
|
|
2019-06-24 17:36:42 -08:00
|
|
|
// UserUpdateRequest defines what information may be provided to modify an existing
|
2019-06-22 17:48:44 -08:00
|
|
|
// User. All fields are optional so clients can send just the fields they want
|
|
|
|
// changed. It uses pointer fields so we can differentiate between a field that
|
|
|
|
// was not provided and a field that was provided as explicitly blank. Normally
|
|
|
|
// we do not want to use pointers to basic types but we make exceptions around
|
|
|
|
// marshalling/unmarshalling.
|
2019-06-24 17:36:42 -08:00
|
|
|
type UserUpdateRequest struct {
|
2019-07-31 13:47:30 -08:00
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
FirstName *string `json:"first_name,omitempty" validate:"omitempty" example:"Gabi May Not"`
|
|
|
|
LastName *string `json:"last_name,omitempty" validate:"omitempty" example:"Gabi May Not"`
|
|
|
|
Email *string `json:"email,omitempty" validate:"omitempty,email,unique" example:"gabi.may@geeksinthewoods.com"`
|
|
|
|
Timezone *string `json:"timezone,omitempty" validate:"omitempty" example:"America/Anchorage"`
|
2019-06-22 17:48:44 -08:00
|
|
|
}
|
|
|
|
|
2019-06-24 17:36:42 -08:00
|
|
|
// UserUpdatePasswordRequest defines what information is required to update a user password.
|
|
|
|
type UserUpdatePasswordRequest struct {
|
2019-06-25 22:31:54 -08:00
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
Password string `json:"password" validate:"required" example:"NeverTellSecret"`
|
2019-07-31 13:47:30 -08:00
|
|
|
PasswordConfirm string `json:"password_confirm" validate:"required,eqfield=Password" example:"NeverTellSecret"`
|
2019-06-25 22:31:54 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// UserArchiveRequest defines the information needed to archive an user. This will archive (soft-delete) the
|
|
|
|
// existing database entry.
|
|
|
|
type UserArchiveRequest struct {
|
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
2019-06-22 17:48:44 -08:00
|
|
|
}
|
|
|
|
|
2019-08-04 14:48:43 -08:00
|
|
|
// UserRestoreRequest defines the information needed to restore an user.
|
|
|
|
type UserRestoreRequest struct {
|
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UserDeleteRequest defines the information needed to delete a user.
|
|
|
|
type UserDeleteRequest struct {
|
2019-08-02 15:03:32 -08:00
|
|
|
ID string `json:"id" validate:"required,uuid" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
}
|
|
|
|
|
2019-06-22 17:48:44 -08:00
|
|
|
// UserFindRequest defines the possible options to search for users. By default
|
|
|
|
// archived users will be excluded from response.
|
|
|
|
type UserFindRequest struct {
|
2019-08-04 14:48:43 -08:00
|
|
|
Where *string `json:"where" example:"name = ? and email = ?"`
|
|
|
|
Args []interface{} `json:"args" swaggertype:"array,string" example:"Company Name,gabi.may@geeksinthewoods.com"`
|
|
|
|
Order []string `json:"order" example:"created_at desc"`
|
|
|
|
Limit *uint `json:"limit" example:"10"`
|
|
|
|
Offset *uint `json:"offset" example:"20"`
|
|
|
|
IncludeArchived bool `json:"include-archived" example:"false"`
|
2019-06-22 17:48:44 -08:00
|
|
|
}
|
|
|
|
|
2019-08-02 15:03:32 -08:00
|
|
|
// UserResetPasswordRequest defines the fields need to reset a user password.
|
|
|
|
type UserResetPasswordRequest struct {
|
|
|
|
Email string `json:"email" validate:"required,email" example:"gabi.may@geeksinthewoods.com"`
|
|
|
|
TTL time.Duration `json:"ttl,omitempty" `
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResetHash
|
|
|
|
type ResetHash struct {
|
|
|
|
ResetID string `json:"reset_id" validate:"required" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
CreatedAt int `json:"created_at" validate:"required"`
|
|
|
|
ExpiresAt int `json:"expires_at" validate:"required"`
|
|
|
|
RequestIP string `json:"request_ip" validate:"required,ip" example:"69.56.104.36"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UserResetConfirmRequest defines the fields need to reset a user password.
|
|
|
|
type UserResetConfirmRequest struct {
|
|
|
|
ResetHash string `json:"reset_hash" validate:"required" example:"d69bdef7-173f-4d29-b52c-3edc60baf6a2"`
|
|
|
|
Password string `json:"password" validate:"required" example:"SecretString"`
|
|
|
|
PasswordConfirm string `json:"password_confirm" validate:"required,eqfield=Password" example:"SecretString"`
|
|
|
|
}
|