Package voldemort.xml

Source Code of voldemort.xml.ClusterMapper

/*
* Copyright 2008-2009 LinkedIn, Inc
*
* 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.
*/

package voldemort.xml;

import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;
import java.util.ArrayList;
import java.util.List;

import javax.xml.XMLConstants;
import javax.xml.transform.Source;
import javax.xml.transform.stream.StreamSource;
import javax.xml.validation.Schema;
import javax.xml.validation.SchemaFactory;
import javax.xml.validation.Validator;

import org.apache.commons.lang.StringUtils;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.JDOMException;
import org.jdom.input.SAXBuilder;
import org.jdom.output.Format;
import org.jdom.output.XMLOutputter;
import org.jdom.transform.JDOMSource;
import org.xml.sax.SAXException;

import voldemort.cluster.Cluster;
import voldemort.cluster.Node;
import voldemort.cluster.Zone;
import voldemort.utils.Utils;

/**
* Parse a cluster xml file
*
*
*/
public class ClusterMapper {

    private static final String SERVER_ID_ELMT = "id";
    private static final String SERVER_PARTITIONS_ELMT = "partitions";
    private static final String ZONE_ELMT = "zone";
    private static final String ZONE_ID_ELMT = "zone-id";
    private static final String ZONE_PROXIMITY_LIST_ELMT = "proximity-list";
    private static final String SERVER_ELMT = "server";
    private static final String CLUSTER_NAME_ELMT = "name";
    private static final String CLUSTER_ELMT = "cluster";
    private static final String HOST_ELMT = "host";
    private static final String HTTP_PORT_ELMT = "http-port";
    private static final String SOCKET_PORT_ELMT = "socket-port";
    private static final String ADMIN_PORT_ELMT = "admin-port";
    private static final String REST_PORT_ELMT = "rest-port";

    public static final Integer MAX_PARTITIONID = 65535;

    private final Schema schema;

    public ClusterMapper() {
        try {
            SchemaFactory factory = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
            Source source = new StreamSource(ClusterMapper.class.getResourceAsStream("cluster.xsd"));
            this.schema = factory.newSchema(source);
        } catch(SAXException e) {
            throw new MappingException(e);
        }
    }

    public Cluster readCluster(File f) throws IOException {
        FileReader reader = null;
        try {
            reader = new FileReader(f);
            return readCluster(reader);
        } finally {
            if(reader != null)
                reader.close();
        }
    }

    public Cluster readCluster(Reader input) {
        return readCluster(input, true);
    }

    @SuppressWarnings("unchecked")
    public Cluster readCluster(Reader input, boolean verifySchema) {
        try {
            SAXBuilder builder = new SAXBuilder(false);
            Document doc = builder.build(input);
            if(verifySchema) {
                Validator validator = this.schema.newValidator();
                validator.validate(new JDOMSource(doc));
            }
            Element root = doc.getRootElement();
            if(!root.getName().equals(CLUSTER_ELMT))
                throw new MappingException("Invalid root element: "
                                           + doc.getRootElement().getName());
            String name = root.getChildText(CLUSTER_NAME_ELMT);

            List<Zone> zones = new ArrayList<Zone>();
            for(Element node: (List<Element>) root.getChildren(ZONE_ELMT))
                zones.add(readZone(node));

            List<Node> servers = new ArrayList<Node>();
            for(Element node: (List<Element>) root.getChildren(SERVER_ELMT))
                servers.add(readServer(node));
            return new Cluster(name, servers, zones);
        } catch(JDOMException e) {
            throw new MappingException(e);
        } catch(SAXException e) {
            throw new MappingException(e);
        } catch(IOException e) {
            throw new MappingException(e);
        }
    }

    private Zone readZone(Element zone) {
        int zoneId = Integer.parseInt(zone.getChildText(ZONE_ID_ELMT));
        String proximitListStr = zone.getChildText(ZONE_PROXIMITY_LIST_ELMT).trim();
        List<Integer> proximityList = new ArrayList<Integer>();
        for(String node: Utils.COMMA_SEP.split(proximitListStr))
            if(node.trim().length() > 0) {
                int proximityZoneId = Integer.parseInt(node.trim());
                if(proximityZoneId == zoneId)
                    continue;
                // Not using a set here, as the ordering of the zone ids are
                // important.
                if(proximityList.contains(proximityZoneId) == false) {
                    proximityList.add(proximityZoneId);
                }
            }
        return new Zone(zoneId, proximityList);
    }

    public Node readServer(Element server) throws SAXException {
        int id = Integer.parseInt(server.getChildText(SERVER_ID_ELMT));
        String host = server.getChildText(HOST_ELMT);
        int httpPort = Integer.parseInt(server.getChildText(HTTP_PORT_ELMT));
        int socketPort = Integer.parseInt(server.getChildText(SOCKET_PORT_ELMT));

        // read admin-port default to -1 if not available
        int adminPort = (null != server.getChildText(ADMIN_PORT_ELMT)) ? Integer.parseInt(server.getChildText(ADMIN_PORT_ELMT))
                                                                      : -1;
        // read rest-port . Default to -1 if not available
        int restPort = (null != server.getChildText(REST_PORT_ELMT)) ? Integer.parseInt(server.getChildText(REST_PORT_ELMT))
                                                                    : -1;
        int zoneId = (null != server.getChildText(ZONE_ID_ELMT)) ? Integer.parseInt(server.getChildText(ZONE_ID_ELMT))
                                                                : Zone.DEFAULT_ZONE_ID;
        String partitionsText = server.getChildText(SERVER_PARTITIONS_ELMT).trim();
        List<Integer> partitions = new ArrayList<Integer>();
        for(String aPartition: Utils.COMMA_SEP.split(partitionsText))
            if(aPartition.trim().length() > 0) {
                Integer partition = Integer.parseInt(aPartition.trim());
                if(partition > MAX_PARTITIONID) {
                    throw new SAXException("Partition id cannot be greater than " + MAX_PARTITIONID);
                }
                partitions.add(partition);
            }

        if(restPort == -1) {
            return new Node(id, host, httpPort, socketPort, adminPort, zoneId, partitions);
        } else {
            return new Node(id, host, httpPort, socketPort, adminPort, zoneId, partitions, restPort);
        }
    }

    public String writeCluster(Cluster cluster) {
        Document doc = new Document(new Element(CLUSTER_ELMT));
        doc.getRootElement().addContent(new Element(CLUSTER_NAME_ELMT).setText(cluster.getName()));
        boolean displayZones = cluster.getZones().size() > 1;
        if(displayZones) {
            for(Zone n: cluster.getZones())
                doc.getRootElement().addContent(mapZone(n));
        }
        for(Node n: cluster.getNodes())
            doc.getRootElement().addContent(mapServer(n, displayZones));
        XMLOutputter serializer = new XMLOutputter(Format.getPrettyFormat());
        return serializer.outputString(doc.getRootElement());
    }

    private Element mapZone(Zone zone) {
        Element zoneElement = new Element(ZONE_ELMT);
        zoneElement.addContent(new Element(ZONE_ID_ELMT).setText(Integer.toString(zone.getId())));
        String proximityListTest = StringUtils.join(zone.getProximityList().toArray(), ", ");
        zoneElement.addContent(new Element(ZONE_PROXIMITY_LIST_ELMT).setText(proximityListTest));
        return zoneElement;
    }

    private Element mapServer(Node node, boolean displayZones) {
        Element server = new Element(SERVER_ELMT);
        server.addContent(new Element(SERVER_ID_ELMT).setText(Integer.toString(node.getId())));
        server.addContent(new Element(HOST_ELMT).setText(node.getHost()));
        server.addContent(new Element(HTTP_PORT_ELMT).setText(Integer.toString(node.getHttpPort())));
        server.addContent(new Element(SOCKET_PORT_ELMT).setText(Integer.toString(node.getSocketPort())));
        server.addContent(new Element(ADMIN_PORT_ELMT).setText(Integer.toString(node.getAdminPort())));
        String serverPartitionsText = StringUtils.join(node.getPartitionIds().toArray(), ", ");
        server.addContent(new Element(SERVER_PARTITIONS_ELMT).setText(serverPartitionsText));
        if(displayZones)
            server.addContent(new Element(ZONE_ID_ELMT).setText(Integer.toString(node.getZoneId())));
        return server;
    }
}
TOP

Related Classes of voldemort.xml.ClusterMapper

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.