/**
* File ./src/main/java/de/lemo/dms/processing/questions/QUserByParameter.java
* Lemo-Data-Management-Server for learning analytics.
* Copyright (C) 2013
* Leonard Kappe, Andreas Pursian, Sebastian Schwarzrock, Boris Wenzlaff
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
**/
/**
* File ./main/java/de/lemo/dms/processing/questions/QUserByParameter.java
* Date 2013-01-24
* Project Lemo Learning Analytics
* Copyright TODO (INSERT COPYRIGHT)
*/
package de.lemo.dms.processing.questions;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import javax.ws.rs.FormParam;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import org.hibernate.Criteria;
import org.hibernate.Session;
import org.hibernate.criterion.Restrictions;
import de.lemo.dms.core.config.ServerConfiguration;
import de.lemo.dms.db.IDBHandler;
import de.lemo.dms.db.mapping.CourseUserMining;
import de.lemo.dms.db.mapping.abstractions.ILogMining;
import de.lemo.dms.processing.ELearningObjectType;
import de.lemo.dms.processing.MetaParam;
import de.lemo.dms.processing.Question;
import de.lemo.dms.processing.resulttype.ResultListLongObject;
/**
* Service for retrieval of user-identifiers (Long) that are filtered by the given restrictions.
*
* @author Leonard Kappe
* @author Sebastian Schwarzrock
*/
@Path("userbyparameter")
public class QUserByParameter extends Question {
/**
* Service for retrieval of user-identifiers (Long) that are filtered by the given restrictions.
*
* @param courses
* (Optional, can be combined with types and objects) Only users are returned, that performed actions in
* courses with the given course-identifiers (Long)
* @param types
* (Optional, can be combined with courses and objects) Only users are returned, that performed actions
* on the given types of learning-objects.
* @param objects
* (Optional, can be combined with types and courses) Only users are returned, that performed actions
* specified learning objects. The list should contain pairs of user-identifiers and object-types
* (["1234"],["assignment"] for example).
* @param roles
* @param startTime
* (Mandatory) Only users are returned, that performed actions after the given time.
* @param endTime
* (Mandatory) Only users are returned, that performed actions before the given time.
* @return
*/
@POST
public ResultListLongObject compute(
@FormParam(MetaParam.COURSE_IDS) final List<Long> courses,
@FormParam(MetaParam.TYPES) final List<String> types,
@FormParam(MetaParam.LOG_OBJECT_IDS) final List<String> objects,
@FormParam(MetaParam.ROLE_IDS) final List<Long> roles,
@FormParam(MetaParam.START_TIME) final Long startTime,
@FormParam(MetaParam.END_TIME) final Long endTime) {
validateTimestamps(startTime, endTime);
ResultListLongObject userIds = null;
// Database initialization
final IDBHandler dbHandler = ServerConfiguration.getInstance().getMiningDbHandler();
final Session session = dbHandler.getMiningSession();
// Global list for log items
List<ILogMining> logs = new ArrayList<ILogMining>();
// look up if types-restriction is set or the object-list contains at least one pair
if (!types.isEmpty() || objects.size() > 1) {
for (ELearningObjectType loType : ELearningObjectType.values()) {
String typeName = loType.name().toLowerCase();
if (types.contains(typeName) || objects.contains(typeName)) {
Criteria criteria = session.createCriteria(loType.getLogMiningType(), "log");
criteria.add(Restrictions.between("log.timestamp", startTime, endTime));
if (!courses.isEmpty()) {
criteria.add(Restrictions.in("log.course.id", courses));
}
List<ILogMining> loadLogMining = loadLogMining(criteria, objects, typeName);
if (loadLogMining != null) {
logs.addAll(loadLogMining);
}
}
}
} else {
// If object list and type restriction aren't set, do a more general scan of the database
final Criteria criteria = session.createCriteria(ILogMining.class, "log")
.add(Restrictions.between("log.timestamp", startTime, endTime));
if (!courses.isEmpty()) {
criteria.add(Restrictions.in("log.course.id", courses));
}
@SuppressWarnings("unchecked")
List<ILogMining> loadLogMining = criteria.list();
logs = loadLogMining;
}
HashMap<Long, Long> usersWithinRoles = null;
// Doesn't make any sense unless role-identifiers are linked with respective course-identifiers. Standard
// setting should be 1 course n roles.
// Working with several courses and roles-per-course will only mess up everything
if (!roles.isEmpty()) {
usersWithinRoles = new HashMap<Long, Long>();
final Criteria criteria = session.createCriteria(CourseUserMining.class, "log")
.add(Restrictions.in("log.role.id", roles))
.add(Restrictions.in("log.course.id", courses));
@SuppressWarnings("unchecked")
final List<CourseUserMining> uwr = criteria.list();
for (int i = 0; i < uwr.size(); i++) {
if (uwr.get(i).getUser() != null) {
usersWithinRoles.put(uwr.get(i).getUser().getId(), uwr.get(i).getUser().getId());
}
}
}
// Create HashMap for user-identifiers to prevent multiple entries for the same user-identifier
final HashMap<Long, Long> users = new HashMap<Long, Long>();
for (int i = 0; i < logs.size(); i++)
{
if ((users.get(logs.get(i).getUser().getId()) == null) && usersWithinRoles == null) {
users.put(logs.get(i).getUser().getId(), logs.get(i).getUser().getId());
}
// If there are role restrictions, only user with the specified role get added
if ((usersWithinRoles != null) && (usersWithinRoles.get(logs.get(i).getUser().getId()) != null)) {
users.put(logs.get(i).getUser().getId(), logs.get(i).getUser().getId());
}
}
userIds = new ResultListLongObject(new ArrayList<Long>(users.values()));
session.close();
return userIds;
}
@SuppressWarnings("unchecked")
private List<ILogMining> loadLogMining(Criteria criteria, List<String> objects, String type) {
List<Long> ids = new ArrayList<Long>(objects.size() / 2);
if (objects.contains(type)) {
Long value = 0L;
for (int i = 0; i < objects.size(); i++) {
if ((i % 2) == 0) {
value = Long.valueOf(objects.get(i));
}
if ((i % 2) != 0 && objects.get(i).equals(type)) {
ids.add(value);
}
}
}
if (!ids.isEmpty()) {
criteria.add(Restrictions.in("log." + type + ".id", ids));
}
List<ILogMining> r = criteria.list();
return r;
}
}