com.adobe.acs.samples.events.cq5.impl.SampleSlingEventHandler.java Source code

Java tutorial

Introduction

Here is the source code for com.adobe.acs.samples.events.cq5.impl.SampleSlingEventHandler.java

Source

/*
 * #%L
 * ACS AEM Samples
 * %%
 * Copyright (C) 2015 Adobe
 * %%
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */

package com.adobe.acs.samples.events.cq5.impl;

import com.day.cq.replication.ReplicationAction;
import org.apache.commons.lang.ArrayUtils;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.Properties;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.Reference;
import org.apache.felix.scr.annotations.Service;
import org.apache.sling.api.SlingConstants;
import org.apache.sling.discovery.TopologyEvent;
import org.apache.sling.discovery.TopologyEventListener;
import org.apache.sling.event.EventUtil;
import org.apache.sling.event.jobs.JobProcessor;
import org.apache.sling.event.jobs.JobUtil;
import org.osgi.service.event.Event;
import org.osgi.service.event.EventAdmin;
import org.osgi.service.event.EventConstants;
import org.osgi.service.event.EventHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Component(label = "ACS AEM Samples - CQ5 Sling Event Handler", description = "Deprecated in AEM6+; Sample implementation of a Custom Event Listener based on Sling",

        // One of the few cases where immediate = true; this is so the Event Listener starts listening immediately
        immediate = true)
@Properties({ @Property(label = "Event Topics", value = { "samples/events/poked",
        "samples/events/*" }, description = "[Required] Event Topics this event handler will to respond to.", name = EventConstants.EVENT_TOPIC, propertyPrivate = true),
        @Property(label = "Event Filters", value = "(&(" + ReplicationAction.PROPERTY_TYPE
                + "=DEACTIVATE))", description = "[Optional] Event Filters used to further restrict this event handler; Uses LDAP expression against event properties.", name = EventConstants.EVENT_FILTER, propertyPrivate = true) })
@Service
public class SampleSlingEventHandler implements JobProcessor, EventHandler, TopologyEventListener {
    private static final Logger log = LoggerFactory.getLogger(SampleSlingEventHandler.class);

    // EventAdmin is used to manually trigger other events
    @Reference
    private EventAdmin eventAdmin;

    private boolean isLeader;

    @Override
    public void handleEvent(final Event event) {
        boolean handleLocally = false;
        boolean handleWithLeader = !handleLocally;

        for (final String name : event.getPropertyNames()) {
            log.debug("{} : {}", name, event.getProperty(name));
        }

        if (!ArrayUtils.contains(event.getPropertyNames(), EventUtil.PROPERTY_DISTRIBUTE)) {
            // This is the check for a distributed event or not; if this property does not exist, it usually
            // means that this event handler should process the job, as no other event handlers
            // will see this event.

            JobUtil.processJob(event, this);

        } else if (handleLocally && EventUtil.isLocal(event)) {
            // This is a distributed event (first 'if' condition failed)

            // If this server created the event
            // then only this server should process the event

            // This will call this's process(..) method, passing in the event obj
            // JobUtil.processJob(..) sends/checks for an ack for this job

            // Jobs guarantee the event will be processed (though doesn't guarantee the job will be
            // processed SUCCESSFULLY)
            JobUtil.processJob(event, this);

        } else if (handleWithLeader && this.isLeader) {
            // This is a distributed event (first 'if' condition failed)

            // If a event is distributed, you may only want to execute it the Master node in
            // the cluster.

            JobUtil.processJob(event, this);
        } else {
            // DO NOTHING!
        }
    }

    @Override
    public boolean process(final Event event) {

        // This is the Job's process method where the work will be
        // Node that Jobs status is persisted in the JCR under /var/eventing so the management
        // of Jobs is NOT a wholly "in-memory" operations.

        // If you have guaranteed VERY FAST processing; it can be pushed to "handleEvent(...)"

        /**
         * Sling Event Properties - VERY handy
         *
         * This aren't guaranteed to have non-null values; so check before using.
         */

        // Resource path "undergoing" the event
        event.getProperty(SlingConstants.PROPERTY_PATH);

        // Resource type
        event.getProperty(SlingConstants.PROPERTY_RESOURCE_TYPE);

        // Resource super type
        event.getProperty(SlingConstants.PROPERTY_RESOURCE_SUPER_TYPE);

        // Properties names that were added/changes/removed
        event.getProperty(SlingConstants.PROPERTY_ADDED_ATTRIBUTES);
        event.getProperty(SlingConstants.PROPERTY_CHANGED_ATTRIBUTES);
        event.getProperty(SlingConstants.PROPERTY_REMOVED_ATTRIBUTES);

        // User id
        event.getProperty(SlingConstants.PROPERTY_USERID);

        /**
         * Event Properties
         */

        // Specifies application node
        event.getProperty(EventUtil.PROPERTY_APPLICATION);

        // Specifies if the event should be distributed in the cluster (defaults to false)
        event.getProperty(EventUtil.PROPERTY_DISTRIBUTE);

        // Timed Event properties
        // Unique event id for Timed event
        event.getProperty(EventUtil.PROPERTY_TIMED_EVENT_ID);
        event.getProperty(EventUtil.PROPERTY_TIMED_EVENT_DATE);
        event.getProperty(EventUtil.PROPERTY_TIMED_EVENT_PERIOD);
        event.getProperty(EventUtil.PROPERTY_TIMED_EVENT_SCHEDULE);
        event.getProperty(EventUtil.PROPERTY_TIMED_EVENT_TOPIC);

        /**
         * Available for Events that are processed as Jobs
         */
        if (JobUtil.isJobEvent(event)) {
            event.getProperty(JobUtil.JOB_ID);
            event.getProperty(JobUtil.PROPERTY_JOB_NAME);
            event.getProperty(JobUtil.PROPERTY_JOB_QUEUE_NAME);
            event.getProperty(JobUtil.PROPERTY_JOB_CREATED);
            event.getProperty(JobUtil.PROPERTY_JOB_PARALLEL);
            event.getProperty(JobUtil.PROPERTY_JOB_PRIORITY);
            event.getProperty(JobUtil.PROPERTY_JOB_QUEUE_ORDERED);
            event.getProperty(JobUtil.PROPERTY_JOB_RETRIES);
            event.getProperty(JobUtil.PROPERTY_JOB_RETRY_COUNT);
            event.getProperty(JobUtil.PROPERTY_JOB_RETRY_DELAY);
            event.getProperty(JobUtil.PROPERTY_JOB_RUN_LOCAL);
            event.getProperty(JobUtil.PROPERTY_JOB_TOPIC);
            event.getProperty(JobUtil.PROPERTY_NOTIFICATION_JOB);
        }

        // Only return false if job processing failed and the job should be rescheduled
        return true;
    }

    // This method makes this Listener cluster aware so it only responds to events on the Leader,
    // Without this every node in the cluster will process the event (generally resulting w duplicate work)
    @Override
    public void handleTopologyEvent(final TopologyEvent event) {
        if (event.getType() == TopologyEvent.Type.TOPOLOGY_CHANGED
                || event.getType() == TopologyEvent.Type.TOPOLOGY_INIT) {
            this.isLeader = event.getNewView().getLocalInstance().isLeader();
        }
    }
}