Package org.exoplatform.services.jcr.ext.initializer

Source Code of org.exoplatform.services.jcr.ext.initializer.RemoteWorkspaceInitializationService$WorkspaceDataPublisher

/*
* Copyright (C) 2009 eXo Platform SAS.
*
* 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 org.exoplatform.services.jcr.ext.initializer;

import org.exoplatform.container.xml.InitParams;
import org.exoplatform.container.xml.PropertiesParam;
import org.exoplatform.services.jcr.RepositoryService;
import org.exoplatform.services.jcr.config.RepositoryConfigurationException;
import org.exoplatform.services.jcr.ext.app.ThreadLocalSessionProviderService;
import org.exoplatform.services.jcr.ext.backup.BackupChain;
import org.exoplatform.services.jcr.ext.backup.BackupConfig;
import org.exoplatform.services.jcr.ext.backup.BackupJob;
import org.exoplatform.services.jcr.ext.backup.BackupManager;
import org.exoplatform.services.jcr.ext.backup.BackupOperationException;
import org.exoplatform.services.jcr.ext.initializer.impl.RemoteTransportImpl;
import org.exoplatform.services.jcr.ext.replication.transport.ChannelManager;
import org.exoplatform.services.jcr.util.IdGenerator;
import org.exoplatform.services.log.ExoLogger;
import org.exoplatform.services.log.Log;
import org.exoplatform.services.rest.resource.ResourceContainer;

import java.io.File;

import javax.jcr.LoginException;
import javax.jcr.NoSuchWorkspaceException;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
import javax.ws.rs.Produces;
import javax.ws.rs.core.Response;

/**
* Created by The eXo Platform SAS.
*
* <br/>
* Date: 16.03.2009
*
* @author <a href="mailto:alex.reshetnyak@exoplatform.com.ua">Alex Reshetnyak</a>
* @version $Id: RemoteWorkspaceInitializerService.java 111 2008-11-11 11:11:11Z rainf0x $
*/

@Path("/jcr-remote-workspace-initializer/")
@Produces("text/plain")
public class RemoteWorkspaceInitializationService implements ResourceContainer
{

   /**
    * The template for ip-address in configuration.
    */
   private static final String IP_ADRESS_TEMPLATE = "[$]bind-ip-address";

   /**
    * Definition the constants to ReplicationTestService.
    */
   public static final class Constants
   {
      /**
       * The base path to this service.
       */
      public static final String BASE_URL = "/rest/jcr-remote-workspace-initializer";

      /**
       * Definition the operation types.
       */
      public static final class OperationType
      {
         /**
          * Full backup only operation.
          */
         public static final String GET_WORKSPACE = "getWorkspaceData";

         /**
          * OperationType constructor.
          */
         private OperationType()
         {
         }
      }

      /**
       * Constants constructor.
       */
      private Constants()
      {
      }
   }

   /**
    * The apache logger.
    */
   private static Log log = ExoLogger.getLogger("exo.jcr.component.ext.RemoteWorkspaceInitializationService");

   /**
    * The repository service.
    */
   private final RepositoryService repositoryService;

   /**
    * The backup manager.
    */
   private final BackupManager backupManager;

   /**
    * Will be get session over base authenticate.
    */
   private final ThreadLocalSessionProviderService sessionProviderService;

   /**
    * Url to remote data source.
    */
   private final String dataSourceUrl;

   /**
    * Bind to IP address.
    */
   private final String bindIpAddress;

   /**
    * The channel configuration.
    */
   private final String channelConfig;

   /**
    * The channel name prefix.
    */
   private final String channelName;

   /**
    * The folder for temporary files.
    */
   private final File tempDir;

   /**
    * ReplicationTestService constructor.
    *
    * @param repoService
    *          the RepositoryService
    * @param backupManager
    *          the BackupManager
    * @param sessionProviderService
    *          the ThreadLocalSessionProviderService
    * @param params
    *          the InitParams from configuration
    */
   public RemoteWorkspaceInitializationService(RepositoryService repoService, BackupManager backupManager,
      ThreadLocalSessionProviderService sessionProviderService, InitParams params)
   {
      this.repositoryService = repoService;
      this.backupManager = backupManager;
      this.sessionProviderService = sessionProviderService;

      PropertiesParam pps = params.getPropertiesParam("remote-initializer-properties");

      if (pps == null)
         throw new RuntimeException("remote-initializer-properties not specified");

      if (pps.getProperty("remote-source-url") == null)
         throw new RuntimeException("remote-source-url not specified");
      dataSourceUrl = pps.getProperty("remote-source-url");

      if (pps.getProperty("bind-ip-address") == null)
         throw new RuntimeException("bind-ip-address not specified");
      bindIpAddress = pps.getProperty("bind-ip-address");

      if (pps.getProperty("channel-config") == null)
         throw new RuntimeException("channel-config not specified");
      channelConfig = pps.getProperty("channel-config");

      if (pps.getProperty("channel-name") == null)
         throw new RuntimeException("channel-name not specified");
      channelName = pps.getProperty("channel-name");

      if (pps.getProperty("temp-dir") == null)
         throw new RuntimeException("temp-dir not specified");
      String tempD = pps.getProperty("temp-dir");

      tempDir = new File(tempD);
      if (!tempDir.exists())
         tempDir.mkdirs();

      log.info("RemoteWorkspaceInitializerService");
   }

   /**
    * getWorkspaceData.
    *
    * @param repository
    *          the repository name
    * @param workspace
    *          the workspace name
    * @return File with workspace data
    * @throws RemoteWorkspaceInitializationException
    *           will be generated the RemoteWorkspaceInitializerException
    */
   public File getWorkspaceData(String repository, String workspace) throws RemoteWorkspaceInitializationException
   {

      String id = IdGenerator.generate();

      ChannelManager channelManager =
         new ChannelManager(channelConfig.replaceAll(IP_ADRESS_TEMPLATE, bindIpAddress), channelName + "_" + repository
            + "_" + workspace + "_" + id, 2);

      RemoteTransport remoteTransport = new RemoteTransportImpl(channelManager, tempDir, dataSourceUrl);

      try
      {
         remoteTransport.init();

         return remoteTransport.getWorkspaceData(repository, workspace, id);
      }
      catch (Throwable t)
      {
         throw new RemoteWorkspaceInitializationException("Can't get workspace data", t);
      }
      finally
      {
         remoteTransport.close();
      }
   }

   /**
    * startFullBackup.
    *
    * @param repositoryName
    *          the repository name
    * @param workspaceName
    *          the workspace name
    * @param id
    *          the unique identifier for channel
    * @return Response return the response
    */
   @GET
   @Path("/{repositoryName}/{workspaceName}/{id}/getWorkspaceData")
   public Response startFullBackup(@PathParam("repositoryName") String repositoryName,
      @PathParam("workspaceName") String workspaceName, @PathParam("id") String id)
   {
      String result = "OK";

      // init transport
      ChannelManager channelManager =
         new ChannelManager(channelConfig.replaceAll(IP_ADRESS_TEMPLATE, bindIpAddress), channelName + "_"
            + repositoryName + "_" + workspaceName + "_" + id, 2);
      RemoteTransport remoteTransport = new RemoteTransportImpl(channelManager, tempDir, dataSourceUrl);

      try
      {
         remoteTransport.init();

         // start backup
         BackupConfig config = new BackupConfig();
         config.setBackupType(BackupManager.FULL_BACKUP_ONLY);
         config.setRepository(repositoryName);
         config.setWorkspace(workspaceName);
         config.setBackupDir(backupManager.getBackupDirectory());

         try
         {
            validateRepositoryName(repositoryName);
            validateWorkspaceName(repositoryName, workspaceName);
            BackupChain backupChain = backupManager.startBackup(config);

            WorkspaceDataPublisher publisher = new WorkspaceDataPublisher(backupChain, remoteTransport);
            publisher.start();

         }
         catch (Exception e)
         {
            result = "FAIL\n" + e.getMessage();
            log.error("Can't start backup", e);

            remoteTransport.close();
         }

      }
      catch (RemoteWorkspaceInitializationException e1)
      {
         result = "FAIL\n" + e1.getMessage();
         log.error("Can't initialization transport", e1);
      }

      return Response.ok(result).build();
   }

   /**
    * validateRepositoryName.
    *
    * @param repositoryName
    *          the repository name
    */
   private void validateRepositoryName(String repositoryName)
   {
      try
      {
         repositoryService.getRepository(repositoryName);
      }
      catch (RepositoryException e)
      {
         throw new RuntimeException("Can not get repository '" + repositoryName + "'", e);
      }
      catch (RepositoryConfigurationException e)
      {
         throw new RuntimeException("Can not get repository '" + repositoryName + "'", e);
      }
   }

   /**
    * validateWorkspaceName.
    *
    * @param repositoryName
    *          the repository name
    * @param workspaceName
    *          the workspace name
    */
   private void validateWorkspaceName(String repositoryName, String workspaceName)
   {
      try
      {
         Session ses =
            sessionProviderService.getSessionProvider(null).getSession(workspaceName,
               repositoryService.getRepository(repositoryName));
         ses.logout();
      }
      catch (LoginException e)
      {
         throw new RuntimeException("Can not loogin to workspace '" + workspaceName + "'", e);
      }
      catch (NoSuchWorkspaceException e)
      {
         throw new RuntimeException("Can not get workspace '" + workspaceName + "'", e);
      }
      catch (RepositoryException e)
      {
         throw new RuntimeException("Can not get workspace '" + workspaceName + "'", e);
      }
      catch (RepositoryConfigurationException e)
      {
         throw new RuntimeException("Can not get workspace '" + workspaceName + "'", e);
      }
   }

   /**
    * The WorkspaceDataPublisher will be published the workspace data.
    *
    */
   class WorkspaceDataPublisher extends Thread
   {

      /**
       * BACKUP_WAIT_INTERVAL. the constants for backup wait interval.
       */
      private static final int BACKUP_WAIT_INTERVAL = 50;

      /**
       * The BackupChain for current backup.
       */
      private BackupChain backupChain;

      /**
       * The RemoteTransport will be send workspace data .
       */
      private RemoteTransport transport;

      /**
       * WorkspaceDataPublisher constructor.
       *
       * @param chain
       *          the BackupChain for current backup
       * @param transport
       *          the RemoteTransport
       */
      public WorkspaceDataPublisher(BackupChain chain, RemoteTransport transport)
      {
         this.backupChain = chain;
         this.transport = transport;
      }

      /**
       * {@inheritDoc}
       */
      public void run()
      {
         try
         {
            try
            {
               // wait till full backup will be stopped
               while (backupChain.getFullBackupState() != BackupJob.FINISHED)
               {
                  Thread.yield();
                  Thread.sleep(BACKUP_WAIT_INTERVAL);
               }

               // get path to file with full backup
               String path = backupChain.getBackupJobs().get(0).getStorageURL().getPath();

               // stop backup
               backupManager.stopBackup(backupChain);

               // send data
               transport.sendWorkspaceData(new File(path));
               Thread.sleep(BACKUP_WAIT_INTERVAL * 20 * 30);
            }
            catch (RemoteWorkspaceInitializationException e)
            {
               try
               {
                  transport.sendError("Can not send the workspace data : " + e.getMessage());
               }
               catch (RemoteWorkspaceInitializationException e1)
               {
                  log.error("Can not send error message : " + e.getMessage(), e);
               }
            }
            catch (BackupOperationException e)
            {
               try
               {
                  transport.sendError("Can not send the workspace data : " + e.getMessage());
               }
               catch (RemoteWorkspaceInitializationException e1)
               {
                  log.error("Can not send error message : " + e.getMessage(), e);
               }
            }
            catch (InterruptedException e)
            {
               try
               {
                  transport.sendError("Can not send the workspace data : " + e.getMessage());
               }
               catch (RemoteWorkspaceInitializationException e1)
               {
                  log.error("Can not send error message : " + e.getMessage(), e);
               }
            }
         }
         catch (NoMemberToSendException e)
         {
            log.error("Can not send the data  : " + e.getMessage(), e);
         }
         finally
         {
            try
            {
               transport.close();
            }
            catch (RemoteWorkspaceInitializationException e)
            {
               log.error("Can not close the transport : " + e.getMessage(), e);
            }
         }
      }
   }

}
TOP

Related Classes of org.exoplatform.services.jcr.ext.initializer.RemoteWorkspaceInitializationService$WorkspaceDataPublisher

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.