mirror of
https://github.com/apache/cloudstack.git
synced 2025-10-26 08:42:29 +01:00
398 lines
14 KiB
Java
Executable File
398 lines
14 KiB
Java
Executable File
/**
|
|
* Copyright (C) 2010 Cloud.com, Inc. All rights reserved.
|
|
*
|
|
* This software is licensed under the GNU General Public License v3 or later.
|
|
*
|
|
* It is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or any later version.
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
package com.cloud.cluster;
|
|
|
|
import java.sql.Connection;
|
|
import java.sql.PreparedStatement;
|
|
import java.sql.SQLException;
|
|
import java.util.ArrayList;
|
|
import java.util.HashMap;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
import java.util.concurrent.ConcurrentHashMap;
|
|
|
|
import javax.ejb.Local;
|
|
import javax.naming.ConfigurationException;
|
|
|
|
import junit.framework.TestCase;
|
|
|
|
import org.apache.log4j.Logger;
|
|
import org.junit.After;
|
|
import org.junit.Before;
|
|
|
|
import com.cloud.agent.Listener;
|
|
import com.cloud.agent.api.Answer;
|
|
import com.cloud.agent.api.Command;
|
|
import com.cloud.cluster.dao.StackMaidDao;
|
|
import com.cloud.cluster.dao.StackMaidDaoImpl;
|
|
import com.cloud.configuration.Config;
|
|
import com.cloud.configuration.DefaultInterceptorLibrary;
|
|
import com.cloud.configuration.dao.ConfigurationDaoImpl;
|
|
import com.cloud.exception.AgentUnavailableException;
|
|
import com.cloud.exception.OperationTimedoutException;
|
|
import com.cloud.host.Status.Event;
|
|
import com.cloud.serializer.SerializerHelper;
|
|
import com.cloud.utils.component.ComponentLocator;
|
|
import com.cloud.utils.component.MockComponentLocator;
|
|
import com.cloud.utils.db.Transaction;
|
|
import com.cloud.utils.exception.CloudRuntimeException;
|
|
|
|
public class CheckPointManagerTest extends TestCase {
|
|
private final static Logger s_logger = Logger.getLogger(CheckPointManagerTest.class);
|
|
|
|
@Override
|
|
@Before
|
|
public void setUp() {
|
|
MockComponentLocator locator = new MockComponentLocator("management-server");
|
|
locator.addDao("StackMaidDao", StackMaidDaoImpl.class);
|
|
locator.addDao("ConfigurationDao", ConfigurationDaoImpl.class);
|
|
locator.addManager("ClusterManager", MockClusterManager.class);
|
|
locator.makeActive(new DefaultInterceptorLibrary());
|
|
MockMaid.map.clear();
|
|
s_logger.info("Cleaning up the database");
|
|
Connection conn = Transaction.getStandaloneConnection();
|
|
try {
|
|
conn.setAutoCommit(true);
|
|
PreparedStatement stmt = conn.prepareStatement("DELETE FROM stack_maid");
|
|
stmt.executeUpdate();
|
|
stmt.close();
|
|
conn.close();
|
|
} catch (SQLException e) {
|
|
throw new CloudRuntimeException("Unable to setup database", e);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
@After
|
|
public void tearDown() throws Exception {
|
|
}
|
|
|
|
public void testCompleteCase() throws Exception {
|
|
ComponentLocator locator = ComponentLocator.getCurrentLocator();
|
|
|
|
CheckPointManagerImpl taskMgr = ComponentLocator.inject(CheckPointManagerImpl.class);
|
|
assertTrue(taskMgr.configure("TaskManager", new HashMap<String, Object>()));
|
|
assertTrue(taskMgr.start());
|
|
|
|
MockMaid delegate = new MockMaid();
|
|
delegate.setValue("first");
|
|
long taskId = taskMgr.pushCheckPoint(delegate);
|
|
|
|
StackMaidDao maidDao = locator.getDao(StackMaidDao.class);
|
|
CheckPointVO task = maidDao.findById(taskId);
|
|
|
|
assertEquals(task.getDelegate(), MockMaid.class.getName());
|
|
MockMaid retrieved = (MockMaid)SerializerHelper.fromSerializedString(task.getContext());
|
|
assertEquals(retrieved.getValue(), delegate.getValue());
|
|
|
|
delegate.setValue("second");
|
|
taskMgr.updateCheckPointState(taskId, delegate);
|
|
|
|
task = maidDao.findById(taskId);
|
|
assertEquals(task.getDelegate(), MockMaid.class.getName());
|
|
retrieved = (MockMaid)SerializerHelper.fromSerializedString(task.getContext());
|
|
assertEquals(retrieved.getValue(), delegate.getValue());
|
|
|
|
taskMgr.popCheckPoint(taskId);
|
|
assertNull(maidDao.findById(taskId));
|
|
}
|
|
|
|
public void testSimulatedReboot() throws Exception {
|
|
ComponentLocator locator = ComponentLocator.getCurrentLocator();
|
|
|
|
CheckPointManagerImpl taskMgr = ComponentLocator.inject(CheckPointManagerImpl.class);
|
|
assertTrue(taskMgr.configure("TaskManager", new HashMap<String, Object>()));
|
|
assertTrue(taskMgr.start());
|
|
|
|
MockMaid maid = new MockMaid();
|
|
maid.setValue("first");
|
|
long taskId = taskMgr.pushCheckPoint(maid);
|
|
|
|
StackMaidDao maidDao = locator.getDao(StackMaidDao.class);
|
|
CheckPointVO task = maidDao.findById(taskId);
|
|
|
|
assertEquals(task.getDelegate(), MockMaid.class.getName());
|
|
MockMaid retrieved = (MockMaid)SerializerHelper.fromSerializedString(task.getContext());
|
|
assertEquals(retrieved.getValue(), maid.getValue());
|
|
|
|
taskMgr.stop();
|
|
|
|
assertNotNull(MockMaid.map.get(maid.getSeq()));
|
|
|
|
taskMgr = ComponentLocator.inject(CheckPointManagerImpl.class);
|
|
HashMap<String, Object> params = new HashMap<String, Object>();
|
|
params.put(Config.TaskCleanupRetryInterval.key(), "1");
|
|
taskMgr.configure("TaskManager", params);
|
|
taskMgr.start();
|
|
|
|
int i = 0;
|
|
while (MockMaid.map.get(maid.getSeq()) != null && i++ < 5) {
|
|
Thread.sleep(1000);
|
|
}
|
|
|
|
assertNull(MockMaid.map.get(maid.getSeq()));
|
|
}
|
|
|
|
public void testTakeover() throws Exception {
|
|
ComponentLocator locator = ComponentLocator.getCurrentLocator();
|
|
|
|
CheckPointManagerImpl taskMgr = ComponentLocator.inject(CheckPointManagerImpl.class);
|
|
assertTrue(taskMgr.configure("TaskManager", new HashMap<String, Object>()));
|
|
assertTrue(taskMgr.start());
|
|
|
|
MockMaid delegate = new MockMaid();
|
|
delegate.setValue("first");
|
|
long taskId = taskMgr.pushCheckPoint(delegate);
|
|
|
|
StackMaidDao maidDao = locator.getDao(StackMaidDao.class);
|
|
CheckPointVO task = maidDao.findById(taskId);
|
|
|
|
assertEquals(task.getDelegate(), MockMaid.class.getName());
|
|
MockMaid retrieved = (MockMaid)SerializerHelper.fromSerializedString(task.getContext());
|
|
assertEquals(retrieved.getValue(), delegate.getValue());
|
|
|
|
Connection conn = Transaction.getStandaloneConnection();
|
|
try {
|
|
conn.setAutoCommit(true);
|
|
PreparedStatement stmt = conn.prepareStatement("update stack_maid set msid=? where msid=?");
|
|
stmt.setLong(1, 1234);
|
|
stmt.setLong(2, ManagementServerNode.getManagementServerId());
|
|
stmt.executeUpdate();
|
|
stmt.close();
|
|
} finally {
|
|
conn.close();
|
|
}
|
|
|
|
MockClusterManager clusterMgr = (MockClusterManager)locator.getManager(ClusterManager.class);
|
|
clusterMgr.triggerTakeover(1234);
|
|
|
|
int i = 0;
|
|
while (MockMaid.map.get(delegate.getSeq()) != null && i++ < 500) {
|
|
Thread.sleep(1000);
|
|
}
|
|
|
|
assertNull(MockMaid.map.get(delegate.getSeq()));
|
|
}
|
|
|
|
public static class MockMaid implements CleanupMaid {
|
|
private static int s_seq = 1;
|
|
public static Map<Integer, MockMaid> map = new ConcurrentHashMap<Integer, MockMaid>();
|
|
|
|
int seq;
|
|
boolean canBeCleanup;
|
|
String value;
|
|
|
|
protected MockMaid() {
|
|
canBeCleanup = true;
|
|
seq = s_seq++;
|
|
map.put(seq, this);
|
|
}
|
|
|
|
public int getSeq() {
|
|
return seq;
|
|
}
|
|
|
|
public String getValue() {
|
|
return value;
|
|
}
|
|
|
|
public void setCanBeCleanup(boolean canBeCleanup) {
|
|
this.canBeCleanup = canBeCleanup;
|
|
}
|
|
|
|
@Override
|
|
public int cleanup(CheckPointManager checkPointMgr) {
|
|
s_logger.debug("Cleanup called for " + seq);
|
|
map.remove(seq);
|
|
return canBeCleanup ? 0 : -1;
|
|
}
|
|
|
|
public void setValue(String value) {
|
|
this.value = value;
|
|
}
|
|
|
|
@Override
|
|
public String getCleanupProcedure() {
|
|
return "No cleanup necessary";
|
|
}
|
|
}
|
|
|
|
@Local(value=ClusterManager.class)
|
|
public static class MockClusterManager implements ClusterManager {
|
|
String _name;
|
|
ClusterManagerListener _listener;
|
|
|
|
@Override
|
|
public boolean configure(String name, Map<String, Object> params) throws ConfigurationException {
|
|
_name = name;
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean start() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean stop() {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public String getName() {
|
|
return _name;
|
|
}
|
|
|
|
@Override
|
|
public Answer[] execute(String strPeer, long agentId, Command[] cmds, boolean stopOnError) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public long executeAsync(String strPeer, long agentId, Command[] cmds, boolean stopOnError, Listener listener) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public boolean onAsyncResult(String executingPeer, long agentId, long seq, Answer[] answers) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public boolean forwardAnswer(String targetPeer, long agentId, long seq, Answer[] answers) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public Answer[] sendToAgent(Long hostId, Command[] cmds, boolean stopOnError) throws AgentUnavailableException, OperationTimedoutException {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public long sendToAgent(Long hostId, Command[] cmds, boolean stopOnError, Listener listener) throws AgentUnavailableException {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public boolean executeAgentUserRequest(long agentId, Event event) throws AgentUnavailableException {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public Boolean propagateAgentEvent(long agentId, Event event) throws AgentUnavailableException {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public int getHeartbeatThreshold() {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public long getManagementNodeId() {
|
|
return ManagementServerNode.getManagementServerId();
|
|
}
|
|
|
|
@Override
|
|
public boolean isManagementNodeAlive(long msid) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public boolean pingManagementNode(long msid) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public long getCurrentRunId() {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public String getSelfPeerName() {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public String getSelfNodeIP() {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public String getPeerName(long agentHostId) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public void registerListener(ClusterManagerListener listener) {
|
|
_listener = listener;
|
|
}
|
|
|
|
@Override
|
|
public void unregisterListener(ClusterManagerListener listener) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public ManagementServerHostVO getPeer(String peerName) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
@Override
|
|
public void broadcast(long agentId, Command[] cmds) {
|
|
throw new UnsupportedOperationException("Not implemented");
|
|
}
|
|
|
|
public void triggerTakeover(long msId) {
|
|
ManagementServerHostVO node = new ManagementServerHostVO();
|
|
node.setMsid(msId);
|
|
|
|
List<ManagementServerHostVO> lst = new ArrayList<ManagementServerHostVO>();
|
|
lst.add(node);
|
|
|
|
_listener.onManagementNodeLeft(lst, ManagementServerNode.getManagementServerId());
|
|
}
|
|
|
|
protected MockClusterManager() {
|
|
}
|
|
|
|
@Override
|
|
public boolean rebalanceAgent(long agentId, Event event, long currentOwnerId, long futureOwnerId) throws AgentUnavailableException, OperationTimedoutException {
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean isAgentRebalanceEnabled() {
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public Boolean propagateResourceEvent(long agentId, com.cloud.resource.ResourceState.Event event) throws AgentUnavailableException {
|
|
// TODO Auto-generated method stub
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public boolean executeResourceUserRequest(long hostId, com.cloud.resource.ResourceState.Event event) throws AgentUnavailableException {
|
|
// TODO Auto-generated method stub
|
|
return false;
|
|
}
|
|
}
|
|
|
|
}
|