Ranking algorithm now considers time. Button to show all hashtags whene there are more than ten added. Portlet moved to liferay 6.2 and version number to 2.0.0

git-svn-id: http://svn.research-infrastructures.eu/public/d4science/gcube/trunk/portlets/user/top-topics@128534 82a268e6-3cf1-43bd-a215-b396298e98cf
This commit is contained in:
Costantino Perciante 2016-05-10 10:03:57 +00:00
parent f7948fafce
commit f08a74a8a7
7 changed files with 307 additions and 30 deletions

14
pom.xml
View File

@ -25,7 +25,7 @@
</scm> </scm>
<properties> <properties>
<!-- Convenience property to set the GWT version --> <!-- Convenience property to set the GWT version -->
<gwtVersion>2.5.1</gwtVersion> <gwtVersion>2.7.0</gwtVersion>
<distroDirectory>distro</distroDirectory> <distroDirectory>distro</distroDirectory>
<liferayVersion>6.2.5</liferayVersion> <liferayVersion>6.2.5</liferayVersion>
<!-- GWT needs at least java 1.6 --> <!-- GWT needs at least java 1.6 -->
@ -52,10 +52,18 @@
<dependency> <dependency>
<groupId>com.google.gwt</groupId> <groupId>com.google.gwt</groupId>
<artifactId>gwt-user</artifactId> <artifactId>gwt-user</artifactId>
<version>${gwtVersion}</version>
</dependency> </dependency>
<dependency> <dependency>
<groupId>com.google.gwt</groupId> <groupId>com.google.gwt</groupId>
<artifactId>gwt-servlet</artifactId> <artifactId>gwt-servlet</artifactId>
<version>${gwtVersion}</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>com.google.gwt</groupId>
<artifactId>gwt-dev</artifactId>
<version>${gwtVersion}</version>
<scope>provided</scope> <scope>provided</scope>
</dependency> </dependency>
<dependency> <dependency>
@ -106,6 +114,10 @@
<artifactId>portlet-api</artifactId> <artifactId>portlet-api</artifactId>
<scope>provided</scope> <scope>provided</scope>
</dependency> </dependency>
<dependency>
<groupId>com.github.gwtbootstrap</groupId>
<artifactId>gwt-bootstrap</artifactId>
</dependency>
</dependencies> </dependencies>
<build> <build>

View File

@ -6,7 +6,10 @@ import org.gcube.portlets.user.topics.client.TopicService;
import org.gcube.portlets.user.topics.client.TopicServiceAsync; import org.gcube.portlets.user.topics.client.TopicServiceAsync;
import org.gcube.portlets.user.topics.shared.HashtagsWrapper; import org.gcube.portlets.user.topics.shared.HashtagsWrapper;
import com.github.gwtbootstrap.client.ui.Button;
import com.google.gwt.core.client.GWT; import com.google.gwt.core.client.GWT;
import com.google.gwt.event.dom.client.ClickEvent;
import com.google.gwt.event.dom.client.ClickHandler;
import com.google.gwt.user.client.rpc.AsyncCallback; import com.google.gwt.user.client.rpc.AsyncCallback;
import com.google.gwt.user.client.ui.Composite; import com.google.gwt.user.client.ui.Composite;
import com.google.gwt.user.client.ui.HTML; import com.google.gwt.user.client.ui.HTML;
@ -27,6 +30,7 @@ public class TopicsPanel extends Composite {
public static final String loading = GWT.getModuleBaseURL() + "../images/topics-loader.gif"; public static final String loading = GWT.getModuleBaseURL() + "../images/topics-loader.gif";
public static final String DISPLAY_NAME = "Top Topics"; public static final String DISPLAY_NAME = "Top Topics";
public static final int THRESHOLD_SHOW_HASHTAGS = 10; // show the first X ones
private Image loadingImage; private Image loadingImage;
@ -50,13 +54,42 @@ public class TopicsPanel extends Composite {
showServError(); showServError();
} }
else { else {
int counter = 0;
if (hashtags != null) { if (hashtags != null) {
for (String hashtag : hashtags) { for (String hashtag : hashtags) {
counter ++;
HTML toAdd = new HTML(hashtag); HTML toAdd = new HTML(hashtag);
toAdd.addStyleName("hashtag-label"); toAdd.addStyleName("hashtag-label");
mainPanel.add(toAdd); mainPanel.add(toAdd);
if(counter > THRESHOLD_SHOW_HASHTAGS) // 11, 12...
toAdd.setVisible(false);
} }
} }
// add a show all button if needed
if(counter > THRESHOLD_SHOW_HASHTAGS){
final Button showAllHashtags = new Button("Show All");
showAllHashtags.addClickHandler(new ClickHandler() {
@Override
public void onClick(ClickEvent event) {
int numberChildren = mainPanel.getWidgetCount();
for (int i = THRESHOLD_SHOW_HASHTAGS; i < numberChildren; i++) {
mainPanel.getWidget(i).setVisible(true);
}
// hide the button
showAllHashtags.setVisible(false);
}
});
mainPanel.add(showAllHashtags);
}
} }
} }

View File

@ -1,9 +1,16 @@
package org.gcube.portlets.user.topics.server; package org.gcube.portlets.user.topics.server;
import java.text.SimpleDateFormat;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collections; import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Map.Entry;
import org.apache.commons.codec.binary.Base64; import org.apache.commons.codec.binary.Base64;
import org.gcube.application.framework.core.session.ASLSession; import org.gcube.application.framework.core.session.ASLSession;
@ -12,6 +19,7 @@ import org.gcube.portal.custom.scopemanager.scopehelper.ScopeHelper;
import org.gcube.portal.databook.client.GCubeSocialNetworking; import org.gcube.portal.databook.client.GCubeSocialNetworking;
import org.gcube.portal.databook.server.DBCassandraAstyanaxImpl; import org.gcube.portal.databook.server.DBCassandraAstyanaxImpl;
import org.gcube.portal.databook.server.DatabookStore; import org.gcube.portal.databook.server.DatabookStore;
import org.gcube.portal.databook.shared.Feed;
import org.gcube.portlets.user.topics.client.TopicService; import org.gcube.portlets.user.topics.client.TopicService;
import org.gcube.portlets.user.topics.shared.HashTagAndOccurrence; import org.gcube.portlets.user.topics.shared.HashTagAndOccurrence;
import org.gcube.portlets.user.topics.shared.HashtagsWrapper; import org.gcube.portlets.user.topics.shared.HashtagsWrapper;
@ -34,6 +42,8 @@ public class TopicServiceImpl extends RemoteServiceServlet implements TopicServi
private static final Logger _log = LoggerFactory.getLogger(TopicServiceImpl.class); private static final Logger _log = LoggerFactory.getLogger(TopicServiceImpl.class);
public static final String TEST_USER = "test.user"; public static final String TEST_USER = "test.user";
private static final int WINDOW_SIZE_IN_MONTHS = 6; // it must not exceed 12
/** /**
* The Cassandra store interface * The Cassandra store interface
*/ */
@ -68,20 +78,32 @@ public class TopicServiceImpl extends RemoteServiceServlet implements TopicServi
*/ */
public String getDevelopmentUser() { public String getDevelopmentUser() {
String user = TEST_USER; String user = TEST_USER;
//user = "massimiliano.assante"; // user = "massimiliano.assante";
return user; return user;
} }
/** /**
* return the top 10 hashtag with max occurrence * return trending hashtags
*/ */
@Override @Override
public HashtagsWrapper getHashtags() { public HashtagsWrapper getHashtags() {
ArrayList<String> hashtagsChart = new ArrayList<>(); ArrayList<String> hashtagsChart = new ArrayList<>();
ASLSession session = getASLSession(); ASLSession session = getASLSession();
long timestampStart = System.currentTimeMillis();
// get the reference time
Calendar referenceTime = Calendar.getInstance();
int currentMonth = referenceTime.get(Calendar.MONTH); // jan = 0, ..... dec = 11
referenceTime.set(Calendar.MONTH, currentMonth - WINDOW_SIZE_IN_MONTHS); // the year is automatically decreased if needed
// print it
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd");
_log.debug("Reference time for trending topics is " + format.format(referenceTime.getTime()));
String userName = session.getUsername(); String userName = session.getUsername();
boolean isInfrastructure = isInfrastructureScope(); boolean isInfrastructure = isInfrastructureScope();
try { try {
//in case the portal is restarted and you have the social home open it will get test.user (no callback to set session info) //in case the portal is restarted and you have the social home open it will get test.user (no callback to set session info)
//this check just return nothing if that happens //this check just return nothing if that happens
@ -90,41 +112,87 @@ public class TopicServiceImpl extends RemoteServiceServlet implements TopicServi
return null; return null;
} }
ArrayList<HashTagAndOccurrence> toSort = new ArrayList<HashTagAndOccurrence>(); ArrayList<HashTagAndOccurrence> toSort = new ArrayList<HashTagAndOccurrence>();
/**
* this handles the case where the portlet is deployed outside of VREs (regular)
*/
if (isInfrastructure) { if (isInfrastructure) {
_log.debug("****** retrieving hashtags for user VREs"); _log.debug("****** retrieving hashtags for user VREs");
// different vres could have a same hashtag, we need to merge them
Map<String, Integer> hashtags = new HashMap<String, Integer>();
// we need a map for the couple <hashtag, vre in which it is present>
// it is needed because later we need to retrieve the most recent feed among the ones
// containing the hashtag itself
Map<String, List<String>> hashtagsInVres = new HashMap<String, List<String>>();
GroupManager gm = new LiferayGroupManager(); GroupManager gm = new LiferayGroupManager();
UserManager um = new LiferayUserManager(); UserManager um = new LiferayUserManager();
GCubeUser user = um.getUserByUsername(userName); GCubeUser user = um.getUserByUsername(userName);
List<GCubeGroup> groups = gm.listGroupsByUser(user.getUserId()); List<GCubeGroup> groups = gm.listGroupsByUser(user.getUserId());
for (GCubeGroup group : groups) { for (GCubeGroup group : groups) {
if (gm.isVRE(group.getGroupId())) { if (gm.isVRE(group.getGroupId())) {
_log.debug("Retrieving hashtags from VRE " + group.getGroupName());
String vreid = gm.getInfrastructureScope(group.getGroupId()); //get the scope String vreid = gm.getInfrastructureScope(group.getGroupId()); //get the scope
Map<String, Integer> map = store.getVREHashtagsWithOccurrence(vreid); Map<String, Integer> map = store.getVREHashtagsWithOccurrenceFilteredByTime(vreid, referenceTime.getTimeInMillis());
// merge the values if needed
for (String hashtag : map.keySet()) { for (String hashtag : map.keySet()) {
toSort.add(new HashTagAndOccurrence(hashtag, map.get(hashtag)));
if(hashtags.containsKey(hashtag)){
int currentValue = hashtags.get(hashtag);
int newValue = currentValue + map.get(hashtag);
// remove and re-add
hashtags.remove(hashtag);
hashtags.put(hashtag, newValue);
// get the current list of vres in which the hashtag is present and add this new one
List<String> vres = hashtagsInVres.get(hashtag);
vres.add(vreid);
hashtagsInVres.remove(hashtag);
hashtagsInVres.put(hashtag, vres);
}else{
hashtags.put(hashtag, map.get(hashtag));
// put in the hashmap hashtagsInVres too
List<String> vres = new ArrayList<String>();
vres.add(vreid);
hashtagsInVres.put(hashtag, vres);
}
} }
} }
} }
// now we need to evaluate score for each element
Map<String, Double> weights = evaluateWeight(hashtags, WINDOW_SIZE_IN_MONTHS, currentMonth, referenceTime, null, hashtagsInVres);
// at the end build the list
for (String hashtag : hashtags.keySet()) {
toSort.add(new HashTagAndOccurrence(hashtag, hashtags.get(hashtag), weights.get(hashtag)));
}
} }
//else must be in a VRE scope //else must be in a VRE scope
else { else {
String scope = session.getScope(); String scope = session.getScope();
_log.debug("****** retrieving hashtags for scope " + scope); _log.debug("****** retrieving hashtags for scope " + scope);
Map<String, Integer> map = store.getVREHashtagsWithOccurrence(scope); Map<String, Integer> hashtags = store.getVREHashtagsWithOccurrenceFilteredByTime(scope, referenceTime.getTimeInMillis());
for (String hashtag : map.keySet()) { // now we need to evaluate the wiehgt for each element
toSort.add(new HashTagAndOccurrence(hashtag, map.get(hashtag))); Map<String, Double> weights = evaluateWeight(hashtags, WINDOW_SIZE_IN_MONTHS, currentMonth, referenceTime, scope, null);
for (String hashtag : hashtags.keySet()) {
toSort.add(new HashTagAndOccurrence(hashtag, hashtags.get(hashtag), weights.get(hashtag)));
} }
} }
_log.debug("Number of topics retrieved is " + toSort.size()); _log.debug("Number of topics retrieved is " + toSort.size());
Collections.sort(toSort, Collections.reverseOrder());
int i = 0; Collections.sort(toSort); // sort for weight
for (HashTagAndOccurrence wrapper : toSort) { for (HashTagAndOccurrence wrapper : toSort) {
_log.debug("Entry is " + wrapper.toString() + " with weight " + wrapper.getWeight());
String hashtag = wrapper.getHashtag(); String hashtag = wrapper.getHashtag();
String href="\"?"+ String href="\"?"+
@ -132,19 +200,165 @@ public class TopicServiceImpl extends RemoteServiceServlet implements TopicServi
new String(Base64.encodeBase64(hashtag.getBytes()))+"\""; new String(Base64.encodeBase64(hashtag.getBytes()))+"\"";
String hashtagLink = "<a class=\"topiclink\" href=" + href + ">"+hashtag+"</a>"; String hashtagLink = "<a class=\"topiclink\" href=" + href + ">"+hashtag+"</a>";
hashtagsChart.add(hashtagLink); hashtagsChart.add(hashtagLink);
i++;
if (i >= 10)
break;
} }
} }
catch (Exception e) { catch (Exception e) {
e.printStackTrace(); e.printStackTrace();
return null; return null;
} }
long timestampEnd = System.currentTimeMillis() - timestampStart;
_log.debug("Overall time to retrieve hastags is " + timestampEnd);
return new HashtagsWrapper(isInfrastructure, hashtagsChart); return new HashtagsWrapper(isInfrastructure, hashtagsChart);
} }
/**
* Evaluate the weight for each element as w = 0.6 * s + 0.4 * f
* where s is the score: a normalized value given by counter_i / counter_max
* f is the freshness: evaluated taking into account the most recent feed containing that hashtag into the window w (that is,
* the period taken into account)
* @param hashtags
* @param hashtagsInVres (present if vreid is null)
* @param window size
* @param current month
* @param referenceTime
* @param vreid (present if hashtagsInVres is null)
* @return a Map of weight for each hashtag
*/
private Map<String, Double> evaluateWeight(Map<String, Integer> hashtags, int windowSize, int currentMonth, Calendar referenceTime, String vreId, Map<String, List<String>> hashtagsInVres) {
Map<String, Double> weights = new HashMap<String, Double>();
// find max score inside the list (counter)
int max = 0;
for(Entry<String, Integer> entry : hashtags.entrySet()){
max = max < entry.getValue() ? entry.getValue() : max;
}
// normalize
Map<String, Double> normalized = new HashMap<String, Double>();
for(Entry<String, Integer> entry : hashtags.entrySet()){
normalized.put(entry.getKey(), (double)entry.getValue() / (double)max);
}
// create the weight for each entry as:
// w = 0.6 * normalized_score + 0.4 * freshness
// freshness is evaluated as (window_size - latest_feed_for_hashtag_in_window_month)/window_size
for(Entry<String, Integer> entry : hashtags.entrySet()){
// first part of the weight
double weight = 0.6 * normalized.get(entry.getKey());
List<Feed> mostRecentFeedForHashtag = null;
// we are in the simplest case.. the hashtag belongs (or the request comes) from a single vre
if(vreId != null){
try{
mostRecentFeedForHashtag = store.getVREFeedsByHashtag(vreId, entry.getKey());
}catch(Exception e){
_log.error("Unable to retrieve the most recent feeds for hashtag " + entry.getKey() + " in " + vreId);
// put a weight of zero for this hashtag
weights.put(entry.getKey(), 0.0);
continue;
}
}else{ // we are not so lucky
// get the list of vres for this hashtag
List<String> vres = hashtagsInVres.get(entry.getKey());
// init list
mostRecentFeedForHashtag = new ArrayList<Feed>();
List<Feed> feedsForVre;
for (String vre : vres) {
try{
feedsForVre = store.getVREFeedsByHashtag(vre, entry.getKey());
}catch(Exception e){
_log.error("Unable to retrieve the most recent feeds for hashtag " + entry.getKey() + " in " + vreId);
continue;
}
// add to the list
mostRecentFeedForHashtag.addAll(feedsForVre);
}
// check if there is at least a feed or it is empty
if(mostRecentFeedForHashtag.isEmpty()){
// put a weight of zero for this hashtag
weights.put(entry.getKey(), 0.0);
continue;
}
}
// retrieve the most recent one among these feeds
Collections.sort(mostRecentFeedForHashtag, Collections.reverseOrder());
// get month of the last recent feed for this hashtag
Calendar monstRecentFeedForHashTagTime = Calendar.getInstance();
monstRecentFeedForHashTagTime.setTimeInMillis(mostRecentFeedForHashtag.get(0).getTime().getTime());
int sub = currentMonth - monstRecentFeedForHashTagTime.get(Calendar.MONTH);
int value = sub >= 0? sub : 12 - Math.abs(sub);
double freshness = 1.0 - (double)(value) / (double)(windowSize);
_log.debug("freshness is " + freshness + " for hashtag " + entry.getKey() +
" because the last feed has month " + monstRecentFeedForHashTagTime.get(Calendar.MONTH));
// update the weight
weight += 0.4 * freshness;
// put it into the hashmap
weights.put(entry.getKey(), weight);
}
// print sorted
Map<String, Double> scoredListSorted = sortByWeight(weights);
for(Entry<String, Double> entry : scoredListSorted.entrySet()){
System.out.println("[hashtag=" + entry.getKey() + " , weight=" + entry.getValue() + "]");
}
return weights;
}
/**
* Sort a map by its values
* @param map
* @return
*/
private static <K, V extends Comparable<? super V>> Map<K, V>
sortByWeight( Map<K, V> map )
{
List<Map.Entry<K, V>> list =
new LinkedList<Map.Entry<K, V>>( map.entrySet() );
Collections.sort( list, new Comparator<Map.Entry<K, V>>()
{
public int compare( Map.Entry<K, V> o1, Map.Entry<K, V> o2 )
{
return (o2.getValue()).compareTo( o1.getValue() );
}
});
Map<K, V> result = new LinkedHashMap<K, V>();
for (Map.Entry<K, V> entry : list)
{
result.put( entry.getKey(), entry.getValue() );
}
return result;
}
/** /**
* Indicates whether the scope is the whole infrastructure. * Indicates whether the scope is the whole infrastructure.
* @return <code>true</code> if it is, <code>false</code> otherwise. * @return <code>true</code> if it is, <code>false</code> otherwise.

View File

@ -3,11 +3,18 @@ package org.gcube.portlets.user.topics.shared;
public class HashTagAndOccurrence implements Comparable<HashTagAndOccurrence>{ public class HashTagAndOccurrence implements Comparable<HashTagAndOccurrence>{
private String hashtag; private String hashtag;
private Integer occurrence; private Integer occurrence;
private double weight;
public HashTagAndOccurrence(String hashtag, Integer occurrence) { public HashTagAndOccurrence(String hashtag, Integer occurrence) {
super(); super();
this.hashtag = hashtag; this.hashtag = hashtag;
this.occurrence = occurrence; this.occurrence = occurrence;
} }
public HashTagAndOccurrence(String hashtag, Integer occurrence, double weight) {
super();
this.hashtag = hashtag;
this.occurrence = occurrence;
this.weight = weight;
}
public String getHashtag() { public String getHashtag() {
return hashtag; return hashtag;
} }
@ -20,16 +27,20 @@ public class HashTagAndOccurrence implements Comparable<HashTagAndOccurrence>{
public void setOccurrence(Integer occurrence) { public void setOccurrence(Integer occurrence) {
this.occurrence = occurrence; this.occurrence = occurrence;
} }
public double getWeight() {
return weight;
}
public void setWeight(double weight) {
this.weight = weight;
}
@Override @Override
public String toString() { public String toString() {
return "HashTagAndOccurrence [hashtag=" + hashtag + ", occurrence=" return "HashTagAndOccurrence [hashtag=" + hashtag + ", occurrence="
+ occurrence + "]"; + occurrence + ", weight=" + weight + "]";
} }
@Override @Override
public int compareTo(HashTagAndOccurrence o) { public int compareTo(HashTagAndOccurrence o) {
if (this.occurrence == o.getOccurrence()) return 0; return Double.compare(o.getWeight(), this.weight);
return (this.occurrence > o.getOccurrence()) ? 1 : -1;
} }
} }

View File

@ -2,8 +2,14 @@ package org.gcube.portlets.user.topics.shared;
import java.io.Serializable; import java.io.Serializable;
import java.util.ArrayList; import java.util.ArrayList;
/**
* @author Massimiliano Assante at ISTI-CNR
* (massimiliano.assante@isti.cnr.it)
* @author Costantino Perciante at ISTI-CNR
* (costantino.perciante@isti.cnr.it)
*/
public class HashtagsWrapper implements Serializable { public class HashtagsWrapper implements Serializable {
private static final long serialVersionUID = -532083077958376460L;
private boolean isInfrastructure; private boolean isInfrastructure;
private ArrayList<String> hashtags; private ArrayList<String> hashtags;
public HashtagsWrapper(boolean isInfrastructure, ArrayList<String> hashtags) { public HashtagsWrapper(boolean isInfrastructure, ArrayList<String> hashtags) {

View File

@ -3,13 +3,15 @@
<!-- Inherit the core Web Toolkit stuff. --> <!-- Inherit the core Web Toolkit stuff. -->
<inherits name='com.google.gwt.user.User' /> <inherits name='com.google.gwt.user.User' />
<!-- To Comment out --> <!-- To Comment out -->
<!-- <set-property name="user.agent" value="safari,gecko1_8" /> --> <!-- <set-property name="user.agent" value="safari,gecko1_8" /> -->
<!-- Bootstrap import -->
<inherits name="com.github.gwtbootstrap.Bootstrap" />
<!-- Other module inherits --> <!-- Other module inherits -->
<inherits name='org.gcube.portlets.user.gcubewidgets.WidgetFactory' /> <inherits name='org.gcube.portlets.user.gcubewidgets.WidgetFactory' />
<inherits name='org.gcube.portal.databook.GCubeSocialNetworking' /> <inherits name='org.gcube.portal.databook.GCubeSocialNetworking' />
<entry-point <entry-point class='org.gcube.portlets.user.topics.client.TopTopics' />
class='org.gcube.portlets.user.topics.client.TopTopics' />
<!-- Specify the paths for translatable code --> <!-- Specify the paths for translatable code -->
<source path='client' /> <source path='client' />

View File

@ -6,7 +6,6 @@ a.topiclink, a.topiclink:active, a.topiclink:visited {
cursor: hand; cursor: hand;
text-decoration: none; text-decoration: none;
color: #0078b2 !important; color: #0078b2 !important;
font-weight: bold !important;
} }
a.topiclink:hover { a.topiclink:hover {