diff --git a/src/main/java/org/olat/core/util/WorkThreadInformations.java b/src/main/java/org/olat/core/util/WorkThreadInformations.java index 8d05d62a14628cde05c557ae99dbf5a85a0b116e..568e8dbb678ecd5f51854612f71e207e244238ea 100644 --- a/src/main/java/org/olat/core/util/WorkThreadInformations.java +++ b/src/main/java/org/olat/core/util/WorkThreadInformations.java @@ -21,6 +21,8 @@ package org.olat.core.util; import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; import java.util.ArrayList; import java.util.HashMap; import java.util.Iterator; @@ -101,4 +103,19 @@ public class WorkThreadInformations { log.error("Cannot write info message about FolderIndexerWorker: " + filePath, e); } } + + /** + * The method return an exception if something happens. + * + * @param filePath + * @throws IOException + */ + public static void setInfoFiles(String filePath) throws IOException { + File file = new File(WebappHelper.getUserDataRoot(), "threadInfos"); + if(!file.exists()) { + file.mkdirs(); + } + File infoFile = new File(file, Thread.currentThread().getName()); + FileUtils.save(new FileOutputStream(infoFile), filePath, "UTF-8"); + } } diff --git a/src/main/java/org/olat/restapi/system/MonitoringWebService.java b/src/main/java/org/olat/restapi/system/MonitoringWebService.java index 9c86f9140ffef3e3b2b89864371f5d6e20064bf8..30463bb24fc8c64cf02aeca783beb6549ee9db60 100644 --- a/src/main/java/org/olat/restapi/system/MonitoringWebService.java +++ b/src/main/java/org/olat/restapi/system/MonitoringWebService.java @@ -49,6 +49,11 @@ public class MonitoringWebService { public MonitoringWebService() { //make Spring happy } + + @Path("status") + public StatusWebservice getStatus() { + return new StatusWebservice(); + } @Path("runtime") public RuntimeWebService getCompilationXml() { diff --git a/src/main/java/org/olat/restapi/system/StatusWebservice.java b/src/main/java/org/olat/restapi/system/StatusWebservice.java new file mode 100644 index 0000000000000000000000000000000000000000..493a481aefaf48e3f9395ee4288feb4c0ee0b021 --- /dev/null +++ b/src/main/java/org/olat/restapi/system/StatusWebservice.java @@ -0,0 +1,139 @@ +/** + * <a href="http://www.openolat.org"> + * OpenOLAT - Online Learning and Training</a><br> + * <p> + * Licensed under the Apache License, Version 2.0 (the "License"); <br> + * you may not use this file except in compliance with the License.<br> + * You may obtain a copy of the License at the + * <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache homepage</a> + * <p> + * Unless required by applicable law or agreed to in writing,<br> + * software distributed under the License is distributed on an "AS IS" BASIS, <br> + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br> + * See the License for the specific language governing permissions and <br> + * limitations under the License. + * <p> + * Initial code contributed and copyrighted by<br> + * frentix GmbH, http://www.frentix.com + * <p> + */ +package org.olat.restapi.system; + +import java.util.List; +import java.util.Set; + +import javax.ws.rs.GET; +import javax.ws.rs.Produces; +import javax.ws.rs.core.MediaType; +import javax.ws.rs.core.Response; + +import org.olat.admin.sysinfo.manager.SessionStatsManager; +import org.olat.core.CoreSpringFactory; +import org.olat.core.commons.persistence.DBFactory; +import org.olat.core.id.Identity; +import org.olat.core.id.OLATResourceable; +import org.olat.core.logging.OLog; +import org.olat.core.logging.Tracing; +import org.olat.core.util.CodeHelper; +import org.olat.core.util.SessionInfo; +import org.olat.core.util.UserSession; +import org.olat.core.util.WorkThreadInformations; +import org.olat.core.util.resource.OresHelper; +import org.olat.core.util.session.UserSessionManager; +import org.olat.group.BusinessGroup; +import org.olat.properties.Property; +import org.olat.properties.PropertyManager; +import org.olat.restapi.system.vo.StatusVO; + + +/** + * + * Initial date: 15.02.2016<br> + * @author srosse, stephane.rosse@frentix.com, http://www.frentix.com + * + */ +public class StatusWebservice { + + private static final OLog log = Tracing.createLoggerFor(StatusWebservice.class); + + private static final String PING_REF = "REST-Ping"; + private static final OLATResourceable PING_RESOURCE = OresHelper.createOLATResourceableInstance(PING_REF, 42l); + + /** + * Return the statistics about runtime: uptime, classes loaded, memory + * summary, threads count... + * + * @response.representation.200.qname {http://www.example.com}runtimeVO + * @response.representation.200.mediaType application/xml, application/json + * @response.representation.200.doc The version of the instance + * @response.representation.200.example {@link org.olat.restapi.system.vo.Examples#SAMPLE_RUNTIMEVO} + * @response.representation.401.doc The roles of the authenticated user are not sufficient + * @param request The HTTP request + * @return The informations about runtime, uptime, classes loaded, memory summary... + */ + @GET + @Produces({MediaType.APPLICATION_XML, MediaType.APPLICATION_JSON}) + public Response getSystemSummaryVO() { + StatusVO stats = new StatusVO(); + + //File + try { + stats.setWriteFile(true); + long startFile = System.nanoTime(); + WorkThreadInformations.setInfoFiles("ping"); + WorkThreadInformations.unset(); + stats.setWriteFileInMilliseconds(CodeHelper.nanoToMilliTime(startFile)); + } catch (Exception e) { + stats.setWriteFile(false); + stats.setWriteFileInMilliseconds(-1l); + log.error("", e); + } + + //Datebase + try { + stats.setWriteDb(true); + + PropertyManager propertyManager = CoreSpringFactory.getImpl(PropertyManager.class); + List<Property> props = propertyManager.findProperties((Identity)null, (BusinessGroup)null, PING_RESOURCE, PING_REF, PING_REF); + if(props != null && props.size() > 0) { + for(Property prop:props) { + propertyManager.deleteProperty(prop); + } + DBFactory.getInstance().commit(); + } + + long startDB = System.nanoTime(); + Property prop = propertyManager.createPropertyInstance(null, null, PING_RESOURCE, PING_REF, PING_REF, 0f, 0l, "-", "-"); + DBFactory.getInstance().commit(); + stats.setWriteDbInMilliseconds(CodeHelper.nanoToMilliTime(startDB)); + + propertyManager.deleteProperty(prop); + DBFactory.getInstance().commit(); + } catch (Exception e) { + stats.setWriteDb(false); + stats.setWriteDbInMilliseconds(-1l); + log.error("", e); + } + + //Secure authenticated user + UserSessionManager sessionManager = CoreSpringFactory.getImpl(UserSessionManager.class); + Set<UserSession> userSessions = sessionManager.getAuthenticatedUserSessions(); + int secureAuthenticatedCount = 0; + for (UserSession usess:userSessions) { + SessionInfo sessInfo = usess.getSessionInfo(); + if (sessInfo.isWebDAV() || sessInfo.isREST()) { + // + } else if (sessInfo.isSecure()) { + secureAuthenticatedCount++; + } + } + stats.setSecureAuthenticatedCount(secureAuthenticatedCount); + + //Concurrent dispatch threads + SessionStatsManager sessionStatsManager = CoreSpringFactory.getImpl(SessionStatsManager.class); + stats.setConcurrentDispatchThreads(sessionStatsManager.getConcurrentCounter()); + + return Response.ok(stats).build(); + } + +} diff --git a/src/main/java/org/olat/restapi/system/vo/StatusVO.java b/src/main/java/org/olat/restapi/system/vo/StatusVO.java new file mode 100644 index 0000000000000000000000000000000000000000..3180cdf7c19f02589b02e387f31ad3601c87c413 --- /dev/null +++ b/src/main/java/org/olat/restapi/system/vo/StatusVO.java @@ -0,0 +1,98 @@ +/** + * <a href="http://www.openolat.org"> + * OpenOLAT - Online Learning and Training</a><br> + * <p> + * Licensed under the Apache License, Version 2.0 (the "License"); <br> + * you may not use this file except in compliance with the License.<br> + * You may obtain a copy of the License at the + * <a href="http://www.apache.org/licenses/LICENSE-2.0">Apache homepage</a> + * <p> + * Unless required by applicable law or agreed to in writing,<br> + * software distributed under the License is distributed on an "AS IS" BASIS, <br> + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. <br> + * See the License for the specific language governing permissions and <br> + * limitations under the License. + * <p> + * Initial code contributed and copyrighted by<br> + * frentix GmbH, http://www.frentix.com + * <p> + */ +package org.olat.restapi.system.vo; + +import javax.xml.bind.annotation.XmlAccessType; +import javax.xml.bind.annotation.XmlAccessorType; +import javax.xml.bind.annotation.XmlAttribute; +import javax.xml.bind.annotation.XmlRootElement; + +/** + * + * Initial date: 15.02.2016<br> + * @author srosse, stephane.rosse@frentix.com, http://www.frentix.com + * + */ +@XmlAccessorType(XmlAccessType.FIELD) +@XmlRootElement(name = "statusVO") +public class StatusVO { + + @XmlAttribute(name="writeFile", required=true) + private boolean writeFile; + @XmlAttribute(name="writeFileInMilliseconds", required=true) + private long writeFileInMilliseconds; + @XmlAttribute(name="writeDb", required=true) + private boolean writeDb; + @XmlAttribute(name="writeDbInMilliseconds", required=true) + private long writeDbInMilliseconds; + @XmlAttribute(name="secureAuthenticatedCount", required=true) + private int secureAuthenticatedCount; + @XmlAttribute(name="concurrentDispatchThreads", required=true) + private long concurrentDispatchThreads; + + public boolean isWriteFile() { + return writeFile; + } + + public void setWriteFile(boolean writeFile) { + this.writeFile = writeFile; + } + + public long getWriteFileInMilliseconds() { + return writeFileInMilliseconds; + } + + public void setWriteFileInMilliseconds(long writeFileInMilliseconds) { + this.writeFileInMilliseconds = writeFileInMilliseconds; + } + + public boolean isWriteDb() { + return writeDb; + } + + public void setWriteDb(boolean writeDb) { + this.writeDb = writeDb; + } + + public long getWriteDbInMilliseconds() { + return writeDbInMilliseconds; + } + + public void setWriteDbInMilliseconds(long writeDbInMilliseconds) { + this.writeDbInMilliseconds = writeDbInMilliseconds; + } + + public int getSecureAuthenticatedCount() { + return secureAuthenticatedCount; + } + + public void setSecureAuthenticatedCount(int secureAuthenticatedCount) { + this.secureAuthenticatedCount = secureAuthenticatedCount; + } + + public long getConcurrentDispatchThreads() { + return concurrentDispatchThreads; + } + + public void setConcurrentDispatchThreads(long concurrentDispatchThreads) { + this.concurrentDispatchThreads = concurrentDispatchThreads; + } + +} diff --git a/src/test/java/org/olat/restapi/SystemTest.java b/src/test/java/org/olat/restapi/SystemTest.java index dc6f8194822c96c85fea5d49d8eb062bed1de4f1..26c0c3a403ee2caa5d0ab44ee6cdcde2bc0d5167 100644 --- a/src/test/java/org/olat/restapi/SystemTest.java +++ b/src/test/java/org/olat/restapi/SystemTest.java @@ -36,6 +36,7 @@ import org.olat.restapi.system.vo.MemoryStatisticsVO; import org.olat.restapi.system.vo.MemoryVO; import org.olat.restapi.system.vo.MonitoringInfosVO; import org.olat.restapi.system.vo.OpenOLATStatisticsVO; +import org.olat.restapi.system.vo.StatusVO; import org.olat.restapi.system.vo.ReleaseInfosVO; import org.olat.restapi.system.vo.RepositoryStatisticsVO; import org.olat.restapi.system.vo.RuntimeStatisticsVO; @@ -59,6 +60,21 @@ import org.olat.test.OlatJerseyTestCase; */ public class SystemTest extends OlatJerseyTestCase { + @Test + public void testMonitoringStatus() throws IOException, URISyntaxException { + RestConnection conn = new RestConnection(); + assertTrue(conn.login("administrator", "openolat")); + + URI systemUri = conn.getContextURI().path("system").path("monitoring").path("status").build(); + StatusVO stats = conn.get(systemUri, StatusVO.class); + Assert.assertNotNull(stats); + Assert.assertTrue(stats.isWriteDb()); + Assert.assertTrue(stats.isWriteFile()); + Assert.assertTrue(stats.getConcurrentDispatchThreads() >= 0l); + Assert.assertTrue(stats.getSecureAuthenticatedCount() >= 0l); + conn.shutdown(); + } + @Test public void testRuntimeStatisticsInfos() throws IOException, URISyntaxException { RestConnection conn = new RestConnection();