voldemort.store.readonly.mr.azkaban.AbstractHadoopJob.java Source code

Java tutorial

Introduction

Here is the source code for voldemort.store.readonly.mr.azkaban.AbstractHadoopJob.java

Source

/*
 * 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.store.readonly.mr.azkaban;

import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Arrays;
import java.util.List;

import org.apache.hadoop.filecache.DistributedCache;
import org.apache.hadoop.fs.FileStatus;
import org.apache.hadoop.fs.FileSystem;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.fs.PathFilter;
import org.apache.hadoop.mapred.Counters;
import org.apache.hadoop.mapred.Counters.Counter;
import org.apache.hadoop.mapred.FileOutputFormat;
import org.apache.hadoop.mapred.JobClient;
import org.apache.hadoop.mapred.JobConf;
import org.apache.hadoop.mapred.Mapper;
import org.apache.hadoop.mapred.Reducer;
import org.apache.hadoop.mapred.RunningJob;
import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;

import voldemort.store.readonly.mr.IdentityJsonReducer;
import voldemort.store.readonly.mr.utils.HadoopUtils;
import azkaban.common.jobs.AbstractJob;
import azkaban.common.utils.Props;

/**
 * An abstract Base class for Hadoop Jobs
 * 
 * @author bbansal
 * 
 */
public abstract class AbstractHadoopJob extends AbstractJob {

    public static String COMMON_FILE_DATE_PATTERN = "yyyy-MM-dd-HH-mm";
    public static final String HADOOP_PREFIX = "hadoop-conf.";
    public static final String LATEST_SUFFIX = "#LATEST";
    public static final String CURRENT_SUFFIX = "#CURRENT";
    private final Props _props;
    private RunningJob _runningJob;

    public AbstractHadoopJob(String name, Props props) {
        super(name);
        this._props = props;
    }

    public void run(JobConf conf) throws Exception {
        _runningJob = new JobClient(conf).submitJob(conf);
        info("See " + _runningJob.getTrackingURL() + " for details.");
        _runningJob.waitForCompletion();

        if (!_runningJob.isSuccessful()) {
            throw new Exception("Hadoop job:" + getId() + " failed!");
        }

        // dump all counters
        Counters counters = _runningJob.getCounters();
        for (String groupName : counters.getGroupNames()) {
            Counters.Group group = counters.getGroup(groupName);
            info("Group: " + group.getDisplayName());
            for (Counter counter : group)
                info(counter.getDisplayName() + ":\t" + counter.getValue());
        }
    }

    public JobConf createJobConf(Class<? extends Mapper> mapperClass) throws IOException, URISyntaxException {
        JobConf conf = createJobConf(mapperClass, IdentityJsonReducer.class);
        conf.setNumReduceTasks(0);

        return conf;
    }

    public JobConf createJobConf(Class<? extends Mapper> mapperClass, Class<? extends Reducer> reducerClass,
            Class<? extends Reducer> combinerClass) throws IOException, URISyntaxException {
        JobConf conf = createJobConf(mapperClass, reducerClass);
        conf.setCombinerClass(combinerClass);

        return conf;
    }

    public JobConf createJobConf(Class<? extends Mapper> mapperClass, Class<? extends Reducer> reducerClass)
            throws IOException, URISyntaxException {
        JobConf conf = new JobConf();
        // set custom class loader with custom find resource strategy.

        conf.setJobName(getId());
        conf.setMapperClass(mapperClass);
        conf.setReducerClass(reducerClass);

        String hadoop_ugi = _props.getString("hadoop.job.ugi", null);
        if (hadoop_ugi != null) {
            conf.set("hadoop.job.ugi", hadoop_ugi);
        }

        if (_props.getBoolean("is.local", false)) {
            conf.set("mapred.job.tracker", "local");
            conf.set("fs.default.name", "file:///");
            conf.set("mapred.local.dir", "/tmp/map-red");

            info("Running locally, no hadoop jar set.");
        } else {
            setClassLoaderAndJar(conf, getClass());
            info("Setting hadoop jar file for class:" + getClass() + "  to " + conf.getJar());
            info("*************************************************************************");
            info("          Running on Real Hadoop Cluster(" + conf.get("mapred.job.tracker") + ")           ");
            info("*************************************************************************");
        }

        // set JVM options if present
        if (_props.containsKey("mapred.child.java.opts")) {
            conf.set("mapred.child.java.opts", _props.getString("mapred.child.java.opts"));
            info("mapred.child.java.opts set to " + _props.getString("mapred.child.java.opts"));
        }

        // set input and output paths if they are present
        if (_props.containsKey("input.paths")) {
            List<String> inputPaths = _props.getStringList("input.paths");
            if (inputPaths.size() == 0)
                throw new IllegalArgumentException("Must specify at least one value for property 'input.paths'");
            for (String path : inputPaths) {
                // Implied stuff, but good implied stuff
                if (path.endsWith(LATEST_SUFFIX)) {
                    FileSystem fs = FileSystem.get(conf);

                    PathFilter filter = new PathFilter() {

                        @Override
                        public boolean accept(Path arg0) {
                            return !arg0.getName().startsWith("_") && !arg0.getName().startsWith(".");
                        }
                    };

                    String latestPath = path.substring(0, path.length() - LATEST_SUFFIX.length());
                    FileStatus[] statuses = fs.listStatus(new Path(latestPath), filter);

                    Arrays.sort(statuses);

                    path = statuses[statuses.length - 1].getPath().toString();
                    System.out.println("Using latest folder: " + path);
                }
                HadoopUtils.addAllSubPaths(conf, new Path(path));
            }
        }

        if (_props.containsKey("output.path")) {
            String location = _props.get("output.path");
            if (location.endsWith("#CURRENT")) {
                DateTimeFormatter format = DateTimeFormat.forPattern(COMMON_FILE_DATE_PATTERN);
                String destPath = format.print(new DateTime());
                location = location.substring(0, location.length() - "#CURRENT".length()) + destPath;
                System.out.println("Store location set to " + location);
            }

            FileOutputFormat.setOutputPath(conf, new Path(location));
            // For testing purpose only remove output file if exists
            if (_props.getBoolean("force.output.overwrite", false)) {
                FileSystem fs = FileOutputFormat.getOutputPath(conf).getFileSystem(conf);
                fs.delete(FileOutputFormat.getOutputPath(conf), true);
            }
        }

        // Adds External jars to hadoop classpath
        String externalJarList = _props.getString("hadoop.external.jarFiles", null);
        if (externalJarList != null) {
            String[] jarFiles = externalJarList.split(",");
            for (String jarFile : jarFiles) {
                info("Adding extenral jar File:" + jarFile);
                DistributedCache.addFileToClassPath(new Path(jarFile), conf);
            }
        }

        // Adds distributed cache files
        String cacheFileList = _props.getString("hadoop.cache.files", null);
        if (cacheFileList != null) {
            String[] cacheFiles = cacheFileList.split(",");
            for (String cacheFile : cacheFiles) {
                info("Adding Distributed Cache File:" + cacheFile);
                DistributedCache.addCacheFile(new URI(cacheFile), conf);
            }
        }

        // Adds distributed cache files
        String archiveFileList = _props.getString("hadoop.cache.archives", null);
        if (archiveFileList != null) {
            String[] archiveFiles = archiveFileList.split(",");
            for (String archiveFile : archiveFiles) {
                info("Adding Distributed Cache Archive File:" + archiveFile);
                DistributedCache.addCacheArchive(new URI(archiveFile), conf);
            }
        }

        String hadoopCacheJarDir = _props.getString("hdfs.default.classpath.dir", null);
        if (hadoopCacheJarDir != null) {
            FileSystem fs = FileSystem.get(conf);
            if (fs != null) {
                FileStatus[] status = fs.listStatus(new Path(hadoopCacheJarDir));

                if (status != null) {
                    for (int i = 0; i < status.length; ++i) {
                        if (!status[i].isDir()) {
                            Path path = new Path(hadoopCacheJarDir, status[i].getPath().getName());
                            info("Adding Jar to Distributed Cache Archive File:" + path);

                            DistributedCache.addFileToClassPath(path, conf);
                        }
                    }
                } else {
                    info("hdfs.default.classpath.dir " + hadoopCacheJarDir + " is empty.");
                }
            } else {
                info("hdfs.default.classpath.dir " + hadoopCacheJarDir + " filesystem doesn't exist");
            }
        }

        // May want to add this to HadoopUtils, but will await refactoring
        for (String key : getProps().keySet()) {
            String lowerCase = key.toLowerCase();
            if (lowerCase.startsWith(HADOOP_PREFIX)) {
                String newKey = key.substring(HADOOP_PREFIX.length());
                conf.set(newKey, getProps().get(key));
            }
        }

        HadoopUtils.setPropsInJob(conf, getProps());
        return conf;
    }

    public Props getProps() {
        return this._props;
    }

    public void cancel() throws Exception {
        if (_runningJob != null)
            _runningJob.killJob();
    }

    public double getProgress() throws IOException {
        if (_runningJob == null)
            return 0.0;
        else
            return (double) (_runningJob.mapProgress() + _runningJob.reduceProgress()) / 2.0d;
    }

    public Counters getCounters() throws IOException {
        return _runningJob.getCounters();
    }

    public static void setClassLoaderAndJar(JobConf conf, Class jobClass) {
        conf.setClassLoader(Thread.currentThread().getContextClassLoader());
        String jar = HadoopUtils.findContainingJar(jobClass, Thread.currentThread().getContextClassLoader());
        if (jar != null) {
            conf.setJar(jar);
        }
    }
}