Refactor SystemInfo, and store SystemInfo into singleton instance

- Implement Bill Pugh singleton for ControlMetricsFactory and
  SystemInfo class

Change-Id: Ia97538d9f1be9ea900b0e87371bf50877eaf6483
This commit is contained in:
Jian Li 2016-02-03 16:21:18 -08:00 committed by Gerrit Code Review
parent f53d0b2d1e
commit 1a42469471
9 changed files with 321 additions and 194 deletions

View File

@ -0,0 +1,94 @@
/*
* Copyright 2016 Open Networking Laboratory
*
* 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.onosproject.cpman;
/**
* System information interface.
*/
public interface SystemInfo {
/**
* Returns number of CPU cores.
*
* @return number of CPU cores
*/
int coreCount();
/**
* Returns number of CPUs.
*
* @return number of CPUs
*/
int cpuCount();
/**
* Returns CPU speed in MHz.
*
* @return CPU speed
*/
int cpuSpeed();
/**
* Returns the total amount of memory in Mega Bytes.
*
* @return memory size
*/
int totalMemory();
/**
* A builder of SystemInfo.
*/
interface Builder {
/**
* Sets number of CPU cores.
*
* @param numOfCores number of CPU cores
* @return Builder object
*/
Builder numOfCores(int numOfCores);
/**
* Sets number of CPUs.
* @param numOfCpus number of CPUs
* @return Builder object
*/
Builder numOfCpus(int numOfCpus);
/**
* Sets CPU speed.
*
* @param cpuSpeedMhz CPU speed in Mhz
* @return Builder object
*/
Builder cpuSpeed(int cpuSpeedMhz);
/**
* Sets total amount of memory.
*
* @param totalMemoryMbytes memory size in Mega Bytes
* @return Builder object
*/
Builder totalMemory(int totalMemoryMbytes);
/**
* Builds a SystemInfo object.
*
* @return SystemInfo object
*/
SystemInfo build();
}
}

View File

@ -32,13 +32,9 @@ import java.util.concurrent.ConcurrentHashMap;
* Singleton class to provide various control plane metrics to other components. * Singleton class to provide various control plane metrics to other components.
*/ */
public final class ControlMetricsFactory { public final class ControlMetricsFactory {
private static volatile ControlMetricsFactory uniqueInstance;
private MetricsService metricsService; private MetricsService metricsService;
private boolean enableMonitor = false; private boolean enableMonitor = false;
private Boolean isInitialized = false;
// define a ControlMetricsSystemSpec
private ControlMetricsSystemSpec cmss;
// define a set of MetricsAggregators // define a set of MetricsAggregators
private MetricsAggregator cpuLoad; private MetricsAggregator cpuLoad;
@ -68,49 +64,24 @@ public final class ControlMetricsFactory {
private Set<String> nwInterfaces = Sets.newConcurrentHashSet(); private Set<String> nwInterfaces = Sets.newConcurrentHashSet();
/** /**
* Constructs a control metrics factory using the given metrics and device services. * Initializes the control metrics factory instance using the given
* * metric service and device service. Makes sure that we only initialize
* @param metricsService metric service reference * control metrics factory instance once.
* @param deviceService device service reference
*/
private ControlMetricsFactory(MetricsService metricsService, DeviceService deviceService) {
this.metricsService = metricsService;
registerMetrics();
deviceService.getDevices().forEach(d->deviceIds.add(d.id()));
addAllControlMessageMetrics(deviceIds);
}
/**
* Obtains the unique instance of ControlMetricsFactory.
* *
* @param metricsService metric service * @param metricsService metric service
* @param deviceService device service * @param deviceService device service
* @return instance of ControlMetricsFactory
*/ */
public static ControlMetricsFactory getInstance(MetricsService metricsService, public void initialization(MetricsService metricsService, DeviceService deviceService) {
DeviceService deviceService) { synchronized (isInitialized) {
if (uniqueInstance == null) { if (!isInitialized) {
synchronized (ControlMetricsFactory.class) { this.metricsService = metricsService;
if (uniqueInstance == null) { registerMetrics();
uniqueInstance = new ControlMetricsFactory(metricsService, deviceService); deviceService.getDevices().forEach(d->deviceIds.add(d.id()));
addAllControlMessageMetrics(deviceIds);
isInitialized = true;
} }
} }
} }
return uniqueInstance;
}
/**
* Sets system specification.
*
* @param cmss ControlMetricsSystemSpec object
*/
public void setSystemSpec(ControlMetricsSystemSpec cmss) {
if (this.cmss == null) {
this.cmss = cmss;
}
}
/** /**
* Adds control metrics of a new device. * Adds control metrics of a new device.
@ -579,4 +550,17 @@ public final class ControlMetricsFactory {
public MetricsAggregator replyPacket(DeviceId deviceId) { public MetricsAggregator replyPacket(DeviceId deviceId) {
return replyPacket.get(deviceId); return replyPacket.get(deviceId);
} }
/**
* Returns an instance of control metrics factory.
*
* @return instance of control metrics factory
*/
public static ControlMetricsFactory getInstance() {
return SingletonHelper.INSTANCE;
}
private static class SingletonHelper {
private static final ControlMetricsFactory INSTANCE = new ControlMetricsFactory();
}
} }

View File

@ -1,132 +0,0 @@
/*
* Copyright 2016 Open Networking Laboratory
*
* 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.onosproject.cpman.impl;
/**
* Control metrics class for storing system specification.
*/
public final class ControlMetricsSystemSpec {
private int numOfCores;
private int numOfCpus;
private int cpuSpeed; // in MHz
private long totalMemory; // in bytes
private ControlMetricsSystemSpec(int numOfCores, int numOfCpus,
int cpuSpeed, long totalMemory) {
this.numOfCores = numOfCores;
this.numOfCpus = numOfCpus;
this.cpuSpeed = cpuSpeed;
this.totalMemory = totalMemory;
}
/**
* Returns number of CPU cores.
*
* @return number of CPU cores
*/
public int numOfCores() {
return this.numOfCores;
}
/**
* Returns number of CPUs.
*
* @return number of CPUs
*/
public int numOfCpus() {
return this.numOfCpus;
}
/**
* Returns CPU speed in MHz.
*
* @return CPU speed
*/
public int cpuSpeed() {
return this.cpuSpeed;
}
/**
* Returns the total amount of memory.
*
* @return memory size
*/
public long totalMemory() {
return this.totalMemory;
}
/**
* ControlMetricsSystemSpec builder class.
*/
public static final class Builder {
private int numOfCores;
private int numOfCpus;
private int cpuSpeed; // in MHz
private long totalMemory; // in bytes
/**
* Sets number of CPU cores.
*
* @param numOfCores number of CPU cores
* @return Builder object
*/
public Builder numOfCores(int numOfCores) {
this.numOfCores = numOfCores;
return this;
}
/**
* Sets number of CPUs.
* @param numOfCpus number of CPUs
* @return Builder object
*/
public Builder numOfCpus(int numOfCpus) {
this.numOfCpus = numOfCpus;
return this;
}
/**
* Sets CPU speed.
*
* @param cpuSpeed CPU speed
* @return Builder object
*/
public Builder cpuSpeed(int cpuSpeed) {
this.cpuSpeed = cpuSpeed;
return this;
}
/**
* Sets total amount of memory.
*
* @param totalMemory memory size
* @return Builder object
*/
public Builder totalMemory(long totalMemory) {
this.totalMemory = totalMemory;
return this;
}
/**
* Builds a ControlMetricsSystemSpec object.
*
* @return ControlMetricsSystemSpec object
*/
public ControlMetricsSystemSpec build() {
return new ControlMetricsSystemSpec(numOfCores, numOfCpus, cpuSpeed, totalMemory);
}
}
}

View File

@ -0,0 +1,95 @@
/*
* Copyright 2016 Open Networking Laboratory
*
* 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.onosproject.cpman.impl;
import org.onosproject.cpman.SystemInfo;
/**
* Implementation class of storing system specification.
*/
public final class DefaultSystemInfo implements SystemInfo {
private int numOfCores;
private int numOfCpus;
private int cpuSpeedMhz;
private int totalMemoryMbytes;
private DefaultSystemInfo(int numOfCores, int numOfCpus,
int cpuSpeedMhz, int totalMemoryMbytes) {
this.numOfCores = numOfCores;
this.numOfCpus = numOfCpus;
this.cpuSpeedMhz = cpuSpeedMhz;
this.totalMemoryMbytes = totalMemoryMbytes;
}
@Override
public int coreCount() {
return this.numOfCores;
}
@Override
public int cpuCount() {
return this.numOfCpus;
}
@Override
public int cpuSpeed() {
return this.cpuSpeedMhz;
}
@Override
public int totalMemory() {
return this.totalMemoryMbytes;
}
/**
* ControlMetricsSystemSpec builder class.
*/
public static final class Builder implements SystemInfo.Builder {
private int numOfCores;
private int numOfCpus;
private int cpuSpeedMHz;
private int totalMemoryBytes;
@Override
public SystemInfo.Builder numOfCores(int numOfCores) {
this.numOfCores = numOfCores;
return this;
}
@Override
public Builder numOfCpus(int numOfCpus) {
this.numOfCpus = numOfCpus;
return this;
}
@Override
public Builder cpuSpeed(int cpuSpeedMhz) {
this.cpuSpeedMHz = cpuSpeedMhz;
return this;
}
@Override
public Builder totalMemory(int totalMemoryBytes) {
this.totalMemoryBytes = totalMemoryBytes;
return this;
}
@Override
public DefaultSystemInfo build() {
return new DefaultSystemInfo(numOfCores, numOfCpus, cpuSpeedMHz, totalMemoryBytes);
}
}
}

View File

@ -0,0 +1,74 @@
/*
* Copyright 2016 Open Networking Laboratory
*
* 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.onosproject.cpman.impl;
import org.onosproject.cpman.SystemInfo;
import org.slf4j.Logger;
import static org.slf4j.LoggerFactory.getLogger;
/**
* A factory class which instantiates a system info object.
*/
public final class SystemInfoFactory {
private final Logger log = getLogger(getClass());
private SystemInfo systemInfo;
// non-instantiable (except for our Singleton)
private SystemInfoFactory() {
}
/**
* Returns system information.
*
* @return reference object of system info
*/
public SystemInfo getSystemInfo() {
synchronized (systemInfo) {
return this.systemInfo;
}
}
/**
* Set system information only if it is empty.
*
* @param systemInfo reference object of system info
*/
public void setSystemInfo(SystemInfo systemInfo) {
synchronized (systemInfo) {
if (this.systemInfo == null) {
this.systemInfo = systemInfo;
} else {
log.warn("System information has already been set");
}
}
}
/**
* Returns an instance of system info factory.
*
* @return instance of system info factory
*/
public static SystemInfoFactory getInstance() {
return SingletonHelper.INSTANCE;
}
private static class SingletonHelper {
private static final SystemInfoFactory INSTANCE = new SystemInfoFactory();
}
}

View File

@ -22,7 +22,9 @@ import org.onosproject.cpman.ControlMetric;
import org.onosproject.cpman.ControlMetricType; import org.onosproject.cpman.ControlMetricType;
import org.onosproject.cpman.ControlPlaneMonitorService; import org.onosproject.cpman.ControlPlaneMonitorService;
import org.onosproject.cpman.MetricValue; import org.onosproject.cpman.MetricValue;
import org.onosproject.cpman.impl.ControlMetricsSystemSpec; import org.onosproject.cpman.SystemInfo;
import org.onosproject.cpman.impl.DefaultSystemInfo;
import org.onosproject.cpman.impl.SystemInfoFactory;
import org.onosproject.rest.AbstractWebResource; import org.onosproject.rest.AbstractWebResource;
import javax.ws.rs.Consumes; import javax.ws.rs.Consumes;
@ -227,19 +229,19 @@ public class ControlMetricsCollectorWebResource extends AbstractWebResource {
} }
/** /**
* Collects system specifications. * Collects system information.
* The system specs include the various control metrics * The system information includes the various control metrics
* which do not require aggregation. * which do not require aggregation.
* *
* @param stream JSON stream * @param stream JSON stream
* @return 200 OK * @return 200 OK
* @onos.rsModel SystemSpecsPost * @onos.rsModel SystemInfoPost
*/ */
@POST @POST
@Path("system_specs") @Path("system_info")
@Consumes(MediaType.APPLICATION_JSON) @Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON) @Produces(MediaType.APPLICATION_JSON)
public Response systemSpecs(InputStream stream) { public Response systemInfo(InputStream stream) {
ObjectNode root = mapper().createObjectNode(); ObjectNode root = mapper().createObjectNode();
try { try {
@ -249,15 +251,17 @@ public class ControlMetricsCollectorWebResource extends AbstractWebResource {
JsonNode cpuSpeed = jsonTree.get("cpuSpeed"); JsonNode cpuSpeed = jsonTree.get("cpuSpeed");
JsonNode totalMemory = jsonTree.get("totalMemory"); JsonNode totalMemory = jsonTree.get("totalMemory");
if (numOfCores != null && numOfCpus != null && cpuSpeed != null && totalMemory != null) { if (numOfCores != null && numOfCpus != null &&
ControlMetricsSystemSpec.Builder builder = new ControlMetricsSystemSpec.Builder(); cpuSpeed != null && totalMemory != null) {
ControlMetricsSystemSpec cmss = builder.numOfCores(numOfCores.asInt()) SystemInfo systemInfo = new DefaultSystemInfo.Builder()
.numOfCores(numOfCores.asInt())
.numOfCpus(numOfCpus.asInt()) .numOfCpus(numOfCpus.asInt())
.cpuSpeed(cpuSpeed.asInt()) .cpuSpeed(cpuSpeed.asInt())
.totalMemory(totalMemory.asLong()) .totalMemory(totalMemory.asInt())
.build(); .build();
// TODO: need to implement spec store
// try to store the system info.
SystemInfoFactory.getInstance().setSystemInfo(systemInfo);
} else { } else {
throw new IllegalArgumentException(INVALID_SYSTEM_SPECS); throw new IllegalArgumentException(INVALID_SYSTEM_SPECS);
} }

View File

@ -27,6 +27,8 @@ import org.onlab.osgi.ServiceDirectory;
import org.onlab.osgi.TestServiceDirectory; import org.onlab.osgi.TestServiceDirectory;
import org.onlab.rest.BaseResource; import org.onlab.rest.BaseResource;
import org.onosproject.cpman.ControlPlaneMonitorService; import org.onosproject.cpman.ControlPlaneMonitorService;
import org.onosproject.cpman.SystemInfo;
import org.onosproject.cpman.impl.SystemInfoFactory;
import org.onosproject.net.DeviceId; import org.onosproject.net.DeviceId;
import javax.ws.rs.core.MediaType; import javax.ws.rs.core.MediaType;
@ -123,8 +125,14 @@ public class ControlMetricsCollectorResourceTest extends JerseyTest {
} }
@Test @Test
public void testSystemSpecsPost() { public void testSystemInfoPost() {
basePostTest("system-spec-post.json", PREFIX + "/system_specs"); basePostTest("system-info-post.json", PREFIX + "/system_info");
SystemInfo si = SystemInfoFactory.getInstance().getSystemInfo();
assertThat(si.cpuSpeed(), is(2048));
assertThat(si.coreCount(), is(6));
assertThat(si.cpuCount(), is(2));
assertThat(si.totalMemory(), is(4096));
} }
private ClientResponse baseTest(String jsonFile, String path) { private ClientResponse baseTest(String jsonFile, String path) {