com.xpn.xwiki.stats.impl.XWikiStatsServiceImpl.java Source code

Java tutorial

Introduction

Here is the source code for com.xpn.xwiki.stats.impl.XWikiStatsServiceImpl.java

Source

/*
 * See the NOTICE file distributed with this work for additional
 * information regarding copyright ownership.
 *
 * This is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 *
 * This software 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package com.xpn.xwiki.stats.impl;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Date;
import java.util.List;
import java.util.Map;

import org.apache.commons.collections.buffer.CircularFifoBuffer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.xwiki.bridge.event.ActionExecutedEvent;
import org.xwiki.model.reference.DocumentReferenceResolver;
import org.xwiki.observation.EventListener;
import org.xwiki.observation.ObservationManager;
import org.xwiki.observation.event.Event;
import org.xwiki.observation.remote.RemoteObservationManagerContext;

import com.xpn.xwiki.XWikiContext;
import com.xpn.xwiki.XWikiException;
import com.xpn.xwiki.criteria.impl.Duration;
import com.xpn.xwiki.criteria.impl.Period;
import com.xpn.xwiki.criteria.impl.Range;
import com.xpn.xwiki.criteria.impl.Scope;
import com.xpn.xwiki.doc.XWikiDocument;
import com.xpn.xwiki.stats.api.XWikiStatsService;
import com.xpn.xwiki.stats.impl.xwiki.XWikiStatsReader;
import com.xpn.xwiki.stats.impl.xwiki.XWikiStatsStoreService;
import com.xpn.xwiki.web.DownloadAction;
import com.xpn.xwiki.web.SaveAction;
import com.xpn.xwiki.web.Utils;
import com.xpn.xwiki.web.ViewAction;

/**
 * Store and retrieve statistics.
 * 
 * @version $Id: b731dca64b3e5f351978c815e15cd0a03bb619ff $
 */
public class XWikiStatsServiceImpl implements XWikiStatsService, EventListener {
    /**
     * Logging tools.
     */
    private static final Logger LOGGER = LoggerFactory.getLogger(XWikiStatsServiceImpl.class);

    /**
     * The name of the listener.
     */
    private static final String NAME = "statistics";

    /**
     * User actions statistics module saves.
     */
    private static final List<Event> EVENTS = new ArrayList<Event>() {
        {
            add(new ActionExecutedEvent(ViewAction.VIEW_ACTION));
            add(new ActionExecutedEvent(SaveAction.ACTION_NAME));
            add(new ActionExecutedEvent(DownloadAction.ACTION_NAME));
        }
    };

    /**
     * Used to resolve reference based on context.
     */
    private DocumentReferenceResolver<String> currentDocumentReferenceResolver = Utils
            .getComponent(DocumentReferenceResolver.class, "current");

    /**
     * The statistics storing thread.
     */
    private XWikiStatsStoreService statsRegister;

    /**
     * The statistics database reader.
     */
    private XWikiStatsReader statsReader = new XWikiStatsReader();

    @Override
    public String getName() {
        return NAME;
    }

    @Override
    public List<Event> getEvents() {
        return EVENTS;
    }

    @Override
    public void init(XWikiContext context) {
        if (LOGGER.isInfoEnabled()) {
            LOGGER.info("Start statistics service initialization");
        }

        if (StatsUtil.isStatsEnabled(context)) {
            // Start statistics store thread
            this.statsRegister = new XWikiStatsStoreService(context);
            this.statsRegister.start();

            // Adding the rule which will allow this module to be called on each page view
            Utils.getComponent(ObservationManager.class).addListener(this);
        }
    }

    @Override
    public Collection<Object> getRecentActions(String action, int size, XWikiContext context) {
        return this.statsReader.getRecentActions(action, size, context);
    }

    @Override
    public void onEvent(Event event, Object source, Object data) {
        if (Utils.getComponent(RemoteObservationManagerContext.class).isRemoteState()) {
            // we do nothing when the event comes from remote instance since the remote instance is supposed to already
            // take care of this
            return;
        }

        ActionExecutedEvent actionEvent = (ActionExecutedEvent) event;
        XWikiDocument document = (XWikiDocument) source;
        XWikiContext context = (XWikiContext) data;

        // If the server is in read-only mode, forget about the statistics (since it's in read only mode we don't write
        // anything in the database)
        if (context.getWiki().isReadOnly()) {
            return;
        }

        // Initialize cookie used as unique identifier of a user visit and put it in the context
        StatsUtil.findCookie(context);

        String action = actionEvent.getActionName();

        // Let's save in the session the last elements view, saved
        synchronized (this) {
            if (!action.equals(DownloadAction.ACTION_NAME)) {
                Collection actions = StatsUtil.getRecentActionFromSessions(context, action);
                if (actions == null) {
                    actions = new CircularFifoBuffer(StatsUtil.getRecentVisitSize(context));
                    StatsUtil.setRecentActionsFromSession(context, action, actions);
                }

                String element = document.getPrefixedFullName();
                if (actions.contains(element)) {
                    actions.remove(element);
                }
                actions.add(element);
            }
        }

        try {
            if (StatsUtil.isWikiStatsEnabled(context) && !StatsUtil.getStorageFilteredUsers(context)
                    .contains(this.currentDocumentReferenceResolver.resolve(context.getUser()))) {
                this.statsRegister.addStats(document, action, context);
            }
        } catch (Exception e) {
            LOGGER.error("Faild to get filter users list", e);
        }
    }

    @Override
    public Map<?, ?> getActionStatistics(String action, Scope scope, Period period, Duration step,
            XWikiContext context) {
        return this.statsReader.getActionStatistics(action, scope, period, step, context);
    }

    @Override
    public List<DocumentStats> getDocumentStatistics(String action, Scope scope, Period period, Range range,
            XWikiContext context) {
        return this.statsReader.getDocumentStatistics(action, scope, period, range, context);
    }

    @Override
    public List<DocumentStats> getBackLinkStatistics(String domain, Scope scope, Period period, Range range,
            XWikiContext context) {
        return this.statsReader.getBackLinkStatistics(domain, scope, period, range, context);
    }

    @Override
    public List<RefererStats> getRefererStatistics(String domain, Scope scope, Period period, Range range,
            XWikiContext context) {
        return this.statsReader.getRefererStatistics(domain, scope, period, range, context);
    }

    @Override
    public List<VisitStats> getVisitStatistics(String action, Period period, Range range, XWikiContext context) {
        return this.statsReader.getVisitStatistics(action, period, range, context);
    }

    // ////////////////////////////////////////////////////////////////////////////////////////
    // Deprecated methods
    // ////////////////////////////////////////////////////////////////////////////////////////

    @Override
    public DocumentStats getDocTotalStats(String docname, String action, XWikiContext context) {
        return new DocumentStats();
    }

    @Override
    public DocumentStats getDocMonthStats(String docname, String action, Date month, XWikiContext context) {
        return this.statsReader.getDocMonthStats(docname, action, month, context);
    }

    @Override
    public DocumentStats getDocDayStats(String docname, String action, Date day, XWikiContext context) {
        return new DocumentStats();
    }

    @Override
    public List<?> getRefMonthStats(String docName, Date month, XWikiContext context) throws XWikiException {
        return this.statsReader.getRefMonthStats(docName, month, context);
    }
}