2021-02-02 20:34:10 +00:00
|
|
|
package graph
|
|
|
|
|
|
|
|
// This file will be automatically regenerated based on the schema, any resolver implementations
|
|
|
|
// will be copied through when generating and any unknown code will be moved to the end.
|
|
|
|
|
|
|
|
import (
|
2021-02-06 03:01:51 +00:00
|
|
|
"bytes"
|
2021-02-02 20:34:10 +00:00
|
|
|
"context"
|
2021-02-06 03:01:51 +00:00
|
|
|
"errors"
|
2021-02-02 20:34:10 +00:00
|
|
|
"fmt"
|
2021-02-06 03:01:51 +00:00
|
|
|
"io"
|
2021-02-04 10:16:13 +00:00
|
|
|
"net/http"
|
2021-02-06 03:01:51 +00:00
|
|
|
"os"
|
|
|
|
"path"
|
2021-02-04 10:16:13 +00:00
|
|
|
"strings"
|
2021-02-06 03:01:51 +00:00
|
|
|
"time"
|
2021-02-02 20:34:10 +00:00
|
|
|
|
2021-02-06 03:01:51 +00:00
|
|
|
"github.com/gabriel-vasile/mimetype"
|
2021-02-04 10:16:13 +00:00
|
|
|
"github.com/google/uuid"
|
2021-02-02 20:34:10 +00:00
|
|
|
"reichard.io/imagini/graph/generated"
|
|
|
|
"reichard.io/imagini/graph/model"
|
|
|
|
)
|
|
|
|
|
2021-02-06 03:01:51 +00:00
|
|
|
// Done
|
2021-02-02 20:34:10 +00:00
|
|
|
func (r *mutationResolver) CreateMediaItem(ctx context.Context, input model.NewMediaItem) (*model.MediaItem, error) {
|
2021-02-06 03:01:51 +00:00
|
|
|
// Get Context
|
|
|
|
authContext := ctx.Value("auth").(*model.AuthContext)
|
|
|
|
accessToken := *authContext.AccessToken
|
|
|
|
userID, ok := accessToken.Get("sub")
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
2021-02-02 20:34:10 +00:00
|
|
|
|
2021-02-06 03:01:51 +00:00
|
|
|
// File header placeholder
|
|
|
|
fileHeader := make([]byte, 64)
|
|
|
|
|
|
|
|
// Copy headers into the buffer
|
|
|
|
if _, err := input.File.File.Read(fileHeader); err != nil {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Determine media type
|
|
|
|
fileMime := mimetype.Detect(fileHeader)
|
|
|
|
contentType := fileMime.String()
|
|
|
|
var isVideo bool
|
|
|
|
if strings.HasPrefix(contentType, "image/") {
|
|
|
|
isVideo = false
|
|
|
|
} else if strings.HasPrefix(contentType, "video/") {
|
|
|
|
isVideo = true
|
|
|
|
} else {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Derive Folder & File Path
|
|
|
|
mediaItemID := uuid.New().String()
|
|
|
|
fileName := mediaItemID + fileMime.Extension()
|
|
|
|
folderPath := path.Join("/" + r.Config.DataPath + "/media/" + userID.(string))
|
|
|
|
os.MkdirAll(folderPath, 0700)
|
|
|
|
filePath := path.Join(folderPath + "/" + fileName)
|
|
|
|
|
|
|
|
// Create File
|
|
|
|
f, err := os.OpenFile(filePath, os.O_WRONLY|os.O_CREATE, 0600)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
// Copy header to file
|
|
|
|
_, err = io.Copy(f, bytes.NewReader(fileHeader))
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy remaining file
|
|
|
|
_, err = io.Copy(f, input.File.File)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create MediaItem From EXIF Data
|
|
|
|
mediaItem, err := mediaItemFromEXIFData(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add Additional MediaItem Fields
|
|
|
|
mediaItem.ID = mediaItemID
|
|
|
|
mediaItem.UserID = userID.(string)
|
|
|
|
mediaItem.IsVideo = isVideo
|
|
|
|
mediaItem.FileName = fileName
|
|
|
|
mediaItem.OrigName = input.File.Filename
|
|
|
|
|
|
|
|
// Create MediaItem in DB
|
|
|
|
err = r.DB.CreateMediaItem(mediaItem)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Upload Failed")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Success
|
|
|
|
return mediaItem, nil
|
2021-02-02 20:34:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mutationResolver) CreateAlbum(ctx context.Context, input model.NewAlbum) (*model.Album, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mutationResolver) CreateTag(ctx context.Context, input model.NewTag) (*model.Tag, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *mutationResolver) CreateUser(ctx context.Context, input model.NewUser) (*model.User, error) {
|
2021-02-03 03:55:35 +00:00
|
|
|
user := &model.User{
|
|
|
|
Email: input.Email,
|
|
|
|
Username: input.Username,
|
|
|
|
FirstName: input.FirstName,
|
|
|
|
LastName: input.LastName,
|
|
|
|
Role: input.Role,
|
|
|
|
AuthType: input.AuthType,
|
|
|
|
Password: input.Password,
|
|
|
|
}
|
|
|
|
|
|
|
|
err := r.DB.CreateUser(user)
|
|
|
|
if err != nil {
|
2021-02-04 10:16:13 +00:00
|
|
|
return nil, err
|
2021-02-03 03:55:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return user, nil
|
|
|
|
}
|
|
|
|
|
2021-02-04 10:16:13 +00:00
|
|
|
func (r *queryResolver) Login(ctx context.Context, user string, password string, deviceID *string) (*model.AuthResponse, error) {
|
2021-02-06 03:01:51 +00:00
|
|
|
// Get Context
|
2021-02-03 03:55:35 +00:00
|
|
|
authContext := ctx.Value("auth").(*model.AuthContext)
|
2021-02-04 10:16:13 +00:00
|
|
|
resp := authContext.AuthResponse
|
|
|
|
req := authContext.AuthRequest
|
|
|
|
|
2021-02-06 03:01:51 +00:00
|
|
|
// Clear All Cookies By Default
|
|
|
|
accessCookie := http.Cookie{Name: "AccessToken", Path: "/", HttpOnly: true, MaxAge: -1, Expires: time.Now().Add(-100 * time.Hour)}
|
|
|
|
refreshCookie := http.Cookie{Name: "RefreshToken", Path: "/", HttpOnly: true, MaxAge: -1, Expires: time.Now().Add(-100 * time.Hour)}
|
|
|
|
http.SetCookie(*resp, &accessCookie)
|
|
|
|
http.SetCookie(*resp, &refreshCookie)
|
|
|
|
|
2021-02-04 10:16:13 +00:00
|
|
|
// Do Login
|
|
|
|
foundUser, success := r.Auth.AuthenticateUser(user, password)
|
|
|
|
if !success {
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultFailure}, nil
|
|
|
|
}
|
2021-02-03 03:55:35 +00:00
|
|
|
|
2021-02-04 10:16:13 +00:00
|
|
|
// Upsert Device
|
2021-02-06 03:01:51 +00:00
|
|
|
foundDevice := model.Device{UserID: foundUser.ID}
|
2021-02-04 10:16:13 +00:00
|
|
|
if deviceID != nil {
|
|
|
|
parsedDeviceID, err := uuid.Parse(*deviceID)
|
|
|
|
if err != nil {
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultFailure}, nil
|
|
|
|
}
|
|
|
|
foundDevice.ID = parsedDeviceID.String()
|
|
|
|
count, err := r.DB.Device(&foundDevice)
|
|
|
|
if count != 1 || err != nil {
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultFailure}, nil
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
foundDevice.Type = deriveDeviceType(req)
|
|
|
|
err := r.DB.CreateDevice(&foundDevice)
|
|
|
|
if err != nil {
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultFailure}, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create Tokens
|
|
|
|
accessToken, err := r.Auth.CreateJWTAccessToken(foundUser, foundDevice)
|
|
|
|
if err != nil {
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultFailure}, nil
|
|
|
|
}
|
|
|
|
refreshToken, err := r.Auth.CreateJWTRefreshToken(foundUser, foundDevice)
|
|
|
|
if err != nil {
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultFailure}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set appropriate cookies
|
2021-02-06 03:01:51 +00:00
|
|
|
accessCookie = http.Cookie{Name: "AccessToken", Value: accessToken, Path: "/", HttpOnly: true}
|
|
|
|
refreshCookie = http.Cookie{Name: "RefreshToken", Value: refreshToken, Path: "/", HttpOnly: true}
|
2021-02-04 10:16:13 +00:00
|
|
|
http.SetCookie(*resp, &accessCookie)
|
|
|
|
http.SetCookie(*resp, &refreshCookie)
|
|
|
|
|
|
|
|
return &model.AuthResponse{Result: model.AuthResultSuccess, Device: &foundDevice}, nil
|
2021-02-03 03:55:35 +00:00
|
|
|
}
|
|
|
|
|
2021-02-04 10:16:13 +00:00
|
|
|
func (r *queryResolver) Logout(ctx context.Context) (*model.AuthResponse, error) {
|
2021-02-06 03:01:51 +00:00
|
|
|
// Set Cookie From Context
|
|
|
|
authContext := ctx.Value("auth").(*model.AuthContext)
|
|
|
|
resp := authContext.AuthResponse
|
|
|
|
|
|
|
|
// Clear All Cookies
|
|
|
|
accessCookie := http.Cookie{Name: "AccessToken", Path: "/", HttpOnly: true, MaxAge: -1, Expires: time.Now().Add(-100 * time.Hour)}
|
|
|
|
refreshCookie := http.Cookie{Name: "RefreshToken", Path: "/", HttpOnly: true, MaxAge: -1, Expires: time.Now().Add(-100 * time.Hour)}
|
|
|
|
http.SetCookie(*resp, &accessCookie)
|
|
|
|
http.SetCookie(*resp, &refreshCookie)
|
|
|
|
|
2021-02-04 10:16:13 +00:00
|
|
|
return &model.AuthResponse{Result: model.AuthResultSuccess}, nil
|
2021-02-02 20:34:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) MediaItem(ctx context.Context, id string) (*model.MediaItem, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) Device(ctx context.Context, id string) (*model.Device, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) Album(ctx context.Context, id string) (*model.Album, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
2021-02-03 03:55:35 +00:00
|
|
|
func (r *queryResolver) User(ctx context.Context, id string) (*model.User, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
2021-02-02 20:34:10 +00:00
|
|
|
func (r *queryResolver) Tag(ctx context.Context, id string) (*model.Tag, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
2021-02-03 03:55:35 +00:00
|
|
|
func (r *queryResolver) Me(ctx context.Context) (*model.User, error) {
|
2021-02-02 20:34:10 +00:00
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) MediaItems(ctx context.Context, filter *model.MediaItemFilter, count *int, page *int) (*model.MediaItemResponse, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) Devices(ctx context.Context, filter *model.DeviceFilter, count *int, page *int) (*model.DeviceResponse, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) Albums(ctx context.Context, filter *model.AlbumFilter, count *int, page *int) (*model.AlbumResponse, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) Tags(ctx context.Context, filter *model.TagFilter, count *int, page *int) (*model.TagResponse, error) {
|
|
|
|
panic(fmt.Errorf("not implemented"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *queryResolver) Users(ctx context.Context, filter *model.UserFilter, count *int, page *int) (*model.UserResponse, error) {
|
2021-02-03 03:55:35 +00:00
|
|
|
resp, totalCount, err := r.DB.Users()
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Errorf("DB Error"))
|
|
|
|
}
|
|
|
|
return &model.UserResponse{
|
|
|
|
Data: resp,
|
|
|
|
PageInfo: &model.PageInfo{
|
|
|
|
Count: int(totalCount),
|
|
|
|
Page: 0,
|
|
|
|
Total: int(totalCount),
|
|
|
|
},
|
|
|
|
}, nil
|
2021-02-02 20:34:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Mutation returns generated.MutationResolver implementation.
|
|
|
|
func (r *Resolver) Mutation() generated.MutationResolver { return &mutationResolver{r} }
|
|
|
|
|
|
|
|
// Query returns generated.QueryResolver implementation.
|
|
|
|
func (r *Resolver) Query() generated.QueryResolver { return &queryResolver{r} }
|
|
|
|
|
|
|
|
type mutationResolver struct{ *Resolver }
|
|
|
|
type queryResolver struct{ *Resolver }
|