/* * 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.geode.management.internal.web.controllers; import java.util.concurrent.Callable; import org.apache.geode.internal.lang.StringUtils; import org.apache.geode.management.internal.cli.i18n.CliStrings; import org.apache.geode.management.internal.cli.util.CommandStringBuilder; import org.springframework.http.ResponseEntity; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.PathVariable; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RequestParam; import org.springframework.web.bind.annotation.ResponseBody; /** * The MiscellaneousCommandsController class implements GemFire Management REST API web service * endpoints for the Gfsh Miscellaneous Commands. * <p/> * * @see org.apache.geode.management.internal.cli.commands.MiscellaneousCommands * @see org.apache.geode.management.internal.web.controllers.AbstractCommandsController * @see org.springframework.stereotype.Controller * @see org.springframework.web.bind.annotation.PathVariable * @see org.springframework.web.bind.annotation.RequestMapping * @see org.springframework.web.bind.annotation.RequestMethod * @see org.springframework.web.bind.annotation.RequestParam * @see org.springframework.web.bind.annotation.ResponseBody * @since GemFire 8.0 */ @Controller("miscellaneousController") @RequestMapping(AbstractCommandsController.REST_API_VERSION) @SuppressWarnings("unused") public class MiscellaneousCommandsController extends AbstractCommandsController { @RequestMapping(method = RequestMethod.GET, value = "/logs") public Callable<ResponseEntity<String>> exportLogs( @RequestParam(CliStrings.EXPORT_LOGS__DIR) final String directory, @RequestParam(value = CliStrings.EXPORT_LOGS__GROUP, required = false) final String[] groups, @RequestParam(value = CliStrings.EXPORT_LOGS__MEMBER, required = false) final String memberNameId, @RequestParam(value = CliStrings.EXPORT_LOGS__LOGLEVEL, required = false) final String logLevel, @RequestParam(value = CliStrings.EXPORT_LOGS__UPTO_LOGLEVEL, defaultValue = "false") final Boolean onlyLogLevel, @RequestParam(value = CliStrings.EXPORT_LOGS__MERGELOG, defaultValue = "false") final Boolean mergeLog, @RequestParam(value = CliStrings.EXPORT_LOGS__STARTTIME, required = false) final String startTime, @RequestParam(value = CliStrings.EXPORT_LOGS__ENDTIME, required = false) final String endTime) { final CommandStringBuilder command = new CommandStringBuilder(CliStrings.EXPORT_LOGS); command.addOption(CliStrings.EXPORT_LOGS__DIR, decode(directory)); if (hasValue(groups)) { command.addOption(CliStrings.EXPORT_LOGS__GROUP, StringUtils.concat(groups, StringUtils.COMMA_DELIMITER)); } if (hasValue(memberNameId)) { command.addOption(CliStrings.EXPORT_LOGS__MEMBER, memberNameId); } if (hasValue(logLevel)) { command.addOption(CliStrings.EXPORT_LOGS__LOGLEVEL, logLevel); } command.addOption(CliStrings.EXPORT_LOGS__UPTO_LOGLEVEL, String.valueOf(Boolean.TRUE.equals(onlyLogLevel))); command.addOption(CliStrings.EXPORT_LOGS__MERGELOG, String.valueOf(Boolean.TRUE.equals(mergeLog))); if (hasValue(startTime)) { command.addOption(CliStrings.EXPORT_LOGS__STARTTIME, startTime); } if (hasValue(endTime)) { command.addOption(CliStrings.EXPORT_LOGS__ENDTIME, endTime); } return getProcessCommandCallable(command.toString()); } // TODO determine whether Async functionality is required @RequestMapping(method = RequestMethod.GET, value = "/stacktraces") @ResponseBody public String exportStackTraces( @RequestParam(value = CliStrings.EXPORT_STACKTRACE__FILE, required = false) final String file, @RequestParam(value = CliStrings.EXPORT_STACKTRACE__GROUP, required = false) final String groupName, @RequestParam(value = CliStrings.EXPORT_STACKTRACE__MEMBER, required = false) final String memberNameId, @RequestParam(value = CliStrings.EXPORT_STACKTRACE__FAIL__IF__FILE__PRESENT, required = false) final boolean failIfFilePresent) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.EXPORT_STACKTRACE); if (hasValue(file)) { command.addOption(CliStrings.EXPORT_STACKTRACE__FILE, decode(file)); } if (hasValue(groupName)) { command.addOption(CliStrings.EXPORT_STACKTRACE__GROUP, groupName); } if (hasValue(memberNameId)) { command.addOption(CliStrings.EXPORT_STACKTRACE__MEMBER, memberNameId); } if (hasValue(failIfFilePresent)) { command.addOption(CliStrings.EXPORT_STACKTRACE__FAIL__IF__FILE__PRESENT, String.valueOf(failIfFilePresent)); } return processCommand(command.toString()); } // TODO add Async functionality @RequestMapping(method = RequestMethod.POST, value = "/gc") @ResponseBody public String gc( @RequestParam(value = CliStrings.GC__GROUP, required = false) final String[] groups) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.GC); if (hasValue(groups)) { command.addOption(CliStrings.GC__GROUP, StringUtils.concat(groups, StringUtils.COMMA_DELIMITER)); } return processCommand(command.toString()); } // TODO add Async functionality @RequestMapping(method = RequestMethod.POST, value = "/members/{member}/gc") @ResponseBody public String gc(@PathVariable("member") final String memberNameId) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.GC); command.addOption(CliStrings.GC__MEMBER, decode(memberNameId)); return processCommand(command.toString()); } // TODO add Async functionality @RequestMapping(method = RequestMethod.GET, value = "/netstat") @ResponseBody public String netstat( @RequestParam(value = CliStrings.NETSTAT__MEMBER, required = false) final String[] members, @RequestParam(value = CliStrings.NETSTAT__GROUP, required = false) final String group, @RequestParam(value = CliStrings.NETSTAT__FILE, required = false) final String file, @RequestParam(value = CliStrings.NETSTAT__WITHLSOF, defaultValue = "false") final Boolean withLsof) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.NETSTAT); addCommandOption(null, command, CliStrings.NETSTAT__MEMBER, members); addCommandOption(null, command, CliStrings.NETSTAT__GROUP, group); addCommandOption(null, command, CliStrings.NETSTAT__FILE, file); addCommandOption(null, command, CliStrings.NETSTAT__WITHLSOF, withLsof); return processCommand(command.toString()); } // TODO determine if Async functionality is required @RequestMapping(method = RequestMethod.GET, value = "/deadlocks") @ResponseBody public String showDeadLock( @RequestParam(CliStrings.SHOW_DEADLOCK__DEPENDENCIES__FILE) final String dependenciesFile) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.SHOW_DEADLOCK); command.addOption(CliStrings.SHOW_DEADLOCK__DEPENDENCIES__FILE, decode(dependenciesFile)); return processCommand(command.toString()); } // TODO determine if Async functionality is required @RequestMapping(method = RequestMethod.GET, value = "/members/{member}/log") @ResponseBody public String showLog(@PathVariable("member") final String memberNameId, @RequestParam(value = CliStrings.SHOW_LOG_LINE_NUM, defaultValue = "0") final Integer lines) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.SHOW_LOG); command.addOption(CliStrings.SHOW_LOG_MEMBER, decode(memberNameId)); command.addOption(CliStrings.SHOW_LOG_LINE_NUM, String.valueOf(lines)); return processCommand(command.toString()); } // TODO determine if Async functionality is required @RequestMapping(method = RequestMethod.GET, value = "/metrics") @ResponseBody public String showMetrics( @RequestParam(value = CliStrings.SHOW_METRICS__MEMBER, required = false) final String memberNameId, @RequestParam(value = CliStrings.SHOW_METRICS__REGION, required = false) final String regionNamePath, @RequestParam(value = CliStrings.SHOW_METRICS__FILE, required = false) final String file, @RequestParam(value = CliStrings.SHOW_METRICS__CACHESERVER__PORT, required = false) final String cacheServerPort, @RequestParam(value = CliStrings.SHOW_METRICS__CATEGORY, required = false) final String[] categories) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.SHOW_METRICS); if (hasValue(memberNameId)) { command.addOption(CliStrings.SHOW_METRICS__MEMBER, memberNameId); } if (hasValue(regionNamePath)) { command.addOption(CliStrings.SHOW_METRICS__REGION, regionNamePath); } if (hasValue(file)) { command.addOption(CliStrings.SHOW_METRICS__FILE, file); } if (hasValue(cacheServerPort)) { command.addOption(CliStrings.SHOW_METRICS__CACHESERVER__PORT, cacheServerPort); } if (hasValue(categories)) { command.addOption(CliStrings.SHOW_METRICS__CATEGORY, StringUtils.concat(categories, StringUtils.COMMA_DELIMITER)); } return processCommand(command.toString()); } @RequestMapping(method = RequestMethod.POST, value = "/shutdown") @ResponseBody public String shutdown( @RequestParam(value = CliStrings.SHUTDOWN__TIMEOUT, defaultValue = "-1") final Integer timeout, @RequestParam(value = CliStrings.INCLUDE_LOCATORS, defaultValue = "false") final boolean includeLocators) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.SHUTDOWN); command.addOption(CliStrings.SHUTDOWN__TIMEOUT, String.valueOf(timeout)); command.addOption(CliStrings.INCLUDE_LOCATORS, String.valueOf(includeLocators)); return processCommand(command.toString()); } // TODO determine whether the {groups} and {members} path variables corresponding to the --groups // and --members // command-line options in the 'change loglevel' Gfsh command actually accept multiple values, // and... // TODO if so, then change the groups and members method parameters to String[] types. // TODO If not, then these options should be renamed! @RequestMapping(method = RequestMethod.POST, value = "/groups/{groups}/loglevel") @ResponseBody public String changeLogLevelForGroups(@PathVariable("groups") final String groups, @RequestParam(value = CliStrings.CHANGE_LOGLEVEL__LOGLEVEL, required = true) final String logLevel) { return internalChangeLogLevel(groups, null, logLevel); } @RequestMapping(method = RequestMethod.POST, value = "/members/{members}/loglevel") @ResponseBody public String changeLogLevelForMembers(@PathVariable("members") final String members, @RequestParam(value = CliStrings.CHANGE_LOGLEVEL__LOGLEVEL, required = true) final String logLevel) { return internalChangeLogLevel(null, members, logLevel); } @RequestMapping(method = RequestMethod.POST, value = "/members/{members}/groups/{groups}/loglevel") @ResponseBody public String changeLogLevelForMembersAndGroups(@PathVariable("members") final String members, @PathVariable("groups") final String groups, @RequestParam(value = CliStrings.CHANGE_LOGLEVEL__LOGLEVEL) final String logLevel) { return internalChangeLogLevel(groups, members, logLevel); } // NOTE since "logLevel" is "required", then just set the option; no need to validate it's value. private String internalChangeLogLevel(final String groups, final String members, final String logLevel) { CommandStringBuilder command = new CommandStringBuilder(CliStrings.CHANGE_LOGLEVEL); command.addOption(CliStrings.CHANGE_LOGLEVEL__LOGLEVEL, decode(logLevel)); if (hasValue(groups)) { command.addOption(CliStrings.CHANGE_LOGLEVEL__GROUPS, decode(groups)); } if (hasValue(members)) { command.addOption(CliStrings.CHANGE_LOGLEVEL__MEMBER, decode(members)); } return processCommand(command.toString()); } }