001package model.repository;
002
003import model.entity.Post;
004import model.entity.Section;
005import model.entity.User;
006
007import javax.persistence.EntityManager;
008import javax.persistence.PersistenceContext;
009import javax.persistence.TypedQuery;
010import javax.persistence.criteria.*;
011import java.io.Serializable;
012import java.time.Instant;
013import java.util.ArrayList;
014import java.util.Collections;
015import java.util.List;
016
017/**
018 * Classe che incapsula la logica per il recupero di entità di tipo {@link Post}
019 */
020public class PostRepository implements Serializable {
021
022    @PersistenceContext
023    protected EntityManager em;
024
025    /**
026     * Restituisce una nuova istanza di PostFinder
027     * @return nuova istanza di PostFinder
028     */
029    public PostFinder getFinder(){
030        return new PostFinder();
031    }
032
033    private enum SortCriteria {OLDEST, NEWEST, MOSTVOTED};
034
035    /**
036     *  Classe interna usata per specificare i parametri di ricerca di un post
037     */
038    public class PostFinder {
039        private User author;
040        private List<Section> sections;
041        private Instant before;
042        private Instant after;
043        private int offset = 0;
044        private int pageSize = 30;
045        private String content;
046        private boolean includeBody = false;
047        private User joinUserFollows;
048        private SortCriteria sortCriteria = PostRepository.SortCriteria.NEWEST;
049
050        protected PostFinder(){ }
051
052        /**
053         * Setta il campo author e restituisce l'istanza passata di PostFinder
054         * @param author entità User dell'autore dei post
055         * @return istanza passata di PostFinder
056         */
057        public PostFinder byAuthor(User author){
058            this.author = author;
059            return this;
060        }
061
062        /**
063         * Setta il campo content e restituisce l'istanza passata di PostFinder
064         * @param author text contenuto dei post
065         * @return istanza passata di PostFinder
066         */
067        public PostFinder byContent(String text){
068            this.content = text;
069            return this;
070        }
071
072        /**
073         * Setta il campo sections e restituisce l'istanza passata di PostFinder
074         * @param sections lista di sezioni
075         * @return istanza passata di PostFinder
076         */
077        public PostFinder bySections(List<Section> sections){
078            this.sections = sections;
079            return this;
080        }
081
082        /**
083         * Setta il campo sections e restituisce l'istanza passata di PostFinder
084         * @param sections sezione dei post
085         * @return istanza passata di PostFinder
086         */
087        public PostFinder bySection(Section section){
088            this.sections = Collections.singletonList(section);
089            return this;
090        }
091
092        /**
093         * Setta il campo postedAfter e restituisce l'istanza passata di PostFinder
094         * @param after data dopo la quale i post da trovare sono stati postati
095         * @return istanza passata di PostFinder
096         */
097        public PostFinder postedAfter(Instant after){
098            this.after = after;
099            return this;
100        }
101
102        /**
103         * Setta il campo postedBefore e restituisce l'istanza passata di PostFinder
104         * @param before data prima della quale i post da trovare sono stati postati
105         * @return istanza passata di PostFinder
106         */
107        public PostFinder postedBefore(Instant before){
108            this.before = before;
109            return this;
110        }
111
112        /**
113         * Setta il campo offset e restituisce l'istanza passata di PostFinder
114         * @param n offset per la paginazione
115         * @return istanza passata di PostFinder
116         */
117        public PostFinder offset(int n){
118            offset = n;
119            return this;
120        }
121
122        /**
123         * Setta il campo limit e restituisce l'istanza passata di PostFinder
124         * @param n limite di post da caricare
125         * @return istanza passata di PostFinder
126         */
127        public PostFinder limit(int n){
128            pageSize = n;
129            return this;
130        }
131
132        /**
133         * Setta il campo sortCriteria a oldest e restituisce l'istanza passata di PostFinder
134         * @return istanza passata di PostFinder
135         */
136        public PostFinder getOldest(){
137            sortCriteria = SortCriteria.OLDEST;
138            return this;
139        }
140
141        /**
142         * Setta il campo sortCriteria a newest e restituisce l'istanza passata di PostFinder
143         * @return istanza passata di PostFinder
144         */
145        public PostFinder getNewest(){
146            sortCriteria = SortCriteria.NEWEST;
147            return this;
148        }
149
150        /**
151         * Setta il campo sortCriteria a most voted e restituisce l'istanza passata di PostFinder
152         * @return istanza passata di PostFinder
153         */
154        public PostFinder getMostVoted(){
155            sortCriteria = SortCriteria.MOSTVOTED;
156            return this;
157        }
158
159        /**
160         * Setta il campo includeBody a true e restituisce l'istanza passata di PostFinder
161         * @return istanza passata di PostFinder
162         */
163        public PostFinder includeBody(){
164            includeBody = true;
165            return this;
166        }
167
168        /**
169         * Setta il campo joinUserFollows e restituisce l'istanza passata di PostFinder
170         * @param user entita utente da cui ottenere le sezioni seguite
171         * @return istanza passata di PostFinder
172         */
173        public PostFinder joinUserFollows(User user){
174            joinUserFollows = user;
175            return this;
176        }
177
178        /**
179         * Restituisce tutti i post che rispettano i criteri di ricerca
180         * @return lista di entità Post
181         */
182        public List<Post> getResults(){
183            CriteriaBuilder cb = em.getCriteriaBuilder();
184            CriteriaQuery<Post> cq = cb.createQuery(Post.class);
185            Root<Post> root = cq.from(Post.class);
186
187            List<Predicate> predicates = new ArrayList<>();
188
189            if(author != null) {
190                predicates.add(cb.equal(root.get("author"), author));
191            }
192
193            if(sections != null && !sections.isEmpty()) {
194                predicates.add(root.get("section").in(sections));
195            }
196
197            if(after != null) {
198                predicates.add(cb.greaterThanOrEqualTo(root.get("creationDate"), after));
199            }
200
201            if(before != null) {
202                predicates.add(cb.lessThanOrEqualTo(root.get("creationDate"), before));
203            }
204
205            if(content != null) {
206                if(includeBody) {
207                    predicates.add(
208                            cb.or(
209                                    cb.like(root.get("title"), '%' + content + '%'),
210                                    cb.and(
211                                            cb.notEqual(root.get("type"), Post.Type.IMG),
212                                            cb.like(root.get("content"), '%' + content + '%')
213                                    )));
214                } else {
215                    predicates.add(cb.like(root.get("title"), '%' + content + '%'));
216                }
217            }
218
219            if(joinUserFollows != null){
220                Join<Object, Object> joinFollow = root.join("section").join("follows");
221                predicates.add(cb.equal(joinFollow.get("user"), joinUserFollows));
222            }
223
224            cq.where(cb.and(predicates.toArray(Predicate[]::new)));
225
226            switch (sortCriteria) {
227                case MOSTVOTED:
228                    cq.orderBy(cb.desc(root.get("votesCount")));
229                    break;
230                case OLDEST:
231                    cq.orderBy(cb.asc(root.get("creationDate")));
232                    break;
233                case NEWEST:
234                default:
235                    cq.orderBy(cb.desc(root.get("creationDate")));
236                    break;
237            }
238
239
240            TypedQuery<Post> tq = em.createQuery(cq);
241            if(offset >= 0)
242                tq.setFirstResult(offset);
243            if(pageSize>=0)
244                tq.setMaxResults(pageSize);
245
246            return tq.getResultList();
247        }
248    }
249}