12 changed files with 493 additions and 4 deletions
			
			
		- 
					12database/mongodb/db.go
- 
					208database/mongodb/stories.go
- 
					5internal/generate/id.go
- 
					15models/chapter.go
- 
					17models/comment.go
- 
					23models/story.go
- 
					14repositories/chapter.go
- 
					14repositories/comment.go
- 
					6repositories/repository.go
- 
					16repositories/story.go
- 
					15services/changes.go
- 
					152services/stories.go
| @ -0,0 +1,208 @@ | |||
| package mongodb | |||
| 
 | |||
| import ( | |||
| 	"context" | |||
| 	"git.aiterp.net/rpdata/api/internal/generate" | |||
| 	"git.aiterp.net/rpdata/api/models" | |||
| 	"git.aiterp.net/rpdata/api/repositories" | |||
| 	"github.com/globalsign/mgo" | |||
| 	"github.com/globalsign/mgo/bson" | |||
| 	"log" | |||
| ) | |||
| 
 | |||
| type storyRepository struct { | |||
| 	stories  *mgo.Collection | |||
| 	chapters *mgo.Collection | |||
| 	comments *mgo.Collection | |||
| } | |||
| 
 | |||
| func newStoryRepository(db *mgo.Database) (repositories.StoryRepository, error) { | |||
| 	collection := db.C("story.stories") | |||
| 
 | |||
| 	err := collection.EnsureIndexKey("tags") | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 	err = collection.EnsureIndexKey("author") | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 	err = collection.EnsureIndexKey("updatedDate") | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 	err = collection.EnsureIndexKey("fictionalDate") | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 	err = collection.EnsureIndexKey("listed") | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	return &storyRepository{ | |||
| 		stories:  collection, | |||
| 		chapters: db.C("story.chapters"), | |||
| 		comments: db.C("story.comments"), | |||
| 	}, nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) Find(ctx context.Context, id string) (*models.Story, error) { | |||
| 	story := new(models.Story) | |||
| 	err := r.stories.FindId(id).One(story) | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	return story, nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) List(ctx context.Context, filter models.StoryFilter) ([]*models.Story, error) { | |||
| 	query := bson.M{} | |||
| 	if filter.Author != nil { | |||
| 		query["author"] = *filter.Author | |||
| 	} | |||
| 	if filter.Category != nil { | |||
| 		query["category"] = *filter.Category | |||
| 	} | |||
| 	if filter.Open != nil { | |||
| 		query["open"] = *filter.Open | |||
| 	} | |||
| 	if len(filter.Tags) > 0 { | |||
| 		query["tags"] = bson.M{"$all": filter.Tags} | |||
| 	} | |||
| 	if filter.Unlisted != nil { | |||
| 		query["listed"] = *filter.Unlisted | |||
| 	} | |||
| 	if !filter.EarliestFictionalDate.IsZero() && !filter.LatestFictionalDate.IsZero() { | |||
| 		query["fictionalDate"] = bson.M{ | |||
| 			"$gte": filter.EarliestFictionalDate, | |||
| 			"$lte": filter.LatestFictionalDate, | |||
| 		} | |||
| 	} else if !filter.EarliestFictionalDate.IsZero() { | |||
| 		query["fictionalDate"] = bson.M{ | |||
| 			"$gte": filter.EarliestFictionalDate, | |||
| 		} | |||
| 	} else if !filter.LatestFictionalDate.IsZero() { | |||
| 		query["fictionalDate"] = bson.M{ | |||
| 			"$lte": filter.LatestFictionalDate, | |||
| 		} | |||
| 	} | |||
| 
 | |||
| 	stories := make([]*models.Story, 0, 32) | |||
| 	err := r.stories.Find(query).Sort("-updatedDate	").Limit(filter.Limit).All(&stories) | |||
| 	if err != nil { | |||
| 		if err == mgo.ErrNotFound { | |||
| 			return stories, nil | |||
| 		} | |||
| 
 | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	return stories, nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) Insert(ctx context.Context, story models.Story) (*models.Story, error) { | |||
| 	story.ID = generate.StoryID() | |||
| 
 | |||
| 	err := r.stories.Insert(story) | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	return &story, nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) Update(ctx context.Context, story models.Story, update models.StoryUpdate) (*models.Story, error) { | |||
| 	updateBson := bson.M{} | |||
| 	if update.Name != nil { | |||
| 		updateBson["name"] = *update.Name | |||
| 		story.Name = *update.Name | |||
| 	} | |||
| 	if update.Open != nil { | |||
| 		updateBson["open"] = *update.Open | |||
| 		story.Open = *update.Open | |||
| 	} | |||
| 	if update.Category != nil { | |||
| 		updateBson["category"] = *update.Category | |||
| 		story.Category = *update.Category | |||
| 	} | |||
| 	if update.Author != nil { | |||
| 		updateBson["author"] = *update.Author | |||
| 		story.Author = *update.Author | |||
| 	} | |||
| 	if update.FictionalDate != nil { | |||
| 		updateBson["fictionalDate"] = *update.FictionalDate | |||
| 		story.FictionalDate = *update.FictionalDate | |||
| 	} | |||
| 	if update.UpdatedDate != nil { | |||
| 		updateBson["updatedDate"] = *update.UpdatedDate | |||
| 		story.UpdatedDate = *update.UpdatedDate | |||
| 	} | |||
| 	if update.Listed != nil { | |||
| 		updateBson["listed"] = *update.Listed | |||
| 		story.Listed = *update.Listed | |||
| 	} | |||
| 
 | |||
| 	err := r.stories.UpdateId(story.ID, bson.M{"$set": updateBson}) | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	return &story, nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) AddTag(ctx context.Context, story models.Story, tag models.Tag) error { | |||
| 	ci, err := r.stories.UpdateAll(bson.M{"_id": story.ID}, bson.M{"$addToSet": bson.M{"tags": tag}}) | |||
| 	if err != nil { | |||
| 		return err | |||
| 	} | |||
| 	if ci.Updated == 0 { | |||
| 		return repositories.ErrTagExists | |||
| 	} | |||
| 
 | |||
| 	return nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) RemoveTag(ctx context.Context, story models.Story, tag models.Tag) error { | |||
| 	ci, err := r.stories.UpdateAll(bson.M{"_id": story.ID}, bson.M{"$pull": bson.M{"tags": tag}}) | |||
| 	if err != nil { | |||
| 		return err | |||
| 	} | |||
| 	if ci.Updated == 0 { | |||
| 		return repositories.ErrTagDoesNotExist | |||
| 	} | |||
| 
 | |||
| 	return nil | |||
| } | |||
| 
 | |||
| func (r *storyRepository) Delete(ctx context.Context, story models.Story) error { | |||
| 	err := r.stories.RemoveId(story.ID) | |||
| 	if err != nil { | |||
| 		return err | |||
| 	} | |||
| 
 | |||
| 	chapterIds := make([]string, 0, 8) | |||
| 	err = r.chapters.Find(bson.M{"storyId": story.ID}).Distinct("_id", &chapterIds) | |||
| 	if err != nil { | |||
| 		log.Println("Failed to find chapterIds:", err) | |||
| 		return nil | |||
| 	} | |||
| 	if len(chapterIds) > 0 { | |||
| 		c1, err := r.chapters.RemoveAll(bson.M{"storyId": story.ID}) | |||
| 		if err != nil { | |||
| 			log.Println("Failed to remove chapters:", err) | |||
| 			return nil | |||
| 		} | |||
| 
 | |||
| 		c2, err := r.chapters.RemoveAll(bson.M{"comments": bson.M{"$in": chapterIds}}) | |||
| 		if err != nil { | |||
| 			log.Println("Failed to remove comments:", err) | |||
| 			return nil | |||
| 		} | |||
| 
 | |||
| 		log.Printf("Removed story %s (%d chapters, %d comments)", story.ID, c1.Removed, c2.Removed) | |||
| 	} | |||
| 
 | |||
| 	return nil | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| package repositories | |||
| 
 | |||
| import ( | |||
| 	"context" | |||
| 	"git.aiterp.net/rpdata/api/models" | |||
| ) | |||
| 
 | |||
| type ChapterRepository interface { | |||
| 	Find(ctx context.Context, id string) (*models.Chapter, error) | |||
| 	List(ctx context.Context, filter models.ChapterFilter) ([]*models.Chapter, error) | |||
| 	Insert(ctx context.Context, chapter models.Chapter) (*models.Chapter, error) | |||
| 	Update(ctx context.Context, chapter models.Chapter, update models.ChapterUpdate) (*models.Story, error) | |||
| 	Delete(ctx context.Context, chapter models.Chapter) error | |||
| } | |||
| @ -0,0 +1,14 @@ | |||
| package repositories | |||
| 
 | |||
| import ( | |||
| 	"context" | |||
| 	"git.aiterp.net/rpdata/api/models" | |||
| ) | |||
| 
 | |||
| type CommentRepository interface { | |||
| 	Find(ctx context.Context, id string) (*models.Comment, error) | |||
| 	List(ctx context.Context, filter models.CommentFilter) ([]*models.Comment, error) | |||
| 	Insert(ctx context.Context, comment models.Comment) (*models.Comment, error) | |||
| 	Update(ctx context.Context, comment models.Comment, update models.CommentUpdate) (*models.Story, error) | |||
| 	Delete(ctx context.Context, comment models.Comment) error | |||
| } | |||
| @ -0,0 +1,16 @@ | |||
| package repositories | |||
| 
 | |||
| import ( | |||
| 	"context" | |||
| 	"git.aiterp.net/rpdata/api/models" | |||
| ) | |||
| 
 | |||
| type StoryRepository interface { | |||
| 	Find(ctx context.Context, id string) (*models.Story, error) | |||
| 	List(ctx context.Context, filter models.StoryFilter) ([]*models.Story, error) | |||
| 	Insert(ctx context.Context, story models.Story) (*models.Story, error) | |||
| 	Update(ctx context.Context, story models.Story, update models.StoryUpdate) (*models.Story, error) | |||
| 	AddTag(ctx context.Context, story models.Story, tag models.Tag) error | |||
| 	RemoveTag(ctx context.Context, story models.Story, tag models.Tag) error | |||
| 	Delete(ctx context.Context, story models.Story) error | |||
| } | |||
| @ -0,0 +1,152 @@ | |||
| package services | |||
| 
 | |||
| import ( | |||
| 	"context" | |||
| 	"errors" | |||
| 	"git.aiterp.net/rpdata/api/internal/auth" | |||
| 	"git.aiterp.net/rpdata/api/internal/generate" | |||
| 	"git.aiterp.net/rpdata/api/models" | |||
| 	"git.aiterp.net/rpdata/api/models/changekeys" | |||
| 	"git.aiterp.net/rpdata/api/repositories" | |||
| 	"time" | |||
| ) | |||
| 
 | |||
| // StoryService is a service governing all operations on stories and child objects.
 | |||
| type StoryService struct { | |||
| 	stories       repositories.StoryRepository | |||
| 	chapters      repositories.ChapterRepository | |||
| 	comments      repositories.CommentRepository | |||
| 	changeService *ChangeService | |||
| } | |||
| 
 | |||
| func (s *StoryService) FindStory(ctx context.Context, id string) (*models.Story, error) { | |||
| 	return s.stories.Find(ctx, id) | |||
| } | |||
| 
 | |||
| func (s *StoryService) ListStories(ctx context.Context, filter models.StoryFilter) ([]*models.Story, error) { | |||
| 	return s.stories.List(ctx, filter) | |||
| } | |||
| 
 | |||
| func (s *StoryService) ListChapters(ctx context.Context, story models.Story) ([]*models.Chapter, error) { | |||
| 	return s.chapters.List(ctx, models.ChapterFilter{StoryID: &story.ID, Limit: 0}) | |||
| } | |||
| 
 | |||
| func (s *StoryService) ListComments(ctx context.Context, chapter models.Chapter, limit int) ([]*models.Comment, error) { | |||
| 	return s.comments.List(ctx, models.CommentFilter{ChapterID: &chapter.ID, Limit: limit}) | |||
| } | |||
| 
 | |||
| func (s *StoryService) CreateStory(ctx context.Context, name, author string, category models.StoryCategory, listed, open bool, tags []models.Tag, createdDate, fictionalDate time.Time) (*models.Story, error) { | |||
| 	story := &models.Story{ | |||
| 		Name:          name, | |||
| 		Author:        author, | |||
| 		Category:      category, | |||
| 		Listed:        listed, | |||
| 		Open:          open, | |||
| 		Tags:          tags, | |||
| 		CreatedDate:   createdDate, | |||
| 		FictionalDate: fictionalDate, | |||
| 		UpdatedDate:   createdDate, | |||
| 	} | |||
| 
 | |||
| 	if err := auth.CheckPermission(ctx, "add", story); err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	story, err := s.stories.Insert(ctx, *story) | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	s.changeService.Submit(ctx, "Story", "add", story.Listed, changekeys.Listed(story), story) | |||
| 
 | |||
| 	return story, nil | |||
| } | |||
| 
 | |||
| func (s *StoryService) CreateChapter(ctx context.Context, story models.Story, title, author, source string, createdDate time.Time, fictionalDate *time.Time, commentMode models.ChapterCommentMode) (*models.Chapter, error) { | |||
| 	chapter := &models.Chapter{ | |||
| 		ID:             generate.ChapterID(), | |||
| 		StoryID:        story.ID, | |||
| 		Title:          title, | |||
| 		Author:         author, | |||
| 		Source:         source, | |||
| 		CreatedDate:    createdDate, | |||
| 		EditedDate:     createdDate, | |||
| 		CommentMode:    commentMode, | |||
| 		CommentsLocked: false, | |||
| 	} | |||
| 	if fictionalDate != nil { | |||
| 		chapter.FictionalDate = *fictionalDate | |||
| 	} | |||
| 
 | |||
| 	if story.Open { | |||
| 		if !auth.TokenFromContext(ctx).Permitted("member", "chapter.add") { | |||
| 			return nil, auth.ErrUnauthorized | |||
| 		} | |||
| 	} else { | |||
| 		if err := auth.CheckPermission(ctx, "add", chapter); err != nil { | |||
| 			return nil, err | |||
| 		} | |||
| 	} | |||
| 
 | |||
| 	chapter, err := s.chapters.Insert(ctx, *chapter) | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	if createdDate.After(story.UpdatedDate) { | |||
| 		_, _ = s.stories.Update(ctx, story, models.StoryUpdate{UpdatedDate: &createdDate}) | |||
| 	} | |||
| 
 | |||
| 	s.changeService.Submit(ctx, "Chapter", "add", story.Listed, changekeys.Many(story, chapter), chapter) | |||
| 
 | |||
| 	return chapter, nil | |||
| } | |||
| 
 | |||
| // CreateComment adds a comment.
 | |||
| func (s *StoryService) CreateComment(ctx context.Context, chapter models.Chapter, subject, author, source, characterName string, character *models.Character, createdDate time.Time, fictionalDate time.Time) (*models.Comment, error) { | |||
| 	characterID := "" | |||
| 	if character != nil { | |||
| 		characterID = character.ID | |||
| 	} | |||
| 
 | |||
| 	if !chapter.CanComment() { | |||
| 		return nil, errors.New("comments are locked or disabled") | |||
| 	} | |||
| 
 | |||
| 	if author == "" { | |||
| 		if token := auth.TokenFromContext(ctx); token != nil { | |||
| 			author = token.UserID | |||
| 		} else { | |||
| 			return nil, auth.ErrUnauthenticated | |||
| 		} | |||
| 	} | |||
| 
 | |||
| 	comment := &models.Comment{ | |||
| 		ID:            generate.CommentID(), | |||
| 		ChapterID:     chapter.ID, | |||
| 		Subject:       subject, | |||
| 		Author:        author, | |||
| 		CharacterName: characterName, | |||
| 		CharacterID:   characterID, | |||
| 		FictionalDate: fictionalDate, | |||
| 		CreatedDate:   createdDate, | |||
| 		EditedDate:    createdDate, | |||
| 		Source:        source, | |||
| 	} | |||
| 	if err := auth.CheckPermission(ctx, "add", comment); err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	comment, err := s.comments.Insert(ctx, *comment) | |||
| 	if err != nil { | |||
| 		return nil, err | |||
| 	} | |||
| 
 | |||
| 	if story, err := s.stories.Find(ctx, chapter.StoryID); err == nil { | |||
| 		s.changeService.Submit(ctx, "Comment", "add", story.Listed, changekeys.Many(story, chapter, comment), comment) | |||
| 	} else { | |||
| 		s.changeService.Submit(ctx, "Comment", "add", false, changekeys.Many(chapter, comment), comment) | |||
| 	} | |||
| 
 | |||
| 	return comment, nil | |||
| } | |||
						Write
						Preview
					
					
					Loading…
					
					Cancel
						Save
					
		Reference in new issue