summaryrefslogtreecommitdiff
path: root/Services/FeedService.cs
blob: b664d624c76eb21e937877463ebb4a22e554d6a3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Query;

namespace HyperBooru.Services;

public enum FeedSortOrder {
    Ascending,
    Descending
}

public enum FeedSortType {
    Chronological,
    Rating
}

public record FeedOptions {
    public FeedSortType  SortType       { get; set; }
    public FeedSortOrder SortOrder      { get; set; }
    public bool          RandomPosition { get; set; }

    public Func<IQueryable<Media>, IQueryable<Media>>? IncludeProperties { get; set; }
}

public interface IFeedService {
    public IEnumerable<Media> Feed { get; }
    public void InitializeFeed(FeedOptions feedOptions);
    public Media[] Next(int count);
}

public class FeedService : IFeedService {
    private const int FeedChunkSize = 50;

    private FeedOptions? feedOptions;
    private Media? last;

    private IDbContextFactory<HBContext> dbFactory;

    public FeedService(IDbContextFactory<HBContext> dbFactory) =>
        this.dbFactory = dbFactory;

    public void InitializeFeed(FeedOptions feedOptions) {
        this.feedOptions = feedOptions;
        last             = null;
    }

    public IEnumerable<Media> Feed {
        get {
            last = null;
            while(true) {
                var media = Next(FeedChunkSize);
                if(media.Count() == 0)
                    break;
                foreach(var m in media)
                    yield return m;
            }
        }
    }

    public Media[] Next(int count) =>
        NextDbChunk(Math.Abs(count), count < 0);

    private Media[] NextDbChunk(int chunkSize, bool reverse = false) {
        if(feedOptions is null)
            throw new InvalidOperationException("Feed must be initialized first");

        while(true) {
            var db = dbFactory.CreateDbContext();

            IQueryable<Media> media = db.Media;

            if(feedOptions.IncludeProperties is not null)
                media = feedOptions.IncludeProperties(media);

            var sortOrder = feedOptions.SortOrder;
            if(reverse)
                sortOrder = sortOrder == FeedSortOrder.Ascending
                    ? FeedSortOrder.Descending
                    : FeedSortOrder.Ascending;

            if(last is not null) {
                switch(feedOptions.SortType) {
                    case FeedSortType.Chronological:
                        if(sortOrder == FeedSortOrder.Descending)
                            media = media.Where(m => m.ObjectId < last.ObjectId);
                        else
                            media = media.Where(m => m.ObjectId > last.ObjectId);
                        break;
                }
                media = media.Where(m => m.ObjectId != last.ObjectId);
            }

            switch(feedOptions.SortType) {
                case FeedSortType.Chronological:
                    if(sortOrder == FeedSortOrder.Descending)
                        media = media.OrderByDescending(m => m.ObjectId);
                    else
                        media = media.OrderBy(m => m.ObjectId);
                    break;
            }

            Media[] mediaArray = media
                .Take(chunkSize)
                .ToArray();

            if(mediaArray.Count() != 0)
                last = mediaArray.Last();
        }
    }
}