All Downloads are FREE. Search and download functionalities are using the official Maven repository.

bboss.org.jgroups.protocols.FILE_PING Maven / Gradle / Ivy

package bboss.org.jgroups.protocols;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;

import bboss.org.jgroups.Address;
import bboss.org.jgroups.Event;
import bboss.org.jgroups.Message;
import bboss.org.jgroups.PhysicalAddress;
import bboss.org.jgroups.View;
import bboss.org.jgroups.annotations.Experimental;
import bboss.org.jgroups.annotations.Property;
import bboss.org.jgroups.util.Promise;
import bboss.org.jgroups.util.UUID;
import bboss.org.jgroups.util.Util;


/**
 * Simple discovery protocol which uses a file on shared storage such as an SMB share, NFS mount or S3. The local
 * address information, e.g. UUID and physical addresses mappings are written to the file and the content is read and
 * added to our transport's UUID-PhysicalAddress cache.

* The design is at doc/design/FILE_PING.txt * @author Bela Ban * @version $Id: FILE_PING.java,v 1.19 2010/06/17 14:57:15 belaban Exp $ */ @Experimental public class FILE_PING extends Discovery { protected static final String SUFFIX=".node"; /* ----------------------------------------- Properties -------------------------------------------------- */ @Property(description="The absolute path of the shared file") protected String location=File.separator + "tmp" + File.separator + "jgroups"; @Property(description="Interval (in milliseconds) at which the own address is written to the file system. 0 disables it.") protected long interval=60000; /* --------------------------------------------- Fields ------------------------------------------------------ */ protected File root_dir=null; protected FilenameFilter filter; private ScheduledFuture writer_future; public void init() throws Exception { super.init(); createRootDir(); } public void start() throws Exception { super.start(); if(interval > 0) writer_future=timer.scheduleWithFixedDelay(new WriterTask(), interval, interval, TimeUnit.MILLISECONDS); } public void stop() { if(writer_future != null) { writer_future.cancel(false); writer_future=null; } super.stop(); } public void sendGetMembersRequest(String cluster_name, Promise promise, boolean return_views_only) throws Exception{ List existing_mbrs=readAll(cluster_name); PhysicalAddress physical_addr=(PhysicalAddress)down(new Event(Event.GET_PHYSICAL_ADDRESS, local_addr)); List physical_addrs=Arrays.asList(physical_addr); PingData data=new PingData(local_addr, null, false, UUID.get(local_addr), physical_addrs); // If we don't find any files, return immediately if(existing_mbrs.isEmpty()) { if(promise != null) { promise.setResult(null); } } else { // 1. Send GET_MBRS_REQ message to members listed in the file for(PingData tmp: existing_mbrs) { Collection dests=tmp != null? tmp.getPhysicalAddrs() : null; if(dests == null) continue; for(final PhysicalAddress dest: dests) { if(dest == null || dest.equals(physical_addr)) continue; PingHeader hdr=new PingHeader(PingHeader.GET_MBRS_REQ, data, cluster_name); hdr.return_view_only=return_views_only; final Message msg=new Message(dest); msg.setFlag(Message.OOB); msg.putHeader(this.id, hdr); // needs to be getName(), so we might get "MPING" ! // down_prot.down(new Event(Event.MSG, msg)); if(log.isTraceEnabled()) log.trace("[FIND_INITIAL_MBRS] sending PING request to " + msg.getDest()); timer.execute(new Runnable() { public void run() { try { down_prot.down(new Event(Event.MSG, msg)); } catch(Exception ex){ if(log.isErrorEnabled()) log.error("failed sending discovery request to " + dest, ex); } } }); } } } // Write my own data to file writeToFile(data, cluster_name); } public Object down(Event evt) { Object retval=super.down(evt); if(evt.getType() == Event.VIEW_CHANGE) handleView((View)evt.getArg()); return retval; } protected void createRootDir() { root_dir=new File(location); if(root_dir.exists()) { if(!root_dir.isDirectory()) throw new IllegalArgumentException("location " + root_dir.getPath() + " is not a directory"); } else { root_dir.mkdirs(); } if(!root_dir.exists()) throw new IllegalArgumentException("location " + root_dir.getPath() + " could not be accessed"); filter=new FilenameFilter() { public boolean accept(File dir, String name) { return name.endsWith(SUFFIX); } }; } // remove all files which are not from the current members protected void handleView(View view) { Collection

mbrs=view.getMembers(); boolean is_coordinator=!mbrs.isEmpty() && mbrs.iterator().next().equals(local_addr); if(is_coordinator) { List data=readAll(group_addr); for(PingData entry: data) { Address addr=entry.getAddress(); if(addr != null && !mbrs.contains(addr)) { remove(group_addr, addr); } } } } protected void remove(String clustername, Address addr) { if(clustername == null || addr == null) return; File dir=new File(root_dir, clustername); if(!dir.exists()) return; try { String filename=addr instanceof UUID? ((UUID)addr).toStringLong() : addr.toString(); File file=new File(dir, filename + SUFFIX); if(log.isTraceEnabled()) log.trace("removing " + file); file.delete(); } catch(Throwable e) { log.error("failure removing data", e); } } /** * Reads all information from the given directory under clustername * @return */ protected List readAll(String clustername) { List retval=new ArrayList(); File dir=new File(root_dir, clustername); if(!dir.exists()) dir.mkdir(); File[] files=dir.listFiles(filter); if(files != null) { for(File file: files) retval.add(readFile(file)); } return retval; } protected static PingData readFile(File file) { PingData retval=null; DataInputStream in=null; try { in=new DataInputStream(new FileInputStream(file)); PingData tmp=new PingData(); tmp.readFrom(in); return tmp; } catch(Exception e) { } finally { Util.close(in); } return retval; } protected void writeToFile(PingData data, String clustername) { DataOutputStream out=null; File dir=new File(root_dir, clustername); if(!dir.exists()) dir.mkdir(); String filename=local_addr instanceof UUID? ((UUID)local_addr).toStringLong() : local_addr.toString(); File file=new File(dir, filename + SUFFIX); file.deleteOnExit(); try { out=new DataOutputStream(new FileOutputStream(file)); data.writeTo(out); } catch(Exception e) { } finally { Util.close(out); } } protected class WriterTask implements Runnable { public void run() { PhysicalAddress physical_addr=(PhysicalAddress)down(new Event(Event.GET_PHYSICAL_ADDRESS, local_addr)); List physical_addrs=Arrays.asList(physical_addr); PingData data=new PingData(local_addr, null, false, UUID.get(local_addr), physical_addrs); writeToFile(data, group_addr); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy