org.apache.druid.indexing.common.tasklogs.FileTaskLogsTest.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.druid.indexing.common.tasklogs.FileTaskLogsTest.java

Source

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.druid.indexing.common.tasklogs;

import com.google.common.collect.ImmutableMap;
import com.google.common.io.ByteStreams;
import com.google.common.io.Files;
import org.apache.commons.io.FileUtils;
import org.apache.druid.indexing.common.config.FileTaskLogsConfig;
import org.apache.druid.java.util.common.StringUtils;
import org.apache.druid.tasklogs.TaskLogs;
import org.junit.Assert;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.junit.rules.TemporaryFolder;

import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Map;

public class FileTaskLogsTest {

    @Rule
    public TemporaryFolder temporaryFolder = new TemporaryFolder();

    @Rule
    public ExpectedException expectedException = ExpectedException.none();

    @Test
    public void testSimple() throws Exception {
        final File tmpDir = temporaryFolder.newFolder();
        try {
            final File logDir = new File(tmpDir, "druid/logs");
            final File logFile = new File(tmpDir, "log");
            Files.write("blah", logFile, StandardCharsets.UTF_8);
            final TaskLogs taskLogs = new FileTaskLogs(new FileTaskLogsConfig(logDir));
            taskLogs.pushTaskLog("foo", logFile);

            final Map<Long, String> expected = ImmutableMap.of(0L, "blah", 1L, "lah", -2L, "ah", -5L, "blah");
            for (Map.Entry<Long, String> entry : expected.entrySet()) {
                final byte[] bytes = ByteStreams
                        .toByteArray(taskLogs.streamTaskLog("foo", entry.getKey()).get().openStream());
                final String string = StringUtils.fromUtf8(bytes);
                Assert.assertEquals(StringUtils.format("Read with offset %,d", entry.getKey()), string,
                        entry.getValue());
            }
        } finally {
            FileUtils.deleteDirectory(tmpDir);
        }
    }

    @Test
    public void testPushTaskLogDirCreationFails() throws Exception {
        final File tmpDir = temporaryFolder.newFolder();
        final File logDir = new File(tmpDir, "druid/logs");
        final File logFile = new File(tmpDir, "log");
        Files.write("blah", logFile, StandardCharsets.UTF_8);

        if (!tmpDir.setWritable(false)) {
            throw new RuntimeException("failed to make tmp dir read-only");
        }

        final TaskLogs taskLogs = new FileTaskLogs(new FileTaskLogsConfig(logDir));

        expectedException.expect(IOException.class);
        expectedException.expectMessage("Unable to create task log dir");
        taskLogs.pushTaskLog("foo", logFile);
    }

    @Test
    public void testKill() throws Exception {
        final File tmpDir = temporaryFolder.newFolder();
        final File logDir = new File(tmpDir, "logs");
        final File logFile = new File(tmpDir, "log");
        final TaskLogs taskLogs = new FileTaskLogs(new FileTaskLogsConfig(logDir));

        Files.write("log1content", logFile, StandardCharsets.UTF_8);
        taskLogs.pushTaskLog("log1", logFile);
        Assert.assertEquals("log1content", readLog(taskLogs, "log1", 0));

        //File modification timestamp is only maintained to seconds resolution, so artificial delay
        //is necessary to separate 2 file creations by a timestamp that would result in only one
        //of them getting deleted
        Thread.sleep(1500);
        long time = (System.currentTimeMillis() / 1000) * 1000;
        Assert.assertTrue(new File(logDir, "log1.log").lastModified() < time);

        Files.write("log2content", logFile, StandardCharsets.UTF_8);
        taskLogs.pushTaskLog("log2", logFile);
        Assert.assertEquals("log2content", readLog(taskLogs, "log2", 0));
        Assert.assertTrue(new File(logDir, "log2.log").lastModified() >= time);

        taskLogs.killOlderThan(time);

        Assert.assertFalse(taskLogs.streamTaskLog("log1", 0).isPresent());
        Assert.assertEquals("log2content", readLog(taskLogs, "log2", 0));

    }

    private String readLog(TaskLogs taskLogs, String logFile, long offset) throws IOException {
        return StringUtils
                .fromUtf8(ByteStreams.toByteArray(taskLogs.streamTaskLog(logFile, offset).get().openStream()));
    }
}