/**
* 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 org.apache.aurora.scheduler.mesos;
import java.nio.charset.StandardCharsets;
import com.google.common.collect.ImmutableList;
import com.google.common.net.InetAddresses;
import org.apache.aurora.common.testing.easymock.EasyMockTest;
import org.apache.mesos.Protos;
import org.apache.mesos.SchedulerDriver;
import org.apache.mesos.v1.Protos.ExecutorID;
import org.apache.mesos.v1.Protos.FrameworkID;
import org.apache.mesos.v1.Protos.OfferID;
import org.apache.mesos.v1.Protos.TaskID;
import org.apache.mesos.v1.Protos.TaskState;
import org.apache.mesos.v1.Protos.TaskStatus;
import org.apache.mesos.v1.Protos.TaskStatus.Reason;
import org.apache.mesos.v1.Protos.TaskStatus.Source;
import org.junit.Before;
import org.junit.Test;
import static org.apache.aurora.scheduler.mesos.ProtosConversion.convert;
import static org.easymock.EasyMock.expectLastCall;
public class MesosSchedulerImplTest extends EasyMockTest {
private static final String FRAMEWORK_ID = "framework-id";
private static final FrameworkID FRAMEWORK =
FrameworkID.newBuilder().setValue(FRAMEWORK_ID).build();
private static final String MASTER_ID = "master-id";
private static final Protos.MasterInfo MASTER = Protos.MasterInfo.newBuilder()
.setId(MASTER_ID)
.setIp(InetAddresses.coerceToInteger(InetAddresses.forString("1.2.3.4"))) //NOPMD
.setPort(5050).build();
private static final String SLAVE_HOST = "slave-hostname";
private static final Protos.SlaveID SLAVE_ID =
Protos.SlaveID.newBuilder().setValue("slave-id").build();
private static final OfferID OFFER_ID = OfferID.newBuilder().setValue("offer-id").build();
private static final Protos.Offer OFFER = Protos.Offer.newBuilder()
.setFrameworkId(convert(FRAMEWORK))
.setSlaveId(SLAVE_ID)
.setHostname(SLAVE_HOST)
.setId(convert(OFFER_ID))
.build();
private static final ExecutorID EXECUTOR_ID =
ExecutorID.newBuilder().setValue("executor-id").build();
private static final TaskStatus STATUS_NO_REASON = TaskStatus.newBuilder()
.setState(TaskState.TASK_RUNNING)
.setSource(Source.SOURCE_AGENT)
.setMessage("message")
.setTimestamp(1D)
.setTaskId(TaskID.newBuilder().setValue("task-id").build())
.build();
private static final TaskStatus STATUS = STATUS_NO_REASON
.toBuilder()
// Only testing data plumbing, this field with TASK_RUNNING would not normally happen,
.setReason(Reason.REASON_COMMAND_EXECUTOR_FAILED)
.build();
private MesosCallbackHandler handler;
private SchedulerDriver driver;
private MesosSchedulerImpl scheduler;
@Before
public void setUp() {
handler = createMock(MesosCallbackHandler.class);
driver = createMock(SchedulerDriver.class);
scheduler = new MesosSchedulerImpl(handler);
}
@Test
public void testSlaveLost() {
handler.handleLostAgent(convert(SLAVE_ID));
expectLastCall().once();
control.replay();
scheduler.slaveLost(driver, SLAVE_ID);
}
@Test
public void testRegistered() {
handler.handleRegistration(FRAMEWORK, convert(MASTER));
expectLastCall().once();
control.replay();
scheduler.registered(driver, convert(FRAMEWORK), MASTER);
}
@Test
public void testDisconnected() {
handler.handleDisconnection();
expectLastCall().once();
control.replay();
scheduler.disconnected(driver);
}
@Test
public void testReRegistered() {
handler.handleReregistration(convert(MASTER));
expectLastCall().once();
control.replay();
scheduler.reregistered(driver, MASTER);
}
@Test
public void testResourceOffers() {
handler.handleRegistration(FRAMEWORK, convert(MASTER));
expectLastCall().once();
handler.handleOffers(ImmutableList.of(convert(OFFER)));
expectLastCall().once();
control.replay();
scheduler.registered(driver, convert(FRAMEWORK), MASTER);
scheduler.resourceOffers(driver, ImmutableList.of(OFFER));
}
@Test(expected = IllegalStateException.class)
public void testBadOrdering() {
control.replay();
// Should fail since the scheduler is not yet registered.
scheduler.resourceOffers(driver, ImmutableList.of());
}
@Test
public void testOfferRescinded() {
handler.handleRescind(OFFER_ID);
expectLastCall().once();
control.replay();
scheduler.offerRescinded(driver, convert(OFFER_ID));
}
@Test
public void testStatusUpdate() {
handler.handleUpdate(STATUS);
expectLastCall().once();
control.replay();
scheduler.statusUpdate(driver, convert(STATUS));
}
@Test
public void testError() {
handler.handleError("Oh No!");
expectLastCall().once();
control.replay();
scheduler.error(driver, "Oh No!");
}
@Test
public void testFrameworkMessage() {
handler.handleMessage(EXECUTOR_ID, convert(SLAVE_ID));
expectLastCall().once();
control.replay();
scheduler.frameworkMessage(
driver,
convert(EXECUTOR_ID),
SLAVE_ID,
"message".getBytes(StandardCharsets.UTF_8));
}
@Test
public void testExecutorLost() {
handler.handleLostExecutor(EXECUTOR_ID, convert(SLAVE_ID), 1);
control.replay();
scheduler.executorLost(driver, convert(EXECUTOR_ID), SLAVE_ID, 1);
}
}